From 074b668148474e5f507e7003c7b7b456e521b7ac Mon Sep 17 00:00:00 2001 From: "opensearch-trigger-bot[bot]" <98922864+opensearch-trigger-bot[bot]@users.noreply.github.com> Date: Fri, 3 Nov 2023 15:15:10 -0700 Subject: [PATCH] [Backport 2.x] Put plugin API behind a feature flag (#145) Put plugin API behind a feature flag (#142) * Put plugin API behind a feature flag * Add test for feature flag disabled --------- (cherry picked from commit aad17d0b07db30d9158f6c148a1c1b40df6bb3e3) Signed-off-by: Daniel Widdis Signed-off-by: github-actions[bot] Co-authored-by: github-actions[bot] --- .../flowframework/FlowFrameworkPlugin.java | 18 +++++- .../FlowFrameworkFeatureEnabledSetting.java | 50 +++++++++++++++++ .../rest/RestCreateWorkflowAction.java | 20 ++++++- .../rest/RestProvisionWorkflowAction.java | 16 +++++- .../FlowFrameworkPluginTests.java | 30 +++++++++- ...owFrameworkFeatureEnabledSettingTests.java | 56 +++++++++++++++++++ .../rest/RestCreateWorkflowActionTests.java | 18 +++++- .../RestProvisionWorkflowActionTests.java | 18 +++++- 8 files changed, 218 insertions(+), 8 deletions(-) create mode 100644 src/main/java/org/opensearch/flowframework/common/FlowFrameworkFeatureEnabledSetting.java create mode 100644 src/test/java/org/opensearch/flowframework/common/FlowFrameworkFeatureEnabledSettingTests.java diff --git a/src/main/java/org/opensearch/flowframework/FlowFrameworkPlugin.java b/src/main/java/org/opensearch/flowframework/FlowFrameworkPlugin.java index 907bde68b..15e61f558 100644 --- a/src/main/java/org/opensearch/flowframework/FlowFrameworkPlugin.java +++ b/src/main/java/org/opensearch/flowframework/FlowFrameworkPlugin.java @@ -16,6 +16,7 @@ import org.opensearch.cluster.service.ClusterService; import org.opensearch.common.settings.ClusterSettings; import org.opensearch.common.settings.IndexScopedSettings; +import org.opensearch.common.settings.Setting; import org.opensearch.common.settings.Settings; import org.opensearch.common.settings.SettingsFilter; import org.opensearch.common.util.concurrent.OpenSearchExecutors; @@ -24,6 +25,7 @@ import org.opensearch.core.xcontent.NamedXContentRegistry; import org.opensearch.env.Environment; import org.opensearch.env.NodeEnvironment; +import org.opensearch.flowframework.common.FlowFrameworkFeatureEnabledSetting; import org.opensearch.flowframework.indices.FlowFrameworkIndicesHandler; import org.opensearch.flowframework.rest.RestCreateWorkflowAction; import org.opensearch.flowframework.rest.RestProvisionWorkflowAction; @@ -57,6 +59,8 @@ */ public class FlowFrameworkPlugin extends Plugin implements ActionPlugin { + private FlowFrameworkFeatureEnabledSetting flowFrameworkFeatureEnabledSetting; + /** * Instantiate this plugin. */ @@ -76,6 +80,9 @@ public Collection createComponents( IndexNameExpressionResolver indexNameExpressionResolver, Supplier repositoriesServiceSupplier ) { + Settings settings = environment.settings(); + flowFrameworkFeatureEnabledSetting = new FlowFrameworkFeatureEnabledSetting(clusterService, settings); + MachineLearningNodeClient mlClient = new MachineLearningNodeClient(client); WorkflowStepFactory workflowStepFactory = new WorkflowStepFactory(clusterService, client, mlClient); WorkflowProcessSorter workflowProcessSorter = new WorkflowProcessSorter(workflowStepFactory, threadPool); @@ -95,7 +102,10 @@ public List getRestHandlers( IndexNameExpressionResolver indexNameExpressionResolver, Supplier nodesInCluster ) { - return ImmutableList.of(new RestCreateWorkflowAction(), new RestProvisionWorkflowAction()); + return ImmutableList.of( + new RestCreateWorkflowAction(flowFrameworkFeatureEnabledSetting), + new RestProvisionWorkflowAction(flowFrameworkFeatureEnabledSetting) + ); } @Override @@ -106,6 +116,12 @@ public List getRestHandlers( ); } + @Override + public List> getSettings() { + List> settings = ImmutableList.of(FlowFrameworkFeatureEnabledSetting.FLOW_FRAMEWORK_ENABLED); + return settings; + } + @Override public List> getExecutorBuilders(Settings settings) { // TODO : Determine final size/queueSize values for the provision thread pool diff --git a/src/main/java/org/opensearch/flowframework/common/FlowFrameworkFeatureEnabledSetting.java b/src/main/java/org/opensearch/flowframework/common/FlowFrameworkFeatureEnabledSetting.java new file mode 100644 index 000000000..f10068f5b --- /dev/null +++ b/src/main/java/org/opensearch/flowframework/common/FlowFrameworkFeatureEnabledSetting.java @@ -0,0 +1,50 @@ +/* + * Copyright OpenSearch Contributors + * SPDX-License-Identifier: Apache-2.0 + * + * The OpenSearch Contributors require contributions made to + * this file be licensed under the Apache-2.0 license or a + * compatible open source license. + */ +package org.opensearch.flowframework.common; + +import org.opensearch.cluster.service.ClusterService; +import org.opensearch.common.settings.Setting; +import org.opensearch.common.settings.Settings; + +/** + * Controls enabling or disabling features of this plugin + */ +public class FlowFrameworkFeatureEnabledSetting { + + /** This setting enables/disables the Flow Framework REST API */ + public static final Setting FLOW_FRAMEWORK_ENABLED = Setting.boolSetting( + "plugins.flow_framework.enabled", + false, + Setting.Property.NodeScope, + Setting.Property.Dynamic + ); + + private volatile Boolean isFlowFrameworkEnabled; + + /** + * Instantiate this class. + * + * @param clusterService OpenSearch cluster service + * @param settings OpenSearch settings + */ + public FlowFrameworkFeatureEnabledSetting(ClusterService clusterService, Settings settings) { + // Currently this is just an on/off switch for the entire plugin's API. + // If desired more fine-tuned feature settings can be added below. + isFlowFrameworkEnabled = FLOW_FRAMEWORK_ENABLED.get(settings); + clusterService.getClusterSettings().addSettingsUpdateConsumer(FLOW_FRAMEWORK_ENABLED, it -> isFlowFrameworkEnabled = it); + } + + /** + * Whether the flow framework feature is enabled. If disabled, no REST APIs will be availble. + * @return whether Flow Framework is enabled. + */ + public boolean isFlowFrameworkEnabled() { + return isFlowFrameworkEnabled; + } +} diff --git a/src/main/java/org/opensearch/flowframework/rest/RestCreateWorkflowAction.java b/src/main/java/org/opensearch/flowframework/rest/RestCreateWorkflowAction.java index b5400e247..645e8ad2b 100644 --- a/src/main/java/org/opensearch/flowframework/rest/RestCreateWorkflowAction.java +++ b/src/main/java/org/opensearch/flowframework/rest/RestCreateWorkflowAction.java @@ -16,6 +16,7 @@ import org.opensearch.core.rest.RestStatus; import org.opensearch.core.xcontent.ToXContent; import org.opensearch.core.xcontent.XContentBuilder; +import org.opensearch.flowframework.common.FlowFrameworkFeatureEnabledSetting; import org.opensearch.flowframework.exception.FlowFrameworkException; import org.opensearch.flowframework.model.Template; import org.opensearch.flowframework.transport.CreateWorkflowAction; @@ -31,6 +32,7 @@ import static org.opensearch.flowframework.common.CommonValue.DRY_RUN; import static org.opensearch.flowframework.common.CommonValue.WORKFLOW_ID; import static org.opensearch.flowframework.common.CommonValue.WORKFLOW_URI; +import static org.opensearch.flowframework.common.FlowFrameworkFeatureEnabledSetting.FLOW_FRAMEWORK_ENABLED; /** * Rest Action to facilitate requests to create and update a use case template @@ -40,10 +42,16 @@ public class RestCreateWorkflowAction extends BaseRestHandler { private static final Logger logger = LogManager.getLogger(RestCreateWorkflowAction.class); private static final String CREATE_WORKFLOW_ACTION = "create_workflow_action"; + private FlowFrameworkFeatureEnabledSetting flowFrameworkFeatureEnabledSetting; + /** * Intantiates a new RestCreateWorkflowAction + * + * @param flowFrameworkFeatureEnabledSetting Whether this API is enabled */ - public RestCreateWorkflowAction() {} + public RestCreateWorkflowAction(FlowFrameworkFeatureEnabledSetting flowFrameworkFeatureEnabledSetting) { + this.flowFrameworkFeatureEnabledSetting = flowFrameworkFeatureEnabledSetting; + } @Override public String getName() { @@ -62,6 +70,15 @@ public List routes() { @Override protected RestChannelConsumer prepareRequest(RestRequest request, NodeClient client) throws IOException { + if (!flowFrameworkFeatureEnabledSetting.isFlowFrameworkEnabled()) { + FlowFrameworkException ffe = new FlowFrameworkException( + "This API is disabled. To enable it, set [" + FLOW_FRAMEWORK_ENABLED.getKey() + "] to true.", + RestStatus.FORBIDDEN + ); + return channel -> channel.sendResponse( + new BytesRestResponse(ffe.getRestStatus(), ffe.toXContent(channel.newErrorBuilder(), ToXContent.EMPTY_PARAMS)) + ); + } try { String workflowId = request.param(WORKFLOW_ID); @@ -89,5 +106,4 @@ protected RestChannelConsumer prepareRequest(RestRequest request, NodeClient cli ); } } - } diff --git a/src/main/java/org/opensearch/flowframework/rest/RestProvisionWorkflowAction.java b/src/main/java/org/opensearch/flowframework/rest/RestProvisionWorkflowAction.java index 1bd07eaf0..81e4fb606 100644 --- a/src/main/java/org/opensearch/flowframework/rest/RestProvisionWorkflowAction.java +++ b/src/main/java/org/opensearch/flowframework/rest/RestProvisionWorkflowAction.java @@ -16,6 +16,7 @@ import org.opensearch.core.rest.RestStatus; import org.opensearch.core.xcontent.ToXContent; import org.opensearch.core.xcontent.XContentBuilder; +import org.opensearch.flowframework.common.FlowFrameworkFeatureEnabledSetting; import org.opensearch.flowframework.exception.FlowFrameworkException; import org.opensearch.flowframework.transport.ProvisionWorkflowAction; import org.opensearch.flowframework.transport.WorkflowRequest; @@ -29,6 +30,7 @@ import static org.opensearch.flowframework.common.CommonValue.WORKFLOW_ID; import static org.opensearch.flowframework.common.CommonValue.WORKFLOW_URI; +import static org.opensearch.flowframework.common.FlowFrameworkFeatureEnabledSetting.FLOW_FRAMEWORK_ENABLED; /** * Rest action to facilitate requests to provision a workflow from an inline defined or stored use case template @@ -39,10 +41,16 @@ public class RestProvisionWorkflowAction extends BaseRestHandler { private static final String PROVISION_WORKFLOW_ACTION = "provision_workflow_action"; + private FlowFrameworkFeatureEnabledSetting flowFrameworkFeatureEnabledSetting; + /** * Instantiates a new RestProvisionWorkflowAction + * + * @param flowFrameworkFeatureEnabledSetting Whether this API is enabled */ - public RestProvisionWorkflowAction() {} + public RestProvisionWorkflowAction(FlowFrameworkFeatureEnabledSetting flowFrameworkFeatureEnabledSetting) { + this.flowFrameworkFeatureEnabledSetting = flowFrameworkFeatureEnabledSetting; + } @Override public String getName() { @@ -61,6 +69,12 @@ public List routes() { protected RestChannelConsumer prepareRequest(RestRequest request, NodeClient client) throws IOException { String workflowId = request.param(WORKFLOW_ID); try { + if (!flowFrameworkFeatureEnabledSetting.isFlowFrameworkEnabled()) { + throw new FlowFrameworkException( + "This API is disabled. To enable it, update the setting [" + FLOW_FRAMEWORK_ENABLED.getKey() + "] to true.", + RestStatus.FORBIDDEN + ); + } // Validate content if (request.hasContent()) { throw new FlowFrameworkException("Invalid request format", RestStatus.BAD_REQUEST); diff --git a/src/test/java/org/opensearch/flowframework/FlowFrameworkPluginTests.java b/src/test/java/org/opensearch/flowframework/FlowFrameworkPluginTests.java index cbbc6d7fe..85029cba9 100644 --- a/src/test/java/org/opensearch/flowframework/FlowFrameworkPluginTests.java +++ b/src/test/java/org/opensearch/flowframework/FlowFrameworkPluginTests.java @@ -11,13 +11,21 @@ import org.opensearch.client.AdminClient; import org.opensearch.client.Client; import org.opensearch.client.ClusterAdminClient; +import org.opensearch.cluster.service.ClusterService; +import org.opensearch.common.settings.ClusterSettings; +import org.opensearch.common.settings.Setting; import org.opensearch.common.settings.Settings; +import org.opensearch.env.Environment; +import org.opensearch.flowframework.common.FlowFrameworkFeatureEnabledSetting; import org.opensearch.test.OpenSearchTestCase; import org.opensearch.threadpool.TestThreadPool; import org.opensearch.threadpool.ThreadPool; import java.io.IOException; +import java.util.Set; import java.util.concurrent.TimeUnit; +import java.util.stream.Collectors; +import java.util.stream.Stream; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.when; @@ -31,6 +39,9 @@ public class FlowFrameworkPluginTests extends OpenSearchTestCase { private ClusterAdminClient clusterAdminClient; private ThreadPool threadPool; private Settings settings; + private Environment environment; + private ClusterSettings clusterSettings; + private ClusterService clusterService; @Override public void setUp() throws Exception { @@ -41,7 +52,18 @@ public void setUp() throws Exception { when(client.admin()).thenReturn(adminClient); when(adminClient.cluster()).thenReturn(clusterAdminClient); threadPool = new TestThreadPool(FlowFrameworkPluginTests.class.getName()); - settings = Settings.EMPTY; + + environment = mock(Environment.class); + settings = Settings.builder().build(); + when(environment.settings()).thenReturn(settings); + + final Set> settingsSet = Stream.concat( + ClusterSettings.BUILT_IN_CLUSTER_SETTINGS.stream(), + Stream.of(FlowFrameworkFeatureEnabledSetting.FLOW_FRAMEWORK_ENABLED) + ).collect(Collectors.toSet()); + clusterSettings = new ClusterSettings(settings, settingsSet); + clusterService = mock(ClusterService.class); + when(clusterService.getClusterSettings()).thenReturn(clusterSettings); } @Override @@ -52,10 +74,14 @@ public void tearDown() throws Exception { public void testPlugin() throws IOException { try (FlowFrameworkPlugin ffp = new FlowFrameworkPlugin()) { - assertEquals(3, ffp.createComponents(client, null, threadPool, null, null, null, null, null, null, null, null).size()); + assertEquals( + 3, + ffp.createComponents(client, clusterService, threadPool, null, null, null, environment, null, null, null, null).size() + ); assertEquals(2, ffp.getRestHandlers(null, null, null, null, null, null, null).size()); assertEquals(2, ffp.getActions().size()); assertEquals(1, ffp.getExecutorBuilders(settings).size()); + assertEquals(1, ffp.getSettings().size()); } } } diff --git a/src/test/java/org/opensearch/flowframework/common/FlowFrameworkFeatureEnabledSettingTests.java b/src/test/java/org/opensearch/flowframework/common/FlowFrameworkFeatureEnabledSettingTests.java new file mode 100644 index 000000000..9ac16c6f3 --- /dev/null +++ b/src/test/java/org/opensearch/flowframework/common/FlowFrameworkFeatureEnabledSettingTests.java @@ -0,0 +1,56 @@ +/* + * Copyright OpenSearch Contributors + * SPDX-License-Identifier: Apache-2.0 + * + * The OpenSearch Contributors require contributions made to + * this file be licensed under the Apache-2.0 license or a + * compatible open source license. + */ +package org.opensearch.flowframework.common; + +import org.opensearch.cluster.service.ClusterService; +import org.opensearch.common.settings.ClusterSettings; +import org.opensearch.common.settings.Setting; +import org.opensearch.common.settings.Settings; +import org.opensearch.test.OpenSearchTestCase; + +import java.io.IOException; +import java.util.Set; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.when; + +public class FlowFrameworkFeatureEnabledSettingTests extends OpenSearchTestCase { + + private Settings settings; + private ClusterSettings clusterSettings; + private ClusterService clusterService; + + private FlowFrameworkFeatureEnabledSetting flowFrameworkFeatureEnabledSetting; + + @Override + public void setUp() throws Exception { + super.setUp(); + + settings = Settings.builder().build(); + final Set> settingsSet = Stream.concat( + ClusterSettings.BUILT_IN_CLUSTER_SETTINGS.stream(), + Stream.of(FlowFrameworkFeatureEnabledSetting.FLOW_FRAMEWORK_ENABLED) + ).collect(Collectors.toSet()); + clusterSettings = new ClusterSettings(settings, settingsSet); + clusterService = mock(ClusterService.class); + when(clusterService.getClusterSettings()).thenReturn(clusterSettings); + flowFrameworkFeatureEnabledSetting = new FlowFrameworkFeatureEnabledSetting(clusterService, settings); + } + + @Override + public void tearDown() throws Exception { + super.tearDown(); + } + + public void testSettings() throws IOException { + assertFalse(flowFrameworkFeatureEnabledSetting.isFlowFrameworkEnabled()); + } +} diff --git a/src/test/java/org/opensearch/flowframework/rest/RestCreateWorkflowActionTests.java b/src/test/java/org/opensearch/flowframework/rest/RestCreateWorkflowActionTests.java index 3daaa4536..8a3564abe 100644 --- a/src/test/java/org/opensearch/flowframework/rest/RestCreateWorkflowActionTests.java +++ b/src/test/java/org/opensearch/flowframework/rest/RestCreateWorkflowActionTests.java @@ -14,6 +14,7 @@ import org.opensearch.core.rest.RestStatus; import org.opensearch.core.xcontent.MediaTypeRegistry; import org.opensearch.flowframework.TestHelpers; +import org.opensearch.flowframework.common.FlowFrameworkFeatureEnabledSetting; import org.opensearch.flowframework.model.Template; import org.opensearch.flowframework.model.Workflow; import org.opensearch.flowframework.model.WorkflowEdge; @@ -30,6 +31,7 @@ import static org.opensearch.flowframework.common.CommonValue.WORKFLOW_URI; import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.when; public class RestCreateWorkflowActionTests extends OpenSearchTestCase { @@ -38,10 +40,13 @@ public class RestCreateWorkflowActionTests extends OpenSearchTestCase { private String createWorkflowPath; private String updateWorkflowPath; private NodeClient nodeClient; + private FlowFrameworkFeatureEnabledSetting flowFrameworkFeatureEnabledSetting; @Override public void setUp() throws Exception { super.setUp(); + flowFrameworkFeatureEnabledSetting = mock(FlowFrameworkFeatureEnabledSetting.class); + when(flowFrameworkFeatureEnabledSetting.isFlowFrameworkEnabled()).thenReturn(true); Version templateVersion = Version.fromString("1.0.0"); List compatibilityVersions = List.of(Version.fromString("2.0.0"), Version.fromString("3.0.0")); @@ -64,7 +69,7 @@ public void setUp() throws Exception { // Invalid template configuration, wrong field name this.invalidTemplate = template.toJson().replace("use_case", "invalid"); - this.createWorkflowRestAction = new RestCreateWorkflowAction(); + this.createWorkflowRestAction = new RestCreateWorkflowAction(flowFrameworkFeatureEnabledSetting); this.createWorkflowPath = String.format(Locale.ROOT, "%s", WORKFLOW_URI); this.updateWorkflowPath = String.format(Locale.ROOT, "%s/{%s}", WORKFLOW_URI, "workflow_id"); this.nodeClient = mock(NodeClient.class); @@ -95,4 +100,15 @@ public void testInvalidCreateWorkflowRequest() throws Exception { assertEquals(RestStatus.BAD_REQUEST, channel.capturedResponse().status()); assertTrue(channel.capturedResponse().content().utf8ToString().contains("Unable to parse field [invalid] in a template object.")); } + + public void testFeatureFlagNotEnabled() throws Exception { + when(flowFrameworkFeatureEnabledSetting.isFlowFrameworkEnabled()).thenReturn(false); + RestRequest request = new FakeRestRequest.Builder(xContentRegistry()).withMethod(RestRequest.Method.POST) + .withPath(this.createWorkflowPath) + .build(); + FakeRestChannel channel = new FakeRestChannel(request, false, 1); + createWorkflowRestAction.handleRequest(request, channel, nodeClient); + assertEquals(RestStatus.FORBIDDEN, channel.capturedResponse().status()); + assertTrue(channel.capturedResponse().content().utf8ToString().contains("This API is disabled.")); + } } diff --git a/src/test/java/org/opensearch/flowframework/rest/RestProvisionWorkflowActionTests.java b/src/test/java/org/opensearch/flowframework/rest/RestProvisionWorkflowActionTests.java index 4d9ef22e4..d7bebb9a9 100644 --- a/src/test/java/org/opensearch/flowframework/rest/RestProvisionWorkflowActionTests.java +++ b/src/test/java/org/opensearch/flowframework/rest/RestProvisionWorkflowActionTests.java @@ -12,6 +12,7 @@ import org.opensearch.core.common.bytes.BytesArray; import org.opensearch.core.rest.RestStatus; import org.opensearch.core.xcontent.MediaTypeRegistry; +import org.opensearch.flowframework.common.FlowFrameworkFeatureEnabledSetting; import org.opensearch.rest.RestHandler.Route; import org.opensearch.rest.RestRequest; import org.opensearch.test.OpenSearchTestCase; @@ -23,17 +24,22 @@ import static org.opensearch.flowframework.common.CommonValue.WORKFLOW_URI; import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.when; public class RestProvisionWorkflowActionTests extends OpenSearchTestCase { private RestProvisionWorkflowAction provisionWorkflowRestAction; private String provisionWorkflowPath; private NodeClient nodeClient; + private FlowFrameworkFeatureEnabledSetting flowFrameworkFeatureEnabledSetting; @Override public void setUp() throws Exception { super.setUp(); - this.provisionWorkflowRestAction = new RestProvisionWorkflowAction(); + flowFrameworkFeatureEnabledSetting = mock(FlowFrameworkFeatureEnabledSetting.class); + when(flowFrameworkFeatureEnabledSetting.isFlowFrameworkEnabled()).thenReturn(true); + + this.provisionWorkflowRestAction = new RestProvisionWorkflowAction(flowFrameworkFeatureEnabledSetting); this.provisionWorkflowPath = String.format(Locale.ROOT, "%s/{%s}/%s", WORKFLOW_URI, "workflow_id", "_provision"); this.nodeClient = mock(NodeClient.class); } @@ -81,4 +87,14 @@ public void testInvalidRequestWithContent() { ); } + public void testFeatureFlagNotEnabled() throws Exception { + when(flowFrameworkFeatureEnabledSetting.isFlowFrameworkEnabled()).thenReturn(false); + RestRequest request = new FakeRestRequest.Builder(xContentRegistry()).withMethod(RestRequest.Method.POST) + .withPath(this.provisionWorkflowPath) + .build(); + FakeRestChannel channel = new FakeRestChannel(request, false, 1); + provisionWorkflowRestAction.handleRequest(request, channel, nodeClient); + assertEquals(RestStatus.FORBIDDEN, channel.capturedResponse().status()); + assertTrue(channel.capturedResponse().content().utf8ToString().contains("This API is disabled.")); + } }