Cover register and activation with e2e tests

This commit is contained in:
SleepWalker
2019-12-27 22:00:13 +02:00
parent f6f0aedc65
commit c8b0168c69
5 changed files with 216 additions and 47 deletions

View File

@@ -0,0 +1,145 @@
it('should register', () => {
const username = `test${Date.now()}`;
const email = `${Date.now()}@gmail.com`;
const password = String(Date.now());
const captchaCode = 'captchaCode';
const activationKey = 'activationKey';
cy.server();
cy.route({
method: 'POST',
url: '/api/signup',
response: {
success: true,
},
}).as('signup');
cy.login({
accounts: ['default'],
updateState: false,
rawApiResp: true,
}).then(({ accounts: [account] }) => {
cy.route({
method: 'POST',
url: '/api/signup/confirm',
response: account,
}).as('activate');
});
cy.visit('/');
cy.getByTestId('toolbar')
.contains('Join')
.click();
cy.location('pathname').should('eq', '/register');
cy.get('[name=username]').type(username);
cy.get('[name=email]').type(email);
cy.get('[name=password]').type(password);
cy.get('[name=rePassword]').type(password);
cy.get('[name=rulesAgreement]').should('not.be.checked');
cy.get('[name=rulesAgreement]')
.parent()
.click();
cy.get('[name=rulesAgreement]').should('be.checked');
cy.window().should('have.property', 'e2eCaptchaSetCode');
cy.window().then(win => {
// fake captcha response
// @ts-ignore
win.e2eCaptchaSetCode(captchaCode);
});
cy.get('[type=submit]').click();
cy.wait('@signup')
.its('requestBody')
.should(
'eq',
new URLSearchParams({
email,
username,
password,
rePassword: password,
rulesAgreement: '1',
lang: 'en',
captcha: captchaCode,
}).toString(),
);
cy.location('pathname').should('eq', '/activation');
cy.get('[name=key]').type(`${activationKey}{enter}`);
cy.wait('@activate')
.its('requestBody')
.should('eq', `key=${activationKey}`);
cy.location('pathname').should('eq', '/');
});
it('should allow activation', () => {
const activationKey = 'activationKey';
cy.server();
cy.login({
accounts: ['default'],
updateState: false,
rawApiResp: true,
}).then(({ accounts: [account] }) => {
cy.route({
method: 'POST',
url: '/api/signup/confirm',
response: account,
}).as('activate');
});
cy.visit('/register');
cy.getByTestId('auth-secondary-controls')
.contains('Already have')
.click();
cy.location('pathname').should('eq', '/activation');
cy.get('[name=key]').type(`${activationKey}{enter}`);
cy.wait('@activate')
.its('requestBody')
.should('eq', `key=${activationKey}`);
cy.location('pathname').should('eq', '/');
});
it('should allow resend code', () => {
const email = `${Date.now()}@gmail.com`;
const captchaCode = 'captchaCode';
cy.server();
cy.route({
method: 'POST',
url: '/api/signup/repeat-message',
response: { success: true },
}).as('resend');
cy.visit('/register');
cy.getByTestId('auth-secondary-controls')
.contains('not received')
.click();
cy.location('pathname').should('eq', '/resend-activation');
cy.get('[name=email]').type(email);
cy.window().should('have.property', 'e2eCaptchaSetCode');
cy.window().then(win => {
// fake captcha response
// @ts-ignore
win.e2eCaptchaSetCode(captchaCode);
});
cy.get('[type=submit]').click();
cy.wait('@resend')
.its('requestBody')
.should(
'eq',
new URLSearchParams({ email, captcha: captchaCode }).toString(),
);
cy.location('pathname').should('eq', '/activation');
});

View File

@@ -31,47 +31,56 @@ const accountsMap = {
default2: account1,
};
Cypress.Commands.add('login', async ({ accounts }) => {
const accountsData = await Promise.all(
accounts.map(async account => {
let credentials;
Cypress.Commands.add(
'login',
async ({ accounts, updateState = true, rawApiResp = false }) => {
const accountsData = await Promise.all(
accounts.map(async account => {
let credentials;
if (account) {
credentials = accountsMap[account];
if (account) {
credentials = accountsMap[account];
if (!credentials) {
throw new Error(`Unknown account name: ${account}`);
if (!credentials) {
throw new Error(`Unknown account name: ${account}`);
}
}
}
const resp = await fetch('/api/authentication/login', {
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
},
body: `${new URLSearchParams({
login: credentials.login,
password: credentials.password,
rememberMe: '1',
})}`,
}).then(rawResp => rawResp.json());
const resp = await fetch('/api/authentication/login', {
method: 'POST',
headers: {
'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',
},
body: `${new URLSearchParams({
login: credentials.login,
password: credentials.password,
rememberMe: '1',
})}`,
}).then(rawResp => rawResp.json());
return {
id: credentials.id,
username: credentials.username,
email: credentials.email,
token: resp.access_token,
refreshToken: resp.refresh_token,
};
}),
);
if (rawApiResp) {
return resp;
}
const state = createState(accountsData);
return {
id: credentials.id,
username: credentials.username,
email: credentials.email,
token: resp.access_token,
refreshToken: resp.refresh_token,
};
}),
);
localStorage.setItem('redux-storage', JSON.stringify(state));
if (updateState) {
const state = createState(accountsData);
return { accounts: accountsData };
});
localStorage.setItem('redux-storage', JSON.stringify(state));
}
return { accounts: accountsData };
},
);
Cypress.Commands.add('getByTestId', (id, options) =>
cy.get(`[data-testid=${id}]`, options),

View File

@@ -15,10 +15,21 @@ declare namespace Cypress {
/**
* Custom command to log in the user
*
* @example cy.login(account)
* @example cy.login({ accounts: ['default'] })
*/
login(options: {
accounts: AccountAlias[];
/**
* defaults to `true`. if `false` — than only api response will
* be returned without mutating app state
* (useful for custom scenarios such as mocking of other api responses
* or checking whether account is registered)
*/
updateState?: boolean;
/**
* Whether return raw api response without any conversion. Defaults to: `false`
*/
rawApiResp?: boolean;
}): Promise<{ accounts: Account[] }>;
getByTestId<S = any>(