mirror of https://github.com/portainer/portainer
483 lines
13 KiB
JavaScript
483 lines
13 KiB
JavaScript
import _ from 'lodash-es';
|
|
|
|
import featureFlagModule from '@/react/portainer/feature-flags';
|
|
|
|
import './rbac';
|
|
|
|
import componentsModule from './components';
|
|
import settingsModule from './settings';
|
|
import userActivityModule from './user-activity';
|
|
import servicesModule from './services';
|
|
import { reactModule } from './react';
|
|
import { sidebarModule } from './react/views/sidebar';
|
|
import environmentsModule from './environments';
|
|
import { helpersModule } from './helpers';
|
|
import { AXIOS_UNAUTHENTICATED } from './services/axios';
|
|
|
|
async function initAuthentication(authManager, Authentication, $rootScope, $state) {
|
|
authManager.checkAuthOnRefresh();
|
|
|
|
function handleUnauthenticated(data, performReload) {
|
|
if (!_.includes(data.config.url, '/v2/') && !_.includes(data.config.url, '/api/v4/') && isTransitionRequiresAuthentication($state.transition)) {
|
|
$state.go('portainer.logout', { error: 'Your session has expired' });
|
|
if (performReload) {
|
|
window.location.reload();
|
|
}
|
|
}
|
|
}
|
|
|
|
// The unauthenticated event is broadcasted by the jwtInterceptor when
|
|
// hitting a 401. We're using this instead of the usual combination of
|
|
// authManager.redirectWhenUnauthenticated() + unauthenticatedRedirector
|
|
// to have more controls on which URL should trigger the unauthenticated state.
|
|
$rootScope.$on('unauthenticated', function (event, data) {
|
|
handleUnauthenticated(data, true);
|
|
});
|
|
|
|
// the AXIOS_UNAUTHENTICATED event is emitted by axios when a request returns with a 401 code
|
|
// the event contains the entire AxiosError in detail.err
|
|
window.addEventListener(AXIOS_UNAUTHENTICATED, (event) => {
|
|
const data = event.detail.err;
|
|
handleUnauthenticated(data);
|
|
});
|
|
|
|
return await Authentication.init();
|
|
}
|
|
|
|
angular
|
|
.module('portainer.app', [
|
|
'portainer.oauth',
|
|
'portainer.rbac',
|
|
componentsModule,
|
|
settingsModule,
|
|
featureFlagModule,
|
|
userActivityModule,
|
|
'portainer.shared.datatable',
|
|
servicesModule,
|
|
reactModule,
|
|
sidebarModule,
|
|
environmentsModule,
|
|
helpersModule,
|
|
])
|
|
.config([
|
|
'$stateRegistryProvider',
|
|
function ($stateRegistryProvider) {
|
|
var root = {
|
|
name: 'root',
|
|
abstract: true,
|
|
onEnter: /* @ngInject */ function onEnter($async, StateManager, Authentication, Notifications, authManager, $rootScope, $state) {
|
|
return $async(async () => {
|
|
const appState = StateManager.getState();
|
|
if (!appState.loading) {
|
|
return;
|
|
}
|
|
try {
|
|
const loggedIn = await initAuthentication(authManager, Authentication, $rootScope, $state);
|
|
await StateManager.initialize();
|
|
if (!loggedIn && isTransitionRequiresAuthentication($state.transition)) {
|
|
$state.go('portainer.logout');
|
|
return Promise.reject('Unauthenticated');
|
|
}
|
|
} catch (err) {
|
|
Notifications.error('Failure', err, 'Unable to retrieve application settings');
|
|
throw err;
|
|
}
|
|
});
|
|
},
|
|
views: {
|
|
'sidebar@': {
|
|
component: 'sidebar',
|
|
},
|
|
},
|
|
};
|
|
|
|
var endpointRoot = {
|
|
name: 'endpoint',
|
|
url: '/:endpointId',
|
|
parent: 'root',
|
|
abstract: true,
|
|
resolve: {
|
|
endpoint: /* @ngInject */ function endpoint($async, $state, $transition$, EndpointProvider, EndpointService, Notifications) {
|
|
return $async(async () => {
|
|
try {
|
|
const endpointId = +$transition$.params().endpointId;
|
|
|
|
const endpoint = await EndpointService.endpoint(endpointId);
|
|
if ((endpoint.Type === 4 || endpoint.Type === 7) && !endpoint.EdgeID) {
|
|
$state.go('portainer.endpoints.endpoint', { id: endpoint.Id });
|
|
return;
|
|
}
|
|
|
|
EndpointProvider.setCurrentEndpoint(endpoint);
|
|
|
|
return endpoint;
|
|
} catch (e) {
|
|
Notifications.error('Failed loading environment', e);
|
|
$state.go('portainer.home', {}, { reload: true });
|
|
return;
|
|
}
|
|
});
|
|
},
|
|
},
|
|
};
|
|
|
|
var portainer = {
|
|
name: 'portainer',
|
|
parent: 'root',
|
|
abstract: true,
|
|
};
|
|
|
|
var account = {
|
|
name: 'portainer.account',
|
|
url: '/account',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/account/account.html',
|
|
controller: 'AccountController',
|
|
},
|
|
},
|
|
};
|
|
|
|
const tokenCreation = {
|
|
name: 'portainer.account.new-access-token',
|
|
url: '/tokens/new',
|
|
views: {
|
|
'content@': {
|
|
component: 'createUserAccessToken',
|
|
},
|
|
},
|
|
};
|
|
|
|
var authentication = {
|
|
name: 'portainer.auth',
|
|
url: '/auth',
|
|
params: {
|
|
reload: false,
|
|
},
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/auth/auth.html',
|
|
controller: 'AuthenticationController',
|
|
controllerAs: 'ctrl',
|
|
},
|
|
'sidebar@': {},
|
|
},
|
|
};
|
|
|
|
const logout = {
|
|
name: 'portainer.logout',
|
|
url: '/logout',
|
|
params: {
|
|
error: '',
|
|
},
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/logout/logout.html',
|
|
controller: 'LogoutController',
|
|
controllerAs: 'ctrl',
|
|
},
|
|
'sidebar@': {},
|
|
},
|
|
};
|
|
|
|
var endpoints = {
|
|
name: 'portainer.endpoints',
|
|
url: '/endpoints',
|
|
views: {
|
|
'content@': {
|
|
component: 'environmentsListView',
|
|
},
|
|
},
|
|
};
|
|
|
|
var endpoint = {
|
|
name: 'portainer.endpoints.endpoint',
|
|
url: '/:id?redirectTo',
|
|
params: {
|
|
redirectTo: '',
|
|
},
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/endpoints/edit/endpoint.html',
|
|
controller: 'EndpointController',
|
|
},
|
|
},
|
|
};
|
|
|
|
var deviceImport = {
|
|
name: 'portainer.endpoints.importDevice',
|
|
url: '/device',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/devices/import/importDevice.html',
|
|
controller: 'ImportDeviceController',
|
|
},
|
|
},
|
|
};
|
|
|
|
const edgeAutoCreateScript = {
|
|
name: 'portainer.endpoints.edgeAutoCreateScript',
|
|
url: '/aeec',
|
|
views: {
|
|
'content@': {
|
|
component: 'edgeAutoCreateScriptView',
|
|
},
|
|
},
|
|
};
|
|
|
|
var addFDOProfile = {
|
|
name: 'portainer.endpoints.profile',
|
|
url: '/profile',
|
|
views: {
|
|
'content@': {
|
|
component: 'addProfileView',
|
|
},
|
|
},
|
|
};
|
|
|
|
var editFDOProfile = {
|
|
name: 'portainer.endpoints.profile.edit',
|
|
url: '/:id',
|
|
views: {
|
|
'content@': {
|
|
component: 'editProfileView',
|
|
},
|
|
},
|
|
};
|
|
|
|
var endpointAccess = {
|
|
name: 'portainer.endpoints.endpoint.access',
|
|
url: '/access',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/endpoints/access/endpointAccess.html',
|
|
controller: 'EndpointAccessController',
|
|
controllerAs: 'ctrl',
|
|
},
|
|
},
|
|
};
|
|
|
|
var endpointKVM = {
|
|
name: 'portainer.endpoints.endpoint.kvm',
|
|
url: '/kvm?deviceId&deviceName',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/endpoints/kvm/endpointKVM.html',
|
|
controller: 'EndpointKVMController',
|
|
},
|
|
},
|
|
};
|
|
|
|
var groups = {
|
|
name: 'portainer.groups',
|
|
url: '/groups',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/groups/groups.html',
|
|
controller: 'GroupsController',
|
|
},
|
|
},
|
|
};
|
|
|
|
var group = {
|
|
name: 'portainer.groups.group',
|
|
url: '/:id',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/groups/edit/group.html',
|
|
controller: 'GroupController',
|
|
},
|
|
},
|
|
};
|
|
|
|
var groupCreation = {
|
|
name: 'portainer.groups.new',
|
|
url: '/new',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/groups/create/creategroup.html',
|
|
controller: 'CreateGroupController',
|
|
},
|
|
},
|
|
};
|
|
|
|
var groupAccess = {
|
|
name: 'portainer.groups.group.access',
|
|
url: '/access',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/groups/access/groupAccess.html',
|
|
controller: 'GroupAccessController',
|
|
},
|
|
},
|
|
};
|
|
|
|
var home = {
|
|
name: 'portainer.home',
|
|
url: '/home?redirect&environmentId&environmentName&route',
|
|
views: {
|
|
'content@': {
|
|
component: 'homeView',
|
|
},
|
|
},
|
|
};
|
|
|
|
var init = {
|
|
name: 'portainer.init',
|
|
abstract: true,
|
|
url: '/init',
|
|
views: {
|
|
'sidebar@': {},
|
|
},
|
|
};
|
|
|
|
var initAdmin = {
|
|
name: 'portainer.init.admin',
|
|
url: '/admin',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/init/admin/initAdmin.html',
|
|
controller: 'InitAdminController',
|
|
},
|
|
},
|
|
};
|
|
|
|
var registries = {
|
|
name: 'portainer.registries',
|
|
url: '/registries',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/registries/registries.html',
|
|
controller: 'RegistriesController',
|
|
},
|
|
},
|
|
};
|
|
|
|
var registry = {
|
|
name: 'portainer.registries.registry',
|
|
url: '/:id',
|
|
views: {
|
|
'content@': {
|
|
component: 'editRegistry',
|
|
},
|
|
},
|
|
};
|
|
|
|
const registryCreation = {
|
|
name: 'portainer.registries.new',
|
|
url: '/new',
|
|
views: {
|
|
'content@': {
|
|
component: 'createRegistry',
|
|
},
|
|
},
|
|
};
|
|
|
|
var settings = {
|
|
name: 'portainer.settings',
|
|
url: '/settings',
|
|
views: {
|
|
'content@': {
|
|
component: 'settingsView',
|
|
},
|
|
},
|
|
};
|
|
|
|
var settingsAuthentication = {
|
|
name: 'portainer.settings.authentication',
|
|
url: '/auth',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/settings/authentication/settingsAuthentication.html',
|
|
controller: 'SettingsAuthenticationController',
|
|
},
|
|
},
|
|
};
|
|
|
|
var settingsEdgeCompute = {
|
|
name: 'portainer.settings.edgeCompute',
|
|
url: '/edge',
|
|
views: {
|
|
'content@': {
|
|
component: 'settingsEdgeComputeView',
|
|
},
|
|
},
|
|
};
|
|
|
|
var tags = {
|
|
name: 'portainer.tags',
|
|
url: '/tags',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/tags/tags.html',
|
|
controller: 'TagsController',
|
|
},
|
|
},
|
|
};
|
|
|
|
var users = {
|
|
name: 'portainer.users',
|
|
url: '/users',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/users/users.html',
|
|
controller: 'UsersController',
|
|
},
|
|
},
|
|
};
|
|
|
|
var user = {
|
|
name: 'portainer.users.user',
|
|
url: '/:id',
|
|
views: {
|
|
'content@': {
|
|
templateUrl: './views/users/edit/user.html',
|
|
controller: 'UserController',
|
|
},
|
|
},
|
|
};
|
|
|
|
$stateRegistryProvider.register(root);
|
|
$stateRegistryProvider.register(endpointRoot);
|
|
$stateRegistryProvider.register(portainer);
|
|
$stateRegistryProvider.register(account);
|
|
$stateRegistryProvider.register(tokenCreation);
|
|
$stateRegistryProvider.register(authentication);
|
|
$stateRegistryProvider.register(logout);
|
|
$stateRegistryProvider.register(endpoints);
|
|
$stateRegistryProvider.register(endpoint);
|
|
$stateRegistryProvider.register(endpointAccess);
|
|
$stateRegistryProvider.register(endpointKVM);
|
|
$stateRegistryProvider.register(edgeAutoCreateScript);
|
|
$stateRegistryProvider.register(deviceImport);
|
|
$stateRegistryProvider.register(addFDOProfile);
|
|
$stateRegistryProvider.register(editFDOProfile);
|
|
$stateRegistryProvider.register(groups);
|
|
$stateRegistryProvider.register(group);
|
|
$stateRegistryProvider.register(groupAccess);
|
|
$stateRegistryProvider.register(groupCreation);
|
|
$stateRegistryProvider.register(home);
|
|
$stateRegistryProvider.register(init);
|
|
$stateRegistryProvider.register(initAdmin);
|
|
$stateRegistryProvider.register(registries);
|
|
$stateRegistryProvider.register(registry);
|
|
$stateRegistryProvider.register(registryCreation);
|
|
$stateRegistryProvider.register(settings);
|
|
$stateRegistryProvider.register(settingsAuthentication);
|
|
$stateRegistryProvider.register(settingsEdgeCompute);
|
|
$stateRegistryProvider.register(tags);
|
|
$stateRegistryProvider.register(users);
|
|
$stateRegistryProvider.register(user);
|
|
},
|
|
]);
|
|
|
|
function isTransitionRequiresAuthentication(transition) {
|
|
const UNAUTHENTICATED_ROUTES = ['portainer.logout', 'portainer.auth'];
|
|
if (!transition) {
|
|
return true;
|
|
}
|
|
const nextTransition = transition && transition.to();
|
|
const nextTransitionName = nextTransition ? nextTransition.name : '';
|
|
return !UNAUTHENTICATED_ROUTES.some((route) => nextTransitionName.startsWith(route));
|
|
}
|