@betterportal/config-manager 0.0.1
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/README.md +3 -0
- package/bsb-plugin.json +23 -0
- package/bsb-tests.json +14 -0
- package/lib/.bsb/clients/service-betterportal-config-manager.d.ts +37 -0
- package/lib/.bsb/clients/service-betterportal-config-manager.d.ts.map +1 -0
- package/lib/.bsb/clients/service-betterportal-config-manager.js +40 -0
- package/lib/.bsb/clients/service-betterportal-config-manager.js.map +1 -0
- package/lib/index.d.ts +2 -0
- package/lib/index.d.ts.map +1 -0
- package/lib/index.js +2 -0
- package/lib/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/.bp-generated/registry.d.ts +3 -0
- package/lib/plugins/service-betterportal-config-manager/.bp-generated/registry.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/.bp-generated/registry.js +235 -0
- package/lib/plugins/service-betterportal-config-manager/.bp-generated/registry.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/adminApi.d.ts +4 -0
- package/lib/plugins/service-betterportal-config-manager/adminApi.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/adminApi.js +2319 -0
- package/lib/plugins/service-betterportal-config-manager/adminApi.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bootstrapEndpoint.d.ts +21 -0
- package/lib/plugins/service-betterportal-config-manager/bootstrapEndpoint.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bootstrapEndpoint.js +269 -0
- package/lib/plugins/service-betterportal-config-manager/bootstrapEndpoint.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bootstrapWizardHtml.d.ts +19 -0
- package/lib/plugins/service-betterportal-config-manager/bootstrapWizardHtml.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bootstrapWizardHtml.js +329 -0
- package/lib/plugins/service-betterportal-config-manager/bootstrapWizardHtml.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/auth/_theme.bootstrap1/index.d.ts +4 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/auth/_theme.bootstrap1/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/auth/_theme.bootstrap1/index.js +38 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/auth/_theme.bootstrap1/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/auth/index.d.ts +96 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/auth/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/auth/index.js +78 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/auth/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/_theme.bootstrap1/index.d.ts +4 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/_theme.bootstrap1/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/_theme.bootstrap1/index.js +62 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/_theme.bootstrap1/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/_theme.embedded.d.ts +5 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/_theme.embedded.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/_theme.embedded.js +5 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/_theme.embedded.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/index.d.ts +43 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/index.js +68 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/config/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/fragments/_theme.bootstrap1/index.d.ts +5 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/fragments/_theme.bootstrap1/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/fragments/_theme.bootstrap1/index.js +11 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/fragments/_theme.bootstrap1/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/fragments/index.d.ts +32 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/fragments/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/fragments/index.js +32 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/fragments/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/menu/_theme.bootstrap1/index.d.ts +4 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/menu/_theme.bootstrap1/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/menu/_theme.bootstrap1/index.js +170 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/menu/_theme.bootstrap1/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/menu/index.d.ts +60 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/menu/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/menu/index.js +48 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/menu/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/preview/_theme.bootstrap1/index.d.ts +4 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/preview/_theme.bootstrap1/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/preview/_theme.bootstrap1/index.js +28 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/preview/_theme.bootstrap1/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/preview/index.d.ts +48 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/preview/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/preview/index.js +40 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/preview/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/routes/_theme.bootstrap1/index.d.ts +5 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/routes/_theme.bootstrap1/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/routes/_theme.bootstrap1/index.js +194 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/routes/_theme.bootstrap1/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/routes/index.d.ts +80 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/routes/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/routes/index.js +59 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/routes/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/services/_theme.bootstrap1/index.d.ts +5 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/services/_theme.bootstrap1/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/services/_theme.bootstrap1/index.js +167 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/services/_theme.bootstrap1/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/services/index.d.ts +128 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/services/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/services/index.js +89 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/services/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/settings/_theme.bootstrap1/index.d.ts +5 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/settings/_theme.bootstrap1/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/settings/_theme.bootstrap1/index.js +8 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/settings/_theme.bootstrap1/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/settings/index.d.ts +89 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/settings/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/settings/index.js +93 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/settings/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/tenants/_theme.bootstrap1/index.d.ts +4 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/tenants/_theme.bootstrap1/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/tenants/_theme.bootstrap1/index.js +61 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/tenants/_theme.bootstrap1/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/tenants/index.d.ts +180 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/tenants/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/tenants/index.js +405 -0
- package/lib/plugins/service-betterportal-config-manager/bp-routes/tenants/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/cpBootstrap.d.ts +26 -0
- package/lib/plugins/service-betterportal-config-manager/cpBootstrap.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/cpBootstrap.js +58 -0
- package/lib/plugins/service-betterportal-config-manager/cpBootstrap.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/fragmentsEditor.d.ts +3 -0
- package/lib/plugins/service-betterportal-config-manager/fragmentsEditor.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/fragmentsEditor.js +365 -0
- package/lib/plugins/service-betterportal-config-manager/fragmentsEditor.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/index.d.ts +143 -0
- package/lib/plugins/service-betterportal-config-manager/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/index.js +696 -0
- package/lib/plugins/service-betterportal-config-manager/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/menuEditor.d.ts +3 -0
- package/lib/plugins/service-betterportal-config-manager/menuEditor.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/menuEditor.js +823 -0
- package/lib/plugins/service-betterportal-config-manager/menuEditor.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/routeContext.d.ts +10 -0
- package/lib/plugins/service-betterportal-config-manager/routeContext.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/routeContext.js +11 -0
- package/lib/plugins/service-betterportal-config-manager/routeContext.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/setupTokens.d.ts +18 -0
- package/lib/plugins/service-betterportal-config-manager/setupTokens.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/setupTokens.js +245 -0
- package/lib/plugins/service-betterportal-config-manager/setupTokens.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/storage/core.d.ts +41 -0
- package/lib/plugins/service-betterportal-config-manager/storage/core.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/storage/core.js +396 -0
- package/lib/plugins/service-betterportal-config-manager/storage/core.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/storage/file.d.ts +10 -0
- package/lib/plugins/service-betterportal-config-manager/storage/file.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/storage/file.js +30 -0
- package/lib/plugins/service-betterportal-config-manager/storage/file.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/storage/index.d.ts +36 -0
- package/lib/plugins/service-betterportal-config-manager/storage/index.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/storage/index.js +52 -0
- package/lib/plugins/service-betterportal-config-manager/storage/index.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/storage/postgres.d.ts +15 -0
- package/lib/plugins/service-betterportal-config-manager/storage/postgres.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/storage/postgres.js +60 -0
- package/lib/plugins/service-betterportal-config-manager/storage/postgres.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/syncApi.d.ts +44 -0
- package/lib/plugins/service-betterportal-config-manager/syncApi.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/syncApi.js +280 -0
- package/lib/plugins/service-betterportal-config-manager/syncApi.js.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/webhooks.d.ts +6 -0
- package/lib/plugins/service-betterportal-config-manager/webhooks.d.ts.map +1 -0
- package/lib/plugins/service-betterportal-config-manager/webhooks.js +372 -0
- package/lib/plugins/service-betterportal-config-manager/webhooks.js.map +1 -0
- package/lib/schemas/service-betterportal-config-manager.json +157 -0
- package/lib/schemas/service-betterportal-config-manager.plugin.json +135 -0
- package/package.json +69 -0
|
@@ -0,0 +1,405 @@
|
|
|
1
|
+
import * as av from "anyvali";
|
|
2
|
+
import { createHandler, uuidv7 } from "@betterportal/framework";
|
|
3
|
+
import { getConfigManagerRouteContext } from "../../routeContext.js";
|
|
4
|
+
import { getManifestCache } from "../../syncApi.js";
|
|
5
|
+
const TenantItemSchema = av.object({
|
|
6
|
+
id: av.string().minLength(1),
|
|
7
|
+
slug: av.string().minLength(1),
|
|
8
|
+
title: av.string().minLength(1),
|
|
9
|
+
active: av.bool(),
|
|
10
|
+
serviceCount: av.int().min(0)
|
|
11
|
+
}, { unknownKeys: "strip" });
|
|
12
|
+
const AppItemSchema = av.object({
|
|
13
|
+
id: av.string().minLength(1),
|
|
14
|
+
tenantId: av.string().minLength(1),
|
|
15
|
+
slug: av.string().minLength(1),
|
|
16
|
+
title: av.string().minLength(1),
|
|
17
|
+
hostnames: av.array(av.string()),
|
|
18
|
+
shellServiceId: av.optional(av.string().minLength(1)),
|
|
19
|
+
authServiceId: av.optional(av.string().minLength(1)),
|
|
20
|
+
routeCount: av.int().min(0)
|
|
21
|
+
}, { unknownKeys: "strip" });
|
|
22
|
+
const ShellServiceSchema = av.object({
|
|
23
|
+
id: av.string().minLength(1),
|
|
24
|
+
tenantId: av.string().minLength(1),
|
|
25
|
+
title: av.string().minLength(1),
|
|
26
|
+
serviceId: av.optional(av.string())
|
|
27
|
+
}, { unknownKeys: "strip" });
|
|
28
|
+
export const ResponseSchema = av.object({
|
|
29
|
+
title: av.string().minLength(1),
|
|
30
|
+
tenants: av.array(TenantItemSchema),
|
|
31
|
+
apps: av.array(AppItemSchema),
|
|
32
|
+
shellServices: av.array(ShellServiceSchema).default([]),
|
|
33
|
+
authServices: av.array(ShellServiceSchema).default([]),
|
|
34
|
+
adminApiBase: av.string().minLength(1),
|
|
35
|
+
serviceBaseUrl: av.optional(av.string())
|
|
36
|
+
}, { unknownKeys: "strip" });
|
|
37
|
+
export const title = "Tenants & Apps";
|
|
38
|
+
export const description = "Manage tenants and applications.";
|
|
39
|
+
export const auth = {
|
|
40
|
+
required: true,
|
|
41
|
+
permissions: [
|
|
42
|
+
{ serviceId: "service.betterportal.config-manager", viewId: "tenants.index", permissions: ["read", "create", "update", "delete"] }
|
|
43
|
+
]
|
|
44
|
+
};
|
|
45
|
+
export const cacheHints = { ttlSeconds: 0, varyBy: ["accept", "origin"] };
|
|
46
|
+
export const demoScenarios = [
|
|
47
|
+
{ id: "default", title: "Default", response: { title: "Tenants & Apps", tenants: [], apps: [], shellServices: [], authServices: [], adminApiBase: "/.well-known/bp/admin" } }
|
|
48
|
+
];
|
|
49
|
+
export const handleGet = createHandler({ response: ResponseSchema }, () => buildResponseModel());
|
|
50
|
+
export const handlePost = createHandler({ response: ResponseSchema }, async (ctx) => {
|
|
51
|
+
const body = ctx.request;
|
|
52
|
+
if (stringValue(body.entity) === "app") {
|
|
53
|
+
await createApp(body);
|
|
54
|
+
}
|
|
55
|
+
else {
|
|
56
|
+
await createTenant(body);
|
|
57
|
+
}
|
|
58
|
+
return buildResponseModel();
|
|
59
|
+
});
|
|
60
|
+
export const handlePut = createHandler({ response: ResponseSchema }, async (ctx) => {
|
|
61
|
+
const body = ctx.request;
|
|
62
|
+
if (stringValue(body.entity) === "app") {
|
|
63
|
+
await updateApp(body);
|
|
64
|
+
}
|
|
65
|
+
else {
|
|
66
|
+
await updateTenant(body);
|
|
67
|
+
}
|
|
68
|
+
return buildResponseModel();
|
|
69
|
+
});
|
|
70
|
+
export const handleDelete = createHandler({ response: ResponseSchema }, async (ctx) => {
|
|
71
|
+
const entity = stringValue(ctx.query.entity);
|
|
72
|
+
const id = stringValue(ctx.query.id);
|
|
73
|
+
if (entity === "app") {
|
|
74
|
+
await deleteApp(id);
|
|
75
|
+
}
|
|
76
|
+
else {
|
|
77
|
+
await deleteTenant(id);
|
|
78
|
+
}
|
|
79
|
+
return buildResponseModel();
|
|
80
|
+
});
|
|
81
|
+
async function buildResponseModel() {
|
|
82
|
+
const routeContext = getConfigManagerRouteContext();
|
|
83
|
+
const config = await routeContext.storage.loadConfig();
|
|
84
|
+
return {
|
|
85
|
+
title: "Tenants & Apps",
|
|
86
|
+
tenants: config.tenants.map((t) => ({
|
|
87
|
+
id: t.id,
|
|
88
|
+
slug: t.slug,
|
|
89
|
+
title: t.title,
|
|
90
|
+
active: t.active,
|
|
91
|
+
serviceCount: t.services.length + config.sharedServiceActivations.filter((activation) => activation.enabled && activation.tenantId === t.id).length
|
|
92
|
+
})),
|
|
93
|
+
apps: config.apps.map((a) => ({
|
|
94
|
+
id: a.id,
|
|
95
|
+
tenantId: a.tenantId,
|
|
96
|
+
slug: a.slug,
|
|
97
|
+
title: a.title,
|
|
98
|
+
hostnames: a.hostnames,
|
|
99
|
+
shellServiceId: a.shell?.serviceId,
|
|
100
|
+
authServiceId: a.auth?.serviceId,
|
|
101
|
+
routeCount: a.routes.length
|
|
102
|
+
})),
|
|
103
|
+
shellServices: config.tenants.flatMap((tenant) => themeShellServicesForTenant(config, tenant.id)),
|
|
104
|
+
authServices: config.tenants.flatMap((tenant) => authServicesForTenant(config, tenant.id)),
|
|
105
|
+
adminApiBase: "/.well-known/bp/admin",
|
|
106
|
+
serviceBaseUrl: routeContext.serviceBaseUrl
|
|
107
|
+
};
|
|
108
|
+
}
|
|
109
|
+
async function createTenant(body) {
|
|
110
|
+
const title = stringValue(body.title);
|
|
111
|
+
if (!title)
|
|
112
|
+
return;
|
|
113
|
+
const routeContext = getConfigManagerRouteContext();
|
|
114
|
+
const config = await routeContext.storage.loadConfig();
|
|
115
|
+
const id = uuidv7();
|
|
116
|
+
config.tenants.push({
|
|
117
|
+
id,
|
|
118
|
+
slug: stringValue(body.slug) || id,
|
|
119
|
+
title,
|
|
120
|
+
active: true,
|
|
121
|
+
branding: {},
|
|
122
|
+
services: [],
|
|
123
|
+
activatedPlatformServices: []
|
|
124
|
+
});
|
|
125
|
+
await routeContext.storage.saveConfig(config);
|
|
126
|
+
}
|
|
127
|
+
async function updateTenant(body) {
|
|
128
|
+
const id = stringValue(body.tenantId);
|
|
129
|
+
if (!id)
|
|
130
|
+
return;
|
|
131
|
+
const routeContext = getConfigManagerRouteContext();
|
|
132
|
+
const config = await routeContext.storage.loadConfig();
|
|
133
|
+
const tenant = config.tenants.find((t) => t.id === id);
|
|
134
|
+
if (!tenant)
|
|
135
|
+
return;
|
|
136
|
+
const title = stringValue(body.title);
|
|
137
|
+
const slug = stringValue(body.slug);
|
|
138
|
+
if (title)
|
|
139
|
+
tenant.title = title;
|
|
140
|
+
if (slug)
|
|
141
|
+
tenant.slug = slug;
|
|
142
|
+
tenant.active = boolValue(body.active);
|
|
143
|
+
await routeContext.storage.saveConfig(config);
|
|
144
|
+
}
|
|
145
|
+
async function deleteTenant(id) {
|
|
146
|
+
if (!id)
|
|
147
|
+
return;
|
|
148
|
+
const routeContext = getConfigManagerRouteContext();
|
|
149
|
+
const config = await routeContext.storage.loadConfig();
|
|
150
|
+
config.tenants = config.tenants.filter((tenant) => tenant.id !== id);
|
|
151
|
+
config.apps = config.apps.filter((app) => app.tenantId !== id);
|
|
152
|
+
await routeContext.storage.saveConfig(config);
|
|
153
|
+
}
|
|
154
|
+
async function createApp(body) {
|
|
155
|
+
const tenantId = stringValue(body.tenantId);
|
|
156
|
+
const title = stringValue(body.title);
|
|
157
|
+
if (!tenantId || !title)
|
|
158
|
+
return;
|
|
159
|
+
const routeContext = getConfigManagerRouteContext();
|
|
160
|
+
const config = await routeContext.storage.loadConfig();
|
|
161
|
+
const tenant = config.tenants.find((candidate) => candidate.id === tenantId);
|
|
162
|
+
if (!tenant)
|
|
163
|
+
return;
|
|
164
|
+
const shellServiceId = stringValue(body.shellServiceId);
|
|
165
|
+
if (shellServiceId && !isThemeShellServiceForTenant(config, tenantId, shellServiceId))
|
|
166
|
+
return;
|
|
167
|
+
const authServiceId = stringValue(body.authServiceId);
|
|
168
|
+
if (authServiceId && !isAuthServiceForTenant(config, tenantId, authServiceId))
|
|
169
|
+
return;
|
|
170
|
+
const id = uuidv7();
|
|
171
|
+
const app = {
|
|
172
|
+
id,
|
|
173
|
+
tenantId,
|
|
174
|
+
slug: stringValue(body.slug) || id,
|
|
175
|
+
title,
|
|
176
|
+
hostnames: hostnamesFromBody(body),
|
|
177
|
+
originOverrides: [],
|
|
178
|
+
refererOverrides: [],
|
|
179
|
+
...(shellServiceId ? { shell: { serviceId: shellServiceId } } : {}),
|
|
180
|
+
themeConfig: body.themeConfig ?? { mode: "system", bootstrap: {}, light: {}, dark: {} },
|
|
181
|
+
defaultRoute: stringValue(body.defaultRoute) || "/",
|
|
182
|
+
routes: [],
|
|
183
|
+
menu: [],
|
|
184
|
+
slots: [],
|
|
185
|
+
fragments: {}
|
|
186
|
+
};
|
|
187
|
+
if (authServiceId) {
|
|
188
|
+
app.auth = buildAppAuthConfig(config, tenantId, authServiceId);
|
|
189
|
+
ensureAuthRouteMounts(config, app);
|
|
190
|
+
}
|
|
191
|
+
config.apps.push(app);
|
|
192
|
+
await routeContext.storage.saveConfig(config);
|
|
193
|
+
}
|
|
194
|
+
async function updateApp(body) {
|
|
195
|
+
const id = stringValue(body.appId);
|
|
196
|
+
if (!id)
|
|
197
|
+
return;
|
|
198
|
+
const routeContext = getConfigManagerRouteContext();
|
|
199
|
+
const config = await routeContext.storage.loadConfig();
|
|
200
|
+
const appDef = config.apps.find((app) => app.id === id);
|
|
201
|
+
if (!appDef)
|
|
202
|
+
return;
|
|
203
|
+
const title = stringValue(body.title);
|
|
204
|
+
const slug = stringValue(body.slug);
|
|
205
|
+
if (title)
|
|
206
|
+
appDef.title = title;
|
|
207
|
+
if (slug)
|
|
208
|
+
appDef.slug = slug;
|
|
209
|
+
if (body.hostnames !== undefined || body.hostname !== undefined) {
|
|
210
|
+
appDef.hostnames = hostnamesFromBody(body);
|
|
211
|
+
}
|
|
212
|
+
const shellServiceId = stringValue(body.shellServiceId);
|
|
213
|
+
if (shellServiceId) {
|
|
214
|
+
if (isThemeShellServiceForTenant(config, appDef.tenantId, shellServiceId)) {
|
|
215
|
+
appDef.shell = { serviceId: shellServiceId };
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
else {
|
|
219
|
+
delete appDef.shell;
|
|
220
|
+
}
|
|
221
|
+
const authServiceId = stringValue(body.authServiceId);
|
|
222
|
+
if (authServiceId) {
|
|
223
|
+
if (isAuthServiceForTenant(config, appDef.tenantId, authServiceId)) {
|
|
224
|
+
const existingRoles = appDef.auth?.roles ?? [];
|
|
225
|
+
appDef.auth = {
|
|
226
|
+
...buildAppAuthConfig(config, appDef.tenantId, authServiceId, appDef.auth),
|
|
227
|
+
roles: existingRoles
|
|
228
|
+
};
|
|
229
|
+
ensureAuthRouteMounts(config, appDef);
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
else {
|
|
233
|
+
delete appDef.auth;
|
|
234
|
+
}
|
|
235
|
+
await routeContext.storage.saveConfig(config);
|
|
236
|
+
}
|
|
237
|
+
async function deleteApp(id) {
|
|
238
|
+
if (!id)
|
|
239
|
+
return;
|
|
240
|
+
const routeContext = getConfigManagerRouteContext();
|
|
241
|
+
const config = await routeContext.storage.loadConfig();
|
|
242
|
+
config.apps = config.apps.filter((app) => app.id !== id);
|
|
243
|
+
await routeContext.storage.saveConfig(config);
|
|
244
|
+
}
|
|
245
|
+
function themeShellServicesForTenant(config, tenantId) {
|
|
246
|
+
const tenant = config.tenants.find((candidate) => candidate.id === tenantId);
|
|
247
|
+
const tenantServices = (tenant?.services ?? [])
|
|
248
|
+
.filter((service) => service.enabled && service.capabilities?.includes("theme"))
|
|
249
|
+
.map((service) => ({
|
|
250
|
+
id: service.id,
|
|
251
|
+
tenantId,
|
|
252
|
+
title: service.title ?? service.serviceId ?? service.hostname,
|
|
253
|
+
serviceId: service.serviceId
|
|
254
|
+
}));
|
|
255
|
+
const sharedServices = config.sharedServiceActivations
|
|
256
|
+
.filter((activation) => activation.enabled && activation.tenantId === tenantId)
|
|
257
|
+
.map((activation) => {
|
|
258
|
+
const shared = config.sharedServiceCatalog.find((service) => service.enabled
|
|
259
|
+
&& service.id === activation.sharedServiceId
|
|
260
|
+
&& service.tags.includes("theme"));
|
|
261
|
+
if (!shared)
|
|
262
|
+
return undefined;
|
|
263
|
+
return {
|
|
264
|
+
id: activation.id,
|
|
265
|
+
tenantId,
|
|
266
|
+
title: shared.title,
|
|
267
|
+
serviceId: shared.serviceId ?? shared.id
|
|
268
|
+
};
|
|
269
|
+
})
|
|
270
|
+
.filter((service) => !!service);
|
|
271
|
+
return [...tenantServices, ...sharedServices];
|
|
272
|
+
}
|
|
273
|
+
function isThemeShellServiceForTenant(config, tenantId, shellServiceId) {
|
|
274
|
+
return themeShellServicesForTenant(config, tenantId).some((service) => service.id === shellServiceId);
|
|
275
|
+
}
|
|
276
|
+
function authServicesForTenant(config, tenantId) {
|
|
277
|
+
const tenant = config.tenants.find((candidate) => candidate.id === tenantId);
|
|
278
|
+
const tenantServices = (tenant?.services ?? [])
|
|
279
|
+
.filter((service) => service.enabled && service.capabilities?.includes("auth"))
|
|
280
|
+
.map((service) => ({
|
|
281
|
+
id: service.id,
|
|
282
|
+
tenantId,
|
|
283
|
+
title: service.title ?? service.serviceId ?? service.hostname,
|
|
284
|
+
serviceId: service.serviceId,
|
|
285
|
+
hostname: service.hostname
|
|
286
|
+
}));
|
|
287
|
+
const sharedServices = config.sharedServiceActivations
|
|
288
|
+
.filter((activation) => activation.enabled && activation.tenantId === tenantId)
|
|
289
|
+
.map((activation) => {
|
|
290
|
+
const shared = config.sharedServiceCatalog.find((service) => service.enabled
|
|
291
|
+
&& service.id === activation.sharedServiceId
|
|
292
|
+
&& service.tags.includes("auth"));
|
|
293
|
+
if (!shared)
|
|
294
|
+
return undefined;
|
|
295
|
+
return {
|
|
296
|
+
id: activation.id,
|
|
297
|
+
tenantId,
|
|
298
|
+
title: shared.title,
|
|
299
|
+
serviceId: shared.serviceId ?? shared.id,
|
|
300
|
+
hostname: shared.baseUrl
|
|
301
|
+
};
|
|
302
|
+
})
|
|
303
|
+
.filter((service) => !!service);
|
|
304
|
+
return [...tenantServices, ...sharedServices];
|
|
305
|
+
}
|
|
306
|
+
function isAuthServiceForTenant(config, tenantId, authServiceId) {
|
|
307
|
+
return authServicesForTenant(config, tenantId).some((service) => service.id === authServiceId);
|
|
308
|
+
}
|
|
309
|
+
function buildAppAuthConfig(config, tenantId, authServiceId, existing) {
|
|
310
|
+
const authService = authServicesForTenant(config, tenantId).find((service) => service.id === authServiceId);
|
|
311
|
+
const publicKeys = existing?.publicKeys ?? findKnownAuthPublicKeys(config, authServiceId);
|
|
312
|
+
const providerKind = authService?.serviceId === "service.betterportal.auth.authress-io"
|
|
313
|
+
? "authress.io"
|
|
314
|
+
: "betterportal.default";
|
|
315
|
+
const expectedIssuer = providerKind === "authress.io" && existing?.expectedIssuer === "https://authress.io"
|
|
316
|
+
? undefined
|
|
317
|
+
: existing?.expectedIssuer;
|
|
318
|
+
const expectedAudience = providerKind === "authress.io" && existing?.expectedAudience === "authress"
|
|
319
|
+
? undefined
|
|
320
|
+
: existing?.expectedAudience;
|
|
321
|
+
return {
|
|
322
|
+
serviceId: authServiceId,
|
|
323
|
+
loginViewId: existing?.loginViewId ?? "/login",
|
|
324
|
+
logoutViewId: existing?.logoutViewId ?? "/logout",
|
|
325
|
+
refreshViewId: existing?.refreshViewId ?? "/refresh",
|
|
326
|
+
provider: existing?.provider ?? (providerKind === "authress.io"
|
|
327
|
+
? { kind: "authress.io", roleClaimPath: "roles", subjectClaimPath: "sub" }
|
|
328
|
+
: { kind: "betterportal.default" }),
|
|
329
|
+
expectedIssuer: expectedIssuer ?? (providerKind === "authress.io" ? "https://authress.betterportal.local" : "https://auth.betterportal.local"),
|
|
330
|
+
expectedAudience: expectedAudience ?? "betterportal-runtime",
|
|
331
|
+
jwksUri: existing?.jwksUri ?? `${(authService?.hostname ?? "").replace(/\/+$/, "")}/.well-known/jwks.json`,
|
|
332
|
+
...(publicKeys ? { publicKeys } : {}),
|
|
333
|
+
roles: existing?.roles ?? []
|
|
334
|
+
};
|
|
335
|
+
}
|
|
336
|
+
function numberedPath(basePath, usedPaths) {
|
|
337
|
+
const normalized = basePath.startsWith("/") ? basePath : `/${basePath}`;
|
|
338
|
+
if (!usedPaths.has(normalized))
|
|
339
|
+
return normalized;
|
|
340
|
+
let i = 2;
|
|
341
|
+
while (usedPaths.has(`${normalized}-${i}`))
|
|
342
|
+
i += 1;
|
|
343
|
+
return `${normalized}-${i}`;
|
|
344
|
+
}
|
|
345
|
+
function ensureAuthRouteMounts(config, appDef) {
|
|
346
|
+
const authServiceId = appDef.auth?.serviceId;
|
|
347
|
+
if (!authServiceId)
|
|
348
|
+
return;
|
|
349
|
+
const manifest = getManifestCache().get(authServiceId);
|
|
350
|
+
if (!manifest)
|
|
351
|
+
return;
|
|
352
|
+
const desiredViewIds = ["login.index", "logout.index", "refresh.index", "register.index"];
|
|
353
|
+
const usedPaths = new Set(appDef.routes.map((route) => route.path));
|
|
354
|
+
for (const viewId of desiredViewIds) {
|
|
355
|
+
const view = manifest.viewIndex[viewId];
|
|
356
|
+
if (!view)
|
|
357
|
+
continue;
|
|
358
|
+
if (appDef.routes.some((route) => route.serviceId === authServiceId && route.viewId === viewId))
|
|
359
|
+
continue;
|
|
360
|
+
const path = numberedPath(view.path, usedPaths);
|
|
361
|
+
const methods = view.methods.filter((method) => method === "GET" || method === "POST" || method === "PUT" || method === "PATCH" || method === "DELETE" || method === "OPTIONS");
|
|
362
|
+
usedPaths.add(path);
|
|
363
|
+
appDef.routes.push({
|
|
364
|
+
id: uuidv7(),
|
|
365
|
+
path,
|
|
366
|
+
serviceId: authServiceId,
|
|
367
|
+
viewId,
|
|
368
|
+
targetPath: view.path,
|
|
369
|
+
title: viewId.replace(".index", "").replace(/^\w/, (char) => char.toUpperCase()),
|
|
370
|
+
enabled: true,
|
|
371
|
+
methods: methods.length ? methods : ["GET"]
|
|
372
|
+
});
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
function findKnownAuthPublicKeys(config, authServiceId) {
|
|
376
|
+
const activation = config.sharedServiceActivations.find((candidate) => candidate.id === authServiceId);
|
|
377
|
+
const sharedServiceId = activation?.sharedServiceId;
|
|
378
|
+
for (const app of config.apps) {
|
|
379
|
+
const auth = app.auth;
|
|
380
|
+
if (!auth?.publicKeys)
|
|
381
|
+
continue;
|
|
382
|
+
if (auth.serviceId === authServiceId)
|
|
383
|
+
return auth.publicKeys;
|
|
384
|
+
if (sharedServiceId) {
|
|
385
|
+
const appActivation = config.sharedServiceActivations.find((candidate) => candidate.id === auth.serviceId);
|
|
386
|
+
if (appActivation?.sharedServiceId === sharedServiceId)
|
|
387
|
+
return auth.publicKeys;
|
|
388
|
+
}
|
|
389
|
+
}
|
|
390
|
+
return undefined;
|
|
391
|
+
}
|
|
392
|
+
function hostnamesFromBody(body) {
|
|
393
|
+
if (Array.isArray(body.hostnames)) {
|
|
394
|
+
return body.hostnames.filter((value) => typeof value === "string" && value.length > 0);
|
|
395
|
+
}
|
|
396
|
+
const raw = stringValue(body.hostnames) || stringValue(body.hostname);
|
|
397
|
+
return raw.split(",").map((hostname) => hostname.trim()).filter(Boolean);
|
|
398
|
+
}
|
|
399
|
+
function stringValue(value) {
|
|
400
|
+
return typeof value === "string" ? value.trim() : "";
|
|
401
|
+
}
|
|
402
|
+
function boolValue(value) {
|
|
403
|
+
return value === true || value === "true" || value === "on";
|
|
404
|
+
}
|
|
405
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../src/plugins/service-betterportal-config-manager/bp-routes/tenants/index.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
export interface CpKeyPair {
|
|
2
|
+
privateKeyPem: string;
|
|
3
|
+
publicKeyPem: string;
|
|
4
|
+
kid: string;
|
|
5
|
+
}
|
|
6
|
+
export interface CpBootstrapState {
|
|
7
|
+
keyPair: CpKeyPair;
|
|
8
|
+
/** Public RSA JWK derived from keyPair. */
|
|
9
|
+
jwk: Record<string, unknown>;
|
|
10
|
+
/** Issuer URL used in tokens (iss claim). */
|
|
11
|
+
issuer: string;
|
|
12
|
+
/** Audience for tokens minted by this CP. */
|
|
13
|
+
audience: string;
|
|
14
|
+
/** Stable identifier for this CP instance (cpId). */
|
|
15
|
+
cpId: string;
|
|
16
|
+
/** JWKS URI the CP exposes (built from issuer). */
|
|
17
|
+
jwksUri: string;
|
|
18
|
+
}
|
|
19
|
+
export declare function cpBootstrap(input: {
|
|
20
|
+
keyStorePath: string;
|
|
21
|
+
issuer?: string;
|
|
22
|
+
audience: string;
|
|
23
|
+
host: string;
|
|
24
|
+
port: number;
|
|
25
|
+
}): CpBootstrapState;
|
|
26
|
+
//# sourceMappingURL=cpBootstrap.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cpBootstrap.d.ts","sourceRoot":"","sources":["../../../src/plugins/service-betterportal-config-manager/cpBootstrap.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,SAAS;IACxB,aAAa,EAAE,MAAM,CAAC;IACtB,YAAY,EAAE,MAAM,CAAC;IACrB,GAAG,EAAE,MAAM,CAAC;CACb;AAED,MAAM,WAAW,gBAAgB;IAC/B,OAAO,EAAE,SAAS,CAAC;IACnB,2CAA2C;IAC3C,GAAG,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAC7B,6CAA6C;IAC7C,MAAM,EAAE,MAAM,CAAC;IACf,6CAA6C;IAC7C,QAAQ,EAAE,MAAM,CAAC;IACjB,qDAAqD;IACrD,IAAI,EAAE,MAAM,CAAC;IACb,mDAAmD;IACnD,OAAO,EAAE,MAAM,CAAC;CACjB;AAQD,wBAAgB,WAAW,CAAC,KAAK,EAAE;IACjC,YAAY,EAAE,MAAM,CAAC;IACrB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,MAAM,CAAC;IACjB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;CACd,GAAG,gBAAgB,CAmBnB"}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import { readFileSync, writeFileSync, existsSync, mkdirSync } from "node:fs";
|
|
2
|
+
import { dirname, resolve } from "node:path";
|
|
3
|
+
import { randomBytes, generateKeyPairSync, createPublicKey } from "node:crypto";
|
|
4
|
+
/**
|
|
5
|
+
* Load or generate the CP keypair + derive JWK + assemble bootstrap state.
|
|
6
|
+
* Returns the same object on subsequent calls within a process.
|
|
7
|
+
*/
|
|
8
|
+
let cached = null;
|
|
9
|
+
export function cpBootstrap(input) {
|
|
10
|
+
if (cached)
|
|
11
|
+
return cached;
|
|
12
|
+
const keyPath = resolve(input.keyStorePath);
|
|
13
|
+
const keyPair = loadOrGenerateKeyPair(keyPath);
|
|
14
|
+
const issuer = input.issuer ?? `http://${input.host === "0.0.0.0" ? "localhost" : input.host}:${input.port}`;
|
|
15
|
+
const jwksUri = `${issuer.replace(/\/+$/, "")}/.well-known/jwks.json`;
|
|
16
|
+
const cpId = `cp-${keyPair.kid.slice(0, 12)}`;
|
|
17
|
+
cached = {
|
|
18
|
+
keyPair,
|
|
19
|
+
jwk: publicKeyToJwk(keyPair.publicKeyPem, keyPair.kid),
|
|
20
|
+
issuer: issuer.replace(/\/+$/, ""),
|
|
21
|
+
audience: input.audience,
|
|
22
|
+
cpId,
|
|
23
|
+
jwksUri
|
|
24
|
+
};
|
|
25
|
+
return cached;
|
|
26
|
+
}
|
|
27
|
+
function loadOrGenerateKeyPair(filePath) {
|
|
28
|
+
if (existsSync(filePath)) {
|
|
29
|
+
const parsed = JSON.parse(readFileSync(filePath, "utf8"));
|
|
30
|
+
if (typeof parsed.privateKeyPem !== "string" || typeof parsed.publicKeyPem !== "string" || typeof parsed.kid !== "string") {
|
|
31
|
+
throw new Error(`Malformed CP keypair file: ${filePath}`);
|
|
32
|
+
}
|
|
33
|
+
return parsed;
|
|
34
|
+
}
|
|
35
|
+
const { privateKey, publicKey } = generateKeyPairSync("rsa", {
|
|
36
|
+
modulusLength: 2048,
|
|
37
|
+
publicKeyEncoding: { type: "spki", format: "pem" },
|
|
38
|
+
privateKeyEncoding: { type: "pkcs8", format: "pem" }
|
|
39
|
+
});
|
|
40
|
+
const kid = randomBytes(16).toString("base64url");
|
|
41
|
+
const pair = { privateKeyPem: privateKey, publicKeyPem: publicKey, kid };
|
|
42
|
+
mkdirSync(dirname(filePath), { recursive: true });
|
|
43
|
+
writeFileSync(filePath, JSON.stringify(pair, null, 2), { mode: 0o600 });
|
|
44
|
+
return pair;
|
|
45
|
+
}
|
|
46
|
+
function publicKeyToJwk(publicKeyPem, kid) {
|
|
47
|
+
const keyObject = createPublicKey(publicKeyPem);
|
|
48
|
+
const jwk = keyObject.export({ format: "jwk" });
|
|
49
|
+
return {
|
|
50
|
+
kty: "RSA",
|
|
51
|
+
use: "sig",
|
|
52
|
+
alg: "RS256",
|
|
53
|
+
kid,
|
|
54
|
+
n: jwk.n,
|
|
55
|
+
e: jwk.e
|
|
56
|
+
};
|
|
57
|
+
}
|
|
58
|
+
//# sourceMappingURL=cpBootstrap.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cpBootstrap.js","sourceRoot":"","sources":["../../../src/plugins/service-betterportal-config-manager/cpBootstrap.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fragmentsEditor.d.ts","sourceRoot":"","sources":["../../../src/plugins/service-betterportal-config-manager/fragmentsEditor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,iBAAiB,EAEjB,mBAAmB,EACpB,MAAM,yBAAyB,CAAC;AAyRjC,wBAAgB,6BAA6B,CAAC,GAAG,EAAE,iBAAiB,EAAE,KAAK,EAAE,mBAAmB,GAAG,IAAI,CA+HtG"}
|