2023-08-30 00:30:59 +03:00
|
|
|
|
import { openHomePage } from '@affine-test/kit/utils/load-page';
|
2023-09-02 09:06:47 +03:00
|
|
|
|
import {
|
|
|
|
|
clickNewPageButton,
|
|
|
|
|
waitForEditorLoad,
|
|
|
|
|
} from '@affine-test/kit/utils/page-logic';
|
|
|
|
|
import {
|
|
|
|
|
clickSideBarCurrentWorkspaceBanner,
|
|
|
|
|
clickSideBarSettingButton,
|
|
|
|
|
} from '@affine-test/kit/utils/sidebar';
|
2023-08-29 13:07:05 +03:00
|
|
|
|
import { faker } from '@faker-js/faker';
|
|
|
|
|
import { hash } from '@node-rs/argon2';
|
2023-10-16 11:44:09 +03:00
|
|
|
|
import {
|
|
|
|
|
type BrowserContext,
|
|
|
|
|
type Cookie,
|
|
|
|
|
expect,
|
|
|
|
|
type Page,
|
|
|
|
|
} from '@playwright/test';
|
2023-08-30 00:30:59 +03:00
|
|
|
|
import { z } from 'zod';
|
2023-08-29 13:07:05 +03:00
|
|
|
|
|
2023-09-13 08:11:19 +03:00
|
|
|
|
export async function getCurrentMailMessageCount() {
|
|
|
|
|
const response = await fetch('http://localhost:8025/api/v2/messages');
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
return data.total;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export async function getLatestMailMessage() {
|
|
|
|
|
const response = await fetch('http://localhost:8025/api/v2/messages');
|
|
|
|
|
const data = await response.json();
|
|
|
|
|
return data.items[0];
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-29 13:07:05 +03:00
|
|
|
|
export async function getLoginCookie(
|
|
|
|
|
context: BrowserContext
|
|
|
|
|
): Promise<Cookie | undefined> {
|
|
|
|
|
return (await context.cookies()).find(
|
|
|
|
|
c => c.name === 'next-auth.session-token'
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-30 00:30:59 +03:00
|
|
|
|
const cloudUserSchema = z.object({
|
2023-09-02 09:06:47 +03:00
|
|
|
|
id: z.string(),
|
2023-08-30 00:30:59 +03:00
|
|
|
|
name: z.string(),
|
|
|
|
|
email: z.string().email(),
|
|
|
|
|
password: z.string(),
|
|
|
|
|
});
|
|
|
|
|
|
2023-09-02 09:06:47 +03:00
|
|
|
|
export const runPrisma = async <T>(
|
|
|
|
|
cb: (
|
|
|
|
|
prisma: InstanceType<
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-imports
|
2023-10-18 18:30:08 +03:00
|
|
|
|
typeof import('../../../packages/backend/server/node_modules/@prisma/client').PrismaClient
|
2023-09-02 09:06:47 +03:00
|
|
|
|
>
|
|
|
|
|
) => Promise<T>
|
|
|
|
|
): Promise<T> => {
|
2023-08-29 13:07:05 +03:00
|
|
|
|
const {
|
|
|
|
|
PrismaClient,
|
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
2023-10-18 18:30:08 +03:00
|
|
|
|
} = require('../../../packages/backend/server/node_modules/@prisma/client');
|
2023-08-29 13:07:05 +03:00
|
|
|
|
const client = new PrismaClient();
|
|
|
|
|
await client.$connect();
|
2023-09-02 09:06:47 +03:00
|
|
|
|
try {
|
|
|
|
|
return await cb(client);
|
|
|
|
|
} finally {
|
|
|
|
|
await client.$disconnect();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
export async function addUserToWorkspace(
|
|
|
|
|
workspaceId: string,
|
|
|
|
|
userId: string,
|
|
|
|
|
permission: number
|
|
|
|
|
) {
|
|
|
|
|
await runPrisma(async client => {
|
|
|
|
|
const workspace = await client.workspace.findUnique({
|
|
|
|
|
where: {
|
|
|
|
|
id: workspaceId,
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
if (workspace == null) {
|
|
|
|
|
throw new Error(`workspace ${workspaceId} not found`);
|
|
|
|
|
}
|
|
|
|
|
await client.userWorkspacePermission.create({
|
|
|
|
|
data: {
|
|
|
|
|
workspaceId: workspace.id,
|
|
|
|
|
subPageId: null,
|
|
|
|
|
userId,
|
|
|
|
|
accepted: true,
|
|
|
|
|
type: permission,
|
|
|
|
|
},
|
|
|
|
|
});
|
2023-08-29 13:07:05 +03:00
|
|
|
|
});
|
2023-09-02 09:06:47 +03:00
|
|
|
|
}
|
2023-08-29 13:07:05 +03:00
|
|
|
|
|
2023-09-02 09:06:47 +03:00
|
|
|
|
export async function createRandomUser(): Promise<{
|
|
|
|
|
name: string;
|
|
|
|
|
email: string;
|
|
|
|
|
password: string;
|
|
|
|
|
id: string;
|
|
|
|
|
}> {
|
|
|
|
|
const user = {
|
|
|
|
|
name: faker.internet.userName(),
|
|
|
|
|
email: faker.internet.email().toLowerCase(),
|
|
|
|
|
password: '123456',
|
|
|
|
|
};
|
|
|
|
|
const result = await runPrisma(async client => {
|
|
|
|
|
await client.user.create({
|
|
|
|
|
data: {
|
|
|
|
|
...user,
|
|
|
|
|
emailVerified: new Date(),
|
|
|
|
|
password: await hash(user.password),
|
|
|
|
|
},
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return await client.user.findUnique({
|
|
|
|
|
where: {
|
|
|
|
|
email: user.email,
|
|
|
|
|
},
|
|
|
|
|
});
|
2023-08-29 13:07:05 +03:00
|
|
|
|
});
|
2023-08-30 00:30:59 +03:00
|
|
|
|
cloudUserSchema.parse(result);
|
2023-09-02 09:06:47 +03:00
|
|
|
|
return {
|
|
|
|
|
...result,
|
|
|
|
|
password: user.password,
|
|
|
|
|
} as any;
|
2023-08-29 13:07:05 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export async function deleteUser(email: string) {
|
2023-09-02 09:06:47 +03:00
|
|
|
|
await runPrisma(async client => {
|
|
|
|
|
await client.user.delete({
|
|
|
|
|
where: {
|
|
|
|
|
email,
|
|
|
|
|
},
|
|
|
|
|
});
|
2023-08-29 13:07:05 +03:00
|
|
|
|
});
|
|
|
|
|
}
|
2023-08-30 00:30:59 +03:00
|
|
|
|
|
|
|
|
|
export async function loginUser(
|
|
|
|
|
page: Page,
|
2023-09-02 09:06:47 +03:00
|
|
|
|
userEmail: string,
|
2023-08-30 00:30:59 +03:00
|
|
|
|
config?: {
|
2023-09-19 05:08:35 +03:00
|
|
|
|
isElectron?: boolean;
|
2023-08-30 00:30:59 +03:00
|
|
|
|
beforeLogin?: () => Promise<void>;
|
|
|
|
|
afterLogin?: () => Promise<void>;
|
|
|
|
|
}
|
|
|
|
|
) {
|
2023-09-19 05:08:35 +03:00
|
|
|
|
if (config?.isElectron !== true) {
|
|
|
|
|
await openHomePage(page);
|
|
|
|
|
await waitForEditorLoad(page);
|
|
|
|
|
}
|
2023-08-30 00:30:59 +03:00
|
|
|
|
|
|
|
|
|
await clickSideBarCurrentWorkspaceBanner(page);
|
|
|
|
|
await page.getByTestId('cloud-signin-button').click({
|
|
|
|
|
delay: 200,
|
|
|
|
|
});
|
2023-09-13 19:54:02 +03:00
|
|
|
|
await page.getByPlaceholder('Enter your email address').fill(userEmail);
|
2023-08-30 00:30:59 +03:00
|
|
|
|
await page.getByTestId('continue-login-button').click({
|
|
|
|
|
delay: 200,
|
|
|
|
|
});
|
|
|
|
|
await page.getByTestId('sign-in-with-password').click({
|
|
|
|
|
delay: 200,
|
|
|
|
|
});
|
2023-09-13 19:54:02 +03:00
|
|
|
|
await page.getByTestId('password-input').fill('123456');
|
2023-08-30 00:30:59 +03:00
|
|
|
|
if (config?.beforeLogin) {
|
|
|
|
|
await config.beforeLogin();
|
|
|
|
|
}
|
|
|
|
|
await page.waitForTimeout(200);
|
|
|
|
|
await page.getByTestId('sign-in-button').click();
|
|
|
|
|
await page.waitForTimeout(200);
|
|
|
|
|
if (config?.afterLogin) {
|
|
|
|
|
await config.afterLogin();
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-09-02 09:06:47 +03:00
|
|
|
|
|
|
|
|
|
export async function enableCloudWorkspace(page: Page) {
|
|
|
|
|
await clickSideBarSettingButton(page);
|
|
|
|
|
await page.getByTestId('current-workspace-label').click();
|
|
|
|
|
await page.getByTestId('publish-enable-affine-cloud-button').click();
|
|
|
|
|
await page.getByTestId('confirm-enable-affine-cloud-button').click();
|
|
|
|
|
// wait for upload and delete local workspace
|
|
|
|
|
await page.waitForTimeout(2000);
|
|
|
|
|
await waitForEditorLoad(page);
|
|
|
|
|
await clickNewPageButton(page);
|
|
|
|
|
}
|
2023-10-16 11:44:09 +03:00
|
|
|
|
|
2023-10-12 06:26:13 +03:00
|
|
|
|
export async function enableCloudWorkspaceFromShareButton(page: Page) {
|
2023-10-16 11:44:09 +03:00
|
|
|
|
const shareMenuButton = page.getByTestId('local-share-menu-button');
|
|
|
|
|
expect(await shareMenuButton.isVisible()).toBeTruthy();
|
|
|
|
|
|
|
|
|
|
// FIXME: this is a workaround for the flaky test
|
|
|
|
|
// For unknown reasons,
|
|
|
|
|
// the online ci test on GitHub is unable to detect the local-share-menu,
|
|
|
|
|
// although it works fine in local testing.
|
|
|
|
|
// To ensure the tests pass consistently, I’ve made the following temporary adjustments.
|
|
|
|
|
// {
|
|
|
|
|
const maxAttempts = 5;
|
|
|
|
|
let attempt = 0;
|
|
|
|
|
let menuVisible = false;
|
|
|
|
|
|
|
|
|
|
while (!menuVisible && attempt < maxAttempts) {
|
|
|
|
|
try {
|
|
|
|
|
await shareMenuButton.click();
|
|
|
|
|
menuVisible = await page.getByTestId('local-share-menu').isVisible();
|
|
|
|
|
} catch (e) {
|
|
|
|
|
console.error(`Attempt ${attempt + 1} failed: ${e}`);
|
|
|
|
|
attempt += 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
expect(menuVisible).toBeTruthy();
|
|
|
|
|
// }
|
|
|
|
|
|
2023-10-12 06:26:13 +03:00
|
|
|
|
await page.getByTestId('share-menu-enable-affine-cloud-button').click();
|
|
|
|
|
await page.getByTestId('confirm-enable-affine-cloud-button').click();
|
|
|
|
|
// wait for upload and delete local workspace
|
|
|
|
|
await page.waitForTimeout(2000);
|
|
|
|
|
await waitForEditorLoad(page);
|
|
|
|
|
await clickNewPageButton(page);
|
|
|
|
|
}
|