From b659df811b2015afb06696cf6bb15054c793bfa8 Mon Sep 17 00:00:00 2001 From: Marcel Schwarz Date: Mon, 29 Jul 2024 10:54:42 +0200 Subject: [PATCH] feat: use `serviceinfo` to determine services to compile (#178) Use `serviceinfo` to determine services to compile instead of directly inspecting service definition for `@protocol` or `@graphql` annotations, since this logic is duplicated in `@sap/cds`. --- CHANGELOG.md | 2 + lib/compile.js | 27 ++------ package.json | 2 +- test/resources/annotations/srv/protocols.cds | 40 +++++------ test/schemas/annotations.gql | 72 ++++++++++++++++++++ test/tests/annotations.test.js | 32 +++++++++ 6 files changed, 135 insertions(+), 40 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index f5bd5c99..57d41a43 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -13,8 +13,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ### Changed +- Bump required `@sap/cds` version to `>=7.8` - To improve performance, binary payloads are no longer validated to check if they are properly base64 or base64url encoded - Bump required `node` version to `^16` due to usage of `Buffer.toString('base64url')` +- Use `cds.compile.to.serviceinfo` to determine if a service should be compiled to GraphQL schema ### Fixed diff --git a/lib/compile.js b/lib/compile.js index 54d0e502..93635876 100644 --- a/lib/compile.js +++ b/lib/compile.js @@ -2,29 +2,16 @@ const cds = require('@sap/cds') const { generateSchema4 } = require('./schema') const { lexicographicSortSchema, printSchema } = require('graphql') -const _isServiceAnnotatedWithGraphQL = service => { - const { definition } = service - - if (definition['@graphql']) return true - - const protocol = definition['@protocol'] - if (protocol) { - // @protocol: 'graphql' or @protocol: ['graphql', 'odata'] - const protocols = Array.isArray(protocol) ? protocol : [protocol] - // Normalize objects such as { kind: 'graphql' } to strings - return protocols.map(p => (typeof p === 'object' ? p.kind : p)).some(p => p.match(/graphql/i)) - } - - return false -} - function cds_compile_to_gql(csn, options = {}) { - const m = cds.linked(csn) + const model = cds.linked(csn) + const serviceinfo = cds.compile.to.serviceinfo(csn, options) const services = Object.fromEntries( - m.services - .map(s => [s.name, new cds.ApplicationService(s.name, m)]) + model.services + .map(s => [s.name, new cds.ApplicationService(s.name, model)]) // Only compile services with GraphQL endpoints - .filter(([_, service]) => _isServiceAnnotatedWithGraphQL(service)) + .filter(([_, service]) => + serviceinfo.find(s => s.name === service.name)?.endpoints.some(e => e.kind === 'graphql') + ) ) let schema = generateSchema4(services) diff --git a/package.json b/package.json index dd1629f5..25b5f732 100644 --- a/package.json +++ b/package.json @@ -36,7 +36,7 @@ "graphql-http": "^1.18.0" }, "peerDependencies": { - "@sap/cds": ">=7.3" + "@sap/cds": ">=7.8" }, "devDependencies": { "@cap-js/graphql": "file:.", diff --git a/test/resources/annotations/srv/protocols.cds b/test/resources/annotations/srv/protocols.cds index c4d0947c..654d4235 100644 --- a/test/resources/annotations/srv/protocols.cds +++ b/test/resources/annotations/srv/protocols.cds @@ -1,47 +1,49 @@ -service NotAnnotated { +context protocols { entity A { key id : UUID; } } +service NotAnnotated { + entity A as projection on protocols.A; +} + @protocol: 'none' service AnnotatedWithAtProtocolNone { - entity A { - key id : UUID; - } + entity A as projection on protocols.A; } @protocol: 'odata' service AnnotatedWithNonGraphQL { - entity A { - key id : UUID; - } + entity A as projection on protocols.A; } @graphql service AnnotatedWithAtGraphQL { - entity A { - key id : UUID; - } + entity A as projection on protocols.A; } @protocol: 'graphql' service AnnotatedWithAtProtocolString { - entity A { - key id : UUID; - } + entity A as projection on protocols.A; } @protocol: ['graphql'] service AnnotatedWithAtProtocolStringList { - entity A { - key id : UUID; - } + entity A as projection on protocols.A; } @protocol: [{kind: 'graphql'}] service AnnotatedWithAtProtocolObjectList { - entity A { - key id : UUID; - } + entity A as projection on protocols.A; +} + +@protocol: { graphql } +service AnnotatedWithAtProtocolObjectWithKey { + entity A as projection on protocols.A; +} + +@protocol: { graphql: 'dummy' } +service AnnotatedWithAtProtocolObjectWithKeyAndValue { + entity A as projection on protocols.A; } diff --git a/test/schemas/annotations.gql b/test/schemas/annotations.gql index 9918197c..690bd976 100644 --- a/test/schemas/annotations.gql +++ b/test/schemas/annotations.gql @@ -66,6 +66,74 @@ type AnnotatedWithAtProtocolObjectList_input { A: AnnotatedWithAtProtocolObjectList_A_input } +type AnnotatedWithAtProtocolObjectWithKey { + A(filter: [AnnotatedWithAtProtocolObjectWithKey_A_filter], orderBy: [AnnotatedWithAtProtocolObjectWithKey_A_orderBy], skip: Int, top: Int): AnnotatedWithAtProtocolObjectWithKey_A_connection +} + +type AnnotatedWithAtProtocolObjectWithKeyAndValue { + A(filter: [AnnotatedWithAtProtocolObjectWithKeyAndValue_A_filter], orderBy: [AnnotatedWithAtProtocolObjectWithKeyAndValue_A_orderBy], skip: Int, top: Int): AnnotatedWithAtProtocolObjectWithKeyAndValue_A_connection +} + +type AnnotatedWithAtProtocolObjectWithKeyAndValue_A { + id: ID +} + +input AnnotatedWithAtProtocolObjectWithKeyAndValue_A_C { + id: ID +} + +type AnnotatedWithAtProtocolObjectWithKeyAndValue_A_connection { + nodes: [AnnotatedWithAtProtocolObjectWithKeyAndValue_A] + totalCount: Int +} + +input AnnotatedWithAtProtocolObjectWithKeyAndValue_A_filter { + id: [ID_filter] +} + +type AnnotatedWithAtProtocolObjectWithKeyAndValue_A_input { + create(input: [AnnotatedWithAtProtocolObjectWithKeyAndValue_A_C]!): [AnnotatedWithAtProtocolObjectWithKeyAndValue_A] + delete(filter: [AnnotatedWithAtProtocolObjectWithKeyAndValue_A_filter]!): Int +} + +input AnnotatedWithAtProtocolObjectWithKeyAndValue_A_orderBy { + id: SortDirection +} + +type AnnotatedWithAtProtocolObjectWithKeyAndValue_input { + A: AnnotatedWithAtProtocolObjectWithKeyAndValue_A_input +} + +type AnnotatedWithAtProtocolObjectWithKey_A { + id: ID +} + +input AnnotatedWithAtProtocolObjectWithKey_A_C { + id: ID +} + +type AnnotatedWithAtProtocolObjectWithKey_A_connection { + nodes: [AnnotatedWithAtProtocolObjectWithKey_A] + totalCount: Int +} + +input AnnotatedWithAtProtocolObjectWithKey_A_filter { + id: [ID_filter] +} + +type AnnotatedWithAtProtocolObjectWithKey_A_input { + create(input: [AnnotatedWithAtProtocolObjectWithKey_A_C]!): [AnnotatedWithAtProtocolObjectWithKey_A] + delete(filter: [AnnotatedWithAtProtocolObjectWithKey_A_filter]!): Int +} + +input AnnotatedWithAtProtocolObjectWithKey_A_orderBy { + id: SortDirection +} + +type AnnotatedWithAtProtocolObjectWithKey_input { + A: AnnotatedWithAtProtocolObjectWithKey_A_input +} + type AnnotatedWithAtProtocolString { A(filter: [AnnotatedWithAtProtocolString_A_filter], orderBy: [AnnotatedWithAtProtocolString_A_orderBy], skip: Int, top: Int): AnnotatedWithAtProtocolString_A_connection } @@ -147,6 +215,8 @@ input ID_filter { type Mutation { AnnotatedWithAtGraphQL: AnnotatedWithAtGraphQL_input AnnotatedWithAtProtocolObjectList: AnnotatedWithAtProtocolObjectList_input + AnnotatedWithAtProtocolObjectWithKey: AnnotatedWithAtProtocolObjectWithKey_input + AnnotatedWithAtProtocolObjectWithKeyAndValue: AnnotatedWithAtProtocolObjectWithKeyAndValue_input AnnotatedWithAtProtocolString: AnnotatedWithAtProtocolString_input AnnotatedWithAtProtocolStringList: AnnotatedWithAtProtocolStringList_input } @@ -154,6 +224,8 @@ type Mutation { type Query { AnnotatedWithAtGraphQL: AnnotatedWithAtGraphQL AnnotatedWithAtProtocolObjectList: AnnotatedWithAtProtocolObjectList + AnnotatedWithAtProtocolObjectWithKey: AnnotatedWithAtProtocolObjectWithKey + AnnotatedWithAtProtocolObjectWithKeyAndValue: AnnotatedWithAtProtocolObjectWithKeyAndValue AnnotatedWithAtProtocolString: AnnotatedWithAtProtocolString AnnotatedWithAtProtocolStringList: AnnotatedWithAtProtocolStringList } diff --git a/test/tests/annotations.test.js b/test/tests/annotations.test.js index 629ef23f..e3ec8ad5 100644 --- a/test/tests/annotations.test.js +++ b/test/tests/annotations.test.js @@ -123,5 +123,37 @@ describe('graphql - annotations', () => { const response = await POST(path, { query }) expect(response.data).not.toHaveProperty('errors') }) + + test('service annotated with "@protocol: { graphql }" is served at configured path', async () => { + const query = gql` + { + AnnotatedWithAtProtocolObjectWithKey { + A { + nodes { + id + } + } + } + } + ` + const response = await POST(path, { query }) + expect(response.data).not.toHaveProperty('errors') + }) + + test('service annotated with "@protocol: { graphql: \'dummy\' }" is served at configured path', async () => { + const query = gql` + { + AnnotatedWithAtProtocolObjectWithKeyAndValue { + A { + nodes { + id + } + } + } + } + ` + const response = await POST(path, { query }) + expect(response.data).not.toHaveProperty('errors') + }) }) })