import type { NavigationGuardNext, RouteLocation, RouteRecordRaw, RouteLocationRaw, RouteLocationNormalized, } from 'vue-router'; import { createRouter, createWebHistory, isNavigationFailure } from 'vue-router'; import { useExternalHooks } from '@/composables/useExternalHooks'; import { useSettingsStore } from '@/stores/settings.store'; import { useTemplatesStore } from '@/stores/templates.store'; import { useUIStore } from '@/stores/ui.store'; import { useSSOStore } from '@/stores/sso.store'; import { EnterpriseEditionFeature, VIEWS, EDITABLE_CANVAS_VIEWS } from '@/constants'; import { useTelemetry } from '@/composables/useTelemetry'; import { middleware } from '@/utils/rbac/middleware'; import type { RouterMiddleware } from '@/types/router'; import { initializeAuthenticatedFeatures, initializeCore } from '@/init'; import { tryToParseNumber } from '@/utils/typesUtils'; import { projectsRoutes } from '@/routes/projects.routes'; const ChangePasswordView = async () => await import('./views/ChangePasswordView.vue'); const ErrorView = async () => await import('./views/ErrorView.vue'); const ForgotMyPasswordView = async () => await import('./views/ForgotMyPasswordView.vue'); const MainHeader = async () => await import('@/components/MainHeader/MainHeader.vue'); const MainSidebar = async () => await import('@/components/MainSidebar.vue'); const CanvasChat = async () => await import('@/components/CanvasChat/CanvasChat.vue'); const NodeView = async () => await import('@/views/NodeViewSwitcher.vue'); const WorkflowExecutionsView = async () => await import('@/views/WorkflowExecutionsView.vue'); const WorkflowExecutionsLandingPage = async () => await import('@/components/executions/workflow/WorkflowExecutionsLandingPage.vue'); const WorkflowExecutionsPreview = async () => await import('@/components/executions/workflow/WorkflowExecutionsPreview.vue'); const SettingsView = async () => await import('./views/SettingsView.vue'); const SettingsLdapView = async () => await import('./views/SettingsLdapView.vue'); const SettingsPersonalView = async () => await import('./views/SettingsPersonalView.vue'); const SettingsUsersView = async () => await import('./views/SettingsUsersView.vue'); const SettingsCommunityNodesView = async () => await import('./views/SettingsCommunityNodesView.vue'); const SettingsApiView = async () => await import('./views/SettingsApiView.vue'); const SettingsLogStreamingView = async () => await import('./views/SettingsLogStreamingView.vue'); const SetupView = async () => await import('./views/SetupView.vue'); const SigninView = async () => await import('./views/SigninView.vue'); const SignupView = async () => await import('./views/SignupView.vue'); const TemplatesCollectionView = async () => await import('@/views/TemplatesCollectionView.vue'); const TemplatesWorkflowView = async () => await import('@/views/TemplatesWorkflowView.vue'); const SetupWorkflowFromTemplateView = async () => await import('@/views/SetupWorkflowFromTemplateView/SetupWorkflowFromTemplateView.vue'); const TemplatesSearchView = async () => await import('@/views/TemplatesSearchView.vue'); const VariablesView = async () => await import('@/views/VariablesView.vue'); const SettingsUsageAndPlan = async () => await import('./views/SettingsUsageAndPlan.vue'); const SettingsSso = async () => await import('./views/SettingsSso.vue'); const SignoutView = async () => await import('@/views/SignoutView.vue'); const SamlOnboarding = async () => await import('@/views/SamlOnboarding.vue'); const SettingsSourceControl = async () => await import('./views/SettingsSourceControl.vue'); const SettingsExternalSecrets = async () => await import('./views/SettingsExternalSecrets.vue'); const WorkerView = async () => await import('./views/WorkerView.vue'); const WorkflowHistory = async () => await import('@/views/WorkflowHistory.vue'); const WorkflowOnboardingView = async () => await import('@/views/WorkflowOnboardingView.vue'); const TestDefinitionListView = async () => await import('./views/TestDefinition/TestDefinitionListView.vue'); const TestDefinitionEditView = async () => await import('./views/TestDefinition/TestDefinitionEditView.vue'); function getTemplatesRedirect(defaultRedirect: VIEWS[keyof VIEWS]): { name: string } | false { const settingsStore = useSettingsStore(); const isTemplatesEnabled: boolean = settingsStore.isTemplatesEnabled; if (!isTemplatesEnabled) { return { name: `${defaultRedirect}` || VIEWS.NOT_FOUND }; } return false; } export const routes: RouteRecordRaw[] = [ { path: '/', redirect: '/home/workflows', meta: { middleware: ['authenticated'], }, }, { path: '/collections/:id', name: VIEWS.COLLECTION, components: { default: TemplatesCollectionView, sidebar: MainSidebar, }, meta: { templatesEnabled: true, telemetry: { getProperties(route: RouteLocation) { const templatesStore = useTemplatesStore(); return { collection_id: route.params.id, wf_template_repo_session_id: templatesStore.currentSessionId, }; }, }, getRedirect: getTemplatesRedirect, middleware: ['authenticated'], }, }, // Following two routes are kept in-app: // Single workflow view, used when a custom template host is set // Also, reachable directly from this URL { path: '/templates/:id', name: VIEWS.TEMPLATE, components: { default: TemplatesWorkflowView, sidebar: MainSidebar, }, meta: { templatesEnabled: true, getRedirect: getTemplatesRedirect, telemetry: { getProperties(route: RouteLocation) { const templatesStore = useTemplatesStore(); return { template_id: tryToParseNumber( Array.isArray(route.params.id) ? route.params.id[0] : route.params.id, ), wf_template_repo_session_id: templatesStore.currentSessionId, }; }, }, middleware: ['authenticated'], }, }, // Template setup view, this is the landing view for website users { path: '/templates/:id/setup', name: VIEWS.TEMPLATE_SETUP, components: { default: SetupWorkflowFromTemplateView, sidebar: MainSidebar, }, meta: { templatesEnabled: true, getRedirect: getTemplatesRedirect, telemetry: { getProperties(route: RouteLocation) { const templatesStore = useTemplatesStore(); return { template_id: tryToParseNumber( Array.isArray(route.params.id) ? route.params.id[0] : route.params.id, ), wf_template_repo_session_id: templatesStore.currentSessionId, }; }, }, middleware: ['authenticated'], }, }, { path: '/templates/', name: VIEWS.TEMPLATES, components: { default: TemplatesSearchView, sidebar: MainSidebar, }, meta: { templatesEnabled: true, getRedirect: getTemplatesRedirect, // Templates view remembers it's scroll position on back scrollOffset: 0, telemetry: { getProperties() { const templatesStore = useTemplatesStore(); return { wf_template_repo_session_id: templatesStore.currentSessionId, }; }, }, setScrollPosition(pos: number) { this.scrollOffset = pos; }, middleware: ['authenticated'], }, beforeEnter: (_to, _from, next) => { const templatesStore = useTemplatesStore(); if (!templatesStore.hasCustomTemplatesHost) { window.location.href = templatesStore.websiteTemplateRepositoryURL; } else { next(); } }, }, { path: '/variables', name: VIEWS.VARIABLES, components: { default: VariablesView, sidebar: MainSidebar, }, meta: { middleware: ['authenticated'] }, }, { path: '/workflow/:name/debug/:executionId', name: VIEWS.EXECUTION_DEBUG, components: { default: NodeView, header: MainHeader, sidebar: MainSidebar, }, meta: { nodeView: true, keepWorkflowAlive: true, middleware: ['authenticated', 'enterprise'], middlewareOptions: { enterprise: { feature: [EnterpriseEditionFeature.DebugInEditor], }, }, }, }, { path: '/workflow/:name/executions', name: VIEWS.WORKFLOW_EXECUTIONS, components: { default: WorkflowExecutionsView, header: MainHeader, sidebar: MainSidebar, }, meta: { keepWorkflowAlive: true, middleware: ['authenticated'], }, children: [ { path: '', name: VIEWS.EXECUTION_HOME, components: { executionPreview: WorkflowExecutionsLandingPage, }, meta: { keepWorkflowAlive: true, middleware: ['authenticated'], }, }, { path: ':executionId', name: VIEWS.EXECUTION_PREVIEW, components: { executionPreview: WorkflowExecutionsPreview, }, meta: { keepWorkflowAlive: true, middleware: ['authenticated'], }, }, ], }, { path: '/workflow/:name/evaluation', name: VIEWS.TEST_DEFINITION, meta: { keepWorkflowAlive: true, middleware: ['authenticated'], }, children: [ { path: '', name: VIEWS.TEST_DEFINITION, components: { default: TestDefinitionListView, header: MainHeader, sidebar: MainSidebar, }, meta: { keepWorkflowAlive: true, middleware: ['authenticated'], }, }, { path: 'new', name: VIEWS.NEW_TEST_DEFINITION, components: { default: TestDefinitionEditView, header: MainHeader, sidebar: MainSidebar, }, meta: { keepWorkflowAlive: true, middleware: ['authenticated'], }, }, { path: ':testId', name: VIEWS.TEST_DEFINITION_EDIT, components: { default: TestDefinitionEditView, header: MainHeader, sidebar: MainSidebar, }, meta: { keepWorkflowAlive: true, middleware: ['authenticated'], }, }, ], }, { path: '/workflow/:workflowId/history/:versionId?', name: VIEWS.WORKFLOW_HISTORY, components: { default: WorkflowHistory, sidebar: MainSidebar, }, meta: { middleware: ['authenticated', 'enterprise'], middlewareOptions: { enterprise: { feature: [EnterpriseEditionFeature.WorkflowHistory], }, }, }, }, { path: '/workflows/templates/:id', name: VIEWS.TEMPLATE_IMPORT, components: { default: NodeView, header: MainHeader, sidebar: MainSidebar, }, meta: { templatesEnabled: true, keepWorkflowAlive: true, getRedirect: getTemplatesRedirect, middleware: ['authenticated'], }, }, { path: '/workflows/onboarding/:id', name: VIEWS.WORKFLOW_ONBOARDING, components: { default: WorkflowOnboardingView, header: MainHeader, sidebar: MainSidebar, }, meta: { templatesEnabled: true, keepWorkflowAlive: true, getRedirect: () => getTemplatesRedirect(VIEWS.NEW_WORKFLOW), middleware: ['authenticated'], }, }, { path: '/workflow/new', name: VIEWS.NEW_WORKFLOW, components: { default: NodeView, header: MainHeader, sidebar: MainSidebar, footer: CanvasChat, }, meta: { nodeView: true, keepWorkflowAlive: true, middleware: ['authenticated'], }, }, { path: '/workflows/demo', name: VIEWS.DEMO, components: { default: NodeView, }, meta: { middleware: ['authenticated'], middlewareOptions: { authenticated: { bypass: () => { const settingsStore = useSettingsStore(); return settingsStore.isPreviewMode; }, }, }, }, }, { path: '/workflow/:name', name: VIEWS.WORKFLOW, components: { default: NodeView, header: MainHeader, sidebar: MainSidebar, footer: CanvasChat, }, meta: { nodeView: true, keepWorkflowAlive: true, middleware: ['authenticated'], }, }, { path: '/workflow', redirect: '/workflow/new', }, { path: '/signin', name: VIEWS.SIGNIN, components: { default: SigninView, }, meta: { telemetry: { pageCategory: 'auth', }, middleware: ['guest'], }, }, { path: '/signup', name: VIEWS.SIGNUP, components: { default: SignupView, }, meta: { telemetry: { pageCategory: 'auth', }, middleware: ['guest'], }, }, { path: '/signout', name: VIEWS.SIGNOUT, components: { default: SignoutView, }, meta: { telemetry: { pageCategory: 'auth', }, middleware: ['authenticated'], }, }, { path: '/setup', name: VIEWS.SETUP, components: { default: SetupView, }, meta: { middleware: ['defaultUser'], telemetry: { pageCategory: 'auth', }, }, }, { path: '/forgot-password', name: VIEWS.FORGOT_PASSWORD, components: { default: ForgotMyPasswordView, }, meta: { middleware: ['guest'], telemetry: { pageCategory: 'auth', }, }, }, { path: '/change-password', name: VIEWS.CHANGE_PASSWORD, components: { default: ChangePasswordView, }, meta: { middleware: ['guest'], telemetry: { pageCategory: 'auth', }, }, }, { path: '/settings', component: SettingsView, props: true, children: [ { path: 'usage', name: VIEWS.USAGE, components: { settingsView: SettingsUsageAndPlan, }, meta: { middleware: ['authenticated', 'custom'], middlewareOptions: { custom: () => { const settingsStore = useSettingsStore(); return !settingsStore.settings.hideUsagePage; }, }, telemetry: { pageCategory: 'settings', getProperties() { return { feature: 'usage', }; }, }, }, }, { path: 'personal', name: VIEWS.PERSONAL_SETTINGS, components: { settingsView: SettingsPersonalView, }, meta: { middleware: ['authenticated'], telemetry: { pageCategory: 'settings', getProperties() { return { feature: 'personal', }; }, }, }, }, { path: 'users', name: VIEWS.USERS_SETTINGS, components: { settingsView: SettingsUsersView, }, meta: { middleware: ['authenticated', 'rbac'], middlewareOptions: { rbac: { scope: ['user:create', 'user:update'], }, }, telemetry: { pageCategory: 'settings', getProperties() { return { feature: 'users', }; }, }, }, }, { path: 'api', name: VIEWS.API_SETTINGS, components: { settingsView: SettingsApiView, }, meta: { middleware: ['authenticated'], telemetry: { pageCategory: 'settings', getProperties() { return { feature: 'api', }; }, }, }, }, { path: 'environments', name: VIEWS.SOURCE_CONTROL, components: { settingsView: SettingsSourceControl, }, meta: { middleware: ['authenticated', 'rbac'], middlewareOptions: { rbac: { scope: 'sourceControl:manage', }, }, telemetry: { pageCategory: 'settings', getProperties() { return { feature: 'environments', }; }, }, }, }, { path: 'external-secrets', name: VIEWS.EXTERNAL_SECRETS_SETTINGS, components: { settingsView: SettingsExternalSecrets, }, meta: { middleware: ['authenticated', 'rbac'], middlewareOptions: { rbac: { scope: ['externalSecretsProvider:list', 'externalSecretsProvider:update'], }, }, telemetry: { pageCategory: 'settings', getProperties() { return { feature: 'external-secrets', }; }, }, }, }, { path: 'sso', name: VIEWS.SSO_SETTINGS, components: { settingsView: SettingsSso, }, meta: { middleware: ['authenticated', 'rbac'], middlewareOptions: { rbac: { scope: 'saml:manage', }, }, telemetry: { pageCategory: 'settings', getProperties() { return { feature: 'sso', }; }, }, }, }, { path: 'log-streaming', name: VIEWS.LOG_STREAMING_SETTINGS, components: { settingsView: SettingsLogStreamingView, }, meta: { middleware: ['authenticated', 'rbac'], middlewareOptions: { rbac: { scope: 'logStreaming:manage', }, }, telemetry: { pageCategory: 'settings', }, }, }, { path: 'workers', name: VIEWS.WORKER_VIEW, components: { settingsView: WorkerView, }, meta: { middleware: ['authenticated'], }, }, { path: 'community-nodes', name: VIEWS.COMMUNITY_NODES, components: { settingsView: SettingsCommunityNodesView, }, meta: { middleware: ['authenticated', 'rbac', 'custom'], middlewareOptions: { rbac: { scope: ['communityPackage:list', 'communityPackage:update'], }, custom: () => { const settingsStore = useSettingsStore(); return settingsStore.isCommunityNodesFeatureEnabled; }, }, telemetry: { pageCategory: 'settings', }, }, }, { path: 'ldap', name: VIEWS.LDAP_SETTINGS, components: { settingsView: SettingsLdapView, }, meta: { middleware: ['authenticated', 'rbac'], middlewareOptions: { rbac: { scope: 'ldap:manage', }, }, }, }, ], }, { path: '/saml/onboarding', name: VIEWS.SAML_ONBOARDING, components: { default: SamlOnboarding, }, meta: { middleware: ['authenticated', 'custom'], middlewareOptions: { custom: () => { const settingsStore = useSettingsStore(); const ssoStore = useSSOStore(); return ssoStore.isEnterpriseSamlEnabled && !settingsStore.isCloudDeployment; }, }, telemetry: { pageCategory: 'auth', }, }, }, ...projectsRoutes, { path: '/:pathMatch(.*)*', name: VIEWS.NOT_FOUND, component: ErrorView, props: { messageKey: 'error.pageNotFound', errorCode: 404, redirectTextKey: 'error.goBack', redirectPage: VIEWS.HOMEPAGE, }, meta: { nodeView: true, telemetry: { disabled: true, }, }, }, ]; function withCanvasReadOnlyMeta(route: RouteRecordRaw) { if (!route.meta) { route.meta = {}; } route.meta.readOnlyCanvas = !EDITABLE_CANVAS_VIEWS.includes((route?.name ?? '') as VIEWS); if (route.children) { route.children = route.children.map(withCanvasReadOnlyMeta); } return route; } const router = createRouter({ history: createWebHistory(import.meta.env.DEV ? '/' : (window.BASE_PATH ?? '/')), scrollBehavior(to: RouteLocationNormalized, _, savedPosition) { // saved position == null means the page is NOT visited from history (back button) if (savedPosition === null && to.name === VIEWS.TEMPLATES && to.meta?.setScrollPosition) { // for templates view, reset scroll position in this case to.meta.setScrollPosition(0); } }, routes: routes.map(withCanvasReadOnlyMeta), }); router.beforeEach(async (to: RouteLocationNormalized, from, next) => { try { /** * Initialize application core * This step executes before first route is loaded and is required for permission checks */ await initializeCore(); await initializeAuthenticatedFeatures(); /** * Redirect to setup page. User should be redirected to this only once */ const settingsStore = useSettingsStore(); if (settingsStore.showSetupPage) { if (to.name === VIEWS.SETUP) { return next(); } return next({ name: VIEWS.SETUP }); } /** * Verify user permissions for current route */ const routeMiddleware = to.meta?.middleware ?? []; const routeMiddlewareOptions = to.meta?.middlewareOptions ?? {}; for (const middlewareName of routeMiddleware) { let nextCalled = false; const middlewareNext = ((location: RouteLocationRaw): void => { next(location); nextCalled = true; }) as NavigationGuardNext; const middlewareOptions = routeMiddlewareOptions[middlewareName]; const middlewareFn = middleware[middlewareName] as RouterMiddleware; await middlewareFn(to, from, middlewareNext, middlewareOptions); if (nextCalled) { return; } } return next(); } catch (failure) { if (isNavigationFailure(failure)) { console.log(failure); } else { console.error(failure); } } }); router.afterEach((to, from) => { try { const telemetry = useTelemetry(); const uiStore = useUIStore(); const templatesStore = useTemplatesStore(); /** * Run external hooks */ void useExternalHooks().run('main.routeChange', { from, to }); /** * Track current view for telemetry */ uiStore.currentView = (to.name as string) ?? ''; if (to.meta?.templatesEnabled) { templatesStore.setSessionId(); } else { templatesStore.resetSessionId(); // reset telemetry session id when user leaves template pages } telemetry.page(to); } catch (failure) { if (isNavigationFailure(failure)) { console.log(failure); } else { console.error(failure); } } }); export default router;