@restforgejs/platform 5.0.3 → 5.0.8

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 (181) 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 +1 -1
  5. package/generators/cli/endpoint/create.js +1 -1
  6. package/generators/cli/processor/create.js +1 -1
  7. package/generators/lib/dbschema-kit/apply-executor.js +15 -1
  8. package/generators/lib/dbschema-kit/dialect/mysql.js +2 -0
  9. package/generators/lib/dbschema-kit/dialect/oracle.js +2 -0
  10. package/generators/lib/dbschema-kit/dialect/postgres.js +3 -0
  11. package/generators/lib/dbschema-kit/dialect/sqlite.js +2 -0
  12. package/generators/lib/dbschema-kit/emitters/alter-table.js +7 -0
  13. package/generators/lib/dbschema-kit/emitters/create-table.js +31 -3
  14. package/generators/lib/dbschema-kit/statement-modifier.js +12 -2
  15. package/generators/lib/payload/payload-runner.js +44 -2
  16. package/generators/lib/payload/schema-diff.js +31 -1
  17. package/generators/lib/templates/dashboard-catalog.js +1 -1
  18. package/generators/lib/templates/db-connection-env.js +1 -1
  19. package/generators/lib/templates/dbschema-catalog.js +1 -1
  20. package/generators/lib/templates/field-validation-catalog.js +1 -1
  21. package/generators/lib/templates/mysql-template.js +1 -1
  22. package/generators/lib/templates/oracle-template.js +1 -1
  23. package/generators/lib/templates/postgres-template.js +1 -1
  24. package/generators/lib/templates/query-declarative-catalog.js +1 -1
  25. package/generators/lib/templates/sqlite-template.js +1 -1
  26. package/generators/lib/utils/database-introspector.js +417 -6
  27. package/generators/lib/validators/argument-validator.js +2 -2
  28. package/integrity-manifest.json +18 -18
  29. package/package.json +4 -2
  30. package/scripts/verify-integrity.js +1 -1
  31. package/server.js +1 -1
  32. package/src/components/handlers/adjust_handler.js +1 -1
  33. package/src/components/handlers/audit_handler.js +1 -1
  34. package/src/components/handlers/delete_handler.js +1 -1
  35. package/src/components/handlers/export_handler.js +1 -1
  36. package/src/components/handlers/import_handler.js +1 -1
  37. package/src/components/handlers/insert_handler.js +1 -1
  38. package/src/components/handlers/update_handler.js +1 -1
  39. package/src/components/handlers/upload_handler.js +1 -1
  40. package/src/components/handlers/workflow_handler.js +1 -1
  41. package/src/components/integrations/webhook.js +1 -1
  42. package/src/consumers/baseConsumer.js +1 -1
  43. package/src/consumers/declarativeMapper.js +1 -1
  44. package/src/consumers/handlers/apiHandler.js +1 -1
  45. package/src/consumers/handlers/consoleHandler.js +1 -1
  46. package/src/consumers/handlers/databaseHandler.js +1 -1
  47. package/src/consumers/handlers/index.js +1 -1
  48. package/src/consumers/handlers/kafkaHandler.js +1 -1
  49. package/src/consumers/index.js +1 -1
  50. package/src/consumers/messageTransformer.js +1 -1
  51. package/src/consumers/validator.js +1 -1
  52. package/src/core/db/dialect/base-dialect.js +1 -1
  53. package/src/core/db/dialect/index.js +1 -1
  54. package/src/core/db/dialect/mysql-dialect.js +1 -1
  55. package/src/core/db/dialect/oracle-dialect.js +1 -1
  56. package/src/core/db/dialect/postgres-dialect.js +1 -1
  57. package/src/core/db/dialect/sqlite-dialect.js +1 -1
  58. package/src/core/db/flatten-helper.js +1 -1
  59. package/src/core/db/query-builder-error.js +1 -1
  60. package/src/core/db/query-builder.js +1 -1
  61. package/src/core/db/relation-helper.js +1 -1
  62. package/src/core/handlers/delete_handler.js +1 -1
  63. package/src/core/handlers/insert_handler.js +1 -1
  64. package/src/core/handlers/update_handler.js +1 -1
  65. package/src/core/models/base-model.js +1 -1
  66. package/src/core/utils/cache-manager.js +1 -1
  67. package/src/core/utils/component-engine.js +1 -1
  68. package/src/core/utils/context-builder.js +1 -1
  69. package/src/core/utils/datetime-formatter.js +1 -1
  70. package/src/core/utils/datetime-parser.js +1 -1
  71. package/src/core/utils/db.js +1 -1
  72. package/src/core/utils/logger.js +1 -1
  73. package/src/core/utils/payload-loader.js +1 -1
  74. package/src/core/utils/security-checks.js +1 -1
  75. package/src/middleware/body-options.js +1 -1
  76. package/src/middleware/cors.js +1 -1
  77. package/src/middleware/idempotency.js +1 -1
  78. package/src/middleware/rate-limiter.js +1 -1
  79. package/src/middleware/request-logger.js +1 -1
  80. package/src/middleware/security-headers.js +1 -1
  81. package/src/models/base-model-mysql.js +1 -1
  82. package/src/models/base-model-oracle.js +1 -1
  83. package/src/models/base-model-sqlite.js +1 -1
  84. package/src/models/base-model.js +1 -1
  85. package/src/pro/caching/redis-client.js +1 -1
  86. package/src/pro/caching/redis-helper.js +1 -1
  87. package/src/pro/consumers/baseConsumer.js +1 -1
  88. package/src/pro/consumers/declarativeMapper.js +1 -1
  89. package/src/pro/consumers/handlers/apiHandler.js +1 -1
  90. package/src/pro/consumers/handlers/consoleHandler.js +1 -1
  91. package/src/pro/consumers/handlers/databaseHandler.js +1 -1
  92. package/src/pro/consumers/handlers/index.js +1 -1
  93. package/src/pro/consumers/handlers/kafkaHandler.js +1 -1
  94. package/src/pro/consumers/index.js +1 -1
  95. package/src/pro/consumers/messageTransformer.js +1 -1
  96. package/src/pro/consumers/validator.js +1 -1
  97. package/src/pro/database/base-model-mysql.js +1 -1
  98. package/src/pro/database/base-model-oracle.js +1 -1
  99. package/src/pro/database/base-model-sqlite.js +1 -1
  100. package/src/pro/database/db-mysql.js +1 -1
  101. package/src/pro/database/db-oracle.js +1 -1
  102. package/src/pro/database/db-sqlite.js +1 -1
  103. package/src/pro/excel/excel-generator.js +1 -1
  104. package/src/pro/excel/excel-parser.js +1 -1
  105. package/src/pro/excel/export-service.js +1 -1
  106. package/src/pro/excel/export_handler.js +1 -1
  107. package/src/pro/excel/import-service.js +1 -1
  108. package/src/pro/excel/import-validator.js +1 -1
  109. package/src/pro/excel/import_handler.js +1 -1
  110. package/src/pro/excel/upsert-builder.js +1 -1
  111. package/src/pro/idgen/idgen-routes.js +1 -1
  112. package/src/pro/integrations/lookup-resolver.js +1 -1
  113. package/src/pro/integrations/upload-handler-v2.js +1 -1
  114. package/src/pro/integrations/upload-handler.js +1 -1
  115. package/src/pro/integrations/webhook.js +1 -1
  116. package/src/pro/locking/lock-routes.js +1 -1
  117. package/src/pro/locking/resource-lock-manager.js +1 -1
  118. package/src/pro/messaging/kafkaConsumerService.js +1 -1
  119. package/src/pro/messaging/kafkaService.js +1 -1
  120. package/src/pro/messaging/messagehubService.js +1 -1
  121. package/src/pro/messaging/rabbitmqService.js +1 -1
  122. package/src/pro/scheduler/job-manager.js +1 -1
  123. package/src/pro/scheduler/job-routes.js +1 -1
  124. package/src/pro/scheduler/job-validator.js +1 -1
  125. package/src/pro/storage/base-storage-provider.js +1 -1
  126. package/src/pro/storage/file-metadata-helper.js +1 -1
  127. package/src/pro/storage/index.js +1 -1
  128. package/src/pro/storage/local-storage-provider.js +1 -1
  129. package/src/pro/storage/s3-storage-provider.js +1 -1
  130. package/src/pro/storage/upload-cleanup-job.js +1 -1
  131. package/src/pro/storage/upload-cleanup-scheduler.js +1 -1
  132. package/src/pro/storage/upload-pending-tracker.js +1 -1
  133. package/src/pro/websocket/broadcast-helper.js +1 -1
  134. package/src/pro/websocket/index.js +1 -1
  135. package/src/pro/websocket/livesync-server.js +1 -1
  136. package/src/pro/websocket/ws-broadcaster.js +1 -1
  137. package/src/services/export-service.js +1 -1
  138. package/src/services/import-service.js +1 -1
  139. package/src/services/kafkaConsumerService.js +1 -1
  140. package/src/services/kafkaService.js +1 -1
  141. package/src/services/messagehubService.js +1 -1
  142. package/src/services/rabbitmqService.js +1 -1
  143. package/src/utils/cache-invalidation-registry.js +1 -1
  144. package/src/utils/cache-manager.js +1 -1
  145. package/src/utils/component-engine.js +1 -1
  146. package/src/utils/config-extractor.js +1 -1
  147. package/src/utils/consumerLogger.js +1 -1
  148. package/src/utils/context-builder.js +1 -1
  149. package/src/utils/dashboard-helpers.js +1 -1
  150. package/src/utils/dateHelper.js +1 -1
  151. package/src/utils/datetime-formatter.js +1 -1
  152. package/src/utils/datetime-parser.js +1 -1
  153. package/src/utils/db-bootstrap.js +1 -1
  154. package/src/utils/db-mysql.js +1 -1
  155. package/src/utils/db-oracle.js +1 -1
  156. package/src/utils/db-sqlite.js +1 -1
  157. package/src/utils/db.js +1 -1
  158. package/src/utils/demo-generator.js +1 -1
  159. package/src/utils/excel-generator.js +1 -1
  160. package/src/utils/excel-parser.js +1 -1
  161. package/src/utils/file-watcher.js +1 -1
  162. package/src/utils/id-generator.js +1 -1
  163. package/src/utils/idempotency-manager.js +1 -1
  164. package/src/utils/import-validator.js +1 -1
  165. package/src/utils/license-client.js +1 -1
  166. package/src/utils/lock-manager.js +1 -1
  167. package/src/utils/logger.js +1 -1
  168. package/src/utils/lookup-resolver.js +1 -1
  169. package/src/utils/payload-loader.js +1 -1
  170. package/src/utils/processor-response.js +1 -1
  171. package/src/utils/rabbitmq.js +1 -1
  172. package/src/utils/redis-client.js +1 -1
  173. package/src/utils/redis-helper.js +1 -1
  174. package/src/utils/request-scope.js +1 -1
  175. package/src/utils/security-checks.js +1 -1
  176. package/src/utils/service-resolver.js +1 -1
  177. package/src/utils/shutdown-coordinator.js +1 -1
  178. package/src/utils/trusted-keys.js +1 -1
  179. package/src/utils/upload-handler.js +1 -1
  180. package/src/utils/upsert-builder.js +1 -1
  181. package/src/utils/workflow-hook-executor.js +1 -1
@@ -879,7 +879,7 @@ module.exports = {
879
879
  type: 'string',
880
880
  required: false,
881
881
  default: null,
882
- description: 'Tipe database (postgres|mysql|oracle). Default: postgres'
882
+ description: 'Tipe database (postgres|mysql|oracle|sqlite). Default: postgres'
883
883
  },
884
884
  force: {
885
885
  type: 'boolean',
@@ -59,7 +59,21 @@ async function connectMysql(driver, config) {
59
59
  multipleStatements: false
60
60
  });
61
61
  return {
62
- exec: (sql) => conn.query(sql),
62
+ exec: async (sql) => {
63
+ try {
64
+ return await conn.query(sql);
65
+ } catch (err) {
66
+ // Idempotensi migrate: MySQL tidak punya `CREATE INDEX IF NOT EXISTS`
67
+ // (lihat statement-modifier.js) sehingga re-run pada index yang sudah
68
+ // ada melempar ER_DUP_KEYNAME (errno 1061). Abaikan khusus untuk
69
+ // CREATE INDEX — setara dengan suppression ORA-00955 di jalur Oracle.
70
+ const isDupIndex = err && (err.errno === 1061 || err.code === 'ER_DUP_KEYNAME');
71
+ if (isDupIndex && /^\s*CREATE\s+(?:UNIQUE\s+)?INDEX\b/i.test(sql)) {
72
+ return [{ affectedRows: 0, suppressed: 'ER_DUP_KEYNAME' }];
73
+ }
74
+ throw err;
75
+ }
76
+ },
63
77
  close: () => conn.end()
64
78
  };
65
79
  }
@@ -82,6 +82,8 @@ function formatReferentialAction(action) {
82
82
  module.exports = {
83
83
  name: 'mysql',
84
84
  maxIdentifierLength: 64,
85
+ // Boolean disimpan sebagai VARCHAR(5) 'true'/'false'; bukan native.
86
+ nativeBoolean: false,
85
87
  mapType,
86
88
  formatDefault,
87
89
  quoteIdentifier,
@@ -87,6 +87,8 @@ module.exports = {
87
87
  // RESTForge target Oracle 19c+ (default COMPATIBLE >= 12.2). Override via
88
88
  // --max-name-length=30 if deploying to legacy database.
89
89
  maxIdentifierLength: 128,
90
+ // Boolean disimpan sebagai VARCHAR2(5) 'true'/'false'; bukan native.
91
+ nativeBoolean: false,
90
92
  mapType,
91
93
  formatDefault,
92
94
  quoteIdentifier,
@@ -81,6 +81,9 @@ function formatReferentialAction(action) {
81
81
  module.exports = {
82
82
  name: 'postgres',
83
83
  maxIdentifierLength: 63,
84
+ // PostgreSQL punya tipe BOOLEAN native; dialect lain menyimpan boolean
85
+ // sebagai VARCHAR 'true'/'false' dan butuh CHECK sebagai penanda boolean.
86
+ nativeBoolean: true,
84
87
  mapType,
85
88
  formatDefault,
86
89
  quoteIdentifier,
@@ -85,6 +85,8 @@ module.exports = {
85
85
  name: 'sqlite',
86
86
  // SQLite has no hard identifier limit; cap at 63 to stay portable with PG/MySQL.
87
87
  maxIdentifierLength: 63,
88
+ // Boolean disimpan sebagai VARCHAR(5) 'true'/'false'; bukan native.
89
+ nativeBoolean: false,
88
90
  mapType,
89
91
  formatDefault,
90
92
  quoteIdentifier,
@@ -54,6 +54,13 @@ function emitAddColumn(tableIR, fieldName, field, dialect) {
54
54
  parts.push(`DEFAULT ${dialect.formatDefault(field.default)}`);
55
55
  }
56
56
 
57
+ // Penanda boolean untuk dialect varchar (selaras dengan create-table emitter):
58
+ // CHECK inline pada kolom agar introspeksi dapat mengenalinya kembali sebagai
59
+ // boolean. PostgreSQL (nativeBoolean) memakai tipe BOOLEAN native, dikecualikan.
60
+ if (field.type === 'boolean' && dialect.nativeBoolean !== true) {
61
+ parts.push(`CHECK (${column} IN ('true', 'false'))`);
62
+ }
63
+
57
64
  if (dialect.name === 'oracle') {
58
65
  // Oracle membungkus column definition di dalam parentheses untuk ADD,
59
66
  // konvensi yang juga mendukung multi-column add bila diperlukan nanti.
@@ -44,9 +44,14 @@ function emitCreateTable(ir, dialect) {
44
44
  lines.push(emitForeignKey(tableName, relName, rel, dialect));
45
45
  }
46
46
 
47
- const checkNames = resolveCheckNames(tableName, checks, dialect.maxIdentifierLength);
48
- for (let i = 0; i < checks.length; i++) {
49
- const check = checks[i];
47
+ // Gabungkan CHECK eksplisit (dari SDF) dengan CHECK boolean yang di-derive
48
+ // otomatis untuk dialect non-native-boolean (mysql/oracle/sqlite). CHECK ini
49
+ // menjadi penanda deterministik agar introspeksi dapat mengenali kembali kolom
50
+ // VARCHAR sebagai boolean (lihat database-introspector.getBooleanColumns).
51
+ const allChecks = (Array.isArray(checks) ? checks : []).concat(deriveBooleanChecks(fields, dialect));
52
+ const checkNames = resolveCheckNames(tableName, allChecks, dialect.maxIdentifierLength);
53
+ for (let i = 0; i < allChecks.length; i++) {
54
+ const check = allChecks[i];
50
55
  if (check._invalid) continue;
51
56
  lines.push(emitCheck(check, checkNames[i], dialect));
52
57
  }
@@ -54,6 +59,29 @@ function emitCreateTable(ir, dialect) {
54
59
  return `CREATE TABLE ${dialect.formatTableIdentifier(ir)} (\n${lines.join(',\n')}\n)`;
55
60
  }
56
61
 
62
+ /**
63
+ * Derive CHECK constraint penanda boolean untuk dialect yang menyimpan boolean
64
+ * sebagai VARCHAR (mysql/oracle/sqlite). Bentuk: `col IN ('true','false')`.
65
+ *
66
+ * PostgreSQL (nativeBoolean === true) memakai tipe BOOLEAN native sehingga tidak
67
+ * memerlukan penanda dan dikecualikan. CHECK ini TIDAK ditulis user di SDF;
68
+ * kata kunci `boolean` di field sudah cukup, generator yang menyisipkannya.
69
+ *
70
+ * @param {Object} fields - Map fieldName -> field IR (punya .type)
71
+ * @param {Object} dialect - Dialect aktif
72
+ * @returns {Array<{field:string, op:string, value:string[]}>}
73
+ */
74
+ function deriveBooleanChecks(fields, dialect) {
75
+ if (dialect.nativeBoolean === true) return [];
76
+ const derived = [];
77
+ for (const [fieldName, field] of Object.entries(fields)) {
78
+ if (field && field.type === 'boolean') {
79
+ derived.push({ field: fieldName, op: 'in', value: ['true', 'false'] });
80
+ }
81
+ }
82
+ return derived;
83
+ }
84
+
57
85
  function emitColumn(fieldName, field, dialect) {
58
86
  const parts = [`${COLUMN_INDENT}${dialect.quoteIdentifier(fieldName)}`, dialect.mapType(field)];
59
87
 
@@ -13,8 +13,12 @@ const RE_BEGIN = /^\s*BEGIN\b/i;
13
13
  * `--drop=false` (statement re-run idempotent).
14
14
  *
15
15
  * Rules per dialect:
16
- * - postgres / mysql / sqlite: insert `IF NOT EXISTS ` after `CREATE TABLE `
17
- * or `CREATE INDEX ` (also `CREATE UNIQUE INDEX `).
16
+ * - postgres / sqlite: insert `IF NOT EXISTS ` after `CREATE TABLE ` or
17
+ * `CREATE INDEX ` (also `CREATE UNIQUE INDEX `).
18
+ * - mysql: insert `IF NOT EXISTS ` after `CREATE TABLE ` saja. MySQL TIDAK
19
+ * mendukung `CREATE INDEX IF NOT EXISTS` (syntax error), jadi CREATE INDEX
20
+ * dibiarkan apa adanya — valid pada first-run; re-run index yang sudah ada
21
+ * akan gagal duplicate (keterbatasan MySQL, gunakan --drop untuk recreate).
18
22
  * - oracle: wrap CREATE TABLE / CREATE INDEX in a PL/SQL block that suppresses
19
23
  * SQLCODE -955 (ORA-00955: name is already used by an existing object).
20
24
  * - DROP statements pass through unchanged (the emitter already uses IF EXISTS).
@@ -53,6 +57,12 @@ function applyIfNotExistsModifier(statement, dialect) {
53
57
  return statement.replace(RE_CREATE_TABLE, '$1CREATE TABLE IF NOT EXISTS ');
54
58
  }
55
59
 
60
+ // MySQL tidak mendukung `CREATE INDEX IF NOT EXISTS` (parser error). Biarkan
61
+ // statement CREATE INDEX apa adanya; valid pada first-run.
62
+ if (dialect === 'mysql') {
63
+ return statement;
64
+ }
65
+
56
66
  return statement.replace(RE_CREATE_INDEX, (match, leading, unique) => {
57
67
  const uniquePart = unique ? 'UNIQUE ' : '';
58
68
  return `${leading}CREATE ${uniquePart}INDEX IF NOT EXISTS `;
@@ -189,6 +189,40 @@ class PayloadGenerator {
189
189
  ));
190
190
  }
191
191
 
192
+ /**
193
+ * Enrich detailedColumns dengan menandai kolom boolean berdasarkan output
194
+ * db.getBooleanColumns() (deteksi penanda CHECK `IN ('true','false')` pada
195
+ * dialect varchar mysql/oracle/sqlite). Kolom yang cocok di-set
196
+ * data_type='boolean' agar generateFieldValidation menghasilkan type:'boolean',
197
+ * konsisten dengan PostgreSQL yang punya kolom BOOLEAN native.
198
+ *
199
+ * Default kolom varchar berbentuk string ('true'/'false', kadang dengan tanda
200
+ * kutip) dinormalisasi ke 'true'/'false' tanpa kutip agar default boolean
201
+ * terdeteksi di generateFieldValidation.
202
+ *
203
+ * @param {Array} detailedColumns - Array from getDetailedColumnInfo()
204
+ * @param {string[]} booleanColumns - Nama kolom boolean (lowercase) dari getBooleanColumns()
205
+ * @returns {Array} New array dengan data_type='boolean' pada kolom yang cocok
206
+ */
207
+ enrichDetailedColumnsWithBoolean(detailedColumns, booleanColumns) {
208
+ if (!Array.isArray(detailedColumns)) return [];
209
+ if (!Array.isArray(booleanColumns) || booleanColumns.length === 0) return detailedColumns;
210
+
211
+ const boolSet = new Set(booleanColumns.map(c => String(c).toLowerCase()));
212
+
213
+ return detailedColumns.map(col => {
214
+ if (!boolSet.has(String(col.column_name).toLowerCase())) return col;
215
+
216
+ let def = col.column_default;
217
+ if (typeof def === 'string') {
218
+ const stripped = def.trim().replace(/^'(.*)'$/, '$1').toLowerCase();
219
+ if (stripped === 'true' || stripped === 'false') def = stripped;
220
+ }
221
+
222
+ return { ...col, data_type: 'boolean', udt_name: 'boolean', column_default: def };
223
+ });
224
+ }
225
+
192
226
  /**
193
227
  * Generate fieldValidation array from detailed column info
194
228
  * Includes: primary key (UUID/auto-generate), numeric, boolean, date/time, and
@@ -695,7 +729,11 @@ class PayloadGenerator {
695
729
  // Generate fieldValidation for special fields
696
730
  const detailedColumns = await this.db.getDetailedColumnInfo(args.table);
697
731
  const constraints = await this.db.getConstraints(args.table);
698
- const enrichedColumns = this.enrichDetailedColumnsWithConstraints(detailedColumns, constraints);
732
+ const booleanColumns = typeof this.db.getBooleanColumns === 'function'
733
+ ? await this.db.getBooleanColumns(args.table)
734
+ : [];
735
+ let enrichedColumns = this.enrichDetailedColumnsWithConstraints(detailedColumns, constraints);
736
+ enrichedColumns = this.enrichDetailedColumnsWithBoolean(enrichedColumns, booleanColumns);
699
737
  const fieldValidation = this.generateFieldValidation(enrichedColumns, payloadData.fieldName, payloadData.primaryKey);
700
738
  if (fieldValidation.length > 0) {
701
739
  payloadData.fieldValidation = fieldValidation;
@@ -1229,7 +1267,11 @@ class SchemaValidator {
1229
1267
  }
1230
1268
 
1231
1269
  // Re-generate fieldValidation
1232
- const enrichedColumns = tempGenerator.enrichDetailedColumnsWithConstraints(detailedColumns, constraints);
1270
+ const booleanColumns = typeof this.db.getBooleanColumns === 'function'
1271
+ ? await this.db.getBooleanColumns(tableName)
1272
+ : [];
1273
+ let enrichedColumns = tempGenerator.enrichDetailedColumnsWithConstraints(detailedColumns, constraints);
1274
+ enrichedColumns = tempGenerator.enrichDetailedColumnsWithBoolean(enrichedColumns, booleanColumns);
1233
1275
  const fieldValidation = tempGenerator.generateFieldValidation(enrichedColumns, newFieldName, primaryKey);
1234
1276
  if (fieldValidation.length > 0) {
1235
1277
  updatedPayload.fieldValidation = fieldValidation;
@@ -98,6 +98,13 @@ function normalizeDatabaseType(dataType, udtName, col) {
98
98
  return dt || 'unknown';
99
99
  }
100
100
 
101
+ // String family: varchar/text/char direpresentasikan sebagai satu tipe 'string'
102
+ // di RDF (fieldValidation), sehingga ketiganya interchangeable saat drift check.
103
+ const STRING_FAMILY = new Set(['varchar', 'text', 'char']);
104
+ function isStringFamilyEquivalent(a, b) {
105
+ return STRING_FAMILY.has(a) && STRING_FAMILY.has(b);
106
+ }
107
+
101
108
  /**
102
109
  * Bandingkan satu payload dengan schema database aktual.
103
110
  *
@@ -144,6 +151,25 @@ async function compareSchemaStrict(payload, db, options = {}) {
144
151
  return result;
145
152
  }
146
153
 
154
+ // Boolean-aware: pada dialect varchar (mysql/oracle/sqlite), kolom boolean
155
+ // disimpan sebagai VARCHAR dengan penanda CHECK `IN ('true','false')`.
156
+ // Tanpa normalisasi ini, drift check membandingkan payload type 'boolean'
157
+ // vs DB type 'varchar' dan salah melaporkan type change. Override data_type
158
+ // ke 'boolean' untuk kolom bermarker agar konsisten dengan introspeksi
159
+ // payload generate (enrichDetailedColumnsWithBoolean di payload-runner).
160
+ if (typeof db.getBooleanColumns === 'function') {
161
+ const booleanColumns = await db.getBooleanColumns(tableName);
162
+ if (Array.isArray(booleanColumns) && booleanColumns.length > 0) {
163
+ const boolSet = new Set(booleanColumns.map(c => String(c).toLowerCase()));
164
+ for (const col of dbColumns) {
165
+ if (boolSet.has(String(col.column_name).toLowerCase())) {
166
+ col.data_type = 'boolean';
167
+ col.udt_name = 'boolean';
168
+ }
169
+ }
170
+ }
171
+ }
172
+
147
173
  const dbColumnSet = new Set(dbColumns.map(c => c.column_name));
148
174
  const dbColumnList = dbColumns.map(c => c.column_name);
149
175
  const dbColumnMap = {};
@@ -207,7 +233,11 @@ async function compareSchemaStrict(payload, db, options = {}) {
207
233
  const payloadType = normalizePayloadValidationType(validation);
208
234
  if (!payloadType) continue;
209
235
 
210
- if (dbType !== payloadType) {
236
+ // varchar/text/char saling ekuivalen: RDF hanya punya satu tipe 'string'
237
+ // (dipetakan ke 'varchar') untuk ketiganya, sedangkan introspeksi DB
238
+ // membedakannya. Tanpa ini, kolom TEXT (mis. MySQL yang melaporkan
239
+ // character_maximum_length) salah dilaporkan drift varchar -> text.
240
+ if (dbType !== payloadType && !isStringFamilyEquivalent(dbType, payloadType)) {
211
241
  result.typeChanges.push({
212
242
  column: field,
213
243
  payloadType,
@@ -1 +1 @@
1
- function a0_0x55aa(_0x48a0fb,_0x357c6a){_0x48a0fb=_0x48a0fb-0x175;const _0x285fb0=a0_0x285f();let _0x55aa03=_0x285fb0[_0x48a0fb];if(a0_0x55aa['MTFens']===undefined){var _0x7cb3e2=function(_0x1cabb2){const _0x420d9d='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x45f712='',_0x4b30bf='';for(let _0xc15954=0x0,_0x2400fa,_0x4277ed,_0x18e1f9=0x0;_0x4277ed=_0x1cabb2['charAt'](_0x18e1f9++);~_0x4277ed&&(_0x2400fa=_0xc15954%0x4?_0x2400fa*0x40+_0x4277ed:_0x4277ed,_0xc15954++%0x4)?_0x45f712+=String['fromCharCode'](0xff&_0x2400fa>>(-0x2*_0xc15954&0x6)):0x0){_0x4277ed=_0x420d9d['indexOf'](_0x4277ed);}for(let _0x32dcc7=0x0,_0x31d7e9=_0x45f712['length'];_0x32dcc7<_0x31d7e9;_0x32dcc7++){_0x4b30bf+='%'+('00'+_0x45f712['charCodeAt'](_0x32dcc7)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x4b30bf);};a0_0x55aa['fggdAy']=_0x7cb3e2,a0_0x55aa['qhcvLj']={},a0_0x55aa['MTFens']=!![];}const _0x38ff3f=_0x285fb0[0x0],_0x32d968=_0x48a0fb+_0x38ff3f,_0x7b4475=a0_0x55aa['qhcvLj'][_0x32d968];return!_0x7b4475?(_0x55aa03=a0_0x55aa['fggdAy'](_0x55aa03),a0_0x55aa['qhcvLj'][_0x32d968]=_0x55aa03):_0x55aa03=_0x7b4475,_0x55aa03;}const a0_0xfa6499=a0_0x55aa;(function(_0x267dca,_0x42bce3){const _0x33d074=a0_0x55aa,_0x10587d=_0x267dca();while(!![]){try{const _0x42232a=-parseInt(_0x33d074(0x1c0))/0x1+-parseInt(_0x33d074(0x19b))/0x2*(parseInt(_0x33d074(0x1c3))/0x3)+parseInt(_0x33d074(0x1c8))/0x4*(parseInt(_0x33d074(0x184))/0x5)+-parseInt(_0x33d074(0x176))/0x6+parseInt(_0x33d074(0x192))/0x7*(parseInt(_0x33d074(0x17b))/0x8)+parseInt(_0x33d074(0x1bb))/0x9+-parseInt(_0x33d074(0x182))/0xa*(-parseInt(_0x33d074(0x1a9))/0xb);if(_0x42232a===_0x42bce3)break;else _0x10587d['push'](_0x10587d['shift']());}catch(_0xd872ce){_0x10587d['push'](_0x10587d['shift']());}}}(a0_0x285f,0xc5a5f));const FORBIDDEN_FRONTEND_FIELDS=['widgetType','layout',a0_0xfa6499(0x197),a0_0xfa6499(0x180),'color'],ALLOWED_PARAM_TYPES=[a0_0xfa6499(0x191),'number','boolean',a0_0xfa6499(0x17c)],FRONTEND_CONCERN_REASONS={'widgetType':a0_0xfa6499(0x1bd),'layout':'Layout\x20is\x20a\x20frontend\x20rendering\x20concern.','title':'UI\x20label\x20is\x20a\x20frontend\x20rendering\x20concern.','subtitle':a0_0xfa6499(0x1a0),'color':a0_0xfa6499(0x1c4)},PAYLOAD_SHAPE={'discriminator':{'field':'widgets','presentMeans':'dashboard\x20payload','absentMeans':'Not\x20a\x20dashboard\x20payload\x20(likely\x20CRUD\x20with\x20tableName,\x20or\x20invalid)','conflictsWith':a0_0xfa6499(0x1b5),'conflictRationale':'A\x20payload\x20with\x20both\x20\x27widgets\x27\x20and\x20\x27tableName\x27\x20is\x20rejected\x20by\x20DashboardValidator.\x20Pick\x20one\x20shape.'},'topLevelAllowed':[{'name':'widgets','type':'array','required':!![],'minItems':0x1,'description':'List\x20of\x20widget\x20definitions.\x20Order\x20is\x20informational\x20only\x20(response\x20keys\x20are\x20by\x20widget\x20id,\x20not\x20array\x20index).'},{'name':'params','type':a0_0xfa6499(0x17f),'required':![],'description':a0_0xfa6499(0x18d)},{'name':'cache','type':a0_0xfa6499(0x17f),'required':![],'description':'Optional\x20cache\x20configuration.\x20See\x20cacheSpec\x20for\x20details.'}],'topLevelForbidden':[{'name':'tableName','category':'shape-conflict','reason':'Reserved\x20for\x20CRUD\x20payloads.\x20A\x20dashboard\x20payload\x20must\x20declare\x20\x27widgets\x27\x20instead.'},...FORBIDDEN_FRONTEND_FIELDS[a0_0xfa6499(0x1b9)](_0x45f712=>({'name':_0x45f712,'category':'frontend-concern','reason':FRONTEND_CONCERN_REASONS[_0x45f712]}))]},WIDGET_SPEC={'requiredFields':[{'name':'id','type':a0_0xfa6499(0x191),'constraint':a0_0xfa6499(0x193),'description':'Widget\x20identifier;\x20used\x20as\x20the\x20response\x20key\x20in\x20the\x20dashboard\x20envelope.'}],'exclusiveQueryFields':{'rule':a0_0xfa6499(0x1a1),'options':[{'name':'query','type':'string','format':a0_0xfa6499(0x1ad),'description':'Single\x20SQL\x20query\x20for\x20the\x20widget.','responseShape':a0_0xfa6499(0x19a)},{'name':a0_0xfa6499(0x1bf),'type':'object','format':'key→file:relative/path/to/query.sql','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_0xfa6499(0x1c1),'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':a0_0xfa6499(0x199),'required':!![],'allowedValues':ALLOWED_PARAM_TYPES,'description':'Param\x20data\x20type.\x20Validates\x20request\x20body\x20and\x20shapes\x20runtime\x20parameter\x20binding.'},{'name':'required','required':![],'type':a0_0xfa6499(0x1b0),'default':![],'description':a0_0xfa6499(0x1c6)},{'name':'default','required':![],'type':a0_0xfa6499(0x1c5),'description':a0_0xfa6499(0x179)}]},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':a0_0xfa6499(0x186)},{'appliesTo':'widget.queries.<key>\x20with\x20SQL\x20returning\x201\x20row\x20×\x201\x20column','rule':'Collapse\x20to\x20scalar\x20primitive\x20(the\x20value\x20of\x20the\x20single\x20column).','exampleSqlShape':a0_0xfa6499(0x1a8),'exampleResponse':a0_0xfa6499(0x17a)},{'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':a0_0xfa6499(0x1bc),'rule':'Return\x20as\x20array\x20of\x20objects\x20(no\x20collapse).','exampleSqlShape':a0_0xfa6499(0x178),'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_0xfa6499(0x1aa),'queries':{'value':a0_0xfa6499(0x1ca),'trend':'file:query/<path>/trend.sql','items':'file:query/<path>/breakdown.sql'}},'sqlShapesPerKey':[{'key':'value','shape':'1\x20row\x20×\x201\x20column','outputColumns':['value'],'collapseRule':'scalar\x20primitive'},{'key':'trend','shape':'1\x20row\x20×\x202\x20columns','outputColumns':['direction',a0_0xfa6499(0x1cb)],'collapseRule':'object'},{'key':a0_0xfa6499(0x1b1),'shape':'N\x20rows\x20×\x202\x20columns','outputColumns':['label','value'],'collapseRule':a0_0xfa6499(0x196)}],'responseShape':{'value':a0_0xfa6499(0x1a7),'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':a0_0xfa6499(0x194)},{'id':a0_0xfa6499(0x1ae),'name':a0_0xfa6499(0x1c2),'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_0xfa6499(0x1ca),'trend':'file:query/<path>/trend.sql','points':a0_0xfa6499(0x18f)}},'sqlShapesPerKey':[{'key':'value','shape':a0_0xfa6499(0x19d),'outputColumns':['value'],'collapseRule':'scalar\x20primitive'},{'key':'trend','shape':a0_0xfa6499(0x1ba),'outputColumns':['direction','pct'],'collapseRule':a0_0xfa6499(0x17f)},{'key':a0_0xfa6499(0x19e),'shape':a0_0xfa6499(0x1a6),'outputColumns':['period','value'],'collapseRule':a0_0xfa6499(0x196)}],'responseShape':{'value':'\x222420\x22','trend':a0_0xfa6499(0x1af),'points':a0_0xfa6499(0x181)},'referenceWidgetId':'avg_daily_sales','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_0xfa6499(0x175),'name':a0_0xfa6499(0x19c),'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':'file:query/<path>/trend.sql','target':'file:query/<path>/target.sql'}},'sqlShapesPerKey':[{'key':a0_0xfa6499(0x1c9),'shape':a0_0xfa6499(0x19d),'outputColumns':['value\x20(or\x20current)'],'collapseRule':a0_0xfa6499(0x1a3)},{'key':a0_0xfa6499(0x185),'shape':'1\x20row\x20×\x202\x20columns','outputColumns':[a0_0xfa6499(0x18a),a0_0xfa6499(0x1cb)],'collapseRule':'object'},{'key':'target','shape':'1\x20row\x20×\x201\x20column','outputColumns':['target'],'collapseRule':a0_0xfa6499(0x1a3)}],'responseShape':{'value':a0_0xfa6499(0x177),'trend':'{\x20\x22direction\x22:\x20\x22down\x22,\x20\x22pct\x22:\x20\x222.2\x22\x20}','target':a0_0xfa6499(0x1b7)},'referenceWidgetId':'orders_this_month','socNotes':a0_0xfa6499(0x189)}],NAMING_CONVENTION={'dashboardName':{'constraint':'MUST\x20start\x20with\x20\x27dash-\x27\x20prefix','minLength':0x6,'maxLength':0x32,'regex':'^dash-[a-zA-Z0-9_-]+$','examples':['dash-sales',a0_0xfa6499(0x18e),a0_0xfa6499(0x188)],'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':a0_0xfa6499(0x1a2),'path':'/api/{project}/{name}/dashboard','exampleFull':'POST\x20/api/mini-inventory/dash-inbound/dashboard','requestBodyShape':{'params':'object\x20—\x20values\x20for\x20declared\x20params\x20(validated\x20against\x20params\x20contract;\x20missing\x20required\x20→\x20400,\x20type\x20mismatch\x20→\x20400)','widgets':a0_0xfa6499(0x1b4)},'responseShape':{'envelope':'{\x20success:\x20boolean,\x20data:\x20{\x20<widgetId>:\x20<perWidgetResponse>,\x20...\x20}\x20}','perWidgetResponse':a0_0xfa6499(0x1b2)}},FILE_REFERENCE_CONVENTION={'format':a0_0xfa6499(0x1ad),'pathRelativeTo':a0_0xfa6499(0x195),'fileExtensionPolicy':a0_0xfa6499(0x1ac),'resolvedAt':a0_0xfa6499(0x17d),'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':a0_0xfa6499(0x18b),'regex':a0_0xfa6499(0x17e),'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':a0_0xfa6499(0x1a4),'exampleSql':a0_0xfa6499(0x187),'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':'top-level\x20\x27cache\x27\x20object','optional':!![],'rationale':a0_0xfa6499(0x183),'fields':[{'name':'enabled','type':a0_0xfa6499(0x1b0),'required':!![],'description':a0_0xfa6499(0x1b8)},{'name':'ttl','type':'number','required':![],'constraint':a0_0xfa6499(0x18c),'default':'inherits\x20CACHE_TTL\x20env','description':a0_0xfa6499(0x1a5)},{'name':a0_0xfa6499(0x1c7),'type':'array<string>','required':![],'default':'[]','description':'List\x20of\x20CRUD\x20table\x20names\x20that,\x20when\x20written,\x20will\x20trigger\x20invalidation\x20of\x20this\x20dashboard\x20cache.'}],'validation':{'sqlCrossReference':a0_0xfa6499(0x19f),'errorOn':[a0_0xfa6499(0x1b6),'Table\x20declared\x20in\x20invalidates,\x20but\x20not\x20detected\x20in\x20any\x20widget\x20SQL\x20(typo\x20or\x20dead\x20entry)'],'warningOn':[a0_0xfa6499(0x1be)]}},DOCUMENTATION_URL=a0_0xfa6499(0x1ab),DASHBOARD_CATALOG={'schemaVersion':a0_0xfa6499(0x198),'source':'dashboard-catalog','summary':{'totalAllowedTopLevelFields':PAYLOAD_SHAPE['topLevelAllowed'][a0_0xfa6499(0x190)],'totalForbiddenFrontendFields':FORBIDDEN_FRONTEND_FIELDS[a0_0xfa6499(0x190)],'totalParamTypes':ALLOWED_PARAM_TYPES[a0_0xfa6499(0x190)],'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};function a0_0x285f(){const _0x421ac2=['Bwv0CMLJx3nWyxjRBgLUzq','EYaIzgLYzwn0Aw9UiJOGiNvWiIWGiNbJDci6iciYlJyIih0','yM9VBgvHBG','AxrLBxm','rgv0zxjTAw5LzcbIEsbZy2fSyxjdB2XSyxbZzvj1BgvZlIbgywLSzwqGD2LKz2v0CYbWCM9KDwnLihSGzxjYB3i6icCUlI4Nih0GyMXVy2SGD2L0Acb0B3aTBgv2zwWGC3vJy2vZCYbZDgLSBcb0CNvLicHVBMuGD2LKz2v0igzHAwX1CMuGzg9LCYbot1qGzMfPBcb0AguGzgfZAgjVyxjKks4','zxHWB3j0CW','yxjYyxK8C3rYAw5NpIWGB3b0Aw9UywWG4Ocuihn1yNnLDcbVzIb3AwrNzxqGsurZihrVigv4zwn1DguUie9TAxqGDg8GzxHLy3v0zsbHBgWGzgvJBgfYzwqGD2LKz2v0CY4','DgfIBgvoyw1L','vgfIBguGyxbWzwfYCYbPBIbtuuWGqu5eigLUig1LDgfKyxrHihbYB2PLy3qSigj1DcbTAxnZAw5NigzYB20GAw52ywXPzgf0zxmGkgnHy2HLihn0ywXLihjPC2SP','iJi4odqI','vg9Nz2XLignHy2HLigzLyxr1CMuGzM9YihrOAxmGzgfZAgjVyxjKlG','BwfW','msbYB3CGW5CGmIbJB2X1Bw5Z','nJu4nJeXrfPsve9N','D2LKz2v0lNf1zxjPzxmUpgTLEt4GD2L0AcbtuuWGCMv0DxjUAw5Nie4GCM93CW','vMLZDwfSihzHCMLHBNqGkgrVBNv0lcbIyxiSihbPzsWGyxjLysKGAxmGysbMCM9UDgvUzcbYzw5KzxjPBMCGy29Uy2vYBIaOC2vWyxjHDgLVBIbVzIbJB25JzxjUCYKU','vgfIBguGzgv0zwn0zwqGAw4Gu1fmlcbIDxqGBM90ihjLz2LZDgvYzwqGyxmGq1jvrcbLBMrWB2LUDcbPBIbTzxrHzgf0ysbWCM9Qzwn0icHSAwTLBhKGysb2Awv3lcbdveuGywXPyxmSig9YignYB3nZlxbYB2PLy3qGDgfIBguG4OcuignHC2nHzguGD2LSBcbUB3qGzMLYzsK','CxvLCMLLCW','mtiWota4nNL1D3zAyW','Dg9WlwXLDMvSicDWyxjHBxmNig9IAMvJDa','twv0CMLJicSGu3bHCMTSAw5L','mtmYmJy3BNzuzgPz','vMLZDwfSignVBg9YigLZigeGzNjVBNrLBMqGCMvUzgvYAw5NignVBMnLCM4U','yw55icHTDxn0igjLignVBxbHDgLIBguGD2L0AcbKzwnSyxjLzcaNDhLWzsCP','v2HLBIb0CNvLlcb0AguGCMvXDwvZDcbIB2r5ie1vu1qGAw5JBhvKzsb0AgLZihbHCMfTicHVDgHLCNDPC2uGndaWks4','Aw52ywXPzgf0zxm','ndi1mty4AMrSq21Z','DMfSDwu','zMLSztPXDwvYEs88Cgf0Ad4VDMfSDwuUC3fS','Cgn0','Bwv0CMLJx3bYB2DYzxnZx3rVx2DVywW','ndy0nJC4neHPBxHHqq','iJe4mZyI','tIbYB3DZimoxie0Gy29SCW','rgvMyxvSDcb2ywX1zsbHChbSAwvKihDOzw4GDgHLihjLCxvLC3qGB21PDhmGDgHPCYbWyxjHBs4GvMfSAwrHDg9YigrVzxmGtK9uihn0CMLJDgX5ihr5CguTy2HLy2SGzgvMyxvSDdSGCNvUDgLTzsbPCYbYzxnWB25ZAwjSzsbMB3iGy29TCgf0AwjPBgL0Es4','iNzHBhvLiJOGiJy5nZaWiG','ndC5mdrhDuLTwMu','zgf0zq','z2vUzxjHDgLVBIb0Aw1LicHot1qGCNvUDgLTzsK','kd88itOPoIHBys16qs1Ax11Bys16qs1Amc05x10Qkq','B2jQzwn0','C3vIDgL0Bgu','w3SGiNbLCMLVzci6iciYmdi2lta0lti0iIWGiNzHBhvLiJOGiJe4ntaIih0Sic4UlIbD','mtb6v2nUyNi','rgfZAgjVyxjKigvUzhbVAw50ig1HEsbVChqTAw4GDg8GuMvKAxmTyMfZzwqGy2fJAguUifbHDhrLCM4GzM9SBg93CYbWCM9JzxnZB3iGy2fJAguGkhnLzsbMzwf0lwnHy2HLlM1Kks4Gq2fJAguGC2nVCguGAxmGDgHLigz1BgWGCMvZCg9UC2uGzw52zwXVCgu7ig9UzsbJywnOzsbLBNrYEsbWzxiGkhbHCMfTCYaRihDPzgDLDhnBxsbZDwjZzxqPignVBwjPBMf0Aw9UlG','mZbjyKzjy3a','DhjLBMq','iNnOB3bWAw5Nx2nHDgvNB3jPzxmIoIb7icjPDgvTCYi6ifT7icjUyw1LiJOGiKXHBMrZiIb9lcb7icjUyw1LiJOGiKHVDxnLCYiGFv0GFq','u0vmrunuicOGrLjptsbZDg9JA19PBMjVDw5KifDirvjfievyvfjbq1qOwuvbuIbguK9nigLUyM91BMrFzgf0zsKGpsa6EwvHCG','zgfZAc1HDxrOB3iTC3rHDhm','rNjVBNrLBMqGy29TChv0zxmGDg9Fz29HBca9ihrHCMDLDcaTihzHBhvLigfUzcbWy3qGpsbYB3vUzcH2ywX1zsaVihrHCMDLDcaQideWmcKGzM9YihrOzsbWCM9NCMvZCYbIyxiUifzPC3vHBcb3Awr0AcbPCYbWCMvZzw50yxrPB25HBcbHBMqGBxvZDcbot1qGBgL2zsbPBIb0AguGyMfJA2vUzcbWyxLSB2fKlIbjzIbWCM9NCMvZCYbPBNzVBhzLCYbJB21WBgv4igj1C2LUzxnZihj1BgvZicHLlMCUigv4y2X1zguGD2vLA2vUzhmSihbYB3jHDgvKihDVCMTKyxLZksWGDxnLigeGC2LUz2XLig11BhrPlwnVBhvTBIbXDwvYEsbZBYaNCgn0jYbPCYbHihn0ywjSzsbIDxnPBMvZCYbMywn0ihjHDgHLCIb0AgfUihzPC3vHBcb3Awr0Ac4','zgLYzwn0Aw9U','oNbHCMfTtMfTzq','pJ0GmcaOC2vJB25KCYK','ugfYyw1LDgvYignVBNrYywn0igzVCIb0AguGzgfZAgjVyxjKlIbfywnOigTLEsbPCYbHihbHCMfTig5HBwu7ihzHBhvLCYbKzxnJCMLIzsb0ExbLl3jLCxvPCMvKl2rLzMf1BhqUifbSywnLAg9SzgvYCYbPBNnPzguGD2LKz2v0ifnrtcbTDxn0ihjLzMvYzw5JzsbKzwnSyxjLzcbWyxjHBsbUyw1LCY4','zgfZAc1PBMjVDw5K','zMLSztPXDwvYEs88Cgf0Ad4VCg9PBNrZlNnXBa','BgvUz3rO','C3rYAw5N','mti2wKvuAKLR','BM9UlwvTChr5lcb1BMLXDwuGywnYB3nZihDPzgDLDhmGAw4GDgHLihnHBwuGCgf5Bg9Hza','rNjVBNrLBMqGzgv0zxjTAw5LCYbKB251Dc9WAwuGDMfYAwfUDcWGy29SB3iGCgvYignHDgvNB3j5lcbHBMqGBgfIzwWGB3jKzxiUieLMihbLCI1JyxrLz29YEsbWzxjJzw50ywDLigLZig5LzwrLzcbMB3iGDgHLigrVBNv0igfYyYWGzNjVBNrLBMqGy29TChv0zxmGAxqGzNjVBsbPDgvTC1TPxs52ywX1zsaVihn1BsHPDgvTC1SQxs52ywX1zsKUie5Vig5LzwqGDg8GC2vUzcaNCgn0jYbMCM9TigjHy2TLBMqGDw5SzxnZihrOzsbMAwD1CMuGAxmGysbZDgfIBguGyNvZAw5LC3mGy2fSy3vSyxrPB24GAw5KzxbLBMrLBNqGB2yGDMLZDwfSihjLBMrLCMLUzY4','Cgf5Bg9Hzcbku09oigzPBguGBg9JyxrPB24','yxjYyxKGB2yGB2jQzwn0CW','DgL0Bgu','ms4W','DhLWzq','qwX3yxLZihSGAxrLBxm6ifSUlI5Dih0GCMvNyxjKBgvZCYbVzIbtuuWGCMvZDwX0ihnOyxbLlG','mtHrvK5WrNO','twv0CMLJicSGuhjVz3jLC3mGDg8Gr29HBa','msbYB3CGW5CGmsbJB2X1Bw4','Cg9PBNrZ','v2HLBIbJywnOzs5LBMfIBgvKid09psb0CNvLigfUzcbPBNzHBgLKyxrLCYbPCYbUB24Tzw1WDhK6ihzHBgLKyxrVCIbLEhrYywn0CYb0ywjSzsbJyw5KAwrHDgvZigzYB20GD2LKz2v0ifnrtcaOCMvNzxGGrLjpts9kt0LoksWGy3jVC3mTCMvMzxjLBMnLCYb3AxrOig1LDgfKyxrHl3TWCM9Qzwn0Fs5QC29UicHLBMrWB2LUDhnBkL0UDgfIBgvoyw1LihDOzxjLihr5CguGpt09icjTB2r1BguIksWGyw5KigfZC2vYDhmGzxf1ywXPDhKGB2yGzxHWzwn0zwqGDNmGzgvJBgfYzwqGC2v0CY4GtwLZBwf0y2HLCYbHCMuGCMvWB3j0zwqGCgvYignHDgvNB3j5icHTAxnZAw5NlcbLEhrYysWGDw5TyxrJAgvKks4','vuKGBgfIzwWGAxmGysbMCM9UDgvUzcbYzw5KzxjPBMCGy29Uy2vYBI4','qsb3AwrNzxqGtvvtvcbKzwnSyxjLigv4ywn0BhKGB25Lig9MoIaNCxvLCNKNie9sicDXDwvYAwvZjY4GqM90AcbVCIbUzwL0AgvYigLZihjLAMvJDgvKlG','ue9tva','C2nHBgfYihbYAw1PDgL2zq','rxzLCNKGCgXHy2vOB2XKzxiGDxnLzcbPBIbtuuWGtvvtvcbIzsbKzwnSyxjLzcbPBIaNCgfYyw1ZjY4GvMfSAwrHDg9YihrOCM93CYbfCNjVCIb3AxrOig1LC3nHz2uGzM9YBwf0oIaIv2LKz2v0icC8Awq+jYbXDwvYEsaNpgXHyMvSpICGDxnLCYb1BMrLy2XHCMvKihbSywnLAg9SzgvYicC6phrVA2vUpICGkgrLy2XHCMuGAw4Gj3bHCMfTCYCPiI4','vgLTzs10BY1SAxzLigLUihnLy29UzhmUidaGzwzMzwn0AxzLBhKGzgLZywjSzxmGy2fJAguGzM9YihrOAxmGzw50CNKU','tIbYB3DZimoxidiGy29SDw1UCW','iJy5nZaWiG','msbYB3CGW5CGmsbJB2WSig91Dhb1DcbJB2X1Bw4Gj3zHBhvLjW','mJyWodmYmZnotvnwEMy','phDPzgDLDf9Pzd4','Ahr0Chm6lY9Yzxn0zM9Yz2uUzgv2l2rVy3mVC2vYDMvYl3f1zxj5lwrHDgeVzgfZAgjVyxjK','zNjLztSGlNnXBcbYzwnVBw1LBMrLzcbMB3iGzwrPDg9YigHPz2HSAwDODa','zMLSztPYzwXHDgL2zs9WyxrOl3rVl3f1zxj5lNnXBa'];a0_0x285f=function(){return _0x421ac2;};return a0_0x285f();}module[a0_0xfa6499(0x1b3)]={'DASHBOARD_CATALOG':DASHBOARD_CATALOG};
1
+ function a0_0x3bf7(_0x344e72,_0x2c5bf0){_0x344e72=_0x344e72-0x1b5;const _0x50c030=a0_0x50c0();let _0x3bf7b5=_0x50c030[_0x344e72];if(a0_0x3bf7['EHxYnf']===undefined){var _0xad0780=function(_0x4b6b33){const _0x5740a1='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x5441db='',_0x43e998='';for(let _0x36b160=0x0,_0x1731cb,_0x38e3dd,_0x1bdba0=0x0;_0x38e3dd=_0x4b6b33['charAt'](_0x1bdba0++);~_0x38e3dd&&(_0x1731cb=_0x36b160%0x4?_0x1731cb*0x40+_0x38e3dd:_0x38e3dd,_0x36b160++%0x4)?_0x5441db+=String['fromCharCode'](0xff&_0x1731cb>>(-0x2*_0x36b160&0x6)):0x0){_0x38e3dd=_0x5740a1['indexOf'](_0x38e3dd);}for(let _0x2d08d5=0x0,_0x4f506d=_0x5441db['length'];_0x2d08d5<_0x4f506d;_0x2d08d5++){_0x43e998+='%'+('00'+_0x5441db['charCodeAt'](_0x2d08d5)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x43e998);};a0_0x3bf7['rKAjhf']=_0xad0780,a0_0x3bf7['kWqxGX']={},a0_0x3bf7['EHxYnf']=!![];}const _0x2c88bf=_0x50c030[0x0],_0x4cd53a=_0x344e72+_0x2c88bf,_0x32e915=a0_0x3bf7['kWqxGX'][_0x4cd53a];return!_0x32e915?(_0x3bf7b5=a0_0x3bf7['rKAjhf'](_0x3bf7b5),a0_0x3bf7['kWqxGX'][_0x4cd53a]=_0x3bf7b5):_0x3bf7b5=_0x32e915,_0x3bf7b5;}const a0_0x1a082b=a0_0x3bf7;(function(_0x3e28e0,_0x5cdb9a){const _0x170d4b=a0_0x3bf7,_0x153c54=_0x3e28e0();while(!![]){try{const _0x214def=-parseInt(_0x170d4b(0x1c1))/0x1*(-parseInt(_0x170d4b(0x1d8))/0x2)+-parseInt(_0x170d4b(0x1f9))/0x3*(parseInt(_0x170d4b(0x206))/0x4)+-parseInt(_0x170d4b(0x1c4))/0x5*(-parseInt(_0x170d4b(0x1ef))/0x6)+parseInt(_0x170d4b(0x1e7))/0x7*(-parseInt(_0x170d4b(0x210))/0x8)+-parseInt(_0x170d4b(0x211))/0x9*(parseInt(_0x170d4b(0x1bd))/0xa)+-parseInt(_0x170d4b(0x1e9))/0xb*(-parseInt(_0x170d4b(0x1cb))/0xc)+parseInt(_0x170d4b(0x1da))/0xd;if(_0x214def===_0x5cdb9a)break;else _0x153c54['push'](_0x153c54['shift']());}catch(_0x1bd09a){_0x153c54['push'](_0x153c54['shift']());}}}(a0_0x50c0,0xb4e0d));const FORBIDDEN_FRONTEND_FIELDS=['widgetType','layout',a0_0x1a082b(0x1c0),'subtitle',a0_0x1a082b(0x1dc)],ALLOWED_PARAM_TYPES=['string',a0_0x1a082b(0x1f3),a0_0x1a082b(0x203),'date'],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_0x1a082b(0x1e6),'subtitle':a0_0x1a082b(0x1e6),'color':'Visual\x20color\x20is\x20a\x20frontend\x20rendering\x20concern.'},PAYLOAD_SHAPE={'discriminator':{'field':a0_0x1a082b(0x214),'presentMeans':a0_0x1a082b(0x209),'absentMeans':a0_0x1a082b(0x215),'conflictsWith':a0_0x1a082b(0x1c7),'conflictRationale':'A\x20payload\x20with\x20both\x20\x27widgets\x27\x20and\x20\x27tableName\x27\x20is\x20rejected\x20by\x20DashboardValidator.\x20Pick\x20one\x20shape.'},'topLevelAllowed':[{'name':a0_0x1a082b(0x214),'type':a0_0x1a082b(0x1ff),'required':!![],'minItems':0x1,'description':a0_0x1a082b(0x1fb)},{'name':a0_0x1a082b(0x1f0),'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':'cache','type':'object','required':![],'description':'Optional\x20cache\x20configuration.\x20See\x20cacheSpec\x20for\x20details.'}],'topLevelForbidden':[{'name':'tableName','category':'shape-conflict','reason':'Reserved\x20for\x20CRUD\x20payloads.\x20A\x20dashboard\x20payload\x20must\x20declare\x20\x27widgets\x27\x20instead.'},...FORBIDDEN_FRONTEND_FIELDS[a0_0x1a082b(0x1d0)](_0x5441db=>({'name':_0x5441db,'category':'frontend-concern','reason':FRONTEND_CONCERN_REASONS[_0x5441db]}))]},WIDGET_SPEC={'requiredFields':[{'name':'id','type':a0_0x1a082b(0x213),'constraint':a0_0x1a082b(0x1de),'description':a0_0x1a082b(0x1ba)}],'exclusiveQueryFields':{'rule':'A\x20widget\x20MUST\x20declare\x20exactly\x20one\x20of:\x20\x27query\x27\x20OR\x20\x27queries\x27.\x20Both\x20or\x20neither\x20is\x20rejected.','options':[{'name':'query','type':a0_0x1a082b(0x213),'format':'file:relative/path/to/query.sql','description':a0_0x1a082b(0x1fe),'responseShape':'Always\x20{\x20items:\x20[...]\x20}\x20regardless\x20of\x20SQL\x20result\x20shape.'},{'name':'queries','type':'object','format':'key→file:relative/path/to/query.sql','minKeys':0x1,'description':a0_0x1a082b(0x200),'responseShape':a0_0x1a082b(0x1cf)}]},'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':a0_0x1a082b(0x20d)},{'name':'required','required':![],'type':'boolean','default':![],'description':a0_0x1a082b(0x1ca)},{'name':a0_0x1a082b(0x1cd),'required':![],'type':a0_0x1a082b(0x1b8),'description':a0_0x1a082b(0x1c6)}]},SCALAR_COLLAPSE_RULES=[{'appliesTo':'widget.query\x20(singular)','rule':'Always\x20wrap\x20as\x20{\x20items:\x20[...]\x20}\x20regardless\x20of\x20SQL\x20result\x20shape.','exampleSqlShape':a0_0x1a082b(0x1bf),'exampleResponse':a0_0x1a082b(0x1e0)},{'appliesTo':'widget.queries.<key>\x20with\x20SQL\x20returning\x201\x20row\x20×\x201\x20column','rule':'Collapse\x20to\x20scalar\x20primitive\x20(the\x20value\x20of\x20the\x20single\x20column).','exampleSqlShape':'1\x20row\x20×\x201\x20col,\x20output\x20column\x20\x27value\x27','exampleResponse':'\x22value\x22:\x20\x2269700\x22'},{'appliesTo':a0_0x1a082b(0x1ee),'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':a0_0x1a082b(0x1df),'exampleSqlShape':'N\x20rows\x20×\x20M\x20cols','exampleResponse':a0_0x1a082b(0x20e)}],COMMON_WIDGET_PATTERNS=[{'id':'metric_donut_breakdown','name':a0_0x1a082b(0x1c3),'useCase':a0_0x1a082b(0x1f2),'payloadShape':{'id':'<widget_id>','queries':{'value':'file:query/<path>/value.sql','trend':a0_0x1a082b(0x20c),'items':a0_0x1a082b(0x1e3)}},'sqlShapesPerKey':[{'key':'value','shape':a0_0x1a082b(0x1ec),'outputColumns':[a0_0x1a082b(0x1c5)],'collapseRule':a0_0x1a082b(0x1d4)},{'key':'trend','shape':'1\x20row\x20×\x202\x20columns','outputColumns':[a0_0x1a082b(0x1e2),'pct'],'collapseRule':a0_0x1a082b(0x1d7)},{'key':'items','shape':a0_0x1a082b(0x1ce),'outputColumns':['label',a0_0x1a082b(0x1c5)],'collapseRule':a0_0x1a082b(0x1f4)}],'responseShape':{'value':a0_0x1a082b(0x1b6),'trend':a0_0x1a082b(0x201),'items':a0_0x1a082b(0x20b)},'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':a0_0x1a082b(0x1d1),'name':a0_0x1a082b(0x1c8),'useCase':a0_0x1a082b(0x205),'payloadShape':{'id':'<widget_id>','queries':{'value':a0_0x1a082b(0x1f7),'trend':a0_0x1a082b(0x20c),'points':'file:query/<path>/points.sql'}},'sqlShapesPerKey':[{'key':a0_0x1a082b(0x1c5),'shape':a0_0x1a082b(0x1ec),'outputColumns':['value'],'collapseRule':a0_0x1a082b(0x1d4)},{'key':'trend','shape':a0_0x1a082b(0x1e4),'outputColumns':['direction',a0_0x1a082b(0x1f6)],'collapseRule':'object'},{'key':a0_0x1a082b(0x212),'shape':a0_0x1a082b(0x1ce),'outputColumns':['period','value'],'collapseRule':'array\x20of\x20objects'}],'responseShape':{'value':a0_0x1a082b(0x1d3),'trend':a0_0x1a082b(0x1e1),'points':'[{\x20\x22period\x22:\x20\x222026-04-24\x22,\x20\x22value\x22:\x20\x221850\x22\x20},\x20...\x20]'},'referenceWidgetId':a0_0x1a082b(0x1f1),'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_0x1a082b(0x20f),'name':'Metric\x20+\x20Progress\x20to\x20Goal','useCase':'Headline\x20metric\x20with\x20trend\x20chip\x20and\x20progress\x20bar\x20against\x20a\x20period\x20target.\x20Suitable\x20for\x20widgets\x20like\x20\x27Orders\x20This\x20Month\x27.','payloadShape':{'id':a0_0x1a082b(0x1c2),'queries':{'value':a0_0x1a082b(0x1d9),'trend':'file:query/<path>/trend.sql','target':'file:query/<path>/target.sql'}},'sqlShapesPerKey':[{'key':'value','shape':'1\x20row\x20×\x201\x20column','outputColumns':[a0_0x1a082b(0x1ed)],'collapseRule':'scalar\x20primitive'},{'key':a0_0x1a082b(0x1b5),'shape':'1\x20row\x20×\x202\x20columns','outputColumns':['direction',a0_0x1a082b(0x1f6)],'collapseRule':'object'},{'key':'target','shape':a0_0x1a082b(0x1ec),'outputColumns':['target'],'collapseRule':'scalar\x20primitive'}],'responseShape':{'value':a0_0x1a082b(0x1db),'trend':a0_0x1a082b(0x1fc),'target':a0_0x1a082b(0x1ea)},'referenceWidgetId':'orders_this_month','socNotes':a0_0x1a082b(0x204)}],NAMING_CONVENTION={'dashboardName':{'constraint':'MUST\x20start\x20with\x20\x27dash-\x27\x20prefix','minLength':0x6,'maxLength':0x32,'regex':'^dash-[a-zA-Z0-9_-]+$','examples':[a0_0x1a082b(0x1d5),'dash-inbound','dash-author-stats'],'rationale':a0_0x1a082b(0x1d2)}},URL_PATTERN={'method':'POST','path':'/api/{project}/{name}/dashboard','exampleFull':a0_0x1a082b(0x1fa),'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':a0_0x1a082b(0x207),'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':a0_0x1a082b(0x1fd),'pathRelativeTo':'payload\x20JSON\x20file\x20location','fileExtensionPolicy':'free;\x20.sql\x20recommended\x20for\x20editor\x20highlight','resolvedAt':a0_0x1a082b(0x20a),'embedStrategy':a0_0x1a082b(0x1f5),'implication':a0_0x1a082b(0x1f8)},PLACEHOLDER_CONVENTION={'format':a0_0x1a082b(0x1bc),'regex':a0_0x1a082b(0x1eb),'regexNotes':a0_0x1a082b(0x1cc),'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_0x1a082b(0x1e8),'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':'top-level\x20\x27cache\x27\x20object','optional':!![],'rationale':a0_0x1a082b(0x1dd),'fields':[{'name':'enabled','type':'boolean','required':!![],'description':a0_0x1a082b(0x1b7)},{'name':'ttl','type':a0_0x1a082b(0x1f3),'required':![],'constraint':'>=\x200\x20(seconds)','default':'inherits\x20CACHE_TTL\x20env','description':a0_0x1a082b(0x1bb)},{'name':a0_0x1a082b(0x202),'type':a0_0x1a082b(0x1be),'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_0x1a082b(0x1d6),a0_0x1a082b(0x208)],'warningOn':[a0_0x1a082b(0x1c9)]}},DOCUMENTATION_URL='https://restforge.dev/docs/server/query-data/dashboard',DASHBOARD_CATALOG={'schemaVersion':a0_0x1a082b(0x1e5),'source':'dashboard-catalog','summary':{'totalAllowedTopLevelFields':PAYLOAD_SHAPE['topLevelAllowed'][a0_0x1a082b(0x1b9)],'totalForbiddenFrontendFields':FORBIDDEN_FRONTEND_FIELDS['length'],'totalParamTypes':ALLOWED_PARAM_TYPES[a0_0x1a082b(0x1b9)],'totalScalarCollapseRules':SCALAR_COLLAPSE_RULES[a0_0x1a082b(0x1b9)],'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['exports']={'DASHBOARD_CATALOG':DASHBOARD_CATALOG};function a0_0x50c0(){const _0x2f8fdf=['y29SB3i','rgfZAgjVyxjKigvUzhbVAw50ig1HEsbVChqTAw4GDg8GuMvKAxmTyMfZzwqGy2fJAguUifbHDhrLCM4GzM9SBg93CYbWCM9JzxnZB3iGy2fJAguGkhnLzsbMzwf0lwnHy2HLlM1Kks4Gq2fJAguGC2nVCguGAxmGDgHLigz1BgWGCMvZCg9UC2uGzw52zwXVCgu7ig9UzsbJywnOzsbLBNrYEsbWzxiGkhbHCMfTCYaRihDPzgDLDhnBxsbZDwjZzxqPignVBwjPBMf0Aw9UlG','BM9UlwvTChr5lcb1BMLXDwuGywnYB3nZihDPzgDLDhmGAw4GDgHLihnHBwuGCgf5Bg9Hza','uMv0DxjUigfZigfYCMf5ig9Mig9IAMvJDhmGkg5VignVBgXHChnLks4','iNnOB3bWAw5Nx2nHDgvNB3jPzxmIoIb7icjPDgvTCYi6ifT7icjUyw1LiJOGiKXHBMrZiIb9lcb7icjUyw1LiJOGiKHVDxnLCYiGFv0GFq','EYaIzgLYzwn0Aw9UiJOGiNvWiIWGiNbJDci6iciYlJyIih0','zgLYzwn0Aw9U','zMLSztPXDwvYEs88Cgf0Ad4VyNjLywTKB3DUlNnXBa','msbYB3CGW5CGmIbJB2X1Bw5Z','ms4W','vuKGBgfIzwWGAxmGysbMCM9UDgvUzcbYzw5KzxjPBMCGy29Uy2vYBI4','ndmXndaYm0nhEMjJDq','u0vmrunuicOGrLjptsbZDg9JA19PBMjVDw5KifDirvjfievyvfjbq1qOwuvbuIbguK9nigLUyM91BMrFzgf0zsKGpsa6EwvHCG','mtfIBMrODKS','iJi4odqI','kd88itOPoIHBys16qs1Ax11Bys16qs1Amc05x10Qkq','msbYB3CGW5CGmsbJB2X1Bw4','DMfSDwuGkg9Yign1CNjLBNqP','D2LKz2v0lNf1zxjPzxmUpgTLEt4GD2L0AcbtuuWGCMv0DxjUAw5NideGCM93imoxig11BhrPCgXLignVBhvTBNm','mZbPANfnAve','CgfYyw1Z','yxzNx2rHAwX5x3nHBgvZ','sgvHzgXPBMuGBwv0CMLJihDPDgGGDhjLBMqGy2HPCcbHBMqGyNjLywTKB3DUigfJCM9ZCYbJyxrLz29YAwvZlIbtDwL0ywjSzsbMB3iGD2LKz2v0CYbSAwTLicDfEhbLy3rLzcbfyxjUAw5NCYCGDgHHDcbZAg93ihrVDgfSihzHBhvLlcbWzxjJzw50ywDLignOyw5NzsWGyw5KihbLCI1JyxrLz29YEsbJB250CMLIDxrPB24U','BNvTyMvY','yxjYyxKGB2yGB2jQzwn0CW','u1fmigzPBguGy29UDgvUDcbPCYbLBwjLzgrLzcbHCYbkyxzHu2nYAxb0ihrLBxbSyxrLigXPDgvYywWGAw5ZAwrLihrOzsbNzw5LCMf0zwqGBw9KDwXLigzPBguUifj1BNrPBwuGCgvYzM9YBxmGEMvYBYbKAxnRieKVtYbWzxiGCMvXDwvZDcdIGjqGywXSifnrtcbPCYbPBIbTzw1VCNKGywz0zxiGBw9KDwXLigXVywqU','Cgn0','zMLSztPXDwvYEs88Cgf0Ad4VDMfSDwuUC3fS','vxbKyxrPBMCGyw4Gu1fmigzPBguGCMvXDwLYzxmGCMvNzw5LCMf0Aw5NihrOzsbKyxnOyM9HCMqGBw9KDwXLicGNy29KzwDLBL9JCMvHDgvFzgfZAgjVyxjKjYKGzM9YignOyw5NzxmGDg8GDgfRzsbLzMzLy3qU','mtC3EfPzyMjx','ue9tvcaVyxbPl21PBMKTAw52zw50B3j5l2rHC2GTAw5IB3vUzc9KyxnOyM9HCMq','tgLZDcbVzIb3AwrNzxqGzgvMAw5PDgLVBNmUie9YzgvYigLZigLUzM9YBwf0Aw9UywWGB25SEsaOCMvZCg9UC2uGA2v5CYbHCMuGyNKGD2LKz2v0igLKlcbUB3qGyxjYyxKGAw5KzxGPlG','EYaIzgLYzwn0Aw9UiJOGiMrVD24IlcaICgn0iJOGiJiUmIiGFq','zMLSztPYzwXHDgL2zs9WyxrOl3rVl3f1zxj5lNnXBa','u2LUz2XLifnrtcbXDwvYEsbMB3iGDgHLihDPzgDLDc4','yxjYyxK','txvSDgKTu1fmihDPzgDLDc4GrwfJAcbRzxKGyMvJB21LCYbHigTLEsbPBIb0AguGCMvZCg9UC2uGB2jQzwn0lG','EYaIzgLYzwn0Aw9UiJOGiNvWiIWGiNbJDci6iciYlJiIih0','Aw52ywXPzgf0zxm','yM9VBgvHBG','rNjVBNrLBMqGy29TChv0zxmGDg9Fz29HBca9ihrHCMDLDcaTihzHBhvLigfUzcbWy3qGpsbYB3vUzcH2ywX1zsaVihrHCMDLDcaQideWmcKGzM9YihrOzsbWCM9NCMvZCYbIyxiUifzPC3vHBcb3Awr0AcbPCYbWCMvZzw50yxrPB25HBcbHBMqGBxvZDcbot1qGBgL2zsbPBIb0AguGyMfJA2vUzcbWyxLSB2fKlIbjzIbWCM9NCMvZCYbPBNzVBhzLCYbJB21WBgv4igj1C2LUzxnZihj1BgvZicHLlMCUigv4y2X1zguGD2vLA2vUzhmSihbYB3jHDgvKihDVCMTKyxLZksWGDxnLigeGC2LUz2XLig11BhrPlwnVBhvTBIbXDwvYEsbZBYaNCgn0jYbPCYbHihn0ywjSzsbIDxnPBMvZCYbMywn0ihjHDgHLCIb0AgfUihzPC3vHBcb3Awr0Ac4','sgvHzgXPBMuGBwv0CMLJihDPDgGGDhjLBMqGy2HPCcbHBMqGC3bHCMTSAw5Lig1PBMKTy2HHCNqGzM9YihnOB3j0ihDPBMrVD3mGkdCGzgf5CYWGmtiGBw9UDgHZlcbLDgmUks4Gu3vPDgfIBguGzM9YihDPzgDLDhmGBgLRzsaNqxzLCMfNzsbeywLSEsbtywXLCYCU','nte2mdHQwuHbsvO','EYbZDwnJzxnZoIbIB29SzwfUlcbKyxrHoIb7idX3AwrNzxrjzd46idXWzxjxAwrNzxrszxnWB25Zzt4Sic4UlIb9ih0','vgfIBguGzgvJBgfYzwqGAw4GAw52ywXPzgf0zxmSigj1DcbUB3qGzgv0zwn0zwqGAw4Gyw55ihDPzgDLDcbtuuWGkhr5Cg8GB3iGzgvHzcbLBNrYEsK','zgfZAgjVyxjKihbHEwXVywq','z2vUzxjHDgLVBIb0Aw1LicHot1qGCNvUDgLTzsK','w3SGiMXHyMvSiJOGiLnOB2vZiIWGiNzHBhvLiJOGiJC2nJaIih0SihSGiMXHyMvSiJOGiKDHBwLUzYiSicj2ywX1zsi6iciYodiWiIb9lcb7icjSywjLBci6icjpDgHLCNmIlcaIDMfSDwuIoIaInduYntCIih1D','zMLSztPXDwvYEs88Cgf0Ad4VDhjLBMqUC3fS','ugfYyw0Gzgf0ysb0ExbLlIbwywXPzgf0zxmGCMvXDwvZDcbIB2r5igfUzcbZAgfWzxmGCNvUDgLTzsbWyxjHBwv0zxiGyMLUzgLUzY4','iML0zw1ZiJOGw3SGiMXHyMvSiJOGiLnOB2vZiIWGiNzHBhvLiJOGiJC2nJaIih0Sic4UlL0','Bwv0CMLJx3bYB2DYzxnZx3rVx2DVywW','offhAxnTDW','mtq0u2Xkvenh','Cg9PBNrZ','C3rYAw5N','D2LKz2v0CW','tM90igeGzgfZAgjVyxjKihbHEwXVywqGkgXPA2vSEsbduLveihDPDgGGDgfIBgvoyw1LlcbVCIbPBNzHBgLKkq','DhjLBMq','iJy5nZaWiG','vg9Nz2XLignHy2HLigzLyxr1CMuGzM9YihrOAxmGzgfZAgjVyxjKlG','yw55icHTDxn0igjLignVBxbHDgLIBguGD2L0AcbKzwnSyxjLzcaNDhLWzsCP','BgvUz3rO','v2LKz2v0igLKzw50AwzPzxi7ihvZzwqGyxmGDgHLihjLC3bVBNnLigTLEsbPBIb0AguGzgfZAgjVyxjKigvUDMvSB3bLlG','vgLTzs10BY1SAxzLigLUihnLy29UzhmUidaGzwzMzwn0AxzLBhKGzgLZywjSzxmGy2fJAguGzM9YihrOAxmGzw50CNKU','oNbHCMfTtMfTzq','nte1ntbyC3fOv1a','yxjYyxK8C3rYAw5NpG','yw55icGXihjVDYddLYaXignVBcWGtIbYB3DZimoxie0Gy29SCYWGzxrJlIK','DgL0Bgu','mxzPsKD1Ea','phDPzgDLDf9Pzd4','twv0CMLJicSGrg9UDxqGqNjLywTKB3DU','mJG1oty1z25Rt1zt','DMfSDwu','rgvMyxvSDcb2ywX1zsbHChbSAwvKihDOzw4GDgHLihjLCxvLC3qGB21PDhmGDgHPCYbWyxjHBs4GvMfSAwrHDg9YigrVzxmGtK9uihn0CMLJDgX5ihr5CguTy2HLy2SGzgvMyxvSDdSGCNvUDgLTzsbPCYbYzxnWB25ZAwjSzsbMB3iGy29TCgf0AwjPBgL0Es4','DgfIBgvoyw1L','twv0CMLJicSGu3bHCMTSAw5L','vgfIBguGzgv0zwn0zwqGAw4Gu1fmlcbIDxqGBM90ihjLz2LZDgvYzwqGyxmGq1jvrcbLBMrWB2LUDcbPBIbTzxrHzgf0ysbWCM9Qzwn0icHSAwTLBhKGysb2Awv3lcbdveuGywXPyxmSig9YignYB3nZlxbYB2PLy3qGDgfIBguG4OcuignHC2nHzguGD2LSBcbUB3qGzMLYzsK','v2HLBIb0CNvLlcb0AguGCMvXDwvZDcbIB2r5ie1vu1qGAw5JBhvKzsb0AgLZihbHCMfTicHVDgHLCNDPC2uGndaWks4','mte3otqYnJbvsLbKshC','tMvNyxrPDMuGBg9VA2jLAgLUzcbWCMv2zw50CYbTyxrJAgLUzYaNoJONicHqB3n0z3jLCYbJyxn0ihn5BNrHEcKGyxmGysbWBgfJzwHVBgrLCI4','zgvMyxvSDa','tIbYB3DZimoxidiGy29SDw1UCW','ugvYlwTLEsbIyxnLzcbVBIbZy2fSyxjdB2XSyxbZzvj1BgvZigjLBg93lG','BwfW','Bwv0CMLJx3nWyxjRBgLUzq','vgHLihbYzwzPEcbIzwnVBwvZihbHCNqGB2yGDgHLifvstcbZzwDTzw50lIbuAguGCMvZzxj2zwqGC2nOzw1LigTLzxbZigrHC2HIB2fYzcbLBMrWB2LUDhmGDMLZDwfSBhKGzgLZDgLUy3qGzNjVBsbduLveigvUzhbVAw50CYbPBIb0AguGvvjmihnWywnLigfUzcbHBgXVD3mGzNv0DxjLihjVDxrPBMCGzgLMzMvYzw50Awf0Aw9UlG','iJi0mJaI','C2nHBgfYihbYAw1PDgL2zq','zgfZAc1ZywXLCW','vgfIBguGyxbWzwfYCYbPBIbtuuWGqu5eigLUig1LDgfKyxrHihbYB2PLy3qSigj1DcbTAxnZAw5NigzYB20GAw52ywXPzgf0zxmGkgnHy2HLihn0ywXLihjPC2SP','B2jQzwn0','ndm1mZyYwNbxz0f0','zMLSztPXDwvYEs88Cgf0Ad4Vy3vYCMvUDc5ZCwW','oti4nJCXouncshz0ua','iJe4mZyI'];a0_0x50c0=function(){return _0x2f8fdf;};return a0_0x50c0();}
@@ -1 +1 @@
1
- function a0_0x4a35(){const _0x2e5c82=['mtC2mZy3mLjoAfrgCW','m09Lyvnwwa','tur6rvi','DgvZDa','nZa3nJaWCwjduLnl','odjbrLHbze4','iYbmAwnLBNnLcKXjq0vou0u9wfHywc1ywfHylvHywfGTwfHywaOkiYbtzxj2zxiku0vsvKvsx0ferfjfu1m9mti3lJaUmc4XcLnfuLzfuL9qt1juptmWmdakcImGtgL2zsbtEw5JicHxzwjtB2nRzxqPienVBMzPz3vYyxrPB24kiYbot1rfoIbmsvzfx1nztKnFru5bqKXfrd10CNvLihjLCxvPCMvZigfUiefqssblzxKGkeTfwt0UlI4PihrVigf1DgHLBNrPy2f0zsbxzwjtB2nRzxqGy2XPzw50CWPmsvzfx1nztKnFru5bqKXfrd1MywXZzqPmsvzfx1nztKnFue9svd0ZmdmZcGOJifjLzgLZienVBMzPz3vYyxrPB24kuKvesvnFse9tvd1SB2nHBgHVC3qkuKvesvnFue9svd02mZGWcLjfreLtx1bbu1nxt1jepqPsrurju19eqJ0WcGOJiev4Cg9YDcbdB25MAwD1CMf0Aw9UcKvyue9svf9gsuXfx0vyueLswt0ZnJaWmdaWcKvyue9svf9dsfvos19tsvPfpteWmdakcImGs2fMA2eGq29UzMLNDxjHDgLVBGPlquzlqv9ftKfctevepwzHBhnLcImGqNjVA2vYigXPC3qGkgnVBw1HlxnLCgfYyxrLzcbMB3iGBxvSDgLWBguGyNjVA2vYCZOGyNjVA2vYmtO5mdKYlgjYB2TLCJi6ota5mIXICM9RzxiZoJKWotiPcKTbrKTbx0jst0TfuLm9Bg9JywXOB3n0oJKWotikiYbdBgLLBNqGsuqGkg9WDgLVBMfSlcbKzwzHDwX0oIbYzxn0zM9Yz2uTE3bYB2PLy3r9lxbYB2r1y2vYic8GlwnVBNn1BwvYkqOJieTbrKTbx0nmsuvovf9jrd0ks0fgs0fFq09otKvdveLptL9usu1ft1vuptmWmdaks0fgs0fFuKvrvuvtvf9usu1ft1vupti1mdaWcKTbrKTbx1rpueLdx1bbvfrfuK49E21VzhvSzx0UE2vUzhbVAw50Fs5LDMvUDhmks0fgs0fFvevoqu5ux0LepwrLzMf1Bhqks0fgs0fFu0vtu0LptL9usu1ft1vuptmWmdaWcKTbrKTbx0HfqvjuqKvbvf9jtLrfuLzbtd0ZmdaWcKTbrKTbx01bwf9cwvrfu19qrvjFuefsveLusu9opteWndG1nZyks0fgs0fFqvvut19dt01nsvq9zMfSC2uks0fgs0fFqvvut19dt01nsvrFsu5urvjwquW9ntaWmaPlquzlqv9srvrswv9bvfrftvbuuZ0ZcKTbrKTbx1jfvfjzx0rftefzpteWmdaks0fgs0fFuKvuuLLFtufyx0rftefzptmWmdaWcKTbrKTbx1nttd1MywXZzqPlquzlqv9mt0DFtevwruW9Aw5MBWOJifnbu0WGqxv0AgvUDgLJyxrPB24Gkg9WDgLVBMfSlcb1BMnVBw1LBNqGAwyGDgHLigjYB2TLCIbYzxf1AxjLCYbHDxrOzw50AwnHDgLVBIKkiYbtDxbWB3j0zwqGBwvJAgfUAxnTCZOGCgXHAw4SihnJCMfTlxnOys0YntySihnJCMfTlxnOys01mtikiYblquzlqv9tqvnmx01fq0HbtKLttt1WBgfPBGOJieTbrKTbx1nbu0XFvvnfuK5btuu9cImGs0fgs0fFu0fttf9qqvntv09srd0kcImGrgf0ywjHC2uGq29UzMLNDxjHDgLVBGOJifn1ChbVCNrLzdOGCg9ZDgDYzxnXBcWGBxLZCwWSig9YywnSzsWGC3fSAxrLcKrcx1rzueu9Cg9ZDgDYzxnXBaPeqL9it1nupteYnY4WlJaUmqPeqL9qt1juptu0mZikrejFvvnfuJ1WB3n0z3jLCWPeqL9qqvntv09srd15B3vYx3bHC3n3B3jKx2HLCMukrejFtKfnrt15B3vYx2rHDgfIyxnLx25HBwukiYbgB3iGu1fmAxrLoIbZzxqGrejFvfLqrt1ZCwXPDguGyw5Kiercx05btuu9lI9KyxrHl215yxbWlMrIcImGrejFse9tvcWGrejFue9svcWGrejFvvnfuIWGrejFueftu1DpuKqGyxjLigLNBM9YzwqGzM9YifnrtgL0zqOkiYbmB2DNAw5NienVBMzPz3vYyxrPB24kte9hx0XfvKvmpwrLyNvNcKXpr19ut19gsuXfpxrYDwukcImGu1fmieXVz2DPBMCku1fmx0Xpr19ftKfctevepwzHBhnLcLnrtf9mt0DFtevwruW9zgvIDwCku1fmx0Xpr19qqvjbtvm9zMfSC2uku1fmx0Xpr19tte9xx1riuKvtse9mrd0XmdaWcGOJienHy2HLienVBMzPz3vYyxrPB24kq0fdsevFru5bqKXfrd1MywXZzqPdqunirv9uveW9mZaWcGOJiePVyIbty2HLzhvSzxiksK9cx0voqujmruq9zMfSC2uksK9cx0nptKnvuLjftKnzptuksK9cx1jfvevoveLptL9it1vsuZ03mGPkt0jFrKfjtevex1jfvevoveLptL9it1vsuZ0XnJGksK9cx1nivvret1Dox1rjtuvpvvq9mtaWmdaksK9cx1nuquXmrurFsu5urvjwquW9mZaWmdaksK9cx01bwf9tvefmtevex0npvu5uptikcImGrgLZDhjPyNv0zwqGtg9JAYbdB25MAwD1CMf0Aw9UcKXpq0TFreLtvfjjqLvururFru5bqKXfrd1MywXZzqPmt0nlx0rju1rssujvvevex1rutd0XmaPmt0nlx1jfu09vuKnfx01bwf9uveW9nJaWcKXpq0TFreLtvfjjqLvururFuKvuuLK9mWPmt0nlx0rju1rssujvvevex1jfvfjzx0rftefzpteWmaPmt0nlx0rju1rssujvvevex1nuuKfuruDzpxjLAMvJDaOkiYbjrcbhzw5LCMf0B3iGq29UzMLNDxjHDgLVBGPjreDftL9ftKfctevepwzHBhnLcKLer0vox0Leru1FvfrmptyWmaPjreDftL9dt1vovevsx1rutf9nt05useXzpti3nJq4mdaksurhru5Fq09vtLrfuL9uveXFrefjtfK9mtCYodaWcKLer0vox0rfrKfvtfrFtufyx1jfvfjzpteWcKLer0vox0rfrKfvtfrFueLox0rjr0LuuZ02cKLer0vox0rfrKfvtfrFu0vssufmx1bbvfrfuK49wfHywc1ywfHylvHywfGTwfHywaPjreDftL9eruzbvuXux0nprevFuefuvevstJ05otK5ltK5otKksurhru5FquXmt1DFuKvtrvq9zMfSC2uk','nZG0n01fz2j6Cq','teLdru5trq','mJa0mtaWmhv1Dg10sa','yM9VBgvHBG','zMfSC2u','Dhj1zq','nJq0mteWDuDmEvvr','mti5mhLgCvPzzW','sefXCNC','Aw5KzxHpzG','BgvUz3rO','tg15zM0','AM9PBG','werzDu0','mJi4nZm2ogTMCLfLwq','rejFue9sva','C2XPy2u','mtaYotHnz29xBxi','Aw5JBhvKzxm','rejFvvnfuG','DhjPBq','Aw50zwDLCG','ue1KvgO','AgfZ'];a0_0x4a35=function(){return _0x2e5c82;};return a0_0x4a35();}const a0_0x25dc7d=a0_0x3341;(function(_0x243a0e,_0x5c36bc){const _0x4994fc=a0_0x3341,_0x5dbcbe=_0x243a0e();while(!![]){try{const _0x889ffc=-parseInt(_0x4994fc(0x1fb))/0x1*(-parseInt(_0x4994fc(0x207))/0x2)+parseInt(_0x4994fc(0x203))/0x3*(-parseInt(_0x4994fc(0x202))/0x4)+-parseInt(_0x4994fc(0x20f))/0x5+parseInt(_0x4994fc(0x210))/0x6*(parseInt(_0x4994fc(0x209))/0x7)+parseInt(_0x4994fc(0x206))/0x8+parseInt(_0x4994fc(0x1f8))/0x9+-parseInt(_0x4994fc(0x20b))/0xa;if(_0x889ffc===_0x5c36bc)break;else _0x5dbcbe['push'](_0x5dbcbe['shift']());}catch(_0xf2a3b8){_0x5dbcbe['push'](_0x5dbcbe['shift']());}}}(a0_0x4a35,0x38a3b));function a0_0x3341(_0x230c40,_0x4f48d8){_0x230c40=_0x230c40-0x1f3;const _0x4a3566=a0_0x4a35();let _0x33418f=_0x4a3566[_0x230c40];if(a0_0x3341['eIThZy']===undefined){var _0x3e7273=function(_0x222b42){const _0x1d69f1='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x52ac7d='',_0x43ac90='';for(let _0x8f1b38=0x0,_0x18f714,_0x569841,_0x3efb4b=0x0;_0x569841=_0x222b42['charAt'](_0x3efb4b++);~_0x569841&&(_0x18f714=_0x8f1b38%0x4?_0x18f714*0x40+_0x569841:_0x569841,_0x8f1b38++%0x4)?_0x52ac7d+=String['fromCharCode'](0xff&_0x18f714>>(-0x2*_0x8f1b38&0x6)):0x0){_0x569841=_0x1d69f1['indexOf'](_0x569841);}for(let _0x6f563c=0x0,_0x10df9b=_0x52ac7d['length'];_0x6f563c<_0x10df9b;_0x6f563c++){_0x43ac90+='%'+('00'+_0x52ac7d['charCodeAt'](_0x6f563c)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x43ac90);};a0_0x3341['fkQnWS']=_0x3e7273,a0_0x3341['NjWDXW']={},a0_0x3341['eIThZy']=!![];}const _0x57b8eb=_0x4a3566[0x0],_0x1da770=_0x230c40+_0x57b8eb,_0x5674b9=a0_0x3341['NjWDXW'][_0x1da770];return!_0x5674b9?(_0x33418f=a0_0x3341['fkQnWS'](_0x33418f),a0_0x3341['NjWDXW'][_0x1da770]=_0x33418f):_0x33418f=_0x5674b9,_0x33418f;}const DB_CONNECTION_ENV_TEMPLATE=a0_0x25dc7d(0x208),REQUIRED_KEYS=new Set([a0_0x25dc7d(0x20a),'SERVER_ADDRESS','SERVER_PORT','DB_TYPE','DB_HOST',a0_0x25dc7d(0x1f9),a0_0x25dc7d(0x1fd),'DB_PASSWORD','DB_NAME']);function parseTemplateAsSchema(_0x500001){const _0x1ecfe3=a0_0x25dc7d,_0x33452c={'XDYuM':function(_0xb21099,_0x1d1191){return _0xb21099<_0x1d1191;},'PMdTj':function(_0x46a585,_0x3d6939){return _0x46a585===_0x3d6939;},'vBZzD':function(_0x2ce727,_0x540c7c){return _0x2ce727>_0x540c7c;},'Lmyfm':'string','BeTbr':_0x1ecfe3(0x20e),'HAqrw':_0x1ecfe3(0x20d),'MDzER':_0x1ecfe3(0x20c)},_0x565642=_0x500001||DB_CONNECTION_ENV_TEMPLATE,_0xcaa096=_0x565642['split']('\x0a'),_0x2c1bc4=[];let _0x3e3a9e=null,_0x18fda6=[];for(const _0x1f965c of _0xcaa096){const _0xc7851=_0x1f965c[_0x1ecfe3(0x1fe)]();if(_0xc7851===''){_0x18fda6=[];continue;}if(_0xc7851['startsWith']('#')){const _0x5ee1ab=_0xc7851[_0x1ecfe3(0x1fa)](0x1)['trim'](),_0x4d2ca9=_0x5ee1ab[_0x1ecfe3(0x1f4)]>0x0&&_0x33452c[_0x1ecfe3(0x1f7)](_0x5ee1ab[_0x1ecfe3(0x1f4)],0x3c)&&!_0x5ee1ab[_0x1ecfe3(0x1fc)](':')&&!/^[A-Z_]+=/[_0x1ecfe3(0x205)](_0x5ee1ab)&&/^[A-Z]/['test'](_0x5ee1ab);_0x4d2ca9&&_0x33452c['PMdTj'](_0x18fda6['length'],0x0)?_0x3e3a9e=_0x5ee1ab:_0x18fda6['push'](_0x5ee1ab);continue;}const _0x495a89=_0x1f965c[_0x1ecfe3(0x1f3)]('=');if(_0x33452c['vBZzD'](_0x495a89,0x0)){const _0x25b302=_0x1f965c['slice'](0x0,_0x495a89)[_0x1ecfe3(0x1fe)](),_0x4905e1=_0x1f965c[_0x1ecfe3(0x1fa)](_0x495a89+0x1);let _0x54c8f7=_0x33452c[_0x1ecfe3(0x1f5)];if(_0x4905e1===_0x33452c['BeTbr']||_0x33452c[_0x1ecfe3(0x200)](_0x4905e1,_0x33452c[_0x1ecfe3(0x211)]))_0x54c8f7=_0x33452c[_0x1ecfe3(0x204)];else/^-?\d+$/['test'](_0x4905e1)&&(_0x54c8f7=_0x1ecfe3(0x1ff));_0x2c1bc4['push']({'name':_0x25b302,'section':_0x3e3a9e,'type':_0x54c8f7,'default':_0x4905e1,'description':_0x18fda6[_0x1ecfe3(0x1f6)]('\x20')||null,'required':REQUIRED_KEYS[_0x1ecfe3(0x201)](_0x25b302)}),_0x18fda6=[];}}return _0x2c1bc4;}module['exports']={'DB_CONNECTION_ENV_TEMPLATE':DB_CONNECTION_ENV_TEMPLATE,'REQUIRED_KEYS':REQUIRED_KEYS,'parseTemplateAsSchema':parseTemplateAsSchema};
1
+ function a0_0x3665(){const _0xe10a06=['iYbmAwnLBNnLcKXjq0vou0u9wfHywc1ywfHylvHywfGTwfHywaOkiYbtzxj2zxiku0vsvKvsx0ferfjfu1m9mti3lJaUmc4XcLnfuLzfuL9qt1juptmWmdakcImGtgL2zsbtEw5JicHxzwjtB2nRzxqPienVBMzPz3vYyxrPB24kiYbot1rfoIbmsvzfx1nztKnFru5bqKXfrd10CNvLihjLCxvPCMvZigfUiefqssblzxKGkeTfwt0UlI4PihrVigf1DgHLBNrPy2f0zsbxzwjtB2nRzxqGy2XPzw50CWPmsvzfx1nztKnFru5bqKXfrd1MywXZzqPmsvzfx1nztKnFue9svd0ZmdmZcGOJifjLzgLZienVBMzPz3vYyxrPB24kuKvesvnFse9tvd1SB2nHBgHVC3qkuKvesvnFue9svd02mZGWcLjfreLtx1bbu1nxt1jepqPsrurju19eqJ0WcGOJiev4Cg9YDcbdB25MAwD1CMf0Aw9UcKvyue9svf9gsuXfx0vyueLswt0ZnJaWmdaWcKvyue9svf9dsfvos19tsvPfpteWmdakcImGs2fMA2eGq29UzMLNDxjHDgLVBGPlquzlqv9ftKfctevepwzHBhnLcImGqNjVA2vYigXPC3qGkgnVBw1HlxnLCgfYyxrLzcbMB3iGBxvSDgLWBguGyNjVA2vYCZOGyNjVA2vYmtO5mdKYlgjYB2TLCJi6ota5mIXICM9RzxiZoJKWotiPcKTbrKTbx0jst0TfuLm9Bg9JywXOB3n0oJKWotikiYbdBgLLBNqGsuqGkg9WDgLVBMfSlcbKzwzHDwX0oIbYzxn0zM9Yz2uTE3bYB2PLy3r9lxbYB2r1y2vYic8GlwnVBNn1BwvYkqOJieTbrKTbx0nmsuvovf9jrd0ks0fgs0fFq09otKvdveLptL9usu1ft1vuptmWmdaks0fgs0fFuKvrvuvtvf9usu1ft1vupti1mdaWcKTbrKTbx1rpueLdx1bbvfrfuK49E21VzhvSzx0UE2vUzhbVAw50Fs5LDMvUDhmks0fgs0fFvevoqu5ux0LepwrLzMf1Bhqks0fgs0fFu0vtu0LptL9usu1ft1vuptmWmdaWcKTbrKTbx0HfqvjuqKvbvf9jtLrfuLzbtd0ZmdaWcKTbrKTbx01bwf9cwvrfu19qrvjFuefsveLusu9opteWndG1nZyks0fgs0fFqvvut19dt01nsvq9zMfSC2uks0fgs0fFqvvut19dt01nsvrFsu5urvjwquW9ntaWmaPlquzlqv9srvrswv9bvfrftvbuuZ0ZcKTbrKTbx1jfvfjzx0rftefzpteWmdaks0fgs0fFuKvuuLLFtufyx0rftefzptmWmdaWcKTbrKTbx1nttd1MywXZzqPlquzlqv9mt0DFtevwruW9Aw5MBWOJifnbu0WGqxv0AgvUDgLJyxrPB24Gkg9WDgLVBMfSlcb1BMnVBw1LBNqGAwyGDgHLigjYB2TLCIbYzxf1AxjLCYbHDxrOzw50AwnHDgLVBIKkiYbtDxbWB3j0zwqGBwvJAgfUAxnTCZOGCgXHAw4SihnJCMfTlxnOys0YntySihnJCMfTlxnOys01mtikiYblquzlqv9tqvnmx01fq0HbtKLttt1WBgfPBGOJieTbrKTbx1nbu0XFvvnfuK5btuu9cImGs0fgs0fFu0fttf9qqvntv09srd0kcImGrgf0ywjHC2uGq29UzMLNDxjHDgLVBGOJifn1ChbVCNrLzdOGCg9ZDgDYzxnXBcWGBxLZCwWSig9YywnSzsWGC3fSAxrLcKrcx1rzueu9Cg9ZDgDYzxnXBaPeqL9it1nupteYnY4WlJaUmqPeqL9qt1juptu0mZikrejFvvnfuJ1WB3n0z3jLCWPeqL9qqvntv09srd15B3vYx3bHC3n3B3jKx2HLCMukrejFtKfnrt15B3vYx2rHDgfIyxnLx25HBwukiYbgB3iGu1fmAxrLoIbZzxqGrejFvfLqrt1ZCwXPDguGyw5Kiercx05btuu9lI9KyxrHl215yxbWlMrIcImGrejFse9tvcWGrejFue9svcWGrejFvvnfuIWGrejFueftu1DpuKqGyxjLigLNBM9YzwqGzM9YifnrtgL0zqOkiYbmB2DNAw5NienVBMzPz3vYyxrPB24kte9hx0XfvKvmpwrLyNvNcKXpr19ut19gsuXfpxrYDwukcImGu1fmieXVz2DPBMCku1fmx0Xpr19ftKfctevepwzHBhnLcLnrtf9mt0DFtevwruW9zgvIDwCku1fmx0Xpr19qqvjbtvm9zMfSC2uku1fmx0Xpr19tte9xx1riuKvtse9mrd0XmdaWcGOJienHy2HLienVBMzPz3vYyxrPB24kq0fdsevFru5bqKXfrd1MywXZzqPdqunirv9uveW9mZaWcGOJiePVyIbty2HLzhvSzxiksK9cx0voqujmruq9zMfSC2uksK9cx0nptKnvuLjftKnzptuksK9cx1jfvevoveLptL9it1vsuZ03mGPkt0jFrKfjtevex1jfvevoveLptL9it1vsuZ0XnJGksK9cx1nivvret1Dox1rjtuvpvvq9mtaWmdaksK9cx1nuquXmrurFsu5urvjwquW9mZaWmdaksK9cx01bwf9tvefmtevex0npvu5uptikcImGrgLZDhjPyNv0zwqGtg9JAYbdB25MAwD1CMf0Aw9UcKXpq0TFreLtvfjjqLvururFru5bqKXfrd1MywXZzqPmt0nlx0rju1rssujvvevex1rutd0XmaPmt0nlx1jfu09vuKnfx01bwf9uveW9nJaWcKXpq0TFreLtvfjjqLvururFuKvuuLK9mWPmt0nlx0rju1rssujvvevex1jfvfjzx0rftefzpteWmaPmt0nlx0rju1rssujvvevex1nuuKfuruDzpxjLAMvJDaOkiYbjrcbhzw5LCMf0B3iGq29UzMLNDxjHDgLVBGPjreDftL9ftKfctevepwzHBhnLcKLer0vox0Leru1FvfrmptyWmaPjreDftL9dt1vovevsx1rutf9nt05useXzpti3nJq4mdaksurhru5Fq09vtLrfuL9uveXFrefjtfK9mtCYodaWcKLer0vox0rfrKfvtfrFtufyx1jfvfjzpteWcKLer0vox0rfrKfvtfrFueLox0rjr0LuuZ02cKLer0vox0rfrKfvtfrFu0vssufmx1bbvfrfuK49wfHywc1ywfHylvHywfGTwfHywaPjreDftL9eruzbvuXux0nprevFuefuvevstJ05otK5ltK5otKksurhru5FquXmt1DFuKvtrvq9zMfSC2uk','EvbNyKO','mtu3ntiXnuTTBhHcrG','vvD1rgK','zxHWB3j0CW','ntyYmZKWy2HMD2Pk','Aw5KzxHpzG','Aw50zwDLCG','mtbvvNP2sLC','DgvZDa','C3rHCNrZv2L0Aa','AM9PBG','CLPlD1u','ugHQyu8','BgvUz3rO','rejFue9sva','nhnyELjQCq','mta4nZC1mNLkz3zLDa','nteXnJuYme9jCKfera','rejFse9tva','mtyWnta4n2HStfvVra','C3rYAw5N','ChvZAa','ntu0nJaYy0jkC2vv','Aw5JBhvKzxm','C2XPy2u','mtyZndCXuwrHqvrt','DhjPBq','zhHkAxi','rejFueftu1DpuKq','mtH5vfLRv0G','Dhj1zq'];a0_0x3665=function(){return _0xe10a06;};return a0_0x3665();}const a0_0x4221e3=a0_0x5bdf;function a0_0x5bdf(_0x2778f2,_0x4f6c57){_0x2778f2=_0x2778f2-0x166;const _0x366577=a0_0x3665();let _0x5bdf2a=_0x366577[_0x2778f2];if(a0_0x5bdf['NwuuYu']===undefined){var _0x5b03ec=function(_0x450b67){const _0x2cf4fe='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x5e7e62='',_0x5c04a2='';for(let _0x119647=0x0,_0x765bfe,_0x403b9e,_0x1a914e=0x0;_0x403b9e=_0x450b67['charAt'](_0x1a914e++);~_0x403b9e&&(_0x765bfe=_0x119647%0x4?_0x765bfe*0x40+_0x403b9e:_0x403b9e,_0x119647++%0x4)?_0x5e7e62+=String['fromCharCode'](0xff&_0x765bfe>>(-0x2*_0x119647&0x6)):0x0){_0x403b9e=_0x2cf4fe['indexOf'](_0x403b9e);}for(let _0x44939b=0x0,_0x22ffea=_0x5e7e62['length'];_0x44939b<_0x22ffea;_0x44939b++){_0x5c04a2+='%'+('00'+_0x5e7e62['charCodeAt'](_0x44939b)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x5c04a2);};a0_0x5bdf['hwKwxC']=_0x5b03ec,a0_0x5bdf['wyXbDI']={},a0_0x5bdf['NwuuYu']=!![];}const _0x47b363=_0x366577[0x0],_0x45b609=_0x2778f2+_0x47b363,_0x598f85=a0_0x5bdf['wyXbDI'][_0x45b609];return!_0x598f85?(_0x5bdf2a=a0_0x5bdf['hwKwxC'](_0x5bdf2a),a0_0x5bdf['wyXbDI'][_0x45b609]=_0x5bdf2a):_0x5bdf2a=_0x598f85,_0x5bdf2a;}(function(_0x37d7e4,_0x3cf450){const _0x889cf9=a0_0x5bdf,_0x510723=_0x37d7e4();while(!![]){try{const _0x55be9a=parseInt(_0x889cf9(0x182))/0x1+parseInt(_0x889cf9(0x174))/0x2+parseInt(_0x889cf9(0x16e))/0x3*(parseInt(_0x889cf9(0x16d))/0x4)+-parseInt(_0x889cf9(0x17f))/0x5+-parseInt(_0x889cf9(0x17b))/0x6*(parseInt(_0x889cf9(0x177))/0x7)+-parseInt(_0x889cf9(0x16f))/0x8+-parseInt(_0x889cf9(0x171))/0x9*(-parseInt(_0x889cf9(0x185))/0xa);if(_0x55be9a===_0x3cf450)break;else _0x510723['push'](_0x510723['shift']());}catch(_0x5471e5){_0x510723['push'](_0x510723['shift']());}}}(a0_0x3665,0x56e6f));const DB_CONNECTION_ENV_TEMPLATE=a0_0x4221e3(0x17d),REQUIRED_KEYS=new Set(['LICENSE','SERVER_ADDRESS','SERVER_PORT','DB_TYPE',a0_0x4221e3(0x170),a0_0x4221e3(0x16c),'DB_USER',a0_0x4221e3(0x17a),'DB_NAME']);function parseTemplateAsSchema(_0x55f51a){const _0x356df7=a0_0x4221e3,_0x364566={'PhjaO':function(_0x5f4289,_0x488fed){return _0x5f4289||_0x488fed;},'rZKwU':function(_0x51fe61,_0x12091b){return _0x51fe61>_0x12091b;},'dxJir':function(_0x2c2714,_0x51ad0e){return _0x2c2714<_0x51ad0e;},'zlTLH':function(_0x1fb841,_0x37b2b7){return _0x1fb841===_0x37b2b7;},'DiyDJ':_0x356df7(0x17c),'UWuDi':'false','yPgbJ':_0x356df7(0x184)},_0x59fa76=_0x364566[_0x356df7(0x16a)](_0x55f51a,DB_CONNECTION_ENV_TEMPLATE),_0x3ae049=_0x59fa76['split']('\x0a'),_0xabeb41=[];let _0x46df13=null,_0x90dfe3=[];for(const _0x2d9bdc of _0x3ae049){const _0x2a6dda=_0x2d9bdc[_0x356df7(0x178)]();if(_0x2a6dda===''){_0x90dfe3=[];continue;}if(_0x2a6dda[_0x356df7(0x167)]('#')){const _0x34593e=_0x2a6dda['slice'](0x1)['trim'](),_0x100323=_0x364566[_0x356df7(0x169)](_0x34593e['length'],0x0)&&_0x364566[_0x356df7(0x179)](_0x34593e[_0x356df7(0x16b)],0x3c)&&!_0x34593e[_0x356df7(0x175)](':')&&!/^[A-Z_]+=/[_0x356df7(0x166)](_0x34593e)&&/^[A-Z]/['test'](_0x34593e);_0x100323&&_0x364566['zlTLH'](_0x90dfe3[_0x356df7(0x16b)],0x0)?_0x46df13=_0x34593e:_0x90dfe3[_0x356df7(0x173)](_0x34593e);continue;}const _0x30ac8c=_0x2d9bdc[_0x356df7(0x183)]('=');if(_0x30ac8c>0x0){const _0x238e0c=_0x2d9bdc[_0x356df7(0x176)](0x0,_0x30ac8c)['trim'](),_0x230fe8=_0x2d9bdc['slice'](_0x30ac8c+0x1);let _0x2c450c=_0x356df7(0x172);if(_0x230fe8===_0x364566['DiyDJ']||_0x230fe8===_0x364566[_0x356df7(0x180)])_0x2c450c='boolean';else/^-?\d+$/[_0x356df7(0x166)](_0x230fe8)&&(_0x2c450c=_0x364566[_0x356df7(0x17e)]);_0xabeb41[_0x356df7(0x173)]({'name':_0x238e0c,'section':_0x46df13,'type':_0x2c450c,'default':_0x230fe8,'description':_0x90dfe3[_0x356df7(0x168)]('\x20')||null,'required':REQUIRED_KEYS['has'](_0x238e0c)}),_0x90dfe3=[];}}return _0xabeb41;}module[a0_0x4221e3(0x181)]={'DB_CONNECTION_ENV_TEMPLATE':DB_CONNECTION_ENV_TEMPLATE,'REQUIRED_KEYS':REQUIRED_KEYS,'parseTemplateAsSchema':parseTemplateAsSchema};
@@ -1 +1 @@
1
- 'use strict';const a0_0x1f19cb=a0_0x3586;(function(_0x549582,_0x48561a){const _0x2c77e8=a0_0x3586,_0x41f25b=_0x549582();while(!![]){try{const _0x515600=parseInt(_0x2c77e8(0xd4))/0x1+parseInt(_0x2c77e8(0xde))/0x2+parseInt(_0x2c77e8(0xaf))/0x3+-parseInt(_0x2c77e8(0xbf))/0x4+-parseInt(_0x2c77e8(0xf7))/0x5+parseInt(_0x2c77e8(0xac))/0x6+-parseInt(_0x2c77e8(0xfc))/0x7;if(_0x515600===_0x48561a)break;else _0x41f25b['push'](_0x41f25b['shift']());}catch(_0x5785da){_0x41f25b['push'](_0x41f25b['shift']());}}}(a0_0xad2b,0x4f0ff));const SCHEMA_VERSION=a0_0x1f19cb(0xd8),SOURCE=a0_0x1f19cb(0xae),DOCUMENTATION_URL=a0_0x1f19cb(0xd2),DEFINE_MODEL_OPTIONS=[{'name':a0_0x1f19cb(0xa7),'required':![],'type':'string|null','description':'Database\x20schema\x20namespace\x20(multi-schema\x20database).','notes':'null\x20or\x20empty\x20string\x20means\x20default\x20schema.'},{'name':a0_0x1f19cb(0xc3),'required':!![],'type':a0_0x1f19cb(0xe8),'description':a0_0x1f19cb(0xe0),'example':a0_0x1f19cb(0xb1)},{'name':a0_0x1f19cb(0xb9),'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_0x1f19cb(0x104),'required':![],'type':a0_0x1f19cb(0xe8),'description':'Inter-table\x20relations\x20(belongsTo,\x20hasOne,\x20hasMany).','example':a0_0x1f19cb(0xa8)},{'name':'indexes','required':![],'type':a0_0x1f19cb(0x101),'description':a0_0x1f19cb(0xa6),'example':'[[\x22status\x22,\x20\x22created_at\x22]]'},{'name':a0_0x1f19cb(0xd0),'required':![],'type':a0_0x1f19cb(0x101),'description':a0_0x1f19cb(0xca),'example':a0_0x1f19cb(0xe3)},{'name':a0_0x1f19cb(0xe9),'required':![],'type':a0_0x1f19cb(0x101),'description':a0_0x1f19cb(0xeb),'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_0x1f19cb(0xda),'requiresModifier':!![],'modifierFormat':a0_0x1f19cb(0xe5),'example':a0_0x1f19cb(0xc9),'notes':'Explicit\x20length\x20required\x20for\x20cross-dialect\x20portability.'},{'name':'text','description':'Long-form\x20text\x20without\x20length\x20limit\x20(TEXT/CLOB).','requiresModifier':![],'example':'text'},{'name':a0_0x1f19cb(0xb0),'description':'32-bit\x20signed\x20integer\x20(INT/INTEGER).','requiresModifier':![],'example':'integer'},{'name':a0_0x1f19cb(0xb4),'description':a0_0x1f19cb(0xa4),'requiresModifier':![],'example':a0_0x1f19cb(0xb4)},{'name':a0_0x1f19cb(0xf0),'description':'Fixed-point\x20decimal.\x20Precision\x20and\x20scale\x20required.','requiresModifier':!![],'modifierFormat':'decimal:<precision>,<scale>','example':a0_0x1f19cb(0xf2),'notes':a0_0x1f19cb(0xb8)},{'name':a0_0x1f19cb(0xdb),'description':'Boolean\x20value\x20(native\x20BOOLEAN\x20on\x20PostgreSQL,\x20VARCHAR\x20on\x20others).','requiresModifier':![],'example':'boolean'},{'name':'date','description':'Date\x20only\x20(no\x20time\x20component).','requiresModifier':![],'example':a0_0x1f19cb(0xd9)},{'name':'timestamp','description':'Date\x20and\x20time\x20(TIMESTAMP).','requiresModifier':![],'example':'timestamp','notes':'Default\x20timezone\x20behavior\x20is\x20dialect-specific.'},{'name':'uuid','description':a0_0x1f19cb(0x102),'requiresModifier':![],'example':'uuid'},{'name':'json','description':'JSON\x20column\x20(JSONB\x20on\x20PostgreSQL,\x20JSON\x20on\x20MySQL,\x20CLOB\x20on\x20Oracle).','requiresModifier':![],'example':a0_0x1f19cb(0xe4)}],CONSTRAINTS=[{'name':'pk','kind':a0_0x1f19cb(0xcb),'description':a0_0x1f19cb(0xe2),'example':a0_0x1f19cb(0xaa)},{'name':a0_0x1f19cb(0x107),'kind':'standalone','description':a0_0x1f19cb(0x10a),'example':a0_0x1f19cb(0xcc)},{'name':a0_0x1f19cb(0x109),'kind':'standalone','description':'Single-column\x20unique\x20constraint.','example':'string:32\x20unique'},{'name':'index','kind':a0_0x1f19cb(0xcb),'description':'Single-column\x20non-unique\x20index.','example':'string:64\x20index'},{'name':a0_0x1f19cb(0xcf),'kind':a0_0x1f19cb(0xcb),'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_0x1f19cb(0xd1),'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_0x1f19cb(0xa5),'description':a0_0x1f19cb(0xf9),'valueFormat':'default:<literal>\x20|\x20default:\x27<string>\x27\x20|\x20default:<constant>\x20|\x20default:<function>()','example':a0_0x1f19cb(0xb3)},{'name':'fk','kind':a0_0x1f19cb(0xa5),'description':a0_0x1f19cb(0xbd),'valueFormat':'fk:<table>.<column>','example':'string:36\x20fk:category.id','notes':'Cannot\x20coexist\x20with\x20explicit\x20relation\x20entry\x20referencing\x20the\x20same\x20field.'}],RELATION_TYPES=[{'name':'belongsTo','description':a0_0x1f19cb(0xad),'requiredFields':['type','localKey',a0_0x1f19cb(0xf1)],'optionalFields':[a0_0x1f19cb(0xf4),'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':a0_0x1f19cb(0xd7),'description':a0_0x1f19cb(0xfe),'requiredFields':[a0_0x1f19cb(0xed),'localKey','references'],'optionalFields':[a0_0x1f19cb(0xf4),a0_0x1f19cb(0xc7),a0_0x1f19cb(0xd3)],'example':a0_0x1f19cb(0xb7)},{'name':'hasMany','description':a0_0x1f19cb(0xb2),'requiredFields':['type',a0_0x1f19cb(0xff),a0_0x1f19cb(0xf1)],'optionalFields':['target',a0_0x1f19cb(0xc7),'onUpdate'],'example':'{\x20type:\x20\x22hasMany\x22,\x20target:\x20\x22order_item\x22,\x20localKey:\x20\x22order_id\x22,\x20references:\x20\x22order_id\x22\x20}'}],REFERENTIAL_ACTIONS=[{'name':a0_0x1f19cb(0xc8),'description':a0_0x1f19cb(0xe6),'appliesTo':[a0_0x1f19cb(0xc7),'onUpdate']},{'name':'restrict','description':'Reject\x20delete/update\x20when\x20child\x20rows\x20exist.','appliesTo':['onDelete','onUpdate']},{'name':'setNull','description':a0_0x1f19cb(0xfd),'appliesTo':[a0_0x1f19cb(0xc7),a0_0x1f19cb(0xd3)],'notes':a0_0x1f19cb(0x105)},{'name':'noAction','description':a0_0x1f19cb(0xbc),'appliesTo':['onDelete',a0_0x1f19cb(0xd3)]}],CHECK_OPERATIONS=[{'name':'in','description':'Value\x20must\x20be\x20one\x20of\x20a\x20list.','valueType':a0_0x1f19cb(0x101),'example':'{\x20field:\x20\x22status\x22,\x20in:\x20[\x22active\x22,\x20\x22inactive\x22]\x20}'},{'name':'eq','description':'Equal\x20to.','valueType':'scalar','example':a0_0x1f19cb(0x103)},{'name':a0_0x1f19cb(0x100),'description':a0_0x1f19cb(0xf3),'valueType':'scalar','example':'{\x20field:\x20\x22type\x22,\x20neq:\x20\x22system\x22\x20}'},{'name':'gt','description':'Greater\x20than.','valueType':a0_0x1f19cb(0xc6),'example':a0_0x1f19cb(0xbe)},{'name':'gte','description':a0_0x1f19cb(0xf8),'valueType':a0_0x1f19cb(0xc6),'example':a0_0x1f19cb(0xba)},{'name':'lt','description':'Less\x20than.','valueType':'numeric','example':'{\x20field:\x20\x22discount\x22,\x20lt:\x20100\x20}'},{'name':'lte','description':a0_0x1f19cb(0xea),'valueType':'numeric','example':a0_0x1f19cb(0xa9)}],AUDIT_COLUMNS={'description':a0_0x1f19cb(0xc4),'columns':[{'name':a0_0x1f19cb(0x106),'shorthand':'timestamp\x20default:now()','nullable':!![],'purpose':a0_0x1f19cb(0xc2)},{'name':a0_0x1f19cb(0xe1),'shorthand':a0_0x1f19cb(0xee),'nullable':!![],'purpose':a0_0x1f19cb(0xce)},{'name':'updated_at','shorthand':a0_0x1f19cb(0xd1),'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':'Yes\x20—\x20dbschema:init\x20emits\x20all\x204\x20columns\x20by\x20default.\x20Tables\x20that\x20do\x20not\x20need\x20audit\x20(lookup\x20tables,\x20system\x20tables)\x20can\x20manually\x20remove\x20these\x20fields\x20from\x20the\x20generated\x20skeleton.','nullablePolicy':a0_0x1f19cb(0xe7),'relationToRdf':a0_0x1f19cb(0xc0)}},SHORTHAND_SYNTAX={'format':'<type>[:<modifier>]\x20[<constraint>[:<value>]]...','description':a0_0x1f19cb(0xf6),'rules':[a0_0x1f19cb(0xfa),a0_0x1f19cb(0xdc),a0_0x1f19cb(0xab),'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_0x1f19cb(0xd5)],'examples':[{'input':'string:36\x20pk','meaning':a0_0x1f19cb(0xc5)},{'input':'string:255\x20notnull','meaning':a0_0x1f19cb(0xd6)},{'input':'decimal:15,2\x20notnull\x20default:0','meaning':a0_0x1f19cb(0x108)},{'input':a0_0x1f19cb(0xb3),'meaning':'BOOLEAN\x20DEFAULT\x20TRUE'},{'input':a0_0x1f19cb(0xc1),'meaning':'VARCHAR(100)\x20DEFAULT\x20\x27pending\x27'},{'input':'timestamp\x20default:now()','meaning':'TIMESTAMP\x20DEFAULT\x20<native_now\x20per\x20dialect>'},{'input':a0_0x1f19cb(0xcd),'meaning':'VARCHAR(36)\x20with\x20FK\x20to\x20category(id),\x20auto\x20belongsTo\x20relation'},{'input':'string:64\x20index','meaning':'VARCHAR(64)\x20with\x20single-column\x20index'}]},NAMING_RULES={'tableName':{'format':a0_0x1f19cb(0xbb),'description':'Table\x20name\x20must\x20be\x20lowercase,\x20digits,\x20and\x20underscores\x20only.'},'fieldName':{'format':'snake_case','description':a0_0x1f19cb(0xb5)},'constraintName':{'format':'<prefix>_<table>_<suffix>','defaultMaxLength':0x1e,'fallbackStrategy':a0_0x1f19cb(0xdd),'description':a0_0x1f19cb(0xb6)},'compositeShortName':{'description':'For\x20composite\x20indexes/uniques,\x20name\x20is\x20derived\x20by\x20joining\x20columns\x20with\x20underscore\x20and\x20truncating\x20when\x20needed.'}},DIALECT_SUPPORT=[{'name':a0_0x1f19cb(0xf5),'driver':'pg','booleanStorage':a0_0x1f19cb(0xef)},{'name':'mysql','driver':'mysql2','booleanStorage':a0_0x1f19cb(0xec)},{'name':'oracle','driver':'oracledb','booleanStorage':'VARCHAR2\x20with\x20CHECK\x20constraint'},{'name':a0_0x1f19cb(0xfb),'driver':'better-sqlite3','booleanStorage':a0_0x1f19cb(0xdf)}],DBSCHEMA_CATALOG={'schemaVersion':SCHEMA_VERSION,'source':SOURCE,'summary':{'totalDefineModelOptions':DEFINE_MODEL_OPTIONS[a0_0x1f19cb(0x10b)],'totalFieldTypes':FIELD_TYPES[a0_0x1f19cb(0x10b)],'totalConstraints':CONSTRAINTS['length'],'totalRelationTypes':RELATION_TYPES['length'],'totalReferentialActions':REFERENTIAL_ACTIONS['length'],'totalCheckOperations':CHECK_OPERATIONS['length'],'totalAuditColumns':AUDIT_COLUMNS['columns']['length'],'totalDialects':DIALECT_SUPPORT[a0_0x1f19cb(0x10b)]},'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_0xad2b(){const _0x20e293=['tgvZCYb0AgfUig9YigvXDwfSihrVlG','q0Hfq0SGy29UC3rYywLUDhmUievHy2GGzw50CNKGDxnLCYbVCgvYyxrVCI1HCY1RzxKGC2HHCgu6ihSGBMfTzt8SigzPzwXKlca8B3bLCMf0B3i+oIa8DMfSDwu+ih0GD2HLCMuGpg9WzxjHDg9YpIbPCYbVBMuGB2yGy2HLy2TpCgvYyxrPB25ZicHPBIWGzxeSig5LCsWGz3qSigD0zsWGBhqSigX0zsKUifrOzsbPBNrLCM5HBcbjuIbZAgfWzsbHzNrLCIbPCI1IDwLSzgvYig5VCM1HBgL6yxrPB24GAxmGEYbUyw1LpYWGzMLLBgqSig9Wlcb2ywX1zsb9lcbIDxqGDgHHDcbPCYbot1qGDgHLihvZzxiGAw5WDxqGzM9YBwf0lG','vKfsq0HbuIaOBgL0zxjHBcaIDhj1zsiViMzHBhnLiIK','DhLWzq','C3rYAw5NoJeWma','BMf0AxzLiejpt0Xfqu4','zgvJAw1HBa','CMvMzxjLBMnLCW','zgvJAw1HBdOXnsWY','tM90igvXDwfSihrVlG','DgfYz2v0','Cg9ZDgDYzxm','u3rYAw5NihnOB3j0AgfUzcbMB3iGzMLLBgqGzgvMAw5PDgLVBI4GvhLWzsbYzxf1AxjLzcbHDcbZDgfYDcWGzM9SBg93zwqGyNKGB3b0Aw9UywWGBw9KAwzPzxiGyw5KihnWywnLlxnLCgfYyxrLzcbJB25ZDhjHAw50CY4','mty5mJqYmevkBNbmEG','r3jLyxrLCIb0AgfUig9YigvXDwfSihrVlG','rgvMyxvSDcb2ywX1zs4Gu2LUz2XLlxf1B3rLzcbMB3iGC3rYAw5NCYWGCMf3igzVCIbUDw1LCMLJl2jVB2XLyw4SigjHCMuGAwrLBNrPzMLLCIbMB3iGu1fmignVBNn0yw50CYWGAwrLBNrPzMLLCIGPigzVCIbUyxrPDMuGzNvUy3rPB24Gy2fSBhmU','vhLWzsbPCYbTyw5KyxrVCNKGyxqGDgHLigjLz2LUBMLUzYbHBMqGBxvZDcbIzsbVBMuGB2yGzMLLBgruExbLCY4','C3fSAxrL','nteXnJi1ohDKv0LTAG','u2v0iezlignVBhvTBIb0BYbovuXmig9UignOAwXKihjVD3mU','t25LlxrVlw9UzsaOAw52zxjZzsbVzIbIzwXVBMDZvg8PlIbpChrPB25HBcbPBNzLCNnLigrLy2XHCMf0Aw9Uig9UihrOzsbWyxjLBNqGC2LKztSGzg9LCYbUB3qGywzMzwn0ieretc4','Bg9JywXlzxK','BMvX','yxjYyxK','vvvjrc4GtMf0AxzLifvvsuqGB24Gug9ZDgDYzvnrtcWGvKfsq0HbuIGZnIKGB24GB3rOzxjZlG','EYbMAwvSzdOGiNr5CguIlcbLCtOGiNvZzxiIih0','CMvSyxrPB25Z','rKSGzMLLBgqGBxvZDcbIzsbUDwXSywjSzs4','y3jLyxrLzf9HDa','BM90BNvSBa','revdsu1btcGXnsWYksbot1qGtLvmtcberuzbvuXuida','Dw5PCxvL','tK9uie5vteWGy29UC3rYywLUDc4','BgvUz3rO','nJqTyML0ihnPz25LzcbPBNrLz2vYicHcsuDjtLqPlG','DMfSDwu','tM9UlxvUAxf1zsbPBMrLEgvZlIbfywnOigvUDhj5igLZigfYCMf5ig9MignVBhvTBIbUyw1LCYbVCIbVyMPLy3qGEYbUyw1LlcbJB2X1Bw5Zih0U','C2nOzw1H','EYbJyxrLz29YEtOGEYb0ExbLoIaIyMvSB25NC1rViIWGCMvMzxjLBMnLCZOGiMnHDgvNB3j5kgLKksiGFsb9','EYbMAwvSzdOGiMrPC2nVDw50iIWGBhrLoIaXmdaGFq','C3rYAw5NoJm2ihbR','u3rHBMrHBg9UzsbJB25ZDhjHAw50CYaOCgSSig5VDg51BgWSihvUAxf1zsWGAw5KzxGPihrHA2uGBM8GDMfSDwuUifrOzsbSzwDHy3KGyxv0B1vWzgf0zsb0B2TLBIbPCYbZDgLSBcbWyxjZzwqGzM9YigjHy2T3yxjKignVBxbHDgLIAwXPDhKGyNv0igLZigrLChjLy2f0zwqGyw5KigHHCYbUBYbMDw5JDgLVBMfSigvMzMvJDc4','mtyZotq3mfbLsKzjva','twfUEs10BY1VBMuUifrOAxmGDgfIBguGAg9SzhmGDgHLiezlignVBhvTBIbYzwzLCMvUy2LUzYb0AguGCgfYzw50ihrHyMXLlG','zgjZy2HLBweTy2f0ywXVzW','mtGZmdy5me9Mv2fWwa','Aw50zwDLCG','EYbPzdOGiNn0CMLUzZOZnIbWAYiSig5HBwu6icjZDhjPBMC6mJu1ig5VDg51BgWIih0','t25LlxrVlw1HBNKGkgLUDMvYC2uGB2yGyMvSB25NC1rVks4Gt3b0Aw9UywWGAw52zxjZzsbKzwnSyxjHDgLVBIbVBIb0AguGCgfYzw50ihnPzgu7igrVzxmGBM90igfMzMvJDcbereWU','yM9VBgvHBIbKzwzHDwX0oNrYDwu','yMLNAw50','q29SDw1Uig5HBwuGBxvZDcbIzsbZBMfRzv9JyxnLlG','q29UC3rYywLUDcbUyw1LCYbHDxrVlwDLBMvYyxrLzcb3AxrOihr5CguTC3bLy2LMAwmGChjLzML4icHWAYWGzMSSigLKEcWGDxeSignRksbHBMqGzgLHBgvJDc1ZCgvJAwzPyYbTyxGGBgvUz3rOlG','EYb0ExbLoIaIAgfZt25LiIWGDgfYz2v0oIaIChjVzMLSzsiSigXVy2fSs2v5oIaIDxnLCL9PzciSihjLzMvYzw5Jzxm6icj1C2vYx2LKiIb9','uhjLy2LZAw9UigfUzcbZy2fSzsbYzxf1AxjLzcbMB3iGy3jVC3mTzgLHBgvJDcbWB3j0ywjPBgL0Es4','ChjPBwfYEuTLEq','EYbMAwvSzdOGiNf0EsiSigD0ztOGmcb9','C25HA2vFy2fZzq','rgvMzxiGy29UC3rYywLUDcbJAgvJAYaOC2vTyw50AwnHBgX5ihnPBwLSyxiGDg8GCMvZDhjPy3qGB24GBw9ZDcbKAwfSzwn0CYKU','rM9YzwLNBIbRzxKGCMvMzxjLBMnLlIbbDxrVlwDLBMvYyxrLCYbHigjLBg9Uz3nuBYbYzwXHDgLVBIbLBNrYEs4','EYbMAwvSzdOGiNf0EsiSigD0oIaWih0','mJa3nJK4neTtzLnZyW','vgHLifjerIbNzw5LCMf0B3iGkgnVzgvNzw5Fy3jLyxrLx2vUzhbVAw50ksbHC3n1BwvZihrOzxnLidqGy29SDw1UCYbLEgLZDcb3AgvUihrOzsbWyxLSB2fKlMf1zgL0q29SDw1UCYbRzxKGAxmGywjZzw50lIbtzwuGzMLLBgqTDMfSAwrHDgLVBJPJyxrHBg9Nigf1zgL0q29SDw1UCYbMB3iGDgHLifjerI1ZAwrLigjLAgf2Aw9YlIbeCMLMDcbIzxr3zwvUifnerIaOBwLZC2LUzYbHDwrPDcbJB2X1Bw5ZksbHBMqGuKrgicHHC3n1BwvZigf1zgL0ignVBhvTBNmPignHDxnLCYbYDw50Aw1LigvYCM9YCYb3AgvUihrOzsbNzw5LCMf0zwqGzw5KCg9PBNqGDhjPzxmGDg8GD3jPDguGy3jLyxrLzf9IEs91CgrHDgvKx2j5ihrVigeGBM9Ulwv4Axn0zw50ignVBhvTBI4','C3rYAw5NoJeWmcbKzwzHDwX0oIDWzw5KAw5NjW','uMvJB3jKignYzwf0Aw9UihrPBwvZDgfTCc4Gqxv0BY1ZzxqGB24Gsu5trvjuihzPysberuzbvuXuig5VDYGPlIboyxrPDMuGzNvUy3rPB24GDhjHBNnSyxrLCYbWzxiGzgLHBgvJDcaOCg9ZDgDYzxm6ienvuLjftLrFveLnrvnuqu1qlcbTExnXBdOGtK9xkcKSig9YywnSztOGu1LtveLnrvnuqu1qlcbZCwXPDgu6ienvuLjftLrFveLnrvnuqu1qks4','zMLLBgrZ','u3rHBMrHCMqGnc1JB2X1Bw4GyxvKAxqGy29UDMvUDgLVBIbMB3iGDgfIBgvZig1HBMfNzwqGyNKGuKvtvezVCMDLlIbuAguGC2fTzsbJB252zw50Aw9UigLZigrVy3vTzw50zwqGAw4GzMLLBgqTDMfSAwrHDgLVBJPJyxrHBg9NicHHDwrPDenVBhvTBNmGC2vJDgLVBIKGzM9YihrOzsbsreyVyMfJA2vUzcbSyxLLCI4Gu0rgigfUzcbsreyGC3rHEsbHBgLNBMvKihDOzw4GyM90AcbKzwnSyxjLihrOzxnLidqGy29SDw1UCY4','vKfsq0HbuIGZnIKGufjjtufswsblrvK','BNvTzxjPyW','B25ezwXLDgu','y2fZy2fKzq','C3rYAw5NoJi1nq','vw5PCxvLignVBNn0CMfPBNrZicHZAw5NBguGB3iGy29TCg9ZAxrLks4','C3rHBMrHBg9Uzq','C3rYAw5NoJi1nsbUB3rUDwXS','C3rYAw5NoJm2igzRoMnHDgvNB3j5lMLK','vxnLCIbPzgvUDgLMAwvYicH1C2vYBMfTzsWGDxnLCL9PzcWGB3iGy29TCg9ZAxrLihrVA2vUksbVzIb0AguGCMvJB3jKignYzwf0B3iUifnLDcbIEsbHChbSAwnHDgLVBIbSyxLLCIbVBIbjtLnfuLqSig5VDcbIEsbKyxrHyMfZzsbKzwzHDwX0lG','yxv0B1vWzgf0zq','Dw5PCxvLCW','DgLTzxn0yw1W','Ahr0Chm6lY9Yzxn0zM9Yz2uUzgv2l2rVy3mVy2XPl3nJAgvTys1KzwzPBML0Aw9U','B25vCgrHDgu','ndiXmJiYr1jkDvfX','rM9YzwLNBIbRzxKGDxnLCYbKB3qGBM90yxrPB246igzRoJX0ywjSzt4UpgnVBhvTBJ4GkguUzY4SigzRoMnHDgvNB3j5lMLKks4GvgHLihbHCNnLCIbYzwPLy3rZihbHCMvUDgHLC2vZihn5BNrHEc4','vKfsq0HbuIGYntuPie5pvcbovuXm','AgfZt25L','ms4W','zgf0zq','vMfYAwfIBguTBgvUz3rOihrLEhqUieXLBMD0AcbTB2rPzMLLCIbYzxf1AxjLzc4','yM9VBgvHBG','tw9KAwzPzxiGAxmGBwfUzgf0B3j5igzVCIb0ExbLicjZDhjPBMCIicHSzw5NDgGPigfUzcaIzgvJAw1HBciGkhbYzwnPC2LVBIXZy2fSzsKU','tuq1igHHC2GGoc1JAgfYywn0zxiGC3vMzML4ihDOzw4GBMfTzsbLEgnLzwrZig1HEeXLBMD0Aa','mtiXntuZmLrZyML4AG','vevyva','t2jQzwn0ig9MigzPzwXKigrLzMLUAxrPB25ZigLUihnOB3j0AgfUzcbZDhjPBMCGzM9YBwf0lG','y3jLyxrLzf9IEq','twfYA3mGzMLLBgqGyxmGChjPBwfYEsbRzxKU','w1SIy2f0zwDVCNLFy29KzsjDlcbBiNrLBMfUDf9PzciSicjJyxrLz29YEv9JB2rLiL1D','ANnVBG','C3rYAw5NoJXSzw5NDgG+','q2fZy2fKzsbKzwXLDguVDxbKyxrLihrVignOAwXKihjVD3mU','qwXSidqGy29SDw1UCYbHCMuGBNvSBgfIBguUifrOzsbJCMvHDgvKx2j5l3vWzgf0zwrFyNKGy29SDw1UCYbTyxKGyMuGzw1WDhKGzM9YihjVD3mGAw5Zzxj0zwqGyNKGC3LZDgvTihnLzwrZlcbTAwDYyxrPB25ZlcbVCIbIyxrJAcbPBxbVCNrZihrOyxqGzg8GBM90igHHDMuGysb1C2vYignVBNrLEhqU','B2jQzwn0','y2HLy2TZ'];a0_0xad2b=function(){return _0x20e293;};return a0_0xad2b();}function a0_0x3586(_0x4cd1f0,_0x5d2e45){_0x4cd1f0=_0x4cd1f0-0xa4;const _0xad2bf=a0_0xad2b();let _0x358643=_0xad2bf[_0x4cd1f0];if(a0_0x3586['RxZVoV']===undefined){var _0x49ed6a=function(_0x574774){const _0xffb4b9='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x17b9de='',_0x4eb260='';for(let _0x5c6de2=0x0,_0x5c9c52,_0x976d0d,_0x16c55d=0x0;_0x976d0d=_0x574774['charAt'](_0x16c55d++);~_0x976d0d&&(_0x5c9c52=_0x5c6de2%0x4?_0x5c9c52*0x40+_0x976d0d:_0x976d0d,_0x5c6de2++%0x4)?_0x17b9de+=String['fromCharCode'](0xff&_0x5c9c52>>(-0x2*_0x5c6de2&0x6)):0x0){_0x976d0d=_0xffb4b9['indexOf'](_0x976d0d);}for(let _0x43cdad=0x0,_0x220d12=_0x17b9de['length'];_0x43cdad<_0x220d12;_0x43cdad++){_0x4eb260+='%'+('00'+_0x17b9de['charCodeAt'](_0x43cdad)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x4eb260);};a0_0x3586['quDmVk']=_0x49ed6a,a0_0x3586['wAcsyH']={},a0_0x3586['RxZVoV']=!![];}const _0x9ef3e6=_0xad2bf[0x0],_0x4af586=_0x4cd1f0+_0x9ef3e6,_0x291455=a0_0x3586['wAcsyH'][_0x4af586];return!_0x291455?(_0x358643=a0_0x3586['quDmVk'](_0x358643),a0_0x3586['wAcsyH'][_0x4af586]=_0x358643):_0x358643=_0x291455,_0x358643;}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};
1
+ 'use strict';const a0_0x23f44f=a0_0x196e;(function(_0x22fb9b,_0x314ad9){const _0x164c1a=a0_0x196e,_0x159c9a=_0x22fb9b();while(!![]){try{const _0x3c1d71=-parseInt(_0x164c1a(0x15f))/0x1*(parseInt(_0x164c1a(0x11d))/0x2)+-parseInt(_0x164c1a(0x146))/0x3*(-parseInt(_0x164c1a(0x121))/0x4)+parseInt(_0x164c1a(0x16b))/0x5+parseInt(_0x164c1a(0x11b))/0x6+-parseInt(_0x164c1a(0x109))/0x7*(-parseInt(_0x164c1a(0x12c))/0x8)+parseInt(_0x164c1a(0x13c))/0x9*(-parseInt(_0x164c1a(0x10a))/0xa)+-parseInt(_0x164c1a(0x117))/0xb;if(_0x3c1d71===_0x314ad9)break;else _0x159c9a['push'](_0x159c9a['shift']());}catch(_0x3fb667){_0x159c9a['push'](_0x159c9a['shift']());}}}(a0_0x3eca,0x6cffe));function a0_0x196e(_0x2a33cf,_0x4ba9a6){_0x2a33cf=_0x2a33cf-0xfa;const _0x3ecac0=a0_0x3eca();let _0x196eb2=_0x3ecac0[_0x2a33cf];if(a0_0x196e['MpkgVr']===undefined){var _0x1f8c62=function(_0x143368){const _0xd98d2a='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x24b7e6='',_0x231e93='';for(let _0x572881=0x0,_0x427cb9,_0x504847,_0x459dc0=0x0;_0x504847=_0x143368['charAt'](_0x459dc0++);~_0x504847&&(_0x427cb9=_0x572881%0x4?_0x427cb9*0x40+_0x504847:_0x504847,_0x572881++%0x4)?_0x24b7e6+=String['fromCharCode'](0xff&_0x427cb9>>(-0x2*_0x572881&0x6)):0x0){_0x504847=_0xd98d2a['indexOf'](_0x504847);}for(let _0x1befd0=0x0,_0x408179=_0x24b7e6['length'];_0x1befd0<_0x408179;_0x1befd0++){_0x231e93+='%'+('00'+_0x24b7e6['charCodeAt'](_0x1befd0)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x231e93);};a0_0x196e['IFYymA']=_0x1f8c62,a0_0x196e['uAcgMO']={},a0_0x196e['MpkgVr']=!![];}const _0x1d22c2=_0x3ecac0[0x0],_0x395905=_0x2a33cf+_0x1d22c2,_0x5e8a5d=a0_0x196e['uAcgMO'][_0x395905];return!_0x5e8a5d?(_0x196eb2=a0_0x196e['IFYymA'](_0x196eb2),a0_0x196e['uAcgMO'][_0x395905]=_0x196eb2):_0x196eb2=_0x5e8a5d,_0x196eb2;}const SCHEMA_VERSION=a0_0x23f44f(0x12d),SOURCE=a0_0x23f44f(0x115),DOCUMENTATION_URL='https://restforge.dev/docs/cli/schema-definition',DEFINE_MODEL_OPTIONS=[{'name':'schema','required':![],'type':a0_0x23f44f(0x10f),'description':a0_0x23f44f(0x129),'notes':a0_0x23f44f(0x130)},{'name':a0_0x23f44f(0x10e),'required':!![],'type':'object','description':a0_0x23f44f(0x161),'example':a0_0x23f44f(0x13a)},{'name':'primaryKey','required':![],'type':a0_0x23f44f(0x10b),'description':'Override\x20primary\x20key.\x20Single\x20field\x20name\x20or\x20composite\x20array.','notes':a0_0x23f44f(0x10d)},{'name':'relations','required':![],'type':'object','description':'Inter-table\x20relations\x20(belongsTo,\x20hasOne,\x20hasMany).','example':'{\x20category:\x20{\x20type:\x20\x22belongsTo\x22,\x20references:\x20\x22category(id)\x22\x20}\x20}'},{'name':'indexes','required':![],'type':a0_0x23f44f(0x113),'description':a0_0x23f44f(0x169),'example':a0_0x23f44f(0x154)},{'name':a0_0x23f44f(0x15e),'required':![],'type':a0_0x23f44f(0x113),'description':a0_0x23f44f(0x159),'example':a0_0x23f44f(0x118)},{'name':a0_0x23f44f(0x145),'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':a0_0x23f44f(0x105)}],FIELD_TYPES=[{'name':a0_0x23f44f(0x147),'description':a0_0x23f44f(0x136),'requiresModifier':!![],'modifierFormat':a0_0x23f44f(0x162),'example':a0_0x23f44f(0x149),'notes':a0_0x23f44f(0x15c)},{'name':a0_0x23f44f(0x133),'description':'Long-form\x20text\x20without\x20length\x20limit\x20(TEXT/CLOB).','requiresModifier':![],'example':'text'},{'name':a0_0x23f44f(0xfb),'description':'32-bit\x20signed\x20integer\x20(INT/INTEGER).','requiresModifier':![],'example':a0_0x23f44f(0xfb)},{'name':a0_0x23f44f(0x158),'description':a0_0x23f44f(0x111),'requiresModifier':![],'example':a0_0x23f44f(0x158)},{'name':a0_0x23f44f(0x11a),'description':'Fixed-point\x20decimal.\x20Precision\x20and\x20scale\x20required.','requiresModifier':!![],'modifierFormat':'decimal:<precision>,<scale>','example':a0_0x23f44f(0x164),'notes':a0_0x23f44f(0x132)},{'name':a0_0x23f44f(0x152),'description':'Boolean\x20value\x20(native\x20BOOLEAN\x20on\x20PostgreSQL,\x20VARCHAR\x20on\x20others).','requiresModifier':![],'example':a0_0x23f44f(0x152)},{'name':'date','description':a0_0x23f44f(0x13e),'requiresModifier':![],'example':a0_0x23f44f(0x13b)},{'name':a0_0x23f44f(0x131),'description':a0_0x23f44f(0x13d),'requiresModifier':![],'example':'timestamp','notes':'Default\x20timezone\x20behavior\x20is\x20dialect-specific.'},{'name':'uuid','description':a0_0x23f44f(0x110),'requiresModifier':![],'example':'uuid'},{'name':a0_0x23f44f(0xfc),'description':a0_0x23f44f(0x102),'requiresModifier':![],'example':'json'}],CONSTRAINTS=[{'name':'pk','kind':a0_0x23f44f(0x124),'description':'Marks\x20field\x20as\x20primary\x20key.','example':a0_0x23f44f(0x14b)},{'name':a0_0x23f44f(0xfa),'kind':a0_0x23f44f(0x124),'description':a0_0x23f44f(0x16e),'example':a0_0x23f44f(0x166)},{'name':'unique','kind':'standalone','description':'Single-column\x20unique\x20constraint.','example':'string:32\x20unique'},{'name':a0_0x23f44f(0x157),'kind':a0_0x23f44f(0x124),'description':'Single-column\x20non-unique\x20index.','example':a0_0x23f44f(0x167)},{'name':'autoUpdate','kind':a0_0x23f44f(0x124),'deprecated':!![],'description':a0_0x23f44f(0x12e),'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':a0_0x23f44f(0x106),'kind':'value','description':'Default\x20value.\x20Single-quoted\x20for\x20strings,\x20raw\x20for\x20numeric/boolean,\x20bare\x20identifier\x20for\x20SQL\x20constants,\x20identifier()\x20for\x20native\x20function\x20calls.','valueFormat':'default:<literal>\x20|\x20default:\x27<string>\x27\x20|\x20default:<constant>\x20|\x20default:<function>()','example':a0_0x23f44f(0x100)},{'name':'fk','kind':'value','description':a0_0x23f44f(0x16f),'valueFormat':a0_0x23f44f(0xff),'example':a0_0x23f44f(0x125),'notes':'Cannot\x20coexist\x20with\x20explicit\x20relation\x20entry\x20referencing\x20the\x20same\x20field.'}],RELATION_TYPES=[{'name':'belongsTo','description':a0_0x23f44f(0x16d),'requiredFields':['type','localKey','references'],'optionalFields':['target','onDelete','onUpdate'],'example':a0_0x23f44f(0xfe),'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':a0_0x23f44f(0x12a),'requiredFields':['type','localKey',a0_0x23f44f(0x150)],'optionalFields':[a0_0x23f44f(0x12f),'onDelete','onUpdate'],'example':a0_0x23f44f(0x12b)},{'name':'hasMany','description':'One-to-many\x20(inverse\x20of\x20belongsTo).\x20Optional\x20inverse\x20declaration\x20on\x20the\x20parent\x20side;\x20does\x20not\x20affect\x20DDL.','requiredFields':[a0_0x23f44f(0x120),a0_0x23f44f(0x148),a0_0x23f44f(0x150)],'optionalFields':[a0_0x23f44f(0x12f),'onDelete','onUpdate'],'example':'{\x20type:\x20\x22hasMany\x22,\x20target:\x20\x22order_item\x22,\x20localKey:\x20\x22order_id\x22,\x20references:\x20\x22order_id\x22\x20}'}],REFERENTIAL_ACTIONS=[{'name':a0_0x23f44f(0x134),'description':a0_0x23f44f(0x141),'appliesTo':['onDelete','onUpdate']},{'name':'restrict','description':a0_0x23f44f(0xfd),'appliesTo':['onDelete',a0_0x23f44f(0x103)]},{'name':'setNull','description':a0_0x23f44f(0x104),'appliesTo':[a0_0x23f44f(0x160),a0_0x23f44f(0x103)],'notes':'FK\x20field\x20must\x20be\x20nullable.'},{'name':a0_0x23f44f(0x142),'description':'Defer\x20constraint\x20check\x20(semantically\x20similar\x20to\x20restrict\x20on\x20most\x20dialects).','appliesTo':['onDelete',a0_0x23f44f(0x103)]}],CHECK_OPERATIONS=[{'name':'in','description':'Value\x20must\x20be\x20one\x20of\x20a\x20list.','valueType':'array','example':a0_0x23f44f(0x143)},{'name':'eq','description':a0_0x23f44f(0x11f),'valueType':a0_0x23f44f(0x122),'example':a0_0x23f44f(0x135)},{'name':a0_0x23f44f(0x138),'description':a0_0x23f44f(0x155),'valueType':'scalar','example':a0_0x23f44f(0x14a)},{'name':'gt','description':a0_0x23f44f(0x14d),'valueType':a0_0x23f44f(0x144),'example':'{\x20field:\x20\x22qty\x22,\x20gt:\x200\x20}'},{'name':a0_0x23f44f(0x14c),'description':a0_0x23f44f(0x140),'valueType':a0_0x23f44f(0x144),'example':a0_0x23f44f(0x16a)},{'name':'lt','description':a0_0x23f44f(0x101),'valueType':'numeric','example':'{\x20field:\x20\x22discount\x22,\x20lt:\x20100\x20}'},{'name':'lte','description':a0_0x23f44f(0x165),'valueType':'numeric','example':'{\x20field:\x20\x22discount\x22,\x20lte:\x20100\x20}'}],AUDIT_COLUMNS={'description':a0_0x23f44f(0x156),'columns':[{'name':a0_0x23f44f(0x10c),'shorthand':'timestamp\x20default:now()','nullable':!![],'purpose':a0_0x23f44f(0x108)},{'name':a0_0x23f44f(0x128),'shorthand':'string:100','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':'updated_at','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':'Yes\x20—\x20dbschema:init\x20emits\x20all\x204\x20columns\x20by\x20default.\x20Tables\x20that\x20do\x20not\x20need\x20audit\x20(lookup\x20tables,\x20system\x20tables)\x20can\x20manually\x20remove\x20these\x20fields\x20from\x20the\x20generated\x20skeleton.','nullablePolicy':a0_0x23f44f(0x119),'relationToRdf':a0_0x23f44f(0x14f)}},SHORTHAND_SYNTAX={'format':a0_0x23f44f(0x15a),'description':a0_0x23f44f(0x107),'rules':[a0_0x23f44f(0x126),a0_0x23f44f(0x13f),'Standalone\x20constraints\x20(pk,\x20notnull,\x20unique,\x20index)\x20take\x20no\x20value.\x20The\x20legacy\x20autoUpdate\x20token\x20is\x20still\x20parsed\x20for\x20backward\x20compatibility\x20but\x20is\x20deprecated\x20and\x20has\x20no\x20functional\x20effect.','Value\x20constraints\x20(default,\x20fk)\x20require\x20constraint:value\x20format.',a0_0x23f44f(0x114),a0_0x23f44f(0x139)],'examples':[{'input':'string:36\x20pk','meaning':a0_0x23f44f(0x123)},{'input':'string:255\x20notnull','meaning':'VARCHAR(255)\x20NOT\x20NULL'},{'input':a0_0x23f44f(0x163),'meaning':a0_0x23f44f(0x127)},{'input':'boolean\x20default:true','meaning':'BOOLEAN\x20DEFAULT\x20TRUE'},{'input':a0_0x23f44f(0x116),'meaning':'VARCHAR(100)\x20DEFAULT\x20\x27pending\x27'},{'input':a0_0x23f44f(0x151),'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':'VARCHAR(64)\x20with\x20single-column\x20index'}]},NAMING_RULES={'tableName':{'format':a0_0x23f44f(0x153),'description':a0_0x23f44f(0x168)},'fieldName':{'format':'snake_case','description':'Column\x20name\x20must\x20be\x20snake_case.'},'constraintName':{'format':'<prefix>_<table>_<suffix>','defaultMaxLength':0x1e,'fallbackStrategy':'MD5\x20hash\x208-character\x20suffix\x20when\x20name\x20exceeds\x20maxLength','description':a0_0x23f44f(0x11e)},'compositeShortName':{'description':'For\x20composite\x20indexes/uniques,\x20name\x20is\x20derived\x20by\x20joining\x20columns\x20with\x20underscore\x20and\x20truncating\x20when\x20needed.'}},DIALECT_SUPPORT=[{'name':a0_0x23f44f(0x137),'driver':'pg','booleanStorage':'native\x20BOOLEAN'},{'name':'mysql','driver':a0_0x23f44f(0x112),'booleanStorage':'VARCHAR\x20(literal\x20\x22true\x22/\x22false\x22)'},{'name':'oracle','driver':'oracledb','booleanStorage':a0_0x23f44f(0x11c)},{'name':'sqlite','driver':a0_0x23f44f(0x16c),'booleanStorage':a0_0x23f44f(0x15d)}],DBSCHEMA_CATALOG={'schemaVersion':SCHEMA_VERSION,'source':SOURCE,'summary':{'totalDefineModelOptions':DEFINE_MODEL_OPTIONS[a0_0x23f44f(0x14e)],'totalFieldTypes':FIELD_TYPES[a0_0x23f44f(0x14e)],'totalConstraints':CONSTRAINTS[a0_0x23f44f(0x14e)],'totalRelationTypes':RELATION_TYPES[a0_0x23f44f(0x14e)],'totalReferentialActions':REFERENTIAL_ACTIONS['length'],'totalCheckOperations':CHECK_OPERATIONS['length'],'totalAuditColumns':AUDIT_COLUMNS['columns'][a0_0x23f44f(0x14e)],'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_0x23f44f(0x15b)]={'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};function a0_0x3eca(){const _0x2a2813=['u3rHBMrHCMqGnc1JB2X1Bw4GyxvKAxqGy29UDMvUDgLVBIbMB3iGDgfIBgvZig1HBMfNzwqGyNKGuKvtvezVCMDLlIbuAguGC2fTzsbJB252zw50Aw9UigLZigrVy3vTzw50zwqGAw4GzMLLBgqTDMfSAwrHDgLVBJPJyxrHBg9NicHHDwrPDenVBhvTBNmGC2vJDgLVBIKGzM9YihrOzsbsreyVyMfJA2vUzcbSyxLLCI4Gu0rgigfUzcbsreyGC3rHEsbHBgLNBMvKihDOzw4GyM90AcbKzwnSyxjLihrOzxnLidqGy29SDw1UCY4','Aw5KzxG','yMLNAw50','vw5PCxvLignVBNn0CMfPBNrZicHZAw5NBguGB3iGy29TCg9ZAxrLks4','phr5Cgu+wZO8Bw9KAwzPzxi+xsbBpgnVBNn0CMfPBNq+wZO8DMfSDwu+xv0UlI4','zxHWB3j0CW','rxHWBgLJAxqGBgvUz3rOihjLCxvPCMvKigzVCIbJCM9ZCY1KAwfSzwn0ihbVCNrHyMLSAxr5lG','vevyva','Dw5PCxvLCW','mJq3zLvpte5J','B25ezwXLDgu','t2jQzwn0ig9MigzPzwXKigrLzMLUAxrPB25ZigLUihnOB3j0AgfUzcbZDhjPBMCGzM9YBwf0lG','C3rYAw5NoJXSzw5NDgG+','zgvJAw1HBdOXnsWYig5VDg51BgWGzgvMyxvSDdOW','zgvJAw1HBdOXnsWY','tgvZCYb0AgfUig9YigvXDwfSihrVlG','C3rYAw5NoJi1nsbUB3rUDwXS','C3rYAw5NoJy0igLUzgv4','vgfIBguGBMfTzsbTDxn0igjLigXVD2vYy2fZzsWGzgLNAxrZlcbHBMqGDw5KzxjZy29YzxmGB25SEs4','tM9UlxvUAxf1zsbPBMrLEgvZlIbfywnOigvUDhj5igLZigfYCMf5ig9MignVBhvTBIbUyw1LCYbVCIbVyMPLy3qGEYbUyw1LlcbJB2X1Bw5Zih0U','EYbMAwvSzdOGiNf0EsiSigD0ztOGmcb9','mtC2mdmWmgDHDLrUBW','yMv0DgvYlxnXBgL0ztm','twfUEs10BY1VBMuUifrOAxmGDgfIBguGAg9SzhmGDgHLiezlignVBhvTBIbYzwzLCMvUy2LUzYb0AguGCgfYzw50ihrHyMXLlG','tK9uie5vteWGy29UC3rYywLUDc4','rM9YzwLNBIbRzxKGCMvMzxjLBMnLlIbbDxrVlwDLBMvYyxrLCYbHigjLBg9Uz3nuBYbYzwXHDgLVBIbLBNrYEs4','BM90BNvSBa','Aw50zwDLCG','ANnVBG','uMvQzwn0igrLBgv0zs91CgrHDguGD2HLBIbJAgLSzcbYB3DZigv4Axn0lG','EYb0ExbLoIaIyMvSB25NC1rViIWGBg9JywXlzxK6icjJyxrLz29YEv9PzciSihjLzMvYzw5Jzxm6icjJyxrLz29YEv9PzciSig9UrgvSzxrLoIaICMvZDhjPy3qIih0','zMS6phrHyMXLpI48y29SDw1UpG','yM9VBgvHBIbKzwzHDwX0oNrYDwu','tgvZCYb0AgfUlG','sLnptIbJB2X1Bw4GkePtt05cig9UifbVC3rNCMvtuuWSiePtt04GB24GtxLtuuWSienmt0iGB24Gt3jHy2XLks4','B25vCgrHDgu','u2v0iezlignVBhvTBIb0BYbovuXmig9UignOAwXKihjVD3mU','w3SGzMLLBgq6icjZDgf0DxmIlcbPBJOGwYjHy3rPDMuIlcaIAw5Hy3rPDMuIxsb9lcb7igzPzwXKoIaICxr5iIWGz3rLoIaWih1D','zgvMyxvSDa','u3rYAw5NihnOB3j0AgfUzcbMB3iGzMLLBgqGzgvMAw5PDgLVBI4GvhLWzsbYzxf1AxjLzcbHDcbZDgfYDcWGzM9SBg93zwqGyNKGB3b0Aw9UywWGBw9KAwzPzxiGyw5KihnWywnLlxnLCgfYyxrLzcbJB25ZDhjHAw50CY4','uMvJB3jKignYzwf0Aw9UihrPBwvZDgfTCc4Gqxv0BY1ZzxqGB24Gsu5trvjuihzPysberuzbvuXuig5VDYGPlIboyxrPDMuGzNvUy3rPB24GDhjHBNnSyxrLCYbWzxiGzgLHBgvJDcaOCg9ZDgDYzxm6ienvuLjftLrFveLnrvnuqu1qlcbTExnXBdOGtK9xkcKSig9YywnSztOGu1LtveLnrvnuqu1qlcbZCwXPDgu6ienvuLjftLrFveLnrvnuqu1qks4','n0vIsMrbAa','ntuYmtbZEgf5tNC','C3rYAw5NFgfYCMf5','y3jLyxrLzf9HDa','qxv0BY1KzxrLy3rLzcbMCM9TigzPzwXKihnOB3j0AgfUzcaICgSIihDOzw4GBM90ihnWzwnPzMLLzc4','zMLLBgrZ','C3rYAw5NFg51BgW','vvvjrc4GtMf0AxzLifvvsuqGB24Gug9ZDgDYzvnrtcWGvKfsq0HbuIGZnIKGB24GB3rOzxjZlG','nJqTyML0ihnPz25LzcbPBNrLz2vYicHcsuDjtLqPlG','BxLZCwWY','yxjYyxK','rgvMyxvSDcb2ywX1zsbMB3jTyxqGzgvWzw5KCYbVBIb0ExbLoIbYyxCGzM9YigjVB2XLyw4GkgrLzMf1Bhq6Dhj1zsKSihjHDYbMB3iGAw50zwDLCIaOzgvMyxvSDdOWksWGC2LUz2XLlxf1B3rLzcbMB3iGC3rYAw5NicHKzwzHDwX0oID2ywX1zsCPlcbIyxjLigLKzw50AwzPzxiGzM9YifnrtcbJB25ZDgfUDhmGkgrLzMf1Bhq6y3vYCMvUDf9KyxrLksWGAwrLBNrPzMLLCIGPigzVCIbUyxrPDMuGzNvUy3rPB24Gy2fSBhmGkgrLzMf1Bhq6BM93kcKPlG','zgjZy2HLBweTy2f0ywXVzW','C3rYAw5NoJeWmcbKzwzHDwX0oIDWzw5KAw5NjW','mte2nJyXmtzJuenbtNG','w1SIy2f0zwDVCNLFy29KzsjDlcbBiNrLBMfUDf9PzciSicjJyxrLz29YEv9JB2rLiL1D','qwXSidqGy29SDw1UCYbHCMuGBNvSBgfIBguUifrOzsbJCMvHDgvKx2j5l3vWzgf0zwrFyNKGy29SDw1UCYbTyxKGyMuGzw1WDhKGzM9YihjVD3mGAw5Zzxj0zwqGyNKGC3LZDgvTihnLzwrZlcbTAwDYyxrPB25ZlcbVCIbIyxrJAcbPBxbVCNrZihrOyxqGzg8GBM90igHHDMuGysb1C2vYignVBNrLEhqU','zgvJAw1HBa','mZy2mJq3ohfWyuH3BW','vKfsq0HbuJiGD2L0AcbdsevdsYbJB25ZDhjHAw50','mtKWnLzYB2zMEa','q29UC3rYywLUDcbUyw1LCYbHDxrVlwDLBMvYyxrLzcb3AxrOihr5CguTC3bLy2LMAwmGChjLzML4icHWAYWGzMSSigLKEcWGDxeSignRksbHBMqGzgLHBgvJDc1ZCgvJAwzPyYbTyxGGBgvUz3rOlG','rxf1ywWGDg8U','DhLWzq','mtG0nZK2CLzTA3rq','C2nHBgfY','vKfsq0HbuIGZnIKGufjjtufswsblrvK','C3rHBMrHBg9Uzq','C3rYAw5NoJm2igzRoMnHDgvNB3j5lMLK','vhLWzsbPCYbTyw5KyxrVCNKGyxqGDgHLigjLz2LUBMLUzYbHBMqGBxvZDcbIzsbVBMuGB2yGzMLLBgruExbLCY4','revdsu1btcGXnsWYksbot1qGtLvmtcberuzbvuXuida','y3jLyxrLzf9IEq','rgf0ywjHC2uGC2nOzw1Hig5HBwvZCgfJzsaOBxvSDgKTC2nOzw1HigrHDgfIyxnLks4','t25LlxrVlw9UzsaOAw52zxjZzsbVzIbIzwXVBMDZvg8PlIbpChrPB25HBcbPBNzLCNnLigrLy2XHCMf0Aw9Uig9UihrOzsbWyxjLBNqGC2LKztSGzg9LCYbUB3qGywzMzwn0ieretc4','EYb0ExbLoIaIAgfZt25LiIWGDgfYz2v0oIaIChjVzMLSzsiSigXVy2fSs2v5oIaIDxnLCL9PzciSihjLzMvYzw5Jzxm6icj1C2vYx2LKiIb9','mZi4nti3mKXsz2XMCq','ms4W','revquKvdqvrfrdOGvgHPCYbTB2rPzMLLCIbUBYbSB25NzxiGAgfZigz1BMn0Aw9UywWGzwzMzwn0igf0ihrOzsbereWGB3iGCNvUDgLTzsbSzxzLBc4Gqxv0BY11CgrHDguGzM9YihvWzgf0zwrFyxqGAxmGAgfUzgXLzcbIEsb0AguGuKrgigXHEwvYicHHDwrPDenVBhvTBNmGy29UDMvUDgLVBIbPBIbcyxnLtw9KzwWGCNvUDgLTzsKGyMfZzwqGB24GzMLLBgqGBMfTAw5NignVBNzLBNrPB24Sig5VDcbIEsb0AgLZifnerIbTyxjRzxiUiev4Axn0Aw5NihvZywDLigLZihbYzxnLCNzLzcbMB3iGyMfJA3DHCMqGy29TCgf0AwjPBgL0EsbIDxqGC2HVDwXKigjLihjLBw92zwqGzNjVBsbUzxCGDgvTCgXHDgvZlG','DgfYz2v0','BNvSBcbVCIbLBxb0EsbZDhjPBMCGBwvHBNmGzgvMyxvSDcbZy2HLBweU','DgLTzxn0yw1W','uhjLy2LZAw9UigfUzcbZy2fSzsbYzxf1AxjLzcbMB3iGy3jVC3mTzgLHBgvJDcbWB3j0ywjPBgL0Es4','Dgv4Da','y2fZy2fKzq','EYbMAwvSzdOGiNr5CguIlcbLCtOGiNvZzxiIih0','vMfYAwfIBguTBgvUz3rOihrLEhqUieXLBMD0AcbTB2rPzMLLCIbYzxf1AxjLzc4','Cg9ZDgDYzxm','BMvX','rM9YzwLNBIbRzxKGDxnLCYbKB3qGBM90yxrPB246igzRoJX0ywjSzt4UpgnVBhvTBJ4GkguUzY4SigzRoMnHDgvNB3j5lMLKks4GvgHLihbHCNnLCIbYzwPLy3rZihbHCMvUDgHLC2vZihn5BNrHEc4','EYbPzdOGiNn0CMLUzZOZnIbWAYiSig5HBwu6icjZDhjPBMC6mJu1ig5VDg51BgWIih0','zgf0zq','nJaZD3riuuXc','rgf0zsbHBMqGDgLTzsaOveLnrvnuqu1qks4','rgf0zsbVBMX5icHUBYb0Aw1LignVBxbVBMvUDcKU','tw9KAwzPzxiGAxmGBwfUzgf0B3j5igzVCIb0ExbLicjZDhjPBMCIicHSzw5NDgGPigfUzcaIzgvJAw1HBciGkhbYzwnPC2LVBIXZy2fSzsKU','r3jLyxrLCIb0AgfUig9YigvXDwfSihrVlG','q2fZy2fKzsbKzwXLDguVDxbKyxrLihrVignOAwXKihjVD3mU','BM9by3rPB24','EYbMAwvSzdOGiNn0yxr1CYiSigLUoIbBiMfJDgL2zsiSicjPBMfJDgL2zsjDih0','BNvTzxjPyW','y2HLy2TZ','ndHny1riC2C','C3rYAw5N','Bg9JywXlzxK','C3rYAw5NoJi1nq','EYbMAwvSzdOGiNr5CguIlcbUzxe6icjZExn0zw0Iih0','C3rYAw5NoJm2ihbR','z3rL','r3jLyxrLCIb0AgfUlG','BgvUz3rO','vgHLifjerIbNzw5LCMf0B3iGkgnVzgvNzw5Fy3jLyxrLx2vUzhbVAw50ksbHC3n1BwvZihrOzxnLidqGy29SDw1UCYbLEgLZDcb3AgvUihrOzsbWyxLSB2fKlMf1zgL0q29SDw1UCYbRzxKGAxmGywjZzw50lIbtzwuGzMLLBgqTDMfSAwrHDgLVBJPJyxrHBg9Nigf1zgL0q29SDw1UCYbMB3iGDgHLifjerI1ZAwrLigjLAgf2Aw9YlIbeCMLMDcbIzxr3zwvUifnerIaOBwLZC2LUzYbHDwrPDcbJB2X1Bw5ZksbHBMqGuKrgicHHC3n1BwvZigf1zgL0ignVBhvTBNmPignHDxnLCYbYDw50Aw1LigvYCM9YCYb3AgvUihrOzsbNzw5LCMf0zwqGzw5KCg9PBNqGDhjPzxmGDg8GD3jPDguGy3jLyxrLzf9IEs91CgrHDgvKx2j5ihrVigeGBM9Ulwv4Axn0zw50ignVBhvTBI4','CMvMzxjLBMnLCW','DgLTzxn0yw1WigrLzMf1Bhq6BM93kcK','yM9VBgvHBG','C25HA2vFy2fZzq','w1SIC3rHDhvZiIWGiMnYzwf0zwrFyxqIxv0','tM90igvXDwfSihrVlG'];a0_0x3eca=function(){return _0x2a2813;};return a0_0x3eca();}