@intranefr/superbackend 1.4.3

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 (188) hide show
  1. package/.commiat +4 -0
  2. package/.env.example +47 -0
  3. package/README.md +110 -0
  4. package/index.js +94 -0
  5. package/package.json +67 -0
  6. package/public/css/styles.css +139 -0
  7. package/public/js/animations.js +41 -0
  8. package/sdk/error-tracking/browser/package.json +16 -0
  9. package/sdk/error-tracking/browser/src/core.js +270 -0
  10. package/sdk/error-tracking/browser/src/embed.js +18 -0
  11. package/sdk/error-tracking/browser/src/index.js +1 -0
  12. package/server.js +5 -0
  13. package/src/admin/endpointRegistry.js +300 -0
  14. package/src/controllers/admin.controller.js +321 -0
  15. package/src/controllers/adminAssets.controller.js +530 -0
  16. package/src/controllers/adminAssetsStorage.controller.js +260 -0
  17. package/src/controllers/adminEjsVirtual.controller.js +354 -0
  18. package/src/controllers/adminFeatureFlags.controller.js +155 -0
  19. package/src/controllers/adminHeadless.controller.js +1071 -0
  20. package/src/controllers/adminI18n.controller.js +604 -0
  21. package/src/controllers/adminJsonConfigs.controller.js +97 -0
  22. package/src/controllers/adminLlm.controller.js +273 -0
  23. package/src/controllers/adminMigration.controller.js +257 -0
  24. package/src/controllers/adminSeoConfig.controller.js +515 -0
  25. package/src/controllers/adminStats.controller.js +121 -0
  26. package/src/controllers/adminUploadNamespaces.controller.js +208 -0
  27. package/src/controllers/assets.controller.js +248 -0
  28. package/src/controllers/auth.controller.js +93 -0
  29. package/src/controllers/billing.controller.js +223 -0
  30. package/src/controllers/featureFlags.controller.js +35 -0
  31. package/src/controllers/forms.controller.js +217 -0
  32. package/src/controllers/globalSettings.controller.js +252 -0
  33. package/src/controllers/headlessCrud.controller.js +126 -0
  34. package/src/controllers/i18n.controller.js +12 -0
  35. package/src/controllers/invite.controller.js +249 -0
  36. package/src/controllers/jsonConfigs.controller.js +19 -0
  37. package/src/controllers/metrics.controller.js +149 -0
  38. package/src/controllers/notificationAdmin.controller.js +264 -0
  39. package/src/controllers/notifications.controller.js +131 -0
  40. package/src/controllers/org.controller.js +357 -0
  41. package/src/controllers/orgAdmin.controller.js +491 -0
  42. package/src/controllers/stripeAdmin.controller.js +410 -0
  43. package/src/controllers/user.controller.js +361 -0
  44. package/src/controllers/userAdmin.controller.js +277 -0
  45. package/src/controllers/waitingList.controller.js +167 -0
  46. package/src/controllers/webhook.controller.js +200 -0
  47. package/src/middleware/auth.js +66 -0
  48. package/src/middleware/errorCapture.js +170 -0
  49. package/src/middleware/headlessApiTokenAuth.js +57 -0
  50. package/src/middleware/org.js +108 -0
  51. package/src/middleware.js +901 -0
  52. package/src/models/ActionEvent.js +31 -0
  53. package/src/models/ActivityLog.js +41 -0
  54. package/src/models/Asset.js +84 -0
  55. package/src/models/AuditEvent.js +93 -0
  56. package/src/models/EmailLog.js +28 -0
  57. package/src/models/ErrorAggregate.js +72 -0
  58. package/src/models/FormSubmission.js +41 -0
  59. package/src/models/GlobalSetting.js +38 -0
  60. package/src/models/HeadlessApiToken.js +24 -0
  61. package/src/models/HeadlessModelDefinition.js +41 -0
  62. package/src/models/I18nEntry.js +77 -0
  63. package/src/models/I18nLocale.js +33 -0
  64. package/src/models/Invite.js +70 -0
  65. package/src/models/JsonConfig.js +46 -0
  66. package/src/models/Notification.js +60 -0
  67. package/src/models/Organization.js +57 -0
  68. package/src/models/OrganizationMember.js +43 -0
  69. package/src/models/StripeCatalogItem.js +77 -0
  70. package/src/models/StripeWebhookEvent.js +57 -0
  71. package/src/models/User.js +89 -0
  72. package/src/models/VirtualEjsFile.js +60 -0
  73. package/src/models/VirtualEjsFileVersion.js +43 -0
  74. package/src/models/VirtualEjsGroupChange.js +32 -0
  75. package/src/models/WaitingList.js +41 -0
  76. package/src/models/Webhook.js +63 -0
  77. package/src/models/Workflow.js +29 -0
  78. package/src/models/WorkflowExecution.js +12 -0
  79. package/src/routes/admin.routes.js +26 -0
  80. package/src/routes/adminAssets.routes.js +28 -0
  81. package/src/routes/adminAssetsStorage.routes.js +13 -0
  82. package/src/routes/adminAudit.routes.js +196 -0
  83. package/src/routes/adminEjsVirtual.routes.js +17 -0
  84. package/src/routes/adminErrors.routes.js +164 -0
  85. package/src/routes/adminFeatureFlags.routes.js +12 -0
  86. package/src/routes/adminHeadless.routes.js +38 -0
  87. package/src/routes/adminI18n.routes.js +22 -0
  88. package/src/routes/adminJsonConfigs.routes.js +15 -0
  89. package/src/routes/adminLlm.routes.js +12 -0
  90. package/src/routes/adminMigration.routes.js +81 -0
  91. package/src/routes/adminSeoConfig.routes.js +20 -0
  92. package/src/routes/adminUploadNamespaces.routes.js +13 -0
  93. package/src/routes/assets.routes.js +21 -0
  94. package/src/routes/auth.routes.js +12 -0
  95. package/src/routes/billing.routes.js +11 -0
  96. package/src/routes/errorTracking.routes.js +31 -0
  97. package/src/routes/featureFlags.routes.js +9 -0
  98. package/src/routes/forms.routes.js +9 -0
  99. package/src/routes/formsAdmin.routes.js +13 -0
  100. package/src/routes/globalSettings.routes.js +18 -0
  101. package/src/routes/headless.routes.js +15 -0
  102. package/src/routes/i18n.routes.js +8 -0
  103. package/src/routes/invite.routes.js +9 -0
  104. package/src/routes/jsonConfigs.routes.js +8 -0
  105. package/src/routes/log.routes.js +111 -0
  106. package/src/routes/metrics.routes.js +9 -0
  107. package/src/routes/notificationAdmin.routes.js +15 -0
  108. package/src/routes/notifications.routes.js +12 -0
  109. package/src/routes/org.routes.js +31 -0
  110. package/src/routes/orgAdmin.routes.js +20 -0
  111. package/src/routes/publicAssets.routes.js +7 -0
  112. package/src/routes/stripeAdmin.routes.js +20 -0
  113. package/src/routes/user.routes.js +22 -0
  114. package/src/routes/userAdmin.routes.js +15 -0
  115. package/src/routes/waitingList.routes.js +13 -0
  116. package/src/routes/waitingListAdmin.routes.js +9 -0
  117. package/src/routes/webhook.routes.js +32 -0
  118. package/src/routes/workflowWebhook.routes.js +54 -0
  119. package/src/routes/workflows.routes.js +110 -0
  120. package/src/services/assets.service.js +110 -0
  121. package/src/services/audit.service.js +62 -0
  122. package/src/services/auditLogger.js +165 -0
  123. package/src/services/ejsVirtual.service.js +614 -0
  124. package/src/services/email.service.js +351 -0
  125. package/src/services/errorLogger.js +221 -0
  126. package/src/services/featureFlags.service.js +202 -0
  127. package/src/services/forms.service.js +214 -0
  128. package/src/services/globalSettings.service.js +49 -0
  129. package/src/services/headlessApiTokens.service.js +158 -0
  130. package/src/services/headlessCrypto.service.js +31 -0
  131. package/src/services/headlessModels.service.js +356 -0
  132. package/src/services/i18n.service.js +314 -0
  133. package/src/services/i18nInferredKeys.service.js +337 -0
  134. package/src/services/jsonConfigs.service.js +392 -0
  135. package/src/services/llm.service.js +749 -0
  136. package/src/services/migration.service.js +581 -0
  137. package/src/services/migrationAssets/fsLocal.js +58 -0
  138. package/src/services/migrationAssets/index.js +134 -0
  139. package/src/services/migrationAssets/s3.js +75 -0
  140. package/src/services/migrationAssets/sftp.js +92 -0
  141. package/src/services/notification.service.js +212 -0
  142. package/src/services/objectStorage.service.js +514 -0
  143. package/src/services/seoConfig.service.js +402 -0
  144. package/src/services/storage.js +150 -0
  145. package/src/services/stripe.service.js +185 -0
  146. package/src/services/stripeHelper.service.js +264 -0
  147. package/src/services/uploadNamespaces.service.js +326 -0
  148. package/src/services/webhook.service.js +157 -0
  149. package/src/services/workflow.service.js +271 -0
  150. package/src/utils/asyncHandler.js +5 -0
  151. package/src/utils/encryption.js +80 -0
  152. package/src/utils/jwt.js +40 -0
  153. package/src/utils/orgRoles.js +156 -0
  154. package/src/utils/validation.js +26 -0
  155. package/src/utils/webhookRetry.js +93 -0
  156. package/views/admin-assets.ejs +444 -0
  157. package/views/admin-audit.ejs +283 -0
  158. package/views/admin-coolify-deploy.ejs +207 -0
  159. package/views/admin-dashboard-home.ejs +291 -0
  160. package/views/admin-dashboard.ejs +397 -0
  161. package/views/admin-ejs-virtual.ejs +280 -0
  162. package/views/admin-errors.ejs +368 -0
  163. package/views/admin-feature-flags.ejs +390 -0
  164. package/views/admin-forms.ejs +526 -0
  165. package/views/admin-global-settings.ejs +436 -0
  166. package/views/admin-headless.ejs +2020 -0
  167. package/views/admin-i18n-locales.ejs +221 -0
  168. package/views/admin-i18n.ejs +728 -0
  169. package/views/admin-json-configs.ejs +410 -0
  170. package/views/admin-llm.ejs +884 -0
  171. package/views/admin-metrics.ejs +274 -0
  172. package/views/admin-migration.ejs +814 -0
  173. package/views/admin-notifications.ejs +430 -0
  174. package/views/admin-organizations.ejs +984 -0
  175. package/views/admin-seo-config.ejs +673 -0
  176. package/views/admin-stripe-pricing.ejs +558 -0
  177. package/views/admin-test.ejs +342 -0
  178. package/views/admin-users.ejs +452 -0
  179. package/views/admin-waiting-list.ejs +547 -0
  180. package/views/admin-webhooks.ejs +329 -0
  181. package/views/admin-workflows.ejs +310 -0
  182. package/views/partials/admin-assets-script.ejs +2022 -0
  183. package/views/partials/admin-test-sidebar.ejs +14 -0
  184. package/views/partials/dashboard/nav-items.ejs +66 -0
  185. package/views/partials/dashboard/palette.ejs +63 -0
  186. package/views/partials/dashboard/sidebar.ejs +21 -0
  187. package/views/partials/dashboard/tab-bar.ejs +26 -0
  188. package/views/partials/footer.ejs +3 -0
@@ -0,0 +1,12 @@
1
+ const mongoose = require('mongoose');
2
+
3
+ const workflowExecutionSchema = new mongoose.Schema({
4
+ workflowId: { type: mongoose.Schema.Types.ObjectId, ref: 'Workflow', required: true, index: true },
5
+ status: { type: String, enum: ['running', 'completed', 'failed'], default: 'running' },
6
+ context: { type: Object, default: {} },
7
+ log: { type: Array, default: [] },
8
+ executedAt: { type: Date, default: Date.now },
9
+ duration: Number
10
+ });
11
+
12
+ module.exports = mongoose.model('WorkflowExecution', workflowExecutionSchema);
@@ -0,0 +1,26 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const adminController = require('../controllers/admin.controller');
4
+ const { basicAuth } = require('../middleware/auth');
5
+
6
+ // All admin routes protected by basic auth
7
+ router.use(basicAuth);
8
+
9
+ router.get('/users', adminController.getUsers);
10
+ router.get('/users/:id', adminController.getUser);
11
+ router.put('/users/:id/subscription', adminController.updateUserSubscription);
12
+ router.patch('/users/:id', adminController.updateUserPassword);
13
+ router.post('/users/:id/reconcile', adminController.reconcileUser);
14
+ router.post('/generate-token', adminController.generateToken);
15
+
16
+ // Coolify Headless Deploy
17
+ router.post('/coolify-headless-deploy/provision', adminController.provisionCoolifyDeploy);
18
+
19
+ // Webhook event routes
20
+ router.get('/stripe-webhooks', adminController.getWebhookEvents);
21
+ router.get('/stripe-webhooks/:id', adminController.getWebhookEvent);
22
+ router.post('/stripe-webhooks/retry', adminController.retryFailedWebhookEvents);
23
+ router.post('/stripe-webhooks/:id/retry', adminController.retrySingleWebhookEvent);
24
+ router.get('/stripe-webhooks-stats', adminController.getWebhookStats);
25
+
26
+ module.exports = router;
@@ -0,0 +1,28 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const multer = require('multer');
4
+ const { basicAuth } = require('../middleware/auth');
5
+ const adminAssetsController = require('../controllers/adminAssets.controller');
6
+ const { auditMiddleware } = require('../services/auditLogger');
7
+
8
+ const adminAssetsStorageRoutes = require('./adminAssetsStorage.routes');
9
+
10
+ const upload = multer({
11
+ storage: multer.memoryStorage(),
12
+ limits: {
13
+ fileSize: parseInt(process.env.MULTER_FILE_SIZE_LIMIT || '1073741824', 10)
14
+ }
15
+ });
16
+
17
+ router.get('/info', basicAuth, adminAssetsController.getStorageInfo);
18
+ router.use('/storage', basicAuth, adminAssetsStorageRoutes);
19
+ router.get('/', basicAuth, adminAssetsController.list);
20
+ router.get('/:id', basicAuth, adminAssetsController.get);
21
+ router.post('/bulk/move-namespace', basicAuth, auditMiddleware('admin.assets.bulk.moveNamespace', { entityType: 'Asset' }), adminAssetsController.bulkMoveNamespace);
22
+ router.post('/bulk/set-tags', basicAuth, auditMiddleware('admin.assets.bulk.setTags', { entityType: 'Asset' }), adminAssetsController.bulkSetTags);
23
+ router.post('/upload', basicAuth, upload.single('file'), adminAssetsController.upload);
24
+ router.post('/:id/replace', basicAuth, upload.single('file'), adminAssetsController.replace);
25
+ router.patch('/:id', basicAuth, adminAssetsController.update);
26
+ router.delete('/:id', basicAuth, adminAssetsController.delete);
27
+
28
+ module.exports = router;
@@ -0,0 +1,13 @@
1
+ const express = require('express');
2
+
3
+ const router = express.Router();
4
+
5
+ const adminAssetsStorageController = require('../controllers/adminAssetsStorage.controller');
6
+
7
+ router.get('/', adminAssetsStorageController.getStorageStatus);
8
+ router.put('/s3-config', adminAssetsStorageController.saveS3Config);
9
+ router.post('/s3-check', adminAssetsStorageController.checkS3Connection);
10
+ router.post('/sync', adminAssetsStorageController.sync);
11
+ router.post('/switch', adminAssetsStorageController.switchBackend);
12
+
13
+ module.exports = router;
@@ -0,0 +1,196 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const mongoose = require('mongoose');
4
+ const AuditEvent = require('../models/AuditEvent');
5
+
6
+ router.get('/', async (req, res) => {
7
+ try {
8
+ const {
9
+ actorType,
10
+ actorUserId,
11
+ action,
12
+ outcome,
13
+ targetType,
14
+ targetId,
15
+ q,
16
+ from,
17
+ to,
18
+ page = 1,
19
+ pageSize = 50,
20
+ } = req.query;
21
+
22
+ const filter = {};
23
+
24
+ if (actorType && ['user', 'admin_basic', 'system', 'anonymous'].includes(actorType)) {
25
+ filter.actorType = actorType;
26
+ }
27
+
28
+ if (actorUserId) {
29
+ if (!mongoose.Types.ObjectId.isValid(String(actorUserId))) {
30
+ return res.status(400).json({ error: 'Invalid actorUserId' });
31
+ }
32
+ filter.actorUserId = String(actorUserId);
33
+ }
34
+
35
+ if (action) {
36
+ filter.action = { $regex: action, $options: 'i' };
37
+ }
38
+
39
+ if (outcome && ['success', 'failure'].includes(outcome)) {
40
+ filter.outcome = outcome;
41
+ }
42
+
43
+ if (targetType) {
44
+ filter.targetType = String(targetType);
45
+ }
46
+
47
+ if (targetId) {
48
+ filter.targetId = String(targetId);
49
+ }
50
+
51
+ if (q) {
52
+ filter.$or = [
53
+ { action: { $regex: q, $options: 'i' } },
54
+ { entityType: { $regex: q, $options: 'i' } },
55
+ { targetType: { $regex: q, $options: 'i' } },
56
+ { 'context.path': { $regex: q, $options: 'i' } },
57
+ ];
58
+ }
59
+
60
+ const dateFilter = {};
61
+ if (from) {
62
+ const fromDate = new Date(from);
63
+ if (!isNaN(fromDate.getTime())) {
64
+ dateFilter.$gte = fromDate;
65
+ }
66
+ }
67
+ if (to) {
68
+ const toDate = new Date(to);
69
+ if (!isNaN(toDate.getTime())) {
70
+ dateFilter.$lte = toDate;
71
+ }
72
+ }
73
+ if (Object.keys(dateFilter).length > 0) {
74
+ filter.createdAt = dateFilter;
75
+ }
76
+
77
+ const skip = (Math.max(1, parseInt(page, 10)) - 1) * parseInt(pageSize, 10);
78
+ const limit = Math.min(100, Math.max(1, parseInt(pageSize, 10)));
79
+
80
+ const [events, total] = await Promise.all([
81
+ AuditEvent.find(filter)
82
+ .populate('actorUserId', 'email')
83
+ .sort({ createdAt: -1 })
84
+ .skip(skip)
85
+ .limit(limit)
86
+ .lean(),
87
+ AuditEvent.countDocuments(filter),
88
+ ]);
89
+
90
+ const normalizedEvents = (events || []).map((evt) => {
91
+ const normalizedActorType = evt.actorType === 'admin' ? 'admin_basic' : evt.actorType;
92
+ return {
93
+ ...evt,
94
+ actorType: normalizedActorType,
95
+ at: evt.createdAt,
96
+ details: evt.details || evt.meta,
97
+ };
98
+ });
99
+
100
+ res.json({
101
+ events: normalizedEvents,
102
+ total,
103
+ page: parseInt(page, 10),
104
+ pageSize: limit,
105
+ totalPages: Math.ceil(total / limit),
106
+ });
107
+ } catch (err) {
108
+ console.error('[AdminAudit] Failed to list audit events:', err);
109
+ res.status(500).json({ error: 'Failed to list audit events' });
110
+ }
111
+ });
112
+
113
+ router.get('/stats', async (req, res) => {
114
+ try {
115
+ const now = new Date();
116
+ const last24h = new Date(now.getTime() - 24 * 60 * 60 * 1000);
117
+ const last7d = new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000);
118
+
119
+ const last24hDateFilter = { createdAt: { $gte: last24h } };
120
+ const last7dDateFilter = { createdAt: { $gte: last7d } };
121
+
122
+ const [total, last24hCount, last7dCount, failures24h, byActorType, topActions] = await Promise.all([
123
+ AuditEvent.countDocuments({}),
124
+ AuditEvent.countDocuments(last24hDateFilter),
125
+ AuditEvent.countDocuments(last7dDateFilter),
126
+ AuditEvent.countDocuments({
127
+ ...last24hDateFilter,
128
+ outcome: 'failure',
129
+ }),
130
+ AuditEvent.aggregate([
131
+ { $match: last24hDateFilter },
132
+ { $group: { _id: '$actorType', count: { $sum: 1 } } },
133
+ ]),
134
+ AuditEvent.aggregate([
135
+ { $match: last24hDateFilter },
136
+ { $group: { _id: '$action', count: { $sum: 1 } } },
137
+ { $sort: { count: -1 } },
138
+ { $limit: 10 },
139
+ ]),
140
+ ]);
141
+
142
+ res.json({
143
+ total,
144
+ last24h: last24hCount,
145
+ last7d: last7dCount,
146
+ failures24h,
147
+ byActorType: byActorType.reduce((acc, a) => {
148
+ acc[a._id] = a.count;
149
+ return acc;
150
+ }, {}),
151
+ topActions: topActions.map((a) => ({ action: a._id, count: a.count })),
152
+ });
153
+ } catch (err) {
154
+ console.error('[AdminAudit] Failed to get stats:', err);
155
+ res.status(500).json({ error: 'Failed to get stats' });
156
+ }
157
+ });
158
+
159
+ router.get('/actions', async (req, res) => {
160
+ try {
161
+ const actions = await AuditEvent.distinct('action');
162
+ res.json({ actions: actions.sort() });
163
+ } catch (err) {
164
+ console.error('[AdminAudit] Failed to get actions:', err);
165
+ res.status(500).json({ error: 'Failed to get actions' });
166
+ }
167
+ });
168
+
169
+ router.get('/:id', async (req, res) => {
170
+ try {
171
+ if (!mongoose.Types.ObjectId.isValid(req.params.id)) {
172
+ return res.status(400).json({ error: 'Invalid id' });
173
+ }
174
+
175
+ const event = await AuditEvent.findById(req.params.id)
176
+ .populate('actorUserId', 'email')
177
+ .lean();
178
+ if (!event) {
179
+ return res.status(404).json({ error: 'Audit event not found' });
180
+ }
181
+
182
+ const normalizedEvent = {
183
+ ...event,
184
+ actorType: event.actorType === 'admin' ? 'admin_basic' : event.actorType,
185
+ at: event.createdAt,
186
+ details: event.details || event.meta,
187
+ };
188
+
189
+ res.json(normalizedEvent);
190
+ } catch (err) {
191
+ console.error('[AdminAudit] Failed to get audit event:', err);
192
+ res.status(500).json({ error: 'Failed to get audit event' });
193
+ }
194
+ });
195
+
196
+ module.exports = router;
@@ -0,0 +1,17 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const { basicAuth } = require('../middleware/auth');
4
+ const controller = require('../controllers/adminEjsVirtual.controller');
5
+
6
+ router.use(basicAuth);
7
+
8
+ router.get('/files', controller.list);
9
+ router.get('/file', controller.getFile);
10
+ router.put('/file', controller.saveFile);
11
+ router.post('/file/revert', controller.revertToDefault);
12
+ router.get('/history', controller.listHistory);
13
+ router.post('/rollback', controller.rollback);
14
+ router.post('/vibe', controller.vibe);
15
+ router.post('/cache/clear', controller.clearCache);
16
+
17
+ module.exports = router;
@@ -0,0 +1,164 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const ErrorAggregate = require('../models/ErrorAggregate');
4
+
5
+ router.get('/', async (req, res) => {
6
+ try {
7
+ const {
8
+ source,
9
+ severity,
10
+ status,
11
+ q,
12
+ sort = 'lastSeenAt',
13
+ order = 'desc',
14
+ page = 1,
15
+ pageSize = 20,
16
+ since,
17
+ } = req.query;
18
+
19
+ const filter = {};
20
+
21
+ if (source && ['frontend', 'backend'].includes(source)) {
22
+ filter.source = source;
23
+ }
24
+ if (severity && ['fatal', 'error', 'warn', 'info'].includes(severity)) {
25
+ filter.severity = severity;
26
+ }
27
+ if (status && ['open', 'ignored', 'resolved'].includes(status)) {
28
+ filter.status = status;
29
+ }
30
+ if (q) {
31
+ filter.$or = [
32
+ { messageTemplate: { $regex: q, $options: 'i' } },
33
+ { errorName: { $regex: q, $options: 'i' } },
34
+ ];
35
+ }
36
+ if (since) {
37
+ const sinceDate = new Date(since);
38
+ if (!isNaN(sinceDate.getTime())) {
39
+ filter.lastSeenAt = { $gte: sinceDate };
40
+ }
41
+ }
42
+
43
+ const sortField = ['lastSeenAt', 'countTotal', 'firstSeenAt', 'errorName'].includes(sort) ? sort : 'lastSeenAt';
44
+ const sortOrder = order === 'asc' ? 1 : -1;
45
+
46
+ const skip = (Math.max(1, parseInt(page, 10)) - 1) * parseInt(pageSize, 10);
47
+ const limit = Math.min(100, Math.max(1, parseInt(pageSize, 10)));
48
+
49
+ const [errors, total] = await Promise.all([
50
+ ErrorAggregate.find(filter)
51
+ .select('-samples')
52
+ .sort({ [sortField]: sortOrder })
53
+ .skip(skip)
54
+ .limit(limit)
55
+ .lean(),
56
+ ErrorAggregate.countDocuments(filter),
57
+ ]);
58
+
59
+ res.json({
60
+ errors,
61
+ total,
62
+ page: parseInt(page, 10),
63
+ pageSize: limit,
64
+ totalPages: Math.ceil(total / limit),
65
+ });
66
+ } catch (err) {
67
+ console.error('[AdminErrors] Failed to list errors:', err);
68
+ res.status(500).json({ error: 'Failed to list errors' });
69
+ }
70
+ });
71
+
72
+ router.get('/stats', async (req, res) => {
73
+ try {
74
+ const now = new Date();
75
+ const last24h = new Date(now.getTime() - 24 * 60 * 60 * 1000);
76
+ const last7d = new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000);
77
+
78
+ const [total, open, last24hCount, last7dCount, bySource, bySeverity] = await Promise.all([
79
+ ErrorAggregate.countDocuments({}),
80
+ ErrorAggregate.countDocuments({ status: 'open' }),
81
+ ErrorAggregate.countDocuments({ lastSeenAt: { $gte: last24h } }),
82
+ ErrorAggregate.countDocuments({ lastSeenAt: { $gte: last7d } }),
83
+ ErrorAggregate.aggregate([
84
+ { $group: { _id: '$source', count: { $sum: 1 }, totalOccurrences: { $sum: '$countTotal' } } },
85
+ ]),
86
+ ErrorAggregate.aggregate([
87
+ { $group: { _id: '$severity', count: { $sum: 1 }, totalOccurrences: { $sum: '$countTotal' } } },
88
+ ]),
89
+ ]);
90
+
91
+ res.json({
92
+ total,
93
+ open,
94
+ last24h: last24hCount,
95
+ last7d: last7dCount,
96
+ bySource: bySource.reduce((acc, s) => {
97
+ acc[s._id] = { count: s.count, totalOccurrences: s.totalOccurrences };
98
+ return acc;
99
+ }, {}),
100
+ bySeverity: bySeverity.reduce((acc, s) => {
101
+ acc[s._id] = { count: s.count, totalOccurrences: s.totalOccurrences };
102
+ return acc;
103
+ }, {}),
104
+ });
105
+ } catch (err) {
106
+ console.error('[AdminErrors] Failed to get stats:', err);
107
+ res.status(500).json({ error: 'Failed to get stats' });
108
+ }
109
+ });
110
+
111
+ router.get('/:id', async (req, res) => {
112
+ try {
113
+ const error = await ErrorAggregate.findById(req.params.id).lean();
114
+ if (!error) {
115
+ return res.status(404).json({ error: 'Error not found' });
116
+ }
117
+ res.json(error);
118
+ } catch (err) {
119
+ console.error('[AdminErrors] Failed to get error:', err);
120
+ res.status(500).json({ error: 'Failed to get error' });
121
+ }
122
+ });
123
+
124
+ router.put('/:id/status', async (req, res) => {
125
+ try {
126
+ const { status } = req.body;
127
+ if (!['open', 'ignored', 'resolved'].includes(status)) {
128
+ return res.status(400).json({ error: 'Invalid status' });
129
+ }
130
+
131
+ const update = { status };
132
+ if (status === 'resolved') {
133
+ update.resolvedAt = new Date();
134
+ } else {
135
+ update.resolvedAt = null;
136
+ }
137
+
138
+ const error = await ErrorAggregate.findByIdAndUpdate(req.params.id, { $set: update }, { new: true }).lean();
139
+
140
+ if (!error) {
141
+ return res.status(404).json({ error: 'Error not found' });
142
+ }
143
+
144
+ res.json(error);
145
+ } catch (err) {
146
+ console.error('[AdminErrors] Failed to update error status:', err);
147
+ res.status(500).json({ error: 'Failed to update error status' });
148
+ }
149
+ });
150
+
151
+ router.delete('/:id', async (req, res) => {
152
+ try {
153
+ const result = await ErrorAggregate.findByIdAndDelete(req.params.id);
154
+ if (!result) {
155
+ return res.status(404).json({ error: 'Error not found' });
156
+ }
157
+ res.json({ message: 'Error deleted' });
158
+ } catch (err) {
159
+ console.error('[AdminErrors] Failed to delete error:', err);
160
+ res.status(500).json({ error: 'Failed to delete error' });
161
+ }
162
+ });
163
+
164
+ module.exports = router;
@@ -0,0 +1,12 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const { basicAuth } = require('../middleware/auth');
4
+ const adminFeatureFlagsController = require('../controllers/adminFeatureFlags.controller');
5
+
6
+ router.get('/', basicAuth, adminFeatureFlagsController.listFlags);
7
+ router.get('/:key', basicAuth, adminFeatureFlagsController.getFlag);
8
+ router.post('/', basicAuth, adminFeatureFlagsController.createFlag);
9
+ router.put('/:key', basicAuth, adminFeatureFlagsController.updateFlag);
10
+ router.delete('/:key', basicAuth, adminFeatureFlagsController.deleteFlag);
11
+
12
+ module.exports = router;
@@ -0,0 +1,38 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+
4
+ const { basicAuth } = require('../middleware/auth');
5
+ const adminHeadlessController = require('../controllers/adminHeadless.controller');
6
+
7
+ router.use(basicAuth);
8
+
9
+ // Models
10
+ router.get('/models', adminHeadlessController.listModels);
11
+ router.get('/models/:codeIdentifier', adminHeadlessController.getModel);
12
+ router.post('/models', adminHeadlessController.createModel);
13
+ router.put('/models/:codeIdentifier', adminHeadlessController.updateModel);
14
+ router.delete('/models/:codeIdentifier', adminHeadlessController.deleteModel);
15
+
16
+ // Advanced JSON / bulk helpers
17
+ router.post('/models/validate', adminHeadlessController.validateModelDefinition);
18
+ router.post('/models/apply', adminHeadlessController.applyModelProposal);
19
+
20
+ // AI model builder
21
+ router.post('/ai/model-builder/chat', adminHeadlessController.aiModelBuilderChat);
22
+
23
+ // Admin collections CRUD (UI)
24
+ router.get('/collections/:modelCode', adminHeadlessController.listCollectionItems);
25
+ router.post('/collections/:modelCode', adminHeadlessController.createCollectionItem);
26
+ router.put('/collections/:modelCode/:id', adminHeadlessController.updateCollectionItem);
27
+ router.delete('/collections/:modelCode/:id', adminHeadlessController.deleteCollectionItem);
28
+
29
+ router.post('/collections-api-test', adminHeadlessController.executeCollectionsApiTest);
30
+
31
+ // API tokens
32
+ router.get('/tokens', adminHeadlessController.listTokens);
33
+ router.get('/tokens/:id', adminHeadlessController.getToken);
34
+ router.post('/tokens', adminHeadlessController.createToken);
35
+ router.put('/tokens/:id', adminHeadlessController.updateToken);
36
+ router.delete('/tokens/:id', adminHeadlessController.deleteToken);
37
+
38
+ module.exports = router;
@@ -0,0 +1,22 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const { basicAuth } = require('../middleware/auth');
4
+
5
+ const adminI18nController = require('../controllers/adminI18n.controller');
6
+
7
+ router.use(basicAuth);
8
+
9
+ router.get('/locales', adminI18nController.listLocales);
10
+ router.post('/locales', adminI18nController.createLocale);
11
+ router.put('/locales/:code', adminI18nController.updateLocale);
12
+
13
+ router.get('/entries', adminI18nController.listEntries);
14
+ router.post('/entries', adminI18nController.createEntry);
15
+ router.put('/entries/:id', adminI18nController.updateEntry);
16
+ router.delete('/entries/:id', adminI18nController.deleteEntry);
17
+
18
+ router.post('/ai/preview', adminI18nController.aiPreview);
19
+ router.post('/ai/apply', adminI18nController.aiApply);
20
+ router.post('/ai/translate-text', adminI18nController.aiTranslateText);
21
+
22
+ module.exports = router;
@@ -0,0 +1,15 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+ const { basicAuth } = require('../middleware/auth');
4
+
5
+ const adminJsonConfigsController = require('../controllers/adminJsonConfigs.controller');
6
+
7
+ router.get('/', basicAuth, adminJsonConfigsController.list);
8
+ router.get('/:id', basicAuth, adminJsonConfigsController.get);
9
+ router.post('/', basicAuth, adminJsonConfigsController.create);
10
+ router.put('/:id', basicAuth, adminJsonConfigsController.update);
11
+ router.post('/:id/regenerate-slug', basicAuth, adminJsonConfigsController.regenerateSlug);
12
+ router.post('/:id/clear-cache', basicAuth, adminJsonConfigsController.clearCache);
13
+ router.delete('/:id', basicAuth, adminJsonConfigsController.remove);
14
+
15
+ module.exports = router;
@@ -0,0 +1,12 @@
1
+ const express = require("express");
2
+ const router = express.Router();
3
+ const { basicAuth } = require("../middleware/auth");
4
+ const adminLlmController = require("../controllers/adminLlm.controller");
5
+
6
+ router.get("/config", basicAuth, adminLlmController.getConfig);
7
+ router.post("/config", basicAuth, adminLlmController.saveConfig);
8
+ router.post("/prompts/:key/test", basicAuth, adminLlmController.testPrompt);
9
+ router.get("/audit", basicAuth, adminLlmController.listAudit);
10
+ router.get("/costs", basicAuth, adminLlmController.listCosts);
11
+
12
+ module.exports = router;
@@ -0,0 +1,81 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+
4
+ const { basicAuth } = require('../middleware/auth');
5
+ const { auditMiddleware } = require('../services/auditLogger');
6
+
7
+ const adminMigrationController = require('../controllers/adminMigration.controller');
8
+
9
+ router.get(
10
+ '/environments',
11
+ basicAuth,
12
+ adminMigrationController.listEnvironments,
13
+ );
14
+
15
+ router.get(
16
+ '/environments/:envKey',
17
+ basicAuth,
18
+ adminMigrationController.getEnvironment,
19
+ );
20
+
21
+ router.get(
22
+ '/models',
23
+ basicAuth,
24
+ adminMigrationController.listModels,
25
+ );
26
+
27
+ router.get(
28
+ '/models/:modelName/schema',
29
+ basicAuth,
30
+ adminMigrationController.getModelSchema,
31
+ );
32
+
33
+ router.post(
34
+ '/preview',
35
+ basicAuth,
36
+ adminMigrationController.preview,
37
+ );
38
+
39
+ router.post(
40
+ '/environments',
41
+ basicAuth,
42
+ auditMiddleware('admin.migration.environments.upsert', { entityType: 'GlobalSetting' }),
43
+ adminMigrationController.upsertEnvironment,
44
+ );
45
+
46
+ router.delete(
47
+ '/environments/:envKey',
48
+ basicAuth,
49
+ auditMiddleware('admin.migration.environments.delete', { entityType: 'GlobalSetting' }),
50
+ adminMigrationController.deleteEnvironment,
51
+ );
52
+
53
+ router.post(
54
+ '/test-connection',
55
+ basicAuth,
56
+ auditMiddleware('admin.migration.test_connection', { entityType: 'Migration' }),
57
+ adminMigrationController.testConnection,
58
+ );
59
+
60
+ router.post(
61
+ '/test-assets',
62
+ basicAuth,
63
+ auditMiddleware('admin.migration.test_assets', { entityType: 'Migration' }),
64
+ adminMigrationController.testAssetsTarget,
65
+ );
66
+
67
+ router.post(
68
+ '/test-assets-copy',
69
+ basicAuth,
70
+ auditMiddleware('admin.migration.test_assets_copy', { entityType: 'Migration' }),
71
+ adminMigrationController.testAssetsCopyKey,
72
+ );
73
+
74
+ router.post(
75
+ '/run',
76
+ basicAuth,
77
+ auditMiddleware('admin.migration.run', { entityType: 'Migration' }),
78
+ adminMigrationController.runMigration,
79
+ );
80
+
81
+ module.exports = router;
@@ -0,0 +1,20 @@
1
+ const express = require('express');
2
+ const router = express.Router();
3
+
4
+ const { basicAuth } = require('../middleware/auth');
5
+ const adminSeoConfigController = require('../controllers/adminSeoConfig.controller');
6
+
7
+ router.get('/', basicAuth, adminSeoConfigController.get);
8
+ router.put('/', basicAuth, adminSeoConfigController.update);
9
+
10
+ // SEO Config helpers
11
+ router.get('/ai/views', basicAuth, adminSeoConfigController.seoConfigAiListViews);
12
+ router.post('/ai/generate-entry', basicAuth, adminSeoConfigController.seoConfigAiGenerateEntry);
13
+ router.post('/ai/improve-entry', basicAuth, adminSeoConfigController.seoConfigAiImproveEntry);
14
+ router.post('/pages/apply-entry', basicAuth, adminSeoConfigController.seoConfigApplyEntry);
15
+
16
+ router.put('/og/svg', basicAuth, adminSeoConfigController.updateOgSvg);
17
+ router.post('/og/generate-png', basicAuth, adminSeoConfigController.generateOgPng);
18
+ router.post('/ai/edit-svg', basicAuth, adminSeoConfigController.aiEditSvg);
19
+
20
+ module.exports = router;