diff --git a/packages/dashboard-frontend/src/components/WorkspaceProgress/CreatingSteps/Fetch/Devfile/__tests__/index.spec.tsx b/packages/dashboard-frontend/src/components/WorkspaceProgress/CreatingSteps/Fetch/Devfile/__tests__/index.spec.tsx index 9a58cdcdb..9f91fbf4a 100644 --- a/packages/dashboard-frontend/src/components/WorkspaceProgress/CreatingSteps/Fetch/Devfile/__tests__/index.spec.tsx +++ b/packages/dashboard-frontend/src/components/WorkspaceProgress/CreatingSteps/Fetch/Devfile/__tests__/index.spec.tsx @@ -13,6 +13,7 @@ /* eslint-disable @typescript-eslint/no-non-null-assertion */ import { FACTORY_LINK_ATTR } from '@eclipse-che/common'; +import { AlertVariant } from '@patternfly/react-core'; import { cleanup, screen, waitFor } from '@testing-library/react'; import userEvent, { UserEvent } from '@testing-library/user-event'; import React from 'react'; @@ -704,6 +705,7 @@ describe('Creating steps, fetching a devfile', () => { [FACTORY_URL_ATTR]: factoryUrl, }); + mockIsOAuthResponse.mockReturnValue(false); mockRequestFactoryResolver.mockRejectedValue('Could not reach devfile'); spyWindowLocation = createWindowLocationSpy(host, protocol); @@ -742,6 +744,46 @@ describe('Creating steps, fetching a devfile', () => { expect(mockOpenOAuthPage).not.toHaveBeenCalled(); expect(mockOnError).not.toHaveBeenCalled(); }); + + it('should show warning on SSH url', async () => { + const expectAlertItem = expect.objectContaining({ + title: 'Warning', + variant: AlertVariant.warning, + children: ( + + ), + actionCallbacks: [ + expect.objectContaining({ + title: 'Continue with default devfile', + callback: expect.any(Function), + }), + expect.objectContaining({ + title: 'Reload', + callback: expect.any(Function), + }), + expect.objectContaining({ + title: 'Open Documentation page', + callback: expect.any(Function), + }), + ], + }); + searchParams = new URLSearchParams({ + [FACTORY_URL_ATTR]: 'git@github.com:user/repository.git', + }); + const emptyStore = new MockStoreBuilder().build(); + renderComponent(emptyStore, searchParams, location); + + await jest.advanceTimersByTimeAsync(MIN_STEP_DURATION_MS); + + await waitFor(() => expect(mockOnNextStep).not.toHaveBeenCalled); + + expect(mockOpenOAuthPage).not.toHaveBeenCalled(); + expect(mockOnError).toHaveBeenCalledWith(expectAlertItem); + }); }); }); diff --git a/packages/dashboard-frontend/src/components/WorkspaceProgress/CreatingSteps/Fetch/Devfile/index.tsx b/packages/dashboard-frontend/src/components/WorkspaceProgress/CreatingSteps/Fetch/Devfile/index.tsx index 27e698157..4aca6e4ce 100644 --- a/packages/dashboard-frontend/src/components/WorkspaceProgress/CreatingSteps/Fetch/Devfile/index.tsx +++ b/packages/dashboard-frontend/src/components/WorkspaceProgress/CreatingSteps/Fetch/Devfile/index.tsx @@ -35,6 +35,7 @@ import { AlertItem } from '@/services/helpers/types'; import { isOAuthResponse, OAuthService } from '@/services/oauth'; import SessionStorageService, { SessionStorageKey } from '@/services/session-storage'; import { RootState } from '@/store'; +import { selectBranding } from '@/store/Branding'; import { factoryResolverActionCreators, selectFactoryResolver } from '@/store/FactoryResolver'; import { selectAllWorkspaces } from '@/store/Workspaces/selectors'; @@ -52,6 +53,13 @@ export class UnsupportedGitProviderError extends Error { } } +export class SSHPrivateRepositoryUrlError extends Error { + constructor(message: string) { + super(message); + this.name = 'UnsupportedGitProviderError'; + } +} + const RELOADS_LIMIT = 2; type ReloadsInfo = { [url: string]: number; @@ -179,6 +187,10 @@ class CreatingStepFetchDevfile extends ProgressStep { this.clearStepError(); } + protected handleOpenDocumentationPage(): void { + window.open(this.props.branding.docs.startingAWorkspaceFromAGitRepositoryURL, '_blank'); + } + protected handleTimeout(): void { const timeoutError = new Error( `Devfile hasn't been resolved in the last ${TIMEOUT_TO_RESOLVE_SEC} seconds.`, @@ -220,7 +232,11 @@ class CreatingStepFetchDevfile extends ProgressStep { errorMessage === 'Failed to fetch devfile' || errorMessage.startsWith('Could not reach devfile') ) { - throw new UnsupportedGitProviderError(errorMessage); + if (sourceUrl.startsWith('git@')) { + throw new SSHPrivateRepositoryUrlError(errorMessage); + } else { + throw new UnsupportedGitProviderError(errorMessage); + } } throw e; } @@ -365,6 +381,34 @@ class CreatingStepFetchDevfile extends ProgressStep { ], }; } + if (error instanceof SSHPrivateRepositoryUrlError) { + return { + key, + title: 'Warning', + variant: AlertVariant.warning, + children: ( + + ), + actionCallbacks: [ + { + title: 'Continue with default devfile', + callback: () => this.handleDefaultDevfile(key), + }, + { + title: 'Reload', + callback: () => this.handleRestart(key), + }, + { + title: 'Open Documentation page', + callback: () => this.handleOpenDocumentationPage(), + }, + ], + }; + } return { key, title: 'Failed to create the workspace', @@ -417,6 +461,7 @@ class CreatingStepFetchDevfile extends ProgressStep { const mapStateToProps = (state: RootState) => ({ allWorkspaces: selectAllWorkspaces(state), factoryResolver: selectFactoryResolver(state), + branding: selectBranding(state), }); const connector = connect(mapStateToProps, factoryResolverActionCreators, null, { diff --git a/packages/dashboard-frontend/src/services/bootstrap/branding.constant.ts b/packages/dashboard-frontend/src/services/bootstrap/branding.constant.ts index 7f41b4157..436bb0c8e 100644 --- a/packages/dashboard-frontend/src/services/bootstrap/branding.constant.ts +++ b/packages/dashboard-frontend/src/services/bootstrap/branding.constant.ts @@ -33,6 +33,7 @@ export type BrandingDocs = { faq?: string; storageTypes: string; webSocketTroubleshooting: string; + startingAWorkspaceFromAGitRepositoryURL: string; }; export type BrandingConfiguration = { @@ -83,6 +84,8 @@ export const BRANDING_DEFAULT: BrandingData = { 'https://www.eclipse.org/che/docs/stable/end-user-guide/url-parameter-for-the-workspace-storage/', webSocketTroubleshooting: 'https://www.eclipse.org/che/docs/stable/end-user-guide/troubleshooting-network-problems/', + startingAWorkspaceFromAGitRepositoryURL: + 'https://eclipse.dev/che/docs/stable/end-user-guide/starting-a-workspace-from-a-git-repository-url/', }, configuration: {}, };