@cyberismo/backend 0.0.23 → 0.0.25

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (126) hide show
  1. package/dist/app.d.ts +3 -3
  2. package/dist/app.js +58 -29
  3. package/dist/app.js.map +1 -1
  4. package/dist/domain/cards/index.js +63 -1
  5. package/dist/domain/cards/index.js.map +1 -1
  6. package/dist/domain/cards/lib.js +7 -1
  7. package/dist/domain/cards/lib.js.map +1 -1
  8. package/dist/domain/cards/schema.d.ts +8 -0
  9. package/dist/domain/cards/schema.js +7 -0
  10. package/dist/domain/cards/schema.js.map +1 -1
  11. package/dist/domain/cards/service.d.ts +2 -0
  12. package/dist/domain/cards/service.js +18 -1
  13. package/dist/domain/cards/service.js.map +1 -1
  14. package/dist/domain/mcp/index.d.ts +8 -2
  15. package/dist/domain/mcp/index.js +68 -65
  16. package/dist/domain/mcp/index.js.map +1 -1
  17. package/dist/domain/project/service.js +1 -1
  18. package/dist/domain/project/service.js.map +1 -1
  19. package/dist/domain/projects/index.d.ts +15 -0
  20. package/dist/domain/projects/index.js +35 -0
  21. package/dist/domain/projects/index.js.map +1 -0
  22. package/dist/domain/resources/index.js +63 -1
  23. package/dist/domain/resources/index.js.map +1 -1
  24. package/dist/domain/resources/schema.d.ts +9 -0
  25. package/dist/domain/resources/schema.js +8 -1
  26. package/dist/domain/resources/schema.js.map +1 -1
  27. package/dist/domain/resources/service.d.ts +12 -0
  28. package/dist/domain/resources/service.js +49 -6
  29. package/dist/domain/resources/service.js.map +1 -1
  30. package/dist/export.d.ts +7 -3
  31. package/dist/export.js +32 -16
  32. package/dist/export.js.map +1 -1
  33. package/dist/index.d.ts +5 -4
  34. package/dist/index.js +4 -3
  35. package/dist/index.js.map +1 -1
  36. package/dist/main.js +41 -6
  37. package/dist/main.js.map +1 -1
  38. package/dist/middleware/auth.js +10 -0
  39. package/dist/middleware/auth.js.map +1 -1
  40. package/dist/middleware/commandManager.d.ts +12 -0
  41. package/dist/middleware/commandManager.js +33 -7
  42. package/dist/middleware/commandManager.js.map +1 -1
  43. package/dist/project-registry.d.ts +50 -0
  44. package/dist/project-registry.js +77 -0
  45. package/dist/project-registry.js.map +1 -0
  46. package/dist/public/THIRD-PARTY.txt +2899 -1974
  47. package/dist/public/assets/architecture-7EHR7CIX-BhpB9ddF.js +1 -0
  48. package/dist/public/assets/architectureDiagram-3BPJPVTR-BQYiU_EO.js +36 -0
  49. package/dist/public/assets/blockDiagram-GPEHLZMM-DuyikC3X.js +132 -0
  50. package/dist/public/assets/c4Diagram-AAUBKEIU-BG8_sPEr.js +10 -0
  51. package/dist/public/assets/channel-BxgB7fMy.js +1 -0
  52. package/dist/public/assets/chunk-2J33WTMH-vNq8B1aw.js +1 -0
  53. package/dist/public/assets/chunk-4BX2VUAB-DFDBsmSo.js +1 -0
  54. package/dist/public/assets/chunk-55IACEB6-DCVUQPWM.js +1 -0
  55. package/dist/public/assets/chunk-727SXJPM-C5ihZMyl.js +206 -0
  56. package/dist/public/assets/chunk-AQP2D5EJ-XGOtp2xP.js +231 -0
  57. package/dist/public/assets/chunk-FMBD7UC4-Da1lR6Mn.js +15 -0
  58. package/dist/public/assets/chunk-ND2GUHAM-ZOvpvr6K.js +1 -0
  59. package/dist/public/assets/chunk-QZHKN3VN-D33jzvFT.js +1 -0
  60. package/dist/public/assets/classDiagram-4FO5ZUOK-hWfZv7hZ.js +1 -0
  61. package/dist/public/assets/classDiagram-v2-Q7XG4LA2-hWfZv7hZ.js +1 -0
  62. package/dist/public/assets/cose-bilkent-S5V4N54A-DO4z-ix4.js +1 -0
  63. package/dist/public/assets/cytoscape.esm-C8YCVR3_.js +321 -0
  64. package/dist/public/assets/dagre-BM42HDAG-DlpRfzgA.js +4 -0
  65. package/dist/public/assets/dagre-Bx709z4p.js +1 -0
  66. package/dist/public/assets/diagram-2AECGRRQ-D-t_ImBP.js +43 -0
  67. package/dist/public/assets/diagram-5GNKFQAL-CBgUMlXz.js +10 -0
  68. package/dist/public/assets/diagram-KO2AKTUF-XoB2TgQt.js +3 -0
  69. package/dist/public/assets/diagram-LMA3HP47-D1Sbl_eS.js +24 -0
  70. package/dist/public/assets/diagram-OG6HWLK6-DKP4aiIY.js +24 -0
  71. package/dist/public/assets/erDiagram-TEJ5UH35-DYxfHOOK.js +85 -0
  72. package/dist/public/assets/eventmodeling-FCH6USID-cF_1Mq4g.js +1 -0
  73. package/dist/public/assets/flowDiagram-I6XJVG4X-BDHPsmlq.js +162 -0
  74. package/dist/public/assets/ganttDiagram-6RSMTGT7-bGgIvBPN.js +292 -0
  75. package/dist/public/assets/gitGraph-WXDBUCRP-DOFshjLy.js +1 -0
  76. package/dist/public/assets/gitGraphDiagram-PVQCEYII-xSwLjGd-.js +106 -0
  77. package/dist/public/assets/graphlib-B8gBHxth.js +1 -0
  78. package/dist/public/assets/index-DGPv1qic.js +1028 -0
  79. package/dist/public/assets/index-DvHiopvR.css +1 -0
  80. package/dist/public/assets/info-J43DQDTF-BuJNK7zQ.js +1 -0
  81. package/dist/public/assets/infoDiagram-5YYISTIA-BanxuIib.js +2 -0
  82. package/dist/public/assets/ishikawaDiagram-YF4QCWOH-DWNWYxz5.js +70 -0
  83. package/dist/public/assets/journeyDiagram-JHISSGLW-I58P5XNg.js +139 -0
  84. package/dist/public/assets/kanban-definition-UN3LZRKU-CMRWbDti.js +89 -0
  85. package/dist/public/assets/katex-C4eR7coU.js +257 -0
  86. package/dist/public/assets/mermaid-parser.core-Dz__fM3g.js +161 -0
  87. package/dist/public/assets/mindmap-definition-RKZ34NQL-C47gCcpC.js +96 -0
  88. package/dist/public/assets/packet-YPE3B663-Cczitw2-.js +1 -0
  89. package/dist/public/assets/pie-LRSECV5Y-rO-Aqx6h.js +1 -0
  90. package/dist/public/assets/pieDiagram-4H26LBE5-VZAxHzjD.js +30 -0
  91. package/dist/public/assets/quadrantDiagram-W4KKPZXB-BY8JORvE.js +7 -0
  92. package/dist/public/assets/radar-GUYGQ44K-SSIGuQjW.js +1 -0
  93. package/dist/public/assets/requirementDiagram-4Y6WPE33-XhNBeFwj.js +84 -0
  94. package/dist/public/assets/sankeyDiagram-5OEKKPKP-H7I2OESy.js +40 -0
  95. package/dist/public/assets/sequenceDiagram-3UESZ5HK-jnTLwq-X.js +162 -0
  96. package/dist/public/assets/stateDiagram-AJRCARHV-BKcf2bdX.js +1 -0
  97. package/dist/public/assets/stateDiagram-v2-BHNVJYJU-wpO0gnsG.js +1 -0
  98. package/dist/public/assets/timeline-definition-PNZ67QCA-BZbaBDRH.js +120 -0
  99. package/dist/public/assets/treeView-BLDUP644-DkGx4HkR.js +1 -0
  100. package/dist/public/assets/treemap-LRROVOQU-yCyuONQh.js +1 -0
  101. package/dist/public/assets/vennDiagram-CIIHVFJN-nY9Pep3o.js +34 -0
  102. package/dist/public/assets/wardley-L42UT6IY-CXWWFUgk.js +1 -0
  103. package/dist/public/assets/wardleyDiagram-YWT4CUSO-CM0yrkHd.js +78 -0
  104. package/dist/public/assets/xychartDiagram-2RQKCTM6-1ZAtqvyQ.js +7 -0
  105. package/dist/public/config.json +1 -0
  106. package/dist/public/index.html +2 -2
  107. package/package.json +11 -7
  108. package/src/app.ts +71 -31
  109. package/src/domain/cards/index.ts +73 -0
  110. package/src/domain/cards/lib.ts +8 -1
  111. package/src/domain/cards/schema.ts +9 -0
  112. package/src/domain/cards/service.ts +28 -2
  113. package/src/domain/mcp/index.ts +83 -78
  114. package/src/domain/project/service.ts +1 -1
  115. package/src/domain/projects/index.ts +39 -0
  116. package/src/domain/resources/index.ts +74 -0
  117. package/src/domain/resources/schema.ts +14 -0
  118. package/src/domain/resources/service.ts +52 -4
  119. package/src/export.ts +44 -21
  120. package/src/index.ts +6 -5
  121. package/src/main.ts +46 -6
  122. package/src/middleware/auth.ts +10 -0
  123. package/src/middleware/commandManager.ts +47 -9
  124. package/src/project-registry.ts +110 -0
  125. package/dist/public/assets/index-Cdn_jRWy.js +0 -720
  126. package/dist/public/assets/index-ypsafPwV.css +0 -1
package/src/export.ts CHANGED
@@ -18,6 +18,7 @@ import fs, { readFile } from 'node:fs/promises';
18
18
  import type { CommandManager } from '@cyberismo/data-handler';
19
19
  import { createApp } from './app.js';
20
20
  import { MockAuthProvider } from './auth/mock.js';
21
+ import type { ProjectRegistry } from './project-registry.js';
21
22
  import { cp, writeFile } from 'node:fs/promises';
22
23
  import { staticFrontendDirRelative } from './utils.js';
23
24
  import type { QueryResult } from '@cyberismo/data-handler/types/queries';
@@ -28,7 +29,11 @@ import {
28
29
  findRelevantAttachments,
29
30
  } from './domain/cards/service.js';
30
31
 
31
- let _cardQueryPromise: Promise<QueryResult<'card'>[]> | null = null;
32
+ export interface ExportSiteOptions extends TreeOptions {
33
+ defaultProject?: string;
34
+ }
35
+
36
+ const _cardQueryCache = new Map<string, Promise<QueryResult<'card'>[]>>();
32
37
  const OVERHEAD_CALLS = 6; // estimated number of overhead calls during export in addition to card exports
33
38
 
34
39
  /**
@@ -36,7 +41,7 @@ const OVERHEAD_CALLS = 6; // estimated number of overhead calls during export in
36
41
  * Also resets the card query promise.
37
42
  */
38
43
  export function reset() {
39
- _cardQueryPromise = null;
44
+ _cardQueryCache.clear();
40
45
  }
41
46
 
42
47
  /**
@@ -50,15 +55,14 @@ export async function getCardQueryResult(
50
55
  commands: CommandManager,
51
56
  cardKey?: string,
52
57
  ): Promise<QueryResult<'card'>[]> {
53
- if (!_cardQueryPromise) {
54
- // fetch all cards
55
- _cardQueryPromise = commands.calculateCmd.runQuery(
56
- 'card',
57
- 'exportedSite',
58
- {},
58
+ const prefix = commands.project.configuration.cardKeyPrefix;
59
+ if (!_cardQueryCache.has(prefix)) {
60
+ _cardQueryCache.set(
61
+ prefix,
62
+ commands.calculateCmd.runQuery('card', 'exportedSite', {}),
59
63
  );
60
64
  }
61
- return _cardQueryPromise.then((results) => {
65
+ return _cardQueryCache.get(prefix)!.then((results) => {
62
66
  if (!cardKey) {
63
67
  return results;
64
68
  }
@@ -73,29 +77,38 @@ export async function getCardQueryResult(
73
77
  /**
74
78
  * Export the site to a given directory.
75
79
  * Note: Do not call this function in parallel.
76
- * @param commands - CommandManager instance for the project.
80
+ * @param registry - ProjectRegistry holding all project CommandManagers.
77
81
  * @param exportDir - Directory to export to.
78
82
  * @param options - Export options.
79
83
  * @param options.recursive - Whether to export cards recursively.
80
84
  * @param options.cardKey - Key of the card to export. If not provided, all cards will be exported.
81
- * @param level - Log level for the operation.
85
+ * @param options.defaultProject - Default project prefix to write into config.json.
82
86
  * @param onProgress - Optional progress callback function.
83
87
  * @returns An object containing any errors that occurred during export.
84
88
  */
85
89
  export async function exportSite(
86
- commands: CommandManager,
90
+ registry: ProjectRegistry,
87
91
  exportDir?: string,
88
- options?: TreeOptions,
92
+ options?: ExportSiteOptions,
89
93
  onProgress?: (current: number, total: number) => void,
90
94
  ): Promise<{ errors: string[] }> {
91
95
  exportDir = exportDir || 'static';
92
- const opts = {
96
+ const { defaultProject, ...treeOpts } = options ?? {};
97
+ const opts: TreeOptions = {
93
98
  recursive: false,
94
- cardKey: undefined,
95
- ...options,
99
+ ...treeOpts,
96
100
  };
97
101
 
98
- const app = createApp(new MockAuthProvider(), commands, opts);
102
+ if (defaultProject && !registry.has(defaultProject)) {
103
+ throw new Error(
104
+ `Default project '${defaultProject}' is not in the registry. Available: ${registry
105
+ .list()
106
+ .map((p) => p.prefix)
107
+ .join(', ')}`,
108
+ );
109
+ }
110
+
111
+ const app = createApp(new MockAuthProvider(), registry, opts, true);
99
112
 
100
113
  // copy whole frontend to the same directory
101
114
  await cp(staticFrontendDirRelative, exportDir, { recursive: true });
@@ -103,6 +116,9 @@ export async function exportSite(
103
116
  const config = await readFile(path.join(exportDir, 'config.json'), 'utf-8');
104
117
  const configJson = JSON.parse(config);
105
118
  configJson.staticMode = true;
119
+ if (defaultProject) {
120
+ configJson.defaultProject = defaultProject;
121
+ }
106
122
  await writeFile(
107
123
  path.join(exportDir, 'config.json'),
108
124
  JSON.stringify(configJson),
@@ -110,10 +126,13 @@ export async function exportSite(
110
126
 
111
127
  reset();
112
128
 
113
- // estimate total based on the number of cards to export
114
- const cards = await findAllCards(commands, opts);
115
- const attachments = await findRelevantAttachments(commands, opts);
116
- let total = cards.length + attachments.length + OVERHEAD_CALLS;
129
+ // estimate total based on the number of cards to export across all projects
130
+ let total = OVERHEAD_CALLS;
131
+ for (const commands of registry.values()) {
132
+ const cards = await findAllCards(commands, opts);
133
+ const attachments = await findRelevantAttachments(commands, opts);
134
+ total += cards.length + attachments.length;
135
+ }
117
136
 
118
137
  // Actual export with progress reporting
119
138
  let done = 0;
@@ -130,6 +149,10 @@ export async function exportSite(
130
149
  if (url.pathname.startsWith('/mcp')) {
131
150
  return false;
132
151
  }
152
+ // Skip OIDC/well-known routes — not relevant for static export
153
+ if (url.pathname.startsWith('/.well-known')) {
154
+ return false;
155
+ }
133
156
  return req;
134
157
  },
135
158
  afterResponseHook: async (response) => {
package/src/index.ts CHANGED
@@ -16,14 +16,15 @@ import { Hono } from 'hono';
16
16
  import { serveStatic } from '@hono/node-server/serve-static';
17
17
  import path from 'node:path';
18
18
  import { readFile } from 'node:fs/promises';
19
- import type { CommandManager } from '@cyberismo/data-handler';
20
19
  import { findFreePort } from './utils.js';
21
20
  import { createApp } from './app.js';
22
21
  import type { AuthProvider } from './auth/types.js';
22
+ import type { ProjectRegistry } from './project-registry.js';
23
23
  export { MockAuthProvider } from './auth/mock.js';
24
24
  export type { MockUserConfig } from './auth/mock.js';
25
25
  export type { AuthProvider } from './auth/types.js';
26
- export { exportSite } from './export.js';
26
+ export { exportSite, type ExportSiteOptions } from './export.js';
27
+ export { ProjectRegistry } from './project-registry.js';
27
28
 
28
29
  const DEFAULT_PORT = 3000;
29
30
  const DEFAULT_MAX_PORT = DEFAULT_PORT + 100;
@@ -56,12 +57,12 @@ export async function previewSite(dir: string, findPort: boolean = true) {
56
57
  /**
57
58
  * Start the server
58
59
  * @param authProvider - Authentication provider
59
- * @param commands - CommandManager instance for the project
60
+ * @param registry - ProjectRegistry holding all project CommandManagers
60
61
  * @param findPort - If true, find a free port
61
62
  */
62
63
  export async function startServer(
63
64
  authProvider: AuthProvider,
64
- commands: CommandManager,
65
+ registry: ProjectRegistry,
65
66
  findPort: boolean = true,
66
67
  ) {
67
68
  let port = parseInt(process.env.PORT || DEFAULT_PORT.toString(), 10);
@@ -69,7 +70,7 @@ export async function startServer(
69
70
  if (findPort) {
70
71
  port = await findFreePort(port, DEFAULT_MAX_PORT);
71
72
  }
72
- const app = createApp(authProvider, commands);
73
+ const app = createApp(authProvider, registry);
73
74
  startApp(app, port);
74
75
  }
75
76
 
package/src/main.ts CHANGED
@@ -10,12 +10,14 @@
10
10
  details. You should have received a copy of the GNU Affero General Public
11
11
  License along with this program. If not, see <https://www.gnu.org/licenses/>.
12
12
  */
13
- import { CommandManager } from '@cyberismo/data-handler';
13
+ import { scanForProjects } from '@cyberismo/data-handler';
14
14
  import { startServer } from './index.js';
15
15
  import { exportSite } from './export.js';
16
16
  import { MockAuthProvider } from './auth/mock.js';
17
17
  import { KeycloakAuthProvider } from './auth/keycloak.js';
18
18
  import type { AuthProvider } from './auth/types.js';
19
+ import { ProjectRegistry } from './project-registry.js';
20
+ import { parseArgs } from 'node:util';
19
21
  import dotenv from 'dotenv';
20
22
 
21
23
  // Load environment variables from .env file
@@ -55,12 +57,50 @@ function createAuthProvider(): AuthProvider {
55
57
  process.exit(1);
56
58
  }
57
59
 
58
- const projectPath = process.env.npm_config_project_path || '';
59
- const commands = await CommandManager.getInstance(projectPath);
60
+ const projectPath = process.env.npm_config_project_path || process.cwd();
61
+ let projects;
62
+ try {
63
+ projects = await scanForProjects(projectPath);
64
+ } catch (error) {
65
+ console.error(
66
+ error instanceof Error
67
+ ? error.message
68
+ : `Failed to scan for projects in '${projectPath}'`,
69
+ );
70
+ process.exit(1);
71
+ }
72
+
73
+ const { values: args } = parseArgs({
74
+ options: {
75
+ export: { type: 'boolean', default: false },
76
+ 'default-project': { type: 'string' },
77
+ },
78
+ strict: false,
79
+ });
60
80
 
61
- if (process.argv.includes('--export')) {
62
- await exportSite(commands);
81
+ if (args.export) {
82
+ if (projects.length === 0) {
83
+ console.error('No projects found to export.');
84
+ process.exit(1);
85
+ }
86
+ const registry = await ProjectRegistry.fromScannedProjects(projects);
87
+ await exportSite(registry, undefined, {
88
+ defaultProject:
89
+ typeof args['default-project'] === 'string'
90
+ ? args['default-project']
91
+ : undefined,
92
+ });
63
93
  } else {
94
+ if (projects.length === 0) {
95
+ console.error(
96
+ `No projects found in "${projectPath}". Cannot start the server without at least one project.`,
97
+ );
98
+ process.exit(1);
99
+ }
100
+ const autocommit = process.env.CYBERISMO_AUTOCOMMIT === 'true';
101
+ const registry = await ProjectRegistry.fromScannedProjects(projects, {
102
+ autocommit,
103
+ });
64
104
  const authProvider = createAuthProvider();
65
- await startServer(authProvider, commands);
105
+ await startServer(authProvider, registry);
66
106
  }
@@ -42,6 +42,16 @@ export function createAuthMiddleware(
42
42
  if (user) {
43
43
  c.set('user', user);
44
44
  } else {
45
+ // RFC 9728 §5.1: include resource_metadata in WWW-Authenticate
46
+ // only for MCP routes, where the metadata document applies.
47
+ const issuer = process.env.OIDC_ISSUER;
48
+ if (issuer && c.req.path.startsWith('/mcp')) {
49
+ const origin = new URL(issuer).origin;
50
+ const resourceUrl = `${origin}/.well-known/oauth-protected-resource/mcp`;
51
+ return c.json({ error: 'Unauthorized' }, 401, {
52
+ 'WWW-Authenticate': `Bearer resource_metadata="${resourceUrl}"`,
53
+ });
54
+ }
45
55
  return c.json({ error: 'Unauthorized' }, 401);
46
56
  }
47
57
 
@@ -13,28 +13,66 @@
13
13
  import type { Context, MiddlewareHandler } from 'hono';
14
14
  import type { CommandManager } from '@cyberismo/data-handler';
15
15
  import { getCurrentUser } from './auth.js';
16
+ import type { ProjectRegistry } from '../project-registry.js';
16
17
 
17
18
  // Extend Hono Context type to include our custom properties
18
19
  declare module 'hono' {
19
20
  interface ContextVariableMap {
20
21
  commands: CommandManager;
21
22
  projectPath: string;
23
+ registry: ProjectRegistry;
22
24
  }
23
25
  }
24
26
 
27
+ /**
28
+ * Set CommandManager on context and run the next handler as the authenticated user.
29
+ */
30
+ async function runWithCommands(
31
+ c: Context,
32
+ commands: CommandManager,
33
+ next: () => Promise<void>,
34
+ ) {
35
+ const user = getCurrentUser(c);
36
+ if (!user) {
37
+ throw new Error('CommandManager expects a user');
38
+ }
39
+ c.set('commands', commands);
40
+ c.set('projectPath', commands.project.basePath);
41
+ await commands.runAsAuthor({ name: user.name, email: user.email }, () =>
42
+ next(),
43
+ );
44
+ }
45
+
46
+ // TODO: Remove once MCP is made project-scoped via attachProjectRegistry
25
47
  export const attachCommandManager = (
26
48
  commands: CommandManager,
49
+ ): MiddlewareHandler => {
50
+ return (c, next) => runWithCommands(c, commands, next);
51
+ };
52
+
53
+ /**
54
+ * Middleware that resolves the project from the registry and sets the
55
+ * CommandManager on context.
56
+ *
57
+ * @param registry - Project registry to look up projects.
58
+ * @param fixedPrefix - When provided, used instead of the `:prefix` route
59
+ * param. This is needed in export/SSG mode where routes are mounted at
60
+ * concrete paths (e.g. `/api/projects/decision/...`) with no dynamic param.
61
+ */
62
+ export const attachProjectRegistry = (
63
+ registry: ProjectRegistry,
64
+ fixedPrefix?: string,
27
65
  ): MiddlewareHandler => {
28
66
  return async (c: Context, next) => {
29
- c.set('commands', commands);
30
- c.set('projectPath', commands.project.basePath);
31
- const user = getCurrentUser(c);
32
- if (user) {
33
- await commands.runAsAuthor({ name: user.name, email: user.email }, () =>
34
- next(),
35
- );
36
- } else {
37
- throw new Error('CommandManager expects a user');
67
+ c.set('registry', registry);
68
+ const prefix = c.req.param('prefix') ?? fixedPrefix;
69
+ if (!prefix) {
70
+ return c.json({ error: 'Project prefix is required' }, 400);
71
+ }
72
+ const commands = registry.get(prefix);
73
+ if (!commands) {
74
+ return c.json({ error: `Project '${prefix}' not found` }, 404);
38
75
  }
76
+ return runWithCommands(c, commands, next);
39
77
  };
40
78
  };
@@ -0,0 +1,110 @@
1
+ /**
2
+ Cyberismo
3
+ Copyright © Cyberismo Ltd and contributors 2026
4
+ This program is free software: you can redistribute it and/or modify it under
5
+ the terms of the GNU Affero General Public License version 3 as published by
6
+ the Free Software Foundation.
7
+ This program is distributed in the hope that it will be useful, but WITHOUT
8
+ ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
9
+ FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
10
+ details. You should have received a copy of the GNU Affero General Public
11
+ License along with this program. If not, see <https://www.gnu.org/licenses/>.
12
+ */
13
+
14
+ import { CommandManager, type ProjectProvider } from '@cyberismo/data-handler';
15
+
16
+ export type ProjectRegistryEntry = {
17
+ prefix: string;
18
+ commands: CommandManager;
19
+ };
20
+
21
+ export type ProjectListItem = {
22
+ prefix: string;
23
+ name: string;
24
+ category?: string;
25
+ description?: string;
26
+ };
27
+
28
+ /** Minimal project descriptor returned by scanForProjects. */
29
+ export interface ScannedProject {
30
+ path: string;
31
+ prefix: string;
32
+ name: string;
33
+ }
34
+
35
+ export class ProjectRegistry implements ProjectProvider {
36
+ private projects: Map<string, CommandManager> = new Map();
37
+
38
+ constructor(entries: ProjectRegistryEntry[] = []) {
39
+ for (const entry of entries) {
40
+ this.add(entry.prefix, entry.commands);
41
+ }
42
+ }
43
+
44
+ get(prefix: string): CommandManager | undefined {
45
+ return this.projects.get(prefix);
46
+ }
47
+
48
+ has(prefix: string): boolean {
49
+ return this.projects.has(prefix);
50
+ }
51
+
52
+ add(prefix: string, commands: CommandManager): void {
53
+ if (this.projects.has(prefix)) {
54
+ throw new Error(`Project '${prefix}' is already registered`);
55
+ }
56
+ this.projects.set(prefix, commands);
57
+ }
58
+
59
+ list(): ProjectListItem[] {
60
+ return Array.from(this.projects.entries()).map(([prefix, commands]) => ({
61
+ prefix,
62
+ name: commands.project.configuration.name,
63
+ category: commands.project.configuration.category || undefined,
64
+ description: commands.project.configuration.description || undefined,
65
+ }));
66
+ }
67
+
68
+ /** Iterate over all registered CommandManagers. */
69
+ values(): IterableIterator<CommandManager> {
70
+ return this.projects.values();
71
+ }
72
+
73
+ first(): CommandManager | undefined {
74
+ const [first] = this.projects.values();
75
+ return first;
76
+ }
77
+
78
+ dispose(): void {
79
+ for (const commands of this.projects.values()) {
80
+ commands.project.dispose();
81
+ }
82
+ this.projects.clear();
83
+ }
84
+
85
+ /**
86
+ * Create a single-project registry from a CommandManager.
87
+ * Used by export mode and tests where only one project is needed.
88
+ */
89
+ static fromCommandManager(commands: CommandManager): ProjectRegistry {
90
+ return new ProjectRegistry([
91
+ { prefix: commands.project.configuration.cardKeyPrefix, commands },
92
+ ]);
93
+ }
94
+
95
+ /**
96
+ * Build a registry from scanned project entries, initializing each CommandManager.
97
+ */
98
+ static async fromScannedProjects(
99
+ projects: ScannedProject[],
100
+ options?: ConstructorParameters<typeof CommandManager>[1],
101
+ ): Promise<ProjectRegistry> {
102
+ const entries: ProjectRegistryEntry[] = [];
103
+ for (const project of projects) {
104
+ const commands = new CommandManager(project.path, options);
105
+ await commands.initialize();
106
+ entries.push({ prefix: project.prefix, commands });
107
+ }
108
+ return new ProjectRegistry(entries);
109
+ }
110
+ }