diff --git a/app/gui/src/dashboard/assets/cross.svg b/app/gui/src/dashboard/assets/cross.svg index 3190ad3d1e04..a9e60f676baa 100644 --- a/app/gui/src/dashboard/assets/cross.svg +++ b/app/gui/src/dashboard/assets/cross.svg @@ -1,4 +1,4 @@ - - + + \ No newline at end of file diff --git a/app/gui/src/dashboard/assets/cross2.svg b/app/gui/src/dashboard/assets/cross2.svg index 920ef74ad3d0..0dc9486b155b 100644 --- a/app/gui/src/dashboard/assets/cross2.svg +++ b/app/gui/src/dashboard/assets/cross2.svg @@ -1,7 +1,7 @@ - - + + \ No newline at end of file diff --git a/app/gui/src/dashboard/assets/error_filled.svg b/app/gui/src/dashboard/assets/error_filled.svg new file mode 100644 index 000000000000..7012571bae24 --- /dev/null +++ b/app/gui/src/dashboard/assets/error_filled.svg @@ -0,0 +1,3 @@ + + + diff --git a/app/gui/src/dashboard/assets/error_outline.svg b/app/gui/src/dashboard/assets/error_outline.svg new file mode 100644 index 000000000000..550619be3a84 --- /dev/null +++ b/app/gui/src/dashboard/assets/error_outline.svg @@ -0,0 +1,3 @@ + + + diff --git a/app/gui/src/dashboard/assets/warning.svg b/app/gui/src/dashboard/assets/warning.svg new file mode 100644 index 000000000000..ff83234b968b --- /dev/null +++ b/app/gui/src/dashboard/assets/warning.svg @@ -0,0 +1,3 @@ + + + diff --git a/app/gui/src/dashboard/hooks/projectHooks.ts b/app/gui/src/dashboard/hooks/projectHooks.ts index 6f4fac4a1f9a..2075edc58762 100644 --- a/app/gui/src/dashboard/hooks/projectHooks.ts +++ b/app/gui/src/dashboard/hooks/projectHooks.ts @@ -81,6 +81,51 @@ function useSetProjectAsset() { ) } +export const OPENING_PROJECT_STATES = new Set([ + backendModule.ProjectState.provisioned, + backendModule.ProjectState.scheduled, + backendModule.ProjectState.openInProgress, + backendModule.ProjectState.closing, +]) +export const OPENED_PROJECT_STATES = new Set([backendModule.ProjectState.opened]) +export const CLOSED_PROJECT_STATES = new Set([backendModule.ProjectState.closed]) +export const STATIC_PROJECT_STATES = new Set([ + backendModule.ProjectState.opened, + backendModule.ProjectState.closed, +]) +export const CREATED_PROJECT_STATES = new Set([ + backendModule.ProjectState.created, + backendModule.ProjectState.new, +]) + +/** Stale time for local projects, set to 10 seconds. */ +// eslint-disable-next-line @typescript-eslint/no-magic-numbers +export const LOCAL_PROJECT_OPEN_TIMEOUT_MS = 10 * 1_000 +/** Stale time for cloud projects, set to 5 minutes. */ +// eslint-disable-next-line @typescript-eslint/no-magic-numbers +export const CLOUD_PROJECT_OPEN_TIMEOUT_MS = 5 * 60 * 1_000 + +/** + * Get the timeout based on the backend type. + * @param backendType - The backend type. + * @throws If the backend type is not supported. + * @returns The timeout in milliseconds. + */ +export function getTimeoutBasedOnTheBackendType(backendType: backendModule.BackendType) { + switch (backendType) { + case backendModule.BackendType.local: { + return LOCAL_PROJECT_OPEN_TIMEOUT_MS + } + case backendModule.BackendType.remote: { + return CLOUD_PROJECT_OPEN_TIMEOUT_MS + } + + default: { + throw new Error('Unsupported backend type') + } + } +} + /** Project status query. */ export function createGetProjectDetailsQuery(options: CreateOpenedProjectQueryOptions) { const { assetId, parentId, backend } = options @@ -90,22 +135,19 @@ export function createGetProjectDetailsQuery(options: CreateOpenedProjectQueryOp return reactQuery.queryOptions({ queryKey: createGetProjectDetailsQuery.getQueryKey(assetId), queryFn: () => backend.getProjectDetails(assetId, parentId), - meta: { persist: false }, refetchIntervalInBackground: true, refetchOnWindowFocus: true, refetchOnMount: true, networkMode: backend.type === backendModule.BackendType.remote ? 'online' : 'always', - refetchInterval: ({ state }): number | false => { - const staticStates = [backendModule.ProjectState.opened, backendModule.ProjectState.closed] + meta: { persist: false }, + refetchInterval: (query): number | false => { + const { state } = query + + const staticStates = STATIC_PROJECT_STATES - const openingStates = [ - backendModule.ProjectState.provisioned, - backendModule.ProjectState.scheduled, - backendModule.ProjectState.openInProgress, - backendModule.ProjectState.closing, - ] + const openingStates = OPENING_PROJECT_STATES - const createdStates = [backendModule.ProjectState.created, backendModule.ProjectState.new] + const createdStates = CREATED_PROJECT_STATES if (state.status === 'error') { return false @@ -118,28 +160,28 @@ export function createGetProjectDetailsQuery(options: CreateOpenedProjectQueryOp const currentState = state.data.state.type if (isLocal) { - if (createdStates.includes(currentState)) { + if (createdStates.has(currentState)) { return LOCAL_OPENING_INTERVAL_MS } - if (staticStates.includes(state.data.state.type)) { + if (staticStates.has(state.data.state.type)) { return OPENED_INTERVAL_MS } - if (openingStates.includes(state.data.state.type)) { + if (openingStates.has(state.data.state.type)) { return LOCAL_OPENING_INTERVAL_MS } } - if (createdStates.includes(currentState)) { + if (createdStates.has(currentState)) { return CLOUD_OPENING_INTERVAL_MS } // Cloud project - if (staticStates.includes(state.data.state.type)) { + if (staticStates.has(state.data.state.type)) { return OPENED_INTERVAL_MS } - if (openingStates.includes(state.data.state.type)) { + if (openingStates.has(state.data.state.type)) { return CLOUD_OPENING_INTERVAL_MS } diff --git a/app/gui/src/dashboard/hooks/timeoutHooks.ts b/app/gui/src/dashboard/hooks/timeoutHooks.ts new file mode 100644 index 000000000000..5c48968559cb --- /dev/null +++ b/app/gui/src/dashboard/hooks/timeoutHooks.ts @@ -0,0 +1,124 @@ +/** + * @file Timeout related hooks. + */ +import { useEffect, useRef, useState, type DependencyList } from 'react' +import { useEventCallback } from './eventCallbackHooks' +import { useUnmount } from './unmountHooks' + +/** + * Options for {@link useTimeoutCallback}. + */ +export interface UseTimeoutCallbackOptions { + /** + * Callback to execute after the timeout. + */ + readonly callback: () => void + /** + * Timeout in milliseconds. + */ + readonly ms: number + /** + * Dependencies for {@link useEventCallback}. + * Reset the timeout when the dependencies change. + */ + readonly deps?: DependencyList + /** + * Whether the timeout is disabled. + */ + readonly isDisabled?: boolean +} + +const STABLE_DEPS_ARRAY: DependencyList = [] + +/** + * Hook that executes a callback after a timeout. + */ +export function useTimeoutCallback(options: UseTimeoutCallbackOptions) { + const { callback, ms, deps = STABLE_DEPS_ARRAY, isDisabled = false } = options + + const timeoutRef = useRef | null>(null) + const stableCallback = useEventCallback(callback) + + /** + * Restarts the timer. + */ + const restartTimer = useEventCallback(() => { + stopTimer() + startTimer() + }) + + /** + * Starts the timer. + */ + const startTimer = useEventCallback(() => { + stopTimer() + timeoutRef.current = setTimeout(stableCallback, ms) + }) + + /** + * Stops the timer. + */ + const stopTimer = useEventCallback(() => { + if (timeoutRef.current != null) { + clearTimeout(timeoutRef.current) + timeoutRef.current = null + } + }) + + useEffect(() => { + if (isDisabled) { + return + } + + startTimer() + + return () => { + stopTimer() + } + // There is no way to enable compiler, but it's not needed here + // as everything is stable. + // eslint-disable-next-line react-compiler/react-compiler + // eslint-disable-next-line react-hooks/exhaustive-deps + }, [ms, isDisabled, ...deps]) + + useUnmount(() => { + stopTimer() + }) + + return [restartTimer, stopTimer, startTimer] as const +} + +/** + * Hook that returns a boolean indicating whether the timeout has expired. + */ +export function useTimeout(params: Pick) { + const { ms, deps = STABLE_DEPS_ARRAY } = params + + /** + * Get the default value for the timeout. + */ + const getDefaultValue = useEventCallback(() => { + return ms === 0 + }) + + const [isTimeout, setIsTimeout] = useState(getDefaultValue) + + const [restartTimer] = useTimeoutCallback({ + callback: () => { + setIsTimeout(true) + }, + ms, + deps, + isDisabled: false, + }) + + /** + * Resets the timeout and restarts it. + */ + const restart = useEventCallback(() => { + setIsTimeout(getDefaultValue) + restartTimer() + }) + + return [isTimeout, restart] as const +} diff --git a/app/gui/src/dashboard/layouts/Editor.tsx b/app/gui/src/dashboard/layouts/Editor.tsx index dc41ac689926..073a28951a89 100644 --- a/app/gui/src/dashboard/layouts/Editor.tsx +++ b/app/gui/src/dashboard/layouts/Editor.tsx @@ -20,6 +20,7 @@ import * as backendModule from '#/services/Backend' import { useEventCallback } from '#/hooks/eventCallbackHooks' import * as twMerge from '#/utilities/tailwindMerge' +import { useTimeoutCallback } from '../hooks/timeoutHooks' // ==================== // === StringConfig === @@ -91,9 +92,20 @@ function Editor(props: EditorProps) { backend, }) - const projectQuery = reactQuery.useQuery(projectStatusQuery) + const queryClient = reactQuery.useQueryClient() - const isProjectClosed = projectQuery.data?.state.type === backendModule.ProjectState.closed + const projectQuery = reactQuery.useSuspenseQuery({ + ...projectStatusQuery, + select: (data) => { + const isOpeningProject = projectHooks.OPENING_PROJECT_STATES.has(data.state.type) + const isProjectClosed = projectHooks.CLOSED_PROJECT_STATES.has(data.state.type) + + return { ...data, isOpeningProject, isProjectClosed } + }, + }) + + const isProjectClosed = projectQuery.data.isProjectClosed + const isOpeningProject = projectQuery.data.isOpeningProject React.useEffect(() => { if (isProjectClosed) { @@ -101,12 +113,28 @@ function Editor(props: EditorProps) { } }, [isProjectClosed, startProject, project]) + useTimeoutCallback({ + callback: () => { + const queryState = queryClient.getQueryCache().find({ queryKey: projectStatusQuery.queryKey }) + + queryState?.setState({ + error: new Error('Timeout opening the project'), + status: 'error', + }) + }, + ms: projectHooks.getTimeoutBasedOnTheBackendType(backend.type), + deps: [], + isDisabled: !isOpeningProject || projectQuery.isError, + }) + if (isOpeningFailed) { return ( { - startProject(project) + if (isProjectClosed) { + startProject(project) + } }} /> ) @@ -128,20 +156,17 @@ function Editor(props: EditorProps) { /> ) - case projectQuery.isLoading || - projectQuery.data?.state.type !== backendModule.ProjectState.opened: + case isOpeningProject: return default: return ( - - - + ) } @@ -178,9 +203,7 @@ function EditorInternal(props: EditorInternalProps) { ) React.useEffect(() => { - if (hidden) { - return - } else { + if (!hidden) { return gtagHooks.gtagOpenCloseCallback(gtagEvent, 'open_workflow', 'close_workflow') } }, [hidden, gtagEvent]) diff --git a/app/gui/src/dashboard/layouts/TabBar.tsx b/app/gui/src/dashboard/layouts/TabBar.tsx index 422b3977db2e..00c48ebc5857 100644 --- a/app/gui/src/dashboard/layouts/TabBar.tsx +++ b/app/gui/src/dashboard/layouts/TabBar.tsx @@ -5,8 +5,8 @@ import * as reactQuery from '@tanstack/react-query' import type * as text from 'enso-common/src/text' +import ErrorFilledIcon from '#/assets/warning.svg' import * as projectHooks from '#/hooks/projectHooks' - import type { LaunchedProject } from '#/providers/ProjectsProvider' import * as textProvider from '#/providers/TextProvider' @@ -19,7 +19,6 @@ import { AnimatedBackground } from '#/components/AnimatedBackground' import { useEventCallback } from '#/hooks/eventCallbackHooks' import { useBackendForProjectType } from '#/providers/BackendProvider' import { useInputBindings } from '#/providers/InputBindingsProvider' -import { ProjectState } from '#/services/Backend' import * as sanitizedEventTargets from '#/utilities/sanitizedEventTargets' import * as tailwindMerge from '#/utilities/tailwindMerge' import { motion } from 'framer-motion' @@ -163,13 +162,17 @@ export function ProjectTab(props: ProjectTabProps) { onClose?.(project) }) - const { data: isOpened, isSuccess } = reactQuery.useQuery({ + const { + data: isOpened, + isSuccess, + isError, + } = reactQuery.useQuery({ ...projectHooks.createGetProjectDetailsQuery({ assetId: project.id, parentId: project.parentId, backend, }), - select: (data) => data.state.type === ProjectState.opened, + select: (data) => projectHooks.OPENED_PROJECT_STATES.has(data.state.type), }) const isReady = isSuccess && isOpened @@ -187,7 +190,17 @@ export function ProjectTab(props: ProjectTabProps) { } }, [isReady]) - const icon = isReady ? iconRaw : SPINNER + const icon = (() => { + if (isReady) { + return iconRaw + } + + if (isError) { + return + } + + return SPINNER + })() return } diff --git a/app/gui/src/dashboard/pages/dashboard/DashboardTabPanels.tsx b/app/gui/src/dashboard/pages/dashboard/DashboardTabPanels.tsx index 8ddf9130a14c..ecf77ccc5704 100644 --- a/app/gui/src/dashboard/pages/dashboard/DashboardTabPanels.tsx +++ b/app/gui/src/dashboard/pages/dashboard/DashboardTabPanels.tsx @@ -102,11 +102,11 @@ export function DashboardTabPanels(props: DashboardTabPanelsProps) { return ( {(tabPanelProps) => ( - - - - - + + + {tabPanelProps.children} + + )} )