@rebasepro/server-core 0.0.1-canary.000dc36
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/LICENSE +6 -0
- package/README.md +40 -0
- package/build-errors.txt +52 -0
- package/coverage/clover.xml +3739 -0
- package/coverage/coverage-final.json +31 -0
- package/coverage/lcov-report/base.css +224 -0
- package/coverage/lcov-report/block-navigation.js +87 -0
- package/coverage/lcov-report/favicon.png +0 -0
- package/coverage/lcov-report/index.html +266 -0
- package/coverage/lcov-report/prettify.css +1 -0
- package/coverage/lcov-report/prettify.js +2 -0
- package/coverage/lcov-report/sort-arrow-sprite.png +0 -0
- package/coverage/lcov-report/sorter.js +210 -0
- package/coverage/lcov-report/src/api/ast-schema-editor.ts.html +952 -0
- package/coverage/lcov-report/src/api/errors.ts.html +472 -0
- package/coverage/lcov-report/src/api/graphql/graphql-schema-generator.ts.html +1069 -0
- package/coverage/lcov-report/src/api/graphql/index.html +116 -0
- package/coverage/lcov-report/src/api/index.html +176 -0
- package/coverage/lcov-report/src/api/openapi-generator.ts.html +565 -0
- package/coverage/lcov-report/src/api/rest/api-generator.ts.html +994 -0
- package/coverage/lcov-report/src/api/rest/index.html +131 -0
- package/coverage/lcov-report/src/api/rest/query-parser.ts.html +550 -0
- package/coverage/lcov-report/src/api/schema-editor-routes.ts.html +202 -0
- package/coverage/lcov-report/src/api/server.ts.html +823 -0
- package/coverage/lcov-report/src/auth/admin-routes.ts.html +973 -0
- package/coverage/lcov-report/src/auth/index.html +176 -0
- package/coverage/lcov-report/src/auth/jwt.ts.html +574 -0
- package/coverage/lcov-report/src/auth/middleware.ts.html +745 -0
- package/coverage/lcov-report/src/auth/password.ts.html +310 -0
- package/coverage/lcov-report/src/auth/services.ts.html +2074 -0
- package/coverage/lcov-report/src/collections/index.html +116 -0
- package/coverage/lcov-report/src/collections/loader.ts.html +232 -0
- package/coverage/lcov-report/src/db/auth-schema.ts.html +523 -0
- package/coverage/lcov-report/src/db/data-transformer.ts.html +1753 -0
- package/coverage/lcov-report/src/db/entityService.ts.html +700 -0
- package/coverage/lcov-report/src/db/index.html +146 -0
- package/coverage/lcov-report/src/db/services/EntityFetchService.ts.html +4048 -0
- package/coverage/lcov-report/src/db/services/EntityPersistService.ts.html +883 -0
- package/coverage/lcov-report/src/db/services/RelationService.ts.html +3121 -0
- package/coverage/lcov-report/src/db/services/entity-helpers.ts.html +442 -0
- package/coverage/lcov-report/src/db/services/index.html +176 -0
- package/coverage/lcov-report/src/db/services/index.ts.html +124 -0
- package/coverage/lcov-report/src/generate-drizzle-schema-logic.ts.html +1960 -0
- package/coverage/lcov-report/src/index.html +116 -0
- package/coverage/lcov-report/src/services/driver-registry.ts.html +631 -0
- package/coverage/lcov-report/src/services/index.html +131 -0
- package/coverage/lcov-report/src/services/postgresDataDriver.ts.html +3025 -0
- package/coverage/lcov-report/src/storage/LocalStorageController.ts.html +1189 -0
- package/coverage/lcov-report/src/storage/S3StorageController.ts.html +970 -0
- package/coverage/lcov-report/src/storage/index.html +161 -0
- package/coverage/lcov-report/src/storage/storage-registry.ts.html +646 -0
- package/coverage/lcov-report/src/storage/types.ts.html +451 -0
- package/coverage/lcov-report/src/utils/drizzle-conditions.ts.html +3082 -0
- package/coverage/lcov-report/src/utils/index.html +116 -0
- package/coverage/lcov.info +7179 -0
- package/dist/common/src/collections/CollectionRegistry.d.ts +56 -0
- package/dist/common/src/collections/index.d.ts +1 -0
- package/dist/common/src/data/buildRebaseData.d.ts +14 -0
- package/dist/common/src/index.d.ts +3 -0
- package/dist/common/src/util/builders.d.ts +57 -0
- package/dist/common/src/util/callbacks.d.ts +6 -0
- package/dist/common/src/util/collections.d.ts +11 -0
- package/dist/common/src/util/common.d.ts +2 -0
- package/dist/common/src/util/conditions.d.ts +26 -0
- package/dist/common/src/util/entities.d.ts +58 -0
- package/dist/common/src/util/enums.d.ts +3 -0
- package/dist/common/src/util/index.d.ts +16 -0
- package/dist/common/src/util/navigation_from_path.d.ts +34 -0
- package/dist/common/src/util/navigation_utils.d.ts +20 -0
- package/dist/common/src/util/parent_references_from_path.d.ts +6 -0
- package/dist/common/src/util/paths.d.ts +14 -0
- package/dist/common/src/util/permissions.d.ts +5 -0
- package/dist/common/src/util/references.d.ts +2 -0
- package/dist/common/src/util/relations.d.ts +22 -0
- package/dist/common/src/util/resolutions.d.ts +72 -0
- package/dist/common/src/util/storage.d.ts +24 -0
- package/dist/index-DXVBFp5V.js +37 -0
- package/dist/index-DXVBFp5V.js.map +1 -0
- package/dist/index.es.js +49249 -0
- package/dist/index.es.js.map +1 -0
- package/dist/index.umd.js +49283 -0
- package/dist/index.umd.js.map +1 -0
- package/dist/server-core/src/api/ast-schema-editor.d.ts +21 -0
- package/dist/server-core/src/api/collections_for_test/callbacks_test_collection.d.ts +2 -0
- package/dist/server-core/src/api/errors.d.ts +35 -0
- package/dist/server-core/src/api/graphql/graphql-schema-generator.d.ts +35 -0
- package/dist/server-core/src/api/graphql/index.d.ts +1 -0
- package/dist/server-core/src/api/index.d.ts +9 -0
- package/dist/server-core/src/api/openapi-generator.d.ts +16 -0
- package/dist/server-core/src/api/rest/api-generator.d.ts +76 -0
- package/dist/server-core/src/api/rest/index.d.ts +1 -0
- package/dist/server-core/src/api/rest/query-parser.d.ts +9 -0
- package/dist/server-core/src/api/schema-editor-routes.d.ts +3 -0
- package/dist/server-core/src/api/server.d.ts +40 -0
- package/dist/server-core/src/api/types.d.ts +90 -0
- package/dist/server-core/src/auth/admin-routes.d.ts +21 -0
- package/dist/server-core/src/auth/apple-oauth.d.ts +30 -0
- package/dist/server-core/src/auth/bitbucket-oauth.d.ts +11 -0
- package/dist/server-core/src/auth/discord-oauth.d.ts +14 -0
- package/dist/server-core/src/auth/facebook-oauth.d.ts +14 -0
- package/dist/server-core/src/auth/github-oauth.d.ts +15 -0
- package/dist/server-core/src/auth/gitlab-oauth.d.ts +13 -0
- package/dist/server-core/src/auth/google-oauth.d.ts +14 -0
- package/dist/server-core/src/auth/index.d.ts +23 -0
- package/dist/server-core/src/auth/interfaces.d.ts +309 -0
- package/dist/server-core/src/auth/jwt.d.ts +43 -0
- package/dist/server-core/src/auth/linkedin-oauth.d.ts +18 -0
- package/dist/server-core/src/auth/microsoft-oauth.d.ts +16 -0
- package/dist/server-core/src/auth/middleware.d.ts +81 -0
- package/dist/server-core/src/auth/password.d.ts +22 -0
- package/dist/server-core/src/auth/rate-limiter.d.ts +31 -0
- package/dist/server-core/src/auth/routes.d.ts +27 -0
- package/dist/server-core/src/auth/slack-oauth.d.ts +12 -0
- package/dist/server-core/src/auth/spotify-oauth.d.ts +12 -0
- package/dist/server-core/src/auth/twitter-oauth.d.ts +18 -0
- package/dist/server-core/src/bootstrappers/index.d.ts +0 -0
- package/dist/server-core/src/collections/BackendCollectionRegistry.d.ts +13 -0
- package/dist/server-core/src/collections/loader.d.ts +5 -0
- package/dist/server-core/src/cron/cron-loader.d.ts +17 -0
- package/dist/server-core/src/cron/cron-routes.d.ts +14 -0
- package/dist/server-core/src/cron/cron-scheduler.d.ts +106 -0
- package/dist/server-core/src/cron/cron-store.d.ts +32 -0
- package/dist/server-core/src/cron/index.d.ts +6 -0
- package/dist/server-core/src/db/interfaces.d.ts +18 -0
- package/dist/server-core/src/email/index.d.ts +6 -0
- package/dist/server-core/src/email/smtp-email-service.d.ts +25 -0
- package/dist/server-core/src/email/templates.d.ts +42 -0
- package/dist/server-core/src/email/types.d.ts +107 -0
- package/dist/server-core/src/functions/function-loader.d.ts +17 -0
- package/dist/server-core/src/functions/function-routes.d.ts +10 -0
- package/dist/server-core/src/functions/index.d.ts +3 -0
- package/dist/server-core/src/history/history-routes.d.ts +23 -0
- package/dist/server-core/src/history/index.d.ts +1 -0
- package/dist/server-core/src/index.d.ts +29 -0
- package/dist/server-core/src/init.d.ts +168 -0
- package/dist/server-core/src/serve-spa.d.ts +30 -0
- package/dist/server-core/src/services/driver-registry.d.ts +78 -0
- package/dist/server-core/src/singleton.d.ts +35 -0
- package/dist/server-core/src/storage/LocalStorageController.d.ts +46 -0
- package/dist/server-core/src/storage/S3StorageController.d.ts +36 -0
- package/dist/server-core/src/storage/index.d.ts +25 -0
- package/dist/server-core/src/storage/routes.d.ts +38 -0
- package/dist/server-core/src/storage/storage-registry.d.ts +78 -0
- package/dist/server-core/src/storage/types.d.ts +103 -0
- package/dist/server-core/src/types/index.d.ts +11 -0
- package/dist/server-core/src/utils/dev-port.d.ts +35 -0
- package/dist/server-core/src/utils/logger.d.ts +31 -0
- package/dist/server-core/src/utils/logging.d.ts +9 -0
- package/dist/server-core/src/utils/request-logger.d.ts +19 -0
- package/dist/server-core/src/utils/sql.d.ts +27 -0
- package/dist/types/src/controllers/analytics_controller.d.ts +7 -0
- package/dist/types/src/controllers/auth.d.ts +119 -0
- package/dist/types/src/controllers/client.d.ts +170 -0
- package/dist/types/src/controllers/collection_registry.d.ts +46 -0
- package/dist/types/src/controllers/customization_controller.d.ts +60 -0
- package/dist/types/src/controllers/data.d.ts +168 -0
- package/dist/types/src/controllers/data_driver.d.ts +195 -0
- package/dist/types/src/controllers/database_admin.d.ts +11 -0
- package/dist/types/src/controllers/dialogs_controller.d.ts +36 -0
- package/dist/types/src/controllers/effective_role.d.ts +4 -0
- package/dist/types/src/controllers/email.d.ts +34 -0
- package/dist/types/src/controllers/index.d.ts +18 -0
- package/dist/types/src/controllers/local_config_persistence.d.ts +20 -0
- package/dist/types/src/controllers/navigation.d.ts +213 -0
- package/dist/types/src/controllers/registry.d.ts +54 -0
- package/dist/types/src/controllers/side_dialogs_controller.d.ts +67 -0
- package/dist/types/src/controllers/side_entity_controller.d.ts +90 -0
- package/dist/types/src/controllers/snackbar.d.ts +24 -0
- package/dist/types/src/controllers/storage.d.ts +171 -0
- package/dist/types/src/index.d.ts +4 -0
- package/dist/types/src/rebase_context.d.ts +105 -0
- package/dist/types/src/types/backend.d.ts +536 -0
- package/dist/types/src/types/backend_hooks.d.ts +187 -0
- package/dist/types/src/types/builders.d.ts +15 -0
- package/dist/types/src/types/chips.d.ts +5 -0
- package/dist/types/src/types/collections.d.ts +857 -0
- package/dist/types/src/types/cron.d.ts +102 -0
- package/dist/types/src/types/data_source.d.ts +64 -0
- package/dist/types/src/types/entities.d.ts +145 -0
- package/dist/types/src/types/entity_actions.d.ts +98 -0
- package/dist/types/src/types/entity_callbacks.d.ts +173 -0
- package/dist/types/src/types/entity_link_builder.d.ts +7 -0
- package/dist/types/src/types/entity_overrides.d.ts +10 -0
- package/dist/types/src/types/entity_views.d.ts +59 -0
- package/dist/types/src/types/export_import.d.ts +21 -0
- package/dist/types/src/types/formex.d.ts +40 -0
- package/dist/types/src/types/index.d.ts +25 -0
- package/dist/types/src/types/locales.d.ts +4 -0
- package/dist/types/src/types/modify_collections.d.ts +5 -0
- package/dist/types/src/types/plugins.d.ts +282 -0
- package/dist/types/src/types/properties.d.ts +1148 -0
- package/dist/types/src/types/property_config.d.ts +70 -0
- package/dist/types/src/types/relations.d.ts +336 -0
- package/dist/types/src/types/slots.d.ts +262 -0
- package/dist/types/src/types/translations.d.ts +874 -0
- package/dist/types/src/types/user_management_delegate.d.ts +121 -0
- package/dist/types/src/types/websockets.d.ts +78 -0
- package/dist/types/src/users/index.d.ts +2 -0
- package/dist/types/src/users/roles.d.ts +22 -0
- package/dist/types/src/users/user.d.ts +46 -0
- package/history_diff.log +385 -0
- package/jest.config.cjs +16 -0
- package/package.json +86 -0
- package/scratch.ts +9 -0
- package/src/api/ast-schema-editor.ts +289 -0
- package/src/api/collections_for_test/callbacks_test_collection.ts +60 -0
- package/src/api/errors.ts +179 -0
- package/src/api/graphql/graphql-schema-generator.ts +336 -0
- package/src/api/graphql/index.ts +2 -0
- package/src/api/index.ts +11 -0
- package/src/api/openapi-generator.ts +715 -0
- package/src/api/rest/api-generator-count.test.ts +113 -0
- package/src/api/rest/api-generator.ts +573 -0
- package/src/api/rest/index.ts +2 -0
- package/src/api/rest/query-parser.ts +155 -0
- package/src/api/schema-editor-routes.ts +41 -0
- package/src/api/server.ts +249 -0
- package/src/api/types.ts +90 -0
- package/src/auth/admin-routes.ts +605 -0
- package/src/auth/apple-oauth.ts +120 -0
- package/src/auth/bitbucket-oauth.ts +82 -0
- package/src/auth/discord-oauth.ts +83 -0
- package/src/auth/facebook-oauth.ts +72 -0
- package/src/auth/github-oauth.ts +110 -0
- package/src/auth/gitlab-oauth.ts +70 -0
- package/src/auth/google-oauth.ts +48 -0
- package/src/auth/index.ts +34 -0
- package/src/auth/interfaces.ts +363 -0
- package/src/auth/jwt.ts +181 -0
- package/src/auth/linkedin-oauth.ts +81 -0
- package/src/auth/microsoft-oauth.ts +88 -0
- package/src/auth/middleware.ts +384 -0
- package/src/auth/password.ts +77 -0
- package/src/auth/rate-limiter.ts +133 -0
- package/src/auth/routes.ts +788 -0
- package/src/auth/slack-oauth.ts +71 -0
- package/src/auth/spotify-oauth.ts +67 -0
- package/src/auth/twitter-oauth.ts +120 -0
- package/src/bootstrappers/index.ts +1 -0
- package/src/collections/BackendCollectionRegistry.ts +20 -0
- package/src/collections/loader.ts +49 -0
- package/src/cron/cron-loader.ts +89 -0
- package/src/cron/cron-routes.test.ts +265 -0
- package/src/cron/cron-routes.ts +85 -0
- package/src/cron/cron-scheduler.test.ts +547 -0
- package/src/cron/cron-scheduler.ts +576 -0
- package/src/cron/cron-store.ts +163 -0
- package/src/cron/index.ts +6 -0
- package/src/db/interfaces.ts +60 -0
- package/src/email/index.ts +18 -0
- package/src/email/smtp-email-service.ts +91 -0
- package/src/email/templates.ts +388 -0
- package/src/email/types.ts +105 -0
- package/src/functions/function-loader.ts +119 -0
- package/src/functions/function-routes.ts +31 -0
- package/src/functions/index.ts +3 -0
- package/src/history/history-routes.ts +129 -0
- package/src/history/index.ts +2 -0
- package/src/index.ts +66 -0
- package/src/init.ts +737 -0
- package/src/serve-spa.ts +81 -0
- package/src/services/driver-registry.ts +182 -0
- package/src/singleton.test.ts +28 -0
- package/src/singleton.ts +70 -0
- package/src/storage/LocalStorageController.ts +365 -0
- package/src/storage/S3StorageController.ts +298 -0
- package/src/storage/index.ts +43 -0
- package/src/storage/routes.ts +264 -0
- package/src/storage/storage-registry.ts +187 -0
- package/src/storage/types.ts +134 -0
- package/src/types/index.ts +27 -0
- package/src/utils/dev-port.ts +176 -0
- package/src/utils/logger.ts +143 -0
- package/src/utils/logging.ts +38 -0
- package/src/utils/request-logger.ts +66 -0
- package/src/utils/sql.ts +38 -0
- package/test/admin-routes.test.ts +640 -0
- package/test/api-generator.test.ts +501 -0
- package/test/ast-schema-editor.test.ts +63 -0
- package/test/auth-middleware-hono.test.ts +556 -0
- package/test/auth-routes.test.ts +1047 -0
- package/test/backend-hooks-admin.test.ts +394 -0
- package/test/backend-hooks-data.test.ts +408 -0
- package/test/driver-registry.test.ts +282 -0
- package/test/error-propagation.test.ts +226 -0
- package/test/errors-hono.test.ts +133 -0
- package/test/errors.test.ts +155 -0
- package/test/jwt-security.test.ts +182 -0
- package/test/jwt.test.ts +324 -0
- package/test/middleware.test.ts +300 -0
- package/test/password.test.ts +165 -0
- package/test/query-parser.test.ts +263 -0
- package/test/rate-limiter.test.ts +102 -0
- package/test/safe-compare.test.ts +66 -0
- package/test/singleton.test.ts +59 -0
- package/test/storage-local.test.ts +271 -0
- package/test/storage-registry.test.ts +282 -0
- package/test/storage-routes.test.ts +222 -0
- package/test/storage-s3.test.ts +304 -0
- package/test-ast.ts +28 -0
- package/test.ts +6 -0
- package/test_output.txt +1133 -0
- package/tsconfig.json +49 -0
- package/tsconfig.prod.json +20 -0
- package/vite.config.ts +80 -0
|
@@ -0,0 +1,605 @@
|
|
|
1
|
+
import { Hono } from "hono";
|
|
2
|
+
import { ApiError, errorHandler } from "../api/errors";
|
|
3
|
+
import type { AuthRepository } from "./interfaces";
|
|
4
|
+
import { requireAuth, requireAdmin, createRequireAuth } from "./middleware";
|
|
5
|
+
import { hashPassword, validatePasswordStrength } from "./password";
|
|
6
|
+
import { AuthModuleConfig } from "./routes";
|
|
7
|
+
import type { BackendHooks, AdminUser, AdminRole, BackendHookContext } from "@rebasepro/types";
|
|
8
|
+
|
|
9
|
+
interface AdminRouteOptions extends AuthModuleConfig {
|
|
10
|
+
serviceKey?: string;
|
|
11
|
+
/**
|
|
12
|
+
* Callback to persistently mark bootstrap as completed.
|
|
13
|
+
* Invoked after the first admin user is promoted via POST /admin/bootstrap.
|
|
14
|
+
*/
|
|
15
|
+
setBootstrapCompleted?: () => Promise<void>;
|
|
16
|
+
/**
|
|
17
|
+
* Backend-level hooks for intercepting admin data.
|
|
18
|
+
*/
|
|
19
|
+
hooks?: BackendHooks;
|
|
20
|
+
}
|
|
21
|
+
import { HonoEnv } from "../api/types";
|
|
22
|
+
import { randomBytes, createHash } from "crypto";
|
|
23
|
+
import { getUserInvitationTemplate, getPasswordResetTemplate } from "../email/templates";
|
|
24
|
+
|
|
25
|
+
/**
|
|
26
|
+
* Generate a cryptographically secure random password that meets strength requirements.
|
|
27
|
+
*/
|
|
28
|
+
function generateSecurePassword(): string {
|
|
29
|
+
const upper = "ABCDEFGHJKLMNPQRSTUVWXYZ";
|
|
30
|
+
const lower = "abcdefghjkmnpqrstuvwxyz";
|
|
31
|
+
const digits = "23456789";
|
|
32
|
+
const all = upper + lower + digits;
|
|
33
|
+
|
|
34
|
+
// Guarantee at least one of each required class
|
|
35
|
+
const pick = (chars: string) => chars[Math.floor(Math.random() * chars.length)];
|
|
36
|
+
const parts = [pick(upper), pick(lower), pick(digits)];
|
|
37
|
+
|
|
38
|
+
// Fill remaining with random chars (16 total)
|
|
39
|
+
for (let i = parts.length; i < 16; i++) {
|
|
40
|
+
parts.push(pick(all));
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
// Shuffle
|
|
44
|
+
for (let i = parts.length - 1; i > 0; i--) {
|
|
45
|
+
const j = Math.floor(Math.random() * (i + 1));
|
|
46
|
+
[parts[i], parts[j]] = [parts[j], parts[i]];
|
|
47
|
+
}
|
|
48
|
+
return parts.join("");
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* Generate a secure random token
|
|
53
|
+
*/
|
|
54
|
+
function generateSecureToken(): string {
|
|
55
|
+
return randomBytes(40).toString("hex");
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/**
|
|
59
|
+
* Hash a token for database storage
|
|
60
|
+
*/
|
|
61
|
+
function hashToken(token: string): string {
|
|
62
|
+
return createHash("sha256").update(token).digest("hex");
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
/**
|
|
66
|
+
* Create admin routes for user and role management
|
|
67
|
+
*/
|
|
68
|
+
export function createAdminRoutes(config: AdminRouteOptions): Hono<HonoEnv> {
|
|
69
|
+
const router = new Hono<HonoEnv>();
|
|
70
|
+
const authRepo = config.authRepo;
|
|
71
|
+
const { emailService, emailConfig, hooks } = config;
|
|
72
|
+
|
|
73
|
+
/** Build a BackendHookContext from Hono's context object */
|
|
74
|
+
function buildHookContext(c: { get: (key: string) => unknown }, method: BackendHookContext["method"]): BackendHookContext {
|
|
75
|
+
const user = c.get("user") as { userId: string; roles?: string[] } | undefined;
|
|
76
|
+
return {
|
|
77
|
+
requestUser: user ? { userId: user.userId, roles: user.roles ?? [] } : undefined,
|
|
78
|
+
method
|
|
79
|
+
};
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
/** Apply users.afterRead hook to an AdminUser, returning null to filter out */
|
|
83
|
+
async function applyUserAfterRead(user: AdminUser, ctx: BackendHookContext): Promise<AdminUser | null> {
|
|
84
|
+
if (!hooks?.users?.afterRead) return user;
|
|
85
|
+
return hooks.users.afterRead(user, ctx);
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
/** Apply users.afterRead hook to an array and filter nulls */
|
|
89
|
+
async function applyUserAfterReadBatch(users: AdminUser[], ctx: BackendHookContext): Promise<AdminUser[]> {
|
|
90
|
+
if (!hooks?.users?.afterRead) return users;
|
|
91
|
+
const results = await Promise.all(users.map(u => applyUserAfterRead(u, ctx)));
|
|
92
|
+
return results.filter((u): u is AdminUser => u !== null);
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
/** Apply roles.afterRead hook to an array and filter nulls */
|
|
96
|
+
async function applyRoleAfterReadBatch(roles: AdminRole[], ctx: BackendHookContext): Promise<AdminRole[]> {
|
|
97
|
+
if (!hooks?.roles?.afterRead) return roles;
|
|
98
|
+
const results = await Promise.all(roles.map(r => hooks!.roles!.afterRead!(r, ctx)));
|
|
99
|
+
return results.filter((r): r is AdminRole => r !== null);
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
/** Convert a DB user record + role IDs into the AdminUser API shape */
|
|
103
|
+
function toAdminUser(u: { id: string; email: string; displayName?: string | null; photoUrl?: string | null; createdAt?: Date | string; updatedAt?: Date | string }, roles: string[]): AdminUser {
|
|
104
|
+
return {
|
|
105
|
+
uid: u.id,
|
|
106
|
+
email: u.email,
|
|
107
|
+
displayName: u.displayName ?? null,
|
|
108
|
+
photoURL: u.photoUrl ?? null,
|
|
109
|
+
provider: "custom",
|
|
110
|
+
roles,
|
|
111
|
+
createdAt: u.createdAt instanceof Date ? u.createdAt.toISOString() : (u.createdAt ?? new Date().toISOString()),
|
|
112
|
+
updatedAt: u.updatedAt instanceof Date ? u.updatedAt.toISOString() : (u.updatedAt ?? new Date().toISOString())
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
// Attach Rebase error handler to ensure exceptions are correctly formatted
|
|
117
|
+
// instead of caught by Hono's default error handler from the sub-router.
|
|
118
|
+
router.onError(errorHandler);
|
|
119
|
+
|
|
120
|
+
// Apply auth middleware to all routes (service-key-aware when configured)
|
|
121
|
+
router.use("/*", createRequireAuth({ serviceKey: config.serviceKey }));
|
|
122
|
+
|
|
123
|
+
/**
|
|
124
|
+
* POST /admin/bootstrap
|
|
125
|
+
*
|
|
126
|
+
* One-time endpoint to promote the calling user to admin.
|
|
127
|
+
* Guarded by three layers:
|
|
128
|
+
* 1. Authentication (handled by middleware above)
|
|
129
|
+
* 2. Persistent `bootstrap_completed` flag (when `setBootstrapCompleted` is provided)
|
|
130
|
+
* 3. Database check — no existing admin users
|
|
131
|
+
*
|
|
132
|
+
* Once invoked successfully the persistent flag is set, permanently disabling
|
|
133
|
+
* this endpoint even if all admin users are later deleted.
|
|
134
|
+
*/
|
|
135
|
+
router.post("/bootstrap", async (c) => {
|
|
136
|
+
const user = c.get("user");
|
|
137
|
+
if (!user || typeof user !== "object") {
|
|
138
|
+
throw ApiError.unauthorized("Not authenticated");
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
// ── Guard 1: persistent flag ──────────────────────────────────
|
|
142
|
+
if (config.isBootstrapCompleted) {
|
|
143
|
+
const alreadyDone = await config.isBootstrapCompleted();
|
|
144
|
+
if (alreadyDone) {
|
|
145
|
+
throw ApiError.forbidden("Bootstrap has already been completed.", "BOOTSTRAP_COMPLETED");
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
// ── Guard 2: no existing admin users ─────────────────────────
|
|
150
|
+
const users = await authRepo.listUsers();
|
|
151
|
+
let hasAdmin = false;
|
|
152
|
+
|
|
153
|
+
for (const u of users) {
|
|
154
|
+
const roles = await authRepo.getUserRoleIds(u.id);
|
|
155
|
+
if (roles.includes("admin")) {
|
|
156
|
+
hasAdmin = true;
|
|
157
|
+
break;
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
if (hasAdmin) {
|
|
162
|
+
throw ApiError.forbidden("Admin users already exist. Bootstrap not allowed.", "BOOTSTRAP_COMPLETED");
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
// ── Promote caller ───────────────────────────────────────────
|
|
166
|
+
const userId = "userId" in user ? user.userId : undefined;
|
|
167
|
+
if (!userId) {
|
|
168
|
+
throw ApiError.unauthorized("User ID not found in auth context");
|
|
169
|
+
}
|
|
170
|
+
await authRepo.setUserRoles(userId, ["admin"]);
|
|
171
|
+
|
|
172
|
+
// ── Set persistent flag ──────────────────────────────────────
|
|
173
|
+
if (config.setBootstrapCompleted) {
|
|
174
|
+
await config.setBootstrapCompleted();
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
return c.json({
|
|
178
|
+
success: true,
|
|
179
|
+
message: "You are now an admin",
|
|
180
|
+
user: {
|
|
181
|
+
uid: userId,
|
|
182
|
+
roles: ["admin"]
|
|
183
|
+
}
|
|
184
|
+
});
|
|
185
|
+
});
|
|
186
|
+
|
|
187
|
+
router.get("/users", requireAdmin, async (c) => {
|
|
188
|
+
const limitParam = c.req.query("limit");
|
|
189
|
+
const offsetParam = c.req.query("offset");
|
|
190
|
+
const search = c.req.query("search");
|
|
191
|
+
const orderBy = c.req.query("orderBy");
|
|
192
|
+
const orderDir = c.req.query("orderDir") as "asc" | "desc" | undefined;
|
|
193
|
+
const hookCtx = buildHookContext(c, "GET");
|
|
194
|
+
|
|
195
|
+
// If pagination params are provided, use the paginated path
|
|
196
|
+
if (limitParam !== undefined || search) {
|
|
197
|
+
const limit = limitParam ? parseInt(limitParam, 10) : 25;
|
|
198
|
+
const offset = offsetParam ? parseInt(offsetParam, 10) : 0;
|
|
199
|
+
|
|
200
|
+
const result = await authRepo.listUsersPaginated({
|
|
201
|
+
limit,
|
|
202
|
+
offset,
|
|
203
|
+
search: search || undefined,
|
|
204
|
+
orderBy: orderBy || undefined,
|
|
205
|
+
orderDir: orderDir || undefined,
|
|
206
|
+
roleId: c.req.query("role") || undefined
|
|
207
|
+
});
|
|
208
|
+
|
|
209
|
+
let usersWithRoles: AdminUser[] = await Promise.all(
|
|
210
|
+
result.users.map(async (u) => {
|
|
211
|
+
const roles = await authRepo.getUserRoleIds(u.id);
|
|
212
|
+
return toAdminUser(u, roles);
|
|
213
|
+
})
|
|
214
|
+
);
|
|
215
|
+
|
|
216
|
+
usersWithRoles = await applyUserAfterReadBatch(usersWithRoles, hookCtx);
|
|
217
|
+
|
|
218
|
+
return c.json({
|
|
219
|
+
users: usersWithRoles,
|
|
220
|
+
total: result.total,
|
|
221
|
+
limit: result.limit,
|
|
222
|
+
offset: result.offset
|
|
223
|
+
});
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
// Legacy: return all users (no pagination)
|
|
227
|
+
const users = await authRepo.listUsers();
|
|
228
|
+
let usersWithRoles: AdminUser[] = await Promise.all(
|
|
229
|
+
users.map(async (u) => {
|
|
230
|
+
const roles = await authRepo.getUserRoleIds(u.id);
|
|
231
|
+
return toAdminUser(u, roles);
|
|
232
|
+
})
|
|
233
|
+
);
|
|
234
|
+
|
|
235
|
+
usersWithRoles = await applyUserAfterReadBatch(usersWithRoles, hookCtx);
|
|
236
|
+
|
|
237
|
+
return c.json({ users: usersWithRoles });
|
|
238
|
+
});
|
|
239
|
+
|
|
240
|
+
router.get("/users/:userId", requireAdmin, async (c) => {
|
|
241
|
+
const userId = c.req.param("userId");
|
|
242
|
+
const result = await authRepo.getUserWithRoles(userId);
|
|
243
|
+
|
|
244
|
+
if (!result) {
|
|
245
|
+
throw ApiError.notFound("User not found");
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
const hookCtx = buildHookContext(c, "GET");
|
|
249
|
+
let adminUser: AdminUser | null = toAdminUser(result.user, result.roles.map(r => r.id));
|
|
250
|
+
|
|
251
|
+
adminUser = await applyUserAfterRead(adminUser, hookCtx);
|
|
252
|
+
if (!adminUser) {
|
|
253
|
+
throw ApiError.notFound("User not found");
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
return c.json({ user: adminUser });
|
|
257
|
+
});
|
|
258
|
+
|
|
259
|
+
router.post("/users", requireAdmin, async (c) => {
|
|
260
|
+
const body = await c.req.json();
|
|
261
|
+
let { email, displayName, password, roles } = body;
|
|
262
|
+
|
|
263
|
+
if (!email) {
|
|
264
|
+
throw ApiError.badRequest("Email is required", "INVALID_INPUT");
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
// Apply beforeSave hook
|
|
268
|
+
const hookCtx = buildHookContext(c, "POST");
|
|
269
|
+
if (hooks?.users?.beforeSave) {
|
|
270
|
+
const hooked = await hooks.users.beforeSave({ email, displayName, roles }, hookCtx);
|
|
271
|
+
email = hooked.email ?? email;
|
|
272
|
+
displayName = hooked.displayName ?? displayName;
|
|
273
|
+
roles = hooked.roles ?? roles;
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
const existing = await authRepo.getUserByEmail(email);
|
|
277
|
+
if (existing) {
|
|
278
|
+
throw ApiError.conflict("Email already exists", "EMAIL_EXISTS");
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
// Use provided password or auto-generate one
|
|
282
|
+
const clearPassword = password || generateSecurePassword();
|
|
283
|
+
|
|
284
|
+
const validation = validatePasswordStrength(clearPassword);
|
|
285
|
+
if (!validation.valid) {
|
|
286
|
+
throw ApiError.badRequest(validation.errors.join(". "), "WEAK_PASSWORD");
|
|
287
|
+
}
|
|
288
|
+
const passwordHash = await hashPassword(clearPassword);
|
|
289
|
+
|
|
290
|
+
const user = await authRepo.createUser({
|
|
291
|
+
email: email.toLowerCase(),
|
|
292
|
+
displayName: displayName || null,
|
|
293
|
+
passwordHash
|
|
294
|
+
});
|
|
295
|
+
|
|
296
|
+
if (roles && Array.isArray(roles) && roles.length > 0) {
|
|
297
|
+
await authRepo.setUserRoles(user.id, roles);
|
|
298
|
+
} else if (config.defaultRole) {
|
|
299
|
+
await authRepo.assignDefaultRole(user.id, config.defaultRole);
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
const userRoles = await authRepo.getUserRoleIds(user.id);
|
|
303
|
+
|
|
304
|
+
// Determine if we can send an invitation email
|
|
305
|
+
const isEmailConfigured = !!(emailService && emailService.isConfigured());
|
|
306
|
+
let invitationSent = false;
|
|
307
|
+
let temporaryPassword: string | undefined;
|
|
308
|
+
|
|
309
|
+
if (isEmailConfigured && !password) {
|
|
310
|
+
// Send invitation email via password-reset token flow
|
|
311
|
+
try {
|
|
312
|
+
const token = generateSecureToken();
|
|
313
|
+
const tokenHash = hashToken(token);
|
|
314
|
+
const expiresAt = new Date(Date.now() + 60 * 60 * 1000); // 1 hour
|
|
315
|
+
|
|
316
|
+
await authRepo.createPasswordResetToken(user.id, tokenHash, expiresAt);
|
|
317
|
+
|
|
318
|
+
const baseUrl = emailConfig?.resetPasswordUrl || "";
|
|
319
|
+
const setPasswordUrl = `${baseUrl}/reset-password?token=${token}`;
|
|
320
|
+
|
|
321
|
+
const appName = emailConfig?.appName || "Rebase";
|
|
322
|
+
const templateFn = emailConfig?.templates?.userInvitation;
|
|
323
|
+
const emailContent = templateFn
|
|
324
|
+
? templateFn(setPasswordUrl, { email: user.email,
|
|
325
|
+
displayName: user.displayName })
|
|
326
|
+
: getUserInvitationTemplate(setPasswordUrl, { email: user.email,
|
|
327
|
+
displayName: user.displayName }, appName);
|
|
328
|
+
|
|
329
|
+
await emailService!.send({
|
|
330
|
+
to: user.email,
|
|
331
|
+
subject: emailContent.subject,
|
|
332
|
+
html: emailContent.html,
|
|
333
|
+
text: emailContent.text
|
|
334
|
+
});
|
|
335
|
+
invitationSent = true;
|
|
336
|
+
} catch (emailError: unknown) {
|
|
337
|
+
console.error("Failed to send invitation email:", emailError instanceof Error ? emailError.message : emailError);
|
|
338
|
+
// Fall back to returning the temporary password
|
|
339
|
+
temporaryPassword = clearPassword;
|
|
340
|
+
}
|
|
341
|
+
} else if (!password) {
|
|
342
|
+
// No email service — return the auto-generated password one-time
|
|
343
|
+
temporaryPassword = clearPassword;
|
|
344
|
+
}
|
|
345
|
+
// If admin provided a password explicitly, don't return it or send email
|
|
346
|
+
|
|
347
|
+
const createdAdminUser: AdminUser = toAdminUser(user, userRoles);
|
|
348
|
+
|
|
349
|
+
// Fire afterSave hook (fire-and-forget for side-effects)
|
|
350
|
+
if (hooks?.users?.afterSave) {
|
|
351
|
+
Promise.resolve(hooks.users.afterSave(createdAdminUser, hookCtx)).catch(err => {
|
|
352
|
+
console.error("[BackendHooks] users.afterSave error:", err instanceof Error ? err.message : err);
|
|
353
|
+
});
|
|
354
|
+
}
|
|
355
|
+
|
|
356
|
+
return c.json({
|
|
357
|
+
user: createdAdminUser,
|
|
358
|
+
invitationSent,
|
|
359
|
+
...(temporaryPassword ? { temporaryPassword } : {})
|
|
360
|
+
}, 201);
|
|
361
|
+
});
|
|
362
|
+
|
|
363
|
+
router.post("/users/:userId/reset-password", requireAdmin, async (c) => {
|
|
364
|
+
const userId = c.req.param("userId");
|
|
365
|
+
const existing = await authRepo.getUserById(userId);
|
|
366
|
+
if (!existing) {
|
|
367
|
+
throw ApiError.notFound("User not found");
|
|
368
|
+
}
|
|
369
|
+
|
|
370
|
+
const isEmailConfigured = !!(emailService && emailService.isConfigured());
|
|
371
|
+
let invitationSent = false;
|
|
372
|
+
let temporaryPassword: string | undefined;
|
|
373
|
+
|
|
374
|
+
if (isEmailConfigured) {
|
|
375
|
+
try {
|
|
376
|
+
const token = generateSecureToken();
|
|
377
|
+
const tokenHash = hashToken(token);
|
|
378
|
+
const expiresAt = new Date(Date.now() + 60 * 60 * 1000); // 1 hour
|
|
379
|
+
|
|
380
|
+
await authRepo.createPasswordResetToken(existing.id, tokenHash, expiresAt);
|
|
381
|
+
|
|
382
|
+
const baseUrl = emailConfig?.resetPasswordUrl || "";
|
|
383
|
+
const setPasswordUrl = `${baseUrl}/reset-password?token=${token}`;
|
|
384
|
+
|
|
385
|
+
const appName = emailConfig?.appName || "Rebase";
|
|
386
|
+
const templateFn = emailConfig?.templates?.passwordReset;
|
|
387
|
+
const emailContent = templateFn
|
|
388
|
+
? templateFn(setPasswordUrl, { email: existing.email,
|
|
389
|
+
displayName: existing.displayName })
|
|
390
|
+
: getPasswordResetTemplate(setPasswordUrl, { email: existing.email,
|
|
391
|
+
displayName: existing.displayName }, appName);
|
|
392
|
+
|
|
393
|
+
await emailService!.send({
|
|
394
|
+
to: existing.email,
|
|
395
|
+
subject: emailContent.subject,
|
|
396
|
+
html: emailContent.html,
|
|
397
|
+
text: emailContent.text
|
|
398
|
+
});
|
|
399
|
+
invitationSent = true;
|
|
400
|
+
} catch (emailError: unknown) {
|
|
401
|
+
console.error("Failed to send reset email:", emailError instanceof Error ? emailError.message : emailError);
|
|
402
|
+
// Fall back to returning the temporary password
|
|
403
|
+
const clearPassword = generateSecurePassword();
|
|
404
|
+
const passwordHash = await hashPassword(clearPassword);
|
|
405
|
+
await authRepo.updatePassword(existing.id, passwordHash);
|
|
406
|
+
temporaryPassword = clearPassword;
|
|
407
|
+
}
|
|
408
|
+
} else {
|
|
409
|
+
// No email service — generate password, set it, and return one-time
|
|
410
|
+
const clearPassword = generateSecurePassword();
|
|
411
|
+
const passwordHash = await hashPassword(clearPassword);
|
|
412
|
+
await authRepo.updatePassword(existing.id, passwordHash);
|
|
413
|
+
temporaryPassword = clearPassword;
|
|
414
|
+
}
|
|
415
|
+
|
|
416
|
+
const userRoles = await authRepo.getUserRoleIds(existing.id);
|
|
417
|
+
|
|
418
|
+
return c.json({
|
|
419
|
+
user: {
|
|
420
|
+
uid: existing.id,
|
|
421
|
+
email: existing.email,
|
|
422
|
+
displayName: existing.displayName,
|
|
423
|
+
roles: userRoles
|
|
424
|
+
},
|
|
425
|
+
invitationSent,
|
|
426
|
+
...(temporaryPassword ? { temporaryPassword } : {})
|
|
427
|
+
}, 200);
|
|
428
|
+
});
|
|
429
|
+
|
|
430
|
+
router.put("/users/:userId", requireAdmin, async (c) => {
|
|
431
|
+
const userId = c.req.param("userId");
|
|
432
|
+
const body = await c.req.json();
|
|
433
|
+
let { email, displayName, password, roles } = body;
|
|
434
|
+
|
|
435
|
+
const existing = await authRepo.getUserById(userId);
|
|
436
|
+
if (!existing) {
|
|
437
|
+
throw ApiError.notFound("User not found");
|
|
438
|
+
}
|
|
439
|
+
|
|
440
|
+
// Apply beforeSave hook
|
|
441
|
+
const hookCtx = buildHookContext(c, "PUT");
|
|
442
|
+
if (hooks?.users?.beforeSave) {
|
|
443
|
+
const hooked = await hooks.users.beforeSave({ email, displayName, roles }, hookCtx);
|
|
444
|
+
email = hooked.email ?? email;
|
|
445
|
+
displayName = hooked.displayName ?? displayName;
|
|
446
|
+
roles = hooked.roles ?? roles;
|
|
447
|
+
}
|
|
448
|
+
|
|
449
|
+
const updates: Record<string, unknown> = {};
|
|
450
|
+
if (email !== undefined) updates.email = email.toLowerCase();
|
|
451
|
+
if (displayName !== undefined) updates.displayName = displayName;
|
|
452
|
+
|
|
453
|
+
if (password) {
|
|
454
|
+
const validation = validatePasswordStrength(password);
|
|
455
|
+
if (!validation.valid) {
|
|
456
|
+
throw ApiError.badRequest(validation.errors.join(". "), "WEAK_PASSWORD");
|
|
457
|
+
}
|
|
458
|
+
updates.passwordHash = await hashPassword(password);
|
|
459
|
+
}
|
|
460
|
+
|
|
461
|
+
if (Object.keys(updates).length > 0) {
|
|
462
|
+
await authRepo.updateUser(userId, updates);
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
if (roles !== undefined && Array.isArray(roles)) {
|
|
466
|
+
await authRepo.setUserRoles(userId, roles);
|
|
467
|
+
}
|
|
468
|
+
|
|
469
|
+
const result = await authRepo.getUserWithRoles(userId);
|
|
470
|
+
|
|
471
|
+
const updatedAdminUser: AdminUser = toAdminUser(result!.user, result!.roles.map(r => r.id));
|
|
472
|
+
|
|
473
|
+
// Fire afterSave hook (fire-and-forget)
|
|
474
|
+
if (hooks?.users?.afterSave) {
|
|
475
|
+
Promise.resolve(hooks.users.afterSave(updatedAdminUser, hookCtx)).catch(err => {
|
|
476
|
+
console.error("[BackendHooks] users.afterSave error:", err instanceof Error ? err.message : err);
|
|
477
|
+
});
|
|
478
|
+
}
|
|
479
|
+
|
|
480
|
+
return c.json({ user: updatedAdminUser });
|
|
481
|
+
});
|
|
482
|
+
|
|
483
|
+
router.delete("/users/:userId", requireAdmin, async (c) => {
|
|
484
|
+
const userId = c.req.param("userId");
|
|
485
|
+
const user = c.get("user");
|
|
486
|
+
|
|
487
|
+
const currentUserId = user && typeof user === "object" && "userId" in user ? user.userId : undefined;
|
|
488
|
+
if (currentUserId === userId) {
|
|
489
|
+
throw ApiError.badRequest("Cannot delete your own account", "SELF_DELETE");
|
|
490
|
+
}
|
|
491
|
+
|
|
492
|
+
const existing = await authRepo.getUserById(userId);
|
|
493
|
+
if (!existing) {
|
|
494
|
+
throw ApiError.notFound("User not found");
|
|
495
|
+
}
|
|
496
|
+
|
|
497
|
+
// Apply beforeDelete hook (throw to abort)
|
|
498
|
+
const hookCtx = buildHookContext(c, "DELETE");
|
|
499
|
+
if (hooks?.users?.beforeDelete) {
|
|
500
|
+
await hooks.users.beforeDelete(userId, hookCtx);
|
|
501
|
+
}
|
|
502
|
+
|
|
503
|
+
await authRepo.deleteUser(userId);
|
|
504
|
+
|
|
505
|
+
// Fire afterDelete hook (fire-and-forget)
|
|
506
|
+
if (hooks?.users?.afterDelete) {
|
|
507
|
+
Promise.resolve(hooks.users.afterDelete(userId, hookCtx)).catch(err => {
|
|
508
|
+
console.error("[BackendHooks] users.afterDelete error:", err instanceof Error ? err.message : err);
|
|
509
|
+
});
|
|
510
|
+
}
|
|
511
|
+
|
|
512
|
+
return c.json({ success: true });
|
|
513
|
+
});
|
|
514
|
+
|
|
515
|
+
router.get("/roles", requireAdmin, async (c) => {
|
|
516
|
+
const roles = await authRepo.listRoles();
|
|
517
|
+
const hookCtx = buildHookContext(c, "GET");
|
|
518
|
+
|
|
519
|
+
let adminRoles: AdminRole[] = roles.map(r => ({
|
|
520
|
+
id: r.id,
|
|
521
|
+
name: r.name,
|
|
522
|
+
isAdmin: r.isAdmin,
|
|
523
|
+
defaultPermissions: r.defaultPermissions,
|
|
524
|
+
config: r.config
|
|
525
|
+
}));
|
|
526
|
+
|
|
527
|
+
adminRoles = await applyRoleAfterReadBatch(adminRoles, hookCtx);
|
|
528
|
+
|
|
529
|
+
return c.json({ roles: adminRoles });
|
|
530
|
+
});
|
|
531
|
+
|
|
532
|
+
router.get("/roles/:roleId", requireAdmin, async (c) => {
|
|
533
|
+
const roleId = c.req.param("roleId");
|
|
534
|
+
const role = await authRepo.getRoleById(roleId);
|
|
535
|
+
|
|
536
|
+
if (!role) {
|
|
537
|
+
throw ApiError.notFound("Role not found");
|
|
538
|
+
}
|
|
539
|
+
|
|
540
|
+
return c.json({ role });
|
|
541
|
+
});
|
|
542
|
+
|
|
543
|
+
router.post("/roles", requireAdmin, async (c) => {
|
|
544
|
+
const body = await c.req.json();
|
|
545
|
+
const { id, name, isAdmin, defaultPermissions, config } = body;
|
|
546
|
+
|
|
547
|
+
if (!id || !name) {
|
|
548
|
+
throw ApiError.badRequest("Role ID and name are required", "INVALID_INPUT");
|
|
549
|
+
}
|
|
550
|
+
|
|
551
|
+
const existing = await authRepo.getRoleById(id);
|
|
552
|
+
if (existing) {
|
|
553
|
+
throw ApiError.conflict("Role already exists", "ROLE_EXISTS");
|
|
554
|
+
}
|
|
555
|
+
|
|
556
|
+
const role = await authRepo.createRole({
|
|
557
|
+
id,
|
|
558
|
+
name,
|
|
559
|
+
isAdmin: isAdmin ?? false,
|
|
560
|
+
defaultPermissions: defaultPermissions ?? null,
|
|
561
|
+
config: config ?? null
|
|
562
|
+
});
|
|
563
|
+
|
|
564
|
+
return c.json({ role }, 201);
|
|
565
|
+
});
|
|
566
|
+
|
|
567
|
+
router.put("/roles/:roleId", requireAdmin, async (c) => {
|
|
568
|
+
const roleId = c.req.param("roleId");
|
|
569
|
+
const body = await c.req.json();
|
|
570
|
+
const { name, isAdmin, defaultPermissions, config } = body;
|
|
571
|
+
|
|
572
|
+
const existing = await authRepo.getRoleById(roleId);
|
|
573
|
+
if (!existing) {
|
|
574
|
+
throw ApiError.notFound("Role not found");
|
|
575
|
+
}
|
|
576
|
+
|
|
577
|
+
const role = await authRepo.updateRole(roleId, {
|
|
578
|
+
name,
|
|
579
|
+
isAdmin,
|
|
580
|
+
defaultPermissions,
|
|
581
|
+
config
|
|
582
|
+
});
|
|
583
|
+
|
|
584
|
+
return c.json({ role });
|
|
585
|
+
});
|
|
586
|
+
|
|
587
|
+
router.delete("/roles/:roleId", requireAdmin, async (c) => {
|
|
588
|
+
const roleId = c.req.param("roleId");
|
|
589
|
+
|
|
590
|
+
if (["admin", "editor", "viewer"].includes(roleId)) {
|
|
591
|
+
throw ApiError.badRequest("Cannot delete built-in roles", "BUILTIN_ROLE");
|
|
592
|
+
}
|
|
593
|
+
|
|
594
|
+
const existing = await authRepo.getRoleById(roleId);
|
|
595
|
+
if (!existing) {
|
|
596
|
+
throw ApiError.notFound("Role not found");
|
|
597
|
+
}
|
|
598
|
+
|
|
599
|
+
await authRepo.deleteRole(roleId);
|
|
600
|
+
|
|
601
|
+
return c.json({ success: true });
|
|
602
|
+
});
|
|
603
|
+
|
|
604
|
+
return router;
|
|
605
|
+
}
|