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: add simple notifications test, add page objects for test #27096

Closed
wants to merge 33 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
567c991
test: add simple notifications test, add page objects for test
cmd-ob Sep 12, 2024
89b6f49
Merge branch 'develop' into e2e/enable-notifications
cmd-ob Sep 17, 2024
f0e999c
fix mocks imports
cmd-ob Sep 17, 2024
4a1f27d
fix linting for privacy snapshot
cmd-ob Sep 17, 2024
37268cd
Merge branch 'develop' into e2e/enable-notifications
cmd-ob Sep 17, 2024
049afbb
use .ts for test file
cmd-ob Sep 17, 2024
a717b01
Merge branch 'develop' into e2e/enable-notifications
cmd-ob Sep 17, 2024
7b98884
fix: selector getKnownMethodData should return empty object if user h…
jpuri Sep 17, 2024
c9869bd
fix: Update max gas limit with value returned from eth_estimateGas ap…
jpuri Sep 17, 2024
2e25f51
test: [POM] create AccountListPage for e2e tests and migrate 4 test f…
cmd-ob Sep 18, 2024
7bf5969
chore(3210): deprecated `Application open` metric event (#27102)
DDDDDanica Sep 17, 2024
d7129ea
fix: Fixed flaky test for custom screen size (#27067)
hjetpoluru Sep 17, 2024
c1bdb49
chore: remove dead code (#27211)
cryptotavares Sep 17, 2024
8e40790
feat: upgrade notification controllers (#27224)
Prithpal-Sooriya Sep 17, 2024
f8b2c81
chore: Add currency conversion telemetry (#26876)
gambinish Sep 17, 2024
f1cf41a
fix: :pencil2: fix event property typo (#27228)
matteoscurati Sep 18, 2024
ce892c5
test: add simple notifications test, add page objects for test
cmd-ob Sep 12, 2024
4423794
update user storage id
cmd-ob Sep 18, 2024
0d821e2
Merge branch 'develop' into e2e/enable-notifications
cmd-ob Sep 18, 2024
8714172
lint fix, make test .ts
cmd-ob Sep 18, 2024
c7f7fb2
Merge branch 'develop' into e2e/enable-notifications
cmd-ob Sep 18, 2024
84e6b68
add logging to assertion, move go to notifications into header page
cmd-ob Sep 19, 2024
aa86704
Merge branch 'e2e/enable-notifications' of github.com:MetaMask/metama…
cmd-ob Sep 19, 2024
914d372
Merge branch 'develop' into e2e/enable-notifications
cmd-ob Sep 19, 2024
57f4446
remove account options menu page
cmd-ob Sep 23, 2024
15ab081
update mock var name
cmd-ob Sep 23, 2024
3b01e86
Merge branch 'develop' of github.com:MetaMask/metamask-extension into…
cmd-ob Sep 23, 2024
8420d5b
update mocks
cmd-ob Oct 7, 2024
63de77a
Merge branch 'develop' of github.com:MetaMask/metamask-extension into…
cmd-ob Oct 7, 2024
09e9f2b
remove irrelevant mocks
cmd-ob Oct 13, 2024
8cf86a7
use import onboarding flow, update mocks for account and notification…
cmd-ob Oct 18, 2024
63aa7c1
Merge branch 'develop' of github.com:MetaMask/metamask-extension into…
cmd-ob Oct 18, 2024
c50fb47
update mock, change folder name to profile-sync
cmd-ob Oct 18, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions privacy-snapshot.json
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@
"portfolio.metamask.io",
"price.api.cx.metamask.io",
"proxy.api.cx.metamask.io",
"push.api.cx.metamask.io",
"proxy.dev-api.cx.metamask.io",
"raw.githubusercontent.com",
"registry.npmjs.org",
Expand All @@ -57,6 +58,7 @@
"test.metamask-phishing.io",
"token.api.cx.metamask.io",
"tokens.api.cx.metamask.io",
"trigger.api.cx.metamask.io",
"transaction.api.cx.metamask.io",
"tx-sentinel-ethereum-mainnet.api.cx.metamask.io",
"unresponsive-rpc.test",
Expand Down
34 changes: 34 additions & 0 deletions test/e2e/fixture-builder.js
Original file line number Diff line number Diff line change
Expand Up @@ -146,6 +146,32 @@ function onboardingFixture() {
ignoredTokens: [],
tokens: [],
},
// FIXME: is this correct?
NotificationsController: {
notifications: {},
metamaskNotificationsList: [
{
type: 'features_announcement',
id: 'enhanced-signatures',
createdAt: '2024-09-27T13:25:59.000Z',
data: {
id: 'enhanced-signatures',
category: 'ANNOUNCEMENT',
title: 'Enhanced signatures',
longDescription:
'<p>We’re improving signature requests to increase readability, so you can have more control over your assets. Stay tuned for more updates.</p>',
shortDescription:
"An update that makes it easier to understand and review the actions you're about to approve.",
image: {
title: 'Enhanced signature confirmations',
description: 'Enhanced signature confirmations',
url: '//images.ctfassets.net/jdkgyfmyd9sw/3yXPGO8LxTT7RrrSkotNNz/d3635fcef422aeb607016150e4fd5b85/24-09-10_MwM_Blog-Post-Signature-Redesign_1920x1280.png',
},
},
isRead: false,
},
],
},
TransactionController: {},
config: {},
firstTimeInfo: {
Expand Down Expand Up @@ -1321,6 +1347,14 @@ class FixtureBuilder {
return this.withNameController({ names: {} });
}

// FIXME: Is this how to set the notifications before starting test?
withMetamaskNotifications(notifications) {
merge(this.fixture.data.NotificationsController, {
notifications,
});
return this;
}

withTrezorAccount() {
return this.withAccountTracker({
accounts: {
Expand Down
2 changes: 1 addition & 1 deletion test/e2e/mock-e2e.js
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@ const blacklistedHosts = [
const {
mockEmptyStalelistAndHotlist,
} = require('./tests/phishing-controller/mocks');
const { mockNotificationServices } = require('./tests/notifications/mocks');
const { mockNotificationServices } = require('./tests/profile-sync/mocks');

const emptyHtmlPage = () => `<!DOCTYPE html>
<html lang="en">
Expand Down
24 changes: 24 additions & 0 deletions test/e2e/page-objects/flows/enable-notifications.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
import { Driver } from '../../webdriver/driver';
import FirstTimeTurnOnNotificationsModal from '../pages/notifications/first-time-turn-on-modal';
import HeaderNavbar from '../pages/header-navbar';

/**
* This function enables the notifications through the header options menu from the home page
*
* Note: this flow focuses on the journey of a user who is enabling this feature for the first time
*
* @param driver - The webdriver instance.
*/
export const enableNotificationsFirstTime = async (
driver: Driver,
): Promise<void> => {
console.log(`Start enable notifications from home screen`);
const header = new HeaderNavbar(driver);
await header.goToNotifiationsList();

const turnOnNotificationsModal = new FirstTimeTurnOnNotificationsModal(
driver,
);
await turnOnNotificationsModal.check_pageIsLoaded();
await turnOnNotificationsModal.clickTurnOnNotifications();
};
22 changes: 19 additions & 3 deletions test/e2e/page-objects/pages/header-navbar.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,11 +5,14 @@ class HeaderNavbar {

private readonly accountMenuButton = '[data-testid="account-menu-icon"]';

private accountOptionsMenuButton: string;
private readonly accountOptionMenu =
'[data-testid="account-options-menu-button"]';

private readonly accountSnapButton = { text: 'Snaps', tag: 'div' };

private notificationsMenuItem: string;

private readonly lockMetaMaskButton = '[data-testid="global-menu-lock"]';

private readonly mmiPortfolioButton =
Expand All @@ -19,6 +22,14 @@ class HeaderNavbar {

constructor(driver: Driver) {
this.driver = driver;
this.lockMetaMaskButton = '[data-testid="global-menu-lock"]';
this.accountMenuButton = '[data-testid="account-menu-icon"]';
this.accountOptionsMenuButton =
'[data-testid="account-options-menu-button"]';
this.notificationsMenuItem = '[data-testid="notifications-menu-item"]'
this.mmiPortfolioButton = '[data-testid="global-menu-mmi-portfolio"]';
this.settingsButton = '[data-testid="global-menu-settings"]';
this.accountSnapButton = { text: 'Snaps', tag: 'div' };
}

async check_pageIsLoaded(): Promise<void> {
Expand All @@ -35,7 +46,7 @@ class HeaderNavbar {
}

async lockMetaMask(): Promise<void> {
await this.driver.clickElement(this.accountOptionMenu);
await this.driver.clickElement(this.accountMenuButton);
// fix race condition with mmi build
if (process.env.MMI) {
await this.driver.waitForSelector(this.mmiPortfolioButton);
Expand All @@ -49,13 +60,13 @@ class HeaderNavbar {

async openSnapListPage(): Promise<void> {
console.log('Open account snap page');
await this.driver.clickElement(this.accountOptionMenu);
await this.driver.clickElement(this.accountMenuButton);
await this.driver.clickElement(this.accountSnapButton);
}

async openSettingsPage(): Promise<void> {
console.log('Open settings page');
await this.driver.clickElement(this.accountOptionMenu);
await this.driver.clickElement(this.accountMenuButton);
// fix race condition with mmi build
if (process.env.MMI) {
await this.driver.waitForSelector(this.mmiPortfolioButton);
Expand All @@ -77,6 +88,11 @@ class HeaderNavbar {
text: expectedLabel,
});
}

async goToNotifiationsList(): Promise<void> {
await this.driver.clickElement(this.accountOptionsMenuButton);
await this.driver.clickElement(this.notificationsMenuItem)
}
Copy link
Contributor

Choose a reason for hiding this comment

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

With this implementation, for notifications, we use clickNotificationsMenuItem in the AccountOptionsMenu page, but we click lockMetaMaskButton using the selectors in the header-navbar page directly. However, the notificationsMenuItem and the lockMetaMaskButton are on the same modal, which creates inconsistency. I suggest that we don't create a new page class for AccountOptionsMenu because it only has a few buttons without complex interactions on AccountOptionsMenu. We can just add the notificationsMenuItem selector in the header-navbar page and click on it directly here. This provides an easier implementation.
Screenshot 2024-09-20 at 09 35 19

}

export default HeaderNavbar;
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
import { Driver } from '../../../webdriver/driver';

class FirstTimeTurnOnNotificationsModal {
private driver: Driver;

private readonly turnOnButton =
'[data-testid="turn-on-notifications-button"]';

constructor(driver: Driver) {
this.driver = driver;
}

async check_pageIsLoaded(): Promise<void> {
try {
await this.driver.waitForSelector(this.turnOnButton);
} catch (e) {
console.log(
'Timeout while waiting for TurnOnNotificationsModal to be loaded',
e,
);
throw e;
}
console.log('TurnOnNotificationsModal is loaded');
}

async clickTurnOnNotifications(): Promise<void> {
await this.driver.clickElementAndWaitToDisappear(this.turnOnButton, 10000);
}
}

export default FirstTimeTurnOnNotificationsModal;
49 changes: 49 additions & 0 deletions test/e2e/page-objects/pages/notifications/notifications-list.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
import { Driver } from '../../../webdriver/driver';

class NotificationsListPage {
private driver: Driver;

private readonly backButton = '[data-testid="back-button"]';

private readonly notificationsSettingsButton =
'[data-testid="notifications-settings-button"]';

private readonly noNotificationsReceivedPlaceholder =
'[data-testid="notifications-list-placeholder"]';

constructor(driver: Driver) {
this.driver = driver;
}

async check_pageIsLoaded(): Promise<void> {
try {
await this.driver.waitForMultipleSelectors([
this.backButton,
this.notificationsSettingsButton,
]);
} catch (e) {
console.log(
'Timeout while waiting for notifications list page to be loaded',
e,
);
throw e;
}
console.log('Notifications list page is loaded');
}

async check_noNotificationsReceived(): Promise<void> {
chloeYue marked this conversation as resolved.
Show resolved Hide resolved
try {
await this.driver.waitForSelector(
this.noNotificationsReceivedPlaceholder,
);
} catch (e) {
console.log(
'Timed out while waiting for the empty notifications list placeholder to be displayed',
e,
);
}
console.log('Notifications list is empty as expected');
}
}

export default NotificationsListPage;
45 changes: 45 additions & 0 deletions test/e2e/tests/profile-sync/enable-notifications-first-time.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
import {
withFixtures,
defaultGanacheOptions,
completeImportSRPOnboardingFlow,
} from '../../helpers';
import { enableNotificationsFirstTime } from '../../page-objects/flows/enable-notifications';
import FixtureBuilder from '../../fixture-builder';
import NotificationsListPage from '../../page-objects/pages/notifications/notifications-list';
import { mockNotificationServices } from './mocks';

const NOTIFICATIONS_TEAM_SEED_PHRASE =
'leisure swallow trip elbow prison wait rely keep supply hole general mountain';

const PASSWORD = 'notify_password';

describe('Notifications', function () {
describe('from inside MetaMask', function () {
it('enables notifications for the first time', async function () {
await withFixtures(
{
fixtures: new FixtureBuilder({ onboarding: true })
.withMetamaskNotifications()
.build(),
ganacheOptions: defaultGanacheOptions,
title: this.test?.fullTitle(),
testSpecificMock: mockNotificationServices,
},
async ({ driver }) => {
await driver.navigate();
await completeImportSRPOnboardingFlow(
driver,
NOTIFICATIONS_TEAM_SEED_PHRASE,
PASSWORD,
);
await enableNotificationsFirstTime(driver);

// should land on notifications list
const notificationsListPage = new NotificationsListPage(driver);
await notificationsListPage.check_pageIsLoaded();
await notificationsListPage.check_noNotificationsReceived();
},
);
});
});
});
17 changes: 17 additions & 0 deletions test/e2e/tests/profile-sync/mockData.ts

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
@@ -1,15 +1,12 @@
import { Mockttp, RequestRuleBuilder } from 'mockttp';
import {
AuthenticationController,
UserStorageController,
} from '@metamask/profile-sync-controller';
import { AuthenticationController } from '@metamask/profile-sync-controller';
import {
NotificationServicesController,
NotificationServicesPushController,
} from '@metamask/notification-services-controller';
import { accountsSyncMockResponse, notificationMockResponse } from './mockData';

const AuthMocks = AuthenticationController.Mocks;
const StorageMocks = UserStorageController.Mocks;
const NotificationMocks = NotificationServicesController.Mocks;
const PushMocks = NotificationServicesPushController.Mocks;

Expand All @@ -30,22 +27,32 @@ export async function mockNotificationServices(server: Mockttp) {
mockAPICall(server, AuthMocks.getMockAuthLoginResponse());
mockAPICall(server, AuthMocks.getMockAuthAccessTokenResponse());

// Storage
mockAPICall(server, await StorageMocks.getMockUserStorageGetResponse());
mockAPICall(server, await StorageMocks.getMockUserStoragePutResponse());

// TODO - add better mock responses for other Profile Sync features
// (Account Sync, Network Sync, ...)
server
.forGet(/https:\/\/user-storage\.api\.cx\.metamask\.io\/.*/gu)
?.thenCallback(() => ({
statusCode: 404,
}));
server
.forPut(/https:\/\/user-storage\.api\.cx\.metamask\.io\/.*/gu)
?.thenCallback(() => ({
statusCode: 204,
}));

// Notifications Syncing
const notificationsRegex =
/https:\/\/user-storage\.api\.cx\.metamask\.io\/api\/v1\/userstorage\/notifications\/.*/gu;
server.forGet(notificationsRegex)?.thenCallback(() => ({
statusCode: 200,
json: notificationMockResponse,
}));

server.forPut(notificationsRegex)?.thenCallback(() => ({
statusCode: 204,
}));

// Accounts Syncing
const accountsRegex =
/https:\/\/user-storage\.api\.cx\.metamask\.io\/api\/v1\/userstorage\/accounts/;
server.forGet(accountsRegex)?.thenCallback(() => ({
statusCode: 200,
json: accountsSyncMockResponse,
}));

server.forPut(accountsRegex)?.thenCallback(() => ({
statusCode: 204,
}));

// Notifications
mockAPICall(server, NotificationMocks.getMockFeatureAnnouncementResponse());
Expand All @@ -56,12 +63,13 @@ export async function mockNotificationServices(server: Mockttp) {
server,
NotificationMocks.getMockMarkNotificationsAsReadResponse(),
);

// Push Notifications
mockAPICall(server, PushMocks.getMockRetrievePushNotificationLinksResponse());
mockAPICall(server, PushMocks.getMockUpdatePushNotificationLinksResponse());
mockAPICall(server, PushMocks.getMockCreateFCMRegistrationTokenResponse());
mockAPICall(server, PushMocks.getMockDeleteFCMRegistrationTokenResponse());

mockAPICall(server, PushMocks.getMockDeleteFCMRegistrationTokenResponse());
}

function mockAPICall(server: Mockttp, response: MockResponse) {
Expand Down
1 change: 1 addition & 0 deletions ui/components/multichain/global-menu/global-menu.js
Original file line number Diff line number Diff line change
Expand Up @@ -196,6 +196,7 @@ export const GlobalMenu = ({ closeMenu, anchorElement, isOpen }) => {
<MenuItem
iconName={IconName.Notification}
onClick={() => handleNotificationsClick()}
data-testid="notifications-menu-item"
>
<Box
display={Display.Flex}
Expand Down
Loading