@ruiapp/rapid-core 0.11.5 → 0.11.7

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 (204) hide show
  1. package/CHANGELOG.md +11 -11
  2. package/dist/core/pluginManager.d.ts +2 -0
  3. package/dist/core/server.d.ts +3 -0
  4. package/dist/dataAccess/entityManager.d.ts +3 -3
  5. package/dist/index.js +118 -41
  6. package/dist/server.d.ts +1 -0
  7. package/dist/types.d.ts +2 -0
  8. package/package.json +1 -1
  9. package/rollup.config.js +16 -16
  10. package/src/bootstrapApplicationConfig.ts +904 -904
  11. package/src/core/ExpressionInterpreter.ts +45 -45
  12. package/src/core/actionHandler.ts +24 -24
  13. package/src/core/actionHandlers/ifActionHandler.ts +27 -27
  14. package/src/core/eventManager.ts +20 -20
  15. package/src/core/facility.ts +7 -7
  16. package/src/core/http/formDataParser.ts +87 -87
  17. package/src/core/http-types.ts +4 -4
  18. package/src/core/pluginManager.ts +202 -193
  19. package/src/core/providers/runtimeProvider.ts +5 -5
  20. package/src/core/request.ts +96 -96
  21. package/src/core/response.ts +84 -84
  22. package/src/core/routeContext.ts +127 -127
  23. package/src/core/routes/healthz.ts +20 -20
  24. package/src/core/routes/index.ts +3 -3
  25. package/src/core/routesBuilder.ts +129 -122
  26. package/src/core/server.ts +161 -158
  27. package/src/dataAccess/columnTypeMapper.ts +22 -22
  28. package/src/dataAccess/dataAccessTypes.ts +165 -165
  29. package/src/dataAccess/dataAccessor.ts +135 -135
  30. package/src/dataAccess/entityManager.ts +2031 -1986
  31. package/src/dataAccess/entityMapper.ts +111 -111
  32. package/src/dataAccess/entityValidator.ts +33 -33
  33. package/src/dataAccess/propertyMapper.ts +28 -28
  34. package/src/deno-std/assert/assert.ts +9 -9
  35. package/src/deno-std/assert/assertion_error.ts +7 -7
  36. package/src/deno-std/datetime/to_imf.ts +32 -32
  37. package/src/deno-std/encoding/base64.ts +141 -141
  38. package/src/deno-std/http/cookie.ts +372 -372
  39. package/src/facilities/cache/CacheFacilityTypes.ts +29 -29
  40. package/src/facilities/cache/CacheFactory.ts +31 -31
  41. package/src/facilities/cache/MemoryCache.ts +58 -58
  42. package/src/facilities/cache/MemoryCacheProvider.ts +15 -15
  43. package/src/facilities/log/LogFacility.ts +35 -35
  44. package/src/helpers/dbTransactionHelper.ts +42 -42
  45. package/src/helpers/entityHelper.ts +89 -89
  46. package/src/helpers/filterHelper.ts +148 -148
  47. package/src/helpers/inputHelper.ts +11 -11
  48. package/src/helpers/licenseHelper.ts +29 -29
  49. package/src/helpers/metaHelper.ts +111 -111
  50. package/src/helpers/runCollectionEntityActionHandler.ts +58 -58
  51. package/src/index.ts +85 -85
  52. package/src/plugins/auth/AuthPlugin.ts +107 -107
  53. package/src/plugins/auth/AuthPluginTypes.ts +11 -11
  54. package/src/plugins/auth/actionHandlers/changePassword.ts +61 -61
  55. package/src/plugins/auth/actionHandlers/createSession.ts +68 -68
  56. package/src/plugins/auth/actionHandlers/deleteSession.ts +18 -18
  57. package/src/plugins/auth/actionHandlers/getMyProfile.ts +28 -28
  58. package/src/plugins/auth/actionHandlers/index.ts +8 -8
  59. package/src/plugins/auth/actionHandlers/resetPassword.ts +45 -45
  60. package/src/plugins/auth/models/AccessToken.ts +56 -56
  61. package/src/plugins/auth/models/index.ts +3 -3
  62. package/src/plugins/auth/routes/changePassword.ts +15 -15
  63. package/src/plugins/auth/routes/getMyProfile.ts +15 -15
  64. package/src/plugins/auth/routes/index.ts +7 -7
  65. package/src/plugins/auth/routes/resetPassword.ts +15 -15
  66. package/src/plugins/auth/routes/signin.ts +15 -15
  67. package/src/plugins/auth/routes/signout.ts +15 -15
  68. package/src/plugins/auth/services/AuthService.ts +66 -66
  69. package/src/plugins/cronJob/CronJobPlugin.ts +104 -104
  70. package/src/plugins/cronJob/CronJobPluginTypes.ts +44 -44
  71. package/src/plugins/cronJob/actionHandlers/index.ts +4 -4
  72. package/src/plugins/cronJob/actionHandlers/runCronJob.ts +32 -32
  73. package/src/plugins/cronJob/entityWatchers/cronJobEntityWatchers.ts +24 -24
  74. package/src/plugins/cronJob/entityWatchers/index.ts +4 -4
  75. package/src/plugins/cronJob/models/CronJob.ts +129 -129
  76. package/src/plugins/cronJob/models/index.ts +3 -3
  77. package/src/plugins/cronJob/routes/index.ts +3 -3
  78. package/src/plugins/cronJob/routes/runCronJob.ts +15 -15
  79. package/src/plugins/cronJob/services/CronJobService.ts +255 -255
  80. package/src/plugins/dataManage/DataManagePlugin.ts +165 -165
  81. package/src/plugins/dataManage/actionHandlers/addEntityRelations.ts +15 -15
  82. package/src/plugins/dataManage/actionHandlers/countCollectionEntities.ts +17 -17
  83. package/src/plugins/dataManage/actionHandlers/createCollectionEntitiesBatch.ts +81 -81
  84. package/src/plugins/dataManage/actionHandlers/createCollectionEntity.ts +20 -20
  85. package/src/plugins/dataManage/actionHandlers/deleteCollectionEntities.ts +47 -45
  86. package/src/plugins/dataManage/actionHandlers/deleteCollectionEntityById.ts +20 -20
  87. package/src/plugins/dataManage/actionHandlers/findCollectionEntities.ts +27 -27
  88. package/src/plugins/dataManage/actionHandlers/findCollectionEntityById.ts +30 -30
  89. package/src/plugins/dataManage/actionHandlers/queryDatabase.ts +22 -22
  90. package/src/plugins/dataManage/actionHandlers/removeEntityRelations.ts +15 -15
  91. package/src/plugins/dataManage/actionHandlers/saveEntity.ts +46 -46
  92. package/src/plugins/dataManage/actionHandlers/updateCollectionEntityById.ts +38 -38
  93. package/src/plugins/entityAccessControl/EntityAccessControlPlugin.ts +146 -146
  94. package/src/plugins/fileManage/FileManagePlugin.ts +52 -52
  95. package/src/plugins/fileManage/actionHandlers/downloadDocument.ts +65 -65
  96. package/src/plugins/fileManage/actionHandlers/downloadFile.ts +44 -44
  97. package/src/plugins/fileManage/actionHandlers/uploadFile.ts +33 -33
  98. package/src/plugins/fileManage/routes/downloadDocument.ts +15 -15
  99. package/src/plugins/fileManage/routes/downloadFile.ts +15 -15
  100. package/src/plugins/fileManage/routes/index.ts +5 -5
  101. package/src/plugins/fileManage/routes/uploadFile.ts +15 -15
  102. package/src/plugins/license/LicensePlugin.ts +79 -79
  103. package/src/plugins/license/LicensePluginTypes.ts +95 -95
  104. package/src/plugins/license/LicenseService.ts +141 -141
  105. package/src/plugins/license/actionHandlers/getLicense.ts +18 -18
  106. package/src/plugins/license/actionHandlers/index.ts +5 -5
  107. package/src/plugins/license/actionHandlers/updateLicense.ts +24 -24
  108. package/src/plugins/license/helpers/certHelper.ts +21 -21
  109. package/src/plugins/license/helpers/cryptoHelper.ts +47 -47
  110. package/src/plugins/license/models/index.ts +1 -1
  111. package/src/plugins/license/routes/getLicense.ts +15 -15
  112. package/src/plugins/license/routes/index.ts +4 -4
  113. package/src/plugins/license/routes/updateLicense.ts +15 -15
  114. package/src/plugins/mail/MailPlugin.ts +74 -74
  115. package/src/plugins/mail/MailPluginTypes.ts +27 -27
  116. package/src/plugins/mail/MailService.ts +38 -38
  117. package/src/plugins/mail/actionHandlers/index.ts +3 -3
  118. package/src/plugins/mail/models/index.ts +1 -1
  119. package/src/plugins/mail/routes/index.ts +1 -1
  120. package/src/plugins/metaManage/MetaManagePlugin.ts +198 -198
  121. package/src/plugins/metaManage/actionHandlers/getMetaModelDetail.ts +10 -10
  122. package/src/plugins/metaManage/actionHandlers/listMetaModels.ts +10 -10
  123. package/src/plugins/metaManage/actionHandlers/listMetaRoutes.ts +10 -10
  124. package/src/plugins/metaManage/services/MetaService.ts +376 -376
  125. package/src/plugins/notification/NotificationPlugin.ts +68 -68
  126. package/src/plugins/notification/NotificationPluginTypes.ts +13 -13
  127. package/src/plugins/notification/NotificationService.ts +25 -25
  128. package/src/plugins/notification/actionHandlers/index.ts +3 -3
  129. package/src/plugins/notification/models/Notification.ts +60 -60
  130. package/src/plugins/notification/models/index.ts +3 -3
  131. package/src/plugins/notification/routes/index.ts +1 -1
  132. package/src/plugins/routeManage/RouteManagePlugin.ts +64 -64
  133. package/src/plugins/routeManage/actionHandlers/httpProxy.ts +13 -13
  134. package/src/plugins/routeManage/actionHandlers/mock.ts +28 -28
  135. package/src/plugins/sequence/SequencePlugin.ts +146 -146
  136. package/src/plugins/sequence/SequencePluginTypes.ts +69 -69
  137. package/src/plugins/sequence/SequenceService.ts +92 -92
  138. package/src/plugins/sequence/actionHandlers/generateSn.ts +32 -32
  139. package/src/plugins/sequence/actionHandlers/index.ts +4 -4
  140. package/src/plugins/sequence/models/SequenceAutoIncrementRecord.ts +49 -49
  141. package/src/plugins/sequence/models/SequenceRule.ts +42 -42
  142. package/src/plugins/sequence/models/index.ts +4 -4
  143. package/src/plugins/sequence/routes/generateSn.ts +15 -15
  144. package/src/plugins/sequence/routes/index.ts +3 -3
  145. package/src/plugins/sequence/segment-utility.ts +11 -11
  146. package/src/plugins/sequence/segments/autoIncrement.ts +90 -90
  147. package/src/plugins/sequence/segments/dayOfMonth.ts +19 -19
  148. package/src/plugins/sequence/segments/index.ts +9 -9
  149. package/src/plugins/sequence/segments/literal.ts +16 -16
  150. package/src/plugins/sequence/segments/month.ts +19 -19
  151. package/src/plugins/sequence/segments/parameter.ts +20 -20
  152. package/src/plugins/sequence/segments/year.ts +19 -19
  153. package/src/plugins/serverOperation/ServerOperationPlugin.ts +93 -93
  154. package/src/plugins/serverOperation/ServerOperationPluginTypes.ts +25 -25
  155. package/src/plugins/serverOperation/actionHandlers/index.ts +4 -4
  156. package/src/plugins/serverOperation/actionHandlers/runServerOperation.ts +20 -20
  157. package/src/plugins/setting/SettingPlugin.ts +68 -68
  158. package/src/plugins/setting/SettingPluginTypes.ts +37 -37
  159. package/src/plugins/setting/SettingService.ts +222 -222
  160. package/src/plugins/setting/actionHandlers/getSystemSettingValues.ts +30 -30
  161. package/src/plugins/setting/actionHandlers/getUserSettingValues.ts +38 -38
  162. package/src/plugins/setting/actionHandlers/index.ts +6 -6
  163. package/src/plugins/setting/actionHandlers/setSystemSettingValues.ts +30 -30
  164. package/src/plugins/setting/models/SystemSettingGroupSetting.ts +57 -57
  165. package/src/plugins/setting/models/SystemSettingItem.ts +48 -48
  166. package/src/plugins/setting/models/SystemSettingItemSetting.ts +73 -73
  167. package/src/plugins/setting/models/UserSettingGroupSetting.ts +57 -57
  168. package/src/plugins/setting/models/UserSettingItem.ts +55 -55
  169. package/src/plugins/setting/models/UserSettingItemSetting.ts +73 -73
  170. package/src/plugins/setting/models/index.ts +8 -8
  171. package/src/plugins/setting/routes/getSystemSettingValues.ts +15 -15
  172. package/src/plugins/setting/routes/getUserSettingValues.ts +15 -15
  173. package/src/plugins/setting/routes/index.ts +5 -5
  174. package/src/plugins/setting/routes/setSystemSettingValues.ts +15 -15
  175. package/src/plugins/stateMachine/StateMachinePlugin.ts +196 -196
  176. package/src/plugins/stateMachine/StateMachinePluginTypes.ts +48 -48
  177. package/src/plugins/stateMachine/actionHandlers/index.ts +4 -4
  178. package/src/plugins/stateMachine/actionHandlers/sendStateMachineEvent.ts +54 -54
  179. package/src/plugins/stateMachine/models/StateMachine.ts +42 -42
  180. package/src/plugins/stateMachine/models/index.ts +3 -3
  181. package/src/plugins/stateMachine/routes/index.ts +3 -3
  182. package/src/plugins/stateMachine/routes/sendStateMachineEvent.ts +15 -15
  183. package/src/plugins/stateMachine/stateMachineHelper.ts +36 -36
  184. package/src/plugins/webhooks/WebhooksPlugin.ts +148 -148
  185. package/src/plugins/webhooks/pluginConfig.ts +75 -75
  186. package/src/polyfill.ts +5 -5
  187. package/src/proxy/mod.ts +38 -38
  188. package/src/proxy/types.ts +21 -21
  189. package/src/queryBuilder/index.ts +1 -1
  190. package/src/queryBuilder/queryBuilder.ts +755 -755
  191. package/src/server.ts +567 -556
  192. package/src/types/cron-job-types.ts +71 -71
  193. package/src/types.ts +884 -882
  194. package/src/utilities/accessControlUtility.ts +33 -33
  195. package/src/utilities/entityUtility.ts +18 -18
  196. package/src/utilities/errorUtility.ts +15 -15
  197. package/src/utilities/fsUtility.ts +137 -137
  198. package/src/utilities/httpUtility.ts +19 -19
  199. package/src/utilities/jwtUtility.ts +26 -26
  200. package/src/utilities/passwordUtility.ts +26 -26
  201. package/src/utilities/pathUtility.ts +14 -14
  202. package/src/utilities/timeUtility.ts +17 -17
  203. package/src/utilities/typeUtility.ts +15 -15
  204. package/tsconfig.json +19 -19
@@ -1,755 +1,755 @@
1
- import { find, isBoolean, isNil, isNull, isNumber, isString, isUndefined } from "lodash";
2
- import { RpdDataModel, RpdDataModelProperty, CreateEntityOptions, QuoteTableOptions, DatabaseQuery, IQueryBuilder } from "../types";
3
- import {
4
- CountRowOptions,
5
- DeleteRowOptions,
6
- FindRowLogicalFilterOptions,
7
- FindRowRelationalFilterOptions,
8
- FindRowSetFilterOptions,
9
- FindRowUnaryFilterOptions,
10
- FindRowOptions,
11
- RowFilterOptions,
12
- RowFilterRelationalOperators,
13
- UpdateRowOptions,
14
- ColumnSelectOptions,
15
- ColumnNameWithTableName,
16
- DataAccessPgColumnTypes,
17
- FindRowArrayFilterOptions,
18
- FindRowRangeFilterOptions,
19
- } from "~/dataAccess/dataAccessTypes";
20
- import { pgPropertyTypeColumnMap } from "~/dataAccess/columnTypeMapper";
21
-
22
- const objLeftQuoteChar = '"';
23
- const objRightQuoteChar = '"';
24
-
25
- const relationalOperatorsMap = new Map<RowFilterRelationalOperators, string>([
26
- ["eq", "="],
27
- ["ne", "<>"],
28
- ["gt", ">"],
29
- ["gte", ">="],
30
- ["lt", "<"],
31
- ["lte", "<="],
32
- ]);
33
-
34
- export interface BuildQueryContext {
35
- model: RpdDataModel;
36
- builder: QueryBuilder;
37
- params: any[];
38
- emitTableAlias: boolean;
39
- /**
40
- * emit parameter value to sql literal.
41
- */
42
- paramToLiteral: boolean;
43
- }
44
-
45
- export interface InitQueryBuilderOptions {
46
- dbDefaultSchema: string;
47
- }
48
-
49
- export default class QueryBuilder implements IQueryBuilder {
50
- #dbDefaultSchema: string;
51
-
52
- constructor(options: InitQueryBuilderOptions) {
53
- this.#dbDefaultSchema = options.dbDefaultSchema;
54
- }
55
-
56
- quoteTable(options: QuoteTableOptions) {
57
- const { schema, tableName } = options;
58
- if (schema) {
59
- return `${this.quoteObject(schema)}.${this.quoteObject(tableName)}`;
60
- } else if (this.#dbDefaultSchema) {
61
- return `${this.quoteObject(this.#dbDefaultSchema)}.${this.quoteObject(tableName)}`;
62
- } else {
63
- return this.quoteObject(tableName);
64
- }
65
- }
66
-
67
- quoteObject(name: string) {
68
- return `${objLeftQuoteChar}${name}${objRightQuoteChar}`;
69
- }
70
-
71
- quoteColumn(model: RpdDataModel, column: ColumnSelectOptions, emitTableAlias: boolean) {
72
- if (typeof column === "string") {
73
- if (emitTableAlias) {
74
- return `${objLeftQuoteChar}${model.tableName}${objRightQuoteChar}.${objLeftQuoteChar}${column}${objRightQuoteChar}`;
75
- } else {
76
- return `${objLeftQuoteChar}${column}${objRightQuoteChar}`;
77
- }
78
- } else {
79
- if (emitTableAlias && column.tableName) {
80
- return `${objLeftQuoteChar}${column.tableName}${objRightQuoteChar}.${objLeftQuoteChar}${column.name}${objRightQuoteChar}`;
81
- } else {
82
- return `${objLeftQuoteChar}${column.name}${objRightQuoteChar}`;
83
- }
84
- }
85
- }
86
-
87
- formatValueToSqlLiteral(value: any) {
88
- return formatValueToSqlLiteral(value);
89
- }
90
-
91
- select(model: RpdDataModel, options: FindRowOptions): DatabaseQuery {
92
- const ctx: BuildQueryContext = {
93
- model,
94
- builder: this,
95
- params: [],
96
- emitTableAlias: true,
97
- paramToLiteral: false,
98
- };
99
- let { fields: columns, filters, orderBy, pagination } = options;
100
- let command = "SELECT ";
101
- if (!columns || !columns.length) {
102
- command += `${this.quoteObject(model.tableName)}.* FROM `;
103
- } else {
104
- command += columns.map((column) => this.quoteColumn(ctx.model, column, ctx.emitTableAlias)).join(", ");
105
- command += " FROM ";
106
- }
107
-
108
- command += this.quoteTable(model);
109
-
110
- if (options.orderBy) {
111
- options.orderBy
112
- .filter((orderByItem) => orderByItem.relationField)
113
- .forEach((orderByItem) => {
114
- const { relationField } = orderByItem;
115
- const orderField = orderByItem.field as ColumnNameWithTableName;
116
- command += ` LEFT JOIN ${this.quoteTable({ schema: orderField.schema, tableName: orderField.tableName })} ON ${this.quoteObject(
117
- orderField.tableName,
118
- )}.id = ${this.quoteObject(relationField.tableName)}.${this.quoteObject(relationField.name)}`;
119
- });
120
- }
121
-
122
- if (filters && filters.length) {
123
- command += " WHERE ";
124
- command += buildFiltersQuery(ctx, filters);
125
- }
126
-
127
- if (orderBy && orderBy.length) {
128
- command += " ORDER BY ";
129
- command += orderBy
130
- .map((item) => {
131
- const quotedName = this.quoteColumn(ctx.model, item.field, ctx.emitTableAlias);
132
- return item.desc ? quotedName + " DESC" : quotedName;
133
- })
134
- .join(", ");
135
- }
136
-
137
- if (pagination) {
138
- command += " OFFSET ";
139
- ctx.params.push(pagination.offset);
140
- command += "$" + ctx.params.length;
141
-
142
- command += " LIMIT ";
143
- ctx.params.push(pagination.limit);
144
- command += "$" + ctx.params.length;
145
- }
146
-
147
- return {
148
- command,
149
- params: ctx.params,
150
- };
151
- }
152
-
153
- selectDerived(derivedModel: RpdDataModel, baseModel: RpdDataModel, options: FindRowOptions): DatabaseQuery {
154
- const ctx: BuildQueryContext = {
155
- model: derivedModel,
156
- builder: this,
157
- params: [],
158
- emitTableAlias: true,
159
- paramToLiteral: false,
160
- };
161
- let { fields: columns, filters, orderBy, pagination } = options;
162
- let command = "SELECT ";
163
- if (!columns || !columns.length) {
164
- command += `${this.quoteObject(derivedModel.tableName)}.* FROM `;
165
- } else {
166
- command += columns
167
- .map((column) => {
168
- return this.quoteColumn(derivedModel, column, ctx.emitTableAlias);
169
- })
170
- .join(", ");
171
- command += " FROM ";
172
- }
173
-
174
- command += `${this.quoteTable(derivedModel)} LEFT JOIN ${this.quoteTable(baseModel)} ON ${this.quoteObject(derivedModel.tableName)}.id = ${this.quoteObject(
175
- baseModel.tableName,
176
- )}.id`;
177
-
178
- if (options.orderBy) {
179
- options.orderBy
180
- .filter((orderByItem) => orderByItem.relationField)
181
- .forEach((orderByItem) => {
182
- const { relationField } = orderByItem;
183
- const orderField = orderByItem.field as ColumnNameWithTableName;
184
- command += ` LEFT JOIN ${this.quoteTable({ schema: orderField.schema, tableName: orderField.tableName })} ON ${this.quoteObject(
185
- orderField.tableName,
186
- )}.id = ${this.quoteObject(relationField.tableName)}.${this.quoteObject(relationField.name)}`;
187
- });
188
- }
189
-
190
- if (filters && filters.length) {
191
- command += " WHERE ";
192
- command += buildFiltersQuery(ctx, filters);
193
- }
194
-
195
- if (orderBy && orderBy.length) {
196
- command += " ORDER BY ";
197
- command += orderBy
198
- .map((item) => {
199
- const quotedName = this.quoteColumn(derivedModel, item.field, ctx.emitTableAlias);
200
- return item.desc ? quotedName + " DESC" : quotedName;
201
- })
202
- .join(", ");
203
- }
204
-
205
- if (pagination) {
206
- command += " OFFSET ";
207
- ctx.params.push(pagination.offset);
208
- command += "$" + ctx.params.length;
209
-
210
- command += " LIMIT ";
211
- ctx.params.push(pagination.limit);
212
- command += "$" + ctx.params.length;
213
- }
214
-
215
- return {
216
- command,
217
- params: ctx.params,
218
- };
219
- }
220
-
221
- count(model: RpdDataModel, options: CountRowOptions): DatabaseQuery {
222
- const ctx: BuildQueryContext = {
223
- model,
224
- builder: this,
225
- params: [],
226
- emitTableAlias: false,
227
- paramToLiteral: false,
228
- };
229
- let { filters } = options;
230
- let command = 'SELECT COUNT(*)::int as "count" FROM ';
231
-
232
- command += this.quoteTable(model);
233
-
234
- if (filters && filters.length) {
235
- command += " WHERE ";
236
- command += buildFiltersQuery(ctx, filters);
237
- }
238
-
239
- return {
240
- command,
241
- params: ctx.params,
242
- };
243
- }
244
-
245
- countDerived(derivedModel: RpdDataModel, baseModel: RpdDataModel, options: CountRowOptions): DatabaseQuery {
246
- const ctx: BuildQueryContext = {
247
- model: derivedModel,
248
- builder: this,
249
- params: [],
250
- emitTableAlias: true,
251
- paramToLiteral: false,
252
- };
253
- let { filters } = options;
254
- let command = 'SELECT COUNT(*)::int as "count" FROM ';
255
-
256
- command += `${this.quoteTable(derivedModel)} LEFT JOIN ${this.quoteTable(baseModel)} ON ${this.quoteObject(derivedModel.tableName)}.id = ${this.quoteObject(
257
- baseModel.tableName,
258
- )}.id`;
259
-
260
- if (filters && filters.length) {
261
- command += " WHERE ";
262
- command += buildFiltersQuery(ctx, filters);
263
- }
264
-
265
- return {
266
- command,
267
- params: ctx.params,
268
- };
269
- }
270
-
271
- insert(model: RpdDataModel, options: CreateEntityOptions): DatabaseQuery {
272
- const params: any[] = [];
273
- const ctx: BuildQueryContext = {
274
- model,
275
- builder: this,
276
- params,
277
- emitTableAlias: false,
278
- paramToLiteral: false,
279
- };
280
- const { entity } = options;
281
- let command = "INSERT INTO ";
282
-
283
- command += this.quoteTable(model);
284
-
285
- const columnNames: string[] = Object.keys(entity);
286
- let values = "";
287
- columnNames.forEach((columnName, index) => {
288
- if (index) {
289
- values += ", ";
290
- }
291
-
292
- let property: RpdDataModelProperty | null = null;
293
- if (model) {
294
- property = find(model.properties, (e: RpdDataModelProperty) => (e.columnName || e.code) === columnName);
295
- }
296
- const columnType: DataAccessPgColumnTypes | null = property ? pgPropertyTypeColumnMap[property.type] : null;
297
- if (columnType === "jsonb") {
298
- params.push(JSON.stringify(entity[columnName]));
299
- values += `$${params.length}::jsonb`;
300
- } else {
301
- params.push(entity[columnName]);
302
- values += `$${params.length}`;
303
- }
304
- });
305
-
306
- command += ` (${columnNames.map(this.quoteObject).join(", ")})`;
307
- command += ` VALUES (${values}) RETURNING *`;
308
-
309
- return {
310
- command,
311
- params: ctx.params,
312
- };
313
- }
314
-
315
- update(model: RpdDataModel, options: UpdateRowOptions): DatabaseQuery {
316
- const params: any[] = [];
317
- const ctx: BuildQueryContext = {
318
- model,
319
- builder: this,
320
- params,
321
- emitTableAlias: false,
322
- paramToLiteral: false,
323
- };
324
- let { entity, filters } = options;
325
- let command = "UPDATE ";
326
-
327
- command += this.quoteTable(model);
328
-
329
- command += " SET ";
330
- const columnNames: string[] = Object.keys(entity);
331
- columnNames.forEach((columnName, index) => {
332
- if (index) {
333
- command += ", ";
334
- }
335
-
336
- command += `${this.quoteObject(columnName)}=`;
337
-
338
- let property: RpdDataModelProperty | null = null;
339
- if (model) {
340
- property = find(model.properties, (e: RpdDataModelProperty) => (e.columnName || e.code) === columnName);
341
- }
342
- const columnType: DataAccessPgColumnTypes | null = property ? pgPropertyTypeColumnMap[property.type] : null;
343
- if (columnType === "jsonb") {
344
- params.push(JSON.stringify(entity[columnName]));
345
- command += `$${params.length}::jsonb`;
346
- } else {
347
- params.push(entity[columnName]);
348
- command += `$${params.length}`;
349
- }
350
- });
351
-
352
- if (filters && filters.length) {
353
- command += " WHERE ";
354
- command += buildFiltersQuery(ctx, filters);
355
- }
356
-
357
- command += " RETURNING *";
358
-
359
- return {
360
- command,
361
- params: ctx.params,
362
- };
363
- }
364
-
365
- delete(model: RpdDataModel, options: DeleteRowOptions): DatabaseQuery {
366
- const params: any[] = [];
367
- const ctx: BuildQueryContext = {
368
- model,
369
- builder: this,
370
- params,
371
- emitTableAlias: false,
372
- paramToLiteral: false,
373
- };
374
- let { filters } = options;
375
- let command = "DELETE FROM ";
376
-
377
- command += this.quoteTable(model);
378
-
379
- if (filters && filters.length) {
380
- command += " WHERE ";
381
- command += buildFiltersQuery(ctx, filters);
382
- }
383
-
384
- return {
385
- command,
386
- params: ctx.params,
387
- };
388
- }
389
-
390
- buildFiltersExpression(model: RpdDataModel, filters: RowFilterOptions[]) {
391
- const params: any[] = [];
392
- const ctx: BuildQueryContext = {
393
- model,
394
- builder: this,
395
- params,
396
- emitTableAlias: false,
397
- paramToLiteral: true,
398
- };
399
-
400
- return buildFiltersQuery(ctx, filters);
401
- }
402
- }
403
-
404
- export function buildFiltersQuery(ctx: BuildQueryContext, filters: RowFilterOptions[]) {
405
- return buildFilterQuery(0, ctx, {
406
- operator: "and",
407
- filters,
408
- });
409
- }
410
-
411
- function buildFilterQuery(level: number, ctx: BuildQueryContext, filter: RowFilterOptions): string {
412
- const { operator } = filter;
413
- if (operator === "eq" || operator === "ne" || operator === "gt" || operator === "gte" || operator === "lt" || operator === "lte") {
414
- return buildRelationalFilterQuery(ctx, filter);
415
- } else if (operator === "and" || operator === "or") {
416
- return buildLogicalFilterQuery(level, ctx, filter);
417
- } else if (operator === "null" || operator === "notNull") {
418
- return buildUnaryFilterQuery(ctx, filter);
419
- } else if (operator === "in" || operator === "notIn") {
420
- return buildInFilterQuery(ctx, filter);
421
- } else if (operator === "between" || operator === "range") {
422
- return buildRangeFilterQuery(ctx, filter);
423
- } else if (operator === "matches") {
424
- return buildMatchesFilterQuery(ctx, filter);
425
- } else if (operator === "matchesCS") {
426
- return buildMatchesCSFilterQuery(ctx, filter);
427
- } else if (operator === "contains") {
428
- return buildContainsFilterQuery(ctx, filter);
429
- } else if (operator === "containsCS") {
430
- return buildContainsCSFilterQuery(ctx, filter);
431
- } else if (operator === "notContains") {
432
- return buildNotContainsFilterQuery(ctx, filter);
433
- } else if (operator === "notContainsCS") {
434
- return buildNotContainsCSFilterQuery(ctx, filter);
435
- } else if (operator === "startsWith") {
436
- return buildStartsWithFilterQuery(ctx, filter);
437
- } else if (operator === "notStartsWith") {
438
- return buildNotStartsWithFilterQuery(ctx, filter);
439
- } else if (operator === "endsWith") {
440
- return buildEndsWithFilterQuery(ctx, filter);
441
- } else if (operator === "notEndsWith") {
442
- return buildNotEndsWithFilterQuery(ctx, filter);
443
- } else if (operator === "arrayContains") {
444
- return buildArrayContainsFilterQuery(ctx, filter);
445
- } else if (operator === "arrayOverlap") {
446
- return buildArrayOverlapFilterQuery(ctx, filter);
447
- } else {
448
- throw new Error(`Filter operator '${operator}' is not supported.`);
449
- }
450
- }
451
-
452
- function buildLogicalFilterQuery(level: number, ctx: BuildQueryContext, filter: FindRowLogicalFilterOptions) {
453
- let dbOperator;
454
- if (filter.operator === "and") {
455
- dbOperator = " AND ";
456
- } else {
457
- dbOperator = " OR ";
458
- }
459
-
460
- let command = filter.filters.map(buildFilterQuery.bind(null, level + 1, ctx)).join(dbOperator);
461
- if (level) {
462
- return `(${command})`;
463
- }
464
- return command;
465
- }
466
-
467
- function buildUnaryFilterQuery(ctx: BuildQueryContext, filter: FindRowUnaryFilterOptions) {
468
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
469
- if (filter.operator === "null") {
470
- command += " IS NULL";
471
- } else {
472
- command += " IS NOT NULL";
473
- }
474
- return command;
475
- }
476
-
477
- function buildInFilterQuery(ctx: BuildQueryContext, filter: FindRowSetFilterOptions) {
478
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
479
-
480
- if (filter.operator === "in") {
481
- command += " = ";
482
- } else {
483
- command += " <> ";
484
- }
485
-
486
- if (ctx.paramToLiteral) {
487
- // TODO: implement it
488
- } else {
489
- ctx.params.push(filter.value);
490
- if (filter.operator === "in") {
491
- command += `ANY($${ctx.params.length}::${filter.itemType || "int"}[])`;
492
- } else {
493
- command += `ALL($${ctx.params.length}::${filter.itemType || "int"}[])`;
494
- }
495
- }
496
-
497
- return command;
498
- }
499
-
500
- function buildRangeFilterQuery(ctx: BuildQueryContext, filter: FindRowRangeFilterOptions) {
501
- let command = "";
502
-
503
- if (filter.value.length != 2) {
504
- throw new Error(`Filter operator '${filter.operator}' need two values.`);
505
- }
506
-
507
- if (filter.operator === "between") {
508
- command += ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
509
-
510
- command += " BETWEEN ";
511
-
512
- ctx.params.push(filter.value[0]);
513
- command += `$${ctx.params.length}`;
514
-
515
- command += " AND ";
516
-
517
- ctx.params.push(filter.value[1]);
518
- command += `$${ctx.params.length}`;
519
- } else if (filter.operator === "range") {
520
- ctx.params.push(filter.value[0]);
521
- command += `(${ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias)} >= $${ctx.params.length}`;
522
-
523
- command += " AND ";
524
-
525
- ctx.params.push(filter.value[1]);
526
- command += `${ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias)} < $${ctx.params.length})`;
527
- } else {
528
- throw new Error(`Filter operator '${filter.operator}' is not supported.`);
529
- }
530
-
531
- return command;
532
- }
533
-
534
- function convertSearchTextToLikeParamValue(searchText: string) {
535
- if (isNil(searchText)) {
536
- return searchText;
537
- }
538
-
539
- let result = searchText.replace(/\*/g, "%");
540
- result = result.replace(/\?/g, "_");
541
- return result;
542
- }
543
-
544
- function buildMatchesFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
545
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
546
-
547
- command += " ILIKE ";
548
-
549
- if (ctx.paramToLiteral) {
550
- // TODO: implement it
551
- } else {
552
- ctx.params.push(`%${convertSearchTextToLikeParamValue(filter.value)}%`);
553
- command += "$" + ctx.params.length;
554
- }
555
-
556
- return command;
557
- }
558
-
559
- function buildMatchesCSFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
560
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
561
-
562
- command += " LIKE ";
563
-
564
- if (ctx.paramToLiteral) {
565
- // TODO: implement it
566
- } else {
567
- ctx.params.push(`%${convertSearchTextToLikeParamValue(filter.value)}%`);
568
- command += "$" + ctx.params.length;
569
- }
570
-
571
- return command;
572
- }
573
-
574
- function buildContainsFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
575
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
576
-
577
- command += " ILIKE ";
578
-
579
- if (ctx.paramToLiteral) {
580
- // TODO: implement it
581
- } else {
582
- ctx.params.push(`%${filter.value}%`);
583
- command += "$" + ctx.params.length;
584
- }
585
-
586
- return command;
587
- }
588
-
589
- function buildContainsCSFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
590
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
591
-
592
- command += " LIKE ";
593
-
594
- if (ctx.paramToLiteral) {
595
- // TODO: implement it
596
- } else {
597
- ctx.params.push(`%${filter.value}%`);
598
- command += "$" + ctx.params.length;
599
- }
600
-
601
- return command;
602
- }
603
-
604
- function buildNotContainsFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
605
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
606
-
607
- command += " NOT ILIKE ";
608
- if (ctx.paramToLiteral) {
609
- // TODO: implement it
610
- } else {
611
- ctx.params.push(`%${filter.value}%`);
612
- command += "$" + ctx.params.length;
613
- }
614
-
615
- return command;
616
- }
617
-
618
- function buildNotContainsCSFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
619
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
620
-
621
- command += " NOT LIKE ";
622
- if (ctx.paramToLiteral) {
623
- // TODO: implement it
624
- } else {
625
- ctx.params.push(`%${filter.value}%`);
626
- command += "$" + ctx.params.length;
627
- }
628
-
629
- return command;
630
- }
631
-
632
- function buildStartsWithFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
633
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
634
-
635
- command += " ILIKE ";
636
-
637
- if (ctx.paramToLiteral) {
638
- // TODO: implement it
639
- } else {
640
- ctx.params.push(`${filter.value}%`);
641
- command += "$" + ctx.params.length;
642
- }
643
-
644
- return command;
645
- }
646
-
647
- function buildNotStartsWithFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
648
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
649
-
650
- command += " NOT ILIKE ";
651
-
652
- if (ctx.paramToLiteral) {
653
- // TODO: implement it
654
- } else {
655
- ctx.params.push(`${filter.value}%`);
656
- command += "$" + ctx.params.length;
657
- }
658
-
659
- return command;
660
- }
661
-
662
- function buildEndsWithFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
663
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
664
-
665
- command += " ILIKE ";
666
-
667
- if (ctx.paramToLiteral) {
668
- // TODO: implement it
669
- } else {
670
- ctx.params.push(`%${filter.value}`);
671
- command += "$" + ctx.params.length;
672
- }
673
-
674
- return command;
675
- }
676
-
677
- function buildNotEndsWithFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
678
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
679
-
680
- command += " NOT ILIKE ";
681
-
682
- if (ctx.paramToLiteral) {
683
- // TODO: implement it
684
- } else {
685
- ctx.params.push(`%${filter.value}`);
686
- command += "$" + ctx.params.length;
687
- }
688
-
689
- return command;
690
- }
691
-
692
- function buildRelationalFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
693
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
694
-
695
- command += relationalOperatorsMap.get(filter.operator);
696
-
697
- if (ctx.paramToLiteral) {
698
- command += formatValueToSqlLiteral(filter.value);
699
- } else {
700
- ctx.params.push(filter.value);
701
- command += "$" + ctx.params.length;
702
- }
703
-
704
- return command;
705
- }
706
-
707
- function buildArrayContainsFilterQuery(ctx: BuildQueryContext, filter: FindRowArrayFilterOptions) {
708
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
709
-
710
- command += " @> ";
711
-
712
- if (ctx.paramToLiteral) {
713
- // TODO: implement it
714
- } else {
715
- ctx.params.push(filter.value);
716
- command += "$" + ctx.params.length;
717
- }
718
-
719
- return command;
720
- }
721
-
722
- function buildArrayOverlapFilterQuery(ctx: BuildQueryContext, filter: FindRowArrayFilterOptions) {
723
- let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
724
-
725
- command += " && ";
726
-
727
- if (ctx.paramToLiteral) {
728
- // TODO: implement it
729
- } else {
730
- ctx.params.push(filter.value);
731
- command += "$" + ctx.params.length;
732
- }
733
-
734
- return command;
735
- }
736
-
737
- function formatValueToSqlLiteral(value: any) {
738
- if (isNull(value) || isUndefined(value)) {
739
- return "null";
740
- }
741
-
742
- if (isString(value)) {
743
- return `'${value.replaceAll("'", "''")}'`;
744
- }
745
-
746
- if (isBoolean(value)) {
747
- return value ? "true" : "false";
748
- }
749
-
750
- if (isNumber(value)) {
751
- return value.toString();
752
- }
753
-
754
- return `'${value.toString().replaceAll("'", "''")}'`;
755
- }
1
+ import { find, isBoolean, isNil, isNull, isNumber, isString, isUndefined } from "lodash";
2
+ import { RpdDataModel, RpdDataModelProperty, CreateEntityOptions, QuoteTableOptions, DatabaseQuery, IQueryBuilder } from "../types";
3
+ import {
4
+ CountRowOptions,
5
+ DeleteRowOptions,
6
+ FindRowLogicalFilterOptions,
7
+ FindRowRelationalFilterOptions,
8
+ FindRowSetFilterOptions,
9
+ FindRowUnaryFilterOptions,
10
+ FindRowOptions,
11
+ RowFilterOptions,
12
+ RowFilterRelationalOperators,
13
+ UpdateRowOptions,
14
+ ColumnSelectOptions,
15
+ ColumnNameWithTableName,
16
+ DataAccessPgColumnTypes,
17
+ FindRowArrayFilterOptions,
18
+ FindRowRangeFilterOptions,
19
+ } from "~/dataAccess/dataAccessTypes";
20
+ import { pgPropertyTypeColumnMap } from "~/dataAccess/columnTypeMapper";
21
+
22
+ const objLeftQuoteChar = '"';
23
+ const objRightQuoteChar = '"';
24
+
25
+ const relationalOperatorsMap = new Map<RowFilterRelationalOperators, string>([
26
+ ["eq", "="],
27
+ ["ne", "<>"],
28
+ ["gt", ">"],
29
+ ["gte", ">="],
30
+ ["lt", "<"],
31
+ ["lte", "<="],
32
+ ]);
33
+
34
+ export interface BuildQueryContext {
35
+ model: RpdDataModel;
36
+ builder: QueryBuilder;
37
+ params: any[];
38
+ emitTableAlias: boolean;
39
+ /**
40
+ * emit parameter value to sql literal.
41
+ */
42
+ paramToLiteral: boolean;
43
+ }
44
+
45
+ export interface InitQueryBuilderOptions {
46
+ dbDefaultSchema: string;
47
+ }
48
+
49
+ export default class QueryBuilder implements IQueryBuilder {
50
+ #dbDefaultSchema: string;
51
+
52
+ constructor(options: InitQueryBuilderOptions) {
53
+ this.#dbDefaultSchema = options.dbDefaultSchema;
54
+ }
55
+
56
+ quoteTable(options: QuoteTableOptions) {
57
+ const { schema, tableName } = options;
58
+ if (schema) {
59
+ return `${this.quoteObject(schema)}.${this.quoteObject(tableName)}`;
60
+ } else if (this.#dbDefaultSchema) {
61
+ return `${this.quoteObject(this.#dbDefaultSchema)}.${this.quoteObject(tableName)}`;
62
+ } else {
63
+ return this.quoteObject(tableName);
64
+ }
65
+ }
66
+
67
+ quoteObject(name: string) {
68
+ return `${objLeftQuoteChar}${name}${objRightQuoteChar}`;
69
+ }
70
+
71
+ quoteColumn(model: RpdDataModel, column: ColumnSelectOptions, emitTableAlias: boolean) {
72
+ if (typeof column === "string") {
73
+ if (emitTableAlias) {
74
+ return `${objLeftQuoteChar}${model.tableName}${objRightQuoteChar}.${objLeftQuoteChar}${column}${objRightQuoteChar}`;
75
+ } else {
76
+ return `${objLeftQuoteChar}${column}${objRightQuoteChar}`;
77
+ }
78
+ } else {
79
+ if (emitTableAlias && column.tableName) {
80
+ return `${objLeftQuoteChar}${column.tableName}${objRightQuoteChar}.${objLeftQuoteChar}${column.name}${objRightQuoteChar}`;
81
+ } else {
82
+ return `${objLeftQuoteChar}${column.name}${objRightQuoteChar}`;
83
+ }
84
+ }
85
+ }
86
+
87
+ formatValueToSqlLiteral(value: any) {
88
+ return formatValueToSqlLiteral(value);
89
+ }
90
+
91
+ select(model: RpdDataModel, options: FindRowOptions): DatabaseQuery {
92
+ const ctx: BuildQueryContext = {
93
+ model,
94
+ builder: this,
95
+ params: [],
96
+ emitTableAlias: true,
97
+ paramToLiteral: false,
98
+ };
99
+ let { fields: columns, filters, orderBy, pagination } = options;
100
+ let command = "SELECT ";
101
+ if (!columns || !columns.length) {
102
+ command += `${this.quoteObject(model.tableName)}.* FROM `;
103
+ } else {
104
+ command += columns.map((column) => this.quoteColumn(ctx.model, column, ctx.emitTableAlias)).join(", ");
105
+ command += " FROM ";
106
+ }
107
+
108
+ command += this.quoteTable(model);
109
+
110
+ if (options.orderBy) {
111
+ options.orderBy
112
+ .filter((orderByItem) => orderByItem.relationField)
113
+ .forEach((orderByItem) => {
114
+ const { relationField } = orderByItem;
115
+ const orderField = orderByItem.field as ColumnNameWithTableName;
116
+ command += ` LEFT JOIN ${this.quoteTable({ schema: orderField.schema, tableName: orderField.tableName })} ON ${this.quoteObject(
117
+ orderField.tableName,
118
+ )}.id = ${this.quoteObject(relationField.tableName)}.${this.quoteObject(relationField.name)}`;
119
+ });
120
+ }
121
+
122
+ if (filters && filters.length) {
123
+ command += " WHERE ";
124
+ command += buildFiltersQuery(ctx, filters);
125
+ }
126
+
127
+ if (orderBy && orderBy.length) {
128
+ command += " ORDER BY ";
129
+ command += orderBy
130
+ .map((item) => {
131
+ const quotedName = this.quoteColumn(ctx.model, item.field, ctx.emitTableAlias);
132
+ return item.desc ? quotedName + " DESC" : quotedName;
133
+ })
134
+ .join(", ");
135
+ }
136
+
137
+ if (pagination) {
138
+ command += " OFFSET ";
139
+ ctx.params.push(pagination.offset);
140
+ command += "$" + ctx.params.length;
141
+
142
+ command += " LIMIT ";
143
+ ctx.params.push(pagination.limit);
144
+ command += "$" + ctx.params.length;
145
+ }
146
+
147
+ return {
148
+ command,
149
+ params: ctx.params,
150
+ };
151
+ }
152
+
153
+ selectDerived(derivedModel: RpdDataModel, baseModel: RpdDataModel, options: FindRowOptions): DatabaseQuery {
154
+ const ctx: BuildQueryContext = {
155
+ model: derivedModel,
156
+ builder: this,
157
+ params: [],
158
+ emitTableAlias: true,
159
+ paramToLiteral: false,
160
+ };
161
+ let { fields: columns, filters, orderBy, pagination } = options;
162
+ let command = "SELECT ";
163
+ if (!columns || !columns.length) {
164
+ command += `${this.quoteObject(derivedModel.tableName)}.* FROM `;
165
+ } else {
166
+ command += columns
167
+ .map((column) => {
168
+ return this.quoteColumn(derivedModel, column, ctx.emitTableAlias);
169
+ })
170
+ .join(", ");
171
+ command += " FROM ";
172
+ }
173
+
174
+ command += `${this.quoteTable(derivedModel)} LEFT JOIN ${this.quoteTable(baseModel)} ON ${this.quoteObject(derivedModel.tableName)}.id = ${this.quoteObject(
175
+ baseModel.tableName,
176
+ )}.id`;
177
+
178
+ if (options.orderBy) {
179
+ options.orderBy
180
+ .filter((orderByItem) => orderByItem.relationField)
181
+ .forEach((orderByItem) => {
182
+ const { relationField } = orderByItem;
183
+ const orderField = orderByItem.field as ColumnNameWithTableName;
184
+ command += ` LEFT JOIN ${this.quoteTable({ schema: orderField.schema, tableName: orderField.tableName })} ON ${this.quoteObject(
185
+ orderField.tableName,
186
+ )}.id = ${this.quoteObject(relationField.tableName)}.${this.quoteObject(relationField.name)}`;
187
+ });
188
+ }
189
+
190
+ if (filters && filters.length) {
191
+ command += " WHERE ";
192
+ command += buildFiltersQuery(ctx, filters);
193
+ }
194
+
195
+ if (orderBy && orderBy.length) {
196
+ command += " ORDER BY ";
197
+ command += orderBy
198
+ .map((item) => {
199
+ const quotedName = this.quoteColumn(derivedModel, item.field, ctx.emitTableAlias);
200
+ return item.desc ? quotedName + " DESC" : quotedName;
201
+ })
202
+ .join(", ");
203
+ }
204
+
205
+ if (pagination) {
206
+ command += " OFFSET ";
207
+ ctx.params.push(pagination.offset);
208
+ command += "$" + ctx.params.length;
209
+
210
+ command += " LIMIT ";
211
+ ctx.params.push(pagination.limit);
212
+ command += "$" + ctx.params.length;
213
+ }
214
+
215
+ return {
216
+ command,
217
+ params: ctx.params,
218
+ };
219
+ }
220
+
221
+ count(model: RpdDataModel, options: CountRowOptions): DatabaseQuery {
222
+ const ctx: BuildQueryContext = {
223
+ model,
224
+ builder: this,
225
+ params: [],
226
+ emitTableAlias: false,
227
+ paramToLiteral: false,
228
+ };
229
+ let { filters } = options;
230
+ let command = 'SELECT COUNT(*)::int as "count" FROM ';
231
+
232
+ command += this.quoteTable(model);
233
+
234
+ if (filters && filters.length) {
235
+ command += " WHERE ";
236
+ command += buildFiltersQuery(ctx, filters);
237
+ }
238
+
239
+ return {
240
+ command,
241
+ params: ctx.params,
242
+ };
243
+ }
244
+
245
+ countDerived(derivedModel: RpdDataModel, baseModel: RpdDataModel, options: CountRowOptions): DatabaseQuery {
246
+ const ctx: BuildQueryContext = {
247
+ model: derivedModel,
248
+ builder: this,
249
+ params: [],
250
+ emitTableAlias: true,
251
+ paramToLiteral: false,
252
+ };
253
+ let { filters } = options;
254
+ let command = 'SELECT COUNT(*)::int as "count" FROM ';
255
+
256
+ command += `${this.quoteTable(derivedModel)} LEFT JOIN ${this.quoteTable(baseModel)} ON ${this.quoteObject(derivedModel.tableName)}.id = ${this.quoteObject(
257
+ baseModel.tableName,
258
+ )}.id`;
259
+
260
+ if (filters && filters.length) {
261
+ command += " WHERE ";
262
+ command += buildFiltersQuery(ctx, filters);
263
+ }
264
+
265
+ return {
266
+ command,
267
+ params: ctx.params,
268
+ };
269
+ }
270
+
271
+ insert(model: RpdDataModel, options: CreateEntityOptions): DatabaseQuery {
272
+ const params: any[] = [];
273
+ const ctx: BuildQueryContext = {
274
+ model,
275
+ builder: this,
276
+ params,
277
+ emitTableAlias: false,
278
+ paramToLiteral: false,
279
+ };
280
+ const { entity } = options;
281
+ let command = "INSERT INTO ";
282
+
283
+ command += this.quoteTable(model);
284
+
285
+ const columnNames: string[] = Object.keys(entity);
286
+ let values = "";
287
+ columnNames.forEach((columnName, index) => {
288
+ if (index) {
289
+ values += ", ";
290
+ }
291
+
292
+ let property: RpdDataModelProperty | null = null;
293
+ if (model) {
294
+ property = find(model.properties, (e: RpdDataModelProperty) => (e.columnName || e.code) === columnName);
295
+ }
296
+ const columnType: DataAccessPgColumnTypes | null = property ? pgPropertyTypeColumnMap[property.type] : null;
297
+ if (columnType === "jsonb") {
298
+ params.push(JSON.stringify(entity[columnName]));
299
+ values += `$${params.length}::jsonb`;
300
+ } else {
301
+ params.push(entity[columnName]);
302
+ values += `$${params.length}`;
303
+ }
304
+ });
305
+
306
+ command += ` (${columnNames.map(this.quoteObject).join(", ")})`;
307
+ command += ` VALUES (${values}) RETURNING *`;
308
+
309
+ return {
310
+ command,
311
+ params: ctx.params,
312
+ };
313
+ }
314
+
315
+ update(model: RpdDataModel, options: UpdateRowOptions): DatabaseQuery {
316
+ const params: any[] = [];
317
+ const ctx: BuildQueryContext = {
318
+ model,
319
+ builder: this,
320
+ params,
321
+ emitTableAlias: false,
322
+ paramToLiteral: false,
323
+ };
324
+ let { entity, filters } = options;
325
+ let command = "UPDATE ";
326
+
327
+ command += this.quoteTable(model);
328
+
329
+ command += " SET ";
330
+ const columnNames: string[] = Object.keys(entity);
331
+ columnNames.forEach((columnName, index) => {
332
+ if (index) {
333
+ command += ", ";
334
+ }
335
+
336
+ command += `${this.quoteObject(columnName)}=`;
337
+
338
+ let property: RpdDataModelProperty | null = null;
339
+ if (model) {
340
+ property = find(model.properties, (e: RpdDataModelProperty) => (e.columnName || e.code) === columnName);
341
+ }
342
+ const columnType: DataAccessPgColumnTypes | null = property ? pgPropertyTypeColumnMap[property.type] : null;
343
+ if (columnType === "jsonb") {
344
+ params.push(JSON.stringify(entity[columnName]));
345
+ command += `$${params.length}::jsonb`;
346
+ } else {
347
+ params.push(entity[columnName]);
348
+ command += `$${params.length}`;
349
+ }
350
+ });
351
+
352
+ if (filters && filters.length) {
353
+ command += " WHERE ";
354
+ command += buildFiltersQuery(ctx, filters);
355
+ }
356
+
357
+ command += " RETURNING *";
358
+
359
+ return {
360
+ command,
361
+ params: ctx.params,
362
+ };
363
+ }
364
+
365
+ delete(model: RpdDataModel, options: DeleteRowOptions): DatabaseQuery {
366
+ const params: any[] = [];
367
+ const ctx: BuildQueryContext = {
368
+ model,
369
+ builder: this,
370
+ params,
371
+ emitTableAlias: false,
372
+ paramToLiteral: false,
373
+ };
374
+ let { filters } = options;
375
+ let command = "DELETE FROM ";
376
+
377
+ command += this.quoteTable(model);
378
+
379
+ if (filters && filters.length) {
380
+ command += " WHERE ";
381
+ command += buildFiltersQuery(ctx, filters);
382
+ }
383
+
384
+ return {
385
+ command,
386
+ params: ctx.params,
387
+ };
388
+ }
389
+
390
+ buildFiltersExpression(model: RpdDataModel, filters: RowFilterOptions[]) {
391
+ const params: any[] = [];
392
+ const ctx: BuildQueryContext = {
393
+ model,
394
+ builder: this,
395
+ params,
396
+ emitTableAlias: false,
397
+ paramToLiteral: true,
398
+ };
399
+
400
+ return buildFiltersQuery(ctx, filters);
401
+ }
402
+ }
403
+
404
+ export function buildFiltersQuery(ctx: BuildQueryContext, filters: RowFilterOptions[]) {
405
+ return buildFilterQuery(0, ctx, {
406
+ operator: "and",
407
+ filters,
408
+ });
409
+ }
410
+
411
+ function buildFilterQuery(level: number, ctx: BuildQueryContext, filter: RowFilterOptions): string {
412
+ const { operator } = filter;
413
+ if (operator === "eq" || operator === "ne" || operator === "gt" || operator === "gte" || operator === "lt" || operator === "lte") {
414
+ return buildRelationalFilterQuery(ctx, filter);
415
+ } else if (operator === "and" || operator === "or") {
416
+ return buildLogicalFilterQuery(level, ctx, filter);
417
+ } else if (operator === "null" || operator === "notNull") {
418
+ return buildUnaryFilterQuery(ctx, filter);
419
+ } else if (operator === "in" || operator === "notIn") {
420
+ return buildInFilterQuery(ctx, filter);
421
+ } else if (operator === "between" || operator === "range") {
422
+ return buildRangeFilterQuery(ctx, filter);
423
+ } else if (operator === "matches") {
424
+ return buildMatchesFilterQuery(ctx, filter);
425
+ } else if (operator === "matchesCS") {
426
+ return buildMatchesCSFilterQuery(ctx, filter);
427
+ } else if (operator === "contains") {
428
+ return buildContainsFilterQuery(ctx, filter);
429
+ } else if (operator === "containsCS") {
430
+ return buildContainsCSFilterQuery(ctx, filter);
431
+ } else if (operator === "notContains") {
432
+ return buildNotContainsFilterQuery(ctx, filter);
433
+ } else if (operator === "notContainsCS") {
434
+ return buildNotContainsCSFilterQuery(ctx, filter);
435
+ } else if (operator === "startsWith") {
436
+ return buildStartsWithFilterQuery(ctx, filter);
437
+ } else if (operator === "notStartsWith") {
438
+ return buildNotStartsWithFilterQuery(ctx, filter);
439
+ } else if (operator === "endsWith") {
440
+ return buildEndsWithFilterQuery(ctx, filter);
441
+ } else if (operator === "notEndsWith") {
442
+ return buildNotEndsWithFilterQuery(ctx, filter);
443
+ } else if (operator === "arrayContains") {
444
+ return buildArrayContainsFilterQuery(ctx, filter);
445
+ } else if (operator === "arrayOverlap") {
446
+ return buildArrayOverlapFilterQuery(ctx, filter);
447
+ } else {
448
+ throw new Error(`Filter operator '${operator}' is not supported.`);
449
+ }
450
+ }
451
+
452
+ function buildLogicalFilterQuery(level: number, ctx: BuildQueryContext, filter: FindRowLogicalFilterOptions) {
453
+ let dbOperator;
454
+ if (filter.operator === "and") {
455
+ dbOperator = " AND ";
456
+ } else {
457
+ dbOperator = " OR ";
458
+ }
459
+
460
+ let command = filter.filters.map(buildFilterQuery.bind(null, level + 1, ctx)).join(dbOperator);
461
+ if (level) {
462
+ return `(${command})`;
463
+ }
464
+ return command;
465
+ }
466
+
467
+ function buildUnaryFilterQuery(ctx: BuildQueryContext, filter: FindRowUnaryFilterOptions) {
468
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
469
+ if (filter.operator === "null") {
470
+ command += " IS NULL";
471
+ } else {
472
+ command += " IS NOT NULL";
473
+ }
474
+ return command;
475
+ }
476
+
477
+ function buildInFilterQuery(ctx: BuildQueryContext, filter: FindRowSetFilterOptions) {
478
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
479
+
480
+ if (filter.operator === "in") {
481
+ command += " = ";
482
+ } else {
483
+ command += " <> ";
484
+ }
485
+
486
+ if (ctx.paramToLiteral) {
487
+ // TODO: implement it
488
+ } else {
489
+ ctx.params.push(filter.value);
490
+ if (filter.operator === "in") {
491
+ command += `ANY($${ctx.params.length}::${filter.itemType || "int"}[])`;
492
+ } else {
493
+ command += `ALL($${ctx.params.length}::${filter.itemType || "int"}[])`;
494
+ }
495
+ }
496
+
497
+ return command;
498
+ }
499
+
500
+ function buildRangeFilterQuery(ctx: BuildQueryContext, filter: FindRowRangeFilterOptions) {
501
+ let command = "";
502
+
503
+ if (filter.value.length != 2) {
504
+ throw new Error(`Filter operator '${filter.operator}' need two values.`);
505
+ }
506
+
507
+ if (filter.operator === "between") {
508
+ command += ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
509
+
510
+ command += " BETWEEN ";
511
+
512
+ ctx.params.push(filter.value[0]);
513
+ command += `$${ctx.params.length}`;
514
+
515
+ command += " AND ";
516
+
517
+ ctx.params.push(filter.value[1]);
518
+ command += `$${ctx.params.length}`;
519
+ } else if (filter.operator === "range") {
520
+ ctx.params.push(filter.value[0]);
521
+ command += `(${ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias)} >= $${ctx.params.length}`;
522
+
523
+ command += " AND ";
524
+
525
+ ctx.params.push(filter.value[1]);
526
+ command += `${ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias)} < $${ctx.params.length})`;
527
+ } else {
528
+ throw new Error(`Filter operator '${filter.operator}' is not supported.`);
529
+ }
530
+
531
+ return command;
532
+ }
533
+
534
+ function convertSearchTextToLikeParamValue(searchText: string) {
535
+ if (isNil(searchText)) {
536
+ return searchText;
537
+ }
538
+
539
+ let result = searchText.replace(/\*/g, "%");
540
+ result = result.replace(/\?/g, "_");
541
+ return result;
542
+ }
543
+
544
+ function buildMatchesFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
545
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
546
+
547
+ command += " ILIKE ";
548
+
549
+ if (ctx.paramToLiteral) {
550
+ // TODO: implement it
551
+ } else {
552
+ ctx.params.push(`%${convertSearchTextToLikeParamValue(filter.value)}%`);
553
+ command += "$" + ctx.params.length;
554
+ }
555
+
556
+ return command;
557
+ }
558
+
559
+ function buildMatchesCSFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
560
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
561
+
562
+ command += " LIKE ";
563
+
564
+ if (ctx.paramToLiteral) {
565
+ // TODO: implement it
566
+ } else {
567
+ ctx.params.push(`%${convertSearchTextToLikeParamValue(filter.value)}%`);
568
+ command += "$" + ctx.params.length;
569
+ }
570
+
571
+ return command;
572
+ }
573
+
574
+ function buildContainsFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
575
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
576
+
577
+ command += " ILIKE ";
578
+
579
+ if (ctx.paramToLiteral) {
580
+ // TODO: implement it
581
+ } else {
582
+ ctx.params.push(`%${filter.value}%`);
583
+ command += "$" + ctx.params.length;
584
+ }
585
+
586
+ return command;
587
+ }
588
+
589
+ function buildContainsCSFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
590
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
591
+
592
+ command += " LIKE ";
593
+
594
+ if (ctx.paramToLiteral) {
595
+ // TODO: implement it
596
+ } else {
597
+ ctx.params.push(`%${filter.value}%`);
598
+ command += "$" + ctx.params.length;
599
+ }
600
+
601
+ return command;
602
+ }
603
+
604
+ function buildNotContainsFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
605
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
606
+
607
+ command += " NOT ILIKE ";
608
+ if (ctx.paramToLiteral) {
609
+ // TODO: implement it
610
+ } else {
611
+ ctx.params.push(`%${filter.value}%`);
612
+ command += "$" + ctx.params.length;
613
+ }
614
+
615
+ return command;
616
+ }
617
+
618
+ function buildNotContainsCSFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
619
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
620
+
621
+ command += " NOT LIKE ";
622
+ if (ctx.paramToLiteral) {
623
+ // TODO: implement it
624
+ } else {
625
+ ctx.params.push(`%${filter.value}%`);
626
+ command += "$" + ctx.params.length;
627
+ }
628
+
629
+ return command;
630
+ }
631
+
632
+ function buildStartsWithFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
633
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
634
+
635
+ command += " ILIKE ";
636
+
637
+ if (ctx.paramToLiteral) {
638
+ // TODO: implement it
639
+ } else {
640
+ ctx.params.push(`${filter.value}%`);
641
+ command += "$" + ctx.params.length;
642
+ }
643
+
644
+ return command;
645
+ }
646
+
647
+ function buildNotStartsWithFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
648
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
649
+
650
+ command += " NOT ILIKE ";
651
+
652
+ if (ctx.paramToLiteral) {
653
+ // TODO: implement it
654
+ } else {
655
+ ctx.params.push(`${filter.value}%`);
656
+ command += "$" + ctx.params.length;
657
+ }
658
+
659
+ return command;
660
+ }
661
+
662
+ function buildEndsWithFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
663
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
664
+
665
+ command += " ILIKE ";
666
+
667
+ if (ctx.paramToLiteral) {
668
+ // TODO: implement it
669
+ } else {
670
+ ctx.params.push(`%${filter.value}`);
671
+ command += "$" + ctx.params.length;
672
+ }
673
+
674
+ return command;
675
+ }
676
+
677
+ function buildNotEndsWithFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
678
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
679
+
680
+ command += " NOT ILIKE ";
681
+
682
+ if (ctx.paramToLiteral) {
683
+ // TODO: implement it
684
+ } else {
685
+ ctx.params.push(`%${filter.value}`);
686
+ command += "$" + ctx.params.length;
687
+ }
688
+
689
+ return command;
690
+ }
691
+
692
+ function buildRelationalFilterQuery(ctx: BuildQueryContext, filter: FindRowRelationalFilterOptions) {
693
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
694
+
695
+ command += relationalOperatorsMap.get(filter.operator);
696
+
697
+ if (ctx.paramToLiteral) {
698
+ command += formatValueToSqlLiteral(filter.value);
699
+ } else {
700
+ ctx.params.push(filter.value);
701
+ command += "$" + ctx.params.length;
702
+ }
703
+
704
+ return command;
705
+ }
706
+
707
+ function buildArrayContainsFilterQuery(ctx: BuildQueryContext, filter: FindRowArrayFilterOptions) {
708
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
709
+
710
+ command += " @> ";
711
+
712
+ if (ctx.paramToLiteral) {
713
+ // TODO: implement it
714
+ } else {
715
+ ctx.params.push(filter.value);
716
+ command += "$" + ctx.params.length;
717
+ }
718
+
719
+ return command;
720
+ }
721
+
722
+ function buildArrayOverlapFilterQuery(ctx: BuildQueryContext, filter: FindRowArrayFilterOptions) {
723
+ let command = ctx.builder.quoteColumn(ctx.model, filter.field, ctx.emitTableAlias);
724
+
725
+ command += " && ";
726
+
727
+ if (ctx.paramToLiteral) {
728
+ // TODO: implement it
729
+ } else {
730
+ ctx.params.push(filter.value);
731
+ command += "$" + ctx.params.length;
732
+ }
733
+
734
+ return command;
735
+ }
736
+
737
+ function formatValueToSqlLiteral(value: any) {
738
+ if (isNull(value) || isUndefined(value)) {
739
+ return "null";
740
+ }
741
+
742
+ if (isString(value)) {
743
+ return `'${value.replaceAll("'", "''")}'`;
744
+ }
745
+
746
+ if (isBoolean(value)) {
747
+ return value ? "true" : "false";
748
+ }
749
+
750
+ if (isNumber(value)) {
751
+ return value.toString();
752
+ }
753
+
754
+ return `'${value.toString().replaceAll("'", "''")}'`;
755
+ }