@entity-access/entity-access 1.0.2 → 1.0.6

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 (69) hide show
  1. package/.github/workflows/node.yml +7 -2
  2. package/.vscode/settings.json +1 -0
  3. package/README.md +57 -27
  4. package/package.json +2 -2
  5. package/src/common/EntityAccessError.ts +10 -0
  6. package/src/common/IDisposable.ts +25 -0
  7. package/src/common/ImmutableObject.ts +53 -0
  8. package/src/common/Logger.ts +59 -0
  9. package/src/common/TypeInfo.ts +3 -0
  10. package/src/common/cache/TimedCache.ts +2 -2
  11. package/src/common/usingAsync.ts +42 -12
  12. package/src/compiler/QueryCompiler.ts +28 -30
  13. package/src/compiler/postgres/PostgreSqlMethodTransformer.ts +23 -0
  14. package/src/compiler/sql-server/SqlServerSqlMethodTransformer.ts +23 -0
  15. package/src/decorators/ForeignKey.ts +1 -1
  16. package/src/decorators/IClassOf.ts +2 -1
  17. package/src/decorators/IColumn.ts +2 -0
  18. package/src/decorators/parser/NameParser.ts +15 -0
  19. package/src/di/di.ts +224 -0
  20. package/src/drivers/base/BaseDriver.ts +28 -3
  21. package/src/drivers/sql-server/ExpressionToSqlServer.ts +34 -9
  22. package/src/drivers/sql-server/SqlServerDriver.ts +7 -16
  23. package/src/entity-query/EntityType.ts +45 -3
  24. package/src/model/EntityContext.ts +167 -22
  25. package/src/model/EntityQuery.ts +118 -59
  26. package/src/model/EntitySource.ts +38 -46
  27. package/src/model/IFilterWithParameter.ts +5 -0
  28. package/src/model/SourceExpression.ts +21 -25
  29. package/src/model/{ChangeEntry.ts → changes/ChangeEntry.ts} +35 -5
  30. package/src/model/{ChangeSet.ts → changes/ChangeSet.ts} +16 -11
  31. package/src/model/events/ContextEvents.ts +26 -0
  32. package/src/model/events/EntityEvents.ts +96 -0
  33. package/src/model/{IdentityService.ts → identity/IdentityService.ts} +9 -1
  34. package/src/model/identity/RelationMapper.ts +71 -0
  35. package/src/model/symbols.ts +1 -0
  36. package/src/model/verification/VerificationSession.ts +173 -0
  37. package/src/query/ast/DebugStringVisitor.ts +175 -0
  38. package/src/query/ast/ExpressionToSql.ts +277 -119
  39. package/src/query/ast/Expressions.ts +130 -13
  40. package/src/query/ast/IStringTransformer.ts +19 -5
  41. package/src/query/ast/ParameterScope.ts +97 -0
  42. package/src/query/ast/ReplaceParameter.ts +40 -0
  43. package/src/query/ast/Types.ts +0 -0
  44. package/src/query/ast/Visitor.ts +26 -5
  45. package/src/query/expander/QueryExpander.ts +147 -0
  46. package/src/query/parser/ArrowToExpression.ts +134 -19
  47. package/src/query/parser/BabelVisitor.ts +31 -43
  48. package/src/query/parser/NotSupportedError.ts +5 -0
  49. package/src/query/parser/Restructure.ts +66 -0
  50. package/src/query/parser/TransformVisitor.ts +83 -0
  51. package/src/sql/ISql.ts +10 -0
  52. package/src/tests/db-tests/tests/select-items.ts +12 -0
  53. package/src/tests/expressions/left-joins/child-joins.ts +54 -34
  54. package/src/tests/expressions/sanitize/sanitize-test.ts +17 -0
  55. package/src/tests/expressions/select/select.ts +24 -0
  56. package/src/tests/expressions/simple/parse-arrow.ts +10 -0
  57. package/src/tests/model/ShoppingContext.ts +7 -3
  58. package/src/tests/model/createContext.ts +68 -17
  59. package/src/tests/security/ShoppingContextEvents.ts +20 -0
  60. package/src/tests/security/events/OrderEvents.ts +72 -0
  61. package/src/tests/security/events/ProductEvents.ts +92 -0
  62. package/src/tests/security/events/UserEvents.ts +28 -0
  63. package/src/tests/security/events/UserInfo.ts +7 -0
  64. package/src/tests/security/tests/include-items.ts +19 -0
  65. package/src/tests/security/tests/place-order.ts +104 -0
  66. package/test.js +11 -4
  67. package/tsconfig.json +2 -0
  68. package/src/decorators/parser/MemberParser.ts +0 -8
  69. package/src/model/EntitySchema.ts +0 -21
@@ -0,0 +1,175 @@
1
+ import { ArrowFunctionExpression, BigIntLiteral, BinaryExpression, BooleanLiteral, CallExpression, CoalesceExpression, ConditionalExpression, Constant, DeleteStatement, ExistsExpression, Expression, ExpressionAs, Identifier, InsertStatement, JoinExpression, MemberExpression, NewObjectExpression, NullExpression, NumberLiteral, OrderByExpression, ParameterExpression, QuotedLiteral, ReturnUpdated, SelectStatement, StringLiteral, TableLiteral, TemplateElement, TemplateLiteral, UpdateStatement, ValuesStatement } from "./Expressions.js";
2
+ import Visitor from "./Visitor.js";
3
+
4
+ const isBinary = (type) => /^(BinaryExpression|CoalesceExpression)$/.test(type);
5
+
6
+ export default class DebugStringVisitor extends Visitor<string> {
7
+
8
+ static expressionToString(e: Expression) {
9
+ const dsv = new DebugStringVisitor();
10
+ return dsv.visit(e);
11
+ }
12
+
13
+ visitArrowFunctionExpression(e: ArrowFunctionExpression) {
14
+ return `(${this.visitArray(e.params)}) => ${this.visit(e.body)}`;
15
+ }
16
+
17
+ visitBigIntLiteral(e: BigIntLiteral): string {
18
+ return e.value.toString() + "n";
19
+ }
20
+
21
+ visitBinaryExpression(e: BinaryExpression): string {
22
+ const left = isBinary(e.left.type)
23
+ ? `(${this.visit(e.left)})`
24
+ : this.visit(e.left);
25
+ const right = isBinary(e.right.type)
26
+ ? `(${this.visit(e.right)})`
27
+ : this.visit(e.right);
28
+ return `${left} ${e.operator} ${right}`;
29
+ }
30
+
31
+ visitBooleanLiteral(e: BooleanLiteral): string {
32
+ return String(e.value);
33
+ }
34
+
35
+ visitCallExpression(e: CallExpression): string {
36
+ return `${this.visit(e.callee)}(${this.visitArray(e.arguments)})`;
37
+ }
38
+
39
+ visitCoalesceExpression(e: CoalesceExpression): string {
40
+ const left = isBinary(e.left.type)
41
+ ? `(${this.visit(e.left)})`
42
+ : this.visit(e.left);
43
+ const right = isBinary(e.right.type)
44
+ ? `(${this.visit(e.right)})`
45
+ : this.visit(e.right);
46
+ return `${left} ?? ${right}`;
47
+ }
48
+
49
+ visitConditionalExpression(e: ConditionalExpression): string {
50
+ return `${this.visit(e.test)} ? ${this.visit(e.consequent)} : ${this.visit(e.alternate)}`;
51
+ }
52
+
53
+ visitConstant(e: Constant): string {
54
+ return `"Constant:${e.value}"`;
55
+ }
56
+
57
+ visitExpressionAs(e: ExpressionAs): string {
58
+ return `${this.visit(e.expression)} as ${this.visit(e.alias)}`;
59
+ }
60
+
61
+ visitIdentifier(e: Identifier): string {
62
+ return e.value;
63
+ }
64
+
65
+ visitMemberExpression(e: MemberExpression): string {
66
+ return `${this.visit(e.target)}.${this.visit(e.property)}`;
67
+ }
68
+
69
+ visitTableLiteral(e: TableLiteral): string {
70
+ if (!e.schema) {
71
+ return this.visit(e.name);
72
+ }
73
+ return `${this.visit(e.schema)}.${this.visit(e.name)}`;
74
+ }
75
+
76
+ visitNewObjectExpression(e: NewObjectExpression): string {
77
+ return `({${this.visitArray(e.properties)}})`;
78
+ }
79
+
80
+ visitNullExpression(e: NullExpression): string {
81
+ return "null";
82
+ }
83
+
84
+ visitNumberLiteral(e: NumberLiteral): string {
85
+ return e.value.toString();
86
+ }
87
+
88
+ visitParameterExpression(e: ParameterExpression): string {
89
+ return e.name;
90
+ }
91
+
92
+ visitQuotedLiteral(e: QuotedLiteral): string {
93
+ return `"${e.literal}"`;
94
+ }
95
+
96
+ visitStringLiteral(e: StringLiteral): string {
97
+ return `'${e.value}'`;
98
+ }
99
+
100
+ visitTemplateElement(e: TemplateElement): string {
101
+ return `${e.value.cooked}`;
102
+ }
103
+
104
+ visitTemplateLiteral(e: TemplateLiteral): string {
105
+ const items = [];
106
+ if (e.quasis?.length) {
107
+ for (let i = 0; i<e.quasis.length; i++) {
108
+ items.push(this.visit(e.quasis[i]));
109
+ if (i<e.value.length) {
110
+ items.push("${" + this.visit(e.value[i]) + "}" );
111
+ }
112
+ }
113
+ } else {
114
+ for (const iterator of e.value) {
115
+ if (iterator.type === "StringLiteral") {
116
+ items.push((iterator as StringLiteral).value as string);
117
+ continue;
118
+ }
119
+ items.push("${" + this.visit(iterator) + "}" );
120
+ }
121
+ }
122
+ return "`" + items.join("") + "`";
123
+ }
124
+
125
+ visitDeleteStatement(e: DeleteStatement): string {
126
+ if (e.where) {
127
+ return `DELETE FROM ${this.visit(e.table)} WHERE ${this.visit(e.where)}`;
128
+ }
129
+ return `DELETE FROM ${this.visit(e.table)}`;
130
+ }
131
+
132
+ visitExistsExpression(e: ExistsExpression): string {
133
+ return `EXISTS (${this.visit(e.target)})`;
134
+ }
135
+
136
+ visitInsertStatement(e: InsertStatement): string {
137
+ return `INSERT INTO ${this.visit(e.table)} ${e.values} ${this.visit(e.returnValues)}`;
138
+ }
139
+
140
+ visitJoinExpression(e: JoinExpression): string {
141
+ return `\n${e.joinType} JOIN ${this.visit(e.source)}\n\t\tON ${this.visit(e.where)}\n`;
142
+ }
143
+
144
+ visitOrderByExpression(e: OrderByExpression): string {
145
+ return `${e.target} ${e.descending ? "DESC" : "ASC"}`;
146
+ }
147
+
148
+ visitReturnUpdated(e: ReturnUpdated): string {
149
+ return `\nRETURNING ${this.visitArray(e.fields)}`;
150
+ }
151
+
152
+ visitValuesStatement(e: ValuesStatement): string {
153
+ const rows = e.values.map((x) => `(${this.visit(x[0])})`).join(",\n\t");
154
+ return `(VALUES ${rows}) as ${this.visit(e.as)})`;
155
+ }
156
+
157
+ visitSelectStatement(e: SelectStatement): string {
158
+ const select = `SELECT\n\t${this.visitArray(e.fields, ",\n\t")}\n\tFROM ${this.visit(e.source)}`;
159
+ const as = e.as ? this.visit(e.as): "";
160
+ const joins = e.joins?.length > 0 ? this.visitArray(e.joins, "\n\t") : "";
161
+ const where = e.where ? `\n\tWHERE ${this.visit(e.where)}` : "";
162
+ const orderBy = e.orderBy ? `\n\tORDER BY ${this.visitArray(e.orderBy, "\n\t\tTHEN BY")}`: "";
163
+ const limit = e.limit > 0 ? `\n\tLIMIT ${e.limit}` : "";
164
+ const offset = e.offset > 0 ? `\n\OFFSET ${e.offset}` : "";
165
+ return `${select}${as}${joins}${where}${orderBy}${limit}${offset}`;
166
+ }
167
+
168
+ visitUpdateStatement(e: UpdateStatement): string {
169
+ return "UPDATE";
170
+ }
171
+
172
+ private visitArray(e: Expression[], separator = ", ") {
173
+ return e.map((x) => this.visit(x)).join(separator);
174
+ }
175
+ }