X-Git-Url: https://git.arvados.org/arvados-workbench2.git/blobdiff_plain/e5ff916017b1f215dc1d0ae5b82cd3d7e669a5fc..f9dde5c781766b8be71d43d0f031c201a0edcfbb:/src/services/auth-service/auth-service.ts diff --git a/src/services/auth-service/auth-service.ts b/src/services/auth-service/auth-service.ts index 5a08d3b5..8601e208 100644 --- a/src/services/auth-service/auth-service.ts +++ b/src/services/auth-service/auth-service.ts @@ -2,16 +2,41 @@ // // SPDX-License-Identifier: AGPL-3.0 -import Axios from "axios"; -import { API_HOST } from "../../common/server-api"; -import { User } from "../../models/user"; +import { getUserFullname, User, UserPrefs } from "~/models/user"; +import { AxiosInstance } from "axios"; +import { ApiActions } from "~/services/api/api-actions"; +import * as uuid from "uuid/v4"; +import { Session, SessionStatus } from "~/models/session"; +import { Config } from "~/common/config"; +import { uniqBy } from "lodash"; -const API_TOKEN_KEY = 'apiToken'; -const USER_EMAIL_KEY = 'userEmail'; -const USER_FIRST_NAME_KEY = 'userFirstName'; -const USER_LAST_NAME_KEY = 'userLastName'; +export const API_TOKEN_KEY = 'apiToken'; +export const USER_EMAIL_KEY = 'userEmail'; +export const USER_FIRST_NAME_KEY = 'userFirstName'; +export const USER_LAST_NAME_KEY = 'userLastName'; +export const USER_UUID_KEY = 'userUuid'; +export const USER_OWNER_UUID_KEY = 'userOwnerUuid'; +export const USER_IS_ADMIN = 'isAdmin'; +export const USER_IDENTITY_URL = 'identityUrl'; +export const USER_PREFS = 'prefs'; -export default class AuthService { +export interface UserDetailsResponse { + email: string; + first_name: string; + last_name: string; + uuid: string; + owner_uuid: string; + is_admin: boolean; + identity_url: string; + prefs: UserPrefs; +} + +export class AuthService { + + constructor( + protected apiClient: AxiosInstance, + protected baseUrl: string, + protected actions: ApiActions) { } public saveApiToken(token: string) { localStorage.setItem(API_TOKEN_KEY, token); @@ -25,12 +50,30 @@ export default class AuthService { return localStorage.getItem(API_TOKEN_KEY) || undefined; } + public getUuid() { + return localStorage.getItem(USER_UUID_KEY) || undefined; + } + + public getOwnerUuid() { + return localStorage.getItem(USER_OWNER_UUID_KEY) || undefined; + } + + public getIsAdmin(): boolean { + return localStorage.getItem(USER_IS_ADMIN) === 'true'; + } + public getUser(): User | undefined { const email = localStorage.getItem(USER_EMAIL_KEY); const firstName = localStorage.getItem(USER_FIRST_NAME_KEY); const lastName = localStorage.getItem(USER_LAST_NAME_KEY); - return email && firstName && lastName - ? { email, firstName, lastName } + const uuid = this.getUuid(); + const ownerUuid = this.getOwnerUuid(); + const isAdmin = this.getIsAdmin(); + const identityUrl = localStorage.getItem(USER_IDENTITY_URL); + const prefs = JSON.parse(localStorage.getItem(USER_PREFS) || '{"profile": {}}'); + + return email && firstName && lastName && uuid && ownerUuid && identityUrl && prefs + ? { email, firstName, lastName, uuid, ownerUuid, isAdmin, identityUrl, prefs } : undefined; } @@ -38,21 +81,112 @@ export default class AuthService { localStorage.setItem(USER_EMAIL_KEY, user.email); localStorage.setItem(USER_FIRST_NAME_KEY, user.firstName); localStorage.setItem(USER_LAST_NAME_KEY, user.lastName); + localStorage.setItem(USER_UUID_KEY, user.uuid); + localStorage.setItem(USER_OWNER_UUID_KEY, user.ownerUuid); + localStorage.setItem(USER_IS_ADMIN, JSON.stringify(user.isAdmin)); + localStorage.setItem(USER_IDENTITY_URL, user.identityUrl); + localStorage.setItem(USER_PREFS, JSON.stringify(user.prefs)); } public removeUser() { localStorage.removeItem(USER_EMAIL_KEY); localStorage.removeItem(USER_FIRST_NAME_KEY); localStorage.removeItem(USER_LAST_NAME_KEY); + localStorage.removeItem(USER_UUID_KEY); + localStorage.removeItem(USER_OWNER_UUID_KEY); + localStorage.removeItem(USER_IS_ADMIN); + localStorage.removeItem(USER_IDENTITY_URL); + localStorage.removeItem(USER_PREFS); } public login() { const currentUrl = `${window.location.protocol}//${window.location.host}/token`; - window.location.href = `${API_HOST}/login?return_to=${currentUrl}`; + window.location.assign(`${this.baseUrl || ""}/login?return_to=${currentUrl}`); } public logout() { const currentUrl = `${window.location.protocol}//${window.location.host}`; - window.location.href = `${API_HOST}/logout?return_to=${currentUrl}`; + window.location.assign(`${this.baseUrl || ""}/logout?return_to=${currentUrl}`); + } + + public getUserDetails = (): Promise => { + const reqId = uuid(); + this.actions.progressFn(reqId, true); + return this.apiClient + .get('/users/current') + .then(resp => { + this.actions.progressFn(reqId, false); + const prefs = resp.data.prefs.profile ? resp.data.prefs : { profile: {}}; + return { + email: resp.data.email, + firstName: resp.data.first_name, + lastName: resp.data.last_name, + uuid: resp.data.uuid, + ownerUuid: resp.data.owner_uuid, + isAdmin: resp.data.is_admin, + identityUrl: resp.data.identity_url, + prefs + }; + }) + .catch(e => { + this.actions.progressFn(reqId, false); + this.actions.errorFn(reqId, e); + throw e; + }); + } + + public getRootUuid() { + const uuid = this.getOwnerUuid(); + const uuidParts = uuid ? uuid.split('-') : []; + return uuidParts.length > 1 ? `${uuidParts[0]}-${uuidParts[1]}` : undefined; + } + + public getSessions(): Session[] { + try { + const sessions = JSON.parse(localStorage.getItem("sessions") || ''); + return sessions; + } catch { + return []; + } + } + + public saveSessions(sessions: Session[]) { + localStorage.setItem("sessions", JSON.stringify(sessions)); + } + + public buildSessions(cfg: Config, user?: User) { + const currentSession = { + clusterId: cfg.uuidPrefix, + remoteHost: cfg.rootUrl, + baseUrl: cfg.baseUrl, + username: getUserFullname(user), + email: user ? user.email : '', + token: this.getApiToken(), + loggedIn: true, + active: true, + status: SessionStatus.VALIDATED + } as Session; + const localSessions = this.getSessions(); + const cfgSessions = Object.keys(cfg.remoteHosts).map(clusterId => { + const remoteHost = cfg.remoteHosts[clusterId]; + return { + clusterId, + remoteHost, + baseUrl: '', + username: '', + email: '', + token: '', + loggedIn: false, + active: false, + status: SessionStatus.INVALIDATED + } as Session; + }); + const sessions = [currentSession] + .concat(localSessions) + .concat(cfgSessions); + + const uniqSessions = uniqBy(sessions, 'clusterId'); + + return uniqSessions; } }