@shadanai/openme 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 (147) hide show
  1. package/bin/openme.js +2 -0
  2. package/dist/agents/agent.d.ts +10 -0
  3. package/dist/agents/agent.js +2 -0
  4. package/dist/agents/agent.js.map +1 -0
  5. package/dist/agents/happy.d.ts +15 -0
  6. package/dist/agents/happy.js +43 -0
  7. package/dist/agents/happy.js.map +1 -0
  8. package/dist/agents/openclaw.d.ts +16 -0
  9. package/dist/agents/openclaw.js +55 -0
  10. package/dist/agents/openclaw.js.map +1 -0
  11. package/dist/agents/proxy.d.ts +15 -0
  12. package/dist/agents/proxy.js +36 -0
  13. package/dist/agents/proxy.js.map +1 -0
  14. package/dist/cli/cmd-agent.d.ts +1 -0
  15. package/dist/cli/cmd-agent.js +73 -0
  16. package/dist/cli/cmd-agent.js.map +1 -0
  17. package/dist/cli/cmd-datasource.d.ts +1 -0
  18. package/dist/cli/cmd-datasource.js +78 -0
  19. package/dist/cli/cmd-datasource.js.map +1 -0
  20. package/dist/cli/cmd-init.d.ts +7 -0
  21. package/dist/cli/cmd-init.js +26 -0
  22. package/dist/cli/cmd-init.js.map +1 -0
  23. package/dist/cli/cmd-install.d.ts +5 -0
  24. package/dist/cli/cmd-install.js +22 -0
  25. package/dist/cli/cmd-install.js.map +1 -0
  26. package/dist/cli/cmd-start.d.ts +5 -0
  27. package/dist/cli/cmd-start.js +136 -0
  28. package/dist/cli/cmd-start.js.map +1 -0
  29. package/dist/cli/cmd-status.d.ts +1 -0
  30. package/dist/cli/cmd-status.js +26 -0
  31. package/dist/cli/cmd-status.js.map +1 -0
  32. package/dist/cli/cmd-stop.d.ts +1 -0
  33. package/dist/cli/cmd-stop.js +17 -0
  34. package/dist/cli/cmd-stop.js.map +1 -0
  35. package/dist/cli/cmd-workspace.d.ts +1 -0
  36. package/dist/cli/cmd-workspace.js +32 -0
  37. package/dist/cli/cmd-workspace.js.map +1 -0
  38. package/dist/cli/index.d.ts +1 -0
  39. package/dist/cli/index.js +46 -0
  40. package/dist/cli/index.js.map +1 -0
  41. package/dist/core/config-sync.d.ts +6 -0
  42. package/dist/core/config-sync.js +41 -0
  43. package/dist/core/config-sync.js.map +1 -0
  44. package/dist/core/config.d.ts +96 -0
  45. package/dist/core/config.js +91 -0
  46. package/dist/core/config.js.map +1 -0
  47. package/dist/core/node-id.d.ts +1 -0
  48. package/dist/core/node-id.js +5 -0
  49. package/dist/core/node-id.js.map +1 -0
  50. package/dist/core/register.d.ts +19 -0
  51. package/dist/core/register.js +31 -0
  52. package/dist/core/register.js.map +1 -0
  53. package/dist/data/connectors/baidupan.d.ts +20 -0
  54. package/dist/data/connectors/baidupan.js +69 -0
  55. package/dist/data/connectors/baidupan.js.map +1 -0
  56. package/dist/data/connectors/connector.d.ts +12 -0
  57. package/dist/data/connectors/connector.js +2 -0
  58. package/dist/data/connectors/connector.js.map +1 -0
  59. package/dist/data/connectors/dingtalk.d.ts +18 -0
  60. package/dist/data/connectors/dingtalk.js +81 -0
  61. package/dist/data/connectors/dingtalk.js.map +1 -0
  62. package/dist/data/connectors/email.d.ts +18 -0
  63. package/dist/data/connectors/email.js +191 -0
  64. package/dist/data/connectors/email.js.map +1 -0
  65. package/dist/data/connectors/feishu.d.ts +18 -0
  66. package/dist/data/connectors/feishu.js +78 -0
  67. package/dist/data/connectors/feishu.js.map +1 -0
  68. package/dist/data/connectors/github.d.ts +10 -0
  69. package/dist/data/connectors/github.js +36 -0
  70. package/dist/data/connectors/github.js.map +1 -0
  71. package/dist/data/connectors/index.d.ts +3 -0
  72. package/dist/data/connectors/index.js +23 -0
  73. package/dist/data/connectors/index.js.map +1 -0
  74. package/dist/data/connectors/local-fs.d.ts +20 -0
  75. package/dist/data/connectors/local-fs.js +57 -0
  76. package/dist/data/connectors/local-fs.js.map +1 -0
  77. package/dist/data/connectors/notion.d.ts +10 -0
  78. package/dist/data/connectors/notion.js +46 -0
  79. package/dist/data/connectors/notion.js.map +1 -0
  80. package/dist/data/connectors/wecom.d.ts +18 -0
  81. package/dist/data/connectors/wecom.js +74 -0
  82. package/dist/data/connectors/wecom.js.map +1 -0
  83. package/dist/data/keys/store.d.ts +16 -0
  84. package/dist/data/keys/store.js +106 -0
  85. package/dist/data/keys/store.js.map +1 -0
  86. package/dist/data/profile/builder.d.ts +10 -0
  87. package/dist/data/profile/builder.js +48 -0
  88. package/dist/data/profile/builder.js.map +1 -0
  89. package/dist/data/skills/generator.d.ts +3 -0
  90. package/dist/data/skills/generator.js +72 -0
  91. package/dist/data/skills/generator.js.map +1 -0
  92. package/dist/data/sync/scheduler.d.ts +12 -0
  93. package/dist/data/sync/scheduler.js +51 -0
  94. package/dist/data/sync/scheduler.js.map +1 -0
  95. package/dist/deps/detector.d.ts +8 -0
  96. package/dist/deps/detector.js +19 -0
  97. package/dist/deps/detector.js.map +1 -0
  98. package/dist/deps/installer.d.ts +1 -0
  99. package/dist/deps/installer.js +38 -0
  100. package/dist/deps/installer.js.map +1 -0
  101. package/dist/deps/platform.d.ts +4 -0
  102. package/dist/deps/platform.js +24 -0
  103. package/dist/deps/platform.js.map +1 -0
  104. package/dist/health/monitor.d.ts +2 -0
  105. package/dist/health/monitor.js +18 -0
  106. package/dist/health/monitor.js.map +1 -0
  107. package/dist/proxy/auth.d.ts +7 -0
  108. package/dist/proxy/auth.js +22 -0
  109. package/dist/proxy/auth.js.map +1 -0
  110. package/dist/proxy/routes-agent.d.ts +3 -0
  111. package/dist/proxy/routes-agent.js +51 -0
  112. package/dist/proxy/routes-agent.js.map +1 -0
  113. package/dist/proxy/routes-config.d.ts +2 -0
  114. package/dist/proxy/routes-config.js +51 -0
  115. package/dist/proxy/routes-config.js.map +1 -0
  116. package/dist/proxy/routes-datasource.d.ts +2 -0
  117. package/dist/proxy/routes-datasource.js +30 -0
  118. package/dist/proxy/routes-datasource.js.map +1 -0
  119. package/dist/proxy/routes-keys.d.ts +7 -0
  120. package/dist/proxy/routes-keys.js +44 -0
  121. package/dist/proxy/routes-keys.js.map +1 -0
  122. package/dist/proxy/routes-profile.d.ts +2 -0
  123. package/dist/proxy/routes-profile.js +29 -0
  124. package/dist/proxy/routes-profile.js.map +1 -0
  125. package/dist/proxy/routes-status.d.ts +2 -0
  126. package/dist/proxy/routes-status.js +9 -0
  127. package/dist/proxy/routes-status.js.map +1 -0
  128. package/dist/proxy/routes-ttyd.d.ts +3 -0
  129. package/dist/proxy/routes-ttyd.js +41 -0
  130. package/dist/proxy/routes-ttyd.js.map +1 -0
  131. package/dist/proxy/routes-workspace.d.ts +2 -0
  132. package/dist/proxy/routes-workspace.js +77 -0
  133. package/dist/proxy/routes-workspace.js.map +1 -0
  134. package/dist/proxy/server.d.ts +22 -0
  135. package/dist/proxy/server.js +130 -0
  136. package/dist/proxy/server.js.map +1 -0
  137. package/dist/proxy/ttyd-manager.d.ts +19 -0
  138. package/dist/proxy/ttyd-manager.js +68 -0
  139. package/dist/proxy/ttyd-manager.js.map +1 -0
  140. package/dist/proxy/utils.d.ts +3 -0
  141. package/dist/proxy/utils.js +13 -0
  142. package/dist/proxy/utils.js.map +1 -0
  143. package/dist/proxy/ws-proxy.d.ts +6 -0
  144. package/dist/proxy/ws-proxy.js +30 -0
  145. package/dist/proxy/ws-proxy.js.map +1 -0
  146. package/package.json +40 -0
  147. package/ui/index.html +631 -0
@@ -0,0 +1,18 @@
1
+ export function startHealthMonitor(agents, intervalMs = 30_000) {
2
+ return setInterval(async () => {
3
+ for (const agent of agents) {
4
+ try {
5
+ const h = await agent.health();
6
+ if (!h.ok) {
7
+ console.log(`[openme] ${agent.name} unhealthy, restarting...`);
8
+ await agent.stop().catch(() => { });
9
+ await agent.start();
10
+ }
11
+ }
12
+ catch (err) {
13
+ console.error(`[openme] health check error for ${agent.name}:`, err);
14
+ }
15
+ }
16
+ }, intervalMs);
17
+ }
18
+ //# sourceMappingURL=monitor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"monitor.js","sourceRoot":"","sources":["../../src/health/monitor.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,kBAAkB,CAAC,MAAe,EAAE,UAAU,GAAG,MAAM;IACrE,OAAO,WAAW,CAAC,KAAK,IAAI,EAAE;QAC5B,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;YAC3B,IAAI,CAAC;gBACH,MAAM,CAAC,GAAG,MAAM,KAAK,CAAC,MAAM,EAAE,CAAC;gBAC/B,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC;oBACV,OAAO,CAAC,GAAG,CAAC,YAAY,KAAK,CAAC,IAAI,2BAA2B,CAAC,CAAC;oBAC/D,MAAM,KAAK,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;oBACnC,MAAM,KAAK,CAAC,KAAK,EAAE,CAAC;gBACtB,CAAC;YACH,CAAC;YAAC,OAAO,GAAG,EAAE,CAAC;gBACb,OAAO,CAAC,KAAK,CAAC,mCAAmC,KAAK,CAAC,IAAI,GAAG,EAAE,GAAG,CAAC,CAAC;YACvE,CAAC;QACH,CAAC;IACH,CAAC,EAAE,UAAU,CAAC,CAAC;AACjB,CAAC"}
@@ -0,0 +1,7 @@
1
+ import type { IncomingMessage, ServerResponse } from 'node:http';
2
+ /**
3
+ * Validate Bearer token or ?token= query param.
4
+ * Returns true if authenticated.
5
+ */
6
+ export declare function authenticate(req: IncomingMessage, token: string): boolean;
7
+ export declare function sendUnauthorized(res: ServerResponse): void;
@@ -0,0 +1,22 @@
1
+ /**
2
+ * Validate Bearer token or ?token= query param.
3
+ * Returns true if authenticated.
4
+ */
5
+ export function authenticate(req, token) {
6
+ // Allow health check without auth
7
+ if (req.url === '/openme/api/status')
8
+ return true;
9
+ const authHeader = req.headers.authorization;
10
+ if (authHeader === `Bearer ${token}`)
11
+ return true;
12
+ // Check query param
13
+ const url = new URL(req.url || '/', `http://${req.headers.host || 'localhost'}`);
14
+ if (url.searchParams.get('token') === token)
15
+ return true;
16
+ return false;
17
+ }
18
+ export function sendUnauthorized(res) {
19
+ res.writeHead(401, { 'Content-Type': 'application/json' });
20
+ res.end(JSON.stringify({ error: 'Unauthorized' }));
21
+ }
22
+ //# sourceMappingURL=auth.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth.js","sourceRoot":"","sources":["../../src/proxy/auth.ts"],"names":[],"mappings":"AAEA;;;GAGG;AACH,MAAM,UAAU,YAAY,CAAC,GAAoB,EAAE,KAAa;IAC9D,kCAAkC;IAClC,IAAI,GAAG,CAAC,GAAG,KAAK,oBAAoB;QAAE,OAAO,IAAI,CAAC;IAElD,MAAM,UAAU,GAAG,GAAG,CAAC,OAAO,CAAC,aAAa,CAAC;IAC7C,IAAI,UAAU,KAAK,UAAU,KAAK,EAAE;QAAE,OAAO,IAAI,CAAC;IAElD,oBAAoB;IACpB,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,GAAG,CAAC,GAAG,IAAI,GAAG,EAAE,UAAU,GAAG,CAAC,OAAO,CAAC,IAAI,IAAI,WAAW,EAAE,CAAC,CAAC;IACjF,IAAI,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,KAAK,KAAK;QAAE,OAAO,IAAI,CAAC;IAEzD,OAAO,KAAK,CAAC;AACf,CAAC;AAED,MAAM,UAAU,gBAAgB,CAAC,GAAmB;IAClD,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;IAC3D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,cAAc,EAAE,CAAC,CAAC,CAAC;AACrD,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { IncomingMessage, ServerResponse } from 'node:http';
2
+ import { Agent } from '../agents/agent.js';
3
+ export declare function createAgentRoutes(agents: Agent[]): (req: IncomingMessage, res: ServerResponse, path: string) => boolean;
@@ -0,0 +1,51 @@
1
+ export function createAgentRoutes(agents) {
2
+ return function handleAgentRoute(req, res, path) {
3
+ const json = (data, status = 200) => {
4
+ res.writeHead(status, { 'Content-Type': 'application/json' });
5
+ res.end(JSON.stringify(data));
6
+ };
7
+ // GET /openme/api/agents
8
+ if (req.method === 'GET' && path === '/openme/api/agents') {
9
+ Promise.all(agents.map(async (a) => {
10
+ const h = await a.health();
11
+ return { name: a.name, ...h, ...a.info() };
12
+ })).then((list) => json(list));
13
+ return true;
14
+ }
15
+ // POST /openme/api/agents/:name/restart
16
+ const restartMatch = path.match(/^\/openme\/api\/agents\/(.+)\/restart$/);
17
+ if (req.method === 'POST' && restartMatch) {
18
+ const agent = agents.find((a) => a.name === restartMatch[1]);
19
+ if (!agent) {
20
+ json({ error: 'Agent not found' }, 404);
21
+ return true;
22
+ }
23
+ agent.stop().then(() => agent.start()).then(() => json({ ok: true })).catch((e) => json({ error: e.message }, 500));
24
+ return true;
25
+ }
26
+ // POST /openme/api/agents/:name/stop
27
+ const stopMatch = path.match(/^\/openme\/api\/agents\/(.+)\/stop$/);
28
+ if (req.method === 'POST' && stopMatch) {
29
+ const agent = agents.find((a) => a.name === stopMatch[1]);
30
+ if (!agent) {
31
+ json({ error: 'Agent not found' }, 404);
32
+ return true;
33
+ }
34
+ agent.stop().then(() => json({ ok: true })).catch((e) => json({ error: e.message }, 500));
35
+ return true;
36
+ }
37
+ // POST /openme/api/agents/:name/start
38
+ const startMatch = path.match(/^\/openme\/api\/agents\/(.+)\/start$/);
39
+ if (req.method === 'POST' && startMatch) {
40
+ const agent = agents.find((a) => a.name === startMatch[1]);
41
+ if (!agent) {
42
+ json({ error: 'Agent not found' }, 404);
43
+ return true;
44
+ }
45
+ agent.start().then(() => json({ ok: true })).catch((e) => json({ error: e.message }, 500));
46
+ return true;
47
+ }
48
+ return false;
49
+ };
50
+ }
51
+ //# sourceMappingURL=routes-agent.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"routes-agent.js","sourceRoot":"","sources":["../../src/proxy/routes-agent.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import type { IncomingMessage, ServerResponse } from 'node:http';
2
+ export declare function handleConfigRoute(req: IncomingMessage, res: ServerResponse, path: string): boolean;
@@ -0,0 +1,51 @@
1
+ import { readFileSync, writeFileSync, existsSync } from 'node:fs';
2
+ import { homedir } from 'node:os';
3
+ import { join } from 'node:path';
4
+ const CONFIG_FILE = join(homedir(), '.openme', 'config.json');
5
+ export function handleConfigRoute(req, res, path) {
6
+ const json = (data, status = 200) => {
7
+ res.writeHead(status, { 'Content-Type': 'application/json' });
8
+ res.end(JSON.stringify(data));
9
+ };
10
+ // GET /openme/api/config
11
+ if (req.method === 'GET' && path === '/openme/api/config') {
12
+ if (!existsSync(CONFIG_FILE)) {
13
+ json({ error: 'Config not found' }, 404);
14
+ return true;
15
+ }
16
+ json(JSON.parse(readFileSync(CONFIG_FILE, 'utf-8')));
17
+ return true;
18
+ }
19
+ // PATCH /openme/api/config (JSON Merge Patch)
20
+ if (req.method === 'PATCH' && path === '/openme/api/config') {
21
+ let body = '';
22
+ req.on('data', (c) => (body += c));
23
+ req.on('end', () => {
24
+ try {
25
+ const current = JSON.parse(readFileSync(CONFIG_FILE, 'utf-8'));
26
+ const patch = JSON.parse(body);
27
+ const merged = deepMerge(current, patch);
28
+ writeFileSync(CONFIG_FILE, JSON.stringify(merged, null, 2));
29
+ json({ ok: true });
30
+ }
31
+ catch (err) {
32
+ json({ error: err.message }, 400);
33
+ }
34
+ });
35
+ return true;
36
+ }
37
+ return false;
38
+ }
39
+ function deepMerge(target, source) {
40
+ const result = { ...target };
41
+ for (const key of Object.keys(source)) {
42
+ if (source[key] && typeof source[key] === 'object' && !Array.isArray(source[key]) && target[key]) {
43
+ result[key] = deepMerge(target[key], source[key]);
44
+ }
45
+ else {
46
+ result[key] = source[key];
47
+ }
48
+ }
49
+ return result;
50
+ }
51
+ //# sourceMappingURL=routes-config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"routes-config.js","sourceRoot":"","sources":["../../src/proxy/routes-config.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import type { IncomingMessage, ServerResponse } from 'node:http';
2
+ export declare function createDatasourceRoutes(getConfig: () => any, syncOne: (type: string) => Promise<any>): (req: IncomingMessage, res: ServerResponse, path: string) => boolean;
@@ -0,0 +1,30 @@
1
+ export function createDatasourceRoutes(getConfig, syncOne) {
2
+ return function handleDatasourceRoute(req, res, path) {
3
+ const json = (data, status = 200) => {
4
+ res.writeHead(status, { 'Content-Type': 'application/json' });
5
+ res.end(JSON.stringify(data));
6
+ };
7
+ // GET /openme/api/datasources
8
+ if (req.method === 'GET' && path === '/openme/api/datasources') {
9
+ const config = getConfig();
10
+ const sources = Object.entries(config.dataSources || {}).map(([type, ds]) => ({
11
+ type,
12
+ enabled: ds.enabled,
13
+ syncTo: ds.syncTo,
14
+ syncInterval: ds.syncInterval,
15
+ }));
16
+ json(sources);
17
+ return true;
18
+ }
19
+ // POST /openme/api/datasources/:type/sync
20
+ const syncMatch = path.match(/^\/openme\/api\/datasources\/(.+)\/sync$/);
21
+ if (req.method === 'POST' && syncMatch) {
22
+ syncOne(syncMatch[1])
23
+ .then((r) => json({ ok: true, ...r }))
24
+ .catch((e) => json({ error: e.message }, 500));
25
+ return true;
26
+ }
27
+ return false;
28
+ };
29
+ }
30
+ //# sourceMappingURL=routes-datasource.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"routes-datasource.js","sourceRoot":"","sources":["../../src/proxy/routes-datasource.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,sBAAsB,CACpC,SAAoB,EACpB,OAAuC;IAEvC,OAAO,SAAS,qBAAqB,CAAC,GAAoB,EAAE,GAAmB,EAAE,IAAY;QAC3F,MAAM,IAAI,GAAG,CAAC,IAAa,EAAE,MAAM,GAAG,GAAG,EAAE,EAAE;YAC3C,GAAG,CAAC,SAAS,CAAC,MAAM,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;YAC9D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;QAChC,CAAC,CAAC;QAEF,8BAA8B;QAC9B,IAAI,GAAG,CAAC,MAAM,KAAK,KAAK,IAAI,IAAI,KAAK,yBAAyB,EAAE,CAAC;YAC/D,MAAM,MAAM,GAAG,SAAS,EAAE,CAAC;YAC3B,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,WAAW,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,EAAE,EAAE,CAAgB,EAAE,EAAE,CAAC,CAAC;gBAC3F,IAAI;gBACJ,OAAO,EAAE,EAAE,CAAC,OAAO;gBACnB,MAAM,EAAE,EAAE,CAAC,MAAM;gBACjB,YAAY,EAAE,EAAE,CAAC,YAAY;aAC9B,CAAC,CAAC,CAAC;YACJ,IAAI,CAAC,OAAO,CAAC,CAAC;YACd,OAAO,IAAI,CAAC;QACd,CAAC;QAED,0CAA0C;QAC1C,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,0CAA0C,CAAC,CAAC;QACzE,IAAI,GAAG,CAAC,MAAM,KAAK,MAAM,IAAI,SAAS,EAAE,CAAC;YACvC,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC;iBAClB,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,CAAC;iBACrC,KAAK,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC;YACtD,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC,CAAC;AACJ,CAAC"}
@@ -0,0 +1,7 @@
1
+ import type { IncomingMessage, ServerResponse } from 'node:http';
2
+ export declare function createKeysRoutes(keyStore: {
3
+ list(): string[];
4
+ load(type: string): Promise<any>;
5
+ save(type: string, creds: any): Promise<void>;
6
+ delete(type: string): Promise<boolean>;
7
+ }): (req: IncomingMessage, res: ServerResponse, path: string) => boolean;
@@ -0,0 +1,44 @@
1
+ export function createKeysRoutes(keyStore) {
2
+ return function handleKeysRoute(req, res, path) {
3
+ const json = (data, status = 200) => {
4
+ res.writeHead(status, { 'Content-Type': 'application/json' });
5
+ res.end(JSON.stringify(data));
6
+ };
7
+ // GET /openme/api/keys
8
+ if (req.method === 'GET' && path === '/openme/api/keys') {
9
+ json(keyStore.list().map((k) => ({ type: k, configured: true })));
10
+ return true;
11
+ }
12
+ const typeMatch = path.match(/^\/openme\/api\/keys\/(.+)$/);
13
+ if (!typeMatch)
14
+ return false;
15
+ const type = typeMatch[1];
16
+ // GET /openme/api/keys/:type — return decrypted credentials
17
+ if (req.method === 'GET') {
18
+ keyStore.load(type)
19
+ .then((creds) => json(creds))
20
+ .catch((e) => json({ error: e.message }, 404));
21
+ return true;
22
+ }
23
+ // PUT /openme/api/keys/:type
24
+ if (req.method === 'PUT') {
25
+ let body = '';
26
+ req.on('data', (c) => (body += c));
27
+ req.on('end', () => {
28
+ keyStore.save(type, JSON.parse(body))
29
+ .then(() => json({ ok: true }))
30
+ .catch((e) => json({ error: e.message }, 500));
31
+ });
32
+ return true;
33
+ }
34
+ // DELETE /openme/api/keys/:type
35
+ if (req.method === 'DELETE') {
36
+ keyStore.delete(type)
37
+ .then((ok) => json({ ok }, ok ? 200 : 404))
38
+ .catch((e) => json({ error: e.message }, 500));
39
+ return true;
40
+ }
41
+ return false;
42
+ };
43
+ }
44
+ //# sourceMappingURL=routes-keys.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"routes-keys.js","sourceRoot":"","sources":["../../src/proxy/routes-keys.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,gBAAgB,CAAC,QAAuJ;IACtL,OAAO,SAAS,eAAe,CAAC,GAAoB,EAAE,GAAmB,EAAE,IAAY;QACrF,MAAM,IAAI,GAAG,CAAC,IAAa,EAAE,MAAM,GAAG,GAAG,EAAE,EAAE;YAC3C,GAAG,CAAC,SAAS,CAAC,MAAM,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;YAC9D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;QAChC,CAAC,CAAC;QAEF,uBAAuB;QACvB,IAAI,GAAG,CAAC,MAAM,KAAK,KAAK,IAAI,IAAI,KAAK,kBAAkB,EAAE,CAAC;YACxD,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,CAAC,EAAE,UAAU,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;YAClE,OAAO,IAAI,CAAC;QACd,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,6BAA6B,CAAC,CAAC;QAC5D,IAAI,CAAC,SAAS;YAAE,OAAO,KAAK,CAAC;QAC7B,MAAM,IAAI,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC;QAE1B,4DAA4D;QAC5D,IAAI,GAAG,CAAC,MAAM,KAAK,KAAK,EAAE,CAAC;YACzB,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC;iBAChB,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;iBAC5B,KAAK,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC;YACtD,OAAO,IAAI,CAAC;QACd,CAAC;QAED,6BAA6B;QAC7B,IAAI,GAAG,CAAC,MAAM,KAAK,KAAK,EAAE,CAAC;YACzB,IAAI,IAAI,GAAG,EAAE,CAAC;YACd,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,CAAC;YAC3C,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE;gBACjB,QAAQ,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;qBAClC,IAAI,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC;qBAC9B,KAAK,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC;YACxD,CAAC,CAAC,CAAC;YACH,OAAO,IAAI,CAAC;QACd,CAAC;QAED,gCAAgC;QAChC,IAAI,GAAG,CAAC,MAAM,KAAK,QAAQ,EAAE,CAAC;YAC5B,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC;iBAClB,IAAI,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;iBAC1C,KAAK,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC;YACtD,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC,CAAC;AACJ,CAAC"}
@@ -0,0 +1,2 @@
1
+ import type { IncomingMessage, ServerResponse } from 'node:http';
2
+ export declare function createProfileRoutes(workspaceDir: string, refreshProfile: () => Promise<void>): (req: IncomingMessage, res: ServerResponse, path: string) => boolean;
@@ -0,0 +1,29 @@
1
+ import { readFileSync, existsSync } from 'node:fs';
2
+ import { join } from 'node:path';
3
+ export function createProfileRoutes(workspaceDir, refreshProfile) {
4
+ return function handleProfileRoute(req, res, path) {
5
+ const json = (data, status = 200) => {
6
+ res.writeHead(status, { 'Content-Type': 'application/json' });
7
+ res.end(JSON.stringify(data));
8
+ };
9
+ // GET /openme/api/profile
10
+ if (req.method === 'GET' && path === '/openme/api/profile') {
11
+ const profilePath = join(workspaceDir, 'USER.md');
12
+ if (!existsSync(profilePath)) {
13
+ json({ content: null });
14
+ return true;
15
+ }
16
+ json({ content: readFileSync(profilePath, 'utf-8') });
17
+ return true;
18
+ }
19
+ // POST /openme/api/profile/refresh
20
+ if (req.method === 'POST' && path === '/openme/api/profile/refresh') {
21
+ refreshProfile()
22
+ .then(() => json({ ok: true }))
23
+ .catch((e) => json({ error: e.message }, 500));
24
+ return true;
25
+ }
26
+ return false;
27
+ };
28
+ }
29
+ //# sourceMappingURL=routes-profile.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"routes-profile.js","sourceRoot":"","sources":["../../src/proxy/routes-profile.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACnD,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAEjC,MAAM,UAAU,mBAAmB,CAAC,YAAoB,EAAE,cAAmC;IAC3F,OAAO,SAAS,kBAAkB,CAAC,GAAoB,EAAE,GAAmB,EAAE,IAAY;QACxF,MAAM,IAAI,GAAG,CAAC,IAAa,EAAE,MAAM,GAAG,GAAG,EAAE,EAAE;YAC3C,GAAG,CAAC,SAAS,CAAC,MAAM,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;YAC9D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;QAChC,CAAC,CAAC;QAEF,0BAA0B;QAC1B,IAAI,GAAG,CAAC,MAAM,KAAK,KAAK,IAAI,IAAI,KAAK,qBAAqB,EAAE,CAAC;YAC3D,MAAM,WAAW,GAAG,IAAI,CAAC,YAAY,EAAE,SAAS,CAAC,CAAC;YAClD,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;gBAAC,IAAI,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC;gBAAC,OAAO,IAAI,CAAC;YAAC,CAAC;YACvE,IAAI,CAAC,EAAE,OAAO,EAAE,YAAY,CAAC,WAAW,EAAE,OAAO,CAAC,EAAE,CAAC,CAAC;YACtD,OAAO,IAAI,CAAC;QACd,CAAC;QAED,mCAAmC;QACnC,IAAI,GAAG,CAAC,MAAM,KAAK,MAAM,IAAI,IAAI,KAAK,6BAA6B,EAAE,CAAC;YACpE,cAAc,EAAE;iBACb,IAAI,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC;iBAC9B,KAAK,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,OAAO,EAAE,EAAE,GAAG,CAAC,CAAC,CAAC;YACtD,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC,CAAC;AACJ,CAAC"}
@@ -0,0 +1,2 @@
1
+ import type { IncomingMessage, ServerResponse } from 'node:http';
2
+ export declare function handleStatus(_req: IncomingMessage, res: ServerResponse): void;
@@ -0,0 +1,9 @@
1
+ import { json } from './utils.js';
2
+ export function handleStatus(_req, res) {
3
+ json(res, {
4
+ ok: true,
5
+ uptime: process.uptime(),
6
+ version: '0.1.0',
7
+ });
8
+ }
9
+ //# sourceMappingURL=routes-status.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"routes-status.js","sourceRoot":"","sources":["../../src/proxy/routes-status.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,IAAI,EAAE,MAAM,YAAY,CAAC;AAElC,MAAM,UAAU,YAAY,CAAC,IAAqB,EAAE,GAAmB;IACrE,IAAI,CAAC,GAAG,EAAE;QACR,EAAE,EAAE,IAAI;QACR,MAAM,EAAE,OAAO,CAAC,MAAM,EAAE;QACxB,OAAO,EAAE,OAAO;KACjB,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,3 @@
1
+ import type { IncomingMessage, ServerResponse } from 'node:http';
2
+ import { TtydManager } from './ttyd-manager.js';
3
+ export declare function createTtydRoutes(manager: TtydManager): (req: IncomingMessage, res: ServerResponse, path: string) => boolean;
@@ -0,0 +1,41 @@
1
+ export function createTtydRoutes(manager) {
2
+ return function handleTtydRoute(req, res, path) {
3
+ // GET /openme/api/ttyd/sessions
4
+ if (req.method === 'GET' && path === '/openme/api/ttyd/sessions') {
5
+ res.writeHead(200, { 'Content-Type': 'application/json' });
6
+ res.end(JSON.stringify(manager.list()));
7
+ return true;
8
+ }
9
+ // POST /openme/api/ttyd/sessions
10
+ if (req.method === 'POST' && path === '/openme/api/ttyd/sessions') {
11
+ let body = '';
12
+ req.on('data', (c) => (body += c));
13
+ req.on('end', () => {
14
+ try {
15
+ const { cmd } = JSON.parse(body);
16
+ if (!cmd)
17
+ throw new Error('cmd required');
18
+ const session = manager.create(cmd);
19
+ res.writeHead(201, { 'Content-Type': 'application/json' });
20
+ res.end(JSON.stringify(session));
21
+ }
22
+ catch (err) {
23
+ const status = err.message.includes('Max sessions') ? 429 : 400;
24
+ res.writeHead(status, { 'Content-Type': 'application/json' });
25
+ res.end(JSON.stringify({ error: err.message }));
26
+ }
27
+ });
28
+ return true;
29
+ }
30
+ // DELETE /openme/api/ttyd/sessions/:id
31
+ const delMatch = path.match(/^\/openme\/api\/ttyd\/sessions\/(.+)$/);
32
+ if (req.method === 'DELETE' && delMatch) {
33
+ const ok = manager.remove(delMatch[1]);
34
+ res.writeHead(ok ? 200 : 404, { 'Content-Type': 'application/json' });
35
+ res.end(JSON.stringify({ ok }));
36
+ return true;
37
+ }
38
+ return false;
39
+ };
40
+ }
41
+ //# sourceMappingURL=routes-ttyd.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"routes-ttyd.js","sourceRoot":"","sources":["../../src/proxy/routes-ttyd.ts"],"names":[],"mappings":"AAGA,MAAM,UAAU,gBAAgB,CAAC,OAAoB;IACnD,OAAO,SAAS,eAAe,CAAC,GAAoB,EAAE,GAAmB,EAAE,IAAY;QACrF,gCAAgC;QAChC,IAAI,GAAG,CAAC,MAAM,KAAK,KAAK,IAAI,IAAI,KAAK,2BAA2B,EAAE,CAAC;YACjE,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;YAC3D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;YACxC,OAAO,IAAI,CAAC;QACd,CAAC;QAED,iCAAiC;QACjC,IAAI,GAAG,CAAC,MAAM,KAAK,MAAM,IAAI,IAAI,KAAK,2BAA2B,EAAE,CAAC;YAClE,IAAI,IAAI,GAAG,EAAE,CAAC;YACd,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,CAAC;YAC3C,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE;gBACjB,IAAI,CAAC;oBACH,MAAM,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;oBACjC,IAAI,CAAC,GAAG;wBAAE,MAAM,IAAI,KAAK,CAAC,cAAc,CAAC,CAAC;oBAC1C,MAAM,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;oBACpC,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;oBAC3D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC;gBACnC,CAAC;gBAAC,OAAO,GAAQ,EAAE,CAAC;oBAClB,MAAM,MAAM,GAAG,GAAG,CAAC,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC;oBAChE,GAAG,CAAC,SAAS,CAAC,MAAM,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;oBAC9D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;gBAClD,CAAC;YACH,CAAC,CAAC,CAAC;YACH,OAAO,IAAI,CAAC;QACd,CAAC;QAED,uCAAuC;QACvC,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,uCAAuC,CAAC,CAAC;QACrE,IAAI,GAAG,CAAC,MAAM,KAAK,QAAQ,IAAI,QAAQ,EAAE,CAAC;YACxC,MAAM,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;YACvC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;YACtE,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC;YAChC,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC,CAAC;AACJ,CAAC"}
@@ -0,0 +1,2 @@
1
+ import type { IncomingMessage, ServerResponse } from 'node:http';
2
+ export declare function createWorkspaceRoutes(workspaceDir: string): (req: IncomingMessage, res: ServerResponse, path: string) => boolean;
@@ -0,0 +1,77 @@
1
+ import { readdirSync, readFileSync, statSync, existsSync } from 'node:fs';
2
+ import { join } from 'node:path';
3
+ export function createWorkspaceRoutes(workspaceDir) {
4
+ return function handleWorkspaceRoute(req, res, path) {
5
+ const json = (data, status = 200) => {
6
+ res.writeHead(status, { 'Content-Type': 'application/json' });
7
+ res.end(JSON.stringify(data));
8
+ };
9
+ // GET /openme/api/workspace/files
10
+ if (req.method === 'GET' && path === '/openme/api/workspace/files') {
11
+ try {
12
+ const files = listDir(workspaceDir, '');
13
+ json(files);
14
+ }
15
+ catch (err) {
16
+ json({ error: err.message }, 500);
17
+ }
18
+ return true;
19
+ }
20
+ // GET /openme/api/workspace/files/:path
21
+ const fileMatch = path.match(/^\/openme\/api\/workspace\/files\/(.+?)(\?.*)?$/);
22
+ if (req.method === 'GET' && fileMatch) {
23
+ const rel = decodeURIComponent(fileMatch[1]);
24
+ const qs = fileMatch[2] || '';
25
+ const recursive = qs.includes('recursive=true');
26
+ const filePath = join(workspaceDir, rel);
27
+ if (!filePath.startsWith(workspaceDir)) {
28
+ json({ error: 'Access denied' }, 403);
29
+ return true;
30
+ }
31
+ if (!existsSync(filePath)) {
32
+ json({ error: 'Not found' }, 404);
33
+ return true;
34
+ }
35
+ try {
36
+ if (statSync(filePath).isDirectory()) {
37
+ json(recursive ? listDirRecursive(workspaceDir, rel) : listDir(workspaceDir, rel));
38
+ }
39
+ else {
40
+ json({ path: rel, content: readFileSync(filePath, 'utf-8') });
41
+ }
42
+ }
43
+ catch (err) {
44
+ json({ error: err.message }, 500);
45
+ }
46
+ return true;
47
+ }
48
+ return false;
49
+ };
50
+ }
51
+ function listDir(base, rel) {
52
+ const dir = join(base, rel);
53
+ if (!existsSync(dir))
54
+ return [];
55
+ return readdirSync(dir, { withFileTypes: true }).map((d) => ({
56
+ name: d.name,
57
+ type: d.isDirectory() ? 'dir' : 'file',
58
+ path: rel ? `${rel}/${d.name}` : d.name,
59
+ }));
60
+ }
61
+ function listDirRecursive(base, rel) {
62
+ const results = [];
63
+ const walk = (r) => {
64
+ const dir = join(base, r);
65
+ if (!existsSync(dir))
66
+ return;
67
+ for (const d of readdirSync(dir, { withFileTypes: true })) {
68
+ const p = r ? `${r}/${d.name}` : d.name;
69
+ results.push({ name: d.name, type: d.isDirectory() ? 'dir' : 'file', path: p });
70
+ if (d.isDirectory())
71
+ walk(p);
72
+ }
73
+ };
74
+ walk(rel);
75
+ return results;
76
+ }
77
+ //# sourceMappingURL=routes-workspace.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"routes-workspace.js","sourceRoot":"","sources":["../../src/proxy/routes-workspace.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,22 @@
1
+ import http from 'node:http';
2
+ import type { Agent } from '../agents/agent.js';
3
+ export interface ProxyServerOptions {
4
+ port: number;
5
+ authToken: string;
6
+ openclawPort: number;
7
+ openclawHost?: string;
8
+ openclawToken?: string;
9
+ workspaceDir: string;
10
+ agents: Agent[];
11
+ keyStore: {
12
+ list(): string[];
13
+ load(type: string): Promise<any>;
14
+ save(type: string, creds: any): Promise<void>;
15
+ delete(type: string): Promise<boolean>;
16
+ };
17
+ getConfig: () => any;
18
+ syncOne: (type: string) => Promise<any>;
19
+ refreshProfile: () => Promise<void>;
20
+ ttydMaxSessions?: number;
21
+ }
22
+ export declare function createProxyServer(opts: ProxyServerOptions): Promise<http.Server>;