@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,130 @@
1
+ import http from 'node:http';
2
+ import { readFileSync, existsSync } from 'node:fs';
3
+ import { join, dirname } from 'node:path';
4
+ import { fileURLToPath } from 'node:url';
5
+ import { authenticate, sendUnauthorized } from './auth.js';
6
+ import { proxyWebSocket } from './ws-proxy.js';
7
+ import { handleStatus } from './routes-status.js';
8
+ import { handleConfigRoute } from './routes-config.js';
9
+ import { createWorkspaceRoutes } from './routes-workspace.js';
10
+ import { createAgentRoutes } from './routes-agent.js';
11
+ import { createTtydRoutes } from './routes-ttyd.js';
12
+ import { createKeysRoutes } from './routes-keys.js';
13
+ import { createDatasourceRoutes } from './routes-datasource.js';
14
+ import { createProfileRoutes } from './routes-profile.js';
15
+ import { TtydManager } from './ttyd-manager.js';
16
+ const __dir = dirname(fileURLToPath(import.meta.url));
17
+ const UI_DIR = join(__dir, '..', '..', 'ui');
18
+ export function createProxyServer(opts) {
19
+ const { port, authToken, openclawPort, openclawHost = '127.0.0.1' } = opts;
20
+ const openclawToken = opts.openclawToken || '';
21
+ const ttydManager = new TtydManager(7681, opts.ttydMaxSessions || 4);
22
+ const handleTtyd = createTtydRoutes(ttydManager);
23
+ const handleWorkspace = createWorkspaceRoutes(opts.workspaceDir);
24
+ const handleAgents = createAgentRoutes(opts.agents);
25
+ const handleKeys = createKeysRoutes(opts.keyStore);
26
+ const handleDatasource = createDatasourceRoutes(opts.getConfig, opts.syncOne);
27
+ const handleProfile = createProfileRoutes(opts.workspaceDir, opts.refreshProfile);
28
+ function proxyToOpenclaw(req, res, path) {
29
+ const headers = { ...req.headers, host: `${openclawHost}:${openclawPort}` };
30
+ if (openclawToken)
31
+ headers['authorization'] = `Bearer ${openclawToken}`;
32
+ const proxyReq = http.request({ hostname: openclawHost, port: openclawPort, path, method: req.method, headers }, (upstream) => {
33
+ const headers = { ...upstream.headers };
34
+ delete headers['content-security-policy'];
35
+ delete headers['x-frame-options'];
36
+ res.writeHead(upstream.statusCode || 502, headers);
37
+ upstream.pipe(res);
38
+ });
39
+ proxyReq.on('error', () => { res.writeHead(502); res.end('openclaw unavailable'); });
40
+ req.pipe(proxyReq);
41
+ }
42
+ function serveStatic(res, filePath, inject) {
43
+ if (!existsSync(filePath)) {
44
+ res.writeHead(404);
45
+ res.end('Not Found');
46
+ return;
47
+ }
48
+ const ext = filePath.split('.').pop() || '';
49
+ const mime = { html: 'text/html', css: 'text/css', js: 'application/javascript', json: 'application/json', png: 'image/png', svg: 'image/svg+xml' };
50
+ let content = readFileSync(filePath, 'utf-8');
51
+ if (inject && ext === 'html') {
52
+ const script = `<script>window.__OPENME__=${JSON.stringify(inject)};</script>`;
53
+ content = content.replace('</head>', `${script}\n</head>`);
54
+ }
55
+ res.writeHead(200, { 'Content-Type': mime[ext] || 'text/plain' });
56
+ res.end(content);
57
+ }
58
+ const server = http.createServer((req, res) => {
59
+ const url = req.url || '/';
60
+ // Auth for API routes
61
+ if (url.startsWith('/openme/api/') && !authenticate(req, authToken)) {
62
+ return sendUnauthorized(res);
63
+ }
64
+ // ── OpenMe REST API ──
65
+ if (url === '/openme/api/status')
66
+ return handleStatus(req, res);
67
+ if (handleConfigRoute(req, res, url))
68
+ return;
69
+ if (handleTtyd(req, res, url))
70
+ return;
71
+ if (handleWorkspace(req, res, url))
72
+ return;
73
+ if (handleAgents(req, res, url))
74
+ return;
75
+ if (handleKeys(req, res, url))
76
+ return;
77
+ if (handleDatasource(req, res, url))
78
+ return;
79
+ if (handleProfile(req, res, url))
80
+ return;
81
+ // ── Static UI ──
82
+ if (url === '/' || url === '/index.html')
83
+ return serveStatic(res, join(UI_DIR, 'index.html'), { token: authToken, ocToken: openclawToken });
84
+ if (url.startsWith('/ui/'))
85
+ return serveStatic(res, join(UI_DIR, url.slice(4)));
86
+ // ── Proxy to openclaw ──
87
+ if (url.startsWith('/openclaw/'))
88
+ return proxyToOpenclaw(req, res, url.slice('/openclaw'.length) || '/');
89
+ // Default: proxy to openclaw
90
+ proxyToOpenclaw(req, res, url);
91
+ });
92
+ server.on('upgrade', (req, socket, head) => {
93
+ const url = req.url || '/';
94
+ // ttyd WS proxy: /ttyd/s/:id/*
95
+ const ttydMatch = url.match(/^\/ttyd\/s\/([^/]+)(\/.*)?$/);
96
+ if (ttydMatch) {
97
+ const sessionPort = ttydManager.getPort(ttydMatch[1]);
98
+ if (sessionPort) {
99
+ proxyWebSocket(req, socket, head, '127.0.0.1', sessionPort, ttydMatch[2] || '/');
100
+ return;
101
+ }
102
+ }
103
+ // openclaw WS
104
+ const ocAuth = openclawToken ? { authorization: `Bearer ${openclawToken}` } : {};
105
+ if (url.startsWith('/openclaw/')) {
106
+ proxyWebSocket(req, socket, head, openclawHost, openclawPort, url.slice('/openclaw'.length) || '/', ocAuth);
107
+ }
108
+ else {
109
+ proxyWebSocket(req, socket, head, openclawHost, openclawPort, undefined, ocAuth);
110
+ }
111
+ });
112
+ return new Promise((resolve, reject) => {
113
+ server.on('error', (err) => {
114
+ if (err.code === 'EADDRINUSE') {
115
+ console.error(`[openme] Port ${port} is already in use. Run 'openme stop' first or kill the process.`);
116
+ }
117
+ reject(err);
118
+ });
119
+ server.listen(port, () => {
120
+ console.log(`[openme] Proxy listening on http://localhost:${port}`);
121
+ console.log(`[openme] ├─ UI: http://localhost:${port}/`);
122
+ console.log(`[openme] ├─ openclaw: http://localhost:${port}/openclaw/`);
123
+ console.log(`[openme] ├─ API: http://localhost:${port}/openme/api/`);
124
+ console.log(`[openme] └─ ttyd: http://localhost:${port}/ttyd/s/:id/`);
125
+ resolve(server);
126
+ });
127
+ server.on('close', () => ttydManager.stopAll());
128
+ });
129
+ }
130
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/proxy/server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,19 @@
1
+ export interface TtydSession {
2
+ id: string;
3
+ cmd: string;
4
+ port: number;
5
+ pid?: number;
6
+ createdAt: string;
7
+ }
8
+ export declare class TtydManager {
9
+ private basePort;
10
+ private maxSessions;
11
+ private sessions;
12
+ private nextPort;
13
+ constructor(basePort?: number, maxSessions?: number);
14
+ create(cmd: string): TtydSession;
15
+ remove(id: string): boolean;
16
+ list(): TtydSession[];
17
+ getPort(id: string): number | undefined;
18
+ stopAll(): void;
19
+ }
@@ -0,0 +1,68 @@
1
+ import { spawn, execSync } from 'node:child_process';
2
+ import { randomUUID } from 'node:crypto';
3
+ export class TtydManager {
4
+ basePort;
5
+ maxSessions;
6
+ sessions = new Map();
7
+ nextPort;
8
+ constructor(basePort = 7681, maxSessions = 4) {
9
+ this.basePort = basePort;
10
+ this.maxSessions = maxSessions;
11
+ this.nextPort = basePort;
12
+ }
13
+ create(cmd) {
14
+ if (this.sessions.size >= this.maxSessions) {
15
+ throw new Error(`Max sessions (${this.maxSessions}) reached`);
16
+ }
17
+ const id = randomUUID().slice(0, 8);
18
+ const port = this.nextPort++;
19
+ // Kill anything on this port first
20
+ try {
21
+ execSync(`lsof -ti:${port} | xargs kill -9 2>/dev/null`, { stdio: 'ignore' });
22
+ }
23
+ catch { }
24
+ const args = ['-W', '-p', String(port), '-t', 'fontSize=14',
25
+ '-t', 'theme={"background":"#0d1117","foreground":"#c9d1d9"}',
26
+ '-t', 'disableResizeOverlay=true',
27
+ ...cmd.split(/\s+/)];
28
+ const proc = spawn('ttyd', args, { stdio: 'ignore', detached: false });
29
+ proc.on('exit', () => this.sessions.delete(id));
30
+ const meta = {
31
+ id,
32
+ cmd,
33
+ port,
34
+ pid: proc.pid,
35
+ createdAt: new Date().toISOString(),
36
+ };
37
+ this.sessions.set(id, { proc, meta });
38
+ console.log(`[openme] ttyd session ${id} :${port} → ${cmd}`);
39
+ return meta;
40
+ }
41
+ remove(id) {
42
+ const s = this.sessions.get(id);
43
+ if (!s)
44
+ return false;
45
+ try {
46
+ s.proc.kill('SIGTERM');
47
+ }
48
+ catch { }
49
+ this.sessions.delete(id);
50
+ return true;
51
+ }
52
+ list() {
53
+ return [...this.sessions.values()].map((s) => s.meta);
54
+ }
55
+ getPort(id) {
56
+ return this.sessions.get(id)?.meta.port;
57
+ }
58
+ stopAll() {
59
+ for (const [id, s] of this.sessions) {
60
+ try {
61
+ s.proc.kill('SIGTERM');
62
+ }
63
+ catch { }
64
+ }
65
+ this.sessions.clear();
66
+ }
67
+ }
68
+ //# sourceMappingURL=ttyd-manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ttyd-manager.js","sourceRoot":"","sources":["../../src/proxy/ttyd-manager.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { ServerResponse } from 'node:http';
2
+ export declare function json(res: ServerResponse, data: unknown, status?: number): void;
3
+ export declare function readBody(req: import('node:http').IncomingMessage): Promise<string>;
@@ -0,0 +1,13 @@
1
+ export function json(res, data, status = 200) {
2
+ res.writeHead(status, { 'Content-Type': 'application/json' });
3
+ res.end(JSON.stringify(data));
4
+ }
5
+ export function readBody(req) {
6
+ return new Promise((resolve, reject) => {
7
+ let body = '';
8
+ req.on('data', (c) => (body += c));
9
+ req.on('end', () => resolve(body));
10
+ req.on('error', reject);
11
+ });
12
+ }
13
+ //# sourceMappingURL=utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"utils.js","sourceRoot":"","sources":["../../src/proxy/utils.ts"],"names":[],"mappings":"AAEA,MAAM,UAAU,IAAI,CAAC,GAAmB,EAAE,IAAa,EAAE,MAAM,GAAG,GAAG;IACnE,GAAG,CAAC,SAAS,CAAC,MAAM,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;IAC9D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;AAChC,CAAC;AAED,MAAM,UAAU,QAAQ,CAAC,GAAwC;IAC/D,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACrC,IAAI,IAAI,GAAG,EAAE,CAAC;QACd,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,CAAS,EAAE,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,CAAC;QAC3C,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC;QACnC,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;IAC1B,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,6 @@
1
+ import http from 'node:http';
2
+ import type { Duplex } from 'node:stream';
3
+ /**
4
+ * Proxy a WebSocket upgrade request to a target host:port.
5
+ */
6
+ export declare function proxyWebSocket(req: http.IncomingMessage, socket: Duplex, head: Buffer, targetHost: string, targetPort: number, targetPath?: string, extraHeaders?: Record<string, string>): void;
@@ -0,0 +1,30 @@
1
+ import http from 'node:http';
2
+ /**
3
+ * Proxy a WebSocket upgrade request to a target host:port.
4
+ */
5
+ export function proxyWebSocket(req, socket, head, targetHost, targetPort, targetPath, extraHeaders) {
6
+ const path = targetPath ?? req.url ?? '/';
7
+ socket.on('error', () => socket.destroy());
8
+ const proxy = http.request({
9
+ hostname: targetHost,
10
+ port: targetPort,
11
+ path,
12
+ method: 'GET',
13
+ headers: { ...req.headers, host: `${targetHost}:${targetPort}`, ...extraHeaders },
14
+ });
15
+ proxy.on('upgrade', (_proxyRes, proxySocket, proxyHead) => {
16
+ proxySocket.on('error', () => proxySocket.destroy());
17
+ socket.write(`HTTP/1.1 101 Switching Protocols\r\n` +
18
+ Object.entries(_proxyRes.headers)
19
+ .map(([k, v]) => `${k}: ${v}`)
20
+ .join('\r\n') +
21
+ '\r\n\r\n');
22
+ if (proxyHead.length)
23
+ socket.write(proxyHead);
24
+ proxySocket.pipe(socket);
25
+ socket.pipe(proxySocket);
26
+ });
27
+ proxy.on('error', () => socket.destroy());
28
+ proxy.end();
29
+ }
30
+ //# sourceMappingURL=ws-proxy.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws-proxy.js","sourceRoot":"","sources":["../../src/proxy/ws-proxy.ts"],"names":[],"mappings":"AAAA,OAAO,IAAI,MAAM,WAAW,CAAC;AAG7B;;GAEG;AACH,MAAM,UAAU,cAAc,CAC5B,GAAyB,EACzB,MAAc,EACd,IAAY,EACZ,UAAkB,EAClB,UAAkB,EAClB,UAAmB,EACnB,YAAqC;IAErC,MAAM,IAAI,GAAG,UAAU,IAAI,GAAG,CAAC,GAAG,IAAI,GAAG,CAAC;IAE1C,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC,CAAC;IAE3C,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC;QACzB,QAAQ,EAAE,UAAU;QACpB,IAAI,EAAE,UAAU;QAChB,IAAI;QACJ,MAAM,EAAE,KAAK;QACb,OAAO,EAAE,EAAE,GAAG,GAAG,CAAC,OAAO,EAAE,IAAI,EAAE,GAAG,UAAU,IAAI,UAAU,EAAE,EAAE,GAAG,YAAY,EAAE;KAClF,CAAC,CAAC;IAEH,KAAK,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,SAAS,EAAE,WAAW,EAAE,SAAS,EAAE,EAAE;QACxD,WAAW,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC,CAAC;QAErD,MAAM,CAAC,KAAK,CACV,sCAAsC;YACpC,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,OAAO,CAAC;iBAC9B,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;iBAC7B,IAAI,CAAC,MAAM,CAAC;YACf,UAAU,CACb,CAAC;QACF,IAAI,SAAS,CAAC,MAAM;YAAE,MAAM,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAC9C,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACzB,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IAC3B,CAAC,CAAC,CAAC;IAEH,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC,CAAC;IAC1C,KAAK,CAAC,GAAG,EAAE,CAAC;AACd,CAAC"}
package/package.json ADDED
@@ -0,0 +1,40 @@
1
+ {
2
+ "name": "@shadanai/openme",
3
+ "version": "0.1.0",
4
+ "description": "OpenMe — Personal AI Data Runtime. Unified agent orchestration + personal data engine.",
5
+ "type": "module",
6
+ "bin": {
7
+ "openme": "./bin/openme.js"
8
+ },
9
+ "scripts": {
10
+ "build": "tsc",
11
+ "dev": "tsc --watch",
12
+ "start": "node dist/cli/index.js"
13
+ },
14
+ "engines": {
15
+ "node": ">=22"
16
+ },
17
+ "dependencies": {
18
+ "commander": "^12.1.0",
19
+ "mailparser": "^3.9.3"
20
+ },
21
+ "peerDependencies": {
22
+ "openclaw": ">=2025.0.0"
23
+ },
24
+ "peerDependenciesMeta": {
25
+ "openclaw": {
26
+ "optional": true
27
+ }
28
+ },
29
+ "devDependencies": {
30
+ "@types/mailparser": "^3.4.6",
31
+ "@types/node": "^22.0.0",
32
+ "typescript": "^5.3.0"
33
+ },
34
+ "files": [
35
+ "dist",
36
+ "bin",
37
+ "ui"
38
+ ],
39
+ "license": "MIT"
40
+ }