@restforgejs/platform 4.3.2 → 4.3.4

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 (176) hide show
  1. package/build-info.json +2 -2
  2. package/cli/consumer-deploy.js +1 -1
  3. package/cli/consumer.js +1 -1
  4. package/generators/cli/dashboard/create.js +0 -1
  5. package/generators/cli/endpoint/create.js +1 -1
  6. package/generators/lib/generators/model-generator.js +1 -1
  7. package/generators/lib/payload/endpoint-schema-validator.js +11 -1
  8. package/generators/lib/payload/payload-runner.js +3 -3
  9. package/generators/lib/payload/schema-diff.js +186 -3
  10. package/generators/lib/templates/dashboard-catalog.js +1 -1
  11. package/generators/lib/templates/db-connection-env.js +1 -1
  12. package/generators/lib/templates/dbschema-catalog.js +1 -1
  13. package/generators/lib/templates/field-validation-catalog.js +1 -1
  14. package/generators/lib/templates/mysql-template.js +1 -1
  15. package/generators/lib/templates/oracle-template.js +1 -1
  16. package/generators/lib/templates/postgres-template.js +1 -1
  17. package/generators/lib/templates/query-declarative-catalog.js +1 -1
  18. package/generators/lib/templates/sqlite-template.js +1 -1
  19. package/generators/lib/utils/conflict-checker.js +9 -57
  20. package/generators/lib/utils/database-introspector.js +84 -0
  21. package/generators/lib/utils/file-utils.js +0 -159
  22. package/generators/lib/utils/payload-processor.js +21 -112
  23. package/integrity-manifest.json +18 -18
  24. package/package.json +1 -1
  25. package/scripts/verify-integrity.js +1 -1
  26. package/server.js +1 -1
  27. package/src/components/handlers/adjust_handler.js +1 -1
  28. package/src/components/handlers/audit_handler.js +1 -1
  29. package/src/components/handlers/delete_handler.js +1 -1
  30. package/src/components/handlers/export_handler.js +1 -1
  31. package/src/components/handlers/import_handler.js +1 -1
  32. package/src/components/handlers/insert_handler.js +1 -1
  33. package/src/components/handlers/update_handler.js +1 -1
  34. package/src/components/handlers/upload_handler.js +1 -1
  35. package/src/components/handlers/workflow_handler.js +1 -1
  36. package/src/components/integrations/webhook.js +1 -1
  37. package/src/consumers/baseConsumer.js +1 -1
  38. package/src/consumers/declarativeMapper.js +1 -1
  39. package/src/consumers/handlers/apiHandler.js +1 -1
  40. package/src/consumers/handlers/consoleHandler.js +1 -1
  41. package/src/consumers/handlers/databaseHandler.js +1 -1
  42. package/src/consumers/handlers/index.js +1 -1
  43. package/src/consumers/handlers/kafkaHandler.js +1 -1
  44. package/src/consumers/index.js +1 -1
  45. package/src/consumers/messageTransformer.js +1 -1
  46. package/src/consumers/validator.js +1 -1
  47. package/src/core/db/dialect/base-dialect.js +1 -1
  48. package/src/core/db/dialect/index.js +1 -1
  49. package/src/core/db/dialect/mysql-dialect.js +1 -1
  50. package/src/core/db/dialect/oracle-dialect.js +1 -1
  51. package/src/core/db/dialect/postgres-dialect.js +1 -1
  52. package/src/core/db/dialect/sqlite-dialect.js +1 -1
  53. package/src/core/db/flatten-helper.js +1 -1
  54. package/src/core/db/query-builder-error.js +1 -1
  55. package/src/core/db/query-builder.js +1 -1
  56. package/src/core/db/relation-helper.js +1 -1
  57. package/src/core/handlers/delete_handler.js +1 -1
  58. package/src/core/handlers/insert_handler.js +1 -1
  59. package/src/core/handlers/update_handler.js +1 -1
  60. package/src/core/models/base-model.js +1 -1
  61. package/src/core/utils/cache-manager.js +1 -1
  62. package/src/core/utils/component-engine.js +1 -1
  63. package/src/core/utils/context-builder.js +1 -1
  64. package/src/core/utils/datetime-formatter.js +1 -1
  65. package/src/core/utils/datetime-parser.js +1 -1
  66. package/src/core/utils/db.js +1 -1
  67. package/src/core/utils/logger.js +1 -1
  68. package/src/core/utils/payload-loader.js +1 -1
  69. package/src/core/utils/security-checks.js +1 -1
  70. package/src/middleware/body-options.js +1 -1
  71. package/src/middleware/cors.js +1 -1
  72. package/src/middleware/idempotency.js +1 -1
  73. package/src/middleware/rate-limiter.js +1 -1
  74. package/src/middleware/request-logger.js +1 -1
  75. package/src/middleware/security-headers.js +1 -1
  76. package/src/models/base-model-mysql.js +1 -1
  77. package/src/models/base-model-oracle.js +1 -1
  78. package/src/models/base-model-sqlite.js +1 -1
  79. package/src/models/base-model.js +1 -1
  80. package/src/pro/caching/redis-client.js +1 -1
  81. package/src/pro/caching/redis-helper.js +1 -1
  82. package/src/pro/consumers/baseConsumer.js +1 -1
  83. package/src/pro/consumers/declarativeMapper.js +1 -1
  84. package/src/pro/consumers/handlers/apiHandler.js +1 -1
  85. package/src/pro/consumers/handlers/consoleHandler.js +1 -1
  86. package/src/pro/consumers/handlers/databaseHandler.js +1 -1
  87. package/src/pro/consumers/handlers/index.js +1 -1
  88. package/src/pro/consumers/handlers/kafkaHandler.js +1 -1
  89. package/src/pro/consumers/index.js +1 -1
  90. package/src/pro/consumers/messageTransformer.js +1 -1
  91. package/src/pro/consumers/validator.js +1 -1
  92. package/src/pro/database/base-model-mysql.js +1 -1
  93. package/src/pro/database/base-model-oracle.js +1 -1
  94. package/src/pro/database/base-model-sqlite.js +1 -1
  95. package/src/pro/database/db-mysql.js +1 -1
  96. package/src/pro/database/db-oracle.js +1 -1
  97. package/src/pro/database/db-sqlite.js +1 -1
  98. package/src/pro/excel/excel-generator.js +1 -1
  99. package/src/pro/excel/excel-parser.js +1 -1
  100. package/src/pro/excel/export-service.js +1 -1
  101. package/src/pro/excel/export_handler.js +1 -1
  102. package/src/pro/excel/import-service.js +1 -1
  103. package/src/pro/excel/import-validator.js +1 -1
  104. package/src/pro/excel/import_handler.js +1 -1
  105. package/src/pro/excel/upsert-builder.js +1 -1
  106. package/src/pro/idgen/idgen-routes.js +1 -1
  107. package/src/pro/integrations/lookup-resolver.js +1 -1
  108. package/src/pro/integrations/upload-handler-v2.js +1 -1
  109. package/src/pro/integrations/upload-handler.js +1 -1
  110. package/src/pro/integrations/webhook.js +1 -1
  111. package/src/pro/locking/lock-routes.js +1 -1
  112. package/src/pro/locking/resource-lock-manager.js +1 -1
  113. package/src/pro/messaging/kafkaConsumerService.js +1 -1
  114. package/src/pro/messaging/kafkaService.js +1 -1
  115. package/src/pro/messaging/messagehubService.js +1 -1
  116. package/src/pro/messaging/rabbitmqService.js +1 -1
  117. package/src/pro/scheduler/job-manager.js +1 -1
  118. package/src/pro/scheduler/job-routes.js +1 -1
  119. package/src/pro/scheduler/job-validator.js +1 -1
  120. package/src/pro/storage/base-storage-provider.js +1 -1
  121. package/src/pro/storage/file-metadata-helper.js +1 -1
  122. package/src/pro/storage/index.js +1 -1
  123. package/src/pro/storage/local-storage-provider.js +1 -1
  124. package/src/pro/storage/s3-storage-provider.js +1 -1
  125. package/src/pro/storage/upload-cleanup-job.js +1 -1
  126. package/src/pro/storage/upload-cleanup-scheduler.js +1 -1
  127. package/src/pro/storage/upload-pending-tracker.js +1 -1
  128. package/src/pro/websocket/broadcast-helper.js +1 -1
  129. package/src/pro/websocket/index.js +1 -1
  130. package/src/pro/websocket/livesync-server.js +1 -1
  131. package/src/pro/websocket/ws-broadcaster.js +1 -1
  132. package/src/services/export-service.js +1 -1
  133. package/src/services/import-service.js +1 -1
  134. package/src/services/kafkaConsumerService.js +1 -1
  135. package/src/services/kafkaService.js +1 -1
  136. package/src/services/messagehubService.js +1 -1
  137. package/src/services/rabbitmqService.js +1 -1
  138. package/src/utils/cache-invalidation-registry.js +1 -1
  139. package/src/utils/cache-manager.js +1 -1
  140. package/src/utils/component-engine.js +1 -1
  141. package/src/utils/config-extractor.js +1 -1
  142. package/src/utils/consumerLogger.js +1 -1
  143. package/src/utils/context-builder.js +1 -1
  144. package/src/utils/dashboard-helpers.js +1 -1
  145. package/src/utils/dateHelper.js +1 -1
  146. package/src/utils/datetime-formatter.js +1 -1
  147. package/src/utils/datetime-parser.js +1 -1
  148. package/src/utils/db-bootstrap.js +1 -1
  149. package/src/utils/db-mysql.js +1 -1
  150. package/src/utils/db-oracle.js +1 -1
  151. package/src/utils/db-sqlite.js +1 -1
  152. package/src/utils/db.js +1 -1
  153. package/src/utils/demo-generator.js +1 -1
  154. package/src/utils/excel-generator.js +1 -1
  155. package/src/utils/excel-parser.js +1 -1
  156. package/src/utils/file-watcher.js +1 -1
  157. package/src/utils/id-generator.js +1 -1
  158. package/src/utils/idempotency-manager.js +1 -1
  159. package/src/utils/import-validator.js +1 -1
  160. package/src/utils/license-client.js +1 -1
  161. package/src/utils/lock-manager.js +1 -1
  162. package/src/utils/logger.js +1 -1
  163. package/src/utils/lookup-resolver.js +1 -1
  164. package/src/utils/payload-loader.js +1 -1
  165. package/src/utils/processor-response.js +1 -1
  166. package/src/utils/rabbitmq.js +1 -1
  167. package/src/utils/redis-client.js +1 -1
  168. package/src/utils/redis-helper.js +1 -1
  169. package/src/utils/request-scope.js +1 -1
  170. package/src/utils/security-checks.js +1 -1
  171. package/src/utils/service-resolver.js +1 -1
  172. package/src/utils/shutdown-coordinator.js +1 -1
  173. package/src/utils/trusted-keys.js +1 -1
  174. package/src/utils/upload-handler.js +1 -1
  175. package/src/utils/upsert-builder.js +1 -1
  176. package/src/utils/workflow-hook-executor.js +1 -1
@@ -34,7 +34,7 @@ class ModelGenerator {
34
34
  return false;
35
35
  }
36
36
 
37
- // Note: SQL file copying sudah ditangani di PayloadProcessor.copyQueryFilesForPreserveMode
37
+ // Note: SQL file references di-expand inline ke output model oleh PayloadProcessor.processFileReferences
38
38
 
39
39
  // Generate model content
40
40
  const modelContent = this.generateModelContent(moduleName, endpointName, processedPayload, databaseType);
@@ -20,6 +20,7 @@
20
20
  * @module lib/payload/endpoint-schema-validator
21
21
  */
22
22
 
23
+ const path = require('path');
23
24
  const { DatabaseIntrospector } = require('../utils/database-introspector');
24
25
  const { resolveConfig, printDefaultConfigWarning } = require('../utils/config-resolver');
25
26
  const { compareSchemaStrict, formatDriftReport } = require('./schema-diff');
@@ -45,6 +46,10 @@ function createExitError(message, exitCode) {
45
46
  * @param {Object} options
46
47
  * @param {Object} options.payload - Processed payload object (harus punya tableName)
47
48
  * @param {string} options.payloadFileName - Nama file payload untuk pesan error
49
+ * @param {string} [options.payloadFilePath] - Absolute path ke payload file, untuk
50
+ * resolve `file:` reference di viewQuery/datatablesQuery/exportQuery. Jika tidak
51
+ * di-set, JOIN field validation tetap jalan untuk inline SQL tapi `file:` reference
52
+ * akan masuk warning.
48
53
  * @param {string} options.configArg - Nilai flag --config dari CLI
49
54
  * @param {boolean} options.skipSchemaCheck - true bila --skip-schema-check di-set
50
55
  * @param {string} [options.workingDir] - cwd, default process.cwd()
@@ -60,6 +65,7 @@ async function validateEndpointSchema(options) {
60
65
  const {
61
66
  payload,
62
67
  payloadFileName,
68
+ payloadFilePath,
63
69
  configArg,
64
70
  skipSchemaCheck,
65
71
  workingDir,
@@ -117,9 +123,13 @@ async function validateEndpointSchema(options) {
117
123
  );
118
124
  }
119
125
 
126
+ const payloadDir = payloadFilePath
127
+ ? path.dirname(payloadFilePath)
128
+ : path.join(workingDir || process.cwd(), 'payload');
129
+
120
130
  let comparison;
121
131
  try {
122
- comparison = await compareSchemaStrict(payload, db);
132
+ comparison = await compareSchemaStrict(payload, db, { payloadDir });
123
133
  } finally {
124
134
  try { await db.close(); } catch (_e) { /* ignore */ }
125
135
  }
@@ -809,7 +809,7 @@ class SchemaValidator {
809
809
  let ok = 0, drift = 0, error = 0;
810
810
 
811
811
  for (const { fileName, payload } of payloads) {
812
- const comparison = await compareSchemaStrict(payload, this.db);
812
+ const comparison = await compareSchemaStrict(payload, this.db, { payloadDir: this.outputDir });
813
813
  results.push({ fileName, comparison });
814
814
  this.printComparisonResult(comparison, fileName, false);
815
815
 
@@ -880,7 +880,7 @@ class SchemaValidator {
880
880
  console.log('-'.repeat(60));
881
881
  console.log();
882
882
 
883
- const comparison = await compareSchemaStrict(payload, this.db);
883
+ const comparison = await compareSchemaStrict(payload, this.db, { payloadDir: this.outputDir });
884
884
  results.push({ fileName, comparison });
885
885
  this.printComparisonResult(comparison, fileName, true);
886
886
  console.log();
@@ -987,7 +987,7 @@ class SchemaValidator {
987
987
  let synced = 0, skipped = 0, errorCount = 0;
988
988
 
989
989
  for (const { filePath, fileName, payload } of payloads) {
990
- const comparison = await compareSchemaStrict(payload, this.db);
990
+ const comparison = await compareSchemaStrict(payload, this.db, { payloadDir: this.outputDir });
991
991
 
992
992
  if (comparison.status === 'error') {
993
993
  console.log(` [ERROR] ${fileName} - ${comparison.summary}`);
@@ -16,9 +16,18 @@
16
16
  * *required* tapi tidak ada di database secara eksplisit, sehingga error
17
17
  * runtime "column does not exist" dapat di-detect saat generate-time.
18
18
  *
19
+ * Query-source-awareness:
20
+ * `fieldName` dapat berisi kolom hasil JOIN (mis. `category_name` dari
21
+ * `viewQuery`/`datatablesQuery`/`exportQuery`/`viewName`) sesuai spec
22
+ * `catalogs/rdf/data-source.md`. Validator meng-compute UNION columns dari
23
+ * seluruh query sources sebelum drift check, sehingga JOIN field tidak
24
+ * di-flag sebagai drift `[-]`.
25
+ *
19
26
  * @module lib/payload/schema-diff
20
27
  */
21
28
 
29
+ const fs = require('fs');
30
+ const path = require('path');
22
31
  const { resolveEffectiveFieldList, resolveAuditColumnNames, DEFAULT_AUDIT_COLUMNS } = require('../utils/audit-columns');
23
32
 
24
33
  const DEFAULT_EXCLUDED_COLUMNS = DEFAULT_AUDIT_COLUMNS;
@@ -115,7 +124,7 @@ function normalizeDatabaseType(dataType, udtName, col) {
115
124
  * summary: string
116
125
  * }>}
117
126
  */
118
- async function compareSchemaStrict(payload, db) {
127
+ async function compareSchemaStrict(payload, db, options = {}) {
119
128
  const tableName = payload.tableName;
120
129
  const result = {
121
130
  tableName,
@@ -124,6 +133,7 @@ async function compareSchemaStrict(payload, db) {
124
133
  added: [],
125
134
  typeChanges: [],
126
135
  auditMissing: [],
136
+ querySourceWarnings: [],
127
137
  summary: ''
128
138
  };
129
139
 
@@ -141,11 +151,21 @@ async function compareSchemaStrict(payload, db) {
141
151
  dbColumnMap[col.column_name] = col;
142
152
  }
143
153
 
154
+ // Resolve UNION column dari query sources (viewName/viewQuery/datatablesQuery/exportQuery).
155
+ // JOIN field di fieldName (mis. category_name dari viewQuery) valid kalau ada di salah
156
+ // satu output query. Sesuai catalogs/rdf/data-source.md.
157
+ const querySourceResult = await resolveQuerySourceColumns(payload, db, options);
158
+ const validColumnSet = new Set(dbColumnSet);
159
+ for (const col of querySourceResult.columns) {
160
+ validColumnSet.add(col);
161
+ }
162
+ result.querySourceWarnings = querySourceResult.warnings;
163
+
144
164
  const { explicit, audit } = resolveEffectiveFieldList(payload);
145
165
 
146
166
  const explicitSet = new Set(explicit);
147
167
  for (const field of explicit) {
148
- if (!dbColumnSet.has(field)) {
168
+ if (!validColumnSet.has(field)) {
149
169
  result.removed.push({ column: field, source: 'payload' });
150
170
  }
151
171
  }
@@ -254,6 +274,14 @@ function formatDriftReport(comparison, options = {}) {
254
274
  lines.push(' (required by auditColumns=true, not in database)');
255
275
  }
256
276
 
277
+ if (Array.isArray(comparison.querySourceWarnings) && comparison.querySourceWarnings.length > 0) {
278
+ lines.push('');
279
+ lines.push(' Query source warnings (column resolution may be incomplete):');
280
+ for (const w of comparison.querySourceWarnings) {
281
+ lines.push(` [!] ${w.source}: ${w.message}`);
282
+ }
283
+ }
284
+
257
285
  lines.push('');
258
286
  lines.push(' Resolution:');
259
287
  lines.push(` 1. Run: npx restforge payload sync --table=${tableName} --config=<ENV>`);
@@ -268,10 +296,165 @@ function formatDriftReport(comparison, options = {}) {
268
296
  return lines;
269
297
  }
270
298
 
299
+ /**
300
+ * Resolve UNION column dari query sources (viewName, viewQuery, datatablesQuery,
301
+ * exportQuery) untuk dipakai sebagai expanded valid column set saat drift check.
302
+ *
303
+ * `detailQuery` (di masterDetail.detailConfig) sengaja TIDAK di-resolve di scope
304
+ * root karena master-detail punya `fieldName` terpisah yang di-validate di scope
305
+ * detail sendiri.
306
+ *
307
+ * Behavior tolerant: query source yang gagal di-describe (SQL syntax error,
308
+ * file tidak ditemukan, dll) dicatat sebagai warning tapi tidak abort. Drift
309
+ * check tetap lanjut dengan UNION dari sources yang berhasil. Trade-off:
310
+ * false-positive drift mungkin terjadi kalau query gagal, tapi user dapat
311
+ * warning yang explicit.
312
+ *
313
+ * @param {Object} payload
314
+ * @param {Object} db - DatabaseIntrospector terkoneksi
315
+ * @param {Object} options
316
+ * @param {string} [options.payloadDir] - Folder payload untuk resolve file: reference
317
+ * @returns {Promise<{columns: string[], warnings: Array<{source: string, message: string}>}>}
318
+ */
319
+ async function resolveQuerySourceColumns(payload, db, options = {}) {
320
+ const columns = new Set();
321
+ const warnings = [];
322
+ const payloadDir = options.payloadDir || null;
323
+ const tableName = payload.tableName;
324
+
325
+ // 1. viewName: introspect sebagai table (information_schema.columns covers VIEW)
326
+ if (typeof payload.viewName === 'string' && payload.viewName.trim().length > 0) {
327
+ try {
328
+ const viewCols = await db.getDetailedColumnInfo(payload.viewName.trim());
329
+ if (Array.isArray(viewCols) && viewCols.length > 0) {
330
+ for (const c of viewCols) columns.add(c.column_name);
331
+ } else {
332
+ warnings.push({
333
+ source: `viewName=${payload.viewName}`,
334
+ message: 'View not found or has no columns'
335
+ });
336
+ }
337
+ } catch (err) {
338
+ warnings.push({
339
+ source: `viewName=${payload.viewName}`,
340
+ message: err.message || String(err)
341
+ });
342
+ }
343
+ }
344
+
345
+ // 2. SQL query sources
346
+ const sqlSources = [
347
+ { key: 'viewQuery', value: payload.viewQuery },
348
+ { key: 'datatablesQuery', value: payload.datatablesQuery },
349
+ { key: 'exportQuery', value: payload.exportQuery }
350
+ ];
351
+
352
+ for (const { key, value } of sqlSources) {
353
+ if (!value || typeof value !== 'string') continue;
354
+
355
+ const resolved = resolveSqlSource(value, payloadDir, key);
356
+ if (!resolved.ok) {
357
+ warnings.push({ source: key, message: resolved.error });
358
+ continue;
359
+ }
360
+
361
+ const sql = substituteQueryPlaceholders(resolved.sql, tableName);
362
+ if (typeof db.describeQueryColumns !== 'function') {
363
+ // Backward compat: introspector versi lama tanpa describeQueryColumns.
364
+ // Skip silently agar tidak break caller existing.
365
+ continue;
366
+ }
367
+
368
+ try {
369
+ const describe = await db.describeQueryColumns(sql);
370
+ if (describe.ok && Array.isArray(describe.columns)) {
371
+ for (const c of describe.columns) columns.add(c);
372
+ } else if (describe.error) {
373
+ warnings.push({
374
+ source: key,
375
+ message: `${describe.error.code}: ${describe.error.message}`
376
+ });
377
+ }
378
+ } catch (err) {
379
+ warnings.push({
380
+ source: key,
381
+ message: err.message || String(err)
382
+ });
383
+ }
384
+ }
385
+
386
+ return {
387
+ columns: [...columns],
388
+ warnings
389
+ };
390
+ }
391
+
392
+ /**
393
+ * Resolve SQL source dari string (inline SQL atau file: reference).
394
+ * Base path file: reference = options.payloadDir (folder payload/).
395
+ *
396
+ * @param {string} value - Inline SQL atau "file:relative/path.sql"
397
+ * @param {string|null} payloadDir - Base directory untuk file: reference
398
+ * @param {string} sourceKey - Key untuk error message context
399
+ * @returns {{ok: boolean, sql?: string, error?: string}}
400
+ */
401
+ function resolveSqlSource(value, payloadDir, sourceKey) {
402
+ const trimmed = value.trim();
403
+
404
+ if (trimmed.startsWith('file:')) {
405
+ const relativePath = trimmed.substring(5);
406
+ if (!payloadDir) {
407
+ return {
408
+ ok: false,
409
+ error: `Cannot resolve file: reference (${relativePath}) — payloadDir not provided`
410
+ };
411
+ }
412
+ const fullPath = path.join(payloadDir, relativePath);
413
+ if (!fs.existsSync(fullPath)) {
414
+ return {
415
+ ok: false,
416
+ error: `Referenced SQL file not found: ${relativePath}`
417
+ };
418
+ }
419
+ try {
420
+ const content = fs.readFileSync(fullPath, 'utf8');
421
+ if (!content || content.trim().length === 0) {
422
+ return {
423
+ ok: false,
424
+ error: `Referenced SQL file is empty: ${relativePath}`
425
+ };
426
+ }
427
+ return { ok: true, sql: content };
428
+ } catch (err) {
429
+ return {
430
+ ok: false,
431
+ error: `Cannot read SQL file ${relativePath}: ${err.message}`
432
+ };
433
+ }
434
+ }
435
+
436
+ return { ok: true, sql: trimmed };
437
+ }
438
+
439
+ /**
440
+ * Substitusi placeholder runtime dalam SQL sebelum describe.
441
+ * Saat ini hanya `${tableName}` yang di-handle (konsisten dengan generator).
442
+ *
443
+ * @param {string} sql
444
+ * @param {string} tableName
445
+ * @returns {string}
446
+ */
447
+ function substituteQueryPlaceholders(sql, tableName) {
448
+ if (!sql || !tableName) return sql;
449
+ return sql.replace(/\$\{tableName\}/g, tableName);
450
+ }
451
+
271
452
  module.exports = {
272
453
  DEFAULT_EXCLUDED_COLUMNS,
273
454
  normalizeDatabaseType,
274
455
  normalizePayloadValidationType,
275
456
  compareSchemaStrict,
276
- formatDriftReport
457
+ formatDriftReport,
458
+ resolveQuerySourceColumns,
459
+ resolveSqlSource
277
460
  };
@@ -1 +1 @@
1
- function a0_0x163f(_0x2fff76,_0x3dad32){_0x2fff76=_0x2fff76-0x160;const _0x38bfad=a0_0x38bf();let _0x163fb2=_0x38bfad[_0x2fff76];if(a0_0x163f['SAJaCv']===undefined){var _0xe116b1=function(_0x1825a5){const _0x2aec6a='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x2b8fe5='',_0x2d0f21='';for(let _0x53cfad=0x0,_0x13283a,_0xeaa967,_0x5b14df=0x0;_0xeaa967=_0x1825a5['charAt'](_0x5b14df++);~_0xeaa967&&(_0x13283a=_0x53cfad%0x4?_0x13283a*0x40+_0xeaa967:_0xeaa967,_0x53cfad++%0x4)?_0x2b8fe5+=String['fromCharCode'](0xff&_0x13283a>>(-0x2*_0x53cfad&0x6)):0x0){_0xeaa967=_0x2aec6a['indexOf'](_0xeaa967);}for(let _0x571693=0x0,_0x279c6a=_0x2b8fe5['length'];_0x571693<_0x279c6a;_0x571693++){_0x2d0f21+='%'+('00'+_0x2b8fe5['charCodeAt'](_0x571693)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x2d0f21);};a0_0x163f['pKrfaV']=_0xe116b1,a0_0x163f['kVtkIC']={},a0_0x163f['SAJaCv']=!![];}const _0x2e82a7=_0x38bfad[0x0],_0x3ae501=_0x2fff76+_0x2e82a7,_0x217380=a0_0x163f['kVtkIC'][_0x3ae501];return!_0x217380?(_0x163fb2=a0_0x163f['pKrfaV'](_0x163fb2),a0_0x163f['kVtkIC'][_0x3ae501]=_0x163fb2):_0x163fb2=_0x217380,_0x163fb2;}function a0_0x38bf(){const _0x444f17=['Cgf5Bg9Hzcbku09oigzPBguGBg9JyxrPB24','rgfZAgjVyxjKigvUzhbVAw50ig1HEsbVChqTAw4GDg8GuMvKAxmTyMfZzwqGy2fJAguUifbHDhrLCM4GzM9SBg93CYbWCM9JzxnZB3iGy2fJAguGkhnLzsbMzwf0lwnHy2HLlM1Kks4Gq2fJAguGC2nVCguGAxmGDgHLigz1BgWGCMvZCg9UC2uGzw52zwXVCgu7ig9UzsbJywnOzsbLBNrYEsbWzxiGkhbHCMfTCYaRihDPzgDLDhnBxsbZDwjZzxqPignVBwjPBMf0Aw9UlG','w3SGiNbLCMLVzci6iciYmdi2lta0lti0iIWGiNzHBhvLiJOGiJe4ntaIih0Sic4UlIbD','mti1nZa5mMvUvxLRDa','twv0CMLJicSGuhjVz3jLC3mGDg8Gr29HBa','yxjYyxKGB2yGB2jQzwn0CW','msbYB3CGW5CGmIbJB2X1Bw5Z','A2v54OAszMLSztPYzwXHDgL2zs9WyxrOl3rVl3f1zxj5lNnXBa','l2fWAs97ChjVAMvJDh0VE25HBwv9l2rHC2HIB2fYza','DgfYz2v0','yxjYyxK8C3rYAw5NpIWGB3b0Aw9UywWG4Ocuihn1yNnLDcbVzIb3AwrNzxqGsurZihrVigv4zwn1DguUie9TAxqGDg8GzxHLy3v0zsbHBgWGzgvJBgfYzwqGD2LKz2v0CY4','zgvMyxvSDa','pJ0GmcaOC2vJB25KCYK','yM9VBgvHBG','mti0ntyYowPwwvnMEa','zw5HyMXLza','zMLSztPXDwvYEs88Cgf0Ad4VDhjLBMqUC3fS','v2LKz2v0igLKzw50AwzPzxi7ihvZzwqGyxmGDgHLihjLC3bVBNnLigTLEsbPBIb0AguGzgfZAgjVyxjKigvUDMvSB3bLlG','yxzNx2rHAwX5x3nHBgvZ','zMLSztPXDwvYEs88Cgf0Ad4VCg9PBNrZlNnXBa','sgvHzgXPBMuGBwv0CMLJihDPDgGGDhjLBMqGy2HPCcbHBMqGChjVz3jLC3mGyMfYigfNywLUC3qGysbWzxjPB2qGDgfYz2v0lIbtDwL0ywjSzsbMB3iGD2LKz2v0CYbSAwTLicDpCMrLCNmGvgHPCYbnB250AcCU','ugfYyw0Gzgf0ysb0ExbLlIbwywXPzgf0zxmGCMvXDwvZDcbIB2r5igfUzcbZAgfWzxmGCNvUDgLTzsbWyxjHBwv0zxiGyMLUzgLUzY4','mJa3u0HlyxPI','Bgf5B3v0','EYaIzgLYzwn0Aw9UiJOGiMrVD24IlcaICgn0iJOGiJiUmIiGFq','Cg9PBNrZ','C2nHBgfYihbYAw1PDgL2zq','uMvZzxj2zwqGzM9YiensvuqGCgf5Bg9HzhmUieeGzgfZAgjVyxjKihbHEwXVywqGBxvZDcbKzwnSyxjLicD3AwrNzxrZjYbPBNn0zwfKlG','rNjVBNrLBMqGy29TChv0zxmGDg9Fz29HBca9ihrHCMDLDcaTihzHBhvLigfUzcbWy3qGpsbYB3vUzcH2ywX1zsaVihrHCMDLDcaQideWmcKGzM9YihrOzsbWCM9NCMvZCYbIyxiUifzPC3vHBcb3Awr0AcbPCYbWCMvZzw50yxrPB25HBcbHBMqGBxvZDcbot1qGBgL2zsbPBIb0AguGyMfJA2vUzcbWyxLSB2fKlIbjzIbWCM9NCMvZCYbPBNzVBhzLCYbJB21WBgv4igj1C2LUzxnZihj1BgvZicHLlMCUigv4y2X1zguGD2vLA2vUzhmSihbYB3jHDgvKihDVCMTKyxLZksWGDxnLigeGC2LUz2XLig11BhrPlwnVBhvTBIbXDwvYEsbZBYaNCgn0jYbPCYbHihn0ywjSzsbIDxnPBMvZCYbMywn0ihjHDgHLCIb0AgfUihzPC3vHBcb3Awr0Ac4','ue9tvcaVyxbPl21PBMKTAw52zw50B3j5l2rHC2GTAw5IB3vUzc9KyxnOyM9HCMq','q29SBgfWC2uGDg8GC2nHBgfYihbYAw1PDgL2zsaODgHLihzHBhvLig9MihrOzsbZAw5NBguGy29SDw1Uks4','zNjLztSGlNnXBcbYzwnVBw1LBMrLzcbMB3iGzwrPDg9YigHPz2HSAwDODa','C2HHCguTy29UzMXPy3q','C3rYAw5N','BgvUz3rO','zMLSztPYzwXHDgL2zs9WyxrOl3rVl3f1zxj5lNnXBa','CxvLCMLLCW','vuKGBgfIzwWGAxmGysbMCM9UDgvUzcbYzw5KzxjPBMCGy29Uy2vYBI4','tIbYB3DZimoxidiGy29SDw1UCW','ugfYyw1LDgvYignVBNrYywn0igzVCIb0AguGzgfZAgjVyxjKlIbfywnOigTLEsbPCYbHihbHCMfTig5HBwu7ihzHBhvLCYbKzxnJCMLIzsb0ExbLl3jLCxvPCMvKl2rLzMf1BhqUifbSywnLAg9SzgvYCYbPBNnPzguGD2LKz2v0ifnrtcbTDxn0ihjLzMvYzw5JzsbKzwnSyxjLzcbWyxjHBsbUyw1LCY4','u1fmigzPBguGy29UDgvUDcbPCYbLBwjLzgrLzcbHCYbkyxzHu2nYAxb0ihrLBxbSyxrLigXPDgvYywWGAw5ZAwrLihrOzsbNzw5LCMf0zwqGBw9KDwXLigzPBguUifj1BNrPBwuGCgvYzM9YBxmGEMvYBYbKAxnRieKVtYbWzxiGCMvXDwvZDcdIGjqGywXSifnrtcbPCYbPBIbTzw1VCNKGywz0zxiGBw9KDwXLigXVywqU','u3bHCMTSAw5LigXPyNjHCMLLCYaOqxbLEenOyxj0CYWGq2HHCNrPC3qSigv0yY4Pihr5CgLJywXSEsbUzwvKigeGCgXHAw4GBNvTyMvYigfYCMf5lIbgCM9UDgvUzcbTyxbZihbVAw50CY5TyxaOCca9pIbWlNzHBhvLks4GvgHLicDWzxjPB2qNigzPzwXKihn0yxLZigzVCIb0B29SDgLWigfUzcbNyxaTCMvZAwXPzw5JzsbHz2fPBNn0ig1PC3nPBMCGzgf5CY4GvxnLigDLBMvYyxrLx3nLCMLLCYbPBIbtuuWGDg8Gzw5ZDxjLignVBNnPC3rLBNqGCM93ignVDw50igv2zw4GzM9YigrHExmGD2L0AcbUBYb0CMfUC2fJDgLVBNmU','Dg9WlwXLDMvSicDWyxjHBxmNig9IAMvJDa','DhjLBMq','zgLYzwn0Aw9U','iNzHBhvLiJOGiJy5nZaWiG','mJKWotKXmMrqEu1UBa','CgvYAw9K','zgfZAgjVyxjKihbHEwXVywq','ndC3otC1mgr6DfnRsG','vMLZDwfSignVBg9YigLZigeGzNjVBNrLBMqGCMvUzgvYAw5NignVBMnLCM4U','D2LKz2v0CW','Cgn0','C3vIDgL0Bgu','B2jQzwn0','yxjYyxK8C3rYAw5NpG','B3jKzxjZx3rOAxnFBw9UDgG','zgf0zq','v2HLBIb0CNvLlcb0AguGCMvXDwvZDcbIB2r5ie1vu1qGAw5JBhvKzsb0AgLZihbHCMfTicHVDgHLCNDPC2uGndaWks4','ntiWnZG1nLLfy2P5rW','CxvLCNK','EYaIzgLYzwn0Aw9UiJOGiNvWiIWGiNbJDci6iciYlJyIih0','u0vmrunuicOGrLjptsbZDg9JA19PBMjVDw5KifDirvjfievyvfjbq1qOwuvbuIbguK9nigLUyM91BMrFzgf0zsKGpsa6EwvHCG','iJi0mJaI','DMfSDwu','zgfZAc1HDxrOB3iTC3rHDhm','AxrLBxm','Ahr0Chm6lY9Yzxn0zM9Yz2uUzgv2l2rVy3mVC2vYDMvYl3f1zxj5lwrHDgeVzgfZAgjVyxjK','zMLSztPXDwvYEs88Cgf0Ad4VDMfSDwuUC3fS','iJe4mZyI','tvvtvcbZDgfYDcb3AxrOicDKyxnOlsCGChjLzML4','nZK1nZG1D0P1yMzT','kd88itOPoIHBys16qs1Ax11Bys16qs1Amc05x10Qkq','msbYB3CGW5CGmsbJB2X1Bw4','zMLSztPXDwvYEs88Cgf0Ad4Vy3vYCMvUDc5ZCwW','BgfIzwW','DgL0Bgu','tgLZDcbVzIbduLveihrHyMXLig5HBwvZihrOyxqSihDOzw4GD3jPDhrLBIWGD2LSBcb0CMLNz2vYigLUDMfSAwrHDgLVBIbVzIb0AgLZigrHC2HIB2fYzcbJywnOzs4','EYaICgfYyw1ZiJOGEYaIEwvHCIi6ihSGiNr5CguIoIaIBNvTyMvYiIWGiNjLCxvPCMvKiJOGDhj1zsb9ih0GFq','EYaIzgLYzwn0Aw9UiJOGiNvWiIWGiNbJDci6iciYlJiIih0','Dg9Wtgv2zwXbBgXVD2vK','zxHWB3j0CW','CgfYyw1Z','BNvTyMvY','yxjYyxK','zgfZAc1ZywXLCW','oNbHCMfTtMfTzq','EYbZDwnJzxnZoIbIB29SzwfUlcbKyxrHoIb7idX3AwrNzxrjzd46idXWzxjxAwrNzxrszxnWB25Zzt4Sic4UlIb9ih0','qsb3AwrNzxqGtvvtvcbKzwnSyxjLigv4ywn0BhKGB25Lig9MoIaNCxvLCNKNie9sicDXDwvYAwvZjY4GqM90AcbVCIbUzwL0AgvYigLZihjLAMvJDgvKlG','mtK0otC2ELnvuwvH','u2LUz2XLifnrtcbXDwvYEsbMB3iGDgHLihDPzgDLDc4','m3HmBeXjzq','phDPzgDLDf9Pzd4'];a0_0x38bf=function(){return _0x444f17;};return a0_0x38bf();}const a0_0xaeedec=a0_0x163f;(function(_0x292394,_0x4f6702){const _0x24b504=a0_0x163f,_0x4569f4=_0x292394();while(!![]){try{const _0x599420=parseInt(_0x24b504(0x19c))/0x1+-parseInt(_0x24b504(0x1b5))/0x2+parseInt(_0x24b504(0x1b0))/0x3*(-parseInt(_0x24b504(0x183))/0x4)+parseInt(_0x24b504(0x186))/0x5+parseInt(_0x24b504(0x190))/0x6+-parseInt(_0x24b504(0x163))/0x7+-parseInt(_0x24b504(0x1ae))/0x8*(parseInt(_0x24b504(0x16b))/0x9);if(_0x599420===_0x4f6702)break;else _0x4569f4['push'](_0x4569f4['shift']());}catch(_0x2a3107){_0x4569f4['push'](_0x4569f4['shift']());}}}(a0_0x38bf,0x80380));const FORBIDDEN_FRONTEND_FIELDS=['widgetType',a0_0xaeedec(0x16c),a0_0xaeedec(0x1a1),a0_0xaeedec(0x18a),'color'],ALLOWED_PARAM_TYPES=['string','number','boolean',a0_0xaeedec(0x18e)],FRONTEND_CONCERN_REASONS={'widgetType':'Visual\x20variant\x20(donut,\x20bar,\x20pie,\x20area)\x20is\x20a\x20frontend\x20rendering\x20concern\x20(separation\x20of\x20concerns).','layout':'Layout\x20is\x20a\x20frontend\x20rendering\x20concern.','title':a0_0xaeedec(0x17a),'subtitle':'UI\x20label\x20is\x20a\x20frontend\x20rendering\x20concern.','color':a0_0xaeedec(0x187)},PAYLOAD_SHAPE={'discriminator':{'field':a0_0xaeedec(0x188),'presentMeans':a0_0xaeedec(0x185),'absentMeans':'Not\x20a\x20dashboard\x20payload\x20(likely\x20CRUD\x20with\x20tableName,\x20or\x20invalid)','conflictsWith':'tableName','conflictRationale':'A\x20payload\x20with\x20both\x20\x27widgets\x27\x20and\x20\x27tableName\x27\x20is\x20rejected\x20by\x20DashboardValidator.\x20Pick\x20one\x20shape.'},'topLevelAllowed':[{'name':'widgets','type':a0_0xaeedec(0x1a9),'required':!![],'minItems':0x1,'description':'List\x20of\x20widget\x20definitions.\x20Order\x20is\x20informational\x20only\x20(response\x20keys\x20are\x20by\x20widget\x20id,\x20not\x20array\x20index).'},{'name':a0_0xaeedec(0x1a7),'type':a0_0xaeedec(0x18b),'required':![],'description':a0_0xaeedec(0x17c)},{'name':'cache','type':'object','required':![],'description':'Optional\x20cache\x20configuration.\x20See\x20cacheSpec\x20for\x20details.'}],'topLevelForbidden':[{'name':'tableName','category':a0_0xaeedec(0x175),'reason':a0_0xaeedec(0x170)},...FORBIDDEN_FRONTEND_FIELDS['map'](_0x2b8fe5=>({'name':_0x2b8fe5,'category':'frontend-concern','reason':FRONTEND_CONCERN_REASONS[_0x2b8fe5]}))]},WIDGET_SPEC={'requiredFields':[{'name':'id','type':a0_0xaeedec(0x176),'constraint':'non-empty,\x20unique\x20across\x20widgets\x20in\x20the\x20same\x20payload','description':a0_0xaeedec(0x166)}],'exclusiveQueryFields':{'rule':a0_0xaeedec(0x1ad),'options':[{'name':a0_0xaeedec(0x191),'type':a0_0xaeedec(0x176),'format':a0_0xaeedec(0x178),'description':a0_0xaeedec(0x1af),'responseShape':'Always\x20{\x20items:\x20[...]\x20}\x20regardless\x20of\x20SQL\x20result\x20shape.'},{'name':a0_0xaeedec(0x179),'type':'object','format':a0_0xaeedec(0x1b9),'minKeys':0x1,'description':'Multi-SQL\x20widget.\x20Each\x20key\x20becomes\x20a\x20key\x20in\x20the\x20response\x20object.','responseShape':'Per-key\x20based\x20on\x20scalarCollapseRules\x20below.'}]},'forbiddenFields':FORBIDDEN_FRONTEND_FIELDS},PARAM_SPEC={'container':a0_0xaeedec(0x17f),'keyConvention':'Param\x20name\x20must\x20match\x20the\x20placeholder\x20regex\x20`[a-zA-Z_][a-zA-Z0-9_]*`\x20(alphanumeric\x20+\x20underscore,\x20must\x20start\x20with\x20letter\x20or\x20underscore).','perEntryFields':[{'name':'type','required':!![],'allowedValues':ALLOWED_PARAM_TYPES,'description':a0_0xaeedec(0x16a)},{'name':'required','required':![],'type':'boolean','default':![],'description':a0_0xaeedec(0x18f)},{'name':a0_0xaeedec(0x160),'required':![],'type':'any\x20(must\x20be\x20compatible\x20with\x20declared\x20\x27type\x27)','description':'Default\x20value\x20applied\x20when\x20the\x20request\x20omits\x20this\x20param.\x20Validator\x20does\x20NOT\x20strictly\x20type-check\x20default;\x20runtime\x20is\x20responsible\x20for\x20compatibility.'}]},SCALAR_COLLAPSE_RULES=[{'appliesTo':'widget.query\x20(singular)','rule':'Always\x20wrap\x20as\x20{\x20items:\x20[...]\x20}\x20regardless\x20of\x20SQL\x20result\x20shape.','exampleSqlShape':'any\x20(1\x20row\x20×\x201\x20col,\x20N\x20rows\x20×\x20M\x20cols,\x20etc.)','exampleResponse':'\x22shopping_categories\x22:\x20{\x20\x22items\x22:\x20[{\x20\x22name\x22:\x20\x22Lands\x22\x20},\x20{\x20\x22name\x22:\x20\x22Houses\x22\x20}]\x20}'},{'appliesTo':'widget.queries.<key>\x20with\x20SQL\x20returning\x201\x20row\x20×\x201\x20column','rule':a0_0xaeedec(0x173),'exampleSqlShape':'1\x20row\x20×\x201\x20col,\x20output\x20column\x20\x27value\x27','exampleResponse':a0_0xaeedec(0x182)},{'appliesTo':'widget.queries.<key>\x20with\x20SQL\x20returning\x201\x20row\x20×\x20multiple\x20columns','rule':'Collapse\x20to\x20object\x20whose\x20keys\x20are\x20SQL\x20column\x20names\x20(lowercased).','exampleSqlShape':'1\x20row\x20×\x202\x20cols,\x20output\x20columns\x20\x27direction\x27,\x20\x27pct\x27','exampleResponse':'\x22trend\x22:\x20{\x20\x22direction\x22:\x20\x22up\x22,\x20\x22pct\x22:\x20\x222.2\x22\x20}'},{'appliesTo':'widget.queries.<key>\x20with\x20SQL\x20returning\x20N\x20rows','rule':'Return\x20as\x20array\x20of\x20objects\x20(no\x20collapse).','exampleSqlShape':'N\x20rows\x20×\x20M\x20cols','exampleResponse':'\x22items\x22:\x20[{\x20\x22label\x22:\x20\x22Shoes\x22,\x20\x22value\x22:\x20\x227660\x22\x20},\x20...]'}],COMMON_WIDGET_PATTERNS=[{'id':'metric_donut_breakdown','name':'Metric\x20+\x20Donut\x20Breakdown','useCase':'Headline\x20metric\x20with\x20trend\x20chip\x20and\x20breakdown\x20across\x20categories.\x20Suitable\x20for\x20widgets\x20like\x20\x27Expected\x20Earnings\x27\x20that\x20show\x20total\x20value,\x20percentage\x20change,\x20and\x20per-category\x20contribution.','payloadShape':{'id':a0_0xaeedec(0x1b1),'queries':{'value':'file:query/<path>/value.sql','trend':a0_0xaeedec(0x165),'items':'file:query/<path>/breakdown.sql'}},'sqlShapesPerKey':[{'key':a0_0xaeedec(0x195),'shape':a0_0xaeedec(0x19e),'outputColumns':['value'],'collapseRule':a0_0xaeedec(0x16f)},{'key':'trend','shape':'1\x20row\x20×\x202\x20columns','outputColumns':['direction',a0_0xaeedec(0x189)],'collapseRule':'object'},{'key':a0_0xaeedec(0x197),'shape':a0_0xaeedec(0x17b),'outputColumns':[a0_0xaeedec(0x1a0),a0_0xaeedec(0x195)],'collapseRule':'array\x20of\x20objects'}],'responseShape':{'value':'\x2269700\x22','trend':a0_0xaeedec(0x1a4),'items':'[{\x20\x22label\x22:\x20\x22Shoes\x22,\x20\x22value\x22:\x20\x227660\x22\x20},\x20{\x20\x22label\x22:\x20\x22Gaming\x22,\x20\x22value\x22:\x20\x222820\x22\x20},\x20{\x20\x22label\x22:\x20\x22Others\x22,\x20\x22value\x22:\x20\x2245257\x22\x20}]'},'referenceWidgetId':'expected_earnings','socNotes':'Frontend\x20determines\x20donut/pie\x20variant,\x20color\x20per\x20category,\x20and\x20label\x20order.\x20If\x20per-category\x20percentage\x20is\x20needed\x20for\x20the\x20donut\x20arc,\x20frontend\x20computes\x20it\x20from\x20items[i].value\x20/\x20sum(items[*].value).\x20No\x20need\x20to\x20send\x20\x27pct\x27\x20from\x20backend\x20unless\x20the\x20figure\x20is\x20a\x20stable\x20business\x20calculation\x20independent\x20of\x20visual\x20rendering.'},{'id':'metric_sparkline','name':'Metric\x20+\x20Sparkline','useCase':'Headline\x20metric\x20with\x20trend\x20chip\x20and\x20sparkline\x20mini-chart\x20for\x20short\x20windows\x20(7\x20days,\x2012\x20months,\x20etc.).\x20Suitable\x20for\x20widgets\x20like\x20\x27Average\x20Daily\x20Sales\x27.','payloadShape':{'id':a0_0xaeedec(0x1b1),'queries':{'value':a0_0xaeedec(0x199),'trend':a0_0xaeedec(0x165),'points':a0_0xaeedec(0x168)}},'sqlShapesPerKey':[{'key':a0_0xaeedec(0x195),'shape':'1\x20row\x20×\x201\x20column','outputColumns':[a0_0xaeedec(0x195)],'collapseRule':a0_0xaeedec(0x16f)},{'key':a0_0xaeedec(0x180),'shape':a0_0xaeedec(0x1b8),'outputColumns':[a0_0xaeedec(0x181),'pct'],'collapseRule':'object'},{'key':a0_0xaeedec(0x16e),'shape':'N\x20rows\x20×\x202\x20columns','outputColumns':[a0_0xaeedec(0x184),a0_0xaeedec(0x195)],'collapseRule':a0_0xaeedec(0x1b7)}],'responseShape':{'value':a0_0xaeedec(0x194),'trend':a0_0xaeedec(0x192),'points':a0_0xaeedec(0x1b4)},'referenceWidgetId':a0_0xaeedec(0x167),'socNotes':a0_0xaeedec(0x17e)},{'id':'metric_progress_to_goal','name':a0_0xaeedec(0x1b6),'useCase':a0_0xaeedec(0x169),'payloadShape':{'id':'<widget_id>','queries':{'value':a0_0xaeedec(0x19f),'trend':a0_0xaeedec(0x165),'target':'file:query/<path>/target.sql'}},'sqlShapesPerKey':[{'key':'value','shape':a0_0xaeedec(0x19e),'outputColumns':['value\x20(or\x20current)'],'collapseRule':a0_0xaeedec(0x16f)},{'key':'trend','shape':'1\x20row\x20×\x202\x20columns','outputColumns':['direction','pct'],'collapseRule':'object'},{'key':a0_0xaeedec(0x1bb),'shape':'1\x20row\x20×\x201\x20column','outputColumns':['target'],'collapseRule':a0_0xaeedec(0x16f)}],'responseShape':{'value':a0_0xaeedec(0x19a),'trend':a0_0xaeedec(0x16d),'target':'\x222884\x22'},'referenceWidgetId':a0_0xaeedec(0x18d),'socNotes':a0_0xaeedec(0x171)}],NAMING_CONVENTION={'dashboardName':{'constraint':a0_0xaeedec(0x19b),'minLength':0x6,'maxLength':0x32,'regex':'^dash-[a-zA-Z0-9_-]+$','examples':[a0_0xaeedec(0x1aa),'dash-inbound',a0_0xaeedec(0x196)],'rationale':'The\x20prefix\x20becomes\x20part\x20of\x20the\x20URL\x20segment.\x20The\x20reserved\x20scheme\x20keeps\x20dashboard\x20endpoints\x20visually\x20distinct\x20from\x20CRUD\x20endpoints\x20in\x20the\x20URL\x20space\x20and\x20allows\x20future\x20routing\x20differentiation.'}},URL_PATTERN={'method':'POST','path':a0_0xaeedec(0x1ba),'exampleFull':a0_0xaeedec(0x172),'requestBodyShape':{'params':'object\x20—\x20values\x20for\x20declared\x20params\x20(validated\x20against\x20params\x20contract;\x20missing\x20required\x20→\x20400,\x20type\x20mismatch\x20→\x20400)','widgets':a0_0xaeedec(0x1bc)},'responseShape':{'envelope':a0_0xaeedec(0x1ac),'perWidgetResponse':'Determined\x20by\x20scalarCollapseRules.\x20Failed\x20widgets\x20produce\x20{\x20error:\x20\x27...\x27\x20}\x20block\x20with\x20top-level\x20success\x20still\x20true\x20(one\x20widget\x20failure\x20does\x20NOT\x20fail\x20the\x20dashboard).'}},FILE_REFERENCE_CONVENTION={'format':'file:relative/path/to/query.sql','pathRelativeTo':a0_0xaeedec(0x1b2),'fileExtensionPolicy':a0_0xaeedec(0x174),'resolvedAt':'generation\x20time\x20(NOT\x20runtime)','embedStrategy':a0_0xaeedec(0x17d),'implication':'Updating\x20an\x20SQL\x20file\x20requires\x20regenerating\x20the\x20dashboard\x20module\x20(\x27codegen_create_dashboard\x27)\x20for\x20changes\x20to\x20take\x20effect.'},PLACEHOLDER_CONVENTION={'format':a0_0xaeedec(0x1ab),'regex':a0_0xaeedec(0x19d),'regexNotes':'Negative\x20lookbehind\x20prevents\x20matching\x20\x27::\x27\x20(Postgres\x20cast\x20syntax)\x20as\x20a\x20placeholder.','scanScope':'All\x20widget\x20SQL\x20—\x20both\x20\x27query\x27\x20(singular)\x20and\x20every\x20\x27queries.<key>\x27.','constraint':'Every\x20placeholder\x20used\x20in\x20SQL\x20MUST\x20be\x20declared\x20in\x20\x27params\x27.\x20Validator\x20throws\x20Error\x20with\x20message\x20format:\x20\x22Widget\x20\x27<id>\x27\x20query\x20\x27<label>\x27\x20uses\x20undeclared\x20placeholder\x20\x27:<token>\x27\x20(declare\x20in\x20\x27params\x27)\x22.','exampleSql':a0_0xaeedec(0x193),'exampleParamDeclaration':a0_0xaeedec(0x1a3)},CACHE_SPEC={'container':'top-level\x20\x27cache\x27\x20object','optional':!![],'rationale':a0_0xaeedec(0x1b3),'fields':[{'name':a0_0xaeedec(0x164),'type':a0_0xaeedec(0x162),'required':!![],'description':'Toggle\x20cache\x20feature\x20for\x20this\x20dashboard.'},{'name':'ttl','type':a0_0xaeedec(0x1a8),'required':![],'constraint':a0_0xaeedec(0x161),'default':'inherits\x20CACHE_TTL\x20env','description':'Time-to-live\x20in\x20seconds.\x200\x20effectively\x20disables\x20cache\x20for\x20this\x20entry.'},{'name':'invalidates','type':a0_0xaeedec(0x18c),'required':![],'default':'[]','description':a0_0xaeedec(0x1a2)}],'validation':{'sqlCrossReference':'When\x20cache.enabled\x20===\x20true\x20and\x20invalidates\x20is\x20non-empty:\x20validator\x20extracts\x20table\x20candidates\x20from\x20widget\x20SQL\x20(regex\x20FROM/JOIN),\x20cross-references\x20with\x20metadata/{project}.json\x20(endpoints[*].tableName\x20where\x20type\x20===\x20\x22module\x22),\x20and\x20asserts\x20equality\x20of\x20expected\x20vs\x20declared\x20sets.\x20Mismatches\x20are\x20reported\x20per\x20category\x20(missing,\x20extra,\x20unmatched).','errorOn':['Table\x20appears\x20in\x20SQL\x20AND\x20in\x20metadata\x20project,\x20but\x20missing\x20from\x20invalidates\x20(cache\x20stale\x20risk)','Table\x20declared\x20in\x20invalidates,\x20but\x20not\x20detected\x20in\x20any\x20widget\x20SQL\x20(typo\x20or\x20dead\x20entry)'],'warningOn':['Table\x20detected\x20in\x20SQL,\x20but\x20not\x20registered\x20as\x20CRUD\x20endpoint\x20in\x20metadata\x20project\x20(likely\x20a\x20view,\x20CTE\x20alias,\x20or\x20cross-project\x20table\x20—\x20cascade\x20will\x20not\x20fire)']}},DOCUMENTATION_URL=a0_0xaeedec(0x198),DASHBOARD_CATALOG={'schemaVersion':'1.0','source':'dashboard-catalog','summary':{'totalAllowedTopLevelFields':PAYLOAD_SHAPE[a0_0xaeedec(0x1a5)]['length'],'totalForbiddenFrontendFields':FORBIDDEN_FRONTEND_FIELDS['length'],'totalParamTypes':ALLOWED_PARAM_TYPES[a0_0xaeedec(0x177)],'totalScalarCollapseRules':SCALAR_COLLAPSE_RULES['length'],'totalCommonWidgetPatterns':COMMON_WIDGET_PATTERNS['length']},'payloadShape':PAYLOAD_SHAPE,'widgetSpec':WIDGET_SPEC,'paramSpec':PARAM_SPEC,'scalarCollapseRules':SCALAR_COLLAPSE_RULES,'commonWidgetPatterns':COMMON_WIDGET_PATTERNS,'namingConvention':NAMING_CONVENTION,'urlPattern':URL_PATTERN,'fileReferenceConvention':FILE_REFERENCE_CONVENTION,'placeholderConvention':PLACEHOLDER_CONVENTION,'cacheSpec':CACHE_SPEC,'documentationUrl':DOCUMENTATION_URL};module[a0_0xaeedec(0x1a6)]={'DASHBOARD_CATALOG':DASHBOARD_CATALOG};
1
+ const a0_0x3cf759=a0_0x58e2;function a0_0x4838(){const _0x51f5e6=['qsbWyxLSB2fKihDPDgGGyM90AcaND2LKz2v0CYCGyw5KicD0ywjSzu5HBwuNigLZihjLAMvJDgvKigj5ierHC2HIB2fYzfzHBgLKyxrVCI4GugLJAYbVBMuGC2HHCguU','Cg9PBNrZ','y2fJAgu','DMfSDwuGkg9Yign1CNjLBNqP','tIbYB3DZimoxidiGy29SDw1UCW','ms4W','z2vUzxjHDgLVBIb0Aw1LicHot1qGCNvUDgLTzsK','q29SBgfWC2uGDg8GC2nHBgfYihbYAw1PDgL2zsaODgHLihzHBhvLig9MihrOzsbZAw5NBguGy29SDw1Uks4','mtKWntKWmevprufgzW','rgv0zxjTAw5LzcbIEsbZy2fSyxjdB2XSyxbZzvj1BgvZlIbgywLSzwqGD2LKz2v0CYbWCM9KDwnLihSGzxjYB3i6icCUlI4Nih0GyMXVy2SGD2L0Acb0B3aTBgv2zwWGC3vJy2vZCYbZDgLSBcb0CNvLicHVBMuGD2LKz2v0igzHAwX1CMuGzg9LCYbot1qGzMfPBcb0AguGzgfZAgjVyxjKks4','mZG1mMvICMrNsW','msbYB3CGW5CGmIbJB2X1Bw5Z','txvSDgKTu1fmihDPzgDLDc4GrwfJAcbRzxKGyMvJB21LCYbHigTLEsbPBIb0AguGCMvZCg9UC2uGB2jQzwn0lG','yxzNx2rHAwX5x3nHBgvZ','msbYB3CGW5CGmIbJB2XZlcbVDxrWDxqGy29SDw1UCYaNzgLYzwn0Aw9UjYWGj3bJDcC','n1f2EMvRra','qsb3AwrNzxqGtvvtvcbKzwnSyxjLigv4ywn0BhKGB25Lig9MoIaNCxvLCNKNie9sicDXDwvYAwvZjY4GqM90AcbVCIbUzwL0AgvYigLZihjLAMvJDgvKlG','B3jKzxjZx3rOAxnFBw9UDgG','Cgn0','pJ0GmcaOC2vJB25KCYK','tIbYB3DZimoxie0Gy29SCW','DMfSDwu','kd88itOPoIHBys16qs1Ax11Bys16qs1Amc05x10Qkq','CMvXDwLYzwq','AxrLBxm','vgLTzs10BY1SAxzLigLUihnLy29UzhmUidaGzwzMzwn0AxzLBhKGzgLZywjSzxmGy2fJAguGzM9YihrOAxmGzw50CNKU','mZmZmty1nKPhtKDIBq','msbYB3CGW5CGmsbJB2X1Bw4','A2v54OAszMLSztPYzwXHDgL2zs9WyxrOl3rVl3f1zxj5lNnXBa','iNzHBhvLiJOGiJy5nZaWiG','msbYB3CGW5CGmsbJB2WSig91Dhb1DcbJB2X1Bw4Gj3zHBhvLjW','DgfIBgvoyw1L','CxvLCNK','mZCYnJq2CvDzAvfR','yM9VBgvHBG','tMvNyxrPDMuGBg9VA2jLAgLUzcbWCMv2zw50CYbTyxrJAgLUzYaNoJONicHqB3n0z3jLCYbJyxn0ihn5BNrHEcKGyxmGysbWBgfJzwHVBgrLCI4','vMLZDwfSignVBg9YigLZigeGzNjVBNrLBMqGCMvUzgvYAw5NignVBMnLCM4U','twv0CMLJicSGuhjVz3jLC3mGDg8Gr29HBa','y29SB3i','uMvZzxj2zwqGzM9YiensvuqGCgf5Bg9HzhmUieeGzgfZAgjVyxjKihbHEwXVywqGBxvZDcbKzwnSyxjLicD3AwrNzxrZjYbPBNn0zwfKlG','D2LKz2v0lNf1zxjPzxmUpgTLEt4GD2L0AcbtuuWGCMv0DxjUAw5NideGCM93imoxig11BhrPCgXLignVBhvTBNm','Bgf5B3v0','zgfZAgjVyxjKihbHEwXVywq','nZK4odmYC2vxr0fs','tgf5B3v0igLZigeGzNjVBNrLBMqGCMvUzgvYAw5NignVBMnLCM4U','zMLSztPXDwvYEs88Cgf0Ad4VDhjLBMqUC3fS','vgfIBguGzgv0zwn0zwqGAw4Gu1fmlcbIDxqGBM90ihjLz2LZDgvYzwqGyxmGq1jvrcbLBMrWB2LUDcbPBIbTzxrHzgf0ysbWCM9Qzwn0icHSAwTLBhKGysb2Awv3lcbdveuGywXPyxmSig9YignYB3nZlxbYB2PLy3qGDgfIBguG4OcuignHC2nHzguGD2LSBcbUB3qGzMLYzsK','vgHLihbYzwzPEcbIzwnVBwvZihbHCNqGB2yGDgHLifvstcbZzwDTzw50lIbuAguGCMvZzxj2zwqGC2nOzw1LigTLzxbZigrHC2HIB2fYzcbLBMrWB2LUDhmGDMLZDwfSBhKGzgLZDgLUy3qGzNjVBsbduLveigvUzhbVAw50CYbPBIb0AguGvvjmihnWywnLigfUzcbHBgXVD3mGzNv0DxjLihjVDxrPBMCGzgLMzMvYzw50Awf0Aw9UlG','zMLSztPXDwvYEs88Cgf0Ad4VDMfSDwuUC3fS','C3rYAw5N','zNjVBNrLBMqTy29Uy2vYBG','zMLSztPXDwvYEs88Cgf0Ad4VCg9PBNrZlNnXBa','tM90igeGzgfZAgjVyxjKihbHEwXVywqGkgXPA2vSEsbduLveihDPDgGGDgfIBgvoyw1LlcbVCIbPBNzHBgLKkq','zxHWB3j0CW','zMLSztPYzwXHDgL2zs9WyxrOl3rVl3f1zxj5lNnXBa','Aw52ywXPzgf0zxm','yw55icGXihjVDYddLYaXignVBcWGtIbYB3DZimoxie0Gy29SCYWGzxrJlIK','otiWmtKWv2jStNzy','Dg9Wtgv2zwXbBgXVD2vK','DgL0Bgu','tvvtvcbZDgfYDcb3AxrOicDKyxnOlsCGChjLzML4','mtqXnZb4qKfksNK','EYaIzgLYzwn0Aw9UiJOGiMrVD24IlcaICgn0iJOGiJiUmIiGFq','BNvTyMvY','rxzLCNKGCgXHy2vOB2XKzxiGDxnLzcbPBIbtuuWGtvvtvcbIzsbKzwnSyxjLzcbPBIaNCgfYyw1ZjY4GvMfSAwrHDg9YihrOCM93CYbfCNjVCIb3AxrOig1LC3nHz2uGzM9YBwf0oIaIv2LKz2v0icC8Awq+jYbXDwvYEsaNpgXHyMvSpICGDxnLCYb1BMrLy2XHCMvKihbSywnLAg9SzgvYicC6phrVA2vUpICGkgrLy2XHCMuGAw4Gj3bHCMfTCYCPiI4','DhjLBMq','q29SBgfWC2uGDg8GB2jQzwn0ihDOB3nLigTLExmGyxjLifnrtcbJB2X1Bw4GBMfTzxmGkgXVD2vYy2fZzwqPlG','m3jVtu1Kva','qwX3yxLZihSGAxrLBxm6ifSUlI5Dih0GCMvNyxjKBgvZCYbVzIbtuuWGCMvZDwX0ihnOyxbLlG','C2nHBgfYihbYAw1PDgL2zq','Bwv0CMLJx3bYB2DYzxnZx3rVx2DVywW','ue9tva','ugvYlwTLEsbIyxnLzcbVBIbZy2fSyxjdB2XSyxbZzvj1BgvZigjLBg93lG','mJmZmtG3nKnit3D6qG','BgvUz3rO','vgfIBguGyxbWzwfYCYbPBIbtuuWGqu5eigLUig1LDgfKyxrHihbYB2PLy3qSigj1DcbTAxnZAw5NigzYB20GAw52ywXPzgf0zxmGkgnHy2HLihn0ywXLihjPC2SP','zgfZAc1HDxrOB3iTC3rHDhm','B2jQzwn0','D2LKz2v0lNf1zxjPzxmUpgTLEt4GD2L0AcbtuuWGCMv0DxjUAw5Nie4GCM93CW','yxjYyxK8C3rYAw5NpG','Dg9WlwXLDMvSicDJywnOzsCGB2jQzwn0','iJi0mJaI','ue9tvcaVyxbPl21PBMKTAw52zw50B3j5l2rHC2GTAw5IB3vUzc9KyxnOyM9HCMq','EYaIzgLYzwn0Aw9UiJOGiNvWiIWGiNbJDci6iciYlJyIih0','zMLSztPXDwvYEs88Cgf0Ad4VyNjLywTKB3DUlNnXBa','tgLZDcbVzIb3AwrNzxqGzgvMAw5PDgLVBNmUie9YzgvYigLZigLUzM9YBwf0Aw9UywWGB25SEsaOCMvZCg9UC2uGA2v5CYbHCMuGyNKGD2LKz2v0igLKlcbUB3qGyxjYyxKGAw5KzxGPlG','D2LKz2v0CW','CgfYyw1Z'];a0_0x4838=function(){return _0x51f5e6;};return a0_0x4838();}function a0_0x58e2(_0xa2e963,_0x25ad60){_0xa2e963=_0xa2e963-0x6f;const _0x483836=a0_0x4838();let _0x58e232=_0x483836[_0xa2e963];if(a0_0x58e2['HUnuDR']===undefined){var _0x148f00=function(_0x1ef57f){const _0x5684cc='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x1ee3f6='',_0x1583e8='';for(let _0x5babbe=0x0,_0x178db2,_0x5d8f2f,_0x11b206=0x0;_0x5d8f2f=_0x1ef57f['charAt'](_0x11b206++);~_0x5d8f2f&&(_0x178db2=_0x5babbe%0x4?_0x178db2*0x40+_0x5d8f2f:_0x5d8f2f,_0x5babbe++%0x4)?_0x1ee3f6+=String['fromCharCode'](0xff&_0x178db2>>(-0x2*_0x5babbe&0x6)):0x0){_0x5d8f2f=_0x5684cc['indexOf'](_0x5d8f2f);}for(let _0x5b6157=0x0,_0x27d4da=_0x1ee3f6['length'];_0x5b6157<_0x27d4da;_0x5b6157++){_0x1583e8+='%'+('00'+_0x1ee3f6['charCodeAt'](_0x5b6157)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x1583e8);};a0_0x58e2['inQQQl']=_0x148f00,a0_0x58e2['bBfVFX']={},a0_0x58e2['HUnuDR']=!![];}const _0x3171dd=_0x483836[0x0],_0x2a48fb=_0xa2e963+_0x3171dd,_0x4264bb=a0_0x58e2['bBfVFX'][_0x2a48fb];return!_0x4264bb?(_0x58e232=a0_0x58e2['inQQQl'](_0x58e232),a0_0x58e2['bBfVFX'][_0x2a48fb]=_0x58e232):_0x58e232=_0x4264bb,_0x58e232;}(function(_0x156d91,_0x5a1c8d){const _0x3c24ad=a0_0x58e2,_0x3f6472=_0x156d91();while(!![]){try{const _0x30e644=-parseInt(_0x3c24ad(0xbe))/0x1*(-parseInt(_0x3c24ad(0x9c))/0x2)+-parseInt(_0x3c24ad(0xb4))/0x3+parseInt(_0x3c24ad(0xc4))/0x4+-parseInt(_0x3c24ad(0x83))/0x5+parseInt(_0x3c24ad(0x95))/0x6+-parseInt(_0x3c24ad(0x8a))/0x7*(parseInt(_0x3c24ad(0xa6))/0x8)+-parseInt(_0x3c24ad(0x85))/0x9*(parseInt(_0x3c24ad(0xb8))/0xa);if(_0x30e644===_0x5a1c8d)break;else _0x3f6472['push'](_0x3f6472['shift']());}catch(_0xdd7c0e){_0x3f6472['push'](_0x3f6472['shift']());}}}(a0_0x4838,0x49f7e));const FORBIDDEN_FRONTEND_FIELDS=['widgetType',a0_0x3cf759(0xa4),a0_0x3cf759(0xb6),'subtitle',a0_0x3cf759(0xa1)],ALLOWED_PARAM_TYPES=[a0_0x3cf759(0xac),a0_0x3cf759(0xba),a0_0x3cf759(0x9d),'date'],FRONTEND_CONCERN_REASONS={'widgetType':'Visual\x20variant\x20(donut,\x20bar,\x20pie,\x20area)\x20is\x20a\x20frontend\x20rendering\x20concern\x20(separation\x20of\x20concerns).','layout':a0_0x3cf759(0xa7),'title':'UI\x20label\x20is\x20a\x20frontend\x20rendering\x20concern.','subtitle':'UI\x20label\x20is\x20a\x20frontend\x20rendering\x20concern.','color':a0_0x3cf759(0x9f)},PAYLOAD_SHAPE={'discriminator':{'field':a0_0x3cf759(0x79),'presentMeans':a0_0x3cf759(0xa5),'absentMeans':a0_0x3cf759(0xaf),'conflictsWith':'tableName','conflictRationale':a0_0x3cf759(0x7b)},'topLevelAllowed':[{'name':a0_0x3cf759(0x79),'type':'array','required':!![],'minItems':0x1,'description':a0_0x3cf759(0x78)},{'name':a0_0x3cf759(0x7a),'type':'object','required':![],'description':'Parameter\x20contract\x20for\x20the\x20dashboard.\x20Each\x20key\x20is\x20a\x20param\x20name;\x20values\x20describe\x20type/required/default.\x20Placeholders\x20inside\x20widget\x20SQL\x20must\x20reference\x20declared\x20param\x20names.'},{'name':a0_0x3cf759(0x7d),'type':'object','required':![],'description':'Optional\x20cache\x20configuration.\x20See\x20cacheSpec\x20for\x20details.'}],'topLevelForbidden':[{'name':a0_0x3cf759(0x9a),'category':'shape-conflict','reason':a0_0x3cf759(0xa2)},...FORBIDDEN_FRONTEND_FIELDS['map'](_0x1ee3f6=>({'name':_0x1ee3f6,'category':a0_0x3cf759(0xad),'reason':FRONTEND_CONCERN_REASONS[_0x1ee3f6]}))]},WIDGET_SPEC={'requiredFields':[{'name':'id','type':'string','constraint':'non-empty,\x20unique\x20across\x20widgets\x20in\x20the\x20same\x20payload','description':'Widget\x20identifier;\x20used\x20as\x20the\x20response\x20key\x20in\x20the\x20dashboard\x20envelope.'}],'exclusiveQueryFields':{'rule':a0_0x3cf759(0x8b),'options':[{'name':a0_0x3cf759(0x9b),'type':a0_0x3cf759(0xac),'format':a0_0x3cf759(0xb1),'description':'Single\x20SQL\x20query\x20for\x20the\x20widget.','responseShape':a0_0x3cf759(0xbf)},{'name':'queries','type':a0_0x3cf759(0x70),'format':a0_0x3cf759(0x97),'minKeys':0x1,'description':a0_0x3cf759(0x87),'responseShape':a0_0x3cf759(0xc3)}]},'forbiddenFields':FORBIDDEN_FRONTEND_FIELDS},PARAM_SPEC={'container':'top-level\x20\x27params\x27\x20object','keyConvention':'Param\x20name\x20must\x20match\x20the\x20placeholder\x20regex\x20`[a-zA-Z_][a-zA-Z0-9_]*`\x20(alphanumeric\x20+\x20underscore,\x20must\x20start\x20with\x20letter\x20or\x20underscore).','perEntryFields':[{'name':'type','required':!![],'allowedValues':ALLOWED_PARAM_TYPES,'description':'Param\x20data\x20type.\x20Validates\x20request\x20body\x20and\x20shapes\x20runtime\x20parameter\x20binding.'},{'name':a0_0x3cf759(0x92),'required':![],'type':'boolean','default':![],'description':'When\x20true,\x20the\x20request\x20body\x20MUST\x20include\x20this\x20param\x20(otherwise\x20400).'},{'name':'default','required':![],'type':'any\x20(must\x20be\x20compatible\x20with\x20declared\x20\x27type\x27)','description':'Default\x20value\x20applied\x20when\x20the\x20request\x20omits\x20this\x20param.\x20Validator\x20does\x20NOT\x20strictly\x20type-check\x20default;\x20runtime\x20is\x20responsible\x20for\x20compatibility.'}]},SCALAR_COLLAPSE_RULES=[{'appliesTo':'widget.query\x20(singular)','rule':'Always\x20wrap\x20as\x20{\x20items:\x20[...]\x20}\x20regardless\x20of\x20SQL\x20result\x20shape.','exampleSqlShape':a0_0x3cf759(0xb3),'exampleResponse':'\x22shopping_categories\x22:\x20{\x20\x22items\x22:\x20[{\x20\x22name\x22:\x20\x22Lands\x22\x20},\x20{\x20\x22name\x22:\x20\x22Houses\x22\x20}]\x20}'},{'appliesTo':'widget.queries.<key>\x20with\x20SQL\x20returning\x201\x20row\x20×\x201\x20column','rule':a0_0x3cf759(0x82),'exampleSqlShape':a0_0x3cf759(0x99),'exampleResponse':a0_0x3cf759(0x98)},{'appliesTo':a0_0x3cf759(0xa3),'rule':a0_0x3cf759(0xbd),'exampleSqlShape':a0_0x3cf759(0x89),'exampleResponse':'\x22trend\x22:\x20{\x20\x22direction\x22:\x20\x22up\x22,\x20\x22pct\x22:\x20\x222.2\x22\x20}'},{'appliesTo':a0_0x3cf759(0x71),'rule':'Return\x20as\x20array\x20of\x20objects\x20(no\x20collapse).','exampleSqlShape':a0_0x3cf759(0x8f),'exampleResponse':'\x22items\x22:\x20[{\x20\x22label\x22:\x20\x22Shoes\x22,\x20\x22value\x22:\x20\x227660\x22\x20},\x20...]'}],COMMON_WIDGET_PATTERNS=[{'id':'metric_donut_breakdown','name':'Metric\x20+\x20Donut\x20Breakdown','useCase':'Headline\x20metric\x20with\x20trend\x20chip\x20and\x20breakdown\x20across\x20categories.\x20Suitable\x20for\x20widgets\x20like\x20\x27Expected\x20Earnings\x27\x20that\x20show\x20total\x20value,\x20percentage\x20change,\x20and\x20per-category\x20contribution.','payloadShape':{'id':'<widget_id>','queries':{'value':'file:query/<path>/value.sql','trend':a0_0x3cf759(0xa8),'items':a0_0x3cf759(0x77)}},'sqlShapesPerKey':[{'key':a0_0x3cf759(0x90),'shape':a0_0x3cf759(0x96),'outputColumns':[a0_0x3cf759(0x90)],'collapseRule':a0_0x3cf759(0xc0)},{'key':a0_0x3cf759(0xbc),'shape':'1\x20row\x20×\x202\x20columns','outputColumns':['direction','pct'],'collapseRule':'object'},{'key':a0_0x3cf759(0x93),'shape':a0_0x3cf759(0x7f),'outputColumns':['label',a0_0x3cf759(0x90)],'collapseRule':'array\x20of\x20objects'}],'responseShape':{'value':'\x2269700\x22','trend':'{\x20\x22direction\x22:\x20\x22up\x22,\x20\x22pct\x22:\x20\x222.2\x22\x20}','items':'[{\x20\x22label\x22:\x20\x22Shoes\x22,\x20\x22value\x22:\x20\x227660\x22\x20},\x20{\x20\x22label\x22:\x20\x22Gaming\x22,\x20\x22value\x22:\x20\x222820\x22\x20},\x20{\x20\x22label\x22:\x20\x22Others\x22,\x20\x22value\x22:\x20\x2245257\x22\x20}]'},'referenceWidgetId':'expected_earnings','socNotes':'Frontend\x20determines\x20donut/pie\x20variant,\x20color\x20per\x20category,\x20and\x20label\x20order.\x20If\x20per-category\x20percentage\x20is\x20needed\x20for\x20the\x20donut\x20arc,\x20frontend\x20computes\x20it\x20from\x20items[i].value\x20/\x20sum(items[*].value).\x20No\x20need\x20to\x20send\x20\x27pct\x27\x20from\x20backend\x20unless\x20the\x20figure\x20is\x20a\x20stable\x20business\x20calculation\x20independent\x20of\x20visual\x20rendering.'},{'id':'metric_sparkline','name':'Metric\x20+\x20Sparkline','useCase':'Headline\x20metric\x20with\x20trend\x20chip\x20and\x20sparkline\x20mini-chart\x20for\x20short\x20windows\x20(7\x20days,\x2012\x20months,\x20etc.).\x20Suitable\x20for\x20widgets\x20like\x20\x27Average\x20Daily\x20Sales\x27.','payloadShape':{'id':'<widget_id>','queries':{'value':a0_0x3cf759(0xab),'trend':a0_0x3cf759(0xa8),'points':a0_0x3cf759(0xae)}},'sqlShapesPerKey':[{'key':'value','shape':'1\x20row\x20×\x201\x20column','outputColumns':[a0_0x3cf759(0x90)],'collapseRule':a0_0x3cf759(0xc0)},{'key':'trend','shape':'1\x20row\x20×\x202\x20columns','outputColumns':['direction',a0_0x3cf759(0x8d)],'collapseRule':'object'},{'key':a0_0x3cf759(0x7c),'shape':a0_0x3cf759(0x7f),'outputColumns':['period',a0_0x3cf759(0x90)],'collapseRule':'array\x20of\x20objects'}],'responseShape':{'value':a0_0x3cf759(0x74),'trend':a0_0x3cf759(0x76),'points':'[{\x20\x22period\x22:\x20\x222026-04-24\x22,\x20\x22value\x22:\x20\x221850\x22\x20},\x20...\x20]'},'referenceWidgetId':a0_0x3cf759(0x88),'socNotes':'Sparkline\x20libraries\x20(ApexCharts,\x20Chartist,\x20etc.)\x20typically\x20need\x20a\x20plain\x20number\x20array.\x20Frontend\x20maps\x20points.map(p\x20=>\x20p.value).\x20The\x20\x27period\x27\x20field\x20stays\x20for\x20tooltip\x20and\x20gap-resilience\x20against\x20missing\x20days.\x20Use\x20generate_series\x20in\x20SQL\x20to\x20ensure\x20consistent\x20row\x20count\x20even\x20for\x20days\x20with\x20no\x20transactions.'},{'id':a0_0x3cf759(0xc1),'name':a0_0x3cf759(0xa0),'useCase':'Headline\x20metric\x20with\x20trend\x20chip\x20and\x20progress\x20bar\x20against\x20a\x20period\x20target.\x20Suitable\x20for\x20widgets\x20like\x20\x27Orders\x20This\x20Month\x27.','payloadShape':{'id':'<widget_id>','queries':{'value':'file:query/<path>/current.sql','trend':a0_0x3cf759(0xa8),'target':'file:query/<path>/target.sql'}},'sqlShapesPerKey':[{'key':'value','shape':'1\x20row\x20×\x201\x20column','outputColumns':[a0_0x3cf759(0x7e)],'collapseRule':'scalar\x20primitive'},{'key':'trend','shape':a0_0x3cf759(0x86),'outputColumns':['direction','pct'],'collapseRule':a0_0x3cf759(0x70)},{'key':'target','shape':a0_0x3cf759(0x96),'outputColumns':['target'],'collapseRule':'scalar\x20primitive'}],'responseShape':{'value':'\x221836\x22','trend':a0_0x3cf759(0xb9),'target':'\x222884\x22'},'referenceWidgetId':a0_0x3cf759(0x8c),'socNotes':'Frontend\x20computes\x20to_goal\x20=\x20target\x20-\x20value\x20and\x20pct\x20=\x20round(value\x20/\x20target\x20*\x20100)\x20for\x20the\x20progress\x20bar.\x20Visual\x20width\x20is\x20presentational\x20and\x20must\x20NOT\x20live\x20in\x20the\x20backend\x20payload.\x20If\x20progress\x20involves\x20complex\x20business\x20rules\x20(e.g.\x20exclude\x20weekends,\x20prorated\x20workdays),\x20use\x20a\x20single\x20multi-column\x20query\x20so\x20\x27pct\x27\x20is\x20a\x20stable\x20business\x20fact\x20rather\x20than\x20visual\x20width.'}],NAMING_CONVENTION={'dashboardName':{'constraint':a0_0x3cf759(0xb7),'minLength':0x6,'maxLength':0x32,'regex':'^dash-[a-zA-Z0-9_-]+$','examples':['dash-sales','dash-inbound',a0_0x3cf759(0x6f)],'rationale':a0_0x3cf759(0xaa)}},URL_PATTERN={'method':a0_0x3cf759(0xc2),'path':'/api/{project}/{name}/dashboard','exampleFull':a0_0x3cf759(0x75),'requestBodyShape':{'params':'object\x20—\x20values\x20for\x20declared\x20params\x20(validated\x20against\x20params\x20contract;\x20missing\x20required\x20→\x20400,\x20type\x20mismatch\x20→\x20400)','widgets':'array<string>,\x20optional\x20—\x20subset\x20of\x20widget\x20IDs\x20to\x20execute.\x20Omit\x20to\x20execute\x20all\x20declared\x20widgets.'},'responseShape':{'envelope':'{\x20success:\x20boolean,\x20data:\x20{\x20<widgetId>:\x20<perWidgetResponse>,\x20...\x20}\x20}','perWidgetResponse':a0_0x3cf759(0x84)}},FILE_REFERENCE_CONVENTION={'format':a0_0x3cf759(0xb1),'pathRelativeTo':'payload\x20JSON\x20file\x20location','fileExtensionPolicy':'free;\x20.sql\x20recommended\x20for\x20editor\x20highlight','resolvedAt':a0_0x3cf759(0x81),'embedStrategy':'SQL\x20file\x20content\x20is\x20embedded\x20as\x20JavaScript\x20template\x20literal\x20inside\x20the\x20generated\x20module\x20file.\x20Runtime\x20performs\x20zero\x20disk\x20I/O\x20per\x20request\x20—\x20all\x20SQL\x20is\x20in\x20memory\x20after\x20module\x20load.','implication':'Updating\x20an\x20SQL\x20file\x20requires\x20regenerating\x20the\x20dashboard\x20module\x20(\x27codegen_create_dashboard\x27)\x20for\x20changes\x20to\x20take\x20effect.'},PLACEHOLDER_CONVENTION={'format':':paramName','regex':a0_0x3cf759(0x91),'regexNotes':a0_0x3cf759(0x9e),'scanScope':'All\x20widget\x20SQL\x20—\x20both\x20\x27query\x27\x20(singular)\x20and\x20every\x20\x27queries.<key>\x27.','constraint':a0_0x3cf759(0xbb),'exampleSql':'SELECT\x20*\x20FROM\x20stock_inbound\x20WHERE\x20EXTRACT(YEAR\x20FROM\x20inbound_date)\x20=\x20:year','exampleParamDeclaration':'{\x20\x22params\x22:\x20{\x20\x22year\x22:\x20{\x20\x22type\x22:\x20\x22number\x22,\x20\x22required\x22:\x20true\x20}\x20}\x20}'},CACHE_SPEC={'container':a0_0x3cf759(0x73),'optional':!![],'rationale':'Dashboard\x20endpoint\x20may\x20opt-in\x20to\x20Redis-based\x20cache.\x20Pattern\x20follows\x20processor\x20cache\x20(see\x20feat-cache.md).\x20Cache\x20scope\x20is\x20the\x20full\x20response\x20envelope;\x20one\x20cache\x20entry\x20per\x20(params\x20+\x20widgets[]\x20subset)\x20combination.','fields':[{'name':'enabled','type':'boolean','required':!![],'description':'Toggle\x20cache\x20feature\x20for\x20this\x20dashboard.'},{'name':'ttl','type':a0_0x3cf759(0xba),'required':![],'constraint':a0_0x3cf759(0x8e),'default':'inherits\x20CACHE_TTL\x20env','description':a0_0x3cf759(0x94)},{'name':a0_0x3cf759(0xb2),'type':a0_0x3cf759(0x72),'required':![],'default':'[]','description':'List\x20of\x20CRUD\x20table\x20names\x20that,\x20when\x20written,\x20will\x20trigger\x20invalidation\x20of\x20this\x20dashboard\x20cache.'}],'validation':{'sqlCrossReference':'When\x20cache.enabled\x20===\x20true\x20and\x20invalidates\x20is\x20non-empty:\x20validator\x20extracts\x20table\x20candidates\x20from\x20widget\x20SQL\x20(regex\x20FROM/JOIN),\x20cross-references\x20with\x20metadata/{project}.json\x20(endpoints[*].tableName\x20where\x20type\x20===\x20\x22module\x22),\x20and\x20asserts\x20equality\x20of\x20expected\x20vs\x20declared\x20sets.\x20Mismatches\x20are\x20reported\x20per\x20category\x20(missing,\x20extra,\x20unmatched).','errorOn':[a0_0x3cf759(0xc6),'Table\x20declared\x20in\x20invalidates,\x20but\x20not\x20detected\x20in\x20any\x20widget\x20SQL\x20(typo\x20or\x20dead\x20entry)'],'warningOn':[a0_0x3cf759(0xa9)]}},DOCUMENTATION_URL='https://restforge.dev/docs/server/query-data/dashboard',DASHBOARD_CATALOG={'schemaVersion':a0_0x3cf759(0x80),'source':'dashboard-catalog','summary':{'totalAllowedTopLevelFields':PAYLOAD_SHAPE[a0_0x3cf759(0xb5)][a0_0x3cf759(0xc5)],'totalForbiddenFrontendFields':FORBIDDEN_FRONTEND_FIELDS[a0_0x3cf759(0xc5)],'totalParamTypes':ALLOWED_PARAM_TYPES['length'],'totalScalarCollapseRules':SCALAR_COLLAPSE_RULES[a0_0x3cf759(0xc5)],'totalCommonWidgetPatterns':COMMON_WIDGET_PATTERNS['length']},'payloadShape':PAYLOAD_SHAPE,'widgetSpec':WIDGET_SPEC,'paramSpec':PARAM_SPEC,'scalarCollapseRules':SCALAR_COLLAPSE_RULES,'commonWidgetPatterns':COMMON_WIDGET_PATTERNS,'namingConvention':NAMING_CONVENTION,'urlPattern':URL_PATTERN,'fileReferenceConvention':FILE_REFERENCE_CONVENTION,'placeholderConvention':PLACEHOLDER_CONVENTION,'cacheSpec':CACHE_SPEC,'documentationUrl':DOCUMENTATION_URL};module[a0_0x3cf759(0xb0)]={'DASHBOARD_CATALOG':DASHBOARD_CATALOG};
@@ -1 +1 @@
1
- const a0_0x5de97d=a0_0x91b9;(function(_0x198f86,_0x4dfc9c){const _0x272159=a0_0x91b9,_0x5f1bae=_0x198f86();while(!![]){try{const _0x17c7c0=-parseInt(_0x272159(0x101))/0x1+-parseInt(_0x272159(0xf5))/0x2*(-parseInt(_0x272159(0xfb))/0x3)+parseInt(_0x272159(0xee))/0x4+-parseInt(_0x272159(0xf1))/0x5*(parseInt(_0x272159(0xef))/0x6)+parseInt(_0x272159(0xf4))/0x7*(-parseInt(_0x272159(0xf7))/0x8)+-parseInt(_0x272159(0xec))/0x9+parseInt(_0x272159(0xf8))/0xa;if(_0x17c7c0===_0x4dfc9c)break;else _0x5f1bae['push'](_0x5f1bae['shift']());}catch(_0x26e052){_0x5f1bae['push'](_0x5f1bae['shift']());}}}(a0_0x43ed,0x7c88e));const DB_CONNECTION_ENV_TEMPLATE=a0_0x5de97d(0xf6),REQUIRED_KEYS=new Set([a0_0x5de97d(0xf3),a0_0x5de97d(0xff),'SERVER_PORT','DB_TYPE','DB_HOST',a0_0x5de97d(0xe7),'DB_USER',a0_0x5de97d(0xe8),'DB_NAME']);function a0_0x91b9(_0x1b93ed,_0x121317){_0x1b93ed=_0x1b93ed-0xe4;const _0x43ed04=a0_0x43ed();let _0x91b9df=_0x43ed04[_0x1b93ed];if(a0_0x91b9['cOYVVq']===undefined){var _0x3bdc18=function(_0x43efc9){const _0x1d448c='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x44ac1a='',_0x58edcf='';for(let _0x112e28=0x0,_0x2ca93a,_0x59d462,_0x353874=0x0;_0x59d462=_0x43efc9['charAt'](_0x353874++);~_0x59d462&&(_0x2ca93a=_0x112e28%0x4?_0x2ca93a*0x40+_0x59d462:_0x59d462,_0x112e28++%0x4)?_0x44ac1a+=String['fromCharCode'](0xff&_0x2ca93a>>(-0x2*_0x112e28&0x6)):0x0){_0x59d462=_0x1d448c['indexOf'](_0x59d462);}for(let _0x1e01eb=0x0,_0x307d28=_0x44ac1a['length'];_0x1e01eb<_0x307d28;_0x1e01eb++){_0x58edcf+='%'+('00'+_0x44ac1a['charCodeAt'](_0x1e01eb)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x58edcf);};a0_0x91b9['OGBtzj']=_0x3bdc18,a0_0x91b9['rFTwcJ']={},a0_0x91b9['cOYVVq']=!![];}const _0x52e694=_0x43ed04[0x0],_0x4d7804=_0x1b93ed+_0x52e694,_0x476674=a0_0x91b9['rFTwcJ'][_0x4d7804];return!_0x476674?(_0x91b9df=a0_0x91b9['OGBtzj'](_0x91b9df),a0_0x91b9['rFTwcJ'][_0x4d7804]=_0x91b9df):_0x91b9df=_0x476674,_0x91b9df;}function parseTemplateAsSchema(_0x5e54c9){const _0x3114e8=a0_0x5de97d,_0x388d3a={'NMKer':function(_0x4e4974,_0x810017){return _0x4e4974===_0x810017;},'dQOSW':function(_0x3b8b69,_0x4f20cb){return _0x3b8b69>_0x4f20cb;},'tSMMy':function(_0xabff8f,_0x5917f1){return _0xabff8f===_0x5917f1;},'WwGoI':function(_0x310ff8,_0x8b5898){return _0x310ff8+_0x8b5898;},'Esiuy':function(_0x5a7a57,_0x34aac4){return _0x5a7a57===_0x34aac4;},'XjcKI':_0x3114e8(0x102),'fzAXQ':'false','VwSLo':_0x3114e8(0x103)},_0x5e95aa=_0x5e54c9||DB_CONNECTION_ENV_TEMPLATE,_0x954523=_0x5e95aa[_0x3114e8(0xeb)]('\x0a'),_0x4dcd46=[];let _0x33e4ec=null,_0xb1654b=[];for(const _0x42ed0b of _0x954523){const _0x3416b3=_0x42ed0b[_0x3114e8(0xe6)]();if(_0x388d3a[_0x3114e8(0xea)](_0x3416b3,'')){_0xb1654b=[];continue;}if(_0x3416b3[_0x3114e8(0xf9)]('#')){const _0x1fc721=_0x3416b3[_0x3114e8(0x104)](0x1)['trim'](),_0x30c191=_0x388d3a['dQOSW'](_0x1fc721[_0x3114e8(0xe9)],0x0)&&_0x1fc721[_0x3114e8(0xe9)]<0x3c&&!_0x1fc721[_0x3114e8(0xf2)](':')&&!/^[A-Z_]+=/['test'](_0x1fc721)&&/^[A-Z]/[_0x3114e8(0xed)](_0x1fc721);_0x30c191&&_0x388d3a[_0x3114e8(0xf0)](_0xb1654b['length'],0x0)?_0x33e4ec=_0x1fc721:_0xb1654b['push'](_0x1fc721);continue;}const _0x379f05=_0x42ed0b[_0x3114e8(0x100)]('=');if(_0x379f05>0x0){const _0x509cf9=_0x42ed0b['slice'](0x0,_0x379f05)['trim'](),_0x5352e1=_0x42ed0b['slice'](_0x388d3a['WwGoI'](_0x379f05,0x1));let _0x2d61bf='string';if(_0x388d3a[_0x3114e8(0xfd)](_0x5352e1,_0x388d3a[_0x3114e8(0xfa)])||_0x5352e1===_0x388d3a['fzAXQ'])_0x2d61bf=_0x388d3a[_0x3114e8(0xfc)];else/^-?\d+$/[_0x3114e8(0xed)](_0x5352e1)&&(_0x2d61bf=_0x3114e8(0x105));_0x4dcd46[_0x3114e8(0xe4)]({'name':_0x509cf9,'section':_0x33e4ec,'type':_0x2d61bf,'default':_0x5352e1,'description':_0xb1654b[_0x3114e8(0xfe)]('\x20')||null,'required':REQUIRED_KEYS['has'](_0x509cf9)}),_0xb1654b=[];}}return _0x4dcd46;}function a0_0x43ed(){const _0x124336=['vNDttg8','rxnPDxK','AM9PBG','u0vsvKvsx0ferfjfu1m','Aw5KzxHpzG','mtqZotmWr25UAhHV','Dhj1zq','yM9VBgvHBG','C2XPy2u','Aw50zwDLCG','ChvZAa','zxHWB3j0CW','DhjPBq','rejFue9sva','rejFueftu1DpuKq','BgvUz3rO','tK1lzxi','C3bSAxq','mZK1otKZn050AgDlCW','DgvZDa','mZK3mtK4mgTqyMHjtG','nK9lsK5Vzq','DfnntxK','mJa3ndq5nvjvCLDWua','Aw5JBhvKzxm','teLdru5trq','mtryyLvuyvG','mM1szeTdAG','iYbmAwnLBNnLcKXjq0vou0u9wfHywc1ywfHylvHywfGTwfHywaOkiYbtzxj2zxiku0vsvKvsx0ferfjfu1m9mti3lJaUmc4XcLnfuLzfuL9qt1juptmWmdakcImGtgL2zsbtEw5JicHxzwjtB2nRzxqPienVBMzPz3vYyxrPB24kiYbot1rfoIbmsvzfx1nztKnFru5bqKXfrd10CNvLihjLCxvPCMvZigfUiefqssblzxKGkeTfwt0UlI4PihrVigf1DgHLBNrPy2f0zsbxzwjtB2nRzxqGy2XPzw50CWPmsvzfx1nztKnFru5bqKXfrd1MywXZzqPmsvzfx1nztKnFue9svd0ZmdmZcGOJifjLzgLZienVBMzPz3vYyxrPB24kuKvesvnFse9tvd1SB2nHBgHVC3qkuKvesvnFue9svd02mZC5cLjfreLtx1bbu1nxt1jepqPsrurju19eqJ0WcGOJiev4Cg9YDcbdB25MAwD1CMf0Aw9UcKvyue9svf9gsuXfx0vyueLswt0ZnJaWmdaWcKvyue9svf9dsfvos19tsvPfpteWmdakcImGs2fMA2eGq29UzMLNDxjHDgLVBGPlquzlqv9ftKfctevepwzHBhnLcImGqNjVA2vYigXPC3qGkgnVBw1HlxnLCgfYyxrLzcbMB3iGBxvSDgLWBguGyNjVA2vYCZOGyNjVA2vYmtO5mdKYlgjYB2TLCJi6ota5mIXICM9RzxiZoJKWotiPcKTbrKTbx0jst0TfuLm9Bg9JywXOB3n0oJKWotikiYbdBgLLBNqGsuqGkg9WDgLVBMfSlcbKzwzHDwX0oIbYzxn0zM9Yz2uTE3bYB2PLy3r9lxbYB2r1y2vYic8GlwnVBNn1BwvYkqOJieTbrKTbx0nmsuvovf9jrd0ks0fgs0fFq09otKvdveLptL9usu1ft1vuptmWmdaks0fgs0fFuKvrvuvtvf9usu1ft1vupti1mdaWcKTbrKTbx1rpueLdx1bbvfrfuK49E21VzhvSzx0UE2vUzhbVAw50Fs5LDMvUDhmks0fgs0fFvevoqu5ux0LepwrLzMf1Bhqks0fgs0fFu0vtu0LptL9usu1ft1vuptmWmdaWcKTbrKTbx0HfqvjuqKvbvf9jtLrfuLzbtd0ZmdaWcKTbrKTbx01bwf9cwvrfu19qrvjFuefsveLusu9opteWndG1nZyks0fgs0fFqvvut19dt01nsvq9zMfSC2uks0fgs0fFqvvut19dt01nsvrFsu5urvjwquW9ntaWmaPlquzlqv9srvrswv9bvfrftvbuuZ0ZcKTbrKTbx1jfvfjzx0rftefzpteWmdaks0fgs0fFuKvuuLLFtufyx0rftefzptmWmdaWcKTbrKTbx1nttd1MywXZzqPlquzlqv9mt0DFtevwruW9Aw5MBWOJifnbu0WGqxv0AgvUDgLJyxrPB24Gkg9WDgLVBMfSlcb1BMnVBw1LBNqGAwyGDgHLigjYB2TLCIbYzxf1AxjLCYbHDxrOzw50AwnHDgLVBIKkiYbtDxbWB3j0zwqGBwvJAgfUAxnTCZOGCgXHAw4SihnJCMfTlxnOys0YntySihnJCMfTlxnOys01mtikiYblquzlqv9tqvnmx01fq0HbtKLttt1WBgfPBGOJieTbrKTbx1nbu0XFvvnfuK5btuu9cImGs0fgs0fFu0fttf9qqvntv09srd0kcImGrgf0ywjHC2uGq29UzMLNDxjHDgLVBGOJifn1ChbVCNrLzdOGCg9ZDgDYzxnXBcWGBxLZCwWSig9YywnSzsWGC3fSAxrLcKrcx1rzueu9Cg9ZDgDYzxnXBaPeqL9it1nupteYnY4WlJaUmqPeqL9qt1juptu0mZikrejFvvnfuJ1WB3n0z3jLCWPeqL9qqvntv09srd15B3vYx3bHC3n3B3jKx2HLCMukrejFtKfnrt15B3vYx2rHDgfIyxnLx25HBwukiYbgB3iGu1fmAxrLoIbZzxqGrejFvfLqrt1ZCwXPDguGyw5Kiercx05btuu9lI9KyxrHl215yxbWlMrIcImGrejFse9tvcWGrejFue9svcWGrejFvvnfuIWGrejFueftu1DpuKqGyxjLigLNBM9YzwqGzM9YifnrtgL0zqOkiYbmB2DNAw5NienVBMzPz3vYyxrPB24kte9hx0XfvKvmpwrLyNvNcKXpr19ut19gsuXfpxrYDwukcImGu1fmieXVz2DPBMCku1fmx0Xpr19ftKfctevepwzHBhnLcLnrtf9mt0DFtevwruW9zgvIDwCku1fmx0Xpr19qqvjbtvm9zMfSC2uku1fmx0Xpr19tte9xx1riuKvtse9mrd0XmdaWcGOJienHy2HLienVBMzPz3vYyxrPB24kq0fdsevFru5bqKXfrd1MywXZzqPdqunirv9uveW9mZaWcGOJiePVyIbty2HLzhvSzxiksK9cx0voqujmruq9zMfSC2uksK9cx0nptKnvuLjftKnzptuksK9cx1jfvevoveLptL9it1vsuZ03mGPkt0jFrKfjtevex1jfvevoveLptL9it1vsuZ0XnJGksK9cx1nivvret1Dox1rjtuvpvvq9mtaWmdaksK9cx1nuquXmrurFsu5urvjwquW9mZaWmdaksK9cx01bwf9tvefmtevex0npvu5uptikcImGrgLZDhjPyNv0zwqGtg9JAYbdB25MAwD1CMf0Aw9UcKXpq0TFreLtvfjjqLvururFru5bqKXfrd1MywXZzqPmt0nlx0rju1rssujvvevex1rutd0XmaPmt0nlx1jfu09vuKnfx01bwf9uveW9nJaWcKXpq0TFreLtvfjjqLvururFuKvuuLK9mWPmt0nlx0rju1rssujvvevex1jfvfjzx0rftefzpteWmaPmt0nlx0rju1rssujvvevex1nuuKfuruDzpxjLAMvJDaOkiYbjrcbhzw5LCMf0B3iGq29UzMLNDxjHDgLVBGPjreDftL9ftKfctevepwzHBhnLcKLer0vox0Leru1FvfrmptyWmaPjreDftL9dt1vovevsx1rutf9nt05useXzpti3nJq4mdaksurhru5Fq09vtLrfuL9uveXFrefjtfK9mtCYodaWcKLer0vox0rfrKfvtfrFtufyx1jfvfjzpteWcKLer0vox0rfrKfvtfrFueLox0rjr0LuuZ02cKLer0vox0rfrKfvtfrFu0vssufmx1bbvfrfuK49wfHywc1ywfHylvHywfGTwfHywaPjreDftL9eruzbvuXux0nprevFuefuvevstJ05otK5ltK5otKksurhru5FquXmt1DFuKvtrvq9zMfSC2uk','mJu1nJy4ofHIzuDYvG','mta5mJyZndbczfjKANq','C3rHCNrZv2L0Aa','wgPJs0K','mtG3mZC3CwTAEuji'];a0_0x43ed=function(){return _0x124336;};return a0_0x43ed();}module[a0_0x5de97d(0xe5)]={'DB_CONNECTION_ENV_TEMPLATE':DB_CONNECTION_ENV_TEMPLATE,'REQUIRED_KEYS':REQUIRED_KEYS,'parseTemplateAsSchema':parseTemplateAsSchema};
1
+ const a0_0x2ea007=a0_0x152e;(function(_0x2bf275,_0x1b318){const _0x11f39c=a0_0x152e,_0x2f97f4=_0x2bf275();while(!![]){try{const _0x38ba90=-parseInt(_0x11f39c(0x1da))/0x1*(-parseInt(_0x11f39c(0x1c5))/0x2)+-parseInt(_0x11f39c(0x1db))/0x3*(-parseInt(_0x11f39c(0x1ca))/0x4)+-parseInt(_0x11f39c(0x1ce))/0x5*(-parseInt(_0x11f39c(0x1d5))/0x6)+parseInt(_0x11f39c(0x1d1))/0x7*(-parseInt(_0x11f39c(0x1d6))/0x8)+parseInt(_0x11f39c(0x1d7))/0x9*(-parseInt(_0x11f39c(0x1c4))/0xa)+-parseInt(_0x11f39c(0x1c7))/0xb+parseInt(_0x11f39c(0x1d2))/0xc;if(_0x38ba90===_0x1b318)break;else _0x2f97f4['push'](_0x2f97f4['shift']());}catch(_0x2703e1){_0x2f97f4['push'](_0x2f97f4['shift']());}}}(a0_0x71dc,0x6d5d4));function a0_0x71dc(){const _0x519ed5=['mJHHtwDOrK4','ntmWodG5nKzzuvD3sq','DwjOCKq','ChvZAa','nLjlq3nHqq','mZqXotjdyMDkELi','mJyZodeWn2T0Efz1ra','zxHWB3j0CW','Aw50zwDLCG','ndmYnufNEefrra','mJC0mKLJtMfSza','mJbfuKHND3G','mJjzBhPlDM4','u0vsvKvsx1bpuLq','ndu1nZiZnhzZwfLova','Aw5JBhvKzxm','Aw5KzxHpzG','mZG2og5wufrhvG','C2XPy2u','BgvUz3rO','DhjPBq','ndu4odu1y0rOzefR','C3rHCNrZv2L0Aa','DgvZDa'];a0_0x71dc=function(){return _0x519ed5;};return a0_0x71dc();}const DB_CONNECTION_ENV_TEMPLATE='#\x20License\x0aLICENSE=XXXX-XXXX-XXXX-XXXX\x0a\x0a#\x20Server\x0aSERVER_ADDRESS=127.0.0.1\x0aSERVER_PORT=3000\x0a\x0a#\x20Live\x20Sync\x20(WebSocket)\x20Configuration\x0a#\x20NOTE:\x20LIVE_SYNC_ENABLED=true\x20requires\x20an\x20API\x20Key\x20(KEY=...)\x20to\x20authenticate\x20WebSocket\x20clients\x0aLIVE_SYNC_ENABLED=false\x0aLIVE_SYNC_PORT=3033\x0a\x0a#\x20Redis\x20Configuration\x0aREDIS_HOST=localhost\x0aREDIS_PORT=6379\x0aREDIS_PASSWORD=\x0aREDIS_DB=0\x0a\x0a#\x20Export\x20Configuration\x0aEXPORT_FILE_EXPIRY=3600000\x0aEXPORT_CHUNK_SIZE=1000\x0a\x0a#\x20Kafka\x20Configuration\x0aKAFKA_ENABLED=false\x0a#\x20Broker\x20list\x20(comma-separated\x20for\x20multiple\x20brokers:\x20broker1:9092,broker2:9092,broker3:9092)\x0aKAFKA_BROKERS=localhost:9092\x0a#\x20Client\x20ID\x20(optional,\x20default:\x20restforge-{project}-producer\x20/\x20-consumer)\x0a#\x20KAFKA_CLIENT_ID=\x0aKAFKA_CONNECTION_TIMEOUT=3000\x0aKAFKA_REQUEST_TIMEOUT=25000\x0aKAFKA_TOPIC_PATTERN={module}.{endpoint}.events\x0aKAFKA_TENANT_ID=default\x0aKAFKA_SESSION_TIMEOUT=30000\x0aKAFKA_HEARTBEAT_INTERVAL=3000\x0aKAFKA_MAX_BYTES_PER_PARTITION=1048576\x0aKAFKA_AUTO_COMMIT=false\x0aKAFKA_AUTO_COMMIT_INTERVAL=5000\x0aKAFKA_RETRY_ATTEMPTS=3\x0aKAFKA_RETRY_DELAY=1000\x0aKAFKA_RETRY_MAX_DELAY=30000\x0aKAFKA_SSL=false\x0aKAFKA_LOG_LEVEL=info\x0a#\x20SASL\x20Authentication\x20(optional,\x20uncomment\x20if\x20the\x20broker\x20requires\x20authentication)\x0a#\x20Supported\x20mechanisms:\x20plain,\x20scram-sha-256,\x20scram-sha-512\x0a#\x20KAFKA_SASL_MECHANISM=plain\x0a#\x20KAFKA_SASL_USERNAME=\x0a#\x20KAFKA_SASL_PASSWORD=\x0a\x0a#\x20Database\x20Configuration\x0a#\x20Supported:\x20postgresql,\x20mysql,\x20oracle,\x20sqlite\x0aDB_TYPE=postgresql\x0aDB_HOST=127.0.0.1\x0aDB_PORT=5432\x0aDB_USER=postgres\x0aDB_PASSWORD=your_password_here\x0aDB_NAME=your_database_name\x0a#\x20For\x20SQLite:\x20set\x20DB_TYPE=sqlite\x20and\x20DB_NAME=./data/myapp.db\x0a#\x20DB_HOST,\x20DB_PORT,\x20DB_USER,\x20DB_PASSWORD\x20are\x20ignored\x20for\x20SQLite\x0a\x0a#\x20Logging\x20Configuration\x0aLOG_LEVEL=debug\x0aLOG_TO_FILE=true\x0a\x0a#\x20SQL\x20Logging\x0aSQL_LOG_ENABLED=false\x0aSQL_LOG_LEVEL=debug\x0aSQL_LOG_PARAMS=false\x0aSQL_LOG_SLOW_THRESHOLD=1000\x0a\x0a#\x20Cache\x20Configuration\x0aCACHE_ENABLED=false\x0aCACHE_TTL=300\x0a\x0a#\x20Job\x20Scheduler\x0aJOB_ENABLED=false\x0aJOB_CONCURRENCY=5\x0aJOB_RETENTION_HOURS=72\x0aJOB_FAILED_RETENTION_HOURS=168\x0aJOB_SHUTDOWN_TIMEOUT=10000\x0aJOB_STALLED_INTERVAL=30000\x0aJOB_MAX_STALLED_COUNT=2\x0a\x0a#\x20Distributed\x20Lock\x20Configuration\x0aLOCK_DISTRIBUTED_ENABLED=false\x0aLOCK_DISTRIBUTED_TTL=10\x0aLOCK_RESOURCE_MAX_TTL=600\x0aLOCK_DISTRIBUTED_RETRY=3\x0aLOCK_DISTRIBUTED_RETRY_DELAY=100\x0aLOCK_DISTRIBUTED_STRATEGY=reject\x0a\x0a#\x20ID\x20Generator\x20Configuration\x0aIDGEN_ENABLED=false\x0aIDGEN_IDEM_TTL=600\x0aIDGEN_COUNTER_TTL_MONTHLY=2764800\x0aIDGEN_COUNTER_TTL_DAILY=172800\x0aIDGEN_DEFAULT_MAX_RETRY=10\x0aIDGEN_DEFAULT_PIN_DIGITS=6\x0aIDGEN_DEFAULT_SERIAL_PATTERN=XXXX-XXXX-XXXX-XXXX\x0aIDGEN_DEFAULT_CODE_PATTERN=9999-9999\x0aIDGEN_ALLOW_RESET=false\x0a',REQUIRED_KEYS=new Set(['LICENSE','SERVER_ADDRESS',a0_0x2ea007(0x1c6),'DB_TYPE','DB_HOST','DB_PORT','DB_USER','DB_PASSWORD','DB_NAME']);function a0_0x152e(_0xbcdf7,_0x5b9d34){_0xbcdf7=_0xbcdf7-0x1c4;const _0x71dc2=a0_0x71dc();let _0x152e60=_0x71dc2[_0xbcdf7];if(a0_0x152e['cHooUT']===undefined){var _0x3bb8a5=function(_0x47a89c){const _0x581abd='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x422273='',_0x473350='';for(let _0x98bcd4=0x0,_0x5e1110,_0x479db,_0x1cbb78=0x0;_0x479db=_0x47a89c['charAt'](_0x1cbb78++);~_0x479db&&(_0x5e1110=_0x98bcd4%0x4?_0x5e1110*0x40+_0x479db:_0x479db,_0x98bcd4++%0x4)?_0x422273+=String['fromCharCode'](0xff&_0x5e1110>>(-0x2*_0x98bcd4&0x6)):0x0){_0x479db=_0x581abd['indexOf'](_0x479db);}for(let _0x2f8676=0x0,_0x5a2b80=_0x422273['length'];_0x2f8676<_0x5a2b80;_0x2f8676++){_0x473350+='%'+('00'+_0x422273['charCodeAt'](_0x2f8676)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x473350);};a0_0x152e['BBJRrb']=_0x3bb8a5,a0_0x152e['wvEvTt']={},a0_0x152e['cHooUT']=!![];}const _0x51ae5a=_0x71dc2[0x0],_0x517030=_0xbcdf7+_0x51ae5a,_0x5df69c=a0_0x152e['wvEvTt'][_0x517030];return!_0x5df69c?(_0x152e60=a0_0x152e['BBJRrb'](_0x152e60),a0_0x152e['wvEvTt'][_0x517030]=_0x152e60):_0x152e60=_0x5df69c,_0x152e60;}function parseTemplateAsSchema(_0xe53828){const _0x5c931=a0_0x2ea007,_0x2a17f3={'ubhrD':function(_0x1cd036,_0x3c5261){return _0x1cd036||_0x3c5261;},'syELe':function(_0x5b1f04,_0x3d6950){return _0x5b1f04===_0x3d6950;},'pFLxW':'string','JEXwn':_0x5c931(0x1d9)},_0xb5709=_0x2a17f3[_0x5c931(0x1d3)](_0xe53828,DB_CONNECTION_ENV_TEMPLATE),_0x2c4a9c=_0xb5709['split']('\x0a'),_0x226c95=[];let _0x3d02af=null,_0x3813dd=[];for(const _0x14684f of _0x2c4a9c){const _0xce2c0e=_0x14684f[_0x5c931(0x1cd)]();if(_0xce2c0e===''){_0x3813dd=[];continue;}if(_0xce2c0e[_0x5c931(0x1cf)]('#')){const _0x3662dd=_0xce2c0e['slice'](0x1)[_0x5c931(0x1cd)](),_0x289ba6=_0x3662dd[_0x5c931(0x1cc)]>0x0&&_0x3662dd[_0x5c931(0x1cc)]<0x3c&&!_0x3662dd[_0x5c931(0x1c8)](':')&&!/^[A-Z_]+=/[_0x5c931(0x1d0)](_0x3662dd)&&/^[A-Z]/['test'](_0x3662dd);_0x289ba6&&_0x2a17f3['syELe'](_0x3813dd['length'],0x0)?_0x3d02af=_0x3662dd:_0x3813dd['push'](_0x3662dd);continue;}const _0x5e9981=_0x14684f[_0x5c931(0x1c9)]('=');if(_0x5e9981>0x0){const _0x41efb4=_0x14684f[_0x5c931(0x1cb)](0x0,_0x5e9981)['trim'](),_0x361cd7=_0x14684f['slice'](_0x5e9981+0x1);let _0x33c94a=_0x2a17f3['pFLxW'];if(_0x2a17f3['syELe'](_0x361cd7,'true')||_0x361cd7==='false')_0x33c94a='boolean';else/^-?\d+$/['test'](_0x361cd7)&&(_0x33c94a=_0x2a17f3['JEXwn']);_0x226c95[_0x5c931(0x1d4)]({'name':_0x41efb4,'section':_0x3d02af,'type':_0x33c94a,'default':_0x361cd7,'description':_0x3813dd['join']('\x20')||null,'required':REQUIRED_KEYS['has'](_0x41efb4)}),_0x3813dd=[];}}return _0x226c95;}module[a0_0x2ea007(0x1d8)]={'DB_CONNECTION_ENV_TEMPLATE':DB_CONNECTION_ENV_TEMPLATE,'REQUIRED_KEYS':REQUIRED_KEYS,'parseTemplateAsSchema':parseTemplateAsSchema};
@@ -1 +1 @@
1
- 'use strict';function a0_0x4142(_0x283d5b,_0x1e7f34){_0x283d5b=_0x283d5b-0x1d5;const _0x2048e8=a0_0x2048();let _0x414206=_0x2048e8[_0x283d5b];if(a0_0x4142['iFpwGZ']===undefined){var _0x14b98f=function(_0x580ebd){const _0x47cf31='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x439ffb='',_0x58fdd8='';for(let _0x326ef3=0x0,_0x23582b,_0x3b8ffe,_0x509933=0x0;_0x3b8ffe=_0x580ebd['charAt'](_0x509933++);~_0x3b8ffe&&(_0x23582b=_0x326ef3%0x4?_0x23582b*0x40+_0x3b8ffe:_0x3b8ffe,_0x326ef3++%0x4)?_0x439ffb+=String['fromCharCode'](0xff&_0x23582b>>(-0x2*_0x326ef3&0x6)):0x0){_0x3b8ffe=_0x47cf31['indexOf'](_0x3b8ffe);}for(let _0x3b29dc=0x0,_0x571d07=_0x439ffb['length'];_0x3b29dc<_0x571d07;_0x3b29dc++){_0x58fdd8+='%'+('00'+_0x439ffb['charCodeAt'](_0x3b29dc)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x58fdd8);};a0_0x4142['Mbkjar']=_0x14b98f,a0_0x4142['vhKQba']={},a0_0x4142['iFpwGZ']=!![];}const _0x15fa02=_0x2048e8[0x0],_0x5126d5=_0x283d5b+_0x15fa02,_0x412bfb=a0_0x4142['vhKQba'][_0x5126d5];return!_0x412bfb?(_0x414206=a0_0x4142['Mbkjar'](_0x414206),a0_0x4142['vhKQba'][_0x5126d5]=_0x414206):_0x414206=_0x412bfb,_0x414206;}const a0_0xb18477=a0_0x4142;function a0_0x2048(){const _0x146c8c=['rgf0zsbHBMqGDgLTzsaOveLnrvnuqu1qks4','ouXIsKnTqW','qM9VBgvHBIb2ywX1zsaOBMf0AxzLiejpt0Xfqu4GB24Gug9ZDgDYzvnrtcWGvKfsq0HbuIbVBIbVDgHLCNmPlG','zgvJAw1HBdOXnsWYig5VDg51BgWGzgvMyxvSDdOW','zgvMyxvSDdO8BgL0zxjHBd4GFcbKzwzHDwX0oIC8C3rYAw5NpICGFcbKzwzHDwX0oJXJB25ZDgfUDd4GFcbKzwzHDwX0oJXMDw5JDgLVBJ4Okq','C3rHBMrHBg9Uzq','wwvZiokaLcbKyNnJAgvTytPPBML0igvTAxrZigfSBca0ignVBhvTBNmGyNKGzgvMyxvSDc4GvgfIBgvZihrOyxqGzg8GBM90ig5LzwqGyxvKAxqGkgXVB2T1Ccb0ywjSzxmSihn5C3rLBsb0ywjSzxmPignHBIbTyw51ywXSEsbYzw1VDMuGDgHLC2uGzMLLBgrZigzYB20GDgHLigDLBMvYyxrLzcbZA2vSzxrVBI4','uMvJB3jKignYzwf0Aw9UihrPBwvZDgfTCc4Gqxv0BY1ZzxqGB24Gsu5trvjuihzPysberuzbvuXuig5VDYGPlIboyxrPDMuGzNvUy3rPB24GDhjHBNnSyxrLCYbWzxiGzgLHBgvJDcaOCg9ZDgDYzxm6ienvuLjftLrFveLnrvnuqu1qlcbTExnXBdOGtK9xkcKSig9YywnSztOGu1LtveLnrvnuqu1qlcbZCwXPDgu6ienvuLjftLrFveLnrvnuqu1qks4','mJu4D1Dgwfbc','EYbMAwvSzdOGiMrPC2nVDw50iIWGBhq6ideWmcb9','vevyva','B25vCgrHDgu','t2jQzwn0ig9MigzPzwXKigrLzMLUAxrPB25ZigLUihnOB3j0AgfUzcbZDhjPBMCGzM9YBwf0lG','y2HLy2TZ','rgvMyxvSDcb0Aw1LEM9UzsbIzwHHDMLVCIbPCYbKAwfSzwn0lxnWzwnPzMLJlG','revquKvdqvrfrdOGvgHPCYbTB2rPzMLLCIbUBYbSB25NzxiGAgfZigz1BMn0Aw9UywWGzwzMzwn0igf0ihrOzsbereWGB3iGCNvUDgLTzsbSzxzLBc4Gqxv0BY11CgrHDguGzM9YihvWzgf0zwrFyxqGAxmGAgfUzgXLzcbIEsb0AguGuKrgigXHEwvYicHHDwrPDenVBhvTBNmGy29UDMvUDgLVBIbPBIbcyxnLtw9KzwWGCNvUDgLTzsKGyMfZzwqGB24GzMLLBgqGBMfTAw5NignVBNzLBNrPB24Sig5VDcbIEsb0AgLZifnerIbTyxjRzxiUiev4Axn0Aw5NihvZywDLigLZihbYzxnLCNzLzcbMB3iGyMfJA3DHCMqGy29TCgf0AwjPBgL0EsbIDxqGC2HVDwXKigjLihjLBw92zwqGzNjVBsbUzxCGDgvTCgXHDgvZlG','C3rYAw5NoJm2igzRoMnHDgvNB3j5lMLK','rgvMyxvSDcb2ywX1zs4Gu2LUz2XLlxf1B3rLzcbMB3iGC3rYAw5NCYWGCMf3igzVCIbUDw1LCMLJl2jVB2XLyw4SigjHCMuGAwrLBNrPzMLLCIbMB3iGu1fmignVBNn0yw50CYWGAwrLBNrPzMLLCIGPigzVCIbUyxrPDMuGzNvUy3rPB24Gy2fSBhmU','EYb0ExbLoIaIAgfZtwfUEsiSihrHCMDLDdOGiM9YzgvYx2L0zw0IlcbSB2nHBeTLEtOGiM9YzgvYx2LKiIWGCMvMzxjLBMnLCZOGiM9YzgvYx2LKiIb9','vMfSDwuGBxvZDcbIzsbVBMuGB2yGysbSAxn0lG','rgvMzxiGy29UC3rYywLUDcbJAgvJAYaOC2vTyw50AwnHBgX5ihnPBwLSyxiGDg8GCMvZDhjPy3qGB24GBw9ZDcbKAwfSzwn0CYKU','uMvQzwn0igrLBgv0zs91CgrHDguGD2HLBIbJAgLSzcbYB3DZigv4Axn0lG','tK9uie5vteWGy29UC3rYywLUDc4','C3rYAw5NoJmYihvUAxf1zq','yxv0B1vWzgf0zq','C3rYAw5NoJy0igLUzgv4','tM90igvXDwfSihrVlG','veLnrvnuqu1qierfrKfvtfqGpg5HDgL2zv9UB3CGCgvYigrPywXLy3q+','u3rHBMrHBg9UzsbJB25ZDhjHAw50CYaOCgSSig5VDg51BgWSihvUAxf1zsWGAw5KzxGPihrHA2uGBM8GDMfSDwuUifrOzsbSzwDHy3KGyxv0B1vWzgf0zsb0B2TLBIbPCYbZDgLSBcbWyxjZzwqGzM9YigjHy2T3yxjKignVBxbHDgLIAwXPDhKGyNv0igLZigrLChjLy2f0zwqGyw5KigHHCYbUBYbMDw5JDgLVBMfSigvMzMvJDc4','DgLTzxn0yw1W','mJHXsLHRA0m','DgLTzxn0yw1WigrLzMf1Bhq6BM93kcK','BhrL','rgf0ywjHC2uGC2nOzw1Hig5HBwvZCgfJzsaOBxvSDgKTC2nOzw1HigrHDgfIyxnLks4','twfYA3mGzMLLBgqGyxmGChjPBwfYEsbRzxKU','q29UC3rYywLUDcbUyw1LCYbHDxrVlwDLBMvYyxrLzcb3AxrOihr5CguTC3bLy2LMAwmGChjLzML4icHWAYWGzMSSigLKEcWGDxeSignRksbHBMqGzgLHBgvJDc1ZCgvJAwzPyYbTyxGGBgvUz3rOlG','nJqTyML0ihnPz25LzcbPBNrLz2vYicHcsuDjtLqPlG','EYbPzdOGiNn0CMLUzZOZnIbWAYiSig5HBwu6icjZDhjPBMC6mJu1ig5VDg51BgWIih0','Dw5PCxvLCW','tuq1igHHC2GGoc1JAgfYywn0zxiGC3vMzML4ihDOzw4GBMfTzsbLEgnLzwrZig1HEeXLBMD0Aa','B25ezwXLDgu','otKXnhvNqNjuyW','Ahr0Chm6lY9Yzxn0zM9Yz2uUzgv2l2rVy3mVy2XPl3nJAgvTys1KzwzPBML0Aw9U','nZeZmZe0n1Lou2H2AW','C3rYAw5NoJeWmcbKzwzHDwX0oIDWzw5KAw5NjW','phr5Cgu+wZO8Bw9KAwzPzxi+xsbBpgnVBNn0CMfPBNq+wZO8DMfSDwu+xv0UlI4','mtK4ntuXmZbQqMLduvy','mJCZmde1Du9qywPk','Bg9JywXlzxK','C3rYAw5NoJeWma','rM9YignVBxbVC2L0zsbPBMrLEgvZl3vUAxf1zxmSig5HBwuGAxmGzgvYAxzLzcbIEsbQB2LUAw5NignVBhvTBNmGD2L0Acb1BMrLCNnJB3jLigfUzcb0CNvUy2f0Aw5NihDOzw4GBMvLzgvKlG','DhLWzq','t25LlxrVlw9UzsaOAw52zxjZzsbVzIbIzwXVBMDZvg8PlIbpChrPB25HBcbPBNzLCNnLigrLy2XHCMf0Aw9Uig9UihrOzsbWyxjLBNqGC2LKztSGzg9LCYbUB3qGywzMzwn0ieretc4','zxHWB3j0CW','tgvZCYb0AgfUig9YigvXDwfSihrVlG','EYbJyxrLz29YEtOGEYb0ExbLoIaIyMvSB25NC1rViIWGCMvMzxjLBMnLCZOGiMnHDgvNB3j5kgLKksiGFsb9','DxbKyxrLzf9HDa','y3jLyxrLzf9HDa','rKSGzMLLBgqGBxvZDcbIzsbUDwXSywjSzs4','rgf0zsbVBMX5icHUBYb0Aw1LignVBxbVBMvUDcKU','B2jQzwn0','zgvJAw1HBdOXnsWY','zMS6phrHyMXLpI48y29SDw1UpG','rM9YzwLNBIbRzxKGDxnLCYbKB3qGBM90yxrPB246igzRoJX0ywjSzt4UpgnVBhvTBJ4GkguUzY4SigzRoMnHDgvNB3j5lMLKks4GvgHLihbHCNnLCIbYzwPLy3rZihbHCMvUDgHLC2vZihn5BNrHEc4','BxLZCwWY','r3jLyxrLCIb0AgfUig9YigvXDwfSihrVlG','vKfsq0HbuIG2ncKGD2L0AcbZAw5NBguTy29SDw1UigLUzgv4','mJiWntK5mhzguML1DG','DMfSDwu','vMfSDwuGy29UC3rYywLUDhmGkgrLzMf1BhqSigzRksbYzxf1AxjLignVBNn0CMfPBNq6DMfSDwuGzM9YBwf0lG','mJG0mJq5nLHwCu5uCa','yhjLzMvYzw5JzxnGigLZihrOzsbIyxjLignVBhvTBIbUyw1LigLUihrOzsb0yxjNzxqGDgfIBguUifrHCMDLDcb0ywjSzsbPCYbHDxrVlwrLCML2zwqGzNjVBsb0AguGCMvSyxrPB24GA2v5ig5HBwuSig9Yig92zxjYAwrLihDPDgGGyhrHCMDLDgaU','DgfYz2v0','yxjYyxK','qxv0BY1KzxrLy3rLzcbMCM9TigzPzwXKihnOB3j0AgfUzcaICgSIihDOzw4GBM90ihnWzwnPzMLLzc4','vKfsq0HbuIGZnIKGufjjtufswsblrvK','uhjLy2LZAw9UigfUzcbZy2fSzsbYzxf1AxjLzcbMB3iGy3jVC3mTzgLHBgvJDcbWB3j0ywjPBgL0Es4','yMLNAw50','EYb0ExbLoIaIyMvSB25NC1rViIWGBg9JywXlzxK6icjJyxrLz29YEv9PzciSihjLzMvYzw5Jzxm6icjJyxrLz29YEv9PzciSig9UrgvSzxrLoIaICMvZDhjPy3qIih0','Dgv4Da','yM9VBgvHBIbKzwzHDwX0oNrYDwu','Aw50zwDLCG','u3rYAw5NihnOB3j0AgfUzcbMB3iGzMLLBgqGzgvMAw5PDgLVBI4GvhLWzsbYzxf1AxjLzcbHDcbZDgfYDcWGzM9SBg93zwqGyNKGB3b0Aw9UywWGBw9KAwzPzxiGyw5KihnWywnLlxnLCgfYyxrLzcbJB25ZDhjHAw50CY4','vKfsq0HbuIGXmdaPierfrKfvtfqGj3bLBMrPBMCN','EYbMAwvSzdOGiNf0EsiSigD0ztOGmcb9','BM9by3rPB24','y3jLyxrLzf9IEq','C2v0tNvSBa','vKfsq0HbuJiGD2L0AcbdsevdsYbJB25ZDhjHAw50','u2LUz2XLlwnVBhvTBIb1BMLXDwuGy29UC3rYywLUDc4','qK9ptevbtIberuzbvuXuifrsvuu','BxLZCwW','ChjPBwfYEuTLEq','BgvUz3rO','vvvjrc4GtMf0AxzLifvvsuqGB24Gug9ZDgDYzvnrtcWGvKfsq0HbuIGZnIKGB24GB3rOzxjZlG','EYbMAwvSzdOGiMrPC2nVDw50iIWGBhrLoIaXmdaGFq','tM9UlxvUAxf1zsbPBMrLEgvZlIbfywnOigvUDhj5igLZigfYCMf5ig9MignVBhvTBIbUyw1LCYbVCIbVyMPLy3qGEYbUyw1LlcbJB2X1Bw5Zih0U','C2nOzw1H','mJu0otC5q0TstwfV','DxbKyxrLzf9IEq','mZiTyML0ihnPz25LzcbPBNrLz2vYicHjtLqVsu5uruDfuIKU','CMvMzxjLBMnLCW'];a0_0x2048=function(){return _0x146c8c;};return a0_0x2048();}(function(_0x33cbc2,_0x27e63e){const _0x354ab6=a0_0x4142,_0x24b695=_0x33cbc2();while(!![]){try{const _0x12f86c=-parseInt(_0x354ab6(0x1f0))/0x1*(-parseInt(_0x354ab6(0x211))/0x2)+parseInt(_0x354ab6(0x1e4))/0x3+parseInt(_0x354ab6(0x206))/0x4*(-parseInt(_0x354ab6(0x217))/0x5)+parseInt(_0x354ab6(0x22b))/0x6+parseInt(_0x354ab6(0x213))/0x7+parseInt(_0x354ab6(0x22e))/0x8+-parseInt(_0x354ab6(0x1e9))/0x9*(parseInt(_0x354ab6(0x216))/0xa);if(_0x12f86c===_0x27e63e)break;else _0x24b695['push'](_0x24b695['shift']());}catch(_0x7f8d39){_0x24b695['push'](_0x24b695['shift']());}}}(a0_0x2048,0xb4373));const SCHEMA_VERSION='1.0',SOURCE='dbschema-catalog',DOCUMENTATION_URL=a0_0xb18477(0x212),DEFINE_MODEL_OPTIONS=[{'name':a0_0xb18477(0x1e3),'required':![],'type':'string|null','description':a0_0xb18477(0x209),'notes':'null\x20or\x20empty\x20string\x20means\x20default\x20schema.'},{'name':'fields','required':!![],'type':a0_0xb18477(0x224),'description':a0_0xb18477(0x1f4),'example':a0_0xb18477(0x20d)},{'name':a0_0xb18477(0x1de),'required':![],'type':'string|array','description':'Override\x20primary\x20key.\x20Single\x20field\x20name\x20or\x20composite\x20array.','notes':a0_0xb18477(0x232)},{'name':'relations','required':![],'type':'object','description':'Inter-table\x20relations\x20(belongsTo,\x20hasOne,\x20hasMany).','example':a0_0xb18477(0x21f)},{'name':'indexes','required':![],'type':'array','description':a0_0xb18477(0x1e2),'example':'[[\x22status\x22,\x20\x22created_at\x22]]'},{'name':a0_0xb18477(0x20e),'required':![],'type':'array','description':'Unique\x20constraints\x20(single\x20or\x20composite).','example':'[[\x22category_code\x22],\x20[\x22tenant_id\x22,\x20\x22category_code\x22]]'},{'name':a0_0xb18477(0x1f5),'required':![],'type':'array','description':'CHECK\x20constraints.\x20Each\x20entry\x20uses\x20operator-as-key\x20shape:\x20{\x20name?,\x20field,\x20<operator>:\x20<value>\x20}\x20where\x20<operator>\x20is\x20one\x20of\x20checkOperations\x20(in,\x20eq,\x20neq,\x20gt,\x20gte,\x20lt,\x20lte).\x20The\x20internal\x20IR\x20shape\x20after\x20ir-builder\x20normalization\x20is\x20{\x20name?,\x20field,\x20op,\x20value\x20},\x20but\x20that\x20is\x20NOT\x20the\x20user\x20input\x20format.','example':'[{\x20field:\x20\x22status\x22,\x20in:\x20[\x22active\x22,\x20\x22inactive\x22]\x20},\x20{\x20field:\x20\x22qty\x22,\x20gte:\x200\x20}]'}],FIELD_TYPES=[{'name':'string','description':'Variable-length\x20text.\x20Length\x20modifier\x20required.','requiresModifier':!![],'modifierFormat':'string:<length>','example':'string:255','notes':'Explicit\x20length\x20required\x20for\x20cross-dialect\x20portability.'},{'name':a0_0xb18477(0x237),'description':'Long-form\x20text\x20without\x20length\x20limit\x20(TEXT/CLOB).','requiresModifier':![],'example':a0_0xb18477(0x237)},{'name':a0_0xb18477(0x239),'description':a0_0xb18477(0x1e6),'requiresModifier':![],'example':'integer'},{'name':'bigint','description':a0_0xb18477(0x20c),'requiresModifier':![],'example':a0_0xb18477(0x235)},{'name':'decimal','description':'Fixed-point\x20decimal.\x20Precision\x20and\x20scale\x20required.','requiresModifier':!![],'modifierFormat':'decimal:<precision>,<scale>','example':a0_0xb18477(0x225),'notes':a0_0xb18477(0x234)},{'name':'boolean','description':a0_0xb18477(0x1ea),'requiresModifier':![],'example':'boolean'},{'name':'date','description':a0_0xb18477(0x223),'requiresModifier':![],'example':'date'},{'name':'timestamp','description':a0_0xb18477(0x1e8),'requiresModifier':![],'example':'timestamp','notes':a0_0xb18477(0x1f6)},{'name':'uuid','description':a0_0xb18477(0x1e0),'requiresModifier':![],'example':'uuid'},{'name':'json','description':'JSON\x20column\x20(JSONB\x20on\x20PostgreSQL,\x20JSON\x20on\x20MySQL,\x20CLOB\x20on\x20Oracle).','requiresModifier':![],'example':'json'}],CONSTRAINTS=[{'name':'pk','kind':a0_0xb18477(0x1ed),'description':a0_0xb18477(0x20a),'example':'string:36\x20pk'},{'name':'notnull','kind':'standalone','description':a0_0xb18477(0x1fe),'example':'string:255\x20notnull'},{'name':'unique','kind':'standalone','description':a0_0xb18477(0x1db),'example':a0_0xb18477(0x1ff)},{'name':'index','kind':a0_0xb18477(0x1ed),'description':'Single-column\x20non-unique\x20index.','example':'string:64\x20index'},{'name':a0_0xb18477(0x200),'kind':a0_0xb18477(0x1ed),'deprecated':!![],'description':a0_0xb18477(0x1f7),'example':'timestamp','notes':'Engine\x20still\x20parses\x20this\x20token\x20(backward\x20compatibility),\x20but\x20DDL\x20output\x20is\x20identical\x20to\x20plain\x20\x27timestamp\x27.\x20Use\x20\x27timestamp\x27\x20alone.\x20The\x20actual\x20auto-update\x20behavior\x20is\x20documented\x20in\x20the\x20RDF\x20auditColumns\x20catalog\x20(field-validation:catalog).'},{'name':'default','kind':a0_0xb18477(0x22c),'description':a0_0xb18477(0x1f9),'valueFormat':a0_0xb18477(0x1ec),'example':a0_0xb18477(0x238)},{'name':'fk','kind':'value','description':'Foreign\x20key\x20reference.\x20Auto-generates\x20a\x20belongsTo\x20relation\x20entry.','valueFormat':a0_0xb18477(0x226),'example':'string:36\x20fk:category.id','notes':'Cannot\x20coexist\x20with\x20explicit\x20relation\x20entry\x20referencing\x20the\x20same\x20field.'}],RELATION_TYPES=[{'name':'belongsTo','description':'Many-to-one.\x20This\x20table\x20holds\x20the\x20FK\x20column\x20referencing\x20the\x20parent\x20table.','requiredFields':[a0_0xb18477(0x21b),'localKey',a0_0xb18477(0x1e7)],'optionalFields':[a0_0xb18477(0x230),a0_0xb18477(0x210),a0_0xb18477(0x1f3)],'example':a0_0xb18477(0x236),'notes':a0_0xb18477(0x22f)},{'name':'hasOne','description':a0_0xb18477(0x21c),'requiredFields':['type','localKey',a0_0xb18477(0x1e7)],'optionalFields':[a0_0xb18477(0x230),'onDelete','onUpdate'],'example':'{\x20type:\x20\x22hasOne\x22,\x20target:\x20\x22profile\x22,\x20localKey:\x20\x22user_id\x22,\x20references:\x20\x22user_id\x22\x20}'},{'name':'hasMany','description':'One-to-many\x20(inverse\x20of\x20belongsTo).\x20Optional\x20inverse\x20declaration\x20on\x20the\x20parent\x20side;\x20does\x20not\x20affect\x20DDL.','requiredFields':[a0_0xb18477(0x21b),a0_0xb18477(0x218),a0_0xb18477(0x1e7)],'optionalFields':[a0_0xb18477(0x230),a0_0xb18477(0x210),a0_0xb18477(0x1f3)],'example':a0_0xb18477(0x1fa)}],REFERENTIAL_ACTIONS=[{'name':'cascade','description':'Cascade\x20delete/update\x20to\x20child\x20rows.','appliesTo':[a0_0xb18477(0x210),'onUpdate']},{'name':'restrict','description':a0_0xb18477(0x1fd),'appliesTo':['onDelete','onUpdate']},{'name':a0_0xb18477(0x1d9),'description':'Set\x20FK\x20column\x20to\x20NULL\x20on\x20child\x20rows.','appliesTo':[a0_0xb18477(0x210),a0_0xb18477(0x1f3)],'notes':a0_0xb18477(0x222)},{'name':a0_0xb18477(0x1d7),'description':a0_0xb18477(0x1fc),'appliesTo':['onDelete','onUpdate']}],CHECK_OPERATIONS=[{'name':'in','description':a0_0xb18477(0x1fb),'valueType':a0_0xb18477(0x231),'example':'{\x20field:\x20\x22status\x22,\x20in:\x20[\x22active\x22,\x20\x22inactive\x22]\x20}'},{'name':'eq','description':'Equal\x20to.','valueType':'scalar','example':'{\x20field:\x20\x22type\x22,\x20eq:\x20\x22user\x22\x20}'},{'name':'neq','description':a0_0xb18477(0x202),'valueType':'scalar','example':'{\x20field:\x20\x22type\x22,\x20neq:\x20\x22system\x22\x20}'},{'name':'gt','description':'Greater\x20than.','valueType':'numeric','example':'{\x20field:\x20\x22qty\x22,\x20gt:\x200\x20}'},{'name':'gte','description':a0_0xb18477(0x229),'valueType':'numeric','example':a0_0xb18477(0x1d6)},{'name':'lt','description':'Less\x20than.','valueType':'numeric','example':a0_0xb18477(0x1f1)},{'name':a0_0xb18477(0x208),'description':a0_0xb18477(0x21e),'valueType':'numeric','example':a0_0xb18477(0x1e1)}],AUDIT_COLUMNS={'description':'Standard\x204-column\x20audit\x20convention\x20for\x20tables\x20managed\x20by\x20RESTForge.\x20The\x20same\x20convention\x20is\x20documented\x20in\x20field-validation:catalog\x20(auditColumns\x20section)\x20for\x20the\x20RDF/backend\x20layer.\x20SDF\x20and\x20RDF\x20stay\x20aligned\x20when\x20both\x20declare\x20these\x204\x20columns.','columns':[{'name':a0_0xb18477(0x221),'shorthand':a0_0xb18477(0x207),'nullable':!![],'purpose':a0_0xb18477(0x1ef)},{'name':a0_0xb18477(0x1d8),'shorthand':a0_0xb18477(0x219),'nullable':!![],'purpose':'User\x20identifier\x20(username,\x20user_id,\x20or\x20composite\x20token)\x20of\x20the\x20record\x20creator.\x20Set\x20by\x20application\x20layer\x20on\x20INSERT,\x20not\x20by\x20database\x20default.'},{'name':a0_0xb18477(0x220),'shorthand':a0_0xb18477(0x205),'nullable':!![],'purpose':'Last\x20modification\x20timestamp.\x20Auto-update\x20is\x20handled\x20by\x20the\x20RDF\x20runtime\x20(BaseModel\x20auditColumns\x20convention)\x20which\x20injects\x20updated_at\x20=\x20CURRENT_TIMESTAMP\x20into\x20every\x20UPDATE\x20statement\x20based\x20on\x20field\x20naming,\x20not\x20by\x20any\x20SDF\x20marker.\x20Initial\x20INSERT\x20value\x20is\x20also\x20handled\x20by\x20the\x20RDF\x20runtime\x20auditColumns\x20helper.'},{'name':a0_0xb18477(0x1e5),'shorthand':'string:100','nullable':!![],'purpose':'User\x20identifier\x20of\x20the\x20last\x20modifier.\x20Set\x20by\x20application\x20layer\x20on\x20UPDATE,\x20not\x20by\x20database\x20default.'}],'convention':{'emitInSkeleton':a0_0xb18477(0x1ee),'nullablePolicy':'All\x204\x20columns\x20are\x20nullable.\x20The\x20created_by/updated_by\x20columns\x20may\x20be\x20empty\x20for\x20rows\x20inserted\x20by\x20system\x20seeds,\x20migrations,\x20or\x20batch\x20imports\x20that\x20do\x20not\x20have\x20a\x20user\x20context.','relationToRdf':'The\x20RDF\x20generator\x20(codegen_create_endpoint)\x20assumes\x20these\x204\x20columns\x20exist\x20when\x20the\x20payload.auditColumns\x20key\x20is\x20absent.\x20See\x20field-validation:catalog\x20auditColumns\x20for\x20the\x20RDF-side\x20behavior.\x20Drift\x20between\x20SDF\x20(missing\x20audit\x20columns)\x20and\x20RDF\x20(assumes\x20audit\x20columns)\x20causes\x20runtime\x20errors\x20when\x20the\x20generated\x20endpoint\x20tries\x20to\x20write\x20created_by/updated_by\x20to\x20a\x20non-existent\x20column.'}},SHORTHAND_SYNTAX={'format':a0_0xb18477(0x215),'description':a0_0xb18477(0x23a),'rules':['Type\x20is\x20mandatory\x20at\x20the\x20beginning\x20and\x20must\x20be\x20one\x20of\x20fieldTypes.','Modifier\x20is\x20mandatory\x20for\x20type\x20\x22string\x22\x20(length)\x20and\x20\x22decimal\x22\x20(precision,scale).',a0_0xb18477(0x204),a0_0xb18477(0x22d),'Default\x20value\x20format\x20depends\x20on\x20type:\x20raw\x20for\x20boolean\x20(default:true),\x20raw\x20for\x20integer\x20(default:0),\x20single-quoted\x20for\x20string\x20(default:\x27value\x27),\x20bare\x20identifier\x20for\x20SQL\x20constants\x20(default:current_date),\x20identifier()\x20for\x20native\x20function\x20calls\x20(default:now()).',a0_0xb18477(0x227)],'examples':[{'input':'string:36\x20pk','meaning':a0_0xb18477(0x233)},{'input':'string:255\x20notnull','meaning':'VARCHAR(255)\x20NOT\x20NULL'},{'input':a0_0xb18477(0x1eb),'meaning':'DECIMAL(15,2)\x20NOT\x20NULL\x20DEFAULT\x200'},{'input':a0_0xb18477(0x238),'meaning':a0_0xb18477(0x1dc)},{'input':a0_0xb18477(0x214),'meaning':a0_0xb18477(0x1d5)},{'input':a0_0xb18477(0x207),'meaning':a0_0xb18477(0x203)},{'input':a0_0xb18477(0x1f8),'meaning':'VARCHAR(36)\x20with\x20FK\x20to\x20category(id),\x20auto\x20belongsTo\x20relation'},{'input':a0_0xb18477(0x201),'meaning':a0_0xb18477(0x22a)}]},NAMING_RULES={'tableName':{'format':'snake_case','description':'Table\x20name\x20must\x20be\x20lowercase,\x20digits,\x20and\x20underscores\x20only.'},'fieldName':{'format':'snake_case','description':'Column\x20name\x20must\x20be\x20snake_case.'},'constraintName':{'format':'<prefix>_<table>_<suffix>','defaultMaxLength':0x1e,'fallbackStrategy':a0_0xb18477(0x20f),'description':a0_0xb18477(0x20b)},'compositeShortName':{'description':a0_0xb18477(0x21a)}},DIALECT_SUPPORT=[{'name':'postgres','driver':'pg','booleanStorage':'native\x20BOOLEAN'},{'name':a0_0xb18477(0x1dd),'driver':a0_0xb18477(0x228),'booleanStorage':'VARCHAR\x20(literal\x20\x22true\x22/\x22false\x22)'},{'name':'oracle','driver':'oracledb','booleanStorage':a0_0xb18477(0x1da)},{'name':'sqlite','driver':'better-sqlite3','booleanStorage':a0_0xb18477(0x1f2)}],DBSCHEMA_CATALOG={'schemaVersion':SCHEMA_VERSION,'source':SOURCE,'summary':{'totalDefineModelOptions':DEFINE_MODEL_OPTIONS['length'],'totalFieldTypes':FIELD_TYPES['length'],'totalConstraints':CONSTRAINTS[a0_0xb18477(0x1df)],'totalRelationTypes':RELATION_TYPES[a0_0xb18477(0x1df)],'totalReferentialActions':REFERENTIAL_ACTIONS['length'],'totalCheckOperations':CHECK_OPERATIONS[a0_0xb18477(0x1df)],'totalAuditColumns':AUDIT_COLUMNS['columns'][a0_0xb18477(0x1df)],'totalDialects':DIALECT_SUPPORT['length']},'defineModelOptions':DEFINE_MODEL_OPTIONS,'fieldTypes':FIELD_TYPES,'constraints':CONSTRAINTS,'relationTypes':RELATION_TYPES,'referentialActions':REFERENTIAL_ACTIONS,'checkOperations':CHECK_OPERATIONS,'auditColumns':AUDIT_COLUMNS,'shorthandSyntax':SHORTHAND_SYNTAX,'namingRules':NAMING_RULES,'dialectSupport':DIALECT_SUPPORT,'documentationUrl':DOCUMENTATION_URL};module[a0_0xb18477(0x21d)]={'DBSCHEMA_CATALOG':DBSCHEMA_CATALOG,'DEFINE_MODEL_OPTIONS':DEFINE_MODEL_OPTIONS,'FIELD_TYPES':FIELD_TYPES,'CONSTRAINTS':CONSTRAINTS,'RELATION_TYPES':RELATION_TYPES,'REFERENTIAL_ACTIONS':REFERENTIAL_ACTIONS,'CHECK_OPERATIONS':CHECK_OPERATIONS,'AUDIT_COLUMNS':AUDIT_COLUMNS,'SHORTHAND_SYNTAX':SHORTHAND_SYNTAX,'NAMING_RULES':NAMING_RULES,'DIALECT_SUPPORT':DIALECT_SUPPORT};
1
+ 'use strict';const a0_0x292f05=a0_0xe08c;(function(_0x12cc8d,_0x2d9720){const _0x39e102=a0_0xe08c,_0x1e8e44=_0x12cc8d();while(!![]){try{const _0x54ac97=-parseInt(_0x39e102(0xce))/0x1*(-parseInt(_0x39e102(0x9e))/0x2)+-parseInt(_0x39e102(0x84))/0x3*(-parseInt(_0x39e102(0xc5))/0x4)+-parseInt(_0x39e102(0xa1))/0x5*(parseInt(_0x39e102(0xb0))/0x6)+-parseInt(_0x39e102(0xda))/0x7+parseInt(_0x39e102(0x86))/0x8*(-parseInt(_0x39e102(0xae))/0x9)+-parseInt(_0x39e102(0x82))/0xa+parseInt(_0x39e102(0xa0))/0xb*(parseInt(_0x39e102(0xcd))/0xc);if(_0x54ac97===_0x2d9720)break;else _0x1e8e44['push'](_0x1e8e44['shift']());}catch(_0x3afb21){_0x1e8e44['push'](_0x1e8e44['shift']());}}}(a0_0x27b7,0x59ecf));const SCHEMA_VERSION=a0_0x292f05(0xd2),SOURCE='dbschema-catalog',DOCUMENTATION_URL='https://restforge.dev/docs/cli/schema-definition',DEFINE_MODEL_OPTIONS=[{'name':a0_0x292f05(0x94),'required':![],'type':'string|null','description':a0_0x292f05(0xd0),'notes':'null\x20or\x20empty\x20string\x20means\x20default\x20schema.'},{'name':'fields','required':!![],'type':'object','description':'Object\x20of\x20field\x20definitions\x20in\x20shorthand\x20string\x20format.','example':a0_0x292f05(0x96)},{'name':'primaryKey','required':![],'type':'string|array','description':'Override\x20primary\x20key.\x20Single\x20field\x20name\x20or\x20composite\x20array.','notes':'Auto-detected\x20from\x20field\x20shorthand\x20\x22pk\x22\x20when\x20not\x20specified.'},{'name':a0_0x292f05(0xc8),'required':![],'type':a0_0x292f05(0xbd),'description':a0_0x292f05(0xc7),'example':'{\x20category:\x20{\x20type:\x20\x22belongsTo\x22,\x20references:\x20\x22category(id)\x22\x20}\x20}'},{'name':'indexes','required':![],'type':a0_0x292f05(0xdc),'description':a0_0x292f05(0x89),'example':a0_0x292f05(0x8a)},{'name':'uniques','required':![],'type':'array','description':'Unique\x20constraints\x20(single\x20or\x20composite).','example':'[[\x22category_code\x22],\x20[\x22tenant_id\x22,\x20\x22category_code\x22]]'},{'name':'checks','required':![],'type':a0_0x292f05(0xdc),'description':'CHECK\x20constraints.\x20Each\x20entry\x20uses\x20operator-as-key\x20shape:\x20{\x20name?,\x20field,\x20<operator>:\x20<value>\x20}\x20where\x20<operator>\x20is\x20one\x20of\x20checkOperations\x20(in,\x20eq,\x20neq,\x20gt,\x20gte,\x20lt,\x20lte).\x20The\x20internal\x20IR\x20shape\x20after\x20ir-builder\x20normalization\x20is\x20{\x20name?,\x20field,\x20op,\x20value\x20},\x20but\x20that\x20is\x20NOT\x20the\x20user\x20input\x20format.','example':'[{\x20field:\x20\x22status\x22,\x20in:\x20[\x22active\x22,\x20\x22inactive\x22]\x20},\x20{\x20field:\x20\x22qty\x22,\x20gte:\x200\x20}]'}],FIELD_TYPES=[{'name':'string','description':a0_0x292f05(0x9d),'requiresModifier':!![],'modifierFormat':a0_0x292f05(0xbc),'example':'string:255','notes':a0_0x292f05(0xb7)},{'name':'text','description':a0_0x292f05(0xab),'requiresModifier':![],'example':a0_0x292f05(0xbb)},{'name':'integer','description':a0_0x292f05(0x8c),'requiresModifier':![],'example':a0_0x292f05(0xcc)},{'name':a0_0x292f05(0xa8),'description':a0_0x292f05(0xb6),'requiresModifier':![],'example':'bigint'},{'name':a0_0x292f05(0x88),'description':a0_0x292f05(0xaf),'requiresModifier':!![],'modifierFormat':'decimal:<precision>,<scale>','example':'decimal:15,2','notes':a0_0x292f05(0x81)},{'name':a0_0x292f05(0x83),'description':'Boolean\x20value\x20(native\x20BOOLEAN\x20on\x20PostgreSQL,\x20VARCHAR\x20on\x20others).','requiresModifier':![],'example':a0_0x292f05(0x83)},{'name':a0_0x292f05(0xcb),'description':a0_0x292f05(0x9f),'requiresModifier':![],'example':a0_0x292f05(0xcb)},{'name':a0_0x292f05(0x95),'description':'Date\x20and\x20time\x20(TIMESTAMP).','requiresModifier':![],'example':'timestamp','notes':a0_0x292f05(0xd7)},{'name':'uuid','description':'UUID.\x20Native\x20UUID\x20on\x20PostgreSQL,\x20VARCHAR(36)\x20on\x20others.','requiresModifier':![],'example':'uuid'},{'name':'json','description':a0_0x292f05(0xb4),'requiresModifier':![],'example':'json'}],CONSTRAINTS=[{'name':'pk','kind':a0_0x292f05(0xa4),'description':'Marks\x20field\x20as\x20primary\x20key.','example':'string:36\x20pk'},{'name':'notnull','kind':a0_0x292f05(0xa4),'description':a0_0x292f05(0xb8),'example':'string:255\x20notnull'},{'name':a0_0x292f05(0xb9),'kind':a0_0x292f05(0xa4),'description':'Single-column\x20unique\x20constraint.','example':'string:32\x20unique'},{'name':'index','kind':'standalone','description':'Single-column\x20non-unique\x20index.','example':'string:64\x20index'},{'name':'autoUpdate','kind':a0_0x292f05(0xa4),'deprecated':!![],'description':'DEPRECATED:\x20This\x20modifier\x20no\x20longer\x20has\x20functional\x20effect\x20at\x20the\x20DDL\x20or\x20runtime\x20level.\x20Auto-update\x20for\x20updated_at\x20is\x20handled\x20by\x20the\x20RDF\x20layer\x20(auditColumns\x20convention\x20in\x20BaseModel\x20runtime)\x20based\x20on\x20field\x20naming\x20convention,\x20not\x20by\x20this\x20SDF\x20marker.\x20Existing\x20usage\x20is\x20preserved\x20for\x20backward\x20compatibility\x20but\x20should\x20be\x20removed\x20from\x20new\x20templates.','example':a0_0x292f05(0x95),'notes':'Engine\x20still\x20parses\x20this\x20token\x20(backward\x20compatibility),\x20but\x20DDL\x20output\x20is\x20identical\x20to\x20plain\x20\x27timestamp\x27.\x20Use\x20\x27timestamp\x27\x20alone.\x20The\x20actual\x20auto-update\x20behavior\x20is\x20documented\x20in\x20the\x20RDF\x20auditColumns\x20catalog\x20(field-validation:catalog).'},{'name':'default','kind':a0_0x292f05(0xa9),'description':'Default\x20value.\x20Single-quoted\x20for\x20strings,\x20raw\x20for\x20numeric/boolean,\x20bare\x20identifier\x20for\x20SQL\x20constants,\x20identifier()\x20for\x20native\x20function\x20calls.','valueFormat':a0_0x292f05(0x9a),'example':a0_0x292f05(0xbf)},{'name':'fk','kind':'value','description':'Foreign\x20key\x20reference.\x20Auto-generates\x20a\x20belongsTo\x20relation\x20entry.','valueFormat':'fk:<table>.<column>','example':'string:36\x20fk:category.id','notes':a0_0x292f05(0x92)}],RELATION_TYPES=[{'name':'belongsTo','description':'Many-to-one.\x20This\x20table\x20holds\x20the\x20FK\x20column\x20referencing\x20the\x20parent\x20table.','requiredFields':['type',a0_0x292f05(0xd1),'references'],'optionalFields':['target','onDelete','onUpdate'],'example':'{\x20type:\x20\x22belongsTo\x22,\x20localKey:\x20\x22category_id\x22,\x20references:\x20\x22category_id\x22,\x20onDelete:\x20\x22restrict\x22\x20}','notes':'`references`\x20is\x20the\x20bare\x20column\x20name\x20in\x20the\x20target\x20table.\x20Target\x20table\x20is\x20auto-derived\x20from\x20the\x20relation\x20key\x20name,\x20or\x20override\x20with\x20`target`.'},{'name':'hasOne','description':'One-to-one\x20(inverse\x20of\x20belongsTo).\x20Optional\x20inverse\x20declaration\x20on\x20the\x20parent\x20side;\x20does\x20not\x20affect\x20DDL.','requiredFields':['type',a0_0x292f05(0xd1),'references'],'optionalFields':[a0_0x292f05(0xba),'onDelete','onUpdate'],'example':'{\x20type:\x20\x22hasOne\x22,\x20target:\x20\x22profile\x22,\x20localKey:\x20\x22user_id\x22,\x20references:\x20\x22user_id\x22\x20}'},{'name':a0_0x292f05(0xd9),'description':a0_0x292f05(0x90),'requiredFields':['type','localKey','references'],'optionalFields':['target','onDelete','onUpdate'],'example':a0_0x292f05(0xd8)}],REFERENTIAL_ACTIONS=[{'name':a0_0x292f05(0xc2),'description':a0_0x292f05(0x8d),'appliesTo':[a0_0x292f05(0xc0),a0_0x292f05(0x80)]},{'name':a0_0x292f05(0xbe),'description':a0_0x292f05(0xc6),'appliesTo':['onDelete',a0_0x292f05(0x80)]},{'name':a0_0x292f05(0x9c),'description':'Set\x20FK\x20column\x20to\x20NULL\x20on\x20child\x20rows.','appliesTo':['onDelete',a0_0x292f05(0x80)],'notes':a0_0x292f05(0x97)},{'name':a0_0x292f05(0xb5),'description':a0_0x292f05(0xad),'appliesTo':[a0_0x292f05(0xc0),a0_0x292f05(0x80)]}],CHECK_OPERATIONS=[{'name':'in','description':a0_0x292f05(0x8e),'valueType':'array','example':'{\x20field:\x20\x22status\x22,\x20in:\x20[\x22active\x22,\x20\x22inactive\x22]\x20}'},{'name':'eq','description':a0_0x292f05(0x9b),'valueType':a0_0x292f05(0x87),'example':a0_0x292f05(0x91)},{'name':'neq','description':'Not\x20equal\x20to.','valueType':a0_0x292f05(0x87),'example':a0_0x292f05(0xc9)},{'name':'gt','description':'Greater\x20than.','valueType':'numeric','example':'{\x20field:\x20\x22qty\x22,\x20gt:\x200\x20}'},{'name':a0_0x292f05(0xd3),'description':a0_0x292f05(0xdd),'valueType':'numeric','example':a0_0x292f05(0xc1)},{'name':'lt','description':a0_0x292f05(0xc3),'valueType':'numeric','example':'{\x20field:\x20\x22discount\x22,\x20lt:\x20100\x20}'},{'name':'lte','description':'Less\x20than\x20or\x20equal\x20to.','valueType':'numeric','example':'{\x20field:\x20\x22discount\x22,\x20lte:\x20100\x20}'}],AUDIT_COLUMNS={'description':a0_0x292f05(0xaa),'columns':[{'name':'created_at','shorthand':a0_0x292f05(0xb3),'nullable':!![],'purpose':a0_0x292f05(0x98)},{'name':'created_by','shorthand':a0_0x292f05(0xd5),'nullable':!![],'purpose':'User\x20identifier\x20(username,\x20user_id,\x20or\x20composite\x20token)\x20of\x20the\x20record\x20creator.\x20Set\x20by\x20application\x20layer\x20on\x20INSERT,\x20not\x20by\x20database\x20default.'},{'name':a0_0x292f05(0xd6),'shorthand':'timestamp','nullable':!![],'purpose':'Last\x20modification\x20timestamp.\x20Auto-update\x20is\x20handled\x20by\x20the\x20RDF\x20runtime\x20(BaseModel\x20auditColumns\x20convention)\x20which\x20injects\x20updated_at\x20=\x20CURRENT_TIMESTAMP\x20into\x20every\x20UPDATE\x20statement\x20based\x20on\x20field\x20naming,\x20not\x20by\x20any\x20SDF\x20marker.\x20Initial\x20INSERT\x20value\x20is\x20also\x20handled\x20by\x20the\x20RDF\x20runtime\x20auditColumns\x20helper.'},{'name':'updated_by','shorthand':'string:100','nullable':!![],'purpose':'User\x20identifier\x20of\x20the\x20last\x20modifier.\x20Set\x20by\x20application\x20layer\x20on\x20UPDATE,\x20not\x20by\x20database\x20default.'}],'convention':{'emitInSkeleton':a0_0x292f05(0xa7),'nullablePolicy':'All\x204\x20columns\x20are\x20nullable.\x20The\x20created_by/updated_by\x20columns\x20may\x20be\x20empty\x20for\x20rows\x20inserted\x20by\x20system\x20seeds,\x20migrations,\x20or\x20batch\x20imports\x20that\x20do\x20not\x20have\x20a\x20user\x20context.','relationToRdf':a0_0x292f05(0x8f)}},SHORTHAND_SYNTAX={'format':a0_0x292f05(0xc4),'description':'String\x20shorthand\x20for\x20field\x20definition.\x20Type\x20required\x20at\x20start,\x20followed\x20by\x20optional\x20modifier\x20and\x20space-separated\x20constraints.','rules':['Type\x20is\x20mandatory\x20at\x20the\x20beginning\x20and\x20must\x20be\x20one\x20of\x20fieldTypes.',a0_0x292f05(0xb2),a0_0x292f05(0x99),'Value\x20constraints\x20(default,\x20fk)\x20require\x20constraint:value\x20format.','Default\x20value\x20format\x20depends\x20on\x20type:\x20raw\x20for\x20boolean\x20(default:true),\x20raw\x20for\x20integer\x20(default:0),\x20single-quoted\x20for\x20string\x20(default:\x27value\x27),\x20bare\x20identifier\x20for\x20SQL\x20constants\x20(default:current_date),\x20identifier()\x20for\x20native\x20function\x20calls\x20(default:now()).',a0_0x292f05(0xa2)],'examples':[{'input':'string:36\x20pk','meaning':'VARCHAR(36)\x20PRIMARY\x20KEY'},{'input':a0_0x292f05(0xdb),'meaning':'VARCHAR(255)\x20NOT\x20NULL'},{'input':a0_0x292f05(0xa3),'meaning':'DECIMAL(15,2)\x20NOT\x20NULL\x20DEFAULT\x200'},{'input':a0_0x292f05(0xbf),'meaning':a0_0x292f05(0x93)},{'input':'string:100\x20default:\x27pending\x27','meaning':'VARCHAR(100)\x20DEFAULT\x20\x27pending\x27'},{'input':a0_0x292f05(0xb3),'meaning':'TIMESTAMP\x20DEFAULT\x20<native_now\x20per\x20dialect>'},{'input':'string:36\x20fk:category.id','meaning':'VARCHAR(36)\x20with\x20FK\x20to\x20category(id),\x20auto\x20belongsTo\x20relation'},{'input':'string:64\x20index','meaning':a0_0x292f05(0xa5)}]},NAMING_RULES={'tableName':{'format':'snake_case','description':'Table\x20name\x20must\x20be\x20lowercase,\x20digits,\x20and\x20underscores\x20only.'},'fieldName':{'format':'snake_case','description':a0_0x292f05(0x8b)},'constraintName':{'format':'<prefix>_<table>_<suffix>','defaultMaxLength':0x1e,'fallbackStrategy':'MD5\x20hash\x208-character\x20suffix\x20when\x20name\x20exceeds\x20maxLength','description':'Constraint\x20names\x20auto-generated\x20with\x20type-specific\x20prefix\x20(pk,\x20fk,\x20idx,\x20uq,\x20ck)\x20and\x20dialect-specific\x20max\x20length.'},'compositeShortName':{'description':'For\x20composite\x20indexes/uniques,\x20name\x20is\x20derived\x20by\x20joining\x20columns\x20with\x20underscore\x20and\x20truncating\x20when\x20needed.'}},DIALECT_SUPPORT=[{'name':'postgres','driver':'pg','booleanStorage':'native\x20BOOLEAN'},{'name':a0_0x292f05(0xca),'driver':'mysql2','booleanStorage':'VARCHAR\x20(literal\x20\x22true\x22/\x22false\x22)'},{'name':a0_0x292f05(0xd4),'driver':'oracledb','booleanStorage':a0_0x292f05(0xb1)},{'name':'sqlite','driver':a0_0x292f05(0xcf),'booleanStorage':a0_0x292f05(0xa6)}],DBSCHEMA_CATALOG={'schemaVersion':SCHEMA_VERSION,'source':SOURCE,'summary':{'totalDefineModelOptions':DEFINE_MODEL_OPTIONS['length'],'totalFieldTypes':FIELD_TYPES[a0_0x292f05(0xac)],'totalConstraints':CONSTRAINTS['length'],'totalRelationTypes':RELATION_TYPES['length'],'totalReferentialActions':REFERENTIAL_ACTIONS[a0_0x292f05(0xac)],'totalCheckOperations':CHECK_OPERATIONS[a0_0x292f05(0xac)],'totalAuditColumns':AUDIT_COLUMNS[a0_0x292f05(0x85)][a0_0x292f05(0xac)],'totalDialects':DIALECT_SUPPORT['length']},'defineModelOptions':DEFINE_MODEL_OPTIONS,'fieldTypes':FIELD_TYPES,'constraints':CONSTRAINTS,'relationTypes':RELATION_TYPES,'referentialActions':REFERENTIAL_ACTIONS,'checkOperations':CHECK_OPERATIONS,'auditColumns':AUDIT_COLUMNS,'shorthandSyntax':SHORTHAND_SYNTAX,'namingRules':NAMING_RULES,'dialectSupport':DIALECT_SUPPORT,'documentationUrl':DOCUMENTATION_URL};function a0_0xe08c(_0x3d85b6,_0x5b2156){_0x3d85b6=_0x3d85b6-0x80;const _0x27b7d2=a0_0x27b7();let _0xe08c0d=_0x27b7d2[_0x3d85b6];if(a0_0xe08c['aeoFzu']===undefined){var _0x433f67=function(_0x3002b7){const _0x1ceecc='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x1c70c5='',_0x3d040b='';for(let _0x56aefa=0x0,_0x2ac2a1,_0x26bee9,_0x1967f0=0x0;_0x26bee9=_0x3002b7['charAt'](_0x1967f0++);~_0x26bee9&&(_0x2ac2a1=_0x56aefa%0x4?_0x2ac2a1*0x40+_0x26bee9:_0x26bee9,_0x56aefa++%0x4)?_0x1c70c5+=String['fromCharCode'](0xff&_0x2ac2a1>>(-0x2*_0x56aefa&0x6)):0x0){_0x26bee9=_0x1ceecc['indexOf'](_0x26bee9);}for(let _0x16817e=0x0,_0x31207e=_0x1c70c5['length'];_0x16817e<_0x31207e;_0x16817e++){_0x3d040b+='%'+('00'+_0x1c70c5['charCodeAt'](_0x16817e)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x3d040b);};a0_0xe08c['wKwkBn']=_0x433f67,a0_0xe08c['KwgJyR']={},a0_0xe08c['aeoFzu']=!![];}const _0x3a7f1e=_0x27b7d2[0x0],_0x28d2b0=_0x3d85b6+_0x3a7f1e,_0x2e0f61=a0_0xe08c['KwgJyR'][_0x28d2b0];return!_0x2e0f61?(_0xe08c0d=a0_0xe08c['wKwkBn'](_0xe08c0d),a0_0xe08c['KwgJyR'][_0x28d2b0]=_0xe08c0d):_0xe08c0d=_0x2e0f61,_0xe08c0d;}function a0_0x27b7(){const _0x33a399=['CMvSyxrPB25Z','EYbMAwvSzdOGiNr5CguIlcbUzxe6icjZExn0zw0Iih0','BxLZCwW','zgf0zq','Aw50zwDLCG','ndq0zKPJAgTl','muD2zevJzq','yMv0DgvYlxnXBgL0ztm','rgf0ywjHC2uGC2nOzw1Hig5HBwvZCgfJzsaOBxvSDgKTC2nOzw1HigrHDgfIyxnLks4','Bg9JywXlzxK','ms4W','z3rL','B3jHy2XL','C3rYAw5NoJeWma','DxbKyxrLzf9HDa','rgvMyxvSDcb0Aw1LEM9UzsbIzwHHDMLVCIbPCYbKAwfSzwn0lxnWzwnPzMLJlG','EYb0ExbLoIaIAgfZtwfUEsiSihrHCMDLDdOGiM9YzgvYx2L0zw0IlcbSB2nHBeTLEtOGiM9YzgvYx2LKiIWGCMvMzxjLBMnLCZOGiM9YzgvYx2LKiIb9','AgfZtwfUEq','oty3mtK3C3nktLrr','C3rYAw5NoJi1nsbUB3rUDwXS','yxjYyxK','r3jLyxrLCIb0AgfUig9YigvXDwfSihrVlG','B25vCgrHDgu','uhjLy2LZAw9UigfUzcbZy2fSzsbYzxf1AxjLzcbMB3iGy3jVC3mTzgLHBgvJDcbWB3j0ywjPBgL0Es4','mZG5mdG4mgTMz0PhDG','yM9VBgvHBG','nKnMtgzMta','y29SDw1UCW','nty4EwrqAxHI','C2nHBgfY','zgvJAw1HBa','tM9UlxvUAxf1zsbPBMrLEgvZlIbfywnOigvUDhj5igLZigfYCMf5ig9MignVBhvTBIbUyw1LCYbVCIbVyMPLy3qGEYbUyw1LlcbJB2X1Bw5Zih0U','w1SIC3rHDhvZiIWGiMnYzwf0zwrFyxqIxv0','q29SDw1Uig5HBwuGBxvZDcbIzsbZBMfRzv9JyxnLlG','mZiTyML0ihnPz25LzcbPBNrLz2vYicHjtLqVsu5uruDfuIKU','q2fZy2fKzsbKzwXLDguVDxbKyxrLihrVignOAwXKihjVD3mU','vMfSDwuGBxvZDcbIzsbVBMuGB2yGysbSAxn0lG','vgHLifjerIbNzw5LCMf0B3iGkgnVzgvNzw5Fy3jLyxrLx2vUzhbVAw50ksbHC3n1BwvZihrOzxnLidqGy29SDw1UCYbLEgLZDcb3AgvUihrOzsbWyxLSB2fKlMf1zgL0q29SDw1UCYbRzxKGAxmGywjZzw50lIbtzwuGzMLLBgqTDMfSAwrHDgLVBJPJyxrHBg9Nigf1zgL0q29SDw1UCYbMB3iGDgHLifjerI1ZAwrLigjLAgf2Aw9YlIbeCMLMDcbIzxr3zwvUifnerIaOBwLZC2LUzYbHDwrPDcbJB2X1Bw5ZksbHBMqGuKrgicHHC3n1BwvZigf1zgL0ignVBhvTBNmPignHDxnLCYbYDw50Aw1LigvYCM9YCYb3AgvUihrOzsbNzw5LCMf0zwqGzw5KCg9PBNqGDhjPzxmGDg8GD3jPDguGy3jLyxrLzf9IEs91CgrHDgvKx2j5ihrVigeGBM9Ulwv4Axn0zw50ignVBhvTBI4','t25LlxrVlw1HBNKGkgLUDMvYC2uGB2yGyMvSB25NC1rVks4Gt3b0Aw9UywWGAw52zxjZzsbKzwnSyxjHDgLVBIbVBIb0AguGCgfYzw50ihnPzgu7igrVzxmGBM90igfMzMvJDcbereWU','EYbMAwvSzdOGiNr5CguIlcbLCtOGiNvZzxiIih0','q2fUBM90ignVzxHPC3qGD2L0AcbLEhbSAwnPDcbYzwXHDgLVBIbLBNrYEsbYzwzLCMvUy2LUzYb0AguGC2fTzsbMAwvSzc4','qK9ptevbtIberuzbvuXuifrsvuu','C2nOzw1H','DgLTzxn0yw1W','EYbPzdOGiNn0CMLUzZOZnIbWAYiSig5HBwu6icjZDhjPBMC6mJu1ig5VDg51BgWIih0','rKSGzMLLBgqGBxvZDcbIzsbUDwXSywjSzs4','uMvJB3jKignYzwf0Aw9UihrPBwvZDgfTCc4Gqxv0BY1ZzxqGB24Gsu5trvjuihzPysberuzbvuXuig5VDYGPlIboyxrPDMuGzNvUy3rPB24GDhjHBNnSyxrLCYbWzxiGzgLHBgvJDcaOCg9ZDgDYzxm6ienvuLjftLrFveLnrvnuqu1qlcbTExnXBdOGtK9xkcKSig9YywnSztOGu1LtveLnrvnuqu1qlcbZCwXPDgu6ienvuLjftLrFveLnrvnuqu1qks4','u3rHBMrHBg9UzsbJB25ZDhjHAw50CYaOCgSSig5VDg51BgWSihvUAxf1zsWGAw5KzxGPihrHA2uGBM8GDMfSDwuUifrOzsbSzwDHy3KGyxv0B1vWzgf0zsb0B2TLBIbPCYbZDgLSBcbWyxjZzwqGzM9YigjHy2T3yxjKignVBxbHDgLIAwXPDhKGyNv0igLZigrLChjLy2f0zwqGyw5KigHHCYbUBYbMDw5JDgLVBMfSigvMzMvJDc4','zgvMyxvSDdO8BgL0zxjHBd4GFcbKzwzHDwX0oIC8C3rYAw5NpICGFcbKzwzHDwX0oJXJB25ZDgfUDd4GFcbKzwzHDwX0oJXMDw5JDgLVBJ4Okq','rxf1ywWGDg8U','C2v0tNvSBa','vMfYAwfIBguTBgvUz3rOihrLEhqUieXLBMD0AcbTB2rPzMLLCIbYzxf1AxjLzc4','ndeXodmYAfjnAe1k','rgf0zsbVBMX5icHUBYb0Aw1LignVBxbVBMvUDcKU','mtuWota5v3zyD0DR','mtbfA1z5A2m','rM9YzwLNBIbRzxKGDxnLCYbKB3qGBM90yxrPB246igzRoJX0ywjSzt4UpgnVBhvTBJ4GkguUzY4SigzRoMnHDgvNB3j5lMLKks4GvgHLihbHCNnLCIbYzwPLy3rZihbHCMvUDgHLC2vZihn5BNrHEc4','zgvJAw1HBdOXnsWYig5VDg51BgWGzgvMyxvSDdOW','C3rHBMrHBg9Uzq','vKfsq0HbuIG2ncKGD2L0AcbZAw5NBguTy29SDw1UigLUzgv4','vevyva','wwvZiokaLcbKyNnJAgvTytPPBML0igvTAxrZigfSBca0ignVBhvTBNmGyNKGzgvMyxvSDc4GvgfIBgvZihrOyxqGzg8GBM90ig5LzwqGyxvKAxqGkgXVB2T1Ccb0ywjSzxmSihn5C3rLBsb0ywjSzxmPignHBIbTyw51ywXSEsbYzw1VDMuGDgHLC2uGzMLLBgrZigzYB20GDgHLigDLBMvYyxrLzcbZA2vSzxrVBI4','yMLNAw50','DMfSDwu','u3rHBMrHCMqGnc1JB2X1Bw4GyxvKAxqGy29UDMvUDgLVBIbMB3iGDgfIBgvZig1HBMfNzwqGyNKGuKvtvezVCMDLlIbuAguGC2fTzsbJB252zw50Aw9UigLZigrVy3vTzw50zwqGAw4GzMLLBgqTDMfSAwrHDgLVBJPJyxrHBg9NicHHDwrPDenVBhvTBNmGC2vJDgLVBIKGzM9YihrOzsbsreyVyMfJA2vUzcbSyxLLCI4Gu0rgigfUzcbsreyGC3rHEsbHBgLNBMvKihDOzw4GyM90AcbKzwnSyxjLihrOzxnLidqGy29SDw1UCY4','tg9UzY1MB3jTihrLEhqGD2L0Ag91DcbSzw5NDgGGBgLTAxqGkfrfwfqVq0XpqIKU','BgvUz3rO','rgvMzxiGy29UC3rYywLUDcbJAgvJAYaOC2vTyw50AwnHBgX5ihnPBwLSyxiGDg8GCMvZDhjPy3qGB24GBw9ZDcbKAwfSzwn0CYKU','ndm2ndf2rKHQELe','rML4zwqTCg9PBNqGzgvJAw1HBc4GuhjLy2LZAw9UigfUzcbZy2fSzsbYzxf1AxjLzc4','mJi4ndmYzurowvvw','vKfsq0HbuJiGD2L0AcbdsevdsYbJB25ZDhjHAw50','tw9KAwzPzxiGAxmGBwfUzgf0B3j5igzVCIb0ExbLicjZDhjPBMCIicHSzw5NDgGPigfUzcaIzgvJAw1HBciGkhbYzwnPC2LVBIXZy2fSzsKU','DgLTzxn0yw1WigrLzMf1Bhq6BM93kcK','sLnptIbJB2X1Bw4GkePtt05cig9UifbVC3rNCMvtuuWSiePtt04GB24GtxLtuuWSienmt0iGB24Gt3jHy2XLks4','BM9by3rPB24','nJqTyML0ihnPz25LzcbPBNrLz2vYicHcsuDjtLqPlG','rxHWBgLJAxqGBgvUz3rOihjLCxvPCMvKigzVCIbJCM9ZCY1KAwfSzwn0ihbVCNrHyMLSAxr5lG','tK9uie5vteWGy29UC3rYywLUDc4','Dw5PCxvL','DgfYz2v0','Dgv4Da','C3rYAw5NoJXSzw5NDgG+','B2jQzwn0','CMvZDhjPy3q','yM9VBgvHBIbKzwzHDwX0oNrYDwu','B25ezwXLDgu','EYbMAwvSzdOGiNf0EsiSigD0ztOGmcb9','y2fZy2fKzq','tgvZCYb0AgfUlG','phr5Cgu+wZO8Bw9KAwzPzxi+xsbBpgnVBNn0CMfPBNq+wZO8DMfSDwu+xv0UlI4','mtiWndK5nKzurfLNCG','uMvQzwn0igrLBgv0zs91CgrHDguGD2HLBIbJAgLSzcbYB3DZigv4Axn0lG','sw50zxiTDgfIBguGCMvSyxrPB25ZicHIzwXVBMDZvg8SigHHC09UzsWGAgfZtwfUEsKU'];a0_0x27b7=function(){return _0x33a399;};return a0_0x27b7();}module['exports']={'DBSCHEMA_CATALOG':DBSCHEMA_CATALOG,'DEFINE_MODEL_OPTIONS':DEFINE_MODEL_OPTIONS,'FIELD_TYPES':FIELD_TYPES,'CONSTRAINTS':CONSTRAINTS,'RELATION_TYPES':RELATION_TYPES,'REFERENTIAL_ACTIONS':REFERENTIAL_ACTIONS,'CHECK_OPERATIONS':CHECK_OPERATIONS,'AUDIT_COLUMNS':AUDIT_COLUMNS,'SHORTHAND_SYNTAX':SHORTHAND_SYNTAX,'NAMING_RULES':NAMING_RULES,'DIALECT_SUPPORT':DIALECT_SUPPORT};