@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.
- package/.eslintrc +21 -0
- package/.github/workflows/release.yml +24 -0
- package/.github/workflows/testing.yml +37 -0
- package/.nvmrc +1 -0
- package/.prettierignore +34 -0
- package/.prettierrc.json +1 -0
- package/.releaserc +27 -0
- package/CHANGELOG.md +6 -0
- package/README.md +15 -0
- package/dist/cjs/index.cjs +2646 -0
- package/dist/esm/client/gql.d.ts +1 -0
- package/dist/esm/client/gql.js +5 -0
- package/dist/esm/client/gql.js.map +1 -0
- package/dist/esm/client/index.d.ts +2 -0
- package/dist/esm/client/index.js +4 -0
- package/dist/esm/client/index.js.map +1 -0
- package/dist/esm/client/queries.d.ts +24 -0
- package/dist/esm/client/queries.js +152 -0
- package/dist/esm/client/queries.js.map +1 -0
- package/dist/esm/context.d.ts +30 -0
- package/dist/esm/context.js +2 -0
- package/dist/esm/context.js.map +1 -0
- package/dist/esm/errors.d.ts +17 -0
- package/dist/esm/errors.js +27 -0
- package/dist/esm/errors.js.map +1 -0
- package/dist/esm/generate/generate.d.ts +7 -0
- package/dist/esm/generate/generate.js +211 -0
- package/dist/esm/generate/generate.js.map +1 -0
- package/dist/esm/generate/index.d.ts +3 -0
- package/dist/esm/generate/index.js +5 -0
- package/dist/esm/generate/index.js.map +1 -0
- package/dist/esm/generate/mutations.d.ts +2 -0
- package/dist/esm/generate/mutations.js +18 -0
- package/dist/esm/generate/mutations.js.map +1 -0
- package/dist/esm/generate/utils.d.ts +22 -0
- package/dist/esm/generate/utils.js +150 -0
- package/dist/esm/generate/utils.js.map +1 -0
- package/dist/esm/index.d.ts +10 -0
- package/dist/esm/index.js +12 -0
- package/dist/esm/index.js.map +1 -0
- package/dist/esm/migrations/generate.d.ts +28 -0
- package/dist/esm/migrations/generate.js +516 -0
- package/dist/esm/migrations/generate.js.map +1 -0
- package/dist/esm/migrations/index.d.ts +1 -0
- package/dist/esm/migrations/index.js +3 -0
- package/dist/esm/migrations/index.js.map +1 -0
- package/dist/esm/models.d.ts +170 -0
- package/dist/esm/models.js +27 -0
- package/dist/esm/models.js.map +1 -0
- package/dist/esm/permissions/check.d.ts +15 -0
- package/dist/esm/permissions/check.js +162 -0
- package/dist/esm/permissions/check.js.map +1 -0
- package/dist/esm/permissions/generate.d.ts +45 -0
- package/dist/esm/permissions/generate.js +77 -0
- package/dist/esm/permissions/generate.js.map +1 -0
- package/dist/esm/permissions/index.d.ts +2 -0
- package/dist/esm/permissions/index.js +4 -0
- package/dist/esm/permissions/index.js.map +1 -0
- package/dist/esm/resolvers/arguments.d.ts +26 -0
- package/dist/esm/resolvers/arguments.js +88 -0
- package/dist/esm/resolvers/arguments.js.map +1 -0
- package/dist/esm/resolvers/filters.d.ts +5 -0
- package/dist/esm/resolvers/filters.js +126 -0
- package/dist/esm/resolvers/filters.js.map +1 -0
- package/dist/esm/resolvers/index.d.ts +7 -0
- package/dist/esm/resolvers/index.js +9 -0
- package/dist/esm/resolvers/index.js.map +1 -0
- package/dist/esm/resolvers/mutations.d.ts +3 -0
- package/dist/esm/resolvers/mutations.js +255 -0
- package/dist/esm/resolvers/mutations.js.map +1 -0
- package/dist/esm/resolvers/node.d.ts +44 -0
- package/dist/esm/resolvers/node.js +102 -0
- package/dist/esm/resolvers/node.js.map +1 -0
- package/dist/esm/resolvers/resolver.d.ts +5 -0
- package/dist/esm/resolvers/resolver.js +143 -0
- package/dist/esm/resolvers/resolver.js.map +1 -0
- package/dist/esm/resolvers/resolvers.d.ts +9 -0
- package/dist/esm/resolvers/resolvers.js +39 -0
- package/dist/esm/resolvers/resolvers.js.map +1 -0
- package/dist/esm/resolvers/utils.d.ts +43 -0
- package/dist/esm/resolvers/utils.js +125 -0
- package/dist/esm/resolvers/utils.js.map +1 -0
- package/dist/esm/utils.d.ts +25 -0
- package/dist/esm/utils.js +159 -0
- package/dist/esm/utils.js.map +1 -0
- package/dist/esm/values.d.ts +15 -0
- package/dist/esm/values.js +7 -0
- package/dist/esm/values.js.map +1 -0
- package/jest.config.ts +12 -0
- package/package.json +66 -0
- package/renovate.json +32 -0
- package/src/client/gql.ts +7 -0
- package/src/client/index.ts +4 -0
- package/src/client/queries.ts +251 -0
- package/src/context.ts +27 -0
- package/src/errors.ts +32 -0
- package/src/generate/generate.ts +273 -0
- package/src/generate/index.ts +5 -0
- package/src/generate/mutations.ts +35 -0
- package/src/generate/utils.ts +223 -0
- package/src/index.ts +12 -0
- package/src/migrations/generate.ts +633 -0
- package/src/migrations/index.ts +3 -0
- package/src/models.ts +228 -0
- package/src/permissions/check.ts +239 -0
- package/src/permissions/generate.ts +143 -0
- package/src/permissions/index.ts +4 -0
- package/src/resolvers/arguments.ts +129 -0
- package/src/resolvers/filters.ts +163 -0
- package/src/resolvers/index.ts +9 -0
- package/src/resolvers/mutations.ts +313 -0
- package/src/resolvers/node.ts +193 -0
- package/src/resolvers/resolver.ts +223 -0
- package/src/resolvers/resolvers.ts +40 -0
- package/src/resolvers/utils.ts +188 -0
- package/src/utils.ts +186 -0
- package/src/values.ts +19 -0
- package/tests/unit/__snapshots__/generate.spec.ts.snap +105 -0
- package/tests/unit/__snapshots__/resolve.spec.ts.snap +60 -0
- package/tests/unit/generate.spec.ts +8 -0
- package/tests/unit/resolve.spec.ts +128 -0
- package/tests/unit/utils.ts +82 -0
- package/tsconfig.jest.json +13 -0
- 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,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)));
|