2019-12-07 21:02:00 +02:00
|
|
|
import expect from 'app/test/unexpected';
|
2016-12-05 21:14:38 +02:00
|
|
|
import sinon from 'sinon';
|
2019-12-07 21:02:00 +02:00
|
|
|
import { browserHistory } from 'app/services/history';
|
|
|
|
import { InternalServerError } from 'app/services/request';
|
|
|
|
import { sessionStorage } from 'app/services/localStorage';
|
|
|
|
import * as authentication from 'app/services/api/authentication';
|
2020-05-24 02:08:24 +03:00
|
|
|
import { authenticate, revoke, logoutAll, logoutStrangers } from 'app/components/accounts/actions';
|
|
|
|
import { add, activate, remove, reset } from 'app/components/accounts/actions/pure-actions';
|
2019-12-07 21:02:00 +02:00
|
|
|
import { updateUser, setUser } from 'app/components/user/actions';
|
2020-10-23 01:23:59 +03:00
|
|
|
import { setLogin } from 'app/components/auth/actions';
|
2020-07-22 13:01:12 +03:00
|
|
|
import { Dispatch, State as RootState } from 'app/types';
|
2019-12-07 13:28:52 +02:00
|
|
|
|
|
|
|
import { Account } from './reducer';
|
2020-10-27 01:46:57 +03:00
|
|
|
import { User } from 'app/components/user';
|
2016-10-30 14:12:49 +02:00
|
|
|
|
2020-06-04 21:03:24 +03:00
|
|
|
jest.mock('app/i18n', () => ({
|
|
|
|
en: {
|
|
|
|
code: 'en',
|
|
|
|
name: 'English',
|
|
|
|
englishName: 'English',
|
|
|
|
progress: 100,
|
|
|
|
isReleased: true,
|
|
|
|
},
|
|
|
|
be: {
|
|
|
|
code: 'be',
|
|
|
|
name: 'Беларуская',
|
|
|
|
englishName: 'Belarusian',
|
|
|
|
progress: 97,
|
|
|
|
isReleased: true,
|
|
|
|
},
|
|
|
|
}));
|
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
const token = 'eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJlbHl8MSJ9.pRJ7vakt2eIscjqwG__KhSxKb3qwGsdBBeDbBffJs_I';
|
|
|
|
const legacyToken = 'eyJhbGciOiJIUzI1NiJ9.eyJqdGkiOjF9.cRF-sQNrwWQ94xCb3vWioVdjxAZeefEE7GMGwh7708o';
|
2019-01-27 22:12:58 +03:00
|
|
|
|
2020-10-27 01:46:57 +03:00
|
|
|
const account: Account = {
|
2020-05-24 02:08:24 +03:00
|
|
|
id: 1,
|
|
|
|
username: 'username',
|
|
|
|
email: 'email@test.com',
|
|
|
|
token,
|
|
|
|
refreshToken: 'bar',
|
2020-10-27 01:46:57 +03:00
|
|
|
isDeleted: false,
|
2016-10-30 14:12:49 +02:00
|
|
|
};
|
|
|
|
|
2020-10-27 01:46:57 +03:00
|
|
|
const user: Partial<User> = {
|
2020-05-24 02:08:24 +03:00
|
|
|
id: 1,
|
|
|
|
username: 'username',
|
|
|
|
email: 'email@test.com',
|
|
|
|
lang: 'be',
|
2020-10-27 01:46:57 +03:00
|
|
|
isDeleted: false,
|
2016-10-30 14:12:49 +02:00
|
|
|
};
|
|
|
|
|
2016-11-12 22:31:44 +02:00
|
|
|
describe('components/accounts/actions', () => {
|
2020-05-24 02:08:24 +03:00
|
|
|
let dispatch: Dispatch;
|
|
|
|
let getState: () => RootState;
|
2016-10-30 14:12:49 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
beforeEach(() => {
|
|
|
|
dispatch = sinon
|
|
|
|
.spy((arg) => (typeof arg === 'function' ? arg(dispatch, getState) : arg))
|
|
|
|
.named('store.dispatch');
|
|
|
|
getState = sinon.stub().named('store.getState');
|
2017-12-30 21:04:31 +02:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
(getState as any).returns({
|
2020-05-24 02:08:24 +03:00
|
|
|
accounts: {
|
|
|
|
available: [],
|
|
|
|
active: null,
|
|
|
|
},
|
|
|
|
auth: {
|
|
|
|
credentials: {},
|
2019-11-27 11:03:32 +02:00
|
|
|
},
|
2020-05-24 02:08:24 +03:00
|
|
|
user: {},
|
2016-10-30 14:12:49 +02:00
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
sinon.stub(authentication, 'validateToken').named('authentication.validateToken');
|
|
|
|
sinon.stub(browserHistory, 'push').named('browserHistory.push');
|
|
|
|
sinon.stub(authentication, 'logout').named('authentication.logout');
|
2017-02-24 07:50:32 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
(authentication.logout as any).returns(Promise.resolve());
|
2019-12-07 13:28:52 +02:00
|
|
|
(authentication.validateToken as any).returns(
|
2020-05-24 02:08:24 +03:00
|
|
|
Promise.resolve({
|
|
|
|
token: account.token,
|
|
|
|
refreshToken: account.refreshToken,
|
|
|
|
user,
|
|
|
|
}),
|
2019-12-07 13:28:52 +02:00
|
|
|
);
|
2020-05-24 02:08:24 +03:00
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
afterEach(() => {
|
|
|
|
(authentication.validateToken as any).restore();
|
|
|
|
(authentication.logout as any).restore();
|
|
|
|
(browserHistory.push as any).restore();
|
2019-11-27 11:03:32 +02:00
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('#authenticate()', () => {
|
|
|
|
it('should request user state using token', () =>
|
|
|
|
authenticate(account)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(authentication.validateToken, 'to have a call satisfying', [
|
|
|
|
account.id,
|
|
|
|
account.token,
|
|
|
|
account.refreshToken,
|
|
|
|
]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it('should request user by extracting id from token', () =>
|
|
|
|
authenticate({ token } as Account)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(authentication.validateToken, 'to have a call satisfying', [1, token, undefined]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it('should request user by extracting id from legacy token', () =>
|
|
|
|
authenticate({ token: legacyToken } as Account)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(authentication.validateToken, 'to have a call satisfying', [1, legacyToken, undefined]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it(`dispatches accounts:add action`, () =>
|
|
|
|
authenticate(account)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [add(account)]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it(`dispatches accounts:activate action`, () =>
|
|
|
|
authenticate(account)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [activate(account)]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it(`dispatches i18n:setLocale action`, () =>
|
|
|
|
authenticate(account)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [{ type: 'i18n:setLocale', payload: { locale: 'be' } }]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it('should update user state', () =>
|
|
|
|
authenticate(account)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [updateUser({ ...user, isGuest: false })]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it('resolves with account', () =>
|
|
|
|
authenticate(account)(dispatch, getState, undefined).then((resp) => expect(resp, 'to equal', account)));
|
|
|
|
|
|
|
|
it('rejects when bad auth data', () => {
|
|
|
|
(authentication.validateToken as any).returns(Promise.reject({}));
|
|
|
|
|
|
|
|
return expect(authenticate(account)(dispatch, getState, undefined), 'to be rejected').then(() => {
|
|
|
|
expect(dispatch, 'to have a call satisfying', [setLogin(account.email)]);
|
|
|
|
|
|
|
|
expect(browserHistory.push, 'to have a call satisfying', ['/login']);
|
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
});
|
2017-01-31 08:05:36 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('rejects when 5xx without logouting', () => {
|
|
|
|
const resp = new InternalServerError('500', { status: 500 });
|
2018-02-27 23:17:31 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
(authentication.validateToken as any).rejects(resp);
|
|
|
|
|
|
|
|
return expect(authenticate(account)(dispatch, getState, undefined), 'to be rejected with', resp).then(() =>
|
|
|
|
expect(dispatch, 'to have no calls satisfying', [{ payload: { isGuest: true } }]),
|
|
|
|
);
|
2018-02-27 23:17:31 +02:00
|
|
|
});
|
2016-10-30 14:12:49 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('marks user as stranger, if there is no refreshToken', () => {
|
|
|
|
const expectedKey = `stranger${account.id}`;
|
|
|
|
(authentication.validateToken as any).resolves({
|
|
|
|
token: account.token,
|
|
|
|
user,
|
|
|
|
});
|
2016-11-15 07:55:15 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
sessionStorage.removeItem(expectedKey);
|
2016-11-15 07:55:15 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
return authenticate(account)(dispatch, getState, undefined).then(() => {
|
|
|
|
expect(sessionStorage.getItem(expectedKey), 'not to be null');
|
|
|
|
sessionStorage.removeItem(expectedKey);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when user authenticated during oauth', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
(getState as any).returns({
|
|
|
|
accounts: {
|
|
|
|
available: [],
|
|
|
|
active: null,
|
|
|
|
},
|
|
|
|
user: {},
|
|
|
|
auth: {
|
|
|
|
oauth: {
|
|
|
|
clientId: 'ely.by',
|
|
|
|
prompt: [],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2016-11-15 07:55:15 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('when one account available', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
(getState as any).returns({
|
|
|
|
accounts: {
|
|
|
|
active: account.id,
|
|
|
|
available: [account],
|
|
|
|
},
|
|
|
|
auth: {
|
|
|
|
credentials: {},
|
|
|
|
},
|
|
|
|
user,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should activate account before auth api call', () => {
|
|
|
|
(authentication.validateToken as any).returns(Promise.reject({ error: 'foo' }));
|
|
|
|
|
|
|
|
return expect(authenticate(account)(dispatch, getState, undefined), 'to be rejected with', {
|
|
|
|
error: 'foo',
|
|
|
|
}).then(() => expect(dispatch, 'to have a call satisfying', [activate(account)]));
|
|
|
|
});
|
|
|
|
});
|
2019-11-27 11:03:32 +02:00
|
|
|
});
|
2016-11-15 07:55:15 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('#revoke()', () => {
|
|
|
|
describe('when one account available', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
(getState as any).returns({
|
|
|
|
accounts: {
|
|
|
|
active: account.id,
|
|
|
|
available: [account],
|
|
|
|
},
|
|
|
|
auth: {
|
|
|
|
credentials: {},
|
|
|
|
},
|
|
|
|
user,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should dispatch reset action', () =>
|
|
|
|
revoke(account)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [reset()]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it('should call logout api method in background', () =>
|
|
|
|
revoke(account)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(authentication.logout, 'to have a call satisfying', [account.token]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it('should update user state', () =>
|
|
|
|
revoke(account)(dispatch, getState, undefined).then(
|
|
|
|
() => expect(dispatch, 'to have a call satisfying', [setUser({ isGuest: true })]),
|
|
|
|
// expect(dispatch, 'to have calls satisfying', [
|
|
|
|
// [remove(account)],
|
|
|
|
// [expect.it('to be a function')]
|
|
|
|
// // [logout()] // TODO: this is not a plain action. How should we simplify its testing?
|
|
|
|
// ])
|
|
|
|
));
|
|
|
|
});
|
2016-11-15 07:55:15 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('when multiple accounts available', () => {
|
|
|
|
const account2 = { ...account, id: 2 };
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
(getState as any).returns({
|
|
|
|
accounts: {
|
|
|
|
active: account2.id,
|
|
|
|
available: [account, account2],
|
|
|
|
},
|
|
|
|
user,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should switch to the next account', () =>
|
|
|
|
revoke(account2)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [activate(account)]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it('should remove current account', () =>
|
|
|
|
revoke(account2)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [remove(account2)]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it('should call logout api method in background', () =>
|
|
|
|
revoke(account2)(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(authentication.logout, 'to have a call satisfying', [account2.token]),
|
|
|
|
));
|
|
|
|
});
|
2019-11-27 11:03:32 +02:00
|
|
|
});
|
2016-11-15 07:55:15 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('#logoutAll()', () => {
|
|
|
|
const account2 = { ...account, id: 2 };
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
(getState as any).returns({
|
|
|
|
accounts: {
|
|
|
|
active: account2.id,
|
|
|
|
available: [account, account2],
|
|
|
|
},
|
|
|
|
auth: {
|
|
|
|
credentials: {},
|
|
|
|
},
|
|
|
|
user,
|
|
|
|
});
|
|
|
|
});
|
2017-01-04 07:52:46 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should call logout api method for each account', () => {
|
|
|
|
logoutAll()(dispatch, getState, undefined);
|
2017-01-04 07:52:46 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
expect(authentication.logout, 'to have calls satisfying', [[account.token], [account2.token]]);
|
|
|
|
});
|
2019-11-27 11:03:32 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should dispatch reset', () => {
|
|
|
|
logoutAll()(dispatch, getState, undefined);
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
expect(dispatch, 'to have a call satisfying', [reset()]);
|
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should redirect to /login', () =>
|
|
|
|
logoutAll()(dispatch, getState, undefined).then(() => {
|
|
|
|
expect(browserHistory.push, 'to have a call satisfying', ['/login']);
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should change user to guest', () =>
|
|
|
|
logoutAll()(dispatch, getState, undefined).then(() => {
|
|
|
|
expect(dispatch, 'to have a call satisfying', [
|
|
|
|
setUser({
|
|
|
|
lang: user.lang,
|
|
|
|
isGuest: true,
|
|
|
|
}),
|
|
|
|
]);
|
|
|
|
}));
|
2019-11-27 11:03:32 +02:00
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('#logoutStrangers', () => {
|
|
|
|
const foreignAccount = {
|
|
|
|
...account,
|
|
|
|
id: 2,
|
|
|
|
refreshToken: null,
|
|
|
|
};
|
|
|
|
|
|
|
|
const foreignAccount2 = {
|
|
|
|
...foreignAccount,
|
|
|
|
id: 3,
|
|
|
|
};
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
(getState as any).returns({
|
|
|
|
accounts: {
|
|
|
|
active: foreignAccount.id,
|
|
|
|
available: [account, foreignAccount, foreignAccount2],
|
|
|
|
},
|
|
|
|
user,
|
|
|
|
});
|
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should remove stranger accounts', () => {
|
|
|
|
logoutStrangers()(dispatch, getState, undefined);
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
expect(dispatch, 'to have a call satisfying', [remove(foreignAccount)]);
|
|
|
|
expect(dispatch, 'to have a call satisfying', [remove(foreignAccount2)]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should logout stranger accounts', () => {
|
|
|
|
logoutStrangers()(dispatch, getState, undefined);
|
|
|
|
|
|
|
|
expect(authentication.logout, 'to have calls satisfying', [
|
|
|
|
[foreignAccount.token],
|
|
|
|
[foreignAccount2.token],
|
|
|
|
]);
|
2016-12-05 21:14:38 +02:00
|
|
|
});
|
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should activate another account if available', () =>
|
|
|
|
logoutStrangers()(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [activate(account)]),
|
|
|
|
));
|
|
|
|
|
|
|
|
it('should not activate another account if active account is already not a stranger', () => {
|
|
|
|
(getState as any).returns({
|
|
|
|
accounts: {
|
|
|
|
active: account.id,
|
|
|
|
available: [account, foreignAccount],
|
|
|
|
},
|
|
|
|
user,
|
|
|
|
});
|
|
|
|
|
|
|
|
return logoutStrangers()(dispatch, getState, undefined).then(() =>
|
|
|
|
expect(dispatch, 'not to have calls satisfying', [activate(account)]),
|
|
|
|
);
|
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should not dispatch if no strangers', () => {
|
|
|
|
(getState as any).returns({
|
|
|
|
accounts: {
|
|
|
|
active: account.id,
|
|
|
|
available: [account],
|
|
|
|
},
|
|
|
|
user,
|
|
|
|
});
|
2017-01-12 07:29:39 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
return logoutStrangers()(dispatch, getState, undefined).then(() => expect(dispatch, 'was not called'));
|
|
|
|
});
|
2017-01-12 07:29:39 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('when all accounts are strangers', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
(getState as any).returns({
|
|
|
|
accounts: {
|
|
|
|
active: foreignAccount.id,
|
|
|
|
available: [foreignAccount, foreignAccount2],
|
|
|
|
},
|
|
|
|
auth: {
|
|
|
|
credentials: {},
|
|
|
|
},
|
|
|
|
user,
|
|
|
|
});
|
|
|
|
|
|
|
|
logoutStrangers()(dispatch, getState, undefined);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('logouts all accounts', () => {
|
|
|
|
expect(authentication.logout, 'to have calls satisfying', [
|
|
|
|
[foreignAccount.token],
|
|
|
|
[foreignAccount2.token],
|
|
|
|
]);
|
|
|
|
|
|
|
|
expect(dispatch, 'to have a call satisfying', [setUser({ isGuest: true })]);
|
|
|
|
|
|
|
|
expect(dispatch, 'to have a call satisfying', [reset()]);
|
|
|
|
});
|
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
describe('when a stranger has a mark in sessionStorage', () => {
|
|
|
|
const key = `stranger${foreignAccount.id}`;
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
beforeEach(() => {
|
|
|
|
sessionStorage.setItem(key, '1');
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
logoutStrangers()(dispatch, getState, undefined);
|
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
afterEach(() => {
|
|
|
|
sessionStorage.removeItem(key);
|
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
it('should not log out', () =>
|
|
|
|
expect(dispatch, 'not to have calls satisfying', [{ payload: foreignAccount }]));
|
|
|
|
});
|
2016-12-05 21:14:38 +02:00
|
|
|
});
|
2016-10-30 14:12:49 +02:00
|
|
|
});
|