From 7d3e3006afc7bdfbd08c3e5886f94f4da25cb4c1 Mon Sep 17 00:00:00 2001 From: Arda TANRIKULU Date: Mon, 21 Jun 2021 18:57:30 +0300 Subject: [PATCH] Huge cleanup before major release (breaking) (#3081) * Breaking cleanup before major release * Fix TS * More chore(deps): update typescript-eslint monorepo to v4.28.0 (#3093) Co-authored-by: Renovate Bot fix(deps): update babel monorepo to v7.14.7 (#3094) Co-authored-by: Renovate Bot chore(deps): update jest monorepo to v27.0.5 (#3095) Co-authored-by: Renovate Bot chore(deps): update dependency @vue/compiler-sfc to v3.1.2 (#3097) Co-authored-by: Renovate Bot fix(deps): update dependency the-guild-components to v1.4.0 (#3098) Co-authored-by: Renovate Bot chore(deps): update dependency @types/node to v14.17.4 (#3101) Co-authored-by: Renovate Bot Some other refactor (#3096) * Fix typing errors in tests * ReplaObject.keys/values and forEach with for..in and for..of loops * Improve Type Checking * Update CI workflow * Fix linting * Fix TS errors for v14 * More * More * Allow incremental builds * Finish * Fix Type Check * Disable sourcemaps * Fix type issues Better changelog Respect and in and more changes chore(CI): Ability to release canaries on demand Make AggregateError spec compliant --- .changeset/blue-phones-kick.md | 8 + .changeset/brave-rats-march.md | 7 + .changeset/cuddly-horses-prove.md | 7 + .changeset/forty-ducks-drum.md | 10 + .changeset/heavy-vans-whisper.md | 8 + .changeset/long-rings-happen.md | 7 + .changeset/mean-news-return.md | 15 + .changeset/new-balloons-reply.md | 8 + .changeset/ninety-shirts-crash.md | 7 + .changeset/pink-zoos-fold.md | 5 + .changeset/polite-yaks-drive.md | 6 + .changeset/silent-comics-tell.md | 8 + .changeset/smooth-tips-know.md | 8 + .changeset/twelve-suns-run.md | 5 + .changeset/wicked-tables-beg.md | 5 + .changeset/young-beans-fail.md | 10 + .eslintrc.json | 3 +- .github/workflows/canary.yml | 11 +- .github/workflows/tests.yml | 11 +- package.json | 15 +- .../src/batchDelegateToSchema.ts | 2 +- .../tests/basic.example.test.ts | 2 +- .../tests/typeMerging.example.test.ts | 2 +- .../tests/withTransforms.test.ts | 2 +- .../src/createBatchingExecutor.ts | 18 +- .../batch-execute/src/getBatchingExecutor.ts | 2 +- .../batch-execute/src/mergeExecutionParams.ts | 13 +- packages/batch-execute/src/prefix.ts | 2 +- packages/batch-execute/src/splitResult.ts | 35 +- packages/delegate/package.json | 1 - packages/delegate/src/Subschema.ts | 2 - packages/delegate/src/Transformer.ts | 4 +- packages/delegate/src/createRequest.ts | 18 +- packages/delegate/src/delegateToSchema.ts | 60 +- packages/delegate/src/externalObjects.ts | 26 +- .../delegate/src/getFieldsNotInSubschema.ts | 12 +- packages/delegate/src/mergeFields.ts | 39 +- packages/delegate/src/resolveExternalValue.ts | 9 +- packages/delegate/src/subschemaConfig.ts | 6 +- .../src/transforms/AddArgumentsAsVariables.ts | 9 +- .../src/transforms/AddSelectionSets.ts | 18 +- .../transforms/CheckResultAndHandleErrors.ts | 27 +- .../src/transforms/ExpandAbstractTypes.ts | 44 +- .../delegate/src/transforms/FilterToSchema.ts | 12 +- .../src/transforms/VisitSelectionSets.ts | 21 +- packages/delegate/src/types.ts | 2 +- .../delegate/tests/delegateToSchema.test..ts | 6 +- packages/delegate/tests/errors.test.ts | 13 +- packages/graphql-tag-pluck/package.json | 6 +- packages/graphql-tools/package.json | 23 - packages/graphql-tools/src/index.ts | 28 +- packages/links/src/linkToExecutor.ts | 35 +- packages/links/src/linkToSubscriber.ts | 3 +- packages/links/tests/upload.test.ts | 2 +- .../load-files/tests/file-scanner.spec.ts | 12 +- .../load-files/tests/test-assets/11/1.spec.ts | 4 +- .../load-files/tests/test-assets/11/2.test.ts | 2 +- packages/load/src/filter-document-kind.ts | 10 +- packages/load/src/load-typedefs.ts | 4 +- packages/load/src/load-typedefs/load-file.ts | 11 +- packages/load/src/utils/queue.ts | 4 +- .../documents/documents-from-glob.spec.ts | 10 +- .../tests/loaders/schema/integration.spec.ts | 6 +- .../loaders/schema/schema-from-export.spec.ts | 2 +- .../schema/schema-from-typedefs.spec.ts | 24 +- packages/loaders/apollo-engine/package.json | 3 +- packages/loaders/apollo-engine/src/index.ts | 7 +- packages/loaders/code-file/src/index.ts | 17 +- .../tests/load-from-code-file.spec.ts | 35 +- packages/loaders/prisma/package.json | 1 - .../prisma/src/prisma-yml/Environment.ts | 8 +- .../prisma/src/prisma-yml/PrismaDefinition.ts | 4 +- .../prisma/src/prisma-yml/Variables.ts | 71 +- .../prisma/src/prisma-yml/constants.ts | 4 +- packages/loaders/url/src/index.ts | 113 +- packages/loaders/url/tests/url-loader.spec.ts | 35 +- packages/merge/src/merge-resolvers.ts | 44 +- packages/merge/src/merge-schemas.ts | 32 +- .../merge/src/typedefs-mergers/comments.ts | 17 +- .../src/typedefs-mergers/merge-typedefs.ts | 74 +- .../extract-extensions-from-schema.spec.ts | 70 +- packages/merge/tests/merge-nodes.spec.ts | 44 +- packages/merge/tests/merge-schemas.spec.ts | 50 +- packages/merge/tests/merge-typedefs.spec.ts | 8 - packages/mock/package.json | 1 + packages/mock/src/MockStore.ts | 8 +- packages/mock/src/mockServer.ts | 14 +- packages/mock/src/utils.ts | 8 +- packages/mock/tests/addMocksToSchema.spec.ts | 8 +- .../mock/tests/mocking-compatibility.spec.ts | 162 +- packages/mock/tests/store.spec.ts | 2 +- packages/node-require/src/index.ts | 4 +- .../tests/relay-optimizer.spec.ts | 4 +- .../src/resolvers-composition.ts | 66 +- .../tests/resolvers-composition.spec.ts | 11 +- packages/schema/package.json | 1 + .../schema/src/addCatchUndefinedToSchema.ts | 25 - .../schema/src/addErrorLoggingToSchema.ts | 19 - packages/schema/src/addResolversToSchema.ts | 348 ++-- packages/schema/src/addSchemaLevelResolver.ts | 70 - .../schema/src/attachDirectiveResolvers.ts | 49 - .../src/buildSchemaFromTypeDefinitions.ts | 48 - packages/schema/src/concatenateTypeDefs.ts | 26 - packages/schema/src/decorateWithLogger.ts | 50 - .../src/extendResolversFromInterfaces.ts | 33 +- packages/schema/src/extensionDefinitions.ts | 43 - packages/schema/src/index.ts | 8 - packages/schema/src/makeExecutableSchema.ts | 60 +- packages/schema/src/types.ts | 33 +- packages/schema/tests/Logger.ts | 36 - .../schema/tests/extensionExtraction.test.ts | 65 - packages/schema/tests/logger.test.ts | 409 ----- packages/schema/tests/resolution.test.ts | 148 -- packages/schema/tests/schemaGenerator.test.ts | 560 +----- .../stitch/src/createMergedTypeResolver.ts | 4 +- packages/stitch/src/definitions.ts | 4 +- packages/stitch/src/mergeCandidates.ts | 116 +- packages/stitch/src/mergeValidations.ts | 35 +- packages/stitch/src/stitchSchemas.ts | 117 +- packages/stitch/src/stitchingInfo.ts | 80 +- .../src/subschemaConfigTransforms/index.ts | 2 +- .../isolateComputedFieldsTransformer.ts | 75 +- .../splitMergedTypeEntryPointsTransformer.ts | 13 +- packages/stitch/src/typeCandidates.ts | 59 +- packages/stitch/src/typeFromAST.ts | 4 +- packages/stitch/src/types.ts | 4 +- .../tests/alternateStitchSchemas.test.ts | 34 +- packages/stitch/tests/dataloader.test.ts | 2 +- packages/stitch/tests/example.test.ts | 18 +- .../stitch/tests/extendedInterface.test.ts | 8 +- packages/stitch/tests/fixtures/schemas.ts | 24 +- .../isolateComputedFieldsTransformer.test.ts | 14 +- .../stitch/tests/mergeAbstractTypes.test.ts | 4 +- .../stitch/tests/mergeComputedFields.test.ts | 8 +- packages/stitch/tests/mergeConflicts.test.ts | 6 +- .../stitch/tests/mergeDefinitions.test.ts | 78 +- packages/stitch/tests/mergeInterfaces.test.ts | 8 +- .../tests/mergeMultipleEntryPoints.test.ts | 6 +- packages/stitch/tests/selectionSets.test.ts | 12 +- ...itMergedTypeEntryPointsTransformer.test.ts | 6 +- packages/stitch/tests/stitchSchemas.test.ts | 74 +- .../tests/stitchingFromSubschemas.test.ts | 148 -- packages/stitch/tests/typeMerging.test.ts | 20 +- .../tests/typeMergingWithDirectives.test.ts | 4 +- .../tests/typeMergingWithExtensions.test.ts | 8 +- .../tests/typeMergingWithInterfaces.test.ts | 10 +- .../src/getSourcePaths.ts | 12 +- .../src/parseMergeArgsExpr.ts | 31 +- .../src/pathsFromSelectionSet.ts | 12 +- .../stitching-directives/src/properties.ts | 12 +- .../src/stitchingDirectivesTransformer.ts | 164 +- .../src/stitchingDirectivesValidator.ts | 8 +- .../stitchingDirectivesTransformer.test.ts | 63 +- packages/utils/package.json | 6 +- packages/utils/src/AggregateError.ts | 16 + packages/utils/src/Interfaces.ts | 101 +- packages/utils/src/SchemaDirectiveVisitor.ts | 324 ---- packages/utils/src/SchemaVisitor.ts | 119 -- packages/utils/src/addTypes.ts | 42 +- packages/utils/src/astFromValueUntyped.ts | 9 +- .../utils/src/build-operation-for-field.ts | 13 +- packages/utils/src/debug-log.ts | 6 - packages/utils/src/executor.ts | 51 +- packages/utils/src/fields.ts | 25 +- packages/utils/src/filterSchema.ts | 14 +- packages/utils/src/fix-windows-path.ts | 1 - packages/utils/src/flatten-array.ts | 2 - packages/utils/src/forEachDefaultValue.ts | 16 +- packages/utils/src/forEachField.ts | 8 +- packages/utils/src/get-directives.ts | 10 +- packages/utils/src/getArgumentValues.ts | 2 +- packages/utils/src/getResolversFromSchema.ts | 18 +- packages/utils/src/heal.ts | 25 +- packages/utils/src/index.ts | 8 +- packages/utils/src/inspect.ts | 113 -- packages/utils/src/mapSchema.ts | 135 +- packages/utils/src/mergeDeep.ts | 6 +- .../utils/src/observableToAsyncIterable.ts | 4 +- .../utils/src/print-schema-with-directives.ts | 36 +- packages/utils/src/prune.ts | 51 +- packages/utils/src/rewire.ts | 32 +- packages/utils/src/stub.ts | 12 +- packages/utils/src/toConfig.ts | 51 - packages/utils/src/transformInputValue.ts | 4 +- packages/utils/src/validate-documents.ts | 75 +- packages/utils/src/visitResult.ts | 28 +- packages/utils/src/visitSchema.ts | 320 ---- .../build-operation-node-for-field.spec.ts | 28 +- packages/utils/tests/directives.test.ts | 1601 ----------------- packages/utils/tests/get-directives.spec.ts | 39 +- packages/utils/tests/mapSchema.test.ts | 4 +- .../print-schema-with-directives.spec.ts | 2 +- packages/utils/tests/relocatedError.test.ts | 4 +- packages/utils/tests/schemaTransforms.test.ts | 40 +- .../utils/tests/validate-documents.spec.ts | 7 +- packages/webpack-loader/src/index.ts | 4 +- .../wrap/src/generateProxyingResolvers.ts | 10 +- packages/wrap/src/index.ts | 1 - packages/wrap/src/introspect.ts | 17 +- .../wrap/src/makeRemoteExecutableSchema.ts | 35 - packages/wrap/src/transforms/HoistField.ts | 4 +- packages/wrap/src/transforms/MapLeafValues.ts | 13 +- .../transforms/TransformCompositeFields.ts | 20 +- .../transforms/TransformInputObjectFields.ts | 64 +- packages/wrap/src/transforms/WrapFields.ts | 34 +- packages/wrap/src/types.ts | 16 +- packages/wrap/src/wrapSchema.ts | 12 +- packages/wrap/tests/fixtures/schemas.ts | 34 +- .../tests/fragmentsAreNotDuplicated.test.ts | 4 +- .../tests/makeRemoteExecutableSchema.test.ts | 28 +- packages/wrap/tests/requests.test.ts | 8 +- .../transformFilterInputObjectFields.test.ts | 4 +- ...ansformFilterObjectFieldDirectives.test.ts | 10 +- .../wrap/tests/transformMapLeafValues.test.ts | 4 +- ...sformRemoveObjectFieldDeprecations.test.ts | 24 +- ...ansformRemoveObjectFieldDirectives.test.ts | 64 +- ...mRemoveObjectFieldsWithDeprecation.test.ts | 8 +- ...ormRemoveObjectFieldsWithDirective.test.ts | 32 +- .../transformRenameInputObjectFields.test.ts | 8 +- patches/bob-the-bundler+1.4.1.patch | 22 + scripts/canary-release.js | 9 +- tsconfig.build.es5.json | 5 +- tsconfig.build.json | 5 + tsconfig.json | 6 +- website/docs/directive-resolvers.md | 178 -- website/docs/generate-schema.md | 8 - website/docs/legacy-schema-directives.md | 714 -------- website/docs/remote-schemas.md | 20 - website/docs/schema-directives.md | 14 +- website/package.json | 2 +- website/sidebars.js | 1 - yarn.lock | 621 ++++--- 232 files changed, 2621 insertions(+), 7870 deletions(-) create mode 100644 .changeset/blue-phones-kick.md create mode 100644 .changeset/brave-rats-march.md create mode 100644 .changeset/cuddly-horses-prove.md create mode 100644 .changeset/forty-ducks-drum.md create mode 100644 .changeset/heavy-vans-whisper.md create mode 100644 .changeset/long-rings-happen.md create mode 100644 .changeset/mean-news-return.md create mode 100644 .changeset/new-balloons-reply.md create mode 100644 .changeset/ninety-shirts-crash.md create mode 100644 .changeset/pink-zoos-fold.md create mode 100644 .changeset/polite-yaks-drive.md create mode 100644 .changeset/silent-comics-tell.md create mode 100644 .changeset/smooth-tips-know.md create mode 100644 .changeset/twelve-suns-run.md create mode 100644 .changeset/wicked-tables-beg.md create mode 100644 .changeset/young-beans-fail.md delete mode 100644 packages/schema/src/addCatchUndefinedToSchema.ts delete mode 100644 packages/schema/src/addErrorLoggingToSchema.ts delete mode 100644 packages/schema/src/addSchemaLevelResolver.ts delete mode 100644 packages/schema/src/attachDirectiveResolvers.ts delete mode 100644 packages/schema/src/buildSchemaFromTypeDefinitions.ts delete mode 100644 packages/schema/src/concatenateTypeDefs.ts delete mode 100644 packages/schema/src/decorateWithLogger.ts delete mode 100644 packages/schema/src/extensionDefinitions.ts delete mode 100644 packages/schema/tests/Logger.ts delete mode 100644 packages/schema/tests/extensionExtraction.test.ts delete mode 100644 packages/schema/tests/logger.test.ts delete mode 100644 packages/schema/tests/resolution.test.ts delete mode 100644 packages/stitch/tests/stitchingFromSubschemas.test.ts create mode 100644 packages/utils/src/AggregateError.ts delete mode 100644 packages/utils/src/SchemaDirectiveVisitor.ts delete mode 100644 packages/utils/src/SchemaVisitor.ts delete mode 100644 packages/utils/src/debug-log.ts delete mode 100644 packages/utils/src/fix-windows-path.ts delete mode 100644 packages/utils/src/flatten-array.ts delete mode 100644 packages/utils/src/inspect.ts delete mode 100644 packages/utils/src/toConfig.ts delete mode 100644 packages/utils/src/visitSchema.ts delete mode 100644 packages/utils/tests/directives.test.ts delete mode 100644 packages/wrap/src/makeRemoteExecutableSchema.ts create mode 100644 patches/bob-the-bundler+1.4.1.patch delete mode 100644 website/docs/directive-resolvers.md delete mode 100644 website/docs/legacy-schema-directives.md diff --git a/.changeset/blue-phones-kick.md b/.changeset/blue-phones-kick.md new file mode 100644 index 00000000000..6b45b64a001 --- /dev/null +++ b/.changeset/blue-phones-kick.md @@ -0,0 +1,8 @@ +--- +'@graphql-tools/delegate': major +--- + +BREAKING CHANGE +- Remove `rootValue` from subschemaConfig +- - Pass it through `ExecutionParams` or delegation options +- Do not pass `info.rootValue` if `rootValue` is falsy diff --git a/.changeset/brave-rats-march.md b/.changeset/brave-rats-march.md new file mode 100644 index 00000000000..52e68849001 --- /dev/null +++ b/.changeset/brave-rats-march.md @@ -0,0 +1,7 @@ +--- +'@graphql-tools/wrap': major +--- + +BREAKING CHANGE +- `makeRemoteExecutableSchema` has been removed. +- - You can use [`wrapSchema`](https://www.graphql-tools.com/docs/remote-schemas#creating-an-executor) instead diff --git a/.changeset/cuddly-horses-prove.md b/.changeset/cuddly-horses-prove.md new file mode 100644 index 00000000000..25c2e5aafc0 --- /dev/null +++ b/.changeset/cuddly-horses-prove.md @@ -0,0 +1,7 @@ +--- +'@graphql-tools/utils': major +--- + +BREAKING CHANGE +- Remove `fieldToFieldConfig`, `argsToFieldConfigArgument` and `argumentToArgumentConfig` +- - You can use `.toConfig` method instead for each. diff --git a/.changeset/forty-ducks-drum.md b/.changeset/forty-ducks-drum.md new file mode 100644 index 00000000000..74cdff2e1e4 --- /dev/null +++ b/.changeset/forty-ducks-drum.md @@ -0,0 +1,10 @@ +--- +'@graphql-tools/schema': major +'@graphql-tools/stitch': major +'@graphql-tools/utils': major +--- + +BREAKING CHANGE +- Legacy Schema Directives and Directive Resolvers have been removed +- - You can check the new method for both; +- - - https://www.graphql-tools.com/docs/schema-directives diff --git a/.changeset/heavy-vans-whisper.md b/.changeset/heavy-vans-whisper.md new file mode 100644 index 00000000000..26e9ea892a4 --- /dev/null +++ b/.changeset/heavy-vans-whisper.md @@ -0,0 +1,8 @@ +--- +'@graphql-tools/schema': major +--- + +BREAKING CHANGE +- Remove `logger` and `addErrorLoggingToSchema` +- - You can implement logging and debugging mechanism outside the resolvers using some kind of plugin system based library like [Envelop](https://www.envelop.dev/docs/core#uselogger) + diff --git a/.changeset/long-rings-happen.md b/.changeset/long-rings-happen.md new file mode 100644 index 00000000000..864360d945a --- /dev/null +++ b/.changeset/long-rings-happen.md @@ -0,0 +1,7 @@ +--- +'graphql-tools': major +--- + +BREAKING CHANGE +- Now it only exports `makeExecutableSchema` from `@graphql-tools/schema` +- Please migrate to scoped packages(`@graphql-tools/*`) because this npm package will no longer get updated diff --git a/.changeset/mean-news-return.md b/.changeset/mean-news-return.md new file mode 100644 index 00000000000..7ccdc2605a0 --- /dev/null +++ b/.changeset/mean-news-return.md @@ -0,0 +1,15 @@ +--- +'@graphql-tools/delegate': major +'@graphql-tools/apollo-engine-loader': major +'@graphql-tools/utils': major +'@graphql-tools/wrap': major +--- + +BREAKING CHANGE +- Now it uses the native [`AggregateError`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/AggregateError) implementation. The major difference is the individual errors are kept under `errors` property instead of the object itself with `Symbol.iterator`. +```js +// From; +for (const error of aggregateError) +// To; +for (const error of aggregateError.errors) +``` diff --git a/.changeset/new-balloons-reply.md b/.changeset/new-balloons-reply.md new file mode 100644 index 00000000000..9fc4b50e2f0 --- /dev/null +++ b/.changeset/new-balloons-reply.md @@ -0,0 +1,8 @@ +--- +'@graphql-tools/schema': major +--- + +BREAKING CHANGE +- Remove schema level resolvers feature and `addSchemaLevelResolver` +- - You can wrap your resolvers by using [Resolvers Composition](https://www.graphql-tools.com/docs/resolvers-composition) + diff --git a/.changeset/ninety-shirts-crash.md b/.changeset/ninety-shirts-crash.md new file mode 100644 index 00000000000..f919a768c63 --- /dev/null +++ b/.changeset/ninety-shirts-crash.md @@ -0,0 +1,7 @@ +--- +'@graphql-tools/utils': major +--- + +BREAKING CHANGE +- No longer exports `debugLog` but uses `console.log` directly only if `DEBUG` is available under `process.env` + diff --git a/.changeset/pink-zoos-fold.md b/.changeset/pink-zoos-fold.md new file mode 100644 index 00000000000..cd3565a1f8b --- /dev/null +++ b/.changeset/pink-zoos-fold.md @@ -0,0 +1,5 @@ +--- +'@graphql-tools/links': minor +--- + +feat(links): Respect operationName diff --git a/.changeset/polite-yaks-drive.md b/.changeset/polite-yaks-drive.md new file mode 100644 index 00000000000..96015ab6095 --- /dev/null +++ b/.changeset/polite-yaks-drive.md @@ -0,0 +1,6 @@ +--- +'@graphql-tools/utils': major +--- + +BREAKING CHANGE +- No longer applies `camelCase` naming convention in `buildOperationNodeForField` diff --git a/.changeset/silent-comics-tell.md b/.changeset/silent-comics-tell.md new file mode 100644 index 00000000000..31e77804ba1 --- /dev/null +++ b/.changeset/silent-comics-tell.md @@ -0,0 +1,8 @@ +--- +'@graphql-tools/schema': major +--- + +BREAKING CHANGE +- No longer exports `buildSchemaFromTypeDefinitions`, use `buildSchema` from `graphql-js` instead +- Remove `allowUndefinedResolve` option in `makeExecutableSchema` because GraphQL Schema itself does this checking + diff --git a/.changeset/smooth-tips-know.md b/.changeset/smooth-tips-know.md new file mode 100644 index 00000000000..326bfe7c745 --- /dev/null +++ b/.changeset/smooth-tips-know.md @@ -0,0 +1,8 @@ +--- +'@graphql-tools/utils': major +--- + +BREAKING CHANGE +- No longer exports `SchemaVisitor`, `visitSchema` and `VisitSchemaKind` +- - Use [`mapSchema`](https://www.graphql-tools.com/docs/schema-directives/#full-mapschema-api) instead + diff --git a/.changeset/twelve-suns-run.md b/.changeset/twelve-suns-run.md new file mode 100644 index 00000000000..7c816992aa7 --- /dev/null +++ b/.changeset/twelve-suns-run.md @@ -0,0 +1,5 @@ +--- +'@graphql-tools/schema': patch +--- + +enhance(schema): use merge package to handle typeDefs and resolvers merging diff --git a/.changeset/wicked-tables-beg.md b/.changeset/wicked-tables-beg.md new file mode 100644 index 00000000000..3d44830de3a --- /dev/null +++ b/.changeset/wicked-tables-beg.md @@ -0,0 +1,5 @@ +--- +'@graphql-tools/utils': minor +--- + +feat(utils): Respect operationName and rootValue in ExecutionParams diff --git a/.changeset/young-beans-fail.md b/.changeset/young-beans-fail.md new file mode 100644 index 00000000000..ce34942cdba --- /dev/null +++ b/.changeset/young-beans-fail.md @@ -0,0 +1,10 @@ +--- +'@graphql-tools/url-loader': major +--- + +BREAKING CHANGE +- No more accept arrays or functions for `headers` + +NEW FEATURES +- Respect `operationName` and `extensions` +- Ability to get headers from `extensions.headers` diff --git a/.eslintrc.json b/.eslintrc.json index 12787662740..fd8fff382e9 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -61,7 +61,8 @@ "packages/load/tests/loaders/schema", "website", "scripts", - "packages/loaders/code-file/tests/test-files" + "packages/loaders/code-file/tests/test-files", + "packages/loaders/git/tests/test-files" ], "globals":{ "BigInt": true diff --git a/.github/workflows/canary.yml b/.github/workflows/canary.yml index 2dc25753ff5..45c8b7014bb 100644 --- a/.github/workflows/canary.yml +++ b/.github/workflows/canary.yml @@ -4,12 +4,17 @@ on: pull_request: branches: - master - + workflow_dispatch: + inputs: + confirm: + description: 'Are you sure?' + required: true + default: 'yes' jobs: publish-canary: name: Publish Canary runs-on: ubuntu-latest - if: github.event.pull_request.head.repo.full_name == github.repository + if: github.event.pull_request.head.repo.full_name == github.repository || github.event.inputs.confirm == 'yes' steps: - name: Checkout Master uses: actions/checkout@v2 @@ -44,6 +49,8 @@ jobs: npm-token: ${{ secrets.NODE_AUTH_TOKEN }} npm-script: 'yarn release:canary' changesets: true + env: + ON_DEMAND: ${{github.event.inputs.confirm}} - name: Publish a message if: steps.canary.outputs.released uses: 'kamilkisiela/pr-comment@master' diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index bcd0e515c36..8e34139248e 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -31,7 +31,7 @@ jobs: - name: Lint run: yarn lint build: - name: Build on ${{matrix.os}} GraphQL v${{matrix.graphql_version}} + name: Type Check on GraphQL v${{matrix.graphql_version}} runs-on: ubuntu-latest strategy: matrix: @@ -55,14 +55,13 @@ jobs: - name: Install Dependencies using Yarn run: yarn install --ignore-engines && git checkout yarn.lock - name: Build - run: yarn ts:transpile + run: yarn ts:check test: - name: Test on ${{matrix.os}}, Node ${{matrix.node_version}} and GraphQL v${{matrix.graphql_version}} - runs-on: ${{matrix.os}} + name: Unit Test on Node ${{matrix.node_version}} and GraphQL v${{matrix.graphql_version}} + runs-on: ubuntu-latest strategy: matrix: - os: [ubuntu-latest] # remove windows to speed up the tests - node_version: [10, 16] + node_version: [12, 16] graphql_version: [14, 15] steps: - name: Checkout Master diff --git a/package.json b/package.json index 724cdd021d0..be808c61dfd 100644 --- a/package.json +++ b/package.json @@ -7,7 +7,8 @@ "postinstall": "patch-package && husky install", "predeploy:website": "yarn build:api-docs", "deploy:website": "cd website && yarn deploy", - "ts:transpile": "concurrently \"tsc --project tsconfig.build.json\" \"tsc --project tsconfig.build.es5.json\"", + "ts:check": "tsc --noEmit --incremental", + "ts:transpile": "concurrently \"tsc --project tsconfig.build.json --incremental\" \"tsc --project tsconfig.build.es5.json --incremental\"", "clean-dist": "rimraf \"packages/**/dist\" && rimraf \"packages/**/dist-es5\" && rimraf \".bob\"", "build": "yarn ts:transpile && bob build", "build:api-docs": "node scripts/build-api-docs.js", @@ -45,14 +46,14 @@ "bob-the-bundler": "1.4.1", "@babel/core": "7.14.6", "@babel/plugin-proposal-class-properties": "7.14.5", - "@babel/preset-env": "7.14.5", + "@babel/preset-env": "7.14.7", "@babel/preset-typescript": "7.14.5", "@changesets/cli": "2.16.0", "@types/jest": "26.0.23", - "@types/node": "14.17.3", - "@typescript-eslint/eslint-plugin": "4.27.0", - "@typescript-eslint/parser": "4.27.0", - "babel-jest": "27.0.2", + "@types/node": "14.17.4", + "@typescript-eslint/eslint-plugin": "4.28.0", + "@typescript-eslint/parser": "4.28.0", + "babel-jest": "27.0.5", "concurrently": "6.2.0", "eslint": "7.29.0", "eslint-config-prettier": "8.3.0", @@ -65,7 +66,7 @@ "graphql-helix": "1.6.1", "graphql-subscriptions": "1.2.1", "husky": "6.0.0", - "jest": "27.0.4", + "jest": "27.0.5", "lint-staged": "11.0.0", "nock": "13.1.0", "patch-package": "6.4.7", diff --git a/packages/batch-delegate/src/batchDelegateToSchema.ts b/packages/batch-delegate/src/batchDelegateToSchema.ts index ee5057522b8..ba12dd310fb 100644 --- a/packages/batch-delegate/src/batchDelegateToSchema.ts +++ b/packages/batch-delegate/src/batchDelegateToSchema.ts @@ -2,7 +2,7 @@ import { BatchDelegateOptions } from './types'; import { getLoader } from './getLoader'; -export function batchDelegateToSchema(options: BatchDelegateOptions): any { +export function batchDelegateToSchema(options: BatchDelegateOptions): any { const key = options.key; if (key == null) { return null; diff --git a/packages/batch-delegate/tests/basic.example.test.ts b/packages/batch-delegate/tests/basic.example.test.ts index b05d07039b3..9f6ed75d6f3 100644 --- a/packages/batch-delegate/tests/basic.example.test.ts +++ b/packages/batch-delegate/tests/basic.example.test.ts @@ -89,7 +89,7 @@ describe('batch delegation within basic stitching example', () => { expect(numCalls).toEqual(1); expect(result.errors).toBeUndefined(); - expect(result.data!.trendingChirps[0].chirpedAtUser.email).not.toBe(null); + expect(result.data!['trendingChirps'][0].chirpedAtUser.email).not.toBe(null); }); test('works with key arrays', async () => { diff --git a/packages/batch-delegate/tests/typeMerging.example.test.ts b/packages/batch-delegate/tests/typeMerging.example.test.ts index 171afd9bdf0..e8e000341bc 100644 --- a/packages/batch-delegate/tests/typeMerging.example.test.ts +++ b/packages/batch-delegate/tests/typeMerging.example.test.ts @@ -84,7 +84,7 @@ describe('merging using type merging', () => { Query: { mostStockedProduct: () => inventory.find(i => i.upc === '3'), _products: (_root, { representations }) => { - return representations.map((rep: Record) => ({ ...rep, ...inventory.find(i => i.upc === rep.upc) })); + return representations.map((rep: Record) => ({ ...rep, ...inventory.find(i => i.upc === rep['upc']) })); }, }, }, diff --git a/packages/batch-delegate/tests/withTransforms.test.ts b/packages/batch-delegate/tests/withTransforms.test.ts index a325d146996..66f64086a8e 100644 --- a/packages/batch-delegate/tests/withTransforms.test.ts +++ b/packages/batch-delegate/tests/withTransforms.test.ts @@ -73,7 +73,7 @@ describe('works with complex transforms', () => { ] }), resultTransformer: (results, delegationContext) => { - const userIds = delegationContext.args.userIds; + const userIds = delegationContext.args['userIds']; const booksByUserIds = results.reduce( (acc: any, { userId, books }: { userId: string, books: any[] }) => { acc[userId] = books diff --git a/packages/batch-execute/src/createBatchingExecutor.ts b/packages/batch-execute/src/createBatchingExecutor.ts index 4a92f5ab6ba..7bb8a4201f0 100644 --- a/packages/batch-execute/src/createBatchingExecutor.ts +++ b/packages/batch-execute/src/createBatchingExecutor.ts @@ -51,20 +51,18 @@ function createLoadFn( } } - const executionResults: Array> = []; - execBatches.forEach(execBatch => { + const executionResults: Array> = execBatches.map(execBatch => { const mergedExecutionParams = mergeExecutionParams(execBatch, extensionsReducer); - executionResults.push(new ValueOrPromise(() => executor(mergedExecutionParams))); + return new ValueOrPromise(() => executor(mergedExecutionParams)); }); return ValueOrPromise.all(executionResults) - .then(resultBatches => { - let results: Array = []; - resultBatches.forEach((resultBatch, index) => { - results = [...results, ...splitResult(resultBatch!, execBatches[index].length)]; - }); - return results; - }) + .then(resultBatches => + resultBatches.reduce( + (results, resultBatch, index) => results.concat(splitResult(resultBatch, execBatches[index].length)), + new Array>>() + ) + ) .resolve(); }; } diff --git a/packages/batch-execute/src/getBatchingExecutor.ts b/packages/batch-execute/src/getBatchingExecutor.ts index 8ea01bb7547..6d5e5de32cc 100644 --- a/packages/batch-execute/src/getBatchingExecutor.ts +++ b/packages/batch-execute/src/getBatchingExecutor.ts @@ -5,7 +5,7 @@ import { createBatchingExecutor } from './createBatchingExecutor'; import { memoize2of4 } from './memoize'; export const getBatchingExecutor = memoize2of4(function ( - _context: Record = self ?? window ?? global, + _context: Record, executor: Executor, dataLoaderOptions?: DataLoader.Options | undefined, extensionsReducer?: diff --git a/packages/batch-execute/src/mergeExecutionParams.ts b/packages/batch-execute/src/mergeExecutionParams.ts index 1dd6cbc0375..80e2fce52fc 100644 --- a/packages/batch-execute/src/mergeExecutionParams.ts +++ b/packages/batch-execute/src/mergeExecutionParams.ts @@ -68,22 +68,25 @@ export function mergeExecutionParams( let operation: Maybe; - execs.forEach((executionParams, index) => { + for (const index in execs) { + const executionParams = execs[index]; const prefixedExecutionParams = prefixExecutionParams(createPrefix(index), executionParams); - prefixedExecutionParams.document.definitions.forEach(def => { + for (const def of prefixedExecutionParams.document.definitions) { if (isOperationDefinition(def)) { operation = def.operation; mergedSelections.push(...def.selectionSet.selections); - mergedVariableDefinitions.push(...(def.variableDefinitions ?? [])); + if (def.variableDefinitions) { + mergedVariableDefinitions.push(...def.variableDefinitions); + } } if (isFragmentDefinition(def)) { mergedFragmentDefinitions.push(def); } - }); + } Object.assign(mergedVariables, prefixedExecutionParams.variables); mergedExtensions = extensionsReducer(mergedExtensions, executionParams); - }); + } if (operation == null) { throw new Error('Could not identify operation type. Did the document only include fragment definitions?'); diff --git a/packages/batch-execute/src/prefix.ts b/packages/batch-execute/src/prefix.ts index 4da94fc8e00..97460d521c8 100644 --- a/packages/batch-execute/src/prefix.ts +++ b/packages/batch-execute/src/prefix.ts @@ -1,6 +1,6 @@ // adapted from https://github.com/gatsbyjs/gatsby/blob/master/packages/gatsby-source-graphql/src/batching/merge-queries.js -export function createPrefix(index: number): string { +export function createPrefix(index: string): string { return `graphqlTools${index}_`; } diff --git a/packages/batch-execute/src/splitResult.ts b/packages/batch-execute/src/splitResult.ts index 0cee17f31c6..1626a77186e 100644 --- a/packages/batch-execute/src/splitResult.ts +++ b/packages/batch-execute/src/splitResult.ts @@ -9,60 +9,41 @@ import { parseKey } from './prefix'; /** * Split and transform result of the query produced by the `merge` function */ -export function splitResult(mergedResult: ExecutionResult, numResults: number): Array { +export function splitResult({ data, errors }: ExecutionResult, numResults: number): Array { const splitResults: Array = []; for (let i = 0; i < numResults; i++) { splitResults.push({}); } - const data = mergedResult.data; if (data) { - Object.keys(data).forEach(prefixedKey => { + for (const prefixedKey in data) { const parsedKey = parseKey(prefixedKey); assertSome(parsedKey, "'parsedKey' should not be null."); const { index, originalKey } = parsedKey; const result = splitResults[index]; if (result == null) { - return; + continue; } if (result.data == null) { result.data = { [originalKey]: data[prefixedKey] }; } else { result.data[originalKey] = data[prefixedKey]; } - }); + } } - const errors = mergedResult.errors; if (errors) { - const newErrors: Record> = Object.create(null); - errors.forEach(error => { + for (const error of errors) { if (error.path) { const parsedKey = parseKey(error.path[0] as string); if (parsedKey) { const { index, originalKey } = parsedKey; const newError = relocatedError(error, [originalKey, ...error.path.slice(1)]); - if (!newErrors[index]) { - newErrors[index] = [newError]; - } else { - newErrors[index].push(newError); - } - return; + const errors = (splitResults[index].errors = (splitResults[index].errors || []) as GraphQLError[]); + errors.push(newError); } } - - splitResults.forEach((_splitResult, index) => { - if (!newErrors[index]) { - newErrors[index] = [error]; - } else { - newErrors[index].push(error); - } - }); - }); - - Object.keys(newErrors).forEach(index => { - splitResults[index].errors = newErrors[index]; - }); + } } return splitResults; diff --git a/packages/delegate/package.json b/packages/delegate/package.json index b9dcaa0a76a..663562b2253 100644 --- a/packages/delegate/package.json +++ b/packages/delegate/package.json @@ -35,7 +35,6 @@ "@graphql-tools/batch-execute": "^7.1.2", "@graphql-tools/schema": "^7.1.5", "@graphql-tools/utils": "^7.7.1", - "@ardatan/aggregate-error": "0.0.6", "dataloader": "2.0.0", "tslib": "~2.3.0", "value-or-promise": "1.0.10" diff --git a/packages/delegate/src/Subschema.ts b/packages/delegate/src/Subschema.ts index bcf53286a8e..b16d8fa611c 100644 --- a/packages/delegate/src/Subschema.ts +++ b/packages/delegate/src/Subschema.ts @@ -19,7 +19,6 @@ export class Subschema> { public schema: GraphQLSchema; - public rootValue?: Record; public executor?: Executor; public subscriber?: Subscriber; public batch?: boolean; @@ -34,7 +33,6 @@ export class Subschema> constructor(config: SubschemaConfig) { this.schema = config.schema; - this.rootValue = config.rootValue; this.executor = config.executor; this.subscriber = config.subscriber; this.batch = config.batch; diff --git a/packages/delegate/src/Transformer.ts b/packages/delegate/src/Transformer.ts index d8b0403edb2..376cb24015c 100644 --- a/packages/delegate/src/Transformer.ts +++ b/packages/delegate/src/Transformer.ts @@ -16,7 +16,9 @@ export class Transformer> { constructor(context: DelegationContext, binding: DelegationBinding = defaultDelegationBinding) { this.delegationContext = context; const delegationTransforms: Array = binding(this.delegationContext); - delegationTransforms.forEach(transform => this.addTransform(transform, {})); + for (const transform of delegationTransforms) { + this.addTransform(transform, {}); + } } private addTransform(transform: Transform, context = {}) { diff --git a/packages/delegate/src/createRequest.ts b/packages/delegate/src/createRequest.ts index 73f205e5c3c..f5bde01de0c 100644 --- a/packages/delegate/src/createRequest.ts +++ b/packages/delegate/src/createRequest.ts @@ -10,7 +10,6 @@ import { typeFromAST, NamedTypeNode, GraphQLInputType, - GraphQLArgument, VariableDefinitionNode, SelectionSetNode, DefinitionNode, @@ -107,7 +106,7 @@ export function createRequest({ const variableDefinitionMap = Object.create(null); if (sourceSchema != null && variableDefinitions != null) { - variableDefinitions.forEach(def => { + for (const def of variableDefinitions) { const varName = def.variable.name.value; variableDefinitionMap[varName] = def; const varType = typeFromAST(sourceSchema, def.type as NamedTypeNode) as GraphQLInputType; @@ -115,7 +114,7 @@ export function createRequest({ if (serializedValue !== undefined) { newVariables[varName] = serializedValue; } - }); + } } if (sourceParentType != null && sourceFieldName != null) { @@ -130,7 +129,7 @@ export function createRequest({ const rootfieldNode: FieldNode = { kind: Kind.FIELD, - arguments: Object.keys(argumentNodeMap).map(argName => argumentNodeMap[argName]), + arguments: Object.values(argumentNodeMap), name: { kind: Kind.NAME, value: @@ -152,17 +151,17 @@ export function createRequest({ kind: Kind.OPERATION_DEFINITION, name: operationName, operation: targetOperation, - variableDefinitions: Object.keys(variableDefinitionMap).map(varName => variableDefinitionMap[varName]), + variableDefinitions: Object.values(variableDefinitionMap), selectionSet: { kind: Kind.SELECTION_SET, selections: [rootfieldNode], }, }; - let definitions: Array = [operationDefinition]; + const definitions: Array = [operationDefinition]; if (fragments != null) { - definitions = definitions.concat(Object.keys(fragments).map(fragmentName => fragments[fragmentName])); + definitions.push(...Object.values(fragments)); } const document: DocumentNode = { @@ -173,6 +172,7 @@ export function createRequest({ return { document, variables: newVariables, + operationName: targetOperationName, }; } @@ -184,7 +184,7 @@ function updateArgumentsWithDefaults( variableValues: Record ): void { const sourceField = sourceParentType.getFields()[sourceFieldName]; - sourceField.args.forEach((argument: GraphQLArgument) => { + for (const argument of sourceField.args) { const argName = argument.name; const sourceArgType = argument.type; @@ -202,5 +202,5 @@ function updateArgumentsWithDefaults( ); } } - }); + } } diff --git a/packages/delegate/src/delegateToSchema.ts b/packages/delegate/src/delegateToSchema.ts index 079bd44107f..2632fdae574 100644 --- a/packages/delegate/src/delegateToSchema.ts +++ b/packages/delegate/src/delegateToSchema.ts @@ -14,8 +14,6 @@ import { import { ValueOrPromise } from 'value-or-promise'; -import AggregateError from '@ardatan/aggregate-error'; - import { getBatchingExecutor } from '@graphql-tools/batch-execute'; import { @@ -26,6 +24,7 @@ import { Subscriber, Maybe, assertSome, + AggregateError, } from '@graphql-tools/utils'; import { @@ -40,7 +39,6 @@ import { isSubschemaConfig } from './subschemaConfig'; import { Subschema } from './Subschema'; import { createRequestFromInfo, getDelegatingOperation } from './createRequest'; import { Transformer } from './Transformer'; -import { memoize2 } from './memoize'; export function delegateToSchema, TArgs = any>( options: IDelegateToSchemaOptions @@ -152,18 +150,19 @@ function getDelegationContext({ args = {}, context, info, - rootValue, + rootValue = emptyObject, transforms = [], transformedSchema, - skipTypeMerging, + skipTypeMerging = false, + operationName, }: IDelegateRequestOptions): DelegationContext { let operationDefinition: Maybe; let targetOperation: Maybe; let targetFieldName: string; - skipTypeMerging = skipTypeMerging ?? false; + let targetOperationName: string | undefined; if (operation == null) { - operationDefinition = getOperationAST(request.document, undefined); + operationDefinition = getOperationAST(request.document, request.operationName); assertSome(operationDefinition, 'Could not identify the main operation of the document.'); targetOperation = operationDefinition.operation; } else { @@ -171,12 +170,22 @@ function getDelegationContext({ } if (fieldName == null) { - operationDefinition = operationDefinition ?? getOperationAST(request.document, undefined); + operationDefinition = operationDefinition ?? getOperationAST(request.document, request.operationName); targetFieldName = (operationDefinition?.selectionSet.selections[0] as unknown as FieldDefinitionNode).name.value; } else { targetFieldName = fieldName; } + if (operationName == null) { + if (request.operationName) { + targetOperationName = request.operationName; + } else if (operationDefinition?.name?.value) { + targetOperationName = operationDefinition.name.value; + } + } else { + targetOperationName = operationName; + } + const stitchingInfo: Maybe> = info?.schema.extensions?.['stitchingInfo']; const subschemaOrSubschemaConfig: GraphQLSchema | SubschemaConfig = @@ -189,11 +198,12 @@ function getDelegationContext({ subschemaConfig: subschemaOrSubschemaConfig, targetSchema, operation: targetOperation, + operationName: targetOperationName, fieldName: targetFieldName, args, context, info, - rootValue: rootValue ?? subschemaOrSubschemaConfig?.rootValue ?? info?.rootValue ?? emptyObject, + rootValue: rootValue ?? emptyObject, returnType: returnType ?? info?.returnType ?? getDelegationReturnType(targetSchema, targetOperation, targetFieldName), transforms: @@ -216,7 +226,7 @@ function getDelegationContext({ args, context, info, - rootValue: rootValue ?? info?.rootValue ?? emptyObject, + rootValue: rootValue, returnType: returnType ?? info?.returnType ?? @@ -239,21 +249,15 @@ function validateRequest(delegationContext: DelegationContext, document: Do } } -// Since the memo function relies on WeakMap which needs an object. -// TODO: clarify whether this has been a potential runtime error -const executorFallbackRootValue = {}; - function getExecutor(delegationContext: DelegationContext): Executor { - const { subschemaConfig, targetSchema, context, rootValue } = delegationContext; + const { subschemaConfig, targetSchema, context } = delegationContext; - let executor: Executor = - subschemaConfig?.executor || - createDefaultExecutor(targetSchema, subschemaConfig?.rootValue ?? rootValue ?? executorFallbackRootValue); + let executor: Executor = subschemaConfig?.executor || createDefaultExecutor(targetSchema); if (subschemaConfig?.batch) { const batchingOptions = subschemaConfig?.batchingOptions; executor = getBatchingExecutor( - context as any, + context ?? self ?? window ?? global, executor, batchingOptions?.dataLoaderOptions, batchingOptions?.extensionsReducer @@ -264,28 +268,28 @@ function getExecutor(delegationContext: DelegationContext): } function getSubscriber(delegationContext: DelegationContext): Subscriber { - const { subschemaConfig, targetSchema, rootValue } = delegationContext; - return subschemaConfig?.subscriber || createDefaultSubscriber(targetSchema, subschemaConfig?.rootValue || rootValue); + const { subschemaConfig, targetSchema } = delegationContext; + return subschemaConfig?.subscriber || createDefaultSubscriber(targetSchema); } -const createDefaultExecutor = memoize2(function (schema: GraphQLSchema, rootValue?: Record): Executor { - return (({ document, context, variables, info }: ExecutionParams) => +function createDefaultExecutor(schema: GraphQLSchema): Executor { + return (({ document, context, variables, rootValue }: ExecutionParams) => execute({ schema, document, contextValue: context, variableValues: variables, - rootValue: rootValue ?? info?.rootValue, + rootValue, })) as Executor; -}); +} -function createDefaultSubscriber(schema: GraphQLSchema, rootValue?: Record) { - return ({ document, context, variables, info }: ExecutionParams) => +function createDefaultSubscriber(schema: GraphQLSchema) { + return ({ document, context, variables, rootValue }: ExecutionParams) => subscribe({ schema, document, contextValue: context, variableValues: variables, - rootValue: rootValue ?? info?.rootValue, + rootValue, }) as any; } diff --git a/packages/delegate/src/externalObjects.ts b/packages/delegate/src/externalObjects.ts index 81e2a9679e5..daee3d8035d 100644 --- a/packages/delegate/src/externalObjects.ts +++ b/packages/delegate/src/externalObjects.ts @@ -41,7 +41,8 @@ export function mergeExternalObjects( const results: Array = []; let errors: Array = []; - sources.forEach((source, index) => { + for (const index in sources) { + const source = sources[index]; if (source instanceof Error || source === null) { const selectionSet = selectionSets[index]; const fieldNodes = collectFields( @@ -56,7 +57,7 @@ export function mergeExternalObjects( Object.create(null) ); const nullResult = {}; - Object.keys(fieldNodes).forEach(responseKey => { + for (const responseKey in fieldNodes) { if (source instanceof GraphQLError) { nullResult[responseKey] = relocatedError(source, path.concat([responseKey])); } else if (source instanceof Error) { @@ -64,31 +65,24 @@ export function mergeExternalObjects( } else { nullResult[responseKey] = null; } - }); + } results.push(nullResult); } else { errors = errors.concat(source[UNPATHED_ERRORS_SYMBOL]); results.push(source); } - }); + } const combinedResult: ExternalObject = results.reduce(mergeDeep, target); - const newFieldSubschemaMap = target[FIELD_SUBSCHEMA_MAP_SYMBOL] ?? Object.create(null); - - results.forEach((source: ExternalObject) => { + const newFieldSubschemaMap = results.reduce((newFieldSubschemaMap, source) => { const objectSubschema = source[OBJECT_SUBSCHEMA_SYMBOL]; const fieldSubschemaMap = source[FIELD_SUBSCHEMA_MAP_SYMBOL]; - if (fieldSubschemaMap === undefined) { - Object.keys(source).forEach(responseKey => { - newFieldSubschemaMap[responseKey] = objectSubschema; - }); - } else { - Object.keys(source).forEach(responseKey => { - newFieldSubschemaMap[responseKey] = fieldSubschemaMap[responseKey] ?? objectSubschema; - }); + for (const responseKey in source) { + newFieldSubschemaMap[responseKey] = fieldSubschemaMap?.[responseKey] ?? objectSubschema; } - }); + return newFieldSubschemaMap; + }, target[FIELD_SUBSCHEMA_MAP_SYMBOL] ?? Object.create(null)); combinedResult[FIELD_SUBSCHEMA_MAP_SYMBOL] = newFieldSubschemaMap; combinedResult[OBJECT_SUBSCHEMA_SYMBOL] = target[OBJECT_SUBSCHEMA_SYMBOL]; diff --git a/packages/delegate/src/getFieldsNotInSubschema.ts b/packages/delegate/src/getFieldsNotInSubschema.ts index 6b81dad4db9..92b01eec8bc 100644 --- a/packages/delegate/src/getFieldsNotInSubschema.ts +++ b/packages/delegate/src/getFieldsNotInSubschema.ts @@ -17,7 +17,7 @@ function collectSubFields(info: GraphQLResolveInfo, typeName: string): Record { + for (const fieldNode of info.fieldNodes) { if (fieldNode.selectionSet) { subFieldNodes = collectFields( partialExecutionContext, @@ -27,13 +27,13 @@ function collectSubFields(info: GraphQLResolveInfo, typeName: string): Record = info.schema.extensions?.['stitchingInfo']; const selectionSetsByField = stitchingInfo?.selectionSetsByField; - Object.keys(subFieldNodes).forEach(responseName => { + for (const responseName in subFieldNodes) { const fieldName = subFieldNodes[responseName][0].name.value; const fieldSelectionSet = selectionSetsByField?.[typeName]?.[fieldName]; if (fieldSelectionSet != null) { @@ -45,7 +45,7 @@ function collectSubFields(info: GraphQLResolveInfo, typeName: string): Record = []; - Object.keys(subFieldNodes).forEach(responseName => { + for (const responseName in subFieldNodes) { const fieldName = subFieldNodes[responseName][0].name.value; if (!(fieldName in fields)) { fieldsNotInSchema = fieldsNotInSchema.concat(subFieldNodes[responseName]); } - }); + } return fieldsNotInSchema; }); diff --git a/packages/delegate/src/mergeFields.ts b/packages/delegate/src/mergeFields.ts index 8e711a6d058..e15ae559756 100644 --- a/packages/delegate/src/mergeFields.ts +++ b/packages/delegate/src/mergeFields.ts @@ -30,7 +30,7 @@ const sortSubschemasByProxiability = memoize4(function ( const proxiableSubschemas: Array = []; const nonProxiableSubschemas: Array = []; - targetSubschemas.forEach(t => { + for (const t of targetSubschemas) { const selectionSet = mergedTypeInfo.selectionSets.get(t); const fieldSelectionSets = mergedTypeInfo.fieldSelectionSets.get(t); if ( @@ -55,7 +55,7 @@ const sortSubschemasByProxiability = memoize4(function ( nonProxiableSubschemas.push(t); } } - }); + } return { proxiableSubschemas, @@ -77,9 +77,9 @@ const buildDelegationPlan = memoize3(function ( // 2. for each selection: const delegationMap: Map> = new Map(); - fieldNodes.forEach(fieldNode => { + for (const fieldNode of fieldNodes) { if (fieldNode.name.value === '__typename') { - return; + continue; } // 2a. use uniqueFields map to assign fields to subschema if one of possible subschemas @@ -88,7 +88,7 @@ const buildDelegationPlan = memoize3(function ( if (uniqueSubschema != null) { if (!proxiableSubschemas.includes(uniqueSubschema)) { unproxiableFieldNodes.push(fieldNode); - return; + continue; } const existingSubschema = delegationMap.get(uniqueSubschema); @@ -98,7 +98,7 @@ const buildDelegationPlan = memoize3(function ( delegationMap.set(uniqueSubschema, [fieldNode]); } - return; + continue; } // 2b. use nonUniqueFields to assign to a possible subschema, @@ -107,13 +107,13 @@ const buildDelegationPlan = memoize3(function ( let nonUniqueSubschemas: Array = nonUniqueFields[fieldNode.name.value]; if (nonUniqueSubschemas == null) { unproxiableFieldNodes.push(fieldNode); - return; + continue; } nonUniqueSubschemas = nonUniqueSubschemas.filter(s => proxiableSubschemas.includes(s)); if (!nonUniqueSubschemas.length) { unproxiableFieldNodes.push(fieldNode); - return; + continue; } const existingSubschema = nonUniqueSubschemas.find(s => delegationMap.has(s)); @@ -123,16 +123,16 @@ const buildDelegationPlan = memoize3(function ( } else { delegationMap.set(nonUniqueSubschemas[0], [fieldNode]); } - }); + } const finalDelegationMap: Map = new Map(); - delegationMap.forEach((selections, subschema) => { + for (const [subschema, selections] of delegationMap) { finalDelegationMap.set(subschema, { kind: Kind.SELECTION_SET, selections, }); - }); + } return { delegationMap: finalDelegationMap, @@ -177,14 +177,15 @@ export function mergeFields( } const resultMap: Map, SelectionSetNode> = new Map(); - delegationMap.forEach((selectionSet: SelectionSetNode, s: Subschema) => { - // TODO: Verify whether it is safe that resolver always exists. - const resolver = mergedTypeInfo.resolvers.get(s)!; - const valueOrPromise = new ValueOrPromise(() => resolver(object, context, info, s, selectionSet)).catch( - error => error - ); - resultMap.set(valueOrPromise, selectionSet); - }); + for (const [s, selectionSet] of delegationMap) { + const resolver = mergedTypeInfo.resolvers.get(s); + if (resolver) { + const valueOrPromise = new ValueOrPromise(() => resolver(object, context, info, s, selectionSet)).catch( + error => error + ); + resultMap.set(valueOrPromise, selectionSet); + } + } return ValueOrPromise.all(Array.from(resultMap.keys())) .then(results => diff --git a/packages/delegate/src/resolveExternalValue.ts b/packages/delegate/src/resolveExternalValue.ts index 954a024a30c..66a2e768ee8 100644 --- a/packages/delegate/src/resolveExternalValue.ts +++ b/packages/delegate/src/resolveExternalValue.ts @@ -13,14 +13,13 @@ import { locatedError, } from 'graphql'; -import AggregateError from '@ardatan/aggregate-error'; +import { AggregateError, Maybe } from '@graphql-tools/utils'; import { StitchingInfo, SubschemaConfig } from './types'; import { annotateExternalObject, isExternalObject } from './externalObjects'; import { getFieldsNotInSubschema } from './getFieldsNotInSubschema'; import { mergeFields } from './mergeFields'; import { Subschema } from './Subschema'; -import { Maybe } from '@graphql-tools/utils'; export function resolveExternalValue( result: any, @@ -162,17 +161,17 @@ function resolveExternalListMember( } } -const reportedErrors: WeakMap = new Map(); +const reportedErrors = new WeakMap(); function reportUnpathedErrorsViaNull(unpathedErrors: Array) { if (unpathedErrors.length) { const unreportedErrors: Array = []; - unpathedErrors.forEach(error => { + for (const error of unpathedErrors) { if (!reportedErrors.has(error)) { unreportedErrors.push(error); reportedErrors.set(error, true); } - }); + } if (unreportedErrors.length) { if (unreportedErrors.length === 1) { diff --git a/packages/delegate/src/subschemaConfig.ts b/packages/delegate/src/subschemaConfig.ts index 2c9fd247f11..55c85c9d407 100644 --- a/packages/delegate/src/subschemaConfig.ts +++ b/packages/delegate/src/subschemaConfig.ts @@ -12,7 +12,7 @@ export function cloneSubschemaConfig(subschemaConfig: SubschemaConfig): Subschem if (newSubschemaConfig.merge != null) { newSubschemaConfig.merge = { ...subschemaConfig.merge }; - for (const typeName of Object.keys(newSubschemaConfig.merge)) { + for (const typeName in newSubschemaConfig.merge) { const mergedTypeConfig = (newSubschemaConfig.merge[typeName] = { ...(subschemaConfig.merge?.[typeName] ?? {}) }); if (mergedTypeConfig.entryPoints != null) { @@ -21,9 +21,9 @@ export function cloneSubschemaConfig(subschemaConfig: SubschemaConfig): Subschem if (mergedTypeConfig.fields != null) { const fields = (mergedTypeConfig.fields = { ...mergedTypeConfig.fields }); - Object.keys(fields).forEach(fieldName => { + for (const fieldName in fields) { fields[fieldName] = { ...fields[fieldName] }; - }); + } } } } diff --git a/packages/delegate/src/transforms/AddArgumentsAsVariables.ts b/packages/delegate/src/transforms/AddArgumentsAsVariables.ts index 33fb3881de1..05d96eeddef 100644 --- a/packages/delegate/src/transforms/AddArgumentsAsVariables.ts +++ b/packages/delegate/src/transforms/AddArgumentsAsVariables.ts @@ -2,7 +2,6 @@ import { ArgumentNode, DocumentNode, FragmentDefinitionNode, - GraphQLArgument, GraphQLField, GraphQLObjectType, GraphQLSchema, @@ -104,7 +103,7 @@ function addVariablesToRootField( newSelectionSet.push({ ...selection, - arguments: Object.keys(argumentNodeMap).map(argName => argumentNodeMap[argName]), + arguments: Object.values(argumentNodeMap), }); } else { newSelectionSet.push(selection); @@ -113,7 +112,7 @@ function addVariablesToRootField( return { ...operation, - variableDefinitions: Object.keys(variableDefinitionMap).map(varName => variableDefinitionMap[varName]), + variableDefinitions: Object.values(variableDefinitionMap), selectionSet: { kind: Kind.SELECTION_SET, selections: newSelectionSet, @@ -137,7 +136,7 @@ function updateArguments( variableValues: Record, newArgs: Record ): void { - targetField.args.forEach((argument: GraphQLArgument) => { + for (const argument of targetField.args) { const argName = argument.name; const argType = argument.type; @@ -151,5 +150,5 @@ function updateArguments( serializeInputValue(argType, newArgs[argName]) ); } - }); + } } diff --git a/packages/delegate/src/transforms/AddSelectionSets.ts b/packages/delegate/src/transforms/AddSelectionSets.ts index f663ad6651b..3ffad61e54a 100644 --- a/packages/delegate/src/transforms/AddSelectionSets.ts +++ b/packages/delegate/src/transforms/AddSelectionSets.ts @@ -50,7 +50,7 @@ function visitSelectionSet( } if (parentTypeName in selectionSetsByField) { - node.selections.forEach(selection => { + for (const selection of node.selections) { if (selection.kind === Kind.FIELD) { const name = selection.name.value; const selectionSet = selectionSetsByField[parentTypeName][name]; @@ -58,35 +58,35 @@ function visitSelectionSet( addSelectionsToMap(newSelections, selectionSet); } } - }); + } } if (parentTypeName in dynamicSelectionSetsByField) { - node.selections.forEach(selection => { + for (const selection of node.selections) { if (selection.kind === Kind.FIELD) { const name = selection.name.value; const dynamicSelectionSets = dynamicSelectionSetsByField[parentTypeName][name]; if (dynamicSelectionSets != null) { - dynamicSelectionSets.forEach(selectionSetFn => { + for (const selectionSetFn of dynamicSelectionSets) { const selectionSet = selectionSetFn(selection); if (selectionSet != null) { addSelectionsToMap(newSelections, selectionSet); } - }); + } } } - }); + } } return { ...node, - selections: Array.from(newSelections.values()), + selections: [...newSelections.values()], }; } } const addSelectionsToMap = memoize2(function (map: Map, selectionSet: SelectionSetNode): void { - selectionSet.selections.forEach(selection => { + for (const selection of selectionSet.selections) { map.set(print(selection), selection); - }); + } }); diff --git a/packages/delegate/src/transforms/CheckResultAndHandleErrors.ts b/packages/delegate/src/transforms/CheckResultAndHandleErrors.ts index 912da07bc46..bb1de4a5cdc 100644 --- a/packages/delegate/src/transforms/CheckResultAndHandleErrors.ts +++ b/packages/delegate/src/transforms/CheckResultAndHandleErrors.ts @@ -7,9 +7,7 @@ import { locatedError, } from 'graphql'; -import AggregateError from '@ardatan/aggregate-error'; - -import { getResponseKeyFromInfo, ExecutionResult, relocatedError } from '@graphql-tools/utils'; +import { AggregateError, getResponseKeyFromInfo, ExecutionResult, relocatedError } from '@graphql-tools/utils'; import { SubschemaConfig, Transform, DelegationContext } from '../types'; import { resolveExternalValue } from '../resolveExternalValue'; @@ -86,38 +84,39 @@ export function mergeDataAndErrors( return { data, unpathedErrors: [] }; } - let unpathedErrors: Array = []; + const unpathedErrors: Array = []; - const errorMap: Record> = Object.create(null); - errors.forEach(error => { + const errorMap = new Map>(); + for (const error of errors) { const pathSegment = error.path?.[index]; if (pathSegment != null) { - const pathSegmentErrors = errorMap[pathSegment]; + let pathSegmentErrors = errorMap.get(pathSegment); if (pathSegmentErrors === undefined) { - errorMap[pathSegment] = [error]; + pathSegmentErrors = [error]; + errorMap.set(pathSegment, pathSegmentErrors); } else { pathSegmentErrors.push(error); } } else { unpathedErrors.push(error); } - }); + } - Object.keys(errorMap).forEach(pathSegment => { + for (const [pathSegment, pathSegmentErrors] of errorMap) { if (data[pathSegment] !== undefined) { const { data: newData, unpathedErrors: newErrors } = mergeDataAndErrors( data[pathSegment], - errorMap[pathSegment], + pathSegmentErrors, path, onLocatedError, index + 1 ); data[pathSegment] = newData; - unpathedErrors = unpathedErrors.concat(newErrors); + unpathedErrors.push(...newErrors); } else { - unpathedErrors = unpathedErrors.concat(errorMap[pathSegment]); + unpathedErrors.push(...pathSegmentErrors); } - }); + } return { data, unpathedErrors }; } diff --git a/packages/delegate/src/transforms/ExpandAbstractTypes.ts b/packages/delegate/src/transforms/ExpandAbstractTypes.ts index a2ba8d2abfa..f320397c209 100644 --- a/packages/delegate/src/transforms/ExpandAbstractTypes.ts +++ b/packages/delegate/src/transforms/ExpandAbstractTypes.ts @@ -50,7 +50,7 @@ function extractPossibleTypes(sourceSchema: GraphQLSchema, targetSchema: GraphQL const typeMap = sourceSchema.getTypeMap(); const possibleTypesMap: Record> = Object.create(null); const interfaceExtensionsMap: Record> = Object.create(null); - Object.keys(typeMap).forEach(typeName => { + for (const typeName in typeMap) { const type = typeMap[typeName]; if (isAbstractType(type)) { const targetType = targetSchema.getType(typeName); @@ -58,12 +58,14 @@ function extractPossibleTypes(sourceSchema: GraphQLSchema, targetSchema: GraphQL if (isInterfaceType(type) && isInterfaceType(targetType)) { const targetTypeFields = targetType.getFields(); const extensionFields: Record = Object.create(null); - Object.keys(type.getFields()).forEach((fieldName: string) => { + let isExtensionFieldsEmpty = true; + for (const fieldName in type.getFields()) { if (!targetTypeFields[fieldName]) { extensionFields[fieldName] = true; + isExtensionFieldsEmpty = false; } - }); - if (Object.keys(extensionFields).length) { + } + if (!isExtensionFieldsEmpty) { interfaceExtensionsMap[typeName] = extensionFields; } } @@ -75,21 +77,21 @@ function extractPossibleTypes(sourceSchema: GraphQLSchema, targetSchema: GraphQL .map(impl => impl.name); } } - }); + } return { possibleTypesMap, interfaceExtensionsMap }; } function flipMapping(mapping: Record>): Record> { const result: Record> = Object.create(null); - Object.keys(mapping).forEach(typeName => { + for (const typeName in mapping) { const toTypeNames = mapping[typeName]; - toTypeNames.forEach(toTypeName => { + for (const toTypeName of toTypeNames) { if (!(toTypeName in result)) { result[toTypeName] = []; } result[toTypeName].push(typeName); - }); - }); + } + } return result; } @@ -135,12 +137,12 @@ function expandAbstractTypes( const newFragments: Array = []; const fragmentReplacements: Record> = Object.create(null); - fragments.forEach((fragment: FragmentDefinitionNode) => { + for (const fragment of fragments) { newFragments.push(fragment); const possibleTypes = possibleTypesMap[fragment.typeCondition.name.value]; if (possibleTypes != null) { fragmentReplacements[fragment.name.value] = []; - possibleTypes.forEach(possibleTypeName => { + for (const possibleTypeName of possibleTypes) { const name = generateFragmentName(possibleTypeName); existingFragmentNames.push(name); const newFragment: FragmentDefinitionNode = { @@ -164,9 +166,9 @@ function expandAbstractTypes( fragmentName: name, typeName: possibleTypeName, }); - }); + } } - }); + } const newDocument = { ...document, @@ -184,12 +186,12 @@ function expandAbstractTypes( const parentType: GraphQLNamedType = getNamedType(maybeType); const interfaceExtension = interfaceExtensionsMap[parentType.name]; const interfaceExtensionFields = [] as Array; - node.selections.forEach((selection: SelectionNode) => { + for (const selection of node.selections) { if (selection.kind === Kind.INLINE_FRAGMENT) { if (selection.typeCondition != null) { const possibleTypes = possibleTypesMap[selection.typeCondition.name.value]; if (possibleTypes != null) { - possibleTypes.forEach(possibleType => { + for (const possibleType of possibleTypes) { const maybePossibleType = targetSchema.getType(possibleType); if ( maybePossibleType != null && @@ -197,13 +199,13 @@ function expandAbstractTypes( ) { addedSelections.push(generateInlineFragment(possibleType, selection.selectionSet)); } - }); + } } } } else if (selection.kind === Kind.FRAGMENT_SPREAD) { const fragmentName = selection.name.value; if (fragmentName in fragmentReplacements) { - fragmentReplacements[fragmentName].forEach(replacement => { + for (const replacement of fragmentReplacements[fragmentName]) { const typeName = replacement.typeName; const maybeReplacementType = targetSchema.getType(typeName); if (maybeReplacementType != null && implementsAbstractType(targetSchema, parentType, maybeType)) { @@ -215,7 +217,7 @@ function expandAbstractTypes( }, }); } - }); + } } } else if ( interfaceExtension != null && @@ -224,7 +226,7 @@ function expandAbstractTypes( ) { interfaceExtensionFields.push(selection); } - }); + } if (parentType.name in reversePossibleTypesMap) { addedSelections.push({ @@ -239,14 +241,14 @@ function expandAbstractTypes( if (interfaceExtensionFields.length) { const possibleTypes = possibleTypesMap[parentType.name]; if (possibleTypes != null) { - possibleTypes.forEach(possibleType => { + for (const possibleType of possibleTypes) { addedSelections.push( generateInlineFragment(possibleType, { kind: Kind.SELECTION_SET, selections: interfaceExtensionFields, }) ); - }); + } newSelections = newSelections.filter( (selection: SelectionNode) => diff --git a/packages/delegate/src/transforms/FilterToSchema.ts b/packages/delegate/src/transforms/FilterToSchema.ts index 553f3618903..3c703ed9141 100644 --- a/packages/delegate/src/transforms/FilterToSchema.ts +++ b/packages/delegate/src/transforms/FilterToSchema.ts @@ -71,7 +71,7 @@ function filterToSchema( let fragmentSet = Object.create(null); - operations.forEach((operation: OperationDefinitionNode) => { + for (const operation of operations) { let type: Maybe>; if (operation.operation === 'subscription') { type = targetSchema.getSubscriptionType(); @@ -112,7 +112,7 @@ function filterToSchema( variableDefinitions, selectionSet, }); - }); + } const newVariables = usedVariables.reduce((acc, variableName) => { const variableValue = variables[variableName]; @@ -278,13 +278,13 @@ function filterSelectionSet( function union(...arrays: Array>): Array { const cache: Record = Object.create(null); const result: Array = []; - arrays.forEach(array => { - array.forEach(item => { + for (const array of arrays) { + for (const item of array) { if (!(item in cache)) { cache[item] = true; result.push(item); } - }); - }); + } + } return result; } diff --git a/packages/delegate/src/transforms/VisitSelectionSets.ts b/packages/delegate/src/transforms/VisitSelectionSets.ts index 02c547e9e20..d0021378bc3 100644 --- a/packages/delegate/src/transforms/VisitSelectionSets.ts +++ b/packages/delegate/src/transforms/VisitSelectionSets.ts @@ -54,13 +54,13 @@ function visitSelectionSets( const operations: Array = []; const fragments: Record = Object.create(null); - document.definitions.forEach(def => { + for (const def of document.definitions) { if (def.kind === Kind.OPERATION_DEFINITION) { operations.push(def); } else if (def.kind === Kind.FRAGMENT_DEFINITION) { fragments[def.name.value] = def; } - }); + } const partialExecutionContext = { schema, @@ -87,14 +87,14 @@ function visitSelectionSets( ); const newSelections: Array = []; - Object.keys(fields).forEach(responseKey => { + for (const responseKey in fields) { const fieldNodes = fields[responseKey]; - fieldNodes.forEach(fieldNode => { + for (const fieldNode of fieldNodes) { const selectionSet = fieldNode.selectionSet; if (selectionSet == null) { newSelections.push(fieldNode); - return; + continue; } const newSelectionSet = visit( @@ -106,15 +106,15 @@ function visitSelectionSets( if (newSelectionSet === selectionSet) { newSelections.push(fieldNode); - return; + continue; } newSelections.push({ ...fieldNode, selectionSet: newSelectionSet, }); - }); - }); + } + } return { ...operation, @@ -125,7 +125,8 @@ function visitSelectionSets( }; }); - Object.values(fragments).forEach(fragment => { + for (const fragmentIndex in fragments) { + const fragment = fragments[fragmentIndex]; newDefinitions.push( visit( fragment, @@ -134,7 +135,7 @@ function visitSelectionSets( }) ) ); - }); + } return { ...document, diff --git a/packages/delegate/src/types.ts b/packages/delegate/src/types.ts index 272c421c857..7f1a77a2fa3 100644 --- a/packages/delegate/src/types.ts +++ b/packages/delegate/src/types.ts @@ -56,6 +56,7 @@ export interface DelegationContext> { transforms: Array>; transformedSchema: GraphQLSchema; skipTypeMerging: boolean; + operationName?: string; } export type DelegationBinding> = ( @@ -143,7 +144,6 @@ export interface SubschemaConfig; transforms?: Array>; merge?: Record>; - rootValue?: Record; executor?: Executor; subscriber?: Subscriber; batch?: boolean; diff --git a/packages/delegate/tests/delegateToSchema.test..ts b/packages/delegate/tests/delegateToSchema.test..ts index 4c51bf400c4..4f4c23ce2f0 100644 --- a/packages/delegate/tests/delegateToSchema.test..ts +++ b/packages/delegate/tests/delegateToSchema.test..ts @@ -54,7 +54,7 @@ describe('delegateToSchema', () => { ); assertSome(result.data) - expect(result.data.delegateToSchema).toEqual('test'); + expect(result.data['delegateToSchema']).toEqual('test'); }); test('should work even where there are default fields', async () => { @@ -101,7 +101,7 @@ describe('delegateToSchema', () => { ); assertSome(result.data) - expect(result.data.delegateToSchema).toEqual('test'); + expect(result.data['delegateToSchema']).toEqual('test'); }); test('should work even when there are variables', async () => { @@ -153,6 +153,6 @@ describe('delegateToSchema', () => { ); assertSome(result.data) - expect(result.data.delegateToSchema).toEqual('test'); + expect(result.data['delegateToSchema']).toEqual('test'); }); }); diff --git a/packages/delegate/tests/errors.test.ts b/packages/delegate/tests/errors.test.ts index 56833c27c62..a78982697a4 100644 --- a/packages/delegate/tests/errors.test.ts +++ b/packages/delegate/tests/errors.test.ts @@ -11,7 +11,7 @@ import { delegateToSchema, defaultMergedResolver } from '../src'; class ErrorWithExtensions extends GraphQLError { constructor(message: string, code: string) { - super(message, null, null, null, null, null, { code }); + super(message, null as any, null, null, null, null, { code }); } } @@ -38,7 +38,7 @@ describe('Errors', () => { fieldNodes: [], returnType: {} as any, parentType: {} as any, - path: {prev: undefined, key: "foo", typename: undefined }, + path: {prev: undefined, key: "foo", typename: undefined } as any, schema: {} as any, fragments: {}, rootValue: {}, @@ -100,9 +100,10 @@ describe('Errors', () => { expect(e.originalError).toBeDefined(); expect(e.originalError.errors).toBeDefined(); expect(e.originalError.errors).toHaveLength(result.errors.length); - result.errors.forEach((error, i) => { + for (const i in result.errors) { + const error = result.errors[i]; expect(e.originalError.errors[i]).toEqual(error); - }); + } } }); @@ -119,7 +120,7 @@ describe('Errors', () => { } `; - const unpathedError = locatedError(new Error('TestError'), undefined, ["_entities", 7, "name"]); + const unpathedError = locatedError(new Error('TestError'), undefined as any, ["_entities", 7, "name"]); const remoteSchema = makeExecutableSchema({ typeDefs, @@ -184,7 +185,7 @@ describe('Errors', () => { } `; - const unpathedError = locatedError(new Error('TestError'), undefined, ["_entities", 7, "name"]); + const unpathedError = locatedError(new Error('TestError'), undefined as any, ["_entities", 7, "name"]); const remoteSchema = makeExecutableSchema({ typeDefs, diff --git a/packages/graphql-tag-pluck/package.json b/packages/graphql-tag-pluck/package.json index 1924a4823c4..db3151acc72 100644 --- a/packages/graphql-tag-pluck/package.json +++ b/packages/graphql-tag-pluck/package.json @@ -29,15 +29,15 @@ "graphql": "^14.0.0 || ^15.0.0" }, "dependencies": { - "@babel/parser": "7.14.6", - "@babel/traverse": "7.14.5", + "@babel/parser": "7.14.7", + "@babel/traverse": "7.14.7", "@babel/types": "7.14.5", "@graphql-tools/utils": "^7.0.0", "tslib": "~2.3.0" }, "devDependencies": { "@types/babel__traverse": "7.11.1", - "@vue/compiler-sfc": "3.1.1" + "@vue/compiler-sfc": "3.1.2" }, "buildOptions": { "external": [ diff --git a/packages/graphql-tools/package.json b/packages/graphql-tools/package.json index bce47c57ac0..431ff3299be 100644 --- a/packages/graphql-tools/package.json +++ b/packages/graphql-tools/package.json @@ -36,30 +36,7 @@ "directory": "dist" }, "dependencies": { - "@graphql-tools/optimize": "1.0.1", - "@graphql-tools/batch-delegate": "^7.0.0", - "@graphql-tools/batch-execute": "^7.0.0", - "@graphql-tools/delegate": "^7.0.10", - "@graphql-tools/graphql-tag-pluck": "^6.2.6", - "@graphql-tools/import": "^6.2.4", - "@graphql-tools/links": "^7.0.4", - "@graphql-tools/load": "^6.2.5", - "@graphql-tools/code-file-loader": "^6.2.5", - "@graphql-tools/git-loader": "^6.2.5", - "@graphql-tools/github-loader": "^6.2.5", - "@graphql-tools/graphql-file-loader": "^6.2.5", - "@graphql-tools/json-file-loader": "^6.2.5", - "@graphql-tools/module-loader": "^6.2.5", - "@graphql-tools/url-loader": "^6.3.2", - "@graphql-tools/load-files": "^6.2.4", - "@graphql-tools/merge": "^6.2.14", - "@graphql-tools/mock": "^7.0.0", - "@graphql-tools/relay-operation-optimizer": "^6.2.5", - "@graphql-tools/resolvers-composition": "^6.2.5", "@graphql-tools/schema": "^7.0.0", - "@graphql-tools/stitch": "^7.3.0", - "@graphql-tools/utils": "^7.0.1", - "@graphql-tools/wrap": "^7.0.0", "tslib": "~2.3.0" } } diff --git a/packages/graphql-tools/src/index.ts b/packages/graphql-tools/src/index.ts index dbd653e5d01..b214057bd3d 100644 --- a/packages/graphql-tools/src/index.ts +++ b/packages/graphql-tools/src/index.ts @@ -1,23 +1,5 @@ -export * from '@graphql-tools/batch-delegate'; -export * from '@graphql-tools/batch-execute'; -export * from '@graphql-tools/delegate'; -export * from '@graphql-tools/graphql-tag-pluck'; -export * from '@graphql-tools/import'; -export * from '@graphql-tools/links'; -export * from '@graphql-tools/load'; -export * from '@graphql-tools/code-file-loader'; -export * from '@graphql-tools/git-loader'; -export * from '@graphql-tools/github-loader'; -export * from '@graphql-tools/graphql-file-loader'; -export * from '@graphql-tools/module-loader'; -export * from '@graphql-tools/url-loader'; -export * from '@graphql-tools/load-files'; -export * from '@graphql-tools/merge'; -export * from '@graphql-tools/mock'; -export * from '@graphql-tools/relay-operation-optimizer'; -export * from '@graphql-tools/resolvers-composition'; -export * from '@graphql-tools/schema'; -export * from '@graphql-tools/stitch'; -export * from '@graphql-tools/utils'; -export * from '@graphql-tools/wrap'; -export * from '@graphql-tools/optimize'; +export { makeExecutableSchema } from '@graphql-tools/schema'; + +console.warn(`This package has been deprecated and now it only exports makeExecutableSchema. +We recommend you to migrate to scoped packages. +Check out https://www.graphql-tools.com for more information!`); diff --git a/packages/links/src/linkToExecutor.ts b/packages/links/src/linkToExecutor.ts index f885406aa49..4b6c105027a 100644 --- a/packages/links/src/linkToExecutor.ts +++ b/packages/links/src/linkToExecutor.ts @@ -4,20 +4,21 @@ import { toPromise } from '@apollo/client/link/utils'; import { AsyncExecutor, ExecutionParams, ExecutionResult } from '@graphql-tools/utils'; -export const linkToExecutor = (link: ApolloLink): AsyncExecutor => ( - params: ExecutionParams -): Promise> => { - const { document, variables, extensions, context, info } = params; - return toPromise( - execute(link, { - query: document, - variables: variables, - context: { - graphqlContext: context, - graphqlResolveInfo: info, - clientAwareness: {}, - }, - extensions, - }) as Observable> - ); -}; +export const linkToExecutor = + (link: ApolloLink): AsyncExecutor => + (params: ExecutionParams): Promise> => { + const { document, variables, extensions, context, info, operationName } = params; + return toPromise( + execute(link, { + query: document, + variables: variables, + context: { + graphqlContext: context, + graphqlResolveInfo: info, + clientAwareness: {}, + }, + extensions, + operationName, + }) as Observable> + ); + }; diff --git a/packages/links/src/linkToSubscriber.ts b/packages/links/src/linkToSubscriber.ts index ddd228872f7..3da42805e7f 100644 --- a/packages/links/src/linkToSubscriber.ts +++ b/packages/links/src/linkToSubscriber.ts @@ -8,7 +8,7 @@ export const linkToSubscriber = async ( params: ExecutionParams ): Promise | AsyncIterableIterator>> => { - const { document, variables, extensions, context, info } = params; + const { document, variables, extensions, context, info, operationName } = params; return observableToAsyncIterable>( execute(link, { query: document, @@ -19,6 +19,7 @@ export const linkToSubscriber = clientAwareness: {}, }, extensions, + operationName, }) as Observable> )[Symbol.asyncIterator](); }; diff --git a/packages/links/tests/upload.test.ts b/packages/links/tests/upload.test.ts index f41b9eb424a..1b8b6d76066 100644 --- a/packages/links/tests/upload.test.ts +++ b/packages/links/tests/upload.test.ts @@ -25,7 +25,7 @@ function streamToString(stream: Readable) { function startServer(e: Express): Promise { return new Promise((resolve, reject) => { - e.listen(undefined, 'localhost', function (error: Error) { + e.listen(undefined as any, 'localhost', function (this: any, error: Error) { if (error) { reject(error); } else { diff --git a/packages/load-files/tests/file-scanner.spec.ts b/packages/load-files/tests/file-scanner.spec.ts index 4d7616709ef..da1e1454c60 100644 --- a/packages/load-files/tests/file-scanner.spec.ts +++ b/packages/load-files/tests/file-scanner.spec.ts @@ -19,7 +19,7 @@ function testSchemaDir({ path, expected, note, extensions, ignoreIndex }: TestDi }; }); - syncAndAsync.forEach(([type, loadFiles]) => { + for (const [type, loadFiles] of syncAndAsync) { describe(type, () => { it(`should return the correct schema results for path: ${path} (${note})`, async () => { const result = await loadFiles(path, options); @@ -33,7 +33,7 @@ function testSchemaDir({ path, expected, note, extensions, ignoreIndex }: TestDi })).toEqual(expected.map(stripWhitespaces)); }); }); - }) + } } @@ -55,7 +55,7 @@ function testResolversDir({ path, expected, note, extensions, compareValue, igno }; }); - syncAndAsync.forEach(([type, loadFiles]) => { + for (const [type, loadFiles] of syncAndAsync) { describe(type, () => { it(`should return the correct resolvers results for path: ${path} (${note})`, async () => { const result = await loadFiles(path, options); @@ -67,7 +67,7 @@ function testResolversDir({ path, expected, note, extensions, compareValue, igno } }); }) - }); + } } function stripWhitespaces(str: any): string { @@ -219,7 +219,7 @@ describe('file scanner', function() { note: 'extensions and ignored extensions works with a trailing dot', }); }); - syncAndAsync.forEach(([type, loadFiles]) => { + for (const [type, loadFiles] of syncAndAsync) { it(`${type}: should process custom extractExports properly`, async () => { const customQueryTypeName = 'root_query'; const customExtractExports = (fileExport: any) => { @@ -254,7 +254,7 @@ describe('file scanner', function() { expect(typeof resolvers.Query.foo).toBe('function'); expect(resolvers.Query.foo()).toBe('FOO'); }); - }) + } }); interface TestDirOptions { diff --git a/packages/load-files/tests/test-assets/11/1.spec.ts b/packages/load-files/tests/test-assets/11/1.spec.ts index d3cbba3204d..dcbb9a5413e 100644 --- a/packages/load-files/tests/test-assets/11/1.spec.ts +++ b/packages/load-files/tests/test-assets/11/1.spec.ts @@ -1,3 +1,3 @@ -function foo() {} +function bar(..._args: any[]) {} -foo('dummy', () => {}); +bar('dummy', () => {}); diff --git a/packages/load-files/tests/test-assets/11/2.test.ts b/packages/load-files/tests/test-assets/11/2.test.ts index d3cbba3204d..d23eadd8239 100644 --- a/packages/load-files/tests/test-assets/11/2.test.ts +++ b/packages/load-files/tests/test-assets/11/2.test.ts @@ -1,3 +1,3 @@ -function foo() {} +function foo(..._args: any[]) {} foo('dummy', () => {}); diff --git a/packages/load/src/filter-document-kind.ts b/packages/load/src/filter-document-kind.ts index 0798ff35254..044a55404f4 100644 --- a/packages/load/src/filter-document-kind.ts +++ b/packages/load/src/filter-document-kind.ts @@ -1,5 +1,5 @@ -import { debugLog } from '@graphql-tools/utils'; import { DocumentNode, DefinitionNode, Kind } from 'graphql'; +import { env } from 'process'; /** * @internal @@ -17,9 +17,11 @@ export const filterKind = (content: DocumentNode | undefined, filterKinds: null } if (invalidDefinitions.length > 0) { - invalidDefinitions.forEach(d => { - debugLog(`Filtered document of kind ${d.kind} due to filter policy (${filterKinds.join(', ')})`); - }); + if (env['DEBUG']) { + for (const d of invalidDefinitions) { + console.log(`Filtered document of kind ${d.kind} due to filter policy (${filterKinds.join(', ')})`); + } + } } return { diff --git a/packages/load/src/load-typedefs.ts b/packages/load/src/load-typedefs.ts index 255e86e179a..defb838b0af 100644 --- a/packages/load/src/load-typedefs.ts +++ b/packages/load/src/load-typedefs.ts @@ -86,7 +86,7 @@ export function loadTypedefsSync>( const validSources: Source[] = []; - sources.forEach(partialSource => { + for (const partialSource of sources) { parseSource({ partialSource, options, @@ -96,7 +96,7 @@ export function loadTypedefsSync>( validSources.push(source); }, }); - }); + } return prepareResult({ options, pointerOptionMap, validSources }); } diff --git a/packages/load/src/load-typedefs/load-file.ts b/packages/load/src/load-typedefs/load-file.ts index ef8efadfbc9..7f6acf067e8 100644 --- a/packages/load/src/load-typedefs/load-file.ts +++ b/packages/load/src/load-typedefs/load-file.ts @@ -1,4 +1,5 @@ -import { Source, debugLog, Maybe } from '@graphql-tools/utils'; +import { Source, Maybe } from '@graphql-tools/utils'; +import { env } from 'process'; import { LoadTypedefsOptions } from '../load-typedefs'; export async function loadFile(pointer: string, options: LoadTypedefsOptions): Promise> { @@ -17,7 +18,9 @@ export async function loadFile(pointer: string, options: LoadTypedefsOptions): P return loadedValue; } } catch (error) { - debugLog(`Failed to find any GraphQL type definitions in: ${pointer} - ${error.message}`); + if (env['DEBUG']) { + console.error(`Failed to find any GraphQL type definitions in: ${pointer} - ${error.message}`); + } throw error; } } @@ -41,7 +44,9 @@ export function loadFileSync(pointer: string, options: LoadTypedefsOptions): May return loader.loadSync!(pointer, options); } } catch (error) { - debugLog(`Failed to find any GraphQL type definitions in: ${pointer} - ${error.message}`); + if (env['DEBUG']) { + console.error(`Failed to find any GraphQL type definitions in: ${pointer} - ${error.message}`); + } throw error; } } diff --git a/packages/load/src/utils/queue.ts b/packages/load/src/utils/queue.ts index 89188845b71..1a396be83e7 100644 --- a/packages/load/src/utils/queue.ts +++ b/packages/load/src/utils/queue.ts @@ -22,7 +22,9 @@ export function useSyncQueue() { queue.push(fn); }, runAll() { - queue.forEach(fn => fn()); + for (const fn of queue) { + fn(); + } }, }; } diff --git a/packages/load/tests/loaders/documents/documents-from-glob.spec.ts b/packages/load/tests/loaders/documents/documents-from-glob.spec.ts index 39424514293..f7fe2ce3532 100644 --- a/packages/load/tests/loaders/documents/documents-from-glob.spec.ts +++ b/packages/load/tests/loaders/documents/documents-from-glob.spec.ts @@ -34,9 +34,10 @@ describe('documentsFromGlob', () => { const result = await load(glob, { loaders: [new CodeFileLoader()] }); - const operations = separateOperations(result[0].document); + const { document } = result[0]; + const operations = document && separateOperations(document); - expect(Object.keys(operations)).toHaveLength(2); + expect(operations && Object.keys(operations)).toHaveLength(2); }); test(`Should load GraphQL documents that match custom settings`, async () => { @@ -57,9 +58,10 @@ describe('documentsFromGlob', () => { ] }); - const operations = separateOperations(result[0].document); + const { document } = result[0]; + const operations = document && separateOperations(document); - expect(Object.keys(operations)).toHaveLength(1); + expect(operations && Object.keys(operations)).toHaveLength(1); }); test(`Should throw on syntax errors`, async () => { diff --git a/packages/load/tests/loaders/schema/integration.spec.ts b/packages/load/tests/loaders/schema/integration.spec.ts index aedfd07a2c8..b913ffae1c6 100644 --- a/packages/load/tests/loaders/schema/integration.spec.ts +++ b/packages/load/tests/loaders/schema/integration.spec.ts @@ -33,8 +33,8 @@ describe('loadSchema', () => { const schema = await load(schemaPath, { loaders: [new CodeFileLoader()] }); - expect(schema.getTypeMap().User).toBeDefined(); - expect(schema.getTypeMap().Query).toBeDefined(); + expect(schema.getTypeMap()['User']).toBeDefined(); + expect(schema.getTypeMap()['Query']).toBeDefined(); }); test('should work with graphql single file', async () => { @@ -43,7 +43,7 @@ describe('loadSchema', () => { loaders: [new GraphQLFileLoader()] }); - expect(schema.getTypeMap().User).toBeDefined(); + expect(schema.getTypeMap()['User']).toBeDefined(); }); test('import and merge Query types from few different files', async () => { diff --git a/packages/load/tests/loaders/schema/schema-from-export.spec.ts b/packages/load/tests/loaders/schema/schema-from-export.spec.ts index 0017c8ddf2f..2e8cce1aeaf 100644 --- a/packages/load/tests/loaders/schema/schema-from-export.spec.ts +++ b/packages/load/tests/loaders/schema/schema-from-export.spec.ts @@ -35,7 +35,7 @@ describe('Schema From Export', () => { expect(isSchema(result)).toBeTruthy(); const QueryType = result.getQueryType() assertNonMaybe(QueryType) - expect(QueryType.getFields().hello).toBeDefined(); + expect(QueryType.getFields()['hello']).toBeDefined(); }); test('should load the schema correctly from variable export', async () => { diff --git a/packages/load/tests/loaders/schema/schema-from-typedefs.spec.ts b/packages/load/tests/loaders/schema/schema-from-typedefs.spec.ts index b35d89ce1c9..b534df6b625 100644 --- a/packages/load/tests/loaders/schema/schema-from-typedefs.spec.ts +++ b/packages/load/tests/loaders/schema/schema-from-typedefs.spec.ts @@ -26,8 +26,8 @@ describe('schema from typedefs', () => { loaders: [new GraphQLFileLoader()] }); - expect(schema.getTypeMap().User).toBeDefined(); - expect(schema.getTypeMap().Query).toBeDefined(); + expect(schema.getTypeMap()['User']).toBeDefined(); + expect(schema.getTypeMap()['Query']).toBeDefined(); }); it('should ignore empty files when using glob expressions', async () => { @@ -78,8 +78,8 @@ describe('schema from typedefs', () => { loaders: [new CodeFileLoader()] }); - expect(schema.getTypeMap().User).toBeDefined(); - expect(schema.getTypeMap().Query).toBeDefined(); + expect(schema.getTypeMap()['User']).toBeDefined(); + expect(schema.getTypeMap()['Query']).toBeDefined(); }); it('should work without globs correctly', async () => { @@ -88,8 +88,8 @@ describe('schema from typedefs', () => { loaders: [new CodeFileLoader()] }); - expect(schema.getTypeMap().User).toBeDefined(); - expect(schema.getTypeMap().Query).toBeDefined(); + expect(schema.getTypeMap()['User']).toBeDefined(); + expect(schema.getTypeMap()['Query']).toBeDefined(); }); it('should work with import notations', async () => { @@ -98,8 +98,8 @@ describe('schema from typedefs', () => { loaders: [new GraphQLFileLoader()] }); - expect(schema.getTypeMap().User).toBeDefined(); - expect(schema.getTypeMap().Query).toBeDefined(); + expect(schema.getTypeMap()['User']).toBeDefined(); + expect(schema.getTypeMap()['Query']).toBeDefined(); }); it('should work with extensions (static graphql file)', async () => { @@ -149,9 +149,9 @@ describe('schema from typedefs', () => { includeSources: true, }); assertNonMaybe(schemaWithSources.extensions) - expect(schemaWithSources.extensions.sources).toBeDefined(); - expect(schemaWithSources.extensions.sources).toHaveLength(1); - expect(schemaWithSources.extensions.sources[0]).toMatchObject(expect.objectContaining({ + expect(schemaWithSources.extensions['sources']).toBeDefined(); + expect(schemaWithSources.extensions['sources']).toHaveLength(1); + expect(schemaWithSources.extensions['sources'][0]).toMatchObject(expect.objectContaining({ name: glob })) @@ -159,7 +159,7 @@ describe('schema from typedefs', () => { loaders: [new GraphQLFileLoader()] }); assertNonMaybe(schemaWithoutSources.extensions) - expect(schemaWithoutSources.extensions.sources).not.toBeDefined(); + expect(schemaWithoutSources.extensions['sources']).not.toBeDefined(); }); }) }); diff --git a/packages/loaders/apollo-engine/package.json b/packages/loaders/apollo-engine/package.json index 4e99982fe17..3382d69b11d 100644 --- a/packages/loaders/apollo-engine/package.json +++ b/packages/loaders/apollo-engine/package.json @@ -30,8 +30,7 @@ "graphql": "^14.0.0 || ^15.0.0" }, "dependencies": { - "@ardatan/aggregate-error": "0.0.6", - "@graphql-tools/utils": "^7.0.0", + "@graphql-tools/utils": "^7.10.0", "cross-fetch": "3.1.4", "tslib": "~2.3.0", "sync-fetch": "0.3.0" diff --git a/packages/loaders/apollo-engine/src/index.ts b/packages/loaders/apollo-engine/src/index.ts index 3015ce125b1..2c1c4b47ba5 100644 --- a/packages/loaders/apollo-engine/src/index.ts +++ b/packages/loaders/apollo-engine/src/index.ts @@ -1,6 +1,5 @@ -import { SchemaLoader, Source, SingleFileOptions, parseGraphQLSDL } from '@graphql-tools/utils'; +import { SchemaLoader, Source, SingleFileOptions, parseGraphQLSDL, AggregateError } from '@graphql-tools/utils'; import { fetch } from 'cross-fetch'; -import AggregateError from '@ardatan/aggregate-error'; import syncFetch from 'sync-fetch'; /** @@ -66,7 +65,7 @@ export class ApolloEngineLoader implements SchemaLoader { const { data, errors } = await response.json(); if (errors) { - throw new AggregateError(errors); + throw new AggregateError(errors, 'Introspection from Apollo Engine failed'); } return parseGraphQLSDL(pointer, data.service.schema.document, options); @@ -79,7 +78,7 @@ export class ApolloEngineLoader implements SchemaLoader { const { data, errors } = response.json(); if (errors) { - throw new AggregateError(errors); + throw new AggregateError(errors, 'Introspection from Apollo Engine failed'); } return parseGraphQLSDL(pointer, data.service.schema.document, options); diff --git a/packages/loaders/code-file/src/index.ts b/packages/loaders/code-file/src/index.ts index 37f042e916e..a1a7a7eb266 100644 --- a/packages/loaders/code-file/src/index.ts +++ b/packages/loaders/code-file/src/index.ts @@ -4,7 +4,6 @@ import { isSchema, GraphQLSchema, DocumentNode } from 'graphql'; import { SchemaPointerSingle, DocumentPointerSingle, - debugLog, SingleFileOptions, Source, UniversalLoader, @@ -24,8 +23,9 @@ import isGlob from 'is-glob'; import unixify from 'unixify'; import { tryToLoadFromExport, tryToLoadFromExportSync } from './load-from-module'; import { isAbsolute, resolve } from 'path'; -import { cwd } from 'process'; +import { cwd, env } from 'process'; import { readFileSync, promises as fsPromises, existsSync } from 'fs'; +import { createRequire } from 'module'; const { readFile, access } = fsPromises; @@ -135,7 +135,9 @@ export class CodeFileLoader implements UniversalLoader { return parseGraphQLSDL(pointer, sdl, options); } } catch (e) { - debugLog(`Failed to load schema from code file "${normalizedFilePath}": ${e.message}`); + if (env['DEBUG']) { + console.error(`Failed to load schema from code file "${normalizedFilePath}": ${e.message}`); + } errors.push(e); } } @@ -178,7 +180,9 @@ export class CodeFileLoader implements UniversalLoader { return parseGraphQLSDL(pointer, sdl, options); } } catch (e) { - debugLog(`Failed to load schema from code file "${normalizedFilePath}": ${e.message}`); + if (env['DEBUG']) { + console.error(`Failed to load schema from code file "${normalizedFilePath}": ${e.message}`); + } errors.push(e); } } @@ -186,7 +190,10 @@ export class CodeFileLoader implements UniversalLoader { if (!options.noRequire) { try { if (options && options.require) { - asArray(options.require).forEach(m => require(m)); + const cwdRequire = createRequire(options.cwd || cwd()); + for (const m of asArray(options.require)) { + cwdRequire(m); + } } const loaded = tryToLoadFromExportSync(normalizedFilePath); diff --git a/packages/loaders/code-file/tests/load-from-code-file.spec.ts b/packages/loaders/code-file/tests/load-from-code-file.spec.ts index a18e5163819..06e273905a3 100644 --- a/packages/loaders/code-file/tests/load-from-code-file.spec.ts +++ b/packages/loaders/code-file/tests/load-from-code-file.spec.ts @@ -1,3 +1,4 @@ +/* eslint-disable @typescript-eslint/no-non-null-asserted-optional-chain */ import * as path from 'path'; import { CodeFileLoader } from '../src'; import { parse } from 'graphql'; @@ -11,7 +12,7 @@ describe('loadFromCodeFile', () => { noRequire: true, cwd: __dirname }); - const doc = loaded.document ? loaded.document : parse(loaded.rawSDL); + const doc = loaded?.document ? loaded?.document : parse(loaded?.rawSDL!); expect(doc).toBeFalsy(); } catch (e) { @@ -24,9 +25,9 @@ describe('loadFromCodeFile', () => { noRequire: true, cwd: __dirname }); - const doc = loaded.document ? loaded.document : parse(loaded.rawSDL); + const doc = loaded?.document ? loaded?.document : parse(loaded?.rawSDL!); - expect(doc.kind).toEqual('Document'); + expect(doc?.kind).toEqual('Document'); }); it('should consider options.cwd', async () => { @@ -34,9 +35,9 @@ describe('loadFromCodeFile', () => { cwd: path.resolve(__dirname, 'test-files'), noRequire: true, }); - const doc = loaded.document ? loaded.document : parse(loaded.rawSDL); + const doc = loaded?.document ? loaded?.document : parse(loaded?.rawSDL!); - expect(doc.kind).toEqual('Document'); + expect(doc?.kind).toEqual('Document'); }); it('should load a TypeScript file using decorator', async () => { @@ -44,18 +45,18 @@ describe('loadFromCodeFile', () => { noRequire: true, cwd: __dirname }); - const doc = loaded.document ? loaded.document : parse(loaded.rawSDL); + const doc = loaded?.document ? loaded?.document : parse(loaded?.rawSDL!); - expect(doc.kind).toEqual('Document'); + expect(doc?.kind).toEqual('Document'); }); it('should support string interpolation', async () => { const loaded = await loader.load('./test-files/string-interpolation.js', { cwd: __dirname }); - const doc = loaded.document ? loaded.document : parse(loaded.rawSDL); + const doc = loaded?.document ? loaded?.document : parse(loaded?.rawSDL!); - expect(doc.kind).toEqual('Document'); + expect(doc?.kind).toEqual('Document'); }); }); @@ -68,9 +69,9 @@ describe('loadFromCodeFileSync', () => { noRequire: true, cwd: __dirname }); - const doc = loaded.document ? loaded.document : parse(loaded.rawSDL); + const doc = loaded?.document ? loaded?.document : parse(loaded?.rawSDL!); - expect(doc.kind).toEqual('Document'); + expect(doc?.kind).toEqual('Document'); }).toThrowError('Syntax Error: Unexpected Name "InvalidGetUser"') }); @@ -79,9 +80,9 @@ describe('loadFromCodeFileSync', () => { noRequire: true, cwd: __dirname }); - const doc = loaded.document; + const doc = loaded?.document; - expect(doc.kind).toEqual('Document'); + expect(doc?.kind).toEqual('Document'); }); it('should consider options.cwd', () => { @@ -89,9 +90,9 @@ describe('loadFromCodeFileSync', () => { cwd: path.resolve(__dirname, 'test-files'), noRequire: true, }); - const doc = loaded.document; + const doc = loaded?.document; - expect(doc.kind).toEqual('Document'); + expect(doc?.kind).toEqual('Document'); }); it('should support string interpolation', () => { @@ -99,8 +100,8 @@ describe('loadFromCodeFileSync', () => { cwd: __dirname }); - const doc = loaded.document; + const doc = loaded?.document; - expect(doc.kind).toEqual('Document'); + expect(doc?.kind).toEqual('Document'); }); }); diff --git a/packages/loaders/prisma/package.json b/packages/loaders/prisma/package.json index 066e1c30f91..47fc5c5e654 100644 --- a/packages/loaders/prisma/package.json +++ b/packages/loaders/prisma/package.json @@ -32,7 +32,6 @@ "dependencies": { "@graphql-tools/url-loader": "^6.8.2", "@graphql-tools/utils": "^7.0.0", - "@types/http-proxy-agent": "^4.0.0", "@types/js-yaml": "^4.0.0", "@types/json-stable-stringify": "^1.0.32", "@types/jsonwebtoken": "^8.5.0", diff --git a/packages/loaders/prisma/src/prisma-yml/Environment.ts b/packages/loaders/prisma/src/prisma-yml/Environment.ts index 7bb138edfb3..e6bfb76c01e 100644 --- a/packages/loaders/prisma/src/prisma-yml/Environment.ts +++ b/packages/loaders/prisma/src/prisma-yml/Environment.ts @@ -113,8 +113,8 @@ export class Environment { // clean up all prisma-eu1 and prisma-us1 clusters if they already exist this.clusters = this._getClusters().filter(c => c.name !== 'prisma-eu1' && c.name !== 'prisma-us1'); - res.me.memberships.forEach((m: any) => { - m.workspace.clusters.forEach((cluster: any) => { + for (const m of res.me.memberships) { + for (const cluster of m.workspace.clusters) { const endpoint = cluster.connectInfo ? cluster.connectInfo.endpoint : cluster.customConnectionInfo @@ -132,8 +132,8 @@ export class Environment { m.workspace.slug ) ); - }); - }); + } + } } } catch (e) { debug(e); diff --git a/packages/loaders/prisma/src/prisma-yml/PrismaDefinition.ts b/packages/loaders/prisma/src/prisma-yml/PrismaDefinition.ts index 71d11941b47..4386064f280 100644 --- a/packages/loaders/prisma/src/prisma-yml/PrismaDefinition.ts +++ b/packages/loaders/prisma/src/prisma-yml/PrismaDefinition.ts @@ -255,7 +255,7 @@ and execute ${chalk.bold.green('prisma deploy')} again, to get that value auto-f : []; let allTypes = ''; - typesPaths.forEach(unresolvedTypesPath => { + for (const unresolvedTypesPath of typesPaths) { const typesPath = path.join(this.definitionDir!, unresolvedTypesPath!); try { fs.accessSync(typesPath); @@ -264,7 +264,7 @@ and execute ${chalk.bold.green('prisma deploy')} again, to get that value auto-f } catch { throw new Error(`The types definition file "${typesPath}" could not be found.`); } - }); + } return allTypes; } diff --git a/packages/loaders/prisma/src/prisma-yml/Variables.ts b/packages/loaders/prisma/src/prisma-yml/Variables.ts index d7ed25325c7..6abb8d0bb5a 100644 --- a/packages/loaders/prisma/src/prisma-yml/Variables.ts +++ b/packages/loaders/prisma/src/prisma-yml/Variables.ts @@ -1,4 +1,4 @@ -import * as lodash from 'lodash'; +import _ from 'lodash'; // eslint-disable-next-line // @ts-ignore import replaceall from 'replaceall'; @@ -43,10 +43,10 @@ export class Variables { const deepMapValues = (object: any, callback: any, propertyPath?: string[]): any => { const deepMapValuesIteratee = (value: any, key: any) => deepMapValues(value, callback, propertyPath ? propertyPath.concat(key) : [key]); - if (lodash.isArray(object)) { - return lodash.map(object, deepMapValuesIteratee); - } else if (lodash.isObject(object) && !lodash.isDate(object) && !lodash.isFunction(object)) { - return lodash.extend({}, object, lodash.mapValues(object, deepMapValuesIteratee)); + if (_.isArray(object)) { + return _.map(object, deepMapValuesIteratee); + } else if (_.isObject(object) && !_.isDate(object) && !_.isFunction(object)) { + return _.extend({}, object, _.mapValues(object, deepMapValuesIteratee)); } return callback(object, propertyPath); }; @@ -54,7 +54,7 @@ export class Variables { deepMapValues(objectToPopulate, (property: any, propertyPath: any) => { if (typeof property === 'string') { const populateSingleProperty = this.populateProperty(property, true).then((newProperty: any) => - lodash.set(objectToPopulate, propertyPath, newProperty) + _.set(objectToPopulate, propertyPath, newProperty) ); populateAll.push(populateSingleProperty); } @@ -64,40 +64,43 @@ export class Variables { } populateProperty(propertyParam: any, populateInPlace?: boolean): any { - let property = populateInPlace ? propertyParam : lodash.cloneDeep(propertyParam); + let property = populateInPlace ? propertyParam : _.cloneDeep(propertyParam); const allValuesToPopulate: any[] = []; let warned = false; if (typeof property === 'string' && property.match(this.variableSyntax)) { - property.match(this.variableSyntax)!.forEach(matchedString => { - const variableString = matchedString - .replace(this.variableSyntax, (_, varName) => varName.trim()) - .replace(/\s/g, ''); - - let singleValueToPopulate: Promise | null = null; - if (variableString.match(this.overwriteSyntax)) { - singleValueToPopulate = this.overwrite(variableString); - } else { - singleValueToPopulate = this.getValueFromSource(variableString).then((valueToPopulate: any) => { - if (typeof valueToPopulate === 'object') { - return this.populateObject(valueToPopulate); - } - return valueToPopulate; - }); - } + const matchedStrings = property.match(this.variableSyntax); + if (matchedStrings) { + for (const matchedString of matchedStrings) { + const variableString = matchedString + .replace(this.variableSyntax, (_, varName) => varName.trim()) + .replace(/\s/g, ''); - singleValueToPopulate = singleValueToPopulate!.then(valueToPopulate => { - if (this.warnIfNotFound(variableString, valueToPopulate)) { - warned = true; + let singleValueToPopulate: Promise | null = null; + if (variableString.match(this.overwriteSyntax)) { + singleValueToPopulate = this.overwrite(variableString); + } else { + singleValueToPopulate = this.getValueFromSource(variableString).then((valueToPopulate: any) => { + if (typeof valueToPopulate === 'object') { + return this.populateObject(valueToPopulate); + } + return valueToPopulate; + }); } - return this.populateVariable(property, matchedString, valueToPopulate).then((newProperty: any) => { - property = newProperty; - return Promise.resolve(property); + + singleValueToPopulate = singleValueToPopulate!.then(valueToPopulate => { + if (this.warnIfNotFound(variableString, valueToPopulate)) { + warned = true; + } + return this.populateVariable(property, matchedString, valueToPopulate).then((newProperty: any) => { + property = newProperty; + return Promise.resolve(property); + }); }); - }); - allValuesToPopulate.push(singleValueToPopulate); - }); + allValuesToPopulate.push(singleValueToPopulate); + } + } return Promise.all(allValuesToPopulate).then(() => { if ((property as any) !== (this.json as any) && !warned) { return this.populateProperty(property); @@ -143,7 +146,7 @@ export class Variables { return ( finalValue !== null && typeof finalValue !== 'undefined' && - !(typeof finalValue === 'object' && lodash.isEmpty(finalValue)) + !(typeof finalValue === 'object' && _.isEmpty(finalValue)) ); }); return Promise.resolve(finalValue); @@ -215,7 +218,7 @@ export class Variables { if ( valueToPopulate === null || typeof valueToPopulate === 'undefined' || - (typeof valueToPopulate === 'object' && lodash.isEmpty(valueToPopulate)) + (typeof valueToPopulate === 'object' && _.isEmpty(valueToPopulate)) ) { let varType; if (variableString.match(this.envRefSyntax)) { diff --git a/packages/loaders/prisma/src/prisma-yml/constants.ts b/packages/loaders/prisma/src/prisma-yml/constants.ts index 161a00f0683..93832914b50 100644 --- a/packages/loaders/prisma/src/prisma-yml/constants.ts +++ b/packages/loaders/prisma/src/prisma-yml/constants.ts @@ -1,4 +1,4 @@ -import { invert } from 'lodash'; +import _ from 'lodash'; export const cloudApiEndpoint = process.env['CLOUD_API_ENDPOINT'] || 'https://api.cloud.prisma.sh'; @@ -7,4 +7,4 @@ export const clusterEndpointMap: { [key: string]: string } = { 'prisma-us1': 'https://us1.prisma.sh', }; -export const clusterEndpointMapReverse: { [key: string]: string } = invert(clusterEndpointMap); +export const clusterEndpointMapReverse: { [key: string]: string } = _.invert(clusterEndpointMap); diff --git a/packages/loaders/url/src/index.ts b/packages/loaders/url/src/index.ts index 2fa8cda6872..2b13a5ee38f 100644 --- a/packages/loaders/url/src/index.ts +++ b/packages/loaders/url/src/index.ts @@ -43,15 +43,10 @@ export type SyncResponse = Omit & { }; export type FetchFn = AsyncFetchFn | SyncFetchFn; -type Headers = - | Record - | Array> - | ((executionParams: ExecutionParams) => Array> | Record); - type BuildExecutorOptions = { pointer: string; fetch: TFetchFn; - extraHeaders?: Maybe; + extraHeaders?: HeadersConfig; defaultMethod: 'GET' | 'POST'; useGETForQueries?: Maybe; multipart?: Maybe; @@ -67,19 +62,25 @@ const syncImport: SyncImportFn = (moduleName: string) => require(moduleName); interface ExecutionResult { errors?: ReadonlyArray; - data?: TData | null; + data?: TData; extensions?: TExtensions; } interface ExecutionPatchResult { errors?: ReadonlyArray; - data?: TData | null; + data?: TData; path?: ReadonlyArray; label?: string; hasNext: boolean; extensions?: TExtensions; } +type HeadersConfig = Record; + +interface ExecutionExtensions { + headers?: HeadersConfig; +} + /** * Additional options for loading from a URL */ @@ -87,7 +88,7 @@ export interface LoadFromUrlOptions extends SingleFileOptions, Partial { return !!isWebUri(pointer); } - async createFormDataFromVariables({ query, variables }: { query: string; variables: TVariables }) { + async createFormDataFromVariables({ + query, + variables, + operationName, + extensions, + }: { + query: string; + variables: TVariables; + operationName?: string; + extensions?: any; + }) { const vars = Object.assign({}, variables); const { clone, files } = extractFiles( vars, @@ -175,6 +186,8 @@ export class UrlLoader implements DocumentLoader { JSON.stringify({ query, variables: clone, + operationName, + extensions, }) ); form.append('map', JSON.stringify(map)); @@ -211,11 +224,13 @@ export class UrlLoader implements DocumentLoader { query, variables, operationName, + extensions, }: { baseUrl: string; query: string; variables: any; operationName?: string; + extensions?: any; }) { const HTTP_URL = switchProtocols(baseUrl, { wss: 'https', @@ -235,12 +250,15 @@ export class UrlLoader implements DocumentLoader { if (operationName) { urlObj.searchParams.set('operationName', operationName); } + if (extensions) { + urlObj.searchParams.set('extensions', JSON.stringify(extensions)); + } const finalUrl = urlObj.toString().replace(dummyHostname, ''); return finalUrl; } - buildExecutor(options: BuildExecutorOptions): SyncExecutor; - buildExecutor(options: BuildExecutorOptions): AsyncExecutor; + buildExecutor(options: BuildExecutorOptions): SyncExecutor; + buildExecutor(options: BuildExecutorOptions): AsyncExecutor; buildExecutor({ pointer, fetch, @@ -248,12 +266,17 @@ export class UrlLoader implements DocumentLoader { defaultMethod, useGETForQueries, multipart, - }: BuildExecutorOptions): Executor { + }: BuildExecutorOptions): Executor { const HTTP_URL = switchProtocols(pointer, { wss: 'https', ws: 'http', }); - const executor = ({ document, variables, ...rest }: ExecutionParams) => { + const executor = ({ + document, + variables, + operationName, + extensions, + }: ExecutionParams) => { const controller = new AbortController(); let method = defaultMethod; if (useGETForQueries) { @@ -267,17 +290,13 @@ export class UrlLoader implements DocumentLoader { } } - const headers = this.getHeadersFromOptions(extraHeaders, { - document, - variables, - ...rest, - }); + const headers = Object.assign({}, extraHeaders, extensions?.headers || {}); let fetchResult: SyncResponse | Promise; const query = print(document); switch (method) { case 'GET': - const finalUrl = this.prepareGETUrl({ baseUrl: pointer, query, variables }); + const finalUrl = this.prepareGETUrl({ baseUrl: pointer, query, variables, operationName, extensions }); fetchResult = fetch(finalUrl, { method: 'GET', credentials: 'include', @@ -289,7 +308,7 @@ export class UrlLoader implements DocumentLoader { break; case 'POST': if (multipart) { - fetchResult = this.createFormDataFromVariables({ query, variables }).then(form => + fetchResult = this.createFormDataFromVariables({ query, variables, operationName, extensions }).then(form => (fetch as AsyncFetchFn)(HTTP_URL, { method: 'POST', credentials: 'include', @@ -308,6 +327,8 @@ export class UrlLoader implements DocumentLoader { body: JSON.stringify({ query, variables, + operationName, + extensions, }), headers: { accept: 'application/json, multipart/mixed', @@ -376,7 +397,7 @@ export class UrlLoader implements DocumentLoader { connectionParams, lazy: true, }); - return async ({ document, variables }) => { + return async ({ document, variables, operationName, extensions }) => { const query = print(document); return observableToAsyncIterable({ subscribe: observer => { @@ -384,6 +405,8 @@ export class UrlLoader implements DocumentLoader { { query, variables: variables as Record, + operationName, + extensions, }, observer ); @@ -413,11 +436,12 @@ export class UrlLoader implements DocumentLoader { webSocketImpl ); - return async ({ document, variables }: ExecutionParams) => { + return async ({ document, variables, operationName }: ExecutionParams) => { return observableToAsyncIterable( subscriptionClient.request({ query: document, variables, + operationName, }) ) as AsyncIterableIterator>; }; @@ -425,21 +449,17 @@ export class UrlLoader implements DocumentLoader { buildSSESubscriber( pointer: string, - extraHeaders: Maybe, + extraHeaders: HeadersConfig | undefined, fetch: AsyncFetchFn, options: Maybe - ): Subscriber { - return async ({ document, variables, ...rest }) => { + ): Subscriber { + return async ({ document, variables, extensions }) => { const controller = new AbortController(); const query = print(document); const finalUrl = this.prepareGETUrl({ baseUrl: pointer, query, variables }); - const headers = this.getHeadersFromOptions(extraHeaders, { - document, - variables, - ...rest, - }); return observableToAsyncIterable({ subscribe: observer => { + const headers = Object.assign({}, extraHeaders || {}, extensions?.headers || {}); fetchEventSource(finalUrl, { credentials: 'include', headers, @@ -509,24 +529,6 @@ export class UrlLoader implements DocumentLoader { return async ? (typeof fetch === 'undefined' ? crossFetch : fetch) : syncFetch; } - private getHeadersFromOptions( - customHeaders: Maybe, - executionParams: ExecutionParams - ): Record { - let headers = {}; - if (customHeaders) { - if (typeof customHeaders === 'function') { - customHeaders = customHeaders(executionParams); - } - if (Array.isArray(customHeaders)) { - headers = customHeaders.reduce((prev: any, v: any) => ({ ...prev, ...v }), {}); - } else if (typeof customHeaders === 'object') { - headers = customHeaders; - } - } - return headers; - } - private getDefaultMethodFromOptions(method: LoadFromUrlOptions['method'], defaultMethod: 'GET' | 'POST') { if (method) { defaultMethod = method; @@ -579,7 +581,7 @@ export class UrlLoader implements DocumentLoader { subscriber = this.buildSSESubscriber(subscriptionsEndpoint, options.headers, fetch, options.eventSourceOptions); } else { const webSocketImpl = await this.getWebSocketImpl(options, asyncImport); - const connectionParams = () => ({ headers: this.getHeadersFromOptions(options.headers, {} as any) }); + const connectionParams = () => ({ headers: options.headers }); if (options.useWebSocketLegacyProtocol) { subscriber = this.buildWSLegacySubscriber(subscriptionsEndpoint, webSocketImpl, connectionParams); } else { @@ -621,7 +623,7 @@ export class UrlLoader implements DocumentLoader { ); } else { const webSocketImpl = this.getWebSocketImpl(options, syncImport); - const connectionParams = () => ({ headers: this.getHeadersFromOptions(options.headers, {} as any) }); + const connectionParams = () => ({ headers: options.headers }); if (options.useWebSocketLegacyProtocol) { subscriber = this.buildWSLegacySubscriber(subscriptionsEndpoint, webSocketImpl, connectionParams); } else { @@ -655,11 +657,10 @@ export class UrlLoader implements DocumentLoader { async handleSDLAsync(pointer: SchemaPointerSingle, options: LoadFromUrlOptions): Promise { const fetch = await this.getFetch(options?.customFetch, asyncImport, true); - const headers = this.getHeadersFromOptions(options?.headers, {} as any); const defaultMethod = this.getDefaultMethodFromOptions(options?.method, 'GET'); const response = await fetch(pointer, { method: defaultMethod, - headers, + headers: options.headers, }); const schemaString = await response.text(); return parseGraphQLSDL(pointer, schemaString, options); @@ -667,11 +668,10 @@ export class UrlLoader implements DocumentLoader { handleSDLSync(pointer: SchemaPointerSingle, options: LoadFromUrlOptions): Source { const fetch = this.getFetch(options?.customFetch, syncImport, false); - const headers = this.getHeadersFromOptions(options?.headers, {} as any); const defaultMethod = this.getDefaultMethodFromOptions(options?.method, 'GET'); const response = fetch(pointer, { method: defaultMethod, - headers, + headers: options.headers, }); const schemaString = response.text(); return parseGraphQLSDL(pointer, schemaString, options); @@ -709,8 +709,7 @@ export class UrlLoader implements DocumentLoader { } function switchProtocols(pointer: string, protocolMap: Record): string { - const protocols: [string, string][] = Object.keys(protocolMap).map(source => [source, protocolMap[source]]); - return protocols.reduce( + return Object.entries(protocolMap).reduce( (prev, [source, target]) => prev.replace(`${source}://`, `${target}://`).replace(`${source}:\\`, `${target}:\\`), pointer ); diff --git a/packages/loaders/url/tests/url-loader.spec.ts b/packages/loaders/url/tests/url-loader.spec.ts index 1f6687c7bfb..ff8d599dfc6 100644 --- a/packages/loaders/url/tests/url-loader.spec.ts +++ b/packages/loaders/url/tests/url-loader.spec.ts @@ -156,7 +156,7 @@ input TestInput { assertNonMaybe(source.schema) expect(printSchemaWithDirectives(source.schema)).toBeSimilarGqlDoc(testTypeDefs); - expect(Array.isArray(headers.accept) ? headers.accept.join(',') : headers.accept).toContain(`application/json`); + expect(Array.isArray(headers['accept']) ? headers['accept'].join(',') : headers['accept']).toContain(`application/json`); expect(headers['content-type']).toContain(`application/json`); }); @@ -177,32 +177,9 @@ input TestInput { assertNonMaybe(source.schema) expect(printSchemaWithDirectives(source.schema)).toBeSimilarGqlDoc(testTypeDefs); - expect(Array.isArray(headers.accept) ? headers.accept.join(',') : headers.accept).toContain(`application/json`); + expect(Array.isArray(headers['accept']) ? headers['accept'].join(',') : headers['accept']).toContain(`application/json`); expect(headers['content-type']).toContain(`application/json`); - expect(headers.auth).toContain(`1`); - }); - - it('Should pass extra headers when they are specified as array', async () => { - let headers: Record = {}; - scope = mockGraphQLServer({ - schema: testSchema, - host: testHost, - path: testPathChecker, - intercept(ctx) { - headers = ctx.req.headers; - }, - }); - const source = await loader.load(testUrl, { headers: [{ A: '1' }, { B: '2', C: '3' }] }); - - expect(source).toBeDefined(); - assertNonMaybe(source.schema) - expect(printSchemaWithDirectives(source.schema)).toBeSimilarGqlDoc(testTypeDefs); - - expect(Array.isArray(headers.accept) ? headers.accept.join(',') : headers.accept).toContain(`application/json`); - expect(headers['content-type']).toContain(`application/json`); - expect(headers.a).toContain(`1`); - expect(headers.b).toContain(`2`); - expect(headers.c).toContain(`3`); + expect(headers['auth']).toContain(`1`); }); it('Should utilize extra introspection options', async () => { @@ -247,7 +224,7 @@ input TestInput { expect(result?.errors).toBeFalsy(); - expect(result?.data?.a).toBe(testVariableValue); + expect(result?.data?.['a']).toBe(testVariableValue); }); it('Should preserve "ws" and "http" in the middle of a pointer', async () => { @@ -488,8 +465,8 @@ input TestInput { expect(result.errors).toBeFalsy(); assertNonMaybe(result.data) - expect(result.data.uploadFile?.filename).toBe(fileName); - expect(result.data.uploadFile?.content).toBe(content); + expect(result.data['uploadFile']?.filename).toBe(fileName); + expect(result.data['uploadFile']?.content).toBe(content); }); }); }); diff --git a/packages/merge/src/merge-resolvers.ts b/packages/merge/src/merge-resolvers.ts index 2173eba5911..1f7b26b27fa 100644 --- a/packages/merge/src/merge-resolvers.ts +++ b/packages/merge/src/merge-resolvers.ts @@ -1,8 +1,5 @@ import { IResolvers, Maybe, mergeDeep } from '@graphql-tools/utils'; -export type ResolversFactory = (...args: any[]) => IResolvers; -export type ResolversDefinition = IResolvers | ResolversFactory; - /** * Additional options for merging resolvers */ @@ -39,46 +36,35 @@ export interface MergeResolversOptions { * const resolvers = mergeResolvers(resolversArray) * ``` */ -export function mergeResolvers>( - resolversDefinitions: Maybe, +export function mergeResolvers( + resolversDefinitions: Maybe> | Maybe>[]>, options?: MergeResolversOptions -): T { - if (!resolversDefinitions || resolversDefinitions.length === 0) { - return {} as T; +): IResolvers { + if (!resolversDefinitions || (Array.isArray(resolversDefinitions) && resolversDefinitions.length === 0)) { + return {}; + } + + if (!Array.isArray(resolversDefinitions)) { + return resolversDefinitions; } if (resolversDefinitions.length === 1) { - const singleDefinition = resolversDefinitions[0]; - if (Array.isArray(singleDefinition)) { - return mergeResolvers(singleDefinition); - } - return singleDefinition; + return resolversDefinitions[0] || {}; } - type TFactory = (...args: any[]) => T; - const resolversFactories = new Array(); - const resolvers = new Array(); + const resolvers = new Array>(); for (let resolversDefinition of resolversDefinitions) { if (Array.isArray(resolversDefinition)) { resolversDefinition = mergeResolvers(resolversDefinition); } - if (typeof resolversDefinition === 'function') { - resolversFactories.push(resolversDefinition as unknown as TFactory); - } else if (typeof resolversDefinition === 'object') { + if (typeof resolversDefinition === 'object' && resolversDefinition) { resolvers.push(resolversDefinition); } } - let result: T = {} as T; - if (resolversFactories.length) { - result = ((...args: any[]) => { - const resultsOfFactories = resolversFactories.map(factory => factory(...args)); - return resolvers.concat(resultsOfFactories).reduce(mergeDeep, {}); - }) as any; - } else { - result = resolvers.reduce(mergeDeep, {} as T); - } - if (options && options.exclusions) { + const result = resolvers.reduce(mergeDeep, {}); + + if (options?.exclusions) { for (const exclusion of options.exclusions) { const [typeName, fieldName] = exclusion.split('.'); if (!fieldName || fieldName === '*') { diff --git a/packages/merge/src/merge-schemas.ts b/packages/merge/src/merge-schemas.ts index 5fd180946b6..b20181e3dba 100644 --- a/packages/merge/src/merge-schemas.ts +++ b/packages/merge/src/merge-schemas.ts @@ -1,13 +1,13 @@ import { GraphQLSchema, DocumentNode, buildASTSchema, BuildSchemaOptions, buildSchema } from 'graphql'; -import { addResolversToSchema, addErrorLoggingToSchema, ILogger } from '@graphql-tools/schema'; +import { addResolversToSchema } from '@graphql-tools/schema'; import { mergeTypeDefs, Config } from './typedefs-mergers/merge-typedefs'; import { mergeResolvers } from './merge-resolvers'; import { IResolvers, - SchemaDirectiveVisitor, IResolverValidationOptions, asArray, getResolversFromSchema, + TypeSource, } from '@graphql-tools/utils'; import { mergeExtensions, extractExtensionsFromSchema, applyExtensions, SchemaExtensions } from './extensions'; @@ -22,23 +22,15 @@ export interface MergeSchemasConfig e /** * Additional type definitions to also merge */ - typeDefs?: (DocumentNode | string)[] | DocumentNode | string; + typeDefs?: TypeSource; /** * Additional resolvers to also merge */ resolvers?: Resolvers | Resolvers[]; - /** - * Schema directives to apply to the type definitions being merged, if provided - */ - schemaDirectives?: { [directiveName: string]: typeof SchemaDirectiveVisitor }; /** * Options to validate the resolvers being merged, if provided */ resolverValidationOptions?: IResolverValidationOptions; - /** - * Custom logger instance - */ - logger?: ILogger; } const defaultResolverValidationOptions: Partial = { @@ -54,7 +46,7 @@ const defaultResolverValidationOptions: Partial = { * @param config Configuration object */ export function mergeSchemas(config: MergeSchemasConfig) { - const typeDefs = mergeTypes(config); + const typeDefs = mergeTypeDefs([config.schemas, config.typeDefs || []], config); const extractedResolvers: IResolvers[] = []; const extractedExtensions: SchemaExtensions[] = []; for (const schema of config.schemas) { @@ -75,7 +67,7 @@ export function mergeSchemas(config: MergeSchemasConfig) { */ export async function mergeSchemasAsync(config: MergeSchemasConfig) { const [typeDefs, resolvers, extensions] = await Promise.all([ - mergeTypes(config), + mergeTypeDefs([config.schemas, config.typeDefs || []], config), Promise.all(config.schemas.map(async schema => getResolversFromSchema(schema))).then(extractedResolvers => mergeResolvers([...extractedResolvers, ...ensureResolvers(config)], config) ), @@ -87,10 +79,6 @@ export async function mergeSchemasAsync(config: MergeSchemasConfig) { return makeSchema({ resolvers, typeDefs, extensions }, config); } -function mergeTypes({ schemas, typeDefs, ...config }: MergeSchemasConfig) { - return mergeTypeDefs([...schemas, ...(typeDefs ? asArray(typeDefs) : [])], config); -} - function ensureResolvers(config: MergeSchemasConfig) { return config.resolvers ? asArray(config.resolvers) : []; } @@ -117,16 +105,6 @@ function makeSchema( }); } - // use logger - if (config.logger) { - schema = addErrorLoggingToSchema(schema, config.logger); - } - - // use schema directives - if (config.schemaDirectives) { - SchemaDirectiveVisitor.visitSchemaDirectives(schema, config.schemaDirectives); - } - // extensions applyExtensions(schema, extensions); diff --git a/packages/merge/src/typedefs-mergers/comments.ts b/packages/merge/src/typedefs-mergers/comments.ts index dade851edc9..fdd80c9d54f 100644 --- a/packages/merge/src/typedefs-mergers/comments.ts +++ b/packages/merge/src/typedefs-mergers/comments.ts @@ -2,7 +2,6 @@ import { getDescription, StringValueNode, FieldDefinitionNode, - InputValueDefinitionNode, ASTNode, NameNode, TypeNode, @@ -29,24 +28,26 @@ export function collectComment(node: NamedDefinitionNode): void { switch (node.kind) { case 'EnumTypeDefinition': - node.values?.forEach(value => { - pushComment(value, entityName, value.name.value); - }); + if (node.values) { + for (const value of node.values) { + pushComment(value, entityName, value.name.value); + } + } break; case 'ObjectTypeDefinition': case 'InputObjectTypeDefinition': case 'InterfaceTypeDefinition': if (node.fields) { - node.fields.forEach((field: FieldDefinitionNode | InputValueDefinitionNode) => { + for (const field of node.fields) { pushComment(field, entityName, field.name.value); if (isFieldDefinitionNode(field) && field.arguments) { - field.arguments.forEach(arg => { + for (const arg of field.arguments) { pushComment(arg, entityName, field.name.value, arg.name.value); - }); + } } - }); + } } break; } diff --git a/packages/merge/src/typedefs-mergers/merge-typedefs.ts b/packages/merge/src/typedefs-mergers/merge-typedefs.ts index 2b4a6707ee6..9ccd5112993 100644 --- a/packages/merge/src/typedefs-mergers/merge-typedefs.ts +++ b/packages/merge/src/typedefs-mergers/merge-typedefs.ts @@ -1,24 +1,28 @@ import { DefinitionNode, DocumentNode, - GraphQLSchema, parse, - Source, Kind, isSchema, OperationTypeDefinitionNode, OperationTypeNode, isDefinitionNode, + ParseOptions, } from 'graphql'; import { CompareFn, defaultStringComparator, isSourceTypes, isStringTypes } from './utils'; import { MergedResultMap, mergeGraphQLNodes, schemaDefSymbol } from './merge-nodes'; import { resetComments, printWithComments } from './comments'; -import { getDocumentNodeFromSchema } from '@graphql-tools/utils'; +import { + getDocumentNodeFromSchema, + GetDocumentNodeFromSchemaOptions, + isDocumentNode, + TypeSource, +} from '@graphql-tools/utils'; import { DEFAULT_OPERATION_TYPE_NAME_MAP } from './schema-def'; type Omit = Pick>; -export interface Config { +export interface Config extends ParseOptions, GetDocumentNodeFromSchemaOptions { /** * Produces `schema { query: ..., mutation: ..., subscription: ... }` * @@ -76,24 +80,18 @@ export interface Config { * Merges multiple type definitions into a single `DocumentNode` * @param types The type definitions to be merged */ -export function mergeTypeDefs(types: Array): DocumentNode; -export function mergeTypeDefs( - types: Array, - config?: Partial & { commentDescriptions: true } -): string; +export function mergeTypeDefs(typeSource: TypeSource): DocumentNode; +export function mergeTypeDefs(typeSource: TypeSource, config?: Partial & { commentDescriptions: true }): string; export function mergeTypeDefs( - types: Array, + typeSource: TypeSource, config?: Omit, 'commentDescriptions'> ): DocumentNode; -export function mergeTypeDefs( - types: Array, - config?: Partial -): DocumentNode | string { +export function mergeTypeDefs(typeSource: TypeSource, config?: Partial): DocumentNode | string { resetComments(); const doc = { kind: Kind.DOCUMENT, - definitions: mergeGraphQLTypes(types, { + definitions: mergeGraphQLTypes(typeSource, { useSchemaDefinition: true, forceSchemaDefinition: false, throwOnConflict: false, @@ -116,36 +114,40 @@ export function mergeTypeDefs( } function visitTypeSources( - types: Array, - allNodes: DefinitionNode[] = [] + typeSource: TypeSource, + options: ParseOptions & GetDocumentNodeFromSchemaOptions, + allNodes: DefinitionNode[] = [], + visitedTypeSources = new Set() ) { - for (const type of types) { - if (type) { - if (Array.isArray(type)) { - visitTypeSources(type, allNodes); - } else if (isSchema(type)) { - const documentNode = getDocumentNodeFromSchema(type); - visitTypeSources(documentNode.definitions as DefinitionNode[], allNodes); - } else if (isStringTypes(type) || isSourceTypes(type)) { - const documentNode = parse(type); - visitTypeSources(documentNode.definitions as DefinitionNode[], allNodes); - } else if (isDefinitionNode(type)) { - allNodes.push(type); - } else { - visitTypeSources(type.definitions as DefinitionNode[], allNodes); + if (typeSource && !visitedTypeSources.has(typeSource)) { + visitedTypeSources.add(typeSource); + if (typeof typeSource === 'function') { + visitTypeSources(typeSource(), options, allNodes, visitedTypeSources); + } else if (Array.isArray(typeSource)) { + for (const type of typeSource) { + visitTypeSources(type, options, allNodes, visitedTypeSources); } + } else if (isSchema(typeSource)) { + const documentNode = getDocumentNodeFromSchema(typeSource, options); + visitTypeSources(documentNode.definitions as DefinitionNode[], options, allNodes, visitedTypeSources); + } else if (isStringTypes(typeSource) || isSourceTypes(typeSource)) { + const documentNode = parse(typeSource, options); + visitTypeSources(documentNode.definitions as DefinitionNode[], options, allNodes, visitedTypeSources); + } else if (typeof typeSource === 'object' && isDefinitionNode(typeSource)) { + allNodes.push(typeSource); + } else if (isDocumentNode(typeSource)) { + visitTypeSources(typeSource.definitions as DefinitionNode[], options, allNodes, visitedTypeSources); + } else { + throw new Error(`typeDefs must contain only strings, documents, schemas, or functions, got ${typeof typeSource}`); } } return allNodes; } -export function mergeGraphQLTypes( - types: Array, - config: Config -): DefinitionNode[] { +export function mergeGraphQLTypes(typeSource: TypeSource, config: Config): DefinitionNode[] { resetComments(); - const allNodes = visitTypeSources(types); + const allNodes = visitTypeSources(typeSource, config); const mergedNodes: MergedResultMap = mergeGraphQLNodes(allNodes, config); diff --git a/packages/merge/tests/extract-extensions-from-schema.spec.ts b/packages/merge/tests/extract-extensions-from-schema.spec.ts index 1823fec4d7b..4e360fa3f12 100644 --- a/packages/merge/tests/extract-extensions-from-schema.spec.ts +++ b/packages/merge/tests/extract-extensions-from-schema.spec.ts @@ -1,7 +1,7 @@ -import { buildSchema, GraphQLEnumType, GraphQLSchema, printSchema, buildClientSchema, buildASTSchema, parse } from 'graphql'; +import { buildSchema, GraphQLSchema, printSchema, buildASTSchema, parse } from 'graphql'; +import { assertGraphQLEnumType, assertGraphQLInputObjectType, assertGraphQLObjectType, assertGraphQLInterfaceType, assertGraphQLUnionType, assertGraphQLScalerType } from '../../testing/assertion'; import { assertSome } from '@graphql-tools/utils'; import { extractExtensionsFromSchema, mergeExtensions, applyExtensions } from '../src/extensions' -import { assertGraphQLEnumType, assertGraphQLInputObjectType, assertGraphQLInterfaceType, assertGraphQLObjectType, assertGraphQLScalerType, assertGraphQLUnionType } from '../../testing/assertion'; describe('extensions', () => { let schema: GraphQLSchema; @@ -69,25 +69,25 @@ describe('extensions', () => { MyScalar.extensions = { scalar: true }; const { types: extensions } = extractExtensionsFromSchema(schema); - expect(extensions.MyInput.extensions).toEqual({ input: true }) - expect(extensions.MyType.extensions).toEqual({ type: true }) - expect(extensions.Node.extensions).toEqual({ interface: true }) - expect(extensions.MyEnum.extensions).toEqual({ enum: true }) - expect(extensions.MyUnion.extensions).toEqual({ union: true }) - expect(extensions.MyScalar.extensions).toEqual({ scalar: true }) + expect(extensions['MyInput'].extensions).toEqual({ input: true }) + expect(extensions['MyType'].extensions).toEqual({ type: true }) + expect(extensions['Node'].extensions).toEqual({ interface: true }) + expect(extensions['MyEnum'].extensions).toEqual({ enum: true }) + expect(extensions['MyUnion'].extensions).toEqual({ union: true }) + expect(extensions['MyScalar'].extensions).toEqual({ scalar: true }) }); it('Should extract extensions correctly for fields arguments', () => { const queryType = schema.getQueryType() assertSome(queryType) - queryType.getFields().t.args[0].extensions = { fieldArg: true }; + queryType.getFields()['t'].args[0].extensions = { fieldArg: true }; const { types: extensions } = extractExtensionsFromSchema(schema); - if (extensions.Query.type !== "object") { + if (extensions['Query'].type !== "object") { throw new Error("Unexpected type.") } - expect(extensions.Query.fields.t.arguments.i).toEqual({ fieldArg: true }) + expect(extensions['Query'].fields['t'].arguments['i']).toEqual({ fieldArg: true }) }); it('Should extract extensions correctly for enum values', () => { @@ -95,37 +95,37 @@ describe('extensions', () => { assertGraphQLEnumType(MyEnum) MyEnum.getValues()[0].extensions = { enumValue: true }; - const { types: extensions } = extractExtensionsFromSchema(schema); - if (extensions.MyEnum.type !== "enum") { - throw new Error("Unexpected type.") - } - expect(extensions.MyEnum.values.A).toEqual({ enumValue: true }); - expect(extensions.MyEnum.values.B).toEqual({}); - expect(extensions.MyEnum.values.C).toEqual({}); + const { types: extensions } = extractExtensionsFromSchema(schema); + if (extensions['MyEnum'].type !== "enum") { + throw new Error("Unexpected type.") + } + expect(extensions['MyEnum'].values['A']).toEqual({ enumValue: true }); + expect(extensions['MyEnum'].values['B']).toEqual({}); + expect(extensions['MyEnum'].values['C']).toEqual({}); }); it('Should extract extensions correctly for fields', () => { const queryType = schema.getQueryType() assertSome(queryType) - queryType.getFields().t.extensions = { field: true }; + queryType.getFields()['t'].extensions = { field: true }; const { types: extensions } = extractExtensionsFromSchema(schema); - if (extensions.Query.type !== "object") { + if (extensions['Query'].type !== "object") { throw new Error("Unexpected type.") } - expect(extensions.Query.fields.t.extensions).toEqual({ field: true }) + expect(extensions['Query'].fields['t'].extensions).toEqual({ field: true }) }); it('Should extract extensions correctly for input fields', () => { const MyInput = schema.getType('MyInput') -assertGraphQLInputObjectType(MyInput) - MyInput.getFields().foo.extensions = { inputField: true }; + assertGraphQLInputObjectType(MyInput) + MyInput.getFields()['foo'].extensions = { inputField: true }; - const { types: extensions } = extractExtensionsFromSchema(schema); - if (extensions.MyInput.type !== "input") { - throw new Error("Unexpected type.") - } - expect(extensions.MyInput.fields.foo.extensions).toEqual({ inputField: true }) + const { types: extensions } = extractExtensionsFromSchema(schema); + if (extensions['MyInput'].type !== "input") { + throw new Error("Unexpected type.") + } + expect(extensions['MyInput'].fields['foo'].extensions).toEqual({ inputField: true }) }); }); @@ -146,7 +146,7 @@ assertGraphQLInputObjectType(MyInput) const extensions = extractExtensionsFromSchema(schema); const secondExtensions = extractExtensionsFromSchema(secondSchema); const mergedExtensions = mergeExtensions([extensions, secondExtensions]); - expect(mergedExtensions.types.Query.extensions).toEqual({ queryTest: true, querySecondTest: true }) + expect(mergedExtensions.types['Query'].extensions).toEqual({ queryTest: true, querySecondTest: true }) }) }); @@ -171,12 +171,12 @@ assertGraphQLInputObjectType(MyInput) let MyScalar = schema.getType('MyScalar') assertGraphQLScalerType(MyScalar) MyScalar.extensions = { scalar: true }; - MyInput.getFields().foo.extensions = { inputField: true }; + MyInput.getFields()['foo'].extensions = { inputField: true }; let QueryType = schema.getQueryType(); assertSome(QueryType) - QueryType.getFields().t.extensions = { field: true }; + QueryType.getFields()['t'].extensions = { field: true }; MyEnum.getValues()[0].extensions = { enumValue: true }; - QueryType.getFields().t.args[0].extensions = { fieldArg: true }; + QueryType.getFields()['t'].args[0].extensions = { fieldArg: true }; const result = extractExtensionsFromSchema(schema); const cleanSchema = buildASTSchema(parse(printSchema(schema))); @@ -207,12 +207,12 @@ assertGraphQLInputObjectType(MyInput) MyScalar = modifiedSchema.getType('MyScalar') assertGraphQLScalerType(MyScalar) expect(MyScalar.extensions).toEqual({ scalar: true }); - expect(MyInput.getFields().foo.extensions).toEqual({ inputField: true }); + expect(MyInput.getFields()['foo'].extensions).toEqual({ inputField: true }); QueryType = modifiedSchema.getQueryType(); assertSome(QueryType) - expect(QueryType.getFields().t.extensions).toEqual({ field: true }); + expect(QueryType.getFields()['t'].extensions).toEqual({ field: true }); expect(MyEnum.getValues()[0].extensions).toEqual({ enumValue: true }); - expect(QueryType.getFields().t.args[0].extensions).toEqual({ fieldArg: true }); + expect(QueryType.getFields()['t'].args[0].extensions).toEqual({ fieldArg: true }); }); }) }); diff --git a/packages/merge/tests/merge-nodes.spec.ts b/packages/merge/tests/merge-nodes.spec.ts index 859851612f8..2ca7fbaa83a 100644 --- a/packages/merge/tests/merge-nodes.spec.ts +++ b/packages/merge/tests/merge-nodes.spec.ts @@ -1,6 +1,6 @@ import { mergeGraphQLNodes } from '../src'; -import { parse, InputObjectTypeDefinitionNode } from 'graphql'; -import { assertEnumTypeDefinitionNode, assertInputObjectTypeDefinitionNode, assertInterfaceTypeDefinitionNode, assertNamedTypeNode, assertObjectTypeDefinitionNode, assertScalarTypeDefinitionNode, assertUnionTypeDefinitionNode } from '../../testing/assertion'; +import { parse } from 'graphql'; +import { assertEnumTypeDefinitionNode, assertInputObjectTypeDefinitionNode, assertNamedTypeNode, assertObjectTypeDefinitionNode, assertScalarTypeDefinitionNode, assertUnionTypeDefinitionNode } from '../../testing/assertion'; import { assertSome } from '@graphql-tools/utils'; describe('Merge Nodes', () => { @@ -9,7 +9,7 @@ describe('Merge Nodes', () => { const type1 = parse(`type A { f1: String }`); const type2 = parse(`type A`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const type = merged.A; + const type = merged['A']; assertObjectTypeDefinitionNode(type) assertSome(type.fields) expect(type.fields.length).toBe(1); @@ -22,7 +22,7 @@ describe('Merge Nodes', () => { const type1 = parse(`type A { f1: String }`); const type2 = parse(`type A { f2: Int }`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const type = merged.A; + const type = merged['A']; assertObjectTypeDefinitionNode(type) assertSome(type.fields) @@ -39,7 +39,7 @@ describe('Merge Nodes', () => { const type1 = parse(`type A { f1: String }`); const type2 = parse(`type A { f1: String, f2: Int}`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const type = merged.A; + const type = merged['A']; assertObjectTypeDefinitionNode(type) assertSome(type.fields) @@ -56,7 +56,7 @@ describe('Merge Nodes', () => { const type1 = parse(`interface Base { f1: String } type A implements Base { f1: String }`); const type2 = parse(`interface Base { f1: String } type A implements Base { f2: Int}`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const type = merged.A; + const type = merged['A']; assertObjectTypeDefinitionNode(type) assertSome(type.interfaces) @@ -68,7 +68,7 @@ describe('Merge Nodes', () => { const type1 = parse(`interface Base { f1: String } type A implements Base { f1: String }`); const type2 = parse(`type A { f2: Int}`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const type = merged.A; + const type = merged['A']; assertObjectTypeDefinitionNode(type) assertSome(type.interfaces) @@ -80,7 +80,7 @@ describe('Merge Nodes', () => { const type1 = parse(`type A @test { f1: String }`); const type2 = parse(`type A { f2: Int}`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const type = merged.A; + const type = merged['A']; assertObjectTypeDefinitionNode(type) assertSome(type.directives) @@ -92,7 +92,7 @@ describe('Merge Nodes', () => { const type1 = parse(`type A @test { f1: String }`); const type2 = parse(`type A @other { f2: Int}`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const type = merged.A; + const type = merged['A']; assertObjectTypeDefinitionNode(type) assertSome(type.directives) @@ -105,7 +105,7 @@ describe('Merge Nodes', () => { const type1 = parse(`type A @test { f1: String }`); const type2 = parse(`type A @test { f2: Int}`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const type = merged.A; + const type = merged['A']; assertObjectTypeDefinitionNode(type) assertSome(type.directives) @@ -117,7 +117,7 @@ describe('Merge Nodes', () => { const type1 = parse(`type A @test { f1: String }`); const type2 = parse(`type A @test2 { f2: Int}`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const type = merged.A; + const type = merged['A']; assertObjectTypeDefinitionNode(type) assertSome(type.directives) @@ -132,7 +132,7 @@ describe('Merge Nodes', () => { const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions], { reverseDirectives: true, }); - const type = merged.A; + const type = merged['A']; assertObjectTypeDefinitionNode(type) assertSome(type.directives) @@ -145,7 +145,7 @@ describe('Merge Nodes', () => { const type1 = parse(`interface Base1 { f1: String } type A implements Base1 { f1: String }`); const type2 = parse(`interface Base2 { f2: Int } type A implements Base2 { f2: Int}`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const type = merged.A; + const type = merged['A']; assertObjectTypeDefinitionNode(type) assertSome(type.interfaces) @@ -168,7 +168,7 @@ describe('Merge Nodes', () => { const type1 = parse(`enum A { T }`); const type2 = parse(`enum A { S }`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const result = merged.A + const result = merged['A'] assertEnumTypeDefinitionNode(result) assertSome(result.values) expect(result.values.length).toBe(2); @@ -180,7 +180,7 @@ describe('Merge Nodes', () => { const type1 = parse(`enum A { T }`); const type2 = parse(`enum A { T }`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const result = merged.A; + const result = merged['A']; assertEnumTypeDefinitionNode(result) assertSome(result.values) @@ -192,7 +192,7 @@ describe('Merge Nodes', () => { const type1 = parse(`enum A @test { T }`); const type2 = parse(`enum A @test2 { T }`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const result = merged.A; + const result = merged['A']; assertEnumTypeDefinitionNode(result) assertSome(result.directives) @@ -205,7 +205,7 @@ describe('Merge Nodes', () => { const type1 = parse(`enum A @test { T }`); const type2 = parse(`enum A { S }`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const result = merged.A; + const result = merged['A']; assertEnumTypeDefinitionNode(result) assertSome(result.directives) @@ -219,7 +219,7 @@ describe('Merge Nodes', () => { const type1 = parse(`type A union C = A`); const type2 = parse(`type B union C = B`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const result = merged.C; + const result = merged['C']; assertUnionTypeDefinitionNode(result) assertSome(result.types) @@ -234,7 +234,7 @@ describe('Merge Nodes', () => { const type1 = parse(`scalar A`); const type2 = parse(`scalar A`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const result = merged.A; + const result = merged['A']; assertScalarTypeDefinitionNode(result) expect(result.name.value).toBe('A'); @@ -246,7 +246,7 @@ describe('Merge Nodes', () => { const type1 = parse(`input A { f1: String }`); const type2 = parse(`input A { f2: String }`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const result = merged.A; + const result = merged['A']; assertInputObjectTypeDefinitionNode(result) assertSome(result.fields) @@ -259,7 +259,7 @@ describe('Merge Nodes', () => { const type1 = parse(`input A { f1: String }`); const type2 = parse(`input A { f1: String! }`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const result = merged.A; + const result = merged['A']; assertInputObjectTypeDefinitionNode(result) assertSome(result.fields) @@ -274,7 +274,7 @@ describe('Merge Nodes', () => { const type1 = parse(`type Query { f1: String }`); const type2 = parse(`type Query { f2: String }`); const merged = mergeGraphQLNodes([...type1.definitions, ...type2.definitions]); - const type = merged.Query; + const type = merged['Query']; assertObjectTypeDefinitionNode(type) assertSome(type.fields) diff --git a/packages/merge/tests/merge-schemas.spec.ts b/packages/merge/tests/merge-schemas.spec.ts index 5b5b4de7c4a..03d5da68988 100644 --- a/packages/merge/tests/merge-schemas.spec.ts +++ b/packages/merge/tests/merge-schemas.spec.ts @@ -78,8 +78,8 @@ describe('Merge Schemas', () => { }); expect(errors).toBeFalsy(); assertSome(data) - expect(data.foo).toBe('FOO'); - expect(data.bar).toBe('BAR'); + expect(data['foo']).toBe('FOO'); + expect(data['bar']).toBe('BAR'); }); it('should merge two valid executable schemas async', async () => { const fooSchema = makeExecutableSchema({ @@ -119,8 +119,8 @@ describe('Merge Schemas', () => { }); expect(errors).toBeFalsy(); assertSome(data) - expect(data.foo).toBe('FOO'); - expect(data.bar).toBe('BAR'); + expect(data['foo']).toBe('FOO'); + expect(data['bar']).toBe('BAR'); }); it('should merge two valid executable schemas with extra resolvers', async () => { const fooSchema = makeExecutableSchema({ @@ -167,9 +167,9 @@ describe('Merge Schemas', () => { }); expect(errors).toBeFalsy(); assertSome(data) - expect(data.foo).toBe('FOO'); - expect(data.bar).toBe('BAR'); - expect(data.qux).toBe('QUX'); + expect(data['foo']).toBe('FOO'); + expect(data['bar']).toBe('BAR'); + expect(data['qux']).toBe('QUX'); }); it('should merge two valid executable schemas with extra typeDefs and resolvers', async () => { const fooSchema = makeExecutableSchema({ @@ -220,9 +220,9 @@ describe('Merge Schemas', () => { }); expect(errors).toBeFalsy(); assertSome(data) - expect(data.foo).toBe('FOO'); - expect(data.bar).toBe('BAR'); - expect(data.qux).toBe('QUX'); + expect(data['foo']).toBe('FOO'); + expect(data['bar']).toBe('BAR'); + expect(data['qux']).toBe('QUX'); }); it('should merge two valid schemas by keeping their directives to be used in extra typeDefs', async () => { const fooSchema = makeExecutableSchema({ @@ -274,9 +274,9 @@ describe('Merge Schemas', () => { }); expect(errors).toBeFalsy(); assertSome(data) - expect(data.foo).toBe('FOO'); - expect(data.bar).toBe('BAR'); - expect(data.qux).toBe('QUX'); + expect(data['foo']).toBe('FOO'); + expect(data['bar']).toBe('BAR'); + expect(data['qux']).toBe('QUX'); }); it('should merge valid schemas with interfaces correctly', async () => { const fooSchema = makeExecutableSchema({ @@ -345,10 +345,10 @@ describe('Merge Schemas', () => { }); expect(errors).toBeFalsy(); assertSome(data) - expect(data.bar.foo).toBe('foo'); - expect(data.bar.bar).toBe('bar'); - expect(data.qux.foo).toBe('foo'); - expect(data.qux.qux).toBe('qux'); + expect(data['bar'].foo).toBe('foo'); + expect(data['bar'].bar).toBe('bar'); + expect(data['qux'].foo).toBe('foo'); + expect(data['qux'].qux).toBe('qux'); }); it('should merge scalars (part of resolvers)', async () => { @@ -389,7 +389,7 @@ describe('Merge Schemas', () => { }); assertSome(dataA) - expect(dataA.a).toEqual(now.toISOString()); + expect(dataA['a']).toEqual(now.toISOString()); // merged schema const { data } = await graphql({ @@ -397,7 +397,7 @@ describe('Merge Schemas', () => { source: /* GraphQL */` { a } ` }); assertSome(data) - expect(data.a).toEqual(now.toISOString()); + expect(data['a']).toEqual(now.toISOString()); }); it.only('should not duplicate directives of scalars', () => { @@ -447,13 +447,13 @@ describe('Merge Schemas', () => { const QueryType = prev.getQueryType() assertSome(QueryType) const fields = QueryType.getFields() - assertSome(fields.test.astNode) - assertSome(fields.test.astNode.directives) - assertSome(fields.test.astNode.directives[0]) - assertSome(fields.test.astNode.directives[0].arguments) - assertListValueNode(fields.test.astNode.directives[0].arguments[0].value) + assertSome(fields['test'].astNode) + assertSome(fields['test'].astNode.directives) + assertSome(fields['test'].astNode.directives[0]) + assertSome(fields['test'].astNode.directives[0].arguments) + assertListValueNode(fields['test'].astNode.directives[0].arguments[0].value) - expect(fields.test.astNode.directives[0].arguments[0].value.values).toHaveLength(1); + expect(fields['test'].astNode.directives[0].arguments[0].value.values).toHaveLength(1); }); it('should merge schemas with custom scalars', () => { const GraphQLUUID = new GraphQLScalarType({ diff --git a/packages/merge/tests/merge-typedefs.spec.ts b/packages/merge/tests/merge-typedefs.spec.ts index bb52a8c0a8b..eb3df437285 100644 --- a/packages/merge/tests/merge-typedefs.spec.ts +++ b/packages/merge/tests/merge-typedefs.spec.ts @@ -7,7 +7,6 @@ import { stripWhitespaces } from './utils'; import gql from 'graphql-tag'; import { readFileSync } from 'fs'; import { join } from 'path'; -import { jest } from '@jest/globals'; import { assertSome } from '@graphql-tools/utils'; const introspectionSchema = JSON.parse(readFileSync(join(__dirname, './schema.json'), 'utf8')); @@ -657,7 +656,6 @@ describe('Merge TypeDefs', () => { const merged = mergeTypeDefs([ makeExecutableSchema({ typeDefs: ['type Query { f1: String }'], - allowUndefinedInResolve: true, }), 'type Query { f2: String }', ]); @@ -679,11 +677,9 @@ describe('Merge TypeDefs', () => { const merged = mergeTypeDefs([ makeExecutableSchema({ typeDefs: ['type RootQuery { f1: String }'], - allowUndefinedInResolve: true, }), makeExecutableSchema({ typeDefs: ['type RootQuery { f2: String }', 'schema { query: RootQuery }'], - allowUndefinedInResolve: true, }), ]); @@ -704,7 +700,6 @@ describe('Merge TypeDefs', () => { const merged = mergeTypeDefs([ makeExecutableSchema({ typeDefs: ['type Query { f1: String }'], - allowUndefinedInResolve: true, }), 'type Query { f2: String }', gql` @@ -732,7 +727,6 @@ describe('Merge TypeDefs', () => { const merged = mergeTypeDefs([ makeExecutableSchema({ typeDefs: ['type MyType { f1: String }'], - allowUndefinedInResolve: true, }), ]); @@ -747,11 +741,9 @@ describe('Merge TypeDefs', () => { const merged = mergeTypeDefs([ makeExecutableSchema({ typeDefs: ['type MyType { f1: String }'], - allowUndefinedInResolve: true, }), makeExecutableSchema({ typeDefs: ['type MyType { f2: String }'], - allowUndefinedInResolve: true, }), ]); diff --git a/packages/mock/package.json b/packages/mock/package.json index 33bb65ad49e..fa16db31983 100644 --- a/packages/mock/package.json +++ b/packages/mock/package.json @@ -33,6 +33,7 @@ }, "dependencies": { "@graphql-tools/schema": "^7.0.0", + "@graphql-tools/merge": "6.2.14", "@graphql-tools/utils": "^7.0.0", "fast-json-stable-stringify": "^2.1.0", "ts-is-defined": "^1.0.0", diff --git a/packages/mock/src/MockStore.ts b/packages/mock/src/MockStore.ts index b21eac7e9b5..69e4ea10cf7 100644 --- a/packages/mock/src/MockStore.ts +++ b/packages/mock/src/MockStore.ts @@ -237,7 +237,7 @@ export class MockStore implements IMockStore { if (!isRecord(value)) { throw new Error('When no `fieldName` is provided, `value` should be a record.'); } - for (const fieldName of Object.keys(value)) { + for (const fieldName in value) { this.setImpl({ typeName, key, @@ -368,7 +368,7 @@ export class MockStore implements IMockStore { } const toInsert = { ...otherValues, ...values }; - for (const fieldName of Object.keys(toInsert)) { + for (const fieldName in toInsert) { if (fieldName === '$ref') continue; if (fieldName === '__typename') continue; this.set({ @@ -410,7 +410,7 @@ export class MockStore implements IMockStore { throw new Error(`Value returned by the mock for ${typeName} is not an object`); } - for (const otherFieldName of Object.keys(values)) { + for (const otherFieldName in values) { if (otherFieldName === fieldName) continue; if (typeof (values as any)[otherFieldName] === 'function') continue; onOtherFieldsGenerated && onOtherFieldsGenerated(otherFieldName, (values as any)[otherFieldName]); @@ -498,7 +498,7 @@ export class MockStore implements IMockStore { } const toInsert = {}; - for (const fieldName of Object.keys(values)) { + for (const fieldName in values) { if (fieldName === '__typename') continue; const fieldValue = (values as any)[fieldName]; toInsert[fieldName] = typeof fieldValue === 'function' ? fieldValue() : fieldValue; diff --git a/packages/mock/src/mockServer.ts b/packages/mock/src/mockServer.ts index 301e9ebcf55..bb3bc2a9123 100644 --- a/packages/mock/src/mockServer.ts +++ b/packages/mock/src/mockServer.ts @@ -1,6 +1,6 @@ -import { ITypeDefinitions } from '@graphql-tools/utils'; -import { GraphQLSchema, isSchema, graphql } from 'graphql'; -import { buildSchemaFromTypeDefinitions } from '@graphql-tools/schema'; +import { TypeSource } from '@graphql-tools/utils'; +import { GraphQLSchema, isSchema, graphql, buildASTSchema } from 'graphql'; +import { mergeTypeDefs } from '@graphql-tools/merge'; import { addMocksToSchema } from './addMocksToSchema'; import { IMockServer, IMocks } from './types'; @@ -16,15 +16,11 @@ import { IMockServer, IMocks } from './types'; * overwritten to provide mock data. This can be used to mock some parts of the * server and not others. */ -export function mockServer( - schema: GraphQLSchema | ITypeDefinitions, - mocks: IMocks, - preserveResolvers = false -): IMockServer { +export function mockServer(schema: TypeSource, mocks: IMocks, preserveResolvers = false): IMockServer { let mySchema: GraphQLSchema; if (!isSchema(schema)) { // TODO: provide useful error messages here if this fails - mySchema = buildSchemaFromTypeDefinitions(schema); + mySchema = buildASTSchema(mergeTypeDefs(schema)); } else { mySchema = schema; } diff --git a/packages/mock/src/utils.ts b/packages/mock/src/utils.ts index e36d784446b..76edccb11ba 100644 --- a/packages/mock/src/utils.ts +++ b/packages/mock/src/utils.ts @@ -26,21 +26,21 @@ export function isObject(thing: any) { } export function copyOwnPropsIfNotPresent(target: Record, source: Record) { - Object.getOwnPropertyNames(source).forEach(prop => { + for (const prop of Object.getOwnPropertyNames(source)) { if (!Object.getOwnPropertyDescriptor(target, prop)) { const propertyDescriptor = Object.getOwnPropertyDescriptor(source, prop); Object.defineProperty(target, prop, propertyDescriptor == null ? {} : propertyDescriptor); } - }); + } } export function copyOwnProps(target: Record, ...sources: Array>) { - sources.forEach(source => { + for (const source of sources) { let chain = source; while (chain != null) { copyOwnPropsIfNotPresent(target, chain); chain = Object.getPrototypeOf(chain); } - }); + } return target; } diff --git a/packages/mock/tests/addMocksToSchema.spec.ts b/packages/mock/tests/addMocksToSchema.spec.ts index 0cba83b226d..24f46e010b7 100644 --- a/packages/mock/tests/addMocksToSchema.spec.ts +++ b/packages/mock/tests/addMocksToSchema.spec.ts @@ -267,8 +267,8 @@ describe('addMocksToSchema', () => { source: query }); - expect(data.foo.field1).toBe('text'); - expect(data.foo.field2).toBe(null); + expect(data?.['foo'].field1).toBe('text'); + expect(data?.['foo'].field2).toBe(null); }) it('should handle null fields correctly in nested fields', async () => { const schema = buildSchema(/* GraphQL */` @@ -309,8 +309,8 @@ describe('addMocksToSchema', () => { }); expect(errors).toBeFalsy(); - expect(data.foo.foo_field).toBe('text'); - expect(data.foo.boo).toBe(null); + expect(data?.['foo'].foo_field).toBe('text'); + expect(data?.['foo'].boo).toBe(null); }); it('handle objects without object prototype correctly', () => { const maybeRef = Object.create(null); diff --git a/packages/mock/tests/mocking-compatibility.spec.ts b/packages/mock/tests/mocking-compatibility.spec.ts index c2c85745137..fd94b338287 100644 --- a/packages/mock/tests/mocking-compatibility.spec.ts +++ b/packages/mock/tests/mocking-compatibility.spec.ts @@ -11,7 +11,6 @@ import { sentence, first_name } from 'casual'; import { addMocksToSchema, MockList, mockServer, IMocks, IMockStore } from '../src'; import { addResolversToSchema, - buildSchemaFromTypeDefinitions, makeExecutableSchema, } from '@graphql-tools/schema'; @@ -100,7 +99,7 @@ describe('Mock retro-compatibility', () => { }); test('throws an error if second argument is not a Map', () => { - const jsSchema = buildSchemaFromTypeDefinitions(shorthand); + const jsSchema = buildSchema(shorthand); expect(() => addMocksToSchema({ schema: jsSchema, @@ -110,7 +109,7 @@ describe('Mock retro-compatibility', () => { }); test('mocks the default types for you', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = {}; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ @@ -121,13 +120,13 @@ describe('Mock retro-compatibility', () => { returnID }`; return graphql(jsSchema, testQuery).then((res) => { - expect(res.data.returnInt).toBeGreaterThanOrEqual(-1000); - expect(res.data.returnInt).toBeLessThanOrEqual(1000); - expect(res.data.returnFloat).toBeGreaterThanOrEqual(-1000); - expect(res.data.returnFloat).toBeLessThanOrEqual(1000); - expect(typeof res.data.returnBoolean).toBe('boolean'); - expect(typeof res.data.returnString).toBe('string'); - expect(typeof res.data.returnID).toBe('string'); + expect(res.data?.['returnInt']).toBeGreaterThanOrEqual(-1000); + expect(res.data?.['returnInt']).toBeLessThanOrEqual(1000); + expect(res.data?.['returnFloat']).toBeGreaterThanOrEqual(-1000); + expect(res.data?.['returnFloat']).toBeLessThanOrEqual(1000); + expect(typeof res.data?.['returnBoolean']).toBe('boolean'); + expect(typeof res.data?.['returnString']).toBe('string'); + expect(typeof res.data?.['returnID']).toBe('string'); }); }); @@ -171,7 +170,7 @@ describe('Mock retro-compatibility', () => { }); test('mockServer is able to preserveResolvers of a prebuilt schema', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const resolvers = { RootQuery: { returnString: () => 'someString', @@ -208,7 +207,7 @@ describe('Mock retro-compatibility', () => { }); test('lets you use mockServer with prebuilt schema', () => { - const jsSchema = buildSchemaFromTypeDefinitions(shorthand); + const jsSchema = buildSchema(shorthand); const testQuery = `{ returnInt returnFloat @@ -248,7 +247,7 @@ describe('Mock retro-compatibility', () => { }); test('does not mask resolveType functions if you tell it not to', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); let spy = 0; const resolvers = { BirdsAndBees: { @@ -284,19 +283,19 @@ describe('Mock retro-compatibility', () => { // TODO test mockServer with precompiled schema test('can mock Enum', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = {}; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ returnEnum }`; return graphql(jsSchema, testQuery).then((res) => { - expect(['A', 'B', 'C']).toContain(res.data.returnEnum); + expect(['A', 'B', 'C']).toContain(res.data?.['returnEnum']); }); }); test('can mock Enum with a certain value', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { SomeEnum: () => 'C', }; @@ -305,12 +304,12 @@ describe('Mock retro-compatibility', () => { returnEnum }`; return graphql(jsSchema, testQuery).then((res) => { - expect('C').toBe(res.data.returnEnum); + expect('C').toBe(res.data?.['returnEnum']); }); }); test('can mock Unions', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { Int: () => 10, String: () => 'aha', @@ -334,13 +333,13 @@ describe('Mock retro-compatibility', () => { }`; return graphql(jsSchema, testQuery).then((res) => { // XXX this test is expected to fail once every 2^40 times ;-) - expect(res.data.returnBirdsAndBees).toContainEqual( + expect(res.data?.['returnBirdsAndBees']).toContainEqual( expect.objectContaining({ returnInt: 10, returnString: 'aha', }), ); - return expect(res.data.returnBirdsAndBees).toContainEqual( + return expect(res.data?.['returnBirdsAndBees']).toContainEqual( expect.objectContaining({ returnInt: 10, returnEnum: 'A', @@ -350,7 +349,7 @@ describe('Mock retro-compatibility', () => { }); test('can mock Interfaces by default', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { Int: () => 10, String: () => 'aha', @@ -376,13 +375,13 @@ describe('Mock retro-compatibility', () => { } }`; return graphql(jsSchema, testQuery).then((res) => { - expect(res.data.returnFlying).toContainEqual( + expect(res.data?.['returnFlying']).toContainEqual( expect.objectContaining({ returnInt: 10, returnString: 'aha', }), ); - return expect(res.data.returnFlying).toContainEqual( + return expect(res.data?.['returnFlying']).toContainEqual( expect.objectContaining({ returnInt: 10, returnEnum: 'A', @@ -392,7 +391,7 @@ describe('Mock retro-compatibility', () => { }); it('can mock nullable Interfaces', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { Bird: (): null => null, @@ -421,7 +420,7 @@ describe('Mock retro-compatibility', () => { }`; return graphql(jsSchema, testQuery).then((res) => { - expect(res.data.node).toEqual(null); + expect(res.data?.['node']).toEqual(null); }); }); @@ -449,7 +448,7 @@ describe('Mock retro-compatibility', () => { const res = await server.query(testQuery); - expect(res.data.node).toEqual({ + expect(res.data?.['node']).toEqual({ id: 'bee:hardcoded', returnAbility: { name: 'Hello World', @@ -458,7 +457,7 @@ describe('Mock retro-compatibility', () => { }); test('can support explicit Interface mock with resolver', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); let spy = 0; const mockMap = { Bird: () => ({ @@ -480,7 +479,7 @@ describe('Mock retro-compatibility', () => { const __typename = ['Bird', 'Bee'].find( (r) => r.toLowerCase() === type, ); - return store.get(__typename, id); + return __typename && store.get(__typename, id); } } }); @@ -499,7 +498,7 @@ describe('Mock retro-compatibility', () => { return graphql(jsSchema, testQuery).then((res) => { expect(spy).toBe(1); // to make sure that Flying possible types are not randomly selected - expect(res.data.node).toMatchObject({ + expect(res.data?.['node']).toMatchObject({ id: 'bee:123456', returnSong: 'I believe i can fly', returnInt: 200, @@ -508,7 +507,7 @@ describe('Mock retro-compatibility', () => { }); test('can support explicit UnionType mock', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); let spy = 0; const mockMap = { Bird: () => ({ @@ -541,7 +540,7 @@ describe('Mock retro-compatibility', () => { return graphql(jsSchema, testQuery).then((res) => { expect(spy).toBe(1); - expect(res.data.node2).toMatchObject({ + expect(res.data?.['node2']).toMatchObject({ id: 'bee:hardcoded', returnEnum: 'A', }); @@ -549,7 +548,7 @@ describe('Mock retro-compatibility', () => { }); test('throws an error when __typename is not returned within an explicit interface mock', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { Bird: (_root: any, args: any) => ({ id: args.id, @@ -570,12 +569,12 @@ describe('Mock retro-compatibility', () => { }`; const expected = 'Please return a __typename in "Flying"'; return graphql(jsSchema, testQuery).then((res) => { - expect(res.errors[0].originalError.message).toBe(expected); + expect(res.errors?.[0].originalError?.message).toBe(expected); }); }); test('throws an error in resolve if mock type is not defined', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = {}; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ @@ -583,12 +582,12 @@ describe('Mock retro-compatibility', () => { }`; const expected = 'No mock defined for type "MissingMockType"'; return graphql(jsSchema, testQuery).then((res) => { - expect(res.errors[0].originalError.message).toBe(expected); + expect(res.errors?.[0].originalError?.message).toBe(expected); }); }); test('throws an error in resolve if mock type is not defined and resolver failed', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const resolvers = { MissingMockType: { __serialize: (val: string) => val, @@ -596,7 +595,7 @@ describe('Mock retro-compatibility', () => { __parseLiteral: (val: string) => val, }, RootQuery: { - returnMockError: (): string => undefined, + returnMockError: () => undefined, }, }; jsSchema = addResolversToSchema(jsSchema, resolvers); @@ -612,12 +611,12 @@ describe('Mock retro-compatibility', () => { }`; const expected = 'No mock defined for type "MissingMockType"'; return graphql(jsSchema, testQuery).then((res) => { - expect(res.errors[0].originalError.message).toBe(expected); + expect(res.errors?.[0].originalError?.message).toBe(expected); }); }); test('can preserve scalar resolvers', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const resolvers = { MissingMockType: { __serialize: (val: string) => val, @@ -649,85 +648,85 @@ describe('Mock retro-compatibility', () => { }); test('can mock an Int', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { Int: () => 55 }; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ returnInt }`; return graphql(jsSchema, testQuery).then((res) => { - expect(res.data.returnInt).toBe(55); + expect(res.data?.['returnInt']).toBe(55); }); }); test('can mock a Float', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { Float: () => 55.5 }; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ returnFloat }`; return graphql(jsSchema, testQuery).then((res) => { - expect(res.data.returnFloat).toBe(55.5); + expect(res.data?.['returnFloat']).toBe(55.5); }); }); test('can mock a String', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { String: () => 'a string' }; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ returnString }`; return graphql(jsSchema, testQuery).then((res) => { - expect(res.data.returnString).toBe('a string'); + expect(res.data?.['returnString']).toBe('a string'); }); }); test('can mock a Boolean', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { Boolean: () => true }; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ returnBoolean }`; return graphql(jsSchema, testQuery).then((res) => { - expect(res.data.returnBoolean).toBe(true); + expect(res.data?.['returnBoolean']).toBe(true); }); }); test('can mock an ID', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { ID: () => 'ea5bdc19' }; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ returnID }`; return graphql(jsSchema, testQuery).then((res) => { - expect(res.data.returnID).toBe('ea5bdc19'); + expect(res.data?.['returnID']).toBe('ea5bdc19'); }); }); test('nullable type is nullable', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { String: (): null => null }; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ returnNullableString }`; return graphql(jsSchema, testQuery).then((res) => { - expect(res.data.returnNullableString).toBe(null); + expect(res.data?.['returnNullableString']).toBe(null); }); }); test('can mock a nonNull type', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { String: () => 'nonnull' }; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ returnNonNullString }`; return graphql(jsSchema, testQuery).then((res) => { - expect(res.data.returnNonNullString).toBe('nonnull'); + expect(res.data?.['returnNonNullString']).toBe('nonnull'); }); }); test('nonNull type is not nullable', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { String: (): null => null }; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ @@ -735,11 +734,11 @@ describe('Mock retro-compatibility', () => { }`; return graphql(jsSchema, testQuery).then((res) => { expect(res.data).toBe(null); - expect(res.errors.length).toBe(1); + expect(res.errors?.length).toBe(1); }); }); test('can mock object types', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { String: () => 'abc', Int: () => 123, @@ -757,7 +756,7 @@ describe('Mock retro-compatibility', () => { }); test('can mock a list of ints', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { Int: () => 123 }; jsSchema = addMocksToSchema({ schema: jsSchema, mocks: mockMap }); const testQuery = `{ @@ -772,7 +771,7 @@ describe('Mock retro-compatibility', () => { }); test('can mock a list of lists of objects', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { String: () => 'a', Int: () => 1, @@ -799,7 +798,7 @@ describe('Mock retro-compatibility', () => { }); test('does not mask resolvers if you tell it not to', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { RootQuery: () => ({ returnInt: (_root: any, _args: Record) => 42, // a) in resolvers, will not be used @@ -836,7 +835,7 @@ describe('Mock retro-compatibility', () => { }); test('lets you mock non-leaf types conveniently', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { Bird: () => ({ returnInt: 12, @@ -865,7 +864,7 @@ describe('Mock retro-compatibility', () => { }); test('lets you mock and resolve non-leaf types concurrently', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const resolvers = { RootQuery: { returnListOfInt: () => [1, 2, 3], @@ -908,7 +907,7 @@ describe('Mock retro-compatibility', () => { }); test('lets you mock and resolve non-leaf types concurrently, support promises', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const resolvers = { RootQuery: { returnObject: () => @@ -948,7 +947,7 @@ describe('Mock retro-compatibility', () => { }); test('lets you mock and resolve non-leaf types concurrently, support defineProperty', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const objProxy = {}; Object.defineProperty( objProxy, @@ -989,7 +988,7 @@ describe('Mock retro-compatibility', () => { }); }); - test('let you mock with preserving resolvers, also when using logger', () => { + test('let you mock with preserving resolvers', () => { const resolvers = { RootQuery: { returnString: () => 'woot!?', // a) resolve of a string @@ -998,7 +997,6 @@ describe('Mock retro-compatibility', () => { let jsSchema = makeExecutableSchema({ typeDefs: [shorthand], resolvers, - logger: console, }); const mockMap = { Int: () => 123, // b) mock of Int. @@ -1028,10 +1026,10 @@ describe('Mock retro-compatibility', () => { }); test('let you resolve null with mocking and preserving resolvers', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const resolvers = { RootQuery: { - returnString: (): string => null, // a) resolve of a string + returnString: () => null, // a) resolve of a string }, }; jsSchema = addResolversToSchema(jsSchema, resolvers); @@ -1055,7 +1053,7 @@ describe('Mock retro-compatibility', () => { returnInt: 666, // from the mock, see b) returnString: 'Hello World', // from mock default values. }, - returnString: null as string, // from the mock, see a) + returnString: null as unknown as string, // from the mock, see a) }; return graphql(jsSchema, testQuery, undefined, {}).then((res) => { expect(res.data).toEqual(expected); @@ -1063,10 +1061,10 @@ describe('Mock retro-compatibility', () => { }); test('lets you mock root query fields', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const resolvers = { RootQuery: { - returnStringArgument: (_: void, a: Record) => a.s, + returnStringArgument: (_: void, a: Record) => a['s'], }, }; jsSchema = addMocksToSchema({ schema: jsSchema, resolvers }); @@ -1082,10 +1080,10 @@ describe('Mock retro-compatibility', () => { }); test('lets you mock root mutation fields', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const resolvers = { RootMutation: { - returnStringArgument: (_: void, a: Record) => a.s, + returnStringArgument: (_: void, a: Record) => a['s'], }, }; jsSchema = addMocksToSchema({ schema: jsSchema, resolvers }); @@ -1101,7 +1099,7 @@ describe('Mock retro-compatibility', () => { }); test('lets you mock a list of a certain length', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { RootQuery: () => ({ returnListOfInt: () => new MockList(3) }), Int: () => 12, @@ -1119,7 +1117,7 @@ describe('Mock retro-compatibility', () => { }); test('lets you mock a list of a random length', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { RootQuery: () => ({ returnListOfInt: () => new MockList([10, 20]) }), Int: () => 12, @@ -1129,14 +1127,14 @@ describe('Mock retro-compatibility', () => { returnListOfInt }`; return graphql(jsSchema, testQuery).then((res) => { - expect(res.data.returnListOfInt.length).toBeGreaterThanOrEqual(10); - expect(res.data.returnListOfInt.length).toBeLessThanOrEqual(20); - expect(res.data.returnListOfInt[0]).toBe(12); + expect(res.data?.['returnListOfInt'].length).toBeGreaterThanOrEqual(10); + expect(res.data?.['returnListOfInt'].length).toBeLessThanOrEqual(20); + expect(res.data?.['returnListOfInt'][0]).toBe(12); }); }); test('lets you provide a function for your MockList', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { RootQuery: () => ({ returnListOfInt: () => new MockList(2, () => 33), @@ -1165,7 +1163,7 @@ describe('Mock retro-compatibility', () => { }); test('lets you nest MockList in MockList', () => { - let jsSchema = buildSchemaFromTypeDefinitions(shorthand); + let jsSchema = buildSchema(shorthand); const mockMap = { RootQuery: () => ({ returnListOfListOfInt: () => new MockList(2, () => new MockList(3)), @@ -1368,9 +1366,9 @@ describe('Mock retro-compatibility', () => { `, }); - expect(result.data?.reviews?.length <= 4).toBeTruthy(); - expect(typeof result.data?.reviews[0]?.sentence).toBe('string'); - expect(typeof result.data?.reviews[0]?.user?.first_name).toBe('string'); + expect(result.data?.['reviews']?.length <= 4).toBeTruthy(); + expect(typeof result.data?.['reviews'][0]?.sentence).toBe('string'); + expect(typeof result.data?.['reviews'][0]?.user?.first_name).toBe('string'); }); it('resolves subscriptions only once', async () => { diff --git a/packages/mock/tests/store.spec.ts b/packages/mock/tests/store.spec.ts index 2d27062ffc1..869bd123359 100644 --- a/packages/mock/tests/store.spec.ts +++ b/packages/mock/tests/store.spec.ts @@ -250,7 +250,7 @@ describe('MockStore', () => { it('should be able to generate a list of Object type', () => { const store = createMockStore({ schema }); - const friends = store.get('User', '123', 'friends'); + const friends = store.get('User', '123', 'friends') as any[]; expect(friends).toBeInstanceOf(Array); expect(friends[0]).toHaveProperty('$ref'); diff --git a/packages/node-require/src/index.ts b/packages/node-require/src/index.ts index 37006940120..d238c3b50d5 100644 --- a/packages/node-require/src/index.ts +++ b/packages/node-require/src/index.ts @@ -23,9 +23,9 @@ function handleModule(m: NodeModule, filename: string) { } export function registerGraphQLExtensions(require: NodeRequire) { - VALID_EXTENSIONS.forEach(ext => { + for (const ext of VALID_EXTENSIONS) { require.extensions[`.${ext}`] = handleModule; - }); + } } registerGraphQLExtensions(require); diff --git a/packages/relay-operation-optimizer/tests/relay-optimizer.spec.ts b/packages/relay-operation-optimizer/tests/relay-optimizer.spec.ts index 62a9c03a058..44d04ae4e84 100644 --- a/packages/relay-operation-optimizer/tests/relay-optimizer.spec.ts +++ b/packages/relay-operation-optimizer/tests/relay-optimizer.spec.ts @@ -68,7 +68,7 @@ it('can inline @argumentDefinitions/@arguments annotated fragments', async () => const queryDoc = output.find(doc => doc.definitions[0].kind === 'OperationDefinition'); expect(queryDoc).toBeDefined(); - expect(print(queryDoc)).toBeSimilarGqlDoc(/* GraphQL */ ` + expect(print(queryDoc!)).toBeSimilarGqlDoc(/* GraphQL */ ` query user { users { id @@ -127,7 +127,7 @@ it('handles unions with interfaces the correct way', async () => { const queryDoc = output.find(doc => doc.definitions[0].kind === 'OperationDefinition'); expect(queryDoc).toBeDefined(); - expect(print(queryDoc)).toBeSimilarGqlDoc(/* GraphQL */ ` + expect(print(queryDoc!)).toBeSimilarGqlDoc(/* GraphQL */ ` query user { user { ... on User { diff --git a/packages/resolvers-composition/src/resolvers-composition.ts b/packages/resolvers-composition/src/resolvers-composition.ts index 927c4b2875d..f07bad3174d 100644 --- a/packages/resolvers-composition/src/resolvers-composition.ts +++ b/packages/resolvers-composition/src/resolvers-composition.ts @@ -1,5 +1,5 @@ import { chainFunctions } from './chain-functions'; -import { get, set, flatten } from 'lodash'; +import _ from 'lodash'; import { GraphQLFieldResolver, GraphQLScalarTypeConfig } from 'graphql'; import { asArray } from '@graphql-tools/utils'; @@ -45,11 +45,14 @@ function resolveRelevantMappings = Record< if (!resolvers) { return []; } - return flatten( - Object.keys(resolvers).map(typeName => - resolveRelevantMappings(resolvers, `${typeName}.${fieldName}`, allMappings) - ) - ); + const mappings: string[] = []; + for (const typeName in resolvers) { + const relevantMappings = resolveRelevantMappings(resolvers, `${typeName}.${fieldName}`, allMappings); + for (const relevantMapping of relevantMappings) { + mappings.push(relevantMapping); + } + } + return mappings; } if (fieldName === '*') { @@ -57,9 +60,16 @@ function resolveRelevantMappings = Record< if (!fieldMap) { return []; } - return flatten( - Object.keys(fieldMap).map(field => resolveRelevantMappings(resolvers, `${typeName}.${field}`, allMappings)) - ).filter(mapItem => !allMappings[mapItem]); + const mappings: string[] = []; + for (const field in fieldMap) { + const relevantMappings = resolveRelevantMappings(resolvers, `${typeName}.${field}`, allMappings); + for (const relevantMapping of relevantMappings) { + if (!allMappings[relevantMapping]) { + mappings.push(relevantMapping); + } + } + } + return mappings; } else { const paths = []; @@ -87,11 +97,15 @@ function resolveRelevantMappings = Record< return []; } - return flatten( - Object.keys(fieldMap).map(fieldName => - resolveRelevantMappings(resolvers, `${typeName}.${fieldName}`, allMappings) - ) - ); + const mappings: string[] = []; + + for (const fieldName in fieldMap) { + const relevantMappings = resolveRelevantMappings(resolvers, `${typeName}.${fieldName}`, allMappings); + for (const relevantMapping of relevantMappings) { + mappings.push(relevantMapping); + } + } + return mappings; } return []; @@ -111,32 +125,32 @@ export function composeResolvers>( ): Resolvers { const mappingResult: { [path: string]: ((...args: any[]) => any)[] } = {}; - Object.keys(mapping).forEach((resolverPath: string) => { + for (const resolverPath in mapping) { const resolverPathMapping = mapping[resolverPath]; if (resolverPathMapping instanceof Array || typeof resolverPathMapping === 'function') { const composeFns = resolverPathMapping as ResolversComposition | ResolversComposition[]; const relevantFields = resolveRelevantMappings(resolvers, resolverPath, mapping); - relevantFields.forEach((path: string) => { + for (const path of relevantFields) { mappingResult[path] = asArray(composeFns); - }); + } } else if (resolverPathMapping) { - Object.keys(resolverPathMapping).forEach(fieldName => { + for (const fieldName in resolverPathMapping) { const composeFns = resolverPathMapping[fieldName]; const relevantFields = resolveRelevantMappings(resolvers, resolverPath + '.' + fieldName, mapping); - relevantFields.forEach((path: string) => { + for (const path of relevantFields) { mappingResult[path] = asArray(composeFns); - }); - }); + } + } } - }); + } - Object.keys(mappingResult).forEach(path => { - const fns = chainFunctions([...asArray(mappingResult[path]), () => get(resolvers, path)]); + for (const path in mappingResult) { + const fns = chainFunctions([...asArray(mappingResult[path]), () => _.get(resolvers, path)]); - set(resolvers, path, fns()); - }); + _.set(resolvers, path, fns()); + } return resolvers; } diff --git a/packages/resolvers-composition/tests/resolvers-composition.spec.ts b/packages/resolvers-composition/tests/resolvers-composition.spec.ts index 123f7582100..8449371c3ab 100644 --- a/packages/resolvers-composition/tests/resolvers-composition.spec.ts +++ b/packages/resolvers-composition/tests/resolvers-composition.spec.ts @@ -2,7 +2,6 @@ import gql from 'graphql-tag'; import { composeResolvers, ResolversComposerMapping } from '../src'; import { makeExecutableSchema } from '@graphql-tools/schema'; import { execute, GraphQLScalarType, Kind } from 'graphql'; -import { IResolvers } from 'packages/graphql-tools/src'; function createAsyncIterator(array: T[]): AsyncIterator { let i = 0; @@ -53,7 +52,7 @@ describe('Resolvers composition', () => { `, }); expect(result.errors).toBeFalsy(); - expect(result.data!.foo).toBe('FOOFOO'); + expect(result.data!['foo']).toBe('FOOFOO'); }); it('should compose resolvers with resolve field', async () => { const getFoo = () => 'FOO'; @@ -91,7 +90,7 @@ describe('Resolvers composition', () => { `, }); expect(result.errors).toBeFalsy(); - expect(result.data!.foo).toBe('FOOFOO'); + expect(result.data!['foo']).toBe('FOOFOO'); }); it('should compose subscription resolvers', async () => { const array1 = [1, 2]; @@ -170,7 +169,7 @@ describe('Resolvers composition', () => { `, }); expect(result.errors).toBeFalsy(); - expect(result.data!.foo).toBe('FOOFOO'); + expect(result.data!['foo']).toBe('FOOFOO'); }); it('should be able to take nested composition objects for subscription resolvers', async () => { const array1 = [1, 2]; @@ -293,14 +292,14 @@ describe('Resolvers composition', () => { it('should handle nullish properties correctly', async () => { const getFoo = () => 'FOO'; - const resolvers: IResolvers = { + const resolvers = { Query: { foo: async () => getFoo(), bar: undefined, }, Mutation: undefined }; - const resolversComposition: ResolversComposerMapping = { + const resolversComposition: any = { 'Query.foo': (next: (arg0: any, arg1: any, arg2: any, arg3: any) => void) => async (root: any, args: any, context: any, info: any) => { const prevResult = await next(root, args, context, info); return getFoo() + prevResult; diff --git a/packages/schema/package.json b/packages/schema/package.json index 95052175ff1..85efbcc0a70 100644 --- a/packages/schema/package.json +++ b/packages/schema/package.json @@ -35,6 +35,7 @@ "input": "./src/index.ts" }, "dependencies": { + "@graphql-tools/merge": "6.2.14", "@graphql-tools/utils": "^7.1.2", "tslib": "~2.3.0", "value-or-promise": "1.0.10" diff --git a/packages/schema/src/addCatchUndefinedToSchema.ts b/packages/schema/src/addCatchUndefinedToSchema.ts deleted file mode 100644 index 806f81e4a1f..00000000000 --- a/packages/schema/src/addCatchUndefinedToSchema.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { GraphQLFieldResolver, defaultFieldResolver, GraphQLSchema } from 'graphql'; -import { mapSchema, MapperKind, Maybe } from '@graphql-tools/utils'; - -function decorateToCatchUndefined( - fn: Maybe>, - hint: string -): GraphQLFieldResolver { - const resolve = fn == null ? defaultFieldResolver : fn; - return (root, args, ctx, info) => { - const result = resolve(root, args, ctx, info); - if (typeof result === 'undefined') { - throw new Error(`Resolver for "${hint}" returned undefined`); - } - return result; - }; -} - -export function addCatchUndefinedToSchema(schema: GraphQLSchema): GraphQLSchema { - return mapSchema(schema, { - [MapperKind.OBJECT_FIELD]: (fieldConfig, fieldName, typeName) => ({ - ...fieldConfig, - resolve: decorateToCatchUndefined(fieldConfig.resolve, `${typeName}.${fieldName}`), - }), - }); -} diff --git a/packages/schema/src/addErrorLoggingToSchema.ts b/packages/schema/src/addErrorLoggingToSchema.ts deleted file mode 100644 index ac42a375b48..00000000000 --- a/packages/schema/src/addErrorLoggingToSchema.ts +++ /dev/null @@ -1,19 +0,0 @@ -import { GraphQLSchema } from 'graphql'; -import { mapSchema, MapperKind } from '@graphql-tools/utils'; -import { decorateWithLogger } from './decorateWithLogger'; -import { ILogger } from './types'; - -export function addErrorLoggingToSchema(schema: GraphQLSchema, logger?: ILogger): GraphQLSchema { - if (!logger) { - throw new Error('Must provide a logger'); - } - if (typeof logger.log !== 'function') { - throw new Error('Logger.log must be a function'); - } - return mapSchema(schema, { - [MapperKind.OBJECT_FIELD]: (fieldConfig, fieldName, typeName) => ({ - ...fieldConfig, - resolve: decorateWithLogger(fieldConfig.resolve, logger, `${typeName}.${fieldName}`), - }), - }); -} diff --git a/packages/schema/src/addResolversToSchema.ts b/packages/schema/src/addResolversToSchema.ts index 5364bacf13b..756f4b1f34b 100644 --- a/packages/schema/src/addResolversToSchema.ts +++ b/packages/schema/src/addResolversToSchema.ts @@ -61,92 +61,82 @@ export function addResolversToSchema( ? extendResolversFromInterfaces(schema, inputResolvers) : inputResolvers; - Object.getOwnPropertyNames(resolvers).forEach(typeName => { + for (const typeName in resolvers) { const resolverValue = resolvers[typeName]; const resolverType = typeof resolverValue; - if (typeName === '__schema') { - if (resolverType !== 'function') { - throw new Error( - `"${typeName}" defined in resolvers, but has invalid value "${ - resolverValue as unknown as string - }". A schema resolver's value must be of type object or function.` - ); - } - } else { - if (resolverType !== 'object') { - throw new Error( - `"${typeName}" defined in resolvers, but has invalid value "${ - resolverValue as unknown as string - }". The resolver's value must be of type object.` - ); - } + if (resolverType !== 'object') { + throw new Error( + `"${typeName}" defined in resolvers, but has invalid value "${ + resolverValue as unknown as string + }". The resolver's value must be of type object.` + ); + } - const type = schema.getType(typeName); + const type = schema.getType(typeName); - if (type == null) { - if (requireResolversToMatchSchema === 'ignore') { - return; + if (type == null) { + if (requireResolversToMatchSchema === 'ignore') { + break; + } + + throw new Error(`"${typeName}" defined in resolvers, but not in schema`); + } else if (isSpecifiedScalarType(type)) { + // allow -- without recommending -- overriding of specified scalar types + for (const fieldName in resolverValue) { + if (fieldName.startsWith('__')) { + type[fieldName.substring(2)] = resolverValue[fieldName]; + } else { + type[fieldName] = resolverValue[fieldName]; + } + } + } else if (isEnumType(type)) { + const values = type.getValues(); + + for (const fieldName in resolverValue) { + if ( + !fieldName.startsWith('__') && + !values.some(value => value.name === fieldName) && + requireResolversToMatchSchema && + requireResolversToMatchSchema !== 'ignore' + ) { + throw new Error(`${type.name}.${fieldName} was defined in resolvers, but not present within ${type.name}`); } + } + } else if (isUnionType(type)) { + for (const fieldName in resolverValue) { + if ( + !fieldName.startsWith('__') && + requireResolversToMatchSchema && + requireResolversToMatchSchema !== 'ignore' + ) { + throw new Error( + `${type.name}.${fieldName} was defined in resolvers, but ${type.name} is not an object or interface type` + ); + } + } + } else if (isObjectType(type) || isInterfaceType(type)) { + for (const fieldName in resolverValue) { + if (!fieldName.startsWith('__')) { + const fields = type.getFields(); + const field = fields[fieldName]; - throw new Error(`"${typeName}" defined in resolvers, but not in schema`); - } else if (isSpecifiedScalarType(type)) { - // allow -- without recommending -- overriding of specified scalar types - Object.getOwnPropertyNames(resolverValue).forEach(fieldName => { - if (fieldName.startsWith('__')) { - type[fieldName.substring(2)] = resolverValue[fieldName]; + if (field == null) { + // Field present in resolver but not in schema + if (requireResolversToMatchSchema && requireResolversToMatchSchema !== 'ignore') { + throw new Error(`${typeName}.${fieldName} defined in resolvers, but not in schema`); + } } else { - type[fieldName] = resolverValue[fieldName]; - } - }); - } else if (isEnumType(type)) { - const values = type.getValues(); - - Object.getOwnPropertyNames(resolverValue).forEach(fieldName => { - if ( - !fieldName.startsWith('__') && - !values.some(value => value.name === fieldName) && - requireResolversToMatchSchema && - requireResolversToMatchSchema !== 'ignore' - ) { - throw new Error(`${type.name}.${fieldName} was defined in resolvers, but not present within ${type.name}`); - } - }); - } else if (isUnionType(type)) { - Object.getOwnPropertyNames(resolverValue).forEach(fieldName => { - if ( - !fieldName.startsWith('__') && - requireResolversToMatchSchema && - requireResolversToMatchSchema !== 'ignore' - ) { - throw new Error( - `${type.name}.${fieldName} was defined in resolvers, but ${type.name} is not an object or interface type` - ); - } - }); - } else if (isObjectType(type) || isInterfaceType(type)) { - Object.getOwnPropertyNames(resolverValue).forEach(fieldName => { - if (!fieldName.startsWith('__')) { - const fields = type.getFields(); - const field = fields[fieldName]; - - if (field == null) { - // Field present in resolver but not in schema - if (requireResolversToMatchSchema && requireResolversToMatchSchema !== 'ignore') { - throw new Error(`${typeName}.${fieldName} defined in resolvers, but not in schema`); - } - } else { - // Field present in both the resolver and schema - const fieldResolve = resolverValue[fieldName]; - if (typeof fieldResolve !== 'function' && typeof fieldResolve !== 'object') { - throw new Error(`Resolver ${typeName}.${fieldName} must be object or function`); - } + // Field present in both the resolver and schema + const fieldResolve = resolverValue[fieldName]; + if (typeof fieldResolve !== 'function' && typeof fieldResolve !== 'object') { + throw new Error(`Resolver ${typeName}.${fieldName} must be object or function`); } } - }); + } } } - }); + } schema = updateResolversInPlace ? addResolversToExistingSchema(schema, resolvers, defaultFieldResolver) @@ -165,98 +155,96 @@ function addResolversToExistingSchema( defaultFieldResolver?: GraphQLFieldResolver ): GraphQLSchema { const typeMap = schema.getTypeMap(); - getAllPropertyNames(resolvers).forEach(typeName => { - if (typeName !== '__schema') { - const type = schema.getType(typeName); - const resolverValue = resolvers[typeName]; + for (const typeName in resolvers) { + const type = schema.getType(typeName); + const resolverValue = resolvers[typeName]; - if (isScalarType(type)) { - getAllPropertyNames(resolverValue).forEach(fieldName => { - if (fieldName.startsWith('__')) { - type[fieldName.substring(2)] = resolverValue[fieldName]; - } else if (fieldName === 'astNode' && type.astNode != null) { - type.astNode = { - ...type.astNode, - description: (resolverValue as GraphQLScalarType)?.astNode?.description ?? type.astNode.description, - directives: (type.astNode.directives ?? []).concat( - (resolverValue as GraphQLScalarType)?.astNode?.directives ?? [] - ), - }; - } else if (fieldName === 'extensionASTNodes' && type.extensionASTNodes != null) { - type.extensionASTNodes = type.extensionASTNodes.concat( - (resolverValue as GraphQLScalarType)?.extensionASTNodes ?? [] - ); - } else if ( - fieldName === 'extensions' && - type.extensions != null && - (resolverValue as GraphQLScalarType).extensions != null - ) { - type.extensions = Object.assign({}, type.extensions, (resolverValue as GraphQLScalarType).extensions); - } else { - type[fieldName] = resolverValue[fieldName]; - } - }); - } else if (isEnumType(type)) { - const config = type.toConfig(); - const enumValueConfigMap = config.values; + if (isScalarType(type)) { + for (const fieldName in resolverValue) { + if (fieldName.startsWith('__')) { + type[fieldName.substring(2)] = resolverValue[fieldName]; + } else if (fieldName === 'astNode' && type.astNode != null) { + type.astNode = { + ...type.astNode, + description: (resolverValue as GraphQLScalarType)?.astNode?.description ?? type.astNode.description, + directives: (type.astNode.directives ?? []).concat( + (resolverValue as GraphQLScalarType)?.astNode?.directives ?? [] + ), + }; + } else if (fieldName === 'extensionASTNodes' && type.extensionASTNodes != null) { + type.extensionASTNodes = type.extensionASTNodes.concat( + (resolverValue as GraphQLScalarType)?.extensionASTNodes ?? [] + ); + } else if ( + fieldName === 'extensions' && + type.extensions != null && + (resolverValue as GraphQLScalarType).extensions != null + ) { + type.extensions = Object.assign({}, type.extensions, (resolverValue as GraphQLScalarType).extensions); + } else { + type[fieldName] = resolverValue[fieldName]; + } + } + } else if (isEnumType(type)) { + const config = type.toConfig(); + const enumValueConfigMap = config.values; - getAllPropertyNames(resolverValue).forEach(fieldName => { - if (fieldName.startsWith('__')) { - config[fieldName.substring(2)] = resolverValue[fieldName]; - } else if (fieldName === 'astNode' && config.astNode != null) { - config.astNode = { - ...config.astNode, - description: (resolverValue as GraphQLScalarType)?.astNode?.description ?? config.astNode.description, - directives: (config.astNode.directives ?? []).concat( - (resolverValue as GraphQLEnumType)?.astNode?.directives ?? [] - ), - }; - } else if (fieldName === 'extensionASTNodes' && config.extensionASTNodes != null) { - config.extensionASTNodes = config.extensionASTNodes.concat( - (resolverValue as GraphQLEnumType)?.extensionASTNodes ?? [] - ); - } else if ( - fieldName === 'extensions' && - type.extensions != null && - (resolverValue as GraphQLEnumType).extensions != null - ) { - type.extensions = Object.assign({}, type.extensions, (resolverValue as GraphQLEnumType).extensions); - } else if (enumValueConfigMap[fieldName]) { - enumValueConfigMap[fieldName].value = resolverValue[fieldName]; - } - }); + for (const fieldName in resolverValue) { + if (fieldName.startsWith('__')) { + config[fieldName.substring(2)] = resolverValue[fieldName]; + } else if (fieldName === 'astNode' && config.astNode != null) { + config.astNode = { + ...config.astNode, + description: (resolverValue as GraphQLScalarType)?.astNode?.description ?? config.astNode.description, + directives: (config.astNode.directives ?? []).concat( + (resolverValue as GraphQLEnumType)?.astNode?.directives ?? [] + ), + }; + } else if (fieldName === 'extensionASTNodes' && config.extensionASTNodes != null) { + config.extensionASTNodes = config.extensionASTNodes.concat( + (resolverValue as GraphQLEnumType)?.extensionASTNodes ?? [] + ); + } else if ( + fieldName === 'extensions' && + type.extensions != null && + (resolverValue as GraphQLEnumType).extensions != null + ) { + type.extensions = Object.assign({}, type.extensions, (resolverValue as GraphQLEnumType).extensions); + } else if (enumValueConfigMap[fieldName]) { + enumValueConfigMap[fieldName].value = resolverValue[fieldName]; + } + } - typeMap[typeName] = new GraphQLEnumType(config); - } else if (isUnionType(type)) { - getAllPropertyNames(resolverValue).forEach(fieldName => { - if (fieldName.startsWith('__')) { - type[fieldName.substring(2)] = resolverValue[fieldName]; - } - }); - } else if (isObjectType(type) || isInterfaceType(type)) { - getAllPropertyNames(resolverValue).forEach(fieldName => { - if (fieldName.startsWith('__')) { - // this is for isTypeOf and resolveType and all the other stuff. - type[fieldName.substring(2)] = resolverValue[fieldName]; - return; - } + typeMap[typeName] = new GraphQLEnumType(config); + } else if (isUnionType(type)) { + for (const fieldName in resolverValue) { + if (fieldName.startsWith('__')) { + type[fieldName.substring(2)] = resolverValue[fieldName]; + } + } + } else if (isObjectType(type) || isInterfaceType(type)) { + for (const fieldName in resolverValue) { + if (fieldName.startsWith('__')) { + // this is for isTypeOf and resolveType and all the other stuff. + type[fieldName.substring(2)] = resolverValue[fieldName]; + break; + } - const fields = type.getFields(); - const field = fields[fieldName]; + const fields = type.getFields(); + const field = fields[fieldName]; - if (field != null) { - const fieldResolve = resolverValue[fieldName]; - if (typeof fieldResolve === 'function') { - // for convenience. Allows shorter syntax in resolver definition file - field.resolve = fieldResolve.bind(resolverValue); - } else { - setFieldProperties(field, fieldResolve); - } + if (field != null) { + const fieldResolve = resolverValue[fieldName]; + if (typeof fieldResolve === 'function') { + // for convenience. Allows shorter syntax in resolver definition file + field.resolve = fieldResolve.bind(resolverValue); + } else { + setFieldProperties(field, fieldResolve); } - }); + } } } - }); + } // serialize all default values prior to healing fields with new scalar/enum types. forEachDefaultValue(schema, serializeInputValue); @@ -286,7 +274,7 @@ function createNewSchemaWithResolvers( const config = type.toConfig(); const resolverValue = resolvers[type.name]; if (!isSpecifiedScalarType(type) && resolverValue != null) { - getAllPropertyNames(resolverValue).forEach(fieldName => { + for (const fieldName in resolverValue) { if (fieldName.startsWith('__')) { config[fieldName.substring(2)] = resolverValue[fieldName]; } else if (fieldName === 'astNode' && config.astNode != null) { @@ -310,7 +298,7 @@ function createNewSchemaWithResolvers( } else { config[fieldName] = resolverValue[fieldName]; } - }); + } return new GraphQLScalarType(config); } @@ -322,7 +310,7 @@ function createNewSchemaWithResolvers( const enumValueConfigMap = config.values; if (resolverValue != null) { - getAllPropertyNames(resolverValue).forEach(fieldName => { + for (const fieldName in resolverValue) { if (fieldName.startsWith('__')) { config[fieldName.substring(2)] = resolverValue[fieldName]; } else if (fieldName === 'astNode' && config.astNode != null) { @@ -346,7 +334,7 @@ function createNewSchemaWithResolvers( } else if (enumValueConfigMap[fieldName]) { enumValueConfigMap[fieldName].value = resolverValue[fieldName]; } - }); + } return new GraphQLEnumType(config); } @@ -356,11 +344,10 @@ function createNewSchemaWithResolvers( if (resolverValue != null) { const config = type.toConfig(); - getAllPropertyNames(resolverValue).forEach(fieldName => { - if (fieldName.startsWith('__')) { - config[fieldName.substring(2)] = resolverValue[fieldName]; - } - }); + + if (resolverValue['__resolveType']) { + config.resolveType = resolverValue['__resolveType']; + } return new GraphQLUnionType(config); } @@ -370,11 +357,9 @@ function createNewSchemaWithResolvers( if (resolverValue != null) { const config = type.toConfig(); - getAllPropertyNames(resolverValue).forEach(fieldName => { - if (fieldName.startsWith('__')) { - config[fieldName.substring(2)] = resolverValue[fieldName]; - } - }); + if (resolverValue['__isTypeOf']) { + config.isTypeOf = resolverValue['__isTypeOf']; + } return new GraphQLObjectType(config); } @@ -384,11 +369,9 @@ function createNewSchemaWithResolvers( if (resolverValue != null) { const config = type.toConfig(); - getAllPropertyNames(resolverValue).forEach(fieldName => { - if (fieldName.startsWith('__')) { - config[fieldName.substring(2)] = resolverValue[fieldName]; - } - }); + if (resolverValue['__resolveType']) { + config.resolveType = resolverValue['__resolveType']; + } return new GraphQLInterfaceType(config); } @@ -428,14 +411,7 @@ function setFieldProperties( field: GraphQLField | GraphQLFieldConfig, propertiesObj: Record ) { - Object.keys(propertiesObj).forEach(propertyName => { + for (const propertyName in propertiesObj) { field[propertyName] = propertiesObj[propertyName]; - }); -} - -function getAllPropertyNames(obj: any): string[] { - const prototype = Object.getPrototypeOf(obj); - let inherited = prototype ? getAllPropertyNames(prototype) : []; - inherited = inherited.filter(property => property !== 'constructor'); - return [...new Set(Object.getOwnPropertyNames(obj).concat(inherited))]; + } } diff --git a/packages/schema/src/addSchemaLevelResolver.ts b/packages/schema/src/addSchemaLevelResolver.ts deleted file mode 100644 index c82ba97354e..00000000000 --- a/packages/schema/src/addSchemaLevelResolver.ts +++ /dev/null @@ -1,70 +0,0 @@ -import { defaultFieldResolver, GraphQLSchema, GraphQLFieldResolver } from 'graphql'; - -import { ValueOrPromise } from 'value-or-promise'; - -import { mapSchema, MapperKind } from '@graphql-tools/utils'; - -// wraps all resolvers of query, mutation or subscription fields -// with the provided function to simulate a root schema level resolver -export function addSchemaLevelResolver(schema: GraphQLSchema, fn: GraphQLFieldResolver): GraphQLSchema { - // TODO test that schema is a schema, fn is a function - const fnToRunOnlyOnce = runAtMostOncePerRequest(fn); - return mapSchema(schema, { - [MapperKind.ROOT_FIELD]: (fieldConfig, _fieldName, typeName, schema) => { - // XXX this should run at most once per request to simulate a true root resolver - // for graphql-js this is an approximation that works with queries but not mutations - // XXX if the type is a subscription, a same query AST will be ran multiple times so we - // deactivate here the runOnce if it's a subscription. This may not be optimal though... - const subscription = schema.getSubscriptionType(); - if (subscription != null && subscription.name === typeName) { - return { - ...fieldConfig, - resolve: wrapResolver(fieldConfig.resolve, fn), - }; - } - - return { - ...fieldConfig, - resolve: wrapResolver(fieldConfig.resolve, fnToRunOnlyOnce), - }; - }, - }); -} - -// XXX badly named function. this doesn't really wrap, it just chains resolvers... -function wrapResolver( - innerResolver: GraphQLFieldResolver | undefined, - outerResolver: GraphQLFieldResolver -): GraphQLFieldResolver { - return (obj, args, ctx, info) => { - return new ValueOrPromise(() => outerResolver(obj, args, ctx, info)) - .then(root => { - if (innerResolver != null) { - return innerResolver(root, args, ctx, info); - } - return defaultFieldResolver(root, args, ctx, info); - }) - .resolve(); - } -} - -// XXX this function only works for resolvers -// XXX very hacky way to remember if the function -// already ran for this request. This will only work -// if people don't actually cache the operation. -// if they do cache the operation, they will have to -// manually remove the __runAtMostOnce before every request. -function runAtMostOncePerRequest(fn: GraphQLFieldResolver): GraphQLFieldResolver { - let value: any; - const randomNumber = Math.random(); - return (root, args, ctx, info) => { - if (!info.operation['__runAtMostOnce']) { - info.operation['__runAtMostOnce'] = {}; - } - if (!info.operation['__runAtMostOnce'][randomNumber]) { - info.operation['__runAtMostOnce'][randomNumber] = true; - value = fn(root, args, ctx, info); - } - return value; - }; -} diff --git a/packages/schema/src/attachDirectiveResolvers.ts b/packages/schema/src/attachDirectiveResolvers.ts deleted file mode 100644 index 8ced8d0fc20..00000000000 --- a/packages/schema/src/attachDirectiveResolvers.ts +++ /dev/null @@ -1,49 +0,0 @@ -import { GraphQLSchema, defaultFieldResolver } from 'graphql'; - -import { IDirectiveResolvers, mapSchema, MapperKind, getDirectives } from '@graphql-tools/utils'; - -export function attachDirectiveResolvers( - schema: GraphQLSchema, - directiveResolvers: IDirectiveResolvers -): GraphQLSchema { - if (typeof directiveResolvers !== 'object') { - throw new Error(`Expected directiveResolvers to be of type object, got ${typeof directiveResolvers}`); - } - - if (Array.isArray(directiveResolvers)) { - throw new Error('Expected directiveResolvers to be of type object, got Array'); - } - - return mapSchema(schema, { - [MapperKind.OBJECT_FIELD]: fieldConfig => { - const newFieldConfig = { ...fieldConfig }; - - const directives = getDirectives(schema, fieldConfig); - Object.keys(directives).forEach(directiveName => { - if (directiveResolvers[directiveName]) { - const resolver = directiveResolvers[directiveName]; - const originalResolver = newFieldConfig.resolve != null ? newFieldConfig.resolve : defaultFieldResolver; - const directiveArgs = directives[directiveName]; - newFieldConfig.resolve = (source, originalArgs, context, info) => { - return resolver( - () => - new Promise((resolve, reject) => { - const result = originalResolver(source, originalArgs, context, info); - if (result instanceof Error) { - reject(result); - } - resolve(result); - }), - source, - directiveArgs, - context, - info - ); - }; - } - }); - - return newFieldConfig; - }, - }); -} diff --git a/packages/schema/src/buildSchemaFromTypeDefinitions.ts b/packages/schema/src/buildSchemaFromTypeDefinitions.ts deleted file mode 100644 index 6836602caa9..00000000000 --- a/packages/schema/src/buildSchemaFromTypeDefinitions.ts +++ /dev/null @@ -1,48 +0,0 @@ -import { extendSchema, buildASTSchema, GraphQLSchema, DocumentNode } from 'graphql'; - -import { ITypeDefinitions, GraphQLParseOptions, parseGraphQLSDL, isDocumentNode } from '@graphql-tools/utils'; - -import { filterAndExtractExtensionDefinitions } from './extensionDefinitions'; -import { concatenateTypeDefs } from './concatenateTypeDefs'; - -export function buildSchemaFromTypeDefinitions( - typeDefinitions: ITypeDefinitions, - parseOptions?: GraphQLParseOptions, - noExtensionExtraction?: boolean -): GraphQLSchema { - const document = buildDocumentFromTypeDefinitions(typeDefinitions, parseOptions); - - if (noExtensionExtraction) { - return buildASTSchema(document); - } - - const { typesAst, extensionsAst } = filterAndExtractExtensionDefinitions(document); - - const backcompatOptions = { commentDescriptions: true }; - let schema: GraphQLSchema = buildASTSchema(typesAst, backcompatOptions); - - if (extensionsAst.definitions.length > 0) { - schema = extendSchema(schema, extensionsAst, backcompatOptions); - } - - return schema; -} - -export function buildDocumentFromTypeDefinitions( - typeDefinitions: ITypeDefinitions, - parseOptions?: GraphQLParseOptions -): DocumentNode { - let document: DocumentNode; - if (typeof typeDefinitions === 'string') { - document = parseGraphQLSDL('', typeDefinitions, parseOptions).document; - } else if (Array.isArray(typeDefinitions)) { - document = parseGraphQLSDL('', concatenateTypeDefs(typeDefinitions), parseOptions).document; - } else if (isDocumentNode(typeDefinitions)) { - document = typeDefinitions; - } else { - const type = typeof typeDefinitions; - throw new Error(`typeDefs must be a string, array or schema AST, got ${type}`); - } - - return document; -} diff --git a/packages/schema/src/concatenateTypeDefs.ts b/packages/schema/src/concatenateTypeDefs.ts deleted file mode 100644 index 546c60e3b04..00000000000 --- a/packages/schema/src/concatenateTypeDefs.ts +++ /dev/null @@ -1,26 +0,0 @@ -import { print, ASTNode } from 'graphql'; - -import { ITypedef } from '@graphql-tools/utils'; - -export function concatenateTypeDefs( - typeDefinitionsAry: Array, - calledFunctionRefs = new Set() -): string { - const resolvedTypeDefinitions = new Set(); - typeDefinitionsAry.forEach((typeDef: ITypedef) => { - if (typeof typeDef === 'function') { - if (!calledFunctionRefs.has(typeDef)) { - calledFunctionRefs.add(typeDef); - resolvedTypeDefinitions.add(concatenateTypeDefs(typeDef(), calledFunctionRefs)); - } - } else if (typeof typeDef === 'string') { - resolvedTypeDefinitions.add(typeDef.trim()); - } else if ((typeDef as ASTNode).kind !== undefined) { - resolvedTypeDefinitions.add(print(typeDef).trim()); - } else { - const type = typeof typeDef; - throw new Error(`typeDef array must contain only strings, documents, or functions, got ${type}`); - } - }); - return [...resolvedTypeDefinitions].join('\n'); -} diff --git a/packages/schema/src/decorateWithLogger.ts b/packages/schema/src/decorateWithLogger.ts deleted file mode 100644 index 3345a1e8de3..00000000000 --- a/packages/schema/src/decorateWithLogger.ts +++ /dev/null @@ -1,50 +0,0 @@ -import { defaultFieldResolver, GraphQLFieldResolver } from 'graphql'; -import { Maybe } from 'packages/graphql-tools/src'; -import { ILogger } from './types'; - -/* - * fn: The function to decorate with the logger - * logger: an object instance of type Logger - * hint: an optional hint to add to the error's message - */ -export function decorateWithLogger( - fn: Maybe>, - logger: ILogger, - hint: string -): GraphQLFieldResolver { - const resolver = fn != null ? fn : defaultFieldResolver; - - const logError = (e: Error) => { - // TODO: clone the error properly - const newE = new Error(); - newE.stack = e.stack; - /* istanbul ignore else: always get the hint from addErrorLoggingToSchema */ - if (hint) { - newE['originalMessage'] = e.message; - newE.message = `Error in resolver ${hint}\n${e.message}`; - } - logger.log(newE); - }; - - return (root, args, ctx, info) => { - try { - const result = resolver(root, args, ctx, info); - // If the resolver returns a Promise log any Promise rejects. - if (result && typeof result.then === 'function' && typeof result.catch === 'function') { - result.catch((reason: Error | string) => { - // make sure that it's an error we're logging. - const error = reason instanceof Error ? reason : new Error(reason); - logError(error); - - // We don't want to leave an unhandled exception so pass on error. - return reason; - }); - } - return result; - } catch (e) { - logError(e); - // we want to pass on the error, just in case. - throw e; - } - }; -} diff --git a/packages/schema/src/extendResolversFromInterfaces.ts b/packages/schema/src/extendResolversFromInterfaces.ts index b9233ef60ce..e5c9b1abea0 100644 --- a/packages/schema/src/extendResolversFromInterfaces.ts +++ b/packages/schema/src/extendResolversFromInterfaces.ts @@ -3,28 +3,21 @@ import { GraphQLSchema } from 'graphql'; import { IResolvers, IObjectTypeResolver } from '@graphql-tools/utils'; export function extendResolversFromInterfaces(schema: GraphQLSchema, resolvers: IResolvers): IResolvers { - const typeNames = Object.keys({ - ...schema.getTypeMap(), - ...resolvers, - }); - const extendedResolvers = {}; - typeNames.forEach(typeName => { - const type = schema.getType(typeName); - if (type && 'getInterfaces' in type) { - const allInterfaceResolvers = type - .getInterfaces() - .map(iFace => resolvers[iFace.name]) - .filter(interfaceResolvers => interfaceResolvers != null); - + const typeMap = schema.getTypeMap(); + for (const typeName in typeMap) { + const type = typeMap[typeName]; + if ('getInterfaces' in type) { extendedResolvers[typeName] = {}; - allInterfaceResolvers.forEach(interfaceResolvers => { - Object.keys(interfaceResolvers).forEach(fieldName => { - if (fieldName === '__isTypeOf' || !fieldName.startsWith('__')) { - extendedResolvers[typeName][fieldName] = interfaceResolvers[fieldName]; + for (const iFace of type.getInterfaces()) { + if (resolvers[iFace.name]) { + for (const fieldName in resolvers[iFace.name]) { + if (fieldName === '__isTypeOf' || !fieldName.startsWith('__')) { + extendedResolvers[typeName][fieldName] = resolvers[iFace.name][fieldName]; + } } - }); - }); + } + } const typeResolvers = resolvers[typeName] as Record; extendedResolvers[typeName] = { @@ -37,7 +30,7 @@ export function extendResolversFromInterfaces(schema: GraphQLSchema, resolvers: extendedResolvers[typeName] = typeResolvers; } } - }); + } return extendedResolvers; } diff --git a/packages/schema/src/extensionDefinitions.ts b/packages/schema/src/extensionDefinitions.ts deleted file mode 100644 index d509b1da691..00000000000 --- a/packages/schema/src/extensionDefinitions.ts +++ /dev/null @@ -1,43 +0,0 @@ -import { DocumentNode, DefinitionNode, Kind } from 'graphql'; - -const isExtensionNode = (def: DefinitionNode) => - def.kind === Kind.OBJECT_TYPE_EXTENSION || - def.kind === Kind.INTERFACE_TYPE_EXTENSION || - def.kind === Kind.INPUT_OBJECT_TYPE_EXTENSION || - def.kind === Kind.UNION_TYPE_EXTENSION || - def.kind === Kind.ENUM_TYPE_EXTENSION || - def.kind === Kind.SCALAR_TYPE_EXTENSION || - def.kind === Kind.SCHEMA_EXTENSION; - -export function filterAndExtractExtensionDefinitions(ast: DocumentNode) { - const extensionDefs: DefinitionNode[] = []; - const typesDefs: DefinitionNode[] = []; - ast.definitions.forEach(def => { - if (isExtensionNode(def)) { - extensionDefs.push(def); - } else { - typesDefs.push(def); - } - }); - - return { - typesAst: { - ...ast, - definitions: typesDefs, - }, - extensionsAst: { - ...ast, - definitions: extensionDefs, - }, - }; -} - -export function filterExtensionDefinitions(ast: DocumentNode) { - const { typesAst } = filterAndExtractExtensionDefinitions(ast); - return typesAst; -} - -export function extractExtensionDefinitions(ast: DocumentNode) { - const { extensionsAst } = filterAndExtractExtensionDefinitions(ast); - return extensionsAst; -} diff --git a/packages/schema/src/index.ts b/packages/schema/src/index.ts index 15ad2b2fafd..4d090541b25 100644 --- a/packages/schema/src/index.ts +++ b/packages/schema/src/index.ts @@ -1,15 +1,7 @@ -export { addSchemaLevelResolver } from './addSchemaLevelResolver'; export { assertResolversPresent } from './assertResolversPresent'; -export { attachDirectiveResolvers } from './attachDirectiveResolvers'; -export { buildSchemaFromTypeDefinitions, buildDocumentFromTypeDefinitions } from './buildSchemaFromTypeDefinitions'; export { chainResolvers } from './chainResolvers'; -export { concatenateTypeDefs } from './concatenateTypeDefs'; -export { decorateWithLogger } from './decorateWithLogger'; -export * from './extensionDefinitions'; export { addResolversToSchema } from './addResolversToSchema'; export { checkForResolveTypeResolver } from './checkForResolveTypeResolver'; export { extendResolversFromInterfaces } from './extendResolversFromInterfaces'; -export { addErrorLoggingToSchema } from './addErrorLoggingToSchema'; -export { addCatchUndefinedToSchema } from './addCatchUndefinedToSchema'; export * from './makeExecutableSchema'; export * from './types'; diff --git a/packages/schema/src/makeExecutableSchema.ts b/packages/schema/src/makeExecutableSchema.ts index 552b47520cf..8dc6c0de675 100644 --- a/packages/schema/src/makeExecutableSchema.ts +++ b/packages/schema/src/makeExecutableSchema.ts @@ -1,15 +1,11 @@ -import { GraphQLFieldResolver } from 'graphql'; +import { buildASTSchema, buildSchema, GraphQLSchema } from 'graphql'; -import { mergeDeep, SchemaDirectiveVisitor, pruneSchema } from '@graphql-tools/utils'; +import { pruneSchema } from '@graphql-tools/utils'; import { addResolversToSchema } from './addResolversToSchema'; -import { attachDirectiveResolvers } from './attachDirectiveResolvers'; import { assertResolversPresent } from './assertResolversPresent'; -import { addSchemaLevelResolver } from './addSchemaLevelResolver'; -import { buildSchemaFromTypeDefinitions } from './buildSchemaFromTypeDefinitions'; -import { addErrorLoggingToSchema } from './addErrorLoggingToSchema'; -import { addCatchUndefinedToSchema } from './addCatchUndefinedToSchema'; import { ExecutableSchemaTransformation, IExecutableSchemaDefinition } from './types'; +import { mergeResolvers, mergeTypeDefs } from '@graphql-tools/merge'; /** * Builds a schema from the provided type definitions and resolvers. @@ -58,17 +54,12 @@ import { ExecutableSchemaTransformation, IExecutableSchemaDefinition } from './t export function makeExecutableSchema({ typeDefs, resolvers = {}, - logger, - allowUndefinedInResolve = true, resolverValidationOptions = {}, - directiveResolvers, - schemaDirectives, schemaTransforms: userProvidedSchemaTransforms, parseOptions = {}, inheritResolversFromInterfaces = false, pruningOptions, updateResolversInPlace = false, - noExtensionExtraction = false, }: IExecutableSchemaDefinition) { // Validate and clean up arguments if (typeof resolverValidationOptions !== 'object') { @@ -83,11 +74,10 @@ export function makeExecutableSchema({ const schemaTransforms: ExecutableSchemaTransformation[] = [ schema => { // We allow passing in an array of resolver maps, in which case we merge them - const resolverMap: any = Array.isArray(resolvers) ? resolvers.reduce(mergeDeep, {}) : resolvers; const schemaWithResolvers = addResolversToSchema({ schema, - resolvers: resolverMap, + resolvers: mergeResolvers(resolvers), resolverValidationOptions, inheritResolversFromInterfaces, updateResolversInPlace, @@ -101,46 +91,28 @@ export function makeExecutableSchema({ }, ]; - if (!allowUndefinedInResolve) { - schemaTransforms.push(addCatchUndefinedToSchema); - } - - if (logger != null) { - schemaTransforms.push(schema => addErrorLoggingToSchema(schema, logger)); - } - - if (typeof resolvers['__schema'] === 'function') { - // TODO a bit of a hack now, better rewrite generateSchema to attach it there. - // not doing that now, because I'd have to rewrite a lot of tests. - schemaTransforms.push(schema => - addSchemaLevelResolver(schema, resolvers['__schema'] as GraphQLFieldResolver) - ); - } - if (userProvidedSchemaTransforms) { schemaTransforms.push(schema => userProvidedSchemaTransforms.reduce((s, schemaTransform) => schemaTransform(s), schema) ); } - // directive resolvers are implemented using SchemaDirectiveVisitor.visitSchemaDirectives - // schema visiting modifies the schema in place - if (directiveResolvers != null) { - schemaTransforms.push(schema => attachDirectiveResolvers(schema, directiveResolvers)); - } - - if (schemaDirectives != null) { - schemaTransforms.push(schema => { - SchemaDirectiveVisitor.visitSchemaDirectives(schema, schemaDirectives); - return schema; - }); - } - if (pruningOptions) { schemaTransforms.push(pruneSchema); } - const schemaFromTypeDefs = buildSchemaFromTypeDefinitions(typeDefs, parseOptions, noExtensionExtraction); + let schemaFromTypeDefs: GraphQLSchema; + + if (parseOptions?.commentDescriptions) { + const mergedTypeDefs = mergeTypeDefs(typeDefs, { + ...parseOptions, + commentDescriptions: true, + }); + schemaFromTypeDefs = buildSchema(mergedTypeDefs, parseOptions); + } else { + const mergedTypeDefs = mergeTypeDefs(typeDefs, parseOptions); + schemaFromTypeDefs = buildASTSchema(mergedTypeDefs, parseOptions); + } return schemaTransforms.reduce((schema, schemaTransform) => schemaTransform(schema), schemaFromTypeDefs); } diff --git a/packages/schema/src/types.ts b/packages/schema/src/types.ts index e263d1340c1..ed28779d686 100644 --- a/packages/schema/src/types.ts +++ b/packages/schema/src/types.ts @@ -1,19 +1,13 @@ import { GraphQLSchema } from 'graphql'; import { - ITypeDefinitions, + TypeSource, IResolvers, IResolverValidationOptions, - IDirectiveResolvers, - SchemaDirectiveVisitorClass, GraphQLParseOptions, PruneSchemaOptions, } from '@graphql-tools/utils'; -export interface ILogger { - log: (error: Error) => void; -} - /** * Configuration object for creating an executable schema */ @@ -21,34 +15,15 @@ export interface IExecutableSchemaDefinition { /** * The type definitions used to create the schema */ - typeDefs: ITypeDefinitions; + typeDefs: TypeSource; /** * Object describing the field resolvers for the provided type definitions */ resolvers?: IResolvers | Array>; - /** - * Logger instance used to print errors to the server console that are - * usually swallowed by GraphQL. - */ - logger?: ILogger; - /** - * Set to `false` to have resolvers throw an if they return undefined, which - * can help make debugging easier - */ - allowUndefinedInResolve?: boolean; /** * Additional options for validating the provided resolvers */ resolverValidationOptions?: IResolverValidationOptions; - /** - * Map of directive resolvers - */ - directiveResolvers?: IDirectiveResolvers; - /** - * A map of schema directives used with the legacy class-based implementation - * of schema directives - */ - schemaDirectives?: Record; /** * An array of schema transformation functions */ @@ -71,10 +46,6 @@ export interface IExecutableSchemaDefinition { * Do not create a schema again and use the one from `buildASTSchema` */ updateResolversInPlace?: boolean; - /** - * Do not extract and apply extensions separately and leave it to `buildASTSchema` - */ - noExtensionExtraction?: boolean; } export type ExecutableSchemaTransformation = (schema: GraphQLSchema) => GraphQLSchema; diff --git a/packages/schema/tests/Logger.ts b/packages/schema/tests/Logger.ts deleted file mode 100644 index 5a50ace70e9..00000000000 --- a/packages/schema/tests/Logger.ts +++ /dev/null @@ -1,36 +0,0 @@ -/* - * A very simple class for logging errors - */ - -import { ILogger } from '@graphql-tools/schema'; - -export class Logger implements ILogger { - public errors: Array; - public name: string | undefined; - private readonly callback: undefined | ((...args: any[]) => any); - - constructor(name?: string, callback?: (...args: any[]) => any) { - this.name = name; - this.errors = []; - this.callback = callback; - // TODO: should assert that callback is a function - } - - public log(err: Error) { - this.errors.push(err); - if (typeof this.callback === 'function') { - this.callback(err); - } - } - - public printOneError(e: Error): string { - return e.stack ? e.stack : ''; - } - - public printAllErrors() { - return this.errors.reduce( - (agg: string, e: Error) => `${agg}\n${this.printOneError(e)}`, - '', - ); - } -} diff --git a/packages/schema/tests/extensionExtraction.test.ts b/packages/schema/tests/extensionExtraction.test.ts deleted file mode 100644 index b972530441d..00000000000 --- a/packages/schema/tests/extensionExtraction.test.ts +++ /dev/null @@ -1,65 +0,0 @@ -import { parse } from 'graphql'; - -import { extractExtensionDefinitions } from '@graphql-tools/schema'; - -describe('Extension extraction', () => { - test('extracts extended inputs', () => { - const typeDefs = ` - input Input { - foo: String - } - - extend input Input { - bar: String - } - `; - - const astDocument = parse(typeDefs); - const extensionAst = extractExtensionDefinitions(astDocument); - - expect(extensionAst.definitions).toHaveLength(1); - expect(extensionAst.definitions[0].kind).toBe('InputObjectTypeExtension'); - }); - - test('extracts extended unions', () => { - const typeDefs = ` - type Person { - name: String! - } - type Location { - name: String! - } - union Searchable = Person | Location - - type Post { - name: String! - } - extend union Searchable = Post - `; - - const astDocument = parse(typeDefs); - const extensionAst = extractExtensionDefinitions(astDocument); - - expect(extensionAst.definitions).toHaveLength(1); - expect(extensionAst.definitions[0].kind).toBe('UnionTypeExtension'); - }); - - test('extracts extended enums', () => { - const typeDefs = ` - enum Color { - RED - GREEN - } - - extend enum Color { - BLUE - } - `; - - const astDocument = parse(typeDefs); - const extensionAst = extractExtensionDefinitions(astDocument); - - expect(extensionAst.definitions).toHaveLength(1); - expect(extensionAst.definitions[0].kind).toBe('EnumTypeExtension'); - }); -}); diff --git a/packages/schema/tests/logger.test.ts b/packages/schema/tests/logger.test.ts deleted file mode 100644 index d7fc6c0e1e0..00000000000 --- a/packages/schema/tests/logger.test.ts +++ /dev/null @@ -1,409 +0,0 @@ -/* eslint-disable promise/param-names */ -import { graphql } from 'graphql'; - -import { makeExecutableSchema } from '@graphql-tools/schema'; - -import { Logger } from './Logger'; - -describe('Logger', () => { - test('logs the errors', () => { - const shorthand = ` - type RootQuery { - just_a_field: Int - } - type RootMutation { - species(name: String): String - stuff: String - } - schema { - query: RootQuery - mutation: RootMutation - } - `; - const resolve = { - RootMutation: { - species: () => { - throw new Error('oops!'); - }, - stuff: () => { - throw new Error('oh noes!'); - }, - }, - }; - const logger = new Logger(); - const jsSchema = makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolve, - logger, - }); - // calling the mutation here so the errors will be ordered. - const testQuery = 'mutation { species, stuff }'; - const expected0 = 'Error in resolver RootMutation.species\noops!'; - const expected1 = 'Error in resolver RootMutation.stuff\noh noes!'; - return graphql(jsSchema, testQuery).then(() => { - expect(logger.errors.length).toEqual(2); - expect(logger.errors[0].message).toEqual(expected0); - expect(logger.errors[1].message).toEqual(expected1); - }); - }); - - test('also forwards the errors when you tell it to', () => { - const shorthand = ` - type RootQuery { - species(name: String): String - } - schema { - query: RootQuery - } - `; - const resolve = { - RootQuery: { - species: () => { - throw new Error('oops!'); - }, - }, - }; - let loggedErr: Error | null = null; - const logger = new Logger('LoggyMcLogface', (e: Error) => { - loggedErr = e; - }); - const jsSchema = makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolve, - logger, - }); - const testQuery = '{ species }'; - return graphql(jsSchema, testQuery).then(() => { - expect(loggedErr).toEqual(logger.errors[0]); - }); - }); - - test('prints the errors when you want it to', () => { - const shorthand = ` - type RootQuery { - species(name: String): String - } - schema { - query: RootQuery - } - `; - const resolve = { - RootQuery: { - species: (_root: any, { name }: { name: string }) => { - if (name) { - throw new Error(name); - } - throw new Error('oops!'); - }, - }, - }; - const logger = new Logger(); - const jsSchema = makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolve, - logger, - }); - const testQuery = '{ q: species, p: species(name: "Peter") }'; - return graphql(jsSchema, testQuery).then(() => { - const allErrors = logger.printAllErrors(); - expect(allErrors).toMatch(/oops/); - expect(allErrors).toMatch(/Peter/); - }); - }); - - test('logs any Promise reject errors', () => { - const shorthand = ` - type RootQuery { - just_a_field: Int - } - type RootMutation { - species(name: String): String - stuff: String - } - schema { - query: RootQuery - mutation: RootMutation - } - `; - const resolve = { - RootMutation: { - species: () => - new Promise((_, reject) => { - reject(new Error('oops!')); - }), - stuff: () => - new Promise((_, reject) => { - reject(new Error('oh noes!')); - }), - }, - }; - const logger = new Logger(); - const jsSchema = makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolve, - logger, - }); - - const testQuery = 'mutation { species, stuff }'; - const expected0 = 'Error in resolver RootMutation.species\noops!'; - const expected1 = 'Error in resolver RootMutation.stuff\noh noes!'; - return graphql(jsSchema, testQuery).then(() => { - expect(logger.errors.length).toEqual(2); - expect(logger.errors[0].message).toEqual(expected0); - expect(logger.errors[1].message).toEqual(expected1); - }); - }); - - test('all Promise rejects will log an Error', () => { - const shorthand = ` - type RootQuery { - species(name: String): String - } - schema { - query: RootQuery - } - `; - const resolve = { - RootQuery: { - species: () => - new Promise((_, reject) => { - reject(new Error('oops!')); - }), - }, - }; - - let loggedErr: Error | null = null; - const logger = new Logger('LoggyMcLogface', (e: Error) => { - loggedErr = e; - }); - const jsSchema = makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolve, - logger, - }); - - const testQuery = '{ species }'; - return graphql(jsSchema, testQuery).then(() => { - expect(loggedErr).toEqual(logger.errors[0]); - }); - }); -}); - - - -describe('providing useful errors from resolvers', () => { - test('logs an error if a resolver fails', () => { - const shorthand = ` - type RootQuery { - species(name: String): String - } - schema { - query: RootQuery - } - `; - const resolve = { - RootQuery: { - species: (): string => { - throw new Error('oops!'); - }, - }, - }; - - // TODO: Should use a spy here instead of logger class - // to make sure we don't duplicate tests from Logger. - const logger = new Logger(); - const jsSchema = makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolve, - logger, - }); - const testQuery = '{ species }'; - const expected = 'Error in resolver RootQuery.species\noops!'; - return graphql(jsSchema, testQuery).then((_res) => { - expect(logger.errors.length).toEqual(1); - expect(logger.errors[0].message).toEqual(expected); - }); - }); - - test('will throw errors on undefined if you tell it to', () => { - const shorthand = ` - type RootQuery { - species(name: String): String - stuff: String - } - schema { - query: RootQuery - } - `; - const resolve = { - RootQuery: { - species: (): string | undefined => undefined, - stuff: () => 'stuff', - }, - }; - - const logger = new Logger(); - const jsSchema = makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolve, - logger, - allowUndefinedInResolve: false, - }); - const testQuery = '{ species, stuff }'; - const expectedErr = /Resolver for "RootQuery.species" returned undefined/; - const expectedResData = { species: null as string | null, stuff: 'stuff' }; - return graphql(jsSchema, testQuery).then((res) => { - expect(logger.errors.length).toEqual(1); - expect(logger.errors[0].message).toMatch(expectedErr); - expect(res.data).toEqual(expectedResData); - }); - }); - - test('decorateToCatchUndefined preserves default resolvers', () => { - const shorthand = ` - type Thread { - name: String - } - type RootQuery { - thread(name: String): Thread - } - schema { - query: RootQuery - } - `; - const resolve = { - RootQuery: { - thread(_root: any, args: Record) { - return args; - }, - }, - }; - - const jsSchema = makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolve, - allowUndefinedInResolve: false, - }); - const testQuery = `{ - thread(name: "SomeThread") { - name - } - }`; - const expectedResData = { - thread: { - name: 'SomeThread', - }, - }; - return graphql(jsSchema, testQuery).then((res) => { - expect(res.data).toEqual(expectedResData); - }); - }); - - test('decorateToCatchUndefined throws even if default resolvers are preserved', () => { - const shorthand = ` - type Thread { - name: String - } - type RootQuery { - thread(name: String): Thread - } - schema { - query: RootQuery - } - `; - const resolve = { - RootQuery: { - thread(_root: any, _args: Record) { - return { name: (): any => undefined }; - }, - }, - }; - - const jsSchema = makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolve, - allowUndefinedInResolve: false, - }); - const testQuery = `{ - thread { - name - } - }`; - return graphql(jsSchema, testQuery).then((res) => { - expect(res.errors?.[0].originalError?.message).toBe( - 'Resolver for "Thread.name" returned undefined', - ); - }); - }); - - test('will use default resolver when returning function properties ', () => { - const shorthand = ` - type Thread { - name: String - } - type RootQuery { - thread(name: String): Thread - } - schema { - query: RootQuery - } - `; - const resolve = { - RootQuery: { - thread(_root: any, args: Record) { - return { name: () => args.name }; - }, - }, - }; - - const jsSchema = makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolve, - allowUndefinedInResolve: false, - }); - const testQuery = `{ - thread(name: "SomeThread") { - name - } - }`; - const expectedResData = { - thread: { - name: 'SomeThread', - }, - }; - return graphql(jsSchema, testQuery).then((res) => { - expect(res.data).toEqual(expectedResData); - }); - }); - - test('will not throw errors on undefined by default', () => { - const shorthand = ` - type RootQuery { - species(name: String): String - stuff: String - } - schema { - query: RootQuery - } - `; - const resolve = { - RootQuery: { - species: (): string | undefined => undefined, - stuff: () => 'stuff', - }, - }; - - const logger = new Logger(); - const jsSchema = makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolve, - logger, - }); - const testQuery = '{ species, stuff }'; - const expectedResData = { species: null as string | null, stuff: 'stuff' }; - return graphql(jsSchema, testQuery).then((res) => { - expect(logger.errors.length).toEqual(0); - expect(res.data).toEqual(expectedResData); - }); - }); -}); diff --git a/packages/schema/tests/resolution.test.ts b/packages/schema/tests/resolution.test.ts deleted file mode 100644 index 9cc71a3c43c..00000000000 --- a/packages/schema/tests/resolution.test.ts +++ /dev/null @@ -1,148 +0,0 @@ -/* eslint-disable promise/param-names */ -import { - parse, - graphql, - subscribe, - graphqlSync, -} from 'graphql'; -import { PubSub } from 'graphql-subscriptions'; - -import { makeExecutableSchema, addSchemaLevelResolver } from '@graphql-tools/schema'; - -import { ExecutionResult } from '@graphql-tools/utils'; - -describe('Resolve', () => { - describe('addSchemaLevelResolver', () => { - const pubsub = new PubSub(); - const typeDefs = ` - type RootQuery { - printRoot: String! - printRootAgain: String! - } - - type RootMutation { - printRoot: String! - } - - type RootSubscription { - printRoot: String! - } - - schema { - query: RootQuery - mutation: RootMutation - subscription: RootSubscription - } - `; - const printRoot = (root: any) => root.toString(); - const resolvers = { - RootQuery: { - printRoot, - printRootAgain: printRoot, - }, - RootMutation: { - printRoot, - }, - RootSubscription: { - printRoot: { - subscribe: () => pubsub.asyncIterator('printRootChannel'), - }, - }, - }; - let schema = makeExecutableSchema({ typeDefs, resolvers }); - let schemaLevelResolverCalls = 0; - schema = addSchemaLevelResolver(schema, (root) => { - schemaLevelResolverCalls += 1; - return root; - }); - - test('should run the schema level resolver once in a same query', () => { - schemaLevelResolverCalls = 0; - const root = 'queryRoot'; - return graphql( - schema, - ` - query TestOnce { - printRoot - printRootAgain - } - `, - root, - ).then(({ data }) => { - expect(data).toEqual({ - printRoot: root, - printRootAgain: root, - }); - expect(schemaLevelResolverCalls).toEqual(1); - }); - }); - - test('should isolate roots from the different operation types', async () => { - schemaLevelResolverCalls = 0; - const queryRoot = 'queryRoot'; - const mutationRoot = 'mutationRoot'; - const subscriptionRoot = 'subscriptionRoot'; - const subscriptionRoot2 = 'subscriptionRoot2'; - - const sub = await subscribe( - schema, - parse(` - subscription TestSubscription { - printRoot - } - `), - ) as AsyncIterableIterator; - - const payload1 = sub.next(); - await pubsub.publish('printRootChannel', { printRoot: subscriptionRoot }); - - expect(await payload1).toEqual({ done: false, value: { data: { printRoot: subscriptionRoot } } }); - expect(schemaLevelResolverCalls).toEqual(1); - - const queryResult = await graphql( - schema, - ` - query TestQuery { - printRoot - } - `, - queryRoot, - ); - - expect(queryResult).toEqual({ data: { printRoot: queryRoot } }); - expect(schemaLevelResolverCalls).toEqual(2); - - const mutationResult = await graphql( - schema, - ` - mutation TestMutation { - printRoot - } - `, - mutationRoot, - ); - - expect(mutationResult).toEqual({ data: { printRoot: mutationRoot } }); - expect(schemaLevelResolverCalls).toEqual(3); - - await pubsub.publish('printRootChannel', { printRoot: subscriptionRoot2 }); - - expect(await sub.next()).toEqual({ done: false, value: { data: { printRoot: subscriptionRoot2 } } }); - expect(schemaLevelResolverCalls).toEqual(4); - }); - - it('should not force an otherwise synchronous operation to be asynchronous', () => { - const queryRoot = 'queryRoot'; - // This will throw an error if schema has any asynchronous resolvers - graphqlSync( - schema, - ` - query TestQuery { - printRoot - } - `, - queryRoot, - ); - }); - }); -}); diff --git a/packages/schema/tests/schemaGenerator.test.ts b/packages/schema/tests/schemaGenerator.test.ts index 483192650a7..ce83fdae67d 100644 --- a/packages/schema/tests/schemaGenerator.test.ts +++ b/packages/schema/tests/schemaGenerator.test.ts @@ -20,30 +20,20 @@ import { DocumentNode, GraphQLBoolean, graphqlSync, - GraphQLSchema, GraphQLFieldResolver, } from 'graphql'; import { makeExecutableSchema, - addErrorLoggingToSchema, - addSchemaLevelResolver, addResolversToSchema, - attachDirectiveResolvers, chainResolvers, - concatenateTypeDefs, - ILogger, } from '@graphql-tools/schema'; import { IResolverValidationOptions, IResolvers, - IDirectiveResolvers, - NextResolverFn, - VisitSchemaKind, - ITypeDefinitions, - visitSchema, - ExecutionResult + ExecutionResult, + TypeSource } from '@graphql-tools/utils'; import TypeA from './fixtures/circularSchemaA'; @@ -88,7 +78,6 @@ const testSchema = ` } `; const testResolvers = { - __schema: () => ({ stuff: 'stuff', species: 'ROOT' }), RootQuery: { usecontext: (_r: any, _a: Record, ctx: any) => ctx.usecontext, species: (root: any, { name }: { name: string }) => @@ -118,22 +107,22 @@ describe('generating schema from shorthand', () => { test('throws an error if typeDefinitionNodes is neither string nor array nor schema AST', () => { expect(() => makeExecutableSchema({ - typeDefs: ({} as unknown) as ITypeDefinitions, + typeDefs: ({} as unknown) as TypeSource, resolvers: {}, }), ).toThrowError( - 'typeDefs must be a string, array or schema AST, got object', + 'typeDefs must contain only strings, documents, schemas, or functions, got object', ); }); test('throws an error if typeDefinitionNode array contains not only functions and strings', () => { expect(() => makeExecutableSchema({ - typeDefs: ([17] as unknown) as ITypeDefinitions, + typeDefs: ([17] as unknown) as TypeSource, resolvers: {}, }), ).toThrowError( - 'typeDef array must contain only strings, documents, or functions, got number', + 'typeDefs must contain only strings, documents, schemas, or functions, got number', ); }); @@ -360,8 +349,8 @@ describe('generating schema from shorthand', () => { resolvers: {}, }); expect(jsSchema.getQueryType()?.name).toBe('Query'); - expect(jsSchema.getQueryType()?.getFields().foo).toBeDefined(); - expect(jsSchema.getQueryType()?.getFields().bar).toBeDefined(); + expect(jsSchema.getQueryType()?.getFields()['foo']).toBeDefined(); + expect(jsSchema.getQueryType()?.getFields()['bar']).toBeDefined(); }); test('allow for a map of extensions in field resolver', () => { @@ -384,33 +373,9 @@ describe('generating schema from shorthand', () => { }, }, }); - const extensions = jsSchema.getQueryType()?.getFields().foo.extensions; + const extensions = jsSchema.getQueryType()?.getFields()['foo'].extensions; expect(extensions).toHaveProperty('verbose'); - expect(extensions!.verbose).toBe(true); - }); - - test('can concatenateTypeDefs created by a function inside a closure', () => { - const typeA = { typeDefs: () => ['type TypeA { foo: String }'] }; - const typeB = { typeDefs: () => ['type TypeB { bar: String }'] }; - const typeC = { typeDefs: () => ['type TypeC { foo: String }'] }; - const typeD = { typeDefs: () => ['type TypeD { bar: String }'] }; - - function combineTypeDefs(...args: Array): any { - return { typeDefs: () => args.map((o) => o.typeDefs) }; - } - - const combinedAandB = combineTypeDefs(typeA, typeB); - const combinedCandD = combineTypeDefs(typeC, typeD); - - const result = concatenateTypeDefs([ - combinedAandB.typeDefs, - combinedCandD.typeDefs, - ]); - - expect(result).toMatch('type TypeA'); - expect(result).toMatch('type TypeB'); - expect(result).toMatch('type TypeC'); - expect(result).toMatch('type TypeD'); + expect(extensions!['verbose']).toBe(true); }); test('properly deduplicates the array of type DefinitionNodes', () => { @@ -751,7 +716,7 @@ describe('generating schema from shorthand', () => { const QueryResolver = class QueryResolver { private internalVersion = 1 - version(root: any, args: any, context: any) { + version() { return this.internalVersion } } @@ -871,7 +836,7 @@ describe('generating schema from shorthand', () => { } `; const result = graphqlSync(jsSchema, testQuery); - expect(result.data!.foo.aField).toBe(false); + expect(result.data!['foo'].aField).toBe(false); jsSchema = addResolversToSchema({ schema: jsSchema, resolvers: { @@ -911,80 +876,6 @@ describe('generating schema from shorthand', () => { expect(testType!.astNode!.directives!.length).toBe(1); }); - test('retains scalars after walking/recreating the schema', () => { - const shorthand = ` - scalar Test - - type Foo { - testField: Test - } - - type Query { - test: Test - testIn(input: Test): Test - } - `; - const resolveFunctions = { - Test: new GraphQLScalarType({ - name: 'Test', - description: 'Test resolver', - serialize(value) { - if (typeof value !== 'string' || value.indexOf('scalar:') !== 0) { - return `scalar:${value as string}`; - } - return value; - }, - parseValue(value) { - return `scalar:${value as string}`; - }, - parseLiteral(ast: any) { - switch (ast.kind) { - case Kind.STRING: - case Kind.INT: - return `scalar:${ast.value as string}`; - default: - return null; - } - }, - }), - Query: { - testIn(_: any, { input }: any) { - expect(input).toMatch('scalar:'); - return input; - }, - test() { - return 42; - }, - }, - }; - const walkedSchema = visitSchema( - makeExecutableSchema({ - typeDefs: shorthand, - resolvers: resolveFunctions, - }), - { - [VisitSchemaKind.ENUM_TYPE](type: GraphQLEnumType) { - return type; - }, - }, - ); - expect(walkedSchema.getType('Test')).toBeInstanceOf(GraphQLScalarType); - expect(walkedSchema.getType('Test')).toHaveProperty('description'); - expect(walkedSchema.getType('Test')!.description).toBe('Test resolver'); - const testQuery = ` - { - test - testIn(input: 1) - }`; - const resultPromise = graphql(walkedSchema, testQuery); - return resultPromise.then((result) => - expect(result.data).toEqual({ - test: 'scalar:42', - testIn: 'scalar:1', - }), - ); - }); - test('should support custom scalar usage on client-side query execution', () => { const shorthand = ` scalar CustomScalar @@ -1099,7 +990,7 @@ describe('generating schema from shorthand', () => { `; const resultPromise = graphql(jsSchema, testQuery); return resultPromise.then((result) => { - expect(result.data!.post.something).toEqual(testValue); + expect(result.data!['post'].something).toEqual(testValue); expect(result.errors).toEqual(undefined); }); }); @@ -1169,7 +1060,7 @@ describe('generating schema from shorthand', () => { `; const resultPromise = graphql(jsSchema, testQuery); return resultPromise.then((result) => { - expect(result.data!.post.something).toEqual(testDate.getTime()); + expect(result.data!['post'].something).toEqual(testDate.getTime()); expect(result.errors).toEqual(undefined); }); }); @@ -1271,9 +1162,9 @@ describe('generating schema from shorthand', () => { const resultPromise = graphql(jsSchema, testQuery); return resultPromise.then((result) => { - expect(result.data!.redColor).toEqual('RED'); - expect(result.data!.blueColor).toEqual('BLUE'); - expect(result.data!.numericEnum).toEqual('TEST'); + expect(result.data!['redColor']).toEqual('RED'); + expect(result.data!['blueColor']).toEqual('BLUE'); + expect(result.data!['numericEnum']).toEqual('TEST'); expect(result.errors).toEqual(undefined); }); }); @@ -1330,9 +1221,9 @@ describe('generating schema from shorthand', () => { const resultPromise = graphql(jsSchema, testQuery); return resultPromise.then((result) => { - expect(result.data!.red).toEqual(resolveFunctions.Color.RED); - expect(result.data!.blue).toEqual(resolveFunctions.Color.BLUE); - expect(result.data!.num).toEqual(resolveFunctions.NumericEnum.TEST); + expect(result.data!['red']).toEqual(resolveFunctions.Color.RED); + expect(result.data!['blue']).toEqual(resolveFunctions.Color.BLUE); + expect(result.data!['num']).toEqual(resolveFunctions.NumericEnum.TEST); expect(result.errors).toEqual(undefined); }); }); @@ -1376,7 +1267,7 @@ describe('generating schema from shorthand', () => { const resultPromise = graphql(jsSchema, testQuery); return resultPromise.then((result) => { - expect(result.data!.red).toEqual(resolveFunctions.Color.RED); + expect(result.data!['red']).toEqual(resolveFunctions.Color.RED); expect(result.errors).toEqual(undefined); }); }); @@ -1427,7 +1318,7 @@ describe('generating schema from shorthand', () => { const resultPromise = graphql(jsSchema, testQuery); return resultPromise.then((result) => { - expect(result.data!.red).toEqual('override'); + expect(result.data!['red']).toEqual('override'); expect(result.errors).toEqual(undefined); }); }); @@ -1475,7 +1366,7 @@ describe('generating schema from shorthand', () => { const resultPromise = graphql(jsSchema, testQuery); return resultPromise.then((result) => { - expect(result.data!.red).toEqual('#EA3232'); + expect(result.data!['red']).toEqual('#EA3232'); expect(result.errors).toEqual(undefined); }); }); @@ -2012,157 +1903,6 @@ To disable this validator, use: }); }); -describe('Add error logging to schema', () => { - test('throws an error if no logger is provided', () => { - expect(() => - addErrorLoggingToSchema(({} as unknown) as GraphQLSchema), - ).toThrow('Must provide a logger'); - }); - test('throws an error if logger.log is not a function', () => { - expect(() => - addErrorLoggingToSchema( - ({} as unknown) as GraphQLSchema, - ({ log: '1' } as unknown) as ILogger, - ), - ).toThrow('Logger.log must be a function'); - }); -}); - -describe('Attaching external data fetchers to schema', () => { - describe('Schema level resolver', () => { - test('actually runs', () => { - let jsSchema = makeExecutableSchema({ - typeDefs: testSchema, - resolvers: testResolvers, - }); - const rootResolver = () => ({ species: 'ROOT' }); - jsSchema = addSchemaLevelResolver(jsSchema, rootResolver); - const query = `{ - species(name: "strix") - }`; - return graphql(jsSchema, query).then((res) => { - expect(res.data!.species).toBe('ROOTstrix'); - }); - }); - - test('can wrap fields that do not have a resolver defined', () => { - let jsSchema = makeExecutableSchema({ - typeDefs: testSchema, - resolvers: testResolvers, - }); - const rootResolver = () => ({ stuff: 'stuff' }); - jsSchema = addSchemaLevelResolver(jsSchema, rootResolver); - const query = `{ - stuff - }`; - return graphql(jsSchema, query).then((res) => { - expect(res.data!.stuff).toBe('stuff'); - }); - }); - - test('runs only once per query', () => { - const simpleResolvers = { - RootQuery: { - usecontext: (_r: any, _a: Record, ctx: any) => - ctx.usecontext, - species: (root: any, { name }: { name: string }) => - (root.species as string) + name, - }, - }; - let jsSchema = makeExecutableSchema({ - typeDefs: testSchema, - resolvers: simpleResolvers, - }); - let count = 0; - const rootResolver = () => { - if (count === 0) { - count += 1; - return { stuff: 'stuff', species: 'some ' }; - } - return { stuff: 'EEE', species: 'EEE' }; - }; - jsSchema = addSchemaLevelResolver(jsSchema, rootResolver); - const query = `{ - species(name: "strix") - stuff - }`; - const expected = { - species: 'some strix', - stuff: 'stuff', - }; - return graphql(jsSchema, query).then((res) => { - expect(res.data).toEqual(expected); - }); - }); - - test('runs twice for two queries', () => { - const simpleResolvers = { - RootQuery: { - usecontext: (_r: any, _a: Record, ctx: any) => - ctx.usecontext, - species: (root: any, { name }: { name: string }) => - (root.species as string) + name, - }, - }; - let jsSchema = makeExecutableSchema({ - typeDefs: testSchema, - resolvers: simpleResolvers, - }); - let count = 0; - const rootResolver = () => { - if (count === 0) { - count += 1; - return { stuff: 'stuff', species: 'some ' }; - } - if (count === 1) { - count += 1; - return { stuff: 'stuff2', species: 'species2 ' }; - } - return { stuff: 'EEE', species: 'EEE' }; - }; - jsSchema = addSchemaLevelResolver(jsSchema, rootResolver); - const query = `{ - species(name: "strix") - stuff - }`; - const expected = { - species: 'some strix', - stuff: 'stuff', - }; - const expected2 = { - species: 'species2 strix', - stuff: 'stuff2', - }; - return graphql(jsSchema, query).then((res) => { - expect(res.data).toEqual(expected); - return graphql(jsSchema, query).then((res2) => - expect(res2.data).toEqual(expected2), - ); - }); - }); - - test('can attach things to context', () => { - let jsSchema = makeExecutableSchema({ - typeDefs: testSchema, - resolvers: testResolvers, - }); - const rootResolver = (_o: any, _a: Record, ctx: any) => { - ctx.usecontext = 'ABC'; - }; - jsSchema = addSchemaLevelResolver(jsSchema, rootResolver); - const query = `{ - usecontext - }`; - const expected = { - usecontext: 'ABC', - }; - return graphql(jsSchema, query, {}, {}).then((res) => { - expect(res.data).toEqual(expected); - }); - }); - }); -}); - describe('Generating a full graphQL schema with resolvers and connectors', () => { test('outputs a working GraphQL schema', () => { const schema = makeExecutableSchema({ @@ -2179,7 +1919,7 @@ describe('Generating a full graphQL schema with resolvers and connectors', () => stuff: 'stuff', usecontext: 'ABC', }; - return graphql(schema, query, {}, { usecontext: 'ABC' }).then((res) => { + return graphql(schema, query, { stuff: 'stuff', species: 'ROOT' }, { usecontext: 'ABC' }).then((res) => { expect(res.data).toEqual(expected); }); }); @@ -2212,248 +1952,6 @@ describe('chainResolvers', () => { }); }); -describe('attachDirectiveResolvers on field', () => { - const testSchemaWithDirectives = ` - directive @upper on FIELD_DEFINITION - directive @lower on FIELD_DEFINITION - directive @default(value: String!) on FIELD_DEFINITION - directive @catchError on FIELD_DEFINITION - - type TestObject { - hello: String @upper - } - type RootQuery { - hello: String @upper - withDefault: String @default(value: "some default_value") - object: TestObject - asyncResolver: String @upper - multiDirectives: String @upper @lower - throwError: String @catchError - } - schema { - query: RootQuery - } - `; - - const testObject = { - hello: 'giau. tran minh', - }; - - const testResolversDirectives = { - RootQuery: { - hello: () => 'giau. tran minh', - object: () => testObject, - asyncResolver: async () => Promise.resolve('giau. tran minh'), - multiDirectives: () => 'Giau. Tran Minh', - throwError: () => { - throw new Error('This error for testing'); - }, - }, - }; - - const directiveResolvers: IDirectiveResolvers = { - lower( - next: NextResolverFn, - _src: any, - _args: { [argName: string]: any }, - _context: any, - ) { - return next().then((str) => { - if (typeof str === 'string') { - return str.toLowerCase(); - } - return str; - }); - }, - upper( - next: NextResolverFn, - _src: any, - _args: { [argName: string]: any }, - _context: any, - ) { - return next().then((str) => { - if (typeof str === 'string') { - return str.toUpperCase(); - } - return str; - }); - }, - default( - next: NextResolverFn, - _src: any, - args: { [argName: string]: any }, - _context: any, - ) { - return next().then((res) => { - if (undefined === res) { - return args.value; - } - return res; - }); - }, - catchError( - next: NextResolverFn, - _src: any, - _args: { [argName: string]: any }, - _context: any, - ) { - return next().catch((error) => error.message); - }, - }; - - test('throws error if directiveResolvers argument is an array', () => { - const jsSchema = makeExecutableSchema({ - typeDefs: testSchema, - resolvers: testResolvers, - }); - expect(() => - attachDirectiveResolvers(jsSchema, ([ - 1, - ] as unknown) as IDirectiveResolvers), - ).toThrowError( - 'Expected directiveResolvers to be of type object, got Array', - ); - }); - - test('throws error if directiveResolvers argument is not an object', () => { - const jsSchema = makeExecutableSchema({ - typeDefs: testSchema, - resolvers: testResolvers, - }); - return expect(() => - attachDirectiveResolvers( - jsSchema, - ('a' as unknown) as IDirectiveResolvers, - ), - ).toThrowError( - 'Expected directiveResolvers to be of type object, got string', - ); - }); - - test('upper String from resolvers', () => { - const schema = makeExecutableSchema({ - typeDefs: testSchemaWithDirectives, - resolvers: testResolversDirectives, - directiveResolvers, - }); - const query = `{ - hello - }`; - const expected = { - hello: 'GIAU. TRAN MINH', - }; - return graphql(schema, query, {}, {}).then((res) => { - expect(res.data).toEqual(expected); - }); - }); - - test('using default resolver for object property', () => { - const schema = makeExecutableSchema({ - typeDefs: testSchemaWithDirectives, - resolvers: testResolversDirectives, - directiveResolvers, - }); - const query = `{ - object { - hello - } - }`; - const expected = { - object: { - hello: 'GIAU. TRAN MINH', - }, - }; - return graphql(schema, query, {}, {}).then((res) => { - expect(res.data).toEqual(expected); - }); - }); - - test('passes in directive arguments to the directive resolver', () => { - const schema = makeExecutableSchema({ - typeDefs: testSchemaWithDirectives, - resolvers: testResolversDirectives, - directiveResolvers, - }); - const query = `{ - withDefault - }`; - const expected = { - withDefault: 'some default_value', - }; - return graphql(schema, query, {}, {}).then((res) => { - expect(res.data).toEqual(expected); - }); - }); - - test('No effect if missing directive resolvers', () => { - const schema = makeExecutableSchema({ - typeDefs: testSchemaWithDirectives, - resolvers: testResolversDirectives, - directiveResolvers: {}, // Empty resolver - }); - const query = `{ - hello - }`; - const expected = { - hello: 'giau. tran minh', - }; - return graphql(schema, query, {}, {}).then((res) => { - expect(res.data).toEqual(expected); - }); - }); - - test('If resolver return Promise, keep using it', () => { - const schema = makeExecutableSchema({ - typeDefs: testSchemaWithDirectives, - resolvers: testResolversDirectives, - directiveResolvers, - }); - const query = `{ - asyncResolver - }`; - const expected = { - asyncResolver: 'GIAU. TRAN MINH', - }; - return graphql(schema, query, {}, {}).then((res) => { - expect(res.data).toEqual(expected); - }); - }); - - test('Multi directives apply with LTR order', () => { - const schema = makeExecutableSchema({ - typeDefs: testSchemaWithDirectives, - resolvers: testResolversDirectives, - directiveResolvers, - }); - const query = `{ - multiDirectives - }`; - const expected = { - multiDirectives: 'giau. tran minh', - }; - return graphql(schema, query, {}, {}).then((res) => { - expect(res.data).toEqual(expected); - }); - }); - - test('Allow to catch error from next resolver', () => { - const schema = makeExecutableSchema({ - typeDefs: testSchemaWithDirectives, - resolvers: testResolversDirectives, - directiveResolvers, - }); - const query = `{ - throwError - }`; - const expected = { - throwError: 'This error for testing', - }; - return graphql(schema, query, {}, {}).then((res) => { - expect(res.data).toEqual(expected); - }); - }); -}); - describe('can specify lexical parser options', () => { test("can specify 'noLocation' option", () => { const schema = makeExecutableSchema({ @@ -2531,13 +2029,13 @@ describe('can specify lexical parser options', () => { const parsedQuery = parse(query, { experimentalFragmentVariables: true }); const hoist = (document: DocumentNode) => { - let variableDefs: Array = []; + const variableDefs: Array = []; - document.definitions.forEach((def) => { - if (def.kind === Kind.FRAGMENT_DEFINITION) { - variableDefs = variableDefs.concat(def.variableDefinitions!); + for (const def of document.definitions) { + if (def.kind === Kind.FRAGMENT_DEFINITION && def.variableDefinitions) { + variableDefs.push(...def.variableDefinitions); } - }); + } return { kind: Kind.DOCUMENT, diff --git a/packages/stitch/src/createMergedTypeResolver.ts b/packages/stitch/src/createMergedTypeResolver.ts index dfae35b8abf..3ceec4b3162 100644 --- a/packages/stitch/src/createMergedTypeResolver.ts +++ b/packages/stitch/src/createMergedTypeResolver.ts @@ -2,9 +2,9 @@ import { getNamedType, GraphQLOutputType, GraphQLList } from 'graphql'; import { delegateToSchema, MergedTypeResolver, MergedTypeResolverOptions } from '@graphql-tools/delegate'; import { batchDelegateToSchema } from '@graphql-tools/batch-delegate'; -export function createMergedTypeResolver( +export function createMergedTypeResolver( mergedTypeResolverOptions: MergedTypeResolverOptions -): MergedTypeResolver | undefined { +): MergedTypeResolver | undefined { const { fieldName, argsFromKeys, valuesFromResults, args } = mergedTypeResolverOptions; if (argsFromKeys != null) { diff --git a/packages/stitch/src/definitions.ts b/packages/stitch/src/definitions.ts index f11648113df..bf683125027 100644 --- a/packages/stitch/src/definitions.ts +++ b/packages/stitch/src/definitions.ts @@ -14,7 +14,7 @@ export function extractDefinitions(ast: DocumentNode) { const schemaDefs: SchemaDefinitionNode[] = []; const schemaExtensions: SchemaExtensionNode[] = []; const extensionDefs: TypeExtensionNode[] = []; - ast.definitions.forEach(def => { + for (const def of ast.definitions) { switch (def.kind) { case Kind.OBJECT_TYPE_DEFINITION: case Kind.INTERFACE_TYPE_DEFINITION: @@ -42,7 +42,7 @@ export function extractDefinitions(ast: DocumentNode) { extensionDefs.push(def); break; } - }); + } return { typeDefinitions, diff --git a/packages/stitch/src/mergeCandidates.ts b/packages/stitch/src/mergeCandidates.ts index bc69006f02c..cf8b7f30a24 100644 --- a/packages/stitch/src/mergeCandidates.ts +++ b/packages/stitch/src/mergeCandidates.ts @@ -50,8 +50,8 @@ import { validateInputObjectConsistency, } from './mergeValidations'; -import { fieldToFieldConfig, inputFieldToFieldConfig, Maybe } from '@graphql-tools/utils'; import { isSubschemaConfig } from '@graphql-tools/delegate'; +import { Maybe } from '@graphql-tools/utils'; export function mergeCandidates>( typeName: string, @@ -92,15 +92,15 @@ function mergeObjectTypeCandidates>( const typeConfigs = candidates.map(candidate => (candidate.type as GraphQLObjectType).toConfig()); const interfaceMap = typeConfigs .map(typeConfig => typeConfig.interfaces) - .reduce((acc, interfaces) => { + .reduce>((acc, interfaces) => { if (interfaces != null) { - interfaces.forEach(iface => { + for (const iface of interfaces) { acc[iface.name] = iface; - }); + } } return acc; }, Object.create(null)); - const interfaces = Object.keys(interfaceMap).map(interfaceName => interfaceMap[interfaceName]); + const interfaces = Object.values(interfaceMap); const astNodes = pluck('astNode', candidates); const fieldAstNodes = canonicalFieldNamesForType(candidates) @@ -198,18 +198,18 @@ function mergeInterfaceTypeCandidates>( const description = mergeTypeDescriptions(candidates, typeMergingOptions); const fields = fieldConfigMapFromTypeCandidates(candidates, typeMergingOptions); - const typeConfigs = candidates.map(candidate => (candidate.type as GraphQLInterfaceType).toConfig()); + const typeConfigs = candidates.map(candidate => candidate.type.toConfig()); const interfaceMap = typeConfigs - .map(typeConfig => (typeConfig as unknown as { interfaces: Array }).interfaces) - .reduce((acc, interfaces) => { + .map(typeConfig => ('interfaces' in typeConfig ? typeConfig.interfaces : [])) + .reduce>((acc, interfaces) => { if (interfaces != null) { - interfaces.forEach(iface => { + for (const iface of interfaces) { acc[iface.name] = iface; - }); + } } return acc; }, Object.create(null)); - const interfaces = Object.keys(interfaceMap).map(interfaceName => interfaceMap[interfaceName]); + const interfaces = Object.values(interfaceMap); const astNodes = pluck('astNode', candidates); const fieldAstNodes = canonicalFieldNamesForType(candidates) @@ -256,14 +256,19 @@ function mergeUnionTypeCandidates>( candidates = orderedTypeCandidates(candidates, typeMergingOptions); const description = mergeTypeDescriptions(candidates, typeMergingOptions); - const typeConfigs = candidates.map(candidate => (candidate.type as GraphQLUnionType).toConfig()); - const typeMap = typeConfigs.reduce((acc, typeConfig) => { - typeConfig.types.forEach(type => { + const typeConfigs = candidates.map(candidate => { + if (!isUnionType(candidate.type)) { + throw new Error(`Expected ${candidate.type} to be a union type!`); + } + return candidate.type.toConfig(); + }); + const typeMap = typeConfigs.reduce>((acc, typeConfig) => { + for (const type of typeConfig.types) { acc[type.name] = type; - }); + } return acc; }, Object.create(null)); - const types = Object.keys(typeMap).map(typeName => typeMap[typeName]); + const types = Object.values(typeMap); const astNodes = pluck('astNode', candidates); const astNode = astNodes @@ -330,9 +335,9 @@ function enumValueConfigMapFromTypeCandidates( ): GraphQLEnumValueConfigMap { const enumValueConfigCandidatesMap: Record> = Object.create(null); - candidates.forEach(candidate => { + for (const candidate of candidates) { const valueMap = (candidate.type as GraphQLEnumType).toConfig().values; - Object.keys(valueMap).forEach(enumValue => { + for (const enumValue in valueMap) { const enumValueConfigCandidate = { enumValueConfig: valueMap[enumValue], enumValue, @@ -346,15 +351,15 @@ function enumValueConfigMapFromTypeCandidates( } else { enumValueConfigCandidatesMap[enumValue] = [enumValueConfigCandidate]; } - }); - }); + } + } const enumValueConfigMap = Object.create(null); - Object.keys(enumValueConfigCandidatesMap).forEach(enumValue => { + for (const enumValue in enumValueConfigCandidatesMap) { const enumValueConfigMerger = typeMergingOptions?.enumValueConfigMerger ?? defaultEnumValueConfigMerger; enumValueConfigMap[enumValue] = enumValueConfigMerger(enumValueConfigCandidatesMap[enumValue]); - }); + } return enumValueConfigMap; } @@ -455,11 +460,14 @@ function fieldConfigMapFromTypeCandidates>( ): GraphQLFieldConfigMap { const fieldConfigCandidatesMap: Record>> = Object.create(null); - candidates.forEach(candidate => { - const fieldMap = (candidate.type as GraphQLObjectType | GraphQLInterfaceType).getFields(); - Object.keys(fieldMap).forEach(fieldName => { + for (const candidate of candidates) { + const typeConfig = (candidate.type as GraphQLObjectType | GraphQLInterfaceType).toConfig(); + const fieldConfigMap = typeConfig.fields; + for (const fieldName in fieldConfigMap) { + const fieldConfig = fieldConfigMap[fieldName]; + const fieldConfigCandidate = { - fieldConfig: fieldToFieldConfig(fieldMap[fieldName]), + fieldConfig, fieldName, type: candidate.type as GraphQLObjectType | GraphQLInterfaceType, subschema: candidate.subschema, @@ -471,14 +479,14 @@ function fieldConfigMapFromTypeCandidates>( } else { fieldConfigCandidatesMap[fieldName] = [fieldConfigCandidate]; } - }); - }); + } + } const fieldConfigMap = Object.create(null); - Object.keys(fieldConfigCandidatesMap).forEach(fieldName => { + for (const fieldName in fieldConfigCandidatesMap) { fieldConfigMap[fieldName] = mergeFieldConfigs(fieldConfigCandidatesMap[fieldName], typeMergingOptions); - }); + } return fieldConfigMap; } @@ -499,14 +507,14 @@ function defaultFieldConfigMerger>( const canonicalByField: Array> = []; const canonicalByType: Array> = []; - candidates.forEach(({ type, fieldName, fieldConfig, transformedSubschema }) => { - if (!isSubschemaConfig(transformedSubschema)) return; + for (const { type, fieldName, fieldConfig, transformedSubschema } of candidates) { + if (!isSubschemaConfig(transformedSubschema)) continue; if (transformedSubschema.merge?.[type.name]?.fields?.[fieldName]?.canonical) { canonicalByField.push(fieldConfig); } else if (transformedSubschema.merge?.[type.name]?.canonical) { canonicalByType.push(fieldConfig); } - }); + } if (canonicalByField.length > 1) { throw new Error(`Multiple canonical definitions for "${candidates[0].type.name}.${candidates[0].fieldName}"`); @@ -523,18 +531,21 @@ function inputFieldConfigMapFromTypeCandidates>( candidates: Array>, typeMergingOptions?: TypeMergingOptions ): GraphQLInputFieldConfigMap { - const inputFieldConfigCandidatesMap: Record>> = - Object.create(null); + const inputFieldConfigCandidatesMap: Record>> = Object.create( + null + ); const fieldInclusionMap: Record = Object.create(null); - candidates.forEach(candidate => { - const inputFieldMap = (candidate.type as GraphQLInputObjectType).getFields(); - Object.keys(inputFieldMap).forEach(fieldName => { + for (const candidate of candidates) { + const typeConfig = (candidate.type as GraphQLInputObjectType).toConfig(); + const inputFieldConfigMap = typeConfig.fields; + for (const fieldName in inputFieldConfigMap) { + const inputFieldConfig = inputFieldConfigMap[fieldName]; fieldInclusionMap[fieldName] = fieldInclusionMap[fieldName] || 0; fieldInclusionMap[fieldName] += 1; const inputFieldConfigCandidate = { - inputFieldConfig: inputFieldToFieldConfig(inputFieldMap[fieldName]), + inputFieldConfig, fieldName, type: candidate.type as GraphQLInputObjectType, subschema: candidate.subschema, @@ -546,14 +557,14 @@ function inputFieldConfigMapFromTypeCandidates>( } else { inputFieldConfigCandidatesMap[fieldName] = [inputFieldConfigCandidate]; } - }); - }); + } + } validateInputObjectConsistency(fieldInclusionMap, candidates, typeMergingOptions); const inputFieldConfigMap = Object.create(null); - Object.keys(inputFieldConfigCandidatesMap).forEach(fieldName => { + for (const fieldName in inputFieldConfigCandidatesMap) { const inputFieldConfigMerger = typeMergingOptions?.inputFieldConfigMerger ?? defaultInputFieldConfigMerger; inputFieldConfigMap[fieldName] = inputFieldConfigMerger(inputFieldConfigCandidatesMap[fieldName]); validateInputFieldConsistency( @@ -561,7 +572,7 @@ function inputFieldConfigMapFromTypeCandidates>( inputFieldConfigCandidatesMap[fieldName], typeMergingOptions ); - }); + } return inputFieldConfigMap; } @@ -572,14 +583,14 @@ function defaultInputFieldConfigMerger>( const canonicalByField: Array = []; const canonicalByType: Array = []; - candidates.forEach(({ type, fieldName, inputFieldConfig, transformedSubschema }) => { - if (!isSubschemaConfig(transformedSubschema)) return; + for (const { type, fieldName, inputFieldConfig, transformedSubschema } of candidates) { + if (!isSubschemaConfig(transformedSubschema)) continue; if (transformedSubschema.merge?.[type.name]?.fields?.[fieldName]?.canonical) { canonicalByField.push(inputFieldConfig); } else if (transformedSubschema.merge?.[type.name]?.canonical) { canonicalByType.push(inputFieldConfig); } - }); + } if (canonicalByField.length > 1) { throw new Error(`Multiple canonical definitions for "${candidates[0].type.name}.${candidates[0].fieldName}"`); @@ -595,19 +606,18 @@ function defaultInputFieldConfigMerger>( function canonicalFieldNamesForType(candidates: Array>): Array { const canonicalFieldNames: Record = Object.create(null); - candidates.forEach(({ type, transformedSubschema }) => { - if (!isSubschemaConfig(transformedSubschema)) { - return; - } + for (const { type, transformedSubschema } of candidates) { + if (!isSubschemaConfig(transformedSubschema)) continue; const mergeConfig = transformedSubschema.merge?.[type.name]; if (mergeConfig != null && mergeConfig.fields != null && !mergeConfig.canonical) { - Object.entries(mergeConfig.fields).forEach(([fieldName, mergedFieldConfig]) => { + for (const fieldName in mergeConfig.fields) { + const mergedFieldConfig = mergeConfig.fields[fieldName]; if (mergedFieldConfig.canonical) { canonicalFieldNames[fieldName] = true; } - }); + } } - }); + } return Object.keys(canonicalFieldNames); } diff --git a/packages/stitch/src/mergeValidations.ts b/packages/stitch/src/mergeValidations.ts index c79a03bb1e7..4ed13197d4e 100644 --- a/packages/stitch/src/mergeValidations.ts +++ b/packages/stitch/src/mergeValidations.ts @@ -55,15 +55,16 @@ export function validateFieldConsistency>( } const argCandidatesMap: Record> = Object.create(null); - candidates.forEach(({ fieldConfig }) => { + for (const { fieldConfig } of candidates) { if (fieldConfig.args == null) { - return; + continue; } - Object.entries(fieldConfig.args).forEach(([argName, arg]) => { + for (const argName in fieldConfig.args) { + const arg = fieldConfig.args[argName]; argCandidatesMap[argName] = argCandidatesMap[argName] || []; argCandidatesMap[argName].push(arg); - }); - }); + } + } if (Object.values(argCandidatesMap).some(argCandidates => candidates.length !== argCandidates.length)) { validationMessage( @@ -73,10 +74,11 @@ export function validateFieldConsistency>( ); } - Object.entries(argCandidatesMap).forEach(([argName, argCandidates]) => { + for (const argName in argCandidatesMap) { if (finalFieldConfig.args == null) { - return; + continue; } + const argCandidates = argCandidatesMap[argName]; const argNamespace = `${fieldNamespace}.${argName}`; const finalArgConfig = finalFieldConfig.args[argName] || argCandidates[argCandidates.length - 1]; const finalArgType = getNamedType(finalArgConfig.type); @@ -104,7 +106,7 @@ export function validateFieldConsistency>( if (isEnumType(finalArgType)) { validateInputEnumConsistency(finalArgType, argCandidates, typeMergingOptions); } - }); + } } export function validateInputObjectConsistency>( @@ -112,7 +114,8 @@ export function validateInputObjectConsistency>( candidates: Array>, typeMergingOptions?: TypeMergingOptions ): void { - Object.entries(fieldInclusionMap).forEach(([fieldName, count]) => { + for (const fieldName in fieldInclusionMap) { + const count = fieldInclusionMap[fieldName]; if (candidates.length !== count) { const namespace = `${candidates[0].type.name}.${fieldName}`; validationMessage( @@ -121,7 +124,7 @@ export function validateInputObjectConsistency>( typeMergingOptions ); } - }); + } } export function validateInputFieldConsistency>( @@ -175,7 +178,7 @@ export function validateTypeConsistency>( const finalIsScalar = isScalarType(finalNamedType); const finalIsList = hasListType(finalElementConfig.type); - candidates.forEach(c => { + for (const c of candidates) { if (finalIsList !== hasListType(c.type)) { throw new Error( `Definitions of ${definitionType} "${settingNamespace}" implement inconsistent list types across subschemas and cannot be merged.` @@ -201,7 +204,7 @@ export function validateTypeConsistency>( ); } } - }); + } } function hasListType(type: GraphQLType): boolean { @@ -215,15 +218,15 @@ export function validateInputEnumConsistency>( ): void { const enumValueInclusionMap: Record = Object.create(null); - candidates.forEach(candidate => { + for (const candidate of candidates) { const enumType = getNamedType(candidate.type) as GraphQLEnumType; if (isEnumType(enumType)) { - enumType.getValues().forEach(({ value }) => { + for (const { value } of enumType.getValues()) { enumValueInclusionMap[value] = enumValueInclusionMap[value] || 0; enumValueInclusionMap[value] += 1; - }); + } } - }); + } if (Object.values(enumValueInclusionMap).some(count => candidates.length !== count)) { validationMessage( diff --git a/packages/stitch/src/stitchSchemas.ts b/packages/stitch/src/stitchSchemas.ts index 0530e79235d..5bc564dc6ac 100644 --- a/packages/stitch/src/stitchSchemas.ts +++ b/packages/stitch/src/stitchSchemas.ts @@ -7,17 +7,9 @@ import { extendSchema, } from 'graphql'; -import { SchemaDirectiveVisitor, mergeDeep, IResolvers, pruneSchema } from '@graphql-tools/utils'; +import { IResolvers, pruneSchema } from '@graphql-tools/utils'; -import { - addResolversToSchema, - addSchemaLevelResolver, - addErrorLoggingToSchema, - addCatchUndefinedToSchema, - assertResolversPresent, - attachDirectiveResolvers, - extendResolversFromInterfaces, -} from '@graphql-tools/schema'; +import { addResolversToSchema, assertResolversPresent, extendResolversFromInterfaces } from '@graphql-tools/schema'; import { SubschemaConfig, isSubschemaConfig, Subschema, defaultMergedResolver } from '@graphql-tools/delegate'; @@ -30,6 +22,7 @@ import { isolateComputedFieldsTransformer, splitMergedTypeEntryPointsTransformer, } from './subschemaConfigTransforms'; +import { mergeResolvers } from '@graphql-tools/merge'; export function stitchSchemas>({ subschemas = [], @@ -41,12 +34,8 @@ export function stitchSchemas>({ typeMergingOptions, subschemaConfigTransforms = defaultSubschemaConfigTransforms, resolvers = {}, - schemaDirectives, inheritResolversFromInterfaces = false, - logger, - allowUndefinedInResolve = true, resolverValidationOptions = {}, - directiveResolvers, schemaTransforms = [], parseOptions = {}, pruningOptions, @@ -56,7 +45,7 @@ export function stitchSchemas>({ throw new Error('Expected `resolverValidationOptions` to be an object'); } - let transformedSubschemas: Array> = []; + const transformedSubschemas: Array> = []; const subschemaMap: Map< GraphQLSchema | SubschemaConfig, Subschema @@ -66,24 +55,29 @@ export function stitchSchemas>({ GraphQLSchema | SubschemaConfig > = new Map(); - subschemas.forEach(subschemaOrSubschemaArray => { + for (const subschemaOrSubschemaArray of subschemas) { if (Array.isArray(subschemaOrSubschemaArray)) { - subschemaOrSubschemaArray.forEach(s => { - transformedSubschemas = transformedSubschemas.concat( - applySubschemaConfigTransforms(subschemaConfigTransforms, s, subschemaMap, originalSubschemaMap) - ); - }); - } else { - transformedSubschemas = transformedSubschemas.concat( - applySubschemaConfigTransforms( + for (const s of subschemaOrSubschemaArray) { + for (const transformedSubschemaConfig of applySubschemaConfigTransforms( subschemaConfigTransforms, - subschemaOrSubschemaArray, + s, subschemaMap, originalSubschemaMap - ) - ); + )) { + transformedSubschemas.push(transformedSubschemaConfig); + } + } + } else { + for (const transformedSubschemaConfig of applySubschemaConfigTransforms( + subschemaConfigTransforms, + subschemaOrSubschemaArray, + subschemaMap, + originalSubschemaMap + )) { + transformedSubschemas.push(transformedSubschemaConfig); + } } - }); + } const extensions: Array = []; const directives: Array = []; @@ -102,7 +96,7 @@ export function stitchSchemas>({ subschemas: transformedSubschemas, originalSubschemaMap, types, - typeDefs, + typeDefs: typeDefs || [], parseOptions, extensions, directiveMap, @@ -111,9 +105,9 @@ export function stitchSchemas>({ mergeDirectives, }); - Object.keys(directiveMap).forEach(directiveName => { + for (const directiveName in directiveMap) { directives.push(directiveMap[directiveName]); - }); + } let stitchingInfo = createStitchingInfo(subschemaMap, typeCandidates, mergeTypes); @@ -138,14 +132,14 @@ export function stitchSchemas>({ extensions: null, }); - extensions.forEach(extension => { + for (const extension of extensions) { schema = extendSchema(schema, extension, { commentDescriptions: true, }); - }); + } // We allow passing in an array of resolver maps, in which case we merge them - const resolverMap: IResolvers = Array.isArray(resolvers) ? resolvers.reduce(mergeDeep, {}) : resolvers; + const resolverMap: IResolvers = mergeResolvers(resolvers); const finalResolvers = inheritResolversFromInterfaces ? extendResolversFromInterfaces(schema, resolverMap) @@ -168,30 +162,8 @@ export function stitchSchemas>({ schema = addStitchingInfo(schema, stitchingInfo); - if (!allowUndefinedInResolve) { - schema = addCatchUndefinedToSchema(schema); - } - - if (logger != null) { - schema = addErrorLoggingToSchema(schema, logger); - } - - if (typeof finalResolvers['__schema'] === 'function') { - // TODO a bit of a hack now, better rewrite generateSchema to attach it there. - // not doing that now, because I'd have to rewrite a lot of tests. - schema = addSchemaLevelResolver(schema, finalResolvers['__schema']); - } - - schemaTransforms.forEach(schemaTransform => { + for (const schemaTransform of schemaTransforms) { schema = schemaTransform(schema); - }); - - if (directiveResolvers != null) { - schema = attachDirectiveResolvers(schema, directiveResolvers); - } - - if (schemaDirectives != null) { - SchemaDirectiveVisitor.visitSchemaDirectives(schema, schemaDirectives); } if (pruningOptions) { @@ -215,7 +187,7 @@ function applySubschemaConfigTransforms>( GraphQLSchema | SubschemaConfig > ): Array> { - let subschemaConfig: SubschemaConfig; + let subschemaConfig: SubschemaConfig; if (isSubschemaConfig(subschemaOrSubschemaConfig)) { subschemaConfig = subschemaOrSubschemaConfig; } else if (subschemaOrSubschemaConfig instanceof GraphQLSchema) { @@ -224,24 +196,13 @@ function applySubschemaConfigTransforms>( throw new TypeError('Received invalid input.'); } - let transformedSubschemaConfigs: Array> = [subschemaConfig]; - subschemaConfigTransforms - .concat(subschemaConfigTransformerPresets as Array>) - .forEach(subschemaConfigTransform => { - const mapped: Array | Array>> = - transformedSubschemaConfigs.map(ssConfig => subschemaConfigTransform(ssConfig)); - - transformedSubschemaConfigs = mapped.reduce( - (acc: Array>, configOrList) => { - if (Array.isArray(configOrList)) { - return acc.concat(configOrList); - } - acc.push(configOrList); - return acc; - }, - [] - ); - }); + const transformedSubschemaConfigs = subschemaConfigTransforms + .concat(subschemaConfigTransformerPresets) + .reduce( + (transformedSubschemaConfigs, subschemaConfigTransform) => + transformedSubschemaConfigs.map(ssConfig => subschemaConfigTransform(ssConfig)).flat(), + [subschemaConfig] + ); const transformedSubschemas = transformedSubschemaConfigs.map( ssConfig => new Subschema(ssConfig) @@ -251,7 +212,9 @@ function applySubschemaConfigTransforms>( subschemaMap.set(subschemaOrSubschemaConfig, baseSubschema); - transformedSubschemas.forEach(subschema => originalSubschemaMap.set(subschema, subschemaOrSubschemaConfig)); + for (const subschema of transformedSubschemas) { + originalSubschemaMap.set(subschema, subschemaOrSubschemaConfig); + } return transformedSubschemas; } diff --git a/packages/stitch/src/stitchingInfo.ts b/packages/stitch/src/stitchingInfo.ts index 255a074ad19..8f048d2b5dd 100644 --- a/packages/stitch/src/stitchingInfo.ts +++ b/packages/stitch/src/stitchingInfo.ts @@ -29,22 +29,23 @@ export function createStitchingInfo>( const mergedTypes = createMergedTypes(typeCandidates, mergeTypes); const selectionSetsByField: Record> = Object.create(null); - Object.entries(mergedTypes).forEach(([typeName, mergedTypeInfo]) => { + for (const typeName in mergedTypes) { + const mergedTypeInfo = mergedTypes[typeName]; if (mergedTypeInfo.selectionSets == null && mergedTypeInfo.fieldSelectionSets == null) { - return; + continue; } selectionSetsByField[typeName] = Object.create(null); - mergedTypeInfo.selectionSets.forEach((selectionSet, subschemaConfig) => { + for (const [subschemaConfig, selectionSet] of mergedTypeInfo.selectionSets) { const schema = subschemaConfig.transformedSchema; const type = schema.getType(typeName) as GraphQLObjectType; const fields = type.getFields(); - Object.keys(fields).forEach(fieldName => { + for (const fieldName in fields) { const field = fields[fieldName]; const fieldType = getNamedType(field.type); if (selectionSet && isLeafType(fieldType) && selectionSetContainsTopLevelField(selectionSet, fieldName)) { - return; + continue; } if (selectionSetsByField[typeName][fieldName] == null) { selectionSetsByField[typeName][fieldName] = { @@ -55,11 +56,11 @@ export function createStitchingInfo>( selectionSetsByField[typeName][fieldName].selections = selectionSetsByField[typeName][ fieldName ].selections.concat(selectionSet.selections); - }); - }); + } + } - mergedTypeInfo.fieldSelectionSets.forEach(selectionSetFieldMap => { - Object.keys(selectionSetFieldMap).forEach(fieldName => { + for (const [, selectionSetFieldMap] of mergedTypeInfo.fieldSelectionSets) { + for (const fieldName in selectionSetFieldMap) { if (selectionSetsByField[typeName][fieldName] == null) { selectionSetsByField[typeName][fieldName] = { kind: Kind.SELECTION_SET, @@ -69,9 +70,9 @@ export function createStitchingInfo>( selectionSetsByField[typeName][fieldName].selections = selectionSetsByField[typeName][ fieldName ].selections.concat(selectionSetFieldMap[fieldName].selections); - }); - }); - }); + } + } + } return { subschemaMap, @@ -88,7 +89,7 @@ function createMergedTypes>( ): Record> { const mergedTypes: Record> = Object.create(null); - Object.keys(typeCandidates).forEach(typeName => { + for (const typeName in typeCandidates) { if ( typeCandidates[typeName].length > 1 && (isObjectType(typeCandidates[typeName][0].type) || isInterfaceType(typeCandidates[typeName][0].type)) @@ -114,11 +115,11 @@ function createMergedTypes>( const fieldSelectionSets: Map, Record> = new Map(); const resolvers: Map, MergedTypeResolver> = new Map(); - typeCandidates[typeName].forEach(typeCandidate => { + for (const typeCandidate of typeCandidates[typeName]) { const subschema = typeCandidate.transformedSubschema; if (subschema == null) { - return; + continue; } typeMaps.set(subschema, subschema.transformedSchema.getTypeMap()); @@ -126,7 +127,7 @@ function createMergedTypes>( const mergedTypeConfig = subschema?.merge?.[typeName]; if (mergedTypeConfig == null) { - return; + continue; } if (mergedTypeConfig.selectionSet) { @@ -150,7 +151,7 @@ function createMergedTypes>( const resolver = mergedTypeConfig.resolve ?? createMergedTypeResolver(mergedTypeConfig); if (resolver == null) { - return; + continue; } const keyFn = mergedTypeConfig.key; @@ -169,18 +170,18 @@ function createMergedTypes>( const type = subschema.transformedSchema.getType(typeName) as GraphQLObjectType | GraphQLInterfaceType; const fieldMap = type.getFields(); const selectionSet = selectionSets.get(subschema); - Object.keys(fieldMap).forEach(fieldName => { + for (const fieldName in fieldMap) { const field = fieldMap[fieldName]; const fieldType = getNamedType(field.type); if (selectionSet && isLeafType(fieldType) && selectionSetContainsTopLevelField(selectionSet, fieldName)) { - return; + continue; } if (!(fieldName in supportedBySubschemas)) { supportedBySubschemas[fieldName] = []; } supportedBySubschemas[fieldName].push(subschema); - }); - }); + } + } const sourceSubschemas = typeCandidates[typeName] .map(typeCandidate => typeCandidate?.transformedSubschema) @@ -189,12 +190,12 @@ function createMergedTypes>( Subschema, Array> > = new Map(); - sourceSubschemas.forEach(subschema => { + for (const subschema of sourceSubschemas) { const filteredSubschemas = targetSubschemas.filter(s => s !== subschema); if (filteredSubschemas.length) { targetSubschemasBySubschema.set(subschema, filteredSubschemas); } - }); + } mergedTypes[typeName] = { typeName, @@ -207,16 +208,16 @@ function createMergedTypes>( resolvers, }; - Object.keys(supportedBySubschemas).forEach(fieldName => { + for (const fieldName in supportedBySubschemas) { if (supportedBySubschemas[fieldName].length === 1) { mergedTypes[typeName].uniqueFields[fieldName] = supportedBySubschemas[fieldName][0]; } else { mergedTypes[typeName].nonUniqueFields[fieldName] = supportedBySubschemas[fieldName]; } - }); + } } } - }); + } return mergedTypes; } @@ -227,21 +228,22 @@ export function completeStitchingInfo>( schema: GraphQLSchema ): StitchingInfo { const selectionSetsByType = Object.create(null); - [schema.getQueryType(), schema.getMutationType()].forEach(rootType => { + const rootTypes = [schema.getQueryType(), schema.getMutationType()]; + for (const rootType of rootTypes) { if (rootType) { selectionSetsByType[rootType.name] = parseSelectionSet('{ __typename }', { noLocation: true }); } - }); + } const selectionSetsByField = stitchingInfo.selectionSetsByField; const dynamicSelectionSetsByField = Object.create(null); - Object.keys(resolvers).forEach(typeName => { + for (const typeName in resolvers) { const type = resolvers[typeName]; if (isScalarType(type)) { - return; + continue; } - Object.keys(type).forEach(fieldName => { + for (const fieldName in type) { const field = type[fieldName] as IFieldResolverOptions; if (field.selectionSet) { if (typeof field.selectionSet === 'function') { @@ -271,20 +273,20 @@ export function completeStitchingInfo>( ].selections.concat(selectionSet.selections); } } - }); - }); + } + } - Object.keys(selectionSetsByField).forEach(typeName => { + for (const typeName in selectionSetsByField) { const typeSelectionSets = selectionSetsByField[typeName]; - Object.keys(typeSelectionSets).forEach(fieldName => { + for (const fieldName in typeSelectionSets) { const consolidatedSelections: Map = new Map(); const selectionSet = typeSelectionSets[fieldName]; - selectionSet.selections.forEach(selection => { + for (const selection of selectionSet.selections) { consolidatedSelections.set(print(selection), selection); - }); + } selectionSet.selections = Array.from(consolidatedSelections.values()); - }); - }); + } + } stitchingInfo.selectionSetsByType = selectionSetsByType; stitchingInfo.selectionSetsByField = selectionSetsByField; diff --git a/packages/stitch/src/subschemaConfigTransforms/index.ts b/packages/stitch/src/subschemaConfigTransforms/index.ts index ebf70699ab4..f9ea965783e 100644 --- a/packages/stitch/src/subschemaConfigTransforms/index.ts +++ b/packages/stitch/src/subschemaConfigTransforms/index.ts @@ -1,4 +1,4 @@ -import { SubschemaConfigTransform } from 'packages/graphql-tools/src'; +import { SubschemaConfigTransform } from '../types'; import { computedDirectiveTransformer } from './computedDirectiveTransformer'; export { computedDirectiveTransformer } from './computedDirectiveTransformer'; diff --git a/packages/stitch/src/subschemaConfigTransforms/isolateComputedFieldsTransformer.ts b/packages/stitch/src/subschemaConfigTransforms/isolateComputedFieldsTransformer.ts index 0090fc714e3..1b128d87a9d 100644 --- a/packages/stitch/src/subschemaConfigTransforms/isolateComputedFieldsTransformer.ts +++ b/packages/stitch/src/subschemaConfigTransforms/isolateComputedFieldsTransformer.ts @@ -1,4 +1,4 @@ -import { GraphQLObjectType, GraphQLInterfaceType, isObjectType, isInterfaceType } from 'graphql'; +import { GraphQLObjectType, isObjectType, isInterfaceType } from 'graphql'; import { SubschemaConfig, MergedTypeConfig, MergedFieldConfig } from '@graphql-tools/delegate'; @@ -14,12 +14,15 @@ export function isolateComputedFieldsTransformer(subschemaConfig: SubschemaConfi const baseSchemaTypes: Record = Object.create(null); const isolatedSchemaTypes: Record = Object.create(null); - Object.entries(subschemaConfig.merge).forEach(([typeName, mergedTypeConfig]) => { + for (const typeName in subschemaConfig.merge) { + const mergedTypeConfig = subschemaConfig.merge[typeName]; + baseSchemaTypes[typeName] = mergedTypeConfig; if (mergedTypeConfig.computedFields) { const mergeConfigFields = mergedTypeConfig.fields ?? Object.create(null); - Object.entries(mergedTypeConfig.computedFields).forEach(([fieldName, mergedFieldConfig]) => { + for (const fieldName in mergedTypeConfig.computedFields) { + const mergedFieldConfig = mergedTypeConfig.computedFields[fieldName]; console.warn( `The "computedFields" setting is deprecated. Update your @graphql-tools/stitching-directives package, and/or update static merged type config to "${typeName}.fields.${fieldName} = { selectionSet: '${mergedFieldConfig.selectionSet}', computed: true }"` ); @@ -28,7 +31,7 @@ export function isolateComputedFieldsTransformer(subschemaConfig: SubschemaConfi ...mergedFieldConfig, computed: true, }; - }); + } delete mergedTypeConfig.computedFields; mergedTypeConfig.fields = mergeConfigFields; } @@ -37,7 +40,8 @@ export function isolateComputedFieldsTransformer(subschemaConfig: SubschemaConfi const baseFields: Record = Object.create(null); const isolatedFields: Record = Object.create(null); - Object.entries(mergedTypeConfig.fields).forEach(([fieldName, mergedFieldConfig]) => { + for (const fieldName in mergedTypeConfig.fields) { + const mergedFieldConfig = mergedTypeConfig.fields[fieldName]; if (mergedFieldConfig.computed && mergedFieldConfig.selectionSet) { isolatedFields[fieldName] = mergedFieldConfig; } else if (mergedFieldConfig.computed) { @@ -45,7 +49,7 @@ export function isolateComputedFieldsTransformer(subschemaConfig: SubschemaConfi } else { baseFields[fieldName] = mergedFieldConfig; } - }); + } const isolatedFieldCount = Object.keys(isolatedFields).length; const objectType = subschemaConfig.schema.getType(typeName) as GraphQLObjectType; @@ -62,7 +66,7 @@ export function isolateComputedFieldsTransformer(subschemaConfig: SubschemaConfi }; } } - }); + } if (Object.keys(isolatedSchemaTypes).length) { return [ @@ -96,16 +100,16 @@ function filterBaseSubschema( ); const filteredFields: Record> = {}; - Object.keys(filteredSchema.getTypeMap()).forEach(typeName => { + for (const typeName in filteredSchema.getTypeMap()) { const type = filteredSchema.getType(typeName); if (isObjectType(type) || isInterfaceType(type)) { filteredFields[typeName] = { __typename: true }; const fieldMap = type.getFields(); - Object.keys(fieldMap).forEach(fieldName => { + for (const fieldName in fieldMap) { filteredFields[typeName][fieldName] = true; - }); + } } - }); + } const filteredSubschema = { ...subschemaConfig, @@ -125,11 +129,11 @@ function filterBaseSubschema( const remainingTypes = filteredSchema.getTypeMap(); const mergeConfig = filteredSubschema.merge; if (mergeConfig) { - Object.keys(mergeConfig).forEach(mergeType => { + for (const mergeType in mergeConfig) { if (!remainingTypes[mergeType]) { delete mergeConfig[mergeType]; } - }); + } if (!Object.keys(mergeConfig).length) { delete filteredSubschema.merge; @@ -146,28 +150,35 @@ type IsolatedSubschemaInput = Exclude & { function filterIsolatedSubschema(subschemaConfig: IsolatedSubschemaInput): SubschemaConfig { const rootFields: Record = {}; - Object.values(subschemaConfig.merge).forEach(mergedTypeConfig => { + for (const typeName in subschemaConfig.merge) { + const mergedTypeConfig = subschemaConfig.merge[typeName]; const entryPoints = mergedTypeConfig.entryPoints ?? [mergedTypeConfig]; - entryPoints.forEach(entryPoint => { + for (const entryPoint of entryPoints) { if (entryPoint.fieldName != null) { rootFields[entryPoint.fieldName] = true; } - }); - }); + } + } const interfaceFields: Record> = {}; - Object.keys(subschemaConfig.merge).forEach(typeName => { - (subschemaConfig.schema.getType(typeName) as GraphQLObjectType).getInterfaces().forEach(int => { - Object.keys((subschemaConfig.schema.getType(int.name) as GraphQLInterfaceType).getFields()).forEach( - intFieldName => { - if (subschemaConfig.merge[typeName].fields?.[intFieldName]) { - interfaceFields[int.name] = interfaceFields[int.name] || {}; - interfaceFields[int.name][intFieldName] = true; - } + for (const typeName in subschemaConfig.merge) { + const type = subschemaConfig.schema.getType(typeName); + if (!type || !('getInterfaces' in type)) { + throw new Error(`${typeName} expected to have 'getInterfaces' method`); + } + for (const int of type.getInterfaces()) { + const intType = subschemaConfig.schema.getType(int.name); + if (!intType || !('getFields' in intType)) { + throw new Error(`${int.name} expected to have 'getFields' method`); + } + for (const intFieldName in intType.getFields()) { + if (subschemaConfig.merge[typeName].fields?.[intFieldName]) { + interfaceFields[int.name] = interfaceFields[int.name] || {}; + interfaceFields[int.name][intFieldName] = true; } - ); - }); - }); + } + } + } const filteredSchema = pruneSchema( filterSchema({ @@ -179,16 +190,16 @@ function filterIsolatedSubschema(subschemaConfig: IsolatedSubschemaInput): Subsc ); const filteredFields: Record> = {}; - Object.keys(filteredSchema.getTypeMap()).forEach(typeName => { + for (const typeName in filteredSchema.getTypeMap()) { const type = filteredSchema.getType(typeName); if (isObjectType(type) || isInterfaceType(type)) { filteredFields[typeName] = { __typename: true }; const fieldMap = type.getFields(); - Object.keys(fieldMap).forEach(fieldName => { + for (const fieldName in fieldMap) { filteredFields[typeName][fieldName] = true; - }); + } } - }); + } return { ...subschemaConfig, diff --git a/packages/stitch/src/subschemaConfigTransforms/splitMergedTypeEntryPointsTransformer.ts b/packages/stitch/src/subschemaConfigTransforms/splitMergedTypeEntryPointsTransformer.ts index d4499f47e83..493772bbfc9 100644 --- a/packages/stitch/src/subschemaConfigTransforms/splitMergedTypeEntryPointsTransformer.ts +++ b/packages/stitch/src/subschemaConfigTransforms/splitMergedTypeEntryPointsTransformer.ts @@ -13,15 +13,15 @@ export function splitMergedTypeEntryPointsTransformer(subschemaConfig: Subschema for (let i = 0; i < maxEntryPoints; i += 1) { const subschemaPermutation = cloneSubschemaConfig(subschemaConfig); - const mergedTypesCopy: Record> = - subschemaPermutation.merge ?? Object.create(null); + const mergedTypesCopy: Record> = subschemaPermutation.merge ?? + Object.create(null); let currentMerge = mergedTypesCopy; if (i > 0) { subschemaPermutation.merge = currentMerge = Object.create(null); } - Object.keys(mergedTypesCopy).forEach(typeName => { + for (const typeName in mergedTypesCopy) { const mergedTypeConfig = mergedTypesCopy[typeName]; const mergedTypeEntryPoint = mergedTypeConfig?.entryPoints?.[i]; @@ -38,15 +38,16 @@ export function splitMergedTypeEntryPointsTransformer(subschemaConfig: Subschema if (i > 0) { delete mergedTypeConfig.canonical; if (mergedTypeConfig.fields != null) { - Object.values(mergedTypeConfig.fields).forEach(mergedFieldConfig => { + for (const mergedFieldName in mergedTypeConfig.fields) { + const mergedFieldConfig = mergedTypeConfig.fields[mergedFieldName]; delete mergedFieldConfig.canonical; - }); + } } } currentMerge[typeName] = mergedTypeConfig; } - }); + } subschemaPermutations.push(subschemaPermutation); } diff --git a/packages/stitch/src/typeCandidates.ts b/packages/stitch/src/typeCandidates.ts index 363667e4f5a..7d356cdc535 100644 --- a/packages/stitch/src/typeCandidates.ts +++ b/packages/stitch/src/typeCandidates.ts @@ -8,17 +8,18 @@ import { SchemaExtensionNode, isSpecifiedScalarType, GraphQLSchema, + isDirective, } from 'graphql'; import { wrapSchema } from '@graphql-tools/wrap'; import { Subschema, SubschemaConfig, StitchingInfo } from '@graphql-tools/delegate'; -import { GraphQLParseOptions, ITypeDefinitions, rewireTypes, TypeMap } from '@graphql-tools/utils'; -import { buildDocumentFromTypeDefinitions } from '@graphql-tools/schema'; +import { GraphQLParseOptions, TypeSource, rewireTypes, TypeMap } from '@graphql-tools/utils'; import typeFromAST from './typeFromAST'; import { MergeTypeCandidate, MergeTypeFilter, OnTypeConflict, TypeMergingOptions } from './types'; import { mergeCandidates } from './mergeCandidates'; import { extractDefinitions } from './definitions'; +import { mergeTypeDefs } from '@graphql-tools/merge'; type CandidateSelector> = ( candidates: Array> @@ -42,7 +43,7 @@ export function buildTypeCandidates>({ GraphQLSchema | SubschemaConfig >; types: Array; - typeDefs: ITypeDefinitions | undefined; + typeDefs: TypeSource; parseOptions: GraphQLParseOptions; extensions: Array; directiveMap: Record; @@ -61,7 +62,7 @@ export function buildTypeCandidates>({ let document: DocumentNode | undefined; let extraction: ReturnType | undefined; if ((typeDefs && !Array.isArray(typeDefs)) || (Array.isArray(typeDefs) && typeDefs.length)) { - document = buildDocumentFromTypeDefinitions(typeDefs, parseOptions); + document = mergeTypeDefs(typeDefs, parseOptions); extraction = extractDefinitions(document); schemaDef = extraction.schemaDefs[0]; schemaExtensions = schemaExtensions.concat(extraction.schemaExtensions); @@ -72,7 +73,7 @@ export function buildTypeCandidates>({ setOperationTypeNames(schemaDefs, operationTypeNames); - subschemas.forEach(subschema => { + for (const subschema of subschemas) { const schema = wrapSchema(subschema); const operationTypes = { @@ -81,7 +82,7 @@ export function buildTypeCandidates>({ subscription: schema.getSubscriptionType(), }; - Object.keys(operationTypes).forEach(operationType => { + for (const operationType in operationTypes) { if (operationTypes[operationType] != null) { addTypeCandidate(typeCandidates, operationTypeNames[operationType], { type: operationTypes[operationType], @@ -89,16 +90,16 @@ export function buildTypeCandidates>({ transformedSubschema: subschema, }); } - }); + } if (mergeDirectives === true) { - schema.getDirectives().forEach(directive => { + for (const directive of schema.getDirectives()) { directiveMap[directive.name] = directive; - }); + } } const originalTypeMap = schema.getTypeMap(); - Object.keys(originalTypeMap).forEach(typeName => { + for (const typeName in originalTypeMap) { const type: GraphQLNamedType = originalTypeMap[typeName]; if ( isNamedType(type) && @@ -113,21 +114,27 @@ export function buildTypeCandidates>({ transformedSubschema: subschema, }); } - }); - }); + } + } if (document != null && extraction != null) { - extraction.typeDefinitions.forEach(def => { - const type = typeFromAST(def) as GraphQLNamedType; + for (const def of extraction.typeDefinitions) { + const type = typeFromAST(def); + if (!isNamedType(type)) { + throw new Error(`Expected to get named typed but got ${JSON.stringify(def)}`); + } if (type != null) { addTypeCandidate(typeCandidates, type.name, { type }); } - }); + } - extraction.directiveDefs.forEach(def => { - const directive = typeFromAST(def) as GraphQLDirective; + for (const def of extraction.directiveDefs) { + const directive = typeFromAST(def); + if (!isDirective(directive)) { + throw new Error(`Expected to get directive type but got ${JSON.stringify(def)}`); + } directiveMap[directive.name] = directive; - }); + } if (extraction.extensionDefs.length > 0) { extensions.push({ @@ -137,7 +144,9 @@ export function buildTypeCandidates>({ } } - types.forEach(type => addTypeCandidate(typeCandidates, type.name, { type })); + for (const type of types) { + addTypeCandidate(typeCandidates, type.name, { type }); + } return typeCandidates; } @@ -157,13 +166,13 @@ function setOperationTypeNames( allNodes.unshift(schemaDef); } - allNodes.forEach(node => { + for (const node of allNodes) { if (node.operationTypes != null) { - node.operationTypes.forEach(operationType => { + for (const operationType of node.operationTypes) { operationTypeNames[operationType.operation] = operationType.type.name.value; - }); + } } - }); + } } function addTypeCandidate>( @@ -196,7 +205,7 @@ export function buildTypes>({ }): { typeMap: TypeMap; directives: Array } { const typeMap: TypeMap = Object.create(null); - Object.keys(typeCandidates).forEach(typeName => { + for (const typeName in typeCandidates) { if ( typeName === operationTypeNames['query'] || typeName === operationTypeNames['mutation'] || @@ -214,7 +223,7 @@ export function buildTypes>({ : (cands: Array>) => cands[cands.length - 1]; typeMap[typeName] = candidateSelector(typeCandidates[typeName]).type; } - }); + } return rewireTypes(typeMap, directives); } diff --git a/packages/stitch/src/typeFromAST.ts b/packages/stitch/src/typeFromAST.ts index 0f17a8d830e..f72355dc1cd 100644 --- a/packages/stitch/src/typeFromAST.ts +++ b/packages/stitch/src/typeFromAST.ts @@ -165,11 +165,11 @@ function makeValues(nodes: ReadonlyArray): GraphQLFiel function makeDirective(node: DirectiveDefinitionNode): GraphQLDirective { const locations: Array = []; - node.locations.forEach(location => { + for (const location of node.locations) { if (location.value in DirectiveLocation) { locations.push(location.value as DirectiveLocationEnum); } - }); + } return new GraphQLDirective({ name: node.name.value, description: node.description != null ? node.description.value : null, diff --git a/packages/stitch/src/types.ts b/packages/stitch/src/types.ts index d8a49b16a72..f7ebed4bcc7 100644 --- a/packages/stitch/src/types.ts +++ b/packages/stitch/src/types.ts @@ -11,7 +11,7 @@ import { GraphQLEnumValueConfig, GraphQLEnumType, } from 'graphql'; -import { ITypeDefinitions, Maybe } from '@graphql-tools/utils'; +import { Maybe, TypeSource } from '@graphql-tools/utils'; import { Subschema, SubschemaConfig } from '@graphql-tools/delegate'; import { IExecutableSchemaDefinition } from '@graphql-tools/schema'; @@ -55,7 +55,7 @@ export interface IStitchSchemasOptions> subschemas?: Array< GraphQLSchema | SubschemaConfig | Array> >; - typeDefs?: ITypeDefinitions; + typeDefs?: TypeSource; types?: Array; onTypeConflict?: OnTypeConflict; mergeDirectives?: boolean | undefined; diff --git a/packages/stitch/tests/alternateStitchSchemas.test.ts b/packages/stitch/tests/alternateStitchSchemas.test.ts index e957cdb841e..28621405339 100644 --- a/packages/stitch/tests/alternateStitchSchemas.test.ts +++ b/packages/stitch/tests/alternateStitchSchemas.test.ts @@ -441,11 +441,11 @@ describe('optional arguments', () => { const originalResult = await graphql(schema, query); assertSome(originalResult.data) - expect(originalResult.data.test).toEqual(true); + expect(originalResult.data['test']).toEqual(true); const stitchedResult = await graphql(stitchedSchema, query); assertSome(stitchedResult.data) - expect(stitchedResult.data.test).toEqual(true); + expect(stitchedResult.data['test']).toEqual(true); }); it('work with schema stitching when using variables', async () => { @@ -457,11 +457,11 @@ describe('optional arguments', () => { const originalResult = await graphql(schema, query); assertSome(originalResult.data) - expect(originalResult.data.test).toEqual(true); + expect(originalResult.data['test']).toEqual(true); const stitchedResult = await graphql(stitchedSchema, query); assertSome(stitchedResult.data) - expect(stitchedResult.data.test).toEqual(true); + expect(stitchedResult.data['test']).toEqual(true); }); // See https://github.com/graphql/graphql-js/issues/2533 @@ -480,7 +480,7 @@ describe('optional arguments', () => { { arg: undefined }, ); assertSome(originalResult.data) - expect(originalResult.data.test).toEqual(false); + expect(originalResult.data['test']).toEqual(false); const stitchedResult = await graphql( stitchedSchema, @@ -490,7 +490,7 @@ describe('optional arguments', () => { { arg: undefined }, ); assertSome(stitchedResult.data) - expect(stitchedResult.data.test).toEqual(false); + expect(stitchedResult.data['test']).toEqual(false); }); }); @@ -515,7 +515,7 @@ describe('default values', () => { args: { ...fieldConfig.args, input: { - ...fieldConfig.args.input, + ...fieldConfig.args['input'], defaultValue: { test: 'test' } } } @@ -1516,7 +1516,7 @@ describe('schema transformation with wrapping of object fields', () => { const query = '{ wrapped { user { dummy } } }'; const result = await graphql(stitchedSchema, query); assertSome(result.data) - expect(result.data.wrapped.user.dummy).not.toEqual(null); + expect(result.data['wrapped'].user.dummy).not.toEqual(null); }); }); }); @@ -1640,7 +1640,7 @@ describe('stitchSchemas', () => { const query = '{ test { field } }'; const response = await graphql(stitchedSchema, query); assertSome(response.data) - expect(response.data.test).toBe(null); + expect(response.data['test']).toBe(null); expect(response.errors).toBeUndefined(); }); @@ -1682,7 +1682,7 @@ type Query { } `.trim(), ); - expect(response.data?.getInput).toBe('test'); + expect(response.data?.['getInput']).toBe('test'); }); test('can override scalars with new internal values', async () => { @@ -1722,7 +1722,7 @@ type Query { const query = '{ getTestScalar }'; const response = await graphql(stitchedSchema, query); - expect(response.data?.getTestScalar).toBe('test'); + expect(response.data?.['getTestScalar']).toBe('test'); }); test('can override scalars with new internal values when using default input types', async () => { @@ -1762,7 +1762,7 @@ type Query { const query = '{ getTestScalar }'; const response = await graphql(stitchedSchema, query); - expect(response.data?.getTestScalar).toBe('test'); + expect(response.data?.['getTestScalar']).toBe('test'); }); test('can use @include directives', async () => { @@ -1810,7 +1810,7 @@ type Query { } `; const response = await graphql(stitchedSchema, query); - expect(response.data?.get2.subfield).toBe('test'); + expect(response.data?.['get2'].subfield).toBe('test'); }); test('can use functions in subfields', async () => { @@ -1838,7 +1838,7 @@ type Query { const query = '{ wrappingObject { functionField } }'; const response = await graphql(stitchedSchema, query); - expect(response.data?.wrappingObject.functionField).toBe(8); + expect(response.data?.['wrappingObject'].functionField).toBe(8); }); }); @@ -1902,7 +1902,7 @@ describe('onTypeConflict', () => { mergeTypes: false, }); const result1 = await graphql(stitchedSchema, '{ test2 { fieldC } }'); - expect(result1.data?.test2.fieldC).toBe('C'); + expect(result1.data?.['test2'].fieldC).toBe('C'); const result2 = await graphql(stitchedSchema, '{ test2 { fieldB } }'); expect(result2.data).toBeUndefined(); }); @@ -1914,7 +1914,7 @@ describe('onTypeConflict', () => { onTypeConflict: (_left, right) => right, }); const result1 = await graphql(stitchedSchema, '{ test2 { fieldC } }'); - expect(result1.data?.test2.fieldC).toBe('C'); + expect(result1.data?.['test2'].fieldC).toBe('C'); const result2 = await graphql(stitchedSchema, '{ test2 { fieldB } }'); expect(result2.data).toBeUndefined(); }); @@ -1926,7 +1926,7 @@ describe('onTypeConflict', () => { onTypeConflict: (left) => left, }); const result1 = await graphql(stitchedSchema, '{ test1 { fieldB } }'); - expect(result1.data?.test1.fieldB).toBe('B'); + expect(result1.data?.['test1'].fieldB).toBe('B'); const result2 = await graphql(stitchedSchema, '{ test1 { fieldC } }'); expect(result2.data).toBeUndefined(); }); diff --git a/packages/stitch/tests/dataloader.test.ts b/packages/stitch/tests/dataloader.test.ts index 2c5389ea2ae..4dcacf27d7e 100644 --- a/packages/stitch/tests/dataloader.test.ts +++ b/packages/stitch/tests/dataloader.test.ts @@ -60,7 +60,7 @@ describe('dataloader', () => { user: { selectionSet: '{ userId }', resolve(task, _args, context, info) { - return context.usersLoader.load({ id: task.userId, info }); + return context['usersLoader'].load({ id: task.userId, info }); }, }, }, diff --git a/packages/stitch/tests/example.test.ts b/packages/stitch/tests/example.test.ts index 6757bd0af7b..9e20c76a278 100644 --- a/packages/stitch/tests/example.test.ts +++ b/packages/stitch/tests/example.test.ts @@ -109,9 +109,9 @@ describe('basic stitching example', () => { expect(result.errors).toBeUndefined(); assertSome(result.data) - expect(result.data.userById.chirps[1].id).not.toBe(null); - expect(result.data.userById.chirps[1].text).not.toBe(null); - expect(result.data.userById.chirps[1].author.email).not.toBe(null); + expect(result.data['userById'].chirps[1].id).not.toBe(null); + expect(result.data['userById'].chirps[1].text).not.toBe(null); + expect(result.data['userById'].chirps[1].author.email).not.toBe(null); }); }); @@ -232,9 +232,9 @@ describe('stitching to interfaces', () => { expect(resultWithFragments.errors).toBeUndefined(); assertSome(resultWithFragments.data) - expect(resultWithFragments.data.node.chirps[1].id).not.toBe(null); - expect(resultWithFragments.data.node.chirps[1].text).not.toBe(null); - expect(resultWithFragments.data.node.chirps[1].author.email).not.toBe(null); + expect(resultWithFragments.data['node'].chirps[1].id).not.toBe(null); + expect(resultWithFragments.data['node'].chirps[1].text).not.toBe(null); + expect(resultWithFragments.data['node'].chirps[1].author.email).not.toBe(null); }); test('it works without fragments', async () => { @@ -257,9 +257,9 @@ describe('stitching to interfaces', () => { expect(resultWithoutFragments.errors).toBeUndefined(); assertSome(resultWithoutFragments.data) - expect(resultWithoutFragments.data.node.chirps[1].id).not.toBe(null); - expect(resultWithoutFragments.data.node.chirps[1].text).not.toBe(null); - expect(resultWithoutFragments.data.node.chirps[1].author.email).not.toBe(null); + expect(resultWithoutFragments.data['node'].chirps[1].id).not.toBe(null); + expect(resultWithoutFragments.data['node'].chirps[1].text).not.toBe(null); + expect(resultWithoutFragments.data['node'].chirps[1].author.email).not.toBe(null); }); }); diff --git a/packages/stitch/tests/extendedInterface.test.ts b/packages/stitch/tests/extendedInterface.test.ts index 45c5205b3b3..4e855c11433 100644 --- a/packages/stitch/tests/extendedInterface.test.ts +++ b/packages/stitch/tests/extendedInterface.test.ts @@ -47,7 +47,7 @@ describe('extended interfaces', () => { } `); assertSome(data) - expect(data.slot).toEqual({ id: '23', name: 'The Item' }); + expect(data['slot']).toEqual({ id: '23', name: 'The Item' }); }); test('merges types behind gateway interface extension', async () => { @@ -63,7 +63,7 @@ describe('extended interfaces', () => { `, resolvers: { Query: { - itemById(obj, args, context, info) { + itemById(_obj, args, _context, _info) { return { id: args.id, name: `Item ${args.id}` }; } } @@ -84,7 +84,7 @@ describe('extended interfaces', () => { `, resolvers: { Query: { - placementById(obj, args, context, info) { + placementById(_obj, args, _context, _info) { return { __typename: 'Item', id: args.id }; } } @@ -121,6 +121,6 @@ describe('extended interfaces', () => { } `); assertSome(result.data) - expect(result.data.placement).toEqual({ id: '23', name: 'Item 23' }); + expect(result.data['placement']).toEqual({ id: '23', name: 'Item 23' }); }); }); diff --git a/packages/stitch/tests/fixtures/schemas.ts b/packages/stitch/tests/fixtures/schemas.ts index 140d5d9b11e..a6840a8b890 100644 --- a/packages/stitch/tests/fixtures/schemas.ts +++ b/packages/stitch/tests/fixtures/schemas.ts @@ -190,10 +190,6 @@ export const sampleData: { }, }; -function values(o: { [s: string]: T }): Array { - return Object.keys(o).map((k) => o[k]); -} - function coerceString(value: any): string { if (Array.isArray(value)) { throw new TypeError( @@ -227,9 +223,9 @@ function parseLiteral(ast: ValueNode): any { return parseFloat(ast.value); case Kind.OBJECT: { const value = Object.create(null); - ast.fields.forEach((field) => { + for (const field of ast.fields) { value[field.name.value] = parseLiteral(field.value); - }); + } return value; } @@ -351,7 +347,7 @@ const propertyResolvers: IResolvers = { }, properties(_root, { limit }) { - const list = values(sampleData.Property); + const list = Object.values(sampleData.Property); return limit ? list.slice(0, limit) : list; }, @@ -466,7 +462,7 @@ const productTypeDefs = ` const productResolvers: IResolvers = { Query: { products(_root) { - const list = values(sampleData.Product); + const list = Object.values(sampleData.Product); return list; }, }, @@ -552,7 +548,7 @@ const bookingResolvers: IResolvers = { return sampleData.Booking[id]; }, bookingsByPropertyId(_parent, { propertyId, limit }) { - const list = values(sampleData.Booking).filter( + const list = Object.values(sampleData.Booking).filter( (booking: Booking) => booking.propertyId === propertyId, ); return limit ? list.slice(0, limit) : list; @@ -561,11 +557,11 @@ const bookingResolvers: IResolvers = { return sampleData.Customer[id]; }, bookings(_parent, { limit }) { - const list = values(sampleData.Booking); + const list = Object.values(sampleData.Booking); return limit ? list.slice(0, limit) : list; }, customers(_parent, { limit }) { - const list = values(sampleData.Customer); + const list = Object.values(sampleData.Customer); return limit ? list.slice(0, limit) : list; }, }, @@ -602,13 +598,13 @@ const bookingResolvers: IResolvers = { Customer: { bookings(parent: Customer, { limit }) { - const list = values(sampleData.Booking).filter( + const list = Object.values(sampleData.Booking).filter( (booking: Booking) => booking.customerId === parent.id, ); return limit ? list.slice(0, limit) : list; }, vehicle(parent: Customer) { - return sampleData.Vehicle[parent.vehicleId]; + return parent.vehicleId && sampleData.Vehicle[parent.vehicleId]; }, error() { throw new Error('Customer.error error'); @@ -717,7 +713,7 @@ function makeSubscriberFromSchema(schema: GraphQLSchema) { if (isPromise(result)) { return result.then(asyncIterator => { assertAsyncIterable(asyncIterator) - return mapAsyncIterator(asyncIterator, (originalResult: ExecutionResult) => JSON.parse(JSON.stringify(originalResult))) + return mapAsyncIterator(asyncIterator, (originalResult: ExecutionResult) => JSON.parse(JSON.stringify(originalResult))) }); } return JSON.parse(JSON.stringify(result)); diff --git a/packages/stitch/tests/isolateComputedFieldsTransformer.test.ts b/packages/stitch/tests/isolateComputedFieldsTransformer.test.ts index e8ffeb88df8..d8c3a043c73 100644 --- a/packages/stitch/tests/isolateComputedFieldsTransformer.test.ts +++ b/packages/stitch/tests/isolateComputedFieldsTransformer.test.ts @@ -75,13 +75,13 @@ describe('isolateComputedFieldsTransformer', () => { expect(computedSubschema.transformedSchema.getType('ProductRepresentation')).toBeDefined(); assertSome(baseSubschema.merge) - expect(baseSubschema.merge.Product.canonical).toEqual(true); - expect(baseSubschema.merge.Product.fields).toEqual({ + expect(baseSubschema.merge['Product'].canonical).toEqual(true); + expect(baseSubschema.merge['Product'].fields).toEqual({ deliveryService: { canonical: true }, }); assertSome(computedSubschema.merge) - expect(computedSubschema.merge.Product.canonical).toBeUndefined(); - expect(computedSubschema.merge.Product.fields).toEqual({ + expect(computedSubschema.merge['Product'].canonical).toBeUndefined(); + expect(computedSubschema.merge['Product'].fields).toEqual({ shippingEstimate: { selectionSet: '{ price }', computed: true, canonical: true }, }); }); @@ -198,16 +198,16 @@ describe('isolateComputedFieldsTransformer', () => { expect(Object.keys((baseSubschema.transformedSchema.getType('Product') as GraphQLObjectType).getFields())).toEqual(['base']); expect(Object.keys((baseSubschema.transformedSchema.getType('Storefront') as GraphQLObjectType).getFields())).toEqual(['base']); assertSome(baseSubschema.merge) - expect(baseSubschema.merge.Storefront.fields).toEqual({}); + expect(baseSubschema.merge['Storefront'].fields).toEqual({}); expect(Object.keys((computedSubschema.transformedSchema.getType('Query') as GraphQLObjectType).getFields())).toEqual(['storefront', '_products']); expect(Object.keys((computedSubschema.transformedSchema.getType('Product') as GraphQLObjectType).getFields())).toEqual(['computeMe']); expect(Object.keys((computedSubschema.transformedSchema.getType('Storefront') as GraphQLObjectType).getFields())).toEqual(['computeMe']); assertSome(computedSubschema.merge) - expect(computedSubschema.merge.Storefront.fields).toEqual({ + expect(computedSubschema.merge['Storefront'].fields).toEqual({ computeMe: { selectionSet: '{ availableProductIds }', computed: true }, }); - expect(computedSubschema.merge.Product.fields).toEqual({ + expect(computedSubschema.merge['Product'].fields).toEqual({ computeMe: { selectionSet: '{ price }', computed: true }, }); }); diff --git a/packages/stitch/tests/mergeAbstractTypes.test.ts b/packages/stitch/tests/mergeAbstractTypes.test.ts index 51628593d52..a8f25bd8365 100644 --- a/packages/stitch/tests/mergeAbstractTypes.test.ts +++ b/packages/stitch/tests/mergeAbstractTypes.test.ts @@ -86,7 +86,7 @@ describe('Abstract type merge', () => { } `); assertSome(data) - expect(data.post.leadArt).toEqual({ + expect(data['post'].leadArt).toEqual({ __typename: 'Image', url: '/path/to/23', id: '23', @@ -185,7 +185,7 @@ describe('Merged associations', () => { `); assertSome(data) - expect(data.slots).toEqual([{ + expect(data['slots']).toEqual([{ id: '55', network: { domain: 'network56.com' } }]); diff --git a/packages/stitch/tests/mergeComputedFields.test.ts b/packages/stitch/tests/mergeComputedFields.test.ts index 0016e87135d..c63c718b97d 100644 --- a/packages/stitch/tests/mergeComputedFields.test.ts +++ b/packages/stitch/tests/mergeComputedFields.test.ts @@ -110,7 +110,7 @@ describe('merge computed fields via config', () => { `); assertSome(data) - expect(data.product).toEqual({ + expect(data['product']).toEqual({ id: '77', price: 77.99, weight: 77, @@ -135,7 +135,7 @@ describe('merge computed fields via config', () => { `); assertSome(data) - expect(data.storefront.availableProducts).toEqual([ + expect(data['storefront'].availableProducts).toEqual([ { id: '23', price: 23.99, @@ -188,7 +188,7 @@ describe('merge computed fields via config', () => { `); assertSome(data) - expect(data.product).toEqual({ + expect(data['product']).toEqual({ id: '77', price: 77.99, weight: 77, @@ -276,7 +276,7 @@ describe('merge computed fields via SDL (Apollo Federation-style directive annot `); assertSome(data) - expect(data.storefront.availableProducts).toEqual([ + expect(data['storefront'].availableProducts).toEqual([ { id: '23', price: 23.99, diff --git a/packages/stitch/tests/mergeConflicts.test.ts b/packages/stitch/tests/mergeConflicts.test.ts index b99d122a3b7..99a83a549ba 100644 --- a/packages/stitch/tests/mergeConflicts.test.ts +++ b/packages/stitch/tests/mergeConflicts.test.ts @@ -98,8 +98,8 @@ describe('merge conflict handlers', () => { expect(Listing.description).toEqual('A type'); expect(IListing.description).toEqual('An interface'); expect(ListingInput.description).toEqual('An input'); - expect(Listing.getFields().id.description).toEqual('type identifier'); - expect(IListing.getFields().id.description).toEqual('interface identifier'); - expect(ListingInput.getFields().id.description).toEqual('input identifier'); + expect(Listing.getFields()['id'].description).toEqual('type identifier'); + expect(IListing.getFields()['id'].description).toEqual('interface identifier'); + expect(ListingInput.getFields()['id'].description).toEqual('input identifier'); }); }); diff --git a/packages/stitch/tests/mergeDefinitions.test.ts b/packages/stitch/tests/mergeDefinitions.test.ts index 3b4f2def57c..17af8555698 100644 --- a/packages/stitch/tests/mergeDefinitions.test.ts +++ b/packages/stitch/tests/mergeDefinitions.test.ts @@ -4,8 +4,6 @@ import { getDirectives } from '@graphql-tools/utils'; import { stitchingDirectives } from '@graphql-tools/stitching-directives'; import { GraphQLObjectType, - GraphQLInputObjectType, - GraphQLEnumType, graphql, } from 'graphql'; import { assertGraphQLEnumType, assertGraphQLInputObjectType, assertGraphQLInterfaceType, assertGraphQLObjectType, assertGraphQLScalerType, assertGraphQLUnionType } from '../../testing/assertion'; @@ -252,20 +250,20 @@ describe('merge canonical types', () => { const enumType = gatewaySchema.getType('ProductEnum'); assertGraphQLEnumType(enumType) - expect(queryType.getFields().field1.description).toEqual('first'); - expect(queryType.getFields().field2.description).toEqual('second'); + expect(queryType.getFields()['field1'].description).toEqual('first'); + expect(queryType.getFields()['field2'].description).toEqual('second'); - expect(objectType.getFields().id.description).toEqual('first'); - expect(interfaceType.getFields().id.description).toEqual('first'); - expect(inputType.getFields().id.description).toEqual('first'); + expect(objectType.getFields()['id'].description).toEqual('first'); + expect(interfaceType.getFields()['id'].description).toEqual('first'); + expect(inputType.getFields()['id'].description).toEqual('first'); - expect(objectType.getFields().url.description).toEqual('second'); - expect(interfaceType.getFields().url.description).toEqual('second'); - expect(inputType.getFields().url.description).toEqual('second'); + expect(objectType.getFields()['url'].description).toEqual('second'); + expect(interfaceType.getFields()['url'].description).toEqual('second'); + expect(inputType.getFields()['url'].description).toEqual('second'); - expect(enumType.toConfig().values.YES.description).toEqual('first'); - expect(enumType.toConfig().values.NO.description).toEqual('first'); - expect(enumType.toConfig().values.MAYBE.description).toEqual('second'); + expect(enumType.toConfig().values['YES'].description).toEqual('first'); + expect(enumType.toConfig().values['NO'].description).toEqual('first'); + expect(enumType.toConfig().values['MAYBE'].description).toEqual('second'); }); it('merges prioritized ASTs', () => { @@ -284,35 +282,35 @@ describe('merge canonical types', () => { const scalarType = gatewaySchema.getType('ProductScalar'); assertGraphQLScalerType(scalarType) - expect(getDirectives(firstSchema, queryType.toConfig()).mydir.value).toEqual('first'); - expect(getDirectives(firstSchema, objectType.toConfig()).mydir.value).toEqual('first'); - expect(getDirectives(firstSchema, interfaceType.toConfig()).mydir.value).toEqual('first'); - expect(getDirectives(firstSchema, inputType.toConfig()).mydir.value).toEqual('first'); - expect(getDirectives(firstSchema, enumType.toConfig()).mydir.value).toEqual('first'); - expect(getDirectives(firstSchema, unionType.toConfig()).mydir.value).toEqual('first'); - expect(getDirectives(firstSchema, scalarType.toConfig()).mydir.value).toEqual('first'); - - expect(getDirectives(firstSchema, queryType.getFields().field1).mydir.value).toEqual('first'); - expect(getDirectives(firstSchema, queryType.getFields().field2).mydir.value).toEqual('second'); - expect(getDirectives(firstSchema, objectType.getFields().id).mydir.value).toEqual('first'); - expect(getDirectives(firstSchema, objectType.getFields().url).mydir.value).toEqual('second'); - expect(getDirectives(firstSchema, interfaceType.getFields().id).mydir.value).toEqual('first'); - expect(getDirectives(firstSchema, interfaceType.getFields().url).mydir.value).toEqual('second'); - expect(getDirectives(firstSchema, inputType.getFields().id).mydir.value).toEqual('first'); - expect(getDirectives(firstSchema, inputType.getFields().url).mydir.value).toEqual('second'); + expect(getDirectives(firstSchema, queryType.toConfig())['mydir'].value).toEqual('first'); + expect(getDirectives(firstSchema, objectType.toConfig())['mydir'].value).toEqual('first'); + expect(getDirectives(firstSchema, interfaceType.toConfig())['mydir'].value).toEqual('first'); + expect(getDirectives(firstSchema, inputType.toConfig())['mydir'].value).toEqual('first'); + expect(getDirectives(firstSchema, enumType.toConfig())['mydir'].value).toEqual('first'); + expect(getDirectives(firstSchema, unionType.toConfig())['mydir'].value).toEqual('first'); + expect(getDirectives(firstSchema, scalarType.toConfig())['mydir'].value).toEqual('first'); + + expect(getDirectives(firstSchema, queryType.getFields()['field1'])['mydir'].value).toEqual('first'); + expect(getDirectives(firstSchema, queryType.getFields()['field2'])['mydir'].value).toEqual('second'); + expect(getDirectives(firstSchema, objectType.getFields()['id'])['mydir'].value).toEqual('first'); + expect(getDirectives(firstSchema, objectType.getFields()['url'])['mydir'].value).toEqual('second'); + expect(getDirectives(firstSchema, interfaceType.getFields()['id'])['mydir'].value).toEqual('first'); + expect(getDirectives(firstSchema, interfaceType.getFields()['url'])['mydir'].value).toEqual('second'); + expect(getDirectives(firstSchema, inputType.getFields()['id'])['mydir'].value).toEqual('first'); + expect(getDirectives(firstSchema, inputType.getFields()['url'])['mydir'].value).toEqual('second'); expect(enumType.toConfig().astNode?.values?.map(v => v.description?.value)).toEqual(['first', 'first', 'second']); - expect(enumType.toConfig().values.YES.astNode?.description?.value).toEqual('first'); - expect(enumType.toConfig().values.NO.astNode?.description?.value).toEqual('first'); - expect(enumType.toConfig().values.MAYBE.astNode?.description?.value).toEqual('second'); + expect(enumType.toConfig().values['YES'].astNode?.description?.value).toEqual('first'); + expect(enumType.toConfig().values['NO'].astNode?.description?.value).toEqual('first'); + expect(enumType.toConfig().values['MAYBE'].astNode?.description?.value).toEqual('second'); }); it('merges prioritized deprecations', () => { const objectType = gatewaySchema.getType('Product') as GraphQLObjectType; - expect(objectType.getFields().id.deprecationReason).toEqual('first'); - expect(objectType.getFields().url.deprecationReason).toEqual('second'); - expect(getDirectives(firstSchema, objectType.getFields().id).deprecated.reason).toEqual('first'); - expect(getDirectives(firstSchema, objectType.getFields().url).deprecated.reason).toEqual('second'); + expect(objectType.getFields()['id'].deprecationReason).toEqual('first'); + expect(objectType.getFields()['url'].deprecationReason).toEqual('second'); + expect(getDirectives(firstSchema, objectType.getFields()['id'])['deprecated'].reason).toEqual('first'); + expect(getDirectives(firstSchema, objectType.getFields()['url'])['deprecated'].reason).toEqual('second'); }); it('promotes canonical root field definitions', async () => { @@ -398,9 +396,9 @@ describe('merge @canonical directives', () => { expect(objectType.description).toEqual('first'); expect(inputType.description).toEqual('first'); expect(enumType.description).toEqual('first'); - expect(queryType.getFields().product.description).toEqual('first'); - expect(objectType.getFields().id.description).toEqual('first'); - expect(objectType.getFields().name.description).toEqual('second'); - expect(inputType.getFields().value.description).toEqual('second'); + expect(queryType.getFields()['product'].description).toEqual('first'); + expect(objectType.getFields()['id'].description).toEqual('first'); + expect(objectType.getFields()['name'].description).toEqual('second'); + expect(inputType.getFields()['value'].description).toEqual('second'); }); }); diff --git a/packages/stitch/tests/mergeInterfaces.test.ts b/packages/stitch/tests/mergeInterfaces.test.ts index 04d7a9e1593..5555c525c51 100644 --- a/packages/stitch/tests/mergeInterfaces.test.ts +++ b/packages/stitch/tests/mergeInterfaces.test.ts @@ -79,7 +79,7 @@ describe('merged interfaces via concrete type', () => { `); assertSome(result.data) - expect(result.data.placement).toEqual({ id: '23', index: 23, name: 'Item 23' }); + expect(result.data['placement']).toEqual({ id: '23', index: 23, name: 'Item 23' }); }); test('works without selection set key', async () => { @@ -93,7 +93,7 @@ describe('merged interfaces via concrete type', () => { `); assertSome(result.data) - expect(result.data.placement).toEqual({ index: 23, name: 'Item 23' }); + expect(result.data['placement']).toEqual({ index: 23, name: 'Item 23' }); }); }); @@ -174,7 +174,7 @@ describe('merged interfaces via abstract type', () => { assertSome(result.data) - expect(result.data.placement).toEqual({ id: '23', index: 23, name: 'Item 23' }); + expect(result.data['placement']).toEqual({ id: '23', index: 23, name: 'Item 23' }); }); test('works without selection set key', async () => { @@ -187,6 +187,6 @@ describe('merged interfaces via abstract type', () => { } `); assertSome(result.data) - expect(result.data.placement).toEqual({ index: 23, name: 'Item 23' }); + expect(result.data['placement']).toEqual({ index: 23, name: 'Item 23' }); }); }); diff --git a/packages/stitch/tests/mergeMultipleEntryPoints.test.ts b/packages/stitch/tests/mergeMultipleEntryPoints.test.ts index 410d29928a0..b57e3e0dc98 100644 --- a/packages/stitch/tests/mergeMultipleEntryPoints.test.ts +++ b/packages/stitch/tests/mergeMultipleEntryPoints.test.ts @@ -126,7 +126,7 @@ describe('merge on multiple keys', () => { } `); assertSome(data) - expect(data.productsByKey).toEqual(result); + expect(data['productsByKey']).toEqual(result); }); test('works from upc -> join -> id', async () => { @@ -142,7 +142,7 @@ describe('merge on multiple keys', () => { } `); assertSome(data) - expect(data.productsByUpc).toEqual(result); + expect(data['productsByUpc']).toEqual(result); }); test('works from id -> join -> upc', async () => { @@ -158,6 +158,6 @@ describe('merge on multiple keys', () => { } `); assertSome(data) - expect(data.productsById).toEqual(result); + expect(data['productsById']).toEqual(result); }); }); diff --git a/packages/stitch/tests/selectionSets.test.ts b/packages/stitch/tests/selectionSets.test.ts index 665342bc219..efa300c5974 100644 --- a/packages/stitch/tests/selectionSets.test.ts +++ b/packages/stitch/tests/selectionSets.test.ts @@ -20,7 +20,7 @@ describe('delegateToSchema ', () => { properties: Record, name: string, ): Property | undefined { - for (const key of Object.keys(properties)) { + for (const key in properties) { const property = properties[key]; if (property.location.name === name) { return property; @@ -96,7 +96,7 @@ describe('delegateToSchema ', () => { bookingById: { property: { location: { - coordinates: sampleData.Property.p1.location.coordinates, + coordinates: sampleData.Property['p1'].location.coordinates, }, }, }, @@ -214,7 +214,7 @@ describe('delegateToSchema ', () => { ); assertSome(data) - expect(data.posts).toEqual(expectedData); + expect(data['posts']).toEqual(expectedData); }); it('should resolve with a fragment', async () => { @@ -239,7 +239,7 @@ describe('delegateToSchema ', () => { `, ); assertSome(data) - expect(data.posts).toEqual(expectedData); + expect(data['posts']).toEqual(expectedData); }); it('should resolve with deep fragment', async () => { @@ -264,7 +264,7 @@ describe('delegateToSchema ', () => { `, ); assertSome(data) - expect(data.posts).toEqual(expectedData); + expect(data['posts']).toEqual(expectedData); }); it('should resolve with nested fragments', async () => { @@ -293,7 +293,7 @@ describe('delegateToSchema ', () => { `, ) assertSome(data) - expect(data.posts).toEqual(expectedData); + expect(data['posts']).toEqual(expectedData); }); }); }); diff --git a/packages/stitch/tests/splitMergedTypeEntryPointsTransformer.test.ts b/packages/stitch/tests/splitMergedTypeEntryPointsTransformer.test.ts index bda4d300c5f..16f52ac6dba 100644 --- a/packages/stitch/tests/splitMergedTypeEntryPointsTransformer.test.ts +++ b/packages/stitch/tests/splitMergedTypeEntryPointsTransformer.test.ts @@ -20,7 +20,7 @@ describe('splitMergedTypeEntryPointsTransformer', () => { expect(results.length).toEqual(1); assertSome(results[0].merge) - expect(results[0].merge.Product).toEqual({ + expect(results[0].merge['Product']).toEqual({ selectionSet: '{ yep }', fieldName: 'yep', }); @@ -82,12 +82,12 @@ describe('splitMergedTypeEntryPointsTransformer', () => { expect(results.length).toEqual(2); assertSome(results[0].merge) - expect(results[0].merge.Product).toEqual({ + expect(results[0].merge['Product']).toEqual({ selectionSet: '{ id }', fieldName: 'productById', }); assertSome(results[1].merge) - expect(results[1].merge.Product).toEqual({ + expect(results[1].merge['Product']).toEqual({ selectionSet: '{ upc }', fieldName: 'productByUpc', }); diff --git a/packages/stitch/tests/stitchSchemas.test.ts b/packages/stitch/tests/stitchSchemas.test.ts index beaf1a7b1ac..91fec6016e9 100644 --- a/packages/stitch/tests/stitchSchemas.test.ts +++ b/packages/stitch/tests/stitchSchemas.test.ts @@ -1,12 +1,10 @@ import { graphql, GraphQLSchema, - GraphQLField, GraphQLObjectType, GraphQLScalarType, subscribe, parse, - defaultFieldResolver, findDeprecatedUsages, printSchema, GraphQLResolveInfo, @@ -18,7 +16,6 @@ import { stitchSchemas } from '../src/stitchSchemas'; import { cloneSchema, getResolversFromSchema, - SchemaDirectiveVisitor, IResolvers, ExecutionResult, assertSome, @@ -328,7 +325,7 @@ const schemaDirectiveTypeDefs = ` } `; -testCombinations.forEach((combination) => { +for (const combination of testCombinations) { describe('merging ' + combination.name, () => { let stitchedSchema: GraphQLSchema; let propertySchema: GraphQLSchema | SubschemaConfig; @@ -356,20 +353,6 @@ testCombinations.forEach((combination) => { codeCoverageTypeDefs, schemaDirectiveTypeDefs, ], - schemaDirectives: { - upper: class extends SchemaDirectiveVisitor { - public visitFieldDefinition(field: GraphQLField) { - const { resolve = defaultFieldResolver } = field; - field.resolve = async function (...args) { - const result = await resolve.apply(this, args); - if (typeof result === 'string') { - return result.toUpperCase(); - } - return result; - }; - } - }, - }, mergeDirectives: true, resolvers: { Property: { @@ -2240,8 +2223,8 @@ fragment BookingFragment on Booking { expect(originalResult.errors).toBeUndefined(); expect(originalResult.data).toBeDefined(); assertSome(originalResult.data) - expect(originalResult.data.persona.transactions.items.length).toBe(2); - expect(originalResult.data.persona.transactions.items[1].debt).toBeDefined(); + expect(originalResult.data['persona'].transactions.items.length).toBe(2); + expect(originalResult.data['persona'].transactions.items[1].debt).toBeDefined(); const mergedSchema = stitchSchemas({ subschemas: [remoteSchema], @@ -2340,8 +2323,8 @@ fragment BookingFragment on Booking { }, transformResult: (originalResult: ExecutionResult) => { assertSome(originalResult.data) - originalResult.data.persona = { - page: originalResult.data.persona.transactions.items, + originalResult.data['persona'] = { + page: originalResult.data['persona'].transactions.items, }; return originalResult; }, @@ -2383,8 +2366,8 @@ fragment BookingFragment on Booking { expect(result.errors).toBeUndefined(); assertSome(result.data) - expect(result.data.flattenedTransactions.page.length).toBe(2); - expect(result.data.flattenedTransactions.page[1].debt).toBeDefined(); + expect(result.data['flattenedTransactions'].page.length).toBe(2); + expect(result.data['flattenedTransactions'].page[1].debt).toBeDefined(); }); test('aliases', async () => { @@ -2696,13 +2679,13 @@ fragment BookingFragment on Booking { const stitchedResult = await graphql(stitchedSchema, propertyQuery, undefined, {}); - [propertyResult, stitchedResult].forEach((result) => { + for (const result of [propertyResult, stitchedResult]) { assertSome(result.errors) expect(result.errors.length > 0).toBe(true); const error = result.errors[0]; assertSome(error.extensions) - expect(error.extensions.code).toBe('SOME_CUSTOM_CODE'); - }); + expect(error.extensions['code']).toBe('SOME_CUSTOM_CODE'); + } }, ); }); @@ -2737,19 +2720,19 @@ fragment BookingFragment on Booking { test('should parse descriptions on new fields', () => { const Query = stitchedSchema.getQueryType(); assertSome(Query) - expect(Query.getFields().linkTest.description).toBe( + expect(Query.getFields()['linkTest'].description).toBe( 'A new field on the root query.', ); const Booking = stitchedSchema.getType('Booking') as GraphQLObjectType; - expect(Booking.getFields().property.description).toBe( + expect(Booking.getFields()['property'].description).toBe( 'The property of the booking.', ); const Property = stitchedSchema.getType( 'Property', ) as GraphQLObjectType; - const bookingsField = Property.getFields().bookings; + const bookingsField = Property.getFields()['bookings']; expect(bookingsField.description).toBe('A list of bookings.'); expect(bookingsField.args[0].description).toBe( 'The maximum number of bookings to retrieve.', @@ -3087,31 +3070,6 @@ fragment BookingFragment on Booking { }); }); - describe('schema directives', () => { - test('should work with schema directives', async () => { - const result = await graphql( - stitchedSchema, - ` - query { - propertyById(id: "p1") { - someField - } - } - `, - undefined, - {}, - ); - - expect(result).toEqual({ - data: { - propertyById: { - someField: 'SOMEFIELD', - }, - }, - }); - }); - }); - describe('regression', () => { test('should not pass extra arguments to delegates', async () => { const result = await graphql( @@ -3161,7 +3119,7 @@ fragment BookingFragment on Booking { const result = await graphql(schema, '{ book { cat: category } }'); assertSome(result.data) - expect(result.data.book.cat).toBe('Test'); + expect(result.data['book'].cat).toBe('Test'); }); }); @@ -3294,7 +3252,7 @@ assertSome(result.data) const result = await graphql(schema, '{ book { cat: category } }'); assertSome(result.data) - expect(result.data.book.cat).toBe('Test'); + expect(result.data['book'].cat).toBe('Test'); }); }); @@ -3552,4 +3510,4 @@ assertSome(result.data) }); }); }); -}); +} diff --git a/packages/stitch/tests/stitchingFromSubschemas.test.ts b/packages/stitch/tests/stitchingFromSubschemas.test.ts deleted file mode 100644 index a30a2e0afe9..00000000000 --- a/packages/stitch/tests/stitchingFromSubschemas.test.ts +++ /dev/null @@ -1,148 +0,0 @@ -// The below is meant to be an alternative canonical schema stitching example -// which intermingles local (mocked) resolvers and stitched schemas and does -// not require use of the fragment field, because it follows best practices of -// always returning the necessary object fields: -// https://medium.com/paypal-engineering/graphql-resolvers-best-practices-cd36fdbcef55 - -// This is achieved at the considerable cost of moving all of the delegation -// logic from the gateway to each subschema so that each subschema imports all -// the required types and performs all delegation. - -// The fragment field is still necessary when working with a remote schema -// where this is not possible. - -import { graphql, GraphQLSchema } from 'graphql'; - -import { delegateToSchema } from '@graphql-tools/delegate'; -import { addMocksToSchema } from '@graphql-tools/mock'; -import { assertSome } from '@graphql-tools/utils'; - -import { stitchSchemas } from '../src/stitchSchemas'; - -const chirpTypeDefs = ` - type Chirp { - id: ID! - text: String - authorId: ID! - author: User - } -`; - -const authorTypeDefs = ` - type User { - id: ID! - email: String - chirps: [Chirp] - } -`; - -const schemas: Record = {}; -const getSchema = (name: string) => schemas[name]; - -let chirpSchema = stitchSchemas({ - typeDefs: [ - chirpTypeDefs, - authorTypeDefs, - ` - type Query { - chirpById(id: ID!): Chirp - chirpsByAuthorId(authorId: ID!): [Chirp] - } - `, - ], - resolvers: { - Chirp: { - author: (chirp, _args, context, info) => - delegateToSchema({ - schema: getSchema('authorSchema'), - operation: 'query', - fieldName: 'userById', - args: { - id: chirp.authorId, - }, - context, - info, - }), - }, - }, -}); - -chirpSchema = addMocksToSchema({ - schema: chirpSchema, - mocks: { - Chirp: () => ({ - authorId: '1', - }), - }, - preserveResolvers: true, -}); - -let authorSchema = stitchSchemas({ - typeDefs: [ - chirpTypeDefs, - authorTypeDefs, - ` - type Query { - userById(id: ID!): User - } - `, - ], - resolvers: { - User: { - chirps: (user, _args, context, info) => - delegateToSchema({ - schema: getSchema('chirpSchema'), - operation: 'query', - fieldName: 'chirpsByAuthorId', - args: { - authorId: user.id, - }, - context, - info, - }), - }, - }, -}); - -authorSchema = addMocksToSchema({ - schema: authorSchema, - mocks: { - User: () => ({ - id: '1', - }), - }, - preserveResolvers: true, -}); - -schemas.chirpSchema = chirpSchema; -schemas.authorSchema = authorSchema; - -const stitchedSchema = stitchSchemas({ - subschemas:Object.keys(schemas).map((schemaName) => schemas[schemaName]), -}); - -describe('merging without specifying fragments', () => { - test.skip('works', async () => { - const query = ` - query { - userById(id: 5) { - chirps { - id - textAlias: text - author { - email - } - } - } - } - `; - - const result = await graphql(stitchedSchema, query); - - expect(result.errors).toBeUndefined(); - assertSome(result.data) - expect(result.data.userById.chirps[1].id).not.toBe(null); - expect(result.data.userById.chirps[1].text).not.toBe(null); - expect(result.data.userById.chirps[1].author.email).not.toBe(null); - }); -}); diff --git a/packages/stitch/tests/typeMerging.test.ts b/packages/stitch/tests/typeMerging.test.ts index 1b6db5ce120..284c1f8b0c0 100644 --- a/packages/stitch/tests/typeMerging.test.ts +++ b/packages/stitch/tests/typeMerging.test.ts @@ -110,10 +110,10 @@ describe('merging using type merging', () => { expect(result.errors).toBeUndefined(); assertSome(result.data) - expect(result.data.userById.__typename).toBe('User'); - expect(result.data.userById.chirps[1].id).not.toBe(null); - expect(result.data.userById.chirps[1].text).not.toBe(null); - expect(result.data.userById.chirps[1].author.email).not.toBe(null); + expect(result.data['userById'].__typename).toBe('User'); + expect(result.data['userById'].chirps[1].id).not.toBe(null); + expect(result.data['userById'].chirps[1].text).not.toBe(null); + expect(result.data['userById'].chirps[1].author.email).not.toBe(null); }); test("handle top level failures on subschema queries", async() => { @@ -426,7 +426,7 @@ describe('Merged associations', () => { } `); assertSome(data) - expect(data.posts).toEqual([{ + expect(data['posts']).toEqual([{ title: 'Post 55', network: { domain: 'network57.com' }, sections: ['News'] @@ -531,10 +531,10 @@ describe('merging using type merging when renaming', () => { expect(result.errors).toBeUndefined(); assertSome(result.data) - expect(result.data.User_userById.__typename).toBe('Gateway_User'); - expect(result.data.User_userById.chirps[1].id).not.toBe(null); - expect(result.data.User_userById.chirps[1].text).not.toBe(null); - expect(result.data.User_userById.chirps[1].author.email).not.toBe(null); + expect(result.data['User_userById'].__typename).toBe('Gateway_User'); + expect(result.data['User_userById'].chirps[1].id).not.toBe(null); + expect(result.data['User_userById'].chirps[1].text).not.toBe(null); + expect(result.data['User_userById'].chirps[1].author.email).not.toBe(null); }); }); @@ -622,7 +622,7 @@ describe('external object annotation with batchDelegateToSchema', () => { `, ) assertSome(data) - expect(data.posts).toEqual([ + expect(data['posts']).toEqual([ { network: { id: '57', domains: [{ id: '60', name: 'network57.com' }] }, }, diff --git a/packages/stitch/tests/typeMergingWithDirectives.test.ts b/packages/stitch/tests/typeMergingWithDirectives.test.ts index 389ef1d8246..de32835db5d 100644 --- a/packages/stitch/tests/typeMergingWithDirectives.test.ts +++ b/packages/stitch/tests/typeMergingWithDirectives.test.ts @@ -56,7 +56,7 @@ describe('merging using type merging with directives', () => { resolvers: { Query: { me: () => users[0], - _users: (_root, { keys }) => keys.map((key: Record) => users.find(u => u.id === key.id)), + _users: (_root, { keys }) => keys.map((key: Record) => users.find(u => u.id === key['id'])), }, }, schemaTransforms: [stitchingDirectivesValidator], @@ -122,7 +122,7 @@ describe('merging using type merging with directives', () => { Query: { mostStockedProduct: () => inventory.find(i => i.upc === '3'), _products: (_root, { keys }) => { - return keys.map((key: Record) => ({ ...key, ...inventory.find(i => i.upc === key.upc) })); + return keys.map((key: Record) => ({ ...key, ...inventory.find(i => i.upc === key['upc']) })); }, }, }, diff --git a/packages/stitch/tests/typeMergingWithExtensions.test.ts b/packages/stitch/tests/typeMergingWithExtensions.test.ts index e47b7247b3a..a84c33d130d 100644 --- a/packages/stitch/tests/typeMergingWithExtensions.test.ts +++ b/packages/stitch/tests/typeMergingWithExtensions.test.ts @@ -35,7 +35,7 @@ describe('merging using type merging', () => { accountsSchemaTypes._Key = new GraphQLScalarType({ name: '_Key', - }); + } as any); accountsSchemaTypes.Query = new GraphQLObjectType({ name: 'Query', fields: () => ({ @@ -50,7 +50,7 @@ describe('merging using type merging', () => { type: new GraphQLNonNull(new GraphQLList(new GraphQLNonNull(accountsSchemaTypes._Key))) }, }, - resolve: (_root, { keys }) => keys.map((key: Record) => users.find(u => u.id === key.id)), + resolve: (_root, { keys }) => keys.map((key: Record) => users.find(u => u.id === key['id'])), extensions: { directives: { merge: {}, @@ -138,7 +138,7 @@ describe('merging using type merging', () => { keys: { type: new GraphQLNonNull(new GraphQLList(new GraphQLNonNull(inventorySchemaTypes.ProductKey))) } }, resolve: (_root, { keys }) => { - return keys.map((key: Record) => ({ ...key, ...inventory.find(i => i.upc === key.upc) })); + return keys.map((key: Record) => ({ ...key, ...inventory.find(i => i.upc === key['upc']) })); }, extensions: { directives: { @@ -204,7 +204,7 @@ describe('merging using type merging', () => { defaultValue: 2, }, }, - resolve: (_root, args) => products.slice(0, args.first), + resolve: (_root, args) => products.slice(0, args['first']), }, _productsByUpc: { type: new GraphQLList(productsSchemaTypes.Product), diff --git a/packages/stitch/tests/typeMergingWithInterfaces.test.ts b/packages/stitch/tests/typeMergingWithInterfaces.test.ts index d575cd20dc5..5b0fa9511a2 100644 --- a/packages/stitch/tests/typeMergingWithInterfaces.test.ts +++ b/packages/stitch/tests/typeMergingWithInterfaces.test.ts @@ -58,7 +58,7 @@ describe('merging using type merging', () => { Query: { me: () => users[0], _entities: (_root, { keys }) => { - return keys.map((key: Record) => ({ ...key, ...users.find(u => u.id === key.id) })); + return keys.map((key: Record) => ({ ...key, ...users.find(u => u.id === key['id']) })); }, }, }, @@ -122,7 +122,7 @@ describe('merging using type merging', () => { Query: { mostStockedProduct: () => inventory.find(i => i.upc === '3'), _entities: (_root, { keys }) => { - return keys.map((key: Record) => ({ ...key, ...inventory.find(i => i.upc === key.upc) })); + return keys.map((key: Record) => ({ ...key, ...inventory.find(i => i.upc === key['upc']) })); }, }, }, @@ -187,7 +187,7 @@ describe('merging using type merging', () => { Query: { topProducts: (_root, args) => products.slice(0, args.first), _entities: (_root, { keys }) => { - return keys.map((key: Record) => ({ ...key, ...products.find(product => product.upc === key.upc) })); + return keys.map((key: Record) => ({ ...key, ...products.find(product => product.upc === key['upc']) })); } } }, @@ -281,8 +281,8 @@ describe('merging using type merging', () => { Query: { _entities: (_root, { keys }) => { return keys.map((key: Record) => { - if (key.__typename === 'Review') { - return ({ ...key, ...reviews.find(review => review.id === key.id) }); + if (key['__typename'] === 'Review') { + return ({ ...key, ...reviews.find(review => review.id === key['id']) }); } return { ...key }; diff --git a/packages/stitching-directives/src/getSourcePaths.ts b/packages/stitching-directives/src/getSourcePaths.ts index b361130d959..c5dbf7b6f9d 100644 --- a/packages/stitching-directives/src/getSourcePaths.ts +++ b/packages/stitching-directives/src/getSourcePaths.ts @@ -9,22 +9,24 @@ export function getSourcePaths( selectionSet?: SelectionSetNode ): Array> { const sourcePaths: Array> = []; - mappingInstructions.forEach(mappingInstruction => { + for (const mappingInstruction of mappingInstructions) { const { sourcePath } = mappingInstruction; if (sourcePath.length) { sourcePaths.push(sourcePath); - return; + continue; } if (selectionSet == null) { - return; + continue; } const paths = pathsFromSelectionSet(selectionSet); - paths.forEach(path => sourcePaths.push(path)); + for (const path of paths) { + sourcePaths.push(path); + } sourcePaths.push([TypeNameMetaFieldDef.name]); - }); + } return sourcePaths; } diff --git a/packages/stitching-directives/src/parseMergeArgsExpr.ts b/packages/stitching-directives/src/parseMergeArgsExpr.ts index c3cfd73c3ab..ae1fcb23f27 100644 --- a/packages/stitching-directives/src/parseMergeArgsExpr.ts +++ b/packages/stitching-directives/src/parseMergeArgsExpr.ts @@ -9,10 +9,7 @@ import { getSourcePaths } from './getSourcePaths'; type VariablePaths = Record>; -export function parseMergeArgsExpr( - mergeArgsExpr: string, - selectionSet?: SelectionSetNode, -): ParsedMergeArgsExpr { +export function parseMergeArgsExpr(mergeArgsExpr: string, selectionSet?: SelectionSetNode): ParsedMergeArgsExpr { const { mergeArgsExpr: newMergeArgsExpr, expansionExpressions } = preparseMergeArgsExpr(mergeArgsExpr); const inputValue = parseValue(`{ ${newMergeArgsExpr} }`, { noLocation: true }); @@ -32,15 +29,15 @@ export function parseMergeArgsExpr( } const expansionRegEx = new RegExp(`^${EXPANSION_PREFIX}[0-9]+$`); - Object.keys(variablePaths).forEach(variableName => { + for (const variableName in variablePaths) { if (!variableName.match(expansionRegEx)) { throw new Error('Expansions cannot be mixed with single key declarations.'); } - }); + } const expansions: Array = []; const sourcePaths: Array> = []; - Object.keys(expansionExpressions).forEach(variableName => { + for (const variableName in expansionExpressions) { const str = expansionExpressions[variableName]; const valuePath = variablePaths[variableName]; const { inputValue: expansionInputValue, variablePaths: expansionVariablePaths } = extractVariables( @@ -57,12 +54,13 @@ export function parseMergeArgsExpr( sourcePaths.push(...getSourcePaths(mappingInstructions, selectionSet)); + assertNotWithinList(valuePath); expansions.push({ - valuePath: assertNotWithinList(valuePath), + valuePath, value, mappingInstructions, }); - }); + } const usedProperties = propertyTreeFromPaths(sourcePaths); @@ -71,23 +69,24 @@ export function parseMergeArgsExpr( function getMappingInstructions(variablePaths: VariablePaths): Array { const mappingInstructions: Array = []; - Object.entries(variablePaths).forEach(([keyPath, valuePath]) => { + for (const keyPath in variablePaths) { + const valuePath = variablePaths[keyPath]; const splitKeyPath = keyPath.split(KEY_DELIMITER).slice(1); + assertNotWithinList(valuePath); mappingInstructions.push({ - destinationPath: assertNotWithinList(valuePath), + destinationPath: valuePath, sourcePath: splitKeyPath, }); - }); + } return mappingInstructions; } -function assertNotWithinList(path: Array): Array { - path.forEach(pathSegment => { +function assertNotWithinList(path: Array): asserts path is string[] { + for (const pathSegment of path) { if (typeof pathSegment === 'number') { throw new Error('Insertions cannot be made into a list.'); } - }); - return path as Array; + } } diff --git a/packages/stitching-directives/src/pathsFromSelectionSet.ts b/packages/stitching-directives/src/pathsFromSelectionSet.ts index bca1586562f..9d452734e49 100644 --- a/packages/stitching-directives/src/pathsFromSelectionSet.ts +++ b/packages/stitching-directives/src/pathsFromSelectionSet.ts @@ -1,11 +1,13 @@ import { Kind, SelectionNode, SelectionSetNode } from 'graphql'; export function pathsFromSelectionSet(selectionSet: SelectionSetNode, path: Array = []): Array> { - let paths: Array> = []; - selectionSet.selections.forEach(selection => { - const addition = pathsFromSelection(selection, path) ?? []; - paths = [...paths, ...addition]; - }); + const paths: Array> = []; + for (const selection of selectionSet.selections) { + const additions = pathsFromSelection(selection, path) ?? []; + for (const addition of additions) { + paths.push(addition); + } + } return paths; } diff --git a/packages/stitching-directives/src/properties.ts b/packages/stitching-directives/src/properties.ts index b0166b3e6e0..deca3896bef 100644 --- a/packages/stitching-directives/src/properties.ts +++ b/packages/stitching-directives/src/properties.ts @@ -44,25 +44,27 @@ export function getProperties(object: Record, propertyTree: Propert const newObject = Object.create(null); - Object.entries(propertyTree).forEach(([key, subKey]) => { + for (const key in propertyTree) { + const subKey = propertyTree[key]; + if (subKey == null) { newObject[key] = object[key]; - return; + continue; } const prop = object[key]; newObject[key] = deepMap(prop, item => getProperties(item, subKey)); - }); + } return newObject; } export function propertyTreeFromPaths(paths: Array>): PropertyTree { const propertyTree = Object.create(null); - paths.forEach(path => { + for (const path of paths) { addProperty(propertyTree, path, null); - }); + } return propertyTree; } diff --git a/packages/stitching-directives/src/stitchingDirectivesTransformer.ts b/packages/stitching-directives/src/stitchingDirectivesTransformer.ts index 78cb22311c4..27691311b2b 100644 --- a/packages/stitching-directives/src/stitchingDirectivesTransformer.ts +++ b/packages/stitching-directives/src/stitchingDirectivesTransformer.ts @@ -194,7 +194,8 @@ export function stitchingDirectivesTransformer( }); if (subschemaConfig.merge) { - Object.entries(subschemaConfig.merge).forEach(([typeName, mergedTypeConfig]) => { + for (const typeName in subschemaConfig.merge) { + const mergedTypeConfig = subschemaConfig.merge[typeName]; if (mergedTypeConfig.selectionSet) { const selectionSet = parseSelectionSet(mergedTypeConfig.selectionSet, { noLocation: true }); if (selectionSet) { @@ -206,8 +207,9 @@ export function stitchingDirectivesTransformer( } } if (mergedTypeConfig.fields) { - Object.entries(mergedTypeConfig.fields).forEach(([fieldName, fieldConfig]) => { - if (!fieldConfig.selectionSet) return; + for (const fieldName in mergedTypeConfig.fields) { + const fieldConfig = mergedTypeConfig.fields[fieldName]; + if (!fieldConfig.selectionSet) continue; const selectionSet = parseSelectionSet(fieldConfig.selectionSet, { noLocation: true }); if (selectionSet) { @@ -223,30 +225,27 @@ export function stitchingDirectivesTransformer( computedFieldSelectionSets[typeName][fieldName] = selectionSet; } } - }); + } } - }); + } } const allSelectionSetsByType: Record> = Object.create(null); - Object.entries(selectionSetsByType).forEach(([typeName, selectionSet]) => { - if (allSelectionSetsByType[typeName] == null) { - allSelectionSetsByType[typeName] = [selectionSet]; - } else { + for (const typeName in selectionSetsByType) { + allSelectionSetsByType[typeName] = allSelectionSetsByType[typeName] || []; + const selectionSet = selectionSetsByType[typeName]; + allSelectionSetsByType[typeName].push(selectionSet); + } + + for (const typeName in computedFieldSelectionSets) { + const selectionSets = computedFieldSelectionSets[typeName]; + for (const i in selectionSets) { + allSelectionSetsByType[typeName] = allSelectionSetsByType[typeName] || []; + const selectionSet = selectionSets[i]; allSelectionSetsByType[typeName].push(selectionSet); } - }); - - Object.entries(computedFieldSelectionSets).forEach(([typeName, selectionSets]) => { - Object.values(selectionSets).forEach(selectionSet => { - if (allSelectionSetsByType[typeName] == null) { - allSelectionSetsByType[typeName] = [selectionSet]; - } else { - allSelectionSetsByType[typeName].push(selectionSet); - } - }); - }); + } mapSchema(schema, { [MapperKind.OBJECT_FIELD]: (fieldConfig, fieldName) => { @@ -272,9 +271,9 @@ export function stitchingDirectivesTransformer( const lastArgName = argNames.pop(); mergeArgsExpr = returnsList ? `${lastArgName}: [[${keyExpr}]]` : `${lastArgName}: ${keyExpr}`; - argNames.reverse().forEach(argName => { + for (const argName of argNames.reverse()) { mergeArgsExpr = `${argName}: { ${mergeArgsExpr} }`; - }); + } } const typeNames: Array = directiveArgumentMap.types; @@ -307,9 +306,10 @@ export function stitchingDirectivesTransformer( }, }); - Object.entries(selectionSetsByType).forEach(([typeName, selectionSet]) => { - const mergeConfig: Record> = - newSubschemaConfig.merge ?? Object.create(null); + for (const typeName in selectionSetsByType) { + const selectionSet = selectionSetsByType[typeName]; + const mergeConfig: Record> = newSubschemaConfig.merge ?? + Object.create(null); newSubschemaConfig.merge = mergeConfig; if (mergeConfig[typeName] == null) { @@ -319,11 +319,12 @@ export function stitchingDirectivesTransformer( const mergeTypeConfig = mergeConfig[typeName]; mergeTypeConfig.selectionSet = print(selectionSet); - }); + } - Object.entries(computedFieldSelectionSets).forEach(([typeName, selectionSets]) => { - const mergeConfig: Record> = - newSubschemaConfig.merge ?? Object.create(null); + for (const typeName in computedFieldSelectionSets) { + const selectionSets = computedFieldSelectionSets[typeName]; + const mergeConfig: Record> = newSubschemaConfig.merge ?? + Object.create(null); newSubschemaConfig.merge = mergeConfig; if (mergeConfig[typeName] == null) { @@ -334,18 +335,21 @@ export function stitchingDirectivesTransformer( const mergeTypeConfigFields: Record = mergeTypeConfig.fields ?? Object.create(null); mergeTypeConfig.fields = mergeTypeConfigFields; - Object.entries(selectionSets).forEach(([fieldName, selectionSet]) => { + for (const fieldName in selectionSets) { + const selectionSet = selectionSets[fieldName]; const fieldConfig: MergedFieldConfig = mergeTypeConfigFields[fieldName] ?? Object.create(null); mergeTypeConfigFields[fieldName] = fieldConfig; fieldConfig.selectionSet = print(selectionSet); fieldConfig.computed = true; - }); - }); + } + } - Object.entries(mergedTypesResolversInfo).forEach(([typeName, mergedTypeResolverInfo]) => { - const mergeConfig: Record> = - newSubschemaConfig.merge ?? Object.create(null); + for (const typeName in mergedTypesResolversInfo) { + const mergedTypeResolverInfo = mergedTypesResolversInfo[typeName]; + + const mergeConfig: Record> = newSubschemaConfig.merge ?? + Object.create(null); newSubschemaConfig.merge = mergeConfig; if (newSubschemaConfig.merge[typeName] == null) { @@ -362,11 +366,12 @@ export function stitchingDirectivesTransformer( } else { mergeTypeConfig.args = generateArgsFn(mergedTypeResolverInfo); } - }); + } - Object.entries(canonicalTypesInfo).forEach(([typeName, canonicalTypeInfo]) => { - const mergeConfig: Record> = - newSubschemaConfig.merge ?? Object.create(null); + for (const typeName in canonicalTypesInfo) { + const canonicalTypeInfo = canonicalTypesInfo[typeName]; + const mergeConfig: Record> = newSubschemaConfig.merge ?? + Object.create(null); newSubschemaConfig.merge = mergeConfig; if (newSubschemaConfig.merge[typeName] == null) { @@ -382,14 +387,14 @@ export function stitchingDirectivesTransformer( if (canonicalTypeInfo.fields) { const mergeTypeConfigFields: Record = mergeTypeConfig.fields ?? Object.create(null); mergeTypeConfig.fields = mergeTypeConfigFields; - Object.keys(canonicalTypeInfo.fields).forEach(fieldName => { + for (const fieldName in canonicalTypeInfo.fields) { if (mergeTypeConfigFields[fieldName] == null) { mergeTypeConfigFields[fieldName] = Object.create(null); } mergeTypeConfigFields[fieldName].canonical = true; - }); + } } - }); + } return newSubschemaConfig; }; @@ -402,17 +407,17 @@ function forEachConcreteType( fn: (typeName: string) => void ) { if (isInterfaceType(type)) { - getImplementingTypes(type.name, schema).forEach(typeName => { + for (const typeName of getImplementingTypes(type.name, schema)) { if (typeNames == null || typeNames.includes(typeName)) { fn(typeName); } - }); + } } else if (isUnionType(type)) { - type.getTypes().forEach(({ name: typeName }) => { + for (const { name: typeName } of type.getTypes()) { if (typeNames == null || typeNames.includes(typeName)) { fn(typeName); } - }); + } } else if (isObjectType(type)) { fn(type.name); } @@ -428,23 +433,24 @@ function generateArgsFromKeysFn( const { expansions, args } = mergedTypeResolverInfo; return (keys: ReadonlyArray): Record => { const newArgs = mergeDeep({}, args); - expansions?.forEach(expansion => { - const mappingInstructions = expansion.mappingInstructions; - const expanded: Array = []; - keys.forEach(key => { - let newValue = mergeDeep({}, expansion.valuePath); - mappingInstructions.forEach(mappingInstruction => { - const { destinationPath, sourcePath } = mappingInstruction; - if (destinationPath.length) { - addProperty(newValue, destinationPath, getProperty(key, sourcePath)); - } else { - newValue = getProperty(key, sourcePath); + if (expansions) { + for (const expansion of expansions) { + const mappingInstructions = expansion.mappingInstructions; + const expanded: Array = []; + for (const key of keys) { + let newValue = mergeDeep({}, expansion.valuePath); + for (const { destinationPath, sourcePath } of mappingInstructions) { + if (destinationPath.length) { + addProperty(newValue, destinationPath, getProperty(key, sourcePath)); + } else { + newValue = getProperty(key, sourcePath); + } } - }); - expanded.push(newValue); - }); - addProperty(newArgs, expansion.valuePath, expanded); - }); + expanded.push(newValue); + } + addProperty(newArgs, expansion.valuePath, expanded); + } + } return newArgs; }; } @@ -455,17 +461,19 @@ function generateArgsFn(mergedTypeResolverInfo: MergedTypeResolverInfo): (origin return (originalResult: any): Record => { const newArgs = mergeDeep({}, args); const filteredResult = getProperties(originalResult, usedProperties); - mappingInstructions?.forEach(mappingInstruction => { - const { destinationPath, sourcePath } = mappingInstruction; - addProperty(newArgs, destinationPath, getProperty(filteredResult, sourcePath)); - }); + if (mappingInstructions) { + for (const mappingInstruction of mappingInstructions) { + const { destinationPath, sourcePath } = mappingInstruction; + addProperty(newArgs, destinationPath, getProperty(filteredResult, sourcePath)); + } + } return newArgs; }; } function buildKeyExpr(key: Array): string { let mergedObject = {}; - key.forEach(keyDef => { + for (const keyDef of key) { let [aliasOrKeyPath, keyPath] = keyDef.split(':'); let aliasPath: string; if (keyPath == null) { @@ -478,11 +486,11 @@ function buildKeyExpr(key: Array): string { assertSome(lastAliasPart); let object: Record = { [lastAliasPart]: `$key.${keyPath}` }; - aliasParts.reverse().forEach(aliasPart => { + for (const aliasPart of aliasParts.reverse()) { object = { [aliasPart]: object }; - }); + } mergedObject = mergeDeep(mergedObject, object); - }); + } return JSON.stringify(mergedObject).replace(/"/g, ''); } @@ -490,12 +498,12 @@ function buildKeyExpr(key: Array): string { function mergeSelectionSets(...selectionSets: Array): SelectionSetNode { const normalizedSelections: Record = Object.create(null); - selectionSets.forEach(selectionSet => { - selectionSet.selections.forEach(selection => { + for (const selectionSet of selectionSets) { + for (const selection of selectionSet.selections) { const normalizedSelection = print(selection); normalizedSelections[normalizedSelection] = selection; - }); - }); + } + } const newSelectionSet = { kind: Kind.SELECTION_SET, @@ -512,17 +520,17 @@ function forEachConcreteTypeName( fn: (typeName: string) => void ): void { if (isInterfaceType(returnType)) { - getImplementingTypes(returnType.name, schema).forEach(typeName => { + for (const typeName of getImplementingTypes(returnType.name, schema)) { if (typeNames == null || typeNames.includes(typeName)) { fn(typeName); } - }); + } } else if (isUnionType(returnType)) { - returnType.getTypes().forEach(type => { + for (const type of returnType.getTypes()) { if (typeNames == null || typeNames.includes(type.name)) { fn(type.name); } - }); + } } else if (isObjectType(returnType) && (typeNames == null || typeNames.includes(returnType.name))) { fn(returnType.name); } diff --git a/packages/stitching-directives/src/stitchingDirectivesValidator.ts b/packages/stitching-directives/src/stitchingDirectivesValidator.ts index bda6241d88b..92ab3ddd9b5 100644 --- a/packages/stitching-directives/src/stitchingDirectivesValidator.ts +++ b/packages/stitching-directives/src/stitchingDirectivesValidator.ts @@ -108,7 +108,7 @@ export function stitchingDirectivesValidator( throw new Error('Cannot use @merge directive with both `keyField` and `key` arguments.'); } - key.forEach(keyDef => { + for (const keyDef of key) { let [aliasOrKeyPath, keyPath] = keyDef.split(':'); let aliasPath: string; if (keyPath == null) { @@ -129,7 +129,7 @@ export function stitchingDirectivesValidator( ); // TODO: ideally we should check that the arg exists within the resolver } - }); + } } const additionalArgs = directiveArgumentMap.additionalArgs; @@ -156,13 +156,13 @@ export function stitchingDirectivesValidator( ? getImplementingTypes(returnType.name, schema).map(typeName => schema.getType(typeName)) : returnType.getTypes(); const implementingTypeNames = implementingTypes.filter(isSome).map(type => type.name); - typeNames.forEach(typeName => { + for (const typeName of typeNames) { if (!implementingTypeNames.includes(typeName)) { throw new Error( `Types argument can only include only type names that implement the field return type's abstract type.` ); } - }); + } } } diff --git a/packages/stitching-directives/tests/stitchingDirectivesTransformer.test.ts b/packages/stitching-directives/tests/stitchingDirectivesTransformer.test.ts index c15e5de5dfb..29bf970afbd 100644 --- a/packages/stitching-directives/tests/stitchingDirectivesTransformer.test.ts +++ b/packages/stitching-directives/tests/stitchingDirectivesTransformer.test.ts @@ -1,3 +1,4 @@ +/* eslint-disable @typescript-eslint/no-non-null-asserted-optional-chain */ import { print } from 'graphql'; import { makeExecutableSchema } from '@graphql-tools/schema'; @@ -31,8 +32,8 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - expect(transformedSubschemaConfig.merge.User.selectionSet).toEqual(print(parseSelectionSet('{ id }'))); - expect(transformedSubschemaConfig.merge.User.fieldName).toEqual('_user'); + expect(transformedSubschemaConfig.merge?.['User'].selectionSet).toEqual(print(parseSelectionSet('{ id }'))); + expect(transformedSubschemaConfig.merge?.['User'].fieldName).toEqual('_user'); }); test('adds type selection sets when returns union', () => { @@ -60,8 +61,8 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - expect(transformedSubschemaConfig.merge.User.selectionSet).toEqual(print(parseSelectionSet('{ id }'))); - expect(transformedSubschemaConfig.merge.User.fieldName).toEqual('_entity'); + expect(transformedSubschemaConfig.merge?.['User'].selectionSet).toEqual(print(parseSelectionSet('{ id }'))); + expect(transformedSubschemaConfig.merge?.['User'].fieldName).toEqual('_entity'); }); test('adds type selection sets when returns interface', () => { @@ -91,8 +92,8 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - expect(transformedSubschemaConfig.merge.User.selectionSet).toEqual(print(parseSelectionSet('{ id }'))); - expect(transformedSubschemaConfig.merge.User.fieldName).toEqual('_entity'); + expect(transformedSubschemaConfig.merge?.['User'].selectionSet).toEqual(print(parseSelectionSet('{ id }'))); + expect(transformedSubschemaConfig.merge?.['User'].fieldName).toEqual('_entity'); }); test('adds type selection sets when returns list', () => { @@ -117,7 +118,7 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const argsFn = transformedSubschemaConfig.merge.User.args; + const argsFn = transformedSubschemaConfig.merge?.['User'].args!; const originalResult = { relations: [ @@ -168,7 +169,7 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const argsFn = transformedSubschemaConfig.merge.User.args; + const argsFn = transformedSubschemaConfig.merge?.['User'].args!; const originalResult = { relationSets: [ @@ -233,7 +234,7 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const argsFn = transformedSubschemaConfig.merge.User.args; + const argsFn = transformedSubschemaConfig.merge?.['User'].args!; const originalResult: { nestedField: null } = { nestedField: null @@ -271,9 +272,9 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - expect(transformedSubschemaConfig.merge.User.fields.name.selectionSet).toEqual(print(parseSelectionSet('{ id }'))); - expect(transformedSubschemaConfig.merge.User.fields.name.computed).toEqual(true); - expect(transformedSubschemaConfig.merge.User.fieldName).toEqual('_user'); + expect(transformedSubschemaConfig.merge?.['User']?.fields?.['name']?.selectionSet).toEqual(print(parseSelectionSet('{ id }'))); + expect(transformedSubschemaConfig.merge?.['User']?.fields?.['name']?.computed).toEqual(true); + expect(transformedSubschemaConfig.merge?.['User'].fieldName).toEqual('_user'); }); test('adds args function when used without arguments', () => { @@ -299,7 +300,7 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const argsFn = transformedSubschemaConfig.merge.User.args; + const argsFn = transformedSubschemaConfig.merge?.['User'].args!; const originalResult = { id: '5', @@ -338,7 +339,7 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const argsFn = transformedSubschemaConfig.merge.User.args; + const argsFn = transformedSubschemaConfig.merge?.['User'].args!; const originalResult = { id: '5', @@ -377,7 +378,7 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const argsFn = transformedSubschemaConfig.merge.User.args; + const argsFn = transformedSubschemaConfig.merge?.['User'].args!; const originalResult = { id: '5', @@ -416,7 +417,7 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const argsFn = transformedSubschemaConfig.merge.User.args; + const argsFn = transformedSubschemaConfig.merge?.['User'].args!; const originalResult = { id: '5', @@ -459,7 +460,7 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const argsFn = transformedSubschemaConfig.merge.User.args; + const argsFn = transformedSubschemaConfig.merge?.['User'].args!; const originalResult = { id: '5', @@ -501,7 +502,7 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const argsFn = transformedSubschemaConfig.merge.User.args; + const argsFn = transformedSubschemaConfig.merge?.['User'].args!; const originalResult = { id: '5', @@ -539,9 +540,9 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - expect(transformedSubschemaConfig.merge.User.selectionSet).toEqual(`{\n id\n}`); + expect(transformedSubschemaConfig.merge?.['User'].selectionSet).toEqual(`{\n id\n}`); - const argsFn = transformedSubschemaConfig.merge.User.args; + const argsFn = transformedSubschemaConfig.merge?.['User'].args!; const originalResult = { id: '5', @@ -578,7 +579,7 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const argsFn = transformedSubschemaConfig.merge.User.args; + const argsFn = transformedSubschemaConfig.merge?.['User'].args!; const originalResult = { id: '5', @@ -624,8 +625,8 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const keyFn = transformedSubschemaConfig.merge.User.key; - const argsFromKeysFn = transformedSubschemaConfig.merge.User.argsFromKeys; + const keyFn = transformedSubschemaConfig.merge?.['User'].key!; + const argsFromKeysFn = transformedSubschemaConfig.merge?.['User'].argsFromKeys!; const originalResult = { id: '5', @@ -670,8 +671,8 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const keyFn = transformedSubschemaConfig.merge.User.key; - const argsFromKeysFn = transformedSubschemaConfig.merge.User.argsFromKeys; + const keyFn = transformedSubschemaConfig.merge?.['User'].key!; + const argsFromKeysFn = transformedSubschemaConfig.merge?.['User'].argsFromKeys!; const originalResult = { __typename: 'User', @@ -721,8 +722,8 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const keyFn = transformedSubschemaConfig.merge.User.key; - const argsFromKeysFn = transformedSubschemaConfig.merge.User.argsFromKeys; + const keyFn = transformedSubschemaConfig.merge?.['User'].key!; + const argsFromKeysFn = transformedSubschemaConfig.merge?.['User'].argsFromKeys!; const originalResult = { __typename: 'User', @@ -768,8 +769,8 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const keyFn = transformedSubschemaConfig.merge.User.key; - const argsFromKeysFn = transformedSubschemaConfig.merge.User.argsFromKeys; + const keyFn = transformedSubschemaConfig.merge?.['User'].key!; + const argsFromKeysFn = transformedSubschemaConfig.merge?.['User'].argsFromKeys!; const originalResult = { id: '5', @@ -812,8 +813,8 @@ describe('type merging directives', () => { const transformedSubschemaConfig = stitchingDirectivesTransformer(subschemaConfig); - const keyFn = transformedSubschemaConfig.merge.User.key; - const argsFromKeysFn = transformedSubschemaConfig.merge.User.argsFromKeys; + const keyFn = transformedSubschemaConfig.merge?.['User'].key!; + const argsFromKeysFn = transformedSubschemaConfig.merge?.['User'].argsFromKeys!; const originalResult = { id: '5', diff --git a/packages/utils/package.json b/packages/utils/package.json index 911687c707e..52c2341c81c 100644 --- a/packages/utils/package.json +++ b/packages/utils/package.json @@ -11,8 +11,8 @@ "license": "MIT", "sideEffects": false, "main": "dist/index.js", - "module": "dist/index.mjs", - "exports": { + "module": "dist/index.mjs", + "exports": { ".": { "require": "./dist/index.js", "import": "./dist/index.mjs" @@ -35,8 +35,6 @@ "graphql-scalars": "1.10.0" }, "dependencies": { - "@ardatan/aggregate-error": "0.0.6", - "camel-case": "4.1.2", "tslib": "~2.3.0" }, "publishConfig": { diff --git a/packages/utils/src/AggregateError.ts b/packages/utils/src/AggregateError.ts new file mode 100644 index 00000000000..94af2ec65b4 --- /dev/null +++ b/packages/utils/src/AggregateError.ts @@ -0,0 +1,16 @@ +let AggregateErrorImpl = globalThis.AggregateError; + +if (typeof AggregateErrorImpl === 'undefined') { + class AggregateErrorClass extends Error implements AggregateError { + constructor(public errors: any[], message = '') { + super(message); + this.name = 'AggregateError'; + Error.captureStackTrace(this, AggregateErrorClass); + } + } + AggregateErrorImpl = function (errors: any[], message?: string) { + return new AggregateErrorClass(errors, message); + } as AggregateErrorConstructor; +} + +export { AggregateErrorImpl as AggregateError }; diff --git a/packages/utils/src/Interfaces.ts b/packages/utils/src/Interfaces.ts index bcd11634d6a..23190a3c41a 100644 --- a/packages/utils/src/Interfaces.ts +++ b/packages/utils/src/Interfaces.ts @@ -46,10 +46,10 @@ import { InputObjectTypeExtensionNode, InputObjectTypeDefinitionNode, GraphQLType, + Source, + DefinitionNode, } from 'graphql'; -import { SchemaVisitor } from './SchemaVisitor'; - // graphql-js < v15 backwards compatible ExecutionResult // See: https://github.com/graphql/graphql-js/pull/2490 @@ -237,9 +237,14 @@ export type IFieldResolver, TRetu info: GraphQLResolveInfo ) => TReturn; -export type ITypedef = string | DocumentNode | (() => Array); - -export type ITypeDefinitions = string | DocumentNode | Array; +export type TypeSource = + | string + | Source + | DocumentNode + | GraphQLSchema + | DefinitionNode + | Array + | (() => TypeSource); export type IObjectTypeResolver = { [key: string]: IFieldResolver | IFieldResolverOptions; @@ -304,21 +309,10 @@ export type IDefaultValueIteratorFn = (type: GraphQLInputType, value: any) => vo export type NextResolverFn = () => Promise; -export type DirectiveResolverFn = ( - next: NextResolverFn, - source: TSource, - args: { [argName: string]: any }, - context: TContext, - info: GraphQLResolveInfo -) => any; - -export interface IDirectiveResolvers { - [directiveName: string]: DirectiveResolverFn; -} - export interface Request { document: DocumentNode; variables: Record; + operationName?: string; extensions?: Record; } @@ -336,79 +330,6 @@ export type VisitableSchemaType = | GraphQLEnumType | GraphQLEnumValue; -export type VisitorSelector = ( - type: VisitableSchemaType, - methodName: string -) => Array; - -export enum VisitSchemaKind { - TYPE = 'VisitSchemaKind.TYPE', - SCALAR_TYPE = 'VisitSchemaKind.SCALAR_TYPE', - ENUM_TYPE = 'VisitSchemaKind.ENUM_TYPE', - COMPOSITE_TYPE = 'VisitSchemaKind.COMPOSITE_TYPE', - OBJECT_TYPE = 'VisitSchemaKind.OBJECT_TYPE', - INPUT_OBJECT_TYPE = 'VisitSchemaKind.INPUT_OBJECT_TYPE', - ABSTRACT_TYPE = 'VisitSchemaKind.ABSTRACT_TYPE', - UNION_TYPE = 'VisitSchemaKind.UNION_TYPE', - INTERFACE_TYPE = 'VisitSchemaKind.INTERFACE_TYPE', - ROOT_OBJECT = 'VisitSchemaKind.ROOT_OBJECT', - QUERY = 'VisitSchemaKind.QUERY', - MUTATION = 'VisitSchemaKind.MUTATION', - SUBSCRIPTION = 'VisitSchemaKind.SUBSCRIPTION', -} - -export interface SchemaVisitorMap { - [VisitSchemaKind.TYPE]?: NamedTypeVisitor; - [VisitSchemaKind.SCALAR_TYPE]?: ScalarTypeVisitor; - [VisitSchemaKind.ENUM_TYPE]?: EnumTypeVisitor; - [VisitSchemaKind.COMPOSITE_TYPE]?: CompositeTypeVisitor; - [VisitSchemaKind.OBJECT_TYPE]?: ObjectTypeVisitor; - [VisitSchemaKind.INPUT_OBJECT_TYPE]?: InputObjectTypeVisitor; - [VisitSchemaKind.ABSTRACT_TYPE]?: AbstractTypeVisitor; - [VisitSchemaKind.UNION_TYPE]?: UnionTypeVisitor; - [VisitSchemaKind.INTERFACE_TYPE]?: InterfaceTypeVisitor; - [VisitSchemaKind.ROOT_OBJECT]?: ObjectTypeVisitor; - [VisitSchemaKind.QUERY]?: ObjectTypeVisitor; - [VisitSchemaKind.MUTATION]?: ObjectTypeVisitor; - [VisitSchemaKind.SUBSCRIPTION]?: ObjectTypeVisitor; -} - -export type NamedTypeVisitor = (type: GraphQLNamedType, schema: GraphQLSchema) => GraphQLNamedType | null | undefined; - -export type ScalarTypeVisitor = ( - type: GraphQLScalarType, - schema: GraphQLSchema -) => GraphQLScalarType | null | undefined; - -export type EnumTypeVisitor = (type: GraphQLEnumType, schema: GraphQLSchema) => GraphQLEnumType | null | undefined; - -export type CompositeTypeVisitor = ( - type: GraphQLObjectType | GraphQLInterfaceType | GraphQLUnionType, - schema: GraphQLSchema -) => GraphQLObjectType | GraphQLInterfaceType | GraphQLUnionType | null | undefined; - -export type ObjectTypeVisitor = ( - type: GraphQLObjectType, - schema: GraphQLSchema -) => GraphQLObjectType | null | undefined; - -export type InputObjectTypeVisitor = ( - type: GraphQLInputObjectType, - schema: GraphQLSchema -) => GraphQLInputObjectType | null | undefined; - -export type AbstractTypeVisitor = ( - type: GraphQLInterfaceType | GraphQLUnionType, - schema: GraphQLSchema -) => GraphQLInterfaceType | GraphQLUnionType | null | undefined; - -export type UnionTypeVisitor = (type: GraphQLUnionType, schema: GraphQLSchema) => GraphQLUnionType | null | undefined; - -export type InterfaceTypeVisitor = ( - type: GraphQLInterfaceType, - schema: GraphQLSchema -) => GraphQLInterfaceType | null | undefined; - export enum MapperKind { TYPE = 'MapperKind.TYPE', SCALAR_TYPE = 'MapperKind.SCALAR_TYPE', diff --git a/packages/utils/src/SchemaDirectiveVisitor.ts b/packages/utils/src/SchemaDirectiveVisitor.ts deleted file mode 100644 index 2add8dd28d3..00000000000 --- a/packages/utils/src/SchemaDirectiveVisitor.ts +++ /dev/null @@ -1,324 +0,0 @@ -import { - GraphQLDirective, - GraphQLSchema, - DirectiveLocationEnum, - TypeSystemExtensionNode, - valueFromASTUntyped, -} from 'graphql'; - -import { VisitableSchemaType } from './Interfaces'; - -import { SchemaVisitor } from './SchemaVisitor'; -import { visitSchema } from './visitSchema'; -import { getArgumentValues } from './getArgumentValues'; -import { Maybe } from 'packages/graphql-tools/src'; - -// This class represents a reusable implementation of a @directive that may -// appear in a GraphQL schema written in Schema Definition Language. -// -// By overriding one or more visit{Object,Union,...} methods, a subclass -// registers interest in certain schema types, such as GraphQLObjectType, -// GraphQLUnionType, etc. When SchemaDirectiveVisitor.visitSchemaDirectives is -// called with a GraphQLSchema object and a map of visitor subclasses, the -// overridden methods of those subclasses allow the visitors to obtain -// references to any type objects that have @directives attached to them, -// enabling visitors to inspect or modify the schema as appropriate. -// -// For example, if a directive called @rest(url: "...") appears after a field -// definition, a SchemaDirectiveVisitor subclass could provide meaning to that -// directive by overriding the visitFieldDefinition method (which receives a -// GraphQLField parameter), and then the body of that visitor method could -// manipulate the field's resolver function to fetch data from a REST endpoint -// described by the url argument passed to the @rest directive: -// -// const typeDefs = ` -// type Query { -// people: [Person] @rest(url: "/api/v1/people") -// }`; -// -// const schema = makeExecutableSchema({ typeDefs }); -// -// SchemaDirectiveVisitor.visitSchemaDirectives(schema, { -// rest: class extends SchemaDirectiveVisitor { -// public visitFieldDefinition(field: GraphQLField) { -// const { url } = this.args; -// field.resolve = () => fetch(url); -// } -// } -// }); -// -// The subclass in this example is defined as an anonymous class expression, -// for brevity. A truly reusable SchemaDirectiveVisitor would most likely be -// defined in a library using a named class declaration, and then exported for -// consumption by other modules and packages. -// -// See below for a complete list of overridable visitor methods, their -// parameter types, and more details about the properties exposed by instances -// of the SchemaDirectiveVisitor class. - -export class SchemaDirectiveVisitor extends SchemaVisitor { - // The name of the directive this visitor is allowed to visit (that is, the - // identifier that appears after the @ character in the schema). Note that - // this property is per-instance rather than static because subclasses of - // SchemaDirectiveVisitor can be instantiated multiple times to visit - // directives of different names. In other words, SchemaDirectiveVisitor - // implementations are effectively anonymous, and it's up to the caller of - // SchemaDirectiveVisitor.visitSchemaDirectives to assign names to them. - public name: string; - - // A map from parameter names to argument values, as obtained from a - // specific occurrence of a @directive(arg1: value1, arg2: value2, ...) in - // the schema. Visitor methods may refer to this object via this.args. - public args: TArgs; - - // A reference to the type object that this visitor was created to visit. - public visitedType: VisitableSchemaType; - - // A shared object that will be available to all visitor instances via - // this.context. Callers of visitSchemaDirectives can provide their own - // object, or just use the default empty object. - public context: TContext; - - // Override this method to return a custom GraphQLDirective (or modify one - // already present in the schema) to enforce argument types, provide default - // argument values, or specify schema locations where this @directive may - // appear. By default, any declaration found in the schema will be returned. - public static getDirectiveDeclaration( - directiveName: string, - schema: GraphQLSchema - ): GraphQLDirective | null | undefined { - return schema.getDirective(directiveName); - } - - // Call SchemaDirectiveVisitor.visitSchemaDirectives to visit every - // @directive in the schema and create an appropriate SchemaDirectiveVisitor - // instance to visit the object decorated by the @directive. - public static visitSchemaDirectives( - schema: GraphQLSchema, - // The keys of this object correspond to directive names as they appear - // in the schema, and the values should be subclasses (not instances!) - // of the SchemaDirectiveVisitor class. This distinction is important - // because a new SchemaDirectiveVisitor instance will be created each - // time a matching directive is found in the schema AST, with arguments - // and other metadata specific to that occurrence. To help prevent the - // mistake of passing instances, the SchemaDirectiveVisitor constructor - // method is marked as protected. - directiveVisitors: Record, - // Optional context object that will be available to all visitor instances - // via this.context. Defaults to an empty null-prototype object. - context: Record = Object.create(null), - // The visitSchemaDirectives method returns a map from directive names to - // lists of SchemaDirectiveVisitors created while visiting the schema. - pathToDirectivesInExtensions = ['directives'] - ): Record> { - // If the schema declares any directives for public consumption, record - // them here so that we can properly coerce arguments when/if we encounter - // an occurrence of the directive while walking the schema below. - const declaredDirectives = this.getDeclaredDirectives(schema, directiveVisitors); - - // Map from directive names to lists of SchemaDirectiveVisitor instances - // created while visiting the schema. - const createdVisitors: Record> = Object.keys(directiveVisitors).reduce( - (prev, item) => ({ - ...prev, - [item]: [], - }), - {} - ); - - const directiveVisitorMap: Record = Object.entries(directiveVisitors).reduce( - (prev, [key, value]) => ({ - ...prev, - [key]: value, - }), - {} - ); - - function visitorSelector(type: VisitableSchemaType, methodName: string): Array { - const directivesInExtensions = pathToDirectivesInExtensions.reduce( - (acc, pathSegment) => (acc == null ? acc : acc[pathSegment]), - type?.extensions - ); - - const directives: Record> = Object.create(null); - - if (directivesInExtensions != null) { - Object.entries(directivesInExtensions).forEach(([directiveName, directiveValue]) => { - if (!directives[directiveName]) { - directives[directiveName] = [directiveValue]; - } else { - directives[directiveName].push([directiveValue]); - } - }); - } else { - let directiveNodes = type?.astNode?.directives ?? []; - - const extensionASTNodes: Maybe> = ( - type as { - extensionASTNodes?: Array; - } - ).extensionASTNodes; - - if (extensionASTNodes != null) { - extensionASTNodes.forEach(extensionASTNode => { - if (extensionASTNode.directives != null) { - directiveNodes = directiveNodes.concat(extensionASTNode.directives); - } - }); - } - - directiveNodes.forEach(directiveNode => { - const directiveName = directiveNode.name.value; - - const decl = declaredDirectives[directiveName]; - let args: Record; - - if (decl != null) { - // If this directive was explicitly declared, use the declared - // argument types (and any default values) to check, coerce, and/or - // supply default values for the given arguments. - args = getArgumentValues(decl, directiveNode); - } else { - // If this directive was not explicitly declared, just convert the - // argument nodes to their corresponding JavaScript values. - args = Object.create(null); - if (directiveNode.arguments != null) { - directiveNode.arguments.forEach(arg => { - args[arg.name.value] = valueFromASTUntyped(arg.value); - }); - } - } - - if (!directives[directiveName]) { - directives[directiveName] = [args]; - } else { - directives[directiveName].push(args); - } - }); - } - - const visitors: Array = []; - - Object.entries(directives).forEach(([directiveName, directiveValues]) => { - if (!(directiveName in directiveVisitorMap)) { - return; - } - - const VisitorClass = directiveVisitorMap[directiveName]; - - // Avoid creating visitor objects if visitorClass does not override - // the visitor method named by methodName. - if (!VisitorClass.implementsVisitorMethod(methodName)) { - return; - } - - directiveValues.forEach(directiveValue => { - // As foretold in comments near the top of the visitSchemaDirectives - // method, this is where instances of the SchemaDirectiveVisitor class - // get created and assigned names. While subclasses could override the - // constructor method, the constructor is marked as protected, so - // these are the only arguments that will ever be passed. - visitors.push( - new VisitorClass({ - name: directiveName, - args: directiveValue, - visitedType: type, - schema, - context, - }) - ); - }); - }); - - if (visitors.length > 0) { - visitors.forEach(visitor => { - createdVisitors[visitor.name].push(visitor); - }); - } - - return visitors; - } - - visitSchema(schema, visitorSelector); - - return createdVisitors; - } - - protected static getDeclaredDirectives( - schema: GraphQLSchema, - directiveVisitors: Record - ): Record { - const declaredDirectives: Record = schema.getDirectives().reduce( - (prev, curr) => ({ - ...prev, - [curr.name]: curr, - }), - {} - ); - // If the visitor subclass overrides getDirectiveDeclaration, and it - // returns a non-null GraphQLDirective, use that instead of any directive - // declared in the schema itself. Reasoning: if a SchemaDirectiveVisitor - // goes to the trouble of implementing getDirectiveDeclaration, it should - // be able to rely on that implementation. - Object.entries(directiveVisitors).forEach(([directiveName, visitorClass]) => { - const decl = visitorClass.getDirectiveDeclaration(directiveName, schema); - if (decl != null) { - declaredDirectives[directiveName] = decl; - } - }); - - Object.entries(declaredDirectives).forEach(([name, decl]) => { - if (!(name in directiveVisitors)) { - // SchemaDirectiveVisitors.visitSchemaDirectives might be called - // multiple times with partial directiveVisitors maps, so it's not - // necessarily an error for directiveVisitors to be missing an - // implementation of a directive that was declared in the schema. - return; - } - const visitorClass = directiveVisitors[name]; - - decl.locations.forEach(loc => { - const visitorMethodName = directiveLocationToVisitorMethodName(loc); - if ( - SchemaVisitor.implementsVisitorMethod(visitorMethodName) && - !visitorClass.implementsVisitorMethod(visitorMethodName) - ) { - // While visitor subclasses may implement extra visitor methods, - // it's definitely a mistake if the GraphQLDirective declares itself - // applicable to certain schema locations, and the visitor subclass - // does not implement all the corresponding methods. - throw new Error(`SchemaDirectiveVisitor for @${name} must implement ${visitorMethodName} method`); - } - }); - }); - - return declaredDirectives; - } - - // Mark the constructor protected to enforce passing SchemaDirectiveVisitor - // subclasses (not instances) to visitSchemaDirectives. - protected constructor(config: { - name: string; - args: TArgs; - visitedType: VisitableSchemaType; - schema: GraphQLSchema; - context: TContext; - }) { - super(); - this.name = config.name; - this.args = config.args; - this.visitedType = config.visitedType; - this.schema = config.schema; - this.context = config.context; - } -} - -// Convert a string like "FIELD_DEFINITION" to "visitFieldDefinition". -function directiveLocationToVisitorMethodName(loc: DirectiveLocationEnum) { - return ( - 'visit' + - loc.replace(/([^_]*)_?/g, (_wholeMatch, part: string) => part.charAt(0).toUpperCase() + part.slice(1).toLowerCase()) - ); -} - -export type SchemaDirectiveVisitorClass = typeof SchemaDirectiveVisitor; diff --git a/packages/utils/src/SchemaVisitor.ts b/packages/utils/src/SchemaVisitor.ts deleted file mode 100644 index 7f49ba0aa82..00000000000 --- a/packages/utils/src/SchemaVisitor.ts +++ /dev/null @@ -1,119 +0,0 @@ -import { - GraphQLArgument, - GraphQLEnumType, - GraphQLEnumValue, - GraphQLField, - GraphQLInputField, - GraphQLInputObjectType, - GraphQLInterfaceType, - GraphQLObjectType, - GraphQLScalarType, - GraphQLSchema, - GraphQLUnionType, -} from 'graphql'; - -// Abstract base class of any visitor implementation, defining the available -// visitor methods along with their parameter types, and providing a static -// helper function for determining whether a subclass implements a given -// visitor method, as opposed to inheriting one of the stubs defined here. -export abstract class SchemaVisitor { - // All SchemaVisitor instances are created while visiting a specific - // GraphQLSchema object, so this property holds a reference to that object, - // in case a visitor method needs to refer to this.schema. - public schema!: GraphQLSchema; - - // Determine if this SchemaVisitor (sub)class implements a particular - // visitor method. - public static implementsVisitorMethod(methodName: string): boolean { - if (!methodName.startsWith('visit')) { - return false; - } - - const method = this.prototype[methodName]; - if (typeof method !== 'function') { - return false; - } - - if (this.name === 'SchemaVisitor') { - // The SchemaVisitor class implements every visitor method. - return true; - } - - const stub = SchemaVisitor.prototype[methodName]; - if (method === stub) { - // If this.prototype[methodName] was just inherited from SchemaVisitor, - // then this class does not really implement the method. - return false; - } - - return true; - } - - // Concrete subclasses of SchemaVisitor should override one or more of these - // visitor methods, in order to express their interest in handling certain - // schema types/locations. Each method may return null to remove the given - // type from the schema, a non-null value of the same type to update the - // type in the schema, or nothing to leave the type as it was. - - // eslint-disable-next-line @typescript-eslint/no-empty-function - public visitSchema(_schema: GraphQLSchema): void {} - - public visitScalar( - _scalar: GraphQLScalarType - // eslint-disable-next-line @typescript-eslint/no-empty-function - ): GraphQLScalarType | void | null {} - - public visitObject( - _object: GraphQLObjectType - // eslint-disable-next-line @typescript-eslint/no-empty-function - ): GraphQLObjectType | void | null {} - - public visitFieldDefinition( - _field: GraphQLField, - _details: { - objectType: GraphQLObjectType | GraphQLInterfaceType; - } - // eslint-disable-next-line @typescript-eslint/no-empty-function - ): GraphQLField | void | null {} - - public visitArgumentDefinition( - _argument: GraphQLArgument, - _details: { - field: GraphQLField; - objectType: GraphQLObjectType | GraphQLInterfaceType; - } - // eslint-disable-next-line @typescript-eslint/no-empty-function - ): GraphQLArgument | void | null {} - - public visitInterface( - _iface: GraphQLInterfaceType - // eslint-disable-next-line @typescript-eslint/no-empty-function - ): GraphQLInterfaceType | void | null {} - - // eslint-disable-next-line @typescript-eslint/no-empty-function - public visitUnion(_union: GraphQLUnionType): GraphQLUnionType | void | null {} - - // eslint-disable-next-line @typescript-eslint/no-empty-function - public visitEnum(_type: GraphQLEnumType): GraphQLEnumType | void | null {} - - public visitEnumValue( - _value: GraphQLEnumValue, - _details: { - enumType: GraphQLEnumType; - } - // eslint-disable-next-line @typescript-eslint/no-empty-function - ): GraphQLEnumValue | void | null {} - - public visitInputObject( - _object: GraphQLInputObjectType - // eslint-disable-next-line @typescript-eslint/no-empty-function - ): GraphQLInputObjectType | void | null {} - - public visitInputFieldDefinition( - _field: GraphQLInputField, - _details: { - objectType: GraphQLInputObjectType; - } - // eslint-disable-next-line @typescript-eslint/no-empty-function - ): GraphQLInputField | void | null {} -} diff --git a/packages/utils/src/addTypes.ts b/packages/utils/src/addTypes.ts index 849b5f52681..ae4859c3c2d 100644 --- a/packages/utils/src/addTypes.ts +++ b/packages/utils/src/addTypes.ts @@ -33,6 +33,7 @@ import { GraphQLDirective, isNamedType, isDirective, + isObjectType, } from 'graphql'; import { rewireTypes } from './rewire'; @@ -50,35 +51,44 @@ export function addTypes( const config = schema.toConfig(); - const originalTypeMap = {}; - config.types.forEach(type => { + const originalTypeMap: Record = {}; + for (const type of config.types) { originalTypeMap[type.name] = type; - }); + } - const originalDirectiveMap = {}; - config.directives.forEach(directive => { + const originalDirectiveMap: Record = {}; + for (const directive of config.directives) { originalDirectiveMap[directive.name] = directive; - }); + } - newTypesOrDirectives.forEach(newTypeOrDirective => { + for (const newTypeOrDirective of newTypesOrDirectives) { if (isNamedType(newTypeOrDirective)) { originalTypeMap[newTypeOrDirective.name] = newTypeOrDirective; } else if (isDirective(newTypeOrDirective)) { originalDirectiveMap[newTypeOrDirective.name] = newTypeOrDirective; } - }); + } - const { typeMap, directives } = rewireTypes( - originalTypeMap, - Object.keys(originalDirectiveMap).map(directiveName => originalDirectiveMap[directiveName]) - ); + const { typeMap, directives } = rewireTypes(originalTypeMap, Object.values(originalDirectiveMap)); return new GraphQLSchema({ ...config, - query: queryTypeName ? (typeMap[queryTypeName] as GraphQLObjectType) : undefined, - mutation: mutationTypeName ? (typeMap[mutationTypeName] as GraphQLObjectType) : undefined, - subscription: subscriptionTypeName != null ? (typeMap[subscriptionTypeName] as GraphQLObjectType) : undefined, - types: Object.keys(typeMap).map(typeName => typeMap[typeName]), + query: getObjectTypeFromTypeMap(typeMap, queryTypeName), + mutation: getObjectTypeFromTypeMap(typeMap, mutationTypeName), + subscription: getObjectTypeFromTypeMap(typeMap, subscriptionTypeName), + types: Object.values(typeMap), directives, }); } + +export function getObjectTypeFromTypeMap( + typeMap: Record, + typeName?: string +): GraphQLObjectType | undefined { + if (typeName) { + const maybeObjectType = typeMap[typeName]; + if (isObjectType(maybeObjectType)) { + return maybeObjectType; + } + } +} diff --git a/packages/utils/src/astFromValueUntyped.ts b/packages/utils/src/astFromValueUntyped.ts index 2719adb05ab..a7ed8213702 100644 --- a/packages/utils/src/astFromValueUntyped.ts +++ b/packages/utils/src/astFromValueUntyped.ts @@ -30,18 +30,19 @@ export function astFromValueUntyped(value: any): ValueNode | null { // the value is not an array, convert the value using the list's item type. if (Array.isArray(value)) { const valuesNodes: Array = []; - value.forEach(item => { + for (const item of value) { const itemNode = astFromValueUntyped(item); if (itemNode != null) { valuesNodes.push(itemNode); } - }); + } return { kind: Kind.LIST, values: valuesNodes }; } if (typeof value === 'object') { const fieldNodes: Array = []; - Object.entries(value).forEach(([fieldName, fieldValue]) => { + for (const fieldName in value) { + const fieldValue = value[fieldName]; const ast = astFromValueUntyped(fieldValue); if (ast) { fieldNodes.push({ @@ -50,7 +51,7 @@ export function astFromValueUntyped(value: any): ValueNode | null { value: ast, }); } - }); + } return { kind: Kind.OBJECT, fields: fieldNodes }; } diff --git a/packages/utils/src/build-operation-for-field.ts b/packages/utils/src/build-operation-for-field.ts index 15e17e24fc1..87652876f51 100644 --- a/packages/utils/src/build-operation-for-field.ts +++ b/packages/utils/src/build-operation-for-field.ts @@ -27,7 +27,6 @@ import { isEnumType, Kind, } from 'graphql'; -import { camelCase } from 'camel-case'; let operationVariables: VariableDefinitionNode[] = []; let fieldTypeMap = new Map(); @@ -44,10 +43,6 @@ function resetFieldMap() { fieldTypeMap = new Map(); } -function buildOperationName(name: string) { - return camelCase(name); -} - export type Skip = string[]; export type Force = string[]; export type Ignore = string[]; @@ -131,15 +126,15 @@ function buildOperationAndCollectVariables({ }; const type = typeMap[kind]; const field = type.getFields()[fieldName]; - const operationName = buildOperationName(`${fieldName}_${kind}`); + const operationName = `${fieldName}_${kind}`; if (field.args) { - field.args.forEach(arg => { + for (const arg of field.args) { const argName = arg.name; if (!argNames || argNames.includes(argName)) { addOperationVariable(resolveVariable(arg, argName)); } - }); + } } return { @@ -394,7 +389,7 @@ function resolveVariable(arg: GraphQLArgument, name?: string): VariableDefinitio } function getArgumentName(name: string, path: string[]): string { - return camelCase([...path, name].join('_')); + return [...path, name].join('_'); } function resolveField({ diff --git a/packages/utils/src/debug-log.ts b/packages/utils/src/debug-log.ts deleted file mode 100644 index 292ec974cee..00000000000 --- a/packages/utils/src/debug-log.ts +++ /dev/null @@ -1,6 +0,0 @@ -export function debugLog(...args: any[]): void { - if (process && process.env && process.env['DEBUG'] && !process.env['GQL_tools_NODEBUG']) { - // tslint:disable-next-line: no-console - console.log(...args); - } -} diff --git a/packages/utils/src/executor.ts b/packages/utils/src/executor.ts index f91a36d6840..eb984db581d 100644 --- a/packages/utils/src/executor.ts +++ b/packages/utils/src/executor.ts @@ -1,39 +1,54 @@ import { DocumentNode, GraphQLResolveInfo } from 'graphql'; import { ExecutionResult } from './Interfaces'; -export interface ExecutionParams = Record, TContext = any> { +export interface ExecutionParams< + TArgs extends Record = Record, + TContext = any, + TRootValue = any, + TExtensions = Record +> { document: DocumentNode; variables?: TArgs; - extensions?: Record; + extensions?: TExtensions; context?: TContext; info?: GraphQLResolveInfo; + rootValue?: TRootValue; + operationName?: string; } -export type AsyncExecutor> = < - TReturn = Record, +export type AsyncExecutor, TBaseExtensions = Record> = < + TReturn = any, TArgs = Record, - TContext extends TBaseContext = TBaseContext + TContext extends TBaseContext = TBaseContext, + TRoot = any, + TExtensions extends TBaseExtensions = TBaseExtensions >( - params: ExecutionParams + params: ExecutionParams ) => Promise>; -export type SyncExecutor> = < - TReturn = Record, +export type SyncExecutor, TBaseExtensions = Record> = < + TReturn = any, TArgs = Record, - TContext extends TBaseContext = TBaseContext + TContext extends TBaseContext = TBaseContext, + TRoot = any, + TExtensions extends TBaseExtensions = TBaseExtensions >( - params: ExecutionParams + params: ExecutionParams ) => ExecutionResult; -export type Executor> = < - TReturn = Record, +export type Executor, TBaseExtensions = Record> = < + TReturn = any, TArgs = Record, - TContext extends TBaseContext = TBaseContext + TContext extends TBaseContext = TBaseContext, + TRoot = any, + TExtensions extends TBaseExtensions = TBaseExtensions >( - params: ExecutionParams + params: ExecutionParams ) => ExecutionResult | Promise>; -export type Subscriber> = < - TReturn = Record, +export type Subscriber, TBaseExtensions = Record> = < + TReturn = any, TArgs = Record, - TContext extends TBaseContext = TBaseContext + TContext extends TBaseContext = TBaseContext, + TRoot = any, + TExtensions extends TBaseExtensions = TBaseExtensions >( - params: ExecutionParams + params: ExecutionParams ) => Promise> | ExecutionResult>; diff --git a/packages/utils/src/fields.ts b/packages/utils/src/fields.ts index 15a16eaca10..e33180753e6 100644 --- a/packages/utils/src/fields.ts +++ b/packages/utils/src/fields.ts @@ -24,12 +24,13 @@ export function appendObjectFields( const originalFieldConfigMap = config.fields; const newFieldConfigMap = {}; - Object.keys(originalFieldConfigMap).forEach(fieldName => { + + for (const fieldName in originalFieldConfigMap) { newFieldConfigMap[fieldName] = originalFieldConfigMap[fieldName]; - }); - Object.keys(additionalFields).forEach(fieldName => { + } + for (const fieldName in additionalFields) { newFieldConfigMap[fieldName] = additionalFields[fieldName]; - }); + } return correctASTNodes( new GraphQLObjectType({ @@ -55,14 +56,14 @@ export function removeObjectFields( const originalFieldConfigMap = config.fields; const newFieldConfigMap = {}; - Object.keys(originalFieldConfigMap).forEach(fieldName => { + for (const fieldName in originalFieldConfigMap) { const originalFieldConfig = originalFieldConfigMap[fieldName]; if (testFn(fieldName, originalFieldConfig)) { removedFields[fieldName] = originalFieldConfig; } else { newFieldConfigMap[fieldName] = originalFieldConfig; } - }); + } return correctASTNodes( new GraphQLObjectType({ @@ -89,12 +90,12 @@ export function selectObjectFields( const config = type.toConfig(); const originalFieldConfigMap = config.fields; - Object.keys(originalFieldConfigMap).forEach(fieldName => { + for (const fieldName in originalFieldConfigMap) { const originalFieldConfig = originalFieldConfigMap[fieldName]; if (testFn(fieldName, originalFieldConfig)) { selectedFields[fieldName] = originalFieldConfig; } - }); + } } return undefined; @@ -118,19 +119,19 @@ export function modifyObjectFields( const originalFieldConfigMap = config.fields; const newFieldConfigMap = {}; - Object.keys(originalFieldConfigMap).forEach(fieldName => { + for (const fieldName in originalFieldConfigMap) { const originalFieldConfig = originalFieldConfigMap[fieldName]; if (testFn(fieldName, originalFieldConfig)) { removedFields[fieldName] = originalFieldConfig; } else { newFieldConfigMap[fieldName] = originalFieldConfig; } - }); + } - Object.keys(newFields).forEach(fieldName => { + for (const fieldName in newFields) { const fieldConfig = newFields[fieldName]; newFieldConfigMap[fieldName] = fieldConfig; - }); + } return correctASTNodes( new GraphQLObjectType({ diff --git a/packages/utils/src/filterSchema.ts b/packages/utils/src/filterSchema.ts index 5648d5ee15a..74e04b06232 100644 --- a/packages/utils/src/filterSchema.ts +++ b/packages/utils/src/filterSchema.ts @@ -81,17 +81,18 @@ function filterRootFields( ): GraphQLObjectType { if (rootFieldFilter || argumentFilter) { const config = type.toConfig(); - Object.entries(config.fields).forEach(([fieldName, field]) => { + for (const fieldName in config.fields) { + const field = config.fields[fieldName]; if (rootFieldFilter && !rootFieldFilter(operation, fieldName, config.fields[fieldName])) { delete config.fields[fieldName]; } else if (argumentFilter && field.args) { - for (const argName of Object.keys(field.args)) { + for (const argName in field.args) { if (!argumentFilter(operation, fieldName, argName, field.args[argName])) { delete field.args[argName]; } } } - }); + } return new GraphQLObjectType(config); } return type; @@ -105,17 +106,18 @@ function filterElementFields( ): ElementType | undefined { if (fieldFilter || argumentFilter) { const config = type.toConfig(); - Object.entries(config.fields).forEach(([fieldName, field]) => { + for (const fieldName in config.fields) { + const field = config.fields[fieldName]; if (fieldFilter && !fieldFilter(type.name, fieldName, config.fields[fieldName])) { delete config.fields[fieldName]; } else if (argumentFilter && 'args' in field) { - for (const argName of Object.keys(field.args)) { + for (const argName in field.args) { if (!argumentFilter(type.name, fieldName, argName, field.args[argName])) { delete field.args[argName]; } } } - }); + } return new ElementConstructor(config); } } diff --git a/packages/utils/src/fix-windows-path.ts b/packages/utils/src/fix-windows-path.ts deleted file mode 100644 index 746bed0a8d0..00000000000 --- a/packages/utils/src/fix-windows-path.ts +++ /dev/null @@ -1 +0,0 @@ -export const fixWindowsPath = (path: string) => path.replace(/\\/g, '/'); diff --git a/packages/utils/src/flatten-array.ts b/packages/utils/src/flatten-array.ts deleted file mode 100644 index 094f04f4918..00000000000 --- a/packages/utils/src/flatten-array.ts +++ /dev/null @@ -1,2 +0,0 @@ -export const flattenArray = (arr: any): any[] => - arr.reduce((acc: any, next: any) => acc.concat(Array.isArray(next) ? flattenArray(next) : next), []); diff --git a/packages/utils/src/forEachDefaultValue.ts b/packages/utils/src/forEachDefaultValue.ts index 4d15403c186..4ff70bc5a4c 100644 --- a/packages/utils/src/forEachDefaultValue.ts +++ b/packages/utils/src/forEachDefaultValue.ts @@ -4,26 +4,26 @@ import { IDefaultValueIteratorFn } from './Interfaces'; export function forEachDefaultValue(schema: GraphQLSchema, fn: IDefaultValueIteratorFn): void { const typeMap = schema.getTypeMap(); - Object.keys(typeMap).forEach(typeName => { + for (const typeName in typeMap) { const type = typeMap[typeName]; if (!getNamedType(type).name.startsWith('__')) { if (isObjectType(type)) { const fields = type.getFields(); - Object.keys(fields).forEach(fieldName => { + for (const fieldName in fields) { const field = fields[fieldName]; - field.args.forEach(arg => { + for (const arg of field.args) { arg.defaultValue = fn(arg.type, arg.defaultValue); - }); - }); + } + } } else if (isInputObjectType(type)) { const fields = type.getFields(); - Object.keys(fields).forEach(fieldName => { + for (const fieldName in fields) { const field = fields[fieldName]; field.defaultValue = fn(field.type, field.defaultValue); - }); + } } } - }); + } } diff --git a/packages/utils/src/forEachField.ts b/packages/utils/src/forEachField.ts index f6795c8d2c5..a59a9a969cb 100644 --- a/packages/utils/src/forEachField.ts +++ b/packages/utils/src/forEachField.ts @@ -4,16 +4,16 @@ import { IFieldIteratorFn } from './Interfaces'; export function forEachField(schema: GraphQLSchema, fn: IFieldIteratorFn): void { const typeMap = schema.getTypeMap(); - Object.keys(typeMap).forEach(typeName => { + for (const typeName in typeMap) { const type = typeMap[typeName]; // TODO: maybe have an option to include these? if (!getNamedType(type).name.startsWith('__') && isObjectType(type)) { const fields = type.getFields(); - Object.keys(fields).forEach(fieldName => { + for (const fieldName in fields) { const field = fields[fieldName]; fn(field, typeName, fieldName); - }); + } } - }); + } } diff --git a/packages/utils/src/get-directives.ts b/packages/utils/src/get-directives.ts index 24668414f05..d0afe8fb287 100644 --- a/packages/utils/src/get-directives.ts +++ b/packages/utils/src/get-directives.ts @@ -23,7 +23,7 @@ import { GraphQLEnumValueConfig, EnumValueDefinitionNode, } from 'graphql'; -import { Maybe } from 'packages/graphql-tools/src'; +import { Maybe } from '@graphql-tools/utils'; import { getArgumentValues } from './getArgumentValues'; @@ -96,9 +96,9 @@ export function getDirectives( const result: DirectiveUseMap = {}; - astNodes.forEach(astNode => { + for (const astNode of astNodes) { if (astNode.directives) { - astNode.directives.forEach(directiveNode => { + for (const directiveNode of astNode.directives) { const schemaDirective = schemaDirectiveMap[directiveNode.name.value]; if (schemaDirective) { if (schemaDirective.isRepeatable) { @@ -108,9 +108,9 @@ export function getDirectives( result[schemaDirective.name] = getArgumentValues(schemaDirective, directiveNode); } } - }); + } } - }); + } return result; } diff --git a/packages/utils/src/getArgumentValues.ts b/packages/utils/src/getArgumentValues.ts index f3e64b6afd4..1ad1003b6e6 100644 --- a/packages/utils/src/getArgumentValues.ts +++ b/packages/utils/src/getArgumentValues.ts @@ -11,7 +11,7 @@ import { ArgumentNode, } from 'graphql'; -import { inspect } from './inspect'; +import { inspect } from 'util'; /** * Prepares an object map of argument values given a list of argument diff --git a/packages/utils/src/getResolversFromSchema.ts b/packages/utils/src/getResolversFromSchema.ts index 2f0a1746e5d..deeb1814836 100644 --- a/packages/utils/src/getResolversFromSchema.ts +++ b/packages/utils/src/getResolversFromSchema.ts @@ -16,7 +16,7 @@ export function getResolversFromSchema(schema: GraphQLSchema): IResolvers { const typeMap = schema.getTypeMap(); - Object.keys(typeMap).forEach(typeName => { + for (const typeName in typeMap) { if (!typeName.startsWith('__')) { const type = typeMap[typeName]; @@ -30,9 +30,9 @@ export function getResolversFromSchema(schema: GraphQLSchema): IResolvers { resolvers[typeName] = {}; const values = type.getValues(); - values.forEach(value => { + for (const value of values) { resolvers[typeName][value.name] = value.value; - }); + } } else if (isInterfaceType(type)) { if (type.resolveType != null) { resolvers[typeName] = { @@ -53,20 +53,24 @@ export function getResolversFromSchema(schema: GraphQLSchema): IResolvers { } const fields = type.getFields(); - Object.keys(fields).forEach(fieldName => { + for (const fieldName in fields) { const field = fields[fieldName]; if (field.subscribe != null) { resolvers[typeName][fieldName] = resolvers[typeName][fieldName] || {}; resolvers[typeName][fieldName].subscribe = field.subscribe; } - if (field.resolve != null && field.resolve?.name !== 'defaultFieldResolver' && field.resolve?.name !== 'defaultMergedResolver') { + if ( + field.resolve != null && + field.resolve?.name !== 'defaultFieldResolver' && + field.resolve?.name !== 'defaultMergedResolver' + ) { resolvers[typeName][fieldName] = resolvers[typeName][fieldName] || {}; resolvers[typeName][fieldName].resolve = field.resolve; } - }); + } } } - }); + } return resolvers; } diff --git a/packages/utils/src/heal.ts b/packages/utils/src/heal.ts index 73a00cbf224..f0dd92fc8b3 100644 --- a/packages/utils/src/heal.ts +++ b/packages/utils/src/heal.ts @@ -66,14 +66,15 @@ export function healTypes( // schema.getTypeMap() have changed, the keys of the type map need to // be updated accordingly. - Object.entries(originalTypeMap).forEach(([typeName, namedType]) => { + for (const typeName in originalTypeMap) { + const namedType = originalTypeMap[typeName]; if (namedType == null || typeName.startsWith('__')) { - return; + continue; } const actualName = namedType.name; if (actualName.startsWith('__')) { - return; + continue; } if (actualName in actualNamedTypeMap) { @@ -85,31 +86,33 @@ export function healTypes( // Note: we are deliberately leaving namedType in the schema by its // original name (which might be different from actualName), so that // references by that name can be healed. - }); + } // Now add back every named type by its actual name. - Object.entries(actualNamedTypeMap).forEach(([typeName, namedType]) => { + for (const typeName in actualNamedTypeMap) { + const namedType = actualNamedTypeMap[typeName]; originalTypeMap[typeName] = namedType; - }); + } // Directive declaration argument types can refer to named types. - directives.forEach((decl: GraphQLDirective) => { + for (const decl of directives) { decl.args = decl.args.filter(arg => { arg.type = healType(arg.type) as GraphQLInputType; return arg.type !== null; }); - }); + } - Object.entries(originalTypeMap).forEach(([typeName, namedType]) => { + for (const typeName in originalTypeMap) { + const namedType = originalTypeMap[typeName]; // Heal all named types, except for dangling references, kept only to redirect. if (!typeName.startsWith('__') && typeName in actualNamedTypeMap) { if (namedType != null) { healNamedType(namedType); } } - }); + } - for (const typeName of Object.keys(originalTypeMap)) { + for (const typeName in originalTypeMap) { if (!typeName.startsWith('__') && !(typeName in actualNamedTypeMap)) { delete originalTypeMap[typeName]; } diff --git a/packages/utils/src/index.ts b/packages/utils/src/index.ts index 531c3a96ea9..5fc559f2755 100644 --- a/packages/utils/src/index.ts +++ b/packages/utils/src/index.ts @@ -1,8 +1,5 @@ export * from './loaders'; export * from './helpers'; -export * from './debug-log'; -export * from './fix-windows-path'; -export * from './flatten-array'; export * from './get-directives'; export * from './get-fields-with-directives'; export * from './get-implementing-types'; @@ -19,9 +16,6 @@ export * from './types'; export * from './filterSchema'; export * from './clone'; export * from './heal'; -export * from './SchemaVisitor'; -export * from './SchemaDirectiveVisitor'; -export * from './visitSchema'; export * from './getResolversFromSchema'; export * from './forEachField'; export * from './forEachDefaultValue'; @@ -42,7 +36,6 @@ export * from './mapAsyncIterator'; export * from './updateArgument'; export * from './implementsAbstractType'; export * from './errors'; -export * from './toConfig'; export * from './observableToAsyncIterable'; export * from './visitResult'; export * from './getArgumentValues'; @@ -52,3 +45,4 @@ export * from './isDocumentNode'; export * from './astFromValueUntyped'; export * from './executor'; export * from './withCancel'; +export * from './AggregateError'; diff --git a/packages/utils/src/inspect.ts b/packages/utils/src/inspect.ts deleted file mode 100644 index 9f2f474e324..00000000000 --- a/packages/utils/src/inspect.ts +++ /dev/null @@ -1,113 +0,0 @@ -const MAX_ARRAY_LENGTH = 10; -const MAX_RECURSIVE_DEPTH = 2; - -/** - * Used to print values in error messages. - */ -export function inspect(value: any): string { - return formatValue(value, []); -} - -function formatValue(value: any, seenValues: Array): string { - switch (typeof value) { - case 'string': - return JSON.stringify(value); - case 'function': - return value.name ? `[function ${(value as (...args: any[]) => any).name}]` : '[function]'; - case 'object': - if (value === null) { - return 'null'; - } - return formatObjectValue(value, seenValues); - default: - return String(value); - } -} - -function formatObjectValue(value: any, previouslySeenValues: Array): string { - if (previouslySeenValues.indexOf(value) !== -1) { - return '[Circular]'; - } - - const seenValues = [...previouslySeenValues, value]; - const customInspectFn = getCustomFn(value); - - if (customInspectFn !== undefined) { - const customValue = customInspectFn.call(value); - - // check for infinite recursion - if (customValue !== value) { - return typeof customValue === 'string' ? customValue : formatValue(customValue, seenValues); - } - } else if (Array.isArray(value)) { - return formatArray(value, seenValues); - } - - return formatObject(value, seenValues); -} - -function formatObject(object: any, seenValues: Array) { - const keys = Object.keys(object); - if (keys.length === 0) { - return '{}'; - } - - if (seenValues.length > MAX_RECURSIVE_DEPTH) { - return '[' + getObjectTag(object) + ']'; - } - - const properties = keys.map(key => { - const value = formatValue(object[key], seenValues); - return key + ': ' + value; - }); - - return '{ ' + properties.join(', ') + ' }'; -} - -function formatArray(array: Array, seenValues: Array): string { - if (array.length === 0) { - return '[]'; - } - - if (seenValues.length > MAX_RECURSIVE_DEPTH) { - return '[Array]'; - } - - const len = Math.min(MAX_ARRAY_LENGTH, array.length); - const remaining = array.length - len; - const items = []; - - for (let i = 0; i < len; ++i) { - items.push(formatValue(array[i], seenValues)); - } - - if (remaining === 1) { - items.push('... 1 more item'); - } else if (remaining > 1) { - items.push(`... ${remaining.toString(10)} more items`); - } - - return '[' + items.join(', ') + ']'; -} - -function getCustomFn(obj: any) { - if (typeof obj.inspect === 'function') { - return obj.inspect; - } -} - -function getObjectTag(obj: any): string { - const tag = Object.prototype.toString - .call(obj) - .replace(/^\[object /, '') - .replace(/]$/, ''); - - if (tag === 'Object' && typeof obj.constructor === 'function') { - const name = obj.constructor.name; - if (typeof name === 'string' && name !== '') { - return name; - } - } - - return tag; -} diff --git a/packages/utils/src/mapSchema.ts b/packages/utils/src/mapSchema.ts index a561b6890f3..d80b8e1284b 100644 --- a/packages/utils/src/mapSchema.ts +++ b/packages/utils/src/mapSchema.ts @@ -29,6 +29,7 @@ import { Kind, EnumValueDefinitionNode, } from 'graphql'; +import { getObjectTypeFromTypeMap } from './addTypes'; import { SchemaMapper, @@ -47,16 +48,30 @@ import { rewireTypes } from './rewire'; import { serializeInputValue, parseInputValue } from './transformInputValue'; export function mapSchema(schema: GraphQLSchema, schemaMapper: SchemaMapper = {}): GraphQLSchema { - const originalTypeMap = schema.getTypeMap(); - - let newTypeMap = mapDefaultValues(originalTypeMap, schema, serializeInputValue); - newTypeMap = mapTypes(newTypeMap, schema, schemaMapper, type => isLeafType(type)); - newTypeMap = mapEnumValues(newTypeMap, schema, schemaMapper); - newTypeMap = mapDefaultValues(newTypeMap, schema, parseInputValue); - - newTypeMap = mapTypes(newTypeMap, schema, schemaMapper, type => !isLeafType(type)); - newTypeMap = mapFields(newTypeMap, schema, schemaMapper); - newTypeMap = mapArguments(newTypeMap, schema, schemaMapper); + const newTypeMap = mapArguments( + mapFields( + mapTypes( + mapDefaultValues( + mapEnumValues( + mapTypes(mapDefaultValues(schema.getTypeMap(), schema, serializeInputValue), schema, schemaMapper, type => + isLeafType(type) + ), + schema, + schemaMapper + ), + schema, + parseInputValue + ), + schema, + schemaMapper, + type => !isLeafType(type) + ), + schema, + schemaMapper + ), + schema, + schemaMapper + ); const originalDirectives = schema.getDirectives(); const newDirectives = mapDirectives(originalDirectives, schema, schemaMapper); @@ -65,29 +80,18 @@ export function mapSchema(schema: GraphQLSchema, schemaMapper: SchemaMapper = {} const mutationType = schema.getMutationType(); const subscriptionType = schema.getSubscriptionType(); - const newQueryTypeName = - queryType != null ? (newTypeMap[queryType.name] != null ? newTypeMap[queryType.name].name : undefined) : undefined; - const newMutationTypeName = - mutationType != null - ? newTypeMap[mutationType.name] != null - ? newTypeMap[mutationType.name].name - : undefined - : undefined; - const newSubscriptionTypeName = - subscriptionType != null - ? newTypeMap[subscriptionType.name] != null - ? newTypeMap[subscriptionType.name].name - : undefined - : undefined; + const newQueryTypeName = queryType?.name && newTypeMap?.[queryType?.name]?.name; + const newMutationTypeName = mutationType?.name && newTypeMap?.[mutationType?.name]?.name; + const newSubscriptionTypeName = subscriptionType?.name && newTypeMap?.[subscriptionType?.name]?.name; const { typeMap, directives } = rewireTypes(newTypeMap, newDirectives); return new GraphQLSchema({ ...schema.toConfig(), - query: newQueryTypeName ? (typeMap[newQueryTypeName] as GraphQLObjectType) : undefined, - mutation: newMutationTypeName ? (typeMap[newMutationTypeName] as GraphQLObjectType) : undefined, - subscription: newSubscriptionTypeName != null ? (typeMap[newSubscriptionTypeName] as GraphQLObjectType) : undefined, - types: Object.keys(typeMap).map(typeName => typeMap[typeName]), + query: getObjectTypeFromTypeMap(typeMap, newQueryTypeName), + mutation: getObjectTypeFromTypeMap(typeMap, newMutationTypeName), + subscription: getObjectTypeFromTypeMap(typeMap, newSubscriptionTypeName), + types: Object.values(typeMap), directives, }); } @@ -100,32 +104,32 @@ function mapTypes( ): TypeMap { const newTypeMap = {}; - Object.keys(originalTypeMap).forEach(typeName => { + for (const typeName in originalTypeMap) { if (!typeName.startsWith('__')) { const originalType = originalTypeMap[typeName]; if (originalType == null || !testFn(originalType)) { newTypeMap[typeName] = originalType; - return; + continue; } const typeMapper = getTypeMapper(schema, schemaMapper, typeName); if (typeMapper == null) { newTypeMap[typeName] = originalType; - return; + continue; } const maybeNewType = typeMapper(originalType, schema); if (maybeNewType === undefined) { newTypeMap[typeName] = originalType; - return; + continue; } newTypeMap[typeName] = maybeNewType; } - }); + } return newTypeMap; } @@ -144,7 +148,7 @@ function mapEnumValues(originalTypeMap: TypeMap, schema: GraphQLSchema, schemaMa const config = type.toConfig(); const originalEnumValueConfigMap = config.values; const newEnumValueConfigMap = {}; - Object.keys(originalEnumValueConfigMap).forEach(externalValue => { + for (const externalValue in originalEnumValueConfigMap) { const originalEnumValueConfig = originalEnumValueConfigMap[externalValue]; const mappedEnumValue = enumValueMapper(originalEnumValueConfig, type.name, schema, externalValue); if (mappedEnumValue === undefined) { @@ -156,7 +160,7 @@ function mapEnumValues(originalTypeMap: TypeMap, schema: GraphQLSchema, schemaMa } else if (mappedEnumValue !== null) { newEnumValueConfigMap[externalValue] = mappedEnumValue; } - }); + } return correctASTNodes( new GraphQLEnumType({ ...config, @@ -221,26 +225,26 @@ function getNewType(newTypeMap: TypeMap, type: T): T | nu function mapFields(originalTypeMap: TypeMap, schema: GraphQLSchema, schemaMapper: SchemaMapper): TypeMap { const newTypeMap = {}; - Object.keys(originalTypeMap).forEach(typeName => { + for (const typeName in originalTypeMap) { if (!typeName.startsWith('__')) { const originalType = originalTypeMap[typeName]; if (!isObjectType(originalType) && !isInterfaceType(originalType) && !isInputObjectType(originalType)) { newTypeMap[typeName] = originalType; - return; + continue; } const fieldMapper = getFieldMapper(schema, schemaMapper, typeName); if (fieldMapper == null) { newTypeMap[typeName] = originalType; - return; + continue; } const config = originalType.toConfig(); const originalFieldConfigMap = config.fields; const newFieldConfigMap = {}; - Object.keys(originalFieldConfigMap).forEach(fieldName => { + for (const fieldName in originalFieldConfigMap) { const originalFieldConfig = originalFieldConfigMap[fieldName]; const mappedField = fieldMapper(originalFieldConfig, fieldName, typeName, schema); if (mappedField === undefined) { @@ -260,7 +264,7 @@ function mapFields(originalTypeMap: TypeMap, schema: GraphQLSchema, schemaMapper } else if (mappedField !== null) { newFieldConfigMap[fieldName] = mappedField; } - }); + } if (isObjectType(originalType)) { newTypeMap[typeName] = correctASTNodes( @@ -285,7 +289,7 @@ function mapFields(originalTypeMap: TypeMap, schema: GraphQLSchema, schemaMapper ); } } - }); + } return newTypeMap; } @@ -293,44 +297,44 @@ function mapFields(originalTypeMap: TypeMap, schema: GraphQLSchema, schemaMapper function mapArguments(originalTypeMap: TypeMap, schema: GraphQLSchema, schemaMapper: SchemaMapper): TypeMap { const newTypeMap = {}; - Object.keys(originalTypeMap).forEach(typeName => { + for (const typeName in originalTypeMap) { if (!typeName.startsWith('__')) { const originalType = originalTypeMap[typeName]; if (!isObjectType(originalType) && !isInterfaceType(originalType)) { newTypeMap[typeName] = originalType; - return; + continue; } const argumentMapper = getArgumentMapper(schemaMapper); if (argumentMapper == null) { newTypeMap[typeName] = originalType; - return; + continue; } const config = originalType.toConfig(); const originalFieldConfigMap = config.fields; const newFieldConfigMap = {}; - Object.keys(originalFieldConfigMap).forEach(fieldName => { + for (const fieldName in originalFieldConfigMap) { const originalFieldConfig = originalFieldConfigMap[fieldName]; const originalArgumentConfigMap = originalFieldConfig.args; if (originalArgumentConfigMap == null) { newFieldConfigMap[fieldName] = originalFieldConfig; - return; + continue; } const argumentNames = Object.keys(originalArgumentConfigMap); if (!argumentNames.length) { newFieldConfigMap[fieldName] = originalFieldConfig; - return; + continue; } const newArgumentConfigMap = {}; - argumentNames.forEach(argumentName => { + for (const argumentName of argumentNames) { const originalArgumentConfig = originalArgumentConfigMap[argumentName]; const mappedArgument = argumentMapper(originalArgumentConfig, fieldName, typeName, schema); @@ -343,12 +347,13 @@ function mapArguments(originalTypeMap: TypeMap, schema: GraphQLSchema, schemaMap } else if (mappedArgument !== null) { newArgumentConfigMap[argumentName] = mappedArgument; } - }); + } + newFieldConfigMap[fieldName] = { ...originalFieldConfig, args: newArgumentConfigMap, }; - }); + } if (isObjectType(originalType)) { newTypeMap[typeName] = new GraphQLObjectType({ @@ -367,7 +372,7 @@ function mapArguments(originalTypeMap: TypeMap, schema: GraphQLSchema, schemaMap }); } } - }); + } return newTypeMap; } @@ -384,14 +389,14 @@ function mapDirectives( const newDirectives: Array = []; - originalDirectives.forEach(directive => { + for (const directive of originalDirectives) { const mappedDirective = directiveMapper(directive, schema); if (mappedDirective === undefined) { newDirectives.push(directive); } else if (mappedDirective !== null) { newDirectives.push(mappedDirective); } - }); + } return newDirectives; } @@ -507,11 +512,14 @@ export function correctASTNodes(type: GraphQLNamedType): GraphQLNamedType { const config = (type as GraphQLObjectType).toConfig(); if (config.astNode != null) { const fields: Array = []; - Object.values(config.fields).forEach(fieldConfig => { + for (const fieldName in config.fields) { + const fieldConfig = config.fields[fieldName]; + if (fieldConfig.astNode != null) { fields.push(fieldConfig.astNode); } - }); + } + config.astNode = { ...config.astNode, kind: Kind.OBJECT_TYPE_DEFINITION, @@ -532,11 +540,13 @@ export function correctASTNodes(type: GraphQLNamedType): GraphQLNamedType { const config = (type as GraphQLInterfaceType).toConfig(); if (config.astNode != null) { const fields: Array = []; - Object.values(config.fields).forEach(fieldConfig => { + for (const fieldName in config.fields) { + const fieldConfig = config.fields[fieldName]; + if (fieldConfig.astNode != null) { fields.push(fieldConfig.astNode); } - }); + } config.astNode = { ...config.astNode, kind: Kind.INTERFACE_TYPE_DEFINITION, @@ -557,11 +567,13 @@ export function correctASTNodes(type: GraphQLNamedType): GraphQLNamedType { const config = (type as GraphQLInputObjectType).toConfig(); if (config.astNode != null) { const fields: Array = []; - Object.values(config.fields).forEach(fieldConfig => { + for (const fieldName in config.fields) { + const fieldConfig = config.fields[fieldName]; + if (fieldConfig.astNode != null) { fields.push(fieldConfig.astNode); } - }); + } config.astNode = { ...config.astNode, kind: Kind.INPUT_OBJECT_TYPE_DEFINITION, @@ -582,11 +594,12 @@ export function correctASTNodes(type: GraphQLNamedType): GraphQLNamedType { const config = (type as GraphQLEnumType).toConfig(); if (config.astNode != null) { const values: Array = []; - Object.values(config.values).forEach(enumValueConfig => { + for (const enumKey in config.values) { + const enumValueConfig = config.values[enumKey]; if (enumValueConfig.astNode != null) { values.push(enumValueConfig.astNode); } - }); + } config.astNode = { ...config.astNode, values, diff --git a/packages/utils/src/mergeDeep.ts b/packages/utils/src/mergeDeep.ts index 780bdd4815f..79a7b3a241e 100644 --- a/packages/utils/src/mergeDeep.ts +++ b/packages/utils/src/mergeDeep.ts @@ -10,7 +10,7 @@ export function mergeDeep( ...sources: S ): T & UnboxIntersection>> & any { if (isScalarType(target)) { - return target as any; + return target; } const output = {}; Object.setPrototypeOf(output, Object.create(Object.getPrototypeOf(target))); @@ -19,12 +19,12 @@ export function mergeDeep( const outputPrototype = Object.getPrototypeOf(output); const sourcePrototype = Object.getPrototypeOf(source); if (sourcePrototype) { - Object.getOwnPropertyNames(sourcePrototype).forEach(key => { + for (const key of Object.getOwnPropertyNames(sourcePrototype)) { const descriptor = Object.getOwnPropertyDescriptor(sourcePrototype, key); if (isSome(descriptor)) { Object.defineProperty(outputPrototype, key, descriptor); } - }); + } } for (const key in source) { diff --git a/packages/utils/src/observableToAsyncIterable.ts b/packages/utils/src/observableToAsyncIterable.ts index 3fbe7322853..62c256c7559 100644 --- a/packages/utils/src/observableToAsyncIterable.ts +++ b/packages/utils/src/observableToAsyncIterable.ts @@ -72,7 +72,9 @@ export function observableToAsyncIterable(observable: Observable): AsyncIt if (listening) { listening = false; subscription.unsubscribe(); - pullQueue.forEach(resolve => resolve({ value: undefined, done: true })); + for (const resolve of pullQueue) { + resolve({ value: undefined, done: true }); + } pullQueue.length = 0; pushQueue.length = 0; } diff --git a/packages/utils/src/print-schema-with-directives.ts b/packages/utils/src/print-schema-with-directives.ts index 4e1da45b0b2..d89f09e1958 100644 --- a/packages/utils/src/print-schema-with-directives.ts +++ b/packages/utils/src/print-schema-with-directives.ts @@ -127,21 +127,23 @@ export function astFromSchema( subscription: undefined, }; - let nodes: Array = []; + const nodes: Array = []; if (schema.astNode != null) { nodes.push(schema.astNode); } if (schema.extensionASTNodes != null) { - nodes = nodes.concat(schema.extensionASTNodes); + for (const extensionASTNode of schema.extensionASTNodes) { + nodes.push(extensionASTNode); + } } - nodes.forEach(node => { + for (const node of nodes) { if (node.operationTypes) { - node.operationTypes.forEach(operationTypeDefinitionNode => { + for (const operationTypeDefinitionNode of node.operationTypes) { operationTypeMap[operationTypeDefinitionNode.operation] = operationTypeDefinitionNode; - }); + } } - }); + } const rootTypeMap: Record> = { query: schema.getQueryType(), @@ -149,7 +151,7 @@ export function astFromSchema( subscription: schema.getSubscriptionType(), }; - Object.keys(operationTypeMap).forEach(operationTypeNode => { + for (const operationTypeNode in operationTypeMap) { if (rootTypeMap[operationTypeNode] != null) { if (operationTypeMap[operationTypeNode] != null) { operationTypeMap[operationTypeNode].type = astFromType(rootTypeMap[operationTypeNode]); @@ -161,7 +163,7 @@ export function astFromSchema( }; } } - }); + } const operationTypes = Object.values(operationTypeMap).filter(isSome); @@ -598,7 +600,7 @@ export function makeDirectiveNode( const directiveArguments: Array = []; if (directive != null) { - directive.args.forEach(arg => { + for (const arg of directive.args) { const argName = arg.name; const argValue = args[argName]; if (argValue !== undefined) { @@ -614,9 +616,10 @@ export function makeDirectiveNode( }); } } - }); + } } else { - Object.entries(args).forEach(([argName, argValue]) => { + for (const argName in args) { + const argValue = args[argName]; const value = astFromValueUntyped(argValue); if (value) { directiveArguments.push({ @@ -628,7 +631,7 @@ export function makeDirectiveNode( value, }); } - }); + } } return { @@ -646,15 +649,16 @@ export function makeDirectiveNodes( directiveValues: Record ): Array { const directiveNodes: Array = []; - Object.entries(directiveValues).forEach(([directiveName, arrayOrSingleValue]) => { + for (const directiveName in directiveValues) { + const arrayOrSingleValue = directiveValues[directiveName]; const directive = schema?.getDirective(directiveName); if (Array.isArray(arrayOrSingleValue)) { - arrayOrSingleValue.forEach(value => { + for (const value of arrayOrSingleValue) { directiveNodes.push(makeDirectiveNode(directiveName, value, directive)); - }); + } } else { directiveNodes.push(makeDirectiveNode(directiveName, arrayOrSingleValue, directive)); } - }); + } return directiveNodes; } diff --git a/packages/utils/src/prune.ts b/packages/utils/src/prune.ts index 5d046383054..7be287cf9a8 100644 --- a/packages/utils/src/prune.ts +++ b/packages/utils/src/prune.ts @@ -46,18 +46,18 @@ export function pruneSchema(schema: GraphQLSchema, options: PruneSchemaOptions = implementations: Object.create(null), }; - Object.keys(schema.getTypeMap()).forEach(typeName => { + for (const typeName in schema.getTypeMap()) { const type = schema.getType(typeName); if (type && 'getInterfaces' in type) { - type.getInterfaces().forEach(iface => { + for (const iface of type.getInterfaces()) { const implementations = getImplementations(pruningContext, iface); if (implementations == null) { pruningContext.implementations[iface.name] = Object.create(null); } pruningContext.implementations[iface.name][type.name] = true; - }); + } } - }); + } visitTypes(pruningContext, schema); @@ -115,35 +115,36 @@ function visitOutputType( if (isObjectType(type) || isInterfaceType(type)) { const fields = type.getFields(); - Object.keys(fields).forEach(fieldName => { + for (const fieldName in fields) { const field = fields[fieldName]; const namedType = getNamedType(field.type) as NamedOutputType; visitOutputType(visitedTypes, pruningContext, namedType); - const args = field.args; - args.forEach(arg => { + for (const arg of field.args) { const type = getNamedType(arg.type) as NamedInputType; visitInputType(visitedTypes, pruningContext, type); - }); - }); + } + } if (isInterfaceType(type)) { const implementations = getImplementations(pruningContext, type); if (implementations) { - Object.keys(implementations).forEach(typeName => { + for (const typeName in implementations) { visitOutputType(visitedTypes, pruningContext, pruningContext.schema.getType(typeName) as NamedOutputType); - }); + } } } if ('getInterfaces' in type) { - type.getInterfaces().forEach(type => { - visitOutputType(visitedTypes, pruningContext, type); - }); + for (const iFace of type.getInterfaces()) { + visitOutputType(visitedTypes, pruningContext, iFace); + } } } else if (isUnionType(type)) { const types = type.getTypes(); - types.forEach(type => visitOutputType(visitedTypes, pruningContext, type)); + for (const type of types) { + visitOutputType(visitedTypes, pruningContext, type); + } } } @@ -171,31 +172,33 @@ function visitInputType( if (isInputObjectType(type)) { const fields = type.getFields(); - Object.keys(fields).forEach(fieldName => { + for (const fieldName in fields) { const field = fields[fieldName]; const namedType = getNamedType(field.type) as NamedInputType; visitInputType(visitedTypes, pruningContext, namedType); - }); + } } } function visitTypes(pruningContext: PruningContext, schema: GraphQLSchema): void { - Object.keys(schema.getTypeMap()).forEach(typeName => { + for (const typeName in schema.getTypeMap()) { if (!typeName.startsWith('__')) { pruningContext.unusedTypes[typeName] = true; } - }); + } const visitedTypes: Record = Object.create(null); const rootTypes = [schema.getQueryType(), schema.getMutationType(), schema.getSubscriptionType()].filter(isSome); - rootTypes.forEach(rootType => visitOutputType(visitedTypes, pruningContext, rootType)); + for (const rootType of rootTypes) { + visitOutputType(visitedTypes, pruningContext, rootType); + } - schema.getDirectives().forEach(directive => { - directive.args.forEach(arg => { + for (const directive of schema.getDirectives()) { + for (const arg of directive.args) { const type = getNamedType(arg.type) as NamedInputType; visitInputType(visitedTypes, pruningContext, type); - }); - }); + } + } } diff --git a/packages/utils/src/rewire.ts b/packages/utils/src/rewire.ts index 4e257714569..45e6639d688 100644 --- a/packages/utils/src/rewire.ts +++ b/packages/utils/src/rewire.ts @@ -37,21 +37,21 @@ export function rewireTypes( directives: Array; } { const referenceTypeMap = Object.create(null); - Object.keys(originalTypeMap).forEach(typeName => { + for (const typeName in originalTypeMap) { referenceTypeMap[typeName] = originalTypeMap[typeName]; - }); + } const newTypeMap: TypeMap = Object.create(null); - Object.keys(referenceTypeMap).forEach(typeName => { + for (const typeName in referenceTypeMap) { const namedType = referenceTypeMap[typeName]; if (namedType == null || typeName.startsWith('__')) { - return; + continue; } const newName = namedType.name; if (newName.startsWith('__')) { - return; + continue; } if (newTypeMap[newName] != null) { @@ -59,11 +59,11 @@ export function rewireTypes( } newTypeMap[newName] = namedType; - }); + } - Object.keys(newTypeMap).forEach(typeName => { + for (const typeName in newTypeMap) { newTypeMap[typeName] = rewireNamedType(newTypeMap[typeName]); - }); + } const newDirectives = directives.map(directive => rewireDirective(directive)); @@ -83,14 +83,14 @@ export function rewireTypes( function rewireArgs(args: GraphQLFieldConfigArgumentMap): GraphQLFieldConfigArgumentMap { const rewiredArgs = {}; - Object.keys(args).forEach(argName => { + for (const argName in args) { const arg = args[argName]; const rewiredArgType = rewireType(arg.type); if (rewiredArgType != null) { arg.type = rewiredArgType; rewiredArgs[argName] = arg; } - }); + } return rewiredArgs; } @@ -144,7 +144,7 @@ export function rewireTypes( function rewireFields(fields: GraphQLFieldConfigMap): GraphQLFieldConfigMap { const rewiredFields = {}; - Object.keys(fields).forEach(fieldName => { + for (const fieldName in fields) { const field = fields[fieldName]; const rewiredFieldType = rewireType(field.type); if (rewiredFieldType != null && field.args) { @@ -152,31 +152,31 @@ export function rewireTypes( field.args = rewireArgs(field.args); rewiredFields[fieldName] = field; } - }); + } return rewiredFields; } function rewireInputFields(fields: GraphQLInputFieldConfigMap): GraphQLInputFieldConfigMap { const rewiredFields = {}; - Object.keys(fields).forEach(fieldName => { + for (const fieldName in fields) { const field = fields[fieldName]; const rewiredFieldType = rewireType(field.type); if (rewiredFieldType != null) { field.type = rewiredFieldType; rewiredFields[fieldName] = field; } - }); + } return rewiredFields; } function rewireNamedTypes(namedTypes: Array): Array { const rewiredTypes: Array = []; - namedTypes.forEach(namedType => { + for (const namedType of namedTypes) { const rewiredType = rewireType(namedType); if (rewiredType != null) { rewiredTypes.push(rewiredType); } - }); + } return rewiredTypes; } diff --git a/packages/utils/src/stub.ts b/packages/utils/src/stub.ts index 1d6cbdb2c13..de72bbfa995 100644 --- a/packages/utils/src/stub.ts +++ b/packages/utils/src/stub.ts @@ -8,9 +8,6 @@ import { GraphQLFloat, GraphQLBoolean, GraphQLID, - isObjectType, - isInterfaceType, - isInputObjectType, TypeNode, Kind, GraphQLType, @@ -64,10 +61,13 @@ export function createStub(node: TypeNode, type: any): any { } export function isNamedStub(type: GraphQLNamedType): boolean { - if (isObjectType(type) || isInterfaceType(type) || isInputObjectType(type)) { + if ('getFields' in type) { const fields = type.getFields(); - const fieldNames = Object.keys(fields); - return fieldNames.length === 1 && fields[fieldNames[0]].name === '_fake'; + // eslint-disable-next-line no-unreachable-loop + for (const fieldName in fields) { + const field = fields[fieldName]; + return field.name === '_fake'; + } } return false; diff --git a/packages/utils/src/toConfig.ts b/packages/utils/src/toConfig.ts deleted file mode 100644 index 623b68395db..00000000000 --- a/packages/utils/src/toConfig.ts +++ /dev/null @@ -1,51 +0,0 @@ -import { - GraphQLArgument, - GraphQLFieldConfigArgumentMap, - GraphQLField, - GraphQLInputField, - GraphQLInputFieldConfig, - GraphQLArgumentConfig, - GraphQLFieldConfig, -} from 'graphql'; - -export function inputFieldToFieldConfig(field: GraphQLInputField): GraphQLInputFieldConfig { - return { - description: field.description, - type: field.type, - defaultValue: field.defaultValue, - extensions: field.extensions, - astNode: field.astNode, - }; -} - -export function fieldToFieldConfig(field: GraphQLField): GraphQLFieldConfig { - return { - description: field.description, - type: field.type, - args: argsToFieldConfigArgumentMap(field.args), - resolve: field.resolve, - subscribe: field.subscribe, - deprecationReason: field.deprecationReason, - extensions: field.extensions, - astNode: field.astNode, - }; -} - -export function argsToFieldConfigArgumentMap(args: ReadonlyArray): GraphQLFieldConfigArgumentMap { - const newArguments = {}; - args.forEach(arg => { - newArguments[arg.name] = argumentToArgumentConfig(arg); - }); - - return newArguments; -} - -export function argumentToArgumentConfig(arg: GraphQLArgument): GraphQLArgumentConfig { - return { - description: arg.description, - type: arg.type, - defaultValue: arg.defaultValue, - extensions: arg.extensions, - astNode: arg.astNode, - }; -} diff --git a/packages/utils/src/transformInputValue.ts b/packages/utils/src/transformInputValue.ts index db95d48d71c..530a0a7be19 100644 --- a/packages/utils/src/transformInputValue.ts +++ b/packages/utils/src/transformInputValue.ts @@ -23,7 +23,7 @@ export function transformInputValue( } else if (isInputObjectType(nullableType)) { const fields = nullableType.getFields(); const newValue = {}; - Object.keys(value).forEach(key => { + for (const key in value) { const field = fields[key]; if (field != null) { newValue[key] = transformInputValue( @@ -33,7 +33,7 @@ export function transformInputValue( inputObjectValueTransformer ); } - }); + } return inputObjectValueTransformer != null ? inputObjectValueTransformer(nullableType, newValue) : newValue; } diff --git a/packages/utils/src/validate-documents.ts b/packages/utils/src/validate-documents.ts index 0c24bd25f3e..9ec3a978f5f 100644 --- a/packages/utils/src/validate-documents.ts +++ b/packages/utils/src/validate-documents.ts @@ -7,9 +7,12 @@ import { FragmentDefinitionNode, ValidationContext, ASTVisitor, + DefinitionNode, + concatAST, + DocumentNode, } from 'graphql'; import { Source } from './loaders'; -import AggregateError from '@ardatan/aggregate-error'; +import { AggregateError } from './AggregateError'; export type ValidationRule = (context: ValidationContext) => ASTVisitor; @@ -24,41 +27,42 @@ export async function validateGraphQlDocuments( effectiveRules?: ValidationRule[] ): Promise> { effectiveRules = effectiveRules || createDefaultRules(); - const allFragments: FragmentDefinitionNode[] = []; + const allFragmentMap = new Map(); + const documentFileObjectsToValidate: { + location?: string; + document: DocumentNode; + }[] = []; - documentFiles.forEach(documentFile => { + for (const documentFile of documentFiles) { if (documentFile.document) { + const definitionsToValidate: DefinitionNode[] = []; for (const definitionNode of documentFile.document.definitions) { if (definitionNode.kind === Kind.FRAGMENT_DEFINITION) { - allFragments.push(definitionNode); + allFragmentMap.set(definitionNode.name.value, definitionNode); + } else { + definitionsToValidate.push(definitionNode); } } + documentFileObjectsToValidate.push({ + location: documentFile.location, + document: { + kind: Kind.DOCUMENT, + definitions: definitionsToValidate, + }, + }); } - }); + } const allErrors: LoadDocumentError[] = []; + const allFragmentsDocument: DocumentNode = { + kind: Kind.DOCUMENT, + definitions: [...allFragmentMap.values()], + }; + await Promise.all( - documentFiles.map(async documentFile => { - const documentToValidate = { - kind: Kind.DOCUMENT, - definitions: [...allFragments, ...(documentFile.document?.definitions ?? [])].filter( - (definition, index, list) => { - if (definition.kind === Kind.FRAGMENT_DEFINITION) { - const firstIndex = list.findIndex( - def => def.kind === Kind.FRAGMENT_DEFINITION && def.name.value === definition.name.value - ); - const isDuplicated = firstIndex !== index; - - if (isDuplicated) { - return false; - } - } - - return true; - } - ), - }; + documentFileObjectsToValidate.map(async documentFile => { + const documentToValidate = concatAST([allFragmentsDocument, documentFile.document]); const errors = validate(schema, documentToValidate, effectiveRules); @@ -84,26 +88,25 @@ export function checkValidationErrors(loadDocumentErrors: ReadonlyArray (error.stack += `\n at ${loadDocumentError.filePath}:${location.line}:${location.column}`) - ); + if (graphQLError.locations) { + for (const location of graphQLError.locations) { + error.stack += `\n at ${loadDocumentError.filePath}:${location.line}:${location.column}`; + } + } errors.push(error); } } - throw new AggregateError(errors); + throw new AggregateError(errors, `GraphQL Document Validation failed with ${loadDocumentErrors.length} errors`); } } function createDefaultRules() { const ignored = ['NoUnusedFragmentsRule', 'NoUnusedVariablesRule', 'KnownDirectivesRule']; + const v4ignored = ignored.map(rule => rule.replace(/Rule$/, '')); - // GraphQL v14 has no Rule suffix in function names - // Adding `*Rule` makes validation backwards compatible - ignored.forEach(rule => { - ignored.push(rule.replace(/Rule$/, '')); - }); - - return specifiedRules.filter((f: (...args: any[]) => any) => !ignored.includes(f.name)); + return specifiedRules.filter( + (f: (...args: any[]) => any) => !ignored.includes(f.name) && !v4ignored.includes(f.name) + ); } diff --git a/packages/utils/src/visitResult.ts b/packages/utils/src/visitResult.ts index fc870ea29aa..f71725fd41f 100644 --- a/packages/utils/src/visitResult.ts +++ b/packages/utils/src/visitResult.ts @@ -17,7 +17,7 @@ import { import { Request, GraphQLExecutionContext, ExecutionResult } from './Interfaces'; import { collectFields } from './collectFields'; -import { Maybe } from 'packages/graphql-tools/src'; +import { Maybe } from '@graphql-tools/utils'; export type ValueVisitor = (value: any) => any; @@ -57,10 +57,10 @@ export function visitData(data: any, enter?: ValueVisitor, leave?: ValueVisitor) const newData = enter != null ? enter(data) : data; if (newData != null) { - Object.keys(newData).forEach(key => { + for (const key in newData) { const value = newData[key]; newData[key] = visitData(value, enter, leave); - }); + } } return leave != null ? leave(newData) : newData; @@ -195,10 +195,12 @@ function visitObjectValue( if (errors != null) { sortedErrors = sortErrorsByPathSegment(errors, pathIndex); errorMap = sortedErrors.errorMap; - sortedErrors.unpathedErrors.forEach(error => errorInfo.unpathedErrors.add(error)); + for (const error of sortedErrors.unpathedErrors) { + errorInfo.unpathedErrors.add(error); + } } - Object.keys(fieldNodeMap).forEach(responseKey => { + for (const responseKey in fieldNodeMap) { const subFieldNodes = fieldNodeMap[responseKey]; const fieldName = subFieldNodes[0].name.value; const fieldType = fieldName === '__typename' ? TypeNameMetaFieldDef.type : fieldMap[fieldName].type; @@ -226,7 +228,7 @@ function visitObjectValue( ); updateObject(newObject, responseKey, newValue, typeVisitorMap, fieldName); - }); + } const oldTypename = newObject.__typename; if (oldTypename != null) { @@ -353,11 +355,11 @@ function visitFieldValue( function sortErrorsByPathSegment(errors: ReadonlyArray, pathIndex: number): SortedErrors { const errorMap = Object.create(null); const unpathedErrors: Set = new Set(); - errors.forEach(error => { + for (const error of errors) { const pathSegment = error.path?.[pathIndex]; if (pathSegment == null) { unpathedErrors.add(error); - return; + continue; } if (pathSegment in errorMap) { @@ -365,7 +367,7 @@ function sortErrorsByPathSegment(errors: ReadonlyArray, pathIndex: } else { errorMap[pathSegment] = [error]; } - }); + } return { errorMap, @@ -380,7 +382,7 @@ function addPathSegmentInfo( errors: ReadonlyArray = [], errorInfo: ErrorInfo ) { - errors.forEach(error => { + for (const error of errors) { const segmentInfo = { type, fieldName, @@ -392,7 +394,7 @@ function addPathSegmentInfo( } else { pathSegmentsInfo.push(segmentInfo); } - }); + } } function collectSubFields( @@ -403,11 +405,11 @@ function collectSubFields( let subFieldNodes: Record> = Object.create(null); const visitedFragmentNames = Object.create(null); - fieldNodes.forEach(fieldNode => { + for (const fieldNode of fieldNodes) { if (fieldNode.selectionSet) { subFieldNodes = collectFields(exeContext, type, fieldNode.selectionSet, subFieldNodes, visitedFragmentNames); } - }); + } return subFieldNodes; } diff --git a/packages/utils/src/visitSchema.ts b/packages/utils/src/visitSchema.ts deleted file mode 100644 index 7878992e8e7..00000000000 --- a/packages/utils/src/visitSchema.ts +++ /dev/null @@ -1,320 +0,0 @@ -import { - GraphQLInterfaceType, - GraphQLObjectType, - GraphQLSchema, - isNamedType, - GraphQLType, - GraphQLNamedType, - GraphQLInputField, - isSchema, - isObjectType, - isInterfaceType, - isInputObjectType, - isScalarType, - isUnionType, - isEnumType, - isInputType, - GraphQLEnumValue, - GraphQLEnumType, -} from 'graphql'; - -import { - VisitableSchemaType, - VisitorSelector, - VisitSchemaKind, - NamedTypeVisitor, - SchemaVisitorMap, -} from './Interfaces'; - -import { healSchema } from './heal'; -import { SchemaVisitor } from './SchemaVisitor'; -import { isSome } from './helpers'; - -function isSchemaVisitor(obj: any): obj is SchemaVisitor { - if ('schema' in obj && isSchema(obj.schema)) { - if ('visitSchema' in obj && typeof obj.visitSchema === 'function') { - return true; - } - } - return false; -} - -// Generic function for visiting GraphQLSchema objects. -export function visitSchema( - schema: GraphQLSchema, - // To accommodate as many different visitor patterns as possible, the - // visitSchema function does not simply accept a single instance of the - // SchemaVisitor class, but instead accepts a function that takes the - // current VisitableSchemaType object and the name of a visitor method and - // returns an array of SchemaVisitor instances that implement the visitor - // method and have an interest in handling the given VisitableSchemaType - // object. In the simplest case, this function can always return an array - // containing a single visitor object, without even looking at the type or - // methodName parameters. In other cases, this function might sometimes - // return an empty array to indicate there are no visitors that should be - // applied to the given VisitableSchemaType object. For an example of a - // visitor pattern that benefits from this abstraction, see the - // SchemaDirectiveVisitor class below. - visitorOrVisitorSelector: VisitorSelector | Array | SchemaVisitor | SchemaVisitorMap -): GraphQLSchema { - const visitorSelector = - typeof visitorOrVisitorSelector === 'function' ? visitorOrVisitorSelector : () => visitorOrVisitorSelector; - - // Helper function that calls visitorSelector and applies the resulting - // visitors to the given type, with arguments [type, ...args]. - function callMethod(methodName: string, type: T, ...args: Array): T | null { - let visitors = visitorSelector(type, methodName); - visitors = Array.isArray(visitors) ? visitors : [visitors]; - - let finalType: T | null = type; - visitors.every(visitorOrVisitorDef => { - let newType; - if (isSchemaVisitor(visitorOrVisitorDef)) { - newType = visitorOrVisitorDef[methodName](finalType, ...args); - } else if ( - isNamedType(finalType) && - (methodName === 'visitScalar' || - methodName === 'visitEnum' || - methodName === 'visitObject' || - methodName === 'visitInputObject' || - methodName === 'visitUnion' || - methodName === 'visitInterface') - ) { - const specifiers = getTypeSpecifiers(finalType, schema); - const typeVisitor = getVisitor(visitorOrVisitorDef, specifiers); - newType = typeVisitor != null ? typeVisitor(finalType, schema) : undefined; - } - - if (typeof newType === 'undefined') { - // Keep going without modifying type. - return true; - } - - if (methodName === 'visitSchema' || isSchema(finalType)) { - throw new Error(`Method ${methodName} cannot replace schema with ${newType as string}`); - } - - if (newType === null) { - // Stop the loop and return null form callMethod, which will cause - // the type to be removed from the schema. - finalType = null; - return false; - } - - // Update type to the new type returned by the visitor method, so that - // later directives will see the new type, and callMethod will return - // the final type. - finalType = newType; - return true; - }); - - // If there were no directives for this type object, or if all visitor - // methods returned nothing, type will be returned unmodified. - return finalType; - } - - // Recursive helper function that calls any appropriate visitor methods for - // each object in the schema, then traverses the object's children (if any). - function visit(type: T): T | null { - if (isSchema(type)) { - // Unlike the other types, the root GraphQLSchema object cannot be - // replaced by visitor methods, because that would make life very hard - // for SchemaVisitor subclasses that rely on the original schema object. - callMethod('visitSchema', type); - - const typeMap: Record = type.getTypeMap(); - Object.entries(typeMap).forEach(([typeName, namedType]) => { - if (!typeName.startsWith('__') && namedType != null) { - // Call visit recursively to let it determine which concrete - // subclass of GraphQLNamedType we found in the type map. - // We do not use updateEachKey because we want to preserve - // deleted types in the typeMap so that other types that reference - // the deleted types can be healed. - typeMap[typeName] = visit(namedType); - } - }); - - return type; - } - - if (isObjectType(type)) { - // Note that callMethod('visitObject', type) may not actually call any - // methods, if there are no @directive annotations associated with this - // type, or if this SchemaDirectiveVisitor subclass does not override - // the visitObject method. - const newObject = callMethod('visitObject', type); - if (newObject != null) { - visitFields(newObject); - } - return newObject; - } - - if (isInterfaceType(type)) { - const newInterface = callMethod('visitInterface', type); - if (newInterface != null) { - visitFields(newInterface); - } - return newInterface; - } - - if (isInputObjectType(type)) { - const newInputObject = callMethod('visitInputObject', type); - - if (newInputObject != null) { - const fieldMap = newInputObject.getFields() as Record; - for (const key of Object.keys(fieldMap)) { - const result = callMethod('visitInputFieldDefinition', fieldMap[key], { - // Since we call a different method for input object fields, we - // can't reuse the visitFields function here. - objectType: newInputObject, - }); - if (result) { - fieldMap[key] = result; - } else { - delete fieldMap[key]; - } - } - } - - return newInputObject; - } - - if (isScalarType(type)) { - return callMethod('visitScalar', type); - } - - if (isUnionType(type)) { - return callMethod('visitUnion', type); - } - - if (isEnumType(type)) { - let newEnum = callMethod('visitEnum', type); - - if (newEnum != null) { - const newValues: Array = newEnum - .getValues() - .map(value => - callMethod('visitEnumValue', value, { - enumType: newEnum, - }) - ) - .filter(isSome); - - // Recreate the enum type if any of the values changed - const valuesUpdated = newValues.some((value, index) => value !== newEnum!.getValues()[index]); - if (valuesUpdated) { - newEnum = new GraphQLEnumType({ - ...(newEnum as GraphQLEnumType).toConfig(), - values: newValues.reduce( - (prev, value) => ({ - ...prev, - [value.name]: { - value: value.value, - deprecationReason: value.deprecationReason, - description: value.description, - astNode: value.astNode, - }, - }), - {} - ), - }) as GraphQLEnumType & T; - } - } - - return newEnum; - } - - throw new Error(`Unexpected schema type: ${type as unknown as string}`); - } - - function visitFields(type: GraphQLObjectType | GraphQLInterfaceType) { - const fieldMap = type.getFields(); - for (const [key, field] of Object.entries(fieldMap)) { - // It would be nice if we could call visit(field) recursively here, but - // GraphQLField is merely a type, not a value that can be detected using - // an instanceof check, so we have to visit the fields in this lexical - // context, so that TypeScript can validate the call to - // visitFieldDefinition. - const newField = callMethod('visitFieldDefinition', field, { - // While any field visitor needs a reference to the field object, some - // field visitors may also need to know the enclosing (parent) type, - // perhaps to determine if the parent is a GraphQLObjectType or a - // GraphQLInterfaceType. To obtain a reference to the parent, a - // visitor method can have a second parameter, which will be an object - // with an .objectType property referring to the parent. - objectType: type, - }); - - if (newField?.args != null) { - newField.args = newField.args - .map(arg => - callMethod('visitArgumentDefinition', arg, { - // Like visitFieldDefinition, visitArgumentDefinition takes a - // second parameter that provides additional context, namely the - // parent .field and grandparent .objectType. Remember that the - // current GraphQLSchema is always available via this.schema. - field: newField, - objectType: type, - }) - ) - .filter(isSome); - } - - // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition - if (newField) { - fieldMap[key] = newField; - } else { - delete fieldMap[key]; - } - } - } - - visit(schema); - - // Automatically update any references to named schema types replaced - // during the traversal, so implementers don't have to worry about that. - healSchema(schema); - - // Return schema for convenience, even though schema parameter has all updated types. - return schema; -} - -function getTypeSpecifiers(type: GraphQLType, schema: GraphQLSchema): Array { - const specifiers = [VisitSchemaKind.TYPE]; - if (isObjectType(type)) { - specifiers.push(VisitSchemaKind.COMPOSITE_TYPE, VisitSchemaKind.OBJECT_TYPE); - const query = schema.getQueryType(); - const mutation = schema.getMutationType(); - const subscription = schema.getSubscriptionType(); - if (type === query) { - specifiers.push(VisitSchemaKind.ROOT_OBJECT, VisitSchemaKind.QUERY); - } else if (type === mutation) { - specifiers.push(VisitSchemaKind.ROOT_OBJECT, VisitSchemaKind.MUTATION); - } else if (type === subscription) { - specifiers.push(VisitSchemaKind.ROOT_OBJECT, VisitSchemaKind.SUBSCRIPTION); - } - } else if (isInputType(type)) { - specifiers.push(VisitSchemaKind.INPUT_OBJECT_TYPE); - } else if (isInterfaceType(type)) { - specifiers.push(VisitSchemaKind.COMPOSITE_TYPE, VisitSchemaKind.ABSTRACT_TYPE, VisitSchemaKind.INTERFACE_TYPE); - } else if (isUnionType(type)) { - specifiers.push(VisitSchemaKind.COMPOSITE_TYPE, VisitSchemaKind.ABSTRACT_TYPE, VisitSchemaKind.UNION_TYPE); - } else if (isEnumType(type)) { - specifiers.push(VisitSchemaKind.ENUM_TYPE); - } else if (isScalarType(type)) { - specifiers.push(VisitSchemaKind.SCALAR_TYPE); - } - - return specifiers; -} - -function getVisitor(visitorDef: SchemaVisitorMap, specifiers: Array): NamedTypeVisitor | null { - let typeVisitor: NamedTypeVisitor | undefined; - const stack = [...specifiers]; - while (!typeVisitor && stack.length > 0) { - const next = stack.pop()!; - typeVisitor = visitorDef[next] as NamedTypeVisitor; - } - - return typeVisitor != null ? typeVisitor : null; -} diff --git a/packages/utils/tests/build-operation-node-for-field.spec.ts b/packages/utils/tests/build-operation-node-for-field.spec.ts index 01b3e506d1c..53e9612e24f 100644 --- a/packages/utils/tests/build-operation-node-for-field.spec.ts +++ b/packages/utils/tests/build-operation-node-for-field.spec.ts @@ -84,7 +84,7 @@ test('should work with Query', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - query meQuery { + query me_query { me { id name @@ -131,7 +131,7 @@ test('should work with Query and variables', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - query userQuery($id: ID!) { + query user_query($id: ID!) { user(id: $id) { id name @@ -178,7 +178,7 @@ test('should work with Query and complicated variable', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - query menuByIngredientsQuery($ingredients: [String!]!) { + query menuByIngredients_query($ingredients: [String!]!) { menuByIngredients(ingredients: $ingredients) { ... on Pizza { dough @@ -211,7 +211,7 @@ test('should work with Union', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - query menuQuery { + query menu_query { menu { ... on Pizza { dough @@ -244,7 +244,7 @@ test('should work with mutation', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - mutation addSaladMutation($ingredients: [String!]!) { + mutation addSalad_mutation($ingredients: [String!]!) { addSalad(ingredients: $ingredients) { ... on CaeserSalad { ingredients @@ -271,7 +271,7 @@ test('should work with mutation and unions', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - mutation addRandomFoodMutation { + mutation addRandomFood_mutation { addRandomFood { ... on Pizza { dough @@ -304,9 +304,9 @@ test('should work with Query and nested variables', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - query feedQuery($feedCommentsFilter: String!) { + query feed_query($feed_comments_filter: String!) { feed { - comments(filter: $feedCommentsFilter) + comments(filter: $feed_comments_filter) } } `) @@ -324,7 +324,7 @@ test('should be able to ignore using models when requested', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - query userQuery($id: ID!) { + query user_query($id: ID!) { user(id: $id) { id name @@ -373,7 +373,7 @@ test('should work with Subscription', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - subscription onFoodSubscription { + subscription onFood_subscription { onFood { ... on Pizza { dough @@ -423,7 +423,7 @@ test('should work with circular ref (default depth limit === 1)', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - query aQuery { + query a_query { a { b { c { @@ -465,7 +465,7 @@ test('should work with circular ref (custom depth limit)', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - query aQuery { + query a_query { a { b { c { @@ -510,7 +510,7 @@ test('arguments', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - query usersQuery($pageInfo: PageInfoInput!) { + query users_query($pageInfo: PageInfoInput!) { users(pageInfo: $pageInfo) { id name @@ -538,7 +538,7 @@ test('selectedFields', async () => { expect(clean(document)).toEqual( clean(/* GraphQL */ ` - query userQuery($id: ID!) { + query user_query($id: ID!) { user(id: $id) { favoriteFood { ... on Pizza { diff --git a/packages/utils/tests/directives.test.ts b/packages/utils/tests/directives.test.ts deleted file mode 100644 index 7ade95d4bcc..00000000000 --- a/packages/utils/tests/directives.test.ts +++ /dev/null @@ -1,1601 +0,0 @@ -import '../../testing/to-be-similar-string'; -import { createHash } from 'crypto'; - -import { - GraphQLArgument, - GraphQLEnumType, - GraphQLEnumValue, - GraphQLField, - GraphQLID, - GraphQLInputField, - GraphQLInputObjectType, - GraphQLObjectType, - GraphQLScalarType, - GraphQLSchema, - GraphQLString, - defaultFieldResolver, - graphql, - GraphQLNonNull, - GraphQLList, - GraphQLUnionType, - GraphQLInt, - GraphQLOutputType, - isNonNullType, - isScalarType, - isListType, - TypeSystemExtensionNode, - GraphQLDirective, - DirectiveLocation, - print, -} from 'graphql'; -import formatDate from 'dateformat'; - -import { makeExecutableSchema } from '@graphql-tools/schema'; -import { - VisitableSchemaType, - SchemaDirectiveVisitor, - SchemaVisitor, - visitSchema, - ExecutionResult, - astFromDirective, -} from '@graphql-tools/utils'; -import { assertGraphQLEnumType, assertGraphQLInputObjectType, assertGraphQLInterfaceType, assertGraphQLObjectType, assertGraphQLScalerType, assertGraphQLUnionType } from '../../testing/assertion'; - -const typeDefs = ` -directive @schemaDirective(role: String) on SCHEMA -directive @schemaExtensionDirective(role: String) on SCHEMA -directive @queryTypeDirective on OBJECT -directive @queryTypeExtensionDirective on OBJECT -directive @queryFieldDirective on FIELD_DEFINITION -directive @enumTypeDirective on ENUM -directive @enumTypeExtensionDirective on ENUM -directive @enumValueDirective on ENUM_VALUE -directive @dateDirective(tz: String) on SCALAR -directive @dateExtensionDirective(tz: String) on SCALAR -directive @interfaceDirective on INTERFACE -directive @interfaceExtensionDirective on INTERFACE -directive @interfaceFieldDirective on FIELD_DEFINITION -directive @inputTypeDirective on INPUT_OBJECT -directive @inputTypeExtensionDirective on INPUT_OBJECT -directive @inputFieldDirective on INPUT_FIELD_DEFINITION -directive @mutationTypeDirective on OBJECT -directive @mutationTypeExtensionDirective on OBJECT -directive @mutationArgumentDirective on ARGUMENT_DEFINITION -directive @mutationMethodDirective on FIELD_DEFINITION -directive @objectTypeDirective on OBJECT -directive @objectTypeExtensionDirective on OBJECT -directive @objectFieldDirective on FIELD_DEFINITION -directive @unionDirective on UNION -directive @unionExtensionDirective on UNION - -schema @schemaDirective(role: "admin") { - query: Query - mutation: Mutation -} - -extend schema @schemaExtensionDirective(role: "admin") - -type Query @queryTypeDirective { - people: [Person] @queryFieldDirective -} - -extend type Query @queryTypeExtensionDirective - -enum Gender @enumTypeDirective { - NONBINARY @enumValueDirective - FEMALE - MALE -} - -extend enum Gender @enumTypeExtensionDirective -scalar Date @dateDirective(tz: "utc") - -extend scalar Date @dateExtensionDirective(tz: "utc") -interface Named @interfaceDirective { - name: String! @interfaceFieldDirective -} - -extend interface Named @interfaceExtensionDirective -input PersonInput @inputTypeDirective { - name: String! @inputFieldDirective - gender: Gender -} - -extend input PersonInput @inputTypeExtensionDirective -type Mutation @mutationTypeDirective { - addPerson( - input: PersonInput @mutationArgumentDirective - ): Person @mutationMethodDirective -} - -extend type Mutation @mutationTypeExtensionDirective -type Person implements Named @objectTypeDirective { - id: ID! @objectFieldDirective - name: String! -} - -extend type Person @objectTypeExtensionDirective -union WhateverUnion @unionDirective = Person | Query | Mutation - -extend union WhateverUnion @unionExtensionDirective`; - -describe('@directives', () => { - test('are included in the schema AST', () => { - const schema = makeExecutableSchema({ - typeDefs, - resolvers: { - Gender: { - NONBINARY: 'NB', - FEMALE: 'F', - MALE: 'M', - }, - }, - }); - - function checkDirectives( - type: VisitableSchemaType, - typeDirectiveNames: Array, - fieldDirectiveMap: Record> = {}, - ) { - expect(getDirectiveNames(type)).toEqual(typeDirectiveNames); - - Object.keys(fieldDirectiveMap).forEach((key) => { - expect( - getDirectiveNames((type as GraphQLObjectType).getFields()[key]), - ).toEqual(fieldDirectiveMap[key]); - }); - } - - function getDirectiveNames(type: VisitableSchemaType): Array { - let directives = (type.astNode?.directives ?? []).map((d) => d.name.value); - const extensionASTNodes = (type as { - extensionASTNodes?: Array; - }).extensionASTNodes; - if (extensionASTNodes != null) { - extensionASTNodes.forEach((extensionASTNode) => { - directives = directives.concat( - (extensionASTNode.directives ?? []).map((d) => d.name.value), - ); - }); - } - return directives; - } - - expect(getDirectiveNames(schema)).toEqual([ - 'schemaDirective', - 'schemaExtensionDirective', - ]); - - const queryType = schema.getQueryType() - assertGraphQLObjectType(queryType) - checkDirectives( - queryType, - ['queryTypeDirective', 'queryTypeExtensionDirective'], - { - people: ['queryFieldDirective'], - }, - ); - - const GenderType = schema.getType('Gender') - assertGraphQLEnumType(GenderType) - expect(getDirectiveNames(GenderType)).toEqual([ - 'enumTypeDirective', - 'enumTypeExtensionDirective', - ]); - - const nonBinary = (schema.getType( - 'Gender', - ) as GraphQLEnumType).getValues()[0]; - expect(getDirectiveNames(nonBinary)).toEqual(['enumValueDirective']); - - const DateType = schema.getType('Date') - assertGraphQLScalerType(DateType) - checkDirectives(DateType, [ - 'dateDirective', - 'dateExtensionDirective', - ]); - - const NamedType = schema.getType('Named') - assertGraphQLInterfaceType(NamedType) - checkDirectives( - NamedType, - ['interfaceDirective', 'interfaceExtensionDirective'], - { - name: ['interfaceFieldDirective'], - }, - ); - - const PersonInput = schema.getType('PersonInput') - assertGraphQLInputObjectType(PersonInput) - checkDirectives( - PersonInput, - ['inputTypeDirective', 'inputTypeExtensionDirective'], - { - name: ['inputFieldDirective'], - gender: [], - }, - ); - const MutationType = schema.getMutationType() - assertGraphQLObjectType(MutationType) - checkDirectives( - MutationType, - ['mutationTypeDirective', 'mutationTypeExtensionDirective'], - { - addPerson: ['mutationMethodDirective'], - }, - ); - expect( - getDirectiveNames(MutationType.getFields().addPerson.args[0]), - ).toEqual(['mutationArgumentDirective']); - - const PersonType = schema.getType('Person') - assertGraphQLObjectType(PersonType) - checkDirectives( - PersonType, - ['objectTypeDirective', 'objectTypeExtensionDirective'], - { - id: ['objectFieldDirective'], - name: [], - }, - ); - - const WhateverUnionType = schema.getType('WhateverUnion') - assertGraphQLUnionType(WhateverUnionType) - checkDirectives(WhateverUnionType, [ - 'unionDirective', - 'unionExtensionDirective', - ]); - }); - - test('works with enum and its resolvers', () => { - const schema = makeExecutableSchema({ - typeDefs: ` - enum DateFormat { - LOCAL - ISO - } - - directive @date(format: DateFormat) on FIELD_DEFINITION - - scalar Date - - type Query { - today: Date @date(format: LOCAL) - } - `, - resolvers: { - DateFormat: { - LOCAL: 'local', - ISO: 'iso', - }, - }, - }); - - expect(schema.getType('DateFormat')).toBeDefined(); - expect(schema.getDirective('date')).toBeDefined(); - }); - - test('can be implemented with SchemaDirectiveVisitor', () => { - const visited: Set = new Set(); - const schema = makeExecutableSchema({ typeDefs }); - - SchemaDirectiveVisitor.visitSchemaDirectives(schema, { - // The directive subclass can be defined anonymously inline! - queryTypeDirective: class extends SchemaDirectiveVisitor { - public static description = 'A @directive for query object types'; - public visitObject(object: GraphQLObjectType) { - expect(object).toBe(schema.getQueryType()); - visited.add(object); - } - }, - queryTypeExtensionDirective: class extends SchemaDirectiveVisitor { - public static description = 'A @directive for query object types'; - public visitObject(object: GraphQLObjectType) { - expect(object).toBe(schema.getQueryType()); - visited.add(object); - } - }, - }); - - expect(visited.size).toBe(1); - }); - - test('can visit the schema itself', () => { - const visited: Array = []; - const schema = makeExecutableSchema({ typeDefs }); - SchemaDirectiveVisitor.visitSchemaDirectives(schema, { - schemaDirective: class extends SchemaDirectiveVisitor { - public visitSchema(s: GraphQLSchema) { - visited.push(s); - } - }, - schemaExtensionDirective: class extends SchemaDirectiveVisitor { - public visitSchema(s: GraphQLSchema) { - visited.push(s); - } - }, - }); - expect(visited.length).toBe(2); - expect(visited[0]).toBe(schema); - expect(visited[1]).toBe(schema); - }); - - test('can visit fields within object types', () => { - const schema = makeExecutableSchema({ typeDefs }); - - let mutationObjectType: GraphQLObjectType; - let mutationField: GraphQLField; - let enumObjectType: GraphQLEnumType; - let inputObjectType: GraphQLInputObjectType; - - SchemaDirectiveVisitor.visitSchemaDirectives(schema, { - mutationTypeDirective: class extends SchemaDirectiveVisitor { - public visitObject(object: GraphQLObjectType) { - mutationObjectType = object; - expect(this.visitedType).toBe(object); - expect(object.name).toBe('Mutation'); - } - }, - - mutationTypeExtensionDirective: class extends SchemaDirectiveVisitor { - public visitObject(object: GraphQLObjectType) { - mutationObjectType = object; - expect(this.visitedType).toBe(object); - expect(object.name).toBe('Mutation'); - } - }, - - mutationMethodDirective: class extends SchemaDirectiveVisitor { - public visitFieldDefinition( - field: GraphQLField, - details: { - objectType: GraphQLObjectType; - }, - ) { - expect(this.visitedType).toBe(field); - expect(field.name).toBe('addPerson'); - expect(details.objectType).toBe(mutationObjectType); - expect(field.args.length).toBe(1); - mutationField = field; - } - }, - - mutationArgumentDirective: class extends SchemaDirectiveVisitor { - public visitArgumentDefinition( - arg: GraphQLArgument, - details: { - field: GraphQLField; - objectType: GraphQLObjectType; - }, - ) { - expect(this.visitedType).toBe(arg); - expect(arg.name).toBe('input'); - expect(details.field).toBe(mutationField); - expect(details.objectType).toBe(mutationObjectType); - expect(details.field.args[0]).toBe(arg); - } - }, - - enumTypeDirective: class extends SchemaDirectiveVisitor { - public visitEnum(enumType: GraphQLEnumType) { - expect(this.visitedType).toBe(enumType); - expect(enumType.name).toBe('Gender'); - enumObjectType = enumType; - } - }, - - enumTypeExtensionDirective: class extends SchemaDirectiveVisitor { - public visitEnum(enumType: GraphQLEnumType) { - expect(this.visitedType).toBe(enumType); - expect(enumType.name).toBe('Gender'); - enumObjectType = enumType; - } - }, - - enumValueDirective: class extends SchemaDirectiveVisitor { - public visitEnumValue( - value: GraphQLEnumValue, - details: { - enumType: GraphQLEnumType; - }, - ) { - expect(this.visitedType).toBe(value); - expect(value.name).toBe('NONBINARY'); - expect(value.value).toBe('NONBINARY'); - expect(details.enumType).toBe(enumObjectType); - } - }, - - inputTypeDirective: class extends SchemaDirectiveVisitor { - public visitInputObject(object: GraphQLInputObjectType) { - inputObjectType = object; - expect(this.visitedType).toBe(object); - expect(object.name).toBe('PersonInput'); - } - }, - - inputTypeExtensionDirective: class extends SchemaDirectiveVisitor { - public visitInputObject(object: GraphQLInputObjectType) { - inputObjectType = object; - expect(this.visitedType).toBe(object); - expect(object.name).toBe('PersonInput'); - } - }, - - inputFieldDirective: class extends SchemaDirectiveVisitor { - public visitInputFieldDefinition( - field: GraphQLInputField, - details: { - objectType: GraphQLInputObjectType; - }, - ) { - expect(this.visitedType).toBe(field); - expect(field.name).toBe('name'); - expect(details.objectType).toBe(inputObjectType); - } - }, - }); - }); - - test('can check if a visitor method is implemented', () => { - class Visitor extends SchemaVisitor { - // eslint-disable-next-line @typescript-eslint/no-empty-function - public notVisitorMethod() {} - - public visitObject(object: GraphQLObjectType) { - return object; - } - } - - expect(Visitor.implementsVisitorMethod('notVisitorMethod')).toBe(false); - - expect(Visitor.implementsVisitorMethod('visitObject')).toBe(true); - - expect(Visitor.implementsVisitorMethod('visitInputFieldDefinition')).toBe( - false, - ); - - expect(Visitor.implementsVisitorMethod('visitBogusType')).toBe(false); - }); - - test('can use visitSchema for simple visitor patterns', () => { - class SimpleVisitor extends SchemaVisitor { - public visitCount = 0; - public names: Array = []; - - constructor(s: GraphQLSchema) { - super(); - this.schema = s; - } - - public visit() { - // More complicated visitor implementations might use the - // visitorSelector function more selectively, but this SimpleVisitor - // class always volunteers itself to visit any schema type. - visitSchema(this.schema, () => [this]); - } - - public visitObject(object: GraphQLObjectType) { - expect(this.schema.getType(object.name)).toBe(object); - this.names.push(object.name); - } - } - - const schema = makeExecutableSchema({ typeDefs }); - const visitor = new SimpleVisitor(schema); - visitor.visit(); - expect(visitor.names.sort((a, b) => a.localeCompare(b))).toEqual([ - 'Mutation', - 'Person', - 'Query', - ]); - }); - - test('can use SchemaDirectiveVisitor as a no-op visitor', () => { - const schema = makeExecutableSchema({ typeDefs }); - const methodNamesEncountered = new Set(); - - class EnthusiasticVisitor extends SchemaDirectiveVisitor { - public static implementsVisitorMethod(name: string) { - // Pretend this class implements all visitor methods. This is safe - // because the SchemaVisitor base class provides empty stubs for all - // the visitor methods that might be called. - methodNamesEncountered.add(name); - return true; - } - } - - EnthusiasticVisitor.visitSchemaDirectives(schema, { - schemaDirective: EnthusiasticVisitor, - queryTypeDirective: EnthusiasticVisitor, - queryFieldDirective: EnthusiasticVisitor, - enumTypeDirective: EnthusiasticVisitor, - enumValueDirective: EnthusiasticVisitor, - dateDirective: EnthusiasticVisitor, - interfaceDirective: EnthusiasticVisitor, - interfaceFieldDirective: EnthusiasticVisitor, - inputTypeDirective: EnthusiasticVisitor, - inputFieldDirective: EnthusiasticVisitor, - mutationTypeDirective: EnthusiasticVisitor, - mutationArgumentDirective: EnthusiasticVisitor, - mutationMethodDirective: EnthusiasticVisitor, - objectTypeDirective: EnthusiasticVisitor, - objectFieldDirective: EnthusiasticVisitor, - unionDirective: EnthusiasticVisitor, - }); - - for (const methodName of methodNamesEncountered) { - expect(methodName in SchemaVisitor.prototype).toBeTruthy(); - } - }); - - test('can handle declared arguments', () => { - const schemaText = ` - directive @oyez( - times: Int = 5, - party: Party = IMPARTIAL, - ) on OBJECT | FIELD_DEFINITION - - schema { - query: Courtroom - } - - type Courtroom @oyez { - judge: String @oyez(times: 0) - marshall: String @oyez - } - - enum Party { - DEFENSE - PROSECUTION - IMPARTIAL - }`; - - const schema = makeExecutableSchema({ typeDefs: schemaText }); - const context = { - objectCount: 0, - fieldCount: 0, - }; - - const visitors = SchemaDirectiveVisitor.visitSchemaDirectives( - schema, - { - oyez: class extends SchemaDirectiveVisitor { - public static getDirectiveDeclaration( - name: string, - theSchema: GraphQLSchema, - ) { - expect(theSchema).toBe(schema); - const prev = schema.getDirective(name); - prev?.args.some((arg) => { - if (arg.name === 'times') { - // Override the default value of the times argument to be 3 - // instead of 5. - arg.defaultValue = 3; - return true; - } - return false; - }); - return prev; - } - - public visitObject() { - ++this.context.objectCount; - expect(this.args.times).toBe(3); - } - - public visitFieldDefinition(field: GraphQLField) { - ++this.context.fieldCount; - if (field.name === 'judge') { - expect(this.args.times).toBe(0); - } else if (field.name === 'marshall') { - expect(this.args.times).toBe(3); - } - expect(this.args.party).toBe('IMPARTIAL'); - } - }, - }, - context, - ); - - expect(context.objectCount).toBe(1); - expect(context.fieldCount).toBe(2); - - expect(Object.keys(visitors)).toEqual(['oyez']); - expect( - visitors.oyez.map( - (v) => - (v.visitedType as GraphQLObjectType | GraphQLField).name, - ), - ).toEqual(['Courtroom', 'judge', 'marshall']); - }); - - test('can be used to implement the @upper example', () => { - const schema = makeExecutableSchema({ - typeDefs: ` - directive @upper on FIELD_DEFINITION - - type Query { - hello: String @upper - }`, - schemaDirectives: { - upper: class extends SchemaDirectiveVisitor { - public visitFieldDefinition(field: GraphQLField) { - const { resolve = defaultFieldResolver } = field; - field.resolve = async function (...args) { - const result = await resolve.apply(this, args); - if (typeof result === 'string') { - return result.toUpperCase(); - } - return result; - }; - } - }, - }, - resolvers: { - Query: { - hello() { - return 'hello world'; - }, - }, - }, - }); - - return graphql( - schema, - ` - query { - hello - } - `, - ).then(({ data }) => { - expect(data).toEqual({ - hello: 'HELLO WORLD', - }); - }); - }); - - test('can be used to implement the @date example', () => { - const schema = makeExecutableSchema({ - typeDefs: ` - directive @date(format: String) on FIELD_DEFINITION - - scalar Date - - type Query { - today: Date @date(format: "mmmm d, yyyy") - }`, - - schemaDirectives: { - date: class extends SchemaDirectiveVisitor { - public visitFieldDefinition(field: GraphQLField) { - const { resolve = defaultFieldResolver } = field; - const { format } = this.args; - field.type = GraphQLString; - field.resolve = async function (...args) { - const date = await resolve.apply(this, args); - return formatDate(date, format, true); - }; - } - }, - }, - - resolvers: { - Query: { - today() { - return new Date(1519688273858).toUTCString(); - }, - }, - }, - }); - - return graphql( - schema, - ` - query { - today - } - `, - ).then(({ data }) => { - expect(data).toEqual({ - today: 'February 26, 2018', - }); - }); - }); - - test('can be used to implement the @date by adding an argument', async () => { - class FormattableDateDirective extends SchemaDirectiveVisitor { - public visitFieldDefinition(field: GraphQLField) { - const { resolve = defaultFieldResolver } = field; - const { defaultFormat } = this.args; - - field.args.push( - Object.create({ - name: 'format', - type: GraphQLString, - }), - ); - - field.type = GraphQLString; - field.resolve = async function ( - source, - { format, ...args }, - context, - info, - ) { - const newFormat = format || defaultFormat; - const date = await resolve.call(this, source, args, context, info); - return formatDate(date, newFormat, true); - }; - } - } - - const schema = makeExecutableSchema({ - typeDefs: ` - directive @date( - defaultFormat: String = "mmmm d, yyyy" - ) on FIELD_DEFINITION - - scalar Date - - type Query { - today: Date @date - }`, - - schemaDirectives: { - date: FormattableDateDirective, - }, - - resolvers: { - Query: { - today() { - return new Date(1521131357195); - }, - }, - }, - }); - - const resultNoArg = await graphql(schema, 'query { today }'); - - if (resultNoArg.errors != null) { - expect(resultNoArg.errors).toEqual([]); - } - - expect(resultNoArg.data).toEqual({ today: 'March 15, 2018' }); - - const resultWithArg = await graphql( - schema, - ` - query { - today(format: "dd mmm yyyy") - } - `, - ); - - if (resultWithArg.errors != null) { - expect(resultWithArg.errors).toEqual([]); - } - - expect(resultWithArg.data).toEqual({ today: '15 Mar 2018' }); - }); - - test('can be used to implement the @intl example', () => { - function translate(text: string, path: Array, locale: string) { - expect(text).toBe('hello'); - expect(path).toEqual(['Query', 'greeting']); - expect(locale).toBe('fr'); - return 'bonjour'; - } - - const context = { - locale: 'fr', - }; - - const schema = makeExecutableSchema({ - typeDefs: ` - directive @intl on FIELD_DEFINITION - - type Query { - greeting: String @intl - }`, - - schemaDirectives: { - intl: class extends SchemaDirectiveVisitor { - public visitFieldDefinition( - field: GraphQLField, - details: { - objectType: GraphQLObjectType; - }, - ) { - const { resolve = defaultFieldResolver } = field; - field.resolve = async function (...args: Parameters) { - const defaultText = await resolve.apply(this, args); - // In this example, path would be ["Query", "greeting"]: - const path = [details.objectType.name, field.name]; - expect(args[2]).toBe(context); - return translate(defaultText, path, context.locale); - }; - } - }, - }, - - resolvers: { - Query: { - greeting() { - return 'hello'; - }, - }, - }, - }); - - return graphql( - schema, - ` - query { - greeting - } - `, - null, - context, - ).then(({ data }) => { - expect(data).toEqual({ - greeting: 'bonjour', - }); - }); - }); - - test('can be used to implement the @auth example', async () => { - const roles = ['UNKNOWN', 'USER', 'REVIEWER', 'ADMIN']; - - function getUser(token: string) { - return { - hasRole(role: string) { - const tokenIndex = roles.indexOf(token); - const roleIndex = roles.indexOf(role); - return roleIndex >= 0 && tokenIndex >= roleIndex; - }, - }; - } - - class AuthDirective extends SchemaDirectiveVisitor { - public visitObject(type: GraphQLObjectType) { - this.ensureFieldsWrapped(type); - (type as any)._requiredAuthRole = this.args.requires; - } - - // Visitor methods for nested types like fields and arguments - // also receive a details object that provides information about - // the parent and grandparent types. - public visitFieldDefinition( - field: GraphQLField, - details: { objectType: GraphQLObjectType }, - ) { - this.ensureFieldsWrapped(details.objectType); - (field as any)._requiredAuthRole = this.args.requires; - } - - public ensureFieldsWrapped(objectType: GraphQLObjectType) { - // Mark the GraphQLObjectType object to avoid re-wrapping: - if ((objectType as any)._authFieldsWrapped) { - return; - } - (objectType as any)._authFieldsWrapped = true; - - const fields = objectType.getFields(); - - Object.keys(fields).forEach((fieldName) => { - const field = fields[fieldName]; - const { resolve = defaultFieldResolver } = field; - field.resolve = function (...args: Parameters) { - // Get the required Role from the field first, falling back - // to the objectType if no Role is required by the field: - const requiredRole = - (field as any)._requiredAuthRole || - (objectType as any)._requiredAuthRole; - - if (!requiredRole) { - return resolve.apply(this, args); - } - - const context = args[2]; - const user = getUser(context.headers.authToken); - if (!user.hasRole(requiredRole)) { - throw new Error('not authorized'); - } - - return resolve.apply(this, args); - }; - }); - } - } - - const schema = makeExecutableSchema({ - typeDefs: ` - directive @auth( - requires: Role = ADMIN, - ) on OBJECT | FIELD_DEFINITION - - enum Role { - ADMIN - REVIEWER - USER - UNKNOWN - } - - type User @auth(requires: USER) { - name: String - banned: Boolean @auth(requires: ADMIN) - canPost: Boolean @auth(requires: REVIEWER) - } - - type Query { - users: [User] - }`, - - schemaDirectives: { - auth: AuthDirective, - }, - - resolvers: { - Query: { - users() { - return [ - { - banned: true, - canPost: false, - name: 'Ben', - }, - ]; - }, - }, - }, - }); - - function execWithRole(role: string): Promise { - return graphql( - schema, - ` - query { - users { - name - banned - canPost - } - } - `, - null, - { - headers: { - authToken: role, - }, - }, - ); - } - - function assertStringArray(input: Array): asserts input is Array { - if (input.some(item => typeof item !== "string")) { - throw new Error("All items in array should be strings.") - } - } - - function checkErrors( - expectedCount: number, - ...expectedNames: Array - ) { - return function ({ - errors = [], - data, - }: ExecutionResult) { - expect(errors.length).toBe(expectedCount); - expect( - errors.every((error) => error.message === 'not authorized'), - ).toBeTruthy(); - const actualNames = errors.map((error) => error.path!.slice(-1)[0]); - assertStringArray(actualNames) - expect(expectedNames.sort((a, b) => a.localeCompare(b))).toEqual( - actualNames.sort((a, b) => a.localeCompare(b)), - ); - return data; - }; - } - - return Promise.all([ - execWithRole('UNKNOWN').then(checkErrors(3, 'banned', 'canPost', 'name')), - execWithRole('USER').then(checkErrors(2, 'banned', 'canPost')), - execWithRole('REVIEWER').then(checkErrors(1, 'banned')), - execWithRole('ADMIN') - .then(checkErrors(0)) - .then((data) => { - expect(data?.users.length).toBe(1); - expect(data?.users[0].banned).toBe(true); - expect(data?.users[0].canPost).toBe(false); - expect(data?.users[0].name).toBe('Ben'); - }), - ]); - }); - - test('can be used to implement the @length example', async () => { - class LimitedLengthType extends GraphQLScalarType { - constructor(type: GraphQLScalarType, maxLength: number) { - super({ - name: `LengthAtMost${maxLength.toString()}`, - - serialize(value: string) { - const newValue: string = type.serialize(value); - expect(typeof newValue.length).toBe('number'); - if (newValue.length > maxLength) { - throw new Error( - `expected ${newValue.length.toString( - 10, - )} to be at most ${maxLength.toString(10)}`, - ); - } - return newValue; - }, - - parseValue(value: string) { - return type.parseValue(value); - }, - - parseLiteral(ast) { - return type.parseLiteral(ast, {}); - }, - }); - } - } - - const schema = makeExecutableSchema({ - typeDefs: ` - directive @length(max: Int) on FIELD_DEFINITION | INPUT_FIELD_DEFINITION - - type Query { - books: [Book] - } - - type Book { - title: String @length(max: 10) - } - - type Mutation { - createBook(book: BookInput): Book - } - - input BookInput { - title: String! @length(max: 10) - }`, - - schemaDirectives: { - length: class extends SchemaDirectiveVisitor { - public visitInputFieldDefinition(field: GraphQLInputField) { - this.wrapType(field); - } - - public visitFieldDefinition(field: GraphQLField) { - this.wrapType(field); - } - - private wrapType(field: GraphQLInputField | GraphQLField) { - if (isNonNullType(field.type) && isScalarType(field.type.ofType)) { - field.type = new GraphQLNonNull( - new LimitedLengthType(field.type.ofType, this.args.max), - ); - } else if (isScalarType(field.type)) { - field.type = new LimitedLengthType(field.type, this.args.max); - } else { - throw new Error(`Not a scalar type: ${field.type.toString()}`); - } - } - }, - }, - - resolvers: { - Query: { - books() { - return [ - { - title: 'abcdefghijklmnopqrstuvwxyz', - }, - ]; - }, - }, - Mutation: { - createBook(_parent, args) { - return args.book; - }, - }, - }, - }); - - const { errors } = await graphql( - schema, - ` - query { - books { - title - } - } - `, - ); - expect(errors?.length).toBe(1); - expect(errors?.[0].message).toBe('expected 26 to be at most 10'); - - const result = await graphql( - schema, - ` - mutation { - createBook(book: { title: "safe title" }) { - title - } - } - `, - ); - - if (result.errors != null) { - expect(result.errors).toEqual([]); - } - - expect(result.data).toEqual({ - createBook: { - title: 'safe title', - }, - }); - }); - - test('can be used to implement the @uniqueID example', () => { - const schema = makeExecutableSchema({ - typeDefs: ` - directive @uniqueID(name: String, from: [String]) on OBJECT - - type Query { - people: [Person] - locations: [Location] - } - - type Person @uniqueID(name: "uid", from: ["personID"]) { - personID: Int - name: String - } - - type Location @uniqueID(name: "uid", from: ["locationID"]) { - locationID: Int - address: String - }`, - - schemaDirectives: { - uniqueID: class extends SchemaDirectiveVisitor { - public visitObject(type: GraphQLObjectType) { - const { name, from } = this.args; - type.getFields()[name] = Object.create({ - name, - type: GraphQLID, - description: 'Unique ID', - args: [], - resolve(object: any) { - const hash = createHash('sha1'); - hash.update(type.name); - from.forEach((fieldName: string) => { - hash.update(String(object[fieldName])); - }); - return hash.digest('hex'); - }, - }); - } - }, - }, - - resolvers: { - Query: { - people() { - return [ - { - personID: 1, - name: 'Ben', - }, - ]; - }, - locations() { - return [ - { - locationID: 1, - address: '140 10th St', - }, - ]; - }, - }, - }, - }); - - return graphql( - schema, - ` - query { - people { - uid - personID - name - } - locations { - uid - locationID - address - } - } - `, - null, - {}, - ).then((result) => { - const { data } = result; - - expect(data?.people).toEqual([ - { - uid: '580a207c8e94f03b93a2b01217c3cc218490571a', - personID: 1, - name: 'Ben', - }, - ]); - - expect(data?.locations).toEqual([ - { - uid: 'c31b71e6e23a7ae527f94341da333590dd7cba96', - locationID: 1, - address: '140 10th St', - }, - ]); - }); - }); - - test('automatically updates references to changed types', () => { - const schema = makeExecutableSchema({ - typeDefs, - schemaDirectives: { - objectTypeDirective: class extends SchemaDirectiveVisitor { - public visitObject(object: GraphQLObjectType) { - return Object.create(object, { - name: { value: 'Human' }, - }); - } - }, - }, - }); - - const Query = schema.getType('Query') as GraphQLObjectType; - const peopleType = Query.getFields().people.type; - if (isListType(peopleType)) { - expect(peopleType.ofType).toBe(schema.getType('Human')); - } else { - throw new Error('Query.people not a GraphQLList type'); - } - - const Mutation = schema.getType('Mutation') as GraphQLObjectType; - const addPersonResultType = Mutation.getFields().addPerson.type; - expect(addPersonResultType).toBe( - schema.getType('Human') as GraphQLOutputType, - ); - - const WhateverUnion = schema.getType('WhateverUnion') as GraphQLUnionType; - const found = WhateverUnion.getTypes().some((type) => { - if (type.name === 'Human') { - expect(type).toBe(schema.getType('Human')); - return true; - } - return false; - }); - expect(found).toBe(true); - - // Make sure that the Person type was actually removed. - expect(typeof schema.getType('Person')).toBe('undefined'); - }); - - test('can remove enum values', () => { - const schema = makeExecutableSchema({ - typeDefs: ` - directive @remove(if: Boolean) on ENUM_VALUE - - type Query { - age(unit: AgeUnit): Int - } - - enum AgeUnit { - DOG_YEARS - TURTLE_YEARS @remove(if: true) - PERSON_YEARS @remove(if: false) - }`, - - schemaDirectives: { - remove: class extends SchemaDirectiveVisitor { - public visitEnumValue(): any { - if (this.args.if) { - return null; - } - } - }, - }, - }); - - const AgeUnit = schema.getType('AgeUnit') as GraphQLEnumType; - expect(AgeUnit.getValues().map((value) => value.name)).toEqual([ - 'DOG_YEARS', - 'PERSON_YEARS', - ]); - }); - - test("can modify enum value's value", () => { - const schema = makeExecutableSchema({ - typeDefs: ` - directive @value(new: String!) on ENUM_VALUE - - type Query { - device: Device - } - - enum Device { - PHONE - TABLET - LAPTOP @value(new: "COMPUTER") - }`, - - schemaDirectives: { - value: class extends SchemaDirectiveVisitor { - public visitEnumValue(value: GraphQLEnumValue): GraphQLEnumValue { - return { - ...value, - value: this.args.new, - }; - } - }, - }, - }); - - const Device = schema.getType('Device') as GraphQLEnumType; - expect(Device.getValues().map((value) => value.value)).toEqual([ - 'PHONE', - 'TABLET', - 'COMPUTER', - ]); - }); - - test('can swap names of GraphQLNamedType objects', () => { - const schema = makeExecutableSchema({ - typeDefs: ` - directive @rename(to: String) on OBJECT - - type Query { - people: [Person] - } - - type Person @rename(to: "Human") { - heightInInches: Int - } - - scalar Date - - type Human @rename(to: "Person") { - born: Date - }`, - - schemaDirectives: { - rename: class extends SchemaDirectiveVisitor { - public visitObject(object: GraphQLObjectType) { - object.name = this.args.to; - } - }, - }, - }); - - const Human = schema.getType('Human') as GraphQLObjectType; - expect(Human.name).toBe('Human'); - expect(Human.getFields().heightInInches.type).toBe(GraphQLInt); - - const Person = schema.getType('Person') as GraphQLObjectType; - expect(Person.name).toBe('Person'); - expect(Person.getFields().born.type).toBe( - schema.getType('Date') as GraphQLScalarType, - ); - - const Query = schema.getType('Query') as GraphQLObjectType; - const peopleType = Query.getFields().people.type as GraphQLList< - GraphQLObjectType - >; - expect(peopleType.ofType).toBe(Human); - }); - - test('does not enforce query directive locations (issue #680)', () => { - const visited = new Set(); - makeExecutableSchema({ - typeDefs: ` - directive @hasScope(scope: [String]) on QUERY | FIELD | OBJECT - - type Query @hasScope { - oyez: String - }`, - - schemaDirectives: { - hasScope: class extends SchemaDirectiveVisitor { - public visitObject(object: GraphQLObjectType) { - expect(object.name).toBe('Query'); - visited.add(object); - } - }, - }, - }); - - expect(visited.size).toBe(1); - }); - - test('allows multiple directives when first replaces type (issue #851)', () => { - const schema = makeExecutableSchema({ - typeDefs: ` - directive @upper on FIELD_DEFINITION - directive @reverse on FIELD_DEFINITION - - type Query { - hello: String @upper @reverse - }`, - schemaDirectives: { - upper: class extends SchemaDirectiveVisitor { - public visitFieldDefinition(field: GraphQLField) { - const { resolve = defaultFieldResolver } = field; - const newField = { ...field }; - - newField.resolve = async function (...args: Parameters) { - const result = await resolve.apply(this, args); - if (typeof result === 'string') { - return result.toUpperCase(); - } - return result; - }; - - return newField; - } - }, - reverse: class extends SchemaDirectiveVisitor { - public visitFieldDefinition(field: GraphQLField) { - const { resolve = defaultFieldResolver } = field; - field.resolve = async function (...args: Parameters) { - const result = await resolve.apply(this, args); - if (typeof result === 'string') { - return result.split('').reverse().join(''); - } - return result; - }; - } - }, - }, - resolvers: { - Query: { - hello() { - return 'hello world'; - }, - }, - }, - }); - - return graphql( - schema, - ` - query { - hello - } - `, - ).then(({ data }) => { - expect(data).toEqual({ - hello: 'DLROW OLLEH', - }); - }); - }); - - test('preserves ability to create fields of different types with same name (issue 1462)', () => { - function validateStr(value: any, { - min = null, - message = null, - } : { - min: null | number, - message: null | string, - }) { - if(min && value.length < min) { - throw new Error(message || `Please ensure the value is at least ${min} characters.`); - } - } - - class ConstraintType extends GraphQLScalarType { - constructor( - type: GraphQLScalarType, - args: { - min: number, - message: string, - }, - ) { - super({ - name: 'ConstraintType', - serialize: (value) => type.serialize(value), - parseValue: (value) => { - const trimmed = value.trim(); - validateStr(trimmed, args); - return type.parseValue(trimmed); - } - }); - } - } - - class ConstraintDirective extends SchemaDirectiveVisitor { - visitInputFieldDefinition(field: GraphQLInputField) { - if (isNonNullType(field.type) && isScalarType(field.type.ofType)) { - field.type = new GraphQLNonNull( - new ConstraintType(field.type.ofType, this.args) - ); - } else if (isScalarType(field.type)) { - field.type = new ConstraintType(field.type, this.args); - } else { - throw new Error(`Not a scalar type: ${field.type}`); - } - } - } - - const schema = makeExecutableSchema({ - typeDefs: ` - directive @constraint(min: Int, message: String) on INPUT_FIELD_DEFINITION - - input BookInput { - name: String! @constraint(min: 10, message: "Book input error!") - } - - input AuthorInput { - name: String! @constraint(min: 4, message: "Author input error") - } - - type Query { - getBookById(id: Int): String - } - - type Mutation { - createBook(input: BookInput!): String - createAuthor(input: AuthorInput!): String - } - `, - resolvers: { - Mutation: { - createBook() { - return 'yes'; - }, - createAuthor() { - return 'no'; - } - } - }, - schemaDirectives: { - constraint: ConstraintDirective - } - }); - - return graphql( - schema, - ` - mutation { - createAuthor(input: { - name: "M" - }) - } - `, - ).then(({ errors }) => { - expect(errors?.[0].originalError).toEqual(new Error('Author input error')); - }); - }); - it('should print a directive correctly from GraphQLDirective object using astFromDirective and print', () => { - const sampleDirective = new GraphQLDirective({ - name: 'sample', - args: { - foo: { - type: GraphQLString - } - }, - locations: [DirectiveLocation.FIELD_DEFINITION] - }); - expect( - print( - astFromDirective(sampleDirective) - ) - ).toBeSimilarString(/* GraphQL */` - directive @sample(foo: String) on FIELD_DEFINITION - `); - }) -}); diff --git a/packages/utils/tests/get-directives.spec.ts b/packages/utils/tests/get-directives.spec.ts index d1ef5d501e6..19da8413343 100644 --- a/packages/utils/tests/get-directives.spec.ts +++ b/packages/utils/tests/get-directives.spec.ts @@ -1,6 +1,7 @@ import { makeExecutableSchema } from '@graphql-tools/schema'; import { getDirectives } from '../src'; import { assertGraphQLObjectType } from '../../testing/assertion'; +import { GraphQLSchema } from 'graphql'; describe('getDirectives', () => { it('should return the correct directives map when no directives specified', () => { @@ -9,10 +10,8 @@ describe('getDirectives', () => { test: String } `; - const schema = makeExecutableSchema({ typeDefs, resolvers: {}, allowUndefinedInResolve: true }); - const QueryType = schema.getQueryType() - assertGraphQLObjectType(QueryType) - const directivesMap = getDirectives(schema, QueryType); + const schema = makeExecutableSchema({ typeDefs, resolvers: {} }) as GraphQLSchema; + const directivesMap = getDirectives(schema, schema.getQueryType()!); expect(directivesMap).toEqual({}); }); @@ -24,10 +23,8 @@ describe('getDirectives', () => { } `; - const schema = makeExecutableSchema({ typeDefs, resolvers: {}, allowUndefinedInResolve: true }); - const QueryType = schema.getQueryType() - assertGraphQLObjectType(QueryType) - const directivesMap = getDirectives(schema, QueryType.getFields().test); + const schema = makeExecutableSchema({ typeDefs, resolvers: {} }) as GraphQLSchema; + const directivesMap = getDirectives(schema, schema.getQueryType()!.getFields()['test']); expect(directivesMap).toEqual({ deprecated: { reason: 'No longer supported', @@ -44,10 +41,8 @@ describe('getDirectives', () => { directive @mydir on FIELD_DEFINITION `; - const schema = makeExecutableSchema({ typeDefs, resolvers: {}, allowUndefinedInResolve: true }); - const QueryType = schema.getQueryType() - assertGraphQLObjectType(QueryType) - const directivesMap = getDirectives(schema, QueryType.getFields().test); + const schema = makeExecutableSchema({ typeDefs, resolvers: {} }) as GraphQLSchema; + const directivesMap = getDirectives(schema, schema.getQueryType()!.getFields()['test']); expect(directivesMap).toEqual({ mydir: {}, }); @@ -62,10 +57,8 @@ describe('getDirectives', () => { directive @mydir(f1: String) on FIELD_DEFINITION `; - const schema = makeExecutableSchema({ typeDefs, resolvers: {}, allowUndefinedInResolve: true }) - const QueryType = schema.getQueryType() - assertGraphQLObjectType(QueryType) - const directivesMap = getDirectives(schema, QueryType.getFields().test); + const schema = makeExecutableSchema({ typeDefs, resolvers: {} }) as GraphQLSchema; + const directivesMap = getDirectives(schema, schema.getQueryType()!.getFields()['test']); expect(directivesMap).toEqual({ mydir: { f1: 'test', @@ -82,25 +75,23 @@ describe('getDirectives', () => { directive @mydir(f1: String) on FIELD_DEFINITION `; - const schema = makeExecutableSchema({ typeDefs, resolvers: {}, allowUndefinedInResolve: true }) - const QueryType = schema.getQueryType() - assertGraphQLObjectType(QueryType) - const directivesMap = getDirectives(schema, QueryType.getFields().test); + const schema = makeExecutableSchema({ typeDefs, resolvers: {} }) as GraphQLSchema; + const directivesMap = getDirectives(schema, schema.getQueryType()!.getFields()['test']); expect(directivesMap).toEqual({ mydir: {}, }); }); - it('provides the extension definition over base', () => { + it('provides the extension definition', () => { const schema = makeExecutableSchema({ typeDefs: ` directive @mydir(arg: String) on OBJECT - extend type Query @mydir(arg: "ext1") { - second: String - } type Query @mydir(arg: "base") { first: String } + extend type Query @mydir(arg: "ext1") { + second: String + } ` }); const QueryType = schema.getQueryType() diff --git a/packages/utils/tests/mapSchema.test.ts b/packages/utils/tests/mapSchema.test.ts index 6054485db3c..e2622760e3e 100644 --- a/packages/utils/tests/mapSchema.test.ts +++ b/packages/utils/tests/mapSchema.test.ts @@ -29,7 +29,7 @@ describe('mapSchema', () => { const newSchema = mapSchema(schema, { [MapperKind.QUERY]: (type) => { const queryConfig = type.toConfig(); - queryConfig.fields.version.resolve = () => 1; + queryConfig.fields['version'].resolve = () => 1; return new GraphQLObjectType(queryConfig); }, }); @@ -37,7 +37,7 @@ describe('mapSchema', () => { expect(newSchema).toBeInstanceOf(GraphQLSchema); const result = graphqlSync(newSchema, '{ version }'); - expect(result.data?.version).toBe(1); + expect(result.data?.['version']).toBe(1); }); test('can change the root query name', () => { diff --git a/packages/utils/tests/print-schema-with-directives.spec.ts b/packages/utils/tests/print-schema-with-directives.spec.ts index 5eb8ec2a9c6..4108123390b 100644 --- a/packages/utils/tests/print-schema-with-directives.spec.ts +++ b/packages/utils/tests/print-schema-with-directives.spec.ts @@ -224,7 +224,7 @@ describe('printSchemaWithDirectives', () => { name: 'dummy', locations: ['QUERY'], })] - }); + } as any); const output = printSchemaWithDirectives(schema); diff --git a/packages/utils/tests/relocatedError.test.ts b/packages/utils/tests/relocatedError.test.ts index e5d5d427a02..9f4d3ab5204 100644 --- a/packages/utils/tests/relocatedError.test.ts +++ b/packages/utils/tests/relocatedError.test.ts @@ -4,11 +4,11 @@ import { relocatedError } from '../src/errors'; describe('Errors', () => { describe('relocatedError', () => { test('should adjust the path of a GraphqlError', () => { - const originalError = new GraphQLError('test', null, null, null, [ + const originalError = new GraphQLError('test', null as any, null, null, [ 'test', ]); const newError = relocatedError(originalError, ['test', 1]); - const expectedError = new GraphQLError('test', null, null, null, [ + const expectedError = new GraphQLError('test', null as any, null, null, [ 'test', 1, ]); diff --git a/packages/utils/tests/schemaTransforms.test.ts b/packages/utils/tests/schemaTransforms.test.ts index c0b18263499..451314b4a82 100644 --- a/packages/utils/tests/schemaTransforms.test.ts +++ b/packages/utils/tests/schemaTransforms.test.ts @@ -122,12 +122,12 @@ describe('@directives', () => { return schema => mapSchema(schema, { [MapperKind.OBJECT_TYPE]: type => { const directives = getDirectives(schema, type); - Object.keys(directives).forEach(directiveName => { + for (const directiveName in directives) { if (directiveNames.includes(directiveName)) { expect(type.name).toBe(schema.getQueryType()?.name); visited.add(type); } - }); + } return undefined; } }) @@ -149,11 +149,11 @@ describe('@directives', () => { function recordSchemaDirectiveUses(directiveNames: Array): (schema: GraphQLSchema) => GraphQLSchema { return schema => { const directives = getDirectives(schema, schema); - Object.keys(directives).forEach(directiveName => { + for (const directiveName in directives) { if (directiveNames.includes(directiveName)) { visited.push(schema); } - }); + } return schema; } } @@ -269,7 +269,7 @@ describe('@directives', () => { schemaTransforms: [deprecatedDirectiveTransformer], }); - expect((schema.getType('ExampleType') as GraphQLObjectType).getFields().oldField.deprecationReason).toBe('Use \`newField\`.') + expect((schema.getType('ExampleType') as GraphQLObjectType).getFields()['oldField'].deprecationReason).toBe('Use \`newField\`.') }); test('can be used to implement the @date example', () => { @@ -552,10 +552,10 @@ describe('@directives', () => { execWithRole('ADMIN') .then(checkErrors(0)) .then((data) => { - expect(data?.users.length).toBe(1); - expect(data?.users[0].banned).toBe(true); - expect(data?.users[0].canPost).toBe(false); - expect(data?.users[0].name).toBe('Ben'); + expect(data?.['users'].length).toBe(1); + expect(data?.['users'][0].banned).toBe(true); + expect(data?.['users'][0].canPost).toBe(false); + expect(data?.['users'][0].name).toBe('Ben'); }), ]); }); @@ -614,9 +614,9 @@ describe('@directives', () => { function wrapType | GraphQLInputFieldConfig>(fieldConfig: F, directiveArgumentMap: Record): void { if (isNonNullType(fieldConfig.type) && isScalarType(fieldConfig.type.ofType)) { - fieldConfig.type = getLimitedLengthType(fieldConfig.type.ofType, directiveArgumentMap.max); + fieldConfig.type = getLimitedLengthType(fieldConfig.type.ofType, directiveArgumentMap['max']); } else if (isScalarType(fieldConfig.type)) { - fieldConfig.type = getLimitedLengthType(fieldConfig.type, directiveArgumentMap.max); + fieldConfig.type = getLimitedLengthType(fieldConfig.type, directiveArgumentMap['max']); } else { throw new Error(`Not a scalar type: ${fieldConfig.type.toString()}`); } @@ -729,9 +729,9 @@ describe('@directives', () => { resolve(object: any) { const hash = createHash('sha1'); hash.update(type.name); - from.forEach((fieldName: string) => { + for (const fieldName of from ){ hash.update(String(object[fieldName])); - }); + } return hash.digest('hex'); }, }; @@ -805,7 +805,7 @@ describe('@directives', () => { ).then((result) => { const { data } = result; - expect(data?.people).toEqual([ + expect(data?.['people']).toEqual([ { uid: '580a207c8e94f03b93a2b01217c3cc218490571a', personID: 1, @@ -813,7 +813,7 @@ describe('@directives', () => { }, ]); - expect(data?.locations).toEqual([ + expect(data?.['locations']).toEqual([ { uid: 'c31b71e6e23a7ae527f94341da333590dd7cba96', locationID: 1, @@ -844,7 +844,7 @@ describe('@directives', () => { }); const Query = schema.getType('Query') as GraphQLObjectType; - const peopleType = Query.getFields().people.type; + const peopleType = Query.getFields()['people'].type; if (isListType(peopleType)) { expect(peopleType.ofType).toBe(schema.getType('Human')); } else { @@ -852,7 +852,7 @@ describe('@directives', () => { } const Mutation = schema.getType('Mutation') as GraphQLObjectType; - const addPersonResultType = Mutation.getFields().addPerson.type; + const addPersonResultType = Mutation.getFields()['addPerson'].type; expect(addPersonResultType).toBe( schema.getType('Human') as GraphQLOutputType, ); @@ -1023,16 +1023,16 @@ describe('@directives', () => { const Human = schema.getType('Human') as GraphQLObjectType; expect(Human.name).toBe('Human'); - expect(Human.getFields().heightInInches.type).toBe(GraphQLInt); + expect(Human.getFields()['heightInInches'].type).toBe(GraphQLInt); const Person = schema.getType('Person') as GraphQLObjectType; expect(Person.name).toBe('Person'); - expect(Person.getFields().born.type).toBe( + expect(Person.getFields()['born'].type).toBe( schema.getType('Date') as GraphQLScalarType, ); const Query = schema.getType('Query') as GraphQLObjectType; - const peopleType = Query.getFields().people.type as GraphQLList< + const peopleType = Query.getFields()['people'].type as GraphQLList< GraphQLObjectType >; expect(peopleType.ofType).toBe(Human); diff --git a/packages/utils/tests/validate-documents.spec.ts b/packages/utils/tests/validate-documents.spec.ts index 5f1b12e6169..62dff298b84 100644 --- a/packages/utils/tests/validate-documents.spec.ts +++ b/packages/utils/tests/validate-documents.spec.ts @@ -53,7 +53,8 @@ describe('validateGraphQlDocuments', () => { try { checkValidationErrors(result); expect(true).toBeFalsy(); - } catch (errors) { + } catch (aggregateError) { + const { errors } = aggregateError; expect(Symbol.iterator in errors).toBeTruthy(); const generator = errors[Symbol.iterator](); @@ -112,8 +113,8 @@ describe('checkValidationErrors', () => { let errors; try { checkValidationErrors(loadDocumentErrors as any); - } catch (_errors) { - errors = _errors; + } catch (aggregateError) { + errors = aggregateError.errors; } expect(Symbol.iterator in errors).toBeTruthy(); diff --git a/packages/webpack-loader/src/index.ts b/packages/webpack-loader/src/index.ts index 1bc6fc5e103..da45e5aa978 100644 --- a/packages/webpack-loader/src/index.ts +++ b/packages/webpack-loader/src/index.ts @@ -64,10 +64,10 @@ export default function graphqlLoader(this: LoaderContext, source: stri let stringifiedDoc = JSON.stringify(doc); if (options.replaceKinds) { - Object.keys(Kind).forEach(identifier => { + for (const identifier in Kind) { const value = Kind[identifier as keyof typeof Kind]; stringifiedDoc = stringifiedDoc.replace(new RegExp(`"kind":"${value}"`, 'g'), `"kind": Kind.${identifier}`); - }); + } } const headerCode = ` diff --git a/packages/wrap/src/generateProxyingResolvers.ts b/packages/wrap/src/generateProxyingResolvers.ts index c71dccb2705..e394509003d 100644 --- a/packages/wrap/src/generateProxyingResolvers.ts +++ b/packages/wrap/src/generateProxyingResolvers.ts @@ -27,15 +27,15 @@ export function generateProxyingResolvers( }; const resolvers = {}; - // @ts-expect-error: Object.keys typings suck. - Object.keys(operationTypes).forEach((operation: OperationTypeNode) => { + for (const operationAsString in operationTypes) { + const operation = operationAsString as OperationTypeNode; const rootType = operationTypes[operation]; if (rootType != null) { const typeName = rootType.name; const fields = rootType.getFields(); resolvers[typeName] = {}; - Object.keys(fields).forEach(fieldName => { + for (const fieldName in fields) { const proxyingResolver = createProxyingResolver({ subschemaConfig, transformedSchema, @@ -56,9 +56,9 @@ export function generateProxyingResolvers( resolve: finalResolver, }; } - }); + } } - }); + } return resolvers; } diff --git a/packages/wrap/src/index.ts b/packages/wrap/src/index.ts index 7be4ebcc7bf..317d012e34c 100644 --- a/packages/wrap/src/index.ts +++ b/packages/wrap/src/index.ts @@ -3,6 +3,5 @@ export { defaultCreateProxyingResolver, generateProxyingResolvers } from './gene export * from './transforms/index'; -export { makeRemoteExecutableSchema, defaultCreateRemoteResolver } from './makeRemoteExecutableSchema'; export * from './types'; export * from './introspect'; diff --git a/packages/wrap/src/introspect.ts b/packages/wrap/src/introspect.ts index 08ef51fdf9c..34e91c3f8cf 100644 --- a/packages/wrap/src/introspect.ts +++ b/packages/wrap/src/introspect.ts @@ -10,15 +10,14 @@ import { import { ValueOrPromise } from 'value-or-promise'; -import { AsyncExecutor, Executor, SyncExecutor, ExecutionResult } from '@graphql-tools/utils'; -import AggregateError from '@ardatan/aggregate-error'; +import { AsyncExecutor, Executor, SyncExecutor, ExecutionResult, AggregateError } from '@graphql-tools/utils'; function getSchemaFromIntrospection(introspectionResult: ExecutionResult): GraphQLSchema { if (introspectionResult?.data?.__schema) { return buildClientSchema(introspectionResult.data); } else if (introspectionResult?.errors?.length) { if (introspectionResult.errors.length > 1) { - const combinedError = new AggregateError(introspectionResult.errors); + const combinedError = new AggregateError(introspectionResult.errors, 'Could not obtain introspection result'); throw combinedError; } const error = introspectionResult.errors[0]; @@ -34,10 +33,14 @@ export function introspectSchema options?: IntrospectionOptions ): TExecutor extends AsyncExecutor ? Promise : GraphQLSchema { const parsedIntrospectionQuery: DocumentNode = parse(getIntrospectionQuery(options)); - return new ValueOrPromise(() => (executor as Executor)({ - document: parsedIntrospectionQuery, - context, - })).then(introspection => getSchemaFromIntrospection(introspection)).resolve() as any; + return new ValueOrPromise(() => + (executor as Executor)({ + document: parsedIntrospectionQuery, + context, + }) + ) + .then(introspection => getSchemaFromIntrospection(introspection)) + .resolve() as any; } // Keep for backwards compatibility. Will be removed on next release. diff --git a/packages/wrap/src/makeRemoteExecutableSchema.ts b/packages/wrap/src/makeRemoteExecutableSchema.ts deleted file mode 100644 index 7fcf10bd59f..00000000000 --- a/packages/wrap/src/makeRemoteExecutableSchema.ts +++ /dev/null @@ -1,35 +0,0 @@ -import { buildSchema, GraphQLFieldResolver, GraphQLSchema } from 'graphql'; - -import { IMakeRemoteExecutableSchemaOptions } from './types'; -import { delegateToSchema } from '@graphql-tools/delegate'; - -import { wrapSchema } from './wrapSchema'; -import { Executor, Subscriber } from '@graphql-tools/utils'; - -export function makeRemoteExecutableSchema({ - schema: schemaOrTypeDefs, - executor, - subscriber, - createResolver = defaultCreateRemoteResolver, - buildSchemaOptions, -}: IMakeRemoteExecutableSchemaOptions): GraphQLSchema { - const targetSchema = - typeof schemaOrTypeDefs === 'string' ? buildSchema(schemaOrTypeDefs, buildSchemaOptions) : schemaOrTypeDefs; - - return wrapSchema({ - schema: targetSchema, - createProxyingResolver: () => createResolver(executor, subscriber), - }); -} - -export function defaultCreateRemoteResolver( - executor: Executor, - subscriber?: Subscriber | undefined -): GraphQLFieldResolver { - return (_parent, _args, context, info) => - delegateToSchema({ - schema: { schema: info.schema, executor, subscriber }, - context, - info, - }); -} diff --git a/packages/wrap/src/transforms/HoistField.ts b/packages/wrap/src/transforms/HoistField.ts index b9463510028..4c686613016 100644 --- a/packages/wrap/src/transforms/HoistField.ts +++ b/packages/wrap/src/transforms/HoistField.ts @@ -80,12 +80,12 @@ export default class HoistField implements Transform { const argsMap: Record = Object.create(null); const innerType: GraphQLObjectType = this.pathToField.reduce((acc, pathSegment, index) => { const field = acc.getFields()[pathSegment]; - field.args.forEach(arg => { + for (const arg of field.args) { if (this.argFilters[index](arg)) { argsMap[arg.name] = arg; this.argLevels[arg.name] = index; } - }); + } return getNullableType(field.type) as GraphQLObjectType; }, originalWrappingSchema.getType(this.typeName) as GraphQLObjectType); diff --git a/packages/wrap/src/transforms/MapLeafValues.ts b/packages/wrap/src/transforms/MapLeafValues.ts index 028736d6605..cbd4c0141ea 100644 --- a/packages/wrap/src/transforms/MapLeafValues.ts +++ b/packages/wrap/src/transforms/MapLeafValues.ts @@ -8,7 +8,6 @@ import { FragmentDefinitionNode, VariableDefinitionNode, ArgumentNode, - GraphQLArgument, FieldNode, valueFromAST, isLeafType, @@ -62,14 +61,14 @@ export default class MapLeafValues implements Transform { + for (const typeName in typeMap) { const type = typeMap[typeName]; if (!typeName.startsWith('__')) { if (isLeafType(type)) { this.resultVisitorMap[typeName] = (value: any) => this.outputValueTransformer(typeName, value); } } - }); + } this.typeInfo = new TypeInfo(originalWrappingSchema); return originalWrappingSchema; } @@ -142,7 +141,7 @@ export default class MapLeafValues implements Transform variableDefinitionMap[varName]), + variableDefinitions: Object.values(variableDefinitionMap), }; }); } @@ -165,7 +164,7 @@ export default class MapLeafValues implements Transform { + for (const argument of targetField.args) { const argName = argument.name; const argType = argument.type; @@ -188,11 +187,11 @@ export default class MapLeafValues implements Transform argumentNodeMap[argName]), + arguments: Object.values(argumentNodeMap), }; } } diff --git a/packages/wrap/src/transforms/TransformCompositeFields.ts b/packages/wrap/src/transforms/TransformCompositeFields.ts index c8feffe87e2..127fa9db5e9 100644 --- a/packages/wrap/src/transforms/TransformCompositeFields.ts +++ b/packages/wrap/src/transforms/TransformCompositeFields.ts @@ -79,11 +79,11 @@ export default class TransformCompositeFields> im ): Request { const document = originalRequest.document; const fragments = Object.create(null); - document.definitions.forEach(def => { + for (const def of document.definitions) { if (def.kind === Kind.FRAGMENT_DEFINITION) { fragments[def.name.value] = def; } - }); + } return { ...originalRequest, document: this.transformDocument(document, fragments, transformationContext), @@ -135,10 +135,10 @@ export default class TransformCompositeFields> im const parentTypeName = parentType.name; let newSelections: Array = []; - node.selections.forEach(selection => { + for (const selection of node.selections) { if (selection.kind !== Kind.FIELD) { newSelections.push(selection); - return; + continue; } const newName = selection.name.value; @@ -172,25 +172,25 @@ export default class TransformCompositeFields> im } if (transformedSelection == null) { - return; + continue; } else if (Array.isArray(transformedSelection)) { newSelections = newSelections.concat(transformedSelection); - return; + continue; } else if (transformedSelection.kind !== Kind.FIELD) { newSelections.push(transformedSelection); - return; + continue; } const typeMapping = this.mapping[parentTypeName]; if (typeMapping == null) { newSelections.push(transformedSelection); - return; + continue; } const oldName = this.mapping[parentTypeName][newName]; if (oldName == null) { newSelections.push(transformedSelection); - return; + continue; } newSelections.push({ @@ -204,7 +204,7 @@ export default class TransformCompositeFields> im value: transformedSelection.alias?.value ?? newName, }, }); - }); + } return { ...node, diff --git a/packages/wrap/src/transforms/TransformInputObjectFields.ts b/packages/wrap/src/transforms/TransformInputObjectFields.ts index 9adeb7cda0d..8f15bc67dd2 100644 --- a/packages/wrap/src/transforms/TransformInputObjectFields.ts +++ b/packages/wrap/src/transforms/TransformInputObjectFields.ts @@ -8,11 +8,10 @@ import { Kind, FragmentDefinitionNode, GraphQLInputObjectType, - GraphQLInputType, ObjectValueNode, ObjectFieldNode, OperationDefinitionNode, - NamedTypeNode, + isInputType, } from 'graphql'; import { Maybe, Request, MapperKind, mapSchema, transformInputValue, assertSome } from '@graphql-tools/utils'; @@ -79,24 +78,27 @@ export default class TransformInputObjectFields implements Transform { const operations: Array = []; - originalRequest.document.definitions.forEach(def => { - if ((def as OperationDefinitionNode).kind === Kind.OPERATION_DEFINITION) { - operations.push(def as OperationDefinitionNode); - } else { - fragments[(def as FragmentDefinitionNode).name.value] = def; + for (const def of originalRequest.document.definitions) { + if (def.kind === Kind.OPERATION_DEFINITION) { + operations.push(def); + } else if (def.kind === Kind.FRAGMENT_DEFINITION) { + fragments[def.name.value] = def; } - }); + } - operations.forEach(def => { + for (const def of operations) { const variableDefs = def.variableDefinitions; if (variableDefs != null) { - variableDefs.forEach(variableDef => { + for (const variableDef of variableDefs) { const varName = variableDef.variable.name.value; + if (variableDef.type.kind !== Kind.NAMED_TYPE) { + throw new Error(`Expected ${variableDef.type} to be a named type`); + } // requirement for 'as NamedTypeNode' appears to be a bug within types, as function should take any TypeNode - const varType = typeFromAST( - delegationContext.transformedSchema, - variableDef.type as NamedTypeNode - ) as GraphQLInputType; + const varType = typeFromAST(delegationContext.transformedSchema, variableDef.type); + if (!isInputType(varType)) { + throw new Error(`Expected ${varType} to be an input type`); + } variableValues[varName] = transformInputValue( varType, variableValues[varName], @@ -104,7 +106,7 @@ export default class TransformInputObjectFields implements Transform { (type, originalValue) => { const newValue = Object.create(null); const fields = type.getFields(); - Object.keys(originalValue).forEach(key => { + for (const key in originalValue) { const field = fields[key]; if (field != null) { const newFieldName = this.mapping[type.name]?.[field.name]; @@ -114,19 +116,17 @@ export default class TransformInputObjectFields implements Transform { newValue[field.name] = originalValue[field.name]; } } - }); + } return newValue; } ); - }); + } } - }); + } - originalRequest.document.definitions - .filter(def => def.kind === Kind.FRAGMENT_DEFINITION) - .forEach(def => { - fragments[(def as FragmentDefinitionNode).name.value] = def; - }); + for (const def of originalRequest.document.definitions.filter(def => def.kind === Kind.FRAGMENT_DEFINITION)) { + fragments[(def as FragmentDefinitionNode).name.value] = def; + } const document = this.transformDocument( originalRequest.document, this.mapping, @@ -162,7 +162,7 @@ export default class TransformInputObjectFields implements Transform { const parentTypeName = parentType.name; const newInputFields: Array = []; - node.fields.forEach(inputField => { + for (const inputField of node.fields) { const newName = inputField.name.value; const transformedInputField = @@ -171,17 +171,17 @@ export default class TransformInputObjectFields implements Transform { : inputField; if (Array.isArray(transformedInputField)) { - transformedInputField.forEach(individualTransformedInputField => { + for (const individualTransformedInputField of transformedInputField) { const typeMapping = mapping[parentTypeName]; if (typeMapping == null) { newInputFields.push(individualTransformedInputField); - return; + continue; } const oldName = typeMapping[newName]; if (oldName == null) { newInputFields.push(individualTransformedInputField); - return; + continue; } newInputFields.push({ @@ -191,20 +191,20 @@ export default class TransformInputObjectFields implements Transform { value: oldName, }, }); - }); - return; + } + continue; } const typeMapping = mapping[parentTypeName]; if (typeMapping == null) { newInputFields.push(transformedInputField); - return; + continue; } const oldName = typeMapping[newName]; if (oldName == null) { newInputFields.push(transformedInputField); - return; + continue; } newInputFields.push({ @@ -214,7 +214,7 @@ export default class TransformInputObjectFields implements Transform { value: oldName, }, }); - }); + } const newNode = { ...node, diff --git a/packages/wrap/src/transforms/WrapFields.ts b/packages/wrap/src/transforms/WrapFields.ts index 817e692fea8..2ae45201e1c 100644 --- a/packages/wrap/src/transforms/WrapFields.ts +++ b/packages/wrap/src/transforms/WrapFields.ts @@ -89,14 +89,14 @@ export default class WrapFields implements Transform = Object.create(null); - Object.keys(targetFieldConfigMap).forEach(fieldName => { + for (const fieldName in targetFieldConfigMap) { const field = targetFieldConfigMap[fieldName]; const newField: GraphQLFieldConfig = { ...field, resolve: defaultMergedResolver, }; newTargetFieldConfigMap[fieldName] = newField; - }); + } let wrapIndex = this.numWraps - 1; let wrappingTypeName = this.wrappingTypeNames[wrapIndex]; @@ -181,7 +181,7 @@ function collectFields( visitedFragmentNames = {} ): Array { if (selectionSet != null) { - selectionSet.selections.forEach(selection => { + for (const selection of selectionSet.selections) { switch (selection.kind) { case Kind.FIELD: fields.push(selection); @@ -201,7 +201,7 @@ function collectFields( // unreachable break; } - }); + } } return fields; @@ -242,7 +242,7 @@ function hoistFieldNodes({ if (index < path.length) { const pathSegment = path[index]; - collectFields(fieldNode.selectionSet, fragments).forEach((possibleFieldNode: FieldNode) => { + for (const possibleFieldNode of collectFields(fieldNode.selectionSet, fragments)) { if (possibleFieldNode.name.value === pathSegment) { const newWrappingPath = wrappingPath.concat([alias]); @@ -259,9 +259,9 @@ function hoistFieldNodes({ }) ); } - }); + } } else { - collectFields(fieldNode.selectionSet, fragments).forEach((possibleFieldNode: FieldNode) => { + for (const possibleFieldNode of collectFields(fieldNode.selectionSet, fragments)) { if (!fieldNames || fieldNames.includes(possibleFieldNode.name.value)) { const nextIndex = transformationContext.nextIndex; transformationContext.nextIndex++; @@ -272,7 +272,7 @@ function hoistFieldNodes({ }; newFieldNodes.push(aliasFieldNode(possibleFieldNode, indexingAlias)); } - }); + } } return newFieldNodes; @@ -285,22 +285,22 @@ export function dehoistValue(originalValue: any, context: WrapFieldsTransformati const newValue = Object.create(null); - Object.keys(originalValue).forEach(alias => { + for (const alias in originalValue) { let obj = newValue; const path = context.paths[alias]; if (path == null) { newValue[alias] = originalValue[alias]; - return; + continue; } const pathToField = path.pathToField; const fieldAlias = path.alias; - pathToField.forEach(key => { + for (const key of pathToField) { obj = obj[key] = obj[key] || Object.create(null); - }); + } obj[fieldAlias] = originalValue[alias]; - }); + } return newValue; } @@ -320,20 +320,20 @@ function dehoistErrors( } let newPath: Array = []; - originalPath.forEach(pathSegment => { + for (const pathSegment of originalPath) { if (typeof pathSegment !== 'string') { newPath.push(pathSegment); - return; + continue; } const path = context.paths[pathSegment]; if (path == null) { newPath.push(pathSegment); - return; + continue; } newPath = newPath.concat(path.pathToField, [path.alias]); - }); + } return relocatedError(error, newPath); }); diff --git a/packages/wrap/src/types.ts b/packages/wrap/src/types.ts index 3eb9c7ef05e..2ff018fd6c9 100644 --- a/packages/wrap/src/types.ts +++ b/packages/wrap/src/types.ts @@ -1,7 +1,4 @@ import { - GraphQLSchema, - GraphQLFieldResolver, - BuildSchemaOptions, GraphQLInputFieldConfig, GraphQLFieldConfig, FieldNode, @@ -13,18 +10,7 @@ import { GraphQLEnumValueConfig, } from 'graphql'; import { DelegationContext } from '@graphql-tools/delegate'; -import { Executor, Subscriber, Request, Maybe } from '@graphql-tools/utils'; - -export interface IMakeRemoteExecutableSchemaOptions> { - schema: GraphQLSchema | string; - executor: Executor; - subscriber?: Subscriber; - createResolver?: ( - executor: Executor, - subscriber?: Subscriber | undefined - ) => GraphQLFieldResolver; - buildSchemaOptions?: BuildSchemaOptions; -} +import { Request, Maybe } from '@graphql-tools/utils'; export type InputFieldTransformer = ( typeName: string, diff --git a/packages/wrap/src/wrapSchema.ts b/packages/wrap/src/wrapSchema.ts index 6d7f9d1ed9c..2532cd8611c 100644 --- a/packages/wrap/src/wrapSchema.ts +++ b/packages/wrap/src/wrapSchema.ts @@ -33,16 +33,16 @@ function createWrappingSchema( const config = type.toConfig(); const fieldConfigMap = config.fields; - Object.keys(fieldConfigMap).forEach(fieldName => { + for (const fieldName in fieldConfigMap) { const field = fieldConfigMap[fieldName]; if (field == null) { - return; + continue; } fieldConfigMap[fieldName] = { ...field, ...proxyingResolvers[type.name]?.[fieldName], }; - }); + } return new GraphQLObjectType(config); }, @@ -50,14 +50,14 @@ function createWrappingSchema( const config = type.toConfig(); config.isTypeOf = undefined; - Object.keys(config.fields).forEach(fieldName => { + for (const fieldName in config.fields) { const field = config.fields[fieldName]; if (field == null) { - return; + continue; } field.resolve = defaultMergedResolver; field.subscribe = undefined; - }); + } return new GraphQLObjectType(config); }, diff --git a/packages/wrap/tests/fixtures/schemas.ts b/packages/wrap/tests/fixtures/schemas.ts index 8c405f4dfa0..b29511136b2 100644 --- a/packages/wrap/tests/fixtures/schemas.ts +++ b/packages/wrap/tests/fixtures/schemas.ts @@ -12,8 +12,6 @@ import { GraphQLInterfaceType, } from 'graphql'; -import isPromise from 'is-promise'; - import { ValueOrPromise } from 'value-or-promise'; import { introspectSchema } from '../../src/introspect'; @@ -21,6 +19,7 @@ import { IResolvers, ExecutionResult, mapAsyncIterator, + isAsyncIterable, } from '@graphql-tools/utils'; import { makeExecutableSchema } from '@graphql-tools/schema'; import { SubschemaConfig, ExecutionParams } from '@graphql-tools/delegate'; @@ -188,10 +187,6 @@ export const sampleData: { }, }; -function values(o: { [s: string]: T }): Array { - return Object.keys(o).map((k) => o[k]); -} - function coerceString(value: any): string { if (Array.isArray(value)) { throw new TypeError( @@ -225,9 +220,9 @@ function parseLiteral(ast: ValueNode): any { return parseFloat(ast.value); case Kind.OBJECT: { const value = Object.create(null); - ast.fields.forEach((field) => { + for (const field of ast.fields) { value[field.name.value] = parseLiteral(field.value); - }); + } return value; } @@ -349,7 +344,7 @@ const propertyResolvers: IResolvers = { }, properties(_root, { limit }) { - const list = values(sampleData.Property); + const list = Object.values(sampleData.Property); return limit ? list.slice(0, limit) : list; }, @@ -464,7 +459,7 @@ const productTypeDefs = ` const productResolvers: IResolvers = { Query: { products(_root) { - const list = values(sampleData.Product); + const list = Object.values(sampleData.Product); return list; }, }, @@ -550,7 +545,7 @@ const bookingResolvers: IResolvers = { return sampleData.Booking[id]; }, bookingsByPropertyId(_parent, { propertyId, limit }) { - const list = values(sampleData.Booking).filter( + const list = Object.values(sampleData.Booking).filter( (booking: Booking) => booking.propertyId === propertyId, ); return limit ? list.slice(0, limit) : list; @@ -559,11 +554,11 @@ const bookingResolvers: IResolvers = { return sampleData.Customer[id]; }, bookings(_parent, { limit }) { - const list = values(sampleData.Booking); + const list = Object.values(sampleData.Booking); return limit ? list.slice(0, limit) : list; }, customers(_parent, { limit }) { - const list = values(sampleData.Customer); + const list = Object.values(sampleData.Customer); return limit ? list.slice(0, limit) : list; }, }, @@ -600,13 +595,13 @@ const bookingResolvers: IResolvers = { Customer: { bookings(parent: Customer, { limit }) { - const list = values(sampleData.Booking).filter( + const list = Object.values(sampleData.Booking).filter( (booking: Booking) => booking.customerId === parent.id, ); return limit ? list.slice(0, limit) : list; }, vehicle(parent: Customer) { - return sampleData.Vehicle[parent.vehicleId]; + return parent.vehicleId && sampleData.Vehicle[parent.vehicleId]; }, error() { throw new Error('Customer.error error'); @@ -699,16 +694,15 @@ function makeExecutorFromSchema(schema: GraphQLSchema) { function makeSubscriberFromSchema(schema: GraphQLSchema) { return async ({ document, variables, context }: ExecutionParams) => { - const result = subscribe( + const result = await subscribe( schema, document, null, context, variables, - ) as Promise> | ExecutionResult>; - if (isPromise(result)) { - return result.then(asyncIterator => - mapAsyncIterator(asyncIterator as AsyncIterator, (originalResult: ExecutionResult) => JSON.parse(JSON.stringify(originalResult)))); + ); + if (isAsyncIterable>(result)) { + return mapAsyncIterator, TReturn>(result, (originalResult: ExecutionResult) => JSON.parse(JSON.stringify(originalResult))); } return JSON.parse(JSON.stringify(result)); }; diff --git a/packages/wrap/tests/fragmentsAreNotDuplicated.test.ts b/packages/wrap/tests/fragmentsAreNotDuplicated.test.ts index 02df525b46a..7cc4a106df4 100644 --- a/packages/wrap/tests/fragmentsAreNotDuplicated.test.ts +++ b/packages/wrap/tests/fragmentsAreNotDuplicated.test.ts @@ -83,6 +83,8 @@ const variables = { function assertNoDuplicateFragmentErrors(result: ExecutionResult) { // Run assertion against each array element for better test failure output. if (result.errors != null) { - result.errors.forEach((error) => expect(error.message).toBe('')); + for (const error of result.errors) { + expect(error.message).toBe('') + } } } diff --git a/packages/wrap/tests/makeRemoteExecutableSchema.test.ts b/packages/wrap/tests/makeRemoteExecutableSchema.test.ts index 554d656108b..fbfeeb7669a 100644 --- a/packages/wrap/tests/makeRemoteExecutableSchema.test.ts +++ b/packages/wrap/tests/makeRemoteExecutableSchema.test.ts @@ -6,9 +6,10 @@ import { execute, print, ExecutionResult, + buildSchema, } from 'graphql'; -import { makeRemoteExecutableSchema } from '../src/index'; +import { wrapSchema } from '../src/index'; import { propertySchema, @@ -22,7 +23,7 @@ describe('remote queries', () => { let schema: GraphQLSchema; beforeAll(async () => { const remoteSubschemaConfig = await makeSchemaRemote(propertySchema); - schema = makeRemoteExecutableSchema(remoteSubschemaConfig); + schema = wrapSchema(remoteSubschemaConfig); }); test('should work', async () => { @@ -60,7 +61,7 @@ describe('remote subscriptions', () => { let schema: GraphQLSchema; beforeAll(async () => { const remoteSubschemaConfig = await makeSchemaRemote(subscriptionSchema); - schema = makeRemoteExecutableSchema(remoteSubschemaConfig); + schema = wrapSchema(remoteSubschemaConfig); }); test('should work', async () => { @@ -133,7 +134,7 @@ describe('remote subscriptions', () => { }); describe('respects buildSchema options', () => { - const schema = ` + const typeDefs = /* GraphQL */` type Query { # Field description custom: CustomScalar! @@ -144,26 +145,25 @@ describe('respects buildSchema options', () => { `; test('without comment descriptions', () => { - const remoteSchema = makeRemoteExecutableSchema({ schema }); + const remoteSchema = wrapSchema({ schema: buildSchema(typeDefs) }); const customScalar = remoteSchema.getType('CustomScalar'); - expect(customScalar.description).toBeUndefined(); + expect(customScalar?.description).toBeUndefined(); }); test('with comment descriptions', () => { - const remoteSchema = makeRemoteExecutableSchema({ - schema, - buildSchemaOptions: { commentDescriptions: true }, + const remoteSchema = wrapSchema({ + schema: buildSchema(typeDefs, { commentDescriptions: true }), }); - const field = remoteSchema.getQueryType().getFields()['custom']; + const field = remoteSchema.getQueryType()!.getFields()['custom']; expect(field.description).toBe('Field description'); const customScalar = remoteSchema.getType('CustomScalar'); - expect(customScalar.description).toBe('Scalar description'); + expect(customScalar?.description).toBe('Scalar description'); }); describe('when query for multiple fields', () => { - const schema = ` + const typeDefs = ` type Query { fieldA: Int! fieldB: Int! @@ -188,8 +188,8 @@ describe('respects buildSchema options', () => { }, }); }; - const remoteSchema = makeRemoteExecutableSchema({ - schema, + const remoteSchema = wrapSchema({ + schema: buildSchema(typeDefs), executor, }); diff --git a/packages/wrap/tests/requests.test.ts b/packages/wrap/tests/requests.test.ts index f38108cabba..a4f308c2116 100644 --- a/packages/wrap/tests/requests.test.ts +++ b/packages/wrap/tests/requests.test.ts @@ -12,11 +12,11 @@ function removeLocations(value: any): any { return value.map((v) => removeLocations(v)); } else if (typeof value === 'object') { const newValue = {}; - Object.keys(value).forEach((key) => { + for (const key in value) { if (key !== 'loc') { newValue[key] = removeLocations(value[key]); } - }); + } return newValue; } @@ -34,12 +34,13 @@ describe('requests', () => { minor patch }`), + targetOperationName: 'test' }), ); const expectedRequest = removeLocations({ document: parse(` - query { + query test { version { major minor @@ -48,6 +49,7 @@ describe('requests', () => { } `), variables: {}, + operationName: 'test' }); expect(expectedRequest).toMatchObject(request); diff --git a/packages/wrap/tests/transformFilterInputObjectFields.test.ts b/packages/wrap/tests/transformFilterInputObjectFields.test.ts index 810d9b60858..9bd1e99a415 100644 --- a/packages/wrap/tests/transformFilterInputObjectFields.test.ts +++ b/packages/wrap/tests/transformFilterInputObjectFields.test.ts @@ -67,7 +67,7 @@ describe('FilterInputObjectFields', () => { const result = await graphql(transformedSchema, query); assertSome(result.data) - expect(result.data.test.field1).toBe('field1'); - expect(result.data.test.field2).toBe('field2'); + expect(result.data['test'].field1).toBe('field1'); + expect(result.data['test'].field2).toBe('field2'); }); }); diff --git a/packages/wrap/tests/transformFilterObjectFieldDirectives.test.ts b/packages/wrap/tests/transformFilterObjectFieldDirectives.test.ts index 733cf13b34d..d122d815e32 100644 --- a/packages/wrap/tests/transformFilterObjectFieldDirectives.test.ts +++ b/packages/wrap/tests/transformFilterObjectFieldDirectives.test.ts @@ -23,10 +23,10 @@ describe('FilterObjectFieldDirectives', () => { ], }); - const fields = transformedSchema.getType('Query').getFields(); - expect(fields.alpha.astNode.directives.length).toEqual(0); - expect(fields.bravo.astNode.directives.length).toEqual(1); - expect(fields.charlie.astNode.directives.length).toEqual(0); - expect(fields.delta.astNode.directives.length).toEqual(1); + const fields = transformedSchema.getQueryType()?.getFields(); + expect(fields?.['alpha']?.astNode?.directives?.length).toEqual(0); + expect(fields?.['bravo']?.astNode?.directives?.length).toEqual(1); + expect(fields?.['charlie']?.astNode?.directives?.length).toEqual(0); + expect(fields?.['delta']?.astNode?.directives?.length).toEqual(1); }); }); diff --git a/packages/wrap/tests/transformMapLeafValues.test.ts b/packages/wrap/tests/transformMapLeafValues.test.ts index 09650ac9cbb..21712a5f4b8 100644 --- a/packages/wrap/tests/transformMapLeafValues.test.ts +++ b/packages/wrap/tests/transformMapLeafValues.test.ts @@ -50,7 +50,7 @@ describe('MapLeafValues', () => { const result = await graphql(transformedSchema, query); assertSome(result.data) - expect(result.data.testEnum).toBe('THREE'); - expect(result.data.testScalar).toBe(15); + expect(result.data['testEnum']).toBe('THREE'); + expect(result.data['testScalar']).toBe(15); }); }); diff --git a/packages/wrap/tests/transformRemoveObjectFieldDeprecations.test.ts b/packages/wrap/tests/transformRemoveObjectFieldDeprecations.test.ts index f4b06c18ad9..888a87da065 100644 --- a/packages/wrap/tests/transformRemoveObjectFieldDeprecations.test.ts +++ b/packages/wrap/tests/transformRemoveObjectFieldDeprecations.test.ts @@ -25,12 +25,12 @@ describe('RemoveObjectFieldDeprecations', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - assertSome(fields.first) - expect(fields.first.deprecationReason).toEqual('do not remove'); - assertSome(fields.second) - expect(fields.second.deprecationReason).toBeUndefined(); - expect(fields.first.astNode?.directives?.length).toEqual(1); - expect(fields.second.astNode?.directives?.length).toEqual(0); + assertSome(fields['first']) + expect(fields['first'].deprecationReason).toEqual('do not remove'); + assertSome(fields['second']) + expect(fields['second'].deprecationReason).toBeUndefined(); + expect(fields['first'].astNode?.directives?.length).toEqual(1); + expect(fields['second'].astNode?.directives?.length).toEqual(0); }); test('removes deprecations by reason regex', async () => { @@ -44,11 +44,11 @@ describe('RemoveObjectFieldDeprecations', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - assertSome(fields.first) - expect(fields.first.deprecationReason).toBeUndefined(); - assertSome(fields.second) - expect(fields.second.deprecationReason).toBeUndefined(); - expect(fields.first.astNode?.directives?.length).toEqual(0); - expect(fields.second.astNode?.directives?.length).toEqual(0); + assertSome(fields['first']) + expect(fields['first'].deprecationReason).toBeUndefined(); + assertSome(fields['second']) + expect(fields['second'].deprecationReason).toBeUndefined(); + expect(fields['first'].astNode?.directives?.length).toEqual(0); + expect(fields['second'].astNode?.directives?.length).toEqual(0); }); }); diff --git a/packages/wrap/tests/transformRemoveObjectFieldDirectives.test.ts b/packages/wrap/tests/transformRemoveObjectFieldDirectives.test.ts index 6186cada3a6..745a885d748 100644 --- a/packages/wrap/tests/transformRemoveObjectFieldDirectives.test.ts +++ b/packages/wrap/tests/transformRemoveObjectFieldDirectives.test.ts @@ -29,14 +29,14 @@ describe('RemoveObjectFieldDirectives', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - assertSome(fields.id) - expect(fields.id.astNode?.directives?.length).toEqual(1); - assertSome(fields.first) - expect(fields.first.astNode?.directives?.length).toEqual(0); - assertSome(fields.second) - expect(fields.second.astNode?.directives?.length).toEqual(0); - assertSome(fields.third) - expect(fields.third.astNode?.directives?.length).toEqual(0); + assertSome(fields['id']) + expect(fields['id'].astNode?.directives?.length).toEqual(1); + assertSome(fields['first']) + expect(fields['first'].astNode?.directives?.length).toEqual(0); + assertSome(fields['second']) + expect(fields['second'].astNode?.directives?.length).toEqual(0); + assertSome(fields['third']) + expect(fields['third'].astNode?.directives?.length).toEqual(0); }); test('removes directives by name regex', async () => { @@ -50,14 +50,14 @@ describe('RemoveObjectFieldDirectives', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - assertSome(fields.id) - expect(fields.id.astNode?.directives?.length).toEqual(1); - assertSome(fields.first) - expect(fields.first.astNode?.directives?.length).toEqual(0); - assertSome(fields.second) - expect(fields.second.astNode?.directives?.length).toEqual(0); - assertSome(fields.third) - expect(fields.third.astNode?.directives?.length).toEqual(0); + assertSome(fields['id']) + expect(fields['id'].astNode?.directives?.length).toEqual(1); + assertSome(fields['first']) + expect(fields['first'].astNode?.directives?.length).toEqual(0); + assertSome(fields['second']) + expect(fields['second'].astNode?.directives?.length).toEqual(0); + assertSome(fields['third']) + expect(fields['third'].astNode?.directives?.length).toEqual(0); }); test('removes directives by argument', async () => { @@ -71,14 +71,14 @@ describe('RemoveObjectFieldDirectives', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - assertSome(fields.id) - expect(fields.id.astNode?.directives?.length).toEqual(0); - assertSome(fields.first) - expect(fields.first.astNode?.directives?.length).toEqual(1); - assertSome(fields.second) - expect(fields.second.astNode?.directives?.length).toEqual(0); - assertSome(fields.third) - expect(fields.third.astNode?.directives?.length).toEqual(0); + assertSome(fields['id']) + expect(fields['id'].astNode?.directives?.length).toEqual(0); + assertSome(fields['first']) + expect(fields['first'].astNode?.directives?.length).toEqual(1); + assertSome(fields['second']) + expect(fields['second'].astNode?.directives?.length).toEqual(0); + assertSome(fields['third']) + expect(fields['third'].astNode?.directives?.length).toEqual(0); }); test('removes directives by argument regex', async () => { @@ -92,13 +92,13 @@ describe('RemoveObjectFieldDirectives', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - assertSome(fields.id) - expect(fields.id.astNode?.directives?.length).toEqual(0); - assertSome(fields.first) - expect(fields.first.astNode?.directives?.length).toEqual(0); - assertSome(fields.second) - expect(fields.second.astNode?.directives?.length).toEqual(0); - assertSome(fields.third) - expect(fields.third.astNode?.directives?.length).toEqual(0); + assertSome(fields['id']) + expect(fields['id'].astNode?.directives?.length).toEqual(0); + assertSome(fields['first']) + expect(fields['first'].astNode?.directives?.length).toEqual(0); + assertSome(fields['second']) + expect(fields['second'].astNode?.directives?.length).toEqual(0); + assertSome(fields['third']) + expect(fields['third'].astNode?.directives?.length).toEqual(0); }); }); diff --git a/packages/wrap/tests/transformRemoveObjectFieldsWithDeprecation.test.ts b/packages/wrap/tests/transformRemoveObjectFieldsWithDeprecation.test.ts index 1989214acfa..a60dabb5615 100644 --- a/packages/wrap/tests/transformRemoveObjectFieldsWithDeprecation.test.ts +++ b/packages/wrap/tests/transformRemoveObjectFieldsWithDeprecation.test.ts @@ -24,8 +24,8 @@ describe('RemoveObjectFieldsWithDeprecation', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - expect(fields.first).toBeDefined(); - expect(fields.second).toBeUndefined(); + expect(fields['first']).toBeDefined(); + expect(fields['second']).toBeUndefined(); }); test('removes deprecated fields by reason regex', async () => { @@ -39,7 +39,7 @@ describe('RemoveObjectFieldsWithDeprecation', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - expect(fields.first).toBeUndefined(); - expect(fields.second).toBeUndefined(); + expect(fields['first']).toBeUndefined(); + expect(fields['second']).toBeUndefined(); }); }); diff --git a/packages/wrap/tests/transformRemoveObjectFieldsWithDirective.test.ts b/packages/wrap/tests/transformRemoveObjectFieldsWithDirective.test.ts index 8edc9c142ff..978bae84505 100644 --- a/packages/wrap/tests/transformRemoveObjectFieldsWithDirective.test.ts +++ b/packages/wrap/tests/transformRemoveObjectFieldsWithDirective.test.ts @@ -29,10 +29,10 @@ describe('RemoveObjectFieldsWithDirective', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - expect(fields.first).toBeUndefined(); - expect(fields.second).toBeUndefined(); - expect(fields.third).toBeUndefined(); - expect(fields.fourth).toBeDefined(); + expect(fields['first']).toBeUndefined(); + expect(fields['second']).toBeUndefined(); + expect(fields['third']).toBeUndefined(); + expect(fields['fourth']).toBeDefined(); }); test('removes directive fields by name regex', async () => { @@ -46,10 +46,10 @@ describe('RemoveObjectFieldsWithDirective', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - expect(fields.first).toBeUndefined(); - expect(fields.second).toBeUndefined(); - expect(fields.third).toBeUndefined(); - expect(fields.fourth).toBeDefined(); + expect(fields['first']).toBeUndefined(); + expect(fields['second']).toBeUndefined(); + expect(fields['third']).toBeUndefined(); + expect(fields['fourth']).toBeDefined(); }); test('removes directive fields by argument', async () => { @@ -63,10 +63,10 @@ describe('RemoveObjectFieldsWithDirective', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - expect(fields.first).toBeDefined(); - expect(fields.second).toBeUndefined(); - expect(fields.third).toBeUndefined(); - expect(fields.third).toBeUndefined(); + expect(fields['first']).toBeDefined(); + expect(fields['second']).toBeUndefined(); + expect(fields['third']).toBeUndefined(); + expect(fields['third']).toBeUndefined(); }); test('removes directive fields by argument regex', async () => { @@ -80,9 +80,9 @@ describe('RemoveObjectFieldsWithDirective', () => { const Test = transformedSchema.getType('Test') assertGraphQLObjectType(Test) const fields = Test.getFields(); - expect(fields.first).toBeUndefined(); - expect(fields.second).toBeUndefined(); - expect(fields.third).toBeUndefined(); - expect(fields.third).toBeUndefined(); + expect(fields['first']).toBeUndefined(); + expect(fields['second']).toBeUndefined(); + expect(fields['third']).toBeUndefined(); + expect(fields['third']).toBeUndefined(); }); }); diff --git a/packages/wrap/tests/transformRenameInputObjectFields.test.ts b/packages/wrap/tests/transformRenameInputObjectFields.test.ts index 4ad2abd005e..f40682d05d6 100644 --- a/packages/wrap/tests/transformRenameInputObjectFields.test.ts +++ b/packages/wrap/tests/transformRenameInputObjectFields.test.ts @@ -55,8 +55,8 @@ describe('RenameInputObjectFields', () => { const result = await graphql(transformedSchema, query); assertSome(result.data) - expect(result.data.test.field1).toBe('field1'); - expect(result.data.test.field2).toBe('field2'); + expect(result.data['test'].field1).toBe('field1'); + expect(result.data['test'].field2).toBe('field2'); }); test('renaming with variables works', async () => { @@ -118,7 +118,7 @@ describe('RenameInputObjectFields', () => { } const result = await graphql(transformedSchema, query, {}, {}, variables); assertSome(result.data) - expect(result.data.test.field1).toBe('field1'); - expect(result.data.test.field2).toBe('field2'); + expect(result.data['test'].field1).toBe('field1'); + expect(result.data['test'].field2).toBe('field2'); }); }); diff --git a/patches/bob-the-bundler+1.4.1.patch b/patches/bob-the-bundler+1.4.1.patch new file mode 100644 index 00000000000..ed97b313bda --- /dev/null +++ b/patches/bob-the-bundler+1.4.1.patch @@ -0,0 +1,22 @@ +diff --git a/node_modules/bob-the-bundler/dist/commands/build.js b/node_modules/bob-the-bundler/dist/commands/build.js +index e1c8d4e..cb9fa6a 100644 +--- a/node_modules/bob-the-bundler/dist/commands/build.js ++++ b/node_modules/bob-the-bundler/dist/commands/build.js +@@ -87,7 +87,7 @@ async function buildSingle({ distDir, distPath = '' }) { + // generates + const commonOutputOptions = { + preferConst: true, +- sourcemap: true, ++ sourcemap: false, + }; + const generates = [ + { +@@ -148,7 +148,7 @@ async function build({ packagePath, cwd, pkg, fullName, config, reporter, distDi + // generates + const commonOutputOptions = { + preferConst: true, +- sourcemap: true, ++ sourcemap: false, + }; + const generates = [ + { diff --git a/scripts/canary-release.js b/scripts/canary-release.js index 978fac233d5..62e2cf32439 100644 --- a/scripts/canary-release.js +++ b/scripts/canary-release.js @@ -11,7 +11,7 @@ const { getPackages } = require("@manypkg/get-packages"); function getNewVersion(version, type) { const gitHash = cp.spawnSync('git', ['rev-parse', '--short', 'HEAD']).stdout.toString().trim(); - + return semver.inc(version, `pre${type}`, true, 'alpha-' + gitHash); } @@ -27,14 +27,15 @@ async function updateVersions() { const packages = await getPackages(cwd); const config = await readConfig(cwd, packages); const modifiedChangesets = getRelevantChangesets(config.baseBranch); - const changesets = (await readChangesets(cwd)).filter(change => modifiedChangesets.includes(change.id)); - + const allChangesets = await readChangesets(cwd); + const changesets = process.env.ON_DEMAND ? allChangesets : allChangesets.filter(change => modifiedChangesets.includes(change.id)); + if (changesets.length === 0) { console.warn(`Unable to find any relevant package for canary publishing. Please make sure changesets exists!`); process.exit(1); } else { const releasePlan = assembleReleasePlan(changesets, packages, config, [], false); - + if (releasePlan.releases.length === 0) { console.warn(`Unable to find any relevant package for canary releasing. Please make sure changesets exists!`); process.exit(1); diff --git a/tsconfig.build.es5.json b/tsconfig.build.es5.json index c42f1e0d7d6..879594f93ff 100644 --- a/tsconfig.build.es5.json +++ b/tsconfig.build.es5.json @@ -2,6 +2,9 @@ "extends": "./tsconfig.build.json", "compilerOptions": { "outDir": "dist-es5", - "target": "es5" + "target": "es5", + "tsBuildInfoFile": "./node_modules/tsconfig.build.es5.tsbuildinfo", + "sourceMap": false, + "inlineSourceMap": false } } diff --git a/tsconfig.build.json b/tsconfig.build.json index 4abd5dd07b0..a4e8b33e761 100644 --- a/tsconfig.build.json +++ b/tsconfig.build.json @@ -1,5 +1,10 @@ { "extends": "./tsconfig.json", + "compilerOptions": { + "tsBuildInfoFile": "./node_modules/tsconfig.build.tsbuildinfo", + "sourceMap": false, + "inlineSourceMap": false + }, "exclude": [ "**/test/*.ts", "*.spec.ts", diff --git a/tsconfig.json b/tsconfig.json index 86feb85ea87..3221d62b0b1 100755 --- a/tsconfig.json +++ b/tsconfig.json @@ -30,8 +30,10 @@ "paths": { "@graphql-tools/*-loader": ["packages/loaders/*/src/index.ts", "packages/*-loader/src/index.ts"], "@graphql-tools/*": ["packages/*/src/index.ts"] - } + }, + + "tsBuildInfoFile": "./node_modules/tsconfig.tsbuildinfo" }, "include": ["packages"], - "exclude": ["**/node_modules", "**/dist"] + "exclude": ["**/node_modules", "**/dist", "**/test-files"] } diff --git a/website/docs/directive-resolvers.md b/website/docs/directive-resolvers.md deleted file mode 100644 index fcf91884492..00000000000 --- a/website/docs/directive-resolvers.md +++ /dev/null @@ -1,178 +0,0 @@ ---- -id: directive-resolvers -title: Directive resolvers -description: A set of utilities to build your JavaScript GraphQL schema in a concise and powerful way. ---- - -## Directive example - -Let's take a look at how we can create `@upper` Directive to upper-case a string returned from resolve on Field - -To start, let's grab the schema definition string from the `makeExecutableSchema` example [in the "Generating a schema" article](/docs/generate-schema/#example). - -```js -import { makeExecutableSchema } from '@graphql-tools/schema'; -import { graphql } from 'graphql'; - -// Construct a schema, using GraphQL schema language -const typeDefs = ` - directive @upper on FIELD_DEFINITION - - type Query { - hello: String @upper - } -`; - -// Implement resolvers for out custom Directive -const directiveResolvers = { - upper( - next, - src, - args, - context, - ) { - return next().then((str) => { - if (typeof(str) === 'string') { - return str.toUpperCase(); - } - return str; - }); - }, -} - -// Provide resolver functions for your schema fields -const resolvers = { - Query: { - hello: (root, args, context) => { - return 'Hello world!'; - }, - }, -}; - -export const schema = makeExecutableSchema({ - typeDefs, - resolvers, - directiveResolvers, -}); - -const query = ` -query UPPER_HELLO { - hello -} -`; - -graphql(schema, query).then((result) => console.log('Got result', result)); -``` - -> Note: next() always return a Promise for consistency, resolved with original resolver value or rejected with an error. - -## Multi-Directives example - -Multi-Directives on a field will be apply with LTR order. - -```js -// graphql-tools combines a schema string with resolvers. -import { makeExecutableSchema } from '@graphql-tools/schema'; - -// Construct a schema, using GraphQL schema language -const typeDefs = ` - directive @upper on FIELD_DEFINITION - directive @concat(value: String!) on FIELD_DEFINITION - - type Query { - foo: String @concat(value: "@gmail.com") @upper - } -`; - -// Customs directives, check https://github.com/ardatan/graphql-tools/pull/518 -// for more examples -const directiveResolvers = { - upper( - next, - src, - args, - context, - ) { - return next().then((str) => { - if (typeof(str) === 'string') { - return str.toUpperCase(); - } - return str; - }); - }, - concat( - next, - src, - args, - context, - ) { - return next().then((str) => { - if (typeof(str) !== 'undefined') { - return `${str}${args.value}`; - } - return str; - }); - }, -} - -// Provide resolver functions for your schema fields -const resolvers = { - Query: { - foo: (root, args, context) => { - return 'foo'; - }, - }, -}; - -// Required: Export the GraphQL.js schema object as "schema" -export const schema = makeExecutableSchema({ - typeDefs, - resolvers, - directiveResolvers, -}); -``` - -The result with query `{foo}` will be: -```json -{ - "data": { - "foo": "FOO@GMAIL.COM" - } -} -``` - -## API - -### directiveResolvers option - -```js -import { makeExecutableSchema } from '@graphql-tools/schema'; - -const directiveResolvers = { - // directive resolvers implement -}; - -const schema = makeExecutableSchema({ - // ... other options - directiveResolvers, -}) -``` - -`makeExecutableSchema` has new option field is `directiveResolvers`, a map object for custom Directive's resolvers. - -### attachDirectiveResolvers - -```js -import { attachDirectiveResolvers } from '@graphql-tools/schema'; - -const directiveResolvers = { - // directive resolvers implement -}; - -schemaWithDirectiveResolvers = attachDirectiveResolvers( - schema, - directiveResolvers, -); -``` - -Given an instance of GraphQLSchema and a `directiveResolvers` map object, `attachDirectiveResolvers` returns a new schema in which all fields' resolver have been wrapped with directive resolvers. diff --git a/website/docs/generate-schema.md b/website/docs/generate-schema.md index 9f8acafa0e4..426cbf5d222 100644 --- a/website/docs/generate-schema.md +++ b/website/docs/generate-schema.md @@ -219,10 +219,7 @@ const jsSchema = makeExecutableSchema({ typeDefs, resolvers, // optional logger, // optional - allowUndefinedInResolve: false, // optional resolverValidationOptions: {}, // optional - directiveResolvers: null, // optional - schemaDirectives: null, // optional schemaTransforms: [], // optional parseOptions: {}, // optional inheritResolversFromInterfaces: false // optional @@ -233,12 +230,8 @@ const jsSchema = makeExecutableSchema({ - `resolvers` is an optional argument _(empty object by default)_ and should be an object or an array of objects that follow the pattern explained in [article on resolvers](/docs/resolvers/) -- `logger` is an optional argument, which can be used to print errors to the server console that are usually swallowed by GraphQL. The `logger` argument should be an object with a `log` function, eg. `const logger = { log: e => console.log(e) }` - - `parseOptions` is an optional argument which allows customization of parse when specifying `typeDefs` as a string. -- `allowUndefinedInResolve` is an optional argument, which is `true` by default. When set to `false`, causes your resolver to throw errors if they return undefined, which can help make debugging easier. - - `resolverValidationOptions` is an optional argument with the following properties, each of which can be set to `error`, `warn`, or `ignore`: - `requireResolversForArgs` will cause `makeExecutableSchema` to throw an error (`error`) or issue a warning (`warn`)unless a resolver is defined for every field with arguments. The default is `ignore`, causing this validator to be skipped. @@ -254,4 +247,3 @@ const jsSchema = makeExecutableSchema({ - `schemaTransforms` is an optional argument _(empty array by default)_ and should be an array of schema transformation functions, essentially designed to enable the use of [directive-based functional schema transformation](/docs/schema-directives/) -- `schemaDirectives` is an optional argument _(empty object by default)_ and can be used to specify the [earlier class-based implementation of schema directives](/docs/legacy-schema-directives/) diff --git a/website/docs/legacy-schema-directives.md b/website/docs/legacy-schema-directives.md deleted file mode 100644 index 0bd464b614c..00000000000 --- a/website/docs/legacy-schema-directives.md +++ /dev/null @@ -1,714 +0,0 @@ ---- -id: legacy-schema-directives -title: Schema directives -description: Using and implementing custom directives to transform schema types, fields, and arguments ---- - -A _directive_ is an identifier preceded by a `@` character, optionally followed by a list of named arguments, which can appear after almost any form of syntax in the GraphQL query or schema languages. Here's an example from the [GraphQL draft specification](http://facebook.github.io/graphql/draft/#sec-Type-System.Directives) that illustrates several of these possibilities: - -```typescript -directive @deprecated( - reason: String = "No longer supported" -) on FIELD_DEFINITION | ENUM_VALUE - -type ExampleType { - newField: String - oldField: String @deprecated(reason: "Use `newField`.") -} -``` - -As you can see, the usage of `@deprecated(reason: ...)` _follows_ the field that it pertains to (`oldField`), though the syntax might remind you of "decorators" in other languages, which usually appear on the line above. Directives are typically _declared_ once, using the `directive @deprecated ... on ...` syntax, and then _used_ zero or more times throughout the schema document, using the `@deprecated(reason: ...)` syntax. - -The possible applications of directive syntax are numerous: enforcing access permissions, formatting date strings, auto-generating resolver functions for a particular backend API, marking strings for internationalization, synthesizing globally unique object identifiers, specifying caching behavior, skipping or including or deprecating fields, and just about anything else you can imagine. - -This document focuses on directives that appear in GraphQL _schemas_ (as opposed to queries) written in [Schema Definition Language](https://github.com/facebook/graphql/pull/90), or SDL for short. In the following sections, you will see how custom directives can be implemented and used to modify the structure and behavior of a GraphQL schema in ways that would not be possible using SDL syntax alone. - -## (At least) two strategies - -`graphql-tools` provides [a newer functional approach](/docs/schema-directives/) for directive-based schema modification. The remainder of this document describes the earlier class-based mechanism. We believe the newer approach is easier to reason about, but older class-based schema directives are still supported, as long as the in-place schema modification techniques they employ do not yield an invalid schema. - -## Using schema directives - -Most of this document is concerned with _implementing_ schema directives, and some of the examples are quite complicated. No matter how many tools and best practices you have at your disposal, it can be difficult to implement a non-trivial class-based schema directive in a reliable, reusable way. Exhaustive testing is essential, and using a typed language like TypeScript is recommended, because there are so many different schema types to worry about. - -However, the API we provide for _using_ a schema directive is extremely simple. Just import the implementation of the directive, then pass it to `makeExecutableSchema` via the `schemaDirectives` argument, which is an object that maps directive names to directive implementations: - -```js -import { makeExecutableSchema } from '@graphql-tools/schema'; -import { RenameDirective } from 'fake-rename-directive-package'; - -const typeDefs = ` -type Person @rename(to: "Human") { - name: String! - currentDateMinusDateOfBirth: Int @rename(to: "age") -}`; - -const schema = makeExecutableSchema({ - typeDefs, - schemaDirectives: { - rename: RenameDirective, - }, -}); -``` - -That's it. The implementation of `RenameDirective` takes care of everything else. If you understand what the directive is supposed to do to your schema, then you do not have to worry about how it works. - -Everything you read below addresses some aspect of how a directive like `@rename(to: ...)` could be implemented. If that's not something you care about right now, feel free to skip the rest of this document. When you need it, it will be here. - -## Implementing schema directives - -Since the GraphQL specification does not discuss any specific implementation strategy for directives, it's up to each GraphQL server framework to expose an API for implementing new directives. - -GraphQL Tools provides a convenient yet powerful tool for implementing directive syntax: the [`SchemaDirectiveVisitor`](https://github.com/ardatan/graphql-tools/blob/master/src/utils/SchemaDirectiveVisitor.ts) class. - -To implement a schema directive using `SchemaDirectiveVisitor`, simply create a subclass of `SchemaDirectiveVisitor` that overrides one or more of the following visitor methods: - -- `visitSchema(schema: GraphQLSchema)` -- `visitScalar(scalar: GraphQLScalarType)` -- `visitObject(object: GraphQLObjectType)` -- `visitFieldDefinition(field: GraphQLField, details: { objectType: GraphQLObjectType | GraphQLInterfaceType })` -- `visitArgumentDefinition(argument: GraphQLArgument, objectType: GraphQLObjectType | GraphQLInterfaceType })` -- `visitInterface(iface: GraphQLInterfaceType)` -- `visitUnion(union: GraphQLUnionType)` -- `visitEnum(type: GraphQLEnumType)` -- `visitEnumValue(value: GraphQLEnumValue, details: { enumType: GraphQLEnumType })` -- `visitInputObject(object: GraphQLInputObjectType)` -- `visitInputFieldDefinition(field: GraphQLInputField, details: { objectType: GraphQLInputObjectType })` - -By overriding methods like `visitObject`, a subclass of `SchemaDirectiveVisitor` expresses interest in certain schema types such as `GraphQLObjectType` (the first parameter type of `visitObject`). - -These method names correspond to all possible [locations](https://github.com/graphql/graphql-js/blob/a62eea88d5844a3bd9725c0f3c30950a78727f3e/src/language/directiveLocation.js#L22-L33) where a directive may be used in a schema. For example, the location `INPUT_FIELD_DEFINITION` is handled by `visitInputFieldDefinition`. - -Here is one possible implementation of the `@deprecated` directive we saw above: - -```typescript -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; -import { GraphQLField, GraphQLEnumValue } from 'graphql'; - -class DeprecatedDirective extends SchemaDirectiveVisitor { - public visitFieldDefinition(field: GraphQLField) { - field.isDeprecated = true; - field.deprecationReason = this.args.reason; - } - - public visitEnumValue(value: GraphQLEnumValue) { - value.isDeprecated = true; - value.deprecationReason = this.args.reason; - } -} -``` - -In order to apply this implementation to a schema that contains `@deprecated` directives, simply pass the `DeprecatedDirective` class to the `makeExecutableSchema` function via the `schemaDirectives` option: - -```typescript -import { makeExecutableSchema } from '@graphql-tools/schema'; - -const typeDefs = ` -type ExampleType { - newField: String - oldField: String @deprecated(reason: "Use \`newField\`.") -}`; - -const schema = makeExecutableSchema({ - typeDefs, - schemaDirectives: { - deprecated: DeprecatedDirective, - }, -}); -``` - -Alternatively, if you want to modify an existing schema object, you can use the `SchemaDirectiveVisitor.visitSchemaDirectives` interface directly: - -```typescript -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; - -SchemaDirectiveVisitor.visitSchemaDirectives(schema, { - deprecated: DeprecatedDirective, -}); -``` - -This syntax is especially useful for code-first schemas that wish to make use of directive implementations. For code-first schemas, directives are read from the `directives` key within the `extensions` field for each GraphQL entity, unless a different path is provided, as per below: - -```typescript -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; - -SchemaDirectiveVisitor.visitSchemaDirectives( - schema, - { - deprecated: DeprecatedDirective, - }, - undefined, - ['custom', 'path', 'to', 'directives', 'within', 'the', 'extensions', 'object'] -); -``` - -The second argument to `visitSchemaDirectives` refers to a shared context object that may be passed to `SchemaDirectiveVisitor` classes -- it is not often used, and can usually be safely set to undefined. - -See [this `graphql-js` issue](https://github.com/graphql/graphql-js/issues/1343) for more information on directives with code-first schemas. We follow the [Gatsby and graphql-compose convention](https://github.com/graphql/graphql-js/issues/1343#issuecomment-479877640) of reading directives from the `extensions` field, but allow customization as above. - -Note that a subclass of `SchemaDirectiveVisitor` may be instantiated multiple times to visit multiple different occurrences of the `@deprecated` directive. That's why you provide a class rather than an instance of that class. - -If for some reason you have a schema that uses another name for the `@deprecated` directive, but you want to use the same implementation, you can! The same `DeprecatedDirective` class can be passed with a different name, simply by changing its key in the `schemaDirectives` object passed to `makeExecutableSchema`. In other words, `SchemaDirectiveVisitor` implementations are effectively anonymous, so it's up to whoever uses them to assign names to them. - -## Examples - -To appreciate the range of possibilities enabled by `SchemaDirectiveVisitor`, let's examine a variety of practical examples. - -### Uppercasing strings - -Suppose you want to ensure a string-valued field is converted to uppercase. Though this use case is simple, it's a good example of a directive implementation that works by wrapping a field's `resolve` function: - -```js -import { defaultFieldResolver } from 'graphql'; -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; -import { makeExecutableSchema } from '@graphql-tools/schema'; - -const typeDefs = ` -directive @upper on FIELD_DEFINITION - -type Query { - hello: String @upper -}`; - -class UpperCaseDirective extends SchemaDirectiveVisitor { - visitFieldDefinition(field) { - const { resolve = defaultFieldResolver } = field; - field.resolve = async function (...args) { - const result = await resolve.apply(this, args); - if (typeof result === 'string') { - return result.toUpperCase(); - } - return result; - }; - } -} - -const schema = makeExecutableSchema({ - typeDefs, - schemaDirectives: { - upper: UpperCaseDirective, - upperCase: UpperCaseDirective, - }, -}); -``` - -Notice how easy it is to handle both `@upper` and `@upperCase` with the same `UpperCaseDirective` implementation. - -### Fetching data from a REST API - -Suppose you've defined an object type that corresponds to a [REST](https://en.wikipedia.org/wiki/Representational_state_transfer) resource, and you want to avoid implementing resolver functions for every field: - -```js -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; -import { makeExecutableSchema } from '@graphql-tools/schema'; - -const typeDefs = ` -directive @rest(url: String) on FIELD_DEFINITION - -type Query { - people: [Person] @rest(url: "/api/v1/people") -}`; - -class RestDirective extends SchemaDirectiveVisitor { - public visitFieldDefinition(field) { - const { url } = this.args; - field.resolve = () => fetch(url); - } -} - -const schema = makeExecutableSchema({ - typeDefs, - schemaDirectives: { - rest: RestDirective - } -}); -``` - -There are many more issues to consider when implementing a real GraphQL wrapper over a REST endpoint (such as how to do caching or pagination), but this example demonstrates the basic structure. - -### Formatting date strings - -Suppose your resolver returns a `Date` object but you want to return a formatted string to the client: - -```js -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; -import { makeExecutableSchema } from '@graphql-tools/schema'; - -const typeDefs = ` -directive @date(format: String) on FIELD_DEFINITION - -scalar Date - -type Post { - published: Date @date(format: "mmmm d, yyyy") -}`; - -class DateFormatDirective extends SchemaDirectiveVisitor { - visitFieldDefinition(field) { - const { resolve = defaultFieldResolver } = field; - const { format } = this.args; - field.resolve = async function (...args) { - const date = await resolve.apply(this, args); - return require('dateformat')(date, format); - }; - // The formatted Date becomes a String, so the field type must change: - field.type = GraphQLString; - } -} - -const schema = makeExecutableSchema({ - typeDefs, - schemaDirectives: { - date: DateFormatDirective, - }, -}); -``` - -Of course, it would be even better if the schema author did not have to decide on a specific `Date` format, but could instead leave that decision to the client. To make this work, the directive just needs to add an additional argument to the field: - -```js -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; -import { makeExecutableSchema } from '@graphql-tools/schema'; -import formatDate from "dateformat"; -import { defaultFieldResolver, GraphQLString } from "graphql"; - -const typeDefs = ` -directive @date( - defaultFormat: String = "mmmm d, yyyy" -) on FIELD_DEFINITION - -scalar Date - -type Query { - today: Date @date -}`; - -class FormattableDateDirective extends SchemaDirectiveVisitor { - public visitFieldDefinition(field) { - const { resolve = defaultFieldResolver } = field; - const { defaultFormat } = this.args; - - field.args.push({ - name: 'format', - type: GraphQLString - }); - - field.resolve = async function ( - source, - { format, ...otherArgs }, - context, - info, - ) { - const date = await resolve.call(this, source, otherArgs, context, info); - // If a format argument was not provided, default to the optional - // defaultFormat argument taken by the @date directive: - return formatDate(date, format || defaultFormat); - }; - - field.type = GraphQLString; - } -} - -const schema = makeExecutableSchema({ - typeDefs, - schemaDirectives: { - date: FormattableDateDirective - } -}); -``` - -Now the client can specify a desired `format` argument when requesting the `Query.today` field, or omit the argument to use the `defaultFormat` string specified in the schema: - -```js -import { graphql } from 'graphql'; - -graphql( - schema, - ` - query { - today - } - ` -).then(result => { - // Logs with the default "mmmm d, yyyy" format: - console.log(result.data.today); -}); - -graphql( - schema, - ` - query { - today(format: "d mmm yyyy") - } - ` -).then(result => { - // Logs with the requested "d mmm yyyy" format: - console.log(result.data.today); -}); -``` - -### Marking strings for internationalization - -Suppose you have a function called `translate` that takes a string, a path identifying that string's role in your application, and a target locale for the translation. - -Here's how you might make sure `translate` is used to localize the `greeting` field of a `Query` type: - -```js -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; -import { makeExecutableSchema } from '@graphql-tools/schema'; - -const typeDefs = ` -directive @intl on FIELD_DEFINITION - -type Query { - greeting: String @intl -}`; - -class IntlDirective extends SchemaDirectiveVisitor { - visitFieldDefinition(field, details) { - const { resolve = defaultFieldResolver } = field; - field.resolve = async function (...args) { - const context = args[2]; - const defaultText = await resolve.apply(this, args); - // In this example, path would be ["Query", "greeting"]: - const path = [details.objectType.name, field.name]; - return translate(defaultText, path, context.locale); - }; - } -} - -const schema = makeExecutableSchema({ - typeDefs, - schemaDirectives: { - intl: IntlDirective, - }, -}); -``` - -GraphQL is great for internationalization, since a GraphQL server can access unlimited translation data, and clients can simply ask for the translations they need. - -### Enforcing access permissions - -Imagine a hypothetical `@auth` directive that takes an argument `requires` of type `Role`, which defaults to `ADMIN`. This `@auth` directive can appear on an `OBJECT` like `User` to set default access permissions for all `User` fields, as well as appearing on individual fields, to enforce field-specific `@auth` restrictions: - -```graphql -directive @auth(requires: Role = ADMIN) on OBJECT | FIELD_DEFINITION - -enum Role { - ADMIN - REVIEWER - USER - UNKNOWN -} - -type User @auth(requires: USER) { - name: String - banned: Boolean @auth(requires: ADMIN) - canPost: Boolean @auth(requires: REVIEWER) -} -``` - -What makes this example tricky is that the `OBJECT` version of the directive needs to wrap all fields of the object, even though some of those fields may be individually wrapped by `@auth` directives at the `FIELD_DEFINITION` level, and we would prefer not to rewrap resolvers if we can help it: - -```js -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; -import { makeExecutableSchema } from '@graphql-tools/schema'; - -class AuthDirective extends SchemaDirectiveVisitor { - visitObject(type) { - this.ensureFieldsWrapped(type); - type._requiredAuthRole = this.args.requires; - } - // Visitor methods for nested types like fields and arguments - // also receive a details object that provides information about - // the parent and grandparent types. - visitFieldDefinition(field, details) { - this.ensureFieldsWrapped(details.objectType); - field._requiredAuthRole = this.args.requires; - } - - ensureFieldsWrapped(objectType) { - // Mark the GraphQLObjectType object to avoid re-wrapping: - if (objectType._authFieldsWrapped) return; - objectType._authFieldsWrapped = true; - - const fields = objectType.getFields(); - - Object.keys(fields).forEach(fieldName => { - const field = fields[fieldName]; - const { resolve = defaultFieldResolver } = field; - field.resolve = async function (...args) { - // Get the required Role from the field first, falling back - // to the objectType if no Role is required by the field: - const requiredRole = field._requiredAuthRole || objectType._requiredAuthRole; - - if (!requiredRole) { - return resolve.apply(this, args); - } - - const context = args[2]; - const user = await getUser(context.headers.authToken); - if (!user.hasRole(requiredRole)) { - throw new Error('not authorized'); - } - - return resolve.apply(this, args); - }; - }); - } -} - -const schema = makeExecutableSchema({ - typeDefs, - schemaDirectives: { - auth: AuthDirective, - authorized: AuthDirective, - authenticated: AuthDirective, - }, -}); -``` - -One drawback of this approach is that it does not guarantee fields will be wrapped if they are added to the schema after `AuthDirective` is applied, and the whole `getUser(context.headers.authToken)` is a made-up API that would need to be fleshed out. In other words, we’ve glossed over some of the details that would be required for a production-ready implementation of this directive, though we hope the basic structure shown here inspires you to find clever solutions to the remaining problems. - -### Enforcing value restrictions - -Suppose you want to enforce a maximum length for a string-valued field: - -```js -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; -import { makeExecutableSchema } from '@graphql-tools/schema'; -import { GraphQLScalarType, isNonNullType, isScalarType, GraphQLNonNull } from 'graphql'; - -const typeDefs = ` -directive @length(max: Int) on FIELD_DEFINITION | INPUT_FIELD_DEFINITION - -type Query { - books: [Book] -} - -type Book { - title: String @length(max: 50) -} - -type Mutation { - createBook(book: BookInput): Book -} - -input BookInput { - title: String! @length(max: 50) -}`; - -class LengthDirective extends SchemaDirectiveVisitor { - visitInputFieldDefinition(field) { - this.wrapType(field); - } - - visitFieldDefinition(field) { - this.wrapType(field); - } - - // Replace field.type with a custom GraphQLScalarType that enforces the - // length restriction. - wrapType(field) { - if (isNonNullType(field.type) && isScalarType(field.type.ofType)) { - field.type = new GraphQLNonNull(new LimitedLengthType(field.type.ofType, this.args.max)); - } else if (isScalarType(field.type)) { - field.type = new LimitedLengthType(field.type, this.args.max); - } else { - throw new Error(`Not a scalar type: ${field.type}`); - } - } -} - -class LimitedLengthType extends GraphQLScalarType { - constructor(type, maxLength) { - super({ - name: `LengthAtMost${maxLength}`, - - // For more information about GraphQLScalar type (de)serialization, - // see the graphql-js implementation: - // https://github.com/graphql/graphql-js/blob/31ae8a8e8312/src/type/definition.js#L425-L446 - - serialize(value) { - value = type.serialize(value); - assert.isAtMost(value.length, maxLength); - return value; - }, - - parseValue(value) { - return type.parseValue(value); - }, - - parseLiteral(ast) { - return type.parseLiteral(ast); - }, - }); - } -} - -const schema = makeExecutableSchema({ - typeDefs, - schemaDirectives: { - length: LengthDirective, - }, -}); -``` - -### Synthesizing unique IDs - -Suppose your database uses incrementing IDs for each resource type, so IDs are not unique across all resource types. Here’s how you might synthesize a field called `uid` that combines the object type with various field values to produce an ID that’s unique across your schema: - -```js -import { GraphQLID } from 'graphql'; -import { createHash } from 'crypto'; -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; -import { makeExecutableSchema } from '@graphql-tools/schema'; - -const typeDefs = ` -directive @uniqueID( - # The name of the new ID field, "uid" by default: - name: String = "uid" - - # Which fields to include in the new ID: - from: [String] = ["id"] -) on OBJECT - -# Since this type just uses the default values of name and from, -# we don't have to pass any arguments to the directive: -type Location @uniqueID { - id: Int - address: String -} - -# This type uses both the person's name and the personID field, -# in addition to the "Person" type name, to construct the ID: -type Person @uniqueID(from: ["name", "personID"]) { - personID: Int - name: String -}`; - -class UniqueIdDirective extends SchemaDirectiveVisitor { - visitObject(type) { - const { name, from } = this.args; - const fields = type.getFields(); - if (name in fields) { - throw new Error(`Conflicting field name ${name}`); - } - fields[name] = { - name, - type: GraphQLID, - description: 'Unique ID', - args: [], - resolve(object) { - const hash = createHash('sha1'); - hash.update(type.name); - from.forEach(fieldName => { - hash.update(String(object[fieldName])); - }); - return hash.digest('hex'); - }, - }; - } -} - -const schema = makeExecutableSchema({ - typeDefs, - schemaDirectives: { - uniqueID: UniqueIdDirective, - }, -}); -``` - -## Declaring schema directives - -While the above examples should be sufficient to implement any `@directive` used in your schema, SDL syntax also supports declaring the names, argument types, default argument values, and permissible locations of any available directives: - -```js -directive @auth( - requires: Role = ADMIN, -) on OBJECT | FIELD_DEFINITION - -enum Role { - ADMIN - REVIEWER - USER - UNKNOWN -} - -type User @auth(requires: USER) { - name: String - banned: Boolean @auth(requires: ADMIN) - canPost: Boolean @auth(requires: REVIEWER) -} -``` - -This hypothetical `@auth` directive takes an argument named `requires` of type `Role`, which defaults to `ADMIN` if `@auth` is used without passing an explicit `requires` argument. The `@auth` directive can appear on an `OBJECT` like `User` to set a default access control for all `User` fields, and also on individual fields, to enforce field-specific `@auth` restrictions. - -Enforcing the requirements of the declaration is something a `SchemaDirectiveVisitor` implementation could do itself, in theory, but the SDL syntax is easier to read and write, and provides value even if you're not using the `SchemaDirectiveVisitor` abstraction. - -However, if you're implementing a reusable `SchemaDirectiveVisitor` for public consumption, you will probably not be the person writing the SDL syntax, so you may not have control over which directives the schema author decides to declare, and how. That's why a well-implemented, reusable `SchemaDirectiveVisitor` should consider overriding the `getDirectiveDeclaration` method: - -```typescript -import { SchemaDirectiveVisitor } from '@graphql-tools/utils'; -import { makeExecutableSchema } from '@graphql-tools/schema'; -import { - DirectiveLocation, - GraphQLDirective, - GraphQLEnumType, -} from "graphql"; - -class AuthDirective extends SchemaDirectiveVisitor { - public visitObject(object: GraphQLObjectType) {...} - public visitFieldDefinition(field: GraphQLField) {...} - - public static getDirectiveDeclaration( - directiveName: string, - schema: GraphQLSchema, - ): GraphQLDirective { - const previousDirective = schema.getDirective(directiveName); - if (previousDirective) { - // If a previous directive declaration exists in the schema, it may be - // better to modify it than to return a new GraphQLDirective object. - previousDirective.args.forEach(arg => { - if (arg.name === 'requires') { - // Lower the default minimum Role from ADMIN to REVIEWER. - arg.defaultValue = 'REVIEWER'; - } - }); - - return previousDirective; - } - - // If a previous directive with this name was not found in the schema, - // there are several options: - // - // 1. Construct a new GraphQLDirective (see below). - // 2. Throw an exception to force the client to declare the directive. - // 3. Return null, and forget about declaring this directive. - // - // All three are valid options, since the visitor will still work without - // any declared directives. In fact, unless you're publishing a directive - // implementation for public consumption, you can probably just ignore - // getDirectiveDeclaration altogether. - - return new GraphQLDirective({ - name: directiveName, - locations: [ - DirectiveLocation.OBJECT, - DirectiveLocation.FIELD_DEFINITION, - ], - args: { - requires: { - // Having the schema available here is important for obtaining - // references to existing type objects, such as the Role enum. - type: (schema.getType('Role') as GraphQLEnumType), - // Set the default minimum Role to REVIEWER. - defaultValue: 'REVIEWER', - } - }] - }); - } -} -``` - -Since the `getDirectiveDeclaration` method receives not only the name of the directive but also the `GraphQLSchema` object, it can modify and/or reuse previous declarations found in the schema, as an alternative to returning a totally new `GraphQLDirective` object. Either way, if the visitor returns a non-null `GraphQLDirective` from `getDirectiveDeclaration`, that declaration will be used to check arguments and permissible locations. diff --git a/website/docs/remote-schemas.md b/website/docs/remote-schemas.md index 08b774d0a42..a07a70c7edb 100644 --- a/website/docs/remote-schemas.md +++ b/website/docs/remote-schemas.md @@ -261,23 +261,3 @@ const schema = wrapSchema({ ``` Note that within the `defaultCreateProxyingResolver` function, `delegateToSchema` receives `executor` and `subscriber` functions stored on the subschema config object originally passed to `wrapSchema`. As above, use of the the `createProxyingResolver` option is helpful when you want to customize additional functionality at resolver creation time. If you just want to customize how things are proxied at the time that they are proxied, you can make do just with custom executors and subscribers. - -### makeRemoteExecutableSchema(options) - -What about `makeRemoteExecutableSchema`, the function used in older versions to access remote schemas? It still works -- just now under the hood calling `wrapSchema`. There is essentially no longer any need to use `makeRemoteExecutableSchema` directly, but we've kept it around for backwards compatibility. - -You can still pass a `createResolver` function to `makeRemoteExecutableSchema` to override how the fetch resolvers are created and executed. The `createResolver` param accepts an `Executor` as its first argument (and a `Subscriber` as its second) and returns a resolver function. This opens up the possibility for users to create batching mechanisms for fetches. As above, it is likely easier to just customize the `executor` function itself. - -Given a GraphQL.js schema (can be a non-executable client schema made by `buildClientSchema`) and a [executor](#creating-an-executor), `makeRemoteExecutableSchema` produce a GraphQL Schema that routes all requests to the executor: - -```js -import { makeRemoteExecutableSchema } from '@graphql-tools/wrap'; - -const createResolver: (executor: Executor) => GraphQLFieldResolver = // . . . - -const schema = makeRemoteExecutableSchema({ - schema, - executor, - createResolver -}); -``` diff --git a/website/docs/schema-directives.md b/website/docs/schema-directives.md index 36f3caed1f4..124bbfaaf07 100644 --- a/website/docs/schema-directives.md +++ b/website/docs/schema-directives.md @@ -23,10 +23,6 @@ The possible applications of directive syntax are numerous: enforcing access per This document focuses on directives that appear in GraphQL _schemas_ (as opposed to queries) written in [Schema Definition Language](https://github.com/facebook/graphql/pull/90), or SDL for short. In the following sections, you will see how custom directives can be implemented and used to modify the structure and behavior of a GraphQL schema in ways that would not be possible using SDL syntax alone. -## (At least) two strategies - -Earlier versions of `graphql-tools` provides a class-based mechanism for directive-based schema modification. The documentation for the class-based version is [still available](/docs/legacy-schema-directives/), but the remainder of this document describes the newer functional mechanism. We believe the newer approach is easier to reason about, but older class-based schema directives are still supported. - ## Using schema directives Most of this document is concerned with _implementing_ schema directives, and some of the examples may seem quite complicated. No matter how many tools and best practices you have at your disposal, it can be difficult to implement a non-trivial schema directive in a reliable, reusable way. Exhaustive testing is essential, and using a typed language like TypeScript is recommended, because there are so many different schema types to worry about. @@ -619,9 +615,9 @@ function uniqueIDDirective(directiveName: string) { resolve(object: any) { const hash = createHash('sha1'); hash.update(type.name); - from.forEach((fieldName: string) => { + for (const fieldName of from) { hash.update(String(object[fieldName])); - }); + } return hash.digest('hex'); }, }; @@ -692,7 +688,7 @@ In theory, access to the query directives is available within the `info` resolve The `makeExecutableSchema` function also takes a `directiveResolvers` option that can be used for implementing certain kinds of `@directive`s on fields that have resolver functions. -The new abstraction is more general, since it can visit any kind of schema syntax, and do much more than just wrap resolver functions. However, the old `directiveResolvers` API has been [left in place](directive-resolvers) for backwards compatibility, though it is now implemented in terms of `mapSchema`: +The new abstraction is more general, since it can visit any kind of schema syntax, and do much more than just wrap resolver functions. However, the old `directiveResolvers` API has been left in place for backwards compatibility, though it is now implemented in terms of `mapSchema`: ```typescript export function attachDirectiveResolvers( @@ -706,7 +702,7 @@ export function attachDirectiveResolvers( const newFieldConfig = { ...fieldConfig }; const directives = getDirectives(schema, fieldConfig); - Object.keys(directives).forEach(directiveName => { + for (const directiveName in directives) { if (directiveResolvers[directiveName]) { const resolver = directiveResolvers[directiveName]; const originalResolver = newFieldConfig.resolve != null ? newFieldConfig.resolve : defaultFieldResolver; @@ -728,7 +724,7 @@ export function attachDirectiveResolvers( ); }; } - }); + } return newFieldConfig; }, diff --git a/website/package.json b/website/package.json index 3270b8c8bba..e70bcbbcdfd 100644 --- a/website/package.json +++ b/website/package.json @@ -15,7 +15,7 @@ "react": "17.0.2", "react-dom": "17.0.2", "styled-components": "5.3.0", - "the-guild-components": "1.3.1" + "the-guild-components": "1.4.0" }, "browserslist": { "production": [ diff --git a/website/sidebars.js b/website/sidebars.js index 10f25f94a8d..b614f8cd3dc 100644 --- a/website/sidebars.js +++ b/website/sidebars.js @@ -10,7 +10,6 @@ module.exports = { "mocking", "connectors", "schema-directives", - "directive-resolvers", "schema-delegation", "remote-schemas", "schema-wrapping", diff --git a/yarn.lock b/yarn.lock index 4e51053a40f..295b563491b 100644 --- a/yarn.lock +++ b/yarn.lock @@ -144,13 +144,6 @@ tslib "^1.10.0" zen-observable "^0.8.14" -"@ardatan/aggregate-error@0.0.6": - version "0.0.6" - resolved "https://registry.yarnpkg.com/@ardatan/aggregate-error/-/aggregate-error-0.0.6.tgz#fe6924771ea40fc98dc7a7045c2e872dc8527609" - integrity sha512-vyrkEHG1jrukmzTPtyWB4NLPauUw5bQeg4uhn8f+1SSynmrOcyvlb1GKQjjgoBzElLdfXCRYX8UnBlhklOHYRQ== - dependencies: - tslib "~2.0.1" - "@babel/code-frame@7.10.4": version "7.10.4" resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.10.4.tgz#168da1a36e90da68ae8d49c0f1b48c7c6249213a" @@ -172,10 +165,10 @@ dependencies: "@babel/highlight" "^7.14.5" -"@babel/compat-data@^7.13.0", "@babel/compat-data@^7.13.11", "@babel/compat-data@^7.14.5": - version "7.14.5" - resolved "https://registry.yarnpkg.com/@babel/compat-data/-/compat-data-7.14.5.tgz#8ef4c18e58e801c5c95d3c1c0f2874a2680fadea" - integrity sha512-kixrYn4JwfAVPa0f2yfzc2AWti6WRRyO3XjWW5PJAvtE11qhSayrrcrEnee05KAtNaPC+EwehE8Qt1UedEVB8w== +"@babel/compat-data@^7.13.0", "@babel/compat-data@^7.13.11", "@babel/compat-data@^7.14.5", "@babel/compat-data@^7.14.7": + version "7.14.7" + resolved "https://registry.yarnpkg.com/@babel/compat-data/-/compat-data-7.14.7.tgz#7b047d7a3a89a67d2258dc61f604f098f1bc7e08" + integrity sha512-nS6dZaISCXJ3+518CWiBfEr//gHyMO02uDxBkXTKZDN5POruCnOZ1N4YBRZDCabwF8nZMWBpRxIicmXtBs+fvw== "@babel/core@7.12.9": version "7.12.9" @@ -455,10 +448,10 @@ chalk "^2.0.0" js-tokens "^4.0.0" -"@babel/parser@7.14.6", "@babel/parser@^7.0.0", "@babel/parser@^7.1.0", "@babel/parser@^7.12.0", "@babel/parser@^7.12.16", "@babel/parser@^7.12.5", "@babel/parser@^7.12.7", "@babel/parser@^7.13.9", "@babel/parser@^7.14.5", "@babel/parser@^7.14.6", "@babel/parser@^7.7.2": - version "7.14.6" - resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.14.6.tgz#d85cc68ca3cac84eae384c06f032921f5227f4b2" - integrity sha512-oG0ej7efjEXxb4UgE+klVx+3j4MVo+A2vCzm7OUN4CLo6WhQ+vSOD2yJ8m7B+DghObxtLxt3EfgMWpq+AsWehQ== +"@babel/parser@7.14.7", "@babel/parser@^7.0.0", "@babel/parser@^7.1.0", "@babel/parser@^7.12.0", "@babel/parser@^7.12.16", "@babel/parser@^7.12.5", "@babel/parser@^7.12.7", "@babel/parser@^7.13.9", "@babel/parser@^7.14.5", "@babel/parser@^7.14.6", "@babel/parser@^7.14.7", "@babel/parser@^7.7.2": + version "7.14.7" + resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.14.7.tgz#6099720c8839ca865a2637e6c85852ead0bdb595" + integrity sha512-X67Z5y+VBJuHB/RjwECp8kSl5uYi0BvRbNeWqkaJCVh+LiTPl19WBUfG627psSgp9rSf6ojuXghQM3ha6qHHdA== "@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining@^7.14.5": version "7.14.5" @@ -469,10 +462,10 @@ "@babel/helper-skip-transparent-expression-wrappers" "^7.14.5" "@babel/plugin-proposal-optional-chaining" "^7.14.5" -"@babel/plugin-proposal-async-generator-functions@^7.14.5": - version "7.14.5" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.14.5.tgz#4024990e3dd74181f4f426ea657769ff49a2df39" - integrity sha512-tbD/CG3l43FIXxmu4a7RBe4zH7MLJ+S/lFowPFO7HetS2hyOZ/0nnnznegDuzFzfkyQYTxqdTH/hKmuBngaDAA== +"@babel/plugin-proposal-async-generator-functions@^7.14.7": + version "7.14.7" + resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.14.7.tgz#784a48c3d8ed073f65adcf30b57bcbf6c8119ace" + integrity sha512-RK8Wj7lXLY3bqei69/cc25gwS5puEc3dknoFPFbqfy3XxYQBQFvu4ioWpafMBAB+L9NyptQK4nMOa5Xz16og8Q== dependencies: "@babel/helper-plugin-utils" "^7.14.5" "@babel/helper-remap-async-to-generator" "^7.14.5" @@ -552,12 +545,12 @@ "@babel/plugin-syntax-object-rest-spread" "^7.8.0" "@babel/plugin-transform-parameters" "^7.12.1" -"@babel/plugin-proposal-object-rest-spread@^7.0.0", "@babel/plugin-proposal-object-rest-spread@^7.14.5": - version "7.14.5" - resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.14.5.tgz#e581d5ccdfa187ea6ed73f56c6a21c1580b90fbf" - integrity sha512-VzMyY6PWNPPT3pxc5hi9LloKNr4SSrVCg7Yr6aZpW4Ym07r7KqSU/QXYwjXLVxqwSv0t/XSXkFoKBPUkZ8vb2A== +"@babel/plugin-proposal-object-rest-spread@^7.0.0", "@babel/plugin-proposal-object-rest-spread@^7.14.7": + version "7.14.7" + resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.14.7.tgz#5920a2b3df7f7901df0205974c0641b13fd9d363" + integrity sha512-082hsZz+sVabfmDWo1Oct1u1AgbKbUAyVgmX4otIc7bdsRgHBXwTwb3DpDmD4Eyyx6DNiuz5UAATT655k+kL5g== dependencies: - "@babel/compat-data" "^7.14.5" + "@babel/compat-data" "^7.14.7" "@babel/helper-compilation-targets" "^7.14.5" "@babel/helper-plugin-utils" "^7.14.5" "@babel/plugin-syntax-object-rest-spread" "^7.8.3" @@ -796,10 +789,10 @@ dependencies: "@babel/helper-plugin-utils" "^7.14.5" -"@babel/plugin-transform-destructuring@^7.0.0", "@babel/plugin-transform-destructuring@^7.14.5": - version "7.14.5" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.14.5.tgz#d32ad19ff1a6da1e861dc62720d80d9776e3bf35" - integrity sha512-wU9tYisEbRMxqDezKUqC9GleLycCRoUsai9ddlsq54r8QRLaeEhc+d+9DqCG+kV9W2GgQjTZESPTpn5bAFMDww== +"@babel/plugin-transform-destructuring@^7.0.0", "@babel/plugin-transform-destructuring@^7.14.7": + version "7.14.7" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.14.7.tgz#0ad58ed37e23e22084d109f185260835e5557576" + integrity sha512-0mDE99nK+kVh3xlc5vKwB6wnP9ecuSj+zQCa/n0voENtP/zymdT4HH6QEb65wjjcbqr1Jb/7z9Qp7TF5FtwYGw== dependencies: "@babel/helper-plugin-utils" "^7.14.5" @@ -901,10 +894,10 @@ "@babel/helper-module-transforms" "^7.14.5" "@babel/helper-plugin-utils" "^7.14.5" -"@babel/plugin-transform-named-capturing-groups-regex@^7.14.5": - version "7.14.5" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.14.5.tgz#d537e8ee083ee6f6aa4f4eef9d2081d555746e4c" - integrity sha512-+Xe5+6MWFo311U8SchgeX5c1+lJM+eZDBZgD+tvXu9VVQPXwwVzeManMMjYX6xw2HczngfOSZjoFYKwdeB/Jvw== +"@babel/plugin-transform-named-capturing-groups-regex@^7.14.7": + version "7.14.7" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.14.7.tgz#60c06892acf9df231e256c24464bfecb0908fd4e" + integrity sha512-DTNOTaS7TkW97xsDMrp7nycUVh6sn/eq22VaxWfEdzuEbRsiaOU0pqU7DlyUGHVsbQbSghvjKRpEl+nUCKGQSg== dependencies: "@babel/helper-create-regexp-features-plugin" "^7.14.5" @@ -1010,10 +1003,10 @@ dependencies: "@babel/helper-plugin-utils" "^7.14.5" -"@babel/plugin-transform-spread@^7.0.0", "@babel/plugin-transform-spread@^7.14.5": - version "7.14.5" - resolved "https://registry.yarnpkg.com/@babel/plugin-transform-spread/-/plugin-transform-spread-7.14.5.tgz#bd269fb4119754d2ce7f4cc39a96b4f71baae356" - integrity sha512-/3iqoQdiWergnShZYl0xACb4ADeYCJ7X/RgmwtXshn6cIvautRPAFzhd58frQlokLO6Jb4/3JXvmm6WNTPtiTw== +"@babel/plugin-transform-spread@^7.0.0", "@babel/plugin-transform-spread@^7.14.6": + version "7.14.6" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-spread/-/plugin-transform-spread-7.14.6.tgz#6bd40e57fe7de94aa904851963b5616652f73144" + integrity sha512-Zr0x0YroFJku7n7+/HH3A2eIrGMjbmAIbJSVv0IZ+t3U2WUQUA64S/oeied2e+MaGSjmt4alzBCsK9E8gh+fag== dependencies: "@babel/helper-plugin-utils" "^7.14.5" "@babel/helper-skip-transparent-expression-wrappers" "^7.14.5" @@ -1063,17 +1056,17 @@ "@babel/helper-create-regexp-features-plugin" "^7.14.5" "@babel/helper-plugin-utils" "^7.14.5" -"@babel/preset-env@7.14.5", "@babel/preset-env@^7.12.1", "@babel/preset-env@^7.12.16": - version "7.14.5" - resolved "https://registry.yarnpkg.com/@babel/preset-env/-/preset-env-7.14.5.tgz#c0c84e763661fd0e74292c3d511cb33b0c668997" - integrity sha512-ci6TsS0bjrdPpWGnQ+m4f+JSSzDKlckqKIJJt9UZ/+g7Zz9k0N8lYU8IeLg/01o2h8LyNZDMLGgRLDTxpudLsA== +"@babel/preset-env@7.14.7", "@babel/preset-env@^7.12.1", "@babel/preset-env@^7.12.16": + version "7.14.7" + resolved "https://registry.yarnpkg.com/@babel/preset-env/-/preset-env-7.14.7.tgz#5c70b22d4c2d893b03d8c886a5c17422502b932a" + integrity sha512-itOGqCKLsSUl0Y+1nSfhbuuOlTs0MJk2Iv7iSH+XT/mR8U1zRLO7NjWlYXB47yhK4J/7j+HYty/EhFZDYKa/VA== dependencies: - "@babel/compat-data" "^7.14.5" + "@babel/compat-data" "^7.14.7" "@babel/helper-compilation-targets" "^7.14.5" "@babel/helper-plugin-utils" "^7.14.5" "@babel/helper-validator-option" "^7.14.5" "@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining" "^7.14.5" - "@babel/plugin-proposal-async-generator-functions" "^7.14.5" + "@babel/plugin-proposal-async-generator-functions" "^7.14.7" "@babel/plugin-proposal-class-properties" "^7.14.5" "@babel/plugin-proposal-class-static-block" "^7.14.5" "@babel/plugin-proposal-dynamic-import" "^7.14.5" @@ -1082,7 +1075,7 @@ "@babel/plugin-proposal-logical-assignment-operators" "^7.14.5" "@babel/plugin-proposal-nullish-coalescing-operator" "^7.14.5" "@babel/plugin-proposal-numeric-separator" "^7.14.5" - "@babel/plugin-proposal-object-rest-spread" "^7.14.5" + "@babel/plugin-proposal-object-rest-spread" "^7.14.7" "@babel/plugin-proposal-optional-catch-binding" "^7.14.5" "@babel/plugin-proposal-optional-chaining" "^7.14.5" "@babel/plugin-proposal-private-methods" "^7.14.5" @@ -1108,7 +1101,7 @@ "@babel/plugin-transform-block-scoping" "^7.14.5" "@babel/plugin-transform-classes" "^7.14.5" "@babel/plugin-transform-computed-properties" "^7.14.5" - "@babel/plugin-transform-destructuring" "^7.14.5" + "@babel/plugin-transform-destructuring" "^7.14.7" "@babel/plugin-transform-dotall-regex" "^7.14.5" "@babel/plugin-transform-duplicate-keys" "^7.14.5" "@babel/plugin-transform-exponentiation-operator" "^7.14.5" @@ -1120,7 +1113,7 @@ "@babel/plugin-transform-modules-commonjs" "^7.14.5" "@babel/plugin-transform-modules-systemjs" "^7.14.5" "@babel/plugin-transform-modules-umd" "^7.14.5" - "@babel/plugin-transform-named-capturing-groups-regex" "^7.14.5" + "@babel/plugin-transform-named-capturing-groups-regex" "^7.14.7" "@babel/plugin-transform-new-target" "^7.14.5" "@babel/plugin-transform-object-super" "^7.14.5" "@babel/plugin-transform-parameters" "^7.14.5" @@ -1128,7 +1121,7 @@ "@babel/plugin-transform-regenerator" "^7.14.5" "@babel/plugin-transform-reserved-words" "^7.14.5" "@babel/plugin-transform-shorthand-properties" "^7.14.5" - "@babel/plugin-transform-spread" "^7.14.5" + "@babel/plugin-transform-spread" "^7.14.6" "@babel/plugin-transform-sticky-regex" "^7.14.5" "@babel/plugin-transform-template-literals" "^7.14.5" "@babel/plugin-transform-typeof-symbol" "^7.14.5" @@ -1139,7 +1132,7 @@ babel-plugin-polyfill-corejs2 "^0.2.2" babel-plugin-polyfill-corejs3 "^0.2.2" babel-plugin-polyfill-regenerator "^0.2.2" - core-js-compat "^3.14.0" + core-js-compat "^3.15.0" semver "^6.3.0" "@babel/preset-modules@^0.1.4": @@ -1197,17 +1190,17 @@ "@babel/parser" "^7.14.5" "@babel/types" "^7.14.5" -"@babel/traverse@7.14.5", "@babel/traverse@^7.0.0", "@babel/traverse@^7.1.0", "@babel/traverse@^7.12.13", "@babel/traverse@^7.12.9", "@babel/traverse@^7.13.0", "@babel/traverse@^7.14.5", "@babel/traverse@^7.4.5", "@babel/traverse@^7.7.2": - version "7.14.5" - resolved "https://registry.yarnpkg.com/@babel/traverse/-/traverse-7.14.5.tgz#c111b0f58afab4fea3d3385a406f692748c59870" - integrity sha512-G3BiS15vevepdmFqmUc9X+64y0viZYygubAMO8SvBmKARuF6CPSZtH4Ng9vi/lrWlZFGe3FWdXNy835akH8Glg== +"@babel/traverse@7.14.7", "@babel/traverse@^7.0.0", "@babel/traverse@^7.1.0", "@babel/traverse@^7.12.13", "@babel/traverse@^7.12.9", "@babel/traverse@^7.13.0", "@babel/traverse@^7.14.5", "@babel/traverse@^7.4.5", "@babel/traverse@^7.7.2": + version "7.14.7" + resolved "https://registry.yarnpkg.com/@babel/traverse/-/traverse-7.14.7.tgz#64007c9774cfdc3abd23b0780bc18a3ce3631753" + integrity sha512-9vDr5NzHu27wgwejuKL7kIOm4bwEtaPQ4Z6cpCmjSuaRqpH/7xc4qcGEscwMqlkwgcXl6MvqoAjZkQ24uSdIZQ== dependencies: "@babel/code-frame" "^7.14.5" "@babel/generator" "^7.14.5" "@babel/helper-function-name" "^7.14.5" "@babel/helper-hoist-variables" "^7.14.5" "@babel/helper-split-export-declaration" "^7.14.5" - "@babel/parser" "^7.14.5" + "@babel/parser" "^7.14.7" "@babel/types" "^7.14.5" debug "^4.1.0" globals "^11.1.0" @@ -1908,15 +1901,6 @@ dependencies: purgecss "^3.1.3" -"@graphql-tools/mock@^7.0.0": - version "7.0.0" - resolved "https://registry.yarnpkg.com/@graphql-tools/mock/-/mock-7.0.0.tgz#b43858f47fedfbf7d8bbbf7d33e6acb64b8b7da7" - integrity sha512-ShO8D9HudgnhqoWeKb3iejGtPV8elFqSO1U0O70g3FH3W/CBW2abXfuyodBUevXVGIjyqzfkNzVtpIE0qiOVVQ== - dependencies: - "@graphql-tools/schema" "^7.0.0" - "@graphql-tools/utils" "^7.0.0" - tslib "~2.0.1" - "@graphql-typed-document-node/core@^3.0.0": version "3.1.0" resolved "https://registry.yarnpkg.com/@graphql-typed-document-node/core/-/core-3.1.0.tgz#0eee6373e11418bfe0b5638f654df7a4ca6a3950" @@ -1962,15 +1946,15 @@ jest-util "^27.0.2" slash "^3.0.0" -"@jest/core@^27.0.4": - version "27.0.4" - resolved "https://registry.yarnpkg.com/@jest/core/-/core-27.0.4.tgz#679bf9ac07900da2ddbb9667bb1afa8029038f53" - integrity sha512-+dsmV8VUs1h/Szb+rEWk8xBM1fp1I///uFy9nk3wXGvRsF2lBp8EVPmtWc+QFRb3MY2b7u2HbkGF1fzoDzQTLA== +"@jest/core@^27.0.5": + version "27.0.5" + resolved "https://registry.yarnpkg.com/@jest/core/-/core-27.0.5.tgz#59e9e69e7374d65dbb22e3fc1bd52e80991eae72" + integrity sha512-g73//jF0VwsOIrWUC9Cqg03lU3QoAMFxVjsm6n6yNmwZcQPN/o8w+gLWODw5VfKNFZT38otXHWxc6b8eGDUpEA== dependencies: "@jest/console" "^27.0.2" - "@jest/reporters" "^27.0.4" + "@jest/reporters" "^27.0.5" "@jest/test-result" "^27.0.2" - "@jest/transform" "^27.0.2" + "@jest/transform" "^27.0.5" "@jest/types" "^27.0.2" "@types/node" "*" ansi-escapes "^4.2.1" @@ -1979,15 +1963,15 @@ exit "^0.1.2" graceful-fs "^4.2.4" jest-changed-files "^27.0.2" - jest-config "^27.0.4" - jest-haste-map "^27.0.2" + jest-config "^27.0.5" + jest-haste-map "^27.0.5" jest-message-util "^27.0.2" jest-regex-util "^27.0.1" - jest-resolve "^27.0.4" - jest-resolve-dependencies "^27.0.4" - jest-runner "^27.0.4" - jest-runtime "^27.0.4" - jest-snapshot "^27.0.4" + jest-resolve "^27.0.5" + jest-resolve-dependencies "^27.0.5" + jest-runner "^27.0.5" + jest-runtime "^27.0.5" + jest-snapshot "^27.0.5" jest-util "^27.0.2" jest-validate "^27.0.2" jest-watcher "^27.0.2" @@ -1997,20 +1981,20 @@ slash "^3.0.0" strip-ansi "^6.0.0" -"@jest/environment@^27.0.3": - version "27.0.3" - resolved "https://registry.yarnpkg.com/@jest/environment/-/environment-27.0.3.tgz#68769b1dfdd213e3456169d64fbe9bd63a5fda92" - integrity sha512-pN9m7fbKsop5vc3FOfH8NF7CKKdRbEZzcxfIo1n2TT6ucKWLFq0P6gCJH0GpnQp036++yY9utHOxpeT1WnkWTA== +"@jest/environment@^27.0.5": + version "27.0.5" + resolved "https://registry.yarnpkg.com/@jest/environment/-/environment-27.0.5.tgz#a294ad4acda2e250f789fb98dc667aad33d3adc9" + integrity sha512-IAkJPOT7bqn0GiX5LPio6/e1YpcmLbrd8O5EFYpAOZ6V+9xJDsXjdgN2vgv9WOKIs/uA1kf5WeD96HhlBYO+FA== dependencies: - "@jest/fake-timers" "^27.0.3" + "@jest/fake-timers" "^27.0.5" "@jest/types" "^27.0.2" "@types/node" "*" jest-mock "^27.0.3" -"@jest/fake-timers@^27.0.3": - version "27.0.3" - resolved "https://registry.yarnpkg.com/@jest/fake-timers/-/fake-timers-27.0.3.tgz#9899ba6304cc636734c74478df502e18136461dd" - integrity sha512-fQ+UCKRIYKvTCEOyKPnaPnomLATIhMnHC/xPZ7yT1Uldp7yMgMxoYIFidDbpSTgB79+/U+FgfoD30c6wg3IUjA== +"@jest/fake-timers@^27.0.5": + version "27.0.5" + resolved "https://registry.yarnpkg.com/@jest/fake-timers/-/fake-timers-27.0.5.tgz#304d5aedadf4c75cff3696995460b39d6c6e72f6" + integrity sha512-d6Tyf7iDoKqeUdwUKrOBV/GvEZRF67m7lpuWI0+SCD9D3aaejiOQZxAOxwH2EH/W18gnfYaBPLi0VeTGBHtQBg== dependencies: "@jest/types" "^27.0.2" "@sinonjs/fake-timers" "^7.0.2" @@ -2019,24 +2003,24 @@ jest-mock "^27.0.3" jest-util "^27.0.2" -"@jest/globals@^27.0.3": - version "27.0.3" - resolved "https://registry.yarnpkg.com/@jest/globals/-/globals-27.0.3.tgz#1cf8933b7791bba0b99305cbf39fd4d2e3fe4060" - integrity sha512-OzsIuf7uf+QalqAGbjClyezzEcLQkdZ+7PejUrZgDs+okdAK8GwRCGcYCirHvhMBBQh60Jr3NlIGbn/KBPQLEQ== +"@jest/globals@^27.0.5": + version "27.0.5" + resolved "https://registry.yarnpkg.com/@jest/globals/-/globals-27.0.5.tgz#f63b8bfa6ea3716f8df50f6a604b5c15b36ffd20" + integrity sha512-qqKyjDXUaZwDuccpbMMKCCMBftvrbXzigtIsikAH/9ca+kaae8InP2MDf+Y/PdCSMuAsSpHS6q6M25irBBUh+Q== dependencies: - "@jest/environment" "^27.0.3" + "@jest/environment" "^27.0.5" "@jest/types" "^27.0.2" expect "^27.0.2" -"@jest/reporters@^27.0.4": - version "27.0.4" - resolved "https://registry.yarnpkg.com/@jest/reporters/-/reporters-27.0.4.tgz#95609b1be97afb80d55d8aa3d7c3179c15810e65" - integrity sha512-Xa90Nm3JnV0xCe4M6A10M9WuN9krb+WFKxV1A98Y4ePCw40n++r7uxFUNU7DT1i9Behj7fjrAIju9oU0t1QtCg== +"@jest/reporters@^27.0.5": + version "27.0.5" + resolved "https://registry.yarnpkg.com/@jest/reporters/-/reporters-27.0.5.tgz#cd730b77d9667b8ff700ad66d4edc293bb09716a" + integrity sha512-4uNg5+0eIfRafnpgu3jCZws3NNcFzhu5JdRd1mKQ4/53+vkIqwB6vfZ4gn5BdGqOaLtYhlOsPaL5ATkKzyBrJw== dependencies: "@bcoe/v8-coverage" "^0.2.3" "@jest/console" "^27.0.2" "@jest/test-result" "^27.0.2" - "@jest/transform" "^27.0.2" + "@jest/transform" "^27.0.5" "@jest/types" "^27.0.2" chalk "^4.0.0" collect-v8-coverage "^1.0.0" @@ -2048,15 +2032,15 @@ istanbul-lib-report "^3.0.0" istanbul-lib-source-maps "^4.0.0" istanbul-reports "^3.0.2" - jest-haste-map "^27.0.2" - jest-resolve "^27.0.4" + jest-haste-map "^27.0.5" + jest-resolve "^27.0.5" jest-util "^27.0.2" jest-worker "^27.0.2" slash "^3.0.0" source-map "^0.6.0" string-length "^4.0.1" terminal-link "^2.0.0" - v8-to-istanbul "^7.0.0" + v8-to-istanbul "^8.0.0" "@jest/source-map@^27.0.1": version "27.0.1" @@ -2077,20 +2061,20 @@ "@types/istanbul-lib-coverage" "^2.0.0" collect-v8-coverage "^1.0.0" -"@jest/test-sequencer@^27.0.4": - version "27.0.4" - resolved "https://registry.yarnpkg.com/@jest/test-sequencer/-/test-sequencer-27.0.4.tgz#976493b277594d81e589896f0ed21f198308928a" - integrity sha512-6UFEVwdmxYdyNffBxVVZxmXEdBE4riSddXYSnFNH0ELFQFk/bvagizim8WfgJTqF4EKd+j1yFxvhb8BMHfOjSQ== +"@jest/test-sequencer@^27.0.5": + version "27.0.5" + resolved "https://registry.yarnpkg.com/@jest/test-sequencer/-/test-sequencer-27.0.5.tgz#c58b21db49afc36c0e3921d7ddf1fb7954abfded" + integrity sha512-opztnGs+cXzZ5txFG2+omBaV5ge/0yuJNKbhE3DREMiXE0YxBuzyEa6pNv3kk2JuucIlH2Xvgmn9kEEHSNt/SA== dependencies: "@jest/test-result" "^27.0.2" graceful-fs "^4.2.4" - jest-haste-map "^27.0.2" - jest-runtime "^27.0.4" + jest-haste-map "^27.0.5" + jest-runtime "^27.0.5" -"@jest/transform@^27.0.2": - version "27.0.2" - resolved "https://registry.yarnpkg.com/@jest/transform/-/transform-27.0.2.tgz#b073b7c589e3f4b842102468875def2bb722d6b5" - integrity sha512-H8sqKlgtDfVog/s9I4GG2XMbi4Ar7RBxjsKQDUhn2XHAi3NG+GoQwWMER+YfantzExbjNqQvqBHzo/G2pfTiPw== +"@jest/transform@^27.0.2", "@jest/transform@^27.0.5": + version "27.0.5" + resolved "https://registry.yarnpkg.com/@jest/transform/-/transform-27.0.5.tgz#2dcb78953708af713941ac845b06078bc74ed873" + integrity sha512-lBD6OwKXSc6JJECBNk4mVxtSVuJSBsQrJ9WCBisfJs7EZuYq4K6vM9HmoB7hmPiLIDGeyaerw3feBV/bC4z8tg== dependencies: "@babel/core" "^7.1.0" "@jest/types" "^27.0.2" @@ -2099,7 +2083,7 @@ convert-source-map "^1.4.0" fast-json-stable-stringify "^2.0.0" graceful-fs "^4.2.4" - jest-haste-map "^27.0.2" + jest-haste-map "^27.0.5" jest-regex-util "^27.0.1" jest-util "^27.0.2" micromatch "^4.0.4" @@ -2383,10 +2367,10 @@ dependencies: defer-to-connect "^1.0.1" -"@theguild/components@1.3.1": - version "1.3.1" - resolved "https://registry.yarnpkg.com/@theguild/components/-/components-1.3.1.tgz#8a16e2ece3f59a5d2af769d0e21b348585ca303d" - integrity sha512-ZgLZfUVclh42tU3FThjx9hK6sXJzDnhl9wolv42vfzuntmXfK/64St6mmUZzEH8cB0vBPh6CCa5MfMN4u+H/UQ== +"@theguild/components@1.4.0": + version "1.4.0" + resolved "https://registry.yarnpkg.com/@theguild/components/-/components-1.4.0.tgz#7484e9f30196899f435b6f836c1eaa97e332fdb6" + integrity sha512-BKsiq6QN9LXFnGmOyMRgKlR3E3qYdAQsFys8SWOf/1CFidzEMQXmtNNn0RaWxW4f9BtNBPm4BvAu6wR2CHr21w== dependencies: "@emotion/react" "^11.4.0" "@emotion/styled" "^11.3.0" @@ -2399,7 +2383,8 @@ react-player "^2.9.0" tailwindcss "^2.1.4" twin.macro "^2.4.2" - use-debounce "^6.0.1" + use-debounce "^7.0.0" + validator "13.6.0" "@tootallnate/once@1": version "1.1.2" @@ -2516,16 +2501,21 @@ "@types/estree" "*" "@types/json-schema" "*" -"@types/estree@*", "@types/estree@^0.0.47": - version "0.0.47" - resolved "https://registry.yarnpkg.com/@types/estree/-/estree-0.0.47.tgz#d7a51db20f0650efec24cd04994f523d93172ed4" - integrity sha512-c5ciR06jK8u9BstrmJyO97m+klJrrhCf9u3rLu3DEAJBirxRqSCvDQoYKmxuYwQI5SZChAWu+tq9oVlGRuzPAg== +"@types/estree@*", "@types/estree@^0.0.48": + version "0.0.48" + resolved "https://registry.yarnpkg.com/@types/estree/-/estree-0.0.48.tgz#18dc8091b285df90db2f25aa7d906cfc394b7f74" + integrity sha512-LfZwXoGUDo0C3me81HXgkBg5CTQYb6xzEl+fNmbO4JdRiSKQ8A0GD1OBBvKAIsbCUgoyAty7m99GqqMQe784ew== "@types/estree@0.0.39": version "0.0.39" resolved "https://registry.yarnpkg.com/@types/estree/-/estree-0.0.39.tgz#e177e699ee1b8c22d23174caaa7422644389509f" integrity sha512-EYNwp3bU+98cpU4lAWYYL7Zz+2gryWH1qbdDTidVd6hkiR6weksdbMadyXKXNPEkQFhXM+hVO9ZygomHXp+AIw== +"@types/estree@^0.0.47": + version "0.0.47" + resolved "https://registry.yarnpkg.com/@types/estree/-/estree-0.0.47.tgz#d7a51db20f0650efec24cd04994f523d93172ed4" + integrity sha512-c5ciR06jK8u9BstrmJyO97m+klJrrhCf9u3rLu3DEAJBirxRqSCvDQoYKmxuYwQI5SZChAWu+tq9oVlGRuzPAg== + "@types/express-serve-static-core@^4.17.18": version "4.17.19" resolved "https://registry.yarnpkg.com/@types/express-serve-static-core/-/express-serve-static-core-4.17.19.tgz#00acfc1632e729acac4f1530e9e16f6dd1508a1d" @@ -2609,13 +2599,6 @@ resolved "https://registry.yarnpkg.com/@types/http-errors/-/http-errors-1.8.0.tgz#682477dbbbd07cd032731cb3b0e7eaee3d026b69" integrity sha512-2aoSC4UUbHDj2uCsCxcG/vRMXey/m17bC7UwitVm5hn22nI8O8Y9iDpA76Orc+DWkQ4zZrOKEshCqR/jSuXAHA== -"@types/http-proxy-agent@^4.0.0": - version "4.0.0" - resolved "https://registry.yarnpkg.com/@types/http-proxy-agent/-/http-proxy-agent-4.0.0.tgz#6aa52287365115da1667384182ccc2f42603326d" - integrity sha512-KHJ5VLu4L0YBj+ZeDKVnGloSMmgdzLlhfSVL8XSwwv5SRoeWVEcco7kNpEIzzsS95lncn43J9vW3m88eX7Yzeg== - dependencies: - http-proxy-agent "*" - "@types/is-glob@4.0.1": version "4.0.1" resolved "https://registry.yarnpkg.com/@types/is-glob/-/is-glob-4.0.1.tgz#a93eec1714172c8eb3225a1cc5eb88c2477b7d00" @@ -2740,10 +2723,10 @@ resolved "https://registry.yarnpkg.com/@types/node/-/node-15.6.0.tgz#f0ddca5a61e52627c9dcb771a6039d44694597bc" integrity sha512-gCYSfQpy+LYhOFTKAeE8BkyGqaxmlFxe+n4DKM6DR0wzw/HISUE/hAmkC/KT8Sw5PCJblqg062b3z9gucv3k0A== -"@types/node@14.17.3": - version "14.17.3" - resolved "https://registry.yarnpkg.com/@types/node/-/node-14.17.3.tgz#6d327abaa4be34a74e421ed6409a0ae2f47f4c3d" - integrity sha512-e6ZowgGJmTuXa3GyaPbTGxX17tnThl2aSSizrFthQ7m9uLGZBXiGhgE55cjRZTF5kjZvYn9EOPOMljdjwbflxw== +"@types/node@14.17.4": + version "14.17.4" + resolved "https://registry.yarnpkg.com/@types/node/-/node-14.17.4.tgz#218712242446fc868d0e007af29a4408c7765bc0" + integrity sha512-8kQ3+wKGRNN0ghtEn7EGps/B8CzuBz1nXZEIGGLP2GnwbqYn4dbTs7k+VKLTq1HvZLRCIDtN3Snx1Ege8B7L5A== "@types/node@^12.7.1": version "12.20.6" @@ -2888,74 +2871,73 @@ resolved "https://registry.yarnpkg.com/@types/zen-observable/-/zen-observable-0.8.2.tgz#808c9fa7e4517274ed555fa158f2de4b4f468e71" integrity sha512-HrCIVMLjE1MOozVoD86622S7aunluLb2PJdPfb3nYiEtohm8mIB/vyv0Fd37AdeMFrTUQXEunw78YloMA3Qilg== -"@typescript-eslint/eslint-plugin@4.27.0": - version "4.27.0" - resolved "https://registry.yarnpkg.com/@typescript-eslint/eslint-plugin/-/eslint-plugin-4.27.0.tgz#0b7fc974e8bc9b2b5eb98ed51427b0be529b4ad0" - integrity sha512-DsLqxeUfLVNp3AO7PC3JyaddmEHTtI9qTSAs+RB6ja27QvIM0TA8Cizn1qcS6vOu+WDLFJzkwkgweiyFhssDdQ== +"@typescript-eslint/eslint-plugin@4.28.0": + version "4.28.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/eslint-plugin/-/eslint-plugin-4.28.0.tgz#1a66f03b264844387beb7dc85e1f1d403bd1803f" + integrity sha512-KcF6p3zWhf1f8xO84tuBailV5cN92vhS+VT7UJsPzGBm9VnQqfI9AsiMUFUCYHTYPg1uCCo+HyiDnpDuvkAMfQ== dependencies: - "@typescript-eslint/experimental-utils" "4.27.0" - "@typescript-eslint/scope-manager" "4.27.0" + "@typescript-eslint/experimental-utils" "4.28.0" + "@typescript-eslint/scope-manager" "4.28.0" debug "^4.3.1" functional-red-black-tree "^1.0.1" - lodash "^4.17.21" regexpp "^3.1.0" semver "^7.3.5" tsutils "^3.21.0" -"@typescript-eslint/experimental-utils@4.27.0": - version "4.27.0" - resolved "https://registry.yarnpkg.com/@typescript-eslint/experimental-utils/-/experimental-utils-4.27.0.tgz#78192a616472d199f084eab8f10f962c0757cd1c" - integrity sha512-n5NlbnmzT2MXlyT+Y0Jf0gsmAQzCnQSWXKy4RGSXVStjDvS5we9IWbh7qRVKdGcxT0WYlgcCYUK/HRg7xFhvjQ== +"@typescript-eslint/experimental-utils@4.28.0": + version "4.28.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/experimental-utils/-/experimental-utils-4.28.0.tgz#13167ed991320684bdc23588135ae62115b30ee0" + integrity sha512-9XD9s7mt3QWMk82GoyUpc/Ji03vz4T5AYlHF9DcoFNfJ/y3UAclRsfGiE2gLfXtyC+JRA3trR7cR296TEb1oiQ== dependencies: "@types/json-schema" "^7.0.7" - "@typescript-eslint/scope-manager" "4.27.0" - "@typescript-eslint/types" "4.27.0" - "@typescript-eslint/typescript-estree" "4.27.0" + "@typescript-eslint/scope-manager" "4.28.0" + "@typescript-eslint/types" "4.28.0" + "@typescript-eslint/typescript-estree" "4.28.0" eslint-scope "^5.1.1" eslint-utils "^3.0.0" -"@typescript-eslint/parser@4.27.0": - version "4.27.0" - resolved "https://registry.yarnpkg.com/@typescript-eslint/parser/-/parser-4.27.0.tgz#85447e573364bce4c46c7f64abaa4985aadf5a94" - integrity sha512-XpbxL+M+gClmJcJ5kHnUpBGmlGdgNvy6cehgR6ufyxkEJMGP25tZKCaKyC0W/JVpuhU3VU1RBn7SYUPKSMqQvQ== +"@typescript-eslint/parser@4.28.0": + version "4.28.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/parser/-/parser-4.28.0.tgz#2404c16751a28616ef3abab77c8e51d680a12caa" + integrity sha512-7x4D22oPY8fDaOCvkuXtYYTQ6mTMmkivwEzS+7iml9F9VkHGbbZ3x4fHRwxAb5KeuSkLqfnYjs46tGx2Nour4A== dependencies: - "@typescript-eslint/scope-manager" "4.27.0" - "@typescript-eslint/types" "4.27.0" - "@typescript-eslint/typescript-estree" "4.27.0" + "@typescript-eslint/scope-manager" "4.28.0" + "@typescript-eslint/types" "4.28.0" + "@typescript-eslint/typescript-estree" "4.28.0" debug "^4.3.1" -"@typescript-eslint/scope-manager@4.27.0": - version "4.27.0" - resolved "https://registry.yarnpkg.com/@typescript-eslint/scope-manager/-/scope-manager-4.27.0.tgz#b0b1de2b35aaf7f532e89c8e81d0fa298cae327d" - integrity sha512-DY73jK6SEH6UDdzc6maF19AHQJBFVRf6fgAXHPXCGEmpqD4vYgPEzqpFz1lf/daSbOcMpPPj9tyXXDPW2XReAw== +"@typescript-eslint/scope-manager@4.28.0": + version "4.28.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/scope-manager/-/scope-manager-4.28.0.tgz#6a3009d2ab64a30fc8a1e257a1a320067f36a0ce" + integrity sha512-eCALCeScs5P/EYjwo6se9bdjtrh8ByWjtHzOkC4Tia6QQWtQr3PHovxh3TdYTuFcurkYI4rmFsRFpucADIkseg== dependencies: - "@typescript-eslint/types" "4.27.0" - "@typescript-eslint/visitor-keys" "4.27.0" + "@typescript-eslint/types" "4.28.0" + "@typescript-eslint/visitor-keys" "4.28.0" -"@typescript-eslint/types@4.27.0": - version "4.27.0" - resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-4.27.0.tgz#712b408519ed699baff69086bc59cd2fc13df8d8" - integrity sha512-I4ps3SCPFCKclRcvnsVA/7sWzh7naaM/b4pBO2hVxnM3wrU51Lveybdw5WoIktU/V4KfXrTt94V9b065b/0+wA== +"@typescript-eslint/types@4.28.0": + version "4.28.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-4.28.0.tgz#a33504e1ce7ac51fc39035f5fe6f15079d4dafb0" + integrity sha512-p16xMNKKoiJCVZY5PW/AfILw2xe1LfruTcfAKBj3a+wgNYP5I9ZEKNDOItoRt53p4EiPV6iRSICy8EPanG9ZVA== -"@typescript-eslint/typescript-estree@4.27.0": - version "4.27.0" - resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-4.27.0.tgz#189a7b9f1d0717d5cccdcc17247692dedf7a09da" - integrity sha512-KH03GUsUj41sRLLEy2JHstnezgpS5VNhrJouRdmh6yNdQ+yl8w5LrSwBkExM+jWwCJa7Ct2c8yl8NdtNRyQO6g== +"@typescript-eslint/typescript-estree@4.28.0": + version "4.28.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-4.28.0.tgz#e66d4e5aa2ede66fec8af434898fe61af10c71cf" + integrity sha512-m19UQTRtxMzKAm8QxfKpvh6OwQSXaW1CdZPoCaQuLwAq7VZMNuhJmZR4g5281s2ECt658sldnJfdpSZZaxUGMQ== dependencies: - "@typescript-eslint/types" "4.27.0" - "@typescript-eslint/visitor-keys" "4.27.0" + "@typescript-eslint/types" "4.28.0" + "@typescript-eslint/visitor-keys" "4.28.0" debug "^4.3.1" globby "^11.0.3" is-glob "^4.0.1" semver "^7.3.5" tsutils "^3.21.0" -"@typescript-eslint/visitor-keys@4.27.0": - version "4.27.0" - resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-4.27.0.tgz#f56138b993ec822793e7ebcfac6ffdce0a60cb81" - integrity sha512-es0GRYNZp0ieckZ938cEANfEhsfHrzuLrePukLKtY3/KPXcq1Xd555Mno9/GOgXhKzn0QfkDLVgqWO3dGY80bg== +"@typescript-eslint/visitor-keys@4.28.0": + version "4.28.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-4.28.0.tgz#255c67c966ec294104169a6939d96f91c8a89434" + integrity sha512-PjJyTWwrlrvM5jazxYF5ZPs/nl0kHDZMVbuIcbpawVXaDPelp3+S9zpOz5RmVUfS/fD5l5+ZXNKnWhNYjPzCvw== dependencies: - "@typescript-eslint/types" "4.27.0" + "@typescript-eslint/types" "4.28.0" eslint-visitor-keys "^2.0.0" "@vercel/ncc@0.28.3": @@ -2963,36 +2945,37 @@ resolved "https://registry.yarnpkg.com/@vercel/ncc/-/ncc-0.28.3.tgz#9461bdbf334d616759b0e7e5415e2f480b9aa30f" integrity sha512-g3gk4D9itbhUQa5MtN7TOdeoQnNLkPDCox5SBaQ/H3Or5lo59TOaZWrLb+x47StiAJ+8DXZS/9MJ67cIBWSsRw== -"@vue/compiler-core@3.1.1": - version "3.1.1" - resolved "https://registry.yarnpkg.com/@vue/compiler-core/-/compiler-core-3.1.1.tgz#4f2c5d70eabd454675714cc8bd2b97f6a8efb196" - integrity sha512-Z1RO3T6AEtAUFf2EqqovFm3ohAeTvFzRtB0qUENW2nEerJfdlk13/LS1a0EgsqlzxmYfR/S/S/gW9PLbFZZxkA== +"@vue/compiler-core@3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vue/compiler-core/-/compiler-core-3.1.2.tgz#31ab1d88e1706a5c7a545faeeb64c31bd0101db0" + integrity sha512-nHmq7vLjq/XM2IMbZUcKWoH5sPXa2uR/nIKZtjbK5F3TcbnYE/zKsrSUR9WZJ03unlwotNBX1OyxVt9HbWD7/Q== dependencies: "@babel/parser" "^7.12.0" "@babel/types" "^7.12.0" - "@vue/shared" "3.1.1" + "@vue/shared" "3.1.2" estree-walker "^2.0.1" source-map "^0.6.1" -"@vue/compiler-dom@3.1.1": - version "3.1.1" - resolved "https://registry.yarnpkg.com/@vue/compiler-dom/-/compiler-dom-3.1.1.tgz#ef60d856ac2ede5b2ad5c72a7a68122895e3d652" - integrity sha512-nobRIo0t5ibzg+q8nC31m+aJhbq8FbWUoKvk6h3Vs1EqTDJaj6lBTcVTq5or8AYht7FbSpdAJ81isbJ1rWNX7A== +"@vue/compiler-dom@3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vue/compiler-dom/-/compiler-dom-3.1.2.tgz#75a7731bcc5d9718183a3c56c18e992f7c13e7b1" + integrity sha512-k2+SWcWH0jL6WQAX7Or2ONqu5MbtTgTO0dJrvebQYzgqaKMXNI90RNeWeCxS4BnNFMDONpHBeFgbwbnDWIkmRg== dependencies: - "@vue/compiler-core" "3.1.1" - "@vue/shared" "3.1.1" + "@vue/compiler-core" "3.1.2" + "@vue/shared" "3.1.2" -"@vue/compiler-sfc@3.1.1": - version "3.1.1" - resolved "https://registry.yarnpkg.com/@vue/compiler-sfc/-/compiler-sfc-3.1.1.tgz#d4e4507c013d0b219f0b106b317ec5bb1cde3398" - integrity sha512-lSgMsZaYHF+bAgryq5aUqpvyfhu52GJI2/4LoiJCE5uaxc6FCZfxfgqgw/d9ltiZghv+HiISFtmQVAVvlsk+/w== +"@vue/compiler-sfc@3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vue/compiler-sfc/-/compiler-sfc-3.1.2.tgz#23ff1e366d887b964899568bffcb11e3d0511fc4" + integrity sha512-SeG/2+DvwejQ7oAiSx8BrDh5qOdqCYHGClPiTvVIHTfSIHiS2JjMbCANdDCjHkTOh/O7WZzo2JhdKm98bRBxTw== dependencies: "@babel/parser" "^7.13.9" "@babel/types" "^7.13.0" - "@vue/compiler-core" "3.1.1" - "@vue/compiler-dom" "3.1.1" - "@vue/compiler-ssr" "3.1.1" - "@vue/shared" "3.1.1" + "@types/estree" "^0.0.48" + "@vue/compiler-core" "3.1.2" + "@vue/compiler-dom" "3.1.2" + "@vue/compiler-ssr" "3.1.2" + "@vue/shared" "3.1.2" consolidate "^0.16.0" estree-walker "^2.0.1" hash-sum "^2.0.0" @@ -3004,18 +2987,18 @@ postcss-selector-parser "^6.0.4" source-map "^0.6.1" -"@vue/compiler-ssr@3.1.1": - version "3.1.1" - resolved "https://registry.yarnpkg.com/@vue/compiler-ssr/-/compiler-ssr-3.1.1.tgz#1d08b98601397258ed059b75966e0e94a385d770" - integrity sha512-7H6krZtVt3h/YzfNp7eYK41hMDz8ZskiBy+Wby+EDRINX6BD9JQ5C8zyy2xAa7T6Iz2VrQzsaJ/Bb52lTPSS5A== +"@vue/compiler-ssr@3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vue/compiler-ssr/-/compiler-ssr-3.1.2.tgz#e33ad0876d9b96f0950e22b0e174b94c1b049d2d" + integrity sha512-BwXo9LFk5OSWdMyZQ4bX1ELHX0Z/9F+ld/OaVnpUPzAZCHslBYLvyKUVDwv2C/lpLjRffpC2DOUEdl1+RP1aGg== dependencies: - "@vue/compiler-dom" "3.1.1" - "@vue/shared" "3.1.1" + "@vue/compiler-dom" "3.1.2" + "@vue/shared" "3.1.2" -"@vue/shared@3.1.1": - version "3.1.1" - resolved "https://registry.yarnpkg.com/@vue/shared/-/shared-3.1.1.tgz#2287cfc3dc20e5b20aeb65c2c3a56533bdca801c" - integrity sha512-g+4pzAw7PYSjARtLBoDq6DmcblX8i9KJHSCnyM5VDDFFifUaUT9iHbFpOF/KOizQ9f7QAqU2JH3Y6aXjzUMhVA== +"@vue/shared@3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vue/shared/-/shared-3.1.2.tgz#1069c0bc7d6f4bd15ccf3a5f3be29450aca368f9" + integrity sha512-EmH/poaDWBPJaPILXNI/1fvUbArJQmmTyVCwvvyDYDFnkPoTclAbHRAtyIvqfez7jybTDn077HTNILpxlsoWhg== "@webassemblyjs/ast@1.11.0": version "1.11.0" @@ -3571,12 +3554,12 @@ axios@^0.21.1: dependencies: follow-redirects "^1.10.0" -babel-jest@27.0.2, babel-jest@^27.0.2: - version "27.0.2" - resolved "https://registry.yarnpkg.com/babel-jest/-/babel-jest-27.0.2.tgz#7dc18adb01322acce62c2af76ea2c7cd186ade37" - integrity sha512-9OThPl3/IQbo4Yul2vMz4FYwILPQak8XelX4YGowygfHaOl5R5gfjm4iVx4d8aUugkW683t8aq0A74E7b5DU1Q== +babel-jest@27.0.5, babel-jest@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/babel-jest/-/babel-jest-27.0.5.tgz#cd34c033ada05d1362211e5152391fd7a88080c8" + integrity sha512-bTMAbpCX7ldtfbca2llYLeSFsDM257aspyAOpsdrdSrBqoLkWCy4HPYTXtXWaSLgFPjrJGACL65rzzr4RFGadw== dependencies: - "@jest/transform" "^27.0.2" + "@jest/transform" "^27.0.5" "@jest/types" "^27.0.2" "@types/babel__core" "^7.1.14" babel-plugin-istanbul "^6.0.0" @@ -4113,7 +4096,7 @@ callsites@^3.0.0: resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73" integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ== -camel-case@4.1.2, camel-case@^4.1.1: +camel-case@^4.1.1: version "4.1.2" resolved "https://registry.yarnpkg.com/camel-case/-/camel-case-4.1.2.tgz#9728072a954f805228225a6deea6b38461e1bd5a" integrity sha512-gxGWBrTT1JuMx6R+o5PTXMmUnhnVzLQ9SNutD4YqKtI6ap897t3tKECYla6gCWEkplXnlNybEkZg9GEGxKFCgw== @@ -4702,10 +4685,10 @@ copy-webpack-plugin@^8.1.1: schema-utils "^3.0.0" serialize-javascript "^5.0.1" -core-js-compat@^3.14.0, core-js-compat@^3.8.1, core-js-compat@^3.9.1: - version "3.14.0" - resolved "https://registry.yarnpkg.com/core-js-compat/-/core-js-compat-3.14.0.tgz#b574dabf29184681d5b16357bd33d104df3d29a5" - integrity sha512-R4NS2eupxtiJU+VwgkF9WTpnSfZW4pogwKHd8bclWU2sp93Pr5S1uYJI84cMOubJRou7bcfL0vmwtLslWN5p3A== +core-js-compat@^3.15.0, core-js-compat@^3.8.1, core-js-compat@^3.9.1: + version "3.15.0" + resolved "https://registry.yarnpkg.com/core-js-compat/-/core-js-compat-3.15.0.tgz#e14a371123db9d1c5b41206d3f420643d238b8fa" + integrity sha512-8X6lWsG+s7IfOKzV93a7fRYfWRZobOfjw5V5rrq43Vh/W+V6qYxl7Akalsvgab4PFT/4L/pjQbdBUEM36NXKrw== dependencies: browserslist "^4.16.6" semver "7.0.0" @@ -7167,7 +7150,7 @@ http-parser-js@>=0.5.1: resolved "https://registry.yarnpkg.com/http-parser-js/-/http-parser-js-0.5.3.tgz#01d2709c79d41698bb01d4decc5e9da4e4a033d9" integrity sha512-t7hjvef/5HEK7RWTdUzVUhl8zkEu+LlaE0IYzdMuvbSDipxBRpOn4Uhw8ZyECEa808iVT8XCjzo6xmYt4CiLZg== -http-proxy-agent@*, http-proxy-agent@^4.0.1: +http-proxy-agent@^4.0.1: version "4.0.1" resolved "https://registry.yarnpkg.com/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz#8a8c8ef7f5932ccf953c296ca8291b95aa74aa3a" integrity sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg== @@ -7910,12 +7893,12 @@ jest-changed-files@^27.0.2: execa "^5.0.0" throat "^6.0.1" -jest-circus@^27.0.4: - version "27.0.4" - resolved "https://registry.yarnpkg.com/jest-circus/-/jest-circus-27.0.4.tgz#3b261514ee3b3da33def736a6352c98ff56bb6e6" - integrity sha512-QD+eblDiRphta630WRKewuASLs/oY1Zki2G4bccntRvrTHQ63ljwFR5TLduuK4Zg0ZPzW0+8o6AP7KRd1yKOjw== +jest-circus@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-circus/-/jest-circus-27.0.5.tgz#b5e327f1d6857c8485126f8e364aefa4378debaa" + integrity sha512-p5rO90o1RTh8LPOG6l0Fc9qgp5YGv+8M5CFixhMh7gGHtGSobD1AxX9cjFZujILgY8t30QZ7WVvxlnuG31r8TA== dependencies: - "@jest/environment" "^27.0.3" + "@jest/environment" "^27.0.5" "@jest/test-result" "^27.0.2" "@jest/types" "^27.0.2" "@types/node" "*" @@ -7927,54 +7910,54 @@ jest-circus@^27.0.4: jest-each "^27.0.2" jest-matcher-utils "^27.0.2" jest-message-util "^27.0.2" - jest-runtime "^27.0.4" - jest-snapshot "^27.0.4" + jest-runtime "^27.0.5" + jest-snapshot "^27.0.5" jest-util "^27.0.2" pretty-format "^27.0.2" slash "^3.0.0" stack-utils "^2.0.3" throat "^6.0.1" -jest-cli@^27.0.4: - version "27.0.4" - resolved "https://registry.yarnpkg.com/jest-cli/-/jest-cli-27.0.4.tgz#491b12c754c0d7c6873b13a66f26b3a80a852910" - integrity sha512-E0T+/i2lxsWAzV7LKYd0SB7HUAvePqaeIh5vX43/G5jXLhv1VzjYzJAGEkTfvxV774ll9cyE2ljcL73PVMEOXQ== +jest-cli@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-cli/-/jest-cli-27.0.5.tgz#f359ba042624cffb96b713010a94bffb7498a37c" + integrity sha512-kZqY020QFOFQKVE2knFHirTBElw3/Q0kUbDc3nMfy/x+RQ7zUY89SUuzpHHJoSX1kX7Lq569ncvjNqU3Td/FCA== dependencies: - "@jest/core" "^27.0.4" + "@jest/core" "^27.0.5" "@jest/test-result" "^27.0.2" "@jest/types" "^27.0.2" chalk "^4.0.0" exit "^0.1.2" graceful-fs "^4.2.4" import-local "^3.0.2" - jest-config "^27.0.4" + jest-config "^27.0.5" jest-util "^27.0.2" jest-validate "^27.0.2" prompts "^2.0.1" yargs "^16.0.3" -jest-config@^27.0.4: - version "27.0.4" - resolved "https://registry.yarnpkg.com/jest-config/-/jest-config-27.0.4.tgz#c4f41378acf40ca77860fb4e213b12109d87b8cf" - integrity sha512-VkQFAHWnPQefdvHU9A+G3H/Z3NrrTKqWpvxgQz3nkUdkDTWeKJE6e//BL+R7z79dXOMVksYgM/z6ndtN0hfChg== +jest-config@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-config/-/jest-config-27.0.5.tgz#683da3b0d8237675c29c817f6e3aba1481028e19" + integrity sha512-zCUIXag7QIXKEVN4kUKbDBDi9Q53dV5o3eNhGqe+5zAbt1vLs4VE3ceWaYrOub0L4Y7E9pGfM84TX/0ARcE+Qw== dependencies: "@babel/core" "^7.1.0" - "@jest/test-sequencer" "^27.0.4" + "@jest/test-sequencer" "^27.0.5" "@jest/types" "^27.0.2" - babel-jest "^27.0.2" + babel-jest "^27.0.5" chalk "^4.0.0" deepmerge "^4.2.2" glob "^7.1.1" graceful-fs "^4.2.4" is-ci "^3.0.0" - jest-circus "^27.0.4" - jest-environment-jsdom "^27.0.3" - jest-environment-node "^27.0.3" + jest-circus "^27.0.5" + jest-environment-jsdom "^27.0.5" + jest-environment-node "^27.0.5" jest-get-type "^27.0.1" - jest-jasmine2 "^27.0.4" + jest-jasmine2 "^27.0.5" jest-regex-util "^27.0.1" - jest-resolve "^27.0.4" - jest-runner "^27.0.4" + jest-resolve "^27.0.5" + jest-runner "^27.0.5" jest-util "^27.0.2" jest-validate "^27.0.2" micromatch "^4.0.4" @@ -8018,26 +8001,26 @@ jest-each@^27.0.2: jest-util "^27.0.2" pretty-format "^27.0.2" -jest-environment-jsdom@^27.0.3: - version "27.0.3" - resolved "https://registry.yarnpkg.com/jest-environment-jsdom/-/jest-environment-jsdom-27.0.3.tgz#ed73e913ddc03864eb9f934b5cbabf1b63504e2e" - integrity sha512-5KLmgv1bhiimpSA8oGTnZYk6g4fsNyZiA/6gI2tAZUgrufd7heRUSVh4gRokzZVEj8zlwAQYT0Zs6tuJSW/ECA== +jest-environment-jsdom@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-environment-jsdom/-/jest-environment-jsdom-27.0.5.tgz#c36771977cf4490a9216a70473b39161d193c212" + integrity sha512-ToWhViIoTl5738oRaajTMgYhdQL73UWPoV4GqHGk2DPhs+olv8OLq5KoQW8Yf+HtRao52XLqPWvl46dPI88PdA== dependencies: - "@jest/environment" "^27.0.3" - "@jest/fake-timers" "^27.0.3" + "@jest/environment" "^27.0.5" + "@jest/fake-timers" "^27.0.5" "@jest/types" "^27.0.2" "@types/node" "*" jest-mock "^27.0.3" jest-util "^27.0.2" jsdom "^16.6.0" -jest-environment-node@^27.0.3: - version "27.0.3" - resolved "https://registry.yarnpkg.com/jest-environment-node/-/jest-environment-node-27.0.3.tgz#b4acb3679d2552a4215732cab8b0ca7ec4398ee0" - integrity sha512-co2/IVnIFL3cItpFULCvXFg9us4gvWXgs7mutAMPCbFhcqh56QAOdKhNzC2+RycsC/k4mbMj1VF+9F/NzA0ROg== +jest-environment-node@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-environment-node/-/jest-environment-node-27.0.5.tgz#b7238fc2b61ef2fb9563a3b7653a95fa009a6a54" + integrity sha512-47qqScV/WMVz5OKF5TWpAeQ1neZKqM3ySwNveEnLyd+yaE/KT6lSMx/0SOx60+ZUcVxPiESYS+Kt2JS9y4PpkQ== dependencies: - "@jest/environment" "^27.0.3" - "@jest/fake-timers" "^27.0.3" + "@jest/environment" "^27.0.5" + "@jest/fake-timers" "^27.0.5" "@jest/types" "^27.0.2" "@types/node" "*" jest-mock "^27.0.3" @@ -8053,10 +8036,10 @@ jest-get-type@^27.0.1: resolved "https://registry.yarnpkg.com/jest-get-type/-/jest-get-type-27.0.1.tgz#34951e2b08c8801eb28559d7eb732b04bbcf7815" integrity sha512-9Tggo9zZbu0sHKebiAijyt1NM77Z0uO4tuWOxUCujAiSeXv30Vb5D4xVF4UR4YWNapcftj+PbByU54lKD7/xMg== -jest-haste-map@^27.0.2: - version "27.0.2" - resolved "https://registry.yarnpkg.com/jest-haste-map/-/jest-haste-map-27.0.2.tgz#3f1819400c671237e48b4d4b76a80a0dbed7577f" - integrity sha512-37gYfrYjjhEfk37C4bCMWAC0oPBxDpG0qpl8lYg8BT//wf353YT/fzgA7+Dq0EtM7rPFS3JEcMsxdtDwNMi2cA== +jest-haste-map@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-haste-map/-/jest-haste-map-27.0.5.tgz#2e1e55073b5328410a2c0d74b334e513d71f3470" + integrity sha512-3LFryGSHxwPFHzKIs6W0BGA2xr6g1MvzSjR3h3D8K8Uqy4vbRm/grpGHzbPtIbOPLC6wFoViRrNEmd116QWSkw== dependencies: "@jest/types" "^27.0.2" "@types/graceful-fs" "^4.1.2" @@ -8073,13 +8056,13 @@ jest-haste-map@^27.0.2: optionalDependencies: fsevents "^2.3.2" -jest-jasmine2@^27.0.4: - version "27.0.4" - resolved "https://registry.yarnpkg.com/jest-jasmine2/-/jest-jasmine2-27.0.4.tgz#c669519ccf4904a485338555e1e66cad36bb0670" - integrity sha512-yj3WrjjquZwkJw+eA4c9yucHw4/+EHndHWSqgHbHGQfT94ihaaQsa009j1a0puU8CNxPDk0c1oAPeOpdJUElwA== +jest-jasmine2@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-jasmine2/-/jest-jasmine2-27.0.5.tgz#8a6eb2a685cdec3af13881145c77553e4e197776" + integrity sha512-m3TojR19sFmTn79QoaGy1nOHBcLvtLso6Zh7u+gYxZWGcza4rRPVqwk1hciA5ZOWWZIJOukAcore8JRX992FaA== dependencies: "@babel/traverse" "^7.1.0" - "@jest/environment" "^27.0.3" + "@jest/environment" "^27.0.5" "@jest/source-map" "^27.0.1" "@jest/test-result" "^27.0.2" "@jest/types" "^27.0.2" @@ -8091,8 +8074,8 @@ jest-jasmine2@^27.0.4: jest-each "^27.0.2" jest-matcher-utils "^27.0.2" jest-message-util "^27.0.2" - jest-runtime "^27.0.4" - jest-snapshot "^27.0.4" + jest-runtime "^27.0.5" + jest-snapshot "^27.0.5" jest-util "^27.0.2" pretty-format "^27.0.2" throat "^6.0.1" @@ -8148,19 +8131,19 @@ jest-regex-util@^27.0.1: resolved "https://registry.yarnpkg.com/jest-regex-util/-/jest-regex-util-27.0.1.tgz#69d4b1bf5b690faa3490113c47486ed85dd45b68" integrity sha512-6nY6QVcpTgEKQy1L41P4pr3aOddneK17kn3HJw6SdwGiKfgCGTvH02hVXL0GU8GEKtPH83eD2DIDgxHXOxVohQ== -jest-resolve-dependencies@^27.0.4: - version "27.0.4" - resolved "https://registry.yarnpkg.com/jest-resolve-dependencies/-/jest-resolve-dependencies-27.0.4.tgz#a07a242d70d668afd3fcf7f4270755eebb1fe579" - integrity sha512-F33UPfw1YGWCV2uxJl7wD6TvcQn5IC0LtguwY3r4L7R6H4twpLkp5Q2ZfzRx9A2I3G8feiy0O0sqcn/Qoym71A== +jest-resolve-dependencies@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-resolve-dependencies/-/jest-resolve-dependencies-27.0.5.tgz#819ccdddd909c65acddb063aac3a49e4ba1ed569" + integrity sha512-xUj2dPoEEd59P+nuih4XwNa4nJ/zRd/g4rMvjHrZPEBWeWRq/aJnnM6mug+B+Nx+ILXGtfWHzQvh7TqNV/WbuA== dependencies: "@jest/types" "^27.0.2" jest-regex-util "^27.0.1" - jest-snapshot "^27.0.4" + jest-snapshot "^27.0.5" -jest-resolve@^27.0.4: - version "27.0.4" - resolved "https://registry.yarnpkg.com/jest-resolve/-/jest-resolve-27.0.4.tgz#8a27bc3f2f00c8ea28f3bc99bbf6f468300a703d" - integrity sha512-BcfyK2i3cG79PDb/6gB6zFeFQlcqLsQjGBqznFCpA0L/3l1L/oOsltdUjs5eISAWA9HS9qtj8v2PSZr/yWxONQ== +jest-resolve@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-resolve/-/jest-resolve-27.0.5.tgz#937535a5b481ad58e7121eaea46d1424a1e0c507" + integrity sha512-Md65pngRh8cRuWVdWznXBB5eDt391OJpdBaJMxfjfuXCvOhM3qQBtLMCMTykhuUKiBMmy5BhqCW7AVOKmPrW+Q== dependencies: "@jest/types" "^27.0.2" chalk "^4.0.0" @@ -8172,15 +8155,15 @@ jest-resolve@^27.0.4: resolve "^1.20.0" slash "^3.0.0" -jest-runner@^27.0.4: - version "27.0.4" - resolved "https://registry.yarnpkg.com/jest-runner/-/jest-runner-27.0.4.tgz#2787170a9509b792ae129794f6944d27d5d12a4f" - integrity sha512-NfmvSYLCsCJk2AG8Ar2NAh4PhsJJpO+/r+g4bKR5L/5jFzx/indUpnVBdrfDvuqhGLLAvrKJ9FM/Nt8o1dsqxg== +jest-runner@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-runner/-/jest-runner-27.0.5.tgz#b6fdc587e1a5056339205914294555c554efc08a" + integrity sha512-HNhOtrhfKPArcECgBTcWOc+8OSL8GoFoa7RsHGnfZR1C1dFohxy9eLtpYBS+koybAHlJLZzNCx2Y/Ic3iEtJpQ== dependencies: "@jest/console" "^27.0.2" - "@jest/environment" "^27.0.3" + "@jest/environment" "^27.0.5" "@jest/test-result" "^27.0.2" - "@jest/transform" "^27.0.2" + "@jest/transform" "^27.0.5" "@jest/types" "^27.0.2" "@types/node" "*" chalk "^4.0.0" @@ -8188,30 +8171,30 @@ jest-runner@^27.0.4: exit "^0.1.2" graceful-fs "^4.2.4" jest-docblock "^27.0.1" - jest-environment-jsdom "^27.0.3" - jest-environment-node "^27.0.3" - jest-haste-map "^27.0.2" + jest-environment-jsdom "^27.0.5" + jest-environment-node "^27.0.5" + jest-haste-map "^27.0.5" jest-leak-detector "^27.0.2" jest-message-util "^27.0.2" - jest-resolve "^27.0.4" - jest-runtime "^27.0.4" + jest-resolve "^27.0.5" + jest-runtime "^27.0.5" jest-util "^27.0.2" jest-worker "^27.0.2" source-map-support "^0.5.6" throat "^6.0.1" -jest-runtime@^27.0.4: - version "27.0.4" - resolved "https://registry.yarnpkg.com/jest-runtime/-/jest-runtime-27.0.4.tgz#2e4a6aa77cac32ac612dfe12768387a8aa15c2f0" - integrity sha512-voJB4xbAjS/qYPboV+e+gmg3jfvHJJY4CagFWBOM9dQKtlaiTjcpD2tWwla84Z7PtXSQPeIpXY0qksA9Dum29A== +jest-runtime@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-runtime/-/jest-runtime-27.0.5.tgz#cd5d1aa9754d30ddf9f13038b3cb7b95b46f552d" + integrity sha512-V/w/+VasowPESbmhXn5AsBGPfb35T7jZPGZybYTHxZdP7Gwaa+A0EXE6rx30DshHKA98lVCODbCO8KZpEW3hiQ== dependencies: "@jest/console" "^27.0.2" - "@jest/environment" "^27.0.3" - "@jest/fake-timers" "^27.0.3" - "@jest/globals" "^27.0.3" + "@jest/environment" "^27.0.5" + "@jest/fake-timers" "^27.0.5" + "@jest/globals" "^27.0.5" "@jest/source-map" "^27.0.1" "@jest/test-result" "^27.0.2" - "@jest/transform" "^27.0.2" + "@jest/transform" "^27.0.5" "@jest/types" "^27.0.2" "@types/yargs" "^16.0.0" chalk "^4.0.0" @@ -8220,12 +8203,12 @@ jest-runtime@^27.0.4: exit "^0.1.2" glob "^7.1.3" graceful-fs "^4.2.4" - jest-haste-map "^27.0.2" + jest-haste-map "^27.0.5" jest-message-util "^27.0.2" jest-mock "^27.0.3" jest-regex-util "^27.0.1" - jest-resolve "^27.0.4" - jest-snapshot "^27.0.4" + jest-resolve "^27.0.5" + jest-snapshot "^27.0.5" jest-util "^27.0.2" jest-validate "^27.0.2" slash "^3.0.0" @@ -8240,10 +8223,10 @@ jest-serializer@^27.0.1: "@types/node" "*" graceful-fs "^4.2.4" -jest-snapshot@^27.0.4: - version "27.0.4" - resolved "https://registry.yarnpkg.com/jest-snapshot/-/jest-snapshot-27.0.4.tgz#2b96e22ca90382b3e93bd0aae2ce4c78bf51fb5b" - integrity sha512-hnjrvpKGdSMvKfbHyaG5Kul7pDJGZvjVy0CKpzhu28MmAssDXS6GpynhXzgst1wBQoKD8c9b2VS2a5yhDLQRCA== +jest-snapshot@^27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest-snapshot/-/jest-snapshot-27.0.5.tgz#6e3b9e8e193685372baff771ba34af631fe4d4d5" + integrity sha512-H1yFYdgnL1vXvDqMrnDStH6yHFdMEuzYQYc71SnC/IJnuuhW6J16w8GWG1P+qGd3Ag3sQHjbRr0TcwEo/vGS+g== dependencies: "@babel/core" "^7.7.2" "@babel/generator" "^7.7.2" @@ -8251,7 +8234,7 @@ jest-snapshot@^27.0.4: "@babel/plugin-syntax-typescript" "^7.7.2" "@babel/traverse" "^7.7.2" "@babel/types" "^7.0.0" - "@jest/transform" "^27.0.2" + "@jest/transform" "^27.0.5" "@jest/types" "^27.0.2" "@types/babel__traverse" "^7.0.4" "@types/prettier" "^2.1.5" @@ -8261,10 +8244,10 @@ jest-snapshot@^27.0.4: graceful-fs "^4.2.4" jest-diff "^27.0.2" jest-get-type "^27.0.1" - jest-haste-map "^27.0.2" + jest-haste-map "^27.0.5" jest-matcher-utils "^27.0.2" jest-message-util "^27.0.2" - jest-resolve "^27.0.4" + jest-resolve "^27.0.5" jest-util "^27.0.2" natural-compare "^1.4.0" pretty-format "^27.0.2" @@ -8325,14 +8308,14 @@ jest-worker@^27.0.2: merge-stream "^2.0.0" supports-color "^8.0.0" -jest@27.0.4: - version "27.0.4" - resolved "https://registry.yarnpkg.com/jest/-/jest-27.0.4.tgz#91d4d564b36bcf93b98dac1ab19f07089e670f53" - integrity sha512-Px1iKFooXgGSkk1H8dJxxBIrM3tsc5SIuI4kfKYK2J+4rvCvPGr/cXktxh0e9zIPQ5g09kOMNfHQEmusBUf/ZA== +jest@27.0.5: + version "27.0.5" + resolved "https://registry.yarnpkg.com/jest/-/jest-27.0.5.tgz#141825e105514a834cc8d6e44670509e8d74c5f2" + integrity sha512-4NlVMS29gE+JOZvgmSAsz3eOjkSsHqjTajlIsah/4MVSmKvf3zFP/TvgcLoWe2UVHiE9KF741sReqhF0p4mqbQ== dependencies: - "@jest/core" "^27.0.4" + "@jest/core" "^27.0.5" import-local "^3.0.2" - jest-cli "^27.0.4" + jest-cli "^27.0.5" joi@^17.3.0, joi@^17.4.0: version "17.4.0" @@ -12520,12 +12503,12 @@ text-table@0.2.0, text-table@^0.2.0: resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4" integrity sha1-f17oI66AUgfACvLfSoTsP8+lcLQ= -the-guild-components@1.3.1: - version "1.3.1" - resolved "https://registry.yarnpkg.com/the-guild-components/-/the-guild-components-1.3.1.tgz#f660711ee9f035bb0b128bd12b36596d9c97d553" - integrity sha512-Bx2Y5+wbqPPuu6Kcl0f0jW1CEKEK8DziL71DN0o0KH5StYcEJIHmQcMStw+d32BmPPz5WVSdWCRH5TqAcswpDg== +the-guild-components@1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/the-guild-components/-/the-guild-components-1.4.0.tgz#3b9f8c0655c933ed61cf688f073184b9c91461ca" + integrity sha512-cnZk/vgTrV6IP948kkfcH+jOhX0CIrLf7htSdv5VZRomJNXc0uHw/sLuYlJXqObqItmYAQ9Cl0WskmyModfB8g== dependencies: - "@theguild/components" "1.3.1" + "@theguild/components" "1.4.0" throat@^6.0.1: version "6.0.1" @@ -12727,11 +12710,6 @@ tslib@^2.0.3, tslib@^2.1.0, tslib@^2.2.0, tslib@~2.3.0: resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.3.0.tgz#803b8cdab3e12ba581a4ca41c8839bbb0dacb09e" integrity sha512-N82ooyxVNm6h1riLCoyS9e3fuJ3AMG2zIZs2Gd1ATcSFjSA23Q0fzjjZeh0jbJvWVDZ0cJT8yaNNaaXHzueNjg== -tslib@~2.0.1: - version "2.0.3" - resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.0.3.tgz#8e0741ac45fc0c226e58a17bfc3e64b9bc6ca61c" - integrity sha512-uZtkfKblCEQtZKBF6EBXVZeQNl82yqtDQdv+eck8u7tdPxjLu2/lp5/uPW+um2tpuxINHWy3GhiccY7QgEaVHQ== - tslib@~2.2.0: version "2.2.0" resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.2.0.tgz#fb2c475977e35e241311ede2693cee1ec6698f5c" @@ -13147,10 +13125,10 @@ use-composed-ref@^1.0.0: dependencies: ts-essentials "^2.0.3" -use-debounce@^6.0.1: - version "6.0.1" - resolved "https://registry.yarnpkg.com/use-debounce/-/use-debounce-6.0.1.tgz#ed1eb2b30189408fb9792ea2887f4c6c3cb401a3" - integrity sha512-kpvIxpa0vOLz/2I2sfNJ72mUeaT2CMNCu5BT1f2HkV9qZK27UVSOFf1sSSu+wjJE4TcR2VTXS2SM569+m3TN7Q== +use-debounce@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/use-debounce/-/use-debounce-7.0.0.tgz#00a67d23d4fe09905e11145a99278da06c01c880" + integrity sha512-4fvxEEs7ztdNMh+c497HAgysdq2+Ascem6EaDANGlCIap1JzqfL03Xw8xkYc2lShfXm4uO6PA6V5zcXN7gJdFA== use-isomorphic-layout-effect@^1.0.0: version "1.1.1" @@ -13209,10 +13187,10 @@ v8-compile-cache@^2.0.3: resolved "https://registry.yarnpkg.com/v8-compile-cache/-/v8-compile-cache-2.3.0.tgz#2de19618c66dc247dcfb6f99338035d8245a2cee" integrity sha512-l8lCEmLcLYZh4nbunNZvQCJc5pv7+RCwa8q/LdUx8u7lsWvPDKmpodJAJNwkAhJC//dFY48KuIEmjtd4RViDrA== -v8-to-istanbul@^7.0.0: - version "7.1.0" - resolved "https://registry.yarnpkg.com/v8-to-istanbul/-/v8-to-istanbul-7.1.0.tgz#5b95cef45c0f83217ec79f8fc7ee1c8b486aee07" - integrity sha512-uXUVqNUCLa0AH1vuVxzi+MI4RfxEOKt9pBgKwHbgH7st8Kv2P1m+jvWNnektzBh5QShF3ODgKmUFCf38LnVz1g== +v8-to-istanbul@^8.0.0: + version "8.0.0" + resolved "https://registry.yarnpkg.com/v8-to-istanbul/-/v8-to-istanbul-8.0.0.tgz#4229f2a99e367f3f018fa1d5c2b8ec684667c69c" + integrity sha512-LkmXi8UUNxnCC+JlH7/fsfsKr5AU110l+SYGJimWNkWhxbN5EyeOtm1MJ0hhvqMMOhGwBj1Fp70Yv9i+hX0QAg== dependencies: "@types/istanbul-lib-coverage" "^2.0.1" convert-source-map "^1.6.0" @@ -13231,6 +13209,11 @@ validate-npm-package-license@^3.0.1: spdx-correct "^3.0.0" spdx-expression-parse "^3.0.0" +validator@13.6.0: + version "13.6.0" + resolved "https://registry.yarnpkg.com/validator/-/validator-13.6.0.tgz#1e71899c14cdc7b2068463cb24c1cc16f6ec7059" + integrity sha512-gVgKbdbHgtxpRyR8K0O6oFZPhhB5tT1jeEHZR0Znr9Svg03U0+r9DXWMrnRAB+HtCStDQKlaIZm42tVsVjqtjg== + value-equal@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/value-equal/-/value-equal-1.0.1.tgz#1e0b794c734c5c0cade179c437d356d931a34d6c"