@arch-cadre/modules 0.0.63 → 0.0.64
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/dist/client/extension-point-client.d.ts +11 -0
- package/dist/client/extension-point-client.d.ts.map +1 -0
- package/dist/client/extension-point-client.js +19 -0
- package/dist/client/extension-point.d.ts +10 -0
- package/dist/client/extension-point.d.ts.map +1 -0
- package/dist/client/extension-point.js +12 -0
- package/dist/client/index.d.ts +4 -0
- package/dist/client/index.d.ts.map +1 -0
- package/dist/client/index.js +3 -0
- package/dist/client/widget-area.d.ts +10 -0
- package/dist/client/widget-area.d.ts.map +1 -0
- package/dist/client/widget-area.js +12 -0
- package/dist/index.d.ts +5 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +6 -0
- package/dist/server/lifecycle.d.ts +6 -0
- package/dist/server/lifecycle.d.ts.map +1 -0
- package/dist/server/lifecycle.js +250 -0
- package/dist/server/manage.d.ts +27 -0
- package/dist/server/manage.d.ts.map +1 -0
- package/dist/server/manage.js +143 -0
- package/dist/server/registry.d.ts +2 -0
- package/dist/server/registry.d.ts.map +1 -0
- package/dist/server/registry.js +118 -0
- package/dist/server/ui.d.ts +13 -0
- package/dist/server/ui.d.ts.map +1 -0
- package/dist/server/ui.js +235 -0
- package/dist/server.d.ts +6 -0
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +8 -0
- package/dist/types.d.ts +114 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +13 -0
- package/package.json +5 -5
- package/dist/_virtual/_rolldown/runtime.cjs +0 -1
- package/dist/client/extension-point-client.cjs +0 -1
- package/dist/client/extension-point-client.d.cts +0 -20
- package/dist/client/extension-point-client.d.cts.map +0 -1
- package/dist/client/extension-point-client.d.mts +0 -20
- package/dist/client/extension-point-client.d.mts.map +0 -1
- package/dist/client/extension-point-client.mjs +0 -2
- package/dist/client/extension-point-client.mjs.map +0 -1
- package/dist/client/extension-point.cjs +0 -1
- package/dist/client/extension-point.d.cts +0 -21
- package/dist/client/extension-point.d.cts.map +0 -1
- package/dist/client/extension-point.d.mts +0 -21
- package/dist/client/extension-point.d.mts.map +0 -1
- package/dist/client/extension-point.mjs +0 -2
- package/dist/client/extension-point.mjs.map +0 -1
- package/dist/client/widget-area.cjs +0 -1
- package/dist/client/widget-area.d.cts +0 -19
- package/dist/client/widget-area.d.cts.map +0 -1
- package/dist/client/widget-area.d.mts +0 -19
- package/dist/client/widget-area.d.mts.map +0 -1
- package/dist/client/widget-area.mjs +0 -2
- package/dist/client/widget-area.mjs.map +0 -1
- package/dist/index.cjs +0 -1
- package/dist/index.d.cts +0 -5
- package/dist/index.d.mts +0 -5
- package/dist/index.mjs +0 -1
- package/dist/server/lifecycle.cjs +0 -1
- package/dist/server/lifecycle.d.cts +0 -9
- package/dist/server/lifecycle.d.cts.map +0 -1
- package/dist/server/lifecycle.d.mts +0 -9
- package/dist/server/lifecycle.d.mts.map +0 -1
- package/dist/server/lifecycle.mjs +0 -2
- package/dist/server/lifecycle.mjs.map +0 -1
- package/dist/server/manage.cjs +0 -1
- package/dist/server/manage.d.cts +0 -31
- package/dist/server/manage.d.cts.map +0 -1
- package/dist/server/manage.d.mts +0 -31
- package/dist/server/manage.d.mts.map +0 -1
- package/dist/server/manage.mjs +0 -2
- package/dist/server/manage.mjs.map +0 -1
- package/dist/server/registry.cjs +0 -1
- package/dist/server/registry.d.cts +0 -5
- package/dist/server/registry.d.cts.map +0 -1
- package/dist/server/registry.d.mts +0 -5
- package/dist/server/registry.d.mts.map +0 -1
- package/dist/server/registry.mjs +0 -2
- package/dist/server/registry.mjs.map +0 -1
- package/dist/server/ui.cjs +0 -1
- package/dist/server/ui.d.cts +0 -17
- package/dist/server/ui.d.cts.map +0 -1
- package/dist/server/ui.d.mts +0 -17
- package/dist/server/ui.d.mts.map +0 -1
- package/dist/server/ui.mjs +0 -2
- package/dist/server/ui.mjs.map +0 -1
- package/dist/server.cjs +0 -1
- package/dist/server.d.cts +0 -6
- package/dist/server.d.mts +0 -6
- package/dist/server.mjs +0 -1
- package/dist/types.cjs +0 -1
- package/dist/types.d.cts +0 -117
- package/dist/types.d.cts.map +0 -1
- package/dist/types.d.mts +0 -117
- package/dist/types.d.mts.map +0 -1
- package/dist/types.mjs +0 -2
- package/dist/types.mjs.map +0 -1
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
interface ExtensionPointProps {
|
|
3
|
+
module: string;
|
|
4
|
+
point: string;
|
|
5
|
+
className?: string;
|
|
6
|
+
props?: any;
|
|
7
|
+
fallback?: React.ReactNode;
|
|
8
|
+
}
|
|
9
|
+
export declare function ExtensionPointClient({ module, point, className, props, fallback, }: ExtensionPointProps): any;
|
|
10
|
+
export {};
|
|
11
|
+
//# sourceMappingURL=extension-point-client.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"extension-point-client.d.ts","sourceRoot":"","sources":["../../src/client/extension-point-client.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAK/B,UAAU,mBAAmB;IAC3B,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,EAAE,MAAM,CAAC;IACd,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,KAAK,CAAC,EAAE,GAAG,CAAC;IACZ,QAAQ,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;CAC5B;AAED,wBAAgB,oBAAoB,CAAC,EACnC,MAAM,EACN,KAAK,EACL,SAAS,EACT,KAAK,EACL,QAAQ,GACT,EAAE,mBAAmB,OAkBrB"}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
/** biome-ignore-all lint/suspicious/noExplicitAny: <> */
|
|
2
|
+
"use client";
|
|
3
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
4
|
+
import { useEffect, useState } from "react";
|
|
5
|
+
import { getExtensions } from "../server/ui";
|
|
6
|
+
export function ExtensionPointClient({ module, point, className, props, fallback, }) {
|
|
7
|
+
const [extensions, setExtensions] = useState([]);
|
|
8
|
+
useEffect(() => {
|
|
9
|
+
getExtensions(module, point).then(setExtensions);
|
|
10
|
+
}, [module, point]);
|
|
11
|
+
if (extensions.length === 0)
|
|
12
|
+
return fallback || null;
|
|
13
|
+
return (_jsx("div", { className: className, children: extensions.map((ext) => {
|
|
14
|
+
const Component = ext.component;
|
|
15
|
+
if (!Component)
|
|
16
|
+
return null;
|
|
17
|
+
return _jsx(Component, { ...props }, ext.id);
|
|
18
|
+
}) }));
|
|
19
|
+
}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
interface ExtensionPointProps {
|
|
2
|
+
module: string;
|
|
3
|
+
point: string;
|
|
4
|
+
className?: string;
|
|
5
|
+
props?: any;
|
|
6
|
+
fallback?: React.ReactNode;
|
|
7
|
+
}
|
|
8
|
+
export declare function ExtensionPoint({ module, point, className, props, fallback, }: ExtensionPointProps): Promise<string | number | bigint | boolean | Iterable<import("react").ReactNode> | import("react/jsx-runtime").JSX.Element | null | undefined>;
|
|
9
|
+
export {};
|
|
10
|
+
//# sourceMappingURL=extension-point.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"extension-point.d.ts","sourceRoot":"","sources":["../../src/client/extension-point.tsx"],"names":[],"mappings":"AAGA,UAAU,mBAAmB;IAC3B,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,EAAE,MAAM,CAAC;IACd,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,KAAK,CAAC,EAAE,GAAG,CAAC;IACZ,QAAQ,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;CAC5B;AAED,wBAAsB,cAAc,CAAC,EACnC,MAAM,EACN,KAAK,EACL,SAAS,EACT,KAAK,EACL,QAAQ,GACT,EAAE,mBAAmB,kJAarB"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
// import * as React from "react";
|
|
3
|
+
import { getExtensions } from "../server/ui";
|
|
4
|
+
export async function ExtensionPoint({ module, point, className, props, fallback, }) {
|
|
5
|
+
const extensions = await getExtensions(module, point);
|
|
6
|
+
if (extensions.length === 0)
|
|
7
|
+
return fallback || null;
|
|
8
|
+
return (_jsx("div", { className: className, children: extensions.map((ext) => {
|
|
9
|
+
const Component = ext.component;
|
|
10
|
+
return _jsx(Component, { ...props }, ext.id);
|
|
11
|
+
}) }));
|
|
12
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/client/index.ts"],"names":[],"mappings":"AAAA,cAAc,mBAAmB,CAAC;AAClC,cAAc,0BAA0B,CAAC;AACzC,cAAc,eAAe,CAAC"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import * as React from "react";
|
|
2
|
+
interface WidgetAreaProps {
|
|
3
|
+
area: string;
|
|
4
|
+
className?: string;
|
|
5
|
+
fallback?: React.ReactNode;
|
|
6
|
+
props?: any;
|
|
7
|
+
}
|
|
8
|
+
export declare function WidgetArea({ area, className, fallback, props, }: WidgetAreaProps): Promise<string | number | bigint | boolean | Iterable<React.ReactNode> | import("react/jsx-runtime").JSX.Element | null | undefined>;
|
|
9
|
+
export {};
|
|
10
|
+
//# sourceMappingURL=widget-area.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"widget-area.d.ts","sourceRoot":"","sources":["../../src/client/widget-area.tsx"],"names":[],"mappings":"AACA,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAG/B,UAAU,eAAe;IACvB,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;IAE3B,KAAK,CAAC,EAAE,GAAG,CAAC;CACb;AAED,wBAAsB,UAAU,CAAC,EAC/B,IAAI,EACJ,SAAS,EACT,QAAQ,EACR,KAAK,GACN,EAAE,eAAe,wIAejB"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
2
|
+
import { getModuleWidgets } from "../server/ui";
|
|
3
|
+
export async function WidgetArea({ area, className, fallback, props, }) {
|
|
4
|
+
const widgets = await getModuleWidgets(area);
|
|
5
|
+
if (widgets.length === 0) {
|
|
6
|
+
return fallback || null;
|
|
7
|
+
}
|
|
8
|
+
return (_jsx("div", { className: className, children: widgets.map((widget) => {
|
|
9
|
+
const Component = widget.component;
|
|
10
|
+
return _jsx(Component, { ...props }, widget.id);
|
|
11
|
+
}) }));
|
|
12
|
+
}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,0BAA0B,CAAC;AAC1D,OAAO,EAAE,oBAAoB,EAAE,MAAM,iCAAiC,CAAC;AACvE,OAAO,EAAE,UAAU,EAAE,MAAM,sBAAsB,CAAC;AAIlD,cAAc,SAAS,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
export { ExtensionPoint } from "./client/extension-point";
|
|
2
|
+
export { ExtensionPointClient } from "./client/extension-point-client";
|
|
3
|
+
export { WidgetArea } from "./client/widget-area";
|
|
4
|
+
// Shared Constants & Models (Safe for Browser)
|
|
5
|
+
// export * from "./server/ui";
|
|
6
|
+
export * from "./types";
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
export declare function pushModuleSchema(moduleId: string): Promise<void>;
|
|
2
|
+
export declare function toggleModuleState(moduleId: string, isEnabled: boolean): Promise<{
|
|
3
|
+
success: boolean;
|
|
4
|
+
}>;
|
|
5
|
+
export declare function initOperationalModules(): Promise<void>;
|
|
6
|
+
//# sourceMappingURL=lifecycle.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"lifecycle.d.ts","sourceRoot":"","sources":["../../src/server/lifecycle.ts"],"names":[],"mappings":"AA4GA,wBAAsB,gBAAgB,CAAC,QAAQ,EAAE,MAAM,iBA6DtD;AA2HD,wBAAsB,iBAAiB,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,EAAE,OAAO;;GAS3E;AAED,wBAAsB,sBAAsB,kBAc3C"}
|
|
@@ -0,0 +1,250 @@
|
|
|
1
|
+
"use server";
|
|
2
|
+
import { exec } from "node:child_process";
|
|
3
|
+
import fs from "node:fs/promises";
|
|
4
|
+
import path from "node:path";
|
|
5
|
+
import { promisify } from "node:util";
|
|
6
|
+
import { systemModulesTable } from "@arch-cadre/core";
|
|
7
|
+
import { db } from "@arch-cadre/core/server";
|
|
8
|
+
import { eq } from "drizzle-orm";
|
|
9
|
+
import { getModuleInstance, getModules } from "./manage";
|
|
10
|
+
const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
|
|
11
|
+
async function updateStep(moduleId, step) {
|
|
12
|
+
console.log(`[Kernel] "${moduleId}" step: ${step}`);
|
|
13
|
+
await db
|
|
14
|
+
.update(systemModulesTable)
|
|
15
|
+
.set({ lastStep: step })
|
|
16
|
+
.where(eq(systemModulesTable.id, moduleId));
|
|
17
|
+
}
|
|
18
|
+
async function resolveSchemaPath(moduleId) {
|
|
19
|
+
// const path = await import("node:path");
|
|
20
|
+
// const fs = await import("node:fs/promises");
|
|
21
|
+
const root = process.cwd();
|
|
22
|
+
// 1. Special case for core
|
|
23
|
+
if (moduleId === "@arch-cadre/core" || moduleId === "core") {
|
|
24
|
+
const p = path.join(root, "node_modules", "@kryo", "core", "dist", "server", "database", "schema.cjs");
|
|
25
|
+
try {
|
|
26
|
+
await fs.access(p);
|
|
27
|
+
return p;
|
|
28
|
+
}
|
|
29
|
+
catch {
|
|
30
|
+
// Monorepo dev fallback
|
|
31
|
+
const devP = path.resolve(root, "../../packages/kryo-core/src/server/database/schema.ts");
|
|
32
|
+
try {
|
|
33
|
+
await fs.access(devP);
|
|
34
|
+
return devP;
|
|
35
|
+
}
|
|
36
|
+
catch { }
|
|
37
|
+
}
|
|
38
|
+
return null;
|
|
39
|
+
}
|
|
40
|
+
// Normalize moduleId (remove @arch-cadre/ prefix if present for directory searching)
|
|
41
|
+
const cleanId = moduleId.replace("@arch-cadre/", "");
|
|
42
|
+
const possible = [
|
|
43
|
+
// Local module
|
|
44
|
+
path.join(root, "modules", cleanId, "schema.ts"),
|
|
45
|
+
path.join(root, "modules", moduleId, "schema.ts"),
|
|
46
|
+
// node_modules (installed or symlinked)
|
|
47
|
+
path.join(root, "node_modules", moduleId, "dist", "schema.cjs"),
|
|
48
|
+
path.join(root, "node_modules", `@arch-cadre/${cleanId}`, "dist", "schema.cjs"),
|
|
49
|
+
path.join(root, "node_modules", moduleId, "src", "schema.ts"),
|
|
50
|
+
path.join(root, "node_modules", `@arch-cadre/${cleanId}`, "src", "schema.ts"),
|
|
51
|
+
// Monorepo packages (searching by various naming conventions)
|
|
52
|
+
path.resolve(root, "../../packages", cleanId, "src", "schema.ts"),
|
|
53
|
+
path.resolve(root, "../../packages", `kryo-${cleanId}`, "src", "schema.ts"),
|
|
54
|
+
path.resolve(root, "../../packages", cleanId, "schema.ts"),
|
|
55
|
+
];
|
|
56
|
+
for (const p of possible) {
|
|
57
|
+
try {
|
|
58
|
+
await fs.access(p);
|
|
59
|
+
console.log(`[Kernel:Lifecycle] Resolved schema for ${moduleId} at: ${p}`);
|
|
60
|
+
return p;
|
|
61
|
+
}
|
|
62
|
+
catch {
|
|
63
|
+
console.warn(`[Kernel:Lifecycle] Unresolved schema for ${moduleId} at: ${p}`);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
console.warn(`[Kernel:Lifecycle] Could not resolve schema path for: ${moduleId}`);
|
|
67
|
+
return null;
|
|
68
|
+
}
|
|
69
|
+
export async function pushModuleSchema(moduleId) {
|
|
70
|
+
const execAsync = promisify(exec);
|
|
71
|
+
console.log(`[Kernel:Lifecycle] Targeted sync for module: ${moduleId}`);
|
|
72
|
+
const root = process.cwd();
|
|
73
|
+
// 1. Get all currently enabled modules to avoid dropping their tables
|
|
74
|
+
const enabledFromDb = await db
|
|
75
|
+
.select({ id: systemModulesTable.id })
|
|
76
|
+
.from(systemModulesTable)
|
|
77
|
+
.where(eq(systemModulesTable.enabled, true));
|
|
78
|
+
const activeModuleIds = new Set(enabledFromDb.map((m) => m.id));
|
|
79
|
+
activeModuleIds.add("@arch-cadre/core");
|
|
80
|
+
activeModuleIds.add(moduleId);
|
|
81
|
+
// 2. Resolve physical paths for all active schemas
|
|
82
|
+
const schemaPaths = [];
|
|
83
|
+
for (const id of Array.from(activeModuleIds)) {
|
|
84
|
+
const p = await resolveSchemaPath(id);
|
|
85
|
+
if (p)
|
|
86
|
+
schemaPaths.push(p);
|
|
87
|
+
}
|
|
88
|
+
if (schemaPaths.length === 0) {
|
|
89
|
+
console.warn(`[Kernel:Lifecycle] No schema paths resolved for ${moduleId}, skipping push.`);
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
// 3. Prepare command
|
|
93
|
+
const configPath = path.join(root, "drizzle.config.ts");
|
|
94
|
+
const drizzleKitBin = path.join(root, "node_modules", ".bin", "drizzle-kit");
|
|
95
|
+
const cmd = `"${drizzleKitBin}" push --force --config=${configPath}`;
|
|
96
|
+
console.log(`[Kernel:Lifecycle] Executing isolated migration. Modules: ${Array.from(activeModuleIds).join(", ")}`);
|
|
97
|
+
try {
|
|
98
|
+
await execAsync(cmd, {
|
|
99
|
+
env: {
|
|
100
|
+
...process.env,
|
|
101
|
+
CI: "true",
|
|
102
|
+
// Pass the calculated schemas to our dynamic drizzle.config.ts
|
|
103
|
+
DRIZZLE_SCHEMA_OVERRIDE: schemaPaths.join(","),
|
|
104
|
+
},
|
|
105
|
+
cwd: root,
|
|
106
|
+
});
|
|
107
|
+
console.log(`[Kernel:Lifecycle] Isolated migration successful for ${moduleId}`);
|
|
108
|
+
}
|
|
109
|
+
catch (e) {
|
|
110
|
+
console.error(`[Kernel:Lifecycle] Isolated migration failed for ${moduleId}:`);
|
|
111
|
+
console.error(e.stdout || e.message);
|
|
112
|
+
throw new Error(`Migration failed: ${moduleId}`);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
async function performToggle(moduleId, isEnabled) {
|
|
116
|
+
var _a;
|
|
117
|
+
try {
|
|
118
|
+
const allModules = await getModules();
|
|
119
|
+
const manifest = allModules.find((m) => m.id === moduleId);
|
|
120
|
+
await updateStep(moduleId, "Validate module...");
|
|
121
|
+
await sleep(500);
|
|
122
|
+
if (!manifest)
|
|
123
|
+
return;
|
|
124
|
+
if (isEnabled) {
|
|
125
|
+
await updateStep(moduleId, "Queued...");
|
|
126
|
+
await sleep(500);
|
|
127
|
+
/*
|
|
128
|
+
* 1. Check dependencies
|
|
129
|
+
* 2. Enable module in DB (but not marked as installed yet)
|
|
130
|
+
* 3. Run onMigrate (or default push) to sync DB
|
|
131
|
+
* 4. Run onEnable
|
|
132
|
+
* 5. Mark as installed
|
|
133
|
+
*/
|
|
134
|
+
if ((_a = manifest.dependencies) === null || _a === void 0 ? void 0 : _a.length) {
|
|
135
|
+
const pendingDeps = manifest.dependencies.filter((depId) => {
|
|
136
|
+
const dep = allModules.find((m) => m.id === depId);
|
|
137
|
+
return dep && !dep.enabled && !dep.system;
|
|
138
|
+
});
|
|
139
|
+
if (pendingDeps.length > 0) {
|
|
140
|
+
// await updateStep(moduleId, "Waiting for dependencies...");
|
|
141
|
+
for (const depId of pendingDeps) {
|
|
142
|
+
await updateStep(moduleId, `Waiting for dependency "${depId}"...`);
|
|
143
|
+
await performToggle(depId, true);
|
|
144
|
+
await sleep(500);
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
await updateStep(moduleId, "Initializing...");
|
|
149
|
+
await sleep(500);
|
|
150
|
+
await db
|
|
151
|
+
.update(systemModulesTable)
|
|
152
|
+
.set({ enabled: true })
|
|
153
|
+
.where(eq(systemModulesTable.id, moduleId));
|
|
154
|
+
const instance = await getModuleInstance(moduleId);
|
|
155
|
+
await updateStep(moduleId, "Migrate database...");
|
|
156
|
+
await sleep(500);
|
|
157
|
+
try {
|
|
158
|
+
if (instance === null || instance === void 0 ? void 0 : instance.onMigrate) {
|
|
159
|
+
await instance.onMigrate();
|
|
160
|
+
}
|
|
161
|
+
else {
|
|
162
|
+
// Fallback to default push if no hook provided but schema might exist
|
|
163
|
+
await pushModuleSchema(moduleId);
|
|
164
|
+
}
|
|
165
|
+
await updateStep(moduleId, "Migration successful");
|
|
166
|
+
}
|
|
167
|
+
catch (e) {
|
|
168
|
+
console.error(`[Kernel] Migration failed for ${moduleId}:`, e);
|
|
169
|
+
await updateStep(moduleId, `Migration failed: ${e.message}`);
|
|
170
|
+
await sleep(500);
|
|
171
|
+
}
|
|
172
|
+
if (instance === null || instance === void 0 ? void 0 : instance.onEnable) {
|
|
173
|
+
await updateStep(moduleId, "Running setup...");
|
|
174
|
+
await sleep(500);
|
|
175
|
+
await instance.onEnable();
|
|
176
|
+
}
|
|
177
|
+
await updateStep(moduleId, "Finishing...");
|
|
178
|
+
await sleep(500);
|
|
179
|
+
await db
|
|
180
|
+
.update(systemModulesTable)
|
|
181
|
+
.set({ installed: true, lastStep: null })
|
|
182
|
+
.where(eq(systemModulesTable.id, moduleId));
|
|
183
|
+
}
|
|
184
|
+
else {
|
|
185
|
+
const dependents = allModules.filter((m) => {
|
|
186
|
+
var _a, _b;
|
|
187
|
+
if (!m.enabled || m.id === moduleId)
|
|
188
|
+
return false;
|
|
189
|
+
if ((_a = m.dependencies) === null || _a === void 0 ? void 0 : _a.includes(moduleId))
|
|
190
|
+
return true;
|
|
191
|
+
if ((_b = m.extends) === null || _b === void 0 ? void 0 : _b.includes(moduleId)) {
|
|
192
|
+
const otherActiveTargets = m.extends.filter((targetId) => {
|
|
193
|
+
if (targetId === moduleId)
|
|
194
|
+
return false;
|
|
195
|
+
const target = allModules.find((mod) => mod.id === targetId);
|
|
196
|
+
return target === null || target === void 0 ? void 0 : target.enabled;
|
|
197
|
+
});
|
|
198
|
+
return otherActiveTargets.length === 0;
|
|
199
|
+
}
|
|
200
|
+
return false;
|
|
201
|
+
});
|
|
202
|
+
if (dependents.length > 0) {
|
|
203
|
+
await updateStep(moduleId, `Deactivating dependents...`);
|
|
204
|
+
for (const dep of dependents) {
|
|
205
|
+
await performToggle(dep.id, false);
|
|
206
|
+
await sleep(500);
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
await updateStep(moduleId, "Deactivating...");
|
|
210
|
+
const instance = await getModuleInstance(moduleId);
|
|
211
|
+
if (instance === null || instance === void 0 ? void 0 : instance.onDisable)
|
|
212
|
+
await instance.onDisable();
|
|
213
|
+
await db
|
|
214
|
+
.update(systemModulesTable)
|
|
215
|
+
.set({ enabled: false, installed: false })
|
|
216
|
+
.where(eq(systemModulesTable.id, moduleId));
|
|
217
|
+
await updateStep(moduleId, null);
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
catch (e) {
|
|
221
|
+
console.error(`[Kernel] Fatal error for ${moduleId}:`, e);
|
|
222
|
+
await updateStep(moduleId, `Error: ${e.message}`);
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
export async function toggleModuleState(moduleId, isEnabled) {
|
|
226
|
+
const allModules = await getModules();
|
|
227
|
+
const manifest = allModules.find((m) => m.id === moduleId);
|
|
228
|
+
if (!manifest)
|
|
229
|
+
throw new Error(`Module "${moduleId}" not found`);
|
|
230
|
+
if (manifest.enabled === isEnabled)
|
|
231
|
+
return { success: true };
|
|
232
|
+
void performToggle(moduleId, isEnabled);
|
|
233
|
+
return { success: true };
|
|
234
|
+
}
|
|
235
|
+
export async function initOperationalModules() {
|
|
236
|
+
const allModules = await getModules();
|
|
237
|
+
for (const mod of allModules) {
|
|
238
|
+
if (mod.enabled) {
|
|
239
|
+
try {
|
|
240
|
+
const instance = await getModuleInstance(mod.id);
|
|
241
|
+
if (instance === null || instance === void 0 ? void 0 : instance.init) {
|
|
242
|
+
await instance.init();
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
catch (e) {
|
|
246
|
+
console.error(`[Kernel] Failed to init module ${mod.id}:`, e);
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { type IModule } from "../types";
|
|
2
|
+
export declare function registerModules(modules: IModule[]): Promise<void>;
|
|
3
|
+
export declare function getModules(): Promise<{
|
|
4
|
+
enabled: boolean;
|
|
5
|
+
installed: boolean;
|
|
6
|
+
lastStep: string | null | undefined;
|
|
7
|
+
hasDocs: boolean;
|
|
8
|
+
id: string;
|
|
9
|
+
name: string;
|
|
10
|
+
version: string;
|
|
11
|
+
dependencies: string[];
|
|
12
|
+
extends: string[];
|
|
13
|
+
system: boolean;
|
|
14
|
+
description?: string | undefined;
|
|
15
|
+
npmDependencies?: string[] | undefined;
|
|
16
|
+
npmDevDependencies?: string[] | undefined;
|
|
17
|
+
}[]>;
|
|
18
|
+
export declare function getModuleStatus(moduleId: string): Promise<{
|
|
19
|
+
enabled: boolean;
|
|
20
|
+
installed: boolean;
|
|
21
|
+
lastStep: string | null;
|
|
22
|
+
}>;
|
|
23
|
+
export declare function isModuleEnabled(moduleId: string): Promise<boolean>;
|
|
24
|
+
export declare function getModuleInstance(moduleId: string): Promise<IModule | null>;
|
|
25
|
+
export declare function getModuleConfig<T>(moduleId: string): Promise<T | null>;
|
|
26
|
+
export declare function updateModuleConfig(moduleId: string, config: any): Promise<void>;
|
|
27
|
+
//# sourceMappingURL=manage.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"manage.d.ts","sourceRoot":"","sources":["../../src/server/manage.ts"],"names":[],"mappings":"AAOA,OAAO,EAAE,KAAK,OAAO,EAAwB,MAAM,UAAU,CAAC;AAQ9D,wBAAsB,eAAe,CAAC,OAAO,EAAE,OAAO,EAAE,iBAOvD;AAED,wBAAsB,UAAU;;;;;;;;;;;;;;KAiD/B;AAED,wBAAsB,eAAe,CAAC,QAAQ,EAAE,MAAM;;;;GAerD;AAED,wBAAsB,eAAe,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC,CAUxE;AAED,wBAAsB,iBAAiB,CACrC,QAAQ,EAAE,MAAM,GACf,OAAO,CAAC,OAAO,GAAG,IAAI,CAAC,CA2CzB;AAED,wBAAsB,eAAe,CAAC,CAAC,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,CAAC,GAAG,IAAI,CAAC,CAW5E;AAED,wBAAsB,kBAAkB,CAAC,QAAQ,EAAE,MAAM,EAAE,MAAM,EAAE,GAAG,iBASrE"}
|
|
@@ -0,0 +1,143 @@
|
|
|
1
|
+
"use server";
|
|
2
|
+
import fs from "node:fs/promises";
|
|
3
|
+
import path from "node:path";
|
|
4
|
+
import { systemModulesTable } from "@arch-cadre/core";
|
|
5
|
+
import { db, getModulesDir } from "@arch-cadre/core/server";
|
|
6
|
+
import { eq } from "drizzle-orm";
|
|
7
|
+
import { ModuleManifestSchema } from "../types";
|
|
8
|
+
const globalForModules = globalThis;
|
|
9
|
+
if (!globalForModules.__KRYO_REGISTERED_MODULES__) {
|
|
10
|
+
globalForModules.__KRYO_REGISTERED_MODULES__ = new Map();
|
|
11
|
+
}
|
|
12
|
+
export async function registerModules(modules) {
|
|
13
|
+
var _a;
|
|
14
|
+
const store = globalForModules.__KRYO_REGISTERED_MODULES__;
|
|
15
|
+
for (const mod of modules) {
|
|
16
|
+
if ((_a = mod.manifest) === null || _a === void 0 ? void 0 : _a.id) {
|
|
17
|
+
store.set(mod.manifest.id, mod);
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
export async function getModules() {
|
|
22
|
+
try {
|
|
23
|
+
const modulesDir = await getModulesDir();
|
|
24
|
+
const manifests = [];
|
|
25
|
+
const entries = await fs.readdir(modulesDir).catch(() => []);
|
|
26
|
+
for (const dir of entries) {
|
|
27
|
+
if (dir.startsWith("."))
|
|
28
|
+
continue;
|
|
29
|
+
try {
|
|
30
|
+
const content = await fs.readFile(path.join(modulesDir, dir, "manifeston"), "utf-8");
|
|
31
|
+
const manifest = ModuleManifestSchema.parse(JSON.parse(content));
|
|
32
|
+
manifests.push({ ...manifest, hasDocs: false });
|
|
33
|
+
}
|
|
34
|
+
catch { }
|
|
35
|
+
}
|
|
36
|
+
const store = globalForModules.__KRYO_REGISTERED_MODULES__;
|
|
37
|
+
for (const mod of store.values()) {
|
|
38
|
+
if (!manifests.some((m) => m.id === mod.manifest.id)) {
|
|
39
|
+
manifests.push({ ...mod.manifest, hasDocs: false });
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
const dbModules = await db
|
|
43
|
+
.select()
|
|
44
|
+
.from(systemModulesTable)
|
|
45
|
+
.catch(() => []);
|
|
46
|
+
const dbMap = new Map(dbModules.map((m) => [m.id, m]));
|
|
47
|
+
return manifests.map((mod) => {
|
|
48
|
+
var _a, _b;
|
|
49
|
+
const dbEntry = dbMap.get(mod.id);
|
|
50
|
+
return {
|
|
51
|
+
...mod,
|
|
52
|
+
enabled: mod.system ? true : ((_a = dbEntry === null || dbEntry === void 0 ? void 0 : dbEntry.enabled) !== null && _a !== void 0 ? _a : false),
|
|
53
|
+
installed: mod.system ? true : ((_b = dbEntry === null || dbEntry === void 0 ? void 0 : dbEntry.installed) !== null && _b !== void 0 ? _b : false),
|
|
54
|
+
lastStep: dbEntry === null || dbEntry === void 0 ? void 0 : dbEntry.lastStep,
|
|
55
|
+
hasDocs: mod.hasDocs,
|
|
56
|
+
};
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
catch {
|
|
60
|
+
return [];
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
export async function getModuleStatus(moduleId) {
|
|
64
|
+
try {
|
|
65
|
+
const [mod] = await db
|
|
66
|
+
.select()
|
|
67
|
+
.from(systemModulesTable)
|
|
68
|
+
.where(eq(systemModulesTable.id, moduleId));
|
|
69
|
+
return {
|
|
70
|
+
enabled: mod === null || mod === void 0 ? void 0 : mod.enabled,
|
|
71
|
+
installed: mod === null || mod === void 0 ? void 0 : mod.installed,
|
|
72
|
+
lastStep: mod === null || mod === void 0 ? void 0 : mod.lastStep,
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
catch {
|
|
76
|
+
return { enabled: false, installed: false, lastStep: null };
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
export async function isModuleEnabled(moduleId) {
|
|
80
|
+
try {
|
|
81
|
+
const [mod] = await db
|
|
82
|
+
.select()
|
|
83
|
+
.from(systemModulesTable)
|
|
84
|
+
.where(eq(systemModulesTable.id, moduleId));
|
|
85
|
+
return !!(mod === null || mod === void 0 ? void 0 : mod.enabled) || !!(mod === null || mod === void 0 ? void 0 : mod.system);
|
|
86
|
+
}
|
|
87
|
+
catch {
|
|
88
|
+
return false;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
export async function getModuleInstance(moduleId) {
|
|
92
|
+
try {
|
|
93
|
+
const store = globalForModules.__KRYO_REGISTERED_MODULES__;
|
|
94
|
+
// 1. Check if module is already registered in the global store
|
|
95
|
+
if (store.has(moduleId)) {
|
|
96
|
+
console.log(`[Kernel:Manage] Module "${moduleId}" found in registry.`);
|
|
97
|
+
return store.get(moduleId) || null;
|
|
98
|
+
}
|
|
99
|
+
console.log(`[Kernel:Manage] Module "${moduleId}" not in registry. Registry size: ${store.size}. Keys: ${Array.from(store.keys()).join(", ")}`);
|
|
100
|
+
// 2. Try to load as an NPM package (for modules installed via npm/pnpm)
|
|
101
|
+
// We only try this if it doesn't look like a local module path
|
|
102
|
+
if (!moduleId.startsWith(".") && !moduleId.startsWith("/")) {
|
|
103
|
+
try {
|
|
104
|
+
const mod = await import(`@arch-cadre/${moduleId}`);
|
|
105
|
+
console.log(`[Kernel:Manage] Module "${moduleId}" loaded via import().`);
|
|
106
|
+
return mod.default || mod || null;
|
|
107
|
+
}
|
|
108
|
+
catch (e) {
|
|
109
|
+
console.warn(`[Kernel:Manage] Failed to import module "${moduleId} via import().": ${e.message}`);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
return null;
|
|
113
|
+
}
|
|
114
|
+
catch (error) {
|
|
115
|
+
console.error(`[Kernel:Manage] Error in getModuleInstance for "${moduleId}":`, error);
|
|
116
|
+
return null;
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
export async function getModuleConfig(moduleId) {
|
|
120
|
+
try {
|
|
121
|
+
const [mod] = await db
|
|
122
|
+
.select({ config: systemModulesTable.config })
|
|
123
|
+
.from(systemModulesTable)
|
|
124
|
+
.where(eq(systemModulesTable.id, moduleId));
|
|
125
|
+
if (!(mod === null || mod === void 0 ? void 0 : mod.config))
|
|
126
|
+
return null;
|
|
127
|
+
return JSON.parse(mod.config);
|
|
128
|
+
}
|
|
129
|
+
catch {
|
|
130
|
+
return null;
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
export async function updateModuleConfig(moduleId, config) {
|
|
134
|
+
try {
|
|
135
|
+
await db
|
|
136
|
+
.update(systemModulesTable)
|
|
137
|
+
.set({ config: JSON.stringify(config) })
|
|
138
|
+
.where(eq(systemModulesTable.id, moduleId));
|
|
139
|
+
}
|
|
140
|
+
catch (e) {
|
|
141
|
+
console.warn(`[Kernel:Manage] Failed to update config for ${moduleId}:`, e);
|
|
142
|
+
}
|
|
143
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"registry.d.ts","sourceRoot":"","sources":["../../src/server/registry.ts"],"names":[],"mappings":"AAWA,wBAAsB,WAAW,CAAC,KAAK,UAAQ,iBAwI9C"}
|