mirror of
https://github.com/strapi/strapi.git
synced 2025-07-26 02:20:32 +00:00

# Conflicts: # .github/actions/check-pr-status/package.json # docs/docs/docs/01-core/admin/05-features/authentication.md # docs/docusaurus.config.js # docs/yarn.lock # examples/getstarted/package.json # examples/kitchensink-ts/package.json # examples/kitchensink/package.json # examples/plugins/workspace-plugin/package.json # lerna.json # package.json # packages/admin-test-utils/package.json # packages/cli/create-strapi-app/package.json # packages/cli/create-strapi-starter/package.json # packages/cli/create-strapi/package.json # packages/core/admin/admin/src/App.tsx # packages/core/admin/admin/src/components/AuthenticatedApp.tsx # packages/core/admin/admin/src/components/Context.tsx # packages/core/admin/admin/src/components/LeftMenu.tsx # packages/core/admin/admin/src/components/PrivateRoute.tsx # packages/core/admin/admin/src/content-manager/pages/ListSettingsView/index.jsx # packages/core/admin/admin/src/content-manager/pages/ListView/components/BulkActions/PublishAction.tsx # packages/core/admin/admin/src/content-manager/pages/ListView/components/TableCells/CellContent.tsx # packages/core/admin/admin/src/features/Configuration.tsx # packages/core/admin/admin/src/index.ts # packages/core/admin/admin/src/pages/ProfilePage.tsx # packages/core/admin/admin/src/pages/UseCasePage.tsx # packages/core/admin/ee/server/src/services/metrics.ts # packages/core/admin/package.json # packages/core/admin/server/src/bootstrap.ts # packages/core/content-manager/package.json # packages/core/content-releases/admin/src/components/CMReleasesContainer.tsx # packages/core/content-releases/admin/src/components/ReleaseListCell.tsx # packages/core/content-releases/admin/src/index.ts # packages/core/content-releases/admin/src/pages/ReleaseDetailsPage.tsx # packages/core/content-releases/admin/src/pages/tests/ReleaseDetailsPage.test.tsx # packages/core/content-releases/admin/src/services/release.ts # packages/core/content-releases/package.json # packages/core/content-releases/server/src/controllers/release.ts # packages/core/content-type-builder/package.json # packages/core/core/.gitignore # packages/core/data-transfer/package.json # packages/core/database/package.json # packages/core/database/src/query/helpers/populate/apply.ts # packages/core/email/package.json # packages/core/helper-plugin/package.json # packages/core/permissions/package.json # packages/core/strapi/package.json # packages/core/types/package.json # packages/core/upload/admin/src/components/AssetDialog/BrowseStep/tests/__snapshots__/index.test.jsx.snap # packages/core/upload/admin/src/pages/App/ConfigureTheView/components/tests/__snapshots__/Settings.test.jsx.snap # packages/core/upload/admin/src/pages/App/ConfigureTheView/tests/__snapshots__/ConfigureTheView.test.jsx.snap # packages/core/upload/package.json # packages/core/upload/server/services/provider.js # packages/core/upload/server/services/upload.js # packages/core/upload/server/src/services/image-manipulation.ts # packages/core/upload/server/tsconfig.eslint.json # packages/core/utils/package.json # packages/generators/app/package.json # packages/generators/app/src/utils/db-client-dependencies.ts # packages/generators/generators/package.json # packages/plugins/cloud/package.json # packages/plugins/color-picker/package.json # packages/plugins/documentation/package.json # packages/plugins/graphql/package.json # packages/plugins/i18n/admin/src/components/LocaleListCell.tsx # packages/plugins/i18n/admin/src/contentReleasesHooks/releaseDetailsView.ts # packages/plugins/i18n/admin/src/index.ts # packages/plugins/i18n/package.json # packages/plugins/sentry/package.json # packages/plugins/users-permissions/package.json # packages/providers/email-amazon-ses/package.json # packages/providers/email-mailgun/package.json # packages/providers/email-nodemailer/package.json # packages/providers/email-sendgrid/package.json # packages/providers/email-sendmail/package.json # packages/providers/upload-aws-s3/package.json # packages/providers/upload-cloudinary/package.json # packages/providers/upload-local/package.json # packages/utils/api-tests/package.json # packages/utils/eslint-config-custom/package.json # packages/utils/logger/package.json # packages/utils/tsconfig/package.json # packages/utils/typescript/package.json # packages/utils/upgrade/LICENSE # packages/utils/upgrade/bin/upgrade.js # packages/utils/upgrade/package.json # packages/utils/upgrade/packup.config.ts # scripts/front/package.json # tests/e2e/constants.ts # tests/e2e/tests/content-releases/releases-page.spec.ts # yarn.lock
403 lines
8.9 KiB
JavaScript
403 lines
8.9 KiB
JavaScript
'use strict';
|
|
|
|
const fs = require('fs');
|
|
const path = require('path');
|
|
|
|
// Helpers.
|
|
const {createTestBuilder} = require('api-tests/builder');
|
|
const {createStrapiInstance} = require('api-tests/strapi');
|
|
const {createContentAPIRequest} = require('api-tests/request');
|
|
|
|
const builder = createTestBuilder();
|
|
const data = {dogs: []};
|
|
let strapi;
|
|
let rq;
|
|
|
|
const dogModel = {
|
|
displayName: 'Dog',
|
|
singularName: 'dog',
|
|
pluralName: 'dogs',
|
|
kind: 'collectionType',
|
|
attributes: {
|
|
profilePicture: {
|
|
type: 'media',
|
|
},
|
|
},
|
|
};
|
|
|
|
const todoListModel = {
|
|
displayName: 'TodoList',
|
|
singularName: 'todolist',
|
|
pluralName: 'todolists',
|
|
kind: 'collectionType',
|
|
attributes: {
|
|
title: {
|
|
type: 'string',
|
|
},
|
|
todo: {
|
|
displayName: 'todo',
|
|
type: 'component',
|
|
repeatable: true,
|
|
component: 'default.todo',
|
|
},
|
|
},
|
|
};
|
|
|
|
const todoComponent = {
|
|
displayName: 'Todo',
|
|
attributes: {
|
|
docs: {
|
|
allowedTypes: ['images', 'files', 'videos', 'audios'],
|
|
type: 'media',
|
|
multiple: true,
|
|
},
|
|
task: {
|
|
type: 'string',
|
|
},
|
|
},
|
|
};
|
|
|
|
describe('Upload plugin', () => {
|
|
beforeAll(async () => {
|
|
await builder
|
|
.addContentType(dogModel)
|
|
.addComponent(todoComponent)
|
|
.addContentType(todoListModel)
|
|
.build();
|
|
strapi = await createStrapiInstance();
|
|
rq = createContentAPIRequest({strapi});
|
|
});
|
|
|
|
afterAll(async () => {
|
|
await strapi.destroy();
|
|
await builder.cleanup();
|
|
});
|
|
|
|
describe('Create', () => {
|
|
test('Simple image upload', async () => {
|
|
const res = await rq({
|
|
method: 'POST',
|
|
url: '/upload',
|
|
formData: {
|
|
files: fs.createReadStream(path.join(__dirname, '../utils/rec.jpg')),
|
|
},
|
|
});
|
|
|
|
expect(res.statusCode).toBe(201);
|
|
expect(Array.isArray(res.body)).toBe(true);
|
|
expect(res.body.length).toBe(1);
|
|
expect(res.body[0]).toEqual(
|
|
expect.objectContaining({
|
|
id: expect.anything(),
|
|
name: 'rec.jpg',
|
|
ext: '.jpg',
|
|
mime: 'image/jpeg',
|
|
hash: expect.any(String),
|
|
size: expect.any(Number),
|
|
width: expect.any(Number),
|
|
height: expect.any(Number),
|
|
url: expect.any(String),
|
|
provider: 'local',
|
|
})
|
|
);
|
|
});
|
|
|
|
test('Rejects when no files are provided', async () => {
|
|
const res = await rq({method: 'POST', url: '/upload', formData: {}});
|
|
expect(res.statusCode).toBe(400);
|
|
});
|
|
|
|
test('Generates a thumbnail on large enough files', async () => {
|
|
const res = await rq({
|
|
method: 'POST',
|
|
url: '/upload',
|
|
formData: {
|
|
files: fs.createReadStream(path.join(__dirname, '../utils/thumbnail_target.png')),
|
|
},
|
|
});
|
|
|
|
expect(res.statusCode).toBe(201);
|
|
expect(Array.isArray(res.body)).toBe(true);
|
|
expect(res.body.length).toBe(1);
|
|
expect(res.body[0]).toEqual(
|
|
expect.objectContaining({
|
|
id: expect.anything(),
|
|
name: 'thumbnail_target.png',
|
|
ext: '.png',
|
|
mime: 'image/png',
|
|
hash: expect.any(String),
|
|
size: expect.any(Number),
|
|
width: expect.any(Number),
|
|
height: expect.any(Number),
|
|
url: expect.any(String),
|
|
provider: 'local',
|
|
formats: {
|
|
thumbnail: {
|
|
name: 'thumbnail_thumbnail_target.png',
|
|
hash: expect.any(String),
|
|
ext: '.png',
|
|
mime: 'image/png',
|
|
size: expect.any(Number),
|
|
sizeInBytes: expect.any(Number),
|
|
width: expect.any(Number),
|
|
height: expect.any(Number),
|
|
url: expect.any(String),
|
|
path: null,
|
|
},
|
|
},
|
|
})
|
|
);
|
|
});
|
|
});
|
|
|
|
describe('Read', () => {
|
|
test('Get files', async () => {
|
|
const getRes = await rq({method: 'GET', url: '/upload/files'});
|
|
|
|
expect(getRes.statusCode).toBe(200);
|
|
expect(getRes.body).toEqual(
|
|
expect.arrayContaining([
|
|
expect.objectContaining({
|
|
id: expect.anything(),
|
|
url: expect.any(String),
|
|
}),
|
|
])
|
|
);
|
|
});
|
|
|
|
test('Get one file', async () => {
|
|
const res = await rq({
|
|
method: 'POST',
|
|
url: '/upload',
|
|
formData: {
|
|
files: fs.createReadStream(path.join(__dirname, '../utils/thumbnail_target.png')),
|
|
},
|
|
});
|
|
|
|
const dogEntity = await strapi.db.query('api::dog.dog').create({
|
|
data: {
|
|
profilePicture: res.body[0].id,
|
|
},
|
|
populate: {
|
|
profilePicture: true,
|
|
},
|
|
});
|
|
|
|
const getRes = await rq({
|
|
method: 'GET',
|
|
url: `/upload/files/${dogEntity.profilePicture.id}`,
|
|
});
|
|
|
|
expect(getRes.statusCode).toBe(200);
|
|
expect(getRes.body).toEqual(
|
|
expect.objectContaining({
|
|
id: expect.anything(),
|
|
url: expect.any(String),
|
|
})
|
|
);
|
|
|
|
await strapi.db.query('api::dog.dog').delete({where: {id: dogEntity.id}});
|
|
await strapi.db
|
|
.query('plugin::upload.file')
|
|
.delete({where: {id: dogEntity.profilePicture.id}});
|
|
});
|
|
});
|
|
|
|
describe('Filtering data based on media attributes', () => {
|
|
let uploadRes;
|
|
let dogRes;
|
|
|
|
beforeAll(async () => {
|
|
await Promise.all(
|
|
data.dogs.map((dog) => {
|
|
return strapi.entityService.delete('api::dog.dog', dog.data.id);
|
|
})
|
|
);
|
|
|
|
uploadRes = await rq({
|
|
method: 'POST',
|
|
url: '/upload',
|
|
formData: {
|
|
files: fs.createReadStream(path.join(__dirname, '../utils/rec.jpg')),
|
|
fileInfo: JSON.stringify({
|
|
alternativeText: 'rec',
|
|
caption: 'my caption',
|
|
}),
|
|
},
|
|
});
|
|
|
|
dogRes = await rq({
|
|
method: 'POST',
|
|
url: '/dogs',
|
|
body: {
|
|
data: {
|
|
profilePicture: {
|
|
id: uploadRes.body[0].id,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
});
|
|
|
|
afterAll(async () => {
|
|
await rq({
|
|
method: 'DELETE',
|
|
url: `/dogs/${dogRes.body.data.id}`,
|
|
});
|
|
|
|
await rq({
|
|
method: 'DELETE',
|
|
url: `/upload/files/${uploadRes.body[0].id}`,
|
|
});
|
|
});
|
|
|
|
test('can filter on notNull', async () => {
|
|
let res;
|
|
|
|
res = await rq({
|
|
method: 'GET',
|
|
url: '/dogs',
|
|
qs: {
|
|
filters: {
|
|
profilePicture: {$notNull: true},
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(res.body.data.length).toBe(1);
|
|
|
|
res = await rq({
|
|
method: 'GET',
|
|
url: '/dogs',
|
|
qs: {
|
|
filters: {
|
|
profilePicture: {$notNull: false},
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(res.body.data.length).toBe(0);
|
|
});
|
|
|
|
test('can filter on null', async () => {
|
|
let res;
|
|
|
|
res = await rq({
|
|
method: 'GET',
|
|
url: '/dogs',
|
|
qs: {
|
|
filters: {
|
|
profilePicture: {$null: true},
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(res.body.data.length).toBe(0);
|
|
|
|
res = await rq({
|
|
method: 'GET',
|
|
url: '/dogs',
|
|
qs: {
|
|
filters: {
|
|
profilePicture: {$null: false},
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(res.body.data.length).toBe(1);
|
|
});
|
|
|
|
test('can filter on id', async () => {
|
|
let res;
|
|
|
|
res = await rq({
|
|
method: 'GET',
|
|
url: '/dogs',
|
|
qs: {
|
|
filters: {
|
|
profilePicture: uploadRes.body[0].id,
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(res.body.data.length).toBe(1);
|
|
|
|
res = await rq({
|
|
method: 'GET',
|
|
url: '/dogs',
|
|
qs: {
|
|
filters: {
|
|
profilePicture: 999999999,
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(res.body.data.length).toBe(0);
|
|
});
|
|
|
|
test('can filter media attribute', async () => {
|
|
let res;
|
|
|
|
res = await rq({
|
|
method: 'GET',
|
|
url: '/dogs',
|
|
qs: {
|
|
filters: {
|
|
profilePicture: {ext: '.jpg'},
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(res.body.data.length).toBe(1);
|
|
|
|
res = await rq({
|
|
method: 'GET',
|
|
url: '/dogs',
|
|
qs: {
|
|
filters: {
|
|
profilePicture: {ext: '.pdf'},
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(res.body.data.length).toBe(0);
|
|
});
|
|
|
|
test('can filter media attribute with operators', async () => {
|
|
let res;
|
|
|
|
res = await rq({
|
|
method: 'GET',
|
|
url: '/dogs',
|
|
qs: {
|
|
filters: {
|
|
profilePicture: {
|
|
caption: {
|
|
$contains: 'my',
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(res.body.data.length).toBe(1);
|
|
|
|
res = await rq({
|
|
method: 'GET',
|
|
url: '/dogs',
|
|
qs: {
|
|
filters: {
|
|
profilePicture: {
|
|
caption: {
|
|
$contains: 'not',
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
expect(res.body.data.length).toBe(0);
|
|
});
|
|
});
|
|
});
|