Files
payloadcms/test/fields-relationship/e2e.spec.ts
2022-07-28 11:50:23 -04:00

246 lines
7.6 KiB
TypeScript

import type { Page } from '@playwright/test';
import { expect, test } from '@playwright/test';
import payload from '../../src';
import { mapAsync } from '../../src/utilities/mapAsync';
import { AdminUrlUtil } from '../helpers/adminUrlUtil';
import { initPayloadE2E } from '../helpers/configHelpers';
import { login, saveDocAndAssert } from '../helpers';
import type {
FieldsRelationship as CollectionWithRelationships,
RelationOne,
RelationRestricted,
RelationTwo,
RelationWithTitle,
} from './config';
import { relationOneSlug, relationRestrictedSlug, relationTwoSlug, relationWithTitleSlug, slug } from './config';
import wait from '../../src/utilities/wait';
const { beforeAll, beforeEach, describe } = test;
let url: AdminUrlUtil;
describe('fields - relationship', () => {
let page: Page;
let relationOneDoc: RelationOne;
let anotherRelationOneDoc: RelationOne;
let relationTwoDoc: RelationTwo;
let docWithExistingRelations: CollectionWithRelationships;
let restrictedRelation: RelationRestricted;
let relationWithTitle: RelationWithTitle;
beforeAll(async ({ browser }) => {
const { serverURL } = await initPayloadE2E(__dirname);
url = new AdminUrlUtil(serverURL, slug);
const context = await browser.newContext();
page = await context.newPage();
await login({ page, serverURL });
});
beforeEach(async () => {
await clearAllDocs();
// Create docs to relate to
relationOneDoc = await payload.create<RelationOne>({
collection: relationOneSlug,
data: {
name: 'relation',
},
});
anotherRelationOneDoc = await payload.create<RelationOne>({
collection: relationOneSlug,
data: {
name: 'relation',
},
});
relationTwoDoc = await payload.create<RelationTwo>({
collection: relationTwoSlug,
data: {
name: 'second-relation',
},
});
// Create restricted doc
restrictedRelation = await payload.create<RelationRestricted>({
collection: relationRestrictedSlug,
data: {
name: 'restricted',
},
});
// Doc with useAsTitle
relationWithTitle = await payload.create<RelationWithTitle>({
collection: relationWithTitleSlug,
data: {
name: 'relation-title',
},
});
// Add restricted doc as relation
docWithExistingRelations = await payload.create<CollectionWithRelationships>({
collection: slug,
data: {
name: 'with-existing-relations',
relationship: relationOneDoc.id,
relationshipRestricted: restrictedRelation.id,
relationshipWithTitle: relationWithTitle.id,
},
});
});
test('should create relationship', async () => {
await page.goto(url.create);
const field = page.locator('#field-relationship');
await field.click({ delay: 100 });
const options = page.locator('.rs__option');
await expect(options).toHaveCount(3); // None + two docs
// Select a relationship
await options.nth(1).click();
await expect(field).toContainText(relationOneDoc.id);
await saveDocAndAssert(page);
});
test('should create hasMany relationship', async () => {
await page.goto(url.create);
const field = page.locator('#field-relationshipHasMany');
await field.click({ delay: 100 });
const options = page.locator('.rs__option');
await expect(options).toHaveCount(2); // Two relationship options
// Add one relationship
await options.locator(`text=${relationOneDoc.id}`).click();
await expect(field).toContainText(relationOneDoc.id);
await expect(field).not.toContainText(anotherRelationOneDoc.id);
// Add second relationship
await field.click({ delay: 100 });
await options.locator(`text=${anotherRelationOneDoc.id}`).click();
await expect(field).toContainText(anotherRelationOneDoc.id);
// No options left
await field.click({ delay: 100 });
await expect(page.locator('.rs__menu')).toHaveText('No options');
await saveDocAndAssert(page);
});
test('should create relations to multiple collections', async () => {
await page.goto(url.create);
const field = page.locator('#field-relationshipMultiple');
await field.click({ delay: 100 });
const options = page.locator('.rs__option');
await expect(options).toHaveCount(4); // None + 3 docs
// Add one relationship
await options.locator(`text=${relationOneDoc.id}`).click();
await expect(field).toContainText(relationOneDoc.id);
// Add relationship of different collection
await field.click({ delay: 100 });
await options.locator(`text=${relationTwoDoc.id}`).click();
await expect(field).toContainText(relationTwoDoc.id);
await saveDocAndAssert(page);
});
describe('existing relationships', () => {
test('should highlight existing relationship', async () => {
await page.goto(url.edit(docWithExistingRelations.id));
const field = page.locator('#field-relationship');
// Check dropdown options
await field.click({ delay: 100 });
await expect(page.locator('.rs__option--is-selected')).toHaveCount(1);
await expect(page.locator('.rs__option--is-selected')).toHaveText(relationOneDoc.id);
});
test('should show untitled ID on restricted relation', async () => {
await page.goto(url.edit(docWithExistingRelations.id));
const field = page.locator('#field-relationshipRestricted');
// Check existing relationship has untitled ID
await expect(field).toContainText(`Untitled - ID: ${restrictedRelation.id}`);
// Check dropdown options
await field.click({ delay: 100 });
const options = page.locator('.rs__option');
await expect(options).toHaveCount(2); // None + 1 Unitled ID
});
// test.todo('should paginate within the dropdown');
// test.todo('should search within the relationship field');
test('should show useAsTitle on relation', async () => {
await page.goto(url.edit(docWithExistingRelations.id));
const field = page.locator('#field-relationshipWithTitle .react-select');
// Check existing relationship for correct title
await expect(field).toHaveText(relationWithTitle.name);
await field.click({ delay: 100 });
const options = page.locator('.rs__option');
await expect(options).toHaveCount(2); // None + 1 Doc
});
test('should show id on relation in list view', async () => {
await page.goto(url.list);
await wait(110);
const relationship = page.locator('.row-1 .cell-relationship');
await expect(relationship).toHaveText(relationOneDoc.id);
});
test('should show Untitled ID on restricted relation in list view', async () => {
await page.goto(url.list);
await wait(110);
const relationship = page.locator('.row-1 .cell-relationshipRestricted');
await expect(relationship).toContainText('Untitled - ID: ');
});
test('should show useAsTitle on relation in list view', async () => {
await page.goto(url.list);
await wait(110);
const relationship = page.locator('.row-1 .cell-relationshipWithTitle');
await expect(relationship).toHaveText(relationWithTitle.name);
});
});
});
async function clearAllDocs(): Promise<void> {
await clearCollectionDocs(slug);
await clearCollectionDocs(relationOneSlug);
await clearCollectionDocs(relationTwoSlug);
await clearCollectionDocs(relationRestrictedSlug);
await clearCollectionDocs(relationWithTitleSlug);
}
async function clearCollectionDocs(collectionSlug: string): Promise<void> {
const ids = (await payload.find({ collection: collectionSlug, limit: 100 })).docs.map((doc) => doc.id);
await mapAsync(ids, async (id) => {
await payload.delete({ collection: collectionSlug, id });
});
}