zenstack 0.1.41 → 1.0.0

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 (215) hide show
  1. package/.vscode/extensions.json +7 -0
  2. package/.vscode/launch.json +49 -0
  3. package/.vscode/settings.json +4 -0
  4. package/README.md +1 -0
  5. package/package.json +8 -90
  6. package/packages/internal/jest.config.ts +32 -0
  7. package/packages/internal/package.json +42 -0
  8. package/packages/internal/src/constants.ts +1 -0
  9. package/packages/internal/src/handler/data/guard-utils.ts +7 -0
  10. package/packages/internal/src/handler/data/handler.ts +415 -0
  11. package/packages/internal/src/handler/data/query-processor.ts +504 -0
  12. package/packages/internal/src/handler/index.ts +1 -0
  13. package/packages/internal/src/handler/types.ts +20 -0
  14. package/packages/internal/src/index.ts +3 -0
  15. package/packages/internal/src/request-handler.ts +27 -0
  16. package/packages/internal/src/request.ts +101 -0
  17. package/packages/internal/src/types.ts +40 -0
  18. package/packages/internal/tests/query-processor.test.ts +172 -0
  19. package/{out/cli/tsconfig.template.json → packages/internal/tsconfig.json} +7 -3
  20. package/packages/runtime/auth.d.ts +1 -0
  21. package/packages/runtime/auth.js +3 -0
  22. package/packages/runtime/hooks.d.ts +10 -0
  23. package/packages/runtime/hooks.js +3 -0
  24. package/packages/runtime/index.d.ts +3 -0
  25. package/packages/runtime/index.js +1 -0
  26. package/packages/runtime/package-lock.json +512 -0
  27. package/packages/runtime/package.json +16 -0
  28. package/packages/runtime/server.d.ts +1 -0
  29. package/packages/runtime/server.js +3 -0
  30. package/packages/runtime/types.d.ts +1 -0
  31. package/packages/runtime/types.js +3 -0
  32. package/packages/schema/.eslintrc.json +13 -0
  33. package/packages/schema/.vscodeignore +4 -0
  34. package/packages/schema/asset/logo-dark.png +0 -0
  35. package/packages/schema/asset/logo-light.png +0 -0
  36. package/{bin → packages/schema/bin}/cli +0 -0
  37. package/packages/schema/jest.config.ts +32 -0
  38. package/packages/schema/langium-config.json +14 -0
  39. package/packages/schema/langium-quickstart.md +41 -0
  40. package/packages/schema/language-configuration.json +30 -0
  41. package/packages/schema/package.json +96 -0
  42. package/packages/schema/src/cli/cli-util.ts +80 -0
  43. package/packages/schema/src/cli/index.ts +64 -0
  44. package/packages/schema/src/extension.ts +76 -0
  45. package/packages/schema/src/generator/constants.ts +5 -0
  46. package/packages/schema/src/generator/index.ts +92 -0
  47. package/{out/generator/next-auth/index.js → packages/schema/src/generator/next-auth/index.ts} +46 -58
  48. package/{out → packages/schema/src}/generator/package.template.json +0 -0
  49. package/packages/schema/src/generator/prisma/expression-writer.ts +352 -0
  50. package/packages/schema/src/generator/prisma/index.ts +32 -0
  51. package/packages/schema/src/generator/prisma/plain-expression-builder.ts +91 -0
  52. package/packages/schema/src/generator/prisma/prisma-builder.ts +366 -0
  53. package/packages/schema/src/generator/prisma/query-gard-generator.ts +208 -0
  54. package/packages/schema/src/generator/prisma/schema-generator.ts +300 -0
  55. package/packages/schema/src/generator/react-hooks/index.ts +181 -0
  56. package/packages/schema/src/generator/service/index.ts +107 -0
  57. package/{out → packages/schema/src}/generator/tsconfig.template.json +0 -0
  58. package/packages/schema/src/generator/types.ts +17 -0
  59. package/packages/schema/src/generator/utils.ts +9 -0
  60. package/packages/schema/src/language-server/generated/ast.ts +603 -0
  61. package/{out/language-server/generated/grammar.js → packages/schema/src/language-server/generated/grammar.ts} +5 -8
  62. package/packages/schema/src/language-server/generated/module.ts +24 -0
  63. package/packages/schema/src/language-server/main.ts +12 -0
  64. package/{out → packages/schema/src}/language-server/stdlib.zmodel +0 -0
  65. package/packages/schema/src/language-server/types.ts +9 -0
  66. package/packages/schema/src/language-server/zmodel-index.ts +33 -0
  67. package/packages/schema/src/language-server/zmodel-linker.ts +409 -0
  68. package/packages/schema/src/language-server/zmodel-module.ts +90 -0
  69. package/packages/schema/src/language-server/zmodel-scope.ts +21 -0
  70. package/packages/schema/src/language-server/zmodel-validator.ts +35 -0
  71. package/packages/schema/src/language-server/zmodel.langium +186 -0
  72. package/packages/schema/src/utils/exec-utils.ts +5 -0
  73. package/packages/schema/src/utils/indent-string.ts +6 -0
  74. package/packages/schema/syntaxes/zmodel.json +57 -0
  75. package/packages/schema/syntaxes/zmodel.tmLanguage.json +57 -0
  76. package/packages/schema/tests/generator/expression-writer.test.ts +676 -0
  77. package/packages/schema/tests/generator/prisma-builder.test.ts +138 -0
  78. package/packages/schema/tests/schema/parser.test.ts +423 -0
  79. package/packages/schema/tests/schema/sample-todo.test.ts +14 -0
  80. package/packages/schema/tests/utils.ts +38 -0
  81. package/packages/schema/tsconfig.json +23 -0
  82. package/pnpm-workspace.yaml +3 -0
  83. package/samples/todo/.env +2 -0
  84. package/samples/todo/.eslintrc.json +3 -0
  85. package/samples/todo/.vscode/launch.json +11 -0
  86. package/samples/todo/README.md +34 -0
  87. package/samples/todo/components/AuthGuard.tsx +17 -0
  88. package/samples/todo/components/Avatar.tsx +22 -0
  89. package/samples/todo/components/BreadCrumb.tsx +44 -0
  90. package/samples/todo/components/ManageMembers.tsx +134 -0
  91. package/samples/todo/components/NavBar.tsx +57 -0
  92. package/samples/todo/components/SpaceMembers.tsx +76 -0
  93. package/samples/todo/components/Spaces.tsx +28 -0
  94. package/samples/todo/components/TimeInfo.tsx +17 -0
  95. package/samples/todo/components/Todo.tsx +72 -0
  96. package/samples/todo/components/TodoList.tsx +77 -0
  97. package/samples/todo/lib/context.ts +31 -0
  98. package/samples/todo/next.config.js +10 -0
  99. package/samples/todo/package-lock.json +7527 -0
  100. package/samples/todo/package.json +45 -0
  101. package/samples/todo/pages/_app.tsx +50 -0
  102. package/samples/todo/pages/api/auth/[...nextauth].ts +83 -0
  103. package/samples/todo/pages/api/zenstack/[...path].ts +16 -0
  104. package/samples/todo/pages/create-space.tsx +114 -0
  105. package/samples/todo/pages/index.tsx +32 -0
  106. package/samples/todo/pages/space/[slug]/[listId]/index.tsx +88 -0
  107. package/samples/todo/pages/space/[slug]/index.tsx +169 -0
  108. package/samples/todo/postcss.config.js +6 -0
  109. package/samples/todo/public/avatar.jpg +0 -0
  110. package/samples/todo/public/favicon.ico +0 -0
  111. package/samples/todo/public/logo.png +0 -0
  112. package/samples/todo/public/vercel.svg +4 -0
  113. package/samples/todo/styles/globals.css +7 -0
  114. package/samples/todo/tailwind.config.js +11 -0
  115. package/samples/todo/tsconfig.json +28 -0
  116. package/samples/todo/types/next-auth.d.ts +14 -0
  117. package/samples/todo/types/next.d.ts +16 -0
  118. package/samples/todo/zenstack/migrations/20221014084317_init/migration.sql +153 -0
  119. package/samples/todo/zenstack/migrations/20221020094651_upate_cli/migration.sql +23 -0
  120. package/samples/todo/zenstack/migrations/migration_lock.toml +3 -0
  121. package/samples/todo/zenstack/schema.prisma +126 -0
  122. package/samples/todo/zenstack/schema.zmodel +161 -0
  123. package/tests/integration/jest.config.ts +16 -0
  124. package/tests/integration/package-lock.json +1081 -0
  125. package/tests/integration/package.json +27 -0
  126. package/tests/integration/tests/operation-coverate.test.ts +563 -0
  127. package/tests/integration/tests/operations.zmodel +69 -0
  128. package/tests/integration/tests/todo-e2e.test.ts +577 -0
  129. package/tests/integration/tests/todo.zmodel +123 -0
  130. package/tests/integration/tests/tsconfig.template.json +10 -0
  131. package/tests/integration/tests/utils.ts +133 -0
  132. package/tests/integration/tsconfig.json +10 -0
  133. package/out/cli/cli-util.js +0 -64
  134. package/out/cli/cli-util.js.map +0 -1
  135. package/out/cli/generator.js +0 -1
  136. package/out/cli/generator.js.map +0 -1
  137. package/out/cli/index.js +0 -46
  138. package/out/cli/index.js.map +0 -1
  139. package/out/cli/package.template.json +0 -10
  140. package/out/extension.js +0 -81
  141. package/out/extension.js.map +0 -1
  142. package/out/generator/constants.js +0 -9
  143. package/out/generator/constants.js.map +0 -1
  144. package/out/generator/data-server/index.js +0 -1
  145. package/out/generator/data-server/index.js.map +0 -1
  146. package/out/generator/index.js +0 -98
  147. package/out/generator/index.js.map +0 -1
  148. package/out/generator/next-auth/index.js.map +0 -1
  149. package/out/generator/prisma/expression-writer.js +0 -287
  150. package/out/generator/prisma/expression-writer.js.map +0 -1
  151. package/out/generator/prisma/index.js +0 -38
  152. package/out/generator/prisma/index.js.map +0 -1
  153. package/out/generator/prisma/plain-expression-builder.js +0 -69
  154. package/out/generator/prisma/plain-expression-builder.js.map +0 -1
  155. package/out/generator/prisma/prisma-builder.js +0 -307
  156. package/out/generator/prisma/prisma-builder.js.map +0 -1
  157. package/out/generator/prisma/query-gard-generator.js +0 -159
  158. package/out/generator/prisma/query-gard-generator.js.map +0 -1
  159. package/out/generator/prisma/schema-generator.js +0 -201
  160. package/out/generator/prisma/schema-generator.js.map +0 -1
  161. package/out/generator/query-guard/index.js +0 -2
  162. package/out/generator/query-guard/index.js.map +0 -1
  163. package/out/generator/react-hooks/index.js +0 -179
  164. package/out/generator/react-hooks/index.js.map +0 -1
  165. package/out/generator/server/data/data-generator.js +0 -376
  166. package/out/generator/server/data/data-generator.js.map +0 -1
  167. package/out/generator/server/data/expression-writer.js +0 -287
  168. package/out/generator/server/data/expression-writer.js.map +0 -1
  169. package/out/generator/server/data/plain-expression-builder.js +0 -69
  170. package/out/generator/server/data/plain-expression-builder.js.map +0 -1
  171. package/out/generator/server/data-generator.js +0 -82
  172. package/out/generator/server/data-generator.js.map +0 -1
  173. package/out/generator/server/expression-writer.js +0 -1
  174. package/out/generator/server/expression-writer.js.map +0 -1
  175. package/out/generator/server/function/function-generator.js +0 -50
  176. package/out/generator/server/function/function-generator.js.map +0 -1
  177. package/out/generator/server/function-generator.js +0 -13
  178. package/out/generator/server/function-generator.js.map +0 -1
  179. package/out/generator/server/index.js +0 -88
  180. package/out/generator/server/index.js.map +0 -1
  181. package/out/generator/server/js-expression-builder.js +0 -1
  182. package/out/generator/server/js-expression-builder.js.map +0 -1
  183. package/out/generator/server/plain-expression-builder.js +0 -1
  184. package/out/generator/server/plain-expression-builder.js.map +0 -1
  185. package/out/generator/server/server-code-generator.js +0 -3
  186. package/out/generator/server/server-code-generator.js.map +0 -1
  187. package/out/generator/server/server-code-writer.js +0 -1
  188. package/out/generator/server/server-code-writer.js.map +0 -1
  189. package/out/generator/service/index.js +0 -133
  190. package/out/generator/service/index.js.map +0 -1
  191. package/out/generator/types.js +0 -10
  192. package/out/generator/types.js.map +0 -1
  193. package/out/generator/utils.js +0 -10
  194. package/out/generator/utils.js.map +0 -1
  195. package/out/language-server/generated/ast.js +0 -386
  196. package/out/language-server/generated/ast.js.map +0 -1
  197. package/out/language-server/generated/grammar.js.map +0 -1
  198. package/out/language-server/generated/module.js +0 -23
  199. package/out/language-server/generated/module.js.map +0 -1
  200. package/out/language-server/main.js +0 -12
  201. package/out/language-server/main.js.map +0 -1
  202. package/out/language-server/types.js +0 -3
  203. package/out/language-server/types.js.map +0 -1
  204. package/out/language-server/zmodel-index.js +0 -38
  205. package/out/language-server/zmodel-index.js.map +0 -1
  206. package/out/language-server/zmodel-linker.js +0 -241
  207. package/out/language-server/zmodel-linker.js.map +0 -1
  208. package/out/language-server/zmodel-module.js +0 -51
  209. package/out/language-server/zmodel-module.js.map +0 -1
  210. package/out/language-server/zmodel-scope.js +0 -30
  211. package/out/language-server/zmodel-scope.js.map +0 -1
  212. package/out/language-server/zmodel-validator.js +0 -25
  213. package/out/language-server/zmodel-validator.js.map +0 -1
  214. package/out/utils/indent-string.js +0 -9
  215. package/out/utils/indent-string.js.map +0 -1
@@ -0,0 +1,352 @@
1
+ import {
2
+ BinaryExpr,
3
+ Expression,
4
+ isDataModel,
5
+ isDataModelField,
6
+ isEnumField,
7
+ isMemberAccessExpr,
8
+ isReferenceExpr,
9
+ isThisExpr,
10
+ LiteralExpr,
11
+ MemberAccessExpr,
12
+ ReferenceExpr,
13
+ UnaryExpr,
14
+ } from '@lang/generated/ast';
15
+ import { CodeBlockWriter } from 'ts-morph';
16
+ import { GeneratorError } from '../types';
17
+ import { TypedNode } from '@lang/types';
18
+ import PlainExpressionBuilder from './plain-expression-builder';
19
+
20
+ const AUX_GUARD_FIELD = 'zenstack_guard';
21
+
22
+ type ComparisonOperator = '==' | '!=' | '>' | '>=' | '<' | '<=';
23
+
24
+ export default class ExpressionWriter {
25
+ private readonly plainExprBuilder = new PlainExpressionBuilder();
26
+
27
+ constructor(private readonly writer: CodeBlockWriter) {}
28
+
29
+ write(expr: Expression) {
30
+ const _write = () => {
31
+ switch (expr.$type) {
32
+ case LiteralExpr:
33
+ this.writeLiteral(expr as LiteralExpr);
34
+ break;
35
+
36
+ case UnaryExpr:
37
+ this.writeUnary(expr as UnaryExpr);
38
+ break;
39
+
40
+ case BinaryExpr:
41
+ this.writeBinary(expr as BinaryExpr);
42
+ break;
43
+
44
+ case ReferenceExpr:
45
+ this.writeReference(expr as ReferenceExpr);
46
+ break;
47
+
48
+ case MemberAccessExpr:
49
+ this.writeMemberAccess(expr as MemberAccessExpr);
50
+ break;
51
+
52
+ default:
53
+ throw new Error(`Not implemented: ${expr.$type}`);
54
+ }
55
+ };
56
+
57
+ this.block(_write);
58
+ }
59
+
60
+ private writeReference(expr: ReferenceExpr) {
61
+ if (isEnumField(expr.target.ref)) {
62
+ throw new Error('We should never get here');
63
+ } else {
64
+ this.writer.write(`${expr.target.ref!.name}: true`);
65
+ }
66
+ }
67
+
68
+ private writeMemberAccess(expr: MemberAccessExpr) {
69
+ this.writeFieldCondition(
70
+ expr.operand,
71
+ () => {
72
+ this.block(() => {
73
+ this.writer.write(`${expr.member.ref?.name}: true`);
74
+ });
75
+ },
76
+ 'is'
77
+ );
78
+ }
79
+
80
+ private writeExprList(exprs: Expression[]) {
81
+ this.writer.writeLine('[');
82
+ for (let i = 0; i < exprs.length; i++) {
83
+ this.write(exprs[i]);
84
+ if (i !== exprs.length - 1) {
85
+ this.writer.writeLine(',');
86
+ }
87
+ }
88
+ this.writer.writeLine(']');
89
+ }
90
+
91
+ private writeBinary(expr: BinaryExpr) {
92
+ switch (expr.operator) {
93
+ case '&&':
94
+ case '||':
95
+ this.writeLogical(expr, expr.operator);
96
+ break;
97
+
98
+ case '==':
99
+ case '!=':
100
+ case '>':
101
+ case '>=':
102
+ case '<':
103
+ case '<=':
104
+ this.writeComparison(expr, expr.operator);
105
+ break;
106
+
107
+ case '?':
108
+ case '!':
109
+ case '^':
110
+ this.writeCollectionPredicate(expr, expr.operator);
111
+ break;
112
+ }
113
+ }
114
+
115
+ private writeCollectionPredicate(expr: BinaryExpr, operator: string) {
116
+ this.writeFieldCondition(
117
+ expr.left,
118
+ () => {
119
+ this.write(expr.right);
120
+ },
121
+ operator === '?' ? 'some' : operator === '!' ? 'every' : 'none'
122
+ );
123
+ }
124
+
125
+ private isFieldAccess(expr: Expression): boolean {
126
+ if (isThisExpr(expr)) {
127
+ return true;
128
+ }
129
+ if (isMemberAccessExpr(expr)) {
130
+ return this.isFieldAccess(expr.operand);
131
+ }
132
+ if (isReferenceExpr(expr) && isDataModelField(expr.target.ref)) {
133
+ return true;
134
+ }
135
+ return false;
136
+ }
137
+
138
+ private guard(write: () => void) {
139
+ this.writer.write(`${AUX_GUARD_FIELD}: `);
140
+ write();
141
+ }
142
+
143
+ private plain(expr: Expression) {
144
+ this.writer.write(this.plainExprBuilder.build(expr));
145
+ }
146
+
147
+ private writeComparison(expr: BinaryExpr, operator: ComparisonOperator) {
148
+ const leftIsFieldAccess = this.isFieldAccess(expr.left);
149
+ const rightIsFieldAccess = this.isFieldAccess(expr.right);
150
+
151
+ if (leftIsFieldAccess && rightIsFieldAccess) {
152
+ throw new GeneratorError(
153
+ `Comparison between fields are not supported yet`
154
+ );
155
+ }
156
+
157
+ if (!leftIsFieldAccess && !rightIsFieldAccess) {
158
+ // compile down to a plain expression
159
+ this.guard(() => {
160
+ this.plain(expr.left);
161
+ this.writer.write(' ' + operator + ' ');
162
+ this.plain(expr.right);
163
+ });
164
+
165
+ return;
166
+ }
167
+
168
+ let fieldAccess: Expression;
169
+ let operand: Expression;
170
+ if (leftIsFieldAccess) {
171
+ fieldAccess = expr.left;
172
+ operand = expr.right;
173
+ } else {
174
+ fieldAccess = expr.right;
175
+ operand = expr.left;
176
+ operator = this.negateOperator(operator);
177
+ }
178
+
179
+ this.writeFieldCondition(
180
+ fieldAccess,
181
+ () => {
182
+ this.block(
183
+ () => {
184
+ if (this.isModelTyped(fieldAccess)) {
185
+ // comparing with an object, conver to "id" comparison instead
186
+ this.writer.write('id: ');
187
+ this.block(() => {
188
+ this.writeOperator(operator, () => {
189
+ this.plain(operand);
190
+ this.writer.write('?.id');
191
+ });
192
+ });
193
+ } else {
194
+ this.writeOperator(operator, () => {
195
+ this.plain(operand);
196
+ });
197
+ }
198
+ },
199
+ // "this" expression is compiled away (to .id access), so we should
200
+ // avoid generating a new layer
201
+ !isThisExpr(fieldAccess)
202
+ );
203
+ },
204
+ 'is'
205
+ );
206
+ }
207
+
208
+ private writeOperator(
209
+ operator: ComparisonOperator,
210
+ writeOperand: () => void
211
+ ) {
212
+ if (operator === '!=') {
213
+ // wrap a 'not'
214
+ this.writer.write('not: ');
215
+ this.block(() => {
216
+ this.writeOperator('==', writeOperand);
217
+ });
218
+ } else {
219
+ this.writer.write(`${this.mapOperator(operator)}: `);
220
+ writeOperand();
221
+ }
222
+ }
223
+
224
+ private writeFieldCondition(
225
+ fieldAccess: Expression,
226
+ writeCondition: () => void,
227
+ relationOp: 'is' | 'some' | 'every' | 'none'
228
+ ) {
229
+ let selector: string;
230
+ let operand: Expression | undefined;
231
+
232
+ if (isThisExpr(fieldAccess)) {
233
+ // pass on
234
+ writeCondition();
235
+ return;
236
+ } else if (isReferenceExpr(fieldAccess)) {
237
+ selector = fieldAccess.target.ref?.name!;
238
+ } else if (isMemberAccessExpr(fieldAccess)) {
239
+ selector = fieldAccess.member.ref?.name!;
240
+ operand = fieldAccess.operand;
241
+ } else {
242
+ throw new GeneratorError(
243
+ `Unsupported expression type: ${fieldAccess.$type}`
244
+ );
245
+ }
246
+
247
+ if (operand) {
248
+ // member access expression
249
+ this.writeFieldCondition(
250
+ operand,
251
+ () => {
252
+ this.block(
253
+ () => {
254
+ this.writer.write(selector + ': ');
255
+ if (this.isModelTyped(fieldAccess)) {
256
+ // expression is resolved to a model, generate relation query
257
+ this.block(() => {
258
+ this.writer.write(`${relationOp}: `);
259
+ writeCondition();
260
+ });
261
+ } else {
262
+ // generate plain query
263
+ writeCondition();
264
+ }
265
+ },
266
+ // if operand is "this", it doesn't really generate a new layer of query,
267
+ // so we should avoid generating a new block
268
+ !isThisExpr(operand)
269
+ );
270
+ },
271
+ 'is'
272
+ );
273
+ } else if (this.isModelTyped(fieldAccess)) {
274
+ // reference resolved to a model, generate relation query
275
+ this.writer.write(selector + ': ');
276
+ this.block(() => {
277
+ this.writer.write(`${relationOp}: `);
278
+ writeCondition();
279
+ });
280
+ } else {
281
+ // generate a plain query
282
+ this.writer.write(selector + ': ');
283
+ writeCondition();
284
+ }
285
+ }
286
+
287
+ private block(write: () => void, condition = true) {
288
+ if (condition) {
289
+ this.writer.block(write);
290
+ } else {
291
+ write();
292
+ }
293
+ }
294
+
295
+ private isModelTyped(expr: Expression) {
296
+ return isDataModel((expr as TypedNode).$resolvedType?.decl);
297
+ }
298
+
299
+ mapOperator(operator: '==' | '!=' | '>' | '>=' | '<' | '<=') {
300
+ switch (operator) {
301
+ case '==':
302
+ return 'equals';
303
+ case '!=':
304
+ throw new Error('Operation != should have been compiled away');
305
+ case '>':
306
+ return 'gt';
307
+ case '>=':
308
+ return 'ge';
309
+ case '<':
310
+ return 'lt';
311
+ case '<=':
312
+ return 'le';
313
+ }
314
+ }
315
+
316
+ private negateOperator(operator: '==' | '!=' | '>' | '>=' | '<' | '<=') {
317
+ switch (operator) {
318
+ case '>':
319
+ return '<=';
320
+ case '<':
321
+ return '>=';
322
+ case '>=':
323
+ return '<';
324
+ case '<=':
325
+ return '>';
326
+ default:
327
+ return operator;
328
+ }
329
+ }
330
+
331
+ private writeLogical(expr: BinaryExpr, operator: '&&' | '||') {
332
+ this.writer.writeLine(`${operator === '&&' ? 'AND' : 'OR'}: `);
333
+ this.writeExprList([expr.left, expr.right]);
334
+ }
335
+
336
+ private writeUnary(expr: UnaryExpr) {
337
+ if (expr.operator !== '!') {
338
+ throw new GeneratorError(
339
+ `Unary operator "${expr.operator}" is not supported`
340
+ );
341
+ }
342
+
343
+ this.writer.writeLine('NOT: ');
344
+ this.write(expr.operand);
345
+ }
346
+
347
+ private writeLiteral(expr: LiteralExpr) {
348
+ this.guard(() => {
349
+ this.plain(expr);
350
+ });
351
+ }
352
+ }
@@ -0,0 +1,32 @@
1
+ import colors from 'colors';
2
+ import { Context, Generator, GeneratorError } from '../types';
3
+ import { execSync } from '../../utils/exec-utils';
4
+ import PrismaSchemaGenerator from './schema-generator';
5
+ import QueryGuardGenerator from './query-gard-generator';
6
+
7
+ export default class PrismaGenerator implements Generator {
8
+ async generate(context: Context) {
9
+ // generate prisma schema
10
+ const schemaFile = await new PrismaSchemaGenerator(context).generate();
11
+
12
+ // run prisma generate and install @prisma/client
13
+ await this.generatePrismaClient(schemaFile);
14
+
15
+ // generate prisma query guard
16
+ await new QueryGuardGenerator(context).generate();
17
+
18
+ console.log(
19
+ colors.blue(` ✔️ Prisma schema and query guard generated`)
20
+ );
21
+ }
22
+
23
+ async generatePrismaClient(schemaFile: string) {
24
+ try {
25
+ execSync(`npx prisma generate --schema "${schemaFile}"`);
26
+ } catch {
27
+ throw new GeneratorError(
28
+ `Failed to generate client code with Prisma. Check errors above for clues.\nThis normally shouldn't happen. Please file an issue at: http://go.zenstack.dev/bug.`
29
+ );
30
+ }
31
+ }
32
+ }
@@ -0,0 +1,91 @@
1
+ import { GeneratorError } from '../types';
2
+ import {
3
+ ArrayExpr,
4
+ Expression,
5
+ InvocationExpr,
6
+ isEnumField,
7
+ isThisExpr,
8
+ LiteralExpr,
9
+ MemberAccessExpr,
10
+ NullExpr,
11
+ ReferenceExpr,
12
+ ThisExpr,
13
+ } from '@lang/generated/ast';
14
+
15
+ export default class PlainExpressionBuilder {
16
+ build(expr: Expression): string {
17
+ switch (expr.$type) {
18
+ case LiteralExpr:
19
+ return this.literal(expr as LiteralExpr);
20
+
21
+ case ArrayExpr:
22
+ return this.array(expr as ArrayExpr);
23
+
24
+ case NullExpr:
25
+ return this.null();
26
+
27
+ case ThisExpr:
28
+ return this.this(expr as ThisExpr);
29
+
30
+ case ReferenceExpr:
31
+ return this.reference(expr as ReferenceExpr);
32
+
33
+ case InvocationExpr:
34
+ return this.invocation(expr as InvocationExpr);
35
+
36
+ case MemberAccessExpr:
37
+ return this.memberAccess(expr as MemberAccessExpr);
38
+
39
+ default:
40
+ throw new GeneratorError(
41
+ `Unsupported expression type: ${expr.$type}`
42
+ );
43
+ }
44
+ }
45
+
46
+ private this(expr: ThisExpr) {
47
+ // "this" is mapped to id comparison
48
+ return 'id';
49
+ }
50
+
51
+ private memberAccess(expr: MemberAccessExpr) {
52
+ if (isThisExpr(expr.operand)) {
53
+ return expr.member.ref!.name;
54
+ } else {
55
+ return `${this.build(expr.operand)}?.${expr.member.ref!.name}`;
56
+ }
57
+ }
58
+
59
+ private invocation(expr: InvocationExpr) {
60
+ if (expr.function.ref?.name !== 'auth') {
61
+ throw new GeneratorError(
62
+ `Function invocation is not supported: ${expr.function.ref?.name}`
63
+ );
64
+ }
65
+ return 'user';
66
+ }
67
+
68
+ private reference(expr: ReferenceExpr) {
69
+ if (isEnumField(expr.target.ref)) {
70
+ return `${expr.target.ref.$container.name}.${expr.target.ref.name}`;
71
+ } else {
72
+ return expr.target.ref!.name;
73
+ }
74
+ }
75
+
76
+ private null() {
77
+ return 'null';
78
+ }
79
+
80
+ private array(expr: ArrayExpr) {
81
+ return `[${expr.items.map((item) => this.build(item)).join(', ')}]`;
82
+ }
83
+
84
+ private literal(expr: LiteralExpr) {
85
+ if (typeof expr.value === 'string') {
86
+ return `'${expr.value}'`;
87
+ } else {
88
+ return expr.value.toString();
89
+ }
90
+ }
91
+ }