@citadel-platform/platform 0.1.0

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 (65) hide show
  1. package/dist/app.d.ts +5 -0
  2. package/dist/app.d.ts.map +1 -0
  3. package/dist/app.js +2 -0
  4. package/dist/app.js.map +1 -0
  5. package/dist/appContract.d.ts +22 -0
  6. package/dist/appContract.d.ts.map +1 -0
  7. package/dist/appContract.js +2 -0
  8. package/dist/appContract.js.map +1 -0
  9. package/dist/appPackageMetadata.d.ts +14 -0
  10. package/dist/appPackageMetadata.d.ts.map +1 -0
  11. package/dist/appPackageMetadata.js +126 -0
  12. package/dist/appPackageMetadata.js.map +1 -0
  13. package/dist/client.d.ts +2 -0
  14. package/dist/client.d.ts.map +1 -0
  15. package/dist/client.js +2 -0
  16. package/dist/client.js.map +1 -0
  17. package/dist/clientAppContract.d.ts +21 -0
  18. package/dist/clientAppContract.d.ts.map +1 -0
  19. package/dist/clientAppContract.js +2 -0
  20. package/dist/clientAppContract.js.map +1 -0
  21. package/dist/generateAppMetadataCli.d.ts +3 -0
  22. package/dist/generateAppMetadataCli.d.ts.map +1 -0
  23. package/dist/generateAppMetadataCli.js +42 -0
  24. package/dist/generateAppMetadataCli.js.map +1 -0
  25. package/dist/installedAppCatalogContract.d.ts +9 -0
  26. package/dist/installedAppCatalogContract.d.ts.map +1 -0
  27. package/dist/installedAppCatalogContract.js +2 -0
  28. package/dist/installedAppCatalogContract.js.map +1 -0
  29. package/dist/persistence.d.ts +2 -0
  30. package/dist/persistence.d.ts.map +1 -0
  31. package/dist/persistence.js +2 -0
  32. package/dist/persistence.js.map +1 -0
  33. package/dist/server-app.d.ts +2 -0
  34. package/dist/server-app.d.ts.map +1 -0
  35. package/dist/server-app.js +2 -0
  36. package/dist/server-app.js.map +1 -0
  37. package/dist/server.d.ts +16 -0
  38. package/dist/server.d.ts.map +1 -0
  39. package/dist/server.js +249 -0
  40. package/dist/server.js.map +1 -0
  41. package/dist/serverApp.d.ts +6 -0
  42. package/dist/serverApp.d.ts.map +1 -0
  43. package/dist/serverApp.js +2 -0
  44. package/dist/serverApp.js.map +1 -0
  45. package/dist/serverAppContract.d.ts +31 -0
  46. package/dist/serverAppContract.d.ts.map +1 -0
  47. package/dist/serverAppContract.js +2 -0
  48. package/dist/serverAppContract.js.map +1 -0
  49. package/dist/shared.d.ts +46 -0
  50. package/dist/shared.d.ts.map +1 -0
  51. package/dist/shared.js +35 -0
  52. package/dist/shared.js.map +1 -0
  53. package/dist/sqlite.d.ts +7 -0
  54. package/dist/sqlite.d.ts.map +1 -0
  55. package/dist/sqlite.js +16 -0
  56. package/dist/sqlite.js.map +1 -0
  57. package/dist/validation.d.ts +9 -0
  58. package/dist/validation.d.ts.map +1 -0
  59. package/dist/validation.js +18 -0
  60. package/dist/validation.js.map +1 -0
  61. package/dist/version.d.ts +2 -0
  62. package/dist/version.d.ts.map +1 -0
  63. package/dist/version.js +2 -0
  64. package/dist/version.js.map +1 -0
  65. package/package.json +60 -0
package/dist/app.d.ts ADDED
@@ -0,0 +1,5 @@
1
+ export type { AppManifest, AppPackageDescriptor } from './appContract.js';
2
+ export type { InstalledAppCatalogEntry } from './installedAppCatalogContract.js';
3
+ export { DEFAULT_SPACE_ID, APP_ID_MAX_LENGTH, APP_ID_PATTERN, DISPLAY_NAME_MAX_LENGTH, GUEST_ID_MAX_LENGTH, GUEST_ID_PATTERN, SPACE_ID_MAX_LENGTH, SPACE_ID_PATTERN, isAppId, normalizeGuestId, normalizeSpaceId, type AppEventEnvelope, type AppId, type JoinSpacePayload, type Participant, type ParticipantEvent, type PlatformErrorPayload, type SpaceState } from './shared.js';
4
+ export type { ValidationResult } from './validation.js';
5
+ //# sourceMappingURL=app.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"app.d.ts","sourceRoot":"","sources":["../src/app.ts"],"names":[],"mappings":"AAAA,YAAY,EAAE,WAAW,EAAE,oBAAoB,EAAE,MAAM,kBAAkB,CAAC;AAC1E,YAAY,EAAE,wBAAwB,EAAE,MAAM,kCAAkC,CAAC;AACjF,OAAO,EACL,gBAAgB,EAChB,iBAAiB,EACjB,cAAc,EACd,uBAAuB,EACvB,mBAAmB,EACnB,gBAAgB,EAChB,mBAAmB,EACnB,gBAAgB,EAChB,OAAO,EACP,gBAAgB,EAChB,gBAAgB,EAChB,KAAK,gBAAgB,EACrB,KAAK,KAAK,EACV,KAAK,gBAAgB,EACrB,KAAK,WAAW,EAChB,KAAK,gBAAgB,EACrB,KAAK,oBAAoB,EACzB,KAAK,UAAU,EAChB,MAAM,aAAa,CAAC;AACrB,YAAY,EAAE,gBAAgB,EAAE,MAAM,iBAAiB,CAAC"}
package/dist/app.js ADDED
@@ -0,0 +1,2 @@
1
+ export { DEFAULT_SPACE_ID, APP_ID_MAX_LENGTH, APP_ID_PATTERN, DISPLAY_NAME_MAX_LENGTH, GUEST_ID_MAX_LENGTH, GUEST_ID_PATTERN, SPACE_ID_MAX_LENGTH, SPACE_ID_PATTERN, isAppId, normalizeGuestId, normalizeSpaceId } from './shared.js';
2
+ //# sourceMappingURL=app.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"app.js","sourceRoot":"","sources":["../src/app.ts"],"names":[],"mappings":"AAEA,OAAO,EACL,gBAAgB,EAChB,iBAAiB,EACjB,cAAc,EACd,uBAAuB,EACvB,mBAAmB,EACnB,gBAAgB,EAChB,mBAAmB,EACnB,gBAAgB,EAChB,OAAO,EACP,gBAAgB,EAChB,gBAAgB,EAQjB,MAAM,aAAa,CAAC"}
@@ -0,0 +1,22 @@
1
+ import type { AppId } from './shared.js';
2
+ export type AppManifest = {
3
+ appId: AppId;
4
+ label: string;
5
+ defaultSpaceId: string;
6
+ persistence: 'none' | 'sqlite';
7
+ version: string;
8
+ };
9
+ export type AppPackageDescriptor = {
10
+ appId: AppId;
11
+ manifest: AppManifest;
12
+ packageName: string;
13
+ client: {
14
+ subpath: './client';
15
+ registrationExport: string;
16
+ };
17
+ server: {
18
+ subpath: './server';
19
+ registrationExport: string;
20
+ };
21
+ };
22
+ //# sourceMappingURL=appContract.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"appContract.d.ts","sourceRoot":"","sources":["../src/appContract.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,KAAK,EAAE,MAAM,aAAa,CAAC;AAEzC,MAAM,MAAM,WAAW,GAAG;IACxB,KAAK,EAAE,KAAK,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,cAAc,EAAE,MAAM,CAAC;IACvB,WAAW,EAAE,MAAM,GAAG,QAAQ,CAAC;IAC/B,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,oBAAoB,GAAG;IACjC,KAAK,EAAE,KAAK,CAAC;IACb,QAAQ,EAAE,WAAW,CAAC;IACtB,WAAW,EAAE,MAAM,CAAC;IACpB,MAAM,EAAE;QACN,OAAO,EAAE,UAAU,CAAC;QACpB,kBAAkB,EAAE,MAAM,CAAC;KAC5B,CAAC;IACF,MAAM,EAAE;QACN,OAAO,EAAE,UAAU,CAAC;QACpB,kBAAkB,EAAE,MAAM,CAAC;KAC5B,CAAC;CACH,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=appContract.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"appContract.js","sourceRoot":"","sources":["../src/appContract.ts"],"names":[],"mappings":""}
@@ -0,0 +1,14 @@
1
+ import type { AppPackageDescriptor } from './appContract.js';
2
+ type PackageJson = {
3
+ name?: unknown;
4
+ citadel?: unknown;
5
+ };
6
+ type GenerateOptions = {
7
+ check?: boolean;
8
+ packageDirs?: string[];
9
+ };
10
+ export declare function parseAppPackageMetadata(packageName: string, packageJson: PackageJson): AppPackageDescriptor;
11
+ export declare function generateAppMetadataSource(packageDir: string): string;
12
+ export declare function generateAppMetadata(options?: GenerateOptions): void;
13
+ export {};
14
+ //# sourceMappingURL=appPackageMetadata.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"appPackageMetadata.d.ts","sourceRoot":"","sources":["../src/appPackageMetadata.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,kBAAkB,CAAC;AAK7D,KAAK,WAAW,GAAG;IACjB,IAAI,CAAC,EAAE,OAAO,CAAC;IACf,OAAO,CAAC,EAAE,OAAO,CAAC;CACnB,CAAC;AAIF,KAAK,eAAe,GAAG;IACrB,KAAK,CAAC,EAAE,OAAO,CAAC;IAChB,WAAW,CAAC,EAAE,MAAM,EAAE,CAAC;CACxB,CAAC;AAqEF,wBAAgB,uBAAuB,CAAC,WAAW,EAAE,MAAM,EAAE,WAAW,EAAE,WAAW,GAAG,oBAAoB,CA2B3G;AAED,wBAAgB,yBAAyB,CAAC,UAAU,EAAE,MAAM,UAqC3D;AAED,wBAAgB,mBAAmB,CAAC,OAAO,GAAE,eAAoB,QA2BhE"}
@@ -0,0 +1,126 @@
1
+ import { existsSync, readFileSync, writeFileSync } from 'node:fs';
2
+ import { join, resolve } from 'node:path';
3
+ const outputFileName = 'generatedMetadata.ts';
4
+ const header = '// Generated by citadel-generate-app-metadata from package.json#citadel.\n';
5
+ function literal(value) {
6
+ return JSON.stringify(value);
7
+ }
8
+ function readRequiredString(metadata, key, packageName) {
9
+ const value = metadata[key];
10
+ if (typeof value !== 'string' || value.length === 0) {
11
+ throw new Error(`App package ${packageName} citadel.${key} must be a non-empty string`);
12
+ }
13
+ return value;
14
+ }
15
+ function readPersistence(metadata, packageName) {
16
+ const value = readRequiredString(metadata, 'persistence', packageName);
17
+ if (value !== 'none' && value !== 'sqlite') {
18
+ throw new Error(`App package ${packageName} citadel.persistence must be "none" or "sqlite"`);
19
+ }
20
+ return value;
21
+ }
22
+ function readRegistrationMetadata(metadata, environment, packageName) {
23
+ const registration = metadata[environment];
24
+ if (!registration || typeof registration !== 'object' || Array.isArray(registration)) {
25
+ throw new Error(`App package ${packageName} citadel.${environment} must declare registration metadata`);
26
+ }
27
+ const registrationMetadata = registration;
28
+ const subpath = readRequiredString(registrationMetadata, 'subpath', packageName);
29
+ if (subpath !== './client' && subpath !== './server') {
30
+ throw new Error(`App package ${packageName} citadel.${environment}.subpath must be "./client" or "./server"`);
31
+ }
32
+ const registrationExport = readRequiredString(registrationMetadata, 'registrationExport', packageName);
33
+ if (environment === 'client') {
34
+ return {
35
+ subpath: './client',
36
+ registrationExport
37
+ };
38
+ }
39
+ return {
40
+ subpath: './server',
41
+ registrationExport
42
+ };
43
+ }
44
+ export function parseAppPackageMetadata(packageName, packageJson) {
45
+ if (packageJson.name !== packageName) {
46
+ throw new Error(`App package mismatch: expected ${packageName}, package.json declares ${String(packageJson.name)}`);
47
+ }
48
+ const metadata = packageJson.citadel;
49
+ if (!metadata || typeof metadata !== 'object' || Array.isArray(metadata)) {
50
+ throw new Error(`App package ${packageName} must declare citadel metadata`);
51
+ }
52
+ const citadelMetadata = metadata;
53
+ const appId = readRequiredString(citadelMetadata, 'appId', packageName);
54
+ return {
55
+ appId,
56
+ manifest: {
57
+ appId,
58
+ label: readRequiredString(citadelMetadata, 'label', packageName),
59
+ defaultSpaceId: readRequiredString(citadelMetadata, 'defaultSpaceId', packageName),
60
+ persistence: readPersistence(citadelMetadata, packageName),
61
+ version: readRequiredString(citadelMetadata, 'version', packageName)
62
+ },
63
+ packageName,
64
+ client: readRegistrationMetadata(citadelMetadata, 'client', packageName),
65
+ server: readRegistrationMetadata(citadelMetadata, 'server', packageName)
66
+ };
67
+ }
68
+ export function generateAppMetadataSource(packageDir) {
69
+ const packageJson = JSON.parse(readFileSync(join(packageDir, 'package.json'), 'utf8'));
70
+ const packageName = packageJson.name;
71
+ if (typeof packageName !== 'string' || packageName.length === 0) {
72
+ throw new Error(`App package at ${packageDir} must declare package.json name`);
73
+ }
74
+ const descriptor = parseAppPackageMetadata(packageName, packageJson);
75
+ return [
76
+ header,
77
+ "import type { AppManifest, AppPackageDescriptor } from '@citadel-platform/platform/app';",
78
+ '',
79
+ 'export const generatedManifest = {',
80
+ ` appId: ${literal(descriptor.manifest.appId)},`,
81
+ ` label: ${literal(descriptor.manifest.label)},`,
82
+ ` defaultSpaceId: ${literal(descriptor.manifest.defaultSpaceId)},`,
83
+ ` persistence: ${literal(descriptor.manifest.persistence)},`,
84
+ ` version: ${literal(descriptor.manifest.version)}`,
85
+ '} satisfies AppManifest;',
86
+ '',
87
+ 'export const generatedAppPackage = {',
88
+ ' appId: generatedManifest.appId,',
89
+ ' manifest: generatedManifest,',
90
+ ` packageName: ${literal(descriptor.packageName)},`,
91
+ ' client: {',
92
+ ` subpath: ${literal(descriptor.client.subpath)},`,
93
+ ` registrationExport: ${literal(descriptor.client.registrationExport)}`,
94
+ ' },',
95
+ ' server: {',
96
+ ` subpath: ${literal(descriptor.server.subpath)},`,
97
+ ` registrationExport: ${literal(descriptor.server.registrationExport)}`,
98
+ ' }',
99
+ '} satisfies AppPackageDescriptor;',
100
+ ''
101
+ ].join('\n');
102
+ }
103
+ export function generateAppMetadata(options = {}) {
104
+ const packageDirs = options.packageDirs?.length ? options.packageDirs : [process.cwd()];
105
+ const staleFiles = [];
106
+ for (const packageDirInput of packageDirs) {
107
+ const packageDir = resolve(packageDirInput);
108
+ const outputPath = join(packageDir, 'src', outputFileName);
109
+ const expectedSource = generateAppMetadataSource(packageDir);
110
+ const currentSource = existsSync(outputPath) ? readFileSync(outputPath, 'utf8') : undefined;
111
+ if (options.check) {
112
+ if (currentSource !== expectedSource) {
113
+ staleFiles.push(outputPath);
114
+ }
115
+ continue;
116
+ }
117
+ writeFileSync(outputPath, expectedSource);
118
+ }
119
+ if (staleFiles.length > 0) {
120
+ throw new Error([
121
+ 'Generated app metadata is stale. Run citadel-generate-app-metadata.',
122
+ ...staleFiles.map((filePath) => `- ${filePath}`)
123
+ ].join('\n'));
124
+ }
125
+ }
126
+ //# sourceMappingURL=appPackageMetadata.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"appPackageMetadata.js","sourceRoot":"","sources":["../src/appPackageMetadata.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export type { AppViewProps, ClientAppModule, ClientAppRegistration } from './clientAppContract.js';
2
+ //# sourceMappingURL=client.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"client.d.ts","sourceRoot":"","sources":["../src/client.ts"],"names":[],"mappings":"AAAA,YAAY,EACV,YAAY,EACZ,eAAe,EACf,qBAAqB,EACtB,MAAM,wBAAwB,CAAC"}
package/dist/client.js ADDED
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=client.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"client.js","sourceRoot":"","sources":["../src/client.ts"],"names":[],"mappings":""}
@@ -0,0 +1,21 @@
1
+ import type { ComponentType } from 'react';
2
+ import type { AppId, Participant } from './shared.js';
3
+ export type AppViewProps<TState = unknown> = {
4
+ currentParticipant: Participant;
5
+ spaceId: string;
6
+ participants: Participant[];
7
+ appState: TState;
8
+ sendAppEvent(type: string, payload?: unknown): void;
9
+ setNotice(message: string): void;
10
+ };
11
+ export type ClientAppModule<TState = unknown> = {
12
+ appId: AppId;
13
+ label: string;
14
+ defaultSpaceId: string;
15
+ View: ComponentType<AppViewProps<TState>>;
16
+ };
17
+ export type ClientAppRegistration<TState = unknown> = {
18
+ appId: AppId;
19
+ clientApp: ClientAppModule<TState>;
20
+ };
21
+ //# sourceMappingURL=clientAppContract.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"clientAppContract.d.ts","sourceRoot":"","sources":["../src/clientAppContract.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,OAAO,CAAC;AAC3C,OAAO,KAAK,EAAE,KAAK,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAEtD,MAAM,MAAM,YAAY,CAAC,MAAM,GAAG,OAAO,IAAI;IAC3C,kBAAkB,EAAE,WAAW,CAAC;IAChC,OAAO,EAAE,MAAM,CAAC;IAChB,YAAY,EAAE,WAAW,EAAE,CAAC;IAC5B,QAAQ,EAAE,MAAM,CAAC;IACjB,YAAY,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,OAAO,GAAG,IAAI,CAAC;IACpD,SAAS,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI,CAAC;CAClC,CAAC;AAEF,MAAM,MAAM,eAAe,CAAC,MAAM,GAAG,OAAO,IAAI;IAC9C,KAAK,EAAE,KAAK,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,cAAc,EAAE,MAAM,CAAC;IACvB,IAAI,EAAE,aAAa,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC;CAC3C,CAAC;AAEF,MAAM,MAAM,qBAAqB,CAAC,MAAM,GAAG,OAAO,IAAI;IACpD,KAAK,EAAE,KAAK,CAAC;IACb,SAAS,EAAE,eAAe,CAAC,MAAM,CAAC,CAAC;CACpC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=clientAppContract.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"clientAppContract.js","sourceRoot":"","sources":["../src/clientAppContract.ts"],"names":[],"mappings":""}
@@ -0,0 +1,3 @@
1
+ #!/usr/bin/env node
2
+ export {};
3
+ //# sourceMappingURL=generateAppMetadataCli.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"generateAppMetadataCli.d.ts","sourceRoot":"","sources":["../src/generateAppMetadataCli.ts"],"names":[],"mappings":""}
@@ -0,0 +1,42 @@
1
+ #!/usr/bin/env node
2
+ import { generateAppMetadata } from './appPackageMetadata.js';
3
+ function usage() {
4
+ return [
5
+ 'Usage: citadel-generate-app-metadata [--check] [--package-dir <dir>]',
6
+ '',
7
+ 'Generates app runtime metadata source from package.json#citadel.'
8
+ ].join('\n');
9
+ }
10
+ function parseArgs(argv) {
11
+ const packageDirs = [];
12
+ let check = false;
13
+ for (let index = 0; index < argv.length; index += 1) {
14
+ const token = argv[index];
15
+ if (token === '--check') {
16
+ check = true;
17
+ continue;
18
+ }
19
+ if (token === '--package-dir') {
20
+ const packageDir = argv[index + 1];
21
+ if (!packageDir) {
22
+ throw new Error(`${usage()}\n\n--package-dir requires a value`);
23
+ }
24
+ packageDirs.push(packageDir);
25
+ index += 1;
26
+ continue;
27
+ }
28
+ throw new Error(`${usage()}\n\nUnknown option: ${token}`);
29
+ }
30
+ return {
31
+ check,
32
+ packageDirs
33
+ };
34
+ }
35
+ try {
36
+ generateAppMetadata(parseArgs(process.argv.slice(2)));
37
+ }
38
+ catch (error) {
39
+ console.error(error instanceof Error ? error.message : error);
40
+ process.exitCode = 1;
41
+ }
42
+ //# sourceMappingURL=generateAppMetadataCli.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"generateAppMetadataCli.js","sourceRoot":"","sources":["../src/generateAppMetadataCli.ts"],"names":[],"mappings":";AACA,OAAO,EAAE,mBAAmB,EAAE,MAAM,yBAAyB,CAAC;AAE9D,SAAS,KAAK;IACZ,OAAO;QACL,sEAAsE;QACtE,EAAE;QACF,kEAAkE;KACnE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AACf,CAAC;AAED,SAAS,SAAS,CAAC,IAAc;IAC/B,MAAM,WAAW,GAAa,EAAE,CAAC;IACjC,IAAI,KAAK,GAAG,KAAK,CAAC;IAElB,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,IAAI,CAAC,MAAM,EAAE,KAAK,IAAI,CAAC,EAAE,CAAC;QACpD,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC;QAE1B,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;YACxB,KAAK,GAAG,IAAI,CAAC;YACb,SAAS;QACX,CAAC;QAED,IAAI,KAAK,KAAK,eAAe,EAAE,CAAC;YAC9B,MAAM,UAAU,GAAG,IAAI,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;YAEnC,IAAI,CAAC,UAAU,EAAE,CAAC;gBAChB,MAAM,IAAI,KAAK,CAAC,GAAG,KAAK,EAAE,oCAAoC,CAAC,CAAC;YAClE,CAAC;YAED,WAAW,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;YAC7B,KAAK,IAAI,CAAC,CAAC;YACX,SAAS;QACX,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,GAAG,KAAK,EAAE,uBAAuB,KAAK,EAAE,CAAC,CAAC;IAC5D,CAAC;IAED,OAAO;QACL,KAAK;QACL,WAAW;KACZ,CAAC;AACJ,CAAC;AAED,IAAI,CAAC;IACH,mBAAmB,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;AACxD,CAAC;AAAC,OAAO,KAAK,EAAE,CAAC;IACf,OAAO,CAAC,KAAK,CAAC,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;IAC9D,OAAO,CAAC,QAAQ,GAAG,CAAC,CAAC;AACvB,CAAC"}
@@ -0,0 +1,9 @@
1
+ import type { AppPackageDescriptor } from './appContract.js';
2
+ import type { ClientAppRegistration } from './clientAppContract.js';
3
+ import type { ServerAppRegistration } from './serverAppContract.js';
4
+ export type InstalledAppCatalogEntry<TState = any, TServices = any> = {
5
+ descriptor: AppPackageDescriptor;
6
+ clientRegistration: ClientAppRegistration<TState>;
7
+ serverRegistration: ServerAppRegistration<TServices>;
8
+ };
9
+ //# sourceMappingURL=installedAppCatalogContract.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"installedAppCatalogContract.d.ts","sourceRoot":"","sources":["../src/installedAppCatalogContract.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,kBAAkB,CAAC;AAC7D,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,wBAAwB,CAAC;AACpE,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,wBAAwB,CAAC;AAEpE,MAAM,MAAM,wBAAwB,CAClC,MAAM,GAAG,GAAG,EACZ,SAAS,GAAG,GAAG,IACb;IACF,UAAU,EAAE,oBAAoB,CAAC;IACjC,kBAAkB,EAAE,qBAAqB,CAAC,MAAM,CAAC,CAAC;IAClD,kBAAkB,EAAE,qBAAqB,CAAC,SAAS,CAAC,CAAC;CACtD,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=installedAppCatalogContract.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"installedAppCatalogContract.js","sourceRoot":"","sources":["../src/installedAppCatalogContract.ts"],"names":[],"mappings":""}
@@ -0,0 +1,2 @@
1
+ export { openCitadelDatabase, type CitadelDatabase } from './sqlite.js';
2
+ //# sourceMappingURL=persistence.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"persistence.d.ts","sourceRoot":"","sources":["../src/persistence.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,mBAAmB,EAAE,KAAK,eAAe,EAAE,MAAM,aAAa,CAAC"}
@@ -0,0 +1,2 @@
1
+ export { openCitadelDatabase } from './sqlite.js';
2
+ //# sourceMappingURL=persistence.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"persistence.js","sourceRoot":"","sources":["../src/persistence.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,mBAAmB,EAAwB,MAAM,aAAa,CAAC"}
@@ -0,0 +1,2 @@
1
+ export * from './serverApp.js';
2
+ //# sourceMappingURL=server-app.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server-app.d.ts","sourceRoot":"","sources":["../src/server-app.ts"],"names":[],"mappings":"AAAA,cAAc,gBAAgB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export * from './serverApp.js';
2
+ //# sourceMappingURL=server-app.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server-app.js","sourceRoot":"","sources":["../src/server-app.ts"],"names":[],"mappings":"AAAA,cAAc,gBAAgB,CAAC"}
@@ -0,0 +1,16 @@
1
+ import { Server } from 'socket.io';
2
+ import type { AppManifest } from './appContract.js';
3
+ import type { ServerAppModule } from './serverAppContract.js';
4
+ export type PlatformServerOptions = {
5
+ clientOrigin?: string;
6
+ staticDir?: string;
7
+ apps: ServerAppModule[];
8
+ appManifests?: AppManifest[];
9
+ };
10
+ export declare function createPlatformServer(options: PlatformServerOptions): {
11
+ app: import("express-serve-static-core").Express;
12
+ httpServer: import("http").Server<typeof import("http").IncomingMessage, typeof import("http").ServerResponse>;
13
+ io: Server<import("socket.io").DefaultEventsMap, import("socket.io").DefaultEventsMap, import("socket.io").DefaultEventsMap, any>;
14
+ apps: Map<string, ServerAppModule>;
15
+ };
16
+ //# sourceMappingURL=server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,MAAM,EAAe,MAAM,WAAW,CAAC;AAYhD,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,kBAAkB,CAAC;AACpD,OAAO,KAAK,EAAoB,eAAe,EAAE,MAAM,wBAAwB,CAAC;AAIhF,MAAM,MAAM,qBAAqB,GAAG;IAClC,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,IAAI,EAAE,eAAe,EAAE,CAAC;IACxB,YAAY,CAAC,EAAE,WAAW,EAAE,CAAC;CAC9B,CAAC;AAeF,wBAAgB,oBAAoB,CAAC,OAAO,EAAE,qBAAqB;;;;;EA2RlE"}
package/dist/server.js ADDED
@@ -0,0 +1,249 @@
1
+ import express from 'express';
2
+ import { createServer } from 'node:http';
3
+ import { join } from 'node:path';
4
+ import { nanoid } from 'nanoid';
5
+ import { Server } from 'socket.io';
6
+ import { DEFAULT_SPACE_ID, isAppId, normalizeGuestId, normalizeSpaceId } from './shared.js';
7
+ import { validateDisplayName } from './validation.js';
8
+ import { PLATFORM_VERSION } from './version.js';
9
+ export function createPlatformServer(options) {
10
+ const clientOrigin = options.clientOrigin ?? 'http://localhost:5173';
11
+ const app = express();
12
+ const httpServer = createServer(app);
13
+ const io = new Server(httpServer, {
14
+ cors: {
15
+ origin: clientOrigin,
16
+ methods: ['GET', 'POST']
17
+ }
18
+ });
19
+ const modules = new Map();
20
+ const manifests = new Map();
21
+ const sessions = new Map();
22
+ const appState = new Map();
23
+ const defaultAppId = options.apps[0]?.appId;
24
+ for (const module of options.apps) {
25
+ modules.set(module.appId, module);
26
+ }
27
+ for (const manifest of options.appManifests ?? []) {
28
+ if (modules.has(manifest.appId)) {
29
+ manifests.set(manifest.appId, manifest);
30
+ }
31
+ }
32
+ function spaceKey(appId, spaceId) {
33
+ return `${appId}:${spaceId}`;
34
+ }
35
+ function socketRoom(appId, spaceId) {
36
+ return `space:${spaceKey(appId, spaceId)}`;
37
+ }
38
+ function getParticipants(appId, spaceId) {
39
+ return [...sessions.values()]
40
+ .filter((session) => session.appId === appId && session.spaceId === spaceId)
41
+ .map((session) => session.participant)
42
+ .sort((a, b) => a.name.localeCompare(b.name));
43
+ }
44
+ function makeContext(socket, session) {
45
+ const key = spaceKey(session.appId, session.spaceId);
46
+ const module = modules.get(session.appId);
47
+ if (!module) {
48
+ throw new Error(`Unknown app module: ${session.appId}`);
49
+ }
50
+ return {
51
+ appId: session.appId,
52
+ spaceId: session.spaceId,
53
+ socketId: socket.id,
54
+ participant: session.participant,
55
+ participants: getParticipants(session.appId, session.spaceId),
56
+ emitToSpace(type, payload) {
57
+ io.to(socketRoom(session.appId, session.spaceId)).emit('app:event', {
58
+ appId: session.appId,
59
+ type,
60
+ payload
61
+ });
62
+ },
63
+ emitToParticipant(type, payload) {
64
+ socket.emit('app:event', {
65
+ appId: session.appId,
66
+ type,
67
+ payload
68
+ });
69
+ },
70
+ emitSpaceState() {
71
+ emitSpaceState(session.appId, session.spaceId);
72
+ },
73
+ getAppState() {
74
+ return appState.get(key);
75
+ },
76
+ setAppState(state) {
77
+ appState.set(key, state);
78
+ },
79
+ clearAppState() {
80
+ appState.delete(key);
81
+ }
82
+ };
83
+ }
84
+ function getSpaceState(appId, spaceId) {
85
+ const module = modules.get(appId);
86
+ if (!module) {
87
+ return { appId, spaceId, participants: [], appState: null };
88
+ }
89
+ const key = spaceKey(appId, spaceId);
90
+ const baseContext = {
91
+ appId,
92
+ spaceId,
93
+ participants: getParticipants(appId, spaceId),
94
+ emitToSpace(type, payload) {
95
+ io.to(socketRoom(appId, spaceId)).emit('app:event', { appId, type, payload });
96
+ },
97
+ emitToParticipant() { },
98
+ emitSpaceState() {
99
+ emitSpaceState(appId, spaceId);
100
+ },
101
+ getAppState() {
102
+ return appState.get(key);
103
+ },
104
+ setAppState(state) {
105
+ appState.set(key, state);
106
+ },
107
+ clearAppState() {
108
+ appState.delete(key);
109
+ }
110
+ };
111
+ return {
112
+ appId,
113
+ spaceId,
114
+ participants: baseContext.participants,
115
+ appState: module.getInitialState(baseContext)
116
+ };
117
+ }
118
+ function emitSpaceState(appId, spaceId) {
119
+ io.to(socketRoom(appId, spaceId)).emit('space:state', getSpaceState(appId, spaceId));
120
+ }
121
+ function leaveCurrentSpace(socket, notifyParticipant = true) {
122
+ const session = sessions.get(socket.id);
123
+ if (!session) {
124
+ return;
125
+ }
126
+ const module = modules.get(session.appId);
127
+ sessions.delete(socket.id);
128
+ const context = makeContext(socket, session);
129
+ module?.onParticipantLeft?.(context);
130
+ socket.leave(socketRoom(session.appId, session.spaceId));
131
+ if (notifyParticipant) {
132
+ socket.to(socketRoom(session.appId, session.spaceId)).emit('participant:left', {
133
+ id: nanoid(),
134
+ type: 'participant:left',
135
+ appId: session.appId,
136
+ spaceId: session.spaceId,
137
+ participant: session.participant,
138
+ createdAt: new Date().toISOString()
139
+ });
140
+ }
141
+ if (getParticipants(session.appId, session.spaceId).length === 0) {
142
+ appState.delete(spaceKey(session.appId, session.spaceId));
143
+ }
144
+ emitSpaceState(session.appId, session.spaceId);
145
+ }
146
+ function getPlatformMetadata() {
147
+ const apps = [...modules.keys()];
148
+ return {
149
+ apps,
150
+ appCount: apps.length,
151
+ appManifests: apps
152
+ .map((appId) => manifests.get(appId))
153
+ .filter((manifest) => Boolean(manifest)),
154
+ version: PLATFORM_VERSION
155
+ };
156
+ }
157
+ app.get('/health', (_request, response) => {
158
+ response.json({
159
+ ...getPlatformMetadata(),
160
+ ok: true,
161
+ participants: sessions.size
162
+ });
163
+ });
164
+ app.get('/config', (_request, response) => {
165
+ const { apps, appManifests } = getPlatformMetadata();
166
+ response.json({
167
+ apps,
168
+ appManifests
169
+ });
170
+ });
171
+ if (options.staticDir) {
172
+ const indexPath = join(options.staticDir, 'index.html');
173
+ app.use(express.static(options.staticDir, { index: false }));
174
+ app.get(/.*/, (_request, response) => {
175
+ response.sendFile(indexPath);
176
+ });
177
+ }
178
+ io.on('connection', (socket) => {
179
+ socket.on('space:join', (payload = { appId: defaultAppId ?? '', name: '' }) => {
180
+ const appId = isAppId(payload.appId) ? payload.appId : defaultAppId;
181
+ if (!appId) {
182
+ socket.emit('error:notice', { message: 'Unknown app.' });
183
+ return;
184
+ }
185
+ const module = modules.get(appId);
186
+ if (!module) {
187
+ socket.emit('error:notice', { message: 'Unknown app.' });
188
+ return;
189
+ }
190
+ const result = validateDisplayName(payload.name);
191
+ const spaceId = normalizeSpaceId(payload.spaceId ?? DEFAULT_SPACE_ID);
192
+ if (!result.ok) {
193
+ socket.emit('error:notice', { message: result.error });
194
+ return;
195
+ }
196
+ const previousSession = sessions.get(socket.id);
197
+ const previousKey = previousSession
198
+ ? spaceKey(previousSession.appId, previousSession.spaceId)
199
+ : null;
200
+ const nextKey = spaceKey(appId, spaceId);
201
+ if (previousSession && previousKey !== nextKey) {
202
+ leaveCurrentSpace(socket);
203
+ }
204
+ const participant = {
205
+ id: normalizeGuestId(payload.guestId, socket.id),
206
+ socketId: socket.id,
207
+ name: result.value
208
+ };
209
+ const session = { appId, spaceId, participant };
210
+ sessions.set(socket.id, session);
211
+ socket.join(socketRoom(appId, spaceId));
212
+ if (!previousSession || previousKey !== nextKey) {
213
+ socket.to(socketRoom(appId, spaceId)).emit('participant:joined', {
214
+ id: nanoid(),
215
+ type: 'participant:joined',
216
+ appId,
217
+ spaceId,
218
+ participant,
219
+ createdAt: new Date().toISOString()
220
+ });
221
+ }
222
+ module.onParticipantJoined?.(makeContext(socket, session));
223
+ emitSpaceState(appId, spaceId);
224
+ });
225
+ socket.on('app:event', (event) => {
226
+ const session = sessions.get(socket.id);
227
+ if (!session) {
228
+ socket.emit('error:notice', { message: 'Join a space before sending events.' });
229
+ return;
230
+ }
231
+ if (!event || event.appId !== session.appId) {
232
+ socket.emit('error:notice', { message: 'Event app does not match the current space.' });
233
+ return;
234
+ }
235
+ const module = modules.get(session.appId);
236
+ module?.handleEvent(makeContext(socket, session), event);
237
+ });
238
+ socket.on('disconnect', () => {
239
+ leaveCurrentSpace(socket);
240
+ });
241
+ });
242
+ return {
243
+ app,
244
+ httpServer,
245
+ io,
246
+ apps: modules
247
+ };
248
+ }
249
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,6 @@
1
+ import type { CitadelDatabase } from './persistence.js';
2
+ export type { ServerAppBundle, ServerAppContext, ServerAppModule, ServerAppRegistration } from './serverAppContract.js';
3
+ export type ServerAppServices = {
4
+ database: CitadelDatabase;
5
+ };
6
+ //# sourceMappingURL=serverApp.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"serverApp.d.ts","sourceRoot":"","sources":["../src/serverApp.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,kBAAkB,CAAC;AAExD,YAAY,EACV,eAAe,EACf,gBAAgB,EAChB,eAAe,EACf,qBAAqB,EACtB,MAAM,wBAAwB,CAAC;AAEhC,MAAM,MAAM,iBAAiB,GAAG;IAC9B,QAAQ,EAAE,eAAe,CAAC;CAC3B,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=serverApp.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"serverApp.js","sourceRoot":"","sources":["../src/serverApp.ts"],"names":[],"mappings":""}
@@ -0,0 +1,31 @@
1
+ import type { AppEventEnvelope, AppId, Participant } from './shared.js';
2
+ export type ServerAppContext = {
3
+ appId: AppId;
4
+ spaceId: string;
5
+ socketId: string;
6
+ participant: Participant;
7
+ participants: Participant[];
8
+ emitToSpace(type: string, payload?: unknown): void;
9
+ emitToParticipant(type: string, payload?: unknown): void;
10
+ emitSpaceState(): void;
11
+ getAppState<T>(): T | undefined;
12
+ setAppState<T>(state: T): void;
13
+ clearAppState(): void;
14
+ };
15
+ export type ServerAppModule = {
16
+ appId: AppId;
17
+ getInitialState(context: Omit<ServerAppContext, 'participant' | 'socketId'>): unknown;
18
+ handleEvent(context: ServerAppContext, event: AppEventEnvelope): void;
19
+ onParticipantJoined?(context: ServerAppContext): void;
20
+ onParticipantLeft?(context: ServerAppContext): void;
21
+ };
22
+ export type ServerAppBundle<TServices> = {
23
+ appId: AppId;
24
+ createServerApp(services: TServices): ServerAppModule;
25
+ };
26
+ export type ServerAppRegistration<TServices> = {
27
+ appId: AppId;
28
+ bundle: ServerAppBundle<TServices>;
29
+ createServerApp(services: TServices): ServerAppModule;
30
+ };
31
+ //# sourceMappingURL=serverAppContract.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"serverAppContract.d.ts","sourceRoot":"","sources":["../src/serverAppContract.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAExE,MAAM,MAAM,gBAAgB,GAAG;IAC7B,KAAK,EAAE,KAAK,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,MAAM,CAAC;IACjB,WAAW,EAAE,WAAW,CAAC;IACzB,YAAY,EAAE,WAAW,EAAE,CAAC;IAC5B,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,OAAO,GAAG,IAAI,CAAC;IACnD,iBAAiB,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,OAAO,GAAG,IAAI,CAAC;IACzD,cAAc,IAAI,IAAI,CAAC;IACvB,WAAW,CAAC,CAAC,KAAK,CAAC,GAAG,SAAS,CAAC;IAChC,WAAW,CAAC,CAAC,EAAE,KAAK,EAAE,CAAC,GAAG,IAAI,CAAC;IAC/B,aAAa,IAAI,IAAI,CAAC;CACvB,CAAC;AAEF,MAAM,MAAM,eAAe,GAAG;IAC5B,KAAK,EAAE,KAAK,CAAC;IACb,eAAe,CAAC,OAAO,EAAE,IAAI,CAAC,gBAAgB,EAAE,aAAa,GAAG,UAAU,CAAC,GAAG,OAAO,CAAC;IACtF,WAAW,CAAC,OAAO,EAAE,gBAAgB,EAAE,KAAK,EAAE,gBAAgB,GAAG,IAAI,CAAC;IACtE,mBAAmB,CAAC,CAAC,OAAO,EAAE,gBAAgB,GAAG,IAAI,CAAC;IACtD,iBAAiB,CAAC,CAAC,OAAO,EAAE,gBAAgB,GAAG,IAAI,CAAC;CACrD,CAAC;AAEF,MAAM,MAAM,eAAe,CAAC,SAAS,IAAI;IACvC,KAAK,EAAE,KAAK,CAAC;IACb,eAAe,CAAC,QAAQ,EAAE,SAAS,GAAG,eAAe,CAAC;CACvD,CAAC;AAEF,MAAM,MAAM,qBAAqB,CAAC,SAAS,IAAI;IAC7C,KAAK,EAAE,KAAK,CAAC;IACb,MAAM,EAAE,eAAe,CAAC,SAAS,CAAC,CAAC;IACnC,eAAe,CAAC,QAAQ,EAAE,SAAS,GAAG,eAAe,CAAC;CACvD,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=serverAppContract.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"serverAppContract.js","sourceRoot":"","sources":["../src/serverAppContract.ts"],"names":[],"mappings":""}
@@ -0,0 +1,46 @@
1
+ export declare const DISPLAY_NAME_MAX_LENGTH = 24;
2
+ export declare const DEFAULT_SPACE_ID = "general";
3
+ export declare const SPACE_ID_MAX_LENGTH = 32;
4
+ export declare const SPACE_ID_PATTERN: RegExp;
5
+ export declare const GUEST_ID_MAX_LENGTH = 80;
6
+ export declare const GUEST_ID_PATTERN: RegExp;
7
+ export declare const APP_ID_MAX_LENGTH = 64;
8
+ export declare const APP_ID_PATTERN: RegExp;
9
+ export type AppId = string;
10
+ export type Participant = {
11
+ id: string;
12
+ socketId?: string;
13
+ name: string;
14
+ };
15
+ export type SpaceState<TAppState = unknown> = {
16
+ appId: AppId;
17
+ spaceId: string;
18
+ participants: Participant[];
19
+ appState: TAppState;
20
+ };
21
+ export type JoinSpacePayload = {
22
+ appId: AppId;
23
+ spaceId?: string;
24
+ guestId?: string;
25
+ name: string;
26
+ };
27
+ export type PlatformErrorPayload = {
28
+ message: string;
29
+ };
30
+ export type ParticipantEvent = {
31
+ id: string;
32
+ type: 'participant:joined' | 'participant:left';
33
+ appId: AppId;
34
+ spaceId: string;
35
+ participant: Participant;
36
+ createdAt: string;
37
+ };
38
+ export type AppEventEnvelope<TPayload = unknown> = {
39
+ appId: AppId;
40
+ type: string;
41
+ payload?: TPayload;
42
+ };
43
+ export declare function normalizeSpaceId(input: unknown): string;
44
+ export declare function isAppId(value: unknown): value is AppId;
45
+ export declare function normalizeGuestId(input: unknown, fallback: string): string;
46
+ //# sourceMappingURL=shared.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shared.d.ts","sourceRoot":"","sources":["../src/shared.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,uBAAuB,KAAK,CAAC;AAC1C,eAAO,MAAM,gBAAgB,YAAY,CAAC;AAC1C,eAAO,MAAM,mBAAmB,KAAK,CAAC;AACtC,eAAO,MAAM,gBAAgB,QAAiB,CAAC;AAC/C,eAAO,MAAM,mBAAmB,KAAK,CAAC;AACtC,eAAO,MAAM,gBAAgB,QAAqB,CAAC;AACnD,eAAO,MAAM,iBAAiB,KAAK,CAAC;AACpC,eAAO,MAAM,cAAc,QAAiB,CAAC;AAE7C,MAAM,MAAM,KAAK,GAAG,MAAM,CAAC;AAE3B,MAAM,MAAM,WAAW,GAAG;IACxB,EAAE,EAAE,MAAM,CAAC;IACX,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,IAAI,EAAE,MAAM,CAAC;CACd,CAAC;AAEF,MAAM,MAAM,UAAU,CAAC,SAAS,GAAG,OAAO,IAAI;IAC5C,KAAK,EAAE,KAAK,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,YAAY,EAAE,WAAW,EAAE,CAAC;IAC5B,QAAQ,EAAE,SAAS,CAAC;CACrB,CAAC;AAEF,MAAM,MAAM,gBAAgB,GAAG;IAC7B,KAAK,EAAE,KAAK,CAAC;IACb,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,IAAI,EAAE,MAAM,CAAC;CACd,CAAC;AAEF,MAAM,MAAM,oBAAoB,GAAG;IACjC,OAAO,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,MAAM,MAAM,gBAAgB,GAAG;IAC7B,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,oBAAoB,GAAG,kBAAkB,CAAC;IAChD,KAAK,EAAE,KAAK,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,EAAE,WAAW,CAAC;IACzB,SAAS,EAAE,MAAM,CAAC;CACnB,CAAC;AAEF,MAAM,MAAM,gBAAgB,CAAC,QAAQ,GAAG,OAAO,IAAI;IACjD,KAAK,EAAE,KAAK,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,CAAC,EAAE,QAAQ,CAAC;CACpB,CAAC;AAEF,wBAAgB,gBAAgB,CAAC,KAAK,EAAE,OAAO,GAAG,MAAM,CAYvD;AAED,wBAAgB,OAAO,CAAC,KAAK,EAAE,OAAO,GAAG,KAAK,IAAI,KAAK,CAOtD;AAED,wBAAgB,gBAAgB,CAAC,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,GAAG,MAAM,CAYzE"}
package/dist/shared.js ADDED
@@ -0,0 +1,35 @@
1
+ export const DISPLAY_NAME_MAX_LENGTH = 24;
2
+ export const DEFAULT_SPACE_ID = 'general';
3
+ export const SPACE_ID_MAX_LENGTH = 32;
4
+ export const SPACE_ID_PATTERN = /^[a-z0-9-]+$/;
5
+ export const GUEST_ID_MAX_LENGTH = 80;
6
+ export const GUEST_ID_PATTERN = /^[A-Za-z0-9_-]+$/;
7
+ export const APP_ID_MAX_LENGTH = 64;
8
+ export const APP_ID_PATTERN = /^[a-z0-9-]+$/;
9
+ export function normalizeSpaceId(input) {
10
+ if (typeof input !== 'string') {
11
+ return DEFAULT_SPACE_ID;
12
+ }
13
+ const value = input.trim().toLowerCase();
14
+ if (!value || value.length > SPACE_ID_MAX_LENGTH || !SPACE_ID_PATTERN.test(value)) {
15
+ return DEFAULT_SPACE_ID;
16
+ }
17
+ return value;
18
+ }
19
+ export function isAppId(value) {
20
+ return (typeof value === 'string' &&
21
+ value.length > 0 &&
22
+ value.length <= APP_ID_MAX_LENGTH &&
23
+ APP_ID_PATTERN.test(value));
24
+ }
25
+ export function normalizeGuestId(input, fallback) {
26
+ if (typeof input !== 'string') {
27
+ return fallback;
28
+ }
29
+ const value = input.trim();
30
+ if (!value || value.length > GUEST_ID_MAX_LENGTH || !GUEST_ID_PATTERN.test(value)) {
31
+ return fallback;
32
+ }
33
+ return value;
34
+ }
35
+ //# sourceMappingURL=shared.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shared.js","sourceRoot":"","sources":["../src/shared.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,uBAAuB,GAAG,EAAE,CAAC;AAC1C,MAAM,CAAC,MAAM,gBAAgB,GAAG,SAAS,CAAC;AAC1C,MAAM,CAAC,MAAM,mBAAmB,GAAG,EAAE,CAAC;AACtC,MAAM,CAAC,MAAM,gBAAgB,GAAG,cAAc,CAAC;AAC/C,MAAM,CAAC,MAAM,mBAAmB,GAAG,EAAE,CAAC;AACtC,MAAM,CAAC,MAAM,gBAAgB,GAAG,kBAAkB,CAAC;AACnD,MAAM,CAAC,MAAM,iBAAiB,GAAG,EAAE,CAAC;AACpC,MAAM,CAAC,MAAM,cAAc,GAAG,cAAc,CAAC;AA2C7C,MAAM,UAAU,gBAAgB,CAAC,KAAc;IAC7C,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QAC9B,OAAO,gBAAgB,CAAC;IAC1B,CAAC;IAED,MAAM,KAAK,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;IAEzC,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,MAAM,GAAG,mBAAmB,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;QAClF,OAAO,gBAAgB,CAAC;IAC1B,CAAC;IAED,OAAO,KAAK,CAAC;AACf,CAAC;AAED,MAAM,UAAU,OAAO,CAAC,KAAc;IACpC,OAAO,CACL,OAAO,KAAK,KAAK,QAAQ;QACzB,KAAK,CAAC,MAAM,GAAG,CAAC;QAChB,KAAK,CAAC,MAAM,IAAI,iBAAiB;QACjC,cAAc,CAAC,IAAI,CAAC,KAAK,CAAC,CAC3B,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,gBAAgB,CAAC,KAAc,EAAE,QAAgB;IAC/D,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QAC9B,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED,MAAM,KAAK,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC;IAE3B,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,MAAM,GAAG,mBAAmB,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;QAClF,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED,OAAO,KAAK,CAAC;AACf,CAAC"}
@@ -0,0 +1,7 @@
1
+ import { DatabaseSync } from 'node:sqlite';
2
+ export type CitadelDatabase = {
3
+ database: DatabaseSync;
4
+ close(): void;
5
+ };
6
+ export declare function openCitadelDatabase(dbPath: string): CitadelDatabase;
7
+ //# sourceMappingURL=sqlite.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sqlite.d.ts","sourceRoot":"","sources":["../src/sqlite.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAE3C,MAAM,MAAM,eAAe,GAAG;IAC5B,QAAQ,EAAE,YAAY,CAAC;IACvB,KAAK,IAAI,IAAI,CAAC;CACf,CAAC;AAEF,wBAAgB,mBAAmB,CAAC,MAAM,EAAE,MAAM,GAAG,eAAe,CAanE"}
package/dist/sqlite.js ADDED
@@ -0,0 +1,16 @@
1
+ import { mkdirSync } from 'node:fs';
2
+ import { dirname } from 'node:path';
3
+ import { DatabaseSync } from 'node:sqlite';
4
+ export function openCitadelDatabase(dbPath) {
5
+ if (dbPath !== ':memory:') {
6
+ mkdirSync(dirname(dbPath), { recursive: true });
7
+ }
8
+ const database = new DatabaseSync(dbPath);
9
+ return {
10
+ database,
11
+ close() {
12
+ database.close();
13
+ }
14
+ };
15
+ }
16
+ //# sourceMappingURL=sqlite.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sqlite.js","sourceRoot":"","sources":["../src/sqlite.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,SAAS,CAAC;AACpC,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAO3C,MAAM,UAAU,mBAAmB,CAAC,MAAc;IAChD,IAAI,MAAM,KAAK,UAAU,EAAE,CAAC;QAC1B,SAAS,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;IAClD,CAAC;IAED,MAAM,QAAQ,GAAG,IAAI,YAAY,CAAC,MAAM,CAAC,CAAC;IAE1C,OAAO;QACL,QAAQ;QACR,KAAK;YACH,QAAQ,CAAC,KAAK,EAAE,CAAC;QACnB,CAAC;KACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,9 @@
1
+ export type ValidationResult = {
2
+ ok: true;
3
+ value: string;
4
+ } | {
5
+ ok: false;
6
+ error: string;
7
+ };
8
+ export declare function validateDisplayName(input: unknown): ValidationResult;
9
+ //# sourceMappingURL=validation.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validation.d.ts","sourceRoot":"","sources":["../src/validation.ts"],"names":[],"mappings":"AAEA,MAAM,MAAM,gBAAgB,GACxB;IAAE,EAAE,EAAE,IAAI,CAAC;IAAC,KAAK,EAAE,MAAM,CAAA;CAAE,GAC3B;IAAE,EAAE,EAAE,KAAK,CAAC;IAAC,KAAK,EAAE,MAAM,CAAA;CAAE,CAAC;AAEjC,wBAAgB,mBAAmB,CAAC,KAAK,EAAE,OAAO,GAAG,gBAAgB,CAmBpE"}
@@ -0,0 +1,18 @@
1
+ import { DISPLAY_NAME_MAX_LENGTH } from './shared.js';
2
+ export function validateDisplayName(input) {
3
+ if (typeof input !== 'string') {
4
+ return { ok: false, error: 'Enter a display name.' };
5
+ }
6
+ const value = input.trim().replace(/\s+/g, ' ');
7
+ if (!value) {
8
+ return { ok: false, error: 'Enter a display name.' };
9
+ }
10
+ if (value.length > DISPLAY_NAME_MAX_LENGTH) {
11
+ return {
12
+ ok: false,
13
+ error: `Display names must be ${DISPLAY_NAME_MAX_LENGTH} characters or fewer.`
14
+ };
15
+ }
16
+ return { ok: true, value };
17
+ }
18
+ //# sourceMappingURL=validation.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validation.js","sourceRoot":"","sources":["../src/validation.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,MAAM,aAAa,CAAC;AAMtD,MAAM,UAAU,mBAAmB,CAAC,KAAc;IAChD,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QAC9B,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC;IACvD,CAAC;IAED,MAAM,KAAK,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAEhD,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC;IACvD,CAAC;IAED,IAAI,KAAK,CAAC,MAAM,GAAG,uBAAuB,EAAE,CAAC;QAC3C,OAAO;YACL,EAAE,EAAE,KAAK;YACT,KAAK,EAAE,yBAAyB,uBAAuB,uBAAuB;SAC/E,CAAC;IACJ,CAAC;IAED,OAAO,EAAE,EAAE,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC;AAC7B,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare const PLATFORM_VERSION = "0.1.0";
2
+ //# sourceMappingURL=version.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"version.d.ts","sourceRoot":"","sources":["../src/version.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,gBAAgB,UAAU,CAAC"}
@@ -0,0 +1,2 @@
1
+ export const PLATFORM_VERSION = '0.1.0';
2
+ //# sourceMappingURL=version.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"version.js","sourceRoot":"","sources":["../src/version.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,gBAAgB,GAAG,OAAO,CAAC"}
package/package.json ADDED
@@ -0,0 +1,60 @@
1
+ {
2
+ "name": "@citadel-platform/platform",
3
+ "version": "0.1.0",
4
+ "type": "module",
5
+ "files": [
6
+ "dist"
7
+ ],
8
+ "bin": {
9
+ "citadel-generate-app-metadata": "dist/generateAppMetadataCli.js"
10
+ },
11
+ "exports": {
12
+ "./app": {
13
+ "types": "./dist/app.d.ts",
14
+ "import": "./dist/app.js"
15
+ },
16
+ "./client": {
17
+ "types": "./dist/client.d.ts",
18
+ "import": "./dist/client.js"
19
+ },
20
+ "./server-app": {
21
+ "types": "./dist/server-app.d.ts",
22
+ "import": "./dist/server-app.js"
23
+ },
24
+ "./persistence": {
25
+ "types": "./dist/persistence.d.ts",
26
+ "import": "./dist/persistence.js"
27
+ },
28
+ "./server": {
29
+ "types": "./dist/server.d.ts",
30
+ "import": "./dist/server.js"
31
+ },
32
+ "./validation": {
33
+ "types": "./dist/validation.d.ts",
34
+ "import": "./dist/validation.js"
35
+ }
36
+ },
37
+ "scripts": {
38
+ "build": "npm run clean && tsc -p tsconfig.build.json",
39
+ "build:watch": "tsc -p tsconfig.build.json --watch --preserveWatchOutput",
40
+ "clean": "node -e \"fs.rmSync('dist', { recursive: true, force: true })\"",
41
+ "typecheck": "tsc -p tsconfig.json --noEmit",
42
+ "test": "vitest run tests"
43
+ },
44
+ "dependencies": {
45
+ "express": "^5.1.0",
46
+ "nanoid": "^5.1.5",
47
+ "socket.io": "^4.8.1"
48
+ },
49
+ "devDependencies": {
50
+ "vitest": "^4.1.9"
51
+ },
52
+ "peerDependencies": {
53
+ "react": "^19.1.0"
54
+ },
55
+ "peerDependenciesMeta": {
56
+ "react": {
57
+ "optional": true
58
+ }
59
+ }
60
+ }