mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-10 12:27:31 -08:00
841 lines
20 KiB
TypeScript
841 lines
20 KiB
TypeScript
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 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 SettingsFakeDoorView = async () => await import('./views/SettingsFakeDoorView.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 ExecutionsView = async () => await import('@/views/ExecutionsView.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');
|
|
|
|
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: '/executions',
|
|
name: VIEWS.EXECUTIONS,
|
|
components: {
|
|
default: ExecutionsView,
|
|
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/: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,
|
|
},
|
|
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,
|
|
},
|
|
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', 'custom'],
|
|
middlewareOptions: {
|
|
custom: () => {
|
|
const settingsStore = useSettingsStore();
|
|
return !settingsStore.isDesktopDeployment;
|
|
},
|
|
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: 'coming-soon/:featureId',
|
|
name: VIEWS.FAKE_DOOR,
|
|
components: {
|
|
settingsView: SettingsFakeDoorView,
|
|
},
|
|
meta: {
|
|
middleware: ['authenticated'],
|
|
telemetry: {
|
|
pageCategory: 'settings',
|
|
getProperties(route: RouteLocation) {
|
|
return {
|
|
feature: route.params.featureId,
|
|
};
|
|
},
|
|
},
|
|
},
|
|
},
|
|
{
|
|
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 &&
|
|
!settingsStore.isDesktopDeployment
|
|
);
|
|
},
|
|
},
|
|
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<unknown>;
|
|
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;
|