From 11b234e2fcf6f0127033f84beeff268b6de6436e Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 5 Aug 2024 18:08:48 +0700 Subject: [PATCH 01/91] Create uploads package with extension --- packages/uploads/README.md | 7 + packages/uploads/build.mts | 32 +++ packages/uploads/package.json | 42 +++ packages/uploads/prisma/index.js | 2 + packages/uploads/prisma/package.json | 4 + packages/uploads/src/prismaExtension.ts | 347 +++++++++++++++++++++++ packages/uploads/tsconfig.json | 13 + packages/uploads/tsconfig.types-cjs.json | 7 + yarn.lock | 26 ++ 9 files changed, 480 insertions(+) create mode 100644 packages/uploads/README.md create mode 100644 packages/uploads/build.mts create mode 100644 packages/uploads/package.json create mode 100644 packages/uploads/prisma/index.js create mode 100644 packages/uploads/prisma/package.json create mode 100644 packages/uploads/src/prismaExtension.ts create mode 100644 packages/uploads/tsconfig.json create mode 100644 packages/uploads/tsconfig.types-cjs.json diff --git a/packages/uploads/README.md b/packages/uploads/README.md new file mode 100644 index 000000000000..94d6e30debf7 --- /dev/null +++ b/packages/uploads/README.md @@ -0,0 +1,7 @@ +# Uploads + +This package houses + +- Prisma extension +- Form uploader +- diff --git a/packages/uploads/build.mts b/packages/uploads/build.mts new file mode 100644 index 000000000000..d1ec3b003bfb --- /dev/null +++ b/packages/uploads/build.mts @@ -0,0 +1,32 @@ +import { writeFileSync } from 'node:fs' + +import { build, defaultBuildOptions } from '@redwoodjs/framework-tools' +import { generateCjsTypes } from '@redwoodjs/framework-tools/cjsTypes' + +// CJS build +await build({ + buildOptions: { + ...defaultBuildOptions, + outdir: 'dist/cjs', + packages: 'external', + }, +}) + +// ESM build +await build({ + buildOptions: { + ...defaultBuildOptions, + format: 'esm', + packages: 'external', + }, +}) + +// Place a package.json file with `type: commonjs` in the dist/cjs folder so that +// all .js files are treated as CommonJS files. +writeFileSync('dist/cjs/package.json', JSON.stringify({ type: 'commonjs' })) + +// Place a package.json file with `type: module` in the dist folder so that +// all .js files are treated as ES Module files. +writeFileSync('dist/package.json', JSON.stringify({ type: 'module' })) + +await generateCjsTypes() diff --git a/packages/uploads/package.json b/packages/uploads/package.json new file mode 100644 index 000000000000..23f8ed72a863 --- /dev/null +++ b/packages/uploads/package.json @@ -0,0 +1,42 @@ +{ + "name": "@redwoodjs/uploads", + "type": "module", + "version": "7.0.0", + "repository": { + "type": "git", + "url": "git+https://github.com/redwoodjs/redwood.git", + "directory": "packages/uploads" + }, + "license": "MIT", + "exports": { + "./prisma": { + "require": "./dist/cjs/prismaExtension.js", + "import": "./dist/prismaExtension.js" + } + }, + "files": [ + "dist", + "prisma" + ], + "scripts": { + "build": "tsx ./build.mts && run build:types", + "build:pack": "yarn pack -o redwoodjs-uploads.tgz", + "build:types": "tsc --build --verbose", + "build:types-cjs": "tsc --build --verbose tsconfig.types-cjs.json" + }, + "dependencies": { + "@redwoodjs/project-config": "workspace:*", + "fs-extra": "11.2.0", + "mime-types": "2.1.35", + "ulid": "2.3.0" + }, + "devDependencies": { + "@redwoodjs/framework-tools": "workspace:*", + "@types/fs-extra": "11.0.4", + "@types/mime-types": "2.1.4", + "esbuild": "0.23.0", + "tsx": "4.16.2", + "typescript": "5.5.4" + }, + "gitHead": "3905ed045508b861b495f8d5630d76c7a157d8f1" +} diff --git a/packages/uploads/prisma/index.js b/packages/uploads/prisma/index.js new file mode 100644 index 000000000000..f21ea7b2fd3d --- /dev/null +++ b/packages/uploads/prisma/index.js @@ -0,0 +1,2 @@ +/* eslint-env es6, commonjs */ +module.exports = require('../dist/cjs/prismaExtension.js') diff --git a/packages/uploads/prisma/package.json b/packages/uploads/prisma/package.json new file mode 100644 index 000000000000..56705d7ff21b --- /dev/null +++ b/packages/uploads/prisma/package.json @@ -0,0 +1,4 @@ +{ + "main": "./index.js", + "types": "../dist/cjs/prisma/index.d.ts" +} diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts new file mode 100644 index 000000000000..782edb9901fc --- /dev/null +++ b/packages/uploads/src/prismaExtension.ts @@ -0,0 +1,347 @@ +import fs from 'node:fs/promises' +import path from 'node:path' + +import { PrismaClient } from '@prisma/client' +import type * as runtime from '@prisma/client/runtime/library' +import { ulid } from 'ulid' + +import { getPaths } from '@redwoodjs/project-config' + +type FilterOutDollarPrefixed = T extends `$${string}` ? never : T +// Filter out $on, $connect, etc. +type ModelNames = FilterOutDollarPrefixed + +export type UploadConfigForModel = { + fields: [string] | string + savePath?: ((args: unknown) => string) | string + fileName?: (args: unknown) => string + onFileSaved?: (filePath: string) => void | Promise +} + +export type UploadsConfig = { + [key in ModelNames]?: UploadConfigForModel +} + +type TUSServerConfig = { + tusUploadDirectory: string +} + +type QueryExtends = { + [key in ModelNames]?: { + update: any + create: any + delete: any + } +} + +type ResultExtends = { + [key in ModelNames]?: { + withDataUri: { + needs: any + compute: (record: any) => () => Promise + } + withPublicUrl: { + needs: any + compute: (record: any) => () => Promise + } + } +} + +type ExtendsType = runtime.ExtensionArgs + +export const createUploadsExtension = ( + config: UploadsConfig, + tusConfig?: TUSServerConfig, +) => { + // @MARK typing these with ExtendsType['query'] and ExtendsType['result'] + // will create an error when we instiate the PrismaClient 🤷 + // but without these types prisma won''t show types for the new methods + const prismaInstance = new PrismaClient() + + async function deleteUploadsFromDiskForArgs({ + model, + args, + fields, + }: { + model: string + args: runtime.JsArgs // @TODO: type this better this is actually a where + fields: string[] + }) { + const record = await prismaInstance[model].findFirstOrThrow(args) + + // Delete the file from the file system + fields.forEach(async (field) => { + const filePath = record[field] + await fs.unlink(filePath) + }) + } + // This gives us typesafety when we write the extension, + // but we override it on return, because TS complains on instantiation of the PrismaClient + const queryExtends: ExtendsType['query'] = {} + const resultExtends: ExtendsType['result'] = {} + + for (const modelName in config) { + const modelConfig = config[modelName] as UploadConfigForModel + const uploadFields = Array.isArray(modelConfig.fields) + ? modelConfig.fields + : [modelConfig.fields] + + queryExtends[modelName] = { + // @TODO: in update we'll need to delete the old file, if the field is being updated + // THis will depend on whether we have a table for uploads or not + async update({ query, model, args }) { + await deleteUploadsFromDiskForArgs({ + model, + args: { + // The update args contains data, which we don't need to supply to delete + where: args.where, + }, + fields: uploadFields, + }) + + const uploadArgs = await saveUploads( + uploadFields, + args, + modelConfig, + tusConfig, + ) + + return query(uploadArgs) + }, + async create({ query, args }) { + const uploadArgs = await saveUploads( + uploadFields, + args, + modelConfig, + tusConfig, + ) + + return query(uploadArgs) + }, + async delete({ model, query, args }) { + await deleteUploadsFromDiskForArgs({ + model, + args, + fields: uploadFields, + }) + + return query(args) + }, + // findMany({ query, args, operation }) {} + } + + resultExtends[modelName] = { + withDataUri: { + needs: { avatar: true }, // specify the field name here, so it doesn't appear as a function if avatar isn't requested + compute(contact) { + return async () => { + // @TODO: edge cases + // 1. If readfile fails - file not found, etc. + // 2. If not a path, relative or absolute, throw error + const base64Content = await fs.readFile(contact.avatar, 'base64url') + return { + ...contact, + avatar: base64Content, + } + } + }, + }, + withPublicUrl: { + needs: { avatar: true }, // specify the field name here, so it doesn't appear as a function if avatar isn't requested + compute(contact) { + return async () => { + // @TODO: Test cases + // 1. If the avatar is a base64 string, we should return it as is, but warn + // 2. If absolute path, but not public, throw error + // 3. If relative path, but not public, throw error + const webPublicPath = contact.avatar.replace('web/public', '') + return { + ...contact, + avatar: webPublicPath, + } + } + }, + }, + } + } + + return { + name: 'redwood-upload-prisma-plugin', + query: queryExtends as QueryExtends, + result: resultExtends as ResultExtends, + } +} + +/** + * Returns new args to use in create or update. + * + * Pass this to the query function! + */ +async function saveUploads( + uploadFields: string[], + args: runtime.JsArgs & { + data?: { + [key: string]: runtime.JsInputValue + } + }, + modelConfig: UploadConfigForModel, + tusConfig?: TUSServerConfig, +) { + const fieldsToUpdate: { + [key: string]: string + } = {} + + if (!args.data) { + throw new Error('No data in prisma query') + } + + // For each upload property, we need to: + // 1. save the file to the file system (path or name from config) + // 2. replace the value of the field + for await (const field of uploadFields) { + const uploadUrlOrDataUrl = args.data[field] as string + + if (!uploadUrlOrDataUrl) { + continue + } + + const fileName = + modelConfig.fileName && typeof modelConfig.fileName === 'function' + ? modelConfig.fileName(args) + : ulid() + + const saveDir = + typeof modelConfig.savePath === 'function' + ? modelConfig.savePath(args) + : modelConfig.savePath || 'web/public/uploads' + + const savedFilePath = await saveUploadToFile( + uploadUrlOrDataUrl, + { + fileName, + saveDir, + }, + tusConfig, + ) + + fieldsToUpdate[field] = savedFilePath + + // Call the onFileSaved callback + // Having it here means it'll always trigger whether create/update + if (modelConfig.onFileSaved) { + await modelConfig.onFileSaved(savedFilePath) + } + } + + // Can't spread according to TS + const newData = Object.assign(args.data, fieldsToUpdate) + + return { + ...args, + data: newData, + } +} + +async function saveUploadToFile( + uploadUrlOrDataUrl: string, + { fileName, saveDir }: { saveDir: string; fileName: string }, + tusConfig?: TUSServerConfig, +) { + let outputPath: string | null = null + + if (isBase65(uploadUrlOrDataUrl)) { + outputPath = await saveBase65File(uploadUrlOrDataUrl, { + saveDir, + fileName, + }) + } else if (uploadUrlOrDataUrl.startsWith('http')) { + if (!tusConfig) { + throw new Error('TusConfig not supplied.') + } + + outputPath = await saveTusUpload(uploadUrlOrDataUrl, { + tusConfig, + saveDir, + fileName, + }) + } // @TODO: add support for form uploads? + + if (!outputPath) { + throw new Error('Unsupported upload URL') + } + + // @MARK: we can create a new record on the uploads table here + + return outputPath +} + +// @MARK: if we block the TUS GET, we don't really need to move it +// We send the TUS upload URL as the value of the field +async function saveTusUpload( + uploadUrl: string, + { + tusConfig, + saveDir, + fileName, + }: { + tusConfig: TUSServerConfig + saveDir: string + fileName: string + }, +) { + const tusId = uploadUrl.split('/').slice(-1).pop() + + if (!tusId) { + throw new Error('Could not extract upload ID from URL') + } + + if (!tusConfig.tusUploadDirectory) { + throw new Error( + 'You have to configure the TUS Upload Directory in the prisma extension. It is required for TUS uploads', + ) + } + + // Optional Step.... + const metaFile = path.join( + path.isAbsolute(tusConfig.tusUploadDirectory) + ? tusConfig.tusUploadDirectory + : // @MARK: if the directory supplied isn't relative + path.join(getPaths().base, tusConfig.tusUploadDirectory), + `${tusId}.json`, + ) + // Can't await import, because JSON file. + const tusMeta = require(metaFile) + + const fileExtension = tusMeta.metadata.filetype.split('/')[1] + + const savedFilePath = path.join(saveDir, `${fileName}.${fileExtension}`) + + // @MARK: we can also move... + await fs.copyFile( + path.join(tusConfig.tusUploadDirectory, tusId), + savedFilePath, + ) + + return savedFilePath +} + +function isBase65(uploadUrlOrDataUrl: string) { + // Check if the uploadUrlOrDataUrl is a valid base64 string + const base64Regex = /^data:(.*?);base64,/ + return base64Regex.test(uploadUrlOrDataUrl) +} + +async function saveBase65File( + dataUrlString: string, + { saveDir, fileName }: { saveDir: string; fileName: string }, +) { + // @TODO, use mime-types package to get the extension here. + const [dataType, fileContent] = dataUrlString.split(',') + // format is data:image/png;base64,.... + const fileExtension = dataType.split('/')[1].split(';')[0] + const filePath = path.join(saveDir, `${fileName}.${fileExtension}`) + + await fs.writeFile(filePath, Buffer.from(fileContent, 'base64')) + + return filePath +} diff --git a/packages/uploads/tsconfig.json b/packages/uploads/tsconfig.json new file mode 100644 index 000000000000..a8c9c1419bf6 --- /dev/null +++ b/packages/uploads/tsconfig.json @@ -0,0 +1,13 @@ +{ + "extends": "../../tsconfig.compilerOption.json", + "compilerOptions": { + "moduleResolution": "NodeNext", + "module": "NodeNext", + "baseUrl": ".", + "rootDir": "src", + "outDir": "dist" + }, + "include": ["src"], + "references": [ + ] +} diff --git a/packages/uploads/tsconfig.types-cjs.json b/packages/uploads/tsconfig.types-cjs.json new file mode 100644 index 000000000000..6bbdc61737c4 --- /dev/null +++ b/packages/uploads/tsconfig.types-cjs.json @@ -0,0 +1,7 @@ +{ + "extends": "./tsconfig.json", + "compilerOptions": { + "outDir": "dist/cjs", + "tsBuildInfoFile": "./tsconfig.types-cjs.tsbuildinfo" + } +} diff --git a/yarn.lock b/yarn.lock index cd78288431a9..781f7bd104e9 100644 --- a/yarn.lock +++ b/yarn.lock @@ -8670,6 +8670,23 @@ __metadata: languageName: unknown linkType: soft +"@redwoodjs/uploads@workspace:packages/uploads": + version: 0.0.0-use.local + resolution: "@redwoodjs/uploads@workspace:packages/uploads" + dependencies: + "@redwoodjs/framework-tools": "workspace:*" + "@redwoodjs/project-config": "workspace:*" + "@types/fs-extra": "npm:11.0.4" + "@types/mime-types": "npm:2.1.4" + esbuild: "npm:0.23.0" + fs-extra: "npm:11.2.0" + mime-types: "npm:2.1.35" + tsx: "npm:4.16.2" + typescript: "npm:5.5.4" + ulid: "npm:2.3.0" + languageName: unknown + linkType: soft + "@redwoodjs/vite@workspace:packages/vite": version: 0.0.0-use.local resolution: "@redwoodjs/vite@workspace:packages/vite" @@ -28860,6 +28877,15 @@ __metadata: languageName: node linkType: hard +"ulid@npm:2.3.0": + version: 2.3.0 + resolution: "ulid@npm:2.3.0" + bin: + ulid: ./bin/cli.js + checksum: 10c0/070d237502781085e59cf3d8ece752ff96cd3a0990cf1c1be57273f4550597daeb72e9a7db8e5a320de31102509bb3321d280b54bfc44e98025e4628a9629773 + languageName: node + linkType: hard + "unbox-primitive@npm:^1.0.2": version: 1.0.2 resolution: "unbox-primitive@npm:1.0.2" From 2d748b2e836fb969538ed8d800d1420df1563f10 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 5 Aug 2024 19:15:40 +0700 Subject: [PATCH 02/91] Fix types after generating local prisma client for testing --- packages/uploads/attw.ts | 31 +++++++++++++++++++ packages/uploads/package.json | 8 ++++- .../src/__tests__/prismaExtension.test.ts | 0 packages/uploads/src/__tests__/schema.prisma | 19 ++++++++++++ packages/uploads/src/prismaExtension.ts | 8 +++-- packages/uploads/vitest.config.mts | 11 +++++++ yarn.lock | 2 ++ 7 files changed, 76 insertions(+), 3 deletions(-) create mode 100644 packages/uploads/attw.ts create mode 100644 packages/uploads/src/__tests__/prismaExtension.test.ts create mode 100644 packages/uploads/src/__tests__/schema.prisma create mode 100644 packages/uploads/vitest.config.mts diff --git a/packages/uploads/attw.ts b/packages/uploads/attw.ts new file mode 100644 index 000000000000..a377f7b5f320 --- /dev/null +++ b/packages/uploads/attw.ts @@ -0,0 +1,31 @@ +import { $ } from 'zx' + +interface Problem { + kind: string + entrypoint?: string + resolutionKind?: string +} + +await $({ nothrow: true })`yarn attw -P -f json > .attw.json` +const output = await $`cat .attw.json` +await $`rm .attw.json` + +const json = JSON.parse(output.stdout) + +if (!json.analysis.problems || json.analysis.problems.length === 0) { + console.log('No errors found') + process.exit(0) +} + +if ( + json.analysis.problems.every( + (problem: Problem) => problem.resolutionKind === 'node10', + ) +) { + console.log("Only found node10 problems, which we don't care about") + process.exit(0) +} + +console.log('Errors found') +console.log(json.analysis.problems) +process.exit(1) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 23f8ed72a863..c84b141b1220 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -2,6 +2,7 @@ "name": "@redwoodjs/uploads", "type": "module", "version": "7.0.0", + "types": "dist/prismaExtension.d.ts", "repository": { "type": "git", "url": "git+https://github.com/redwoodjs/redwood.git", @@ -22,7 +23,9 @@ "build": "tsx ./build.mts && run build:types", "build:pack": "yarn pack -o redwoodjs-uploads.tgz", "build:types": "tsc --build --verbose", - "build:types-cjs": "tsc --build --verbose tsconfig.types-cjs.json" + "build:types-cjs": "tsc --build --verbose tsconfig.types-cjs.json", + "check:attw": "tsx attw.ts", + "check:package": "concurrently npm:check:attw yarn publint" }, "dependencies": { "@redwoodjs/project-config": "workspace:*", @@ -31,10 +34,13 @@ "ulid": "2.3.0" }, "devDependencies": { + "@arethetypeswrong/cli": "0.15.3", + "@prisma/client": "5.17.0", "@redwoodjs/framework-tools": "workspace:*", "@types/fs-extra": "11.0.4", "@types/mime-types": "2.1.4", "esbuild": "0.23.0", + "publint": "0.2.9", "tsx": "4.16.2", "typescript": "5.5.4" }, diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts new file mode 100644 index 000000000000..e69de29bb2d1 diff --git a/packages/uploads/src/__tests__/schema.prisma b/packages/uploads/src/__tests__/schema.prisma new file mode 100644 index 000000000000..f2a46e6fd5b4 --- /dev/null +++ b/packages/uploads/src/__tests__/schema.prisma @@ -0,0 +1,19 @@ +datasource db { + provider = "sqlite" + url = "file:never_used.db" +} + +generator client { + provider = "prisma-client-js" +} + +model Dummy { + id Int @id @default(autoincrement()) + uploadField String +} + +model Dumbo { + id Int @id @default(autoincrement()) + firstUpload String + secondUpload String +} diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 782edb9901fc..9a5a729d6a8f 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -67,7 +67,10 @@ export const createUploadsExtension = ( args: runtime.JsArgs // @TODO: type this better this is actually a where fields: string[] }) { - const record = await prismaInstance[model].findFirstOrThrow(args) + const record = + // in the project its OK + // @ts-expect-error can't get prisma to stop complaining here + await prismaInstance[model as ModelNames].findFirstOrThrow(args) // Delete the file from the file system fields.forEach(async (field) => { @@ -81,7 +84,8 @@ export const createUploadsExtension = ( const resultExtends: ExtendsType['result'] = {} for (const modelName in config) { - const modelConfig = config[modelName] as UploadConfigForModel + // Guaranteed to have modelConfig, we're looping over config 🙄 + const modelConfig = config[modelName as ModelNames] as UploadConfigForModel const uploadFields = Array.isArray(modelConfig.fields) ? modelConfig.fields : [modelConfig.fields] diff --git a/packages/uploads/vitest.config.mts b/packages/uploads/vitest.config.mts new file mode 100644 index 000000000000..580e8462c027 --- /dev/null +++ b/packages/uploads/vitest.config.mts @@ -0,0 +1,11 @@ +import { defineConfig, configDefaults } from 'vitest/config' + +export default defineConfig({ + test: { + exclude: [...configDefaults.exclude, '**/fixtures'], + deps: { + interopDefault: false, + } + }, + +}) diff --git a/yarn.lock b/yarn.lock index 781f7bd104e9..089f93563eba 100644 --- a/yarn.lock +++ b/yarn.lock @@ -8674,6 +8674,7 @@ __metadata: version: 0.0.0-use.local resolution: "@redwoodjs/uploads@workspace:packages/uploads" dependencies: + "@arethetypeswrong/cli": "npm:0.15.3" "@redwoodjs/framework-tools": "workspace:*" "@redwoodjs/project-config": "workspace:*" "@types/fs-extra": "npm:11.0.4" @@ -8681,6 +8682,7 @@ __metadata: esbuild: "npm:0.23.0" fs-extra: "npm:11.2.0" mime-types: "npm:2.1.35" + publint: "npm:0.2.9" tsx: "npm:4.16.2" typescript: "npm:5.5.4" ulid: "npm:2.3.0" From a32a3c82befce56791becc79484a68f1a3624d7c Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 5 Aug 2024 22:52:07 +0700 Subject: [PATCH 03/91] Add gitignore --- packages/uploads/.gitignore | 3 +++ 1 file changed, 3 insertions(+) create mode 100644 packages/uploads/.gitignore diff --git a/packages/uploads/.gitignore b/packages/uploads/.gitignore new file mode 100644 index 000000000000..0affa40abc3a --- /dev/null +++ b/packages/uploads/.gitignore @@ -0,0 +1,3 @@ +src/__tests__/migrations/* +src/__tests__/for_unit_test.db* + From e08016d102c5a5261f68e3a6e8de96d811d1111d Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 5 Aug 2024 22:53:38 +0700 Subject: [PATCH 04/91] Rename test schema --- .../src/__tests__/{schema.prisma => unit-test-schema.prisma} | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) rename packages/uploads/src/__tests__/{schema.prisma => unit-test-schema.prisma} (87%) diff --git a/packages/uploads/src/__tests__/schema.prisma b/packages/uploads/src/__tests__/unit-test-schema.prisma similarity index 87% rename from packages/uploads/src/__tests__/schema.prisma rename to packages/uploads/src/__tests__/unit-test-schema.prisma index f2a46e6fd5b4..2040c947ba95 100644 --- a/packages/uploads/src/__tests__/schema.prisma +++ b/packages/uploads/src/__tests__/unit-test-schema.prisma @@ -1,6 +1,6 @@ datasource db { provider = "sqlite" - url = "file:never_used.db" + url = "file:for_unit_test.db" } generator client { From 56f008a314ccbd28289dbf7b5b6415aa9112714f Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 5 Aug 2024 22:53:55 +0700 Subject: [PATCH 05/91] Add initial tests --- packages/uploads/package.json | 8 +- packages/uploads/src/__tests__/fileMocks.js | 2 + .../src/__tests__/getFileExtension.test.ts | 21 +++++ .../src/__tests__/prismaExtension.test.ts | 90 +++++++++++++++++++ packages/uploads/src/prismaExtension.ts | 22 +++-- yarn.lock | 3 + 6 files changed, 137 insertions(+), 9 deletions(-) create mode 100644 packages/uploads/src/__tests__/fileMocks.js create mode 100644 packages/uploads/src/__tests__/getFileExtension.test.ts diff --git a/packages/uploads/package.json b/packages/uploads/package.json index c84b141b1220..c5573df58e37 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -25,7 +25,9 @@ "build:types": "tsc --build --verbose", "build:types-cjs": "tsc --build --verbose tsconfig.types-cjs.json", "check:attw": "tsx attw.ts", - "check:package": "concurrently npm:check:attw yarn publint" + "check:package": "concurrently npm:check:attw yarn publint", + "test": "vitest run", + "test:watch": "vitest watch" }, "dependencies": { "@redwoodjs/project-config": "workspace:*", @@ -39,10 +41,12 @@ "@redwoodjs/framework-tools": "workspace:*", "@types/fs-extra": "11.0.4", "@types/mime-types": "2.1.4", + "concurrently": "8.2.2", "esbuild": "0.23.0", "publint": "0.2.9", "tsx": "4.16.2", - "typescript": "5.5.4" + "typescript": "5.5.4", + "vitest": "2.0.4" }, "gitHead": "3905ed045508b861b495f8d5630d76c7a157d8f1" } diff --git a/packages/uploads/src/__tests__/fileMocks.js b/packages/uploads/src/__tests__/fileMocks.js new file mode 100644 index 000000000000..469745009c0b --- /dev/null +++ b/packages/uploads/src/__tests__/fileMocks.js @@ -0,0 +1,2 @@ +export const dataUrlPng = + '' diff --git a/packages/uploads/src/__tests__/getFileExtension.test.ts b/packages/uploads/src/__tests__/getFileExtension.test.ts new file mode 100644 index 000000000000..85629d8b4429 --- /dev/null +++ b/packages/uploads/src/__tests__/getFileExtension.test.ts @@ -0,0 +1,21 @@ +import { describe, it, expect } from 'vitest' + +import { getFileExtension } from '../prismaExtension' + +describe('getFileExtension', () => { + it('should return the correct file extension for a given data type', () => { + const dataType = 'data:image/png;base64' + const extension = getFileExtension(dataType) + expect(extension).toBe('png') + }) + + it('handles svgs', () => { + const dataType = 'data:image/svg+xml;base64' + expect(getFileExtension(dataType)).toBe('svg') + }) + + it('handles gif', () => { + const dataType = 'data:image/gif;base64' + expect(getFileExtension(dataType)).toBe('gif') + }) +}) diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index e69de29bb2d1..eb9b51527a28 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -0,0 +1,90 @@ +import fs from 'node:fs/promises' + +import { PrismaClient } from '@prisma/client' +import { describe, it, vi, expect } from 'vitest' + +import { createUploadsExtension } from '../prismaExtension' + +import { dataUrlPng } from './fileMocks' + +vi.mock('node:fs/promises', () => ({ + default: { + writeFile: vi.fn(), + }, +})) + +describe('Uploads Prisma Extension', () => { + const dummyUploadConfig = { + fields: 'uploadField', + savePath: '/bazinga', + onFileSaved: vi.fn(), + } + + const dumboUploadConfig = { + fields: ['firstUpload', 'secondUpload'], + savePath: '/dumbo', + onFileSaved: vi.fn(), + } + const prismaClient = new PrismaClient().$extends( + createUploadsExtension({ + dummy: dummyUploadConfig, + dumbo: dumboUploadConfig, + }), + ) + + describe('Query extensions', () => { + it('will create a file with base64 encoded png', async () => { + const dum1 = await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) + + expect(dummyUploadConfig.onFileSaved).toHaveBeenCalled() + expect(dum1.uploadField).toMatch(/bazinga\/.*\.png/) + expect(fs.writeFile).toHaveBeenCalledWith( + expect.stringMatching(/bazinga\/.*\.png/), + expect.anything(), // no need to check content here, makes test slow + ) + }) + + it('handles multiple upload fields', async () => { + const dumbo = await prismaClient.dumbo.create({ + data: { + firstUpload: dataUrlPng, + secondUpload: dataUrlPng, + }, + }) + + expect(dumbo.firstUpload).toMatch(/dumbo\/.*\.png/) + expect(dumbo.secondUpload).toMatch(/dumbo\/.*\.png/) + + expect(dumboUploadConfig.onFileSaved).toHaveBeenCalledTimes(2) + }) + + // @TODO implement these tests + it('handles empty, null, or undefined fields', async () => {}) + + // it('handles updates, and removes old files', async () => {}) + + // it('handles deletes, and removes files', async () => {}) + + // it('supports custom file name functions', async () => {}) + + // it('supports custom save path functions', async () => {}) + + // it('will move file to new location with TUS uploads', async () => {}) + + // it('will remove old file when updating with TUS uploads', async () => {}) + + // it('will remove file when deleting with TUS uploads', async () => {}) + }) + + describe('Result extensions', () => { + it('will return a data URL for the file', async () => {}) + // @TODO implement + // it('will return a public URL for the file', async () => {}) + // it('if file is not found, will throw an error', async () => {}) + // it('if saved file is not a path, will throw an error', async () => {}) + }) +}) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 9a5a729d6a8f..4a876c3b4b0c 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -3,6 +3,7 @@ import path from 'node:path' import { PrismaClient } from '@prisma/client' import type * as runtime from '@prisma/client/runtime/library' +import mime from 'mime-types' import { ulid } from 'ulid' import { getPaths } from '@redwoodjs/project-config' @@ -12,7 +13,7 @@ type FilterOutDollarPrefixed = T extends `$${string}` ? never : T type ModelNames = FilterOutDollarPrefixed export type UploadConfigForModel = { - fields: [string] | string + fields: string[] | string savePath?: ((args: unknown) => string) | string fileName?: (args: unknown) => string onFileSaved?: (filePath: string) => void | Promise @@ -38,11 +39,11 @@ type ResultExtends = { [key in ModelNames]?: { withDataUri: { needs: any - compute: (record: any) => () => Promise + compute: (record: T) => () => Promise } withPublicUrl: { needs: any - compute: (record: any) => () => Promise + compute: (record: T) => () => Promise } } } @@ -80,6 +81,7 @@ export const createUploadsExtension = ( } // This gives us typesafety when we write the extension, // but we override it on return, because TS complains on instantiation of the PrismaClient + // its important that the resultsExtends const queryExtends: ExtendsType['query'] = {} const resultExtends: ExtendsType['result'] = {} @@ -91,8 +93,6 @@ export const createUploadsExtension = ( : [modelConfig.fields] queryExtends[modelName] = { - // @TODO: in update we'll need to delete the old file, if the field is being updated - // THis will depend on whether we have a table for uploads or not async update({ query, model, args }) { await deleteUploadsFromDiskForArgs({ model, @@ -339,13 +339,21 @@ async function saveBase65File( dataUrlString: string, { saveDir, fileName }: { saveDir: string; fileName: string }, ) { - // @TODO, use mime-types package to get the extension here. const [dataType, fileContent] = dataUrlString.split(',') // format is data:image/png;base64,.... - const fileExtension = dataType.split('/')[1].split(';')[0] + const fileExtension = getFileExtension(dataType) const filePath = path.join(saveDir, `${fileName}.${fileExtension}`) await fs.writeFile(filePath, Buffer.from(fileContent, 'base64')) return filePath } + +export function getFileExtension(dataType: string): string { + const mimeType = dataType.split(':')[1].split(';')[0] + const extension = mime.extension(mimeType) + if (!extension) { + throw new Error(`Unsupported file type: ${mimeType}`) + } + return extension +} diff --git a/yarn.lock b/yarn.lock index 089f93563eba..29a23a4bccd4 100644 --- a/yarn.lock +++ b/yarn.lock @@ -8675,10 +8675,12 @@ __metadata: resolution: "@redwoodjs/uploads@workspace:packages/uploads" dependencies: "@arethetypeswrong/cli": "npm:0.15.3" + "@prisma/client": "npm:5.17.0" "@redwoodjs/framework-tools": "workspace:*" "@redwoodjs/project-config": "workspace:*" "@types/fs-extra": "npm:11.0.4" "@types/mime-types": "npm:2.1.4" + concurrently: "npm:8.2.2" esbuild: "npm:0.23.0" fs-extra: "npm:11.2.0" mime-types: "npm:2.1.35" @@ -8686,6 +8688,7 @@ __metadata: tsx: "npm:4.16.2" typescript: "npm:5.5.4" ulid: "npm:2.3.0" + vitest: "npm:2.0.4" languageName: unknown linkType: soft From 7133768f9b579ea2b2eb3dfe2ef43da89bed3d45 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 5 Aug 2024 22:58:38 +0700 Subject: [PATCH 06/91] Update gitignore again --- packages/uploads/.gitignore | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/uploads/.gitignore b/packages/uploads/.gitignore index 0affa40abc3a..e5c46c18f78e 100644 --- a/packages/uploads/.gitignore +++ b/packages/uploads/.gitignore @@ -1,3 +1,3 @@ src/__tests__/migrations/* src/__tests__/for_unit_test.db* - +.attw.json From 7d157aeee1d56703b42e9fb89c7e23404ebac164 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 5 Aug 2024 22:58:44 +0700 Subject: [PATCH 07/91] Add test for empty values --- .../uploads/src/__tests__/prismaExtension.test.ts | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index eb9b51527a28..17555cfd1797 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -62,9 +62,17 @@ describe('Uploads Prisma Extension', () => { expect(dumboUploadConfig.onFileSaved).toHaveBeenCalledTimes(2) }) - // @TODO implement these tests - it('handles empty, null, or undefined fields', async () => {}) + it('handles empty fields', async () => { + const emptyUploadFieldPromise = prismaClient.dummy.create({ + data: { + uploadField: '', + }, + }) + await expect(emptyUploadFieldPromise).resolves.not.toThrow() + }) + + // @TODO implement these tests // it('handles updates, and removes old files', async () => {}) // it('handles deletes, and removes files', async () => {}) From 40f0011b1c0a5af6a0d902a77dc2522542d6ee1f Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 5 Aug 2024 23:00:56 +0700 Subject: [PATCH 08/91] Add prisma setup script --- packages/uploads/package.json | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index c5573df58e37..e704d7c6b933 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -26,7 +26,8 @@ "build:types-cjs": "tsc --build --verbose tsconfig.types-cjs.json", "check:attw": "tsx attw.ts", "check:package": "concurrently npm:check:attw yarn publint", - "test": "vitest run", + "setup:test": "npx prisma migrate reset -f --schema src/__tests__/unit-test-schema.prisma", + "test": "yarn run setup:test && vitest run", "test:watch": "vitest watch" }, "dependencies": { From 59f83ccbb996937652e64ddbdb315742c982ea72 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 5 Aug 2024 23:12:01 +0700 Subject: [PATCH 09/91] Update tsconfig --- packages/uploads/tsconfig.json | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/packages/uploads/tsconfig.json b/packages/uploads/tsconfig.json index a8c9c1419bf6..43d163492556 100644 --- a/packages/uploads/tsconfig.json +++ b/packages/uploads/tsconfig.json @@ -9,5 +9,11 @@ }, "include": ["src"], "references": [ + { + "path": "../project-config" + }, + { + "path": "../framework-tools" + }, ] } From 2c28d76bdb54977f578ab2ed59f076516f41e4ad Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 5 Aug 2024 23:13:45 +0700 Subject: [PATCH 10/91] Update readme --- packages/uploads/README.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/uploads/README.md b/packages/uploads/README.md index 94d6e30debf7..b6b7c36ce457 100644 --- a/packages/uploads/README.md +++ b/packages/uploads/README.md @@ -2,6 +2,6 @@ This package houses -- Prisma extension -- Form uploader -- +- Prisma extension for handling uploads +- Base64 file picker uploader (?) +- TUS uploader (?) From bdbdccf25a4da382ae23b2c68a0b866a916b5169 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 5 Aug 2024 23:15:23 +0700 Subject: [PATCH 11/91] Add setup:test to watch command too --- packages/uploads/package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index e704d7c6b933..f0b8cff5e6b3 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -28,7 +28,7 @@ "check:package": "concurrently npm:check:attw yarn publint", "setup:test": "npx prisma migrate reset -f --schema src/__tests__/unit-test-schema.prisma", "test": "yarn run setup:test && vitest run", - "test:watch": "vitest watch" + "test:watch": "yarn run setup:test && vitest watch" }, "dependencies": { "@redwoodjs/project-config": "workspace:*", From 9ab71a7205a5f95b6bc9c7535fd6025f5eac0d0d Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 6 Aug 2024 13:24:01 +0700 Subject: [PATCH 12/91] Change types again --- packages/uploads/src/prismaExtension.ts | 20 +++++++++++++------- 1 file changed, 13 insertions(+), 7 deletions(-) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 4a876c3b4b0c..34d0423d3348 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -8,7 +8,12 @@ import { ulid } from 'ulid' import { getPaths } from '@redwoodjs/project-config' -type FilterOutDollarPrefixed = T extends `$${string}` ? never : T +type FilterOutDollarPrefixed = T extends `$${string}` + ? never + : T extends symbol // Remove symbol here, because it doesn't help users + ? never + : T + // Filter out $on, $connect, etc. type ModelNames = FilterOutDollarPrefixed @@ -59,19 +64,20 @@ export const createUploadsExtension = ( // but without these types prisma won''t show types for the new methods const prismaInstance = new PrismaClient() - async function deleteUploadsFromDiskForArgs({ + async function deleteUploadsFromDiskForArgs({ model, args, fields, }: { model: string - args: runtime.JsArgs // @TODO: type this better this is actually a where + args: T fields: string[] }) { - const record = - // in the project its OK - // @ts-expect-error can't get prisma to stop complaining here - await prismaInstance[model as ModelNames].findFirstOrThrow(args) + // With strict mode you cannot call findFirstOrThrow with the same args, because it is a union type + // Ideally there's a better way to do this + const record = await ( + prismaInstance[model as ModelNames] as any + ).findFirstOrThrow(args) // Delete the file from the file system fields.forEach(async (field) => { From d8a67951c7014e9cda820afe024bfed8ef3c9795 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 6 Aug 2024 13:32:10 +0700 Subject: [PATCH 13/91] Fix types appearing correctly --- packages/uploads/package.json | 10 ++++++++-- packages/uploads/prisma/package.json | 2 +- 2 files changed, 9 insertions(+), 3 deletions(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index f0b8cff5e6b3..395478389cd1 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -11,8 +11,14 @@ "license": "MIT", "exports": { "./prisma": { - "require": "./dist/cjs/prismaExtension.js", - "import": "./dist/prismaExtension.js" + "require": { + "types": "./dist/cjs/prismaExtension.d.ts", + "default": "./dist/cjs/prismaExtension.js" + }, + "import": { + "types": "./dist/prismaExtension.d.ts", + "default": "./dist/prismaExtension.js" + } } }, "files": [ diff --git a/packages/uploads/prisma/package.json b/packages/uploads/prisma/package.json index 56705d7ff21b..9b83f1f2dc6d 100644 --- a/packages/uploads/prisma/package.json +++ b/packages/uploads/prisma/package.json @@ -1,4 +1,4 @@ { "main": "./index.js", - "types": "../dist/cjs/prisma/index.d.ts" + "types": "../dist/cjs/prismaExtension.d.ts" } From 8ddf5d6d0c0821624907505f03cdbb875e617990 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 6 Aug 2024 15:07:15 +0700 Subject: [PATCH 14/91] Try using zx to setup --- packages/uploads/package.json | 4 ++-- packages/uploads/src/prismaExtension.ts | 2 ++ packages/uploads/vitest.config.mts | 3 ++- packages/uploads/vitest.setup.mts | 7 +++++++ 4 files changed, 13 insertions(+), 3 deletions(-) create mode 100644 packages/uploads/vitest.setup.mts diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 395478389cd1..0ff566444471 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -33,8 +33,8 @@ "check:attw": "tsx attw.ts", "check:package": "concurrently npm:check:attw yarn publint", "setup:test": "npx prisma migrate reset -f --schema src/__tests__/unit-test-schema.prisma", - "test": "yarn run setup:test && vitest run", - "test:watch": "yarn run setup:test && vitest watch" + "test": "vitest run", + "test:watch": "vitest watch" }, "dependencies": { "@redwoodjs/project-config": "workspace:*", diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 34d0423d3348..dea174623012 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -44,6 +44,8 @@ type ResultExtends = { [key in ModelNames]?: { withDataUri: { needs: any + // @TODO(TS): this generates unknowns. We dont have access to the Prisma type here + // because it depends on the type it was called from compute: (record: T) => () => Promise } withPublicUrl: { diff --git a/packages/uploads/vitest.config.mts b/packages/uploads/vitest.config.mts index 580e8462c027..cb0188bdb924 100644 --- a/packages/uploads/vitest.config.mts +++ b/packages/uploads/vitest.config.mts @@ -5,7 +5,8 @@ export default defineConfig({ exclude: [...configDefaults.exclude, '**/fixtures'], deps: { interopDefault: false, - } + }, + globalSetup: ['vitest.setup.mts'], }, }) diff --git a/packages/uploads/vitest.setup.mts b/packages/uploads/vitest.setup.mts new file mode 100644 index 000000000000..b673e2cfb2eb --- /dev/null +++ b/packages/uploads/vitest.setup.mts @@ -0,0 +1,7 @@ +import { $ } from 'zx' + +export default async function setup() { + console.log('[setup] Setting up unit test prisma db....') + await $`npx prisma migrate reset -f --skip-seed --schema src/__tests__/unit-test-schema.prisma` + console.log('[setup] Done! \n') +} From a0069db4fa4e9b70980905415ee7afac2b77e679 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 6 Aug 2024 15:58:04 +0700 Subject: [PATCH 15/91] Try cleaning prisma first --- packages/uploads/vitest.setup.mts | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/uploads/vitest.setup.mts b/packages/uploads/vitest.setup.mts index b673e2cfb2eb..01987b744896 100644 --- a/packages/uploads/vitest.setup.mts +++ b/packages/uploads/vitest.setup.mts @@ -2,6 +2,7 @@ import { $ } from 'zx' export default async function setup() { console.log('[setup] Setting up unit test prisma db....') + await $`yarn clean:prisma` await $`npx prisma migrate reset -f --skip-seed --schema src/__tests__/unit-test-schema.prisma` console.log('[setup] Done! \n') } From c2a268e4f678928f4ca6ece9629e9934332fb8cc Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 6 Aug 2024 17:08:14 +0700 Subject: [PATCH 16/91] Remove publicUri extension for now. Add more tests, unhardcode withDataUri --- .../src/__tests__/prismaExtension.test.ts | 131 ++++++++++++++++-- packages/uploads/src/prismaExtension.ts | 42 +++--- 2 files changed, 140 insertions(+), 33 deletions(-) diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index 17555cfd1797..6f1c778d606b 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -1,6 +1,7 @@ import fs from 'node:fs/promises' import { PrismaClient } from '@prisma/client' +import { vol } from 'memfs' import { describe, it, vi, expect } from 'vitest' import { createUploadsExtension } from '../prismaExtension' @@ -10,9 +11,22 @@ import { dataUrlPng } from './fileMocks' vi.mock('node:fs/promises', () => ({ default: { writeFile: vi.fn(), + unlink: vi.fn(), + readFile: vi.fn((path, encoding) => { + if (encoding === 'base64url') { + return 'BASE64::THIS_IS_A_MOCKED_DATA_URL' + } + + return 'MOCKED_FILE_CONTENT' + }), }, })) +vol.fromJSON({ + '/tmp/tus-uploads/123.json': '{}', + '/tmp/tus-uploads/ABCD.json': '{}', +}) + describe('Uploads Prisma Extension', () => { const dummyUploadConfig = { fields: 'uploadField', @@ -25,11 +39,18 @@ describe('Uploads Prisma Extension', () => { savePath: '/dumbo', onFileSaved: vi.fn(), } + + const tusConfig = { + tusUploadDirectory: '/tmp/tus-uploads', + } const prismaClient = new PrismaClient().$extends( - createUploadsExtension({ - dummy: dummyUploadConfig, - dumbo: dumboUploadConfig, - }), + createUploadsExtension( + { + dummy: dummyUploadConfig, + dumbo: dumboUploadConfig, + }, + tusConfig, + ), ) describe('Query extensions', () => { @@ -72,16 +93,91 @@ describe('Uploads Prisma Extension', () => { await expect(emptyUploadFieldPromise).resolves.not.toThrow() }) - // @TODO implement these tests - // it('handles updates, and removes old files', async () => {}) + it('handles updates, and removes old files', async () => { + const dum1 = await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) - // it('handles deletes, and removes files', async () => {}) + const originalPath = dum1.uploadField - // it('supports custom file name functions', async () => {}) + const dum2 = await prismaClient.dummy.update({ + where: { id: dum1.id }, + data: { + uploadField: dataUrlPng, + }, + }) - // it('supports custom save path functions', async () => {}) + expect(dum2.uploadField).not.toEqual(originalPath) + expect(dum2.uploadField).toMatch(/bazinga\/.*\.png/) + expect(fs.unlink).toHaveBeenCalledWith(originalPath) + }) - // it('will move file to new location with TUS uploads', async () => {}) + it('handles deletes, and removes files', async () => { + const dum1 = await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) + + await prismaClient.dummy.delete({ + where: { id: dum1.id }, + }) + + expect(fs.unlink).toHaveBeenCalledWith(dum1.uploadField) + }) + + it('supports custom file name and save path functions', async () => { + const customNameConfig = { + fields: 'firstUpload', + savePath: '/custom', + onFileSaved: vi.fn(), + fileName: (args) => { + // 👇 Using args here + return `my-name-is-dumbo-${args.data.id}` + }, + } + + const clientWithFileName = new PrismaClient().$extends( + createUploadsExtension({ + dumbo: customNameConfig, + }), + ) + + const dumbo = await clientWithFileName.dumbo.create({ + data: { + firstUpload: dataUrlPng, + secondUpload: '', + id: 55, + }, + }) + + expect(customNameConfig.onFileSaved).toHaveBeenCalled() + expect(dumbo.firstUpload).toBe('/custom/my-name-is-dumbo-55.png') + + // Delete it to clean up + await clientWithFileName.dumbo.delete({ + where: { + id: 55, + }, + }) + }) + + // @TODO have to figure out how to mock the + // it('will move file to new location with TUS uploads', async () => { + // const dumbo = await prismaClient.dumbo.create({ + // data: { + // firstUpload: + // 'http://example.com/.redwood/functions/tusUploadEndpoint/123', + // secondUpload: + // 'http://example.com/.redwood/functions/tusUploadEndpoint/ABCD', + // }, + // }) + + // expect(dumbo.firstUpload).toBe('balknsdg') + // expect(dumbo.secondUpload).toBe('balknsdg') + // }) // it('will remove old file when updating with TUS uploads', async () => {}) @@ -89,9 +185,20 @@ describe('Uploads Prisma Extension', () => { }) describe('Result extensions', () => { - it('will return a data URL for the file', async () => {}) + it('will return a data URL for the file', async () => { + const res1 = await ( + await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) + ).withDataUri() // WHY NO TYPES? + + // Mocked in FS mocks + expect(res1.uploadField).toBe('BASE64::THIS_IS_A_MOCKED_DATA_URL') + }) + // @TODO implement - // it('will return a public URL for the file', async () => {}) // it('if file is not found, will throw an error', async () => {}) // it('if saved file is not a path, will throw an error', async () => {}) }) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index dea174623012..4abe870b95ae 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -142,34 +142,30 @@ export const createUploadsExtension = ( // findMany({ query, args, operation }) {} } + // This makes the result extension only available for models with uploadFields + const needs = Object.fromEntries(uploadFields.map((field) => [field, true])) + console.log(`👉 \n ~ needs:`, needs) + resultExtends[modelName] = { withDataUri: { - needs: { avatar: true }, // specify the field name here, so it doesn't appear as a function if avatar isn't requested - compute(contact) { + needs, + compute(modelData) { return async () => { + const base64UploadFields: Record = {} + for await (const field of uploadFields) { + base64UploadFields[field] = await fs.readFile( + modelData[field], + 'base64url', + ) + } + // @TODO: edge cases // 1. If readfile fails - file not found, etc. // 2. If not a path, relative or absolute, throw error - const base64Content = await fs.readFile(contact.avatar, 'base64url') - return { - ...contact, - avatar: base64Content, - } - } - }, - }, - withPublicUrl: { - needs: { avatar: true }, // specify the field name here, so it doesn't appear as a function if avatar isn't requested - compute(contact) { - return async () => { - // @TODO: Test cases - // 1. If the avatar is a base64 string, we should return it as is, but warn - // 2. If absolute path, but not public, throw error - // 3. If relative path, but not public, throw error - const webPublicPath = contact.avatar.replace('web/public', '') + return { - ...contact, - avatar: webPublicPath, + ...modelData, + ...base64UploadFields, } } }, @@ -177,6 +173,8 @@ export const createUploadsExtension = ( } } + console.log('xxxx resultExtends', resultExtends) + return { name: 'redwood-upload-prisma-plugin', query: queryExtends as QueryExtends, @@ -301,6 +299,8 @@ async function saveTusUpload( fileName: string }, ) { + // Get the last part of the TUS upload url + // http://localhost:8910/.redwood/functions/uploadTUS/👉28fa96bf5772338d51👈 const tusId = uploadUrl.split('/').slice(-1).pop() if (!tusId) { From 19c85a4b329fd87fd59f9fc9824ff760bc77b428 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 13:32:45 +0700 Subject: [PATCH 17/91] Get type mapping nearly there! Just need to make sure the type of compute in results extensions is correct --- .../src/__tests__/prismaExtension.test.ts | 2 +- packages/uploads/src/prismaExtension.ts | 77 ++++++++----------- 2 files changed, 35 insertions(+), 44 deletions(-) diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index 6f1c778d606b..09323211781c 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -192,7 +192,7 @@ describe('Uploads Prisma Extension', () => { uploadField: dataUrlPng, }, }) - ).withDataUri() // WHY NO TYPES? + ).withDataUri() // Mocked in FS mocks expect(res1.uploadField).toBe('BASE64::THIS_IS_A_MOCKED_DATA_URL') diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 4abe870b95ae..8f77f0ac7bc7 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -2,6 +2,7 @@ import fs from 'node:fs/promises' import path from 'node:path' import { PrismaClient } from '@prisma/client' +import { Prisma } from '@prisma/client/extension' import type * as runtime from '@prisma/client/runtime/library' import mime from 'mime-types' import { ulid } from 'ulid' @@ -24,41 +25,20 @@ export type UploadConfigForModel = { onFileSaved?: (filePath: string) => void | Promise } -export type UploadsConfig = { - [key in ModelNames]?: UploadConfigForModel -} +export type UploadsConfig = Record< + MName, + UploadConfigForModel +> type TUSServerConfig = { tusUploadDirectory: string } -type QueryExtends = { - [key in ModelNames]?: { - update: any - create: any - delete: any - } -} - -type ResultExtends = { - [key in ModelNames]?: { - withDataUri: { - needs: any - // @TODO(TS): this generates unknowns. We dont have access to the Prisma type here - // because it depends on the type it was called from - compute: (record: T) => () => Promise - } - withPublicUrl: { - needs: any - compute: (record: T) => () => Promise - } - } -} - -type ExtendsType = runtime.ExtensionArgs +// type ExtendsType = runtime.ExtensionArgs +// type ExtendsType = Parameters[0] -export const createUploadsExtension = ( - config: UploadsConfig, +export const createUploadsExtension = ( + config: UploadsConfig, tusConfig?: TUSServerConfig, ) => { // @MARK typing these with ExtendsType['query'] and ExtendsType['result'] @@ -87,15 +67,21 @@ export const createUploadsExtension = ( await fs.unlink(filePath) }) } - // This gives us typesafety when we write the extension, - // but we override it on return, because TS complains on instantiation of the PrismaClient - // its important that the resultsExtends - const queryExtends: ExtendsType['query'] = {} - const resultExtends: ExtendsType['result'] = {} + + const queryExtends: runtime.ExtensionArgs['query'] = {} + + const resultExtends = {} as { + [K in MNames]: { + withDataUri: { + needs: Record + compute: (modelData: T) => () => Promise + } + } + } for (const modelName in config) { // Guaranteed to have modelConfig, we're looping over config 🙄 - const modelConfig = config[modelName as ModelNames] as UploadConfigForModel + const modelConfig = config[modelName] as UploadConfigForModel const uploadFields = Array.isArray(modelConfig.fields) ? modelConfig.fields : [modelConfig.fields] @@ -143,8 +129,9 @@ export const createUploadsExtension = ( } // This makes the result extension only available for models with uploadFields - const needs = Object.fromEntries(uploadFields.map((field) => [field, true])) - console.log(`👉 \n ~ needs:`, needs) + const needs: any = Object.fromEntries( + uploadFields.map((field) => [field, true]), + ) resultExtends[modelName] = { withDataUri: { @@ -152,9 +139,11 @@ export const createUploadsExtension = ( compute(modelData) { return async () => { const base64UploadFields: Record = {} + type ModelField = keyof typeof modelData + for await (const field of uploadFields) { base64UploadFields[field] = await fs.readFile( - modelData[field], + modelData[field as ModelField] as string, 'base64url', ) } @@ -175,11 +164,13 @@ export const createUploadsExtension = ( console.log('xxxx resultExtends', resultExtends) - return { - name: 'redwood-upload-prisma-plugin', - query: queryExtends as QueryExtends, - result: resultExtends as ResultExtends, - } + return Prisma.defineExtension((client) => { + return client.$extends({ + name: 'redwood-upload-prisma-plugin', + query: queryExtends, + result: resultExtends, + }) + }) } /** From 5a884e5dfb21c05a7f53078bb83ec964c8c0690f Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 15:06:17 +0700 Subject: [PATCH 18/91] Hack the result from compute --- packages/uploads/src/prismaExtension.ts | 53 ++++++++++++------------- 1 file changed, 26 insertions(+), 27 deletions(-) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 8f77f0ac7bc7..94c5ca103dcb 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -74,7 +74,12 @@ export const createUploadsExtension = ( [K in MNames]: { withDataUri: { needs: Record - compute: (modelData: T) => () => Promise + compute: ( + // @MARK: this is a hack + // There has to be a better way to type this... because if you used a select or omit + // it would be a different type + modelData: ReturnType, + ) => () => ReturnType } } } @@ -129,45 +134,39 @@ export const createUploadsExtension = ( } // This makes the result extension only available for models with uploadFields - const needs: any = Object.fromEntries( - uploadFields.map((field) => [field, true]), - ) + const needs = Object.fromEntries(uploadFields.map((field) => [field, true])) resultExtends[modelName] = { withDataUri: { needs, - compute(modelData) { - return async () => { - const base64UploadFields: Record = {} - type ModelField = keyof typeof modelData - - for await (const field of uploadFields) { - base64UploadFields[field] = await fs.readFile( - modelData[field as ModelField] as string, - 'base64url', - ) - } - - // @TODO: edge cases - // 1. If readfile fails - file not found, etc. - // 2. If not a path, relative or absolute, throw error - - return { - ...modelData, - ...base64UploadFields, - } + async compute(modelData) { + const base64UploadFields: Record = {} + type ModelField = keyof typeof modelData + + for await (const field of uploadFields) { + base64UploadFields[field] = await fs.readFile( + modelData[field as ModelField] as string, + 'base64url', + ) + } + + // @TODO: edge cases + // 1. If readfile fails - file not found, etc. + // 2. If not a path, relative or absolute, throw error + + return { + ...modelData, + ...base64UploadFields, } }, }, } } - console.log('xxxx resultExtends', resultExtends) - return Prisma.defineExtension((client) => { return client.$extends({ name: 'redwood-upload-prisma-plugin', - query: queryExtends, + // query: queryExtends, result: resultExtends, }) }) From a7d06f4cbe8f7dbcc5f4b8f84bcfa9c56e79612c Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 15:33:49 +0700 Subject: [PATCH 19/91] Update comments --- packages/uploads/src/prismaExtension.ts | 9 ++------- 1 file changed, 2 insertions(+), 7 deletions(-) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 94c5ca103dcb..a3415890fe73 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -33,17 +33,12 @@ export type UploadsConfig = Record< type TUSServerConfig = { tusUploadDirectory: string } - -// type ExtendsType = runtime.ExtensionArgs -// type ExtendsType = Parameters[0] - export const createUploadsExtension = ( config: UploadsConfig, tusConfig?: TUSServerConfig, ) => { - // @MARK typing these with ExtendsType['query'] and ExtendsType['result'] - // will create an error when we instiate the PrismaClient 🤷 - // but without these types prisma won''t show types for the new methods + // @TODO I think we can use Prisma.getExtensionContext(this) + // instead of creating a new PrismaClient instance const prismaInstance = new PrismaClient() async function deleteUploadsFromDiskForArgs({ From 6974d456cb7b19e7c0cf5bfda3b0cf85a13ffcbd Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 15:35:44 +0700 Subject: [PATCH 20/91] Bump prisma client to match main --- packages/uploads/package.json | 2 +- yarn.lock | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 0ff566444471..3a1065e9f0c0 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -44,7 +44,7 @@ }, "devDependencies": { "@arethetypeswrong/cli": "0.15.3", - "@prisma/client": "5.17.0", + "@prisma/client": "5.18.0", "@redwoodjs/framework-tools": "workspace:*", "@types/fs-extra": "11.0.4", "@types/mime-types": "2.1.4", diff --git a/yarn.lock b/yarn.lock index 324f1d04eee3..93d40020a6e7 100644 --- a/yarn.lock +++ b/yarn.lock @@ -8675,7 +8675,7 @@ __metadata: resolution: "@redwoodjs/uploads@workspace:packages/uploads" dependencies: "@arethetypeswrong/cli": "npm:0.15.3" - "@prisma/client": "npm:5.17.0" + "@prisma/client": "npm:5.18.0" "@redwoodjs/framework-tools": "workspace:*" "@redwoodjs/project-config": "workspace:*" "@types/fs-extra": "npm:11.0.4" From f5c696a8007212a10b628f6953c4d8589538a402 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 16:09:33 +0700 Subject: [PATCH 21/91] Fix extension types without prisma generate --- packages/uploads/src/prismaExtension.ts | 42 +++++++++++++------------ 1 file changed, 22 insertions(+), 20 deletions(-) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index a3415890fe73..fd9a4673da70 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -25,7 +25,7 @@ export type UploadConfigForModel = { onFileSaved?: (filePath: string) => void | Promise } -export type UploadsConfig = Record< +export type UploadsConfig = Record< MName, UploadConfigForModel > @@ -81,7 +81,7 @@ export const createUploadsExtension = ( for (const modelName in config) { // Guaranteed to have modelConfig, we're looping over config 🙄 - const modelConfig = config[modelName] as UploadConfigForModel + const modelConfig = config[modelName as MNames] as UploadConfigForModel const uploadFields = Array.isArray(modelConfig.fields) ? modelConfig.fields : [modelConfig.fields] @@ -134,24 +134,26 @@ export const createUploadsExtension = ( resultExtends[modelName] = { withDataUri: { needs, - async compute(modelData) { - const base64UploadFields: Record = {} - type ModelField = keyof typeof modelData - - for await (const field of uploadFields) { - base64UploadFields[field] = await fs.readFile( - modelData[field as ModelField] as string, - 'base64url', - ) - } - - // @TODO: edge cases - // 1. If readfile fails - file not found, etc. - // 2. If not a path, relative or absolute, throw error - - return { - ...modelData, - ...base64UploadFields, + compute(modelData) { + return async () => { + const base64UploadFields: Record = {} + type ModelField = keyof typeof modelData + + for await (const field of uploadFields) { + base64UploadFields[field] = await fs.readFile( + modelData[field as ModelField] as string, + 'base64url', + ) + } + + // @TODO: edge cases + // 1. If readfile fails - file not found, etc. + // 2. If not a path, relative or absolute, throw error + + return { + ...modelData, + ...base64UploadFields, + } } }, }, From fd28dd282b4af5f1e711936e0aa8529134eb5c3c Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 16:57:36 +0700 Subject: [PATCH 22/91] Return to generics --- packages/uploads/src/prismaExtension.ts | 13 ++++++------- 1 file changed, 6 insertions(+), 7 deletions(-) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index fd9a4673da70..531377b48f5a 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -69,12 +69,11 @@ export const createUploadsExtension = ( [K in MNames]: { withDataUri: { needs: Record - compute: ( - // @MARK: this is a hack - // There has to be a better way to type this... because if you used a select or omit - // it would be a different type - modelData: ReturnType, - ) => () => ReturnType + compute: ( + // @MARK: this generic doesn't get picked up by prisma + // the returned type ends up being unknown + modelData: T, + ) => () => Promise } } } @@ -163,7 +162,7 @@ export const createUploadsExtension = ( return Prisma.defineExtension((client) => { return client.$extends({ name: 'redwood-upload-prisma-plugin', - // query: queryExtends, + query: queryExtends, result: resultExtends, }) }) From 1a428402dae872d93451fbd028a5ec92e6f570d7 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 17:49:47 +0700 Subject: [PATCH 23/91] Remove prisma test from CI, fix withDataUri --- packages/uploads/package.json | 3 +- .../src/__tests__/getFileExtension.test.ts | 8 +- .../__tests__/prismaExtension.local.test.ts | 226 ++++++++++++++++++ .../src/__tests__/prismaExtension.test.ts | 205 ---------------- packages/uploads/src/fileSave.utils.ts | 135 +++++++++++ packages/uploads/src/prismaExtension.ts | 128 +--------- packages/uploads/vitest.config.mts | 1 - packages/uploads/vitest.setup.mts | 8 - 8 files changed, 373 insertions(+), 341 deletions(-) create mode 100644 packages/uploads/src/__tests__/prismaExtension.local.test.ts delete mode 100644 packages/uploads/src/__tests__/prismaExtension.test.ts create mode 100644 packages/uploads/src/fileSave.utils.ts delete mode 100644 packages/uploads/vitest.setup.mts diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 3a1065e9f0c0..1bb59627bb80 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -34,7 +34,8 @@ "check:package": "concurrently npm:check:attw yarn publint", "setup:test": "npx prisma migrate reset -f --schema src/__tests__/unit-test-schema.prisma", "test": "vitest run", - "test:watch": "vitest watch" + "test:watch": "vitest watch", + "test:prismaExtension": "PRISMA_EXTENSION_TESTS=true vitest run" }, "dependencies": { "@redwoodjs/project-config": "workspace:*", diff --git a/packages/uploads/src/__tests__/getFileExtension.test.ts b/packages/uploads/src/__tests__/getFileExtension.test.ts index 85629d8b4429..5f72570bc33b 100644 --- a/packages/uploads/src/__tests__/getFileExtension.test.ts +++ b/packages/uploads/src/__tests__/getFileExtension.test.ts @@ -1,21 +1,21 @@ import { describe, it, expect } from 'vitest' -import { getFileExtension } from '../prismaExtension' +import { getFileExtensionFromDataUri } from '../fileSave.utils' describe('getFileExtension', () => { it('should return the correct file extension for a given data type', () => { const dataType = 'data:image/png;base64' - const extension = getFileExtension(dataType) + const extension = getFileExtensionFromDataUri(dataType) expect(extension).toBe('png') }) it('handles svgs', () => { const dataType = 'data:image/svg+xml;base64' - expect(getFileExtension(dataType)).toBe('svg') + expect(getFileExtensionFromDataUri(dataType)).toBe('svg') }) it('handles gif', () => { const dataType = 'data:image/gif;base64' - expect(getFileExtension(dataType)).toBe('gif') + expect(getFileExtensionFromDataUri(dataType)).toBe('gif') }) }) diff --git a/packages/uploads/src/__tests__/prismaExtension.local.test.ts b/packages/uploads/src/__tests__/prismaExtension.local.test.ts new file mode 100644 index 000000000000..ddca4bd28518 --- /dev/null +++ b/packages/uploads/src/__tests__/prismaExtension.local.test.ts @@ -0,0 +1,226 @@ +import fs from 'node:fs/promises' + +import { PrismaClient } from '@prisma/client' +import { vol } from 'memfs' +import { describe, it, vi, expect, beforeAll } from 'vitest' +import { $ } from 'zx' + +import { createUploadsExtension } from '../prismaExtension' + +import { dataUrlPng } from './fileMocks' + +/*** + * NOTE: this test does not run in CI, because it requires a local prisma db + * which causes build failures elsewhere. It's still useful to have locally when adding/changing features though + * + * To run it use the script `yarn test:prismaExtension` + */ + +const shouldRunPrismaExtensionTests = + process.env.PRISMA_EXTENSION_TESTS === 'true' + +vi.mock('node:fs/promises', () => ({ + default: { + writeFile: vi.fn(), + unlink: vi.fn(), + readFile: vi.fn((path, encoding) => { + if (encoding === 'base64url') { + return 'BASE64::THIS_IS_A_MOCKED_DATA_URL' + } + + return 'MOCKED_FILE_CONTENT' + }), + }, +})) + +vol.fromJSON({ + '/tmp/tus-uploads/123.json': '{}', + '/tmp/tus-uploads/ABCD.json': '{}', +}) + +describe.runIf(shouldRunPrismaExtensionTests)( + 'Uploads Prisma Extension', + () => { + beforeAll(async () => { + console.log('[setup] Setting up unit test prisma db....') + await $`yarn clean:prisma` + await $`npx prisma migrate reset -f --skip-seed --schema src/__tests__/unit-test-schema.prisma` + console.log('[setup] Done! \n') + }) + + const dummyUploadConfig = { + fields: 'uploadField', + savePath: '/bazinga', + onFileSaved: vi.fn(), + } + + const dumboUploadConfig = { + fields: ['firstUpload', 'secondUpload'], + savePath: '/dumbo', + onFileSaved: vi.fn(), + } + + const tusConfig = { + tusUploadDirectory: '/tmp/tus-uploads', + } + const prismaClient = new PrismaClient().$extends( + createUploadsExtension( + { + dummy: dummyUploadConfig, + dumbo: dumboUploadConfig, + }, + tusConfig, + ), + ) + + describe('Query extensions', () => { + it('will create a file with base64 encoded png', async () => { + const dum1 = await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) + + expect(dummyUploadConfig.onFileSaved).toHaveBeenCalled() + expect(dum1.uploadField).toMatch(/bazinga\/.*\.png/) + expect(fs.writeFile).toHaveBeenCalledWith( + expect.stringMatching(/bazinga\/.*\.png/), + expect.anything(), // no need to check content here, makes test slow + ) + }) + + it('handles multiple upload fields', async () => { + const dumbo = await prismaClient.dumbo.create({ + data: { + firstUpload: dataUrlPng, + secondUpload: dataUrlPng, + }, + }) + + expect(dumbo.firstUpload).toMatch(/dumbo\/.*\.png/) + expect(dumbo.secondUpload).toMatch(/dumbo\/.*\.png/) + + expect(dumboUploadConfig.onFileSaved).toHaveBeenCalledTimes(2) + }) + + it('handles empty fields', async () => { + const emptyUploadFieldPromise = prismaClient.dummy.create({ + data: { + uploadField: '', + }, + }) + + await expect(emptyUploadFieldPromise).resolves.not.toThrow() + }) + + it('handles updates, and removes old files', async () => { + const dum1 = await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) + + const originalPath = dum1.uploadField + + const dum2 = await prismaClient.dummy.update({ + where: { id: dum1.id }, + data: { + uploadField: dataUrlPng, + }, + }) + + expect(dum2.uploadField).not.toEqual(originalPath) + expect(dum2.uploadField).toMatch(/bazinga\/.*\.png/) + expect(fs.unlink).toHaveBeenCalledWith(originalPath) + }) + + it('handles deletes, and removes files', async () => { + const dum1 = await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) + + await prismaClient.dummy.delete({ + where: { id: dum1.id }, + }) + + expect(fs.unlink).toHaveBeenCalledWith(dum1.uploadField) + }) + + it('supports custom file name and save path functions', async () => { + const customNameConfig = { + fields: 'firstUpload', + savePath: '/custom', + onFileSaved: vi.fn(), + fileName: (args) => { + // 👇 Using args here + return `my-name-is-dumbo-${args.data.id}` + }, + } + + const clientWithFileName = new PrismaClient().$extends( + createUploadsExtension({ + dumbo: customNameConfig, + }), + ) + + const dumbo = await clientWithFileName.dumbo.create({ + data: { + firstUpload: dataUrlPng, + secondUpload: '', + id: 55, + }, + }) + + expect(customNameConfig.onFileSaved).toHaveBeenCalled() + expect(dumbo.firstUpload).toBe('/custom/my-name-is-dumbo-55.png') + + // Delete it to clean up + await clientWithFileName.dumbo.delete({ + where: { + id: 55, + }, + }) + }) + + // @TODO have to figure out how to mock the + // it('will move file to new location with TUS uploads', async () => { + // const dumbo = await prismaClient.dumbo.create({ + // data: { + // firstUpload: + // 'http://example.com/.redwood/functions/tusUploadEndpoint/123', + // secondUpload: + // 'http://example.com/.redwood/functions/tusUploadEndpoint/ABCD', + // }, + // }) + + // expect(dumbo.firstUpload).toBe('balknsdg') + // expect(dumbo.secondUpload).toBe('balknsdg') + // }) + + // it('will remove old file when updating with TUS uploads', async () => {}) + + // it('will remove file when deleting with TUS uploads', async () => {}) + }) + + describe('Result extensions', () => { + it('will return a data URL for the file', async () => { + const res1 = await ( + await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) + ).withDataUri() + + // Mocked in FS mocks + expect(res1.uploadField).toBe('BASE64::THIS_IS_A_MOCKED_DATA_URL') + }) + + // @TODO implement + // it('if file is not found, will throw an error', async () => {}) + // it('if saved file is not a path, will throw an error', async () => {}) + }) + }, +) diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts deleted file mode 100644 index 09323211781c..000000000000 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ /dev/null @@ -1,205 +0,0 @@ -import fs from 'node:fs/promises' - -import { PrismaClient } from '@prisma/client' -import { vol } from 'memfs' -import { describe, it, vi, expect } from 'vitest' - -import { createUploadsExtension } from '../prismaExtension' - -import { dataUrlPng } from './fileMocks' - -vi.mock('node:fs/promises', () => ({ - default: { - writeFile: vi.fn(), - unlink: vi.fn(), - readFile: vi.fn((path, encoding) => { - if (encoding === 'base64url') { - return 'BASE64::THIS_IS_A_MOCKED_DATA_URL' - } - - return 'MOCKED_FILE_CONTENT' - }), - }, -})) - -vol.fromJSON({ - '/tmp/tus-uploads/123.json': '{}', - '/tmp/tus-uploads/ABCD.json': '{}', -}) - -describe('Uploads Prisma Extension', () => { - const dummyUploadConfig = { - fields: 'uploadField', - savePath: '/bazinga', - onFileSaved: vi.fn(), - } - - const dumboUploadConfig = { - fields: ['firstUpload', 'secondUpload'], - savePath: '/dumbo', - onFileSaved: vi.fn(), - } - - const tusConfig = { - tusUploadDirectory: '/tmp/tus-uploads', - } - const prismaClient = new PrismaClient().$extends( - createUploadsExtension( - { - dummy: dummyUploadConfig, - dumbo: dumboUploadConfig, - }, - tusConfig, - ), - ) - - describe('Query extensions', () => { - it('will create a file with base64 encoded png', async () => { - const dum1 = await prismaClient.dummy.create({ - data: { - uploadField: dataUrlPng, - }, - }) - - expect(dummyUploadConfig.onFileSaved).toHaveBeenCalled() - expect(dum1.uploadField).toMatch(/bazinga\/.*\.png/) - expect(fs.writeFile).toHaveBeenCalledWith( - expect.stringMatching(/bazinga\/.*\.png/), - expect.anything(), // no need to check content here, makes test slow - ) - }) - - it('handles multiple upload fields', async () => { - const dumbo = await prismaClient.dumbo.create({ - data: { - firstUpload: dataUrlPng, - secondUpload: dataUrlPng, - }, - }) - - expect(dumbo.firstUpload).toMatch(/dumbo\/.*\.png/) - expect(dumbo.secondUpload).toMatch(/dumbo\/.*\.png/) - - expect(dumboUploadConfig.onFileSaved).toHaveBeenCalledTimes(2) - }) - - it('handles empty fields', async () => { - const emptyUploadFieldPromise = prismaClient.dummy.create({ - data: { - uploadField: '', - }, - }) - - await expect(emptyUploadFieldPromise).resolves.not.toThrow() - }) - - it('handles updates, and removes old files', async () => { - const dum1 = await prismaClient.dummy.create({ - data: { - uploadField: dataUrlPng, - }, - }) - - const originalPath = dum1.uploadField - - const dum2 = await prismaClient.dummy.update({ - where: { id: dum1.id }, - data: { - uploadField: dataUrlPng, - }, - }) - - expect(dum2.uploadField).not.toEqual(originalPath) - expect(dum2.uploadField).toMatch(/bazinga\/.*\.png/) - expect(fs.unlink).toHaveBeenCalledWith(originalPath) - }) - - it('handles deletes, and removes files', async () => { - const dum1 = await prismaClient.dummy.create({ - data: { - uploadField: dataUrlPng, - }, - }) - - await prismaClient.dummy.delete({ - where: { id: dum1.id }, - }) - - expect(fs.unlink).toHaveBeenCalledWith(dum1.uploadField) - }) - - it('supports custom file name and save path functions', async () => { - const customNameConfig = { - fields: 'firstUpload', - savePath: '/custom', - onFileSaved: vi.fn(), - fileName: (args) => { - // 👇 Using args here - return `my-name-is-dumbo-${args.data.id}` - }, - } - - const clientWithFileName = new PrismaClient().$extends( - createUploadsExtension({ - dumbo: customNameConfig, - }), - ) - - const dumbo = await clientWithFileName.dumbo.create({ - data: { - firstUpload: dataUrlPng, - secondUpload: '', - id: 55, - }, - }) - - expect(customNameConfig.onFileSaved).toHaveBeenCalled() - expect(dumbo.firstUpload).toBe('/custom/my-name-is-dumbo-55.png') - - // Delete it to clean up - await clientWithFileName.dumbo.delete({ - where: { - id: 55, - }, - }) - }) - - // @TODO have to figure out how to mock the - // it('will move file to new location with TUS uploads', async () => { - // const dumbo = await prismaClient.dumbo.create({ - // data: { - // firstUpload: - // 'http://example.com/.redwood/functions/tusUploadEndpoint/123', - // secondUpload: - // 'http://example.com/.redwood/functions/tusUploadEndpoint/ABCD', - // }, - // }) - - // expect(dumbo.firstUpload).toBe('balknsdg') - // expect(dumbo.secondUpload).toBe('balknsdg') - // }) - - // it('will remove old file when updating with TUS uploads', async () => {}) - - // it('will remove file when deleting with TUS uploads', async () => {}) - }) - - describe('Result extensions', () => { - it('will return a data URL for the file', async () => { - const res1 = await ( - await prismaClient.dummy.create({ - data: { - uploadField: dataUrlPng, - }, - }) - ).withDataUri() - - // Mocked in FS mocks - expect(res1.uploadField).toBe('BASE64::THIS_IS_A_MOCKED_DATA_URL') - }) - - // @TODO implement - // it('if file is not found, will throw an error', async () => {}) - // it('if saved file is not a path, will throw an error', async () => {}) - }) -}) diff --git a/packages/uploads/src/fileSave.utils.ts b/packages/uploads/src/fileSave.utils.ts new file mode 100644 index 000000000000..6d6eaa94b0ae --- /dev/null +++ b/packages/uploads/src/fileSave.utils.ts @@ -0,0 +1,135 @@ +import fs from 'node:fs/promises' +import path from 'node:path' + +import mime from 'mime-types' + +import { getPaths } from '@redwoodjs/project-config' + +export type TUSServerConfig = { + tusUploadDirectory: string +} + +/** + * This function takes an upload field, determines whether its TUS or Base64, and saves it to the file system. + */ +export async function saveUploadToFile( + uploadUrlOrDataUrl: string, + { fileName, saveDir }: { saveDir: string; fileName: string }, + tusConfig?: TUSServerConfig, +) { + let outputPath: string | null = null + + if (isBase64DataUri(uploadUrlOrDataUrl)) { + outputPath = await saveBase64DataUriToFile(uploadUrlOrDataUrl, { + saveDir, + fileName, + }) + } else if (uploadUrlOrDataUrl.startsWith('http')) { + if (!tusConfig) { + throw new Error('TusConfig not supplied.') + } + + outputPath = await saveTusUpload(uploadUrlOrDataUrl, { + tusConfig, + saveDir, + fileName, + }) + } // @TODO: add support for form uploads? + + if (!outputPath) { + throw new Error('Unsupported upload URL') + } + + // @MARK: we can create a new record on the uploads table here + + return outputPath +} + +// @MARK: if we block the TUS GET, we don't really need to move it +// We send the TUS upload URL as the value of the field +export async function saveTusUpload( + uploadUrl: string, + { + tusConfig, + saveDir, + fileName, + }: { + tusConfig: TUSServerConfig + saveDir: string + fileName: string + }, +) { + // Get the last part of the TUS upload url + // http://localhost:8910/.redwood/functions/uploadTUS/👉28fa96bf5772338d51👈 + const tusId = uploadUrl.split('/').slice(-1).pop() + + if (!tusId) { + throw new Error('Could not extract upload ID from URL') + } + + if (!tusConfig.tusUploadDirectory) { + throw new Error( + 'You have to configure the TUS Upload Directory in the prisma extension. It is required for TUS uploads', + ) + } + + // Optional Step.... + const metaFile = path.join( + path.isAbsolute(tusConfig.tusUploadDirectory) + ? tusConfig.tusUploadDirectory + : // @MARK: if the directory supplied isn't relative + path.join(getPaths().base, tusConfig.tusUploadDirectory), + `${tusId}.json`, + ) + // Can't await import, because JSON file. + const tusMeta = require(metaFile) + + const fileExtension = tusMeta.metadata.filetype.split('/')[1] + + const savedFilePath = path.join(saveDir, `${fileName}.${fileExtension}`) + + // @MARK: we can also move... + await fs.copyFile( + path.join(tusConfig.tusUploadDirectory, tusId), + savedFilePath, + ) + + return savedFilePath +} + +function isBase64DataUri(uploadUrlOrDataUrl: string) { + // Check if the uploadUrlOrDataUrl is a valid base64 string + const base64Regex = /^data:(.*?);base64,/ + return base64Regex.test(uploadUrlOrDataUrl) +} + +async function saveBase64DataUriToFile( + dataUrlString: string, + { saveDir, fileName }: { saveDir: string; fileName: string }, +) { + const [dataType, fileContent] = dataUrlString.split(',') + // format is data:image/png;base64,.... + const fileExtension = getFileExtensionFromDataUri(dataType) + const filePath = path.join(saveDir, `${fileName}.${fileExtension}`) + + await fs.writeFile(filePath, Buffer.from(fileContent, 'base64')) + + return filePath +} + +export function getFileExtensionFromDataUri(dataType: string): string { + const mimeType = dataType.split(':')[1].split(';')[0] + const extension = mime.extension(mimeType) + if (!extension) { + throw new Error(`Unsupported file type: ${mimeType}`) + } + return extension +} + +export async function fileToDataUri(filePath: string) { + const base64Data = await fs.readFile(filePath, 'base64') + const ext = path.extname(filePath) + const mimeType = mime.lookup(ext) + + return `data:${mimeType};base64,${base64Data}` +} diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 531377b48f5a..1a584b3f1ab4 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -1,13 +1,15 @@ import fs from 'node:fs/promises' -import path from 'node:path' import { PrismaClient } from '@prisma/client' import { Prisma } from '@prisma/client/extension' import type * as runtime from '@prisma/client/runtime/library' -import mime from 'mime-types' import { ulid } from 'ulid' -import { getPaths } from '@redwoodjs/project-config' +import { + fileToDataUri, + saveUploadToFile, + type TUSServerConfig, +} from './fileSave.utils.js' type FilterOutDollarPrefixed = T extends `$${string}` ? never @@ -30,9 +32,6 @@ export type UploadsConfig = Record< UploadConfigForModel > -type TUSServerConfig = { - tusUploadDirectory: string -} export const createUploadsExtension = ( config: UploadsConfig, tusConfig?: TUSServerConfig, @@ -139,9 +138,8 @@ export const createUploadsExtension = ( type ModelField = keyof typeof modelData for await (const field of uploadFields) { - base64UploadFields[field] = await fs.readFile( + base64UploadFields[field] = await fileToDataUri( modelData[field as ModelField] as string, - 'base64url', ) } @@ -237,117 +235,3 @@ async function saveUploads( data: newData, } } - -async function saveUploadToFile( - uploadUrlOrDataUrl: string, - { fileName, saveDir }: { saveDir: string; fileName: string }, - tusConfig?: TUSServerConfig, -) { - let outputPath: string | null = null - - if (isBase65(uploadUrlOrDataUrl)) { - outputPath = await saveBase65File(uploadUrlOrDataUrl, { - saveDir, - fileName, - }) - } else if (uploadUrlOrDataUrl.startsWith('http')) { - if (!tusConfig) { - throw new Error('TusConfig not supplied.') - } - - outputPath = await saveTusUpload(uploadUrlOrDataUrl, { - tusConfig, - saveDir, - fileName, - }) - } // @TODO: add support for form uploads? - - if (!outputPath) { - throw new Error('Unsupported upload URL') - } - - // @MARK: we can create a new record on the uploads table here - - return outputPath -} - -// @MARK: if we block the TUS GET, we don't really need to move it -// We send the TUS upload URL as the value of the field -async function saveTusUpload( - uploadUrl: string, - { - tusConfig, - saveDir, - fileName, - }: { - tusConfig: TUSServerConfig - saveDir: string - fileName: string - }, -) { - // Get the last part of the TUS upload url - // http://localhost:8910/.redwood/functions/uploadTUS/👉28fa96bf5772338d51👈 - const tusId = uploadUrl.split('/').slice(-1).pop() - - if (!tusId) { - throw new Error('Could not extract upload ID from URL') - } - - if (!tusConfig.tusUploadDirectory) { - throw new Error( - 'You have to configure the TUS Upload Directory in the prisma extension. It is required for TUS uploads', - ) - } - - // Optional Step.... - const metaFile = path.join( - path.isAbsolute(tusConfig.tusUploadDirectory) - ? tusConfig.tusUploadDirectory - : // @MARK: if the directory supplied isn't relative - path.join(getPaths().base, tusConfig.tusUploadDirectory), - `${tusId}.json`, - ) - // Can't await import, because JSON file. - const tusMeta = require(metaFile) - - const fileExtension = tusMeta.metadata.filetype.split('/')[1] - - const savedFilePath = path.join(saveDir, `${fileName}.${fileExtension}`) - - // @MARK: we can also move... - await fs.copyFile( - path.join(tusConfig.tusUploadDirectory, tusId), - savedFilePath, - ) - - return savedFilePath -} - -function isBase65(uploadUrlOrDataUrl: string) { - // Check if the uploadUrlOrDataUrl is a valid base64 string - const base64Regex = /^data:(.*?);base64,/ - return base64Regex.test(uploadUrlOrDataUrl) -} - -async function saveBase65File( - dataUrlString: string, - { saveDir, fileName }: { saveDir: string; fileName: string }, -) { - const [dataType, fileContent] = dataUrlString.split(',') - // format is data:image/png;base64,.... - const fileExtension = getFileExtension(dataType) - const filePath = path.join(saveDir, `${fileName}.${fileExtension}`) - - await fs.writeFile(filePath, Buffer.from(fileContent, 'base64')) - - return filePath -} - -export function getFileExtension(dataType: string): string { - const mimeType = dataType.split(':')[1].split(';')[0] - const extension = mime.extension(mimeType) - if (!extension) { - throw new Error(`Unsupported file type: ${mimeType}`) - } - return extension -} diff --git a/packages/uploads/vitest.config.mts b/packages/uploads/vitest.config.mts index cb0188bdb924..984991bff461 100644 --- a/packages/uploads/vitest.config.mts +++ b/packages/uploads/vitest.config.mts @@ -6,7 +6,6 @@ export default defineConfig({ deps: { interopDefault: false, }, - globalSetup: ['vitest.setup.mts'], }, }) diff --git a/packages/uploads/vitest.setup.mts b/packages/uploads/vitest.setup.mts deleted file mode 100644 index 01987b744896..000000000000 --- a/packages/uploads/vitest.setup.mts +++ /dev/null @@ -1,8 +0,0 @@ -import { $ } from 'zx' - -export default async function setup() { - console.log('[setup] Setting up unit test prisma db....') - await $`yarn clean:prisma` - await $`npx prisma migrate reset -f --skip-seed --schema src/__tests__/unit-test-schema.prisma` - console.log('[setup] Done! \n') -} From 6dcf81fe6966de5e0779384c91ae3fd699852b13 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 18:57:46 +0700 Subject: [PATCH 24/91] Cleanup tests --- .../uploads/src/__tests__/prismaExtension.local.test.ts | 9 +++++---- packages/uploads/src/fileSave.utils.ts | 2 +- 2 files changed, 6 insertions(+), 5 deletions(-) diff --git a/packages/uploads/src/__tests__/prismaExtension.local.test.ts b/packages/uploads/src/__tests__/prismaExtension.local.test.ts index ddca4bd28518..ce6cc4dc9287 100644 --- a/packages/uploads/src/__tests__/prismaExtension.local.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.local.test.ts @@ -24,8 +24,8 @@ vi.mock('node:fs/promises', () => ({ writeFile: vi.fn(), unlink: vi.fn(), readFile: vi.fn((path, encoding) => { - if (encoding === 'base64url') { - return 'BASE64::THIS_IS_A_MOCKED_DATA_URL' + if (encoding === 'base64') { + return 'BASE64_FILE_CONTENT' } return 'MOCKED_FILE_CONTENT' @@ -43,7 +43,6 @@ describe.runIf(shouldRunPrismaExtensionTests)( () => { beforeAll(async () => { console.log('[setup] Setting up unit test prisma db....') - await $`yarn clean:prisma` await $`npx prisma migrate reset -f --skip-seed --schema src/__tests__/unit-test-schema.prisma` console.log('[setup] Done! \n') }) @@ -215,7 +214,9 @@ describe.runIf(shouldRunPrismaExtensionTests)( ).withDataUri() // Mocked in FS mocks - expect(res1.uploadField).toBe('BASE64::THIS_IS_A_MOCKED_DATA_URL') + expect(res1.uploadField).toBe( + '_FILE_CONTENT', + ) }) // @TODO implement diff --git a/packages/uploads/src/fileSave.utils.ts b/packages/uploads/src/fileSave.utils.ts index 6d6eaa94b0ae..e7d23bd7f548 100644 --- a/packages/uploads/src/fileSave.utils.ts +++ b/packages/uploads/src/fileSave.utils.ts @@ -34,7 +34,7 @@ export async function saveUploadToFile( saveDir, fileName, }) - } // @TODO: add support for form uploads? + } if (!outputPath) { throw new Error('Unsupported upload URL') From 60828f3985622e4df5283a14ac23ec1ffd236295 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 20:38:36 +0700 Subject: [PATCH 25/91] Try generating a local prisma client for tests --- packages/uploads/.gitignore | 1 + .../src/__tests__/fileSave.units.test.ts | 60 +++++ .../src/__tests__/getFileExtension.test.ts | 21 -- .../__tests__/prismaExtension.local.test.ts | 227 ------------------ .../src/__tests__/prismaExtension.test.ts | 219 +++++++++++++++++ .../src/__tests__/unit-test-schema.prisma | 1 + packages/uploads/src/fileSave.utils.ts | 4 +- packages/uploads/src/prismaExtension.ts | 2 +- packages/uploads/vitest.config.mts | 10 + packages/uploads/vitest.setup.mts | 8 + 10 files changed, 303 insertions(+), 250 deletions(-) create mode 100644 packages/uploads/src/__tests__/fileSave.units.test.ts delete mode 100644 packages/uploads/src/__tests__/getFileExtension.test.ts delete mode 100644 packages/uploads/src/__tests__/prismaExtension.local.test.ts create mode 100644 packages/uploads/src/__tests__/prismaExtension.test.ts create mode 100644 packages/uploads/vitest.setup.mts diff --git a/packages/uploads/.gitignore b/packages/uploads/.gitignore index e5c46c18f78e..9b5d07c455ec 100644 --- a/packages/uploads/.gitignore +++ b/packages/uploads/.gitignore @@ -1,3 +1,4 @@ src/__tests__/migrations/* src/__tests__/for_unit_test.db* .attw.json +src/__tests__/prisma-client/* diff --git a/packages/uploads/src/__tests__/fileSave.units.test.ts b/packages/uploads/src/__tests__/fileSave.units.test.ts new file mode 100644 index 000000000000..5907ea216522 --- /dev/null +++ b/packages/uploads/src/__tests__/fileSave.units.test.ts @@ -0,0 +1,60 @@ +import { describe, it, expect, vi } from 'vitest' + +import { + getFileExtensionFromDataUri, + saveUploadToFile, +} from '../fileSave.utils' + +describe('getFileExtension', () => { + it('should return the correct file extension for a given data type', () => { + const dataType = 'data:image/png;base64' + const extension = getFileExtensionFromDataUri(dataType) + expect(extension).toBe('png') + }) + + it('handles svgs', () => { + const dataType = 'data:image/svg+xml;base64' + expect(getFileExtensionFromDataUri(dataType)).toBe('svg') + }) + + it('handles gif', () => { + const dataType = 'data:image/gif;base64' + expect(getFileExtensionFromDataUri(dataType)).toBe('gif') + }) +}) + +describe('saveUploadToFile', () => { + vi.mock('node:fs/promises', () => ({ + default: { + writeFile: vi.fn(), + unlink: vi.fn(), + // readFile: vi.fn((path, encoding) => { + // if (encoding === 'base64') { + // return 'BASE64_FILE_CONTENT' + // } + + // return 'MOCKED_FILE_CONTENT' + // }), + }, + })) + + // it('Should call saveBase64DataUriToFile if the uploadUrlOrDataUrl is a base64 data uri', async () => { + // saveUploadToFile('data:image/png;base64,....', { + // saveDir: 'uploads', + // fileName: 'test', + // }) + // }) + + // it('Should call saveTusFileToFile if the uploadUrlOrDataUrl is a tus url', async () => {}) + + it('Should throw an error if the uploadUrlOrDataUrl is not a base64 data uri or a tus url', async () => { + try { + await saveUploadToFile('/random/path/to/whatever.png', { + saveDir: 'uploads', + fileName: 'test', + }) + } catch (e) { + expect(e.message).toBe('Unsupported upload format') + } + }) +}) diff --git a/packages/uploads/src/__tests__/getFileExtension.test.ts b/packages/uploads/src/__tests__/getFileExtension.test.ts deleted file mode 100644 index 5f72570bc33b..000000000000 --- a/packages/uploads/src/__tests__/getFileExtension.test.ts +++ /dev/null @@ -1,21 +0,0 @@ -import { describe, it, expect } from 'vitest' - -import { getFileExtensionFromDataUri } from '../fileSave.utils' - -describe('getFileExtension', () => { - it('should return the correct file extension for a given data type', () => { - const dataType = 'data:image/png;base64' - const extension = getFileExtensionFromDataUri(dataType) - expect(extension).toBe('png') - }) - - it('handles svgs', () => { - const dataType = 'data:image/svg+xml;base64' - expect(getFileExtensionFromDataUri(dataType)).toBe('svg') - }) - - it('handles gif', () => { - const dataType = 'data:image/gif;base64' - expect(getFileExtensionFromDataUri(dataType)).toBe('gif') - }) -}) diff --git a/packages/uploads/src/__tests__/prismaExtension.local.test.ts b/packages/uploads/src/__tests__/prismaExtension.local.test.ts deleted file mode 100644 index ce6cc4dc9287..000000000000 --- a/packages/uploads/src/__tests__/prismaExtension.local.test.ts +++ /dev/null @@ -1,227 +0,0 @@ -import fs from 'node:fs/promises' - -import { PrismaClient } from '@prisma/client' -import { vol } from 'memfs' -import { describe, it, vi, expect, beforeAll } from 'vitest' -import { $ } from 'zx' - -import { createUploadsExtension } from '../prismaExtension' - -import { dataUrlPng } from './fileMocks' - -/*** - * NOTE: this test does not run in CI, because it requires a local prisma db - * which causes build failures elsewhere. It's still useful to have locally when adding/changing features though - * - * To run it use the script `yarn test:prismaExtension` - */ - -const shouldRunPrismaExtensionTests = - process.env.PRISMA_EXTENSION_TESTS === 'true' - -vi.mock('node:fs/promises', () => ({ - default: { - writeFile: vi.fn(), - unlink: vi.fn(), - readFile: vi.fn((path, encoding) => { - if (encoding === 'base64') { - return 'BASE64_FILE_CONTENT' - } - - return 'MOCKED_FILE_CONTENT' - }), - }, -})) - -vol.fromJSON({ - '/tmp/tus-uploads/123.json': '{}', - '/tmp/tus-uploads/ABCD.json': '{}', -}) - -describe.runIf(shouldRunPrismaExtensionTests)( - 'Uploads Prisma Extension', - () => { - beforeAll(async () => { - console.log('[setup] Setting up unit test prisma db....') - await $`npx prisma migrate reset -f --skip-seed --schema src/__tests__/unit-test-schema.prisma` - console.log('[setup] Done! \n') - }) - - const dummyUploadConfig = { - fields: 'uploadField', - savePath: '/bazinga', - onFileSaved: vi.fn(), - } - - const dumboUploadConfig = { - fields: ['firstUpload', 'secondUpload'], - savePath: '/dumbo', - onFileSaved: vi.fn(), - } - - const tusConfig = { - tusUploadDirectory: '/tmp/tus-uploads', - } - const prismaClient = new PrismaClient().$extends( - createUploadsExtension( - { - dummy: dummyUploadConfig, - dumbo: dumboUploadConfig, - }, - tusConfig, - ), - ) - - describe('Query extensions', () => { - it('will create a file with base64 encoded png', async () => { - const dum1 = await prismaClient.dummy.create({ - data: { - uploadField: dataUrlPng, - }, - }) - - expect(dummyUploadConfig.onFileSaved).toHaveBeenCalled() - expect(dum1.uploadField).toMatch(/bazinga\/.*\.png/) - expect(fs.writeFile).toHaveBeenCalledWith( - expect.stringMatching(/bazinga\/.*\.png/), - expect.anything(), // no need to check content here, makes test slow - ) - }) - - it('handles multiple upload fields', async () => { - const dumbo = await prismaClient.dumbo.create({ - data: { - firstUpload: dataUrlPng, - secondUpload: dataUrlPng, - }, - }) - - expect(dumbo.firstUpload).toMatch(/dumbo\/.*\.png/) - expect(dumbo.secondUpload).toMatch(/dumbo\/.*\.png/) - - expect(dumboUploadConfig.onFileSaved).toHaveBeenCalledTimes(2) - }) - - it('handles empty fields', async () => { - const emptyUploadFieldPromise = prismaClient.dummy.create({ - data: { - uploadField: '', - }, - }) - - await expect(emptyUploadFieldPromise).resolves.not.toThrow() - }) - - it('handles updates, and removes old files', async () => { - const dum1 = await prismaClient.dummy.create({ - data: { - uploadField: dataUrlPng, - }, - }) - - const originalPath = dum1.uploadField - - const dum2 = await prismaClient.dummy.update({ - where: { id: dum1.id }, - data: { - uploadField: dataUrlPng, - }, - }) - - expect(dum2.uploadField).not.toEqual(originalPath) - expect(dum2.uploadField).toMatch(/bazinga\/.*\.png/) - expect(fs.unlink).toHaveBeenCalledWith(originalPath) - }) - - it('handles deletes, and removes files', async () => { - const dum1 = await prismaClient.dummy.create({ - data: { - uploadField: dataUrlPng, - }, - }) - - await prismaClient.dummy.delete({ - where: { id: dum1.id }, - }) - - expect(fs.unlink).toHaveBeenCalledWith(dum1.uploadField) - }) - - it('supports custom file name and save path functions', async () => { - const customNameConfig = { - fields: 'firstUpload', - savePath: '/custom', - onFileSaved: vi.fn(), - fileName: (args) => { - // 👇 Using args here - return `my-name-is-dumbo-${args.data.id}` - }, - } - - const clientWithFileName = new PrismaClient().$extends( - createUploadsExtension({ - dumbo: customNameConfig, - }), - ) - - const dumbo = await clientWithFileName.dumbo.create({ - data: { - firstUpload: dataUrlPng, - secondUpload: '', - id: 55, - }, - }) - - expect(customNameConfig.onFileSaved).toHaveBeenCalled() - expect(dumbo.firstUpload).toBe('/custom/my-name-is-dumbo-55.png') - - // Delete it to clean up - await clientWithFileName.dumbo.delete({ - where: { - id: 55, - }, - }) - }) - - // @TODO have to figure out how to mock the - // it('will move file to new location with TUS uploads', async () => { - // const dumbo = await prismaClient.dumbo.create({ - // data: { - // firstUpload: - // 'http://example.com/.redwood/functions/tusUploadEndpoint/123', - // secondUpload: - // 'http://example.com/.redwood/functions/tusUploadEndpoint/ABCD', - // }, - // }) - - // expect(dumbo.firstUpload).toBe('balknsdg') - // expect(dumbo.secondUpload).toBe('balknsdg') - // }) - - // it('will remove old file when updating with TUS uploads', async () => {}) - - // it('will remove file when deleting with TUS uploads', async () => {}) - }) - - describe('Result extensions', () => { - it('will return a data URL for the file', async () => { - const res1 = await ( - await prismaClient.dummy.create({ - data: { - uploadField: dataUrlPng, - }, - }) - ).withDataUri() - - // Mocked in FS mocks - expect(res1.uploadField).toBe( - '_FILE_CONTENT', - ) - }) - - // @TODO implement - // it('if file is not found, will throw an error', async () => {}) - // it('if saved file is not a path, will throw an error', async () => {}) - }) - }, -) diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts new file mode 100644 index 000000000000..665f660e2189 --- /dev/null +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -0,0 +1,219 @@ +import fs from 'node:fs/promises' + +import { PrismaClient } from '@prisma/client' +import { vol } from 'memfs' +import { describe, it, vi, expect, beforeAll } from 'vitest' +import { $ } from 'zx' + +import { createUploadsExtension } from '../prismaExtension' + +import { dataUrlPng } from './fileMocks' + +/*** + * NOTE: this test does not run in CI, because it requires a local prisma db + * which causes build failures elsewhere. It's still useful to have locally when adding/changing features though + * + * To run it use the script `yarn test:prismaExtension` + */ + +vi.mock('node:fs/promises', () => ({ + default: { + writeFile: vi.fn(), + unlink: vi.fn(), + readFile: vi.fn((path, encoding) => { + if (encoding === 'base64') { + return 'BASE64_FILE_CONTENT' + } + + return 'MOCKED_FILE_CONTENT' + }), + }, +})) + +vol.fromJSON({ + '/tmp/tus-uploads/123.json': '{}', + '/tmp/tus-uploads/ABCD.json': '{}', +}) + +describe('Uploads Prisma Extension', () => { + beforeAll(async () => { + console.log('[setup] Setting up unit test prisma db....') + await $`npx prisma migrate reset -f --skip-seed --schema src/__tests__/unit-test-schema.prisma` + console.log('[setup] Done! \n') + }) + + const dummyUploadConfig = { + fields: 'uploadField', + savePath: '/bazinga', + onFileSaved: vi.fn(), + } + + const dumboUploadConfig = { + fields: ['firstUpload', 'secondUpload'], + savePath: '/dumbo', + onFileSaved: vi.fn(), + } + + const tusConfig = { + tusUploadDirectory: '/tmp/tus-uploads', + } + const prismaClient = new PrismaClient().$extends( + createUploadsExtension( + { + dummy: dummyUploadConfig, + dumbo: dumboUploadConfig, + }, + tusConfig, + ), + ) + + describe('Query extensions', () => { + it('will create a file with base64 encoded png', async () => { + const dum1 = await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) + + expect(dummyUploadConfig.onFileSaved).toHaveBeenCalled() + expect(dum1.uploadField).toMatch(/bazinga\/.*\.png/) + expect(fs.writeFile).toHaveBeenCalledWith( + expect.stringMatching(/bazinga\/.*\.png/), + expect.anything(), // no need to check content here, makes test slow + ) + }) + + it('handles multiple upload fields', async () => { + const dumbo = await prismaClient.dumbo.create({ + data: { + firstUpload: dataUrlPng, + secondUpload: dataUrlPng, + }, + }) + + expect(dumbo.firstUpload).toMatch(/dumbo\/.*\.png/) + expect(dumbo.secondUpload).toMatch(/dumbo\/.*\.png/) + + expect(dumboUploadConfig.onFileSaved).toHaveBeenCalledTimes(2) + }) + + it('handles empty fields', async () => { + const emptyUploadFieldPromise = prismaClient.dummy.create({ + data: { + uploadField: '', + }, + }) + + await expect(emptyUploadFieldPromise).resolves.not.toThrow() + }) + + it('handles updates, and removes old files', async () => { + const dum1 = await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) + + const originalPath = dum1.uploadField + + const dum2 = await prismaClient.dummy.update({ + where: { id: dum1.id }, + data: { + uploadField: dataUrlPng, + }, + }) + + expect(dum2.uploadField).not.toEqual(originalPath) + expect(dum2.uploadField).toMatch(/bazinga\/.*\.png/) + expect(fs.unlink).toHaveBeenCalledWith(originalPath) + }) + + it('handles deletes, and removes files', async () => { + const dum1 = await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) + + await prismaClient.dummy.delete({ + where: { id: dum1.id }, + }) + + expect(fs.unlink).toHaveBeenCalledWith(dum1.uploadField) + }) + + it('supports custom file name and save path functions', async () => { + const customNameConfig = { + fields: 'firstUpload', + savePath: '/custom', + onFileSaved: vi.fn(), + fileName: (args) => { + // 👇 Using args here + return `my-name-is-dumbo-${args.data.id}` + }, + } + + const clientWithFileName = new PrismaClient().$extends( + createUploadsExtension({ + dumbo: customNameConfig, + }), + ) + + const dumbo = await clientWithFileName.dumbo.create({ + data: { + firstUpload: dataUrlPng, + secondUpload: '', + id: 55, + }, + }) + + expect(customNameConfig.onFileSaved).toHaveBeenCalled() + expect(dumbo.firstUpload).toBe('/custom/my-name-is-dumbo-55.png') + + // Delete it to clean up + await clientWithFileName.dumbo.delete({ + where: { + id: 55, + }, + }) + }) + + // @TODO have to figure out how to mock the + // it('will move file to new location with TUS uploads', async () => { + // const dumbo = await prismaClient.dumbo.create({ + // data: { + // firstUpload: + // 'http://example.com/.redwood/functions/tusUploadEndpoint/123', + // secondUpload: + // 'http://example.com/.redwood/functions/tusUploadEndpoint/ABCD', + // }, + // }) + + // expect(dumbo.firstUpload).toBe('balknsdg') + // expect(dumbo.secondUpload).toBe('balknsdg') + // }) + + // it('will remove old file when updating with TUS uploads', async () => {}) + + // it('will remove file when deleting with TUS uploads', async () => {}) + }) + + describe('Result extensions', () => { + it('will return a data URL for the file', async () => { + const res1 = await ( + await prismaClient.dummy.create({ + data: { + uploadField: dataUrlPng, + }, + }) + ).withDataUri() + + // Mocked in FS mocks + expect(res1.uploadField).toBe('_FILE_CONTENT') + }) + + // @TODO implement + // it('if file is not found, will throw an error', async () => {}) + // it('if saved file is not a path, will throw an error', async () => {}) + }) +}) diff --git a/packages/uploads/src/__tests__/unit-test-schema.prisma b/packages/uploads/src/__tests__/unit-test-schema.prisma index 2040c947ba95..e4c606541958 100644 --- a/packages/uploads/src/__tests__/unit-test-schema.prisma +++ b/packages/uploads/src/__tests__/unit-test-schema.prisma @@ -5,6 +5,7 @@ datasource db { generator client { provider = "prisma-client-js" + output = "./prisma-client" // <-- we generated a local prisma client so it doesn't interfere with the mono repo } model Dummy { diff --git a/packages/uploads/src/fileSave.utils.ts b/packages/uploads/src/fileSave.utils.ts index e7d23bd7f548..b39fb3be361f 100644 --- a/packages/uploads/src/fileSave.utils.ts +++ b/packages/uploads/src/fileSave.utils.ts @@ -34,6 +34,8 @@ export async function saveUploadToFile( saveDir, fileName, }) + } else { + throw new Error('Unsupported upload format') } if (!outputPath) { @@ -98,7 +100,7 @@ export async function saveTusUpload( } function isBase64DataUri(uploadUrlOrDataUrl: string) { - // Check if the uploadUrlOrDataUrl is a valid base64 string + // Check if the uploadUrlOrDataUrl is a valid base64 data uri const base64Regex = /^data:(.*?);base64,/ return base64Regex.test(uploadUrlOrDataUrl) } diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 1a584b3f1ab4..0c0ec40191d7 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -1,7 +1,7 @@ import fs from 'node:fs/promises' import { PrismaClient } from '@prisma/client' -import { Prisma } from '@prisma/client/extension' +import { Prisma } from '@prisma/client' import type * as runtime from '@prisma/client/runtime/library' import { ulid } from 'ulid' diff --git a/packages/uploads/vitest.config.mts b/packages/uploads/vitest.config.mts index 984991bff461..92ccad350ca8 100644 --- a/packages/uploads/vitest.config.mts +++ b/packages/uploads/vitest.config.mts @@ -1,4 +1,9 @@ import { defineConfig, configDefaults } from 'vitest/config' +import { fileURLToPath } from 'url'; +import path from 'path'; + +const __filename = fileURLToPath(import.meta.url); +const __dirname = path.dirname(__filename); export default defineConfig({ test: { @@ -6,6 +11,11 @@ export default defineConfig({ deps: { interopDefault: false, }, + globalSetup: ['vitest.setup.mts'], + alias: { + // We alias prisma client, so that it doesn't interfere with other packages in the mono repo + '@prisma/client': path.resolve(__dirname, 'src/__tests__/prisma-client'), + }, }, }) diff --git a/packages/uploads/vitest.setup.mts b/packages/uploads/vitest.setup.mts new file mode 100644 index 000000000000..01987b744896 --- /dev/null +++ b/packages/uploads/vitest.setup.mts @@ -0,0 +1,8 @@ +import { $ } from 'zx' + +export default async function setup() { + console.log('[setup] Setting up unit test prisma db....') + await $`yarn clean:prisma` + await $`npx prisma migrate reset -f --skip-seed --schema src/__tests__/unit-test-schema.prisma` + console.log('[setup] Done! \n') +} From c5f462f04ec4c59a36d46d9b1a9f6c203f541eb7 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 20:48:42 +0700 Subject: [PATCH 26/91] Remove extraneous beforeAll --- packages/uploads/package.json | 3 +-- packages/uploads/src/__tests__/prismaExtension.test.ts | 9 +-------- 2 files changed, 2 insertions(+), 10 deletions(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 1bb59627bb80..3a1065e9f0c0 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -34,8 +34,7 @@ "check:package": "concurrently npm:check:attw yarn publint", "setup:test": "npx prisma migrate reset -f --schema src/__tests__/unit-test-schema.prisma", "test": "vitest run", - "test:watch": "vitest watch", - "test:prismaExtension": "PRISMA_EXTENSION_TESTS=true vitest run" + "test:watch": "vitest watch" }, "dependencies": { "@redwoodjs/project-config": "workspace:*", diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index 665f660e2189..272d7827b597 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -2,8 +2,7 @@ import fs from 'node:fs/promises' import { PrismaClient } from '@prisma/client' import { vol } from 'memfs' -import { describe, it, vi, expect, beforeAll } from 'vitest' -import { $ } from 'zx' +import { describe, it, vi, expect } from 'vitest' import { createUploadsExtension } from '../prismaExtension' @@ -36,12 +35,6 @@ vol.fromJSON({ }) describe('Uploads Prisma Extension', () => { - beforeAll(async () => { - console.log('[setup] Setting up unit test prisma db....') - await $`npx prisma migrate reset -f --skip-seed --schema src/__tests__/unit-test-schema.prisma` - console.log('[setup] Done! \n') - }) - const dummyUploadConfig = { fields: 'uploadField', savePath: '/bazinga', From f2571a796f0858adc720b4319cdf7019aa1bcd57 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 21:08:32 +0700 Subject: [PATCH 27/91] Put . in front of schema :( --- packages/uploads/package.json | 2 +- packages/uploads/vitest.setup.mts | 3 ++- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 3a1065e9f0c0..af2ffffd2b64 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -32,7 +32,7 @@ "build:types-cjs": "tsc --build --verbose tsconfig.types-cjs.json", "check:attw": "tsx attw.ts", "check:package": "concurrently npm:check:attw yarn publint", - "setup:test": "npx prisma migrate reset -f --schema src/__tests__/unit-test-schema.prisma", + "setup:test": "npx prisma migrate reset -f --schema ./src/__tests__/unit-test-schema.prisma", "test": "vitest run", "test:watch": "vitest watch" }, diff --git a/packages/uploads/vitest.setup.mts b/packages/uploads/vitest.setup.mts index 01987b744896..163282422cb4 100644 --- a/packages/uploads/vitest.setup.mts +++ b/packages/uploads/vitest.setup.mts @@ -1,8 +1,9 @@ import { $ } from 'zx' export default async function setup() { + $.verbose = true console.log('[setup] Setting up unit test prisma db....') await $`yarn clean:prisma` - await $`npx prisma migrate reset -f --skip-seed --schema src/__tests__/unit-test-schema.prisma` + await $`npx prisma migrate reset -f --skip-seed --schema ./src/__tests__/unit-test-schema.prisma` console.log('[setup] Done! \n') } From 91c4601835ce40fee4cd3c4548b5b3fcdad4c408 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 21:30:23 +0700 Subject: [PATCH 28/91] Use db push instead --- packages/uploads/package.json | 2 +- packages/uploads/vitest.setup.mts | 3 +-- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index af2ffffd2b64..6b40f3742880 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -32,7 +32,7 @@ "build:types-cjs": "tsc --build --verbose tsconfig.types-cjs.json", "check:attw": "tsx attw.ts", "check:package": "concurrently npm:check:attw yarn publint", - "setup:test": "npx prisma migrate reset -f --schema ./src/__tests__/unit-test-schema.prisma", + "setup:test": "npx prisma db push --schema ./src/__tests__/unit-test-schema.prisma", "test": "vitest run", "test:watch": "vitest watch" }, diff --git a/packages/uploads/vitest.setup.mts b/packages/uploads/vitest.setup.mts index 163282422cb4..80434f24618b 100644 --- a/packages/uploads/vitest.setup.mts +++ b/packages/uploads/vitest.setup.mts @@ -3,7 +3,6 @@ import { $ } from 'zx' export default async function setup() { $.verbose = true console.log('[setup] Setting up unit test prisma db....') - await $`yarn clean:prisma` - await $`npx prisma migrate reset -f --skip-seed --schema ./src/__tests__/unit-test-schema.prisma` + await $`npx prisma db push --schema ./src/__tests__/unit-test-schema.prisma` console.log('[setup] Done! \n') } From e5048a5e9e85e92dd8f1862851e4d16d1242d467 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 21:33:18 +0700 Subject: [PATCH 29/91] Add TUS tests --- .../src/__tests__/prismaExtension.test.ts | 112 ++++++++++++++---- packages/uploads/src/fileSave.utils.ts | 2 +- 2 files changed, 90 insertions(+), 24 deletions(-) diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index 272d7827b597..81cbc410c802 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -8,13 +8,6 @@ import { createUploadsExtension } from '../prismaExtension' import { dataUrlPng } from './fileMocks' -/*** - * NOTE: this test does not run in CI, because it requires a local prisma db - * which causes build failures elsewhere. It's still useful to have locally when adding/changing features though - * - * To run it use the script `yarn test:prismaExtension` - */ - vi.mock('node:fs/promises', () => ({ default: { writeFile: vi.fn(), @@ -26,6 +19,7 @@ vi.mock('node:fs/promises', () => ({ return 'MOCKED_FILE_CONTENT' }), + copyFile: vi.fn(), }, })) @@ -171,24 +165,96 @@ describe('Uploads Prisma Extension', () => { }) }) - // @TODO have to figure out how to mock the - // it('will move file to new location with TUS uploads', async () => { - // const dumbo = await prismaClient.dumbo.create({ - // data: { - // firstUpload: - // 'http://example.com/.redwood/functions/tusUploadEndpoint/123', - // secondUpload: - // 'http://example.com/.redwood/functions/tusUploadEndpoint/ABCD', - // }, - // }) + it('will move file to new location with TUS uploads', async () => { + // Mock TUS metadata files + vi.mock('/tmp/tus-uploads/123.json', () => { + return { + metadata: { + filetype: 'image/png', + }, + } + }) - // expect(dumbo.firstUpload).toBe('balknsdg') - // expect(dumbo.secondUpload).toBe('balknsdg') - // }) + vi.mock('/tmp/tus-uploads/ABCD.json', () => { + return { + metadata: { + filetype: 'application/pdf', + }, + } + }) - // it('will remove old file when updating with TUS uploads', async () => {}) + const dumbo = await prismaClient.dumbo.create({ + data: { + firstUpload: + 'http://example.com/.redwood/functions/tusUploadEndpoint/123', + secondUpload: + 'http://example.com/.redwood/functions/tusUploadEndpoint/ABCD', + }, + }) + + expect(fs.copyFile).toHaveBeenCalledTimes(2) + expect(dumbo.firstUpload).toMatch(/dumbo\/.*\.png/) + expect(dumbo.secondUpload).toMatch(/dumbo\/.*\.pdf/) + }) + + it('will remove old file when updating with TUS uploads', async () => { + // Mock TUS metadata files + vi.mock('/tmp/tus-uploads/512356.json', () => { + return { + metadata: { + filetype: 'image/gif', + }, + } + }) + + const dumbo = await prismaClient.dumbo.create({ + data: { + firstUpload: + 'http://example.com/.redwood/functions/tusUploadEndpoint/123', + secondUpload: '', + }, + }) - // it('will remove file when deleting with TUS uploads', async () => {}) + const originalPath = dumbo.firstUpload + + const dumbo2 = await prismaClient.dumbo.update({ + where: { id: dumbo.id }, + data: { + firstUpload: + 'http://example.com/.redwood/functions/tusUploadEndpoint/512356', + }, + }) + + expect(dumbo2.firstUpload).not.toEqual(originalPath) + expect(dumbo2.firstUpload).toMatch(/dumbo\/.*\.gif/) + + // And deletes it! + expect(fs.unlink).toHaveBeenCalledWith(originalPath) + }) + + it('will remove file when deleting with TUS uploads', async () => { + // Mock TUS metadata files + vi.mock('/tmp/tus-uploads/512356.json', () => { + return { + metadata: { + filetype: 'image/gif', + }, + } + }) + + const dummy = await prismaClient.dummy.create({ + data: { + uploadField: + 'http://example.com/.redwood/functions/tusUploadEndpoint/123', + }, + }) + + await prismaClient.dummy.delete({ + where: { id: dummy.id }, + }) + + expect(fs.unlink).toHaveBeenCalledWith(dummy.uploadField) + }) }) describe('Result extensions', () => { @@ -205,7 +271,7 @@ describe('Uploads Prisma Extension', () => { expect(res1.uploadField).toBe('_FILE_CONTENT') }) - // @TODO implement + // @TODO Handle edge cases (file removed, data modified, etc.) // it('if file is not found, will throw an error', async () => {}) // it('if saved file is not a path, will throw an error', async () => {}) }) diff --git a/packages/uploads/src/fileSave.utils.ts b/packages/uploads/src/fileSave.utils.ts index b39fb3be361f..42b165a8e721 100644 --- a/packages/uploads/src/fileSave.utils.ts +++ b/packages/uploads/src/fileSave.utils.ts @@ -84,7 +84,7 @@ export async function saveTusUpload( `${tusId}.json`, ) // Can't await import, because JSON file. - const tusMeta = require(metaFile) + const tusMeta = await import(metaFile) const fileExtension = tusMeta.metadata.filetype.split('/')[1] From 28041a40b43f7d5d83fb4c54a659987709f0b849 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 21:34:21 +0700 Subject: [PATCH 30/91] Add changeset --- .changesets/11154.md | 15 +++++++++++++++ 1 file changed, 15 insertions(+) create mode 100644 .changesets/11154.md diff --git a/.changesets/11154.md b/.changesets/11154.md new file mode 100644 index 000000000000..44c278ffb1f5 --- /dev/null +++ b/.changesets/11154.md @@ -0,0 +1,15 @@ +- feat(rw-uploads): Create uploads package with prisma extension (#11154) by @dac09 + +This PR does the following: +- creates new `@redwoodjs/uploads` package. This is configured to be a dual esm/cjs package. +- exports prisma extension with legacy support e.g. + ``` + import { + createUploadsExtension, + UploadsConfig, + } from '@redwoodjs/uploads/prisma' + ``` +- scaffolds unit test structure for the prisma extension +- the prisma extension does the following: +a) If a base64 or TUS upload URL string is sent, it saves or moves it to a file +b) Adds a result extension (withDataUri), to make it easier to deal with files From 087ed7f0f7b3d32ee83dc5a0ad335bbb7023bd9d Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 21:43:15 +0700 Subject: [PATCH 31/91] Clean up in progress stuff --- ...e.units.test.ts => fileSave.utils.test.ts} | 25 +------------------ 1 file changed, 1 insertion(+), 24 deletions(-) rename packages/uploads/src/__tests__/{fileSave.units.test.ts => fileSave.utils.test.ts} (59%) diff --git a/packages/uploads/src/__tests__/fileSave.units.test.ts b/packages/uploads/src/__tests__/fileSave.utils.test.ts similarity index 59% rename from packages/uploads/src/__tests__/fileSave.units.test.ts rename to packages/uploads/src/__tests__/fileSave.utils.test.ts index 5907ea216522..940937383778 100644 --- a/packages/uploads/src/__tests__/fileSave.units.test.ts +++ b/packages/uploads/src/__tests__/fileSave.utils.test.ts @@ -1,4 +1,4 @@ -import { describe, it, expect, vi } from 'vitest' +import { describe, it, expect } from 'vitest' import { getFileExtensionFromDataUri, @@ -24,29 +24,6 @@ describe('getFileExtension', () => { }) describe('saveUploadToFile', () => { - vi.mock('node:fs/promises', () => ({ - default: { - writeFile: vi.fn(), - unlink: vi.fn(), - // readFile: vi.fn((path, encoding) => { - // if (encoding === 'base64') { - // return 'BASE64_FILE_CONTENT' - // } - - // return 'MOCKED_FILE_CONTENT' - // }), - }, - })) - - // it('Should call saveBase64DataUriToFile if the uploadUrlOrDataUrl is a base64 data uri', async () => { - // saveUploadToFile('data:image/png;base64,....', { - // saveDir: 'uploads', - // fileName: 'test', - // }) - // }) - - // it('Should call saveTusFileToFile if the uploadUrlOrDataUrl is a tus url', async () => {}) - it('Should throw an error if the uploadUrlOrDataUrl is not a base64 data uri or a tus url', async () => { try { await saveUploadToFile('/random/path/to/whatever.png', { From c6ec5d86b7016fea7c1894bd8439c018e1a65ac5 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 21:45:37 +0700 Subject: [PATCH 32/91] Remove old comment --- packages/uploads/src/fileSave.utils.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/packages/uploads/src/fileSave.utils.ts b/packages/uploads/src/fileSave.utils.ts index 42b165a8e721..757ed221bfc3 100644 --- a/packages/uploads/src/fileSave.utils.ts +++ b/packages/uploads/src/fileSave.utils.ts @@ -83,7 +83,6 @@ export async function saveTusUpload( path.join(getPaths().base, tusConfig.tusUploadDirectory), `${tusId}.json`, ) - // Can't await import, because JSON file. const tusMeta = await import(metaFile) const fileExtension = tusMeta.metadata.filetype.split('/')[1] From ec85ddcd04a150825b913ab63267fa5b5e899cc5 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 7 Aug 2024 21:56:08 +0700 Subject: [PATCH 33/91] Fix weird CJS/ESM issue with JSON import --- packages/uploads/src/fileSave.utils.ts | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/packages/uploads/src/fileSave.utils.ts b/packages/uploads/src/fileSave.utils.ts index 757ed221bfc3..550a388b6014 100644 --- a/packages/uploads/src/fileSave.utils.ts +++ b/packages/uploads/src/fileSave.utils.ts @@ -83,9 +83,14 @@ export async function saveTusUpload( path.join(getPaths().base, tusConfig.tusUploadDirectory), `${tusId}.json`, ) - const tusMeta = await import(metaFile) + const metafile = await import(metaFile, { assert: { type: 'json' } }) - const fileExtension = tusMeta.metadata.filetype.split('/')[1] + // CJS dynamic imports wrap in default + const tusMetadata: { filetype: string } = metafile.metadata + ? metafile.metadata + : metafile.default.metadata + + const fileExtension = tusMetadata.filetype.split('/')[1] const savedFilePath = path.join(saveDir, `${fileName}.${fileExtension}`) From 821c02af3d277ebf9fd79346c02c99449267b5e7 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 8 Aug 2024 14:32:51 +0700 Subject: [PATCH 34/91] Get types working for result extension --- .../src/__tests__/prismaExtension.test.ts | 3 ++- packages/uploads/src/prismaExtension.ts | 16 ++++++---------- 2 files changed, 8 insertions(+), 11 deletions(-) diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index 81cbc410c802..c811392a955b 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -1,12 +1,13 @@ import fs from 'node:fs/promises' -import { PrismaClient } from '@prisma/client' import { vol } from 'memfs' import { describe, it, vi, expect } from 'vitest' import { createUploadsExtension } from '../prismaExtension' import { dataUrlPng } from './fileMocks' +// @MARK: use the local prisma client +import { PrismaClient } from './prisma-client' vi.mock('node:fs/promises', () => ({ default: { diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 0c0ec40191d7..e182e340bf43 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -27,10 +27,8 @@ export type UploadConfigForModel = { onFileSaved?: (filePath: string) => void | Promise } -export type UploadsConfig = Record< - MName, - UploadConfigForModel -> +export type UploadsConfig = + Record export const createUploadsExtension = ( config: UploadsConfig, @@ -68,11 +66,9 @@ export const createUploadsExtension = ( [K in MNames]: { withDataUri: { needs: Record - compute: ( - // @MARK: this generic doesn't get picked up by prisma - // the returned type ends up being unknown - modelData: T, - ) => () => Promise + compute: ( + modelData: Record, + ) => (this: T) => Promise } } } @@ -148,7 +144,7 @@ export const createUploadsExtension = ( // 2. If not a path, relative or absolute, throw error return { - ...modelData, + ...(modelData as any), ...base64UploadFields, } } From 30cc64ed84018a21917f0e73b85c6fcb2ef61556 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 8 Aug 2024 14:44:24 +0700 Subject: [PATCH 35/91] Clean up types and comments for result extends --- packages/uploads/src/prismaExtension.ts | 28 ++++++++++++------------- 1 file changed, 13 insertions(+), 15 deletions(-) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index e182e340bf43..b40957868b4a 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -38,6 +38,17 @@ export const createUploadsExtension = ( // instead of creating a new PrismaClient instance const prismaInstance = new PrismaClient() + type ResultExtends = { + [K in MNames]: { + withDataUri: { + needs: Record + compute: ( + modelData: Record, + ) => (this: T) => Promise + } + } + } + async function deleteUploadsFromDiskForArgs({ model, args, @@ -62,17 +73,7 @@ export const createUploadsExtension = ( const queryExtends: runtime.ExtensionArgs['query'] = {} - const resultExtends = {} as { - [K in MNames]: { - withDataUri: { - needs: Record - compute: ( - modelData: Record, - ) => (this: T) => Promise - } - } - } - + const resultExtends = {} as ResultExtends for (const modelName in config) { // Guaranteed to have modelConfig, we're looping over config 🙄 const modelConfig = config[modelName as MNames] as UploadConfigForModel @@ -139,11 +140,8 @@ export const createUploadsExtension = ( ) } - // @TODO: edge cases - // 1. If readfile fails - file not found, etc. - // 2. If not a path, relative or absolute, throw error - return { + // modelData is of type unknown at this point ...(modelData as any), ...base64UploadFields, } From e80e556b44bef237601590a172341dc8749a4968 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 13 Aug 2024 14:50:47 +0700 Subject: [PATCH 36/91] WIP: adapter --- packages/uploads/src/FileSystemStorage.ts | 29 +++ packages/uploads/src/StorageAdapter.ts | 22 ++ .../src/__tests__/fileSave.utils.test.ts | 37 --- .../src/__tests__/prismaExtension.test.ts | 29 ++- packages/uploads/src/fileSave.utils.ts | 129 ----------- packages/uploads/src/prismaExtension.ts | 97 ++++---- yarn.lock | 210 +++++++++++++++++- 7 files changed, 325 insertions(+), 228 deletions(-) create mode 100644 packages/uploads/src/FileSystemStorage.ts create mode 100644 packages/uploads/src/StorageAdapter.ts delete mode 100644 packages/uploads/src/__tests__/fileSave.utils.test.ts diff --git a/packages/uploads/src/FileSystemStorage.ts b/packages/uploads/src/FileSystemStorage.ts new file mode 100644 index 000000000000..7da595b54be5 --- /dev/null +++ b/packages/uploads/src/FileSystemStorage.ts @@ -0,0 +1,29 @@ +import fs from 'node:fs/promises' +import path from 'node:path' + +import type { StorageAdapter } from './StorageAdapter.js' +import type { SaveOptions } from './StorageAdapter.js' + +export class FileSystemStorage implements StorageAdapter { + // let basePath: string + // @TODO enable base path + // constructor({ basePath }) { + // this.basePath = basePath + // } + + async save(o_file: File, saveOpts: SaveOptions) { + // const file = new File([o_file], o_file.name) + // console.log(`👉 \n ~ FileSystemStorage ~ file:`, file.name) + console.log(`👉 \n ~ FileSystemStorage ~ file:`, await o_file.text()) + + const location = path.join(saveOpts.path, saveOpts.fileName + o_file.type) + const nodeBuffer = await o_file.arrayBuffer() + const extension = path.extname(o_file.name) + + await fs.writeFile(`${location}.${extension}`, Buffer.from(nodeBuffer)) + return { location } + } + async remove(filePath: string) { + await fs.unlink(filePath) + } +} diff --git a/packages/uploads/src/StorageAdapter.ts b/packages/uploads/src/StorageAdapter.ts new file mode 100644 index 000000000000..ecc0f28ef58a --- /dev/null +++ b/packages/uploads/src/StorageAdapter.ts @@ -0,0 +1,22 @@ +/** + * The storage adapter will just save the file and return + * { + * fileId: string, + * location: string, // depending on storage it could be a path + * } + */ + +export type AdapterResult = { + location: string +} + +export type SaveOptions = { + fileName: string + path: string +} + +export abstract class StorageAdapter { + abstract save(file: File, saveOpts?: SaveOptions): Promise + abstract remove(fileLocation: AdapterResult['location']): Promise + // abstract replace(fileId: string, file: File): Promise +} diff --git a/packages/uploads/src/__tests__/fileSave.utils.test.ts b/packages/uploads/src/__tests__/fileSave.utils.test.ts deleted file mode 100644 index 940937383778..000000000000 --- a/packages/uploads/src/__tests__/fileSave.utils.test.ts +++ /dev/null @@ -1,37 +0,0 @@ -import { describe, it, expect } from 'vitest' - -import { - getFileExtensionFromDataUri, - saveUploadToFile, -} from '../fileSave.utils' - -describe('getFileExtension', () => { - it('should return the correct file extension for a given data type', () => { - const dataType = 'data:image/png;base64' - const extension = getFileExtensionFromDataUri(dataType) - expect(extension).toBe('png') - }) - - it('handles svgs', () => { - const dataType = 'data:image/svg+xml;base64' - expect(getFileExtensionFromDataUri(dataType)).toBe('svg') - }) - - it('handles gif', () => { - const dataType = 'data:image/gif;base64' - expect(getFileExtensionFromDataUri(dataType)).toBe('gif') - }) -}) - -describe('saveUploadToFile', () => { - it('Should throw an error if the uploadUrlOrDataUrl is not a base64 data uri or a tus url', async () => { - try { - await saveUploadToFile('/random/path/to/whatever.png', { - saveDir: 'uploads', - fileName: 'test', - }) - } catch (e) { - expect(e.message).toBe('Unsupported upload format') - } - }) -}) diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index c811392a955b..7b7689ae69a2 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -3,9 +3,10 @@ import fs from 'node:fs/promises' import { vol } from 'memfs' import { describe, it, vi, expect } from 'vitest' -import { createUploadsExtension } from '../prismaExtension' +import { FileSystemStorage } from '../FileSystemStorage.js' +import { createUploadsExtension } from '../prismaExtension.js' -import { dataUrlPng } from './fileMocks' +import { dataUrlPng } from './fileMocks.js' // @MARK: use the local prisma client import { PrismaClient } from './prisma-client' @@ -42,24 +43,27 @@ describe('Uploads Prisma Extension', () => { onFileSaved: vi.fn(), } - const tusConfig = { - tusUploadDirectory: '/tmp/tus-uploads', - } const prismaClient = new PrismaClient().$extends( createUploadsExtension( { dummy: dummyUploadConfig, dumbo: dumboUploadConfig, }, - tusConfig, + new FileSystemStorage(), ), ) describe('Query extensions', () => { - it('will create a file with base64 encoded png', async () => { + it.only('will create a file with base64 encoded png', async () => { + const file = new File(['hello'], 'hello.txt', { + type: 'text/plain', + }) + console.log(`👉 \n ~ file:`, file) + console.log(`👉 \n ~ file arraybuf:`, await file.arrayBuffer()) + const dum1 = await prismaClient.dummy.create({ data: { - uploadField: dataUrlPng, + uploadField: file, }, }) @@ -142,9 +146,12 @@ describe('Uploads Prisma Extension', () => { } const clientWithFileName = new PrismaClient().$extends( - createUploadsExtension({ - dumbo: customNameConfig, - }), + createUploadsExtension( + { + dumbo: customNameConfig, + }, + new FileSystemStorage(), + ), ) const dumbo = await clientWithFileName.dumbo.create({ diff --git a/packages/uploads/src/fileSave.utils.ts b/packages/uploads/src/fileSave.utils.ts index 550a388b6014..193626c89d87 100644 --- a/packages/uploads/src/fileSave.utils.ts +++ b/packages/uploads/src/fileSave.utils.ts @@ -3,135 +3,6 @@ import path from 'node:path' import mime from 'mime-types' -import { getPaths } from '@redwoodjs/project-config' - -export type TUSServerConfig = { - tusUploadDirectory: string -} - -/** - * This function takes an upload field, determines whether its TUS or Base64, and saves it to the file system. - */ -export async function saveUploadToFile( - uploadUrlOrDataUrl: string, - { fileName, saveDir }: { saveDir: string; fileName: string }, - tusConfig?: TUSServerConfig, -) { - let outputPath: string | null = null - - if (isBase64DataUri(uploadUrlOrDataUrl)) { - outputPath = await saveBase64DataUriToFile(uploadUrlOrDataUrl, { - saveDir, - fileName, - }) - } else if (uploadUrlOrDataUrl.startsWith('http')) { - if (!tusConfig) { - throw new Error('TusConfig not supplied.') - } - - outputPath = await saveTusUpload(uploadUrlOrDataUrl, { - tusConfig, - saveDir, - fileName, - }) - } else { - throw new Error('Unsupported upload format') - } - - if (!outputPath) { - throw new Error('Unsupported upload URL') - } - - // @MARK: we can create a new record on the uploads table here - - return outputPath -} - -// @MARK: if we block the TUS GET, we don't really need to move it -// We send the TUS upload URL as the value of the field -export async function saveTusUpload( - uploadUrl: string, - { - tusConfig, - saveDir, - fileName, - }: { - tusConfig: TUSServerConfig - saveDir: string - fileName: string - }, -) { - // Get the last part of the TUS upload url - // http://localhost:8910/.redwood/functions/uploadTUS/👉28fa96bf5772338d51👈 - const tusId = uploadUrl.split('/').slice(-1).pop() - - if (!tusId) { - throw new Error('Could not extract upload ID from URL') - } - - if (!tusConfig.tusUploadDirectory) { - throw new Error( - 'You have to configure the TUS Upload Directory in the prisma extension. It is required for TUS uploads', - ) - } - - // Optional Step.... - const metaFile = path.join( - path.isAbsolute(tusConfig.tusUploadDirectory) - ? tusConfig.tusUploadDirectory - : // @MARK: if the directory supplied isn't relative - path.join(getPaths().base, tusConfig.tusUploadDirectory), - `${tusId}.json`, - ) - const metafile = await import(metaFile, { assert: { type: 'json' } }) - - // CJS dynamic imports wrap in default - const tusMetadata: { filetype: string } = metafile.metadata - ? metafile.metadata - : metafile.default.metadata - - const fileExtension = tusMetadata.filetype.split('/')[1] - - const savedFilePath = path.join(saveDir, `${fileName}.${fileExtension}`) - - // @MARK: we can also move... - await fs.copyFile( - path.join(tusConfig.tusUploadDirectory, tusId), - savedFilePath, - ) - - return savedFilePath -} - -function isBase64DataUri(uploadUrlOrDataUrl: string) { - // Check if the uploadUrlOrDataUrl is a valid base64 data uri - const base64Regex = /^data:(.*?);base64,/ - return base64Regex.test(uploadUrlOrDataUrl) -} - -async function saveBase64DataUriToFile( - dataUrlString: string, - { saveDir, fileName }: { saveDir: string; fileName: string }, -) { - const [dataType, fileContent] = dataUrlString.split(',') - // format is data:image/png;base64,.... - const fileExtension = getFileExtensionFromDataUri(dataType) - const filePath = path.join(saveDir, `${fileName}.${fileExtension}`) - - await fs.writeFile(filePath, Buffer.from(fileContent, 'base64')) - - return filePath -} - -export function getFileExtensionFromDataUri(dataType: string): string { - const mimeType = dataType.split(':')[1].split(';')[0] - const extension = mime.extension(mimeType) - if (!extension) { - throw new Error(`Unsupported file type: ${mimeType}`) - } - return extension -} - export async function fileToDataUri(filePath: string) { const base64Data = await fs.readFile(filePath, 'base64') const ext = path.extname(filePath) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index b40957868b4a..710d018cf411 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -1,15 +1,10 @@ -import fs from 'node:fs/promises' - import { PrismaClient } from '@prisma/client' import { Prisma } from '@prisma/client' import type * as runtime from '@prisma/client/runtime/library' import { ulid } from 'ulid' -import { - fileToDataUri, - saveUploadToFile, - type TUSServerConfig, -} from './fileSave.utils.js' +import { fileToDataUri } from './fileSave.utils.js' +import type { StorageAdapter } from './StorageAdapter.js' type FilterOutDollarPrefixed = T extends `$${string}` ? never @@ -32,7 +27,7 @@ export type UploadsConfig = export const createUploadsExtension = ( config: UploadsConfig, - tusConfig?: TUSServerConfig, + storageAdapter: StorageAdapter, ) => { // @TODO I think we can use Prisma.getExtensionContext(this) // instead of creating a new PrismaClient instance @@ -49,15 +44,18 @@ export const createUploadsExtension = ( } } - async function deleteUploadsFromDiskForArgs({ - model, - args, - fields, - }: { - model: string - args: T - fields: string[] - }) { + async function deleteUpload( + { + model, + args, + fields, + }: { + model: string + args: T + fields: string[] + }, + storageAdapter: StorageAdapter, + ) { // With strict mode you cannot call findFirstOrThrow with the same args, because it is a union type // Ideally there's a better way to do this const record = await ( @@ -67,7 +65,7 @@ export const createUploadsExtension = ( // Delete the file from the file system fields.forEach(async (field) => { const filePath = record[field] - await fs.unlink(filePath) + await storageAdapter.remove(filePath) }) } @@ -83,20 +81,23 @@ export const createUploadsExtension = ( queryExtends[modelName] = { async update({ query, model, args }) { - await deleteUploadsFromDiskForArgs({ - model, - args: { - // The update args contains data, which we don't need to supply to delete - where: args.where, + await deleteUpload( + { + model, + args: { + // The update args contains data, which we don't need to supply to delete + where: args.where, + }, + fields: uploadFields, }, - fields: uploadFields, - }) + storageAdapter, + ) const uploadArgs = await saveUploads( uploadFields, args, modelConfig, - tusConfig, + storageAdapter, ) return query(uploadArgs) @@ -106,21 +107,23 @@ export const createUploadsExtension = ( uploadFields, args, modelConfig, - tusConfig, + storageAdapter, ) return query(uploadArgs) }, async delete({ model, query, args }) { - await deleteUploadsFromDiskForArgs({ - model, - args, - fields: uploadFields, - }) + await deleteUpload( + { + model, + args, + fields: uploadFields, + }, + storageAdapter, + ) return query(args) }, - // findMany({ query, args, operation }) {} } // This makes the result extension only available for models with uploadFields @@ -169,11 +172,11 @@ async function saveUploads( uploadFields: string[], args: runtime.JsArgs & { data?: { - [key: string]: runtime.JsInputValue + [key: string]: runtime.JsInputValue | File } }, modelConfig: UploadConfigForModel, - tusConfig?: TUSServerConfig, + storageAdapter: StorageAdapter, ) { const fieldsToUpdate: { [key: string]: string @@ -183,13 +186,14 @@ async function saveUploads( throw new Error('No data in prisma query') } - // For each upload property, we need to: + // For each upload property, we need to:z // 1. save the file to the file system (path or name from config) // 2. replace the value of the field for await (const field of uploadFields) { - const uploadUrlOrDataUrl = args.data[field] as string + const uploadFile = args.data[field] as File + console.log(`👉 \n ~ uploadFile:`, uploadFile) - if (!uploadUrlOrDataUrl) { + if (!uploadFile) { continue } @@ -203,21 +207,18 @@ async function saveUploads( ? modelConfig.savePath(args) : modelConfig.savePath || 'web/public/uploads' - const savedFilePath = await saveUploadToFile( - uploadUrlOrDataUrl, - { - fileName, - saveDir, - }, - tusConfig, - ) + const savedFile = await storageAdapter.save(uploadFile, { + fileName, + path: saveDir, + }) - fieldsToUpdate[field] = savedFilePath + // @TODO should we return location or fileId? + fieldsToUpdate[field] = savedFile.location // Call the onFileSaved callback // Having it here means it'll always trigger whether create/update if (modelConfig.onFileSaved) { - await modelConfig.onFileSaved(savedFilePath) + await modelConfig.onFileSaved(savedFile.location) } } diff --git a/yarn.lock b/yarn.lock index d8e95521b463..2bc153502df2 100644 --- a/yarn.lock +++ b/yarn.lock @@ -205,6 +205,23 @@ __metadata: languageName: node linkType: hard +"@arethetypeswrong/cli@npm:0.15.3": + version: 0.15.3 + resolution: "@arethetypeswrong/cli@npm:0.15.3" + dependencies: + "@arethetypeswrong/core": "npm:0.15.1" + chalk: "npm:^4.1.2" + cli-table3: "npm:^0.6.3" + commander: "npm:^10.0.1" + marked: "npm:^9.1.2" + marked-terminal: "npm:^6.0.0" + semver: "npm:^7.5.4" + bin: + attw: dist/index.js + checksum: 10c0/5998ab4a2195f9036a5c1988f73912a0a82cceeaa6a4e647b04414ad956a62163d8286b2a936941f23065b0c872f2bbdf9196fe3cac19c40b8b62a643d91c3c2 + languageName: node + linkType: hard + "@arethetypeswrong/cli@npm:0.15.4": version: 0.15.4 resolution: "@arethetypeswrong/cli@npm:0.15.4" @@ -11448,6 +11465,18 @@ __metadata: languageName: node linkType: hard +"@vitest/expect@npm:2.0.4": + version: 2.0.4 + resolution: "@vitest/expect@npm:2.0.4" + dependencies: + "@vitest/spy": "npm:2.0.4" + "@vitest/utils": "npm:2.0.4" + chai: "npm:^5.1.1" + tinyrainbow: "npm:^1.2.0" + checksum: 10c0/18acdd6b1f5001830722fab7d41b0bd754e37572dded74d1549c5e8f40e58d9e4bbbb6a8ce6be1200b04653237329ba1aeeb3330c2a41f1024450016464d491e + languageName: node + linkType: hard + "@vitest/expect@npm:2.0.5": version: 2.0.5 resolution: "@vitest/expect@npm:2.0.5" @@ -11460,7 +11489,16 @@ __metadata: languageName: node linkType: hard -"@vitest/pretty-format@npm:2.0.5, @vitest/pretty-format@npm:^2.0.5": +"@vitest/pretty-format@npm:2.0.4": + version: 2.0.4 + resolution: "@vitest/pretty-format@npm:2.0.4" + dependencies: + tinyrainbow: "npm:^1.2.0" + checksum: 10c0/c2ac3ca302b93ad53ea2977209ee4eb31a313c18690034a09f8ec5528d7e82715c233c4927ecf8b364203c5e5475231d9b737b3fb7680eea71882e1eae11e473 + languageName: node + linkType: hard + +"@vitest/pretty-format@npm:2.0.5, @vitest/pretty-format@npm:^2.0.4, @vitest/pretty-format@npm:^2.0.5": version: 2.0.5 resolution: "@vitest/pretty-format@npm:2.0.5" dependencies: @@ -11469,6 +11507,16 @@ __metadata: languageName: node linkType: hard +"@vitest/runner@npm:2.0.4": + version: 2.0.4 + resolution: "@vitest/runner@npm:2.0.4" + dependencies: + "@vitest/utils": "npm:2.0.4" + pathe: "npm:^1.1.2" + checksum: 10c0/b550372ce5e2c6a3f08dbd584ea669723fc0d789ebaa4224b703f12e908813fb76b963ea9ac2265aa751cab0309f637dc1fa7ce3fb3e67e08e52e241d33237ee + languageName: node + linkType: hard + "@vitest/runner@npm:2.0.5": version: 2.0.5 resolution: "@vitest/runner@npm:2.0.5" @@ -11479,6 +11527,17 @@ __metadata: languageName: node linkType: hard +"@vitest/snapshot@npm:2.0.4": + version: 2.0.4 + resolution: "@vitest/snapshot@npm:2.0.4" + dependencies: + "@vitest/pretty-format": "npm:2.0.4" + magic-string: "npm:^0.30.10" + pathe: "npm:^1.1.2" + checksum: 10c0/67608c5b1e2f8b02ebc95286cd644c31ea29344c81d67151375b6eebf088a0eea242756eefb509aac626b8f7f091044fdcbc80d137d811ead1117a4a524e2d74 + languageName: node + linkType: hard + "@vitest/snapshot@npm:2.0.5": version: 2.0.5 resolution: "@vitest/snapshot@npm:2.0.5" @@ -11490,6 +11549,15 @@ __metadata: languageName: node linkType: hard +"@vitest/spy@npm:2.0.4": + version: 2.0.4 + resolution: "@vitest/spy@npm:2.0.4" + dependencies: + tinyspy: "npm:^3.0.0" + checksum: 10c0/ef0d0c5e36bb6dfa3ef7561368b39c92cd89bb52d112ec13345dfc99981796a9af98bafd35ce6952322a6a7534eaad144485fe7764628d94d77edeba5fa773b6 + languageName: node + linkType: hard + "@vitest/spy@npm:2.0.5": version: 2.0.5 resolution: "@vitest/spy@npm:2.0.5" @@ -11499,6 +11567,18 @@ __metadata: languageName: node linkType: hard +"@vitest/utils@npm:2.0.4": + version: 2.0.4 + resolution: "@vitest/utils@npm:2.0.4" + dependencies: + "@vitest/pretty-format": "npm:2.0.4" + estree-walker: "npm:^3.0.3" + loupe: "npm:^3.1.1" + tinyrainbow: "npm:^1.2.0" + checksum: 10c0/48e0bad3aa463d147b125e355b6bc6c5b4a5eab600132ebafac8379800273b2f47df17dbf76fe179b1500cc6b5866ead2d375a39a9114a03f705eb8850b93afa + languageName: node + linkType: hard + "@vitest/utils@npm:2.0.5": version: 2.0.5 resolution: "@vitest/utils@npm:2.0.5" @@ -12036,7 +12116,7 @@ __metadata: languageName: node linkType: hard -"ansi-escapes@npm:^6.0.0": +"ansi-escapes@npm:^6.0.0, ansi-escapes@npm:^6.2.0": version: 6.2.1 resolution: "ansi-escapes@npm:6.2.1" checksum: 10c0/a2c6f58b044be5f69662ee17073229b492daa2425a7fd99a665db6c22eab6e4ab42752807def7281c1c7acfed48f87f2362dda892f08c2c437f1b39c6b033103 @@ -12098,6 +12178,13 @@ __metadata: languageName: node linkType: hard +"ansicolors@npm:~0.3.2": + version: 0.3.2 + resolution: "ansicolors@npm:0.3.2" + checksum: 10c0/e202182895e959c5357db6c60791b2abaade99fcc02221da11a581b26a7f83dc084392bc74e4d3875c22f37b3c9ef48842e896e3bfed394ec278194b8003e0ac + languageName: node + linkType: hard + "any-promise@npm:^1.0.0": version: 1.3.0 resolution: "any-promise@npm:1.3.0" @@ -13470,6 +13557,18 @@ __metadata: languageName: node linkType: hard +"cardinal@npm:^2.1.1": + version: 2.1.1 + resolution: "cardinal@npm:2.1.1" + dependencies: + ansicolors: "npm:~0.3.2" + redeyed: "npm:~2.1.0" + bin: + cdl: ./bin/cdl.js + checksum: 10c0/0051d0e64c0e1dff480c1aace4c018c48ecca44030533257af3f023107ccdeb061925603af6d73710f0345b0ae0eb57e5241d181d9b5fdb595d45c5418161675 + languageName: node + linkType: hard + "caseless@npm:~0.12.0": version: 0.12.0 resolution: "caseless@npm:0.12.0" @@ -16249,7 +16348,7 @@ __metadata: languageName: node linkType: hard -"esbuild@npm:^0.21.3": +"esbuild@npm:^0.21.3, esbuild@npm:~0.21.5": version: 0.21.5 resolution: "esbuild@npm:0.21.5" dependencies: @@ -22044,6 +22143,22 @@ __metadata: languageName: node linkType: hard +"marked-terminal@npm:^6.0.0": + version: 6.2.0 + resolution: "marked-terminal@npm:6.2.0" + dependencies: + ansi-escapes: "npm:^6.2.0" + cardinal: "npm:^2.1.1" + chalk: "npm:^5.3.0" + cli-table3: "npm:^0.6.3" + node-emoji: "npm:^2.1.3" + supports-hyperlinks: "npm:^3.0.0" + peerDependencies: + marked: ">=1 <12" + checksum: 10c0/72d4093cbb1ee864ced1f88fdb6fb8dbfea56d6aa3d8a1ec401ac51866ff3c32382c3f4642b19f2d808c798efde23b10300b99e3b6475b3f79e41e7741581d54 + languageName: node + linkType: hard + "marked-terminal@npm:^7.1.0": version: 7.1.0 resolution: "marked-terminal@npm:7.1.0" @@ -25906,6 +26021,15 @@ __metadata: languageName: node linkType: hard +"redeyed@npm:~2.1.0": + version: 2.1.1 + resolution: "redeyed@npm:2.1.1" + dependencies: + esprima: "npm:~4.0.0" + checksum: 10c0/350f5e39aebab3886713a170235c38155ee64a74f0f7e629ecc0144ba33905efea30c2c3befe1fcbf0b0366e344e7bfa34e6b2502b423c9a467d32f1306ef166 + languageName: node + linkType: hard + "redis-errors@npm:^1.0.0, redis-errors@npm:^1.2.0": version: 1.2.0 resolution: "redis-errors@npm:1.2.0" @@ -28632,6 +28756,22 @@ __metadata: languageName: node linkType: hard +"tsx@npm:4.16.2": + version: 4.16.2 + resolution: "tsx@npm:4.16.2" + dependencies: + esbuild: "npm:~0.21.5" + fsevents: "npm:~2.3.3" + get-tsconfig: "npm:^4.7.5" + dependenciesMeta: + fsevents: + optional: true + bin: + tsx: dist/cli.mjs + checksum: 10c0/9df52264f88be00ca473e7d7eda43bb038cc09028514996b864db78645e9cd297c71485f0fdd4985464d6dc46424f8bef9f8c4bd56692c4fcf4d71621ae21763 + languageName: node + linkType: hard + "tsx@npm:4.17.0": version: 4.17.0 resolution: "tsx@npm:4.17.0" @@ -29474,6 +29614,21 @@ __metadata: languageName: node linkType: hard +"vite-node@npm:2.0.4": + version: 2.0.4 + resolution: "vite-node@npm:2.0.4" + dependencies: + cac: "npm:^6.7.14" + debug: "npm:^4.3.5" + pathe: "npm:^1.1.2" + tinyrainbow: "npm:^1.2.0" + vite: "npm:^5.0.0" + bin: + vite-node: vite-node.mjs + checksum: 10c0/2689b05b391b59cf3d15e1e80884e9b054f2ca90b2150cc7a08b0f234e79e6750a28cc8d107a57f005185e759c3bc020030f687065317fc37fe169ce17f4cdb7 + languageName: node + linkType: hard + "vite-node@npm:2.0.5": version: 2.0.5 resolution: "vite-node@npm:2.0.5" @@ -29557,6 +29712,55 @@ __metadata: languageName: node linkType: hard +"vitest@npm:2.0.4": + version: 2.0.4 + resolution: "vitest@npm:2.0.4" + dependencies: + "@ampproject/remapping": "npm:^2.3.0" + "@vitest/expect": "npm:2.0.4" + "@vitest/pretty-format": "npm:^2.0.4" + "@vitest/runner": "npm:2.0.4" + "@vitest/snapshot": "npm:2.0.4" + "@vitest/spy": "npm:2.0.4" + "@vitest/utils": "npm:2.0.4" + chai: "npm:^5.1.1" + debug: "npm:^4.3.5" + execa: "npm:^8.0.1" + magic-string: "npm:^0.30.10" + pathe: "npm:^1.1.2" + std-env: "npm:^3.7.0" + tinybench: "npm:^2.8.0" + tinypool: "npm:^1.0.0" + tinyrainbow: "npm:^1.2.0" + vite: "npm:^5.0.0" + vite-node: "npm:2.0.4" + why-is-node-running: "npm:^2.3.0" + peerDependencies: + "@edge-runtime/vm": "*" + "@types/node": ^18.0.0 || >=20.0.0 + "@vitest/browser": 2.0.4 + "@vitest/ui": 2.0.4 + happy-dom: "*" + jsdom: "*" + peerDependenciesMeta: + "@edge-runtime/vm": + optional: true + "@types/node": + optional: true + "@vitest/browser": + optional: true + "@vitest/ui": + optional: true + happy-dom: + optional: true + jsdom: + optional: true + bin: + vitest: vitest.mjs + checksum: 10c0/139200d0bda3270fd00641e4bd5524f78a2b1fe9a3d4a0d5ba2b6ed08bbcf6f1e711cc4bfd8b0d823628a2fcab00f822bb210bd5bf3c6a9260fd6115ea085a3d + languageName: node + linkType: hard + "vitest@npm:2.0.5": version: 2.0.5 resolution: "vitest@npm:2.0.5" From 71160fe4cc6aeb90d177add38a9cca8b163775f3 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 14 Aug 2024 20:50:08 +0700 Subject: [PATCH 37/91] WIP: commit changes --- packages/internal/src/generate/graphqlCodeGen.ts | 1 + packages/uploads/src/prismaExtension.ts | 6 ++++-- 2 files changed, 5 insertions(+), 2 deletions(-) diff --git a/packages/internal/src/generate/graphqlCodeGen.ts b/packages/internal/src/generate/graphqlCodeGen.ts index 08b383f0ea36..73f51e42a4d6 100644 --- a/packages/internal/src/generate/graphqlCodeGen.ts +++ b/packages/internal/src/generate/graphqlCodeGen.ts @@ -288,6 +288,7 @@ async function getPluginConfig(side: CodegenSide) { JSONObject: 'Prisma.JsonObject', Time: side === CodegenSide.WEB ? 'string' : 'Date | string', Byte: 'Buffer', + File: 'File', }, // prevent type names being PetQueryQuery, RW generators already append // Query/Mutation/etc diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 710d018cf411..aa0248c9b408 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -22,8 +22,10 @@ export type UploadConfigForModel = { onFileSaved?: (filePath: string) => void | Promise } -export type UploadsConfig = - Record +export type UploadsConfig = Record< + MName, + UploadConfigForModel +> export const createUploadsExtension = ( config: UploadsConfig, From b2dfc189a5480b0d5c354e12a2e043c0ac62bd15 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 15 Aug 2024 15:49:15 +0700 Subject: [PATCH 38/91] Regenerate lock file --- yarn.lock | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/yarn.lock b/yarn.lock index 5675e4cc3f83..690534615d91 100644 --- a/yarn.lock +++ b/yarn.lock @@ -12093,6 +12093,13 @@ __metadata: languageName: node linkType: hard +"ansi-escapes@npm:^6.2.0": + version: 6.2.1 + resolution: "ansi-escapes@npm:6.2.1" + checksum: 10c0/a2c6f58b044be5f69662ee17073229b492daa2425a7fd99a665db6c22eab6e4ab42752807def7281c1c7acfed48f87f2362dda892f08c2c437f1b39c6b033103 + languageName: node + linkType: hard + "ansi-escapes@npm:^7.0.0": version: 7.0.0 resolution: "ansi-escapes@npm:7.0.0" From 3001a5ee0b46d6a76da17abd9bbe1611a26772f9 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 15 Aug 2024 18:29:03 +0700 Subject: [PATCH 39/91] Create processors, add some tests Add various stores --- packages/uploads/src/FileSystemStorage.ts | 30 +++--- packages/uploads/src/MemoryStorage.ts | 42 ++++++++ packages/uploads/src/StorageAdapter.ts | 17 ++- .../src/__tests__/createProcessors.test.ts | 101 ++++++++++++++++++ .../src/__tests__/prismaExtension.test.ts | 100 +---------------- packages/uploads/src/createProcessors.ts | 49 +++++++++ packages/uploads/src/prismaExtension.ts | 2 +- 7 files changed, 225 insertions(+), 116 deletions(-) create mode 100644 packages/uploads/src/MemoryStorage.ts create mode 100644 packages/uploads/src/__tests__/createProcessors.test.ts create mode 100644 packages/uploads/src/createProcessors.ts diff --git a/packages/uploads/src/FileSystemStorage.ts b/packages/uploads/src/FileSystemStorage.ts index 7da595b54be5..6a431b0d79b8 100644 --- a/packages/uploads/src/FileSystemStorage.ts +++ b/packages/uploads/src/FileSystemStorage.ts @@ -1,26 +1,26 @@ import fs from 'node:fs/promises' import path from 'node:path' -import type { StorageAdapter } from './StorageAdapter.js' -import type { SaveOptions } from './StorageAdapter.js' +import mime from 'mime-types' +import { ulid } from 'ulid' + +import type { AdapterOptions, SaveOptionsOverride, StorageAdapter } from './StorageAdapter.js' export class FileSystemStorage implements StorageAdapter { - // let basePath: string - // @TODO enable base path - // constructor({ basePath }) { - // this.basePath = basePath - // } + constructor(adapterOpts: AdapterOptions) { + this.adapterOpts = adapterOpts + } + + adapterOpts: AdapterOptions - async save(o_file: File, saveOpts: SaveOptions) { - // const file = new File([o_file], o_file.name) - // console.log(`👉 \n ~ FileSystemStorage ~ file:`, file.name) - console.log(`👉 \n ~ FileSystemStorage ~ file:`, await o_file.text()) + async save(file: File, saveOpts?: SaveOptionsOverride) { + const randomFileName = ulid() + const extension = mime.extension(file.type) + const location = path.join(saveOpts?.path || this.adapterOpts.baseDir, saveOpts?.fileName || randomFileName + `.${extension}`) + const nodeBuffer = await file.arrayBuffer() - const location = path.join(saveOpts.path, saveOpts.fileName + o_file.type) - const nodeBuffer = await o_file.arrayBuffer() - const extension = path.extname(o_file.name) - await fs.writeFile(`${location}.${extension}`, Buffer.from(nodeBuffer)) + await fs.writeFile(location, Buffer.from(nodeBuffer)) return { location } } async remove(filePath: string) { diff --git a/packages/uploads/src/MemoryStorage.ts b/packages/uploads/src/MemoryStorage.ts new file mode 100644 index 000000000000..1e6d4bd3e3ee --- /dev/null +++ b/packages/uploads/src/MemoryStorage.ts @@ -0,0 +1,42 @@ +import path from 'node:path' + +import mime from 'mime-types' +import { ulid } from 'ulid' + +import type { AdapterOptions, StorageAdapter } from './StorageAdapter.js' +import type { SaveOptionsOverride } from './StorageAdapter.js' + +export class MemoryStorage implements StorageAdapter { + constructor(adapterOpts: AdapterOptions) { + this.adapterOpts = adapterOpts + } + + adapterOpts: AdapterOptions + store: Record = {} + + async save(file: File, saveOpts?: SaveOptionsOverride) { + const randomFileName = ulid() + const extension = mime.extension(file.type) ? `.${mime.extension(file.type)}` : '' + const location = path.join(saveOpts?.path || this.adapterOpts.baseDir, saveOpts?.fileName || randomFileName + `${extension}`) + const nodeBuffer = await file.arrayBuffer() + + const result = `${location}` + this.store[result] = Buffer.from(nodeBuffer) + + return { + location: result + } + } + async remove(filePath: string) { + delete this.store[filePath] + } + +// Not sure about read method... should it be in the base class? + async read(filePath: string) { + return this.store[filePath] + } + + async clear() { + this.store = {} + } +} diff --git a/packages/uploads/src/StorageAdapter.ts b/packages/uploads/src/StorageAdapter.ts index ecc0f28ef58a..b1871ec314b4 100644 --- a/packages/uploads/src/StorageAdapter.ts +++ b/packages/uploads/src/StorageAdapter.ts @@ -10,13 +10,22 @@ export type AdapterResult = { location: string } -export type SaveOptions = { - fileName: string - path: string +export type SaveOptionsOverride = { + fileName?: string + path?: string +} + +export type AdapterOptions = { + baseDir: string } export abstract class StorageAdapter { - abstract save(file: File, saveOpts?: SaveOptions): Promise + adapterOpts: AdapterOptions + constructor(adapterOpts: AdapterOptions){ + this.adapterOpts = adapterOpts + } + + abstract save(file: File, saveOpts?: SaveOptionsOverride): Promise abstract remove(fileLocation: AdapterResult['location']): Promise // abstract replace(fileId: string, file: File): Promise } diff --git a/packages/uploads/src/__tests__/createProcessors.test.ts b/packages/uploads/src/__tests__/createProcessors.test.ts new file mode 100644 index 000000000000..dc9acbd578db --- /dev/null +++ b/packages/uploads/src/__tests__/createProcessors.test.ts @@ -0,0 +1,101 @@ + +import { describe, it, expect } from 'vitest' + +import { createUploadProcessors } from '../createProcessors.js' +import { MemoryStorage } from '../MemoryStorage.js' + +const memStore = new MemoryStorage({ + baseDir: '/memory_store_basedir' +}) + +const uploadsConfig = { + dumbo: { + fields: ['firstUpload', 'secondUpload'], + }, + dummy: { + fields: 'uploadField', + }, +} + +describe('Create processors', () => { + const processors = createUploadProcessors(memStore, uploadsConfig) + + it('should create processors with CapitalCased model name', () => { + expect(processors.processDumboUploads).toBeDefined() + expect(processors.processDummyUploads).toBeDefined() + }) + + it('Should replace file types with location strings', async() => { + const data = { + firstUpload: new File(['Meaow'], 'kitten.txt', { + type: 'text/plain', + }), + secondUpload: new File(['Woof'], 'puppy.txt', { + type: 'text/plain', + }), + } + + const result = await processors.processDumboUploads(data) + + expect(result.firstUpload).toMatch(/\/memory_store_basedir\/.*\.txt/) + expect(result.secondUpload).toMatch(/\/memory_store_basedir\/.*\.txt/) + + const firstContents = await memStore.read(result.firstUpload) + expect(firstContents.toString()).toBe('Meaow') + + const secondContents = await memStore.read(result.secondUpload) + expect(secondContents.toString()).toBe('Woof') + }) + + it('Should be able to override save options', async() => { + const data = { + uploadField: new File(['Hello'], 'hello.png', { + type: 'image/png', + }), + } + + const fileNameOverrideOnly = await processors.processDummyUploads(data, { + fileName: 'overridden.txt', + }) + + const pathOverrideOnly = await processors.processDummyUploads(data, { + path: '/bazinga', +}) + +const bothOverride = await processors.processDummyUploads(data, { + path: '/bazinga', + fileName: 'overridden.png', +}) + + expect(fileNameOverrideOnly.uploadField).toBe('/memory_store_basedir/overridden.txt') // 👈 overrode the extension too + + expect(pathOverrideOnly.uploadField).toMatch(/\/bazinga\/.*\.png/) + // Overriding path ignores the baseDir + expect(pathOverrideOnly.uploadField).not.toContain('memory_store_basedir') + + + expect(bothOverride.uploadField).toBe('/bazinga/overridden.png') +}) + +it('Should not add extension for unknown file type', async() => { + const data = { + uploadField: new File(['Hello'], 'hello', { + type: 'bazinga/unknown', + }), + } + + const noOverride = await processors.processDummyUploads(data) + + // No extension + expect(noOverride.uploadField).toMatch(/\/memory_store_basedir\/.*[^.]+$/); + + const withOverride = await processors.processDummyUploads(data, { + fileName: 'hello', + }) + expect(withOverride.uploadField).toMatch(/[^.]+$/); + expect(withOverride.uploadField).toBe('/memory_store_basedir/hello') + +}) + + +}) \ No newline at end of file diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index 7b7689ae69a2..24919f5a106a 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -33,14 +33,10 @@ vol.fromJSON({ describe('Uploads Prisma Extension', () => { const dummyUploadConfig = { fields: 'uploadField', - savePath: '/bazinga', - onFileSaved: vi.fn(), } const dumboUploadConfig = { fields: ['firstUpload', 'secondUpload'], - savePath: '/dumbo', - onFileSaved: vi.fn(), } const prismaClient = new PrismaClient().$extends( @@ -49,7 +45,10 @@ describe('Uploads Prisma Extension', () => { dummy: dummyUploadConfig, dumbo: dumboUploadConfig, }, - new FileSystemStorage(), + // Test with file system storage, and mock the fs calls + new FileSystemStorage({ + baseDir: '/tmp', + }), ), ) @@ -173,97 +172,6 @@ describe('Uploads Prisma Extension', () => { }) }) - it('will move file to new location with TUS uploads', async () => { - // Mock TUS metadata files - vi.mock('/tmp/tus-uploads/123.json', () => { - return { - metadata: { - filetype: 'image/png', - }, - } - }) - - vi.mock('/tmp/tus-uploads/ABCD.json', () => { - return { - metadata: { - filetype: 'application/pdf', - }, - } - }) - - const dumbo = await prismaClient.dumbo.create({ - data: { - firstUpload: - 'http://example.com/.redwood/functions/tusUploadEndpoint/123', - secondUpload: - 'http://example.com/.redwood/functions/tusUploadEndpoint/ABCD', - }, - }) - - expect(fs.copyFile).toHaveBeenCalledTimes(2) - expect(dumbo.firstUpload).toMatch(/dumbo\/.*\.png/) - expect(dumbo.secondUpload).toMatch(/dumbo\/.*\.pdf/) - }) - - it('will remove old file when updating with TUS uploads', async () => { - // Mock TUS metadata files - vi.mock('/tmp/tus-uploads/512356.json', () => { - return { - metadata: { - filetype: 'image/gif', - }, - } - }) - - const dumbo = await prismaClient.dumbo.create({ - data: { - firstUpload: - 'http://example.com/.redwood/functions/tusUploadEndpoint/123', - secondUpload: '', - }, - }) - - const originalPath = dumbo.firstUpload - - const dumbo2 = await prismaClient.dumbo.update({ - where: { id: dumbo.id }, - data: { - firstUpload: - 'http://example.com/.redwood/functions/tusUploadEndpoint/512356', - }, - }) - - expect(dumbo2.firstUpload).not.toEqual(originalPath) - expect(dumbo2.firstUpload).toMatch(/dumbo\/.*\.gif/) - - // And deletes it! - expect(fs.unlink).toHaveBeenCalledWith(originalPath) - }) - - it('will remove file when deleting with TUS uploads', async () => { - // Mock TUS metadata files - vi.mock('/tmp/tus-uploads/512356.json', () => { - return { - metadata: { - filetype: 'image/gif', - }, - } - }) - - const dummy = await prismaClient.dummy.create({ - data: { - uploadField: - 'http://example.com/.redwood/functions/tusUploadEndpoint/123', - }, - }) - - await prismaClient.dummy.delete({ - where: { id: dummy.id }, - }) - - expect(fs.unlink).toHaveBeenCalledWith(dummy.uploadField) - }) - }) describe('Result extensions', () => { it('will return a data URL for the file', async () => { diff --git a/packages/uploads/src/createProcessors.ts b/packages/uploads/src/createProcessors.ts new file mode 100644 index 000000000000..bcfe78ebc3fc --- /dev/null +++ b/packages/uploads/src/createProcessors.ts @@ -0,0 +1,49 @@ +import type { UploadsConfig } from "./prismaExtension.js" +import type { SaveOptionsOverride, StorageAdapter } from "./StorageAdapter.js" + +// Assumes you pass in the graphql type +type MakeFilesString = { + [K in keyof T]: T[K] extends File ? string : T[K] +} + +export const createUploadProcessors = ( + storage: StorageAdapter, + uploadConfig: UploadsConfig +) => { + type Processors = { + [K in keyof typeof uploadConfig]: >(data: T, overrideSaveOptions?: SaveOptionsOverride) => Promise> + } + + // @TODO TS: how do I get make it process${keyof UploadsConfig}Uploads so it autocompletes? + const processors: Processors = {} + + Object.keys(uploadConfig).forEach((model) => { + const currentModelUploadFields = Array.isArray(uploadConfig[model].fields) ? uploadConfig[model].fields : [uploadConfig[model].fields] + + const capitalCaseModel = `${model.charAt(0).toUpperCase() + model.slice(1)}` + processors[`process${capitalCaseModel}Uploads`] = async( + data, + overrideSaveOptions + ) => { + const updatedFields = {} as Record + for await (const field of currentModelUploadFields) { + if (data[field]) { + // @TODO deal with file lists + const file = data[field] as File + + // @TODO: should we automatically create a directory for each model? + // you can always override it in the saveOpts + const { location } = await storage.save(file, overrideSaveOptions) + + updatedFields[field] = location + } + } + return { + ...data, + ...updatedFields + } + } + }) + + return processors +} diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index aa0248c9b408..3dbd04cbfe12 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -22,7 +22,7 @@ export type UploadConfigForModel = { onFileSaved?: (filePath: string) => void | Promise } -export type UploadsConfig = Record< +export type UploadsConfig = Record< MName, UploadConfigForModel > From 4e6f56becdcc142383bd1a17db6ac8a12d3696c8 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 15 Aug 2024 18:35:25 +0700 Subject: [PATCH 40/91] Remove outdated tests --- packages/uploads/src/__tests__/fileMocks.js | 2 - .../src/__tests__/prismaExtension.test.ts | 134 +----------------- 2 files changed, 3 insertions(+), 133 deletions(-) delete mode 100644 packages/uploads/src/__tests__/fileMocks.js diff --git a/packages/uploads/src/__tests__/fileMocks.js b/packages/uploads/src/__tests__/fileMocks.js deleted file mode 100644 index 469745009c0b..000000000000 --- a/packages/uploads/src/__tests__/fileMocks.js +++ /dev/null @@ -1,2 +0,0 @@ -export const dataUrlPng = - '' diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index 24919f5a106a..c93e8926b638 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -1,12 +1,10 @@ -import fs from 'node:fs/promises' import { vol } from 'memfs' -import { describe, it, vi, expect } from 'vitest' +import { describe, it, vi } from 'vitest' import { FileSystemStorage } from '../FileSystemStorage.js' import { createUploadsExtension } from '../prismaExtension.js' -import { dataUrlPng } from './fileMocks.js' // @MARK: use the local prisma client import { PrismaClient } from './prisma-client' @@ -53,138 +51,12 @@ describe('Uploads Prisma Extension', () => { ) describe('Query extensions', () => { - it.only('will create a file with base64 encoded png', async () => { - const file = new File(['hello'], 'hello.txt', { - type: 'text/plain', - }) - console.log(`👉 \n ~ file:`, file) - console.log(`👉 \n ~ file arraybuf:`, await file.arrayBuffer()) - - const dum1 = await prismaClient.dummy.create({ - data: { - uploadField: file, - }, - }) - - expect(dummyUploadConfig.onFileSaved).toHaveBeenCalled() - expect(dum1.uploadField).toMatch(/bazinga\/.*\.png/) - expect(fs.writeFile).toHaveBeenCalledWith( - expect.stringMatching(/bazinga\/.*\.png/), - expect.anything(), // no need to check content here, makes test slow - ) - }) - - it('handles multiple upload fields', async () => { - const dumbo = await prismaClient.dumbo.create({ - data: { - firstUpload: dataUrlPng, - secondUpload: dataUrlPng, - }, - }) - - expect(dumbo.firstUpload).toMatch(/dumbo\/.*\.png/) - expect(dumbo.secondUpload).toMatch(/dumbo\/.*\.png/) - - expect(dumboUploadConfig.onFileSaved).toHaveBeenCalledTimes(2) - }) - - it('handles empty fields', async () => { - const emptyUploadFieldPromise = prismaClient.dummy.create({ - data: { - uploadField: '', - }, - }) - - await expect(emptyUploadFieldPromise).resolves.not.toThrow() - }) - - it('handles updates, and removes old files', async () => { - const dum1 = await prismaClient.dummy.create({ - data: { - uploadField: dataUrlPng, - }, - }) - - const originalPath = dum1.uploadField - - const dum2 = await prismaClient.dummy.update({ - where: { id: dum1.id }, - data: { - uploadField: dataUrlPng, - }, - }) - - expect(dum2.uploadField).not.toEqual(originalPath) - expect(dum2.uploadField).toMatch(/bazinga\/.*\.png/) - expect(fs.unlink).toHaveBeenCalledWith(originalPath) - }) - - it('handles deletes, and removes files', async () => { - const dum1 = await prismaClient.dummy.create({ - data: { - uploadField: dataUrlPng, - }, - }) - - await prismaClient.dummy.delete({ - where: { id: dum1.id }, - }) - - expect(fs.unlink).toHaveBeenCalledWith(dum1.uploadField) - }) - - it('supports custom file name and save path functions', async () => { - const customNameConfig = { - fields: 'firstUpload', - savePath: '/custom', - onFileSaved: vi.fn(), - fileName: (args) => { - // 👇 Using args here - return `my-name-is-dumbo-${args.data.id}` - }, - } - - const clientWithFileName = new PrismaClient().$extends( - createUploadsExtension( - { - dumbo: customNameConfig, - }, - new FileSystemStorage(), - ), - ) - - const dumbo = await clientWithFileName.dumbo.create({ - data: { - firstUpload: dataUrlPng, - secondUpload: '', - id: 55, - }, - }) - - expect(customNameConfig.onFileSaved).toHaveBeenCalled() - expect(dumbo.firstUpload).toBe('/custom/my-name-is-dumbo-55.png') - - // Delete it to clean up - await clientWithFileName.dumbo.delete({ - where: { - id: 55, - }, - }) - }) + it('not implemented yet after refactor') + }) describe('Result extensions', () => { it('will return a data URL for the file', async () => { - const res1 = await ( - await prismaClient.dummy.create({ - data: { - uploadField: dataUrlPng, - }, - }) - ).withDataUri() - - // Mocked in FS mocks - expect(res1.uploadField).toBe('_FILE_CONTENT') }) // @TODO Handle edge cases (file removed, data modified, etc.) From 808a023f0d399ee638dc4a85a02e39d00fdf8e57 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Fri, 16 Aug 2024 12:42:37 +0700 Subject: [PATCH 41/91] Try setting prisma override --- packages/uploads/prisma-override.d.ts | 9 +++ .../src/__tests__/createProcessors.test.ts | 73 ++++++++++--------- .../src/__tests__/prismaExtension.test.ts | 2 +- .../src/__tests__/unit-test-schema.prisma | 5 ++ packages/uploads/src/createProcessors.ts | 29 +++++--- packages/uploads/src/prismaExtension.ts | 10 ++- packages/uploads/tsconfig.json | 5 +- 7 files changed, 81 insertions(+), 52 deletions(-) create mode 100644 packages/uploads/prisma-override.d.ts diff --git a/packages/uploads/prisma-override.d.ts b/packages/uploads/prisma-override.d.ts new file mode 100644 index 000000000000..ec14d355db25 --- /dev/null +++ b/packages/uploads/prisma-override.d.ts @@ -0,0 +1,9 @@ +// Locally, within this project we override the type for @prisma/client to the one we generate locally +// This is so that we get accurate types (rather than the default anys) - and when the prismaExtension runs +// it will still use the types from '@prisma/client' which points to the user's prisma client and not ours + +import { PrismaClient as LocalPrismaClient } from './src/__tests__/prisma-client/index.d.ts' + +declare module '@prisma/client' { + export class PrismaClient extends LocalPrismaClient {} +} diff --git a/packages/uploads/src/__tests__/createProcessors.test.ts b/packages/uploads/src/__tests__/createProcessors.test.ts index dc9acbd578db..d7e2c018644e 100644 --- a/packages/uploads/src/__tests__/createProcessors.test.ts +++ b/packages/uploads/src/__tests__/createProcessors.test.ts @@ -3,12 +3,15 @@ import { describe, it, expect } from 'vitest' import { createUploadProcessors } from '../createProcessors.js' import { MemoryStorage } from '../MemoryStorage.js' +import type { UploadsConfig } from '../prismaExtension.js' const memStore = new MemoryStorage({ - baseDir: '/memory_store_basedir' + baseDir: '/memory_store_basedir', }) -const uploadsConfig = { +// @TODO(TS): How can I make this accept not all model names? +// This is error-ing out because it wants all the models in my prisma client +const uploadsConfig: UploadsConfig = { dumbo: { fields: ['firstUpload', 'secondUpload'], }, @@ -25,7 +28,7 @@ describe('Create processors', () => { expect(processors.processDummyUploads).toBeDefined() }) - it('Should replace file types with location strings', async() => { + it('Should replace file types with location strings', async () => { const data = { firstUpload: new File(['Meaow'], 'kitten.txt', { type: 'text/plain', @@ -47,7 +50,7 @@ describe('Create processors', () => { expect(secondContents.toString()).toBe('Woof') }) - it('Should be able to override save options', async() => { + it('Should be able to override save options', async () => { const data = { uploadField: new File(['Hello'], 'hello.png', { type: 'image/png', @@ -56,46 +59,44 @@ describe('Create processors', () => { const fileNameOverrideOnly = await processors.processDummyUploads(data, { fileName: 'overridden.txt', - }) - - const pathOverrideOnly = await processors.processDummyUploads(data, { - path: '/bazinga', -}) + }) -const bothOverride = await processors.processDummyUploads(data, { - path: '/bazinga', - fileName: 'overridden.png', -}) + const pathOverrideOnly = await processors.processDummyUploads(data, { + path: '/bazinga', + }) - expect(fileNameOverrideOnly.uploadField).toBe('/memory_store_basedir/overridden.txt') // 👈 overrode the extension too + const bothOverride = await processors.processDummyUploads(data, { + path: '/bazinga', + fileName: 'overridden.png', + }) - expect(pathOverrideOnly.uploadField).toMatch(/\/bazinga\/.*\.png/) - // Overriding path ignores the baseDir - expect(pathOverrideOnly.uploadField).not.toContain('memory_store_basedir') + expect(fileNameOverrideOnly.uploadField).toBe( + '/memory_store_basedir/overridden.txt', + ) // 👈 overrode the extension too + expect(pathOverrideOnly.uploadField).toMatch(/\/bazinga\/.*\.png/) + // Overriding path ignores the baseDir + expect(pathOverrideOnly.uploadField).not.toContain('memory_store_basedir') - expect(bothOverride.uploadField).toBe('/bazinga/overridden.png') -}) + expect(bothOverride.uploadField).toBe('/bazinga/overridden.png') + }) -it('Should not add extension for unknown file type', async() => { - const data = { - uploadField: new File(['Hello'], 'hello', { - type: 'bazinga/unknown', - }), - } + it('Should not add extension for unknown file type', async () => { + const data = { + uploadField: new File(['Hello'], 'hello', { + type: 'bazinga/unknown', + }), + } - const noOverride = await processors.processDummyUploads(data) + const noOverride = await processors.processDummyUploads(data) - // No extension - expect(noOverride.uploadField).toMatch(/\/memory_store_basedir\/.*[^.]+$/); + // No extension + expect(noOverride.uploadField).toMatch(/\/memory_store_basedir\/.*[^.]+$/) - const withOverride = await processors.processDummyUploads(data, { - fileName: 'hello', + const withOverride = await processors.processDummyUploads(data, { + fileName: 'hello', + }) + expect(withOverride.uploadField).toMatch(/[^.]+$/) + expect(withOverride.uploadField).toBe('/memory_store_basedir/hello') }) - expect(withOverride.uploadField).toMatch(/[^.]+$/); - expect(withOverride.uploadField).toBe('/memory_store_basedir/hello') - -}) - - }) \ No newline at end of file diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index c93e8926b638..7236510ff9d5 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -51,7 +51,7 @@ describe('Uploads Prisma Extension', () => { ) describe('Query extensions', () => { - it('not implemented yet after refactor') + it('not implemented yet after refactor', async () => {}) }) diff --git a/packages/uploads/src/__tests__/unit-test-schema.prisma b/packages/uploads/src/__tests__/unit-test-schema.prisma index e4c606541958..905afc366007 100644 --- a/packages/uploads/src/__tests__/unit-test-schema.prisma +++ b/packages/uploads/src/__tests__/unit-test-schema.prisma @@ -18,3 +18,8 @@ model Dumbo { firstUpload String secondUpload String } + +model NoUploadFields { + id Int @id @default(autoincrement()) + name String +} \ No newline at end of file diff --git a/packages/uploads/src/createProcessors.ts b/packages/uploads/src/createProcessors.ts index bcfe78ebc3fc..c35ddd9dced3 100644 --- a/packages/uploads/src/createProcessors.ts +++ b/packages/uploads/src/createProcessors.ts @@ -10,21 +10,32 @@ export const createUploadProcessors = ( storage: StorageAdapter, uploadConfig: UploadsConfig ) => { + type modelNamesInUploadConfig = keyof typeof uploadConfig + + type uploadProcessorNames = + `process${Capitalize}Uploads` type Processors = { - [K in keyof typeof uploadConfig]: >(data: T, overrideSaveOptions?: SaveOptionsOverride) => Promise> + [K in uploadProcessorNames]: >( + data: T, + overrideSaveOptions?: SaveOptionsOverride, + ) => Promise> } - // @TODO TS: how do I get make it process${keyof UploadsConfig}Uploads so it autocompletes? - const processors: Processors = {} + const processors = {} as Processors Object.keys(uploadConfig).forEach((model) => { - const currentModelUploadFields = Array.isArray(uploadConfig[model].fields) ? uploadConfig[model].fields : [uploadConfig[model].fields] + const modelKey = model as keyof typeof uploadConfig + + const currentModelUploadFields = Array.isArray( + uploadConfig[modelKey].fields, + ) + ? uploadConfig[modelKey].fields + : [uploadConfig[modelKey].fields] const capitalCaseModel = `${model.charAt(0).toUpperCase() + model.slice(1)}` - processors[`process${capitalCaseModel}Uploads`] = async( - data, - overrideSaveOptions - ) => { + const processorKey = `process${capitalCaseModel}Uploads` as keyof Processors + + processors[processorKey] = async (data, overrideSaveOptions) => { const updatedFields = {} as Record for await (const field of currentModelUploadFields) { if (data[field]) { @@ -40,7 +51,7 @@ export const createUploadProcessors = ( } return { ...data, - ...updatedFields + ...updatedFields, } } }) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 3dbd04cbfe12..7c3b659391f8 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -3,6 +3,10 @@ import { Prisma } from '@prisma/client' import type * as runtime from '@prisma/client/runtime/library' import { ulid } from 'ulid' +// @TODO(TS): UploadsConfig behaves differently here.. probably +// the prisma-override not quite there yet? +// import { PrismaClient } from './__tests__/prisma-client/index.js' +// import { Prisma } from './__tests__/prisma-client/index.js' import { fileToDataUri } from './fileSave.utils.js' import type { StorageAdapter } from './StorageAdapter.js' @@ -22,10 +26,8 @@ export type UploadConfigForModel = { onFileSaved?: (filePath: string) => void | Promise } -export type UploadsConfig = Record< - MName, - UploadConfigForModel -> +export type UploadsConfig = + Record export const createUploadsExtension = ( config: UploadsConfig, diff --git a/packages/uploads/tsconfig.json b/packages/uploads/tsconfig.json index 43d163492556..3484bfda0d5c 100644 --- a/packages/uploads/tsconfig.json +++ b/packages/uploads/tsconfig.json @@ -7,13 +7,14 @@ "rootDir": "src", "outDir": "dist" }, - "include": ["src"], + "include": ["src", "prisma-override.d.ts"], + "references": [ { "path": "../project-config" }, { "path": "../framework-tools" - }, + } ] } From 1c99b1404aaf345a9a9ff460e6908786e84ac2d5 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Fri, 16 Aug 2024 13:32:26 +0700 Subject: [PATCH 42/91] Add file list processor --- .../src/__tests__/createProcessors.test.ts | 41 +++++++++++++++++-- packages/uploads/src/createProcessors.ts | 18 ++++++++ 2 files changed, 56 insertions(+), 3 deletions(-) diff --git a/packages/uploads/src/__tests__/createProcessors.test.ts b/packages/uploads/src/__tests__/createProcessors.test.ts index d7e2c018644e..060a995259e1 100644 --- a/packages/uploads/src/__tests__/createProcessors.test.ts +++ b/packages/uploads/src/__tests__/createProcessors.test.ts @@ -1,7 +1,9 @@ - import { describe, it, expect } from 'vitest' -import { createUploadProcessors } from '../createProcessors.js' +import { + createFileListProcessor, + createUploadProcessors, +} from '../createProcessors.js' import { MemoryStorage } from '../MemoryStorage.js' import type { UploadsConfig } from '../prismaExtension.js' @@ -99,4 +101,37 @@ describe('Create processors', () => { expect(withOverride.uploadField).toMatch(/[^.]+$/) expect(withOverride.uploadField).toBe('/memory_store_basedir/hello') }) -}) \ No newline at end of file +}) +// FileLists +// Problem is - in the database world, a string[] is not a thing +// so we need a generic way of doing this +describe('FileList processing', () => { + const fileListProcessor = createFileListProcessor(memStore) + + const notPrismaData = [ + new File(['Hello'], 'hello.png', { + type: 'image/png', + }), + new File(['World'], 'world.jpeg', { + type: 'image/jpeg', + }), + ] + + it('Should handle FileLists', async () => { + const result = await fileListProcessor(notPrismaData) + + expect(result).toHaveLength(2) + expect(result[0]).toMatch(/\/memory_store_basedir\/.*\.png/) + expect(result[1]).toMatch(/\/memory_store_basedir\/.*\.jpeg/) + }) + + it('Should handle FileLists with SaveOptions', async () => { + const result = await fileListProcessor(notPrismaData, { + path: '/bazinga_not_mem_store', + }) + + expect(result).toHaveLength(2) + expect(result[0]).toMatch(/\/bazinga_not_mem_store\/.*\.png/) + expect(result[1]).toMatch(/\/bazinga_not_mem_store\/.*\.jpeg/) + }) +}) diff --git a/packages/uploads/src/createProcessors.ts b/packages/uploads/src/createProcessors.ts index c35ddd9dced3..2505a1128cc5 100644 --- a/packages/uploads/src/createProcessors.ts +++ b/packages/uploads/src/createProcessors.ts @@ -6,6 +6,24 @@ type MakeFilesString = { [K in keyof T]: T[K] extends File ? string : T[K] } +export const createFileListProcessor = (storage: StorageAdapter) => { + return async (files: File[], pathOverrideOnly?: { path?: string }) => { + const locations = await Promise.all( + files.map(async (file) => { + const { location } = await storage.save(file, pathOverrideOnly) + return location + }) + ) + + return locations + } + +} + +/* +This creates a processor for each model in the uploads config (i.e. tied to a model in the prisma schema) +The processor will only handle single file uploads, not file lists. +*/ export const createUploadProcessors = ( storage: StorageAdapter, uploadConfig: UploadsConfig From 964433982ba46415ce549a4540f959c18b5e91fd Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Fri, 16 Aug 2024 17:30:25 +0700 Subject: [PATCH 43/91] WIP --- packages/uploads/src/FileSystemStorage.ts | 20 +-- packages/uploads/src/MemoryStorage.ts | 26 ++-- packages/uploads/src/StorageAdapter.ts | 11 +- .../src/__tests__/createProcessors.test.ts | 17 +- .../src/__tests__/prismaExtension.test.ts | 147 ++++++++++++++---- packages/uploads/src/createProcessors.ts | 21 +-- packages/uploads/src/prismaExtension.ts | 112 ++++--------- packages/uploads/src/setup.ts | 27 ++++ 8 files changed, 229 insertions(+), 152 deletions(-) create mode 100644 packages/uploads/src/setup.ts diff --git a/packages/uploads/src/FileSystemStorage.ts b/packages/uploads/src/FileSystemStorage.ts index 6a431b0d79b8..85ef8e1aeb49 100644 --- a/packages/uploads/src/FileSystemStorage.ts +++ b/packages/uploads/src/FileSystemStorage.ts @@ -4,22 +4,22 @@ import path from 'node:path' import mime from 'mime-types' import { ulid } from 'ulid' -import type { AdapterOptions, SaveOptionsOverride, StorageAdapter } from './StorageAdapter.js' - -export class FileSystemStorage implements StorageAdapter { - constructor(adapterOpts: AdapterOptions) { - this.adapterOpts = adapterOpts - } - - adapterOpts: AdapterOptions +import type { SaveOptionsOverride } from './StorageAdapter.js' +import { StorageAdapter } from './StorageAdapter.js' +export class FileSystemStorage + extends StorageAdapter + implements StorageAdapter +{ async save(file: File, saveOpts?: SaveOptionsOverride) { const randomFileName = ulid() const extension = mime.extension(file.type) - const location = path.join(saveOpts?.path || this.adapterOpts.baseDir, saveOpts?.fileName || randomFileName + `.${extension}`) + const location = path.join( + saveOpts?.path || this.adapterOpts.baseDir, + saveOpts?.fileName || randomFileName + `.${extension}`, + ) const nodeBuffer = await file.arrayBuffer() - await fs.writeFile(location, Buffer.from(nodeBuffer)) return { location } } diff --git a/packages/uploads/src/MemoryStorage.ts b/packages/uploads/src/MemoryStorage.ts index 1e6d4bd3e3ee..c478c3bf3944 100644 --- a/packages/uploads/src/MemoryStorage.ts +++ b/packages/uploads/src/MemoryStorage.ts @@ -3,35 +3,35 @@ import path from 'node:path' import mime from 'mime-types' import { ulid } from 'ulid' -import type { AdapterOptions, StorageAdapter } from './StorageAdapter.js' +import { StorageAdapter } from './StorageAdapter.js' import type { SaveOptionsOverride } from './StorageAdapter.js' -export class MemoryStorage implements StorageAdapter { - constructor(adapterOpts: AdapterOptions) { - this.adapterOpts = adapterOpts - } - - adapterOpts: AdapterOptions +export class MemoryStorage extends StorageAdapter implements StorageAdapter { store: Record = {} async save(file: File, saveOpts?: SaveOptionsOverride) { - const randomFileName = ulid() - const extension = mime.extension(file.type) ? `.${mime.extension(file.type)}` : '' - const location = path.join(saveOpts?.path || this.adapterOpts.baseDir, saveOpts?.fileName || randomFileName + `${extension}`) + const fileName = saveOpts?.fileName || ulid() + const extension = mime.extension(file.type) + ? `.${mime.extension(file.type)}` + : '' + const location = path.join( + saveOpts?.path || this.adapterOpts.baseDir, + fileName + `${extension}`, + ) const nodeBuffer = await file.arrayBuffer() const result = `${location}` this.store[result] = Buffer.from(nodeBuffer) return { - location: result + location: result, } } async remove(filePath: string) { - delete this.store[filePath] + delete this.store[filePath] } -// Not sure about read method... should it be in the base class? + // Not sure about read method... should it be in the base class? async read(filePath: string) { return this.store[filePath] } diff --git a/packages/uploads/src/StorageAdapter.ts b/packages/uploads/src/StorageAdapter.ts index b1871ec314b4..f3883b17b935 100644 --- a/packages/uploads/src/StorageAdapter.ts +++ b/packages/uploads/src/StorageAdapter.ts @@ -21,11 +21,18 @@ export type AdapterOptions = { export abstract class StorageAdapter { adapterOpts: AdapterOptions - constructor(adapterOpts: AdapterOptions){ + constructor(adapterOpts: AdapterOptions) { this.adapterOpts = adapterOpts } - abstract save(file: File, saveOpts?: SaveOptionsOverride): Promise + getAdapterOptions() { + return this.adapterOpts + } + + abstract save( + file: File, + saveOpts?: SaveOptionsOverride, + ): Promise abstract remove(fileLocation: AdapterResult['location']): Promise // abstract replace(fileId: string, file: File): Promise } diff --git a/packages/uploads/src/__tests__/createProcessors.test.ts b/packages/uploads/src/__tests__/createProcessors.test.ts index 060a995259e1..e2624e7e8975 100644 --- a/packages/uploads/src/__tests__/createProcessors.test.ts +++ b/packages/uploads/src/__tests__/createProcessors.test.ts @@ -23,7 +23,7 @@ const uploadsConfig: UploadsConfig = { } describe('Create processors', () => { - const processors = createUploadProcessors(memStore, uploadsConfig) + const processors = createUploadProcessors(uploadsConfig, memStore) it('should create processors with CapitalCased model name', () => { expect(processors.processDumboUploads).toBeDefined() @@ -42,8 +42,11 @@ describe('Create processors', () => { const result = await processors.processDumboUploads(data) - expect(result.firstUpload).toMatch(/\/memory_store_basedir\/.*\.txt/) - expect(result.secondUpload).toMatch(/\/memory_store_basedir\/.*\.txt/) + // Location strings in this format: {baseDir/{model}-{field}-{ulid}.{ext} + expect(result.firstUpload).toMatch(/\/memory_store_basedir\/dumbo-*.*\.txt/) + expect(result.secondUpload).toMatch( + /\/memory_store_basedir\/dumbo-*.*\.txt/, + ) const firstContents = await memStore.read(result.firstUpload) expect(firstContents.toString()).toBe('Meaow') @@ -60,7 +63,7 @@ describe('Create processors', () => { } const fileNameOverrideOnly = await processors.processDummyUploads(data, { - fileName: 'overridden.txt', + fileName: 'overridden', }) const pathOverrideOnly = await processors.processDummyUploads(data, { @@ -69,12 +72,12 @@ describe('Create processors', () => { const bothOverride = await processors.processDummyUploads(data, { path: '/bazinga', - fileName: 'overridden.png', + fileName: 'overridden', }) expect(fileNameOverrideOnly.uploadField).toBe( - '/memory_store_basedir/overridden.txt', - ) // 👈 overrode the extension too + '/memory_store_basedir/overridden.png', + ) expect(pathOverrideOnly.uploadField).toMatch(/\/bazinga\/.*\.png/) // Overriding path ignores the baseDir diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index 7236510ff9d5..f891dd080aa2 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -1,9 +1,10 @@ +import fs from 'node:fs/promises' -import { vol } from 'memfs' -import { describe, it, vi } from 'vitest' +import { describe, it, vi, expect, beforeEach } from 'vitest' import { FileSystemStorage } from '../FileSystemStorage.js' -import { createUploadsExtension } from '../prismaExtension.js' +import type { UploadsConfig } from '../prismaExtension.js' +import { setupUploads } from '../setup.js' // @MARK: use the local prisma client import { PrismaClient } from './prisma-client' @@ -12,22 +13,13 @@ vi.mock('node:fs/promises', () => ({ default: { writeFile: vi.fn(), unlink: vi.fn(), - readFile: vi.fn((path, encoding) => { - if (encoding === 'base64') { - return 'BASE64_FILE_CONTENT' - } - + readFile: vi.fn(() => { return 'MOCKED_FILE_CONTENT' }), copyFile: vi.fn(), }, })) -vol.fromJSON({ - '/tmp/tus-uploads/123.json': '{}', - '/tmp/tus-uploads/ABCD.json': '{}', -}) - describe('Uploads Prisma Extension', () => { const dummyUploadConfig = { fields: 'uploadField', @@ -37,28 +29,127 @@ describe('Uploads Prisma Extension', () => { fields: ['firstUpload', 'secondUpload'], } - const prismaClient = new PrismaClient().$extends( - createUploadsExtension( - { - dummy: dummyUploadConfig, - dumbo: dumboUploadConfig, - }, - // Test with file system storage, and mock the fs calls - new FileSystemStorage({ - baseDir: '/tmp', - }), - ), + const uploadConfig: UploadsConfig = { + dummy: dummyUploadConfig, + dumbo: dumboUploadConfig, + } + + const { prismaExtension, uploadsProcessors } = setupUploads( + uploadConfig, + new FileSystemStorage({ + baseDir: '/tmp', + }), ) + const prismaClient = new PrismaClient().$extends(prismaExtension) + describe('Query extensions', () => { - it('not implemented yet after refactor', async () => {}) - }) + beforeEach(() => { + vi.resetAllMocks() + }) + + const sampleFile = new File(['heres-some-content'], 'dummy.txt', { + type: 'text/plain', + }) + describe('create', () => { + it('create will save files', async () => { + const processedData = await uploadsProcessors.processDummyUploads({ + uploadField: sampleFile, + }) - describe('Result extensions', () => { - it('will return a data URL for the file', async () => { + expect(fs.writeFile).toHaveBeenCalled() + const dummy = await prismaClient.dummy.create({ + data: processedData, + }) + + expect(dummy).toMatchObject({ + uploadField: expect.stringMatching(/\/tmp\/.*\.txt$/), + }) + }) + + it('will remove the file if the create fails', async () => { + try { + await prismaClient.dumbo.create({ + data: { + firstUpload: '/tmp/first.txt', + secondUpload: '/bazinga/second.txt', + // @ts-expect-error Checking the error here + id: 'this-is-the-incorrect-type', + }, + }) + } catch { + expect(fs.unlink).toHaveBeenNthCalledWith(1, '/tmp/first.txt') + expect(fs.unlink).toHaveBeenNthCalledWith(2, '/bazinga/second.txt') + } + + expect.assertions(2) + }) + }) + + describe('createMany', () => { + it('createMany will remove files if all the create fails', async () => { + try { + await prismaClient.dumbo.createMany({ + data: [ + { + firstUpload: '/one/first.txt', + secondUpload: '/one/second.txt', + id: 'break', + }, + { + firstUpload: '/two/first.txt', + secondUpload: '/two/second.txt', + id: 'break2', + }, + ], + }) + } catch (e) { + expect(fs.unlink).toHaveBeenCalledTimes(4) + expect(fs.unlink).toHaveBeenNthCalledWith(1, '/one/first.txt') + expect(fs.unlink).toHaveBeenNthCalledWith(2, '/one/second.txt') + expect(fs.unlink).toHaveBeenNthCalledWith(3, '/two/first.txt') + expect(fs.unlink).toHaveBeenNthCalledWith(4, '/two/second.txt') + } + + expect.assertions(4) + }) + + it('createMany will remove files from only the creates that fail', async () => { + try { + await prismaClient.dumbo.createMany({ + data: [ + // This one will go through + { + firstUpload: '/one/first.txt', + secondUpload: '/one/second.txt', + }, + { + firstUpload: '/two/first.txt', + secondUpload: '/two/second.txt', + id: 'break2', + }, + ], + }) + } catch (e) { + console.log(e) + expect(fs.unlink).toHaveBeenCalledTimes(2) + expect(fs.unlink).toHaveBeenNthCalledWith(1, '/two/first.txt') + expect(fs.unlink).toHaveBeenNthCalledWith(2, '/two/second.txt') + } + + expect.assertions(4) + }) }) + // describe('update', () => {}) + + // describe('delete', () => {}) + }) + + describe('Result extensions', () => { + it('will return a data URL for the file', async () => {}) + // @TODO Handle edge cases (file removed, data modified, etc.) // it('if file is not found, will throw an error', async () => {}) // it('if saved file is not a path, will throw an error', async () => {}) diff --git a/packages/uploads/src/createProcessors.ts b/packages/uploads/src/createProcessors.ts index 2505a1128cc5..af29a18a32c9 100644 --- a/packages/uploads/src/createProcessors.ts +++ b/packages/uploads/src/createProcessors.ts @@ -1,5 +1,7 @@ -import type { UploadsConfig } from "./prismaExtension.js" -import type { SaveOptionsOverride, StorageAdapter } from "./StorageAdapter.js" +import { ulid } from 'ulid' + +import type { UploadsConfig } from './prismaExtension.js' +import type { SaveOptionsOverride, StorageAdapter } from './StorageAdapter.js' // Assumes you pass in the graphql type type MakeFilesString = { @@ -12,12 +14,11 @@ export const createFileListProcessor = (storage: StorageAdapter) => { files.map(async (file) => { const { location } = await storage.save(file, pathOverrideOnly) return location - }) + }), ) return locations } - } /* @@ -25,15 +26,17 @@ This creates a processor for each model in the uploads config (i.e. tied to a mo The processor will only handle single file uploads, not file lists. */ export const createUploadProcessors = ( + uploadConfig: UploadsConfig, storage: StorageAdapter, - uploadConfig: UploadsConfig ) => { type modelNamesInUploadConfig = keyof typeof uploadConfig type uploadProcessorNames = `process${Capitalize}Uploads` + type Processors = { [K in uploadProcessorNames]: >( + // @TODO(TS): T should be the type of the model data: T, overrideSaveOptions?: SaveOptionsOverride, ) => Promise> @@ -57,12 +60,12 @@ export const createUploadProcessors = ( const updatedFields = {} as Record for await (const field of currentModelUploadFields) { if (data[field]) { - // @TODO deal with file lists const file = data[field] as File - // @TODO: should we automatically create a directory for each model? - // you can always override it in the saveOpts - const { location } = await storage.save(file, overrideSaveOptions) + const saveOptions = overrideSaveOptions || { + fileName: `${model}-${field}-${ulid()}`, + } + const { location } = await storage.save(file, saveOptions) updatedFields[field] = location } diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 7c3b659391f8..36e22291c340 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -1,7 +1,6 @@ import { PrismaClient } from '@prisma/client' import { Prisma } from '@prisma/client' import type * as runtime from '@prisma/client/runtime/library' -import { ulid } from 'ulid' // @TODO(TS): UploadsConfig behaves differently here.. probably // the prisma-override not quite there yet? @@ -20,6 +19,7 @@ type FilterOutDollarPrefixed = T extends `$${string}` type ModelNames = FilterOutDollarPrefixed export type UploadConfigForModel = { + // @TODO(TS): I want the fields here to be the fields of the model fields: string[] | string savePath?: ((args: unknown) => string) | string fileName?: (args: unknown) => string @@ -84,6 +84,16 @@ export const createUploadsExtension = ( : [modelConfig.fields] queryExtends[modelName] = { + async create({ query, args }) { + try { + const result = await query(args) + return result + } catch (e) { + // If the create fails, we need to delete the uploaded files + await removeUploadedFiles(uploadFields, args) + throw e + } + }, async update({ query, model, args }) { await deleteUpload( { @@ -97,25 +107,17 @@ export const createUploadsExtension = ( storageAdapter, ) - const uploadArgs = await saveUploads( - uploadFields, - args, - modelConfig, - storageAdapter, - ) - - return query(uploadArgs) + // Same as create 👇 + try { + const result = await query(args) + return result + } catch (e) { + // If the create fails, we need to delete the uploaded files + await removeUploadedFiles(uploadFields, args) + throw e + } }, - async create({ query, args }) { - const uploadArgs = await saveUploads( - uploadFields, - args, - modelConfig, - storageAdapter, - ) - return query(uploadArgs) - }, async delete({ model, query, args }) { await deleteUpload( { @@ -165,72 +167,16 @@ export const createUploadsExtension = ( result: resultExtends, }) }) -} -/** - * Returns new args to use in create or update. - * - * Pass this to the query function! - */ -async function saveUploads( - uploadFields: string[], - args: runtime.JsArgs & { - data?: { - [key: string]: runtime.JsInputValue | File - } - }, - modelConfig: UploadConfigForModel, - storageAdapter: StorageAdapter, -) { - const fieldsToUpdate: { - [key: string]: string - } = {} - - if (!args.data) { - throw new Error('No data in prisma query') - } - - // For each upload property, we need to:z - // 1. save the file to the file system (path or name from config) - // 2. replace the value of the field - for await (const field of uploadFields) { - const uploadFile = args.data[field] as File - console.log(`👉 \n ~ uploadFile:`, uploadFile) - - if (!uploadFile) { - continue - } - - const fileName = - modelConfig.fileName && typeof modelConfig.fileName === 'function' - ? modelConfig.fileName(args) - : ulid() - - const saveDir = - typeof modelConfig.savePath === 'function' - ? modelConfig.savePath(args) - : modelConfig.savePath || 'web/public/uploads' - - const savedFile = await storageAdapter.save(uploadFile, { - fileName, - path: saveDir, - }) - - // @TODO should we return location or fileId? - fieldsToUpdate[field] = savedFile.location - - // Call the onFileSaved callback - // Having it here means it'll always trigger whether create/update - if (modelConfig.onFileSaved) { - await modelConfig.onFileSaved(savedFile.location) + // @TODO(TS): According to TS, data could be a non-object... + // Setting args to JsArgs causes errors. This could be a legit issue + async function removeUploadedFiles(uploadFields: string[], args: any) { + for await (const field of uploadFields) { + const uploadLocation = args.data?.[field] as string + if (uploadLocation) { + console.log('Removing file >>>', uploadLocation) + await storageAdapter.remove(uploadLocation) + } } } - - // Can't spread according to TS - const newData = Object.assign(args.data, fieldsToUpdate) - - return { - ...args, - data: newData, - } } diff --git a/packages/uploads/src/setup.ts b/packages/uploads/src/setup.ts new file mode 100644 index 000000000000..e21042326488 --- /dev/null +++ b/packages/uploads/src/setup.ts @@ -0,0 +1,27 @@ +import { + createFileListProcessor, + createUploadProcessors, +} from './createProcessors.js' +import { + createUploadsExtension, + type UploadsConfig, +} from './prismaExtension.js' +import type { StorageAdapter } from './StorageAdapter.js' + +export const setupUploads = ( + uploadsConfig: UploadsConfig, + storageAdapter: StorageAdapter, +) => { + const prismaExtension = createUploadsExtension(uploadsConfig, storageAdapter) + const uploadsProcessors = createUploadProcessors( + uploadsConfig, + storageAdapter, + ) + const fileListProcessor = createFileListProcessor(storageAdapter) + + return { + prismaExtension, + uploadsProcessors, + fileListProcessor, + } +} From 52c57c98f710499fadd459438a1426e256e53e3a Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Fri, 16 Aug 2024 17:36:26 +0700 Subject: [PATCH 44/91] Bit of cleanup --- packages/uploads/src/FileSystemStorage.ts | 7 +++++-- packages/uploads/src/MemoryStorage.ts | 6 +++--- 2 files changed, 8 insertions(+), 5 deletions(-) diff --git a/packages/uploads/src/FileSystemStorage.ts b/packages/uploads/src/FileSystemStorage.ts index 85ef8e1aeb49..0b3312effbec 100644 --- a/packages/uploads/src/FileSystemStorage.ts +++ b/packages/uploads/src/FileSystemStorage.ts @@ -12,11 +12,14 @@ export class FileSystemStorage implements StorageAdapter { async save(file: File, saveOpts?: SaveOptionsOverride) { - const randomFileName = ulid() + const fileName = saveOpts?.fileName || ulid() const extension = mime.extension(file.type) + ? `.${mime.extension(file.type)}` + : '' + const location = path.join( saveOpts?.path || this.adapterOpts.baseDir, - saveOpts?.fileName || randomFileName + `.${extension}`, + fileName + `${extension}`, ) const nodeBuffer = await file.arrayBuffer() diff --git a/packages/uploads/src/MemoryStorage.ts b/packages/uploads/src/MemoryStorage.ts index c478c3bf3944..64d667a1f9cc 100644 --- a/packages/uploads/src/MemoryStorage.ts +++ b/packages/uploads/src/MemoryStorage.ts @@ -14,17 +14,17 @@ export class MemoryStorage extends StorageAdapter implements StorageAdapter { const extension = mime.extension(file.type) ? `.${mime.extension(file.type)}` : '' + const location = path.join( saveOpts?.path || this.adapterOpts.baseDir, fileName + `${extension}`, ) const nodeBuffer = await file.arrayBuffer() - const result = `${location}` - this.store[result] = Buffer.from(nodeBuffer) + this.store[location] = Buffer.from(nodeBuffer) return { - location: result, + location, } } async remove(filePath: string) { From 540bdcc7ffe9732fafcf413a8b9c930995631ada Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 10:08:33 +0700 Subject: [PATCH 45/91] WIP: Start on signature generation --- .../src/__tests__/generateSignature.test.ts | 14 ++++++++++++++ packages/uploads/src/lib/generateSignature.ts | 16 ++++++++++++++++ 2 files changed, 30 insertions(+) create mode 100644 packages/uploads/src/__tests__/generateSignature.test.ts create mode 100644 packages/uploads/src/lib/generateSignature.ts diff --git a/packages/uploads/src/__tests__/generateSignature.test.ts b/packages/uploads/src/__tests__/generateSignature.test.ts new file mode 100644 index 000000000000..d587baa22fa6 --- /dev/null +++ b/packages/uploads/src/__tests__/generateSignature.test.ts @@ -0,0 +1,14 @@ +import { beforeAll, describe, test } from 'vitest' + +import { generateSignature } from '../lib/generateSignature.js' + +describe('Generate signature', () => { + beforeAll(() => { + process.env.RW_UPLOADS_SECRET = 'bazinga' + }) + + test('It creates a signature', () => { + const out = generateSignature('/tmp/myfile.txt', 500) + console.log(`👉 \n ~ out:`, out) + }) +}) diff --git a/packages/uploads/src/lib/generateSignature.ts b/packages/uploads/src/lib/generateSignature.ts new file mode 100644 index 000000000000..055e3b3a3bc8 --- /dev/null +++ b/packages/uploads/src/lib/generateSignature.ts @@ -0,0 +1,16 @@ +import crypto from 'node:crypto' +export const generateSignature = (filePath: string, expiresIn: number) => { + if (!process.env.RW_UPLOADS_SECRET) { + throw new Error( + 'Please configure RW_UPLOADS_SECRET in your environment variables', + ) + } + + const expires = Math.floor(Date.now() / 1000) + expiresIn + const signature = crypto + .createHmac('sha256', process.env.RW_UPLOADS_SECRET) + .update(`${filePath}:${expires}`) + .digest('hex') + + return signature +} From d70ae4b2e73e38a242e28e658b21c64f31d85533 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 10:35:13 +0700 Subject: [PATCH 46/91] Uplaod yarn.lock --- yarn.lock | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/yarn.lock b/yarn.lock index 16367989ff78..3ae6b1e1a939 100644 --- a/yarn.lock +++ b/yarn.lock @@ -25365,6 +25365,19 @@ __metadata: languageName: node linkType: hard +"publint@npm:0.2.9": + version: 0.2.9 + resolution: "publint@npm:0.2.9" + dependencies: + npm-packlist: "npm:^5.1.3" + picocolors: "npm:^1.0.1" + sade: "npm:^1.8.1" + bin: + publint: lib/cli.js + checksum: 10c0/b414f40c2bc9372119346d5684eccb12bdf8066fc821301880d9dcdec0a5d852bbe926cb4583511f3e97736c53d1723e46e49285c9463bcb808cbfd979d5c2fc + languageName: node + linkType: hard + "pump@npm:^2.0.0": version: 2.0.1 resolution: "pump@npm:2.0.1" From 7478dcbea95d87594f7233405cacaf26c9db6391 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 13:34:19 +0700 Subject: [PATCH 47/91] Setup signature generation and validation (no URLs just yet) --- .../src/__tests__/generateSignature.test.ts | 14 -- .../uploads/src/__tests__/signedUrls.test.ts | 133 ++++++++++++++++++ packages/uploads/src/lib/generateSignature.ts | 16 --- packages/uploads/src/lib/signedUrls.ts | 111 +++++++++++++++ 4 files changed, 244 insertions(+), 30 deletions(-) delete mode 100644 packages/uploads/src/__tests__/generateSignature.test.ts create mode 100644 packages/uploads/src/__tests__/signedUrls.test.ts delete mode 100644 packages/uploads/src/lib/generateSignature.ts create mode 100644 packages/uploads/src/lib/signedUrls.ts diff --git a/packages/uploads/src/__tests__/generateSignature.test.ts b/packages/uploads/src/__tests__/generateSignature.test.ts deleted file mode 100644 index d587baa22fa6..000000000000 --- a/packages/uploads/src/__tests__/generateSignature.test.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { beforeAll, describe, test } from 'vitest' - -import { generateSignature } from '../lib/generateSignature.js' - -describe('Generate signature', () => { - beforeAll(() => { - process.env.RW_UPLOADS_SECRET = 'bazinga' - }) - - test('It creates a signature', () => { - const out = generateSignature('/tmp/myfile.txt', 500) - console.log(`👉 \n ~ out:`, out) - }) -}) diff --git a/packages/uploads/src/__tests__/signedUrls.test.ts b/packages/uploads/src/__tests__/signedUrls.test.ts new file mode 100644 index 000000000000..09d6db959eda --- /dev/null +++ b/packages/uploads/src/__tests__/signedUrls.test.ts @@ -0,0 +1,133 @@ +import { + beforeAll, + describe, + expect, + beforeEach, + afterEach, + vi, + it, +} from 'vitest' + +import { + EXPIRES_IN, + generateSignature, + validateSignature, +} from '../lib/signedUrls.js' + +describe('Signed URLs', () => { + beforeAll(() => { + process.env.RW_UPLOADS_SECRET = 'bazinga' + }) + + it('Can creates a signature', () => { + const { signature, expires } = generateSignature( + '/tmp/myfile.txt', + EXPIRES_IN.days(5), + ) + + expect(signature).toBeDefined() + + expect(diffInDaysFromNow(expires as number)).toBeCloseTo(5) + }) + + it('throws with correct error when wrong expires passed', () => { + const { signature, expires } = generateSignature( + '/tmp/myfile.txt', + EXPIRES_IN.days(1), + ) + + expect(() => + validateSignature({ + filePath: '/tmp/myfile.txt', + signature, + expires, + }), + ).not.toThrow() + + expect(() => + validateSignature({ + filePath: '/tmp/myfile.txt', + signature, + expires: 12512351, + }), + ).toThrowError('Signature has expired') + }) + + it('Throws an invalid signature when signature is wrong', () => { + const { signature, expires } = generateSignature( + '/tmp/myfile.txt', + EXPIRES_IN.days(1), + ) + + expect(() => + validateSignature({ + filePath: '/tmp/myfile.txt', + signature, + expires, + }), + ).not.toThrow() + + expect(() => + validateSignature({ + filePath: '/tmp/myfile.txt', + signature: 'im-the-wrong-signature', + expires, + }), + ).toThrowError('Invalid signature') + }) + + it('Throws an invalid signature when file path is wrong', () => { + const { signature, expires } = generateSignature( + '/tmp/myfile.txt', + EXPIRES_IN.days(20), + ) + expect(() => + validateSignature({ + filePath: '/tmp/some-other-file.txt', + signature, + expires, + }), + ).toThrowError('Invalid signature') + }) +}) + +describe('Expired signature', () => { + beforeEach(() => { + vi.useFakeTimers() + }) + + afterEach(() => { + vi.useRealTimers() + }) + + it('throws an error when the signature has expired', () => { + const filePath = '/bazinga/kittens.png' + const { signature, expires } = generateSignature( + filePath, + EXPIRES_IN.minutes(15), + ) + + const validation = () => + validateSignature({ + filePath, + signature, + expires, + }) + + expect(validation).not.toThrow() + + // Time travel to the future + vi.advanceTimersByTime(EXPIRES_IN.days(1)) + + expect(validation).toThrowError('Signature has expired') + }) +}) + +// Some utility function to make tests more readable +function diffInMinsFromNow(time: number) { + return Math.abs(time - Date.now()) / 60000 +} + +function diffInDaysFromNow(time: number) { + return Math.abs(time - Date.now()) / 86400000 +} diff --git a/packages/uploads/src/lib/generateSignature.ts b/packages/uploads/src/lib/generateSignature.ts deleted file mode 100644 index 055e3b3a3bc8..000000000000 --- a/packages/uploads/src/lib/generateSignature.ts +++ /dev/null @@ -1,16 +0,0 @@ -import crypto from 'node:crypto' -export const generateSignature = (filePath: string, expiresIn: number) => { - if (!process.env.RW_UPLOADS_SECRET) { - throw new Error( - 'Please configure RW_UPLOADS_SECRET in your environment variables', - ) - } - - const expires = Math.floor(Date.now() / 1000) + expiresIn - const signature = crypto - .createHmac('sha256', process.env.RW_UPLOADS_SECRET) - .update(`${filePath}:${expires}`) - .digest('hex') - - return signature -} diff --git a/packages/uploads/src/lib/signedUrls.ts b/packages/uploads/src/lib/signedUrls.ts new file mode 100644 index 000000000000..313ef900ca32 --- /dev/null +++ b/packages/uploads/src/lib/signedUrls.ts @@ -0,0 +1,111 @@ +import crypto from 'node:crypto' + +export const generateSignature = (filePath: string, expiresInMs?: number) => { + if (!process.env.RW_UPLOADS_SECRET) { + throw new Error( + 'Please configure RW_UPLOADS_SECRET in your environment variables', + ) + } + + if (expiresInMs) { + const expires = Date.now() + expiresInMs + const signature = crypto + .createHmac('sha256', process.env.RW_UPLOADS_SECRET) + .update(`${filePath}:${expires}`) + .digest('hex') + + return { expires, signature } + } else { + // Does not expire + const signature = crypto + .createHmac('sha256', process.env.RW_UPLOADS_SECRET) + .update(filePath) + .digest('hex') + + return { + signature, + expires: undefined, + } + } +} + +/** + * The signature and expires have to be extracted from the URL + */ +export const validateSignature = ({ + signature, + filePath, + expires, +}: { + filePath: string + signature: string + expires?: number +}) => { + // Note, expires not the same as expiresIn + if (!process.env.RW_UPLOADS_SECRET) { + throw new Error( + 'Please configure RW_UPLOADS_SECRET in your environment variables', + ) + } + + if (expires) { + // No need to validate if the signature has expired + if (Date.now() > expires) { + throw new Error('Signature has expired') + } + } + + const validSignature = expires + ? crypto + .createHmac('sha256', process.env.RW_UPLOADS_SECRET) + .update(`${filePath}:${expires}`) + .digest('hex') + : crypto + .createHmac('sha256', process.env.RW_UPLOADS_SECRET) + .update(`${filePath}`) + .digest('hex') + + if (validSignature !== signature) { + throw new Error('Invalid signature') + } +} + +export const getSignedDetailsFromUrl = (url: string) => { + const urlObj = new URL(url) + const expires = urlObj.searchParams.get('expires') + return { + expires: expires ? parseInt(expires) : undefined, + file: urlObj.searchParams.get('file'), + signature: urlObj.searchParams.get('s'), + } +} + +type SigningParms = { filePath: string; expiresIn?: number } + +export const getSignedUriString = ( + endpoint: string, + { filePath, expiresIn }: SigningParms, +) => { + const { signature, expires } = generateSignature(filePath, expiresIn) + + // This way you can pass in a path with params already + const signedUrl = new URL(endpoint) + signedUrl.searchParams.set('s', signature) + if (expires) { + signedUrl.searchParams.set('expires', expires.toString()) + } + + signedUrl.searchParams.set('path', filePath) + + return signedUrl.toString() +} + +export const EXPIRES_IN = { + seconds: (s: number) => s * 1000, + minutes: (m: number) => m * 60 * 1000, + hours: (h: number) => h * 60 * 60 * 1000, + days: (d: number) => d * 24 * 60 * 60 * 1000, + weeks: (w: number) => w * 7 * 24 * 60 * 60 * 1000, + months: (m: number) => m * 30 * 24 * 60 * 60 * 1000, + years: (y: number) => y * 365 * 24 * 60 * 60 * 1000, +} From 6243643ea2af2796ebf66545881d629a05ef82b6 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 15:01:05 +0700 Subject: [PATCH 48/91] Update extension tests --- .../src/__tests__/prismaExtension.test.ts | 51 +++++++++++++++---- 1 file changed, 42 insertions(+), 9 deletions(-) diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts index f891dd080aa2..498325ebafdd 100644 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ b/packages/uploads/src/__tests__/prismaExtension.test.ts @@ -87,7 +87,9 @@ describe('Uploads Prisma Extension', () => { }) }) - describe('createMany', () => { + // Not implemented yet + // Ideally it would just work automatically... but I guess we need to do all the variants + describe.skip('createMany', () => { it('createMany will remove files if all the create fails', async () => { try { await prismaClient.dumbo.createMany({ @@ -142,16 +144,47 @@ describe('Uploads Prisma Extension', () => { }) }) - // describe('update', () => {}) + describe('update', () => { + it('update will remove the old file, save new one', async () => { + const dummy = await prismaClient.dummy.create({ + data: { + uploadField: '/tmp/old.txt', + }, + }) - // describe('delete', () => {}) - }) + const updatedDummy = await prismaClient.dummy.update({ + data: { + uploadField: '/tmp/new.txt', + }, + where: { + id: dummy.id, + }, + }) + + expect(fs.unlink).toHaveBeenCalledWith('/tmp/old.txt') + expect(updatedDummy.uploadField).toBe('/tmp/new.txt') + }) + }) + + describe('delete', () => { + it('delete will remove all uploads', async () => { + const dumbo = await prismaClient.dumbo.create({ + data: { + firstUpload: '/tmp/first.txt', + secondUpload: '/tmp/second.txt', + }, + }) - describe('Result extensions', () => { - it('will return a data URL for the file', async () => {}) + await prismaClient.dumbo.delete({ + where: { + id: dumbo.id, + }, + }) - // @TODO Handle edge cases (file removed, data modified, etc.) - // it('if file is not found, will throw an error', async () => {}) - // it('if saved file is not a path, will throw an error', async () => {}) + expect(fs.unlink).toHaveBeenCalledTimes(2) + expect(fs.unlink).toHaveBeenCalledWith('/tmp/first.txt') + expect(fs.unlink).toHaveBeenCalledWith('/tmp/second.txt') + }) + }) }) }) From a5fd94a10c308cb39606a96e5412fe3030c7b483 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 15:01:23 +0700 Subject: [PATCH 49/91] Move filename generation to base adapter --- packages/uploads/src/FileSystemStorage.ts | 19 ++++++------------- packages/uploads/src/MemoryStorage.ts | 10 +++------- packages/uploads/src/StorageAdapter.ts | 14 ++++++++++++++ 3 files changed, 23 insertions(+), 20 deletions(-) diff --git a/packages/uploads/src/FileSystemStorage.ts b/packages/uploads/src/FileSystemStorage.ts index 0b3312effbec..653305fd1384 100644 --- a/packages/uploads/src/FileSystemStorage.ts +++ b/packages/uploads/src/FileSystemStorage.ts @@ -1,31 +1,24 @@ import fs from 'node:fs/promises' import path from 'node:path' -import mime from 'mime-types' -import { ulid } from 'ulid' import type { SaveOptionsOverride } from './StorageAdapter.js' import { StorageAdapter } from './StorageAdapter.js' -export class FileSystemStorage - extends StorageAdapter - implements StorageAdapter -{ - async save(file: File, saveOpts?: SaveOptionsOverride) { - const fileName = saveOpts?.fileName || ulid() - const extension = mime.extension(file.type) - ? `.${mime.extension(file.type)}` - : '' +export class FileSystemStorage extends StorageAdapter implements StorageAdapter { + async save(file: File, saveOverride?: SaveOptionsOverride) { + const fileName = this.generateFileNameWithExtension(saveOverride, file) const location = path.join( - saveOpts?.path || this.adapterOpts.baseDir, - fileName + `${extension}`, + saveOverride?.path || this.adapterOpts.baseDir, + fileName, ) const nodeBuffer = await file.arrayBuffer() await fs.writeFile(location, Buffer.from(nodeBuffer)) return { location } } + async remove(filePath: string) { await fs.unlink(filePath) } diff --git a/packages/uploads/src/MemoryStorage.ts b/packages/uploads/src/MemoryStorage.ts index 64d667a1f9cc..f02ea05a9220 100644 --- a/packages/uploads/src/MemoryStorage.ts +++ b/packages/uploads/src/MemoryStorage.ts @@ -1,7 +1,5 @@ import path from 'node:path' -import mime from 'mime-types' -import { ulid } from 'ulid' import { StorageAdapter } from './StorageAdapter.js' import type { SaveOptionsOverride } from './StorageAdapter.js' @@ -10,14 +8,11 @@ export class MemoryStorage extends StorageAdapter implements StorageAdapter { store: Record = {} async save(file: File, saveOpts?: SaveOptionsOverride) { - const fileName = saveOpts?.fileName || ulid() - const extension = mime.extension(file.type) - ? `.${mime.extension(file.type)}` - : '' + const fileName = this.generateFileNameWithExtension(saveOpts, file) const location = path.join( saveOpts?.path || this.adapterOpts.baseDir, - fileName + `${extension}`, + fileName, ) const nodeBuffer = await file.arrayBuffer() @@ -27,6 +22,7 @@ export class MemoryStorage extends StorageAdapter implements StorageAdapter { location, } } + async remove(filePath: string) { delete this.store[filePath] } diff --git a/packages/uploads/src/StorageAdapter.ts b/packages/uploads/src/StorageAdapter.ts index f3883b17b935..64a3d0d288d7 100644 --- a/packages/uploads/src/StorageAdapter.ts +++ b/packages/uploads/src/StorageAdapter.ts @@ -6,6 +6,9 @@ * } */ +import mime from 'mime-types' +import { ulid } from 'ulid' + export type AdapterResult = { location: string } @@ -29,6 +32,17 @@ export abstract class StorageAdapter { return this.adapterOpts } + generateFileNameWithExtension( + saveOpts: SaveOptionsOverride | undefined, + file: File, + ) { + const fileName = saveOpts?.fileName || ulid() + const extension = mime.extension(file.type) + ? `.${mime.extension(file.type)}` + : '' + return `${fileName}${extension}` + } + abstract save( file: File, saveOpts?: SaveOptionsOverride, From f1aaafbaaa7fe648cbfccb76d02656039f5ee785 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 15:01:46 +0700 Subject: [PATCH 50/91] More signed url tests --- .../uploads/src/__tests__/signedUrls.test.ts | 17 +++++++++++++---- 1 file changed, 13 insertions(+), 4 deletions(-) diff --git a/packages/uploads/src/__tests__/signedUrls.test.ts b/packages/uploads/src/__tests__/signedUrls.test.ts index 09d6db959eda..c23650cb2f98 100644 --- a/packages/uploads/src/__tests__/signedUrls.test.ts +++ b/packages/uploads/src/__tests__/signedUrls.test.ts @@ -6,11 +6,13 @@ import { afterEach, vi, it, + test, } from 'vitest' import { EXPIRES_IN, generateSignature, + getSignedDetailsFromUrl, validateSignature, } from '../lib/signedUrls.js' @@ -123,11 +125,18 @@ describe('Expired signature', () => { }) }) -// Some utility function to make tests more readable -function diffInMinsFromNow(time: number) { - return Math.abs(time - Date.now()) / 60000 -} +test('Parses details related to signatures from a url string', () => { + const url = + 'https://example.com/signedFile?file=/path/to/hello.txt&s=s1gnatur3&expires=123123' + + const { file, signature, expires } = getSignedDetailsFromUrl(url) + + expect(file).toBe('/path/to/hello.txt') + expect(signature).toBe('s1gnatur3') + expect(expires).toBe(123123) +}) +// Util functions to make the tests more readable function diffInDaysFromNow(time: number) { return Math.abs(time - Date.now()) / 86400000 } From e48d364b0482b2c0c3a5d0b586fd06bbd639dcf7 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 15:21:57 +0700 Subject: [PATCH 51/91] Few changes --- packages/uploads/src/__tests__/signedUrls.test.ts | 13 +++++++++++++ packages/uploads/src/lib/signedUrls.ts | 12 ++++++------ packages/uploads/src/prismaExtension.ts | 4 +--- 3 files changed, 20 insertions(+), 9 deletions(-) diff --git a/packages/uploads/src/__tests__/signedUrls.test.ts b/packages/uploads/src/__tests__/signedUrls.test.ts index c23650cb2f98..b5e517ef1fb9 100644 --- a/packages/uploads/src/__tests__/signedUrls.test.ts +++ b/packages/uploads/src/__tests__/signedUrls.test.ts @@ -12,6 +12,7 @@ import { import { EXPIRES_IN, generateSignature, + generateSignedQueryParams, getSignedDetailsFromUrl, validateSignature, } from '../lib/signedUrls.js' @@ -136,6 +137,18 @@ test('Parses details related to signatures from a url string', () => { expect(expires).toBe(123123) }) +test('Generates a signed url', () => { + const signedQueryParams = generateSignedQueryParams('/files/bazinga', { + filePath: '/path/to/hello.txt', + expiresIn: EXPIRES_IN.days(1), + }) + + expect(signedQueryParams).toContain('/files/bazinga?s=') + expect(signedQueryParams).toContain('s=') + expect(signedQueryParams).toContain('expires=') + expect(signedQueryParams).toContain('path=') // The actual file path +}) + // Util functions to make the tests more readable function diffInDaysFromNow(time: number) { return Math.abs(time - Date.now()) / 86400000 diff --git a/packages/uploads/src/lib/signedUrls.ts b/packages/uploads/src/lib/signedUrls.ts index 313ef900ca32..b626e8e1dff1 100644 --- a/packages/uploads/src/lib/signedUrls.ts +++ b/packages/uploads/src/lib/signedUrls.ts @@ -82,22 +82,22 @@ export const getSignedDetailsFromUrl = (url: string) => { type SigningParms = { filePath: string; expiresIn?: number } -export const getSignedUriString = ( +export const generateSignedQueryParams = ( endpoint: string, { filePath, expiresIn }: SigningParms, ) => { const { signature, expires } = generateSignature(filePath, expiresIn) // This way you can pass in a path with params already - const signedUrl = new URL(endpoint) - signedUrl.searchParams.set('s', signature) + const params = new URLSearchParams() + params.set('s', signature) if (expires) { - signedUrl.searchParams.set('expires', expires.toString()) + params.set('expires', expires.toString()) } - signedUrl.searchParams.set('path', filePath) + params.set('path', filePath) - return signedUrl.toString() + return `${endpoint}?${params.toString()}` } export const EXPIRES_IN = { diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 36e22291c340..9ea56f21bd74 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -141,11 +141,10 @@ export const createUploadsExtension = ( compute(modelData) { return async () => { const base64UploadFields: Record = {} - type ModelField = keyof typeof modelData for await (const field of uploadFields) { base64UploadFields[field] = await fileToDataUri( - modelData[field as ModelField] as string, + modelData[field] as string, ) } @@ -174,7 +173,6 @@ export const createUploadsExtension = ( for await (const field of uploadFields) { const uploadLocation = args.data?.[field] as string if (uploadLocation) { - console.log('Removing file >>>', uploadLocation) await storageAdapter.remove(uploadLocation) } } From 75b38356ebf47e3f91b7388e718283d2444c692d Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 16:00:41 +0700 Subject: [PATCH 52/91] Add withSignedUrl result extension --- packages/uploads/prisma-override.d.ts | 2 +- .../src/__tests__/prismaExtension.test.ts | 190 ------------------ .../src/__tests__/queryExtensions.test.ts | 183 +++++++++++++++++ .../src/__tests__/resultExtensions.test.ts | 48 +++++ packages/uploads/src/prismaExtension.ts | 33 ++- 5 files changed, 262 insertions(+), 194 deletions(-) delete mode 100644 packages/uploads/src/__tests__/prismaExtension.test.ts create mode 100644 packages/uploads/src/__tests__/queryExtensions.test.ts create mode 100644 packages/uploads/src/__tests__/resultExtensions.test.ts diff --git a/packages/uploads/prisma-override.d.ts b/packages/uploads/prisma-override.d.ts index ec14d355db25..f976517c35be 100644 --- a/packages/uploads/prisma-override.d.ts +++ b/packages/uploads/prisma-override.d.ts @@ -2,7 +2,7 @@ // This is so that we get accurate types (rather than the default anys) - and when the prismaExtension runs // it will still use the types from '@prisma/client' which points to the user's prisma client and not ours -import { PrismaClient as LocalPrismaClient } from './src/__tests__/prisma-client/index.d.ts' +import type { PrismaClient as LocalPrismaClient } from './src/__tests__/prisma-client/index.d.ts' declare module '@prisma/client' { export class PrismaClient extends LocalPrismaClient {} diff --git a/packages/uploads/src/__tests__/prismaExtension.test.ts b/packages/uploads/src/__tests__/prismaExtension.test.ts deleted file mode 100644 index 498325ebafdd..000000000000 --- a/packages/uploads/src/__tests__/prismaExtension.test.ts +++ /dev/null @@ -1,190 +0,0 @@ -import fs from 'node:fs/promises' - -import { describe, it, vi, expect, beforeEach } from 'vitest' - -import { FileSystemStorage } from '../FileSystemStorage.js' -import type { UploadsConfig } from '../prismaExtension.js' -import { setupUploads } from '../setup.js' - -// @MARK: use the local prisma client -import { PrismaClient } from './prisma-client' - -vi.mock('node:fs/promises', () => ({ - default: { - writeFile: vi.fn(), - unlink: vi.fn(), - readFile: vi.fn(() => { - return 'MOCKED_FILE_CONTENT' - }), - copyFile: vi.fn(), - }, -})) - -describe('Uploads Prisma Extension', () => { - const dummyUploadConfig = { - fields: 'uploadField', - } - - const dumboUploadConfig = { - fields: ['firstUpload', 'secondUpload'], - } - - const uploadConfig: UploadsConfig = { - dummy: dummyUploadConfig, - dumbo: dumboUploadConfig, - } - - const { prismaExtension, uploadsProcessors } = setupUploads( - uploadConfig, - new FileSystemStorage({ - baseDir: '/tmp', - }), - ) - - const prismaClient = new PrismaClient().$extends(prismaExtension) - - describe('Query extensions', () => { - beforeEach(() => { - vi.resetAllMocks() - }) - - const sampleFile = new File(['heres-some-content'], 'dummy.txt', { - type: 'text/plain', - }) - - describe('create', () => { - it('create will save files', async () => { - const processedData = await uploadsProcessors.processDummyUploads({ - uploadField: sampleFile, - }) - - expect(fs.writeFile).toHaveBeenCalled() - const dummy = await prismaClient.dummy.create({ - data: processedData, - }) - - expect(dummy).toMatchObject({ - uploadField: expect.stringMatching(/\/tmp\/.*\.txt$/), - }) - }) - - it('will remove the file if the create fails', async () => { - try { - await prismaClient.dumbo.create({ - data: { - firstUpload: '/tmp/first.txt', - secondUpload: '/bazinga/second.txt', - // @ts-expect-error Checking the error here - id: 'this-is-the-incorrect-type', - }, - }) - } catch { - expect(fs.unlink).toHaveBeenNthCalledWith(1, '/tmp/first.txt') - expect(fs.unlink).toHaveBeenNthCalledWith(2, '/bazinga/second.txt') - } - - expect.assertions(2) - }) - }) - - // Not implemented yet - // Ideally it would just work automatically... but I guess we need to do all the variants - describe.skip('createMany', () => { - it('createMany will remove files if all the create fails', async () => { - try { - await prismaClient.dumbo.createMany({ - data: [ - { - firstUpload: '/one/first.txt', - secondUpload: '/one/second.txt', - id: 'break', - }, - { - firstUpload: '/two/first.txt', - secondUpload: '/two/second.txt', - id: 'break2', - }, - ], - }) - } catch (e) { - expect(fs.unlink).toHaveBeenCalledTimes(4) - expect(fs.unlink).toHaveBeenNthCalledWith(1, '/one/first.txt') - expect(fs.unlink).toHaveBeenNthCalledWith(2, '/one/second.txt') - expect(fs.unlink).toHaveBeenNthCalledWith(3, '/two/first.txt') - expect(fs.unlink).toHaveBeenNthCalledWith(4, '/two/second.txt') - } - - expect.assertions(4) - }) - - it('createMany will remove files from only the creates that fail', async () => { - try { - await prismaClient.dumbo.createMany({ - data: [ - // This one will go through - { - firstUpload: '/one/first.txt', - secondUpload: '/one/second.txt', - }, - { - firstUpload: '/two/first.txt', - secondUpload: '/two/second.txt', - id: 'break2', - }, - ], - }) - } catch (e) { - console.log(e) - expect(fs.unlink).toHaveBeenCalledTimes(2) - expect(fs.unlink).toHaveBeenNthCalledWith(1, '/two/first.txt') - expect(fs.unlink).toHaveBeenNthCalledWith(2, '/two/second.txt') - } - - expect.assertions(4) - }) - }) - - describe('update', () => { - it('update will remove the old file, save new one', async () => { - const dummy = await prismaClient.dummy.create({ - data: { - uploadField: '/tmp/old.txt', - }, - }) - - const updatedDummy = await prismaClient.dummy.update({ - data: { - uploadField: '/tmp/new.txt', - }, - where: { - id: dummy.id, - }, - }) - - expect(fs.unlink).toHaveBeenCalledWith('/tmp/old.txt') - expect(updatedDummy.uploadField).toBe('/tmp/new.txt') - }) - }) - - describe('delete', () => { - it('delete will remove all uploads', async () => { - const dumbo = await prismaClient.dumbo.create({ - data: { - firstUpload: '/tmp/first.txt', - secondUpload: '/tmp/second.txt', - }, - }) - - await prismaClient.dumbo.delete({ - where: { - id: dumbo.id, - }, - }) - - expect(fs.unlink).toHaveBeenCalledTimes(2) - expect(fs.unlink).toHaveBeenCalledWith('/tmp/first.txt') - expect(fs.unlink).toHaveBeenCalledWith('/tmp/second.txt') - }) - }) - }) -}) diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts new file mode 100644 index 000000000000..317a0f46ef59 --- /dev/null +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -0,0 +1,183 @@ +import fs from 'node:fs/promises' + +import { describe, it, vi, expect, beforeEach } from 'vitest' + +import { FileSystemStorage } from '../FileSystemStorage.js' +import type { UploadsConfig } from '../prismaExtension.js' +import { setupUploads } from '../setup.js' + +// @MARK: use the local prisma client in the test +import { PrismaClient } from './prisma-client' + +vi.mock('node:fs/promises', () => ({ + default: { + writeFile: vi.fn(), + unlink: vi.fn(), + readFile: vi.fn(() => { + return 'MOCKED_FILE_CONTENT' + }), + copyFile: vi.fn(), + }, +})) +describe('Query extensions', () => { + const uploadConfig: UploadsConfig = { + dummy: { + fields: 'uploadField', + }, + dumbo: { + fields: ['firstUpload', 'secondUpload'], + }, + } + + const { prismaExtension, uploadsProcessors } = setupUploads( + uploadConfig, + new FileSystemStorage({ + baseDir: '/tmp', + }), + ) + + const prismaClient = new PrismaClient().$extends(prismaExtension) + + beforeEach(() => { + vi.resetAllMocks() + }) + + const sampleFile = new File(['heres-some-content'], 'dummy.txt', { + type: 'text/plain', + }) + + describe('create', () => { + it('create will save files', async () => { + const processedData = await uploadsProcessors.processDummyUploads({ + uploadField: sampleFile, + }) + + expect(fs.writeFile).toHaveBeenCalled() + const dummy = await prismaClient.dummy.create({ + data: processedData, + }) + + expect(dummy).toMatchObject({ + uploadField: expect.stringMatching(/\/tmp\/.*\.txt$/), + }) + }) + + it('will remove the file if the create fails', async () => { + try { + await prismaClient.dumbo.create({ + data: { + firstUpload: '/tmp/first.txt', + secondUpload: '/bazinga/second.txt', + // @ts-expect-error Checking the error here + id: 'this-is-the-incorrect-type', + }, + }) + } catch { + expect(fs.unlink).toHaveBeenNthCalledWith(1, '/tmp/first.txt') + expect(fs.unlink).toHaveBeenNthCalledWith(2, '/bazinga/second.txt') + } + + expect.assertions(2) + }) + }) + + // Not implemented yet + // Ideally it would just work automatically... but I guess we need to do all the variants + describe.skip('createMany', () => { + it('createMany will remove files if all the create fails', async () => { + try { + await prismaClient.dumbo.createMany({ + data: [ + { + firstUpload: '/one/first.txt', + secondUpload: '/one/second.txt', + id: 'break', + }, + { + firstUpload: '/two/first.txt', + secondUpload: '/two/second.txt', + id: 'break2', + }, + ], + }) + } catch (e) { + expect(fs.unlink).toHaveBeenCalledTimes(4) + expect(fs.unlink).toHaveBeenNthCalledWith(1, '/one/first.txt') + expect(fs.unlink).toHaveBeenNthCalledWith(2, '/one/second.txt') + expect(fs.unlink).toHaveBeenNthCalledWith(3, '/two/first.txt') + expect(fs.unlink).toHaveBeenNthCalledWith(4, '/two/second.txt') + } + + expect.assertions(4) + }) + + it('createMany will remove files from only the creates that fail', async () => { + try { + await prismaClient.dumbo.createMany({ + data: [ + // This one will go through + { + firstUpload: '/one/first.txt', + secondUpload: '/one/second.txt', + }, + { + firstUpload: '/two/first.txt', + secondUpload: '/two/second.txt', + id: 'break2', + }, + ], + }) + } catch (e) { + console.log(e) + expect(fs.unlink).toHaveBeenCalledTimes(2) + expect(fs.unlink).toHaveBeenNthCalledWith(1, '/two/first.txt') + expect(fs.unlink).toHaveBeenNthCalledWith(2, '/two/second.txt') + } + + expect.assertions(4) + }) + }) + + describe('update', () => { + it('update will remove the old file, save new one', async () => { + const dummy = await prismaClient.dummy.create({ + data: { + uploadField: '/tmp/old.txt', + }, + }) + + const updatedDummy = await prismaClient.dummy.update({ + data: { + uploadField: '/tmp/new.txt', + }, + where: { + id: dummy.id, + }, + }) + + expect(fs.unlink).toHaveBeenCalledWith('/tmp/old.txt') + expect(updatedDummy.uploadField).toBe('/tmp/new.txt') + }) + }) + + describe('delete', () => { + it('delete will remove all uploads', async () => { + const dumbo = await prismaClient.dumbo.create({ + data: { + firstUpload: '/tmp/first.txt', + secondUpload: '/tmp/second.txt', + }, + }) + + await prismaClient.dumbo.delete({ + where: { + id: dumbo.id, + }, + }) + + expect(fs.unlink).toHaveBeenCalledTimes(2) + expect(fs.unlink).toHaveBeenCalledWith('/tmp/first.txt') + expect(fs.unlink).toHaveBeenCalledWith('/tmp/second.txt') + }) + }) +}) diff --git a/packages/uploads/src/__tests__/resultExtensions.test.ts b/packages/uploads/src/__tests__/resultExtensions.test.ts new file mode 100644 index 000000000000..bc77605ec864 --- /dev/null +++ b/packages/uploads/src/__tests__/resultExtensions.test.ts @@ -0,0 +1,48 @@ +import { describe, it, beforeAll, expect } from 'vitest' + +import { MemoryStorage } from '../MemoryStorage.js' +import type { UploadsConfig } from '../prismaExtension.js' +import { setupUploads } from '../setup.js' + +// @MARK: use the local prisma client in the test +import { PrismaClient } from './prisma-client' + +describe('Result extensions', () => { + const uploadConfig: UploadsConfig = { + dummy: { + fields: 'uploadField', + }, + dumbo: { + fields: ['firstUpload', 'secondUpload'], + }, + } + + const { prismaExtension } = setupUploads( + uploadConfig, + new MemoryStorage({ + baseDir: '/tmp', + }), + ) + + const prismaClient = new PrismaClient().$extends(prismaExtension) + + describe('withSignedUrl', () => { + beforeAll(() => { + process.env.RW_UPLOADS_SECRET = 'gasdg' + }) + it('Generates signed urls for each upload field', async () => { + const dumbo = await prismaClient.dumbo.create({ + data: { + firstUpload: '/dumbo/first.txt', + secondUpload: '/dumbo/second.txt', + }, + }) + + const signedUrlDumbo = await dumbo.withSignedUrl(1000) + expect(signedUrlDumbo.firstUpload).toContain('path=%2Fdumbo%2Ffirst.txt') + expect(signedUrlDumbo.secondUpload).toContain( + 'path=%2Fdumbo%2Fsecond.txt', + ) + }) + }) +}) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 9ea56f21bd74..846fedd6f352 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -7,6 +7,7 @@ import type * as runtime from '@prisma/client/runtime/library' // import { PrismaClient } from './__tests__/prisma-client/index.js' // import { Prisma } from './__tests__/prisma-client/index.js' import { fileToDataUri } from './fileSave.utils.js' +import { generateSignedQueryParams } from './lib/signedUrls.js' import type { StorageAdapter } from './StorageAdapter.js' type FilterOutDollarPrefixed = T extends `$${string}` @@ -45,6 +46,12 @@ export const createUploadsExtension = ( modelData: Record, ) => (this: T) => Promise } + withSignedUrl: { + needs: Record + compute: ( + modelData: Record, + ) => (this: T) => Promise + } } } @@ -62,9 +69,8 @@ export const createUploadsExtension = ( ) { // With strict mode you cannot call findFirstOrThrow with the same args, because it is a union type // Ideally there's a better way to do this - const record = await ( - prismaInstance[model as ModelNames] as any - ).findFirstOrThrow(args) + const record = + await prismaInstance[model as ModelNames].findFirstOrThrow(args) // Delete the file from the file system fields.forEach(async (field) => { @@ -156,6 +162,27 @@ export const createUploadsExtension = ( } }, }, + withSignedUrl: { + needs, + compute(modelData) { + return (expiresIn?: number) => { + const signedUrlFields: Record = {} + + for (const field of uploadFields) { + signedUrlFields[field] = generateSignedQueryParams('/HARDCODED', { + filePath: modelData[field] as string, + expiresIn: expiresIn, + }) + } + + return { + // modelData is of type unknown at this point + ...(modelData as any), + ...signedUrlFields, + } + } + }, + }, } } From ef8aa4ac0125a38e16bdf05f106fc193650d9bde Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 17:43:51 +0700 Subject: [PATCH 53/91] Get UploadConfig type working with fields --- packages/uploads/src/prismaExtension.ts | 35 ++++++++++++++++--------- 1 file changed, 23 insertions(+), 12 deletions(-) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 846fedd6f352..bf343b729992 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -2,10 +2,10 @@ import { PrismaClient } from '@prisma/client' import { Prisma } from '@prisma/client' import type * as runtime from '@prisma/client/runtime/library' -// @TODO(TS): UploadsConfig behaves differently here.. probably -// the prisma-override not quite there yet? +// local imports // import { PrismaClient } from './__tests__/prisma-client/index.js' -// import { Prisma } from './__tests__/prisma-client/index.js' +// import type { Prisma } from './__tests__/prisma-client/index.js' +// import type * as PrismaAll from './__tests__/prisma-client/index.js' import { fileToDataUri } from './fileSave.utils.js' import { generateSignedQueryParams } from './lib/signedUrls.js' import type { StorageAdapter } from './StorageAdapter.js' @@ -19,16 +19,24 @@ type FilterOutDollarPrefixed = T extends `$${string}` // Filter out $on, $connect, etc. type ModelNames = FilterOutDollarPrefixed -export type UploadConfigForModel = { - // @TODO(TS): I want the fields here to be the fields of the model - fields: string[] | string +type PrismaModelFields = keyof Prisma.Result< + PrismaClient[MName], + any, + 'findFirstOrThrow' +> + +export type UploadConfigForModel = { + fields: + | PrismaModelFields + | PrismaModelFields[] savePath?: ((args: unknown) => string) | string fileName?: (args: unknown) => string onFileSaved?: (filePath: string) => void | Promise } -export type UploadsConfig = - Record +export type UploadsConfig = { + [K in MNames]?: UploadConfigForModel +} export const createUploadsExtension = ( config: UploadsConfig, @@ -70,6 +78,7 @@ export const createUploadsExtension = ( // With strict mode you cannot call findFirstOrThrow with the same args, because it is a union type // Ideally there's a better way to do this const record = + // @ts-expect-error laskndglkn await prismaInstance[model as ModelNames].findFirstOrThrow(args) // Delete the file from the file system @@ -84,10 +93,12 @@ export const createUploadsExtension = ( const resultExtends = {} as ResultExtends for (const modelName in config) { // Guaranteed to have modelConfig, we're looping over config 🙄 - const modelConfig = config[modelName as MNames] as UploadConfigForModel - const uploadFields = Array.isArray(modelConfig.fields) - ? modelConfig.fields - : [modelConfig.fields] + const modelConfig = config[modelName] + const uploadFields = ( + Array.isArray(modelConfig.fields) + ? modelConfig.fields + : [modelConfig.fields] + ) as string[] queryExtends[modelName] = { async create({ query, args }) { From a75133700e46581019d6c68b12b737e9bd49d655 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 18:18:34 +0700 Subject: [PATCH 54/91] Fix types for resultExtension --- packages/uploads/src/createProcessors.ts | 14 +++++++++----- packages/uploads/src/prismaExtension.ts | 14 ++++++++------ packages/uploads/tsconfig.json | 2 +- packages/uploads/vitest.config.mts | 16 ++++++++++------ 4 files changed, 28 insertions(+), 18 deletions(-) diff --git a/packages/uploads/src/createProcessors.ts b/packages/uploads/src/createProcessors.ts index af29a18a32c9..470d17342a84 100644 --- a/packages/uploads/src/createProcessors.ts +++ b/packages/uploads/src/createProcessors.ts @@ -47,11 +47,15 @@ export const createUploadProcessors = ( Object.keys(uploadConfig).forEach((model) => { const modelKey = model as keyof typeof uploadConfig - const currentModelUploadFields = Array.isArray( - uploadConfig[modelKey].fields, - ) - ? uploadConfig[modelKey].fields - : [uploadConfig[modelKey].fields] + const currentModelConfig = uploadConfig[modelKey] + + if (!currentModelConfig) { + return + } + + const currentModelUploadFields = Array.isArray(currentModelConfig.fields) + ? currentModelConfig.fields + : [currentModelConfig.fields] const capitalCaseModel = `${model.charAt(0).toUpperCase() + model.slice(1)}` const processorKey = `process${capitalCaseModel}Uploads` as keyof Processors diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index bf343b729992..8dd07260f0d8 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -1,11 +1,8 @@ import { PrismaClient } from '@prisma/client' -import { Prisma } from '@prisma/client' +import type { Prisma } from '@prisma/client' +import { Prisma as PrismaExtension } from '@prisma/client/extension' import type * as runtime from '@prisma/client/runtime/library' -// local imports -// import { PrismaClient } from './__tests__/prisma-client/index.js' -// import type { Prisma } from './__tests__/prisma-client/index.js' -// import type * as PrismaAll from './__tests__/prisma-client/index.js' import { fileToDataUri } from './fileSave.utils.js' import { generateSignedQueryParams } from './lib/signedUrls.js' import type { StorageAdapter } from './StorageAdapter.js' @@ -94,6 +91,11 @@ export const createUploadsExtension = ( for (const modelName in config) { // Guaranteed to have modelConfig, we're looping over config 🙄 const modelConfig = config[modelName] + + if (!modelConfig) { + continue + } + const uploadFields = ( Array.isArray(modelConfig.fields) ? modelConfig.fields @@ -197,7 +199,7 @@ export const createUploadsExtension = ( } } - return Prisma.defineExtension((client) => { + return PrismaExtension.defineExtension((client) => { return client.$extends({ name: 'redwood-upload-prisma-plugin', query: queryExtends, diff --git a/packages/uploads/tsconfig.json b/packages/uploads/tsconfig.json index 3484bfda0d5c..84e3d7c1df9d 100644 --- a/packages/uploads/tsconfig.json +++ b/packages/uploads/tsconfig.json @@ -8,7 +8,7 @@ "outDir": "dist" }, "include": ["src", "prisma-override.d.ts"], - + "exclude": ["dist", "node_modules", "**/__mocks__"], "references": [ { "path": "../project-config" diff --git a/packages/uploads/vitest.config.mts b/packages/uploads/vitest.config.mts index 92ccad350ca8..2903bac00e26 100644 --- a/packages/uploads/vitest.config.mts +++ b/packages/uploads/vitest.config.mts @@ -1,9 +1,10 @@ +import path from 'path' +import { fileURLToPath } from 'url' + import { defineConfig, configDefaults } from 'vitest/config' -import { fileURLToPath } from 'url'; -import path from 'path'; -const __filename = fileURLToPath(import.meta.url); -const __dirname = path.dirname(__filename); +const __filename = fileURLToPath(import.meta.url) +const __dirname = path.dirname(__filename) export default defineConfig({ test: { @@ -13,9 +14,12 @@ export default defineConfig({ }, globalSetup: ['vitest.setup.mts'], alias: { - // We alias prisma client, so that it doesn't interfere with other packages in the mono repo + // We alias prisma client, otherwise you'll get "prisma client not initialized" + '@prisma/client/extension': path.resolve( + __dirname, + '../../node_modules/@prisma/client/extension.js', + ), '@prisma/client': path.resolve(__dirname, 'src/__tests__/prisma-client'), }, }, - }) From 8cd00818d5cf7ea857bdd3b581dfaab9cd4e77fc Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 18:48:59 +0700 Subject: [PATCH 55/91] Keep these changes --- .../src/__tests__/queryExtensions.test.ts | 5 +++- .../src/__tests__/resultExtensions.test.ts | 23 +++++++++++++++++-- packages/uploads/tsconfig.json | 1 + 3 files changed, 26 insertions(+), 3 deletions(-) diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts index 317a0f46ef59..7a5b2cc1f4c1 100644 --- a/packages/uploads/src/__tests__/queryExtensions.test.ts +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -7,7 +7,7 @@ import type { UploadsConfig } from '../prismaExtension.js' import { setupUploads } from '../setup.js' // @MARK: use the local prisma client in the test -import { PrismaClient } from './prisma-client' +import { PrismaClient } from './prisma-client/index.js' vi.mock('node:fs/promises', () => ({ default: { @@ -91,11 +91,13 @@ describe('Query extensions', () => { { firstUpload: '/one/first.txt', secondUpload: '/one/second.txt', + // @ts-expect-error Intentional bro id: 'break', }, { firstUpload: '/two/first.txt', secondUpload: '/two/second.txt', + // @ts-expect-error Intentional bro id: 'break2', }, ], @@ -123,6 +125,7 @@ describe('Query extensions', () => { { firstUpload: '/two/first.txt', secondUpload: '/two/second.txt', + // @ts-expect-error Intentional bro id: 'break2', }, ], diff --git a/packages/uploads/src/__tests__/resultExtensions.test.ts b/packages/uploads/src/__tests__/resultExtensions.test.ts index bc77605ec864..5daf296e21cd 100644 --- a/packages/uploads/src/__tests__/resultExtensions.test.ts +++ b/packages/uploads/src/__tests__/resultExtensions.test.ts @@ -5,7 +5,7 @@ import type { UploadsConfig } from '../prismaExtension.js' import { setupUploads } from '../setup.js' // @MARK: use the local prisma client in the test -import { PrismaClient } from './prisma-client' +import { PrismaClient } from './prisma-client/index.js' describe('Result extensions', () => { const uploadConfig: UploadsConfig = { @@ -38,11 +38,30 @@ describe('Result extensions', () => { }, }) - const signedUrlDumbo = await dumbo.withSignedUrl(1000) + const signedUrlDumbo = await dumbo.withSignedUrl() expect(signedUrlDumbo.firstUpload).toContain('path=%2Fdumbo%2Ffirst.txt') expect(signedUrlDumbo.secondUpload).toContain( 'path=%2Fdumbo%2Fsecond.txt', ) }) + + it('laskdng', async () => { + const customClient = new PrismaClient().$extends({ + result: { + dumbo: { + helloJosh: { + compute() { + return () => { + return 'hello josh' + } + }, + }, + }, + }, + }) + + const dumbo = await customClient.dumbo.findFirst({ where: { id: 1 } }) + dumbo?.helloJosh() + }) }) }) diff --git a/packages/uploads/tsconfig.json b/packages/uploads/tsconfig.json index 84e3d7c1df9d..7458881c480d 100644 --- a/packages/uploads/tsconfig.json +++ b/packages/uploads/tsconfig.json @@ -8,6 +8,7 @@ "outDir": "dist" }, "include": ["src", "prisma-override.d.ts"], + // Excluding types here causes types to be inaccurate in tests "exclude": ["dist", "node_modules", "**/__mocks__"], "references": [ { From 32942a4637d6c9ad99dadee290e657ddaa218bc5 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 19:15:25 +0700 Subject: [PATCH 56/91] Warn comment on vitest alias --- packages/uploads/vitest.config.mts | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/uploads/vitest.config.mts b/packages/uploads/vitest.config.mts index 2903bac00e26..ee7fa5cb1f93 100644 --- a/packages/uploads/vitest.config.mts +++ b/packages/uploads/vitest.config.mts @@ -15,6 +15,7 @@ export default defineConfig({ globalSetup: ['vitest.setup.mts'], alias: { // We alias prisma client, otherwise you'll get "prisma client not initialized" + // Important to have the subpath first here '@prisma/client/extension': path.resolve( __dirname, '../../node_modules/@prisma/client/extension.js', From 32e870de3589e6f88b9442122ee80c0eb5a6c5f8 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 19 Aug 2024 19:41:48 +0700 Subject: [PATCH 57/91] Fix typing on keys being hardcoded --- packages/uploads/src/createProcessors.ts | 11 ++++++----- packages/uploads/src/prismaExtension.ts | 2 +- packages/uploads/src/setup.ts | 12 ++++++------ 3 files changed, 13 insertions(+), 12 deletions(-) diff --git a/packages/uploads/src/createProcessors.ts b/packages/uploads/src/createProcessors.ts index 470d17342a84..a7e0349e0ae8 100644 --- a/packages/uploads/src/createProcessors.ts +++ b/packages/uploads/src/createProcessors.ts @@ -1,6 +1,5 @@ import { ulid } from 'ulid' -import type { UploadsConfig } from './prismaExtension.js' import type { SaveOptionsOverride, StorageAdapter } from './StorageAdapter.js' // Assumes you pass in the graphql type @@ -25,14 +24,16 @@ export const createFileListProcessor = (storage: StorageAdapter) => { This creates a processor for each model in the uploads config (i.e. tied to a model in the prisma schema) The processor will only handle single file uploads, not file lists. */ -export const createUploadProcessors = ( - uploadConfig: UploadsConfig, +export const createUploadProcessors = < + TUploadConfig extends Record, +>( + uploadConfig: TUploadConfig, storage: StorageAdapter, ) => { - type modelNamesInUploadConfig = keyof typeof uploadConfig + type modelNamesInUploadConfig = keyof TUploadConfig type uploadProcessorNames = - `process${Capitalize}Uploads` + `process${Capitalize}Uploads` type Processors = { [K in uploadProcessorNames]: >( diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 8dd07260f0d8..de81bb4da1e7 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -14,7 +14,7 @@ type FilterOutDollarPrefixed = T extends `$${string}` : T // Filter out $on, $connect, etc. -type ModelNames = FilterOutDollarPrefixed +export type ModelNames = FilterOutDollarPrefixed type PrismaModelFields = keyof Prisma.Result< PrismaClient[MName], diff --git a/packages/uploads/src/setup.ts b/packages/uploads/src/setup.ts index e21042326488..3b470bc7e501 100644 --- a/packages/uploads/src/setup.ts +++ b/packages/uploads/src/setup.ts @@ -2,21 +2,21 @@ import { createFileListProcessor, createUploadProcessors, } from './createProcessors.js' -import { - createUploadsExtension, - type UploadsConfig, -} from './prismaExtension.js' +import type { ModelNames, UploadsConfig } from './prismaExtension.js' +import { createUploadsExtension } from './prismaExtension.js' import type { StorageAdapter } from './StorageAdapter.js' -export const setupUploads = ( - uploadsConfig: UploadsConfig, +export const setupUploads = ( + uploadsConfig: UploadsConfig, storageAdapter: StorageAdapter, ) => { const prismaExtension = createUploadsExtension(uploadsConfig, storageAdapter) + const uploadsProcessors = createUploadProcessors( uploadsConfig, storageAdapter, ) + const fileListProcessor = createFileListProcessor(storageAdapter) return { From 4274d95d4a0265901975d0bb8d24eb9d05cf644d Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 20 Aug 2024 13:32:05 +0700 Subject: [PATCH 58/91] Refactor signed url generator --- .../src/__tests__/resultExtensions.test.ts | 47 +++-- packages/uploads/src/lib/signedUrls.ts | 163 ++++++++++-------- packages/uploads/src/prismaExtension.ts | 28 ++- packages/uploads/src/setup.ts | 13 +- 4 files changed, 140 insertions(+), 111 deletions(-) diff --git a/packages/uploads/src/__tests__/resultExtensions.test.ts b/packages/uploads/src/__tests__/resultExtensions.test.ts index 5daf296e21cd..28dbec3a296d 100644 --- a/packages/uploads/src/__tests__/resultExtensions.test.ts +++ b/packages/uploads/src/__tests__/resultExtensions.test.ts @@ -1,4 +1,4 @@ -import { describe, it, beforeAll, expect } from 'vitest' +import { describe, it, expect, vi } from 'vitest' import { MemoryStorage } from '../MemoryStorage.js' import type { UploadsConfig } from '../prismaExtension.js' @@ -7,6 +7,20 @@ import { setupUploads } from '../setup.js' // @MARK: use the local prisma client in the test import { PrismaClient } from './prisma-client/index.js' +vi.mock('@redwoodjs/project-config', async (importOriginal) => { + const originalProjectConfig = (await importOriginal()) as any + return { + ...originalProjectConfig, + getConfig: () => { + return { + web: { + apiUrl: '/.redwood/functions', + }, + } + }, + } +}) + describe('Result extensions', () => { const uploadConfig: UploadsConfig = { dummy: { @@ -22,14 +36,15 @@ describe('Result extensions', () => { new MemoryStorage({ baseDir: '/tmp', }), + { + endpoint: '/signed-url', + secret: 'my-sekret', + }, ) const prismaClient = new PrismaClient().$extends(prismaExtension) describe('withSignedUrl', () => { - beforeAll(() => { - process.env.RW_UPLOADS_SECRET = 'gasdg' - }) it('Generates signed urls for each upload field', async () => { const dumbo = await prismaClient.dumbo.create({ data: { @@ -38,30 +53,14 @@ describe('Result extensions', () => { }, }) - const signedUrlDumbo = await dumbo.withSignedUrl() + const signedUrlDumbo = await dumbo.withSignedUrl(254) + expect(signedUrlDumbo.firstUpload).toContain( + '/.redwood/functions/signed-url', + ) expect(signedUrlDumbo.firstUpload).toContain('path=%2Fdumbo%2Ffirst.txt') expect(signedUrlDumbo.secondUpload).toContain( 'path=%2Fdumbo%2Fsecond.txt', ) }) - - it('laskdng', async () => { - const customClient = new PrismaClient().$extends({ - result: { - dumbo: { - helloJosh: { - compute() { - return () => { - return 'hello josh' - } - }, - }, - }, - }, - }) - - const dumbo = await customClient.dumbo.findFirst({ where: { id: 1 } }) - dumbo?.helloJosh() - }) }) }) diff --git a/packages/uploads/src/lib/signedUrls.ts b/packages/uploads/src/lib/signedUrls.ts index b626e8e1dff1..5f4c240c4afb 100644 --- a/packages/uploads/src/lib/signedUrls.ts +++ b/packages/uploads/src/lib/signedUrls.ts @@ -1,72 +1,103 @@ import crypto from 'node:crypto' -export const generateSignature = (filePath: string, expiresInMs?: number) => { - if (!process.env.RW_UPLOADS_SECRET) { - throw new Error( - 'Please configure RW_UPLOADS_SECRET in your environment variables', - ) - } +import { getConfig } from '@redwoodjs/project-config' - if (expiresInMs) { - const expires = Date.now() + expiresInMs - const signature = crypto - .createHmac('sha256', process.env.RW_UPLOADS_SECRET) - .update(`${filePath}:${expires}`) - .digest('hex') - - return { expires, signature } - } else { - // Does not expire - const signature = crypto - .createHmac('sha256', process.env.RW_UPLOADS_SECRET) - .update(filePath) - .digest('hex') - - return { - signature, - expires: undefined, - } - } +export type SignedUrlSettings = { + endpoint: string // The path to the signed url endpoint, or a full url (include http(s)://) + secret: string // The secret to sign the urls with } -/** - * The signature and expires have to be extracted from the URL - */ -export const validateSignature = ({ - signature, - filePath, - expires, -}: { - filePath: string - signature: string - expires?: number -}) => { - // Note, expires not the same as expiresIn - if (!process.env.RW_UPLOADS_SECRET) { - throw new Error( - 'Please configure RW_UPLOADS_SECRET in your environment variables', - ) +export class UrlSigner { + private secret: string + private endpoint: string + + constructor({ secret, endpoint }: SignedUrlSettings) { + this.secret = secret + this.endpoint = endpoint + + this.endpoint = endpoint.startsWith('http') + ? endpoint + : `${getConfig().web.apiUrl}${endpoint}` } - if (expires) { - // No need to validate if the signature has expired - if (Date.now() > expires) { - throw new Error('Signature has expired') + generateSignature(filePath: string, expiresInMs?: number) { + if (!this.secret) { + throw new Error('Please configure the secret') } - } - const validSignature = expires - ? crypto - .createHmac('sha256', process.env.RW_UPLOADS_SECRET) + if (expiresInMs) { + const expires = Date.now() + expiresInMs + const signature = crypto + .createHmac('sha256', this.secret) .update(`${filePath}:${expires}`) .digest('hex') - : crypto - .createHmac('sha256', process.env.RW_UPLOADS_SECRET) - .update(`${filePath}`) + + return { expires, signature } + } else { + // Does not expire + const signature = crypto + .createHmac('sha256', this.secret) + .update(filePath) .digest('hex') - if (validSignature !== signature) { - throw new Error('Invalid signature') + return { + signature, + expires: undefined, + } + } + } + + /** + * The signature and expires have to be extracted from the URL + */ + validateSignature({ + signature, + filePath, + expires, + }: { + filePath: string + signature: string + expires?: number + }) { + if (!this.secret) { + throw new Error('Please configure the secret') + } + + if (expires) { + // No need to validate if the signature has expired + if (Date.now() > expires) { + throw new Error('Signature has expired') + } + } + + const validSignature = expires + ? crypto + .createHmac('sha256', this.secret) + .update(`${filePath}:${expires}`) + .digest('hex') + : crypto + .createHmac('sha256', this.secret) + .update(`${filePath}`) + .digest('hex') + + if (validSignature !== signature) { + throw new Error('Invalid signature') + } + } + + generateSignedUrl(filePath: string, expiresIn?: number) { + const { signature, expires } = this.generateSignature(filePath, expiresIn) + + // This way you can pass in a path with params already + const params = new URLSearchParams() + params.set('s', signature) + if (expires) { + params.set('expires', expires.toString()) + } + + params.set('path', filePath) + + return `${this.endpoint}?${params.toString()}` } } @@ -80,26 +111,6 @@ export const getSignedDetailsFromUrl = (url: string) => { } } -type SigningParms = { filePath: string; expiresIn?: number } - -export const generateSignedQueryParams = ( - endpoint: string, - { filePath, expiresIn }: SigningParms, -) => { - const { signature, expires } = generateSignature(filePath, expiresIn) - - // This way you can pass in a path with params already - const params = new URLSearchParams() - params.set('s', signature) - if (expires) { - params.set('expires', expires.toString()) - } - - params.set('path', filePath) - - return `${endpoint}?${params.toString()}` -} - export const EXPIRES_IN = { seconds: (s: number) => s * 1000, minutes: (m: number) => m * 60 * 1000, diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index de81bb4da1e7..5e30eacf1952 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -3,8 +3,10 @@ import type { Prisma } from '@prisma/client' import { Prisma as PrismaExtension } from '@prisma/client/extension' import type * as runtime from '@prisma/client/runtime/library' + import { fileToDataUri } from './fileSave.utils.js' -import { generateSignedQueryParams } from './lib/signedUrls.js' +import type { SignedUrlSettings } from './lib/signedUrls.js' +import { UrlSigner } from './lib/signedUrls.js' import type { StorageAdapter } from './StorageAdapter.js' type FilterOutDollarPrefixed = T extends `$${string}` @@ -26,9 +28,6 @@ export type UploadConfigForModel = { fields: | PrismaModelFields | PrismaModelFields[] - savePath?: ((args: unknown) => string) | string - fileName?: (args: unknown) => string - onFileSaved?: (filePath: string) => void | Promise } export type UploadsConfig = { @@ -38,11 +37,17 @@ export type UploadsConfig = { export const createUploadsExtension = ( config: UploadsConfig, storageAdapter: StorageAdapter, + signedUrlSettings?: SignedUrlSettings, ) => { // @TODO I think we can use Prisma.getExtensionContext(this) // instead of creating a new PrismaClient instance const prismaInstance = new PrismaClient() + let signedUrlGenerator: UrlSigner + if (signedUrlSettings) { + signedUrlGenerator = new UrlSigner(signedUrlSettings) + } + type ResultExtends = { [K in MNames]: { withDataUri: { @@ -55,7 +60,7 @@ export const createUploadsExtension = ( needs: Record compute: ( modelData: Record, - ) => (this: T) => Promise + ) => (this: T, expiresIn?: number) => Promise } } } @@ -179,13 +184,18 @@ export const createUploadsExtension = ( needs, compute(modelData) { return (expiresIn?: number) => { + if (!signedUrlGenerator) { + throw new Error( + 'Please supply signed url settings in setupUpload()', + ) + } const signedUrlFields: Record = {} for (const field of uploadFields) { - signedUrlFields[field] = generateSignedQueryParams('/HARDCODED', { - filePath: modelData[field] as string, - expiresIn: expiresIn, - }) + signedUrlFields[field] = signedUrlGenerator.generateSignedUrl( + modelData[field] as string, + expiresIn, + ) } return { diff --git a/packages/uploads/src/setup.ts b/packages/uploads/src/setup.ts index 3b470bc7e501..d7d20d894316 100644 --- a/packages/uploads/src/setup.ts +++ b/packages/uploads/src/setup.ts @@ -2,15 +2,24 @@ import { createFileListProcessor, createUploadProcessors, } from './createProcessors.js' -import type { ModelNames, UploadsConfig } from './prismaExtension.js' +import type { + ModelNames, + SignedUrlSettings, + UploadsConfig, +} from './prismaExtension.js' import { createUploadsExtension } from './prismaExtension.js' import type { StorageAdapter } from './StorageAdapter.js' export const setupUploads = ( uploadsConfig: UploadsConfig, storageAdapter: StorageAdapter, + signedUrlSettings?: SignedUrlSettings, ) => { - const prismaExtension = createUploadsExtension(uploadsConfig, storageAdapter) + const prismaExtension = createUploadsExtension( + uploadsConfig, + storageAdapter, + signedUrlSettings, + ) const uploadsProcessors = createUploadProcessors( uploadsConfig, From 33926d6de6d49111feab2e50741b27ff39d84c98 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 20 Aug 2024 13:50:01 +0700 Subject: [PATCH 59/91] Update tests too --- .../uploads/src/__tests__/signedUrls.test.ts | 106 ++++++++---------- packages/uploads/src/lib/signedUrls.ts | 33 ++++-- 2 files changed, 70 insertions(+), 69 deletions(-) diff --git a/packages/uploads/src/__tests__/signedUrls.test.ts b/packages/uploads/src/__tests__/signedUrls.test.ts index b5e517ef1fb9..72a021e28ab7 100644 --- a/packages/uploads/src/__tests__/signedUrls.test.ts +++ b/packages/uploads/src/__tests__/signedUrls.test.ts @@ -1,32 +1,23 @@ -import { - beforeAll, - describe, - expect, - beforeEach, - afterEach, - vi, - it, - test, -} from 'vitest' +import { describe, expect, beforeEach, afterEach, vi, it, test } from 'vitest' import { EXPIRES_IN, - generateSignature, - generateSignedQueryParams, + UrlSigner, getSignedDetailsFromUrl, - validateSignature, } from '../lib/signedUrls.js' -describe('Signed URLs', () => { - beforeAll(() => { - process.env.RW_UPLOADS_SECRET = 'bazinga' - }) +const signer = new UrlSigner({ + // Doing this means we don't need to mock getConfig + endpoint: 'https://myapiside.com/access-signed-file', + secret: 'bazinga-3-32-151', +}) +describe('UrlSigner', () => { it('Can creates a signature', () => { - const { signature, expires } = generateSignature( - '/tmp/myfile.txt', - EXPIRES_IN.days(5), - ) + const { signature, expiry: expires } = signer.generateSignature({ + filePath: '/tmp/myfile.txt', + expiresInMs: EXPIRES_IN.days(5), + }) expect(signature).toBeDefined() @@ -34,67 +25,68 @@ describe('Signed URLs', () => { }) it('throws with correct error when wrong expires passed', () => { - const { signature, expires } = generateSignature( - '/tmp/myfile.txt', - EXPIRES_IN.days(1), - ) + const { signature, expiry: expires } = signer.generateSignature({ + filePath: '/tmp/myfile.txt', + expiresInMs: EXPIRES_IN.days(1), + }) expect(() => - validateSignature({ + signer.validateSignature({ filePath: '/tmp/myfile.txt', signature, - expires, + expiry: expires, }), ).not.toThrow() expect(() => - validateSignature({ + signer.validateSignature({ filePath: '/tmp/myfile.txt', signature, - expires: 12512351, + expiry: 12512351, }), ).toThrowError('Signature has expired') }) it('Throws an invalid signature when signature is wrong', () => { - const { signature, expires } = generateSignature( - '/tmp/myfile.txt', - EXPIRES_IN.days(1), - ) + const { signature, expiry } = signer.generateSignature({ + filePath: '/tmp/myfile.txt', + expiresInMs: EXPIRES_IN.days(1), + }) expect(() => - validateSignature({ + signer.validateSignature({ filePath: '/tmp/myfile.txt', signature, - expires, + expiry, }), ).not.toThrow() expect(() => - validateSignature({ + signer.validateSignature({ filePath: '/tmp/myfile.txt', signature: 'im-the-wrong-signature', - expires, + expiry, }), ).toThrowError('Invalid signature') }) it('Throws an invalid signature when file path is wrong', () => { - const { signature, expires } = generateSignature( - '/tmp/myfile.txt', - EXPIRES_IN.days(20), - ) + const { signature, expiry } = signer.generateSignature({ + filePath: '/tmp/myfile.txt', + expiresInMs: EXPIRES_IN.days(20), + }) expect(() => - validateSignature({ + signer.validateSignature({ filePath: '/tmp/some-other-file.txt', signature, - expires, + expiry, }), ).toThrowError('Invalid signature') }) }) describe('Expired signature', () => { + // Seprate, so we can mock the times beforeEach(() => { vi.useFakeTimers() }) @@ -105,16 +97,16 @@ describe('Expired signature', () => { it('throws an error when the signature has expired', () => { const filePath = '/bazinga/kittens.png' - const { signature, expires } = generateSignature( + const { signature, expiry } = signer.generateSignature({ filePath, - EXPIRES_IN.minutes(15), - ) + expiresInMs: EXPIRES_IN.minutes(15), + }) const validation = () => - validateSignature({ + signer.validateSignature({ filePath, signature, - expires, + expiry, }) expect(validation).not.toThrow() @@ -138,15 +130,15 @@ test('Parses details related to signatures from a url string', () => { }) test('Generates a signed url', () => { - const signedQueryParams = generateSignedQueryParams('/files/bazinga', { - filePath: '/path/to/hello.txt', - expiresIn: EXPIRES_IN.days(1), - }) - - expect(signedQueryParams).toContain('/files/bazinga?s=') - expect(signedQueryParams).toContain('s=') - expect(signedQueryParams).toContain('expires=') - expect(signedQueryParams).toContain('path=') // The actual file path + const signedUrl = signer.generateSignedUrl( + '/files/bazinga', + EXPIRES_IN.days(1), + ) + + expect(signedUrl).toContain('https://myapiside.com/access-signed-file?s=') + expect(signedUrl).toMatch(/s=.*/) + expect(signedUrl).toMatch(/expires=[0-9]+/) + expect(signedUrl).toContain(`path=${encodeURIComponent('/files/bazinga')}`) // The actual file path }) // Util functions to make the tests more readable diff --git a/packages/uploads/src/lib/signedUrls.ts b/packages/uploads/src/lib/signedUrls.ts index 5f4c240c4afb..f15280400eaf 100644 --- a/packages/uploads/src/lib/signedUrls.ts +++ b/packages/uploads/src/lib/signedUrls.ts @@ -20,19 +20,25 @@ export class UrlSigner { : `${getConfig().web.apiUrl}${endpoint}` } - generateSignature(filePath: string, expiresInMs?: number) { + generateSignature({ + filePath, + expiresInMs, + }: { + filePath: string + expiresInMs?: number + }) { if (!this.secret) { throw new Error('Please configure the secret') } if (expiresInMs) { - const expires = Date.now() + expiresInMs + const expiry = Date.now() + expiresInMs const signature = crypto .createHmac('sha256', this.secret) - .update(`${filePath}:${expires}`) + .update(`${filePath}:${expiry}`) .digest('hex') - return { expires, signature } + return { expiry, signature } } else { // Does not expire const signature = crypto @@ -42,7 +48,7 @@ export class UrlSigner { return { signature, - expires: undefined, + expiry: undefined, } } } @@ -53,27 +59,27 @@ export class UrlSigner { validateSignature({ signature, filePath, - expires, + expiry, }: { filePath: string signature: string - expires?: number + expiry?: number }) { if (!this.secret) { throw new Error('Please configure the secret') } - if (expires) { + if (expiry) { // No need to validate if the signature has expired - if (Date.now() > expires) { + if (Date.now() > expiry) { throw new Error('Signature has expired') } } - const validSignature = expires + const validSignature = expiry ? crypto .createHmac('sha256', this.secret) - .update(`${filePath}:${expires}`) + .update(`${filePath}:${expiry}`) .digest('hex') : crypto .createHmac('sha256', this.secret) @@ -86,7 +92,10 @@ export class UrlSigner { } generateSignedUrl(filePath: string, expiresIn?: number) { - const { signature, expires } = this.generateSignature(filePath, expiresIn) + const { signature, expiry: expires } = this.generateSignature({ + filePath, + expiresInMs: expiresIn, + }) // This way you can pass in a path with params already const params = new URLSearchParams() From 4cebfd19112316afe7810291058c6a5442519dc2 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 20 Aug 2024 14:33:49 +0700 Subject: [PATCH 60/91] Pass urlSigner instance to setup --- .../uploads/src/__tests__/resultExtensions.test.ts | 5 +++-- packages/uploads/src/prismaExtension.ts | 14 ++++---------- packages/uploads/src/setup.ts | 11 ++++------- 3 files changed, 11 insertions(+), 19 deletions(-) diff --git a/packages/uploads/src/__tests__/resultExtensions.test.ts b/packages/uploads/src/__tests__/resultExtensions.test.ts index 28dbec3a296d..23b81e051e28 100644 --- a/packages/uploads/src/__tests__/resultExtensions.test.ts +++ b/packages/uploads/src/__tests__/resultExtensions.test.ts @@ -1,5 +1,6 @@ import { describe, it, expect, vi } from 'vitest' +import { UrlSigner } from '../lib/signedUrls.js' import { MemoryStorage } from '../MemoryStorage.js' import type { UploadsConfig } from '../prismaExtension.js' import { setupUploads } from '../setup.js' @@ -36,10 +37,10 @@ describe('Result extensions', () => { new MemoryStorage({ baseDir: '/tmp', }), - { + new UrlSigner({ endpoint: '/signed-url', secret: 'my-sekret', - }, + }), ) const prismaClient = new PrismaClient().$extends(prismaExtension) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 5e30eacf1952..c28f10b3006d 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -5,8 +5,7 @@ import type * as runtime from '@prisma/client/runtime/library' import { fileToDataUri } from './fileSave.utils.js' -import type { SignedUrlSettings } from './lib/signedUrls.js' -import { UrlSigner } from './lib/signedUrls.js' +import type { UrlSigner } from './lib/signedUrls.js' import type { StorageAdapter } from './StorageAdapter.js' type FilterOutDollarPrefixed = T extends `$${string}` @@ -37,17 +36,12 @@ export type UploadsConfig = { export const createUploadsExtension = ( config: UploadsConfig, storageAdapter: StorageAdapter, - signedUrlSettings?: SignedUrlSettings, + urlSigner?: UrlSigner, ) => { // @TODO I think we can use Prisma.getExtensionContext(this) // instead of creating a new PrismaClient instance const prismaInstance = new PrismaClient() - let signedUrlGenerator: UrlSigner - if (signedUrlSettings) { - signedUrlGenerator = new UrlSigner(signedUrlSettings) - } - type ResultExtends = { [K in MNames]: { withDataUri: { @@ -184,7 +178,7 @@ export const createUploadsExtension = ( needs, compute(modelData) { return (expiresIn?: number) => { - if (!signedUrlGenerator) { + if (!urlSigner) { throw new Error( 'Please supply signed url settings in setupUpload()', ) @@ -192,7 +186,7 @@ export const createUploadsExtension = ( const signedUrlFields: Record = {} for (const field of uploadFields) { - signedUrlFields[field] = signedUrlGenerator.generateSignedUrl( + signedUrlFields[field] = urlSigner.generateSignedUrl( modelData[field] as string, expiresIn, ) diff --git a/packages/uploads/src/setup.ts b/packages/uploads/src/setup.ts index d7d20d894316..589840b7e5e5 100644 --- a/packages/uploads/src/setup.ts +++ b/packages/uploads/src/setup.ts @@ -2,23 +2,20 @@ import { createFileListProcessor, createUploadProcessors, } from './createProcessors.js' -import type { - ModelNames, - SignedUrlSettings, - UploadsConfig, -} from './prismaExtension.js' +import type { UrlSigner } from './lib/signedUrls.js' +import type { ModelNames, UploadsConfig } from './prismaExtension.js' import { createUploadsExtension } from './prismaExtension.js' import type { StorageAdapter } from './StorageAdapter.js' export const setupUploads = ( uploadsConfig: UploadsConfig, storageAdapter: StorageAdapter, - signedUrlSettings?: SignedUrlSettings, + urlSigner?: UrlSigner, ) => { const prismaExtension = createUploadsExtension( uploadsConfig, storageAdapter, - signedUrlSettings, + urlSigner, ) const uploadsProcessors = createUploadProcessors( From d2b57f7b72cac5e80472d3482c127e80823d5a74 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 20 Aug 2024 17:08:30 +0700 Subject: [PATCH 61/91] Refactor exports, etc. --- packages/uploads/build.mts | 31 ++++++++++--------- packages/uploads/package.json | 28 +++++++++++++++-- packages/uploads/src/FileSystemStorage.ts | 14 +++++++-- .../src/__tests__/queryExtensions.test.ts | 2 +- .../src/__tests__/resultExtensions.test.ts | 4 +-- .../uploads/src/__tests__/signedUrls.test.ts | 2 +- packages/uploads/src/{setup.ts => index.ts} | 4 ++- packages/uploads/src/prismaExtension.ts | 6 +++- packages/uploads/src/{lib => }/signedUrls.ts | 0 9 files changed, 65 insertions(+), 26 deletions(-) rename packages/uploads/src/{setup.ts => index.ts} (87%) rename packages/uploads/src/{lib => }/signedUrls.ts (100%) diff --git a/packages/uploads/build.mts b/packages/uploads/build.mts index d1ec3b003bfb..da389f21e834 100644 --- a/packages/uploads/build.mts +++ b/packages/uploads/build.mts @@ -1,32 +1,33 @@ -import { writeFileSync } from 'node:fs' - import { build, defaultBuildOptions } from '@redwoodjs/framework-tools' -import { generateCjsTypes } from '@redwoodjs/framework-tools/cjsTypes' +import { + generateTypesCjs, + generateTypesEsm, + insertCommonJsPackageJson, +} from '@redwoodjs/framework-tools/generateTypes' -// CJS build +// ESM build await build({ buildOptions: { ...defaultBuildOptions, - outdir: 'dist/cjs', + format: 'esm', packages: 'external', }, }) -// ESM build +await generateTypesEsm() + +// CJS build await build({ buildOptions: { ...defaultBuildOptions, - format: 'esm', + outdir: 'dist/cjs', packages: 'external', }, }) -// Place a package.json file with `type: commonjs` in the dist/cjs folder so that -// all .js files are treated as CommonJS files. -writeFileSync('dist/cjs/package.json', JSON.stringify({ type: 'commonjs' })) - -// Place a package.json file with `type: module` in the dist folder so that -// all .js files are treated as ES Module files. -writeFileSync('dist/package.json', JSON.stringify({ type: 'module' })) +await generateTypesCjs() -await generateCjsTypes() +await insertCommonJsPackageJson({ + buildFileUrl: import.meta.url, + cjsDir: 'dist/cjs', +}) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 6b40f3742880..7b9fb8372c20 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -1,15 +1,36 @@ { "name": "@redwoodjs/uploads", - "type": "module", "version": "7.0.0", - "types": "dist/prismaExtension.d.ts", "repository": { "type": "git", "url": "git+https://github.com/redwoodjs/redwood.git", "directory": "packages/uploads" }, "license": "MIT", + "type": "module", "exports": { + ".": { + "require": { + "types": "./dist/cjs/index.d.ts", + "default": "./dist/cjs/index.js" + }, + "import": { + "types": "./dist/index.d.ts", + "default": "./dist/index.js" + } + }, + "./FileSystemStorage": { + "require": "./dist/cjs/FileSystemStorage.js", + "import": "./dist/FileSystemStorage.js" + }, + "./MemoryStorage": { + "require": "./dist/cjs/MemoryStorage.js", + "import": "./dist/MemoryStorage.js" + }, + "./signedUrl": { + "require": "./dist/cjs/signedUrls.js", + "import": "./dist/signedUrls.js" + }, "./prisma": { "require": { "types": "./dist/cjs/prismaExtension.d.ts", @@ -21,6 +42,7 @@ } } }, + "types": "dist/prismaExtension.d.ts", "files": [ "dist", "prisma" @@ -56,4 +78,4 @@ "vitest": "2.0.4" }, "gitHead": "3905ed045508b861b495f8d5630d76c7a157d8f1" -} +} \ No newline at end of file diff --git a/packages/uploads/src/FileSystemStorage.ts b/packages/uploads/src/FileSystemStorage.ts index 653305fd1384..d99af186d90a 100644 --- a/packages/uploads/src/FileSystemStorage.ts +++ b/packages/uploads/src/FileSystemStorage.ts @@ -1,11 +1,21 @@ +import { existsSync, mkdirSync } from 'node:fs' import fs from 'node:fs/promises' import path from 'node:path' - import type { SaveOptionsOverride } from './StorageAdapter.js' import { StorageAdapter } from './StorageAdapter.js' -export class FileSystemStorage extends StorageAdapter implements StorageAdapter { +export class FileSystemStorage + extends StorageAdapter + implements StorageAdapter +{ + constructor(opts: { baseDir: string }) { + super(opts) + if (!existsSync(opts.baseDir)) { + console.log('Creating baseDir', opts.baseDir) + mkdirSync(opts.baseDir, { recursive: true }) + } + } async save(file: File, saveOverride?: SaveOptionsOverride) { const fileName = this.generateFileNameWithExtension(saveOverride, file) diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts index 7a5b2cc1f4c1..2e5d1f9914ef 100644 --- a/packages/uploads/src/__tests__/queryExtensions.test.ts +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -4,7 +4,7 @@ import { describe, it, vi, expect, beforeEach } from 'vitest' import { FileSystemStorage } from '../FileSystemStorage.js' import type { UploadsConfig } from '../prismaExtension.js' -import { setupUploads } from '../setup.js' +import { setupUploads } from '../index.js' // @MARK: use the local prisma client in the test import { PrismaClient } from './prisma-client/index.js' diff --git a/packages/uploads/src/__tests__/resultExtensions.test.ts b/packages/uploads/src/__tests__/resultExtensions.test.ts index 23b81e051e28..89165925afb0 100644 --- a/packages/uploads/src/__tests__/resultExtensions.test.ts +++ b/packages/uploads/src/__tests__/resultExtensions.test.ts @@ -1,9 +1,9 @@ import { describe, it, expect, vi } from 'vitest' -import { UrlSigner } from '../lib/signedUrls.js' +import { setupUploads } from '../index.js' import { MemoryStorage } from '../MemoryStorage.js' import type { UploadsConfig } from '../prismaExtension.js' -import { setupUploads } from '../setup.js' +import { UrlSigner } from '../signedUrls.js' // @MARK: use the local prisma client in the test import { PrismaClient } from './prisma-client/index.js' diff --git a/packages/uploads/src/__tests__/signedUrls.test.ts b/packages/uploads/src/__tests__/signedUrls.test.ts index 72a021e28ab7..f02caf5474d0 100644 --- a/packages/uploads/src/__tests__/signedUrls.test.ts +++ b/packages/uploads/src/__tests__/signedUrls.test.ts @@ -4,7 +4,7 @@ import { EXPIRES_IN, UrlSigner, getSignedDetailsFromUrl, -} from '../lib/signedUrls.js' +} from '../signedUrls.js' const signer = new UrlSigner({ // Doing this means we don't need to mock getConfig diff --git a/packages/uploads/src/setup.ts b/packages/uploads/src/index.ts similarity index 87% rename from packages/uploads/src/setup.ts rename to packages/uploads/src/index.ts index 589840b7e5e5..b02b65f0bd5a 100644 --- a/packages/uploads/src/setup.ts +++ b/packages/uploads/src/index.ts @@ -2,9 +2,9 @@ import { createFileListProcessor, createUploadProcessors, } from './createProcessors.js' -import type { UrlSigner } from './lib/signedUrls.js' import type { ModelNames, UploadsConfig } from './prismaExtension.js' import { createUploadsExtension } from './prismaExtension.js' +import type { UrlSigner } from './signedUrls.js' import type { StorageAdapter } from './StorageAdapter.js' export const setupUploads = ( @@ -31,3 +31,5 @@ export const setupUploads = ( fileListProcessor, } } + +export type { ModelNames, UploadsConfig } from './prismaExtension.js' diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index c28f10b3006d..0f5e28b93c2a 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -5,7 +5,7 @@ import type * as runtime from '@prisma/client/runtime/library' import { fileToDataUri } from './fileSave.utils.js' -import type { UrlSigner } from './lib/signedUrls.js' +import type { UrlSigner } from './signedUrls.js' import type { StorageAdapter } from './StorageAdapter.js' type FilterOutDollarPrefixed = T extends `$${string}` @@ -186,6 +186,10 @@ export const createUploadsExtension = ( const signedUrlFields: Record = {} for (const field of uploadFields) { + if (!signedUrlFields[field]) { + continue + } + signedUrlFields[field] = urlSigner.generateSignedUrl( modelData[field] as string, expiresIn, diff --git a/packages/uploads/src/lib/signedUrls.ts b/packages/uploads/src/signedUrls.ts similarity index 100% rename from packages/uploads/src/lib/signedUrls.ts rename to packages/uploads/src/signedUrls.ts From 2119b5d2fd2872393be9b73d4b6d7a976cc0e98a Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 20 Aug 2024 17:11:26 +0700 Subject: [PATCH 62/91] Fix signedurl creation --- packages/uploads/src/__tests__/queryExtensions.test.ts | 9 ++++++++- packages/uploads/src/prismaExtension.ts | 2 +- 2 files changed, 9 insertions(+), 2 deletions(-) diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts index 2e5d1f9914ef..4fb12bcbeaf0 100644 --- a/packages/uploads/src/__tests__/queryExtensions.test.ts +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -3,8 +3,8 @@ import fs from 'node:fs/promises' import { describe, it, vi, expect, beforeEach } from 'vitest' import { FileSystemStorage } from '../FileSystemStorage.js' -import type { UploadsConfig } from '../prismaExtension.js' import { setupUploads } from '../index.js' +import type { UploadsConfig } from '../prismaExtension.js' // @MARK: use the local prisma client in the test import { PrismaClient } from './prisma-client/index.js' @@ -19,6 +19,13 @@ vi.mock('node:fs/promises', () => ({ copyFile: vi.fn(), }, })) + +// For creation of FS adapter +vi.mock('node:fs', () => ({ + existsSync: vi.fn(() => true), + mkdirSync: vi.fn(), +})) + describe('Query extensions', () => { const uploadConfig: UploadsConfig = { dummy: { diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 0f5e28b93c2a..00c216a05d2a 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -186,7 +186,7 @@ export const createUploadsExtension = ( const signedUrlFields: Record = {} for (const field of uploadFields) { - if (!signedUrlFields[field]) { + if (!modelData[field]) { continue } From cb08dc2e7ada4eb5281c1ca22c75db1f2620912c Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 20 Aug 2024 22:16:53 +0700 Subject: [PATCH 63/91] Fix signing by using s instead of signature as argument Add read to base adapter --- packages/uploads/src/FileSystemStorage.ts | 9 ++ packages/uploads/src/MemoryStorage.ts | 7 +- packages/uploads/src/StorageAdapter.ts | 5 +- .../src/__tests__/createProcessors.test.ts | 6 +- .../uploads/src/__tests__/signedUrls.test.ts | 132 ++++++++++++++---- packages/uploads/src/signedUrls.ts | 58 ++++++-- 6 files changed, 169 insertions(+), 48 deletions(-) diff --git a/packages/uploads/src/FileSystemStorage.ts b/packages/uploads/src/FileSystemStorage.ts index d99af186d90a..de054a84ff8d 100644 --- a/packages/uploads/src/FileSystemStorage.ts +++ b/packages/uploads/src/FileSystemStorage.ts @@ -2,6 +2,8 @@ import { existsSync, mkdirSync } from 'node:fs' import fs from 'node:fs/promises' import path from 'node:path' +import mime from 'mime-types' + import type { SaveOptionsOverride } from './StorageAdapter.js' import { StorageAdapter } from './StorageAdapter.js' @@ -29,6 +31,13 @@ export class FileSystemStorage return { location } } + async read(filePath: string) { + return { + contents: await fs.readFile(filePath), + type: mime.lookup(filePath), + } + } + async remove(filePath: string) { await fs.unlink(filePath) } diff --git a/packages/uploads/src/MemoryStorage.ts b/packages/uploads/src/MemoryStorage.ts index f02ea05a9220..905ee7b8e2ff 100644 --- a/packages/uploads/src/MemoryStorage.ts +++ b/packages/uploads/src/MemoryStorage.ts @@ -1,5 +1,6 @@ import path from 'node:path' +import mime from 'mime-types' import { StorageAdapter } from './StorageAdapter.js' import type { SaveOptionsOverride } from './StorageAdapter.js' @@ -27,9 +28,11 @@ export class MemoryStorage extends StorageAdapter implements StorageAdapter { delete this.store[filePath] } - // Not sure about read method... should it be in the base class? async read(filePath: string) { - return this.store[filePath] + return { + contents: this.store[filePath], + type: mime.lookup(filePath), + } } async clear() { diff --git a/packages/uploads/src/StorageAdapter.ts b/packages/uploads/src/StorageAdapter.ts index 64a3d0d288d7..c2df085752bc 100644 --- a/packages/uploads/src/StorageAdapter.ts +++ b/packages/uploads/src/StorageAdapter.ts @@ -48,5 +48,8 @@ export abstract class StorageAdapter { saveOpts?: SaveOptionsOverride, ): Promise abstract remove(fileLocation: AdapterResult['location']): Promise - // abstract replace(fileId: string, file: File): Promise + abstract read(fileLocation: AdapterResult['location']): Promise<{ + contents: Buffer | string + type: ReturnType + }> } diff --git a/packages/uploads/src/__tests__/createProcessors.test.ts b/packages/uploads/src/__tests__/createProcessors.test.ts index e2624e7e8975..5db76b6b46a5 100644 --- a/packages/uploads/src/__tests__/createProcessors.test.ts +++ b/packages/uploads/src/__tests__/createProcessors.test.ts @@ -48,10 +48,12 @@ describe('Create processors', () => { /\/memory_store_basedir\/dumbo-*.*\.txt/, ) - const firstContents = await memStore.read(result.firstUpload) + const { contents: firstContents } = await memStore.read(result.firstUpload) expect(firstContents.toString()).toBe('Meaow') - const secondContents = await memStore.read(result.secondUpload) + const { contents: secondContents } = await memStore.read( + result.secondUpload, + ) expect(secondContents.toString()).toBe('Woof') }) diff --git a/packages/uploads/src/__tests__/signedUrls.test.ts b/packages/uploads/src/__tests__/signedUrls.test.ts index f02caf5474d0..10d9747de74d 100644 --- a/packages/uploads/src/__tests__/signedUrls.test.ts +++ b/packages/uploads/src/__tests__/signedUrls.test.ts @@ -1,10 +1,6 @@ import { describe, expect, beforeEach, afterEach, vi, it, test } from 'vitest' -import { - EXPIRES_IN, - UrlSigner, - getSignedDetailsFromUrl, -} from '../signedUrls.js' +import { EXPIRES_IN, UrlSigner } from '../signedUrls.js' const signer = new UrlSigner({ // Doing this means we don't need to mock getConfig @@ -32,21 +28,36 @@ describe('UrlSigner', () => { expect(() => signer.validateSignature({ - filePath: '/tmp/myfile.txt', - signature, + path: '/tmp/myfile.txt', + s: signature, expiry: expires, }), ).not.toThrow() expect(() => signer.validateSignature({ - filePath: '/tmp/myfile.txt', - signature, + path: '/tmp/myfile.txt', + s: signature, expiry: 12512351, }), ).toThrowError('Signature has expired') }) + it('Handles url encoded filePaths', () => { + const { signature, expiry: expires } = signer.generateSignature({ + filePath: '/tmp/myfile.txt', + expiresInMs: EXPIRES_IN.days(1), + }) + + expect(() => + signer.validateSignature({ + path: encodeURIComponent('/tmp/myfile.txt'), + s: signature, + expiry: expires, + }), + ).not.toThrow() + }) + it('Throws an invalid signature when signature is wrong', () => { const { signature, expiry } = signer.generateSignature({ filePath: '/tmp/myfile.txt', @@ -55,16 +66,16 @@ describe('UrlSigner', () => { expect(() => signer.validateSignature({ - filePath: '/tmp/myfile.txt', - signature, + path: '/tmp/myfile.txt', + s: signature, expiry, }), ).not.toThrow() expect(() => signer.validateSignature({ - filePath: '/tmp/myfile.txt', - signature: 'im-the-wrong-signature', + path: '/tmp/myfile.txt', + s: 'im-the-wrong-signature', expiry, }), ).toThrowError('Invalid signature') @@ -77,8 +88,8 @@ describe('UrlSigner', () => { }) expect(() => signer.validateSignature({ - filePath: '/tmp/some-other-file.txt', - signature, + path: '/tmp/some-other-file.txt', + s: signature, expiry, }), ).toThrowError('Invalid signature') @@ -104,8 +115,8 @@ describe('Expired signature', () => { const validation = () => signer.validateSignature({ - filePath, - signature, + path: filePath, + s: signature, expiry, }) @@ -118,17 +129,6 @@ describe('Expired signature', () => { }) }) -test('Parses details related to signatures from a url string', () => { - const url = - 'https://example.com/signedFile?file=/path/to/hello.txt&s=s1gnatur3&expires=123123' - - const { file, signature, expires } = getSignedDetailsFromUrl(url) - - expect(file).toBe('/path/to/hello.txt') - expect(signature).toBe('s1gnatur3') - expect(expires).toBe(123123) -}) - test('Generates a signed url', () => { const signedUrl = signer.generateSignedUrl( '/files/bazinga', @@ -137,10 +137,84 @@ test('Generates a signed url', () => { expect(signedUrl).toContain('https://myapiside.com/access-signed-file?s=') expect(signedUrl).toMatch(/s=.*/) - expect(signedUrl).toMatch(/expires=[0-9]+/) + expect(signedUrl).toMatch(/expiry=[0-9]+/) expect(signedUrl).toContain(`path=${encodeURIComponent('/files/bazinga')}`) // The actual file path }) +describe('validatePath', () => { + beforeEach(() => { + vi.useFakeTimers() + }) + + afterEach(() => { + vi.useRealTimers() + }) + + it('validates a path or url with a valid signature and expiry', () => { + const filePath = '/tmp/myfile.txt' + const expiresInMs = EXPIRES_IN.days(1) + const { signature, expiry } = signer.generateSignature({ + filePath, + expiresInMs, + }) + + const signedPath = `/bazinga?s=${signature}&expiry=${expiry}&path=${encodeURIComponent( + filePath, + )}` + + // When its just a path + expect(() => signer.validateSignedUrl(signedPath)).not.toThrow() + expect(signer.validateSignedUrl(signedPath)).toBe(filePath) + + // When its a full url + const signedUrl = `https://myredwoodapp.com/bazinga?s=${signature}&expiry=${expiry}&path=${encodeURIComponent( + filePath, + )}` + + expect(() => signer.validateSignedUrl(signedUrl)).not.toThrow() + expect(signer.validateSignedUrl(signedUrl)).toBe(filePath) + }) + + it('throws an error when the signature has expired', () => { + const filePath = '/tmp/myfile.txt' + const expiresInMs = EXPIRES_IN.minutes(15) + const { signature, expiry } = signer.generateSignature({ + filePath, + expiresInMs, + }) + + const url = `/bazinga?s=${signature}&expiry=${expiry}&path=${encodeURIComponent( + filePath, + )}` + + // Time travel to the future + vi.advanceTimersByTime(EXPIRES_IN.days(1)) + + expect(() => signer.validateSignedUrl(url)).toThrowError( + 'Signature has expired', + ) + }) + + it('throws an error when the signature is invalid', () => { + const filePath = '/tmp/myfile.txt' + const expiresInMs = EXPIRES_IN.days(1) + const { signature, expiry } = signer.generateSignature({ + filePath, + expiresInMs, + }) + + const url = `/bazinga?s=${signature}&expiry=${expiry}&path=${encodeURIComponent( + filePath, + )}` + + const invalidSignatureUrl = url.replace(signature, 'invalid-signature') + + expect(() => signer.validateSignedUrl(invalidSignatureUrl)).toThrowError( + 'Invalid signature', + ) + }) +}) + // Util functions to make the tests more readable function diffInDaysFromNow(time: number) { return Math.abs(time - Date.now()) / 86400000 diff --git a/packages/uploads/src/signedUrls.ts b/packages/uploads/src/signedUrls.ts index f15280400eaf..1edb2d64816f 100644 --- a/packages/uploads/src/signedUrls.ts +++ b/packages/uploads/src/signedUrls.ts @@ -7,6 +7,11 @@ export type SignedUrlSettings = { secret: string // The secret to sign the urls with } +export type SignatureValidationArgs = { + path: string + s: string + expiry?: number | string +} export class UrlSigner { private secret: string private endpoint: string @@ -57,42 +62,67 @@ export class UrlSigner { * The signature and expires have to be extracted from the URL */ validateSignature({ - signature, - filePath, + s: signature, + path: filePath, // In the URL we call it path expiry, - }: { - filePath: string - signature: string - expiry?: number - }) { + }: SignatureValidationArgs) { if (!this.secret) { throw new Error('Please configure the secret') } if (expiry) { - // No need to validate if the signature has expired - if (Date.now() > expiry) { + // No need to validate if the signature has expired, + // but make sure its a number! + if (Date.now() > +expiry) { throw new Error('Signature has expired') } } + // Decoded filePath + const decodedFilePath = decodeURIComponent(filePath) + const validSignature = expiry ? crypto .createHmac('sha256', this.secret) - .update(`${filePath}:${expiry}`) + .update(`${decodedFilePath}:${expiry}`) .digest('hex') : crypto .createHmac('sha256', this.secret) - .update(`${filePath}`) + .update(`${decodedFilePath}`) .digest('hex') if (validSignature !== signature) { throw new Error('Invalid signature') } + + return decodedFilePath + } + + validateSignedUrl(fullPathWithQueryParametersOrUrl: string) { + const url = new URL( + fullPathWithQueryParametersOrUrl, + // We don't care about the host, but just need to create a URL object + // to parse search params + fullPathWithQueryParametersOrUrl.startsWith('http') + ? undefined + : 'http://localhost', + ) + + const path = url.searchParams.get('path') as string + + this.validateSignature({ + // Note the signature is called 's' in the URL + s: url.searchParams.get('s') as string, + expiry: url.searchParams.get('expiry') as string, + path, + }) + + // Return the decoded path + return decodeURIComponent(path) } generateSignedUrl(filePath: string, expiresIn?: number) { - const { signature, expiry: expires } = this.generateSignature({ + const { signature, expiry } = this.generateSignature({ filePath, expiresInMs: expiresIn, }) @@ -100,8 +130,8 @@ export class UrlSigner { // This way you can pass in a path with params already const params = new URLSearchParams() params.set('s', signature) - if (expires) { - params.set('expires', expires.toString()) + if (expiry) { + params.set('expiry', expiry.toString()) } params.set('path', filePath) From c6d2b0927333f63589db4fdfac9f2bcc4cc1a3cd Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 21 Aug 2024 14:23:11 +0700 Subject: [PATCH 64/91] Don't throw when invalid upload path passed to delete --- .../src/__tests__/queryExtensions.test.ts | 29 +++++++++++++++++++ packages/uploads/src/prismaExtension.ts | 11 +++++-- 2 files changed, 38 insertions(+), 2 deletions(-) diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts index 4fb12bcbeaf0..f3c07026d126 100644 --- a/packages/uploads/src/__tests__/queryExtensions.test.ts +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -1,5 +1,6 @@ import fs from 'node:fs/promises' +import type { MockedFunction } from 'vitest' import { describe, it, vi, expect, beforeEach } from 'vitest' import { FileSystemStorage } from '../FileSystemStorage.js' @@ -168,6 +169,10 @@ describe('Query extensions', () => { expect(fs.unlink).toHaveBeenCalledWith('/tmp/old.txt') expect(updatedDummy.uploadField).toBe('/tmp/new.txt') }) + + it('should not delete the file if the update fails', async () => {}) + + it('should not delete files from other fields', async () => {}) }) describe('delete', () => { @@ -189,5 +194,29 @@ describe('Query extensions', () => { expect(fs.unlink).toHaveBeenCalledWith('/tmp/first.txt') expect(fs.unlink).toHaveBeenCalledWith('/tmp/second.txt') }) + + it('Should handle if a bad path is provided', async () => { + ;(fs.unlink as MockedFunction).mockRejectedValueOnce( + new Error('unlink error'), + ) + + const invalidPathDumbo = await prismaClient.dumbo.create({ + data: { + firstUpload: '', + secondUpload: 'im-a-invalid-path', + }, + }) + + const deletePromise = prismaClient.dumbo.delete({ + where: { + id: invalidPathDumbo.id, + }, + }) + + await expect(deletePromise).resolves.not.toThrow() + + expect(fs.unlink).toHaveBeenCalledOnce() + expect(fs.unlink).toHaveBeenCalledWith('im-a-invalid-path') + }) }) }) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 00c216a05d2a..77a4a2c3298d 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -74,13 +74,20 @@ export const createUploadsExtension = ( // With strict mode you cannot call findFirstOrThrow with the same args, because it is a union type // Ideally there's a better way to do this const record = - // @ts-expect-error laskndglkn + // @TODO not sure how to resolve this error + // @ts-expect-error Complaning because findFirstOrThrow args is a union type await prismaInstance[model as ModelNames].findFirstOrThrow(args) // Delete the file from the file system fields.forEach(async (field) => { const filePath = record[field] - await storageAdapter.remove(filePath) + if (filePath) { + try { + await storageAdapter.remove(filePath) + } catch { + // Swallow the error, we don't want to stop the delete operation + } + } }) } From a0634c104b98b6eea3a77005c876731e2118063c Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 21 Aug 2024 18:18:59 +0700 Subject: [PATCH 65/91] Handle empty lists in file list processor --- packages/uploads/src/__tests__/createProcessors.test.ts | 6 ++++++ packages/uploads/src/createProcessors.ts | 2 +- 2 files changed, 7 insertions(+), 1 deletion(-) diff --git a/packages/uploads/src/__tests__/createProcessors.test.ts b/packages/uploads/src/__tests__/createProcessors.test.ts index 5db76b6b46a5..b6e7a0aa599a 100644 --- a/packages/uploads/src/__tests__/createProcessors.test.ts +++ b/packages/uploads/src/__tests__/createProcessors.test.ts @@ -139,4 +139,10 @@ describe('FileList processing', () => { expect(result[0]).toMatch(/\/bazinga_not_mem_store\/.*\.png/) expect(result[1]).toMatch(/\/bazinga_not_mem_store\/.*\.jpeg/) }) + + it('Should handle empty FileLists', async () => { + const promise = fileListProcessor() + + await expect(promise).resolves.not.toThrow() + }) }) diff --git a/packages/uploads/src/createProcessors.ts b/packages/uploads/src/createProcessors.ts index a7e0349e0ae8..263773ee5995 100644 --- a/packages/uploads/src/createProcessors.ts +++ b/packages/uploads/src/createProcessors.ts @@ -8,7 +8,7 @@ type MakeFilesString = { } export const createFileListProcessor = (storage: StorageAdapter) => { - return async (files: File[], pathOverrideOnly?: { path?: string }) => { + return async (files: File[] = [], pathOverrideOnly?: { path?: string }) => { const locations = await Promise.all( files.map(async (file) => { const { location } = await storage.save(file, pathOverrideOnly) From 06307f3fe63b168978108258f2eaf69649e02b2d Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 21 Aug 2024 18:19:49 +0700 Subject: [PATCH 66/91] Use storage adapter in dataUri --- packages/uploads/src/fileSave.utils.ts | 11 ++++------- 1 file changed, 4 insertions(+), 7 deletions(-) diff --git a/packages/uploads/src/fileSave.utils.ts b/packages/uploads/src/fileSave.utils.ts index 193626c89d87..aebd14bb302c 100644 --- a/packages/uploads/src/fileSave.utils.ts +++ b/packages/uploads/src/fileSave.utils.ts @@ -1,12 +1,9 @@ -import fs from 'node:fs/promises' -import path from 'node:path' +import type { StorageAdapter } from './StorageAdapter.js' -import mime from 'mime-types' +export async function fileToDataUri(filePath: string, storage: StorageAdapter) { + const { contents, type: mimeType } = await storage.read(filePath) -export async function fileToDataUri(filePath: string) { - const base64Data = await fs.readFile(filePath, 'base64') - const ext = path.extname(filePath) - const mimeType = mime.lookup(ext) + const base64Data = Buffer.from(contents).toString('base64') return `data:${mimeType};base64,${base64Data}` } From d2d207f04f97d5b71be643057cb08d5ed1ed3623 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 22 Aug 2024 12:13:32 +0700 Subject: [PATCH 67/91] Constraints --- packages/uploads/package.json | 12 +- yarn.lock | 238 +--------------------------------- 2 files changed, 11 insertions(+), 239 deletions(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 7b9fb8372c20..244e577a2633 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -60,22 +60,20 @@ }, "dependencies": { "@redwoodjs/project-config": "workspace:*", - "fs-extra": "11.2.0", "mime-types": "2.1.35", "ulid": "2.3.0" }, "devDependencies": { - "@arethetypeswrong/cli": "0.15.3", + "@arethetypeswrong/cli": "0.15.4", "@prisma/client": "5.18.0", "@redwoodjs/framework-tools": "workspace:*", - "@types/fs-extra": "11.0.4", "@types/mime-types": "2.1.4", "concurrently": "8.2.2", "esbuild": "0.23.0", - "publint": "0.2.9", - "tsx": "4.16.2", + "publint": "0.2.10", + "tsx": "4.17.0", "typescript": "5.5.4", - "vitest": "2.0.4" + "vitest": "2.0.5" }, "gitHead": "3905ed045508b861b495f8d5630d76c7a157d8f1" -} \ No newline at end of file +} diff --git a/yarn.lock b/yarn.lock index 1ad808bf70da..f15f2bd545d3 100644 --- a/yarn.lock +++ b/yarn.lock @@ -242,23 +242,6 @@ __metadata: languageName: node linkType: hard -"@arethetypeswrong/cli@npm:0.15.3": - version: 0.15.3 - resolution: "@arethetypeswrong/cli@npm:0.15.3" - dependencies: - "@arethetypeswrong/core": "npm:0.15.1" - chalk: "npm:^4.1.2" - cli-table3: "npm:^0.6.3" - commander: "npm:^10.0.1" - marked: "npm:^9.1.2" - marked-terminal: "npm:^6.0.0" - semver: "npm:^7.5.4" - bin: - attw: dist/index.js - checksum: 10c0/5998ab4a2195f9036a5c1988f73912a0a82cceeaa6a4e647b04414ad956a62163d8286b2a936941f23065b0c872f2bbdf9196fe3cac19c40b8b62a643d91c3c2 - languageName: node - linkType: hard - "@arethetypeswrong/cli@npm:0.15.4": version: 0.15.4 resolution: "@arethetypeswrong/cli@npm:0.15.4" @@ -8725,21 +8708,19 @@ __metadata: version: 0.0.0-use.local resolution: "@redwoodjs/uploads@workspace:packages/uploads" dependencies: - "@arethetypeswrong/cli": "npm:0.15.3" + "@arethetypeswrong/cli": "npm:0.15.4" "@prisma/client": "npm:5.18.0" "@redwoodjs/framework-tools": "workspace:*" "@redwoodjs/project-config": "workspace:*" - "@types/fs-extra": "npm:11.0.4" "@types/mime-types": "npm:2.1.4" concurrently: "npm:8.2.2" esbuild: "npm:0.23.0" - fs-extra: "npm:11.2.0" mime-types: "npm:2.1.35" - publint: "npm:0.2.9" - tsx: "npm:4.16.2" + publint: "npm:0.2.10" + tsx: "npm:4.17.0" typescript: "npm:5.5.4" ulid: "npm:2.3.0" - vitest: "npm:2.0.4" + vitest: "npm:2.0.5" languageName: unknown linkType: soft @@ -11559,18 +11540,6 @@ __metadata: languageName: node linkType: hard -"@vitest/expect@npm:2.0.4": - version: 2.0.4 - resolution: "@vitest/expect@npm:2.0.4" - dependencies: - "@vitest/spy": "npm:2.0.4" - "@vitest/utils": "npm:2.0.4" - chai: "npm:^5.1.1" - tinyrainbow: "npm:^1.2.0" - checksum: 10c0/18acdd6b1f5001830722fab7d41b0bd754e37572dded74d1549c5e8f40e58d9e4bbbb6a8ce6be1200b04653237329ba1aeeb3330c2a41f1024450016464d491e - languageName: node - linkType: hard - "@vitest/expect@npm:2.0.5": version: 2.0.5 resolution: "@vitest/expect@npm:2.0.5" @@ -11583,16 +11552,7 @@ __metadata: languageName: node linkType: hard -"@vitest/pretty-format@npm:2.0.4": - version: 2.0.4 - resolution: "@vitest/pretty-format@npm:2.0.4" - dependencies: - tinyrainbow: "npm:^1.2.0" - checksum: 10c0/c2ac3ca302b93ad53ea2977209ee4eb31a313c18690034a09f8ec5528d7e82715c233c4927ecf8b364203c5e5475231d9b737b3fb7680eea71882e1eae11e473 - languageName: node - linkType: hard - -"@vitest/pretty-format@npm:2.0.5, @vitest/pretty-format@npm:^2.0.4, @vitest/pretty-format@npm:^2.0.5": +"@vitest/pretty-format@npm:2.0.5, @vitest/pretty-format@npm:^2.0.5": version: 2.0.5 resolution: "@vitest/pretty-format@npm:2.0.5" dependencies: @@ -11601,16 +11561,6 @@ __metadata: languageName: node linkType: hard -"@vitest/runner@npm:2.0.4": - version: 2.0.4 - resolution: "@vitest/runner@npm:2.0.4" - dependencies: - "@vitest/utils": "npm:2.0.4" - pathe: "npm:^1.1.2" - checksum: 10c0/b550372ce5e2c6a3f08dbd584ea669723fc0d789ebaa4224b703f12e908813fb76b963ea9ac2265aa751cab0309f637dc1fa7ce3fb3e67e08e52e241d33237ee - languageName: node - linkType: hard - "@vitest/runner@npm:2.0.5": version: 2.0.5 resolution: "@vitest/runner@npm:2.0.5" @@ -11621,17 +11571,6 @@ __metadata: languageName: node linkType: hard -"@vitest/snapshot@npm:2.0.4": - version: 2.0.4 - resolution: "@vitest/snapshot@npm:2.0.4" - dependencies: - "@vitest/pretty-format": "npm:2.0.4" - magic-string: "npm:^0.30.10" - pathe: "npm:^1.1.2" - checksum: 10c0/67608c5b1e2f8b02ebc95286cd644c31ea29344c81d67151375b6eebf088a0eea242756eefb509aac626b8f7f091044fdcbc80d137d811ead1117a4a524e2d74 - languageName: node - linkType: hard - "@vitest/snapshot@npm:2.0.5": version: 2.0.5 resolution: "@vitest/snapshot@npm:2.0.5" @@ -11643,15 +11582,6 @@ __metadata: languageName: node linkType: hard -"@vitest/spy@npm:2.0.4": - version: 2.0.4 - resolution: "@vitest/spy@npm:2.0.4" - dependencies: - tinyspy: "npm:^3.0.0" - checksum: 10c0/ef0d0c5e36bb6dfa3ef7561368b39c92cd89bb52d112ec13345dfc99981796a9af98bafd35ce6952322a6a7534eaad144485fe7764628d94d77edeba5fa773b6 - languageName: node - linkType: hard - "@vitest/spy@npm:2.0.5": version: 2.0.5 resolution: "@vitest/spy@npm:2.0.5" @@ -11661,18 +11591,6 @@ __metadata: languageName: node linkType: hard -"@vitest/utils@npm:2.0.4": - version: 2.0.4 - resolution: "@vitest/utils@npm:2.0.4" - dependencies: - "@vitest/pretty-format": "npm:2.0.4" - estree-walker: "npm:^3.0.3" - loupe: "npm:^3.1.1" - tinyrainbow: "npm:^1.2.0" - checksum: 10c0/48e0bad3aa463d147b125e355b6bc6c5b4a5eab600132ebafac8379800273b2f47df17dbf76fe179b1500cc6b5866ead2d375a39a9114a03f705eb8850b93afa - languageName: node - linkType: hard - "@vitest/utils@npm:2.0.5": version: 2.0.5 resolution: "@vitest/utils@npm:2.0.5" @@ -12197,13 +12115,6 @@ __metadata: languageName: node linkType: hard -"ansi-escapes@npm:^6.2.0": - version: 6.2.1 - resolution: "ansi-escapes@npm:6.2.1" - checksum: 10c0/a2c6f58b044be5f69662ee17073229b492daa2425a7fd99a665db6c22eab6e4ab42752807def7281c1c7acfed48f87f2362dda892f08c2c437f1b39c6b033103 - languageName: node - linkType: hard - "ansi-escapes@npm:^7.0.0": version: 7.0.0 resolution: "ansi-escapes@npm:7.0.0" @@ -12259,13 +12170,6 @@ __metadata: languageName: node linkType: hard -"ansicolors@npm:~0.3.2": - version: 0.3.2 - resolution: "ansicolors@npm:0.3.2" - checksum: 10c0/e202182895e959c5357db6c60791b2abaade99fcc02221da11a581b26a7f83dc084392bc74e4d3875c22f37b3c9ef48842e896e3bfed394ec278194b8003e0ac - languageName: node - linkType: hard - "any-promise@npm:^1.0.0": version: 1.3.0 resolution: "any-promise@npm:1.3.0" @@ -13630,18 +13534,6 @@ __metadata: languageName: node linkType: hard -"cardinal@npm:^2.1.1": - version: 2.1.1 - resolution: "cardinal@npm:2.1.1" - dependencies: - ansicolors: "npm:~0.3.2" - redeyed: "npm:~2.1.0" - bin: - cdl: ./bin/cdl.js - checksum: 10c0/0051d0e64c0e1dff480c1aace4c018c48ecca44030533257af3f023107ccdeb061925603af6d73710f0345b0ae0eb57e5241d181d9b5fdb595d45c5418161675 - languageName: node - linkType: hard - "caseless@npm:~0.12.0": version: 0.12.0 resolution: "caseless@npm:0.12.0" @@ -16450,7 +16342,7 @@ __metadata: languageName: node linkType: hard -"esbuild@npm:^0.21.3, esbuild@npm:~0.21.5": +"esbuild@npm:^0.21.3": version: 0.21.5 resolution: "esbuild@npm:0.21.5" dependencies: @@ -22254,22 +22146,6 @@ __metadata: languageName: node linkType: hard -"marked-terminal@npm:^6.0.0": - version: 6.2.0 - resolution: "marked-terminal@npm:6.2.0" - dependencies: - ansi-escapes: "npm:^6.2.0" - cardinal: "npm:^2.1.1" - chalk: "npm:^5.3.0" - cli-table3: "npm:^0.6.3" - node-emoji: "npm:^2.1.3" - supports-hyperlinks: "npm:^3.0.0" - peerDependencies: - marked: ">=1 <12" - checksum: 10c0/72d4093cbb1ee864ced1f88fdb6fb8dbfea56d6aa3d8a1ec401ac51866ff3c32382c3f4642b19f2d808c798efde23b10300b99e3b6475b3f79e41e7741581d54 - languageName: node - linkType: hard - "marked-terminal@npm:^7.1.0": version: 7.1.0 resolution: "marked-terminal@npm:7.1.0" @@ -25479,19 +25355,6 @@ __metadata: languageName: node linkType: hard -"publint@npm:0.2.9": - version: 0.2.9 - resolution: "publint@npm:0.2.9" - dependencies: - npm-packlist: "npm:^5.1.3" - picocolors: "npm:^1.0.1" - sade: "npm:^1.8.1" - bin: - publint: lib/cli.js - checksum: 10c0/b414f40c2bc9372119346d5684eccb12bdf8066fc821301880d9dcdec0a5d852bbe926cb4583511f3e97736c53d1723e46e49285c9463bcb808cbfd979d5c2fc - languageName: node - linkType: hard - "pump@npm:^2.0.0": version: 2.0.1 resolution: "pump@npm:2.0.1" @@ -26143,15 +26006,6 @@ __metadata: languageName: node linkType: hard -"redeyed@npm:~2.1.0": - version: 2.1.1 - resolution: "redeyed@npm:2.1.1" - dependencies: - esprima: "npm:~4.0.0" - checksum: 10c0/350f5e39aebab3886713a170235c38155ee64a74f0f7e629ecc0144ba33905efea30c2c3befe1fcbf0b0366e344e7bfa34e6b2502b423c9a467d32f1306ef166 - languageName: node - linkType: hard - "redis-errors@npm:^1.0.0, redis-errors@npm:^1.2.0": version: 1.2.0 resolution: "redis-errors@npm:1.2.0" @@ -28892,22 +28746,6 @@ __metadata: languageName: node linkType: hard -"tsx@npm:4.16.2": - version: 4.16.2 - resolution: "tsx@npm:4.16.2" - dependencies: - esbuild: "npm:~0.21.5" - fsevents: "npm:~2.3.3" - get-tsconfig: "npm:^4.7.5" - dependenciesMeta: - fsevents: - optional: true - bin: - tsx: dist/cli.mjs - checksum: 10c0/9df52264f88be00ca473e7d7eda43bb038cc09028514996b864db78645e9cd297c71485f0fdd4985464d6dc46424f8bef9f8c4bd56692c4fcf4d71621ae21763 - languageName: node - linkType: hard - "tsx@npm:4.17.0": version: 4.17.0 resolution: "tsx@npm:4.17.0" @@ -29747,21 +29585,6 @@ __metadata: languageName: node linkType: hard -"vite-node@npm:2.0.4": - version: 2.0.4 - resolution: "vite-node@npm:2.0.4" - dependencies: - cac: "npm:^6.7.14" - debug: "npm:^4.3.5" - pathe: "npm:^1.1.2" - tinyrainbow: "npm:^1.2.0" - vite: "npm:^5.0.0" - bin: - vite-node: vite-node.mjs - checksum: 10c0/2689b05b391b59cf3d15e1e80884e9b054f2ca90b2150cc7a08b0f234e79e6750a28cc8d107a57f005185e759c3bc020030f687065317fc37fe169ce17f4cdb7 - languageName: node - linkType: hard - "vite-node@npm:2.0.5": version: 2.0.5 resolution: "vite-node@npm:2.0.5" @@ -29845,55 +29668,6 @@ __metadata: languageName: node linkType: hard -"vitest@npm:2.0.4": - version: 2.0.4 - resolution: "vitest@npm:2.0.4" - dependencies: - "@ampproject/remapping": "npm:^2.3.0" - "@vitest/expect": "npm:2.0.4" - "@vitest/pretty-format": "npm:^2.0.4" - "@vitest/runner": "npm:2.0.4" - "@vitest/snapshot": "npm:2.0.4" - "@vitest/spy": "npm:2.0.4" - "@vitest/utils": "npm:2.0.4" - chai: "npm:^5.1.1" - debug: "npm:^4.3.5" - execa: "npm:^8.0.1" - magic-string: "npm:^0.30.10" - pathe: "npm:^1.1.2" - std-env: "npm:^3.7.0" - tinybench: "npm:^2.8.0" - tinypool: "npm:^1.0.0" - tinyrainbow: "npm:^1.2.0" - vite: "npm:^5.0.0" - vite-node: "npm:2.0.4" - why-is-node-running: "npm:^2.3.0" - peerDependencies: - "@edge-runtime/vm": "*" - "@types/node": ^18.0.0 || >=20.0.0 - "@vitest/browser": 2.0.4 - "@vitest/ui": 2.0.4 - happy-dom: "*" - jsdom: "*" - peerDependenciesMeta: - "@edge-runtime/vm": - optional: true - "@types/node": - optional: true - "@vitest/browser": - optional: true - "@vitest/ui": - optional: true - happy-dom: - optional: true - jsdom: - optional: true - bin: - vitest: vitest.mjs - checksum: 10c0/139200d0bda3270fd00641e4bd5524f78a2b1fe9a3d4a0d5ba2b6ed08bbcf6f1e711cc4bfd8b0d823628a2fcab00f822bb210bd5bf3c6a9260fd6115ea085a3d - languageName: node - linkType: hard - "vitest@npm:2.0.5": version: 2.0.5 resolution: "vitest@npm:2.0.5" From 0ef44e39a693e571a539fa084cc1fee4dbfc51d9 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 22 Aug 2024 12:13:39 +0700 Subject: [PATCH 68/91] Refactor base64 handling --- packages/uploads/src/__tests__/queryExtensions.test.ts | 8 ++++++-- .../uploads/src/{fileSave.utils.ts => fileHandling.ts} | 0 packages/uploads/src/prismaExtension.ts | 3 ++- 3 files changed, 8 insertions(+), 3 deletions(-) rename packages/uploads/src/{fileSave.utils.ts => fileHandling.ts} (100%) diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts index f3c07026d126..48a9b2a0e368 100644 --- a/packages/uploads/src/__tests__/queryExtensions.test.ts +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -170,9 +170,13 @@ describe('Query extensions', () => { expect(updatedDummy.uploadField).toBe('/tmp/new.txt') }) - it('should not delete the file if the update fails', async () => {}) + it('should not delete the file if the update fails', async () => { + throw new Error('Not implemented yet') + }) - it('should not delete files from other fields', async () => {}) + it('should not delete files from other fields', async () => { + throw new Error('Not implemented yet') + }) }) describe('delete', () => { diff --git a/packages/uploads/src/fileSave.utils.ts b/packages/uploads/src/fileHandling.ts similarity index 100% rename from packages/uploads/src/fileSave.utils.ts rename to packages/uploads/src/fileHandling.ts diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 77a4a2c3298d..08d1222ab75d 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -4,7 +4,7 @@ import { Prisma as PrismaExtension } from '@prisma/client/extension' import type * as runtime from '@prisma/client/runtime/library' -import { fileToDataUri } from './fileSave.utils.js' +import { fileToDataUri } from './fileHandling.js' import type { UrlSigner } from './signedUrls.js' import type { StorageAdapter } from './StorageAdapter.js' @@ -170,6 +170,7 @@ export const createUploadsExtension = ( for await (const field of uploadFields) { base64UploadFields[field] = await fileToDataUri( modelData[field] as string, + storageAdapter, ) } From cd43150d623264f1ef6813d33e7f53e3a86b2741 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 22 Aug 2024 12:44:52 +0700 Subject: [PATCH 69/91] Update README --- packages/uploads/README.md | 125 +++++++++++++++++++++++++++++++++++-- 1 file changed, 121 insertions(+), 4 deletions(-) diff --git a/packages/uploads/README.md b/packages/uploads/README.md index b6b7c36ce457..764a58aec2cd 100644 --- a/packages/uploads/README.md +++ b/packages/uploads/README.md @@ -1,7 +1,124 @@ -# Uploads +# `@redwoodjs/uploads` This package houses -- Prisma extension for handling uploads -- Base64 file picker uploader (?) -- TUS uploader (?) +- Prisma extension for handling uploads. Currently + a) Query Extension: will save, delete, replace files on disk during CRUD + b) Result Extension: gives you functions like `.withSignedUri` on configured prisma results - which will take the paths, and convert it to a signed url +- Storage adapters e.g. FS and Memory to use with the prisma extension +- Processors - i.e. utility functions which will take [`Files`](https://developer.mozilla.org/en-US/docs/Web/API/File) and save them to storage + +## Usage + +In `api/src/uploads.ts` - setup uploads - processors, storage and the prisma extension. + +```ts +// api/src/lib/uploads.ts + +import { UploadsConfig } from '@redwoodjs/uploads' +import { setupUploads } from '@redwoodjs/uploads' +import { FileSystemStorage } from '@redwoodjs/uploads/FileSystemStorage' +import { UrlSigner } from '@redwoodjs/uploads/signedUrl' + +const uploadConfig: UploadsConfig = { + // 👇 prisma model + profile: { + // 👇 pass in fields that are going to be File uploads + // these should be configured as string in the Prisma.schema + fields: ['avatar', 'coverPhoto'], + }, +} + +// 👇 exporting these allows you access elsewhere on the api side +export const storage = new FileSystemStorage({ + baseDir: './uploads', +}) + +// Optional +export const urlSigner = new UrlSigner({ + secret: process.env.UPLOADS_SECRET, + endpoint: '/signedUrl', +}) + +const { uploadsProcessors, prismaExtension, fileListProcessor } = setupUploads( + uploadConfig, + storage, + urlSigner, +) + +export { uploadsProcessors, prismaExtension, fileListProcessor } +``` + +### Configuring db to use the prisma extension + +```ts +// api/src/lib/db.ts + +import { PrismaClient } from '@prisma/client' + +import { emitLogLevels, handlePrismaLogging } from '@redwoodjs/api/logger' + +import { logger } from './logger' +import { prismaExtension } from './uploads' + +// 👇 Notice here we create prisma client, and don't export it yet +export const prismaClient = new PrismaClient({ + log: emitLogLevels(['info', 'warn', 'error']), +}) + +handlePrismaLogging({ + db: prismaClient, + logger, + logLevels: ['info', 'warn', 'error'], +}) + +// 👇 Export db after adding uploads extension +export const db = prismaClient.$extends(prismaExtension) +``` + +## Using Prisma extension + +### A) CRUD operations + +No need to do anything here, but you have to use processors to supply Prisma with data in the correct format. + +### B) Result extensions + +```ts +// api/src/services/profiles/profiles.ts + +export const profile: QueryResolvers['profile'] = async ({ id }) => { + // 👇 await the result from your prisma query + const profile = await db.profile.findUnique({ + where: { id }, + }) + + // Convert the avatar and coverPhoto fields to signed URLs + // Note that you still need to add a api endpoint to handle these signed urls + return profile?.withSignedUrl() +} +``` + +## Using processors + +In your services, you can use the preconfigured "processors" to convert Files to strings for Prisma to save into the database. The processors, and storage adapters determine where the file is saved. + +```ts +// api/src/services/profiles/profiles.ts + +export const updateProfile: MutationResolvers['updateProfile'] = async ({ + id, + input, +}) => { + const processedInput = await uploadsProcessors.processProfileUploads(input) + + // This becomes a string 👇 + // The configuration on where it was saved is passed when we setup uploads in src/lib/uploads.ts + // processedInput.avatar = '/mySavePath/profile/avatar/generatedId.jpg' + + return db.profile.update({ + data: processedInput, + where: { id }, + }) +} +``` From 3b8d4a1e297d233285dfc75ea5c5549d07ab39cb Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 22 Aug 2024 15:33:58 +0700 Subject: [PATCH 70/91] Handle update edgecases --- .../src/__tests__/queryExtensions.test.ts | 76 +++++++++-- .../src/__tests__/unit-test-schema.prisma | 1 + packages/uploads/src/prismaExtension.ts | 128 +++++++++--------- 3 files changed, 131 insertions(+), 74 deletions(-) diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts index 48a9b2a0e368..567596407410 100644 --- a/packages/uploads/src/__tests__/queryExtensions.test.ts +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -1,13 +1,14 @@ import fs from 'node:fs/promises' import type { MockedFunction } from 'vitest' -import { describe, it, vi, expect, beforeEach } from 'vitest' +import { describe, it, vi, expect, beforeEach, beforeAll } from 'vitest' import { FileSystemStorage } from '../FileSystemStorage.js' import { setupUploads } from '../index.js' import type { UploadsConfig } from '../prismaExtension.js' // @MARK: use the local prisma client in the test +import type { Dumbo, Dummy } from './prisma-client/index.js' import { PrismaClient } from './prisma-client/index.js' vi.mock('node:fs/promises', () => ({ @@ -99,13 +100,13 @@ describe('Query extensions', () => { { firstUpload: '/one/first.txt', secondUpload: '/one/second.txt', - // @ts-expect-error Intentional bro + // @ts-expect-error Intentional id: 'break', }, { firstUpload: '/two/first.txt', secondUpload: '/two/second.txt', - // @ts-expect-error Intentional bro + // @ts-expect-error Intentional id: 'break2', }, ], @@ -133,7 +134,7 @@ describe('Query extensions', () => { { firstUpload: '/two/first.txt', secondUpload: '/two/second.txt', - // @ts-expect-error Intentional bro + // @ts-expect-error Intentional id: 'break2', }, ], @@ -150,19 +151,34 @@ describe('Query extensions', () => { }) describe('update', () => { - it('update will remove the old file, save new one', async () => { - const dummy = await prismaClient.dummy.create({ + let ogDummy: Dummy + let ogDumbo: Dumbo + beforeAll(async () => { + ogDummy = await prismaClient.dummy.create({ data: { uploadField: '/tmp/old.txt', }, }) + ogDumbo = await prismaClient.dumbo.create({ + data: { + firstUpload: '/tmp/oldFirst.txt', + secondUpload: '/tmp/oldSecond.txt', + }, + }) + }) + + beforeEach(() => { + vi.resetAllMocks() + }) + + it('update will remove the old file, save new one', async () => { const updatedDummy = await prismaClient.dummy.update({ data: { uploadField: '/tmp/new.txt', }, where: { - id: dummy.id, + id: ogDummy.id, }, }) @@ -171,11 +187,51 @@ describe('Query extensions', () => { }) it('should not delete the file if the update fails', async () => { - throw new Error('Not implemented yet') + const failedUpdatePromise = prismaClient.dummy.update({ + data: { + // @ts-expect-error Intentional + id: 'this-is-the-incorrect-type', + }, + where: { + id: ogDummy.id, + }, + }) + + // Id is invalid, so the update should fail + await expect(failedUpdatePromise).rejects.toThrowError() + + // The old one should NOT be deleted + expect(fs.unlink).not.toHaveBeenCalled() + }) + + it.only('should only delete old files from the fields that are being updated', async () => { + const updatedDumbo = await prismaClient.dumbo.update({ + data: { + firstUpload: '/tmp/newFirst.txt', + }, + where: { + id: ogDumbo.id, + }, + }) + + expect(updatedDumbo.firstUpload).toBe('/tmp/newFirst.txt') + expect(updatedDumbo.secondUpload).toBe('/tmp/oldSecond.txt') + expect(fs.unlink).toHaveBeenCalledOnce() + expect(fs.unlink).toHaveBeenCalledWith('/tmp/oldFirst.txt') }) - it('should not delete files from other fields', async () => { - throw new Error('Not implemented yet') + it('should not delete files on update of non-upload fields', async () => { + // In this case, we're only updating the message field + await prismaClient.dumbo.update({ + data: { + message: 'Hello world', + }, + where: { + id: ogDumbo.id, + }, + }) + + expect(fs.unlink).not.toHaveBeenCalled() }) }) diff --git a/packages/uploads/src/__tests__/unit-test-schema.prisma b/packages/uploads/src/__tests__/unit-test-schema.prisma index 905afc366007..1a8cd0489e04 100644 --- a/packages/uploads/src/__tests__/unit-test-schema.prisma +++ b/packages/uploads/src/__tests__/unit-test-schema.prisma @@ -17,6 +17,7 @@ model Dumbo { id Int @id @default(autoincrement()) firstUpload String secondUpload String + message String? } model NoUploadFields { diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 08d1222ab75d..db5c31262b54 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -3,7 +3,6 @@ import type { Prisma } from '@prisma/client' import { Prisma as PrismaExtension } from '@prisma/client/extension' import type * as runtime from '@prisma/client/runtime/library' - import { fileToDataUri } from './fileHandling.js' import type { UrlSigner } from './signedUrls.js' import type { StorageAdapter } from './StorageAdapter.js' @@ -59,38 +58,6 @@ export const createUploadsExtension = ( } } - async function deleteUpload( - { - model, - args, - fields, - }: { - model: string - args: T - fields: string[] - }, - storageAdapter: StorageAdapter, - ) { - // With strict mode you cannot call findFirstOrThrow with the same args, because it is a union type - // Ideally there's a better way to do this - const record = - // @TODO not sure how to resolve this error - // @ts-expect-error Complaning because findFirstOrThrow args is a union type - await prismaInstance[model as ModelNames].findFirstOrThrow(args) - - // Delete the file from the file system - fields.forEach(async (field) => { - const filePath = record[field] - if (filePath) { - try { - await storageAdapter.remove(filePath) - } catch { - // Swallow the error, we don't want to stop the delete operation - } - } - }) - } - const queryExtends: runtime.ExtensionArgs['query'] = {} const resultExtends = {} as ResultExtends @@ -115,43 +82,64 @@ export const createUploadsExtension = ( return result } catch (e) { // If the create fails, we need to delete the uploaded files - await removeUploadedFiles(uploadFields, args) + await removeUploadedFiles( + uploadFields, + args.data as Record, + ) throw e } }, async update({ query, model, args }) { - await deleteUpload( - { - model, - args: { - // The update args contains data, which we don't need to supply to delete - where: args.where, - }, - fields: uploadFields, - }, - storageAdapter, + // Check if any of the uploadFields are present in args.data + // We only want to process fields that are being updated + const uploadFieldsToUpdate = uploadFields.filter( + (field) => + // All of this non-sense is to make typescript happy. I'm not sure how data could be anything but an object + typeof args.data === 'object' && + args.data !== null && + field in args.data, ) - // Same as create 👇 - try { - const result = await query(args) - return result - } catch (e) { - // If the create fails, we need to delete the uploaded files - await removeUploadedFiles(uploadFields, args) - throw e + // If no upload fields are present, proceed with the original query + // avoid overhead of extra lookups + if (uploadFieldsToUpdate.length == 0) { + return query(args) + } else { + const originalRecord = await prismaInstance[ + model as ModelNames + // @ts-expect-error TS in strict mode will error due to union type. We cannot narrow it down here. + ].findFirstOrThrow({ + where: args.where, + // @TODO: should we select here to reduce the amount of data we're handling + }) + + // Similar, but not same as create + try { + const result = await query(args) + + // **After** we've updated the record, we need to delete the old file. + await removeUploadedFiles(uploadFieldsToUpdate, originalRecord) + + return result + } catch (e) { + // If the update fails, we need to delete the newly uploaded files + // but not the ones that already exist! + await removeUploadedFiles( + uploadFieldsToUpdate, + args.data as Record, + ) + throw e + } } }, async delete({ model, query, args }) { - await deleteUpload( - { - model, - args, - fields: uploadFields, - }, - storageAdapter, - ) + /** Delete args are the same as findFirst, essentially a where clause */ + const record = + // @ts-expect-error TS in strict mode will error due to union type. We cannot narrow it down here. + await prismaInstance[model as ModelNames].findFirstOrThrow(args) + + await removeUploadedFiles(uploadFields, record) return query(args) }, @@ -225,11 +213,23 @@ export const createUploadsExtension = ( // @TODO(TS): According to TS, data could be a non-object... // Setting args to JsArgs causes errors. This could be a legit issue - async function removeUploadedFiles(uploadFields: string[], args: any) { - for await (const field of uploadFields) { - const uploadLocation = args.data?.[field] as string + async function removeUploadedFiles( + fieldsToDelete: string[], + data: Record, + ) { + if (!data) { + console.warn('Empty data object passed to removeUploadedFiles') + return + } + + for await (const field of fieldsToDelete) { + const uploadLocation = data?.[field] if (uploadLocation) { - await storageAdapter.remove(uploadLocation) + try { + await storageAdapter.remove(uploadLocation) + } catch { + // Swallow the error, we don't want to stop the delete operation + } } } } From 9fc42ce3546e4af1d71f0d70df659f4ac953c31e Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 22 Aug 2024 16:31:10 +0700 Subject: [PATCH 71/91] Prettify --- .changesets/11154.md | 139 ++++++++++++++++++++++++++++++++++++++----- 1 file changed, 124 insertions(+), 15 deletions(-) diff --git a/.changesets/11154.md b/.changesets/11154.md index 44c278ffb1f5..920a5b7ae784 100644 --- a/.changesets/11154.md +++ b/.changesets/11154.md @@ -1,15 +1,124 @@ -- feat(rw-uploads): Create uploads package with prisma extension (#11154) by @dac09 - -This PR does the following: -- creates new `@redwoodjs/uploads` package. This is configured to be a dual esm/cjs package. -- exports prisma extension with legacy support e.g. - ``` - import { - createUploadsExtension, - UploadsConfig, - } from '@redwoodjs/uploads/prisma' - ``` -- scaffolds unit test structure for the prisma extension -- the prisma extension does the following: -a) If a base64 or TUS upload URL string is sent, it saves or moves it to a file -b) Adds a result extension (withDataUri), to make it easier to deal with files +- feat(rw-uploads): Create uploads package with prisma extension and upload processor (#11154) by @dac09 + +Introduces `@redwoodjs/uploads` package which houses + +- Prisma extension for handling uploads. Currently + a) Query Extension: will save, delete, replace files on disk during CRUD + b) Result Extension: gives you functions like `.withSignedUri` on configured prisma results - which will take the paths, and convert it to a signed url +- Storage adapters e.g. FS and Memory to use with the prisma extension +- Processors - i.e. utility functions which will take [`Files`](https://developer.mozilla.org/en-US/docs/Web/API/File) and save them to storage + +## Usage + +In `api/src/uploads.ts` - setup uploads - processors, storage and the prisma extension. + +```ts +// api/src/lib/uploads.ts + +import { UploadsConfig } from '@redwoodjs/uploads' +import { setupUploads } from '@redwoodjs/uploads' +import { FileSystemStorage } from '@redwoodjs/uploads/FileSystemStorage' +import { UrlSigner } from '@redwoodjs/uploads/signedUrl' + +const uploadConfig: UploadsConfig = { + // 👇 prisma model + profile: { + // 👇 pass in fields that are going to be File uploads + // these should be configured as string in the Prisma.schema + fields: ['avatar', 'coverPhoto'], + }, +} + +// 👇 exporting these allows you access elsewhere on the api side +export const storage = new FileSystemStorage({ + baseDir: './uploads', +}) + +// Optional +export const urlSigner = new UrlSigner({ + secret: process.env.UPLOADS_SECRET, + endpoint: '/signedUrl', +}) + +const { uploadsProcessors, prismaExtension, fileListProcessor } = setupUploads( + uploadConfig, + storage, + urlSigner, +) + +export { uploadsProcessors, prismaExtension, fileListProcessor } +``` + +### Configuring db to use the prisma extension + +```ts +// api/src/lib/db.ts + +import { PrismaClient } from '@prisma/client' + +import { emitLogLevels, handlePrismaLogging } from '@redwoodjs/api/logger' + +import { logger } from './logger' +import { prismaExtension } from './uploads' + +// 👇 Notice here we create prisma client, and don't export it yet +export const prismaClient = new PrismaClient({ + log: emitLogLevels(['info', 'warn', 'error']), +}) + +handlePrismaLogging({ + db: prismaClient, + logger, + logLevels: ['info', 'warn', 'error'], +}) + +// 👇 Export db after adding uploads extension +export const db = prismaClient.$extends(prismaExtension) +``` + +## Using Prisma extension + +### A) CRUD operations + +No need to do anything here, but you have to use processors to supply Prisma with data in the correct format. + +### B) Result extensions + +```ts +// api/src/services/profiles/profiles.ts + +export const profile: QueryResolvers['profile'] = async ({ id }) => { + // 👇 await the result from your prisma query + const profile = await db.profile.findUnique({ + where: { id }, + }) + + // Convert the avatar and coverPhoto fields to signed URLs + // Note that you still need to add a api endpoint to handle these signed urls + return profile?.withSignedUrl() +} +``` + +## Using processors + +In your services, you can use the preconfigured "processors" to convert Files to strings for Prisma to save into the database. The processors, and storage adapters determine where the file is saved. + +```ts +// api/src/services/profiles/profiles.ts + +export const updateProfile: MutationResolvers['updateProfile'] = async ({ + id, + input, +}) => { + const processedInput = await uploadsProcessors.processProfileUploads(input) + + // This becomes a string 👇 + // The configuration on where it was saved is passed when we setup uploads in src/lib/uploads.ts + // processedInput.avatar = '/mySavePath/profile/avatar/generatedId.jpg' + + return db.profile.update({ + data: processedInput, + where: { id }, + }) +} +``` From e400acf42af5c9b5de4ab4d91e381974d82d9cd4 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 22 Aug 2024 16:36:25 +0700 Subject: [PATCH 72/91] Add Memory storage tests --- .../src/__tests__/MemoryStorage.test.ts | 56 +++++++++++++++++++ 1 file changed, 56 insertions(+) create mode 100644 packages/uploads/src/__tests__/MemoryStorage.test.ts diff --git a/packages/uploads/src/__tests__/MemoryStorage.test.ts b/packages/uploads/src/__tests__/MemoryStorage.test.ts new file mode 100644 index 000000000000..ffad7c70cad8 --- /dev/null +++ b/packages/uploads/src/__tests__/MemoryStorage.test.ts @@ -0,0 +1,56 @@ +import { beforeEach, describe, expect, test } from 'vitest' + +import { MemoryStorage } from '../MemoryStorage.js' + +describe('MemoryStorage', () => { + let storage: MemoryStorage + + beforeEach(() => { + storage = new MemoryStorage({ baseDir: 'uploads' }) + }) + + test('save should store a file in memory', async () => { + const file = new File(['test content'], 'test.txt', { type: 'text/plain' }) + const result = await storage.save(file) + + expect(result).toHaveProperty('location') + expect(result.location).toMatch(/uploads\/.*\.txt$/) + expect(storage.store[result.location]).toBeDefined() + }) + + test('remove should delete a file from memory', async () => { + const file = new File(['test content'], 'test.txt', { type: 'text/plain' }) + const { location } = await storage.save(file) + + await storage.remove(location) + expect(storage.store[location]).toBeUndefined() + }) + + test('read should return file contents and type', async () => { + const file = new File(['ABCDEF'], 'test.txt', { type: 'image/png' }) + const { location } = await storage.save(file) + + const result = await storage.read(location) + expect(result.contents).toBeInstanceOf(Buffer) + expect(result.contents.toString()).toBe('ABCDEF') + expect(result.type).toBe('image/png') + }) + + test('clear should remove all stored files', async () => { + const file1 = new File(['content 1'], 'file1.txt', { type: 'text/plain' }) + const file2 = new File(['content 2'], 'file2.txt', { type: 'text/plain' }) + + await storage.save(file1) + await storage.save(file2) + + await storage.clear() + expect(Object.keys(storage.store).length).toBe(0) + }) + + test('save should use custom path when provided', async () => { + const file = new File(['test content'], 'test.txt', { type: 'text/plain' }) + const result = await storage.save(file, { path: 'custom/path' }) + + expect(result.location).toContain('custom/path') + }) +}) From 2f7d822a72d685c93458aa64572610fdcc16dbb4 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 22 Aug 2024 16:52:23 +0700 Subject: [PATCH 73/91] Add tests for storage adapters --- packages/uploads/src/FileSystemStorage.ts | 2 +- .../src/__tests__/FileSystemsStorage.test.ts | 82 +++++++++++++++++++ .../src/__tests__/MemoryStorage.test.ts | 8 +- 3 files changed, 85 insertions(+), 7 deletions(-) create mode 100644 packages/uploads/src/__tests__/FileSystemsStorage.test.ts diff --git a/packages/uploads/src/FileSystemStorage.ts b/packages/uploads/src/FileSystemStorage.ts index de054a84ff8d..88dfd29b96ca 100644 --- a/packages/uploads/src/FileSystemStorage.ts +++ b/packages/uploads/src/FileSystemStorage.ts @@ -14,7 +14,7 @@ export class FileSystemStorage constructor(opts: { baseDir: string }) { super(opts) if (!existsSync(opts.baseDir)) { - console.log('Creating baseDir', opts.baseDir) + console.log('Creating baseDir >', opts.baseDir) mkdirSync(opts.baseDir, { recursive: true }) } } diff --git a/packages/uploads/src/__tests__/FileSystemsStorage.test.ts b/packages/uploads/src/__tests__/FileSystemsStorage.test.ts new file mode 100644 index 000000000000..5a266a424eb0 --- /dev/null +++ b/packages/uploads/src/__tests__/FileSystemsStorage.test.ts @@ -0,0 +1,82 @@ +import { vol } from 'memfs' +import { beforeEach, describe, expect, test, vi } from 'vitest' + +import { FileSystemStorage } from '../FileSystemStorage.js' + +// Mock the entire fs module +vi.mock('node:fs', async () => { + const memfs = await import('memfs') + return { + ...memfs.fs, + default: memfs.fs, + } +}) + +// Mock the fs/promises module +vi.mock('node:fs/promises', async () => { + const memfs = await import('memfs') + return { + ...memfs.fs.promises, + default: memfs.fs.promises, + } +}) + +describe('FileSystemStorage', () => { + let storage: FileSystemStorage + const baseDir = '/tmp/test_uploads' + + beforeEach(() => { + vol.reset() + storage = new FileSystemStorage({ baseDir }) + }) + + const plainFile = new File(['test content'], 'test.txt', { + type: 'text/plain', + }) + + test('save should store a file on the file system', async () => { + const result = await storage.save(plainFile) + + expect(result).toHaveProperty('location') + expect(result.location).toMatch(/\/tmp\/test_uploads\/.*\.txt$/) + expect(vol.existsSync(result.location)).toBe(true) + }) + + test('remove should delete a file fron ', async () => { + const { location } = await storage.save(plainFile) + + await storage.remove(location) + expect(vol.existsSync(location)).toBe(false) + }) + + test('read should return file contents and type', async () => { + const { location: plainFileLocation } = await storage.save(plainFile) + + const plainFileReadResult = await storage.read(plainFileLocation) + expect(plainFileReadResult.contents).toBeInstanceOf(Buffer) + expect(plainFileReadResult.contents.toString()).toBe('test content') + expect(plainFileReadResult.type).toBe('text/plain') + + const imageFile = new File(['ABCDEF'], 'test.png', { type: 'image/png' }) + const { location } = await storage.save(imageFile) + + const result = await storage.read(location) + expect(result.contents).toBeInstanceOf(Buffer) + expect(result.contents.toString()).toBe('ABCDEF') + expect(result.type).toBe('image/png') + }) + + test('save should use custom path, with no baseDir, when provided', async () => { + // Note that using a custom path means you need to create the directory yourself! + vol.mkdirSync('/my_custom/path', { recursive: true }) + + const result = await storage.save(plainFile, { + path: '/my_custom/path', + fileName: 'bazinga', + }) + + // Note that it doesn't have the baseDir! + expect(result.location).toEqual('/my_custom/path/bazinga.txt') + expect(vol.existsSync(result.location)).toBe(true) + }) +}) diff --git a/packages/uploads/src/__tests__/MemoryStorage.test.ts b/packages/uploads/src/__tests__/MemoryStorage.test.ts index ffad7c70cad8..12bb49972db2 100644 --- a/packages/uploads/src/__tests__/MemoryStorage.test.ts +++ b/packages/uploads/src/__tests__/MemoryStorage.test.ts @@ -1,13 +1,9 @@ -import { beforeEach, describe, expect, test } from 'vitest' +import { describe, expect, test } from 'vitest' import { MemoryStorage } from '../MemoryStorage.js' describe('MemoryStorage', () => { - let storage: MemoryStorage - - beforeEach(() => { - storage = new MemoryStorage({ baseDir: 'uploads' }) - }) + const storage = new MemoryStorage({ baseDir: 'uploads' }) test('save should store a file in memory', async () => { const file = new File(['test content'], 'test.txt', { type: 'text/plain' }) From 1d94e1fbc626194e628846db6e28d42eacd0664d Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 22 Aug 2024 17:05:00 +0700 Subject: [PATCH 74/91] Remove import aliasing for now --- packages/uploads/prisma/index.js | 2 -- packages/uploads/prisma/package.json | 4 ---- 2 files changed, 6 deletions(-) delete mode 100644 packages/uploads/prisma/index.js delete mode 100644 packages/uploads/prisma/package.json diff --git a/packages/uploads/prisma/index.js b/packages/uploads/prisma/index.js deleted file mode 100644 index f21ea7b2fd3d..000000000000 --- a/packages/uploads/prisma/index.js +++ /dev/null @@ -1,2 +0,0 @@ -/* eslint-env es6, commonjs */ -module.exports = require('../dist/cjs/prismaExtension.js') diff --git a/packages/uploads/prisma/package.json b/packages/uploads/prisma/package.json deleted file mode 100644 index 9b83f1f2dc6d..000000000000 --- a/packages/uploads/prisma/package.json +++ /dev/null @@ -1,4 +0,0 @@ -{ - "main": "./index.js", - "types": "../dist/cjs/prismaExtension.d.ts" -} From 5bc4ae45db831d8c0bb700727da53fa0704850a0 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 22 Aug 2024 17:05:36 +0700 Subject: [PATCH 75/91] Clean up package.json --- packages/uploads/package.json | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 244e577a2633..c0893d7d2758 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -42,13 +42,12 @@ } } }, - "types": "dist/prismaExtension.d.ts", "files": [ "dist", "prisma" ], "scripts": { - "build": "tsx ./build.mts && run build:types", + "build": "tsx ./build.mts", "build:pack": "yarn pack -o redwoodjs-uploads.tgz", "build:types": "tsc --build --verbose", "build:types-cjs": "tsc --build --verbose tsconfig.types-cjs.json", @@ -76,4 +75,4 @@ "vitest": "2.0.5" }, "gitHead": "3905ed045508b861b495f8d5630d76c7a157d8f1" -} +} \ No newline at end of file From 33ee3bb97efb0b75f7620489d354fa6df0df9102 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Fri, 23 Aug 2024 14:21:33 +0700 Subject: [PATCH 76/91] Build prisma client before building package Ignore test files from packaging --- packages/uploads/package.json | 4 ++-- packages/uploads/tsconfig.json | 3 ++- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index c0893d7d2758..644d5fa78813 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -44,10 +44,10 @@ }, "files": [ "dist", - "prisma" + "!dist/**/*.test.d.*" ], "scripts": { - "build": "tsx ./build.mts", + "build": "yarn setup:test && tsx ./build.mts", "build:pack": "yarn pack -o redwoodjs-uploads.tgz", "build:types": "tsc --build --verbose", "build:types-cjs": "tsc --build --verbose tsconfig.types-cjs.json", diff --git a/packages/uploads/tsconfig.json b/packages/uploads/tsconfig.json index 7458881c480d..d8a2b7c67426 100644 --- a/packages/uploads/tsconfig.json +++ b/packages/uploads/tsconfig.json @@ -8,7 +8,8 @@ "outDir": "dist" }, "include": ["src", "prisma-override.d.ts"], - // Excluding types here causes types to be inaccurate in tests + // Excluding tests (as in root compilerOption) causes types to be inaccurate in tests + // This overrides the exclude in the root compilerOption "exclude": ["dist", "node_modules", "**/__mocks__"], "references": [ { From d4f0d4433a8a0a8c53c14d20f03de54083fa17c3 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Fri, 23 Aug 2024 14:55:50 +0700 Subject: [PATCH 77/91] Ignore generated prisma client from linting --- .eslintrc.js | 1 + packages/uploads/src/__tests__/queryExtensions.test.ts | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/.eslintrc.js b/.eslintrc.js index a7094e6c548f..3ee6aec61361 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -59,6 +59,7 @@ module.exports = { 'packages/babel-config/src/__tests__/__fixtures__/**/*', 'packages/codemods/**/__testfixtures__/**/*', 'packages/cli/**/__testfixtures__/**/*', + 'packages/uploads/src/__tests__/prisma-client/*', ], rules: { curly: 'error', diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts index 567596407410..8a000e804098 100644 --- a/packages/uploads/src/__tests__/queryExtensions.test.ts +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -111,7 +111,7 @@ describe('Query extensions', () => { }, ], }) - } catch (e) { + } catch { expect(fs.unlink).toHaveBeenCalledTimes(4) expect(fs.unlink).toHaveBeenNthCalledWith(1, '/one/first.txt') expect(fs.unlink).toHaveBeenNthCalledWith(2, '/one/second.txt') From 7eaa1b0cd48e6c9f6d4f877ab1b36b081bef037a Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Fri, 23 Aug 2024 15:38:08 +0700 Subject: [PATCH 78/91] Remove only from tests --- .../src/__tests__/queryExtensions.test.ts | 62 +------------------ 1 file changed, 1 insertion(+), 61 deletions(-) diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts index 8a000e804098..61b2187320e6 100644 --- a/packages/uploads/src/__tests__/queryExtensions.test.ts +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -90,66 +90,6 @@ describe('Query extensions', () => { }) }) - // Not implemented yet - // Ideally it would just work automatically... but I guess we need to do all the variants - describe.skip('createMany', () => { - it('createMany will remove files if all the create fails', async () => { - try { - await prismaClient.dumbo.createMany({ - data: [ - { - firstUpload: '/one/first.txt', - secondUpload: '/one/second.txt', - // @ts-expect-error Intentional - id: 'break', - }, - { - firstUpload: '/two/first.txt', - secondUpload: '/two/second.txt', - // @ts-expect-error Intentional - id: 'break2', - }, - ], - }) - } catch { - expect(fs.unlink).toHaveBeenCalledTimes(4) - expect(fs.unlink).toHaveBeenNthCalledWith(1, '/one/first.txt') - expect(fs.unlink).toHaveBeenNthCalledWith(2, '/one/second.txt') - expect(fs.unlink).toHaveBeenNthCalledWith(3, '/two/first.txt') - expect(fs.unlink).toHaveBeenNthCalledWith(4, '/two/second.txt') - } - - expect.assertions(4) - }) - - it('createMany will remove files from only the creates that fail', async () => { - try { - await prismaClient.dumbo.createMany({ - data: [ - // This one will go through - { - firstUpload: '/one/first.txt', - secondUpload: '/one/second.txt', - }, - { - firstUpload: '/two/first.txt', - secondUpload: '/two/second.txt', - // @ts-expect-error Intentional - id: 'break2', - }, - ], - }) - } catch (e) { - console.log(e) - expect(fs.unlink).toHaveBeenCalledTimes(2) - expect(fs.unlink).toHaveBeenNthCalledWith(1, '/two/first.txt') - expect(fs.unlink).toHaveBeenNthCalledWith(2, '/two/second.txt') - } - - expect.assertions(4) - }) - }) - describe('update', () => { let ogDummy: Dummy let ogDumbo: Dumbo @@ -204,7 +144,7 @@ describe('Query extensions', () => { expect(fs.unlink).not.toHaveBeenCalled() }) - it.only('should only delete old files from the fields that are being updated', async () => { + it('should only delete old files from the fields that are being updated', async () => { const updatedDumbo = await prismaClient.dumbo.update({ data: { firstUpload: '/tmp/newFirst.txt', From 0e74764d94e2c543fcb6b3287f9301fdf2be3de1 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Fri, 23 Aug 2024 16:29:22 +0700 Subject: [PATCH 79/91] Update tests to pass on windows too --- packages/uploads/src/FileSystemStorage.ts | 9 ++-- .../src/__tests__/FileSystemsStorage.test.ts | 5 +- .../src/__tests__/MemoryStorage.test.ts | 6 ++- .../src/__tests__/createProcessors.test.ts | 46 ++++++++++++++----- .../src/__tests__/queryExtensions.test.ts | 9 ++-- 5 files changed, 54 insertions(+), 21 deletions(-) diff --git a/packages/uploads/src/FileSystemStorage.ts b/packages/uploads/src/FileSystemStorage.ts index 88dfd29b96ca..5f03c96a2b53 100644 --- a/packages/uploads/src/FileSystemStorage.ts +++ b/packages/uploads/src/FileSystemStorage.ts @@ -4,6 +4,8 @@ import path from 'node:path' import mime from 'mime-types' +import { ensurePosixPath } from '@redwoodjs/project-config' + import type { SaveOptionsOverride } from './StorageAdapter.js' import { StorageAdapter } from './StorageAdapter.js' @@ -14,15 +16,16 @@ export class FileSystemStorage constructor(opts: { baseDir: string }) { super(opts) if (!existsSync(opts.baseDir)) { - console.log('Creating baseDir >', opts.baseDir) - mkdirSync(opts.baseDir, { recursive: true }) + const posixBaseDir = ensurePosixPath(opts.baseDir) + console.log('Creating baseDir >', posixBaseDir) + mkdirSync(posixBaseDir, { recursive: true }) } } async save(file: File, saveOverride?: SaveOptionsOverride) { const fileName = this.generateFileNameWithExtension(saveOverride, file) const location = path.join( - saveOverride?.path || this.adapterOpts.baseDir, + ensurePosixPath(saveOverride?.path || this.adapterOpts.baseDir), fileName, ) const nodeBuffer = await file.arrayBuffer() diff --git a/packages/uploads/src/__tests__/FileSystemsStorage.test.ts b/packages/uploads/src/__tests__/FileSystemsStorage.test.ts index 5a266a424eb0..88395e360a23 100644 --- a/packages/uploads/src/__tests__/FileSystemsStorage.test.ts +++ b/packages/uploads/src/__tests__/FileSystemsStorage.test.ts @@ -1,6 +1,8 @@ import { vol } from 'memfs' import { beforeEach, describe, expect, test, vi } from 'vitest' +import { ensurePosixPath } from '@redwoodjs/project-config' + import { FileSystemStorage } from '../FileSystemStorage.js' // Mock the entire fs module @@ -38,7 +40,8 @@ describe('FileSystemStorage', () => { const result = await storage.save(plainFile) expect(result).toHaveProperty('location') - expect(result.location).toMatch(/\/tmp\/test_uploads\/.*\.txt$/) + const posixLocation = ensurePosixPath(result.location) + expect(posixLocation).toMatch(/\/tmp\/test_uploads\/.*\.txt$/) expect(vol.existsSync(result.location)).toBe(true) }) diff --git a/packages/uploads/src/__tests__/MemoryStorage.test.ts b/packages/uploads/src/__tests__/MemoryStorage.test.ts index 12bb49972db2..2f5f427ca998 100644 --- a/packages/uploads/src/__tests__/MemoryStorage.test.ts +++ b/packages/uploads/src/__tests__/MemoryStorage.test.ts @@ -1,5 +1,7 @@ import { describe, expect, test } from 'vitest' +import { ensurePosixPath } from '@redwoodjs/project-config' + import { MemoryStorage } from '../MemoryStorage.js' describe('MemoryStorage', () => { @@ -10,7 +12,7 @@ describe('MemoryStorage', () => { const result = await storage.save(file) expect(result).toHaveProperty('location') - expect(result.location).toMatch(/uploads\/.*\.txt$/) + expect(ensurePosixPath(result.location)).toMatch(/uploads\/.*\.txt$/) expect(storage.store[result.location]).toBeDefined() }) @@ -47,6 +49,6 @@ describe('MemoryStorage', () => { const file = new File(['test content'], 'test.txt', { type: 'text/plain' }) const result = await storage.save(file, { path: 'custom/path' }) - expect(result.location).toContain('custom/path') + expect(ensurePosixPath(result.location)).toContain('custom/path') }) }) diff --git a/packages/uploads/src/__tests__/createProcessors.test.ts b/packages/uploads/src/__tests__/createProcessors.test.ts index b6e7a0aa599a..f559c174593c 100644 --- a/packages/uploads/src/__tests__/createProcessors.test.ts +++ b/packages/uploads/src/__tests__/createProcessors.test.ts @@ -1,5 +1,7 @@ import { describe, it, expect } from 'vitest' +import { ensurePosixPath } from '@redwoodjs/project-config' + import { createFileListProcessor, createUploadProcessors, @@ -43,8 +45,10 @@ describe('Create processors', () => { const result = await processors.processDumboUploads(data) // Location strings in this format: {baseDir/{model}-{field}-{ulid}.{ext} - expect(result.firstUpload).toMatch(/\/memory_store_basedir\/dumbo-*.*\.txt/) - expect(result.secondUpload).toMatch( + expect(ensurePosixPath(result.firstUpload)).toMatch( + /\/memory_store_basedir\/dumbo-*.*\.txt/, + ) + expect(ensurePosixPath(result.secondUpload)).toMatch( /\/memory_store_basedir\/dumbo-*.*\.txt/, ) @@ -77,34 +81,43 @@ describe('Create processors', () => { fileName: 'overridden', }) - expect(fileNameOverrideOnly.uploadField).toBe( + expect(ensurePosixPath(fileNameOverrideOnly.uploadField)).toBe( '/memory_store_basedir/overridden.png', ) - expect(pathOverrideOnly.uploadField).toMatch(/\/bazinga\/.*\.png/) + expect(ensurePosixPath(pathOverrideOnly.uploadField)).toMatch( + /\/bazinga\/.*\.png/, + ) // Overriding path ignores the baseDir expect(pathOverrideOnly.uploadField).not.toContain('memory_store_basedir') - expect(bothOverride.uploadField).toBe('/bazinga/overridden.png') + expect(ensurePosixPath(bothOverride.uploadField)).toBe( + '/bazinga/overridden.png', + ) }) it('Should not add extension for unknown file type', async () => { const data = { uploadField: new File(['Hello'], 'hello', { - type: 'bazinga/unknown', + type: 'bazinga/unknown', // we don't use this anyway }), } const noOverride = await processors.processDummyUploads(data) // No extension - expect(noOverride.uploadField).toMatch(/\/memory_store_basedir\/.*[^.]+$/) + expect(ensurePosixPath(noOverride.uploadField)).toMatch( + /\/memory_store_basedir\/.*[^.]+$/, + ) const withOverride = await processors.processDummyUploads(data, { fileName: 'hello', }) + expect(withOverride.uploadField).toMatch(/[^.]+$/) - expect(withOverride.uploadField).toBe('/memory_store_basedir/hello') + expect(ensurePosixPath(withOverride.uploadField)).toBe( + '/memory_store_basedir/hello', + ) }) }) // FileLists @@ -126,8 +139,13 @@ describe('FileList processing', () => { const result = await fileListProcessor(notPrismaData) expect(result).toHaveLength(2) - expect(result[0]).toMatch(/\/memory_store_basedir\/.*\.png/) - expect(result[1]).toMatch(/\/memory_store_basedir\/.*\.jpeg/) + + expect(ensurePosixPath(result[0])).toMatch( + /\/memory_store_basedir\/.*\.png/, + ) + expect(ensurePosixPath(result[1])).toMatch( + /\/memory_store_basedir\/.*\.jpeg/, + ) }) it('Should handle FileLists with SaveOptions', async () => { @@ -136,8 +154,12 @@ describe('FileList processing', () => { }) expect(result).toHaveLength(2) - expect(result[0]).toMatch(/\/bazinga_not_mem_store\/.*\.png/) - expect(result[1]).toMatch(/\/bazinga_not_mem_store\/.*\.jpeg/) + expect(ensurePosixPath(result[0])).toMatch( + /\/bazinga_not_mem_store\/.*\.png/, + ) + expect(ensurePosixPath(result[1])).toMatch( + /\/bazinga_not_mem_store\/.*\.jpeg/, + ) }) it('Should handle empty FileLists', async () => { diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts index 61b2187320e6..f16f25ea9912 100644 --- a/packages/uploads/src/__tests__/queryExtensions.test.ts +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -3,6 +3,8 @@ import fs from 'node:fs/promises' import type { MockedFunction } from 'vitest' import { describe, it, vi, expect, beforeEach, beforeAll } from 'vitest' +import { ensurePosixPath } from '@redwoodjs/project-config' + import { FileSystemStorage } from '../FileSystemStorage.js' import { setupUploads } from '../index.js' import type { UploadsConfig } from '../prismaExtension.js' @@ -66,9 +68,10 @@ describe('Query extensions', () => { data: processedData, }) - expect(dummy).toMatchObject({ - uploadField: expect.stringMatching(/\/tmp\/.*\.txt$/), - }) + // On windows the slahes are different + const uploadFieldPath = ensurePosixPath(dummy.uploadField) + + expect(uploadFieldPath).toMatch(/\/tmp\/.*\.txt$/) }) it('will remove the file if the create fails', async () => { From e0115a24e3e9450ad423cabcfa390b6292be6f70 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Fri, 23 Aug 2024 16:45:41 +0700 Subject: [PATCH 80/91] Missed one test --- packages/uploads/src/__tests__/FileSystemsStorage.test.ts | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/packages/uploads/src/__tests__/FileSystemsStorage.test.ts b/packages/uploads/src/__tests__/FileSystemsStorage.test.ts index 88395e360a23..7da26ef9f7df 100644 --- a/packages/uploads/src/__tests__/FileSystemsStorage.test.ts +++ b/packages/uploads/src/__tests__/FileSystemsStorage.test.ts @@ -79,7 +79,9 @@ describe('FileSystemStorage', () => { }) // Note that it doesn't have the baseDir! - expect(result.location).toEqual('/my_custom/path/bazinga.txt') + expect(ensurePosixPath(result.location)).toEqual( + '/my_custom/path/bazinga.txt', + ) expect(vol.existsSync(result.location)).toBe(true) }) }) From 4cbb3b1a2c16b7977a68dd5ce58b1e2576b651b8 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 26 Aug 2024 16:28:28 +0700 Subject: [PATCH 81/91] =?UTF-8?q?Add=20new=20line=20to=20end=20of=20packag?= =?UTF-8?q?e.json=20=F0=9F=A4=B7?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/uploads/package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 644d5fa78813..4cbb7d6dc6f3 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -75,4 +75,4 @@ "vitest": "2.0.5" }, "gitHead": "3905ed045508b861b495f8d5630d76c7a157d8f1" -} \ No newline at end of file +} From 6a297d4530b5f2a0dd19056cc0ee633ce0e24600 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 26 Aug 2024 18:53:50 +0700 Subject: [PATCH 82/91] Put fileList processor inside processors --- .../uploads/src/__tests__/createProcessors.test.ts | 13 +++++-------- packages/uploads/src/createProcessors.ts | 7 +++++-- packages/uploads/src/index.ts | 8 +------- 3 files changed, 11 insertions(+), 17 deletions(-) diff --git a/packages/uploads/src/__tests__/createProcessors.test.ts b/packages/uploads/src/__tests__/createProcessors.test.ts index f559c174593c..51dab21cbf64 100644 --- a/packages/uploads/src/__tests__/createProcessors.test.ts +++ b/packages/uploads/src/__tests__/createProcessors.test.ts @@ -2,10 +2,7 @@ import { describe, it, expect } from 'vitest' import { ensurePosixPath } from '@redwoodjs/project-config' -import { - createFileListProcessor, - createUploadProcessors, -} from '../createProcessors.js' +import { createUploadProcessors } from '../createProcessors.js' import { MemoryStorage } from '../MemoryStorage.js' import type { UploadsConfig } from '../prismaExtension.js' @@ -124,7 +121,7 @@ describe('Create processors', () => { // Problem is - in the database world, a string[] is not a thing // so we need a generic way of doing this describe('FileList processing', () => { - const fileListProcessor = createFileListProcessor(memStore) + const processors = createUploadProcessors(uploadsConfig, memStore) const notPrismaData = [ new File(['Hello'], 'hello.png', { @@ -136,7 +133,7 @@ describe('FileList processing', () => { ] it('Should handle FileLists', async () => { - const result = await fileListProcessor(notPrismaData) + const result = await processors.processFileList(notPrismaData) expect(result).toHaveLength(2) @@ -149,7 +146,7 @@ describe('FileList processing', () => { }) it('Should handle FileLists with SaveOptions', async () => { - const result = await fileListProcessor(notPrismaData, { + const result = await processors.processFileList(notPrismaData, { path: '/bazinga_not_mem_store', }) @@ -163,7 +160,7 @@ describe('FileList processing', () => { }) it('Should handle empty FileLists', async () => { - const promise = fileListProcessor() + const promise = processors.processFileList() await expect(promise).resolves.not.toThrow() }) diff --git a/packages/uploads/src/createProcessors.ts b/packages/uploads/src/createProcessors.ts index 263773ee5995..7f867226e8bb 100644 --- a/packages/uploads/src/createProcessors.ts +++ b/packages/uploads/src/createProcessors.ts @@ -65,7 +65,7 @@ export const createUploadProcessors = < const updatedFields = {} as Record for await (const field of currentModelUploadFields) { if (data[field]) { - const file = data[field] as File + const file = data[field] const saveOptions = overrideSaveOptions || { fileName: `${model}-${field}-${ulid()}`, @@ -82,5 +82,8 @@ export const createUploadProcessors = < } }) - return processors + return { + ...processors, + processFileList: createFileListProcessor(storage), + } } diff --git a/packages/uploads/src/index.ts b/packages/uploads/src/index.ts index b02b65f0bd5a..5ebda08c89c2 100644 --- a/packages/uploads/src/index.ts +++ b/packages/uploads/src/index.ts @@ -1,7 +1,4 @@ -import { - createFileListProcessor, - createUploadProcessors, -} from './createProcessors.js' +import { createUploadProcessors } from './createProcessors.js' import type { ModelNames, UploadsConfig } from './prismaExtension.js' import { createUploadsExtension } from './prismaExtension.js' import type { UrlSigner } from './signedUrls.js' @@ -23,12 +20,9 @@ export const setupUploads = ( storageAdapter, ) - const fileListProcessor = createFileListProcessor(storageAdapter) - return { prismaExtension, uploadsProcessors, - fileListProcessor, } } From 9b474d789b476fcc4aae60b2baf4baa3a2e9cddc Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 26 Aug 2024 22:03:20 +0700 Subject: [PATCH 83/91] Remove outdated TODOs --- packages/uploads/src/__tests__/createProcessors.test.ts | 2 -- packages/uploads/src/createProcessors.ts | 2 +- packages/uploads/src/prismaExtension.ts | 3 +-- 3 files changed, 2 insertions(+), 5 deletions(-) diff --git a/packages/uploads/src/__tests__/createProcessors.test.ts b/packages/uploads/src/__tests__/createProcessors.test.ts index 51dab21cbf64..632b9d6ac22e 100644 --- a/packages/uploads/src/__tests__/createProcessors.test.ts +++ b/packages/uploads/src/__tests__/createProcessors.test.ts @@ -10,8 +10,6 @@ const memStore = new MemoryStorage({ baseDir: '/memory_store_basedir', }) -// @TODO(TS): How can I make this accept not all model names? -// This is error-ing out because it wants all the models in my prisma client const uploadsConfig: UploadsConfig = { dumbo: { fields: ['firstUpload', 'secondUpload'], diff --git a/packages/uploads/src/createProcessors.ts b/packages/uploads/src/createProcessors.ts index 7f867226e8bb..08a06c7e2028 100644 --- a/packages/uploads/src/createProcessors.ts +++ b/packages/uploads/src/createProcessors.ts @@ -35,9 +35,9 @@ export const createUploadProcessors = < type uploadProcessorNames = `process${Capitalize}Uploads` + // @TODO(TS): Is there a way to make the type of data more specific? type Processors = { [K in uploadProcessorNames]: >( - // @TODO(TS): T should be the type of the model data: T, overrideSaveOptions?: SaveOptionsOverride, ) => Promise> diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index db5c31262b54..d3960959269a 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -211,8 +211,7 @@ export const createUploadsExtension = ( }) }) - // @TODO(TS): According to TS, data could be a non-object... - // Setting args to JsArgs causes errors. This could be a legit issue + async function removeUploadedFiles( fieldsToDelete: string[], data: Record, From b7f6b9f63a52de22c336b16b4247cdce94ec92b1 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Mon, 26 Aug 2024 22:16:27 +0700 Subject: [PATCH 84/91] Prettier --- packages/uploads/src/prismaExtension.ts | 1 - 1 file changed, 1 deletion(-) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index d3960959269a..0c53ddf82b26 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -211,7 +211,6 @@ export const createUploadsExtension = ( }) }) - async function removeUploadedFiles( fieldsToDelete: string[], data: Record, From 3434411c2b742e0277bf91ea4c64de5caecc5cb9 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Tue, 27 Aug 2024 23:08:51 +0700 Subject: [PATCH 85/91] Make with signedUrl result extension take an object instead --- packages/uploads/src/__tests__/resultExtensions.test.ts | 4 +++- packages/uploads/src/prismaExtension.ts | 4 ++-- 2 files changed, 5 insertions(+), 3 deletions(-) diff --git a/packages/uploads/src/__tests__/resultExtensions.test.ts b/packages/uploads/src/__tests__/resultExtensions.test.ts index 89165925afb0..2469f4901093 100644 --- a/packages/uploads/src/__tests__/resultExtensions.test.ts +++ b/packages/uploads/src/__tests__/resultExtensions.test.ts @@ -54,7 +54,9 @@ describe('Result extensions', () => { }, }) - const signedUrlDumbo = await dumbo.withSignedUrl(254) + const signedUrlDumbo = await dumbo.withSignedUrl({ + expiresIn: 254, + }) expect(signedUrlDumbo.firstUpload).toContain( '/.redwood/functions/signed-url', ) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 0c53ddf82b26..f116a79f952b 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -53,7 +53,7 @@ export const createUploadsExtension = ( needs: Record compute: ( modelData: Record, - ) => (this: T, expiresIn?: number) => Promise + ) => (this: T, { expiresIn }: { expiresIn?: number }) => Promise } } } @@ -173,7 +173,7 @@ export const createUploadsExtension = ( withSignedUrl: { needs, compute(modelData) { - return (expiresIn?: number) => { + return ({ expiresIn }: { expiresIn?: number }) => { if (!urlSigner) { throw new Error( 'Please supply signed url settings in setupUpload()', From a0d4183b6df76d8681520ab23265c10a2c6f751c Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Wed, 28 Aug 2024 11:16:42 +0700 Subject: [PATCH 86/91] Make SignedUrl args optional --- packages/uploads/src/prismaExtension.ts | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index f116a79f952b..985045be784c 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -32,6 +32,10 @@ export type UploadsConfig = { [K in MNames]?: UploadConfigForModel } +type WithSignedUrlArgs = { + expiresIn?: number +} + export const createUploadsExtension = ( config: UploadsConfig, storageAdapter: StorageAdapter, @@ -53,7 +57,7 @@ export const createUploadsExtension = ( needs: Record compute: ( modelData: Record, - ) => (this: T, { expiresIn }: { expiresIn?: number }) => Promise + ) => (this: T, signArgs?: WithSignedUrlArgs) => Promise } } } @@ -173,7 +177,7 @@ export const createUploadsExtension = ( withSignedUrl: { needs, compute(modelData) { - return ({ expiresIn }: { expiresIn?: number }) => { + return ({ expiresIn }: WithSignedUrlArgs = {}) => { if (!urlSigner) { throw new Error( 'Please supply signed url settings in setupUpload()', From 6d683c75bfe0cd08f16e10bb424840529670c567 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 29 Aug 2024 13:32:18 +0700 Subject: [PATCH 87/91] Accept dataloss in uploads prisma setup for tests --- packages/uploads/package.json | 4 ++-- packages/uploads/vitest.setup.mts | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 4cbb7d6dc6f3..2f1895d0cb3d 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -53,7 +53,7 @@ "build:types-cjs": "tsc --build --verbose tsconfig.types-cjs.json", "check:attw": "tsx attw.ts", "check:package": "concurrently npm:check:attw yarn publint", - "setup:test": "npx prisma db push --schema ./src/__tests__/unit-test-schema.prisma", + "setup:test": "npx prisma db push --accept-data-loss --schema ./src/__tests__/unit-test-schema.prisma", "test": "vitest run", "test:watch": "vitest watch" }, @@ -75,4 +75,4 @@ "vitest": "2.0.5" }, "gitHead": "3905ed045508b861b495f8d5630d76c7a157d8f1" -} +} \ No newline at end of file diff --git a/packages/uploads/vitest.setup.mts b/packages/uploads/vitest.setup.mts index 80434f24618b..702c707e71da 100644 --- a/packages/uploads/vitest.setup.mts +++ b/packages/uploads/vitest.setup.mts @@ -3,6 +3,6 @@ import { $ } from 'zx' export default async function setup() { $.verbose = true console.log('[setup] Setting up unit test prisma db....') - await $`npx prisma db push --schema ./src/__tests__/unit-test-schema.prisma` + await $`npx prisma db push --accept-data-loss --schema ./src/__tests__/unit-test-schema.prisma` console.log('[setup] Done! \n') } From 1b43a2374661b4f4c24d08365e533798131bac4c Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 29 Aug 2024 16:34:22 +0700 Subject: [PATCH 88/91] Add tests for delete failure --- .../src/__tests__/queryExtensions.test.ts | 27 +++++++++++++++++++ .../src/__tests__/unit-test-schema.prisma | 26 +++++++++++++----- packages/uploads/src/prismaExtension.ts | 14 +++++----- 3 files changed, 55 insertions(+), 12 deletions(-) diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts index f16f25ea9912..faa47cc587d4 100644 --- a/packages/uploads/src/__tests__/queryExtensions.test.ts +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -198,6 +198,33 @@ describe('Query extensions', () => { expect(fs.unlink).toHaveBeenCalledWith('/tmp/second.txt') }) + it('delete will not remove any uploads if the delete fails', async () => { + const bookWithCover = await prismaClient.book.create({ + data: { + name: 'Prisma extensions for dummies', + cover: { + create: { + photo: '/tmp/book-covers/prisma-for-dummies.jpg', + }, + }, + }, + }) + + // This delete will fail because the book is associated with a cover BUTTTT + // test serves more as documentation (and to prevent regression if Prisma changes behavior) + // Because Prisma will throw the validation __before__ the delete in the extension is called + + try { + await prismaClient.bookCover.delete({ + where: { + id: bookWithCover.coverId, + }, + }) + } catch {} + + expect(fs.unlink).not.toHaveBeenCalled() + }) + it('Should handle if a bad path is provided', async () => { ;(fs.unlink as MockedFunction).mockRejectedValueOnce( new Error('unlink error'), diff --git a/packages/uploads/src/__tests__/unit-test-schema.prisma b/packages/uploads/src/__tests__/unit-test-schema.prisma index 1a8cd0489e04..21f28a1f1ccb 100644 --- a/packages/uploads/src/__tests__/unit-test-schema.prisma +++ b/packages/uploads/src/__tests__/unit-test-schema.prisma @@ -9,18 +9,32 @@ generator client { } model Dummy { - id Int @id @default(autoincrement()) + id Int @id @default(autoincrement()) uploadField String } model Dumbo { - id Int @id @default(autoincrement()) - firstUpload String + id Int @id @default(autoincrement()) + firstUpload String secondUpload String - message String? + message String? } model NoUploadFields { - id Int @id @default(autoincrement()) + id Int @id @default(autoincrement()) name String -} \ No newline at end of file +} + +model Book { + id Int @id @default(autoincrement()) + coverId Int @unique + cover BookCover @relation(fields: [coverId], references: [id]) + name String +} + +model BookCover { + id Int @id @default(autoincrement()) + // This is the upload field, + photo String + book Book? +} diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 985045be784c..8be0f0dc9a3b 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -138,14 +138,16 @@ export const createUploadsExtension = ( }, async delete({ model, query, args }) { - /** Delete args are the same as findFirst, essentially a where clause */ - const record = - // @ts-expect-error TS in strict mode will error due to union type. We cannot narrow it down here. - await prismaInstance[model as ModelNames].findFirstOrThrow(args) + const deleteResult = await query(args) + storageAdapter.remove(args.where.id) - await removeUploadedFiles(uploadFields, record) + await removeUploadedFiles( + uploadFields, + // We don't know the exact type here + deleteResult as Record, + ) - return query(args) + return deleteResult }, } From b6586ef22f1169e55ba77d12beff73b463394586 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 29 Aug 2024 17:01:09 +0700 Subject: [PATCH 89/91] Whoopsie --- packages/uploads/src/prismaExtension.ts | 2 -- 1 file changed, 2 deletions(-) diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index 8be0f0dc9a3b..b8a4498baaa1 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -139,8 +139,6 @@ export const createUploadsExtension = ( async delete({ model, query, args }) { const deleteResult = await query(args) - storageAdapter.remove(args.where.id) - await removeUploadedFiles( uploadFields, // We don't know the exact type here From 4508df719bfb9bea9a1efb6df35f3921b4682747 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 29 Aug 2024 17:14:39 +0700 Subject: [PATCH 90/91] Allow empty catch in query extension test --- packages/uploads/src/__tests__/queryExtensions.test.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/uploads/src/__tests__/queryExtensions.test.ts b/packages/uploads/src/__tests__/queryExtensions.test.ts index faa47cc587d4..70cfad682547 100644 --- a/packages/uploads/src/__tests__/queryExtensions.test.ts +++ b/packages/uploads/src/__tests__/queryExtensions.test.ts @@ -220,6 +220,7 @@ describe('Query extensions', () => { id: bookWithCover.coverId, }, }) + // eslint-disable-next-line no-empty } catch {} expect(fs.unlink).not.toHaveBeenCalled() From 5351342f2de535346b96631fb1359994bfd5a408 Mon Sep 17 00:00:00 2001 From: Danny Choudhury Date: Thu, 29 Aug 2024 17:15:06 +0700 Subject: [PATCH 91/91] Lint --- packages/uploads/package.json | 2 +- packages/uploads/src/prismaExtension.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/uploads/package.json b/packages/uploads/package.json index 2f1895d0cb3d..2f58a831df53 100644 --- a/packages/uploads/package.json +++ b/packages/uploads/package.json @@ -75,4 +75,4 @@ "vitest": "2.0.5" }, "gitHead": "3905ed045508b861b495f8d5630d76c7a157d8f1" -} \ No newline at end of file +} diff --git a/packages/uploads/src/prismaExtension.ts b/packages/uploads/src/prismaExtension.ts index b8a4498baaa1..8d532382a98e 100644 --- a/packages/uploads/src/prismaExtension.ts +++ b/packages/uploads/src/prismaExtension.ts @@ -137,7 +137,7 @@ export const createUploadsExtension = ( } }, - async delete({ model, query, args }) { + async delete({ query, args }) { const deleteResult = await query(args) await removeUploadedFiles( uploadFields,