strapi/api-tests/core/content-manager/find-available-relations.test.api.js
Alexandre BODIN 5ae0dc40cd feat!(content-manager): remove the ability to disable draft&publish (#18488)
* feat!: remove draft and publish triggering

* chore: update snapshots and tests

* test: adapt tests
2023-10-24 17:18:59 +01:00

459 lines
12 KiB
JavaScript

'use strict';
const { createTestBuilder } = require('api-tests/builder');
const { createStrapiInstance } = require('api-tests/strapi');
const { createAuthRequest } = require('api-tests/request');
const modelsUtils = require('api-tests/models');
let strapi;
let rq;
const data = {
products: [],
shops: [],
};
const compo = (withRelations = false) => ({
displayName: 'compo',
category: 'default',
attributes: {
name: {
type: 'string',
},
...(!withRelations
? {}
: {
compo_products_ow: {
type: 'relation',
relation: 'oneToOne',
target: 'api::product.product',
},
compo_products_mw: {
type: 'relation',
relation: 'oneToMany',
target: 'api::product.product',
},
}),
},
});
const productModel = () => ({
attributes: {
name: {
type: 'string',
},
},
displayName: 'Product',
singularName: 'product',
pluralName: 'products',
description: '',
collectionName: '',
});
const shopModel = () => ({
attributes: {
name: {
type: 'string',
},
products_ow: {
type: 'relation',
relation: 'oneToOne',
target: 'api::product.product',
},
products_oo: {
type: 'relation',
relation: 'oneToOne',
target: 'api::product.product',
targetAttribute: 'shop',
},
products_mo: {
type: 'relation',
relation: 'manyToOne',
target: 'api::product.product',
targetAttribute: 'shops_mo',
},
products_om: {
type: 'relation',
relation: 'oneToMany',
target: 'api::product.product',
targetAttribute: 'shop_om',
},
products_mm: {
type: 'relation',
relation: 'manyToMany',
target: 'api::product.product',
targetAttribute: 'shops',
},
products_mw: {
type: 'relation',
relation: 'oneToMany',
target: 'api::product.product',
},
myCompo: {
type: 'component',
repeatable: false,
component: 'default.compo',
},
},
displayName: 'Shop',
singularName: 'shop',
pluralName: 'shops',
});
const createEntry = async (uid, data) => {
const { body } = await rq({
method: 'POST',
url: `/content-manager/collection-types/${uid}`,
body: data,
});
return body;
};
describe('Relations, with d&p: %p', () => {
const builder = createTestBuilder();
const addPublishedAtCheck = (value) => {
publishedAt: value;
};
beforeAll(async () => {
await builder.addComponent(compo(false)).addContentTypes([productModel(), shopModel()]).build();
await modelsUtils.modifyComponent(compo(true));
strapi = await createStrapiInstance();
rq = await createAuthRequest({ strapi });
const createdProduct1 = await createEntry('api::product.product', { name: 'Skate' });
const createdProduct2 = await createEntry('api::product.product', { name: 'Candle' });
await rq({
url: `/content-manager/collection-types/api::product.product/${createdProduct1.id}/actions/publish`,
method: 'POST',
});
data.products.push(createdProduct1);
data.products.push(createdProduct2);
const id1 = createdProduct1.id;
const createdShop1 = await createEntry('api::shop.shop', {
name: 'Cazotte Shop',
products_ow: id1,
products_oo: id1,
products_mo: id1,
products_om: [id1],
products_mm: [id1],
products_mw: [id1],
myCompo: {
compo_products_ow: id1,
compo_products_mw: [id1],
},
});
const createdShop2 = await createEntry('api::shop.shop', {
name: 'Empty Shop',
myCompo: {
compo_products_ow: null,
compo_products_mw: [],
},
});
data.shops.push(createdShop1);
data.shops.push(createdShop2);
});
afterAll(async () => {
await strapi.destroy();
await builder.cleanup();
});
describe('findAvailable', () => {
describe('On a content-type', () => {
test('Fail when entity is not found', async () => {
const res = await rq({
method: 'GET',
url: '/content-manager/relations/api::shop.shop/products_ow',
qs: {
entityId: 99999,
},
});
expect(res.status).toBe(404);
expect(res.body).toMatchObject({
data: null,
error: {
details: {},
message: 'Not Found',
name: 'NotFoundError',
status: 404,
},
});
});
test("Fail when the field doesn't exist", async () => {
const res = await rq({
method: 'GET',
url: '/content-manager/relations/api::shop.shop/unkown',
});
expect(res.status).toBe(400);
expect(res.body).toMatchObject({
data: null,
error: {
details: {},
message: "This relational field doesn't exist",
name: 'BadRequestError',
status: 400,
},
});
});
test('Fail when the field exists but is not a relational field', async () => {
const res = await rq({
method: 'GET',
url: '/content-manager/relations/api::shop.shop/name',
});
expect(res.status).toBe(400);
expect(res.body).toMatchObject({
data: null,
error: {
details: {},
message: "This relational field doesn't exist",
name: 'BadRequestError',
status: 400,
},
});
});
});
describe('On a component', () => {
test('Fail when the component is not found', async () => {
const res = await rq({
method: 'GET',
url: '/content-manager/relations/default.compo/compo_products_ow',
qs: {
entityId: 99999,
},
});
expect(res.status).toBe(404);
expect(res.body).toMatchObject({
data: null,
error: {
details: {},
message: 'Not Found',
name: 'NotFoundError',
status: 404,
},
});
});
test("Fail when the field doesn't exist", async () => {
const res = await rq({
method: 'GET',
url: '/content-manager/relations/default.compo/unknown',
});
expect(res.status).toBe(400);
expect(res.body).toMatchObject({
data: null,
error: {
details: {},
message: "This relational field doesn't exist",
name: 'BadRequestError',
status: 400,
},
});
});
test('Fail when the field exists but is not a relational field', async () => {
const res = await rq({
method: 'GET',
url: '/content-manager/relations/default.compo/name',
});
expect(res.status).toBe(400);
expect(res.body).toMatchObject({
data: null,
error: {
details: {},
message: "This relational field doesn't exist",
name: 'BadRequestError',
status: 400,
},
});
});
});
describe.each([
['products_ow', false],
['products_oo', false],
['products_mo', false],
['products_om', false],
['products_mm', false],
['products_mw', false],
['compo_products_ow', true],
['compo_products_mw', true],
])('Relation (%s) (is in component: %s)', (fieldName, isComponent) => {
let entityId;
let entityIdEmptyShop;
let modelUID;
beforeAll(() => {
entityId = isComponent ? data.shops[0].myCompo.id : data.shops[0].id;
entityIdEmptyShop = isComponent ? data.shops[1].myCompo.id : data.shops[1].id;
modelUID = isComponent ? 'default.compo' : 'api::shop.shop';
});
test('Can retrieve available relation(s) for an entity that have some relations', async () => {
let res = await rq({
method: 'GET',
url: `/content-manager/relations/${modelUID}/${fieldName}`,
qs: {
entityId,
},
});
expect(res.status).toBe(200);
expect(res.body.results).toMatchObject([
{
id: expect.any(Number),
name: 'Candle',
...addPublishedAtCheck(null),
},
]);
// can omitIds
res = await rq({
method: 'GET',
url: `/content-manager/relations/${modelUID}/${fieldName}`,
qs: {
entityId,
idsToOmit: [data.products[1].id],
},
});
expect(res.body.results).toHaveLength(0);
});
test("Can retrieve available relation(s) for an entity that don't have relations yet", async () => {
let res = await rq({
method: 'GET',
url: `/content-manager/relations/${modelUID}/${fieldName}`,
qs: {
entityId: entityIdEmptyShop,
},
});
expect(res.status).toBe(200);
expect(res.body.results).toMatchObject([
{
id: expect.any(Number),
name: 'Candle',
...addPublishedAtCheck(null),
},
{
id: expect.any(Number),
name: 'Skate',
...addPublishedAtCheck(expect.any(String)),
},
]);
// can omitIds
res = await rq({
method: 'GET',
url: `/content-manager/relations/${modelUID}/${fieldName}`,
qs: {
entityId,
idsToOmit: [data.products[1].id],
},
});
expect(res.body.results).toHaveLength(0);
});
test('Can retrieve available relation(s) without entity', async () => {
let res = await rq({
method: 'GET',
url: `/content-manager/relations/${modelUID}/${fieldName}`,
});
expect(res.status).toBe(200);
expect(res.body.results).toMatchObject([
{
id: expect.any(Number),
name: 'Candle',
...addPublishedAtCheck(null),
},
{
id: expect.any(Number),
name: 'Skate',
...addPublishedAtCheck(expect.any(String)),
},
]);
// can omitIds
res = await rq({
method: 'GET',
url: `/content-manager/relations/${modelUID}/${fieldName}`,
qs: {
idsToOmit: [data.products[0].id],
},
});
expect(res.body.results).toMatchObject([
{
id: expect.any(Number),
name: 'Candle',
...addPublishedAtCheck(null),
},
]);
});
describe.each([['with'], ['without']])('Can search %s entity', (withOrWithout) => {
const withEntity = withOrWithout === 'with';
test("search ''", async () => {
const res = await rq({
method: 'GET',
url: `/content-manager/relations/${modelUID}/${fieldName}`,
qs: {
_q: '',
...(withEntity ? { entityId } : {}),
},
});
expect(res.status).toBe(200);
expect(res.body.results).toHaveLength(withEntity ? 1 : 2);
expect(res.body.results[0]).toMatchObject({
id: expect.any(Number),
name: 'Candle',
...addPublishedAtCheck(null),
});
});
test("search 'Can'", async () => {
const res = await rq({
method: 'GET',
url: `/content-manager/relations/${modelUID}/${fieldName}`,
qs: {
_q: 'Can',
...(withEntity ? { entityId } : {}),
},
});
expect(res.status).toBe(200);
expect(res.body.results).toMatchObject([
{
id: expect.any(Number),
name: 'Candle',
...addPublishedAtCheck(null),
},
]);
});
});
});
});
});