diff --git a/src/plugins/workspace/public/components/workspace_creator/workspace_create_action_panel.test.tsx b/src/plugins/workspace/public/components/workspace_creator/workspace_create_action_panel.test.tsx
index 6f1dbc58bf9e..3fd6e78ac7d1 100644
--- a/src/plugins/workspace/public/components/workspace_creator/workspace_create_action_panel.test.tsx
+++ b/src/plugins/workspace/public/components/workspace_creator/workspace_create_action_panel.test.tsx
@@ -27,6 +27,7 @@ describe('WorkspaceCreateActionPanel', () => {
formId={formId}
formData={{ name: longName, description: formData.description }}
application={mockApplication}
+ isSubmitting={false}
/>
);
const createButton = screen.getByText('Create workspace');
@@ -40,6 +41,7 @@ describe('WorkspaceCreateActionPanel', () => {
formId={formId}
formData={{ name: formData.name, description: longDescription }}
application={mockApplication}
+ isSubmitting={false}
/>
);
const createButton = screen.getByText('Create workspace');
@@ -52,9 +54,23 @@ describe('WorkspaceCreateActionPanel', () => {
formId={formId}
formData={formData}
application={mockApplication}
+ isSubmitting={false}
/>
);
const createButton = screen.getByText('Create workspace');
expect(createButton.closest('button')).not.toBeDisabled();
});
+
+ it('should disable the "Create Workspace" and "Cancel" button when submitting', () => {
+ render(
+
+ );
+ expect(screen.getByText('Create workspace').closest('button')).toBeDisabled();
+ expect(screen.getByText('Cancel').closest('button')).toBeDisabled();
+ });
});
diff --git a/src/plugins/workspace/public/components/workspace_creator/workspace_create_action_panel.tsx b/src/plugins/workspace/public/components/workspace_creator/workspace_create_action_panel.tsx
index 7fb6e8d512cc..782196e2bbc7 100644
--- a/src/plugins/workspace/public/components/workspace_creator/workspace_create_action_panel.tsx
+++ b/src/plugins/workspace/public/components/workspace_creator/workspace_create_action_panel.tsx
@@ -17,12 +17,14 @@ interface WorkspaceCreateActionPanelProps {
formId: string;
formData: Pick;
application: ApplicationStart;
+ isSubmitting: boolean;
}
export const WorkspaceCreateActionPanel = ({
formId,
formData,
application,
+ isSubmitting,
}: WorkspaceCreateActionPanelProps) => {
const [isCancelModalVisible, setIsCancelModalVisible] = useState(false);
const closeCancelModal = useCallback(() => setIsCancelModalVisible(false), []);
@@ -38,6 +40,7 @@ export const WorkspaceCreateActionPanel = ({
{i18n.translate('workspace.form.right.sidebar.buttons.cancelText', {
defaultMessage: 'Cancel',
@@ -50,7 +53,8 @@ export const WorkspaceCreateActionPanel = ({
form={formId}
data-test-subj="workspaceForm-bottomBar-createButton"
fill
- disabled={createButtonDisabled}
+ disabled={createButtonDisabled || isSubmitting}
+ isLoading={isSubmitting}
>
{i18n.translate('workspace.form.right.sidebar.buttons.createWorkspaceText', {
defaultMessage: 'Create workspace',
diff --git a/src/plugins/workspace/public/components/workspace_creator/workspace_creator.test.tsx b/src/plugins/workspace/public/components/workspace_creator/workspace_creator.test.tsx
index 760c4060de58..083c1d61c4de 100644
--- a/src/plugins/workspace/public/components/workspace_creator/workspace_creator.test.tsx
+++ b/src/plugins/workspace/public/components/workspace_creator/workspace_creator.test.tsx
@@ -344,4 +344,33 @@ describe('WorkspaceCreator', () => {
});
expect(notificationToastsAddDanger).not.toHaveBeenCalled();
});
+
+ it('should not create workspace API when submitting', async () => {
+ workspaceClientCreate.mockImplementationOnce(
+ () =>
+ new Promise((resolve) => {
+ setTimeout(resolve, 100);
+ })
+ );
+ const { getByTestId } = render();
+ // Ensure workspace create form rendered
+ await waitFor(() => {
+ expect(getByTestId('workspaceForm-bottomBar-createButton')).toBeInTheDocument();
+ });
+ const nameInput = getByTestId('workspaceForm-workspaceDetails-nameInputText');
+ fireEvent.input(nameInput, {
+ target: { value: 'test workspace name' },
+ });
+ fireEvent.click(getByTestId('workspaceUseCase-observability'));
+ fireEvent.click(getByTestId('workspaceForm-bottomBar-createButton'));
+ expect(workspaceClientCreate).toHaveBeenCalledTimes(1);
+
+ fireEvent.click(getByTestId('workspaceForm-bottomBar-createButton'));
+ expect(workspaceClientCreate).toHaveBeenCalledTimes(1);
+
+ await waitFor(() => {
+ fireEvent.click(getByTestId('workspaceForm-bottomBar-createButton'));
+ expect(workspaceClientCreate).toHaveBeenCalledTimes(2);
+ });
+ });
});
diff --git a/src/plugins/workspace/public/components/workspace_creator/workspace_creator.tsx b/src/plugins/workspace/public/components/workspace_creator/workspace_creator.tsx
index 8f99ba4731ee..9a96137c143f 100644
--- a/src/plugins/workspace/public/components/workspace_creator/workspace_creator.tsx
+++ b/src/plugins/workspace/public/components/workspace_creator/workspace_creator.tsx
@@ -3,7 +3,7 @@
* SPDX-License-Identifier: Apache-2.0
*/
-import React, { useCallback } from 'react';
+import React, { useCallback, useState } from 'react';
import { EuiPage, EuiPageBody, EuiPageContent, euiPaletteColorBlind } from '@elastic/eui';
import { i18n } from '@osd/i18n';
import { BehaviorSubject } from 'rxjs';
@@ -43,6 +43,7 @@ export const WorkspaceCreator = (props: WorkspaceCreatorProps) => {
dataSourceManagement?: DataSourceManagementPluginSetup;
navigationUI: NavigationPublicPluginStart['ui'];
}>();
+ const [isFormSubmitting, setIsFormSubmitting] = useState(false);
const isPermissionEnabled = application?.capabilities.workspaces.permissionEnabled;
const { isOnlyAllowEssential, availableUseCases } = useFormAvailableUseCases({
@@ -65,6 +66,10 @@ export const WorkspaceCreator = (props: WorkspaceCreatorProps) => {
const handleWorkspaceFormSubmit = useCallback(
async (data: WorkspaceFormSubmitData) => {
let result;
+ if (isFormSubmitting) {
+ return;
+ }
+ setIsFormSubmitting(true);
try {
const { permissionSettings, selectedDataSources, ...attributes } = data;
const selectedDataSourceIds = (selectedDataSources ?? []).map((ds: DataSource) => {
@@ -105,9 +110,11 @@ export const WorkspaceCreator = (props: WorkspaceCreatorProps) => {
text: error instanceof Error ? error.message : JSON.stringify(error),
});
return;
+ } finally {
+ setIsFormSubmitting(false);
}
},
- [notifications?.toasts, http, application, workspaceClient]
+ [notifications?.toasts, http, application, workspaceClient, isFormSubmitting]
);
const isFormReadyToRender =
@@ -146,6 +153,7 @@ export const WorkspaceCreator = (props: WorkspaceCreatorProps) => {
dataSourceManagement={dataSourceManagement}
availableUseCases={availableUseCases}
defaultValues={defaultWorkspaceFormValues}
+ isSubmitting={isFormSubmitting}
/>
)}
diff --git a/src/plugins/workspace/public/components/workspace_creator/workspace_creator_form.test.tsx b/src/plugins/workspace/public/components/workspace_creator/workspace_creator_form.test.tsx
index 81251685fb3e..fbbf4443c9ec 100644
--- a/src/plugins/workspace/public/components/workspace_creator/workspace_creator_form.test.tsx
+++ b/src/plugins/workspace/public/components/workspace_creator/workspace_creator_form.test.tsx
@@ -38,6 +38,7 @@ const setup = (
return render(
{
+interface WorkspaceCreatorFormProps extends WorkspaceFormProps {
+ isSubmitting: boolean;
+}
+
+export const WorkspaceCreatorForm = (props: WorkspaceCreatorFormProps) => {
const {
application,
savedObjects,
@@ -217,6 +221,7 @@ export const WorkspaceCreatorForm = (props: WorkspaceFormProps) => {
formData={formData}
formId={formId}
application={application}
+ isSubmitting={props.isSubmitting}
/>