1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 import { getNewExtraToken, initAuth } from "./auth-action";
6 import { API_TOKEN_KEY } from "~/services/auth-service/auth-service";
8 import 'jest-localstorage-mock';
9 import { ServiceRepository, createServices } from "~/services/services";
10 import { configureStore, RootStore } from "../store";
11 import { createBrowserHistory } from "history";
12 import { Config, mockConfig } from '~/common/config';
13 import { ApiActions } from "~/services/api/api-actions";
14 import { ACCOUNT_LINK_STATUS_KEY } from '~/services/link-account-service/link-account-service';
15 import Axios from "axios";
16 import MockAdapter from "axios-mock-adapter";
17 import { ImportMock } from 'ts-mock-imports';
18 import * as servicesModule from "~/services/services";
19 import { SessionStatus } from "~/models/session";
21 describe('auth-actions', () => {
22 const axiosInst = Axios.create({ headers: {} });
23 const axiosMock = new MockAdapter(axiosInst);
26 let services: ServiceRepository;
27 const config: any = {};
28 const actions: ApiActions = {
29 progressFn: (id: string, working: boolean) => { },
30 errorFn: (id: string, message: string) => { }
32 let importMocks: any[];
36 services = createServices(mockConfig({}), actions, axiosInst);
37 store = configureStore(createBrowserHistory(), services, config);
43 importMocks.map(m => m.restore());
46 it('creates an extra token', async () => {
48 .onGet("/users/current")
50 email: "test@test.com",
53 uuid: "zzzzz-tpzed-abcefg",
54 owner_uuid: "ownerUuid",
60 .onGet("/api_client_authorizations/current")
62 expires_at: "2140-01-01T00:00:00.000Z",
63 api_token: 'extra token',
65 .onPost("/api_client_authorizations")
67 uuid: 'zzzzz-gj3su-xxxxxxxxxx',
68 apiToken: 'extra token',
70 .onPost("/api_client_authorizations")
72 uuid: 'zzzzz-gj3su-xxxxxxxxxx',
73 apiToken: 'extra additional token',
76 importMocks.push(ImportMock.mockFunction(servicesModule, 'createServices', services));
77 sessionStorage.setItem(ACCOUNT_LINK_STATUS_KEY, "0");
78 localStorage.setItem(API_TOKEN_KEY, "token");
81 rootUrl: "https://zzzzz.arvadosapi.com",
84 apiRevision: 12345678,
86 Login: { LoginCluster: "" },
90 // Set up auth, confirm that no extra token was requested
91 await store.dispatch(initAuth(config))
92 expect(store.getState().auth.apiToken).toBeDefined();
93 expect(store.getState().auth.extraApiToken).toBeUndefined();
95 // Ask for an extra token
96 await store.dispatch(getNewExtraToken());
97 expect(store.getState().auth.apiToken).toBeDefined();
98 expect(store.getState().auth.extraApiToken).toBeDefined();
99 const extraToken = store.getState().auth.extraApiToken;
101 // Ask for a cached extra token
102 await store.dispatch(getNewExtraToken(true));
103 expect(store.getState().auth.extraApiToken).toBe(extraToken);
105 // Ask for a new extra token, should make a second api request
106 await store.dispatch(getNewExtraToken(false));
107 expect(store.getState().auth.extraApiToken).toBeDefined();
108 expect(store.getState().auth.extraApiToken).not.toBe(extraToken);
111 it('should initialise state with user and api token from local storage', (done) => {
113 .onGet("/users/current")
115 email: "test@test.com",
118 uuid: "zzzzz-tpzed-abcefg",
119 owner_uuid: "ownerUuid",
125 .onGet("/api_client_authorizations/current")
127 expires_at: "2140-01-01T00:00:00.000Z"
129 .onGet("https://xc59z.arvadosapi.com/discovery/v1/apis/arvados/v1/rest")
131 baseUrl: "https://xc59z.arvadosapi.com/arvados/v1",
132 keepWebServiceUrl: "",
133 keepWebInlineServiceUrl: "",
135 rootUrl: "https://xc59z.arvadosapi.com",
143 importMocks.push(ImportMock.mockFunction(servicesModule, 'createServices', services));
145 // Only test the case when a link account operation is not being cancelled
146 sessionStorage.setItem(ACCOUNT_LINK_STATUS_KEY, "0");
147 localStorage.setItem(API_TOKEN_KEY, "token");
149 const config: any = {
150 rootUrl: "https://zzzzz.arvadosapi.com",
152 remoteHosts: { xc59z: "xc59z.arvadosapi.com" },
153 apiRevision: 12345678,
155 Login: { LoginCluster: "" },
159 store.dispatch(initAuth(config));
161 store.subscribe(() => {
162 const auth = store.getState().auth;
163 if (auth.apiToken === "token" &&
164 auth.sessions.length === 2 &&
165 auth.sessions[0].status === SessionStatus.VALIDATED &&
166 auth.sessions[1].status === SessionStatus.VALIDATED
169 expect(auth).toEqual({
171 apiTokenExpiration: new Date("2140-01-01T00:00:00.000Z"),
173 apiRevision: 12345678,
180 "xc59z": "xc59z.arvadosapi.com",
182 rootUrl: "https://zzzzz.arvadosapi.com",
186 extraApiToken: undefined,
187 extraApiTokenExpiration: undefined,
188 homeCluster: "zzzzz",
189 localCluster: "zzzzz",
190 loginCluster: undefined,
193 "apiRevision": 12345678,
200 "xc59z": "xc59z.arvadosapi.com",
202 "rootUrl": "https://zzzzz.arvadosapi.com",
203 "uuidPrefix": "zzzzz",
205 "xc59z": mockConfig({
206 apiRevision: 12345678,
207 baseUrl: "https://xc59z.arvadosapi.com/arvados/v1",
208 rootUrl: "https://xc59z.arvadosapi.com",
213 zzzzz: "zzzzz.arvadosapi.com",
214 xc59z: "xc59z.arvadosapi.com"
218 "baseUrl": undefined,
219 "clusterId": "zzzzz",
220 "email": "test@test.com",
222 "remoteHost": "https://zzzzz.arvadosapi.com",
226 "apiRevision": 12345678,
227 "uuid": "zzzzz-tpzed-abcefg",
232 "clusterId": "xc59z",
235 "remoteHost": "xc59z.arvadosapi.com",
243 email: "test@test.com",
246 uuid: "zzzzz-tpzed-abcefg",
247 ownerUuid: "ownerUuid",
249 prefs: { profile: {} },
263 // TODO: Add remaining action tests
265 it('should fire external url to login', () => {
266 const initialState = undefined;
267 window.location.assign = jest.fn();
268 reducer(initialState, authActions.LOGIN());
269 expect(window.location.assign).toBeCalledWith(
270 `/login?return_to=${window.location.protocol}//${window.location.host}/token`
274 it('should fire external url to logout', () => {
275 const initialState = undefined;
276 window.location.assign = jest.fn();
277 reducer(initialState, authActions.LOGOUT());
278 expect(window.location.assign).toBeCalledWith(
279 `/logout?return_to=${location.protocol}//${location.host}`