Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

test: [M3-7498] - Add Cypress tests for Parent/Child billing page UX enhancements #10070

Merged
merged 10 commits into from
Jan 19, 2024
Original file line number Diff line number Diff line change
@@ -0,0 +1,284 @@
/**
* @file Integration tests for restricted user billing flows.
*/

import { profileFactory } from '@src/factories';
import { accountUserFactory } from '@src/factories/accountUsers';
import { grantsFactory } from '@src/factories/grants';
import { mockGetUser } from 'support/intercepts/account';
import {
mockAppendFeatureFlags,
mockGetFeatureFlagClientstream,
} from 'support/intercepts/feature-flags';
import {
mockGetProfile,
mockGetProfileGrants,
} from 'support/intercepts/profile';
import { ui } from 'support/ui';
import { makeFeatureFlagData } from 'support/util/feature-flags';
import { randomLabel } from 'support/util/random';

// Tooltip message that appears on disabled billing action buttons for restricted users.
const restrictedUserTooltip =
'To edit this section, please contact your administrator.';

// Tooltip message that appears on disabled billing action buttons for child users.
const childUserTooltip =
'To edit this section, please contact your business partner.';

/**
* Asserts that the billing contact "Edit" button is disabled.
*
* Additionally confirms that clicking the "Edit" button reveals a tooltip and
* does not open the "Edit Billing Contact Info" drawer.
*
* @param tooltipText - Expected tooltip message to be shown to the user.
*/
const assertEditBillingInfoDisabled = (tooltipText: string) => {
// Confirm Billing Contact section "Edit" button is disabled, then click it.
cy.get('[data-qa-contact-summary]')
.should('be.visible')
.within(() => {
ui.button
.findByTitle('Edit')
.should('be.visible')
.should('be.disabled')
.click();
});

// Assert that "Edit Contact Billing Info" drawer does not open and that tooltip is revealed.
cy.get(`[data-qa-drawer-title="Edit Billing Contact Info"]`).should(
'not.exist'
);
ui.tooltip.findByText(tooltipText).should('be.visible');
};

/**
* Asserts that the billing contact "Edit" button is enabled.
*
* Additionally confirms that clicking the "Edit" button opens the "Edit Billing
* Contact Info" drawer, then closes the drawer.
*/
const assertEditBillingInfoEnabled = () => {
cy.get('[data-qa-contact-summary]')
.should('be.visible')
.within(() => {
ui.button
.findByTitle('Edit')
.should('be.visible')
.should('be.enabled')
.click();
});

ui.drawer
.findByTitle('Edit Billing Contact Info')
.should('be.visible')
.within(() => {
ui.drawerCloseButton.find().click();
});
};

/**
* Asserts that the "Add Payment Method" button is disabled.
*
* Additionally confirms that clicking the "Add Payment Method" button reveals
* a tooltip and does not open the "Add Payment Method" drawer.
*
* @param tooltipText - Expected tooltip message to be shown to the user.
*/
const assertAddPaymentMethodDisabled = (tooltipText: string) => {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@jaalah-akamai quick question occurred to me while writing these tests: should the payment method action menus (or their items) be disabled for restricted and child users?

Screenshot 2024-01-17 at 12 53 47 PM

Copy link
Contributor

@jaalah-akamai jaalah-akamai Jan 17, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes! Edit: Well let me triple check this.
Edit edit: 100% yes!

// Confirm Billing Summary section "Add Payment Method" button is disabled, then click it.
cy.get('[data-qa-billing-summary]')
.should('be.visible')
.within(() => {
ui.button
.findByTitle('Add Payment Method')
.should('be.visible')
.should('be.disabled')
.click();
});

// Assert that "Add Payment Method" drawer does not open and that tooltip is revealed.
cy.get(`[data-qa-drawer-title="Add Payment Method"]`).should('not.exist');
ui.tooltip.findByText(tooltipText).should('be.visible');
};

/**
* Asserts that the "Add Payment Method" button is enabled.
*
* Additionally confirms that clicking the "Add Payment Method" button opens the
* "Add Payment Method" drawer, then closes the drawer.
*/
const assertAddPaymentMethodEnabled = () => {
cy.get('[data-qa-billing-summary]')
.should('be.visible')
.within(() => {
ui.button
.findByTitle('Add Payment Method')
.should('be.visible')
.should('be.enabled')
.click();
});

ui.drawer
.findByTitle('Add Payment Method')
.should('be.visible')
.within(() => {
ui.drawerCloseButton.find().click();
});
};

describe('restricted user billing flows', () => {
// TODO Delete all of these tests when Parent/Child launches and flag is removed.
describe('Parent/Child feature disabled', () => {
beforeEach(() => {
// Mock the Parent/Child feature flag to be enabled.
mockAppendFeatureFlags({
parentChildAccountAccess: makeFeatureFlagData(false),
});
mockGetFeatureFlagClientstream();
});

/*
* - Smoke test to confirm that regular users can edit billing information.
* - Confirms that billing action buttons are enabled and open their respective drawers on click.
*/
it('can edit billing information', () => {
// The flow prior to Parent/Child does not account for user privileges, instead relying
// on the API to forbid actions when the user does not have the required privileges.
// Because the API is doing the heavy lifting, we only need to ensure that the billing action
// buttons behave as expected for this smoke test.
const mockProfile = profileFactory.build({
username: randomLabel(),
restricted: false,
});

const mockUser = accountUserFactory.build({
username: mockProfile.username,
user_type: null,
restricted: false,
});

// Confirm button behavior for regular users.
mockGetProfile(mockProfile);
mockGetUser(mockUser);
cy.visitWithLogin('/account/billing');
assertEditBillingInfoEnabled();
assertAddPaymentMethodEnabled();
});
});

describe('Parent/Child feature enabled', () => {
beforeEach(() => {
// Mock the Parent/Child feature flag to be enabled.
// TODO Delete this `beforeEach()` block when Parent/Child launches and flag is removed.
mockAppendFeatureFlags({
parentChildAccountAccess: makeFeatureFlagData(true),
});
mockGetFeatureFlagClientstream();
});

/*
* - Confirms that users with read-only account access cannot edit billing information.
* - Confirms UX enhancements are applied when parent/child feature flag is enabled.
* - Confirms that "Edit" and "Add Payment Method" buttons are disabled and have informational tooltips.
* - Confirms that clicking "Edit" and "Add Payment Method" does not open their respective drawers when disabled.
* - Confirms that button tooltip text reflects read-only account access.
*/
it('cannot edit billing information with read-only account access', () => {
const mockProfile = profileFactory.build({
username: randomLabel(),
restricted: true,
});

const mockUser = accountUserFactory.build({
username: mockProfile.username,
restricted: true,
user_type: null,
});

const mockGrants = grantsFactory.build({
global: {
account_access: 'read_only',
},
});

mockGetProfile(mockProfile);
mockGetProfileGrants(mockGrants);
mockGetUser(mockUser);
cy.visitWithLogin('/account/billing');

assertEditBillingInfoDisabled(restrictedUserTooltip);
assertAddPaymentMethodDisabled(restrictedUserTooltip);
});

/*
* - Confirms that child users cannot edit billing information.
* - Confirms that UX enhancements are applied when parent/child feature flag is enabled.
* - Confirms that "Edit" and "Add Payment Method" buttons are disabled and have informational tooltips.
* - Confirms that clicking "Edit" and "Add Payment Method" does not open their respective drawers when disabled.
* - Confirms that button tooltip text reflects child user access.
*/
it('cannot edit billing information as child account', () => {
const mockProfile = profileFactory.build({
username: randomLabel(),
});

const mockUser = accountUserFactory.build({
username: mockProfile.username,
user_type: 'child',
});

mockGetProfile(mockProfile);
mockGetUser(mockUser);
cy.visitWithLogin('/account/billing');

assertEditBillingInfoDisabled(childUserTooltip);
assertAddPaymentMethodDisabled(childUserTooltip);
});

/*
* - Smoke test to confirm that regular and parent users can edit billing information.
* - Confirms that billing action buttons are enabled and open their respective drawers on click.
*/
it('can edit billing information as a regular user and as a parent user', () => {
const mockProfileRegular = profileFactory.build({
username: randomLabel(),
restricted: false,
});

const mockUserRegular = accountUserFactory.build({
username: mockProfileRegular.username,
user_type: null,
restricted: false,
});

const mockProfileParent = profileFactory.build({
username: randomLabel(),
restricted: false,
});

const mockUserParent = accountUserFactory.build({
username: mockProfileParent.username,
user_type: 'parent',
restricted: false,
});

// Confirm button behavior for regular users.
mockGetProfile(mockProfileRegular);
mockGetUser(mockUserRegular);
cy.visitWithLogin('/account/billing');
cy.findByText(mockProfileRegular.username);
assertEditBillingInfoEnabled();
assertAddPaymentMethodEnabled();

// Confirm button behavior for parent users.
mockGetProfile(mockProfileParent);
mockGetUser(mockUserParent);
cy.visitWithLogin('/account/billing');
cy.findByText(mockProfileParent.username);
assertEditBillingInfoEnabled();
assertAddPaymentMethodEnabled();
});
});
});
20 changes: 19 additions & 1 deletion packages/manager/cypress/support/intercepts/profile.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import { paginateResponse } from 'support/util/paginate';
import { makeResponse } from 'support/util/response';

import type {
Grants,
OAuthClient,
Profile,
SecurityQuestionsData,
Expand Down Expand Up @@ -46,7 +47,24 @@ export const mockGetProfile = (profile: Profile): Cypress.Chainable<null> => {
export const mockUpdateProfile = (
profile: Profile
): Cypress.Chainable<null> => {
return cy.intercept('PUT', apiMatcher(`profile`), makeResponse(profile));
return cy.intercept('PUT', apiMatcher('profile'), makeResponse(profile));
};

/**
* Intercepts GET request to fetch profile grants and mocks response.
*
* @param grants - Grants object with which to mock response.
*
* @returns Cypress chainable.
*/
export const mockGetProfileGrants = (
grants: Grants
): Cypress.Chainable<null> => {
return cy.intercept(
'GET',
apiMatcher('profile/grants'),
makeResponse(grants)
);
};

/**
Expand Down
2 changes: 2 additions & 0 deletions packages/manager/cypress/support/ui/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ import * as select from './select';
import * as tabList from './tab-list';
import * as toast from './toast';
import * as toggle from './toggle';
import * as tooltip from './tooltip';
import * as userMenu from './user-menu';

export const ui = {
Expand All @@ -35,5 +36,6 @@ export const ui = {
...toast,
...tabList,
...toggle,
...tooltip,
...userMenu,
};
11 changes: 11 additions & 0 deletions packages/manager/cypress/support/ui/tooltip.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
/**
* Tooltip UI helper.
*/
export const tooltip = {
/**
* Finds a tooltip that has the given text.
*/
findByText: (text: string): Cypress.Chainable => {
return cy.document().its('body').find(`[data-qa-tooltip="${text}"]`);
},
};
2 changes: 1 addition & 1 deletion packages/manager/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -209,7 +209,7 @@
"ts-node": "^10.9.2",
"vite": "^5.0.7",
"vite-plugin-svgr": "^3.2.0",
"vitest": "^1.0.4"
"vitest": "^1.2.0"
},
"browserslist": [
">1%",
Expand Down
2 changes: 1 addition & 1 deletion packages/manager/src/components/Tooltip.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import type { TooltipProps } from '@mui/material/Tooltip';
* Tooltips display informative text when users hover over, focus on, or tap an element.
*/
export const Tooltip = (props: TooltipProps) => {
return <_Tooltip {...props} />;
return <_Tooltip data-qa-tooltip={props.title} {...props} />;
};
export { tooltipClasses };
export type { TooltipProps };
Loading