mirror of
				https://github.com/strapi/strapi.git
				synced 2025-10-31 18:08:11 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			263 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			263 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 'use strict';
 | |
| 
 | |
| const path = require('path');
 | |
| const os = require('os');
 | |
| const mime = require('mime-types');
 | |
| const fse = require('fs-extra');
 | |
| const {
 | |
|   file: { getStreamSize },
 | |
| } = require('@strapi/utils');
 | |
| 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;
 | |
| 
 | |
|   const fileModel = strapi.getModel(FILE_MODEL_UID);
 | |
|   const fileTypeName = getTypeName(fileModel);
 | |
|   const fileEntityResponseType = getEntityResponseName(fileModel);
 | |
| 
 | |
|   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>}
 | |
|    */
 | |
|   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;
 | |
| 
 | |
|     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
 | |
|             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;
 | |
| 
 | |
|               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
 | |
|               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
 | |
|             const tmpWorkingDirectory = await fse.mkdtemp(path.join(os.tmpdir(), 'strapi-upload-'));
 | |
|             let sanitizedEntities = [];
 | |
| 
 | |
|             try {
 | |
|               const { files: uploads, ...metas } = args;
 | |
| 
 | |
|               const apiUploadFolderService = getUploadService('api-upload-folder');
 | |
| 
 | |
|               const apiUploadFolder = await apiUploadFolderService.getAPIUploadFolder();
 | |
| 
 | |
|               const files = await Promise.all(
 | |
|                 uploads.map((upload) =>
 | |
|                   formatFile(
 | |
|                     upload,
 | |
|                     { folder: apiUploadFolder.id },
 | |
|                     { ...metas, tmpWorkingDirectory }
 | |
|                   )
 | |
|                 )
 | |
|               );
 | |
| 
 | |
|               const uploadService = getUploadService('upload');
 | |
| 
 | |
|               const uploadedFiles = await Promise.all(
 | |
|                 files.map((file) => uploadService.uploadFileAndPersist(file, {}))
 | |
|               );
 | |
| 
 | |
|               sanitizedEntities = uploadedFiles.map((file) =>
 | |
|                 toEntityResponse(file, { args, resourceUID: fileTypeName })
 | |
|               );
 | |
|             } finally {
 | |
|               // delete temporary folder
 | |
|               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;
 | |
| 
 | |
|             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
 | |
|         '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' },
 | |
|         },
 | |
|       },
 | |
|     };
 | |
|   });
 | |
| };
 | 
