@smartive/graphql-magic 1.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (124) hide show
  1. package/.eslintrc +21 -0
  2. package/.github/workflows/release.yml +24 -0
  3. package/.github/workflows/testing.yml +37 -0
  4. package/.nvmrc +1 -0
  5. package/.prettierignore +34 -0
  6. package/.prettierrc.json +1 -0
  7. package/.releaserc +27 -0
  8. package/CHANGELOG.md +6 -0
  9. package/README.md +15 -0
  10. package/dist/cjs/index.cjs +2646 -0
  11. package/dist/esm/client/gql.d.ts +1 -0
  12. package/dist/esm/client/gql.js +5 -0
  13. package/dist/esm/client/gql.js.map +1 -0
  14. package/dist/esm/client/index.d.ts +2 -0
  15. package/dist/esm/client/index.js +4 -0
  16. package/dist/esm/client/index.js.map +1 -0
  17. package/dist/esm/client/queries.d.ts +24 -0
  18. package/dist/esm/client/queries.js +152 -0
  19. package/dist/esm/client/queries.js.map +1 -0
  20. package/dist/esm/context.d.ts +30 -0
  21. package/dist/esm/context.js +2 -0
  22. package/dist/esm/context.js.map +1 -0
  23. package/dist/esm/errors.d.ts +17 -0
  24. package/dist/esm/errors.js +27 -0
  25. package/dist/esm/errors.js.map +1 -0
  26. package/dist/esm/generate/generate.d.ts +7 -0
  27. package/dist/esm/generate/generate.js +211 -0
  28. package/dist/esm/generate/generate.js.map +1 -0
  29. package/dist/esm/generate/index.d.ts +3 -0
  30. package/dist/esm/generate/index.js +5 -0
  31. package/dist/esm/generate/index.js.map +1 -0
  32. package/dist/esm/generate/mutations.d.ts +2 -0
  33. package/dist/esm/generate/mutations.js +18 -0
  34. package/dist/esm/generate/mutations.js.map +1 -0
  35. package/dist/esm/generate/utils.d.ts +22 -0
  36. package/dist/esm/generate/utils.js +150 -0
  37. package/dist/esm/generate/utils.js.map +1 -0
  38. package/dist/esm/index.d.ts +10 -0
  39. package/dist/esm/index.js +12 -0
  40. package/dist/esm/index.js.map +1 -0
  41. package/dist/esm/migrations/generate.d.ts +28 -0
  42. package/dist/esm/migrations/generate.js +516 -0
  43. package/dist/esm/migrations/generate.js.map +1 -0
  44. package/dist/esm/migrations/index.d.ts +1 -0
  45. package/dist/esm/migrations/index.js +3 -0
  46. package/dist/esm/migrations/index.js.map +1 -0
  47. package/dist/esm/models.d.ts +170 -0
  48. package/dist/esm/models.js +27 -0
  49. package/dist/esm/models.js.map +1 -0
  50. package/dist/esm/permissions/check.d.ts +15 -0
  51. package/dist/esm/permissions/check.js +162 -0
  52. package/dist/esm/permissions/check.js.map +1 -0
  53. package/dist/esm/permissions/generate.d.ts +45 -0
  54. package/dist/esm/permissions/generate.js +77 -0
  55. package/dist/esm/permissions/generate.js.map +1 -0
  56. package/dist/esm/permissions/index.d.ts +2 -0
  57. package/dist/esm/permissions/index.js +4 -0
  58. package/dist/esm/permissions/index.js.map +1 -0
  59. package/dist/esm/resolvers/arguments.d.ts +26 -0
  60. package/dist/esm/resolvers/arguments.js +88 -0
  61. package/dist/esm/resolvers/arguments.js.map +1 -0
  62. package/dist/esm/resolvers/filters.d.ts +5 -0
  63. package/dist/esm/resolvers/filters.js +126 -0
  64. package/dist/esm/resolvers/filters.js.map +1 -0
  65. package/dist/esm/resolvers/index.d.ts +7 -0
  66. package/dist/esm/resolvers/index.js +9 -0
  67. package/dist/esm/resolvers/index.js.map +1 -0
  68. package/dist/esm/resolvers/mutations.d.ts +3 -0
  69. package/dist/esm/resolvers/mutations.js +255 -0
  70. package/dist/esm/resolvers/mutations.js.map +1 -0
  71. package/dist/esm/resolvers/node.d.ts +44 -0
  72. package/dist/esm/resolvers/node.js +102 -0
  73. package/dist/esm/resolvers/node.js.map +1 -0
  74. package/dist/esm/resolvers/resolver.d.ts +5 -0
  75. package/dist/esm/resolvers/resolver.js +143 -0
  76. package/dist/esm/resolvers/resolver.js.map +1 -0
  77. package/dist/esm/resolvers/resolvers.d.ts +9 -0
  78. package/dist/esm/resolvers/resolvers.js +39 -0
  79. package/dist/esm/resolvers/resolvers.js.map +1 -0
  80. package/dist/esm/resolvers/utils.d.ts +43 -0
  81. package/dist/esm/resolvers/utils.js +125 -0
  82. package/dist/esm/resolvers/utils.js.map +1 -0
  83. package/dist/esm/utils.d.ts +25 -0
  84. package/dist/esm/utils.js +159 -0
  85. package/dist/esm/utils.js.map +1 -0
  86. package/dist/esm/values.d.ts +15 -0
  87. package/dist/esm/values.js +7 -0
  88. package/dist/esm/values.js.map +1 -0
  89. package/jest.config.ts +12 -0
  90. package/package.json +66 -0
  91. package/renovate.json +32 -0
  92. package/src/client/gql.ts +7 -0
  93. package/src/client/index.ts +4 -0
  94. package/src/client/queries.ts +251 -0
  95. package/src/context.ts +27 -0
  96. package/src/errors.ts +32 -0
  97. package/src/generate/generate.ts +273 -0
  98. package/src/generate/index.ts +5 -0
  99. package/src/generate/mutations.ts +35 -0
  100. package/src/generate/utils.ts +223 -0
  101. package/src/index.ts +12 -0
  102. package/src/migrations/generate.ts +633 -0
  103. package/src/migrations/index.ts +3 -0
  104. package/src/models.ts +228 -0
  105. package/src/permissions/check.ts +239 -0
  106. package/src/permissions/generate.ts +143 -0
  107. package/src/permissions/index.ts +4 -0
  108. package/src/resolvers/arguments.ts +129 -0
  109. package/src/resolvers/filters.ts +163 -0
  110. package/src/resolvers/index.ts +9 -0
  111. package/src/resolvers/mutations.ts +313 -0
  112. package/src/resolvers/node.ts +193 -0
  113. package/src/resolvers/resolver.ts +223 -0
  114. package/src/resolvers/resolvers.ts +40 -0
  115. package/src/resolvers/utils.ts +188 -0
  116. package/src/utils.ts +186 -0
  117. package/src/values.ts +19 -0
  118. package/tests/unit/__snapshots__/generate.spec.ts.snap +105 -0
  119. package/tests/unit/__snapshots__/resolve.spec.ts.snap +60 -0
  120. package/tests/unit/generate.spec.ts +8 -0
  121. package/tests/unit/resolve.spec.ts +128 -0
  122. package/tests/unit/utils.ts +82 -0
  123. package/tsconfig.jest.json +13 -0
  124. package/tsconfig.json +13 -0
package/package.json ADDED
@@ -0,0 +1,66 @@
1
+ {
2
+ "name": "@smartive/graphql-magic",
3
+ "version": "1.0.1",
4
+ "description": "",
5
+ "source": "src/index.ts",
6
+ "type": "module",
7
+ "module": "dist/esm/index.js",
8
+ "main": "dist/cjs/index.cjs",
9
+ "types": "dist/esm/index.d.ts",
10
+ "engines": {
11
+ "node": ">=18"
12
+ },
13
+ "sideEffecs": false,
14
+ "scripts": {
15
+ "bootstrap": "npm ci && npm run generate",
16
+ "generate": "npm run generate:index-files",
17
+ "generate:index-files": "cti create ./src --excludes types --withoutbackup",
18
+ "lint": "eslint src",
19
+ "lint:fix": "eslint src --fix",
20
+ "test": "npm run lint && npm run test:unit && npm run build",
21
+ "test:unit": "jest tests/unit --no-cache --no-watchman",
22
+ "clean": "del-cli dist/**",
23
+ "prebuild": "npm run clean",
24
+ "build": "npm run build:esm && npm run build:cjs",
25
+ "build:esm": "tsc",
26
+ "build:cjs": "esbuild src/index.ts --bundle --platform=node --outdir=dist/cjs --out-extension:.js=.cjs --format=cjs --packages=external",
27
+ "publish": "semantic-release"
28
+ },
29
+ "overrides": {
30
+ "graphql": "$graphql",
31
+ "rollup": "3.26.2"
32
+ },
33
+ "browserslist": "> 0.25%, not dead",
34
+ "publishConfig": {
35
+ "access": "public"
36
+ },
37
+ "dependencies": {
38
+ "@apollo/server": "^4.0.0",
39
+ "code-block-writer": "^12.0.0",
40
+ "graphql": "^15.8.0",
41
+ "inflection": "^2.0.1",
42
+ "knex": "^2.4.2",
43
+ "knex-schema-inspector": "^3.0.1",
44
+ "lodash": "^4.17.21",
45
+ "luxon": "^3.3.0",
46
+ "uuid": "^9.0.0"
47
+ },
48
+ "devDependencies": {
49
+ "@smartive/eslint-config": "3.2.0",
50
+ "@smartive/prettier-config": "3.1.2",
51
+ "@types/jest": "29.5.3",
52
+ "@types/lodash": "4.14.195",
53
+ "@types/luxon": "3.3.0",
54
+ "@types/uuid": "9.0.2",
55
+ "create-ts-index": "1.14.0",
56
+ "del-cli": "5.0.0",
57
+ "esbuild": "0.18.11",
58
+ "eslint": "8.44.0",
59
+ "jest": "29.6.1",
60
+ "mock-knex": "0.4.12",
61
+ "prettier": "2.8.8",
62
+ "ts-jest": "29.1.1",
63
+ "ts-node": "10.9.1",
64
+ "typescript": "5.1.6"
65
+ }
66
+ }
package/renovate.json ADDED
@@ -0,0 +1,32 @@
1
+ {
2
+ "$schema": "https://docs.renovatebot.com/renovate-schema.json",
3
+ "extends": ["gitlab>smartive/internal-it/renovate"],
4
+ "packageRules": [
5
+ {
6
+ "packagePatterns": ["^@types[/]"],
7
+ "groupName": "TypeScript typings"
8
+ },
9
+ {
10
+ "packagePatterns": ["^eslint*", "prettier"],
11
+ "groupName": "Linting"
12
+ },
13
+ {
14
+ "updateTypes": ["major"],
15
+ "automerge": false,
16
+ "gitLabAutomerge": false,
17
+ "labels": ["dependencies", "dependencies-major"]
18
+ },
19
+ {
20
+ "packagePatterns": ["*"],
21
+ "rangeStrategy": "replace"
22
+ },
23
+ {
24
+ "depTypeList": ["devDependencies"],
25
+ "rangeStrategy": "pin"
26
+ },
27
+ {
28
+ "depTypeList": ["peerDependencies"],
29
+ "rangeStrategy": "widen"
30
+ }
31
+ ]
32
+ }
@@ -0,0 +1,7 @@
1
+ // This tag does nothing (just generates a string) - it is here for the tooling (syntax highlighting, formatting and type generation)
2
+ export const gql = (chunks: TemplateStringsArray, ...variables: (string | number | boolean)[]): string => {
3
+ return chunks.reduce(
4
+ (accumulator, chunk, index) => `${accumulator}${chunk}${index in variables ? variables[index] : ''}`,
5
+ ''
6
+ );
7
+ };
@@ -0,0 +1,4 @@
1
+ // created from 'create-ts-index'
2
+
3
+ export * from './gql';
4
+ export * from './queries';
@@ -0,0 +1,251 @@
1
+ import upperFirst from 'lodash/upperFirst';
2
+ import {
3
+ actionableRelations,
4
+ and,
5
+ isQueriableBy,
6
+ isRelation,
7
+ isSimpleField,
8
+ isToOneRelation,
9
+ isUpdatableBy,
10
+ isVisibleRelation,
11
+ Model,
12
+ Models,
13
+ not,
14
+ Relation,
15
+ ReverseRelation,
16
+ VisibleRelationsByRole,
17
+ } from '../models';
18
+ import { getModelPlural, getModelPluralField, summonByName, typeToField } from '../utils';
19
+
20
+ export const getUpdateEntityQuery = (
21
+ model: Model,
22
+ role: any,
23
+ fields?: string[] | undefined,
24
+ additionalFields = ''
25
+ ) => `query Update${model.name}Fields ($id: ID!) {
26
+ data: ${typeToField(model.name)}(where: { id: $id }) {
27
+ id
28
+ ${model.fields
29
+ .filter(({ name }) => !fields || fields.includes(name))
30
+ .filter(not(isRelation))
31
+ .filter(isUpdatableBy(role))
32
+ .map(({ name }) => name)
33
+ .join(' ')}
34
+ ${actionableRelations(model, 'update')
35
+ .filter(({ name }) => !fields || fields.includes(name))
36
+ .map(({ name }) => `${name} { id }`)}
37
+ ${additionalFields}
38
+ }
39
+ }`;
40
+
41
+ export const getEditEntityRelationsQuery = (
42
+ models: Models,
43
+ model: Model,
44
+ action: 'create' | 'update' | 'filter',
45
+ fields?: string[],
46
+ ignoreFields?: string[],
47
+ additionalFields: Record<string, string> = {}
48
+ ) => {
49
+ const relations = actionableRelations(model, action).filter(
50
+ ({ name }) => (!fields || fields.includes(name)) && (!ignoreFields || !ignoreFields.includes(name))
51
+ );
52
+
53
+ return (
54
+ !!relations.length &&
55
+ `query ${upperFirst(action)}${model.name}Relations {
56
+ ${relations
57
+ .map(({ name, type }) => {
58
+ const model = summonByName(models, type);
59
+
60
+ return `${name}: ${getModelPluralField(model)} {
61
+ id
62
+ display: ${model.displayField || ''}
63
+ ${additionalFields[name] || ''}
64
+ }`;
65
+ })
66
+ .join(' ')}
67
+ }`
68
+ );
69
+ };
70
+
71
+ export const getManyToManyRelations = (model: Model, fields?: string[], ignoreFields?: string[]) => {
72
+ const manyToManyRelations: [ReverseRelation, Relation][] = [];
73
+ for (const field of model.reverseRelations) {
74
+ if ((fields && !fields.includes(field.name)) || (ignoreFields && ignoreFields.includes(field.name))) {
75
+ continue;
76
+ }
77
+
78
+ const relation = field.model.relations.find(
79
+ (relation) => !relation.field.generated && relation.field.name !== field.field.name
80
+ );
81
+ if (!relation) {
82
+ continue;
83
+ }
84
+
85
+ const inapplicableFields = field.model.fields.filter(
86
+ (otherField) => !otherField.generated && ![field.field.name, relation.field.name].includes(otherField.name)
87
+ );
88
+ if (inapplicableFields.length) {
89
+ continue;
90
+ }
91
+
92
+ manyToManyRelations.push([field, relation]);
93
+ }
94
+ return manyToManyRelations;
95
+ };
96
+
97
+ export const getManyToManyRelation = (model: Model, name: string) => getManyToManyRelations(model, [name])[0];
98
+
99
+ export const getManyToManyRelationsQuery = (
100
+ model: Model,
101
+ action: 'create' | 'update',
102
+ manyToManyRelations: [ReverseRelation, Relation][]
103
+ ) =>
104
+ !!manyToManyRelations.length &&
105
+ (action === 'update'
106
+ ? `query Update${model.name}ManyToManyRelations($id: ID!) {
107
+ ${typeToField(model.name)}(where: { id: $id }) {
108
+ ${manyToManyRelations
109
+ .map(([reverseRelation, { field }]) => {
110
+ return `${reverseRelation.name} {
111
+ id
112
+ ${field.name} {
113
+ id
114
+ }
115
+ }`;
116
+ })
117
+ .join(' ')}
118
+ }
119
+ ${manyToManyRelations
120
+ .map(([reverseRelation, { model }]) => {
121
+ return `${reverseRelation.name}: ${getModelPluralField(model)} {
122
+ id
123
+ ${model.displayField || ''}
124
+ }`;
125
+ })
126
+ .join(' ')}
127
+ }`
128
+ : `query Create${model.name}ManyToManyRelations {
129
+ ${manyToManyRelations
130
+ .map(([reverseRelation, { model }]) => {
131
+ return `${reverseRelation.name}: ${getModelPluralField(model)} {
132
+ id
133
+ ${model.displayField || ''}
134
+ }`;
135
+ })
136
+ .join(' ')}
137
+ }`);
138
+
139
+ export type MutationQuery = {
140
+ mutated: {
141
+ id: string;
142
+ };
143
+ };
144
+
145
+ export const getMutationQuery = (model: Model, action: 'create' | 'update' | 'delete') =>
146
+ action === 'create'
147
+ ? `
148
+ mutation Create${model.name} ($data: Create${model.name}!) {
149
+ mutated: create${model.name}(data: $data) {
150
+ id
151
+ }
152
+ }
153
+ `
154
+ : action === 'update'
155
+ ? `
156
+ mutation Update${model.name} ($id: ID!, $data: Update${model.name}!) {
157
+ mutated: update${model.name}(where: { id: $id } data: $data) {
158
+ id
159
+ }
160
+ }
161
+ `
162
+ : `
163
+ mutation Delete${model.name} ($id: ID!) {
164
+ mutated: delete${model.name}(where: { id: $id }) {
165
+ id
166
+ }
167
+ }
168
+ `;
169
+
170
+ export const displayField = (model: Model) => `
171
+ ${model.displayField ? `display: ${model.displayField}` : ''}
172
+ `;
173
+
174
+ export const getEntityListQuery = (
175
+ model: Model,
176
+ role: string,
177
+ additionalFields = '',
178
+ root?: {
179
+ model: Model;
180
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
181
+ entity: any;
182
+ reverseRelationName: string;
183
+ }
184
+ ) => `query ${getModelPlural(model)}List(
185
+ ${root ? '$id: ID!,' : ''}
186
+ $limit: Int!,
187
+ $where: ${model.name}Where!,
188
+ ${model.fields.some(({ searchable }) => searchable) ? '$search: String,' : ''}
189
+ ) {
190
+ ${root ? `root: ${typeToField(root.model.name)}(where: { id: $id }) {` : ''}
191
+ data: ${root ? root.reverseRelationName : getModelPluralField(model)}(limit: $limit, where: $where, ${
192
+ model.fields.some(({ searchable }) => searchable) ? ', search: $search' : ''
193
+ }) {
194
+ ${displayField(model)}
195
+ ${model.fields.filter(and(isSimpleField, isQueriableBy(role))).map(({ name }) => name)}
196
+ ${additionalFields}
197
+ }
198
+ ${root ? '}' : ''}
199
+ }`;
200
+
201
+ export const getEntityQuery = (
202
+ models: Models,
203
+ model: Model,
204
+ role: string,
205
+ visibleRelationsByRole: VisibleRelationsByRole,
206
+ typesWithSubRelations: string[]
207
+ ) => `query Admin${model.name} ($id: ID!) {
208
+ data: ${typeToField(model.name)}(where: { id: $id }) {
209
+ ${displayField(model)}
210
+ ${model.fields.filter(and(isSimpleField, isQueriableBy(role))).map(({ name }) => name)}
211
+ ${queryRelations(
212
+ models,
213
+ model.fields.filter(and(isRelation, isVisibleRelation(visibleRelationsByRole, model.name, role))),
214
+ role,
215
+ typesWithSubRelations
216
+ )}
217
+ ${queryRelations(
218
+ models,
219
+ model.reverseRelations.filter(and(isToOneRelation, isVisibleRelation(visibleRelationsByRole, model.name, role))),
220
+ role,
221
+ typesWithSubRelations
222
+ )}
223
+ }
224
+ }`;
225
+
226
+ export const getFindEntityQuery = (model: Model, role: string) => `query Find${model.name}($where: ${
227
+ model.name
228
+ }Where!, $orderBy: [${model.name}OrderBy!]) {
229
+ data: ${getModelPluralField(model)}(limit: 1, where: $where, orderBy: $orderBy) {
230
+ ${model.fields.filter(and(isSimpleField, isQueriableBy(role))).map(({ name }) => name)}
231
+ }
232
+ }`;
233
+
234
+ export const queryRelations = (
235
+ models: Models,
236
+ relations: { name: string; type: string }[],
237
+ role: string,
238
+ typesWithSubRelations: string[]
239
+ ) =>
240
+ relations
241
+ .map(({ name, type }): string => {
242
+ const relatedModel = summonByName(models, type);
243
+ const subRelations = typesWithSubRelations.includes(type) ? relatedModel.fields.filter(isRelation) : [];
244
+
245
+ return `${name} {
246
+ id
247
+ ${displayField(relatedModel)}
248
+ ${subRelations.length > 0 ? queryRelations(models, subRelations, role, typesWithSubRelations) : ''}
249
+ }`;
250
+ })
251
+ .join('\n');
package/src/context.ts ADDED
@@ -0,0 +1,27 @@
1
+ import { DocumentNode, GraphQLResolveInfo } from 'graphql';
2
+ import { IncomingMessage } from 'http';
3
+ import { Knex } from 'knex';
4
+ import { DateTime } from 'luxon';
5
+ import { Entity, Models, MutationHook, RawModels } from './models';
6
+ import { Permissions } from './permissions/generate';
7
+ import { AliasGenerator } from './resolvers/utils';
8
+
9
+ // Minimal user structure required by graphql-magic
10
+ export type User = { id: string; role: string };
11
+
12
+ export type Context = {
13
+ req: IncomingMessage;
14
+ now: DateTime;
15
+ knex: Knex;
16
+ document: DocumentNode;
17
+ locale: string;
18
+ locales: string[];
19
+ user: User;
20
+ rawModels: RawModels;
21
+ models: Models;
22
+ permissions: Permissions;
23
+ mutationHook?: MutationHook;
24
+ handleUploads?: (data: Entity) => Promise<void>;
25
+ };
26
+
27
+ export type FullContext = Context & { info: GraphQLResolveInfo; aliases: AliasGenerator };
package/src/errors.ts ADDED
@@ -0,0 +1,32 @@
1
+ import { GraphQLError as GQLError } from 'graphql';
2
+ import { PermissionAction } from './permissions/generate';
3
+
4
+ export class GraphQLError extends GQLError {
5
+ constructor(message: string, extensions: ConstructorParameters<typeof GQLError>[6]) {
6
+ super(message, undefined, undefined, undefined, undefined, undefined, extensions);
7
+ }
8
+ }
9
+
10
+ export class ForbiddenError extends GraphQLError {
11
+ constructor(what: string) {
12
+ super(what, { code: 'FORBIDDEN' });
13
+ }
14
+ }
15
+
16
+ export class NotFoundError extends GraphQLError {
17
+ constructor(what: string) {
18
+ super(what, { code: 'NOT_FOUND' });
19
+ }
20
+ }
21
+
22
+ export class UserInputError extends GraphQLError {
23
+ constructor(what: string) {
24
+ super(what, { code: 'BAD_USER_INPUT' });
25
+ }
26
+ }
27
+
28
+ export class PermissionError extends ForbiddenError {
29
+ constructor(action: PermissionAction, what: string) {
30
+ super(`You do not have sufficient permissions to ${action.toLowerCase()} ${what}.`);
31
+ }
32
+ }
@@ -0,0 +1,273 @@
1
+ import { buildASTSchema, DefinitionNode, DocumentNode, GraphQLSchema, print } from 'graphql';
2
+ import flatMap from 'lodash/flatMap';
3
+ import {
4
+ Field,
5
+ isEnumModel,
6
+ isJsonObjectModel,
7
+ isQueriableField,
8
+ isRawEnumModel,
9
+ isRawObjectModel,
10
+ isScalarModel,
11
+ RawModels,
12
+ } from '../models';
13
+ import { getModelPluralField, getModels, typeToField } from '../utils';
14
+ import { document, enm, input, object, scalar } from './utils';
15
+
16
+ export const generateDefinitions = (rawModels: RawModels): DefinitionNode[] => {
17
+ const models = getModels(rawModels);
18
+
19
+ return [
20
+ // Predefined types
21
+ enm('Order', ['ASC', 'DESC']),
22
+ scalar('DateTime'),
23
+ scalar('Upload'),
24
+
25
+ ...rawModels.filter(isEnumModel).map((model) => enm(model.name, model.values)),
26
+ ...rawModels.filter(isRawEnumModel).map((model) => enm(model.name, model.values)),
27
+ ...rawModels.filter(isScalarModel).map((model) => scalar(model.name)),
28
+ ...rawModels.filter(isRawObjectModel).map((model) => object(model.name, model.fields)),
29
+ ...rawModels.filter(isJsonObjectModel).map((model) => object(model.name, model.fields)),
30
+ ...rawModels
31
+ .filter(isRawObjectModel)
32
+ .filter(({ rawFilters }) => rawFilters)
33
+ .map((model) =>
34
+ input(
35
+ `${model.name}Where`,
36
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion -- array gets filtered above to only include models with rawFilters
37
+ model.rawFilters!.map(({ name, type, list = false, nonNull = false }) => ({ name, type, list, nonNull }))
38
+ )
39
+ ),
40
+
41
+ ...flatMap(
42
+ models.map((model) => {
43
+ const types = [
44
+ object(
45
+ model.name,
46
+ [
47
+ ...model.fields.filter(isQueriableField).map((field) => ({
48
+ ...field,
49
+ args: [
50
+ ...(field.args || []),
51
+ ...(hasRawFilters(rawModels, field.type) ? [{ name: 'where', type: `${field.type}Where` }] : []),
52
+ ],
53
+ directives: field.directives,
54
+ })),
55
+ ...model.reverseRelations.map(({ name, field, model }) => ({
56
+ name,
57
+ type: model.name,
58
+ list: !field.toOne,
59
+ nonNull: !field.toOne,
60
+ args: [
61
+ { name: 'where', type: `${model.name}Where` },
62
+ ...(model.fields.some(({ searchable }) => searchable) ? [{ name: 'search', type: 'String' }] : []),
63
+ ...(model.fields.some(({ orderable }) => orderable)
64
+ ? [{ name: 'orderBy', type: `${model.name}OrderBy`, list: true }]
65
+ : []),
66
+ { name: 'limit', type: 'Int' },
67
+ { name: 'offset', type: 'Int' },
68
+ ],
69
+ })),
70
+ ],
71
+ model.interfaces
72
+ ),
73
+ input(`${model.name}Where`, [
74
+ ...model.fields
75
+ .filter(({ unique, filterable, relation }) => (unique || filterable) && !relation)
76
+ .map(({ name, type, defaultFilter }) => ({ name, type, list: true, default: defaultFilter })),
77
+ ...flatMap(
78
+ model.fields.filter(({ comparable }) => comparable),
79
+ ({ name, type }) => [
80
+ { name: `${name}_GT`, type },
81
+ { name: `${name}_GTE`, type },
82
+ { name: `${name}_LT`, type },
83
+ { name: `${name}_LTE`, type },
84
+ ]
85
+ ),
86
+ ...model.fields
87
+ .filter(({ filterable, relation }) => filterable && relation)
88
+ .map(({ name, type }) => ({
89
+ name,
90
+ type: `${type}Where`,
91
+ })),
92
+ ]),
93
+ input(
94
+ `${model.name}WhereUnique`,
95
+ model.fields.filter(({ unique }) => unique).map(({ name, type }) => ({ name, type }))
96
+ ),
97
+ ...(model.fields.some(({ orderable }) => orderable)
98
+ ? [
99
+ input(
100
+ `${model.name}OrderBy`,
101
+ model.fields.filter(({ orderable }) => orderable).map(({ name }) => ({ name, type: 'Order' }))
102
+ ),
103
+ ]
104
+ : []),
105
+ ];
106
+
107
+ if (model.creatable) {
108
+ types.push(
109
+ input(
110
+ `Create${model.name}`,
111
+ model.fields
112
+ .filter(({ creatable }) => creatable)
113
+ .map(({ name, relation, type, nonNull, list, default: defaultValue }) =>
114
+ relation
115
+ ? { name: `${name}Id`, type: 'ID', nonNull }
116
+ : { name, type, list, nonNull: nonNull && defaultValue === undefined }
117
+ )
118
+ )
119
+ );
120
+ }
121
+
122
+ if (model.updatable) {
123
+ types.push(
124
+ input(
125
+ `Update${model.name}`,
126
+ model.fields
127
+ .filter(({ updatable }) => updatable)
128
+ .map(({ name, relation, type, list }) =>
129
+ relation ? { name: `${name}Id`, type: 'ID' } : { name, type, list }
130
+ )
131
+ )
132
+ );
133
+ }
134
+ return types;
135
+ })
136
+ ),
137
+
138
+ object('Query', [
139
+ {
140
+ name: 'me',
141
+ type: 'User',
142
+ },
143
+ ...models
144
+ .filter(({ queriable }) => queriable)
145
+ .map(({ name }) => ({
146
+ name: typeToField(name),
147
+ type: name,
148
+ nonNull: true,
149
+ args: [
150
+ {
151
+ name: 'where',
152
+ type: `${name}WhereUnique`,
153
+ nonNull: true,
154
+ },
155
+ ],
156
+ })),
157
+ ...models
158
+ .filter(({ listQueriable }) => listQueriable)
159
+ .map((model) => ({
160
+ name: getModelPluralField(model),
161
+ type: model.name,
162
+ list: true,
163
+ nonNull: true,
164
+ args: [
165
+ { name: 'where', type: `${model.name}Where` },
166
+ ...(model.fields.some(({ searchable }) => searchable) ? [{ name: 'search', type: 'String' }] : []),
167
+ ...(model.fields.some(({ orderable }) => orderable)
168
+ ? [{ name: 'orderBy', type: `${model.name}OrderBy`, list: true }]
169
+ : []),
170
+ { name: 'limit', type: 'Int' },
171
+ { name: 'offset', type: 'Int' },
172
+ ],
173
+ })),
174
+ ]),
175
+
176
+ object('Mutation', [
177
+ ...flatMap(
178
+ models.map((model): Field[] => {
179
+ const mutations: Field[] = [];
180
+
181
+ if (model.creatable) {
182
+ mutations.push({
183
+ name: `create${model.name}`,
184
+ type: model.name,
185
+ nonNull: true,
186
+ args: [
187
+ {
188
+ name: 'data',
189
+ type: `Create${model.name}`,
190
+ nonNull: true,
191
+ },
192
+ ],
193
+ });
194
+ }
195
+
196
+ if (model.updatable) {
197
+ mutations.push({
198
+ name: `update${model.name}`,
199
+ type: model.name,
200
+ nonNull: true,
201
+ args: [
202
+ {
203
+ name: 'where',
204
+ type: `${model.name}WhereUnique`,
205
+ nonNull: true,
206
+ },
207
+ {
208
+ name: 'data',
209
+ type: `Update${model.name}`,
210
+ nonNull: true,
211
+ },
212
+ ],
213
+ });
214
+ }
215
+
216
+ if (model.deletable) {
217
+ mutations.push({
218
+ name: `delete${model.name}`,
219
+ type: 'ID',
220
+ nonNull: true,
221
+ args: [
222
+ {
223
+ name: 'where',
224
+ type: `${model.name}WhereUnique`,
225
+ nonNull: true,
226
+ },
227
+ {
228
+ name: 'dryRun',
229
+ type: 'Boolean',
230
+ },
231
+ ],
232
+ });
233
+ mutations.push({
234
+ name: `restore${model.name}`,
235
+ type: 'ID',
236
+ nonNull: true,
237
+ args: [
238
+ {
239
+ name: 'where',
240
+ type: `${model.name}WhereUnique`,
241
+ nonNull: true,
242
+ },
243
+ ],
244
+ });
245
+ }
246
+
247
+ return mutations;
248
+ })
249
+ ),
250
+ ]),
251
+ ];
252
+ };
253
+
254
+ export const generate = (rawModels: RawModels) => document(generateDefinitions(rawModels));
255
+
256
+ export const printSchema = (schema: GraphQLSchema): string =>
257
+ [
258
+ ...schema.getDirectives().map((d) => d.astNode && print(d.astNode)),
259
+ ...Object.values(schema.getTypeMap())
260
+ .filter((t) => !t.name.match(/^__/))
261
+ .sort((a, b) => (a.name > b.name ? 1 : -1))
262
+ .map((t) => t.astNode && print(t.astNode)),
263
+ ]
264
+ .filter(Boolean)
265
+ .map((s) => `${s}\n`)
266
+ .join('\n');
267
+
268
+ const hasRawFilters = (models: RawModels, type: string) =>
269
+ models.filter(isRawObjectModel).some(({ name, rawFilters }) => name === type && !!rawFilters);
270
+
271
+ export const printSchemaFromDocument = (document: DocumentNode) => printSchema(buildASTSchema(document));
272
+
273
+ export const printSchemaFromModels = (models: RawModels) => printSchema(buildASTSchema(generate(models)));