2016-08-09 22:17:49 +03:00
|
|
|
import expect from 'unexpected';
|
|
|
|
|
|
|
|
import refreshTokenMiddleware from 'components/user/middlewares/refreshTokenMiddleware';
|
|
|
|
|
|
|
|
import authentication from 'services/api/authentication';
|
2016-11-05 12:11:41 +02:00
|
|
|
import { updateToken } from 'components/accounts/actions';
|
2016-08-09 22:17:49 +03:00
|
|
|
|
|
|
|
const refreshToken = 'foo';
|
|
|
|
const expiredToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJuYmYiOjE0NzA3NjE0NDMsImV4cCI6MTQ3MDc2MTQ0MywiaWF0IjoxNDcwNzYxNDQzLCJqdGkiOiJpZDEyMzQ1NiJ9.gWdnzfQQvarGpkbldUvB8qdJZSVkvdNtCbhbbl2yJW8';
|
|
|
|
// valid till 2100 year
|
|
|
|
const validToken = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJuYmYiOjE0NzA3NjE5NzcsImV4cCI6NDEwMjQ0NDgwMCwiaWF0IjoxNDcwNzYxOTc3LCJqdGkiOiJpZDEyMzQ1NiJ9.M4KY4QgHOUzhpAZjWoHJbGsEJPR-RBsJ1c1BKyxvAoU';
|
|
|
|
|
|
|
|
describe('refreshTokenMiddleware', () => {
|
|
|
|
let middleware;
|
|
|
|
let getState;
|
|
|
|
let dispatch;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
sinon.stub(authentication, 'requestToken').named('authentication.requestToken');
|
2016-11-15 07:55:15 +02:00
|
|
|
sinon.stub(authentication, 'logout').named('authentication.logout');
|
2016-08-09 22:17:49 +03:00
|
|
|
|
|
|
|
getState = sinon.stub().named('store.getState');
|
2016-11-05 12:11:41 +02:00
|
|
|
dispatch = sinon.spy((arg) =>
|
|
|
|
typeof arg === 'function' ? arg(dispatch, getState) : arg
|
|
|
|
).named('store.dispatch');
|
2016-08-09 22:17:49 +03:00
|
|
|
|
|
|
|
middleware = refreshTokenMiddleware({getState, dispatch});
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
authentication.requestToken.restore();
|
2016-11-15 07:55:15 +02:00
|
|
|
authentication.logout.restore();
|
2016-08-09 22:17:49 +03:00
|
|
|
});
|
|
|
|
|
2016-11-05 12:11:41 +02:00
|
|
|
it('must be till 2100 to test with validToken', () =>
|
|
|
|
expect(new Date().getFullYear(), 'to be less than', 2100)
|
|
|
|
);
|
|
|
|
|
2016-08-09 22:17:49 +03:00
|
|
|
describe('#before', () => {
|
2016-10-30 14:12:49 +02:00
|
|
|
describe('when token expired', () => {
|
|
|
|
beforeEach(() => {
|
2016-11-15 07:55:15 +02:00
|
|
|
const account = {
|
|
|
|
token: expiredToken,
|
|
|
|
refreshToken
|
|
|
|
};
|
2016-10-30 14:12:49 +02:00
|
|
|
getState.returns({
|
2016-11-05 12:11:41 +02:00
|
|
|
accounts: {
|
2016-11-15 07:55:15 +02:00
|
|
|
active: account,
|
|
|
|
available: [account]
|
2016-11-05 12:11:41 +02:00
|
|
|
},
|
|
|
|
user: {}
|
2016-10-30 14:12:49 +02:00
|
|
|
});
|
2016-08-09 22:17:49 +03:00
|
|
|
});
|
|
|
|
|
2016-10-30 14:12:49 +02:00
|
|
|
it('should request new token', () => {
|
|
|
|
const data = {
|
|
|
|
url: 'foo',
|
|
|
|
options: {
|
|
|
|
headers: {}
|
|
|
|
}
|
|
|
|
};
|
2016-08-09 22:17:49 +03:00
|
|
|
|
2016-10-30 14:12:49 +02:00
|
|
|
authentication.requestToken.returns(Promise.resolve({token: validToken}));
|
|
|
|
|
|
|
|
return middleware.before(data).then((resp) => {
|
|
|
|
expect(resp, 'to satisfy', data);
|
|
|
|
|
|
|
|
expect(authentication.requestToken, 'to have a call satisfying', [
|
|
|
|
refreshToken
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not apply to refresh-token request', () => {
|
2016-11-12 22:31:44 +02:00
|
|
|
const data = {url: '/refresh-token', options: {}};
|
2016-10-30 14:12:49 +02:00
|
|
|
const resp = middleware.before(data);
|
2016-08-09 22:17:49 +03:00
|
|
|
|
|
|
|
expect(resp, 'to satisfy', data);
|
|
|
|
|
2016-10-30 14:12:49 +02:00
|
|
|
expect(authentication.requestToken, 'was not called');
|
2016-08-09 22:17:49 +03:00
|
|
|
});
|
|
|
|
|
2016-11-08 08:30:53 +02:00
|
|
|
it('should not auto refresh token if options.token specified', () => {
|
2016-10-30 14:12:49 +02:00
|
|
|
const data = {
|
|
|
|
url: 'foo',
|
2016-11-08 08:30:53 +02:00
|
|
|
options: {token: 'foo'}
|
2016-10-30 14:12:49 +02:00
|
|
|
};
|
|
|
|
middleware.before(data);
|
|
|
|
|
|
|
|
expect(authentication.requestToken, 'was not called');
|
2016-08-09 22:17:49 +03:00
|
|
|
});
|
|
|
|
|
2016-11-05 12:11:41 +02:00
|
|
|
it('should update user with new token', () => {
|
|
|
|
const data = {
|
|
|
|
url: 'foo',
|
|
|
|
options: {
|
|
|
|
headers: {}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
authentication.requestToken.returns(Promise.resolve({token: validToken}));
|
|
|
|
|
|
|
|
return middleware.before(data).then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [
|
|
|
|
updateToken(validToken)
|
|
|
|
])
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should if token can not be parsed', () => {
|
2016-11-15 07:55:15 +02:00
|
|
|
const account = {
|
|
|
|
token: 'realy bad token',
|
|
|
|
refreshToken
|
|
|
|
};
|
2016-11-05 12:11:41 +02:00
|
|
|
getState.returns({
|
|
|
|
accounts: {
|
2016-11-15 07:55:15 +02:00
|
|
|
active: account,
|
|
|
|
available: [account]
|
2016-11-05 12:11:41 +02:00
|
|
|
},
|
|
|
|
user: {}
|
|
|
|
});
|
|
|
|
|
|
|
|
const req = {url: 'foo', options: {}};
|
|
|
|
|
|
|
|
return expect(middleware.before(req), 'to be fulfilled with', req).then(() => {
|
|
|
|
expect(authentication.requestToken, 'was not called');
|
2016-08-09 22:17:49 +03:00
|
|
|
|
2016-11-05 12:11:41 +02:00
|
|
|
expect(dispatch, 'to have a call satisfying', [
|
|
|
|
{payload: {isGuest: true}}
|
|
|
|
]);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should logout if token request failed', () => {
|
2016-10-30 14:12:49 +02:00
|
|
|
authentication.requestToken.returns(Promise.reject());
|
2016-08-09 22:17:49 +03:00
|
|
|
|
2016-11-05 12:11:41 +02:00
|
|
|
return expect(middleware.before({url: 'foo', options: {}}), 'to be fulfilled').then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [
|
|
|
|
{payload: {isGuest: true}}
|
|
|
|
])
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when token expired legacy user', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
getState.returns({
|
|
|
|
accounts: {
|
2016-11-15 07:55:15 +02:00
|
|
|
active: null,
|
|
|
|
available: []
|
2016-11-05 12:11:41 +02:00
|
|
|
},
|
|
|
|
user: {
|
|
|
|
token: expiredToken,
|
|
|
|
refreshToken
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should request new token', () => {
|
|
|
|
const data = {
|
|
|
|
url: 'foo',
|
|
|
|
options: {
|
|
|
|
headers: {}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
authentication.requestToken.returns(Promise.resolve({token: validToken}));
|
|
|
|
|
|
|
|
return middleware.before(data).then((resp) => {
|
|
|
|
expect(resp, 'to satisfy', data);
|
|
|
|
|
|
|
|
expect(authentication.requestToken, 'to have a call satisfying', [
|
|
|
|
refreshToken
|
|
|
|
]);
|
2016-10-30 14:12:49 +02:00
|
|
|
});
|
|
|
|
});
|
2016-08-09 22:17:49 +03:00
|
|
|
});
|
|
|
|
|
2016-10-30 14:12:49 +02:00
|
|
|
it('should not be applied if no token', () => {
|
2016-08-09 22:17:49 +03:00
|
|
|
getState.returns({
|
2016-11-05 12:11:41 +02:00
|
|
|
accounts: {
|
|
|
|
active: null
|
|
|
|
},
|
2016-08-09 22:17:49 +03:00
|
|
|
user: {}
|
|
|
|
});
|
|
|
|
|
2016-10-30 14:12:49 +02:00
|
|
|
const data = {url: 'foo'};
|
2016-08-09 22:17:49 +03:00
|
|
|
const resp = middleware.before(data);
|
|
|
|
|
|
|
|
expect(resp, 'to satisfy', data);
|
|
|
|
|
|
|
|
expect(authentication.requestToken, 'was not called');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('#catch', () => {
|
2016-10-30 14:12:49 +02:00
|
|
|
const expiredResponse = {
|
|
|
|
name: 'Unauthorized',
|
|
|
|
message: 'Token expired',
|
|
|
|
code: 0,
|
|
|
|
status: 401,
|
|
|
|
type: 'yii\\web\\UnauthorizedHttpException'
|
|
|
|
};
|
|
|
|
|
|
|
|
const badTokenReponse = {
|
|
|
|
name: 'Unauthorized',
|
2016-11-05 12:11:41 +02:00
|
|
|
message: 'You are requesting with an invalid credential.',
|
|
|
|
code: 0,
|
|
|
|
status: 401,
|
|
|
|
type: 'yii\\web\\UnauthorizedHttpException'
|
|
|
|
};
|
|
|
|
|
|
|
|
const incorrectTokenReponse = {
|
|
|
|
name: 'Unauthorized',
|
|
|
|
message: 'Incorrect token',
|
2016-10-30 14:12:49 +02:00
|
|
|
code: 0,
|
|
|
|
status: 401,
|
|
|
|
type: 'yii\\web\\UnauthorizedHttpException'
|
|
|
|
};
|
|
|
|
|
|
|
|
let restart;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
2016-08-09 22:17:49 +03:00
|
|
|
getState.returns({
|
2016-11-05 12:11:41 +02:00
|
|
|
accounts: {
|
2016-11-15 07:55:15 +02:00
|
|
|
active: {refreshToken},
|
|
|
|
available: [{refreshToken}]
|
2016-11-05 12:11:41 +02:00
|
|
|
},
|
|
|
|
user: {}
|
2016-08-09 22:17:49 +03:00
|
|
|
});
|
|
|
|
|
2016-10-30 14:12:49 +02:00
|
|
|
restart = sinon.stub().named('restart');
|
2016-08-09 22:17:49 +03:00
|
|
|
|
|
|
|
authentication.requestToken.returns(Promise.resolve({token: validToken}));
|
2016-10-30 14:12:49 +02:00
|
|
|
});
|
2016-08-09 22:17:49 +03:00
|
|
|
|
2016-10-30 14:12:49 +02:00
|
|
|
it('should request new token if expired', () =>
|
|
|
|
middleware.catch(expiredResponse, {options: {}}, restart).then(() => {
|
2016-08-09 22:17:49 +03:00
|
|
|
expect(authentication.requestToken, 'to have a call satisfying', [
|
|
|
|
refreshToken
|
|
|
|
]);
|
|
|
|
expect(restart, 'was called');
|
2016-10-30 14:12:49 +02:00
|
|
|
})
|
|
|
|
);
|
|
|
|
|
2016-11-05 12:11:41 +02:00
|
|
|
it('should logout user if invalid credential', () =>
|
|
|
|
expect(
|
|
|
|
middleware.catch(badTokenReponse, {options: {}}, restart),
|
|
|
|
'to be rejected'
|
|
|
|
).then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [
|
|
|
|
{payload: {isGuest: true}}
|
|
|
|
])
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
it('should logout user if token is incorrect', () =>
|
|
|
|
expect(
|
|
|
|
middleware.catch(incorrectTokenReponse, {options: {}}, restart),
|
|
|
|
'to be rejected'
|
|
|
|
).then(() =>
|
|
|
|
expect(dispatch, 'to have a call satisfying', [
|
|
|
|
{payload: {isGuest: true}}
|
|
|
|
])
|
|
|
|
)
|
|
|
|
);
|
2016-08-09 22:17:49 +03:00
|
|
|
|
2016-11-08 08:30:53 +02:00
|
|
|
it('should pass the request through if options.token specified', () => {
|
2016-10-30 14:12:49 +02:00
|
|
|
const promise = middleware.catch(expiredResponse, {
|
|
|
|
options: {
|
2016-11-08 08:30:53 +02:00
|
|
|
token: 'foo'
|
2016-10-30 14:12:49 +02:00
|
|
|
}
|
|
|
|
}, restart);
|
|
|
|
|
|
|
|
return expect(promise, 'to be rejected with', expiredResponse).then(() => {
|
|
|
|
expect(restart, 'was not called');
|
|
|
|
expect(authentication.requestToken, 'was not called');
|
|
|
|
});
|
|
|
|
});
|
2016-08-09 22:17:49 +03:00
|
|
|
|
|
|
|
it('should pass the rest of failed requests through', () => {
|
|
|
|
const resp = {};
|
|
|
|
|
2016-10-30 14:12:49 +02:00
|
|
|
const promise = middleware.catch(resp, {
|
|
|
|
options: {}
|
|
|
|
}, restart);
|
2016-08-09 22:17:49 +03:00
|
|
|
|
2016-10-30 14:12:49 +02:00
|
|
|
return expect(promise, 'to be rejected with', resp).then(() => {
|
|
|
|
expect(restart, 'was not called');
|
|
|
|
expect(authentication.requestToken, 'was not called');
|
2016-08-09 22:17:49 +03:00
|
|
|
});
|
|
|
|
});
|
2016-11-05 12:11:41 +02:00
|
|
|
|
|
|
|
describe('legacy user.refreshToken', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
getState.returns({
|
|
|
|
accounts: {
|
|
|
|
active: null
|
|
|
|
},
|
|
|
|
user: {refreshToken}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should request new token if expired', () =>
|
|
|
|
middleware.catch(expiredResponse, {options: {}}, restart).then(() => {
|
|
|
|
expect(authentication.requestToken, 'to have a call satisfying', [
|
|
|
|
refreshToken
|
|
|
|
]);
|
|
|
|
expect(restart, 'was called');
|
|
|
|
})
|
|
|
|
);
|
|
|
|
});
|
2016-08-09 22:17:49 +03:00
|
|
|
});
|
|
|
|
});
|