261 lines
8.2 KiB
JavaScript
Raw Normal View History

'use strict';
const path = require('path');
const os = require('os');
const mime = require('mime-types');
const fse = require('fs-extra');
const { getStreamSize } = require('@strapi/utils/lib/file');
2022-05-13 16:10:18 +02:00
const { FILE_MODEL_UID } = require('./constants');
const UPLOAD_MUTATION_NAME = 'upload';
const MULTIPLE_UPLOAD_MUTATION_NAME = 'multipleUpload';
const UPDATE_FILE_INFO_MUTATION_NAME = 'updateFileInfo';
const DELETE_FILE_MUTATION_NAME = 'removeFile';
const FILE_INFO_INPUT_TYPE_NAME = 'FileInfoInput';
/**
* @param {{ strapi: import('@strapi/strapi').Strapi }}
*/
module.exports = ({ strapi }) => {
const { service: getGraphQLService, config: graphQLConfig } = strapi.plugin('graphql');
const { service: getUploadService } = strapi.plugin('upload');
const isShadowCRUDEnabled = graphQLConfig('shadowCRUD', true);
if (!isShadowCRUDEnabled) {
return;
}
const { getTypeName, getEntityResponseName } = getGraphQLService('utils').naming;
const { toEntityResponse } = getGraphQLService('format').returnTypes;
2022-05-13 16:10:18 +02:00
const fileModel = strapi.getModel(FILE_MODEL_UID);
const fileTypeName = getTypeName(fileModel);
const fileEntityResponseType = getEntityResponseName(fileModel);
2022-02-28 10:49:16 +01:00
const { optimize, isSupportedImage } = getUploadService('image-manipulation');
/**
* Optimize and format a file using the upload services
*
* @param {object} upload
* @param {object} extraInfo
* @param {object} metas
* @return {Promise<object>}
*/
2022-02-16 12:23:11 +01:00
const formatFile = async (upload, extraInfo, metas) => {
const uploadService = getUploadService('upload');
const { filename, mimetype, createReadStream } = await upload;
const currentFile = await uploadService.formatFileInfo(
{
filename,
/**
* in case the mime-type wasn't sent, Strapi tries to guess it
* from the file extension, to avoid a corrupt database state
*/
type: mimetype || mime.lookup(filename) || 'application/octet-stream',
size: await getStreamSize(createReadStream()),
},
extraInfo || {},
metas
);
currentFile.getStream = createReadStream;
2022-02-28 10:49:16 +01:00
if (!(await isSupportedImage(currentFile))) {
return currentFile;
}
return optimize(currentFile);
};
/**
* Register Upload's types, queries & mutations to the content API using the GraphQL extension API
*/
getGraphQLService('extension').use(({ nexus }) => {
const { inputObjectType, extendType, nonNull, list } = nexus;
// Represents the input data payload for the file's information
const fileInfoInputType = inputObjectType({
name: FILE_INFO_INPUT_TYPE_NAME,
definition(t) {
t.string('name');
t.string('alternativeText');
t.string('caption');
},
});
const mutations = extendType({
type: 'Mutation',
definition(t) {
/**
* Upload a single file
*/
t.field(UPLOAD_MUTATION_NAME, {
type: nonNull(fileEntityResponseType),
args: {
refId: 'ID',
ref: 'String',
field: 'String',
info: FILE_INFO_INPUT_TYPE_NAME,
file: nonNull('Upload'),
},
async resolve(parent, args) {
// create temporary folder to store files for stream manipulation
2022-02-16 12:23:11 +01:00
const tmpWorkingDirectory = await fse.mkdtemp(path.join(os.tmpdir(), 'strapi-upload-'));
let sanitizedEntity;
try {
const { file: upload, info = {}, ...metas } = args;
const apiUploadFolderService = getUploadService('api-upload-folder');
const apiUploadFolder = await apiUploadFolderService.getAPIUploadFolder();
info.folder = apiUploadFolder.id;
2022-02-16 12:23:11 +01:00
const file = await formatFile(upload, info, { ...metas, tmpWorkingDirectory });
const uploadedFile = await getUploadService('upload').uploadFileAndPersist(file, {});
sanitizedEntity = await toEntityResponse(uploadedFile, {
args,
resourceUID: fileTypeName,
});
} finally {
// delete temporary folder
2022-02-16 12:23:11 +01:00
await fse.remove(tmpWorkingDirectory);
}
return sanitizedEntity;
},
});
/**
* Upload multiple files
*/
t.field(MULTIPLE_UPLOAD_MUTATION_NAME, {
type: nonNull(list(fileEntityResponseType)),
args: {
refId: 'ID',
ref: 'String',
field: 'String',
files: nonNull(list('Upload')),
},
async resolve(parent, args) {
// create temporary folder to store files for stream manipulation
2022-02-16 12:23:11 +01:00
const tmpWorkingDirectory = await fse.mkdtemp(path.join(os.tmpdir(), 'strapi-upload-'));
let sanitizedEntities = [];
try {
2022-02-16 12:23:11 +01:00
const { files: uploads, ...metas } = args;
const apiUploadFolderService = getUploadService('api-upload-folder');
const apiUploadFolder = await apiUploadFolderService.getAPIUploadFolder();
const files = await Promise.all(
2022-08-08 23:33:39 +02:00
uploads.map((upload) =>
formatFile(
upload,
{ folder: apiUploadFolder.id },
{ ...metas, tmpWorkingDirectory }
)
)
);
const uploadService = getUploadService('upload');
const uploadedFiles = await Promise.all(
2022-08-08 23:33:39 +02:00
files.map((file) => uploadService.uploadFileAndPersist(file, {}))
);
2022-08-08 23:33:39 +02:00
sanitizedEntities = uploadedFiles.map((file) =>
toEntityResponse(file, { args, resourceUID: fileTypeName })
);
} finally {
// delete temporary folder
2022-02-16 12:23:11 +01:00
await fse.remove(tmpWorkingDirectory);
}
return sanitizedEntities;
},
});
/**
* Update some information for a given file
*/
t.field(UPDATE_FILE_INFO_MUTATION_NAME, {
type: nonNull(fileEntityResponseType),
args: {
id: nonNull('ID'),
info: FILE_INFO_INPUT_TYPE_NAME,
},
async resolve(parent, args) {
const { id, info } = args;
const updatedFile = await getUploadService('upload').updateFileInfo(id, info);
return toEntityResponse(updatedFile, { args, resourceUID: fileTypeName });
},
});
/**
* Delete & remove a given file
*/
t.field(DELETE_FILE_MUTATION_NAME, {
type: fileEntityResponseType,
args: {
id: nonNull('ID'),
},
async resolve(parent, args) {
const { id } = args;
2021-09-28 11:11:03 +02:00
const file = await getUploadService('upload').findOne(id);
if (!file) {
return null;
}
const deletedFile = await getUploadService('upload').remove(file);
return toEntityResponse(deletedFile, { args, resourceUID: fileTypeName });
},
});
},
});
return {
types: [fileInfoInputType, mutations],
resolversConfig: {
// Use custom scopes for the upload file CRUD operations
2022-08-08 15:50:34 +02:00
'Query.uploadFiles': { auth: { scope: 'plugin::upload.content-api.find' } },
'Query.uploadFile': { auth: { scope: 'plugin::upload.content-api.findOne' } },
'Mutation.createUploadFile': { auth: { scope: 'plugin::upload.content-api.upload' } },
'Mutation.updateUploadFile': { auth: { scope: 'plugin::upload.content-api.upload' } },
'Mutation.deleteUploadFile': { auth: { scope: 'plugin::upload.content-api.destroy' } },
[`Mutation.${UPLOAD_MUTATION_NAME}`]: {
auth: { scope: 'plugin::upload.content-api.upload' },
},
[`Mutation.${MULTIPLE_UPLOAD_MUTATION_NAME}`]: {
auth: { scope: 'plugin::upload.content-api.upload' },
},
[`Mutation.${UPDATE_FILE_INFO_MUTATION_NAME}`]: {
auth: { scope: 'plugin::upload.content-api.upload' },
},
[`Mutation.${DELETE_FILE_MUTATION_NAME}`]: {
auth: { scope: 'plugin::upload.content-api.destroy' },
},
},
};
});
};