mirror of
https://github.com/Unleash/unleash.git
synced 2025-01-20 00:08:02 +01:00
9f3648dc81
## About the changes This PR will validate that our current migrations are backward compatible with the latest stable release of Unleash. It will do so by starting a database, applying the latest migrations, and then starting a docker container with the last stable unleash release and running UI tests against it. There's a risk that the current version of UI tests will not work with the previous version of our UI. Because of that we copied the previous version of cypress tests (https://github.com/Unleash/unleash/tree/5.6/frontend/cypress) into oss folder and removed the ones that are enterprise only. We can discuss a better way of doing this to avoid having to maintain this folder always in sync with the previous version of Unleash This action will only run when there are changes in migrations or to cypress tests.
350 lines
12 KiB
TypeScript
350 lines
12 KiB
TypeScript
///<reference path="../global.d.ts" />
|
|
|
|
import Chainable = Cypress.Chainable;
|
|
import AddStrategyOptions = Cypress.AddFlexibleRolloutStrategyOptions;
|
|
const AUTH_USER = Cypress.env('AUTH_USER');
|
|
const AUTH_PASSWORD = Cypress.env('AUTH_PASSWORD');
|
|
const ENTERPRISE = Boolean(Cypress.env('ENTERPRISE'));
|
|
|
|
let strategyId: string | undefined;
|
|
|
|
const disableActiveSplashScreens = () => {
|
|
return cy.visit(`/splash/operators`);
|
|
};
|
|
|
|
const disableFeatureStrategiesProdGuard = () => {
|
|
localStorage.setItem(
|
|
'useFeatureStrategyProdGuardSettings:v2',
|
|
JSON.stringify({ hide: true }),
|
|
);
|
|
};
|
|
|
|
export const runBefore = () => {
|
|
disableFeatureStrategiesProdGuard();
|
|
disableActiveSplashScreens();
|
|
};
|
|
|
|
export const login_UI = (
|
|
user = AUTH_USER,
|
|
password = AUTH_PASSWORD,
|
|
): Chainable<any> => {
|
|
return cy.session(user, () => {
|
|
cy.visit('/');
|
|
cy.wait(200);
|
|
cy.get("[data-testid='LOGIN_EMAIL_ID']").type(user);
|
|
|
|
if (AUTH_PASSWORD) {
|
|
cy.get("[data-testid='LOGIN_PASSWORD_ID']").type(password);
|
|
}
|
|
|
|
cy.get("[data-testid='LOGIN_BUTTON']").click();
|
|
|
|
// Wait for the login redirect to complete.
|
|
cy.get("[data-testid='HEADER_USER_AVATAR']");
|
|
|
|
if (document.querySelector("[data-testid='CLOSE_SPLASH']")) {
|
|
cy.get("[data-testid='CLOSE_SPLASH']").click();
|
|
}
|
|
});
|
|
};
|
|
|
|
export const createFeature_UI = (
|
|
name: string,
|
|
shouldWait?: boolean,
|
|
project?: string,
|
|
forceInteractions?: boolean,
|
|
): Chainable<any> => {
|
|
const projectName = project || 'default';
|
|
const uiOpts = forceInteractions ? { force: true } : undefined;
|
|
cy.visit(`/projects/${projectName}`);
|
|
cy.get('[data-testid=NAVIGATE_TO_CREATE_FEATURE').click(uiOpts);
|
|
|
|
cy.intercept('POST', `/api/admin/projects/${projectName}/features`).as(
|
|
'createFeature',
|
|
);
|
|
|
|
cy.wait(300);
|
|
|
|
cy.get("[data-testid='CF_NAME_ID'] input").type(name, uiOpts);
|
|
cy.get("[data-testid='CF_DESC_ID'] textarea")
|
|
.first()
|
|
.type('hello-world', uiOpts);
|
|
if (!shouldWait)
|
|
return cy.get("[data-testid='CF_CREATE_BTN_ID']").click(uiOpts);
|
|
else cy.get("[data-testid='CF_CREATE_BTN_ID']").click(uiOpts);
|
|
return cy.wait('@createFeature');
|
|
};
|
|
|
|
export const createProject_UI = (
|
|
projectName: string,
|
|
defaultStickiness: string,
|
|
): Chainable<any> => {
|
|
cy.get('[data-testid=NAVIGATE_TO_CREATE_PROJECT').click();
|
|
|
|
cy.intercept('POST', `/api/admin/projects`).as('createProject');
|
|
|
|
cy.get("[data-testid='PROJECT_ID_INPUT']").type(projectName);
|
|
cy.get("[data-testid='PROJECT_NAME_INPUT']").type(projectName);
|
|
cy.get("[id='stickiness-select']")
|
|
.first()
|
|
.click()
|
|
.get(`[data-testid=SELECT_ITEM_ID-${defaultStickiness}`)
|
|
.first()
|
|
.click();
|
|
cy.get("[data-testid='CREATE_PROJECT_BTN']").click();
|
|
cy.wait('@createProject');
|
|
return cy.visit(`/projects/${projectName}`);
|
|
};
|
|
|
|
export const createSegment_UI = (segmentName: string): Chainable<any> => {
|
|
cy.get("[data-testid='NAVIGATE_TO_CREATE_SEGMENT']").click();
|
|
|
|
cy.intercept('POST', '/api/admin/segments').as('createSegment');
|
|
|
|
cy.get("[data-testid='SEGMENT_NAME_ID']").type(segmentName);
|
|
cy.get("[data-testid='SEGMENT_DESC_ID']").type('hello-world');
|
|
cy.get("[data-testid='SEGMENT_NEXT_BTN_ID']").click();
|
|
cy.get("[data-testid='SEGMENT_CREATE_BTN_ID']").click();
|
|
return cy.wait('@createSegment');
|
|
};
|
|
|
|
export const deleteSegment_UI = (segmentName: string): Chainable<any> => {
|
|
cy.get(`[data-testid='SEGMENT_DELETE_BTN_ID_${segmentName}']`).click();
|
|
|
|
cy.get("[data-testid='SEGMENT_DIALOG_NAME_ID']").type(segmentName);
|
|
return cy.get("[data-testid='DIALOGUE_CONFIRM_ID'").click();
|
|
};
|
|
|
|
export const addFlexibleRolloutStrategyToFeature_UI = (
|
|
options: AddStrategyOptions,
|
|
): Chainable<any> => {
|
|
const { featureToggleName, project, environment, stickiness } = options;
|
|
const projectName = project || 'default';
|
|
const env = environment || 'development';
|
|
const defaultStickiness = stickiness || 'default';
|
|
|
|
cy.visit(`/projects/${projectName}/features/${featureToggleName}`);
|
|
|
|
cy.intercept(
|
|
'POST',
|
|
`/api/admin/projects/${projectName}/features/${featureToggleName}/environments/development/strategies`,
|
|
(req) => {
|
|
expect(req.body.name).to.equal('flexibleRollout');
|
|
expect(req.body.parameters.groupId).to.equal(featureToggleName);
|
|
expect(req.body.parameters.stickiness).to.equal(defaultStickiness);
|
|
expect(req.body.parameters.rollout).to.equal('50');
|
|
|
|
if (ENTERPRISE) {
|
|
expect(req.body.constraints.length).to.equal(1);
|
|
} else {
|
|
expect(req.body.constraints.length).to.equal(0);
|
|
}
|
|
|
|
req.continue((res) => {
|
|
strategyId = res.body.id;
|
|
});
|
|
},
|
|
).as('addStrategyToFeature');
|
|
|
|
cy.visit(
|
|
`/projects/${projectName}/features/${featureToggleName}/strategies/create?environmentId=${env}&strategyName=flexibleRollout`,
|
|
);
|
|
cy.wait(500);
|
|
// Takes a bit to load the screen - this will wait until it finds it or fail
|
|
cy.get('[data-testid=FLEXIBLE_STRATEGY_STICKINESS_ID]');
|
|
if (ENTERPRISE) {
|
|
cy.get('[data-testid=ADD_CONSTRAINT_ID]').click();
|
|
cy.get('[data-testid=DIALOGUE_CONFIRM_ID]').click();
|
|
}
|
|
cy.get(`[data-testid=STRATEGY_FORM_SUBMIT_ID]`).first().click();
|
|
return cy.wait('@addStrategyToFeature');
|
|
};
|
|
|
|
export const updateFlexibleRolloutStrategy_UI = (
|
|
featureToggleName: string,
|
|
projectName?: string,
|
|
) => {
|
|
const project = projectName || 'default';
|
|
cy.visit(
|
|
`/projects/${project}/features/${featureToggleName}/strategies/edit?environmentId=development&strategyId=${strategyId}`,
|
|
);
|
|
|
|
cy.wait(500);
|
|
|
|
cy.get('[data-testid=FLEXIBLE_STRATEGY_STICKINESS_ID]')
|
|
.first()
|
|
.click()
|
|
.get('[data-testid=SELECT_ITEM_ID-sessionId')
|
|
.first()
|
|
.click();
|
|
|
|
cy.wait(500);
|
|
cy.get('[data-testid=FLEXIBLE_STRATEGY_GROUP_ID]')
|
|
.first()
|
|
.clear()
|
|
.type('new-group-id');
|
|
|
|
cy.intercept(
|
|
'PUT',
|
|
`/api/admin/projects/${project}/features/${featureToggleName}/environments/*/strategies/${strategyId}`,
|
|
(req) => {
|
|
expect(req.body.parameters.groupId).to.equal('new-group-id');
|
|
expect(req.body.parameters.stickiness).to.equal('sessionId');
|
|
expect(req.body.parameters.rollout).to.equal('50');
|
|
|
|
if (ENTERPRISE) {
|
|
expect(req.body.constraints.length).to.equal(1);
|
|
} else {
|
|
expect(req.body.constraints.length).to.equal(0);
|
|
}
|
|
|
|
req.continue((res) => {
|
|
expect(res.statusCode).to.equal(200);
|
|
});
|
|
},
|
|
).as('updateStrategy');
|
|
|
|
cy.get(`[data-testid=STRATEGY_FORM_SUBMIT_ID]`).first().click();
|
|
return cy.wait('@updateStrategy');
|
|
};
|
|
|
|
export const deleteFeatureStrategy_UI = (
|
|
featureToggleName: string,
|
|
shouldWait?: boolean,
|
|
projectName?: string,
|
|
): Chainable<any> => {
|
|
const project = projectName || 'default';
|
|
|
|
cy.intercept(
|
|
'DELETE',
|
|
`/api/admin/projects/${project}/features/${featureToggleName}/environments/*/strategies/${strategyId}`,
|
|
(req) => {
|
|
req.continue((res) => {
|
|
expect(res.statusCode).to.equal(200);
|
|
});
|
|
},
|
|
).as('deleteUserStrategy');
|
|
cy.visit(`/projects/${project}/features/${featureToggleName}`);
|
|
cy.get('[data-testid=FEATURE_ENVIRONMENT_ACCORDION_development]').click();
|
|
cy.get('[data-testid=STRATEGY_REMOVE_MENU_BTN]').first().click();
|
|
cy.get('[data-testid=STRATEGY_FORM_REMOVE_ID]').first().click();
|
|
if (!shouldWait) return cy.get('[data-testid=DIALOGUE_CONFIRM_ID]').click();
|
|
else cy.get('[data-testid=DIALOGUE_CONFIRM_ID]').click();
|
|
return cy.wait('@deleteUserStrategy');
|
|
};
|
|
|
|
export const addUserIdStrategyToFeature_UI = (
|
|
featureToggleName: string,
|
|
projectName: string,
|
|
): Chainable<any> => {
|
|
const project = projectName || 'default';
|
|
cy.visit(
|
|
`/projects/${project}/features/${featureToggleName}/strategies/create?environmentId=development&strategyName=userWithId`,
|
|
);
|
|
|
|
if (ENTERPRISE) {
|
|
cy.get('[data-testid=ADD_CONSTRAINT_ID]').click();
|
|
cy.get('[data-testid=CONSTRAINT_AUTOCOMPLETE_ID]')
|
|
.type('{downArrow}'.repeat(1))
|
|
.type('{enter}');
|
|
cy.get('[data-testid=DIALOGUE_CONFIRM_ID]').click();
|
|
}
|
|
|
|
cy.get('[data-testid=STRATEGY_INPUT_LIST]')
|
|
.type('user1')
|
|
.type('{enter}')
|
|
.type('user2')
|
|
.type('{enter}');
|
|
cy.get('[data-testid=ADD_TO_STRATEGY_INPUT_LIST]').click();
|
|
|
|
cy.intercept(
|
|
'POST',
|
|
`/api/admin/projects/default/features/${featureToggleName}/environments/*/strategies`,
|
|
(req) => {
|
|
expect(req.body.name).to.equal('userWithId');
|
|
|
|
expect(req.body.parameters.userIds.length).to.equal(11);
|
|
|
|
if (ENTERPRISE) {
|
|
expect(req.body.constraints.length).to.equal(1);
|
|
} else {
|
|
expect(req.body.constraints.length).to.equal(0);
|
|
}
|
|
|
|
req.continue((res) => {
|
|
strategyId = res.body.id;
|
|
});
|
|
},
|
|
).as('addStrategyToFeature');
|
|
|
|
cy.get(`[data-testid=STRATEGY_FORM_SUBMIT_ID]`).first().click();
|
|
return cy.wait('@addStrategyToFeature');
|
|
};
|
|
|
|
export const addVariantsToFeature_UI = (
|
|
featureToggleName: string,
|
|
variants: Array<string>,
|
|
projectName: string,
|
|
) => {
|
|
const project = projectName || 'default';
|
|
cy.visit(`/projects/${project}/features/${featureToggleName}/variants`);
|
|
cy.wait(200);
|
|
|
|
cy.intercept(
|
|
'PATCH',
|
|
`/api/admin/projects/${project}/features/${featureToggleName}/environments/development/variants`,
|
|
(req) => {
|
|
variants.forEach((variant, index) => {
|
|
expect(req.body[index].op).to.equal('add');
|
|
expect(req.body[index].path).to.equal(`/${index}`);
|
|
expect(req.body[index].value.name).to.equal(variant);
|
|
expect(req.body[index].value.weight).to.equal(
|
|
1000 / variants.length,
|
|
);
|
|
});
|
|
},
|
|
).as('variantCreation');
|
|
|
|
cy.get('[data-testid=ADD_VARIANT_BUTTON]').first().click();
|
|
cy.wait(500);
|
|
variants.forEach((variant, index) => {
|
|
cy.get('[data-testid=VARIANT_NAME_INPUT]').eq(index).type(variant);
|
|
index + 1 < variants.length &&
|
|
cy.get('[data-testid=MODAL_ADD_VARIANT_BUTTON]').first().click();
|
|
});
|
|
|
|
cy.get('[data-testid=DIALOGUE_CONFIRM_ID]').first().click();
|
|
return cy.wait('@variantCreation');
|
|
};
|
|
|
|
export const deleteVariant_UI = (
|
|
featureToggleName: string,
|
|
variant: string,
|
|
projectName?: string,
|
|
): Chainable<any> => {
|
|
const project = projectName || 'default';
|
|
cy.visit(`/projects/${project}/features/${featureToggleName}/variants`);
|
|
cy.get('[data-testid=EDIT_VARIANTS_BUTTON]').click();
|
|
cy.wait(300);
|
|
cy.get(`[data-testid=VARIANT_DELETE_BUTTON_${variant}]`).first().click();
|
|
|
|
cy.intercept(
|
|
'PATCH',
|
|
`/api/admin/projects/${project}/features/${featureToggleName}/environments/development/variants`,
|
|
(req) => {
|
|
expect(req.body[0].op).to.equal('remove');
|
|
expect(req.body[0].path).to.equal('/1');
|
|
expect(req.body[1].op).to.equal('replace');
|
|
expect(req.body[1].path).to.equal('/0/weight');
|
|
expect(req.body[1].value).to.equal(1000);
|
|
},
|
|
).as('delete');
|
|
|
|
cy.get('[data-testid=DIALOGUE_CONFIRM_ID]').click();
|
|
return cy.wait('@delete');
|
|
};
|
|
|
|
export const logout_UI = (): Chainable<any> => {
|
|
return cy.visit('/logout');
|
|
};
|