@restforgejs/platform 5.2.12 → 5.2.16

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 (190) hide show
  1. package/bin/drift-check-linux +0 -0
  2. package/bin/sdf-tools-linux +0 -0
  3. package/bin/sdf-tools.exe +0 -0
  4. package/build-info.json +2 -2
  5. package/cli/consumer-deploy.js +1 -1
  6. package/cli/consumer.js +1 -1
  7. package/generators/cli/fast-track.js +8 -9
  8. package/generators/cli/project/auth.js +209 -0
  9. package/generators/lib/auth/component-generator.js +58 -0
  10. package/generators/lib/auth/dependency-checker.js +102 -0
  11. package/generators/lib/auth/env-injector.js +81 -0
  12. package/generators/lib/auth/migrate-runner.js +111 -0
  13. package/generators/lib/auth/prefix.js +22 -0
  14. package/generators/lib/auth/processor-generator.js +55 -0
  15. package/generators/lib/auth/sdf-generator.js +102 -0
  16. package/generators/lib/auth/template-renderer.js +29 -0
  17. package/generators/lib/auth/templates/processor/login.js.tmpl +152 -0
  18. package/generators/lib/auth/templates/processor/logout.js.tmpl +58 -0
  19. package/generators/lib/auth/templates/processor/me.js.tmpl +64 -0
  20. package/generators/lib/auth/templates/processor/refresh.js.tmpl +134 -0
  21. package/generators/lib/auth/templates/processor/register.js.tmpl +77 -0
  22. package/generators/lib/auth/templates/processor/reset-password.js.tmpl +106 -0
  23. package/generators/lib/auth/templates/rfx_auth-middleware.js.tmpl +79 -0
  24. package/generators/lib/auth/templates/rfx_auth.js.tmpl +104 -0
  25. package/generators/lib/dbschema-kit/schema-printer.js +10 -1
  26. package/generators/lib/templates/dashboard-catalog.js +1 -1
  27. package/generators/lib/templates/db-connection-env.js +1 -1
  28. package/generators/lib/templates/dbschema-catalog.js +1 -1
  29. package/generators/lib/templates/field-validation-catalog.js +1 -1
  30. package/generators/lib/templates/mysql-template.js +1 -1
  31. package/generators/lib/templates/oracle-template.js +1 -1
  32. package/generators/lib/templates/postgres-template.js +1 -1
  33. package/generators/lib/templates/query-declarative-catalog.js +1 -1
  34. package/generators/lib/templates/sqlite-template.js +1 -1
  35. package/integrity-manifest.json +18 -18
  36. package/package.json +1 -1
  37. package/scripts/verify-integrity.js +1 -1
  38. package/server.js +1 -1
  39. package/src/components/handlers/adjust_handler.js +1 -1
  40. package/src/components/handlers/audit_handler.js +1 -1
  41. package/src/components/handlers/delete_handler.js +1 -1
  42. package/src/components/handlers/export_handler.js +1 -1
  43. package/src/components/handlers/import_handler.js +1 -1
  44. package/src/components/handlers/insert_handler.js +1 -1
  45. package/src/components/handlers/update_handler.js +1 -1
  46. package/src/components/handlers/upload_handler.js +1 -1
  47. package/src/components/handlers/workflow_handler.js +1 -1
  48. package/src/components/integrations/webhook.js +1 -1
  49. package/src/consumers/baseConsumer.js +1 -1
  50. package/src/consumers/declarativeMapper.js +1 -1
  51. package/src/consumers/handlers/apiHandler.js +1 -1
  52. package/src/consumers/handlers/consoleHandler.js +1 -1
  53. package/src/consumers/handlers/databaseHandler.js +1 -1
  54. package/src/consumers/handlers/index.js +1 -1
  55. package/src/consumers/handlers/kafkaHandler.js +1 -1
  56. package/src/consumers/index.js +1 -1
  57. package/src/consumers/messageTransformer.js +1 -1
  58. package/src/consumers/validator.js +1 -1
  59. package/src/core/db/dialect/base-dialect.js +1 -1
  60. package/src/core/db/dialect/index.js +1 -1
  61. package/src/core/db/dialect/mysql-dialect.js +1 -1
  62. package/src/core/db/dialect/oracle-dialect.js +1 -1
  63. package/src/core/db/dialect/postgres-dialect.js +1 -1
  64. package/src/core/db/dialect/sqlite-dialect.js +1 -1
  65. package/src/core/db/flatten-helper.js +1 -1
  66. package/src/core/db/query-builder-error.js +1 -1
  67. package/src/core/db/query-builder.js +1 -1
  68. package/src/core/db/relation-helper.js +1 -1
  69. package/src/core/handlers/delete_handler.js +1 -1
  70. package/src/core/handlers/insert_handler.js +1 -1
  71. package/src/core/handlers/update_handler.js +1 -1
  72. package/src/core/models/base-model.js +1 -1
  73. package/src/core/utils/cache-manager.js +1 -1
  74. package/src/core/utils/component-engine.js +1 -1
  75. package/src/core/utils/context-builder.js +1 -1
  76. package/src/core/utils/datetime-formatter.js +1 -1
  77. package/src/core/utils/datetime-parser.js +1 -1
  78. package/src/core/utils/db.js +1 -1
  79. package/src/core/utils/logger.js +1 -1
  80. package/src/core/utils/payload-loader.js +1 -1
  81. package/src/core/utils/security-checks.js +1 -1
  82. package/src/middleware/body-options.js +1 -1
  83. package/src/middleware/cors.js +1 -1
  84. package/src/middleware/idempotency.js +1 -1
  85. package/src/middleware/rate-limiter.js +1 -1
  86. package/src/middleware/request-logger.js +1 -1
  87. package/src/middleware/security-headers.js +1 -1
  88. package/src/models/base-model-mysql.js +1 -1
  89. package/src/models/base-model-oracle.js +1 -1
  90. package/src/models/base-model-sqlite.js +1 -1
  91. package/src/models/base-model.js +1 -1
  92. package/src/pro/caching/redis-client.js +1 -1
  93. package/src/pro/caching/redis-helper.js +1 -1
  94. package/src/pro/consumers/baseConsumer.js +1 -1
  95. package/src/pro/consumers/declarativeMapper.js +1 -1
  96. package/src/pro/consumers/handlers/apiHandler.js +1 -1
  97. package/src/pro/consumers/handlers/consoleHandler.js +1 -1
  98. package/src/pro/consumers/handlers/databaseHandler.js +1 -1
  99. package/src/pro/consumers/handlers/index.js +1 -1
  100. package/src/pro/consumers/handlers/kafkaHandler.js +1 -1
  101. package/src/pro/consumers/index.js +1 -1
  102. package/src/pro/consumers/messageTransformer.js +1 -1
  103. package/src/pro/consumers/validator.js +1 -1
  104. package/src/pro/database/base-model-mysql.js +1 -1
  105. package/src/pro/database/base-model-oracle.js +1 -1
  106. package/src/pro/database/base-model-sqlite.js +1 -1
  107. package/src/pro/database/db-mysql.js +1 -1
  108. package/src/pro/database/db-oracle.js +1 -1
  109. package/src/pro/database/db-sqlite.js +1 -1
  110. package/src/pro/excel/excel-generator.js +1 -1
  111. package/src/pro/excel/excel-parser.js +1 -1
  112. package/src/pro/excel/export-service.js +1 -1
  113. package/src/pro/excel/export_handler.js +1 -1
  114. package/src/pro/excel/import-service.js +1 -1
  115. package/src/pro/excel/import-validator.js +1 -1
  116. package/src/pro/excel/import_handler.js +1 -1
  117. package/src/pro/excel/upsert-builder.js +1 -1
  118. package/src/pro/idgen/idgen-routes.js +1 -1
  119. package/src/pro/integrations/lookup-resolver.js +1 -1
  120. package/src/pro/integrations/upload-handler-v2.js +1 -1
  121. package/src/pro/integrations/upload-handler.js +1 -1
  122. package/src/pro/integrations/webhook.js +1 -1
  123. package/src/pro/locking/lock-routes.js +1 -1
  124. package/src/pro/locking/resource-lock-manager.js +1 -1
  125. package/src/pro/messaging/kafkaConsumerService.js +1 -1
  126. package/src/pro/messaging/kafkaService.js +1 -1
  127. package/src/pro/messaging/messagehubService.js +1 -1
  128. package/src/pro/messaging/rabbitmqService.js +1 -1
  129. package/src/pro/scheduler/job-manager.js +1 -1
  130. package/src/pro/scheduler/job-routes.js +1 -1
  131. package/src/pro/scheduler/job-validator.js +1 -1
  132. package/src/pro/storage/base-storage-provider.js +1 -1
  133. package/src/pro/storage/file-metadata-helper.js +1 -1
  134. package/src/pro/storage/index.js +1 -1
  135. package/src/pro/storage/local-storage-provider.js +1 -1
  136. package/src/pro/storage/s3-storage-provider.js +1 -1
  137. package/src/pro/storage/upload-cleanup-job.js +1 -1
  138. package/src/pro/storage/upload-cleanup-scheduler.js +1 -1
  139. package/src/pro/storage/upload-pending-tracker.js +1 -1
  140. package/src/pro/websocket/broadcast-helper.js +1 -1
  141. package/src/pro/websocket/index.js +1 -1
  142. package/src/pro/websocket/livesync-server.js +1 -1
  143. package/src/pro/websocket/ws-broadcaster.js +1 -1
  144. package/src/services/export-service.js +1 -1
  145. package/src/services/import-service.js +1 -1
  146. package/src/services/kafkaConsumerService.js +1 -1
  147. package/src/services/kafkaService.js +1 -1
  148. package/src/services/messagehubService.js +1 -1
  149. package/src/services/rabbitmqService.js +1 -1
  150. package/src/utils/cache-invalidation-registry.js +1 -1
  151. package/src/utils/cache-manager.js +1 -1
  152. package/src/utils/component-engine.js +1 -1
  153. package/src/utils/config-extractor.js +1 -1
  154. package/src/utils/consumerLogger.js +1 -1
  155. package/src/utils/context-builder.js +1 -1
  156. package/src/utils/dashboard-helpers.js +1 -1
  157. package/src/utils/dateHelper.js +1 -1
  158. package/src/utils/datetime-formatter.js +1 -1
  159. package/src/utils/datetime-parser.js +1 -1
  160. package/src/utils/db-bootstrap.js +1 -1
  161. package/src/utils/db-mysql.js +1 -1
  162. package/src/utils/db-oracle.js +1 -1
  163. package/src/utils/db-sqlite.js +1 -1
  164. package/src/utils/db.js +1 -1
  165. package/src/utils/demo-generator.js +1 -1
  166. package/src/utils/excel-generator.js +1 -1
  167. package/src/utils/excel-parser.js +1 -1
  168. package/src/utils/file-watcher.js +1 -1
  169. package/src/utils/id-generator.js +1 -1
  170. package/src/utils/idempotency-manager.js +1 -1
  171. package/src/utils/import-validator.js +1 -1
  172. package/src/utils/license-client.js +1 -1
  173. package/src/utils/lock-manager.js +1 -1
  174. package/src/utils/logger.js +1 -1
  175. package/src/utils/lookup-resolver.js +1 -1
  176. package/src/utils/payload-loader.js +1 -1
  177. package/src/utils/processor-response.js +1 -1
  178. package/src/utils/rabbitmq.js +1 -1
  179. package/src/utils/redis-client.js +1 -1
  180. package/src/utils/redis-helper.js +1 -1
  181. package/src/utils/request-scope.js +1 -1
  182. package/src/utils/security-checks.js +1 -1
  183. package/src/utils/service-resolver.js +1 -1
  184. package/src/utils/shutdown-coordinator.js +1 -1
  185. package/src/utils/soft-delete-dashboard-guard.js +1 -1
  186. package/src/utils/sql-table-extractor.js +1 -1
  187. package/src/utils/trusted-keys.js +1 -1
  188. package/src/utils/upload-handler.js +1 -1
  189. package/src/utils/upsert-builder.js +1 -1
  190. package/src/utils/workflow-hook-executor.js +1 -1
@@ -0,0 +1,106 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * Processor: reset-password (prototype, tanpa verifikasi email)
5
+ * Reset password berdasarkan email saja, tanpa token/link verifikasi.
6
+ * Method: POST body: { email, new_password, confirm_password }
7
+ *
8
+ * Catatan keamanan: tanpa verifikasi kepemilikan email, siapa pun yang tahu
9
+ * email terdaftar dapat mengganti password. Hanya layak untuk prototype lokal.
10
+ */
11
+
12
+ const bcrypt = require('bcrypt');
13
+
14
+ const BCRYPT_ROUNDS = 12;
15
+ const MIN_PASSWORD = 6;
16
+
17
+ module.exports = {
18
+ async process(input, services, req) {
19
+ const { db, logger } = services;
20
+
21
+ try {
22
+ const { email, new_password, confirm_password } = input;
23
+
24
+ if (!email || !new_password || !confirm_password) {
25
+ return {
26
+ success: false,
27
+ statusCode: 400,
28
+ message: 'Email, password baru, dan konfirmasi wajib diisi.',
29
+ timestamp: new Date().toISOString()
30
+ };
31
+ }
32
+ if (new_password.length < MIN_PASSWORD) {
33
+ return {
34
+ success: false,
35
+ statusCode: 400,
36
+ message: `Password minimal ${MIN_PASSWORD} karakter.`,
37
+ timestamp: new Date().toISOString()
38
+ };
39
+ }
40
+ if (new_password !== confirm_password) {
41
+ return {
42
+ success: false,
43
+ statusCode: 400,
44
+ message: 'Konfirmasi password tidak cocok.',
45
+ timestamp: new Date().toISOString()
46
+ };
47
+ }
48
+
49
+ const rows = await db.executeQuery(
50
+ 'SELECT user_id FROM public.{{AUTH_USER_TABLE}} WHERE email = $1',
51
+ [email]
52
+ );
53
+
54
+ if (!rows || rows.length === 0) {
55
+ return {
56
+ success: false,
57
+ statusCode: 404,
58
+ message: 'Email tidak terdaftar.',
59
+ timestamp: new Date().toISOString()
60
+ };
61
+ }
62
+ if (rows.length > 1) {
63
+ // Email tidak unik di schema; tolak daripada diam-diam mereset banyak akun.
64
+ return {
65
+ success: false,
66
+ statusCode: 409,
67
+ message: 'Email terdaftar pada lebih dari satu akun. Hubungi administrator.',
68
+ timestamp: new Date().toISOString()
69
+ };
70
+ }
71
+
72
+ const userId = rows[0].user_id;
73
+ const passwordHash = await bcrypt.hash(new_password, BCRYPT_ROUNDS);
74
+
75
+ // Reset password sekaligus buka kunci akun dan reset hitungan gagal login.
76
+ await db.executeQuery(
77
+ `UPDATE public.{{AUTH_USER_TABLE}}
78
+ SET password_hash = $1, password_changed_at = NOW(), updated_at = NOW(),
79
+ failed_login_count = 0, is_locked = FALSE
80
+ WHERE user_id = $2`,
81
+ [passwordHash, userId]
82
+ );
83
+
84
+ // Revoke semua refresh token lama: paksa login ulang di seluruh sesi.
85
+ await db.executeQuery(
86
+ 'UPDATE public.{{AUTH_REFRESH_TOKEN_TABLE}} SET is_revoked = TRUE WHERE user_id = $1',
87
+ [userId]
88
+ );
89
+
90
+ return {
91
+ success: true,
92
+ statusCode: 200,
93
+ message: 'Password berhasil diperbarui. Silakan login dengan password baru.',
94
+ timestamp: new Date().toISOString()
95
+ };
96
+ } catch (error) {
97
+ logger.error({ error: error.message }, '[auth-reset-password] Unexpected error');
98
+ return {
99
+ success: false,
100
+ statusCode: 500,
101
+ message: 'Terjadi kesalahan internal pada server.',
102
+ timestamp: new Date().toISOString()
103
+ };
104
+ }
105
+ }
106
+ };
@@ -0,0 +1,79 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * Auth Middleware (prototype, tanpa RBAC)
5
+ *
6
+ * Hanya menangani JWT: sign access token, verify Bearer token, dan guard
7
+ * `requireAuth` untuk melindungi endpoint lain. Tidak ada role/permission.
8
+ */
9
+
10
+ const jwt = require('jsonwebtoken');
11
+
12
+ function getJwtSecret() {
13
+ return process.env.JWT_SECRET || 'CHANGE-THIS-TO-RANDOM-SECRET-IN-PRODUCTION';
14
+ }
15
+
16
+ function getAccessTokenExpiryMin() {
17
+ return parseInt(process.env.ACCESS_TOKEN_EXPIRY_MIN || '60', 10);
18
+ }
19
+
20
+ /**
21
+ * Decode + verifikasi JWT dari header `Authorization: Bearer <token>`.
22
+ * @throws {Error} err.statusCode = 401 bila tidak valid.
23
+ */
24
+ function verifyToken(req) {
25
+ const authHeader = req.headers['authorization'];
26
+ if (!authHeader || !authHeader.startsWith('Bearer ')) {
27
+ const err = new Error('Token tidak ditemukan');
28
+ err.statusCode = 401;
29
+ throw err;
30
+ }
31
+
32
+ const token = authHeader.substring(7);
33
+ try {
34
+ return jwt.verify(token, getJwtSecret());
35
+ } catch (jwtErr) {
36
+ const err = new Error('Token tidak valid atau sudah kedaluwarsa');
37
+ err.statusCode = 401;
38
+ throw err;
39
+ }
40
+ }
41
+
42
+ /**
43
+ * Express middleware: verifikasi JWT lalu set req.user.
44
+ */
45
+ function requireAuth(req, res, next) {
46
+ try {
47
+ req.user = verifyToken(req);
48
+ next();
49
+ } catch (err) {
50
+ return res.status(err.statusCode || 401).json({
51
+ success: false,
52
+ message: err.message
53
+ });
54
+ }
55
+ }
56
+
57
+ /**
58
+ * Generate JWT access token. Payload ramping: hanya identitas user.
59
+ */
60
+ function generateAccessToken(user) {
61
+ const payload = {
62
+ sub: user.user_id,
63
+ username: user.username,
64
+ email: user.email
65
+ };
66
+
67
+ return jwt.sign(payload, getJwtSecret(), {
68
+ algorithm: process.env.JWT_ALGORITHM || 'HS256',
69
+ expiresIn: `${getAccessTokenExpiryMin()}m`
70
+ });
71
+ }
72
+
73
+ module.exports = {
74
+ verifyToken,
75
+ requireAuth,
76
+ generateAccessToken,
77
+ getJwtSecret,
78
+ getAccessTokenExpiryMin
79
+ };
@@ -0,0 +1,104 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * Auth endpoints — prototype login (tanpa RBAC).
5
+ * Di-mount oleh module loader di prefix /api/{{PROJECT_NAME}}/rfx_auth (dari nama file).
6
+ *
7
+ * POST /api/{{PROJECT_NAME}}/rfx_auth/register — buat user baru
8
+ * POST /api/{{PROJECT_NAME}}/rfx_auth/login — login, terbitkan access + refresh token
9
+ * POST /api/{{PROJECT_NAME}}/rfx_auth/refresh — tukar refresh token (rotation)
10
+ * POST /api/{{PROJECT_NAME}}/rfx_auth/logout — revoke refresh token
11
+ * POST /api/{{PROJECT_NAME}}/rfx_auth/reset-password — reset password via email (tanpa token)
12
+ * GET /api/{{PROJECT_NAME}}/rfx_auth/me — profil user dari Bearer token
13
+ *
14
+ * Router tipis: hanya validasi input dasar lalu delegasikan ke processor.
15
+ */
16
+
17
+ const express = require('express');
18
+ const router = express.Router();
19
+ const { resolveServices } = require('@restforgejs/platform/src/utils/service-resolver');
20
+ const services = resolveServices();
21
+
22
+ const register = require('./processor/auth/register');
23
+ const login = require('./processor/auth/login');
24
+ const refresh = require('./processor/auth/refresh');
25
+ const logout = require('./processor/auth/logout');
26
+ const me = require('./processor/auth/me');
27
+ const resetPassword = require('./processor/auth/reset-password');
28
+
29
+ const SENSITIVE = /password|token/i;
30
+
31
+ function maskSensitive(obj) {
32
+ if (!obj || typeof obj !== 'object') return obj;
33
+ const out = {};
34
+ for (const k of Object.keys(obj)) {
35
+ out[k] = SENSITIVE.test(k) ? '***MASKED***' : obj[k];
36
+ }
37
+ return out;
38
+ }
39
+
40
+ /**
41
+ * Bungkus pemanggilan processor dengan validasi field wajib (string non-kosong).
42
+ */
43
+ function handle(processor, requiredFields, actionLabel) {
44
+ return async (req, res) => {
45
+ try {
46
+ const input = { ...req.body, ...req.params, ...req.query };
47
+
48
+ const errors = [];
49
+ for (const field of requiredFields) {
50
+ const v = input[field];
51
+ if (v === undefined || v === null || v === '') {
52
+ errors.push({ field, message: `Field "${field}" wajib diisi` });
53
+ } else if (typeof v !== 'string') {
54
+ errors.push({ field, message: `Field "${field}" harus berupa string` });
55
+ }
56
+ }
57
+ if (errors.length > 0) {
58
+ return res.status(400).json({
59
+ success: false,
60
+ statusCode: 400,
61
+ message: 'Validation failed',
62
+ errors,
63
+ timestamp: new Date().toISOString()
64
+ });
65
+ }
66
+
67
+ if (process.env.NODE_ENV !== 'production') {
68
+ console.log(`[${actionLabel}] Input:`, JSON.stringify(maskSensitive(input)));
69
+ }
70
+
71
+ const result = await processor.process(input, services, req);
72
+ return res.status(result.statusCode || 200).json(result);
73
+ } catch (error) {
74
+ console.error(`[${actionLabel}] Error:`, error.message);
75
+ return res.status(500).json({
76
+ success: false,
77
+ message: `Gagal memproses ${actionLabel}`,
78
+ timestamp: new Date().toISOString()
79
+ });
80
+ }
81
+ };
82
+ }
83
+
84
+ router.post('/register', handle(register, ['username', 'password'], 'register'));
85
+ router.post('/login', handle(login, ['username', 'password'], 'login'));
86
+ router.post('/refresh', handle(refresh, ['refresh_token'], 'refresh'));
87
+ router.post('/logout', handle(logout, [], 'logout'));
88
+ router.post('/reset-password', handle(resetPassword, ['email', 'new_password', 'confirm_password'], 'reset-password'));
89
+
90
+ router.get('/me', async (req, res) => {
91
+ try {
92
+ const result = await me.process({}, services, req);
93
+ return res.status(result.statusCode || 200).json(result);
94
+ } catch (error) {
95
+ console.error('[me] Error:', error.message);
96
+ return res.status(500).json({
97
+ success: false,
98
+ message: 'Gagal memproses me',
99
+ timestamp: new Date().toISOString()
100
+ });
101
+ }
102
+ });
103
+
104
+ module.exports = router;
@@ -13,12 +13,21 @@
13
13
 
14
14
  const VALID_REL_FIELDS = ['type', 'target', 'localKey', 'references', 'onDelete', 'onUpdate'];
15
15
 
16
+ /**
17
+ * @param {object} ir
18
+ * @param {object} [options]
19
+ * @param {string} [options.timestamp] - ISO timestamp override for `Generated at:`.
20
+ * @param {string} [options.generatedBy] - Provenance label for the `Generated by:` header
21
+ * line. Defaults to `npx restforge schema introspect` when not set, preserving output
22
+ * for all existing callers.
23
+ */
16
24
  function serialize(ir, options) {
17
25
  if (!ir || typeof ir !== 'object' || typeof ir.tableName !== 'string') {
18
26
  throw new Error('serialize: ir must be an object with tableName');
19
27
  }
20
28
  const opts = options || {};
21
29
  const timestamp = opts.timestamp || new Date().toISOString();
30
+ const generatedBy = opts.generatedBy || 'npx restforge schema introspect';
22
31
 
23
32
  const sections = [];
24
33
 
@@ -66,7 +75,7 @@ function serialize(ir, options) {
66
75
  '/**',
67
76
  ` * Schema definition for \`${ir.tableName}\``,
68
77
  ' *',
69
- ' * Generated by: npx restforge schema introspect',
78
+ ` * Generated by: ${generatedBy}`,
70
79
  ` * Generated at: ${timestamp}`,
71
80
  ' */',
72
81
  `module.exports = ({ defineModel }) => defineModel('${ir.tableName}', {`
@@ -1 +1 @@
1
- const a0_0x2659ec=a0_0x3123;function a0_0x3123(_0x3453ca,_0x3ebf10){_0x3453ca=_0x3453ca-0x1ca;const _0xd56b41=a0_0xd56b();let _0x3123bd=_0xd56b41[_0x3453ca];if(a0_0x3123['LjhHqG']===undefined){var _0x160abd=function(_0x228b94){const _0x3b01bd='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x999290='',_0x554244='';for(let _0x3693f0=0x0,_0x3bf121,_0x4c17c9,_0x1bddf2=0x0;_0x4c17c9=_0x228b94['charAt'](_0x1bddf2++);~_0x4c17c9&&(_0x3bf121=_0x3693f0%0x4?_0x3bf121*0x40+_0x4c17c9:_0x4c17c9,_0x3693f0++%0x4)?_0x999290+=String['fromCharCode'](0xff&_0x3bf121>>(-0x2*_0x3693f0&0x6)):0x0){_0x4c17c9=_0x3b01bd['indexOf'](_0x4c17c9);}for(let _0x182fe0=0x0,_0x2910e6=_0x999290['length'];_0x182fe0<_0x2910e6;_0x182fe0++){_0x554244+='%'+('00'+_0x999290['charCodeAt'](_0x182fe0)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x554244);};a0_0x3123['JvXOmB']=_0x160abd,a0_0x3123['VFxKvD']={},a0_0x3123['LjhHqG']=!![];}const _0x4e576c=_0xd56b41[0x0],_0x1e9a8e=_0x3453ca+_0x4e576c,_0x1bdeca=a0_0x3123['VFxKvD'][_0x1e9a8e];return!_0x1bdeca?(_0x3123bd=a0_0x3123['JvXOmB'](_0x3123bd),a0_0x3123['VFxKvD'][_0x1e9a8e]=_0x3123bd):_0x3123bd=_0x1bdeca,_0x3123bd;}(function(_0x49959a,_0x13afe1){const _0x4b3d5b=a0_0x3123,_0x579a11=_0x49959a();while(!![]){try{const _0xaf9556=-parseInt(_0x4b3d5b(0x1d3))/0x1*(-parseInt(_0x4b3d5b(0x211))/0x2)+parseInt(_0x4b3d5b(0x1ef))/0x3+parseInt(_0x4b3d5b(0x21e))/0x4+-parseInt(_0x4b3d5b(0x226))/0x5+-parseInt(_0x4b3d5b(0x22d))/0x6+-parseInt(_0x4b3d5b(0x20f))/0x7*(parseInt(_0x4b3d5b(0x230))/0x8)+parseInt(_0x4b3d5b(0x20d))/0x9*(parseInt(_0x4b3d5b(0x22b))/0xa);if(_0xaf9556===_0x13afe1)break;else _0x579a11['push'](_0x579a11['shift']());}catch(_0x5773d1){_0x579a11['push'](_0x579a11['shift']());}}}(a0_0xd56b,0xdcd92));const FORBIDDEN_FRONTEND_FIELDS=['widgetType',a0_0x2659ec(0x1f6),a0_0x2659ec(0x213),'subtitle','color'],ALLOWED_PARAM_TYPES=[a0_0x2659ec(0x210),'number','boolean',a0_0x2659ec(0x1f5)],FRONTEND_CONCERN_REASONS={'widgetType':a0_0x2659ec(0x206),'layout':'Layout\x20is\x20a\x20frontend\x20rendering\x20concern.','title':'UI\x20label\x20is\x20a\x20frontend\x20rendering\x20concern.','subtitle':a0_0x2659ec(0x200),'color':a0_0x2659ec(0x222)},PAYLOAD_SHAPE={'discriminator':{'field':a0_0x2659ec(0x1fd),'presentMeans':'dashboard\x20payload','absentMeans':a0_0x2659ec(0x1e4),'conflictsWith':'tableName','conflictRationale':'A\x20payload\x20with\x20both\x20\x27widgets\x27\x20and\x20\x27tableName\x27\x20is\x20rejected\x20by\x20DashboardValidator.\x20Pick\x20one\x20shape.'},'topLevelAllowed':[{'name':a0_0x2659ec(0x1fd),'type':a0_0x2659ec(0x1f3),'required':!![],'minItems':0x1,'description':a0_0x2659ec(0x21a)},{'name':a0_0x2659ec(0x1f2),'type':a0_0x2659ec(0x1ec),'required':![],'description':a0_0x2659ec(0x212)},{'name':'cache','type':'object','required':![],'description':a0_0x2659ec(0x1e9)}],'topLevelForbidden':[{'name':a0_0x2659ec(0x1ff),'category':a0_0x2659ec(0x20e),'reason':a0_0x2659ec(0x1df)},...FORBIDDEN_FRONTEND_FIELDS[a0_0x2659ec(0x1ee)](_0x999290=>({'name':_0x999290,'category':a0_0x2659ec(0x1d1),'reason':FRONTEND_CONCERN_REASONS[_0x999290]}))]},WIDGET_SPEC={'requiredFields':[{'name':'id','type':'string','constraint':a0_0x2659ec(0x1f8),'description':a0_0x2659ec(0x221)}],'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_0x2659ec(0x210),'format':'file:relative/path/to/query.sql','description':'Single\x20SQL\x20query\x20for\x20the\x20widget.','responseShape':a0_0x2659ec(0x1da)},{'name':'queries','type':'object','format':a0_0x2659ec(0x1ce),'minKeys':0x1,'description':a0_0x2659ec(0x1dc),'responseShape':'Per-key\x20based\x20on\x20scalarCollapseRules\x20below.'}]},'forbiddenFields':FORBIDDEN_FRONTEND_FIELDS},PARAM_SPEC={'container':'top-level\x20\x27params\x27\x20object','keyConvention':a0_0x2659ec(0x21d),'perEntryFields':[{'name':'type','required':!![],'allowedValues':ALLOWED_PARAM_TYPES,'description':a0_0x2659ec(0x1d0)},{'name':a0_0x2659ec(0x215),'required':![],'type':a0_0x2659ec(0x216),'default':![],'description':'When\x20true,\x20the\x20request\x20body\x20MUST\x20include\x20this\x20param\x20(otherwise\x20400).'},{'name':a0_0x2659ec(0x1fa),'required':![],'type':a0_0x2659ec(0x234),'description':'Default\x20value\x20applied\x20when\x20the\x20request\x20omits\x20this\x20param.\x20Validator\x20does\x20NOT\x20strictly\x20type-check\x20default;\x20runtime\x20is\x20responsible\x20for\x20compatibility.'}]},SCALAR_COLLAPSE_RULES=[{'appliesTo':'widget.query\x20(singular)','rule':a0_0x2659ec(0x1d4),'exampleSqlShape':a0_0x2659ec(0x1d8),'exampleResponse':a0_0x2659ec(0x21f)},{'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':a0_0x2659ec(0x208)},{'appliesTo':a0_0x2659ec(0x205),'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':a0_0x2659ec(0x1d9)},{'appliesTo':'widget.queries.<key>\x20with\x20SQL\x20returning\x20N\x20rows','rule':a0_0x2659ec(0x231),'exampleSqlShape':a0_0x2659ec(0x1eb),'exampleResponse':'\x22items\x22:\x20[{\x20\x22label\x22:\x20\x22Shoes\x22,\x20\x22value\x22:\x20\x227660\x22\x20},\x20...]'}],COMMON_WIDGET_PATTERNS=[{'id':a0_0x2659ec(0x214),'name':a0_0x2659ec(0x1e8),'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_0x2659ec(0x1d5),'queries':{'value':a0_0x2659ec(0x235),'trend':'file:query/<path>/trend.sql','items':'file:query/<path>/breakdown.sql'}},'sqlShapesPerKey':[{'key':a0_0x2659ec(0x21c),'shape':a0_0x2659ec(0x1d7),'outputColumns':[a0_0x2659ec(0x21c)],'collapseRule':'scalar\x20primitive'},{'key':a0_0x2659ec(0x1d6),'shape':'1\x20row\x20×\x202\x20columns','outputColumns':['direction','pct'],'collapseRule':'object'},{'key':'items','shape':'N\x20rows\x20×\x202\x20columns','outputColumns':[a0_0x2659ec(0x20c),a0_0x2659ec(0x21c)],'collapseRule':'array\x20of\x20objects'}],'responseShape':{'value':a0_0x2659ec(0x22c),'trend':a0_0x2659ec(0x1db),'items':a0_0x2659ec(0x218)},'referenceWidgetId':a0_0x2659ec(0x228),'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_0x2659ec(0x22f),'name':a0_0x2659ec(0x209),'useCase':a0_0x2659ec(0x22e),'payloadShape':{'id':'<widget_id>','queries':{'value':'file:query/<path>/value.sql','trend':a0_0x2659ec(0x1f9),'points':'file:query/<path>/points.sql'}},'sqlShapesPerKey':[{'key':'value','shape':'1\x20row\x20×\x201\x20column','outputColumns':[a0_0x2659ec(0x21c)],'collapseRule':a0_0x2659ec(0x229)},{'key':'trend','shape':'1\x20row\x20×\x202\x20columns','outputColumns':[a0_0x2659ec(0x1ed),a0_0x2659ec(0x207)],'collapseRule':a0_0x2659ec(0x1ec)},{'key':a0_0x2659ec(0x201),'shape':'N\x20rows\x20×\x202\x20columns','outputColumns':[a0_0x2659ec(0x1f4),a0_0x2659ec(0x21c)],'collapseRule':'array\x20of\x20objects'}],'responseShape':{'value':'\x222420\x22','trend':a0_0x2659ec(0x1cd),'points':a0_0x2659ec(0x233)},'referenceWidgetId':a0_0x2659ec(0x217),'socNotes':a0_0x2659ec(0x1cc)},{'id':'metric_progress_to_goal','name':'Metric\x20+\x20Progress\x20to\x20Goal','useCase':a0_0x2659ec(0x204),'payloadShape':{'id':a0_0x2659ec(0x1d5),'queries':{'value':a0_0x2659ec(0x22a),'trend':'file:query/<path>/trend.sql','target':'file:query/<path>/target.sql'}},'sqlShapesPerKey':[{'key':a0_0x2659ec(0x21c),'shape':a0_0x2659ec(0x1d7),'outputColumns':['value\x20(or\x20current)'],'collapseRule':a0_0x2659ec(0x229)},{'key':'trend','shape':a0_0x2659ec(0x1fc),'outputColumns':['direction','pct'],'collapseRule':'object'},{'key':'target','shape':'1\x20row\x20×\x201\x20column','outputColumns':[a0_0x2659ec(0x202)],'collapseRule':a0_0x2659ec(0x229)}],'responseShape':{'value':'\x221836\x22','trend':'{\x20\x22direction\x22:\x20\x22down\x22,\x20\x22pct\x22:\x20\x222.2\x22\x20}','target':'\x222884\x22'},'referenceWidgetId':a0_0x2659ec(0x1e0),'socNotes':a0_0x2659ec(0x1d2)}],NAMING_CONVENTION={'dashboardName':{'constraint':a0_0x2659ec(0x224),'minLength':0x6,'maxLength':0x32,'regex':'^dash-[a-zA-Z0-9_-]+$','examples':[a0_0x2659ec(0x1ca),'dash-inbound',a0_0x2659ec(0x1e2)],'rationale':a0_0x2659ec(0x1de)}},URL_PATTERN={'method':a0_0x2659ec(0x1cb),'path':'/api/{project}/{name}/dashboard','exampleFull':'POST\x20/api/mini-inventory/dash-inbound/dashboard','requestBodyShape':{'params':a0_0x2659ec(0x20b),'widgets':a0_0x2659ec(0x203)},'responseShape':{'envelope':a0_0x2659ec(0x21b),'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_0x2659ec(0x1ea),'pathRelativeTo':'payload\x20JSON\x20file\x20location','fileExtensionPolicy':a0_0x2659ec(0x1e6),'resolvedAt':a0_0x2659ec(0x1e7),'embedStrategy':a0_0x2659ec(0x1f7),'implication':a0_0x2659ec(0x1e5)},PLACEHOLDER_CONVENTION={'format':a0_0x2659ec(0x223),'regex':a0_0x2659ec(0x1e1),'regexNotes':'Negative\x20lookbehind\x20prevents\x20matching\x20\x27::\x27\x20(Postgres\x20cast\x20syntax)\x20as\x20a\x20placeholder.','scanScope':a0_0x2659ec(0x20a),'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_0x2659ec(0x1fb),'exampleParamDeclaration':a0_0x2659ec(0x1f1)},CACHE_SPEC={'container':a0_0x2659ec(0x1f0),'optional':!![],'rationale':'Dashboard\x20endpoint\x20may\x20opt-in\x20to\x20Redis-based\x20cache.\x20Pattern\x20follows\x20processor\x20cache\x20(see\x20feat-cache.md).\x20Cache\x20scope\x20is\x20the\x20full\x20response\x20envelope;\x20one\x20cache\x20entry\x20per\x20(params\x20+\x20widgets[]\x20subset)\x20combination.','fields':[{'name':'enabled','type':'boolean','required':!![],'description':a0_0x2659ec(0x227)},{'name':'ttl','type':'number','required':![],'constraint':a0_0x2659ec(0x220),'default':'inherits\x20CACHE_TTL\x20env','description':'Time-to-live\x20in\x20seconds.\x200\x20effectively\x20disables\x20cache\x20for\x20this\x20entry.'},{'name':a0_0x2659ec(0x1cf),'type':'array<string>','required':![],'default':'[]','description':a0_0x2659ec(0x232)}],'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_0x2659ec(0x1fe),'Table\x20declared\x20in\x20invalidates,\x20but\x20not\x20detected\x20in\x20any\x20widget\x20SQL\x20(typo\x20or\x20dead\x20entry)'],'warningOn':[a0_0x2659ec(0x225)]}},DOCUMENTATION_URL=a0_0x2659ec(0x1e3),DASHBOARD_CATALOG={'schemaVersion':a0_0x2659ec(0x219),'source':'dashboard-catalog','summary':{'totalAllowedTopLevelFields':PAYLOAD_SHAPE['topLevelAllowed']['length'],'totalForbiddenFrontendFields':FORBIDDEN_FRONTEND_FIELDS['length'],'totalParamTypes':ALLOWED_PARAM_TYPES[a0_0x2659ec(0x1dd)],'totalScalarCollapseRules':SCALAR_COLLAPSE_RULES[a0_0x2659ec(0x1dd)],'totalCommonWidgetPatterns':COMMON_WIDGET_PATTERNS[a0_0x2659ec(0x1dd)]},'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_0xd56b(){const _0x40fc8a=['tvvtvcbZDgfYDcb3AxrOicDKyxnOlsCGChjLzML4','vgfIBguGzgv0zwn0zwqGAw4Gu1fmlcbIDxqGBM90ihjLz2LZDgvYzwqGyxmGq1jvrcbLBMrWB2LUDcbPBIbTzxrHzgf0ysbWCM9Qzwn0icHSAwTLBhKGysb2Awv3lcbdveuGywXPyxmSig9YignYB3nZlxbYB2PLy3qGDgfIBguG4OcuignHC2nHzguGD2LSBcbUB3qGzMLYzsK','mZC2mdy2mgjyEMjSvW','vg9Nz2XLignHy2HLigzLyxr1CMuGzM9YihrOAxmGzgfZAgjVyxjKlG','zxHWzwn0zwrFzwfYBMLUz3m','C2nHBgfYihbYAw1PDgL2zq','zMLSztPXDwvYEs88Cgf0Ad4Vy3vYCMvUDc5ZCwW','otK1ntKZmgnxwerzqW','iJy5nZaWiG','ntm1mdyWogP3rNH6Aq','sgvHzgXPBMuGBwv0CMLJihDPDgGGDhjLBMqGy2HPCcbHBMqGC3bHCMTSAw5Lig1PBMKTy2HHCNqGzM9YihnOB3j0ihDPBMrVD3mGkdCGzgf5CYWGmtiGBw9UDgHZlcbLDgmUks4Gu3vPDgfIBguGzM9YihDPzgDLDhmGBgLRzsaNqxzLCMfNzsbeywLSEsbtywXLCYCU','Bwv0CMLJx3nWyxjRBgLUzq','ohrbs1nzzW','uMv0DxjUigfZigfYCMf5ig9Mig9IAMvJDhmGkg5VignVBgXHChnLks4','tgLZDcbVzIbduLveihrHyMXLig5HBwvZihrOyxqSihDOzw4GD3jPDhrLBIWGD2LSBcb0CMLNz2vYigLUDMfSAwrHDgLVBIbVzIb0AgLZigrHC2HIB2fYzcbJywnOzs4','w3SGiNbLCMLVzci6iciYmdi2lta0lti0iIWGiNzHBhvLiJOGiJe4ntaIih0Sic4UlIbD','yw55icHTDxn0igjLignVBxbHDgLIBguGD2L0AcbKzwnSyxjLzcaNDhLWzsCP','zMLSztPXDwvYEs88Cgf0Ad4VDMfSDwuUC3fS','zgfZAc1ZywXLCW','ue9tva','u3bHCMTSAw5LigXPyNjHCMLLCYaOqxbLEenOyxj0CYWGq2HHCNrPC3qSigv0yY4Pihr5CgLJywXSEsbUzwvKigeGCgXHAw4GBNvTyMvYigfYCMf5lIbgCM9UDgvUzcbTyxbZihbVAw50CY5TyxaOCca9pIbWlNzHBhvLks4GvgHLicDWzxjPB2qNigzPzwXKihn0yxLZigzVCIb0B29SDgLWigfUzcbNyxaTCMvZAwXPzw5JzsbHz2fPBNn0ig1PC3nPBMCGzgf5CY4GvxnLigDLBMvYyxrLx3nLCMLLCYbPBIbtuuWGDg8Gzw5ZDxjLignVBNnPC3rLBNqGCM93ignVDw50igv2zw4GzM9YigrHExmGD2L0AcbUBYb0CMfUC2fJDgLVBNmU','EYaIzgLYzwn0Aw9UiJOGiNvWiIWGiNbJDci6iciYlJyIih0','A2v54OAszMLSztPYzwXHDgL2zs9WyxrOl3rVl3f1zxj5lNnXBa','Aw52ywXPzgf0zxm','ugfYyw0Gzgf0ysb0ExbLlIbwywXPzgf0zxmGCMvXDwvZDcbIB2r5igfUzcbZAgfWzxmGCNvUDgLTzsbWyxjHBwv0zxiGyMLUzgLUzY4','zNjVBNrLBMqTy29Uy2vYBG','rNjVBNrLBMqGy29TChv0zxmGDg9Fz29HBca9ihrHCMDLDcaTihzHBhvLigfUzcbWy3qGpsbYB3vUzcH2ywX1zsaVihrHCMDLDcaQideWmcKGzM9YihrOzsbWCM9NCMvZCYbIyxiUifzPC3vHBcb3Awr0AcbPCYbWCMvZzw50yxrPB25HBcbHBMqGBxvZDcbot1qGBgL2zsbPBIb0AguGyMfJA2vUzcbWyxLSB2fKlIbjzIbWCM9NCMvZCYbPBNzVBhzLCYbJB21WBgv4igj1C2LUzxnZihj1BgvZicHLlMCUigv4y2X1zguGD2vLA2vUzhmSihbYB3jHDgvKihDVCMTKyxLZksWGDxnLigeGC2LUz2XLig11BhrPlwnVBhvTBIbXDwvYEsbZBYaNCgn0jYbPCYbHihn0ywjSzsbIDxnPBMvZCYbMywn0ihjHDgHLCIb0AgfUihzPC3vHBcb3Awr0Ac4','mtvVy1Drq3u','qwX3yxLZihDYyxaGyxmGEYbPDgvTCZOGwY4UlL0GFsbYzwDHCMrSzxnZig9MifnrtcbYzxn1BhqGC2HHCguU','phDPzgDLDf9Pzd4','DhjLBMq','msbYB3CGW5CGmsbJB2X1Bw4','yw55icGXihjVDYddLYaXignVBcWGtIbYB3DZimoxie0Gy29SCYWGzxrJlIK','iNrYzw5KiJOGEYaIzgLYzwn0Aw9UiJOGiNvWiIWGiNbJDci6iciYlJiIih0','qwX3yxLZihSGAxrLBxm6ifSUlI5Dih0GCMvNyxjKBgvZCYbVzIbtuuWGCMvZDwX0ihnOyxbLlG','EYaIzgLYzwn0Aw9UiJOGiNvWiIWGiNbJDci6iciYlJiIih0','txvSDgKTu1fmihDPzgDLDc4GrwfJAcbRzxKGyMvJB21LCYbHigTLEsbPBIb0AguGCMvZCg9UC2uGB2jQzwn0lG','BgvUz3rO','vgHLihbYzwzPEcbIzwnVBwvZihbHCNqGB2yGDgHLifvstcbZzwDTzw50lIbuAguGCMvZzxj2zwqGC2nOzw1LigTLzxbZigrHC2HIB2fYzcbLBMrWB2LUDhmGDMLZDwfSBhKGzgLZDgLUy3qGzNjVBsbduLveigvUzhbVAw50CYbPBIb0AguGvvjmihnWywnLigfUzcbHBgXVD3mGzNv0DxjLihjVDxrPBMCGzgLMzMvYzw50Awf0Aw9UlG','uMvZzxj2zwqGzM9YiensvuqGCgf5Bg9HzhmUieeGzgfZAgjVyxjKihbHEwXVywqGBxvZDcbKzwnSyxjLicD3AwrNzxrZjYbPBNn0zwfKlG','B3jKzxjZx3rOAxnFBw9UDgG','kd88itOPoIHBys16qs1Ax11Bys16qs1Amc05x10Qkq','zgfZAc1HDxrOB3iTC3rHDhm','Ahr0Chm6lY9Yzxn0zM9Yz2uUzgv2l2rVy3mVC2vYDMvYl3f1zxj5lwrHDgeVzgfZAgjVyxjK','tM90igeGzgfZAgjVyxjKihbHEwXVywqGkgXPA2vSEsbduLveihDPDgGGDgfIBgvoyw1LlcbVCIbPBNzHBgLKkq','vxbKyxrPBMCGyw4Gu1fmigzPBguGCMvXDwLYzxmGCMvNzw5LCMf0Aw5NihrOzsbKyxnOyM9HCMqGBw9KDwXLicGNy29KzwDLBL9JCMvHDgvFzgfZAgjVyxjKjYKGzM9YignOyw5NzxmGDg8GDgfRzsbLzMzLy3qU','zNjLztSGlNnXBcbYzwnVBw1LBMrLzcbMB3iGzwrPDg9YigHPz2HSAwDODa','z2vUzxjHDgLVBIb0Aw1LicHot1qGCNvUDgLTzsK','twv0CMLJicSGrg9UDxqGqNjLywTKB3DU','t3b0Aw9UywWGy2fJAguGy29UzMLNDxjHDgLVBI4Gu2vLignHy2HLu3bLyYbMB3iGzgv0ywLSCY4','zMLSztPYzwXHDgL2zs9WyxrOl3rVl3f1zxj5lNnXBa','tIbYB3DZimoxie0Gy29SCW','B2jQzwn0','zgLYzwn0Aw9U','BwfW','mtm5ndG3n0fWCvjbqW','Dg9WlwXLDMvSicDJywnOzsCGB2jQzwn0','EYaICgfYyw1ZiJOGEYaIEwvHCIi6ihSGiNr5CguIoIaIBNvTyMvYiIWGiNjLCxvPCMvKiJOGDhj1zsb9ih0GFq','CgfYyw1Z','yxjYyxK','CgvYAw9K','zgf0zq','Bgf5B3v0','u1fmigzPBguGy29UDgvUDcbPCYbLBwjLzgrLzcbHCYbkyxzHu2nYAxb0ihrLBxbSyxrLigXPDgvYywWGAw5ZAwrLihrOzsbNzw5LCMf0zwqGBw9KDwXLigzPBguUifj1BNrPBwuGCgvYzM9YBxmGEMvYBYbKAxnRieKVtYbWzxiGCMvXDwvZDcdIGjqGywXSifnrtcbPCYbPBIbTzw1VCNKGywz0zxiGBw9KDwXLigXVywqU','BM9UlwvTChr5lcb1BMLXDwuGywnYB3nZihDPzgDLDhmGAw4GDgHLihnHBwuGCgf5Bg9Hza','zMLSztPXDwvYEs88Cgf0Ad4VDhjLBMqUC3fS','zgvMyxvSDa','u0vmrunuicOGrLjptsbZDg9JA19PBMjVDw5KifDirvjfievyvfjbq1qOwuvbuIbguK9nigLUyM91BMrFzgf0zsKGpsa6EwvHCG','msbYB3CGW5CGmIbJB2X1Bw5Z','D2LKz2v0CW','vgfIBguGyxbWzwfYCYbPBIbtuuWGqu5eigLUig1LDgfKyxrHihbYB2PLy3qSigj1DcbTAxnZAw5NigzYB20GAw52ywXPzgf0zxmGkgnHy2HLihn0ywXLihjPC2SP','DgfIBgvoyw1L','vuKGBgfIzwWGAxmGysbMCM9UDgvUzcbYzw5KzxjPBMCGy29Uy2vYBI4','Cg9PBNrZ','DgfYz2v0','yxjYyxK8C3rYAw5NpIWGB3b0Aw9UywWG4Ocuihn1yNnLDcbVzIb3AwrNzxqGsurZihrVigv4zwn1DguUie9TAxqGDg8GzxHLy3v0zsbHBgWGzgvJBgfYzwqGD2LKz2v0CY4','sgvHzgXPBMuGBwv0CMLJihDPDgGGDhjLBMqGy2HPCcbHBMqGChjVz3jLC3mGyMfYigfNywLUC3qGysbWzxjPB2qGDgfYz2v0lIbtDwL0ywjSzsbMB3iGD2LKz2v0CYbSAwTLicDpCMrLCNmGvgHPCYbnB250AcCU','D2LKz2v0lNf1zxjPzxmUpgTLEt4GD2L0AcbtuuWGCMv0DxjUAw5NideGCM93imoxig11BhrPCgXLignVBhvTBNm','vMLZDwfSihzHCMLHBNqGkgrVBNv0lcbIyxiSihbPzsWGyxjLysKGAxmGysbMCM9UDgvUzcbYzw5KzxjPBMCGy29Uy2vYBIaOC2vWyxjHDgLVBIbVzIbJB25JzxjUCYKU','Cgn0','iNzHBhvLiJOGiJy5nZaWiG','twv0CMLJicSGu3bHCMTSAw5L','qwXSihDPzgDLDcbtuuWG4OcuigjVDgGGj3f1zxj5jYaOC2LUz3vSyxiPigfUzcbLDMvYEsaNCxvLCMLLCY48A2v5pICU','B2jQzwn0iokaLcb2ywX1zxmGzM9YigrLy2XHCMvKihbHCMfTCYaODMfSAwrHDgvKigfNywLUC3qGCgfYyw1ZignVBNrYywn0oYbTAxnZAw5NihjLCxvPCMvKiokgKIa0mdaSihr5CguGBwLZBwf0y2GG4OAsidqWmcK','BgfIzwW','owXvDuT1rq','C2HHCguTy29UzMXPy3q','mtC1otaZn0v2r3LTvW','C3rYAw5N','mZq4ntzgB3HWq1O','ugfYyw1LDgvYignVBNrYywn0igzVCIb0AguGzgfZAgjVyxjKlIbfywnOigTLEsbPCYbHihbHCMfTig5HBwu7ihzHBhvLCYbKzxnJCMLIzsb0ExbLl3jLCxvPCMvKl2rLzMf1BhqUifbSywnLAg9SzgvYCYbPBNnPzguGD2LKz2v0ifnrtcbTDxn0ihjLzMvYzw5JzsbKzwnSyxjLzcbWyxjHBsbUyw1LCY4','DgL0Bgu','Bwv0CMLJx2rVBNv0x2jYzwfRzg93BG','CMvXDwLYzwq','yM9VBgvHBG','yxzNx2rHAwX5x3nHBgvZ','w3SGiMXHyMvSiJOGiLnOB2vZiIWGiNzHBhvLiJOGiJC2nJaIih0SihSGiMXHyMvSiJOGiKDHBwLUzYiSicj2ywX1zsi6iciYodiWiIb9lcb7icjSywjLBci6icjpDgHLCNmIlcaIDMfSDwuIoIaInduYntCIih1D','ms4W','tgLZDcbVzIb3AwrNzxqGzgvMAw5PDgLVBNmUie9YzgvYigLZigLUzM9YBwf0Aw9UywWGB25SEsaOCMvZCg9UC2uGA2v5CYbHCMuGyNKGD2LKz2v0igLKlcbUB3qGyxjYyxKGAw5KzxGPlG','EYbZDwnJzxnZoIbIB29SzwfUlcbKyxrHoIb7idX3AwrNzxrjzd46idXWzxjxAwrNzxrszxnWB25Zzt4Sic4UlIb9ih0','DMfSDwu','ugfYyw0GBMfTzsbTDxn0ig1HDgnOihrOzsbWBgfJzwHVBgrLCIbYzwDLEcbGw2eTEKeTwL9Dw2eTEKeTwJaTov9DkMaGkgfSCgHHBNvTzxjPyYaRihvUzgvYC2nVCMuSig11C3qGC3rHCNqGD2L0AcbSzxr0zxiGB3iGDw5KzxjZy29YzsKU','ndmXmti1mLfqELHJsa','iNnOB3bWAw5Nx2nHDgvNB3jPzxmIoIb7icjPDgvTCYi6ifT7icjUyw1LiJOGiKXHBMrZiIb9lcb7icjUyw1LiJOGiKHVDxnLCYiGFv0GFq','pJ0GmcaOC2vJB25KCYK','v2LKz2v0igLKzw50AwzPzxi7ihvZzwqGyxmGDgHLihjLC3bVBNnLigTLEsbPBIb0AguGzgfZAgjVyxjKigvUDMvSB3bLlG','vMLZDwfSignVBg9YigLZigeGzNjVBNrLBMqGCMvUzgvYAw5NignVBMnLCM4U','oNbHCMfTtMfTzq'];a0_0xd56b=function(){return _0x40fc8a;};return a0_0xd56b();}module['exports']={'DASHBOARD_CATALOG':DASHBOARD_CATALOG};
1
+ function a0_0xaa0e(_0x2bf147,_0x48e76d){_0x2bf147=_0x2bf147-0xa5;const _0x3629e2=a0_0x3629();let _0xaa0e2c=_0x3629e2[_0x2bf147];if(a0_0xaa0e['BlmBEp']===undefined){var _0x5252b4=function(_0x2eed14){const _0x28e87d='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x1b4cb1='',_0x46a020='';for(let _0x3f8d97=0x0,_0x2a8961,_0xae407,_0x2cda1=0x0;_0xae407=_0x2eed14['charAt'](_0x2cda1++);~_0xae407&&(_0x2a8961=_0x3f8d97%0x4?_0x2a8961*0x40+_0xae407:_0xae407,_0x3f8d97++%0x4)?_0x1b4cb1+=String['fromCharCode'](0xff&_0x2a8961>>(-0x2*_0x3f8d97&0x6)):0x0){_0xae407=_0x28e87d['indexOf'](_0xae407);}for(let _0x3637f6=0x0,_0x15fdbe=_0x1b4cb1['length'];_0x3637f6<_0x15fdbe;_0x3637f6++){_0x46a020+='%'+('00'+_0x1b4cb1['charCodeAt'](_0x3637f6)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x46a020);};a0_0xaa0e['cFzqhl']=_0x5252b4,a0_0xaa0e['zSYtma']={},a0_0xaa0e['BlmBEp']=!![];}const _0x26e702=_0x3629e2[0x0],_0x48197d=_0x2bf147+_0x26e702,_0x1093c0=a0_0xaa0e['zSYtma'][_0x48197d];return!_0x1093c0?(_0xaa0e2c=a0_0xaa0e['cFzqhl'](_0xaa0e2c),a0_0xaa0e['zSYtma'][_0x48197d]=_0xaa0e2c):_0xaa0e2c=_0x1093c0,_0xaa0e2c;}const a0_0xcabb97=a0_0xaa0e;function a0_0x3629(){const _0x5bd668=['yxjYyxK8C3rYAw5NpIWGB3b0Aw9UywWG4Ocuihn1yNnLDcbVzIb3AwrNzxqGsurZihrVigv4zwn1DguUie9TAxqGDg8GzxHLy3v0zsbHBgWGzgvJBgfYzwqGD2LKz2v0CY4','zMLSztPXDwvYEs88Cgf0Ad4VyNjLywTKB3DUlNnXBa','EYaIzgLYzwn0Aw9UiJOGiNvWiIWGiNbJDci6iciYlJiIih0','zMLSztPXDwvYEs88Cgf0Ad4VDhjLBMqUC3fS','yxjYyxK','qsb3AwrNzxqGtvvtvcbKzwnSyxjLigv4ywn0BhKGB25Lig9MoIaNCxvLCNKNie9sicDXDwvYAwvZjY4GqM90AcbVCIbUzwL0AgvYigLZihjLAMvJDgvKlG','zMLSztPXDwvYEs88Cgf0Ad4VDMfSDwuUC3fS','zgf0zq','vgfIBguGzgv0zwn0zwqGAw4Gu1fmlcbIDxqGBM90ihjLz2LZDgvYzwqGyxmGq1jvrcbLBMrWB2LUDcbPBIbTzxrHzgf0ysbWCM9Qzwn0icHSAwTLBhKGysb2Awv3lcbdveuGywXPyxmSig9YignYB3nZlxbYB2PLy3qGDgfIBguG4OcuignHC2nHzguGD2LSBcbUB3qGzMLYzsK','yxjYyxKGB2yGB2jQzwn0CW','ugvYlwTLEsbIyxnLzcbVBIbZy2fSyxjdB2XSyxbZzvj1BgvZigjLBg93lG','phDPzgDLDf9Pzd4','qwXSihDPzgDLDcbtuuWG4OcuigjVDgGGj3f1zxj5jYaOC2LUz3vSyxiPigfUzcbLDMvYEsaNCxvLCMLLCY48A2v5pICU','zxHWzwn0zwrFzwfYBMLUz3m','nde5oteYzNHpsKDc','Bwv0CMLJx2rVBNv0x2jYzwfRzg93BG','vuKGBgfIzwWGAxmGysbMCM9UDgvUzcbYzw5KzxjPBMCGy29Uy2vYBI4','z2vUzxjHDgLVBIb0Aw1LicHot1qGCNvUDgLTzsK','vg9Nz2XLignHy2HLigzLyxr1CMuGzM9YihrOAxmGzgfZAgjVyxjKlG','qsbWyxLSB2fKihDPDgGGyM90AcaND2LKz2v0CYCGyw5KicD0ywjSzu5HBwuNigLZihjLAMvJDgvKigj5ierHC2HIB2fYzfzHBgLKyxrVCI4GugLJAYbVBMuGC2HHCguU','Aw5OzxjPDhmGq0fdsevFvfrmigvUDG','zNjVBNrLBMqTy29Uy2vYBG','rgv0zxjTAw5LzcbIEsbZy2fSyxjdB2XSyxbZzvj1BgvZlIbgywLSzwqGD2LKz2v0CYbWCM9KDwnLihSGzxjYB3i6icCUlI4Nih0GyMXVy2SGD2L0Acb0B3aTBgv2zwWGC3vJy2vZCYbZDgLSBcb0CNvLicHVBMuGD2LKz2v0igzHAwX1CMuGzg9LCYbot1qGzMfPBcb0AguGzgfZAgjVyxjKks4','BwfW','uMv0DxjUigfZigfYCMf5ig9Mig9IAMvJDhmGkg5VignVBgXHChnLks4','twv0CMLJicSGrg9UDxqGqNjLywTKB3DU','msbYB3CGW5CGmIbJB2X1Bw5Z','ugfYyw0Gzgf0ysb0ExbLlIbwywXPzgf0zxmGCMvXDwvZDcbIB2r5igfUzcbZAgfWzxmGCNvUDgLTzsbWyxjHBwv0zxiGyMLUzgLUzY4','Cg9PBNrZ','ugfYyw0GBMfTzsbTDxn0ig1HDgnOihrOzsbWBgfJzwHVBgrLCIbYzwDLEcbGw2eTEKeTwL9Dw2eTEKeTwJaTov9DkMaGkgfSCgHHBNvTzxjPyYaRihvUzgvYC2nVCMuSig11C3qGC3rHCNqGD2L0AcbSzxr0zxiGB3iGDw5KzxjZy29YzsKU','w3SGiMXHyMvSiJOGiLnOB2vZiIWGiNzHBhvLiJOGiJC2nJaIih0SihSGiMXHyMvSiJOGiKDHBwLUzYiSicj2ywX1zsi6iciYodiWiIb9lcb7icjSywjLBci6icjpDgHLCNmIlcaIDMfSDwuIoIaInduYntCIih1D','yw55icHTDxn0igjLignVBxbHDgLIBguGD2L0AcbKzwnSyxjLzcaNDhLWzsCP','EYaIzgLYzwn0Aw9UiJOGiMrVD24IlcaICgn0iJOGiJiUmIiGFq','twv0CMLJicSGuhjVz3jLC3mGDg8Gr29HBa','qwX3yxLZihDYyxaGyxmGEYbPDgvTCZOGwY4UlL0GFsbYzwDHCMrSzxnZig9MifnrtcbYzxn1BhqGC2HHCguU','qwX3yxLZihSGAxrLBxm6ifSUlI5Dih0GCMvNyxjKBgvZCYbVzIbtuuWGCMvZDwX0ihnOyxbLlG','yw55icGXihjVDYddLYaXignVBcWGtIbYB3DZimoxie0Gy29SCYWGzxrJlIK','zgfZAgjVyxjKihbHEwXVywq','ndi5D3vrDKXW','zgLYzwn0Aw9U','mJeXoduWD0rct3b3','iJi4odqI','zgfZAc1HDxrOB3iTC3rHDhm','zMLSztPXDwvYEs88Cgf0Ad4VCg9PBNrZlNnXBa','D2LKz2v0CW','DhjLBMq','ndG5mJGWmJb0wvrPzM0','vgfIBguGzgvJBgfYzwqGAw4GAw52ywXPzgf0zxmSigj1DcbUB3qGzgv0zwn0zwqGAw4Gyw55ihDPzgDLDcbtuuWGkhr5Cg8GB3iGzgvHzcbLBNrYEsK','CxvLCNK','vgHLihbYzwzPEcbIzwnVBwvZihbHCNqGB2yGDgHLifvstcbZzwDTzw50lIbuAguGCMvZzxj2zwqGC2nOzw1LigTLzxbZigrHC2HIB2fYzcbLBMrWB2LUDhmGDMLZDwfSBhKGzgLZDgLUy3qGzNjVBsbduLveigvUzhbVAw50CYbPBIb0AguGvvjmihnWywnLigfUzcbHBgXVD3mGzNv0DxjLihjVDxrPBMCGzgLMzMvYzw50Awf0Aw9UlG','vxbKyxrPBMCGyw4Gu1fmigzPBguGCMvXDwLYzxmGCMvNzw5LCMf0Aw5NihrOzsbKyxnOyM9HCMqGBw9KDwXLicGNy29KzwDLBL9JCMvHDgvFzgfZAgjVyxjKjYKGzM9YignOyw5NzxmGDg8GDgfRzsbLzMzLy3qU','ndK2otiZueXoy2j0','mJuZndbgzev3qLK','pJ0GmcaOC2vJB25KCYK','msbYB3CGW5CGmsbJB2X1Bw4','mZzqvuP4yuy','DgfIBgvoyw1L','EYaICgfYyw1ZiJOGEYaIEwvHCIi6ihSGiNr5CguIoIaIBNvTyMvYiIWGiNjLCxvPCMvKiJOGDhj1zsb9ih0GFq','mZzeAffNuKm','msbYB3CGW5CGmsbJB2WSig91Dhb1DcbJB2X1Bw4Gj3zHBhvLjW','C2nHBgfYihbYAw1PDgL2zq','tvvtvcbZDgfYDcb3AxrOicDKyxnOlsCGChjLzML4','D2LKz2v0vhLWzq','yM9VBgvHBG','nZi4nJHjywXUsfu','BgvUz3rO','B2jQzwn0','mJuZmLjVENDyCW','tgf5B3v0igLZigeGzNjVBNrLBMqGCMvUzgvYAw5NignVBMnLCM4U','tM90igeGzgfZAgjVyxjKihbHEwXVywqGkgXPA2vSEsbduLveihDPDgGGDgfIBgvoyw1LlcbVCIbPBNzHBgLKkq','Bgf5B3v0','D2LKz2v0lNf1zxjPzxmUpgTLEt4GD2L0AcbtuuWGCMv0DxjUAw5Nie4GCM93CW','tMvNyxrPDMuGBg9VA2jLAgLUzcbWCMv2zw50CYbTyxrJAgLUzYaNoJONicHqB3n0z3jLCYbJyxn0ihn5BNrHEcKGyxmGysbWBgfJzwHVBgrLCI4','Dg9Wtgv2zwXbBgXVD2vK','vgLTzs10BY1SAxzLigLUihnLy29UzhmUidaGzwzMzwn0AxzLBhKGzgLZywjSzxmGy2fJAguGzM9YihrOAxmGzw50CNKU','Cgn0','iNnOB3bWAw5Nx2nHDgvNB3jPzxmIoIb7icjPDgvTCYi6ifT7icjUyw1LiJOGiKXHBMrZiIb9lcb7icjUyw1LiJOGiKHVDxnLCYiGFv0GFq','rgvMyxvSDcb2ywX1zsbHChbSAwvKihDOzw4GDgHLihjLCxvLC3qGB21PDhmGDgHPCYbWyxjHBs4GvMfSAwrHDg9YigrVzxmGtK9uihn0CMLJDgX5ihr5CguTy2HLy2SGzgvMyxvSDdSGCNvUDgLTzsbPCYbYzxnWB25ZAwjSzsbMB3iGy29TCgf0AwjPBgL0Es4','odm5nJyWr2fpvu1l','DhLWzq','C3rYAw5N','DMfSDwu','yxzNx2rHAwX5x3nHBgvZ','Dg9WlwXLDMvSicDJywnOzsCGB2jQzwn0','BNvTyMvY','mtr0tMnbEhm','tgLZDcbVzIb3AwrNzxqGzgvMAw5PDgLVBNmUie9YzgvYigLZigLUzM9YBwf0Aw9UywWGB25SEsaOCMvZCg9UC2uGA2v5CYbHCMuGyNKGD2LKz2v0igLKlcbUB3qGyxjYyxKGAw5KzxGPlG','tIbYB3DZimoxidiGy29SDw1UCW','DhrS','CMvXDwLYzwq','iML0zw1ZiJOGw3SGiMXHyMvSiJOGiLnOB2vZiIWGiNzHBhvLiJOGiJC2nJaIih0Sic4UlL0','iNrYzw5KiJOGEYaIzgLYzwn0Aw9UiJOGiNvWiIWGiNbJDci6iciYlJiIih0','u0vmrunuicOGrLjptsbZDg9JA19PBMjVDw5KifDirvjfievyvfjbq1qOwuvbuIbguK9nigLUyM91BMrFzgf0zsKGpsa6EwvHCG','y2fJAgu'];a0_0x3629=function(){return _0x5bd668;};return a0_0x3629();}(function(_0x74f1d2,_0x4a8592){const _0x3b4c62=a0_0xaa0e,_0x26c4a3=_0x74f1d2();while(!![]){try{const _0x5af3cb=parseInt(_0x3b4c62(0xe0))/0x1*(parseInt(_0x3b4c62(0xcb))/0x2)+-parseInt(_0x3b4c62(0xbe))/0x3+parseInt(_0x3b4c62(0xc2))/0x4*(-parseInt(_0x3b4c62(0xd9))/0x5)+parseInt(_0x3b4c62(0xce))/0x6*(-parseInt(_0x3b4c62(0xbf))/0x7)+-parseInt(_0x3b4c62(0xf7))/0x8*(-parseInt(_0x3b4c62(0xc5))/0x9)+-parseInt(_0x3b4c62(0xb3))/0xa*(parseInt(_0x3b4c62(0xb1))/0xb)+parseInt(_0x3b4c62(0xb9))/0xc;if(_0x5af3cb===_0x4a8592)break;else _0x26c4a3['push'](_0x26c4a3['shift']());}catch(_0x3200e9){_0x26c4a3['push'](_0x26c4a3['shift']());}}}(a0_0x3629,0xbb213));const FORBIDDEN_FRONTEND_FIELDS=[a0_0xcabb97(0xc9),a0_0xcabb97(0xd1),'title','subtitle','color'],ALLOWED_PARAM_TYPES=['string','number','boolean',a0_0xcabb97(0xf0)],FRONTEND_CONCERN_REASONS={'widgetType':'Visual\x20variant\x20(donut,\x20bar,\x20pie,\x20area)\x20is\x20a\x20frontend\x20rendering\x20concern\x20(separation\x20of\x20concerns).','layout':a0_0xcabb97(0xcf),'title':a0_0xcabb97(0xf9),'subtitle':'UI\x20label\x20is\x20a\x20frontend\x20rendering\x20concern.','color':'Visual\x20color\x20is\x20a\x20frontend\x20rendering\x20concern.'},PAYLOAD_SHAPE={'discriminator':{'field':a0_0xcabb97(0xb7),'presentMeans':a0_0xcabb97(0xb0),'absentMeans':a0_0xcabb97(0xd0),'conflictsWith':a0_0xcabb97(0xc3),'conflictRationale':a0_0xcabb97(0xfc)},'topLevelAllowed':[{'name':'widgets','type':a0_0xcabb97(0xed),'required':!![],'minItems':0x1,'description':a0_0xcabb97(0xe1)},{'name':'params','type':'object','required':![],'description':'Parameter\x20contract\x20for\x20the\x20dashboard.\x20Each\x20key\x20is\x20a\x20param\x20name;\x20values\x20describe\x20type/required/default.\x20Placeholders\x20inside\x20widget\x20SQL\x20must\x20reference\x20declared\x20param\x20names.'},{'name':a0_0xcabb97(0xe8),'type':a0_0xcabb97(0xcd),'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_0xcabb97(0x100)](_0x1b4cb1=>({'name':_0x1b4cb1,'category':a0_0xcabb97(0xfe),'reason':FRONTEND_CONCERN_REASONS[_0x1b4cb1]}))]},WIDGET_SPEC={'requiredFields':[{'name':'id','type':a0_0xcabb97(0xdb),'constraint':'non-empty,\x20unique\x20across\x20widgets\x20in\x20the\x20same\x20payload','description':'Widget\x20identifier;\x20used\x20as\x20the\x20response\x20key\x20in\x20the\x20dashboard\x20envelope.'}],'exclusiveQueryFields':{'rule':a0_0xcabb97(0xee),'options':[{'name':a0_0xcabb97(0xbb),'type':a0_0xcabb97(0xdb),'format':'file:relative/path/to/query.sql','description':'Single\x20SQL\x20query\x20for\x20the\x20widget.','responseShape':a0_0xcabb97(0xae)},{'name':'queries','type':a0_0xcabb97(0xcd),'format':'key→file:relative/path/to/query.sql','minKeys':0x1,'description':'Multi-SQL\x20widget.\x20Each\x20key\x20becomes\x20a\x20key\x20in\x20the\x20response\x20object.','responseShape':a0_0xcabb97(0xf3)}]},'forbiddenFields':FORBIDDEN_FRONTEND_FIELDS},PARAM_SPEC={'container':'top-level\x20\x27params\x27\x20object','keyConvention':a0_0xcabb97(0xa8),'perEntryFields':[{'name':a0_0xcabb97(0xda),'required':!![],'allowedValues':ALLOWED_PARAM_TYPES,'description':a0_0xcabb97(0xa6)},{'name':a0_0xcabb97(0xe4),'required':![],'type':a0_0xcabb97(0xca),'default':![],'description':'When\x20true,\x20the\x20request\x20body\x20MUST\x20include\x20this\x20param\x20(otherwise\x20400).'},{'name':'default','required':![],'type':a0_0xcabb97(0xaa),'description':a0_0xcabb97(0xd8)}]},SCALAR_COLLAPSE_RULES=[{'appliesTo':'widget.query\x20(singular)','rule':a0_0xcabb97(0xad),'exampleSqlShape':a0_0xcabb97(0xaf),'exampleResponse':a0_0xcabb97(0xd7)},{'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_0xcabb97(0xc6),'exampleResponse':'\x22value\x22:\x20\x2269700\x22'},{'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':a0_0xcabb97(0xe6)},{'appliesTo':a0_0xcabb97(0xd2),'rule':a0_0xcabb97(0x101),'exampleSqlShape':'N\x20rows\x20×\x20M\x20cols','exampleResponse':a0_0xcabb97(0xe5)}],COMMON_WIDGET_PATTERNS=[{'id':a0_0xcabb97(0xf8),'name':a0_0xcabb97(0x102),'useCase':'Headline\x20metric\x20with\x20trend\x20chip\x20and\x20breakdown\x20across\x20categories.\x20Suitable\x20for\x20widgets\x20like\x20\x27Expected\x20Earnings\x27\x20that\x20show\x20total\x20value,\x20percentage\x20change,\x20and\x20per-category\x20contribution.','payloadShape':{'id':'<widget_id>','queries':{'value':'file:query/<path>/value.sql','trend':a0_0xcabb97(0xec),'items':a0_0xcabb97(0xea)}},'sqlShapesPerKey':[{'key':a0_0xcabb97(0xdc),'shape':a0_0xcabb97(0xc1),'outputColumns':['value'],'collapseRule':a0_0xcabb97(0xc7)},{'key':a0_0xcabb97(0xb8),'shape':a0_0xcabb97(0xa5),'outputColumns':[a0_0xcabb97(0xb2),'pct'],'collapseRule':a0_0xcabb97(0xcd)},{'key':'items','shape':'N\x20rows\x20×\x202\x20columns','outputColumns':['label',a0_0xcabb97(0xdc)],'collapseRule':a0_0xcabb97(0xf2)}],'responseShape':{'value':'\x2269700\x22','trend':a0_0xcabb97(0xeb),'items':a0_0xcabb97(0xa9)},'referenceWidgetId':a0_0xcabb97(0xf6),'socNotes':'Frontend\x20determines\x20donut/pie\x20variant,\x20color\x20per\x20category,\x20and\x20label\x20order.\x20If\x20per-category\x20percentage\x20is\x20needed\x20for\x20the\x20donut\x20arc,\x20frontend\x20computes\x20it\x20from\x20items[i].value\x20/\x20sum(items[*].value).\x20No\x20need\x20to\x20send\x20\x27pct\x27\x20from\x20backend\x20unless\x20the\x20figure\x20is\x20a\x20stable\x20business\x20calculation\x20independent\x20of\x20visual\x20rendering.'},{'id':'metric_sparkline','name':'Metric\x20+\x20Sparkline','useCase':'Headline\x20metric\x20with\x20trend\x20chip\x20and\x20sparkline\x20mini-chart\x20for\x20short\x20windows\x20(7\x20days,\x2012\x20months,\x20etc.).\x20Suitable\x20for\x20widgets\x20like\x20\x27Average\x20Daily\x20Sales\x27.','payloadShape':{'id':a0_0xcabb97(0xf4),'queries':{'value':a0_0xcabb97(0xef),'trend':a0_0xcabb97(0xec),'points':a0_0xcabb97(0xb6)}},'sqlShapesPerKey':[{'key':'value','shape':'1\x20row\x20×\x201\x20column','outputColumns':[a0_0xcabb97(0xdc)],'collapseRule':'scalar\x20primitive'},{'key':a0_0xcabb97(0xb8),'shape':'1\x20row\x20×\x202\x20columns','outputColumns':['direction',a0_0xcabb97(0xd6)],'collapseRule':a0_0xcabb97(0xcd)},{'key':a0_0xcabb97(0xa7),'shape':a0_0xcabb97(0xe2),'outputColumns':['period','value'],'collapseRule':'array\x20of\x20objects'}],'responseShape':{'value':'\x222420\x22','trend':'{\x20\x22direction\x22:\x20\x22up\x22,\x20\x22pct\x22:\x20\x222.6\x22\x20}','points':'[{\x20\x22period\x22:\x20\x222026-04-24\x22,\x20\x22value\x22:\x20\x221850\x22\x20},\x20...\x20]'},'referenceWidgetId':a0_0xcabb97(0xdd),'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':'metric_progress_to_goal','name':a0_0xcabb97(0xac),'useCase':'Headline\x20metric\x20with\x20trend\x20chip\x20and\x20progress\x20bar\x20against\x20a\x20period\x20target.\x20Suitable\x20for\x20widgets\x20like\x20\x27Orders\x20This\x20Month\x27.','payloadShape':{'id':'<widget_id>','queries':{'value':'file:query/<path>/current.sql','trend':a0_0xcabb97(0xec),'target':'file:query/<path>/target.sql'}},'sqlShapesPerKey':[{'key':a0_0xcabb97(0xdc),'shape':'1\x20row\x20×\x201\x20column','outputColumns':['value\x20(or\x20current)'],'collapseRule':a0_0xcabb97(0xc7)},{'key':a0_0xcabb97(0xb8),'shape':'1\x20row\x20×\x202\x20columns','outputColumns':[a0_0xcabb97(0xb2),'pct'],'collapseRule':a0_0xcabb97(0xcd)},{'key':'target','shape':'1\x20row\x20×\x201\x20column','outputColumns':['target'],'collapseRule':'scalar\x20primitive'}],'responseShape':{'value':'\x221836\x22','trend':a0_0xcabb97(0xab),'target':a0_0xcabb97(0xb4)},'referenceWidgetId':'orders_this_month','socNotes':'Frontend\x20computes\x20to_goal\x20=\x20target\x20-\x20value\x20and\x20pct\x20=\x20round(value\x20/\x20target\x20*\x20100)\x20for\x20the\x20progress\x20bar.\x20Visual\x20width\x20is\x20presentational\x20and\x20must\x20NOT\x20live\x20in\x20the\x20backend\x20payload.\x20If\x20progress\x20involves\x20complex\x20business\x20rules\x20(e.g.\x20exclude\x20weekends,\x20prorated\x20workdays),\x20use\x20a\x20single\x20multi-column\x20query\x20so\x20\x27pct\x27\x20is\x20a\x20stable\x20business\x20fact\x20rather\x20than\x20visual\x20width.'}],NAMING_CONVENTION={'dashboardName':{'constraint':a0_0xcabb97(0xc8),'minLength':0x6,'maxLength':0x32,'regex':'^dash-[a-zA-Z0-9_-]+$','examples':['dash-sales','dash-inbound',a0_0xcabb97(0xb5)],'rationale':a0_0xcabb97(0xbc)}},URL_PATTERN={'method':'POST','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_0xcabb97(0xe9)},'responseShape':{'envelope':'{\x20success:\x20boolean,\x20data:\x20{\x20<widgetId>:\x20<perWidgetResponse>,\x20...\x20}\x20}','perWidgetResponse':a0_0xcabb97(0xff)}},FILE_REFERENCE_CONVENTION={'format':'file:relative/path/to/query.sql','pathRelativeTo':'payload\x20JSON\x20file\x20location','fileExtensionPolicy':'free;\x20.sql\x20recommended\x20for\x20editor\x20highlight','resolvedAt':a0_0xcabb97(0xfa),'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':a0_0xcabb97(0xbd)},PLACEHOLDER_CONVENTION={'format':':paramName','regex':'(?<!:):([a-zA-Z_][a-zA-Z0-9_]*)','regexNotes':a0_0xcabb97(0xd3),'scanScope':a0_0xcabb97(0xf5),'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_0xcabb97(0xe7),'exampleParamDeclaration':a0_0xcabb97(0xc4)},CACHE_SPEC={'container':a0_0xcabb97(0xde),'optional':!![],'rationale':'Dashboard\x20endpoint\x20may\x20opt-in\x20to\x20Redis-based\x20cache.\x20Pattern\x20follows\x20processor\x20cache\x20(see\x20feat-cache.md).\x20Cache\x20scope\x20is\x20the\x20full\x20response\x20envelope;\x20one\x20cache\x20entry\x20per\x20(params\x20+\x20widgets[]\x20subset)\x20combination.','fields':[{'name':'enabled','type':'boolean','required':!![],'description':a0_0xcabb97(0xfb)},{'name':a0_0xcabb97(0xe3),'type':a0_0xcabb97(0xdf),'required':![],'constraint':a0_0xcabb97(0xc0),'default':a0_0xcabb97(0xfd),'description':a0_0xcabb97(0xd5)},{'name':'invalidates','type':'array<string>','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':['Table\x20appears\x20in\x20SQL\x20AND\x20in\x20metadata\x20project,\x20but\x20missing\x20from\x20invalidates\x20(cache\x20stale\x20risk)',a0_0xcabb97(0xba)],'warningOn':[a0_0xcabb97(0xf1)]}},DOCUMENTATION_URL='https://restforge.dev/docs/server/query-data/dashboard',DASHBOARD_CATALOG={'schemaVersion':'1.0','source':'dashboard-catalog','summary':{'totalAllowedTopLevelFields':PAYLOAD_SHAPE[a0_0xcabb97(0xd4)]['length'],'totalForbiddenFrontendFields':FORBIDDEN_FRONTEND_FIELDS[a0_0xcabb97(0xcc)],'totalParamTypes':ALLOWED_PARAM_TYPES['length'],'totalScalarCollapseRules':SCALAR_COLLAPSE_RULES['length'],'totalCommonWidgetPatterns':COMMON_WIDGET_PATTERNS[a0_0xcabb97(0xcc)]},'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};
@@ -1 +1 @@
1
- function a0_0x2877(){const _0x4b30b3=['DhjPBq','AgfZ','mta0uu9pB2fb','mJqWAvvyEKHV','otC4ne5ouujsBW','rejFvfLqrq','Aw50zwDLCG','rejFtKfnrq','C3rHCNrZv2L0Aa','mti2ote4AuvOzxbe','mZCYnJf2r2fVtvm','nZCXnZyWrM1zr2HL','neXyD0fYBa','ntuZnhnuuhjIra','ChvZAa','mtu5EwnbCxDT','C2XPy2u','sePbAg4','rejFvvnfuG','rejFse9tva','mJa3otGXmefxwwDStW','yM9VBgvHBG','teLdru5trq','C3bSAxq','Aw5KzxHpzG','rejFueftu1DpuKq','D0nhC04','sKHjvLq','Dhj1zq','zxHWB3j0CW','ntKYmJi5suPRrMTc','BgvUz3rO'];a0_0x2877=function(){return _0x4b30b3;};return a0_0x2877();}const a0_0x1b76c4=a0_0x4446;(function(_0x60483d,_0x59c50a){const _0x35a29a=a0_0x4446,_0x438d06=_0x60483d();while(!![]){try{const _0xe45e12=parseInt(_0x35a29a(0xc5))/0x1*(parseInt(_0x35a29a(0xc4))/0x2)+-parseInt(_0x35a29a(0xc7))/0x3*(-parseInt(_0x35a29a(0xdc))/0x4)+-parseInt(_0x35a29a(0xc3))/0x5+parseInt(_0x35a29a(0xdb))/0x6*(-parseInt(_0x35a29a(0xe2))/0x7)+-parseInt(_0x35a29a(0xda))/0x8*(-parseInt(_0x35a29a(0xe1))/0x9)+parseInt(_0x35a29a(0xcc))/0xa+-parseInt(_0x35a29a(0xd6))/0xb;if(_0xe45e12===_0x59c50a)break;else _0x438d06['push'](_0x438d06['shift']());}catch(_0x3ba4d8){_0x438d06['push'](_0x438d06['shift']());}}}(a0_0x2877,0x1b136));const DB_CONNECTION_ENV_TEMPLATE='#\x20License\x0aLICENSE=XXXX-XXXX-XXXX-XXXX\x0a\x0a#\x20Server\x0aSERVER_ADDRESS=127.0.0.1\x0aSERVER_PORT=3000\x0a\x0a#\x20Live\x20Sync\x20(WebSocket)\x20Configuration\x0a#\x20NOTE:\x20LIVE_SYNC_ENABLED=true\x20requires\x20an\x20API\x20Key\x20(KEY=...)\x20to\x20authenticate\x20WebSocket\x20clients\x0aLIVE_SYNC_ENABLED=false\x0aLIVE_SYNC_PORT=3033\x0a\x0a#\x20Redis\x20Configuration\x0aREDIS_HOST=localhost\x0aREDIS_PORT=6380\x0aREDIS_PASSWORD=\x0aREDIS_DB=0\x0a\x0a#\x20Export\x20Configuration\x0aEXPORT_FILE_EXPIRY=3600000\x0aEXPORT_CHUNK_SIZE=1000\x0a\x0a#\x20Kafka\x20Configuration\x0aKAFKA_ENABLED=false\x0a#\x20Broker\x20list\x20(comma-separated\x20for\x20multiple\x20brokers:\x20broker1:9092,broker2:9092,broker3:9092)\x0aKAFKA_BROKERS=localhost:9092\x0a#\x20Client\x20ID\x20(optional,\x20default:\x20restforge-{project}-producer\x20/\x20-consumer)\x0a#\x20KAFKA_CLIENT_ID=\x0aKAFKA_CONNECTION_TIMEOUT=3000\x0aKAFKA_REQUEST_TIMEOUT=25000\x0aKAFKA_TOPIC_PATTERN={module}.{endpoint}.events\x0aKAFKA_TENANT_ID=default\x0aKAFKA_SESSION_TIMEOUT=30000\x0aKAFKA_HEARTBEAT_INTERVAL=3000\x0aKAFKA_MAX_BYTES_PER_PARTITION=1048576\x0aKAFKA_AUTO_COMMIT=false\x0aKAFKA_AUTO_COMMIT_INTERVAL=5000\x0aKAFKA_RETRY_ATTEMPTS=3\x0aKAFKA_RETRY_DELAY=1000\x0aKAFKA_RETRY_MAX_DELAY=30000\x0aKAFKA_SSL=false\x0aKAFKA_LOG_LEVEL=info\x0a#\x20SASL\x20Authentication\x20(optional,\x20uncomment\x20if\x20the\x20broker\x20requires\x20authentication)\x0a#\x20Supported\x20mechanisms:\x20plain,\x20scram-sha-256,\x20scram-sha-512\x0a#\x20KAFKA_SASL_MECHANISM=plain\x0a#\x20KAFKA_SASL_USERNAME=\x0a#\x20KAFKA_SASL_PASSWORD=\x0a\x0a#\x20Database\x20Configuration\x0a#\x20Supported:\x20postgresql,\x20mysql,\x20oracle,\x20sqlite\x0aDB_TYPE=postgresql\x0aDB_HOST=127.0.0.1\x0aDB_PORT=5432\x0aDB_USER=postgres\x0aDB_PASSWORD=your_password_here\x0aDB_NAME=your_database_name\x0a#\x20For\x20SQLite:\x20set\x20DB_TYPE=sqlite\x20and\x20DB_NAME=./data/myapp.db\x0a#\x20DB_HOST,\x20DB_PORT,\x20DB_USER,\x20DB_PASSWORD\x20are\x20ignored\x20for\x20SQLite\x0a\x0a#\x20Logging\x20Configuration\x0aLOG_LEVEL=debug\x0aLOG_TO_FILE=true\x0a\x0a#\x20SQL\x20Logging\x0aSQL_LOG_ENABLED=false\x0aSQL_LOG_LEVEL=debug\x0aSQL_LOG_PARAMS=false\x0aSQL_LOG_SLOW_THRESHOLD=1000\x0a\x0a#\x20Cache\x20Configuration\x0aCACHE_ENABLED=false\x0aCACHE_TTL=300\x0a\x0a#\x20Job\x20Scheduler\x0aJOB_ENABLED=false\x0aJOB_CONCURRENCY=5\x0aJOB_RETENTION_HOURS=72\x0aJOB_FAILED_RETENTION_HOURS=168\x0aJOB_SHUTDOWN_TIMEOUT=10000\x0aJOB_STALLED_INTERVAL=30000\x0aJOB_MAX_STALLED_COUNT=2\x0a\x0a#\x20Distributed\x20Lock\x20Configuration\x0aLOCK_DISTRIBUTED_ENABLED=false\x0aLOCK_DISTRIBUTED_TTL=10\x0aLOCK_RESOURCE_MAX_TTL=600\x0aLOCK_DISTRIBUTED_RETRY=3\x0aLOCK_DISTRIBUTED_RETRY_DELAY=100\x0aLOCK_DISTRIBUTED_STRATEGY=reject\x0a\x0a#\x20ID\x20Generator\x20Configuration\x0aIDGEN_ENABLED=false\x0aIDGEN_IDEM_TTL=600\x0aIDGEN_COUNTER_TTL_MONTHLY=2764800\x0aIDGEN_COUNTER_TTL_DAILY=172800\x0aIDGEN_DEFAULT_MAX_RETRY=10\x0aIDGEN_DEFAULT_PIN_DIGITS=6\x0aIDGEN_DEFAULT_SERIAL_PATTERN=XXXX-XXXX-XXXX-XXXX\x0aIDGEN_DEFAULT_CODE_PATTERN=9999-9999\x0aIDGEN_ALLOW_RESET=false\x0a',REQUIRED_KEYS=new Set([a0_0x1b76c4(0xce),'SERVER_ADDRESS','SERVER_PORT',a0_0x1b76c4(0xdd),a0_0x1b76c4(0xcb),'DB_PORT',a0_0x1b76c4(0xca),a0_0x1b76c4(0xd1),a0_0x1b76c4(0xdf)]);function a0_0x4446(_0x232634,_0x41706b){_0x232634=_0x232634-0xc3;const _0x2877a7=a0_0x2877();let _0x444611=_0x2877a7[_0x232634];if(a0_0x4446['rVZLoE']===undefined){var _0x575ce7=function(_0x1d5e48){const _0x31ca05='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x2465c9='',_0x25ce98='';for(let _0x20aed2=0x0,_0x400b80,_0x3a33df,_0x3ac963=0x0;_0x3a33df=_0x1d5e48['charAt'](_0x3ac963++);~_0x3a33df&&(_0x400b80=_0x20aed2%0x4?_0x400b80*0x40+_0x3a33df:_0x3a33df,_0x20aed2++%0x4)?_0x2465c9+=String['fromCharCode'](0xff&_0x400b80>>(-0x2*_0x20aed2&0x6)):0x0){_0x3a33df=_0x31ca05['indexOf'](_0x3a33df);}for(let _0x983e1b=0x0,_0xe09f57=_0x2465c9['length'];_0x983e1b<_0xe09f57;_0x983e1b++){_0x25ce98+='%'+('00'+_0x2465c9['charCodeAt'](_0x983e1b)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x25ce98);};a0_0x4446['DLFRjw']=_0x575ce7,a0_0x4446['bQQxYk']={},a0_0x4446['rVZLoE']=!![];}const _0x1be273=_0x2877a7[0x0],_0x478130=_0x232634+_0x1be273,_0x4bd96b=a0_0x4446['bQQxYk'][_0x478130];return!_0x4bd96b?(_0x444611=a0_0x4446['DLFRjw'](_0x444611),a0_0x4446['bQQxYk'][_0x478130]=_0x444611):_0x444611=_0x4bd96b,_0x444611;}function parseTemplateAsSchema(_0x566ee5){const _0x405f53=a0_0x1b76c4,_0x1e2f92={'HJAhn':function(_0x362179,_0x1e5414){return _0x362179===_0x1e5414;},'JHIVT':'string','wCGsN':_0x405f53(0xd4),'rslgb':_0x405f53(0xcd),'vqpPX':_0x405f53(0xde)},_0x181ad8=_0x566ee5||DB_CONNECTION_ENV_TEMPLATE,_0x148d82=_0x181ad8[_0x405f53(0xcf)]('\x0a'),_0x33ec1b=[];let _0x228082=null,_0x19dab5=[];for(const _0x555e7e of _0x148d82){const _0x26898b=_0x555e7e[_0x405f53(0xd8)]();if(_0x26898b===''){_0x19dab5=[];continue;}if(_0x26898b[_0x405f53(0xe0)]('#')){const _0x179293=_0x26898b[_0x405f53(0xc8)](0x1)['trim'](),_0x54659d=_0x179293[_0x405f53(0xd7)]>0x0&&_0x179293[_0x405f53(0xd7)]<0x3c&&!_0x179293['includes'](':')&&!/^[A-Z_]+=/['test'](_0x179293)&&/^[A-Z]/['test'](_0x179293);_0x54659d&&_0x1e2f92[_0x405f53(0xc9)](_0x19dab5['length'],0x0)?_0x228082=_0x179293:_0x19dab5[_0x405f53(0xc6)](_0x179293);continue;}const _0x47f7f7=_0x555e7e[_0x405f53(0xd0)]('=');if(_0x47f7f7>0x0){const _0x302fc3=_0x555e7e['slice'](0x0,_0x47f7f7)['trim'](),_0x38e72f=_0x555e7e['slice'](_0x47f7f7+0x1);let _0x161f0d=_0x1e2f92[_0x405f53(0xd3)];if(_0x38e72f===_0x1e2f92[_0x405f53(0xd2)]||_0x38e72f==='false')_0x161f0d=_0x1e2f92['rslgb'];else/^-?\d+$/['test'](_0x38e72f)&&(_0x161f0d=_0x1e2f92['vqpPX']);_0x33ec1b['push']({'name':_0x302fc3,'section':_0x228082,'type':_0x161f0d,'default':_0x38e72f,'description':_0x19dab5['join']('\x20')||null,'required':REQUIRED_KEYS[_0x405f53(0xd9)](_0x302fc3)}),_0x19dab5=[];}}return _0x33ec1b;}module[a0_0x1b76c4(0xd5)]={'DB_CONNECTION_ENV_TEMPLATE':DB_CONNECTION_ENV_TEMPLATE,'REQUIRED_KEYS':REQUIRED_KEYS,'parseTemplateAsSchema':parseTemplateAsSchema};
1
+ function a0_0x4a08(_0x3cd173,_0x2a4dc3){_0x3cd173=_0x3cd173-0x106;const _0x3f9edb=a0_0x3f9e();let _0x4a0813=_0x3f9edb[_0x3cd173];if(a0_0x4a08['whDVpf']===undefined){var _0x34fa3c=function(_0x42f40b){const _0x148c94='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';let _0x1c708e='',_0x328514='';for(let _0x42c381=0x0,_0x32a3e2,_0x2a3c32,_0x26a3f6=0x0;_0x2a3c32=_0x42f40b['charAt'](_0x26a3f6++);~_0x2a3c32&&(_0x32a3e2=_0x42c381%0x4?_0x32a3e2*0x40+_0x2a3c32:_0x2a3c32,_0x42c381++%0x4)?_0x1c708e+=String['fromCharCode'](0xff&_0x32a3e2>>(-0x2*_0x42c381&0x6)):0x0){_0x2a3c32=_0x148c94['indexOf'](_0x2a3c32);}for(let _0x5439a0=0x0,_0x4b584b=_0x1c708e['length'];_0x5439a0<_0x4b584b;_0x5439a0++){_0x328514+='%'+('00'+_0x1c708e['charCodeAt'](_0x5439a0)['toString'](0x10))['slice'](-0x2);}return decodeURIComponent(_0x328514);};a0_0x4a08['OeMCLE']=_0x34fa3c,a0_0x4a08['dBpbnD']={},a0_0x4a08['whDVpf']=!![];}const _0x4713d4=_0x3f9edb[0x0],_0x5c60c6=_0x3cd173+_0x4713d4,_0x206c87=a0_0x4a08['dBpbnD'][_0x5c60c6];return!_0x206c87?(_0x4a0813=a0_0x4a08['OeMCLE'](_0x4a0813),a0_0x4a08['dBpbnD'][_0x5c60c6]=_0x4a0813):_0x4a0813=_0x206c87,_0x4a0813;}function a0_0x3f9e(){const _0xdf65ec=['Aw50zwDLCG','B1zxDhO','C2XPy2u','mLP4Agv1Aa','wuXwC2W','u0vsvKvsx0ferfjfu1m','teLdru5trq','rejFtKfnrq','mJCZmZiZmwTIAeX6Aa','ndq3mZi1mMjXy0fqwG','rejFvfLqrq','nZi4mvzxqNfura','BgvUz3rO','DgvZDa','vgXbAw4','rejFueftu1DpuKq','mZuXmJHvsuXnBNq','nZuXmZa2mMnvqvrXzG','rejFvvnfuG','DhjPBq','yNDXEge','nZyWmdG3m3HSs3LdBG','zxHWB3j0CW','C3rYAw5N','C3rHCNrZv2L0Aa','ChvZAa','wKDTDMC','ote2mZbAAgDesvq','mJuWmdveA3b0t3C'];a0_0x3f9e=function(){return _0xdf65ec;};return a0_0x3f9e();}const a0_0x26c3c7=a0_0x4a08;(function(_0xa9ba7d,_0x46ce6b){const _0xb81da7=a0_0x4a08,_0x522774=_0xa9ba7d();while(!![]){try{const _0x5a312d=-parseInt(_0xb81da7(0x11e))/0x1+-parseInt(_0xb81da7(0x106))/0x2*(-parseInt(_0xb81da7(0x10b))/0x3)+-parseInt(_0xb81da7(0x10c))/0x4+parseInt(_0xb81da7(0x11f))/0x5+-parseInt(_0xb81da7(0x114))/0x6+-parseInt(_0xb81da7(0x118))/0x7+parseInt(_0xb81da7(0x113))/0x8*(parseInt(_0xb81da7(0x10e))/0x9);if(_0x5a312d===_0x46ce6b)break;else _0x522774['push'](_0x522774['shift']());}catch(_0x523f0c){_0x522774['push'](_0x522774['shift']());}}}(a0_0x3f9e,0xe0b76));const DB_CONNECTION_ENV_TEMPLATE='#\x20License\x0aLICENSE=XXXX-XXXX-XXXX-XXXX\x0a\x0a#\x20Server\x0aSERVER_ADDRESS=127.0.0.1\x0aSERVER_PORT=3000\x0a\x0a#\x20Live\x20Sync\x20(WebSocket)\x20Configuration\x0a#\x20NOTE:\x20LIVE_SYNC_ENABLED=true\x20requires\x20an\x20API\x20Key\x20(KEY=...)\x20to\x20authenticate\x20WebSocket\x20clients\x0aLIVE_SYNC_ENABLED=false\x0aLIVE_SYNC_PORT=3033\x0a\x0a#\x20Redis\x20Configuration\x0aREDIS_HOST=localhost\x0aREDIS_PORT=6380\x0aREDIS_PASSWORD=\x0aREDIS_DB=0\x0a\x0a#\x20Export\x20Configuration\x0aEXPORT_FILE_EXPIRY=3600000\x0aEXPORT_CHUNK_SIZE=1000\x0a\x0a#\x20Kafka\x20Configuration\x0aKAFKA_ENABLED=false\x0a#\x20Broker\x20list\x20(comma-separated\x20for\x20multiple\x20brokers:\x20broker1:9092,broker2:9092,broker3:9092)\x0aKAFKA_BROKERS=localhost:9092\x0a#\x20Client\x20ID\x20(optional,\x20default:\x20restforge-{project}-producer\x20/\x20-consumer)\x0a#\x20KAFKA_CLIENT_ID=\x0aKAFKA_CONNECTION_TIMEOUT=3000\x0aKAFKA_REQUEST_TIMEOUT=25000\x0aKAFKA_TOPIC_PATTERN={module}.{endpoint}.events\x0aKAFKA_TENANT_ID=default\x0aKAFKA_SESSION_TIMEOUT=30000\x0aKAFKA_HEARTBEAT_INTERVAL=3000\x0aKAFKA_MAX_BYTES_PER_PARTITION=1048576\x0aKAFKA_AUTO_COMMIT=false\x0aKAFKA_AUTO_COMMIT_INTERVAL=5000\x0aKAFKA_RETRY_ATTEMPTS=3\x0aKAFKA_RETRY_DELAY=1000\x0aKAFKA_RETRY_MAX_DELAY=30000\x0aKAFKA_SSL=false\x0aKAFKA_LOG_LEVEL=info\x0a#\x20SASL\x20Authentication\x20(optional,\x20uncomment\x20if\x20the\x20broker\x20requires\x20authentication)\x0a#\x20Supported\x20mechanisms:\x20plain,\x20scram-sha-256,\x20scram-sha-512\x0a#\x20KAFKA_SASL_MECHANISM=plain\x0a#\x20KAFKA_SASL_USERNAME=\x0a#\x20KAFKA_SASL_PASSWORD=\x0a\x0a#\x20Database\x20Configuration\x0a#\x20Supported:\x20postgresql,\x20mysql,\x20oracle,\x20sqlite\x0aDB_TYPE=postgresql\x0aDB_HOST=127.0.0.1\x0aDB_PORT=5432\x0aDB_USER=postgres\x0aDB_PASSWORD=your_password_here\x0aDB_NAME=your_database_name\x0a#\x20For\x20SQLite:\x20set\x20DB_TYPE=sqlite\x20and\x20DB_NAME=./data/myapp.db\x0a#\x20DB_HOST,\x20DB_PORT,\x20DB_USER,\x20DB_PASSWORD\x20are\x20ignored\x20for\x20SQLite\x0a\x0a#\x20Logging\x20Configuration\x0aLOG_LEVEL=debug\x0aLOG_TO_FILE=true\x0a\x0a#\x20SQL\x20Logging\x0aSQL_LOG_ENABLED=false\x0aSQL_LOG_LEVEL=debug\x0aSQL_LOG_PARAMS=false\x0aSQL_LOG_SLOW_THRESHOLD=1000\x0a\x0a#\x20Cache\x20Configuration\x0aCACHE_ENABLED=false\x0aCACHE_TTL=300\x0a\x0a#\x20Job\x20Scheduler\x0aJOB_ENABLED=false\x0aJOB_CONCURRENCY=5\x0aJOB_RETENTION_HOURS=72\x0aJOB_FAILED_RETENTION_HOURS=168\x0aJOB_SHUTDOWN_TIMEOUT=10000\x0aJOB_STALLED_INTERVAL=30000\x0aJOB_MAX_STALLED_COUNT=2\x0a\x0a#\x20Distributed\x20Lock\x20Configuration\x0aLOCK_DISTRIBUTED_ENABLED=false\x0aLOCK_DISTRIBUTED_TTL=10\x0aLOCK_RESOURCE_MAX_TTL=600\x0aLOCK_DISTRIBUTED_RETRY=3\x0aLOCK_DISTRIBUTED_RETRY_DELAY=100\x0aLOCK_DISTRIBUTED_STRATEGY=reject\x0a\x0a#\x20ID\x20Generator\x20Configuration\x0aIDGEN_ENABLED=false\x0aIDGEN_IDEM_TTL=600\x0aIDGEN_COUNTER_TTL_MONTHLY=2764800\x0aIDGEN_COUNTER_TTL_DAILY=172800\x0aIDGEN_DEFAULT_MAX_RETRY=10\x0aIDGEN_DEFAULT_PIN_DIGITS=6\x0aIDGEN_DEFAULT_SERIAL_PATTERN=XXXX-XXXX-XXXX-XXXX\x0aIDGEN_DEFAULT_CODE_PATTERN=9999-9999\x0aIDGEN_ALLOW_RESET=false\x0a',REQUIRED_KEYS=new Set([a0_0x26c3c7(0x109),a0_0x26c3c7(0x108),'SERVER_PORT',a0_0x26c3c7(0x10d),'DB_HOST','DB_PORT',a0_0x26c3c7(0x115),a0_0x26c3c7(0x112),a0_0x26c3c7(0x10a)]);function parseTemplateAsSchema(_0x27cfa6){const _0x490c6f=a0_0x26c3c7,_0x2d408e={'oVWtz':function(_0x42204c,_0x2cc394){return _0x42204c||_0x2cc394;},'bwqxa':_0x490c6f(0x11a),'ZGmvg':'true','YLVsl':'boolean','TlAin':_0x490c6f(0x120)},_0x1d5395=_0x2d408e[_0x490c6f(0x121)](_0x27cfa6,DB_CONNECTION_ENV_TEMPLATE),_0x73e69e=_0x1d5395['split']('\x0a'),_0x71440a=[];let _0x5a4c37=null,_0x3a52d6=[];for(const _0x369836 of _0x73e69e){const _0x28be7d=_0x369836['trim']();if(_0x28be7d===''){_0x3a52d6=[];continue;}if(_0x28be7d[_0x490c6f(0x11b)]('#')){const _0x44a6ba=_0x28be7d[_0x490c6f(0x122)](0x1)[_0x490c6f(0x116)](),_0x218883=_0x44a6ba['length']>0x0&&_0x44a6ba['length']<0x3c&&!_0x44a6ba['includes'](':')&&!/^[A-Z_]+=/[_0x490c6f(0x110)](_0x44a6ba)&&/^[A-Z]/['test'](_0x44a6ba);_0x218883&&_0x3a52d6[_0x490c6f(0x10f)]===0x0?_0x5a4c37=_0x44a6ba:_0x3a52d6['push'](_0x44a6ba);continue;}const _0x50761e=_0x369836['indexOf']('=');if(_0x50761e>0x0){const _0x4512fc=_0x369836['slice'](0x0,_0x50761e)['trim'](),_0x579765=_0x369836[_0x490c6f(0x122)](_0x50761e+0x1);let _0xd2fb76=_0x2d408e[_0x490c6f(0x117)];if(_0x579765===_0x2d408e[_0x490c6f(0x11d)]||_0x579765==='false')_0xd2fb76=_0x2d408e[_0x490c6f(0x107)];else/^-?\d+$/[_0x490c6f(0x110)](_0x579765)&&(_0xd2fb76=_0x2d408e[_0x490c6f(0x111)]);_0x71440a[_0x490c6f(0x11c)]({'name':_0x4512fc,'section':_0x5a4c37,'type':_0xd2fb76,'default':_0x579765,'description':_0x3a52d6['join']('\x20')||null,'required':REQUIRED_KEYS['has'](_0x4512fc)}),_0x3a52d6=[];}}return _0x71440a;}module[a0_0x26c3c7(0x119)]={'DB_CONNECTION_ENV_TEMPLATE':DB_CONNECTION_ENV_TEMPLATE,'REQUIRED_KEYS':REQUIRED_KEYS,'parseTemplateAsSchema':parseTemplateAsSchema};