accounts-frontend/tests-e2e/cypress/integration/auth/oauth.test.ts

419 lines
11 KiB
TypeScript
Raw Normal View History

/* eslint-disable @typescript-eslint/camelcase */
import { account1 } from '../../fixtures/accounts.json';
const defaults = {
client_id: 'ely',
2020-01-10 17:57:20 +05:30
redirect_uri: 'https://dev.ely.by/authorization/oauth',
response_type: 'code',
scope: 'account_info,account_email',
};
2019-12-28 15:55:44 +05:30
describe('OAuth', () => {
it('should complete oauth', () => {
cy.login({ accounts: ['default'] });
2019-12-28 15:55:44 +05:30
cy.visit(`/oauth2/v1/ely?${new URLSearchParams(defaults)}`);
2020-01-10 17:57:20 +05:30
cy.url().should('equal', 'https://dev.ely.by/');
2019-12-28 15:55:44 +05:30
});
2019-12-28 15:55:44 +05:30
it('should restore previous oauthData if any', () => {
localStorage.setItem(
'oauthData',
JSON.stringify({
timestamp: Date.now() - 3600,
payload: {
clientId: 'ely',
2020-01-10 17:57:20 +05:30
redirectUrl: 'https://dev.ely.by/authorization/oauth',
2019-12-28 15:55:44 +05:30
responseType: 'code',
description: null,
scope: 'account_info account_email',
loginHint: null,
state: null,
},
}),
);
cy.login({ accounts: ['default'] });
2019-12-28 15:55:44 +05:30
cy.visit('/');
2020-01-10 17:57:20 +05:30
cy.url().should('equal', 'https://dev.ely.by/');
2019-12-28 15:55:44 +05:30
});
2019-12-28 15:55:44 +05:30
it('should ask to choose an account if user has multiple', () => {
cy.login({ accounts: ['default', 'default2'] }).then(
({ accounts: [account] }) => {
cy.visit(`/oauth2/v1/ely?${new URLSearchParams(defaults)}`);
2019-12-28 15:55:44 +05:30
cy.url().should('include', '/oauth/choose-account');
2019-12-28 15:55:44 +05:30
cy.getByTestId('auth-header').should('contain', 'Choose an account');
cy.getByTestId('auth-body').contains(account.email).click();
2020-01-10 17:57:20 +05:30
cy.url().should('equal', 'https://dev.ely.by/');
2019-12-28 15:55:44 +05:30
},
);
});
2019-12-28 15:55:44 +05:30
// TODO: remove api mocks, when we will be able to revoke permissions
it('should prompt for permissions', () => {
cy.server();
cy.route({
method: 'POST',
// NOTE: can not use cypress glob syntax, because it will break due to
// '%2F%2F' (//) in redirect_uri
// url: '/api/oauth2/v1/complete/*',
url: new RegExp('/api/oauth2/v1/complete'),
response: {
statusCode: 401,
error: 'accept_required',
},
status: 401,
}).as('complete');
2019-12-28 15:55:44 +05:30
cy.login({ accounts: ['default'] });
2019-12-28 15:55:44 +05:30
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
client_id: 'tlauncher',
redirect_uri: 'http://localhost:8080',
})}`,
);
2019-12-28 15:55:44 +05:30
cy.wait('@complete');
2019-12-28 15:55:44 +05:30
assertPermissions();
2019-12-28 15:55:44 +05:30
cy.server({ enable: false });
cy.getByTestId('auth-controls').contains('Approve').click();
2019-12-28 15:55:44 +05:30
cy.url().should('match', /^http:\/\/localhost:8080\/?\?code=[^&]+&state=$/);
});
2019-12-28 15:55:44 +05:30
it('should allow sign in during oauth (guest oauth)', () => {
cy.visit(`/oauth2/v1/ely?${new URLSearchParams(defaults)}`);
2019-12-28 15:55:44 +05:30
cy.url().should('include', '/login');
2019-12-28 15:55:44 +05:30
cy.get('[name=login]').type(`${account1.login}{enter}`);
2019-12-28 15:55:44 +05:30
cy.url().should('include', '/password');
2019-12-28 15:55:44 +05:30
cy.get('[name=password]').type(`${account1.password}{enter}`);
2020-01-10 17:57:20 +05:30
cy.url().should('equal', 'https://dev.ely.by/');
2019-12-28 15:55:44 +05:30
});
2019-12-28 15:55:44 +05:30
// TODO: enable, when backend api will return correct response on auth decline
xit('should redirect to error page, when permission request declined', () => {
cy.server();
cy.route({
method: 'POST',
// NOTE: can not use cypress glob syntax, because it will break due to
// '%2F%2F' (//) in redirect_uri
// url: '/api/oauth2/v1/complete/*',
url: new RegExp('/api/oauth2/v1/complete'),
response: {
statusCode: 401,
error: 'accept_required',
},
status: 401,
}).as('complete');
2019-12-28 15:55:44 +05:30
cy.login({ accounts: ['default'] });
2019-12-28 15:55:44 +05:30
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
client_id: 'tlauncher',
redirect_uri: 'http://localhost:8080',
})}`,
);
2019-12-28 15:55:44 +05:30
cy.wait('@complete');
2019-12-28 15:55:44 +05:30
assertPermissions();
2019-12-28 15:55:44 +05:30
cy.server({ enable: false });
cy.getByTestId('auth-controls-secondary').contains('Decline').click();
2019-12-28 15:55:44 +05:30
cy.url().should('include', 'error=access_denied');
});
2019-12-28 15:55:44 +05:30
describe('login_hint', () => {
it('should automatically choose account, when id in login_hint is present', () => {
cy.login({ accounts: ['default', 'default2'] }).then(
({ accounts: [account] }) => {
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
// suggest preferred username
// https://docs.ely.by/ru/oauth.html#id3
2019-12-29 21:56:51 +05:30
login_hint: String(account.id),
}).toString()}`,
2019-12-28 15:55:44 +05:30
);
2020-01-10 17:57:20 +05:30
cy.url().should('equal', 'https://dev.ely.by/');
2019-12-28 15:55:44 +05:30
},
);
});
2019-12-28 15:55:44 +05:30
it('should automatically choose account, when email in login_hint is present', () => {
cy.login({ accounts: ['default', 'default2'] }).then(
({ accounts: [account] }) => {
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
// suggest preferred username
// https://docs.ely.by/ru/oauth.html#id3
login_hint: account.email,
})}`,
);
2020-01-10 17:57:20 +05:30
cy.url().should('equal', 'https://dev.ely.by/');
2019-12-28 15:55:44 +05:30
},
);
});
2019-12-28 15:55:44 +05:30
it('should automatically choose account, when username in login_hint is present and it is not an active account', () => {
cy.login({ accounts: ['default2', 'default'] }).then(
({
// try to authenticate with an account, that is not currently active one
accounts: [, account],
}) => {
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
// suggest preferred username
// https://docs.ely.by/ru/oauth.html#id3
login_hint: account.username,
})}`,
);
2020-01-10 17:57:20 +05:30
cy.url().should('equal', 'https://dev.ely.by/');
2019-12-28 15:55:44 +05:30
},
);
});
});
2019-12-28 15:55:44 +05:30
describe('prompts', () => {
it('should prompt for account', () => {
cy.login({ accounts: ['default'] }).then(({ accounts: [account] }) => {
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
2019-12-28 15:55:44 +05:30
prompt: 'select_account',
})}`,
);
2019-12-28 15:55:44 +05:30
cy.url().should('include', '/oauth/choose-account');
2019-12-28 15:55:44 +05:30
cy.getByTestId('auth-header').should('contain', 'Choose an account');
cy.getByTestId('auth-body').contains(account.email).click();
2020-01-10 17:57:20 +05:30
cy.url().should('equal', 'https://dev.ely.by/');
2019-12-28 15:55:44 +05:30
});
});
it('should allow sign in with another account', () => {
cy.login({ accounts: ['default2'] });
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
prompt: 'select_account',
})}`,
);
cy.url().should('include', '/oauth/choose-account');
cy.getByTestId('auth-controls').contains('another account').click();
2019-12-28 15:55:44 +05:30
cy.url().should('include', '/login');
2019-12-28 15:55:44 +05:30
cy.get('[name=login]').type(`${account1.login}{enter}`);
2019-12-28 15:55:44 +05:30
cy.url().should('include', '/password');
2019-12-28 15:55:44 +05:30
cy.get('[name=password]').type(`${account1.password}{enter}`);
2020-01-10 17:57:20 +05:30
cy.url().should('equal', 'https://dev.ely.by/');
2019-12-28 15:55:44 +05:30
});
2019-12-28 15:55:44 +05:30
it('should prompt for permissions', () => {
cy.login({ accounts: ['default'] });
2019-12-28 15:55:44 +05:30
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
client_id: 'tlauncher',
redirect_uri: 'http://localhost:8080',
prompt: 'consent',
})}`,
);
2019-12-28 15:55:44 +05:30
assertPermissions();
cy.getByTestId('auth-controls').contains('Approve').click();
2019-12-28 15:55:44 +05:30
cy.url().should(
'match',
/^http:\/\/localhost:8080\/?\?code=[^&]+&state=$/,
);
});
2019-12-28 15:55:44 +05:30
// TODO: enable, when backend api will return correct response on auth decline
xit('should redirect to error page, when permission request declined', () => {
cy.login({ accounts: ['default'] });
2019-12-28 15:55:44 +05:30
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
client_id: 'tlauncher',
redirect_uri: 'http://localhost:8080',
prompt: 'consent',
})}`,
);
2019-12-28 15:55:44 +05:30
cy.url().should('include', '/oauth/permissions');
cy.getByTestId('auth-controls-secondary').contains('Decline').click();
2019-12-28 15:55:44 +05:30
cy.url().should('include', 'error=access_denied');
});
2019-12-28 15:55:44 +05:30
it('should prompt for both account and permissions', () => {
cy.login({ accounts: ['default'] }).then(({ accounts: [account] }) => {
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
client_id: 'tlauncher',
redirect_uri: 'http://localhost:8080',
prompt: 'select_account,consent',
})}`,
);
2019-12-28 15:55:44 +05:30
cy.url().should('include', '/oauth/choose-account');
2019-12-28 15:55:44 +05:30
cy.getByTestId('auth-header').should('contain', 'Choose an account');
cy.getByTestId('auth-body').contains(account.email).click();
2019-12-28 15:55:44 +05:30
assertPermissions();
cy.getByTestId('auth-controls').contains('Approve').click();
2019-12-28 15:55:44 +05:30
cy.url().should(
'match',
/^http:\/\/localhost:8080\/?\?code=[^&]+&state=$/,
);
});
});
2019-12-28 15:55:44 +05:30
it('should allow sign in during oauth (guest oauth)', () => {
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
client_id: 'tlauncher',
redirect_uri: 'http://localhost:8080',
prompt: 'select_account,consent',
})}`,
);
2019-12-28 15:55:44 +05:30
cy.url().should('include', '/login');
2019-12-28 15:55:44 +05:30
cy.get('[name=login]').type(`${account1.login}{enter}`);
2019-12-28 15:55:44 +05:30
cy.url().should('include', '/password');
cy.get('[name=password]').type(`${account1.password}{enter}`);
assertPermissions();
cy.getByTestId('auth-controls').contains('Approve').click();
cy.url().should(
'match',
/^http:\/\/localhost:8080\/?\?code=[^&]+&state=$/,
);
});
});
2019-12-28 15:55:44 +05:30
describe('static pages', () => {
it('should authenticate using static page', () => {
2019-12-29 21:56:51 +05:30
cy.server();
cy.route({
method: 'POST',
url: '/api/oauth2/v1/complete**',
}).as('complete');
2019-12-28 15:55:44 +05:30
cy.login({ accounts: ['default'] });
2019-12-28 15:55:44 +05:30
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
client_id: 'tlauncher',
redirect_uri: 'static_page',
})}`,
);
2019-12-29 21:56:51 +05:30
cy.wait('@complete');
2019-12-28 15:55:44 +05:30
cy.url().should('include', 'oauth/finish#{%22auth_code%22:');
});
2019-12-28 15:55:44 +05:30
it('should authenticate using static page with code', () => {
2019-12-29 21:56:51 +05:30
cy.server();
cy.route({
method: 'POST',
url: '/api/oauth2/v1/complete**',
}).as('complete');
2019-12-28 15:55:44 +05:30
cy.login({ accounts: ['default'] });
2019-12-28 15:55:44 +05:30
cy.visit(
`/oauth2/v1/ely?${new URLSearchParams({
...defaults,
client_id: 'tlauncher',
redirect_uri: 'static_page_with_code',
})}`,
);
2019-12-29 21:56:51 +05:30
cy.wait('@complete');
2019-12-28 15:55:44 +05:30
cy.url().should('include', 'oauth/finish#{%22auth_code%22:');
2019-12-28 15:55:44 +05:30
cy.getByTestId('oauth-code-container').should(
'contain',
'provide the following code',
);
2019-12-28 15:55:44 +05:30
// just click on copy, but we won't assert if the string was copied
// because it is a little bit complicated
// https://github.com/cypress-io/cypress/issues/2752
cy.getByTestId('oauth-code-container').contains('Copy').click();
2019-12-28 15:55:44 +05:30
});
});
});
2019-12-29 21:56:51 +05:30
function assertPermissions() {
cy.url().should('include', '/oauth/permissions');
cy.getByTestId('auth-header').should('contain', 'Application permissions');
cy.getByTestId('auth-body').should(
'contain',
'Access to your profile data (except Email)',
);
cy.getByTestId('auth-body').should(
'contain',
'Access to your Email address',
);
}