@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.
- package/.commiat +4 -0
- package/.env.example +47 -0
- package/README.md +110 -0
- package/index.js +94 -0
- package/package.json +67 -0
- package/public/css/styles.css +139 -0
- package/public/js/animations.js +41 -0
- package/sdk/error-tracking/browser/package.json +16 -0
- package/sdk/error-tracking/browser/src/core.js +270 -0
- package/sdk/error-tracking/browser/src/embed.js +18 -0
- package/sdk/error-tracking/browser/src/index.js +1 -0
- package/server.js +5 -0
- package/src/admin/endpointRegistry.js +300 -0
- package/src/controllers/admin.controller.js +321 -0
- package/src/controllers/adminAssets.controller.js +530 -0
- package/src/controllers/adminAssetsStorage.controller.js +260 -0
- package/src/controllers/adminEjsVirtual.controller.js +354 -0
- package/src/controllers/adminFeatureFlags.controller.js +155 -0
- package/src/controllers/adminHeadless.controller.js +1071 -0
- package/src/controllers/adminI18n.controller.js +604 -0
- package/src/controllers/adminJsonConfigs.controller.js +97 -0
- package/src/controllers/adminLlm.controller.js +273 -0
- package/src/controllers/adminMigration.controller.js +257 -0
- package/src/controllers/adminSeoConfig.controller.js +515 -0
- package/src/controllers/adminStats.controller.js +121 -0
- package/src/controllers/adminUploadNamespaces.controller.js +208 -0
- package/src/controllers/assets.controller.js +248 -0
- package/src/controllers/auth.controller.js +93 -0
- package/src/controllers/billing.controller.js +223 -0
- package/src/controllers/featureFlags.controller.js +35 -0
- package/src/controllers/forms.controller.js +217 -0
- package/src/controllers/globalSettings.controller.js +252 -0
- package/src/controllers/headlessCrud.controller.js +126 -0
- package/src/controllers/i18n.controller.js +12 -0
- package/src/controllers/invite.controller.js +249 -0
- package/src/controllers/jsonConfigs.controller.js +19 -0
- package/src/controllers/metrics.controller.js +149 -0
- package/src/controllers/notificationAdmin.controller.js +264 -0
- package/src/controllers/notifications.controller.js +131 -0
- package/src/controllers/org.controller.js +357 -0
- package/src/controllers/orgAdmin.controller.js +491 -0
- package/src/controllers/stripeAdmin.controller.js +410 -0
- package/src/controllers/user.controller.js +361 -0
- package/src/controllers/userAdmin.controller.js +277 -0
- package/src/controllers/waitingList.controller.js +167 -0
- package/src/controllers/webhook.controller.js +200 -0
- package/src/middleware/auth.js +66 -0
- package/src/middleware/errorCapture.js +170 -0
- package/src/middleware/headlessApiTokenAuth.js +57 -0
- package/src/middleware/org.js +108 -0
- package/src/middleware.js +901 -0
- package/src/models/ActionEvent.js +31 -0
- package/src/models/ActivityLog.js +41 -0
- package/src/models/Asset.js +84 -0
- package/src/models/AuditEvent.js +93 -0
- package/src/models/EmailLog.js +28 -0
- package/src/models/ErrorAggregate.js +72 -0
- package/src/models/FormSubmission.js +41 -0
- package/src/models/GlobalSetting.js +38 -0
- package/src/models/HeadlessApiToken.js +24 -0
- package/src/models/HeadlessModelDefinition.js +41 -0
- package/src/models/I18nEntry.js +77 -0
- package/src/models/I18nLocale.js +33 -0
- package/src/models/Invite.js +70 -0
- package/src/models/JsonConfig.js +46 -0
- package/src/models/Notification.js +60 -0
- package/src/models/Organization.js +57 -0
- package/src/models/OrganizationMember.js +43 -0
- package/src/models/StripeCatalogItem.js +77 -0
- package/src/models/StripeWebhookEvent.js +57 -0
- package/src/models/User.js +89 -0
- package/src/models/VirtualEjsFile.js +60 -0
- package/src/models/VirtualEjsFileVersion.js +43 -0
- package/src/models/VirtualEjsGroupChange.js +32 -0
- package/src/models/WaitingList.js +41 -0
- package/src/models/Webhook.js +63 -0
- package/src/models/Workflow.js +29 -0
- package/src/models/WorkflowExecution.js +12 -0
- package/src/routes/admin.routes.js +26 -0
- package/src/routes/adminAssets.routes.js +28 -0
- package/src/routes/adminAssetsStorage.routes.js +13 -0
- package/src/routes/adminAudit.routes.js +196 -0
- package/src/routes/adminEjsVirtual.routes.js +17 -0
- package/src/routes/adminErrors.routes.js +164 -0
- package/src/routes/adminFeatureFlags.routes.js +12 -0
- package/src/routes/adminHeadless.routes.js +38 -0
- package/src/routes/adminI18n.routes.js +22 -0
- package/src/routes/adminJsonConfigs.routes.js +15 -0
- package/src/routes/adminLlm.routes.js +12 -0
- package/src/routes/adminMigration.routes.js +81 -0
- package/src/routes/adminSeoConfig.routes.js +20 -0
- package/src/routes/adminUploadNamespaces.routes.js +13 -0
- package/src/routes/assets.routes.js +21 -0
- package/src/routes/auth.routes.js +12 -0
- package/src/routes/billing.routes.js +11 -0
- package/src/routes/errorTracking.routes.js +31 -0
- package/src/routes/featureFlags.routes.js +9 -0
- package/src/routes/forms.routes.js +9 -0
- package/src/routes/formsAdmin.routes.js +13 -0
- package/src/routes/globalSettings.routes.js +18 -0
- package/src/routes/headless.routes.js +15 -0
- package/src/routes/i18n.routes.js +8 -0
- package/src/routes/invite.routes.js +9 -0
- package/src/routes/jsonConfigs.routes.js +8 -0
- package/src/routes/log.routes.js +111 -0
- package/src/routes/metrics.routes.js +9 -0
- package/src/routes/notificationAdmin.routes.js +15 -0
- package/src/routes/notifications.routes.js +12 -0
- package/src/routes/org.routes.js +31 -0
- package/src/routes/orgAdmin.routes.js +20 -0
- package/src/routes/publicAssets.routes.js +7 -0
- package/src/routes/stripeAdmin.routes.js +20 -0
- package/src/routes/user.routes.js +22 -0
- package/src/routes/userAdmin.routes.js +15 -0
- package/src/routes/waitingList.routes.js +13 -0
- package/src/routes/waitingListAdmin.routes.js +9 -0
- package/src/routes/webhook.routes.js +32 -0
- package/src/routes/workflowWebhook.routes.js +54 -0
- package/src/routes/workflows.routes.js +110 -0
- package/src/services/assets.service.js +110 -0
- package/src/services/audit.service.js +62 -0
- package/src/services/auditLogger.js +165 -0
- package/src/services/ejsVirtual.service.js +614 -0
- package/src/services/email.service.js +351 -0
- package/src/services/errorLogger.js +221 -0
- package/src/services/featureFlags.service.js +202 -0
- package/src/services/forms.service.js +214 -0
- package/src/services/globalSettings.service.js +49 -0
- package/src/services/headlessApiTokens.service.js +158 -0
- package/src/services/headlessCrypto.service.js +31 -0
- package/src/services/headlessModels.service.js +356 -0
- package/src/services/i18n.service.js +314 -0
- package/src/services/i18nInferredKeys.service.js +337 -0
- package/src/services/jsonConfigs.service.js +392 -0
- package/src/services/llm.service.js +749 -0
- package/src/services/migration.service.js +581 -0
- package/src/services/migrationAssets/fsLocal.js +58 -0
- package/src/services/migrationAssets/index.js +134 -0
- package/src/services/migrationAssets/s3.js +75 -0
- package/src/services/migrationAssets/sftp.js +92 -0
- package/src/services/notification.service.js +212 -0
- package/src/services/objectStorage.service.js +514 -0
- package/src/services/seoConfig.service.js +402 -0
- package/src/services/storage.js +150 -0
- package/src/services/stripe.service.js +185 -0
- package/src/services/stripeHelper.service.js +264 -0
- package/src/services/uploadNamespaces.service.js +326 -0
- package/src/services/webhook.service.js +157 -0
- package/src/services/workflow.service.js +271 -0
- package/src/utils/asyncHandler.js +5 -0
- package/src/utils/encryption.js +80 -0
- package/src/utils/jwt.js +40 -0
- package/src/utils/orgRoles.js +156 -0
- package/src/utils/validation.js +26 -0
- package/src/utils/webhookRetry.js +93 -0
- package/views/admin-assets.ejs +444 -0
- package/views/admin-audit.ejs +283 -0
- package/views/admin-coolify-deploy.ejs +207 -0
- package/views/admin-dashboard-home.ejs +291 -0
- package/views/admin-dashboard.ejs +397 -0
- package/views/admin-ejs-virtual.ejs +280 -0
- package/views/admin-errors.ejs +368 -0
- package/views/admin-feature-flags.ejs +390 -0
- package/views/admin-forms.ejs +526 -0
- package/views/admin-global-settings.ejs +436 -0
- package/views/admin-headless.ejs +2020 -0
- package/views/admin-i18n-locales.ejs +221 -0
- package/views/admin-i18n.ejs +728 -0
- package/views/admin-json-configs.ejs +410 -0
- package/views/admin-llm.ejs +884 -0
- package/views/admin-metrics.ejs +274 -0
- package/views/admin-migration.ejs +814 -0
- package/views/admin-notifications.ejs +430 -0
- package/views/admin-organizations.ejs +984 -0
- package/views/admin-seo-config.ejs +673 -0
- package/views/admin-stripe-pricing.ejs +558 -0
- package/views/admin-test.ejs +342 -0
- package/views/admin-users.ejs +452 -0
- package/views/admin-waiting-list.ejs +547 -0
- package/views/admin-webhooks.ejs +329 -0
- package/views/admin-workflows.ejs +310 -0
- package/views/partials/admin-assets-script.ejs +2022 -0
- package/views/partials/admin-test-sidebar.ejs +14 -0
- package/views/partials/dashboard/nav-items.ejs +66 -0
- package/views/partials/dashboard/palette.ejs +63 -0
- package/views/partials/dashboard/sidebar.ejs +21 -0
- package/views/partials/dashboard/tab-bar.ejs +26 -0
- 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;
|