From 4c2f7e11771821d3e9a878899279d0f47ef1a05a Mon Sep 17 00:00:00 2001 From: Tobbe Lundberg Date: Thu, 18 Jan 2024 12:38:42 +0100 Subject: [PATCH] Add support for Prisma Bytes --- .../generate/__tests__/helpers.test.js | 4 + packages/cli/src/commands/generate/helpers.js | 3 +- .../__tests__/__snapshots__/sdl.test.js.snap | 172 ++++++++++++++++++ .../sdl/__tests__/fixtures/schema.prisma | 5 + .../generate/sdl/__tests__/sdl.test.js | 55 +++++- packages/cli/src/commands/generate/sdl/sdl.js | 12 +- .../src/commands/generate/service/service.js | 17 +- .../service/templates/test.ts.template | 7 +- packages/graphql-server/src/rootSchema.ts | 4 + .../src/__tests__/graphqlCodeGen.test.ts | 1 + packages/internal/src/generate/generate.ts | 1 + .../internal/src/generate/graphqlCodeGen.ts | 3 +- 12 files changed, 264 insertions(+), 20 deletions(-) diff --git a/packages/cli/src/commands/generate/__tests__/helpers.test.js b/packages/cli/src/commands/generate/__tests__/helpers.test.js index 1714901af7f0..5902292277f3 100644 --- a/packages/cli/src/commands/generate/__tests__/helpers.test.js +++ b/packages/cli/src/commands/generate/__tests__/helpers.test.js @@ -541,6 +541,10 @@ describe('mapPrismaScalarToPagePropTsType', () => { expect(helpers.mapPrismaScalarToPagePropTsType('DateTime')).toBe('string') }) + it('maps scalar type Bytes to TS type Buffer', () => { + expect(helpers.mapPrismaScalarToPagePropTsType('Bytes')).toBe('Buffer') + }) + it('maps all other type not-known to TS to unknown', () => { expect(helpers.mapPrismaScalarToPagePropTsType('Json')).toBe('unknown') }) diff --git a/packages/cli/src/commands/generate/helpers.js b/packages/cli/src/commands/generate/helpers.js index e3d48ff99c7d..c17bb6eb2985 100644 --- a/packages/cli/src/commands/generate/helpers.js +++ b/packages/cli/src/commands/generate/helpers.js @@ -298,7 +298,7 @@ export const mapRouteParamTypeToTsType = (paramType) => { return routeParamToTsType[paramType] || 'unknown' } -/** @type {(scalarType: 'String' | 'Boolean' | 'Int' | 'BigInt' | 'Float' | 'Decimal' | 'DateTime' ) => string } **/ +/** @type {(scalarType: 'String' | 'Boolean' | 'Int' | 'BigInt' | 'Float' | 'Decimal' | 'DateTime' | 'Bytes' ) => string } **/ export const mapPrismaScalarToPagePropTsType = (scalarType) => { const prismaScalarToTsType = { String: 'string', @@ -308,6 +308,7 @@ export const mapPrismaScalarToPagePropTsType = (scalarType) => { Float: 'number', Decimal: 'number', DateTime: 'string', + Bytes: 'Buffer', } return prismaScalarToTsType[scalarType] || 'unknown' } diff --git a/packages/cli/src/commands/generate/sdl/__tests__/__snapshots__/sdl.test.js.snap b/packages/cli/src/commands/generate/sdl/__tests__/__snapshots__/sdl.test.js.snap index 4121cc48665d..bbe1df8bc373 100644 --- a/packages/cli/src/commands/generate/sdl/__tests__/__snapshots__/sdl.test.js.snap +++ b/packages/cli/src/commands/generate/sdl/__tests__/__snapshots__/sdl.test.js.snap @@ -728,6 +728,63 @@ exports[`with graphql documentations in javascript mode creates a multi word sdl " `; +exports[`with graphql documentations in javascript mode creates a sdl file with Byte definitions 1`] = ` +"export const schema = gql\` + """ + Representation of Key. + """ + type Key { + "Description for id." + id: Int! + + "Description for publicKey." + publicKey: Byte! + } + + """ + About queries + """ + type Query { + "Fetch Keys." + keys: [Key!]! @requireAuth + + "Fetch a Key by id." + key(id: Int!): Key @requireAuth + } + + """ + Autogenerated input type of InputKey. + """ + input CreateKeyInput { + "Description for publicKey." + publicKey: Byte! + } + + """ + Autogenerated input type of UpdateKey. + """ + input UpdateKeyInput { + "Description for publicKey." + publicKey: Byte + } + + """ + About mutations + """ + type Mutation { + "Creates a new Key." + createKey(input: CreateKeyInput!): Key! @requireAuth + + "Updates an existing Key." + updateKey(id: Int!, input: UpdateKeyInput!): Key! @requireAuth + + "Deletes an existing Key." + deleteKey(id: Int!): Key! @requireAuth + } +\` +" +`; + exports[`with graphql documentations in javascript mode creates a sdl file with enum definitions 1`] = ` "export const schema = gql\` """ @@ -1163,6 +1220,63 @@ exports[`with graphql documentations in typescript mode creates a multi word sdl " `; +exports[`with graphql documentations in typescript mode creates a sdl file with Byte definitions 1`] = ` +"export const schema = gql\` + """ + Representation of Key. + """ + type Key { + "Description for id." + id: Int! + + "Description for publicKey." + publicKey: Byte! + } + + """ + About queries + """ + type Query { + "Fetch Keys." + keys: [Key!]! @requireAuth + + "Fetch a Key by id." + key(id: Int!): Key @requireAuth + } + + """ + Autogenerated input type of InputKey. + """ + input CreateKeyInput { + "Description for publicKey." + publicKey: Byte! + } + + """ + Autogenerated input type of UpdateKey. + """ + input UpdateKeyInput { + "Description for publicKey." + publicKey: Byte + } + + """ + About mutations + """ + type Mutation { + "Creates a new Key." + createKey(input: CreateKeyInput!): Key! @requireAuth + + "Updates an existing Key." + updateKey(id: Int!, input: UpdateKeyInput!): Key! @requireAuth + + "Deletes an existing Key." + deleteKey(id: Int!): Key! @requireAuth + } +\` +" +`; + exports[`with graphql documentations in typescript mode creates a sdl file with enum definitions 1`] = ` "export const schema = gql\` """ @@ -1526,6 +1640,35 @@ exports[`without graphql documentations in javascript mode creates a multi word " `; +exports[`without graphql documentations in javascript mode creates a sdl file with Byte definitions 1`] = ` +"export const schema = gql\` + type Key { + id: Int! + publicKey: Byte! + } + + type Query { + keys: [Key!]! @requireAuth + key(id: Int!): Key @requireAuth + } + + input CreateKeyInput { + publicKey: Byte! + } + + input UpdateKeyInput { + publicKey: Byte + } + + type Mutation { + createKey(input: CreateKeyInput!): Key! @requireAuth + updateKey(id: Int!, input: UpdateKeyInput!): Key! @requireAuth + deleteKey(id: Int!): Key! @requireAuth + } +\` +" +`; + exports[`without graphql documentations in javascript mode creates a sdl file with enum definitions 1`] = ` "export const schema = gql\` type Shoe { @@ -1734,6 +1877,35 @@ exports[`without graphql documentations in typescript mode creates a multi word " `; +exports[`without graphql documentations in typescript mode creates a sdl file with Byte definitions 1`] = ` +"export const schema = gql\` + type Key { + id: Int! + publicKey: Byte! + } + + type Query { + keys: [Key!]! @requireAuth + key(id: Int!): Key @requireAuth + } + + input CreateKeyInput { + publicKey: Byte! + } + + input UpdateKeyInput { + publicKey: Byte + } + + type Mutation { + createKey(input: CreateKeyInput!): Key! @requireAuth + updateKey(id: Int!, input: UpdateKeyInput!): Key! @requireAuth + deleteKey(id: Int!): Key! @requireAuth + } +\` +" +`; + exports[`without graphql documentations in typescript mode creates a sdl file with enum definitions 1`] = ` "export const schema = gql\` type Shoe { diff --git a/packages/cli/src/commands/generate/sdl/__tests__/fixtures/schema.prisma b/packages/cli/src/commands/generate/sdl/__tests__/fixtures/schema.prisma index 86d854f9895d..d31245c497a1 100644 --- a/packages/cli/src/commands/generate/sdl/__tests__/fixtures/schema.prisma +++ b/packages/cli/src/commands/generate/sdl/__tests__/fixtures/schema.prisma @@ -53,6 +53,11 @@ model Photo { metadata Json } +model Key { + id Int @id @default(autoincrement()) + publicKey Bytes +} + /// A list of allowed colors. enum Color { RED diff --git a/packages/cli/src/commands/generate/sdl/__tests__/sdl.test.js b/packages/cli/src/commands/generate/sdl/__tests__/sdl.test.js index 081aebe7ba19..eaeb85d5fa5b 100644 --- a/packages/cli/src/commands/generate/sdl/__tests__/sdl.test.js +++ b/packages/cli/src/commands/generate/sdl/__tests__/sdl.test.js @@ -2,6 +2,24 @@ globalThis.__dirname = __dirname globalThis.mockFs = false +// jest.mock('console', () => { +// return { +// info: jest.fn(), +// log: jest.fn(), +// } +// }) + +// jest.mock('process', () => { +// const actualProcess = jest.requireActual('process') +// return { +// ...actualProcess, +// stdout: { +// ...actualProcess.stdout, +// write: () => true, +// }, +// } +// }) + jest.mock('fs', () => { const actual = jest.requireActual('fs') @@ -37,7 +55,15 @@ import { ensurePosixPath } from '@redwoodjs/project-config' import { getDefaultArgs } from '../../../../lib' import * as sdl from '../sdl' +beforeEach(() => { + // jest.spyOn(console, 'info').mockImplementation(() => {}) + // jest.spyOn(console, 'log').mockImplementation(() => {}) +}) + afterEach(() => { + // console.info.mockRestore() + // console.log.mockRestore() + jest.clearAllMocks() }) @@ -203,6 +229,21 @@ const itCreatesAnSDLFileWithJsonDefinitions = (baseArgs = {}) => { }) } +const itCreatesAnSDLFileWithByteDefinitions = (baseArgs = {}) => { + test('creates a sdl file with Byte definitions', async () => { + const files = await sdl.files({ + ...baseArgs, + name: 'Key', + crud: true, + }) + const ext = extensionForBaseArgs(baseArgs) + + expect( + files[path.normalize(`/path/to/project/api/src/graphql/keys.sdl.${ext}`)] + ).toMatchSnapshot() + }) +} + describe('without graphql documentations', () => { describe('in javascript mode', () => { const baseArgs = { ...getDefaultArgs(sdl.defaults), tests: true } @@ -215,6 +256,7 @@ describe('without graphql documentations', () => { itCreateAMultiWordSDLFileWithCRUD(baseArgs) itCreatesAnSDLFileWithEnumDefinitions(baseArgs) itCreatesAnSDLFileWithJsonDefinitions(baseArgs) + itCreatesAnSDLFileWithByteDefinitions(baseArgs) }) describe('in typescript mode', () => { @@ -232,6 +274,7 @@ describe('without graphql documentations', () => { itCreateAMultiWordSDLFileWithCRUD(baseArgs) itCreatesAnSDLFileWithEnumDefinitions(baseArgs) itCreatesAnSDLFileWithJsonDefinitions(baseArgs) + itCreatesAnSDLFileWithByteDefinitions(baseArgs) }) }) @@ -251,6 +294,7 @@ describe('with graphql documentations', () => { itCreateAMultiWordSDLFileWithCRUD(baseArgs) itCreatesAnSDLFileWithEnumDefinitions(baseArgs) itCreatesAnSDLFileWithJsonDefinitions(baseArgs) + itCreatesAnSDLFileWithByteDefinitions(baseArgs) }) describe('in typescript mode', () => { @@ -269,20 +313,11 @@ describe('with graphql documentations', () => { itCreateAMultiWordSDLFileWithCRUD(baseArgs) itCreatesAnSDLFileWithEnumDefinitions(baseArgs) itCreatesAnSDLFileWithJsonDefinitions(baseArgs) + itCreatesAnSDLFileWithByteDefinitions(baseArgs) }) }) describe('handler', () => { - beforeEach(() => { - jest.spyOn(console, 'info').mockImplementation(() => {}) - jest.spyOn(console, 'log').mockImplementation(() => {}) - }) - - afterEach(() => { - console.info.mockRestore() - console.log.mockRestore() - }) - const canBeCalledWithGivenModelName = (letterCase, model) => { test(`can be called with ${letterCase} model name`, async () => { const spy = jest.spyOn(fs, 'writeFileSync') diff --git a/packages/cli/src/commands/generate/sdl/sdl.js b/packages/cli/src/commands/generate/sdl/sdl.js index 60344bba965a..98fe50356c2f 100644 --- a/packages/cli/src/commands/generate/sdl/sdl.js +++ b/packages/cli/src/commands/generate/sdl/sdl.js @@ -69,13 +69,14 @@ const modelFieldToSDL = ({ field.kind === 'object' ? idType(types[field.type]) : field.type } - const dictionary = { + const prismaTypeToGraphqlType = { Json: 'JSON', Decimal: 'Float', + Bytes: 'Byte', } const fieldContent = `${field.name}: ${field.isList ? '[' : ''}${ - dictionary[field.type] || field.type + prismaTypeToGraphqlType[field.type] || field.type }${field.isList ? ']' : ''}${ (field.isRequired && required) | field.isList ? '!' : '' }` @@ -331,6 +332,7 @@ export const handler = async ({ { title: `Generating types ...`, task: async () => { + console.log('about to generate types') const { errors } = await generateTypes() for (const { message, error } of errors) { @@ -344,7 +346,11 @@ export const handler = async ({ }, }, ].filter(Boolean), - { rendererOptions: { collapseSubtasks: false }, exitOnError: true } + { + rendererOptions: { collapseSubtasks: false }, + exitOnError: true, + silentRendererCondition: process.env.NODE_ENV === 'test', + } ) if (rollback && !force) { diff --git a/packages/cli/src/commands/generate/service/service.js b/packages/cli/src/commands/generate/service/service.js index 3f4d7377493a..b3df5afac1cd 100644 --- a/packages/cli/src/commands/generate/service/service.js +++ b/packages/cli/src/commands/generate/service/service.js @@ -44,6 +44,11 @@ export const parseSchema = async (model) => { export const scenarioFieldValue = (field) => { const randFloat = Math.random() * 10000000 const randInt = parseInt(Math.random() * 10000000) + const randIntArray = [ + parseInt(Math.random() * 300), + parseInt(Math.random() * 300), + parseInt(Math.random() * 300), + ] switch (field.type) { case 'BigInt': @@ -61,6 +66,8 @@ export const scenarioFieldValue = (field) => { return { foo: 'bar' } case 'String': return field.isUnique ? `String${randInt}` : 'String' + case 'Bytes': + return `Buffer.from([${randIntArray}])` default: { if (field.kind === 'enum' && field.enumValues[0]) { return field.enumValues[0].dbName || field.enumValues[0].name @@ -125,8 +132,9 @@ export const buildScenario = async (model) => { Object.keys(scenarioData).forEach((key) => { const value = scenarioData[key] + // Support BigInt if (value && typeof value === 'string' && value.match(/^\d+n$/)) { - scenarioData[key] = `${value.substr(0, value.length - 1)}n` + scenarioData[key] = `${value.slice(0, value.length - 1)}n` } }) @@ -141,17 +149,20 @@ export const buildScenario = async (model) => { export const buildStringifiedScenario = async (model) => { const scenario = await buildScenario(model) - return JSON.stringify(scenario, (key, value) => { + const jsonString = JSON.stringify(scenario, (_key, value) => { if (typeof value === 'bigint') { return value.toString() } if (typeof value === 'string' && value.match(/^\d+n$/)) { - return Number(value.substr(0, value.length - 1)) + return Number(value.slice(0, value.length - 1)) } return value }) + + // Not all values can be represented as JSON, like function invocations + return jsonString.replace(/"Buffer\.from\(([^)]+)\)"/g, 'Buffer.from($1)') } export const fieldTypes = async (model) => { diff --git a/packages/cli/src/commands/generate/service/templates/test.ts.template b/packages/cli/src/commands/generate/service/templates/test.ts.template index e85bfda525df..3d2200e28160 100644 --- a/packages/cli/src/commands/generate/service/templates/test.ts.template +++ b/packages/cli/src/commands/generate/service/templates/test.ts.template @@ -15,18 +15,21 @@ return `new Prisma.Decimal(${obj})` } - return JSON.stringify(obj).replace(/['"].*?['"]/g, (string) => { + const jsonString = JSON.stringify(obj).replace(/['"].*?['"]/g, (string) => { if (string.match(/scenario\./)) { return string.replace(/['"]/g, '') } // BigInt if (string.match(/^\"\d+n\"$/)) { - return string.substr(1, string.length - 2) + return string.slice(1, string.length - 2) } return string }) + + // Not all values can be represented as JSON, like function invocations + return jsonString.replace(/"Buffer\.from\(([^)]+)\)"/g, 'Buffer.from($1)') } %> <% if (prismaImport) { %>import { Prisma, ${prismaModel} } from '@prisma/client'<% } else { %>import type { ${prismaModel} } from '@prisma/client'<% } %> diff --git a/packages/graphql-server/src/rootSchema.ts b/packages/graphql-server/src/rootSchema.ts index 7332d49b94ed..4010e7d87d99 100644 --- a/packages/graphql-server/src/rootSchema.ts +++ b/packages/graphql-server/src/rootSchema.ts @@ -6,6 +6,7 @@ import { DateTimeResolver, JSONResolver, JSONObjectResolver, + ByteResolver, } from 'graphql-scalars' import gql from 'graphql-tag' @@ -29,6 +30,7 @@ export const schema = gql` scalar DateTime scalar JSON scalar JSONObject + scalar Byte """ The RedwoodJS Root Schema @@ -61,6 +63,7 @@ export interface Resolvers { JSON: typeof JSONResolver JSONObject: typeof JSONObjectResolver Query: Record + Byte: typeof ByteResolver } export const resolvers: Resolvers = { @@ -79,4 +82,5 @@ export const resolvers: Resolvers = { }, }), }, + Byte: ByteResolver, } diff --git a/packages/internal/src/__tests__/graphqlCodeGen.test.ts b/packages/internal/src/__tests__/graphqlCodeGen.test.ts index e968b14e1abd..d87613c518fc 100644 --- a/packages/internal/src/__tests__/graphqlCodeGen.test.ts +++ b/packages/internal/src/__tests__/graphqlCodeGen.test.ts @@ -83,6 +83,7 @@ test('Generate gql typedefs api', async () => { // Check that JSON types are imported from prisma expect(data).toContain('JSON: Prisma.JsonValue;') expect(data).toContain('JSONObject: Prisma.JsonObject;') + expect(data).toContain('Byte: Buffer;') // Check that prisma model imports are added to the top of the file expect(data).toContain( diff --git a/packages/internal/src/generate/generate.ts b/packages/internal/src/generate/generate.ts index 51d3020a4d8a..847e02dec82b 100644 --- a/packages/internal/src/generate/generate.ts +++ b/packages/internal/src/generate/generate.ts @@ -8,6 +8,7 @@ import { generatePossibleTypes } from './possibleTypes' import { generateTypeDefs } from './typeDefinitions' export const generate = async () => { + console.log('internal generate') const config = getConfig() const { schemaPath, errors: generateGraphQLSchemaErrors } = await generateGraphQLSchema() diff --git a/packages/internal/src/generate/graphqlCodeGen.ts b/packages/internal/src/generate/graphqlCodeGen.ts index a5e44b7d4fa9..90dafd158133 100644 --- a/packages/internal/src/generate/graphqlCodeGen.ts +++ b/packages/internal/src/generate/graphqlCodeGen.ts @@ -80,7 +80,7 @@ export const generateTypeDefGraphQLApi = async (): Promise => { codegenPlugin: addPlugin, }, { - name: 'print-mapped-moddels', + name: 'print-mapped-models', options: {}, codegenPlugin: printMappedModelsPlugin, }, @@ -285,6 +285,7 @@ function getPluginConfig(side: CodegenSide) { JSON: 'Prisma.JsonValue', JSONObject: 'Prisma.JsonObject', Time: side === CodegenSide.WEB ? 'string' : 'Date | string', + Byte: 'Buffer', }, // prevent type names being PetQueryQuery, RW generators already append // Query/Mutation/etc