2019-12-07 21:02:00 +02:00
|
|
|
import expect from 'app/test/unexpected';
|
2020-01-17 23:37:52 +03:00
|
|
|
import sinon, { SinonStub } from 'sinon';
|
2016-07-30 13:44:43 +03:00
|
|
|
|
2019-12-07 21:02:00 +02:00
|
|
|
import AuthFlow from 'app/services/authFlow/AuthFlow';
|
|
|
|
import AbstractState from 'app/services/authFlow/AbstractState';
|
|
|
|
import localStorage from 'app/services/localStorage';
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2024-12-10 20:42:06 +01:00
|
|
|
import InitOAuthAuthCodeFlowState from './InitOAuthAuthCodeFlowState';
|
2019-12-07 21:02:00 +02:00
|
|
|
import RegisterState from 'app/services/authFlow/RegisterState';
|
|
|
|
import RecoverPasswordState from 'app/services/authFlow/RecoverPasswordState';
|
|
|
|
import ForgotPasswordState from 'app/services/authFlow/ForgotPasswordState';
|
|
|
|
import ActivationState from 'app/services/authFlow/ActivationState';
|
|
|
|
import ResendActivationState from 'app/services/authFlow/ResendActivationState';
|
|
|
|
import LoginState from 'app/services/authFlow/LoginState';
|
|
|
|
import CompleteState from 'app/services/authFlow/CompleteState';
|
|
|
|
import ChooseAccountState from 'app/services/authFlow/ChooseAccountState';
|
2020-01-17 23:37:52 +03:00
|
|
|
import { Store } from 'redux';
|
2016-05-28 01:24:22 +03:00
|
|
|
|
2016-04-12 06:49:58 +03:00
|
|
|
describe('AuthFlow', () => {
|
2020-05-24 02:08:24 +03:00
|
|
|
let flow: AuthFlow;
|
|
|
|
let actions: { test: SinonStub };
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
beforeEach(() => {
|
2020-05-24 02:08:24 +03:00
|
|
|
actions = {
|
|
|
|
test: sinon.stub().named('actions.test'),
|
|
|
|
};
|
|
|
|
actions.test.returns('passed');
|
2016-08-27 13:19:02 +03:00
|
|
|
|
2020-01-17 23:37:52 +03:00
|
|
|
// @ts-ignore
|
2020-05-24 02:08:24 +03:00
|
|
|
flow = new AuthFlow(actions);
|
2019-11-27 11:03:32 +02:00
|
|
|
});
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should not allow to mutate actions', () => {
|
|
|
|
expect(
|
|
|
|
// @ts-ignore
|
|
|
|
() => (flow.actions.foo = 'bar'),
|
|
|
|
'to throw',
|
|
|
|
/readonly|not extensible/,
|
|
|
|
);
|
|
|
|
// @ts-ignore
|
|
|
|
expect(() => (flow.actions.test = 'hacked'), 'to throw', /read ?only/);
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('#setStore', () => {
|
|
|
|
it('should create #navigate, #getState, #dispatch', () => {
|
|
|
|
flow.setStore({
|
|
|
|
// @ts-ignore
|
|
|
|
getState() {},
|
|
|
|
// @ts-ignore
|
|
|
|
dispatch() {},
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(flow.getState, 'to be defined');
|
|
|
|
expect(flow.dispatch, 'to be defined');
|
|
|
|
expect(flow.navigate, 'to be defined');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('#restoreOAuthState', () => {
|
|
|
|
let oauthData: Record<string, string>;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
oauthData = { foo: 'bar' };
|
|
|
|
localStorage.setItem(
|
|
|
|
'oauthData',
|
|
|
|
JSON.stringify({
|
|
|
|
timestamp: Date.now() - 10,
|
|
|
|
payload: oauthData,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
|
|
|
sinon.stub(flow, 'run').named('flow.run');
|
|
|
|
const promiseLike = { then: (fn: Function) => fn() || promiseLike };
|
|
|
|
// @ts-ignore
|
|
|
|
flow.run.returns(promiseLike);
|
|
|
|
sinon.stub(flow, 'setState').named('flow.setState');
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
localStorage.removeItem('oauthData');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should call to restoreOAuthState', () => {
|
|
|
|
// @ts-ignore
|
|
|
|
sinon.stub(flow, 'restoreOAuthState').named('flow.restoreOAuthState');
|
|
|
|
|
|
|
|
flow.handleRequest(
|
|
|
|
{ path: '/', query: new URLSearchParams(''), params: {} },
|
|
|
|
() => {},
|
|
|
|
() => {},
|
|
|
|
);
|
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
expect(flow.restoreOAuthState, 'was called');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should restore oauth state from localStorage', () => {
|
|
|
|
flow.handleRequest(
|
|
|
|
{ path: '/', query: new URLSearchParams(''), params: {} },
|
|
|
|
() => {},
|
|
|
|
() => {},
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(flow.run, 'to have a call satisfying', ['oAuthValidate', oauthData]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should transition to CompleteState', () => {
|
|
|
|
flow.handleRequest(
|
|
|
|
{ path: '/', query: new URLSearchParams(''), params: {} },
|
|
|
|
() => {},
|
|
|
|
() => {},
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(flow.setState, 'to have a call satisfying', [expect.it('to be a', CompleteState)]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not handle current request', () => {
|
|
|
|
flow.handleRequest(
|
|
|
|
{ path: '/', query: new URLSearchParams(''), params: {} },
|
|
|
|
() => {},
|
|
|
|
() => {},
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(flow.setState, 'was called once');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should call onReady after state restoration', () => {
|
|
|
|
const onReady = sinon.stub().named('onReady');
|
|
|
|
|
|
|
|
flow.handleRequest(
|
|
|
|
{ path: '/login', query: new URLSearchParams(''), params: {} },
|
|
|
|
// @ts-ignore
|
|
|
|
null,
|
|
|
|
onReady,
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(onReady, 'was called');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not restore oauth state for /register route', () => {
|
|
|
|
flow.handleRequest(
|
|
|
|
{ path: '/register', query: new URLSearchParams(''), params: {} },
|
|
|
|
() => {},
|
|
|
|
() => {},
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(flow.run, 'was not called'); // this.run('oAuthValidate'...
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not restore outdated (>1h) oauth state', () => {
|
|
|
|
localStorage.setItem(
|
|
|
|
'oauthData',
|
|
|
|
JSON.stringify({
|
|
|
|
timestamp: Date.now() - 2 * 60 * 60 * 1000,
|
|
|
|
payload: oauthData,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
|
|
|
flow.handleRequest(
|
|
|
|
{ path: '/', query: new URLSearchParams(''), params: {} },
|
|
|
|
() => {},
|
|
|
|
() => {},
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(flow.run, 'was not called');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('#setState', () => {
|
|
|
|
it('should change state', () => {
|
|
|
|
const state = new AbstractState();
|
|
|
|
flow.setState(state);
|
|
|
|
|
|
|
|
expect(flow.state, 'to be', state);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should call #enter() on new state and pass reference to itself', () => {
|
|
|
|
const state = new AbstractState();
|
|
|
|
const spy = sinon.spy(state, 'enter').named('state.enter');
|
|
|
|
|
|
|
|
flow.setState(state);
|
|
|
|
|
|
|
|
expect(spy, 'was called once');
|
|
|
|
expect(spy, 'to have a call satisfying', [flow]);
|
|
|
|
});
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should call `leave` on previous state if any', () => {
|
|
|
|
class State1 extends AbstractState {}
|
|
|
|
class State2 extends AbstractState {}
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
const state1 = new State1();
|
|
|
|
const state2 = new State2();
|
|
|
|
const spy1 = sinon.spy(state1, 'leave');
|
|
|
|
const spy2 = sinon.spy(state2, 'leave');
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
flow.setState(state1);
|
|
|
|
flow.setState(state2);
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
expect(spy1, 'was called once');
|
|
|
|
expect(spy1, 'to have a call satisfying', [flow]);
|
|
|
|
expect(spy2, 'was not called');
|
|
|
|
});
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should return promise, if #enter returns it', () => {
|
|
|
|
const state = new AbstractState();
|
|
|
|
const expected = Promise.resolve();
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
state.enter = () => expected;
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
const actual = flow.setState(state);
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
expect(actual, 'to be', expected);
|
|
|
|
});
|
2016-04-12 06:49:58 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should throw if no state', () => {
|
|
|
|
// @ts-ignore
|
|
|
|
expect(() => flow.setState(), 'to throw', 'State is required');
|
|
|
|
});
|
2016-04-12 06:49:58 +03:00
|
|
|
});
|
2016-05-28 01:24:22 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('#run', () => {
|
|
|
|
let store: Store;
|
2016-05-28 01:24:22 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
beforeEach(() => {
|
|
|
|
// @ts-ignore
|
|
|
|
store = {
|
|
|
|
getState() {},
|
|
|
|
dispatch: sinon.stub().named('store.dispatch'),
|
|
|
|
};
|
|
|
|
|
|
|
|
flow.setStore(store);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should dispatch an action', () => {
|
|
|
|
// @ts-ignore
|
|
|
|
flow.run('test');
|
|
|
|
|
|
|
|
expect(store.dispatch, 'was called once');
|
|
|
|
expect(store.dispatch, 'to have a call satisfying', ['passed']);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should dispatch an action with payload given', () => {
|
|
|
|
// @ts-ignore
|
|
|
|
flow.run('test', 'arg');
|
|
|
|
|
|
|
|
expect(actions.test, 'was called once');
|
|
|
|
expect(actions.test, 'to have a call satisfying', ['arg']);
|
|
|
|
});
|
2019-11-27 11:03:32 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should resolve to action dispatch result', () => {
|
|
|
|
const expected = 'dispatch called';
|
|
|
|
// @ts-ignore
|
|
|
|
store.dispatch.returns(expected);
|
2016-05-28 01:24:22 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
// @ts-ignore
|
|
|
|
return expect(flow.run('test'), 'to be fulfilled with', expected);
|
|
|
|
});
|
2019-11-27 11:03:32 +02:00
|
|
|
});
|
2016-06-03 22:10:47 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('#goBack', () => {
|
|
|
|
it('should call goBack on state passing itself as argument', () => {
|
|
|
|
const state = new AbstractState();
|
|
|
|
sinon.stub(state, 'goBack').named('state.goBack');
|
|
|
|
flow.setState(state);
|
2016-06-03 22:10:47 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
flow.goBack();
|
2016-06-15 09:01:41 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
expect(state.goBack, 'was called once');
|
|
|
|
expect(state.goBack, 'to have a call satisfying', [flow]);
|
|
|
|
});
|
2019-11-27 11:03:32 +02:00
|
|
|
});
|
2016-08-07 16:54:59 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('#resolve', () => {
|
|
|
|
it('should call resolve on state passing itself and payload as arguments', () => {
|
|
|
|
const state = new AbstractState();
|
|
|
|
sinon.stub(state, 'resolve').named('state.resolve');
|
|
|
|
flow.setState(state);
|
|
|
|
|
|
|
|
const expectedPayload = { foo: 'bar' };
|
|
|
|
|
|
|
|
flow.resolve(expectedPayload);
|
|
|
|
|
|
|
|
expect(state.resolve, 'was called once');
|
|
|
|
expect(state.resolve, 'to have a call satisfying', [flow, expectedPayload]);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('#reject', () => {
|
|
|
|
it('should call reject on state passing itself and payload as arguments', () => {
|
|
|
|
const state = new AbstractState();
|
|
|
|
sinon.stub(state, 'reject').named('state.reject');
|
|
|
|
flow.setState(state);
|
2016-08-07 16:54:59 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
const expectedPayload = { foo: 'bar' };
|
2019-11-27 11:03:32 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
flow.reject(expectedPayload);
|
2019-11-27 11:03:32 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
expect(state.reject, 'was called once');
|
|
|
|
expect(state.reject, 'to have a call satisfying', [flow, expectedPayload]);
|
|
|
|
});
|
2019-11-27 11:03:32 +02:00
|
|
|
});
|
2016-08-07 16:54:59 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('#handleRequest()', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
sinon.stub(flow, 'setState').named('flow.setState');
|
|
|
|
sinon.stub(flow, 'run').named('flow.run');
|
|
|
|
});
|
|
|
|
|
|
|
|
Object.entries({
|
|
|
|
'/': LoginState,
|
|
|
|
'/login': LoginState,
|
|
|
|
'/password': LoginState,
|
|
|
|
'/accept-rules': LoginState,
|
|
|
|
'/oauth/permissions': LoginState,
|
|
|
|
'/oauth/choose-account': LoginState,
|
|
|
|
'/oauth/finish': LoginState,
|
2024-12-10 20:42:06 +01:00
|
|
|
'/oauth2/v1/foo': InitOAuthAuthCodeFlowState,
|
|
|
|
'/oauth2/v1': InitOAuthAuthCodeFlowState,
|
|
|
|
'/oauth2': InitOAuthAuthCodeFlowState,
|
2020-05-24 02:08:24 +03:00
|
|
|
'/register': RegisterState,
|
|
|
|
'/choose-account': ChooseAccountState,
|
|
|
|
'/recover-password': RecoverPasswordState,
|
|
|
|
'/recover-password/key123': RecoverPasswordState,
|
|
|
|
'/forgot-password': ForgotPasswordState,
|
|
|
|
'/activation': ActivationState,
|
|
|
|
'/resend-activation': ResendActivationState,
|
|
|
|
}).forEach(([path, type]) => {
|
|
|
|
it(`should transition to ${type.name} if ${path}`, () => {
|
|
|
|
flow.handleRequest(
|
|
|
|
{ path, query: new URLSearchParams({}), params: {} },
|
|
|
|
() => {},
|
|
|
|
() => {},
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(flow.setState, 'was called once');
|
|
|
|
expect(flow.setState, 'to have a call satisfying', [expect.it('to be a', type)]);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should call callback', () => {
|
|
|
|
const callback = sinon.stub().named('callback');
|
|
|
|
|
|
|
|
flow.handleRequest({ path: '/', query: new URLSearchParams({}), params: {} }, () => {}, callback);
|
|
|
|
|
|
|
|
expect(callback, 'was called once');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not call callback till returned from #enter() promise will be resolved', () => {
|
|
|
|
let resolve: Function;
|
|
|
|
const promise: Promise<void> = {
|
|
|
|
// @ts-ignore
|
|
|
|
then: (cb: Function) => {
|
|
|
|
resolve = cb;
|
|
|
|
},
|
|
|
|
};
|
|
|
|
const callback = sinon.stub().named('callback');
|
|
|
|
|
|
|
|
const state = new AbstractState();
|
|
|
|
state.enter = () => promise;
|
|
|
|
|
|
|
|
flow.setState = AuthFlow.prototype.setState.bind(flow, state);
|
|
|
|
|
|
|
|
flow.handleRequest({ path: '/', query: new URLSearchParams({}), params: {} }, () => {}, callback);
|
|
|
|
|
|
|
|
// @ts-ignore
|
|
|
|
expect(resolve, 'to be', callback);
|
|
|
|
|
|
|
|
expect(callback, 'was not called');
|
|
|
|
// @ts-ignore
|
|
|
|
resolve();
|
|
|
|
expect(callback, 'was called once');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not handle the same request twice', () => {
|
|
|
|
const path = '/oauth2';
|
|
|
|
const callback = sinon.stub();
|
|
|
|
|
|
|
|
flow.handleRequest({ path, query: new URLSearchParams({}), params: {} }, () => {}, callback);
|
|
|
|
flow.handleRequest({ path, query: new URLSearchParams({}), params: {} }, () => {}, callback);
|
|
|
|
|
|
|
|
expect(flow.setState, 'was called once');
|
2024-12-10 20:42:06 +01:00
|
|
|
expect(flow.setState, 'to have a call satisfying', [expect.it('to be a', InitOAuthAuthCodeFlowState)]);
|
2020-05-24 02:08:24 +03:00
|
|
|
expect(callback, 'was called twice');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('throws if unsupported request', () => {
|
|
|
|
const replace = sinon.stub().named('replace');
|
|
|
|
|
|
|
|
flow.handleRequest({ path: '/foo/bar', query: new URLSearchParams({}), params: {} }, replace);
|
|
|
|
|
|
|
|
expect(replace, 'to have a call satisfying', ['/404']);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('#getRequest()', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
sinon.stub(flow, 'setState').named('flow.setState');
|
|
|
|
sinon.stub(flow, 'run').named('flow.run');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return request with path, query, params', () => {
|
|
|
|
const request = { path: '/', query: new URLSearchParams({}), params: {} };
|
|
|
|
|
|
|
|
flow.handleRequest(
|
|
|
|
request,
|
|
|
|
() => {},
|
|
|
|
() => {},
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(flow.getRequest(), 'to satisfy', {
|
|
|
|
...request,
|
|
|
|
query: expect.it('to be an', URLSearchParams),
|
|
|
|
params: {},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return a copy of current request', () => {
|
|
|
|
const request = {
|
|
|
|
path: '/',
|
|
|
|
query: new URLSearchParams({ foo: 'bar' }),
|
|
|
|
params: { baz: 'bud' },
|
|
|
|
};
|
|
|
|
|
|
|
|
flow.handleRequest(
|
|
|
|
request,
|
|
|
|
() => {},
|
|
|
|
() => {},
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(flow.getRequest(), 'to equal', request);
|
|
|
|
expect(flow.getRequest(), 'not to be', request);
|
|
|
|
});
|
2016-05-28 01:24:22 +03:00
|
|
|
});
|
2016-04-12 06:49:58 +03:00
|
|
|
});
|