@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.
- package/bin/openme.js +2 -0
- package/dist/agents/agent.d.ts +10 -0
- package/dist/agents/agent.js +2 -0
- package/dist/agents/agent.js.map +1 -0
- package/dist/agents/happy.d.ts +15 -0
- package/dist/agents/happy.js +43 -0
- package/dist/agents/happy.js.map +1 -0
- package/dist/agents/openclaw.d.ts +16 -0
- package/dist/agents/openclaw.js +55 -0
- package/dist/agents/openclaw.js.map +1 -0
- package/dist/agents/proxy.d.ts +15 -0
- package/dist/agents/proxy.js +36 -0
- package/dist/agents/proxy.js.map +1 -0
- package/dist/cli/cmd-agent.d.ts +1 -0
- package/dist/cli/cmd-agent.js +73 -0
- package/dist/cli/cmd-agent.js.map +1 -0
- package/dist/cli/cmd-datasource.d.ts +1 -0
- package/dist/cli/cmd-datasource.js +78 -0
- package/dist/cli/cmd-datasource.js.map +1 -0
- package/dist/cli/cmd-init.d.ts +7 -0
- package/dist/cli/cmd-init.js +26 -0
- package/dist/cli/cmd-init.js.map +1 -0
- package/dist/cli/cmd-install.d.ts +5 -0
- package/dist/cli/cmd-install.js +22 -0
- package/dist/cli/cmd-install.js.map +1 -0
- package/dist/cli/cmd-start.d.ts +5 -0
- package/dist/cli/cmd-start.js +136 -0
- package/dist/cli/cmd-start.js.map +1 -0
- package/dist/cli/cmd-status.d.ts +1 -0
- package/dist/cli/cmd-status.js +26 -0
- package/dist/cli/cmd-status.js.map +1 -0
- package/dist/cli/cmd-stop.d.ts +1 -0
- package/dist/cli/cmd-stop.js +17 -0
- package/dist/cli/cmd-stop.js.map +1 -0
- package/dist/cli/cmd-workspace.d.ts +1 -0
- package/dist/cli/cmd-workspace.js +32 -0
- package/dist/cli/cmd-workspace.js.map +1 -0
- package/dist/cli/index.d.ts +1 -0
- package/dist/cli/index.js +46 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/core/config-sync.d.ts +6 -0
- package/dist/core/config-sync.js +41 -0
- package/dist/core/config-sync.js.map +1 -0
- package/dist/core/config.d.ts +96 -0
- package/dist/core/config.js +91 -0
- package/dist/core/config.js.map +1 -0
- package/dist/core/node-id.d.ts +1 -0
- package/dist/core/node-id.js +5 -0
- package/dist/core/node-id.js.map +1 -0
- package/dist/core/register.d.ts +19 -0
- package/dist/core/register.js +31 -0
- package/dist/core/register.js.map +1 -0
- package/dist/data/connectors/baidupan.d.ts +20 -0
- package/dist/data/connectors/baidupan.js +69 -0
- package/dist/data/connectors/baidupan.js.map +1 -0
- package/dist/data/connectors/connector.d.ts +12 -0
- package/dist/data/connectors/connector.js +2 -0
- package/dist/data/connectors/connector.js.map +1 -0
- package/dist/data/connectors/dingtalk.d.ts +18 -0
- package/dist/data/connectors/dingtalk.js +81 -0
- package/dist/data/connectors/dingtalk.js.map +1 -0
- package/dist/data/connectors/email.d.ts +18 -0
- package/dist/data/connectors/email.js +191 -0
- package/dist/data/connectors/email.js.map +1 -0
- package/dist/data/connectors/feishu.d.ts +18 -0
- package/dist/data/connectors/feishu.js +78 -0
- package/dist/data/connectors/feishu.js.map +1 -0
- package/dist/data/connectors/github.d.ts +10 -0
- package/dist/data/connectors/github.js +36 -0
- package/dist/data/connectors/github.js.map +1 -0
- package/dist/data/connectors/index.d.ts +3 -0
- package/dist/data/connectors/index.js +23 -0
- package/dist/data/connectors/index.js.map +1 -0
- package/dist/data/connectors/local-fs.d.ts +20 -0
- package/dist/data/connectors/local-fs.js +57 -0
- package/dist/data/connectors/local-fs.js.map +1 -0
- package/dist/data/connectors/notion.d.ts +10 -0
- package/dist/data/connectors/notion.js +46 -0
- package/dist/data/connectors/notion.js.map +1 -0
- package/dist/data/connectors/wecom.d.ts +18 -0
- package/dist/data/connectors/wecom.js +74 -0
- package/dist/data/connectors/wecom.js.map +1 -0
- package/dist/data/keys/store.d.ts +16 -0
- package/dist/data/keys/store.js +106 -0
- package/dist/data/keys/store.js.map +1 -0
- package/dist/data/profile/builder.d.ts +10 -0
- package/dist/data/profile/builder.js +48 -0
- package/dist/data/profile/builder.js.map +1 -0
- package/dist/data/skills/generator.d.ts +3 -0
- package/dist/data/skills/generator.js +72 -0
- package/dist/data/skills/generator.js.map +1 -0
- package/dist/data/sync/scheduler.d.ts +12 -0
- package/dist/data/sync/scheduler.js +51 -0
- package/dist/data/sync/scheduler.js.map +1 -0
- package/dist/deps/detector.d.ts +8 -0
- package/dist/deps/detector.js +19 -0
- package/dist/deps/detector.js.map +1 -0
- package/dist/deps/installer.d.ts +1 -0
- package/dist/deps/installer.js +38 -0
- package/dist/deps/installer.js.map +1 -0
- package/dist/deps/platform.d.ts +4 -0
- package/dist/deps/platform.js +24 -0
- package/dist/deps/platform.js.map +1 -0
- package/dist/health/monitor.d.ts +2 -0
- package/dist/health/monitor.js +18 -0
- package/dist/health/monitor.js.map +1 -0
- package/dist/proxy/auth.d.ts +7 -0
- package/dist/proxy/auth.js +22 -0
- package/dist/proxy/auth.js.map +1 -0
- package/dist/proxy/routes-agent.d.ts +3 -0
- package/dist/proxy/routes-agent.js +51 -0
- package/dist/proxy/routes-agent.js.map +1 -0
- package/dist/proxy/routes-config.d.ts +2 -0
- package/dist/proxy/routes-config.js +51 -0
- package/dist/proxy/routes-config.js.map +1 -0
- package/dist/proxy/routes-datasource.d.ts +2 -0
- package/dist/proxy/routes-datasource.js +30 -0
- package/dist/proxy/routes-datasource.js.map +1 -0
- package/dist/proxy/routes-keys.d.ts +7 -0
- package/dist/proxy/routes-keys.js +44 -0
- package/dist/proxy/routes-keys.js.map +1 -0
- package/dist/proxy/routes-profile.d.ts +2 -0
- package/dist/proxy/routes-profile.js +29 -0
- package/dist/proxy/routes-profile.js.map +1 -0
- package/dist/proxy/routes-status.d.ts +2 -0
- package/dist/proxy/routes-status.js +9 -0
- package/dist/proxy/routes-status.js.map +1 -0
- package/dist/proxy/routes-ttyd.d.ts +3 -0
- package/dist/proxy/routes-ttyd.js +41 -0
- package/dist/proxy/routes-ttyd.js.map +1 -0
- package/dist/proxy/routes-workspace.d.ts +2 -0
- package/dist/proxy/routes-workspace.js +77 -0
- package/dist/proxy/routes-workspace.js.map +1 -0
- package/dist/proxy/server.d.ts +22 -0
- package/dist/proxy/server.js +130 -0
- package/dist/proxy/server.js.map +1 -0
- package/dist/proxy/ttyd-manager.d.ts +19 -0
- package/dist/proxy/ttyd-manager.js +68 -0
- package/dist/proxy/ttyd-manager.js.map +1 -0
- package/dist/proxy/utils.d.ts +3 -0
- package/dist/proxy/utils.js +13 -0
- package/dist/proxy/utils.js.map +1 -0
- package/dist/proxy/ws-proxy.d.ts +6 -0
- package/dist/proxy/ws-proxy.js +30 -0
- package/dist/proxy/ws-proxy.js.map +1 -0
- package/package.json +40 -0
- package/ui/index.html +631 -0
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { mkdirSync, writeFileSync } from 'node:fs';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
export class NotionConnector {
|
|
4
|
+
type = 'notion';
|
|
5
|
+
token = '';
|
|
6
|
+
async connect(credentials) {
|
|
7
|
+
this.token = credentials.token;
|
|
8
|
+
}
|
|
9
|
+
async sync(targetDir) {
|
|
10
|
+
mkdirSync(targetDir, { recursive: true });
|
|
11
|
+
const result = { added: 0, updated: 0, deleted: 0, errors: [] };
|
|
12
|
+
try {
|
|
13
|
+
const res = await fetch('https://api.notion.com/v1/search', {
|
|
14
|
+
method: 'POST',
|
|
15
|
+
headers: {
|
|
16
|
+
Authorization: `Bearer ${this.token}`,
|
|
17
|
+
'Notion-Version': '2022-06-28',
|
|
18
|
+
'Content-Type': 'application/json',
|
|
19
|
+
},
|
|
20
|
+
body: JSON.stringify({ filter: { property: 'object', value: 'page' }, page_size: 50 }),
|
|
21
|
+
});
|
|
22
|
+
if (!res.ok) {
|
|
23
|
+
result.errors.push(`Notion API: ${res.status}`);
|
|
24
|
+
return result;
|
|
25
|
+
}
|
|
26
|
+
const data = (await res.json());
|
|
27
|
+
for (const page of data.results || []) {
|
|
28
|
+
const title = page.properties?.title?.title?.[0]?.plain_text
|
|
29
|
+
|| page.properties?.Name?.title?.[0]?.plain_text
|
|
30
|
+
|| page.id;
|
|
31
|
+
const safeName = title.replace(/[^a-zA-Z0-9\u4e00-\u9fff_-]/g, '_').slice(0, 80);
|
|
32
|
+
const md = `# ${title}\n\n- ID: ${page.id}\n- Created: ${page.created_time}\n- Updated: ${page.last_edited_time}\n- URL: ${page.url}\n`;
|
|
33
|
+
writeFileSync(join(targetDir, `${safeName}.md`), md);
|
|
34
|
+
result.added++;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
catch (err) {
|
|
38
|
+
result.errors.push(err.message);
|
|
39
|
+
}
|
|
40
|
+
return result;
|
|
41
|
+
}
|
|
42
|
+
async disconnect() {
|
|
43
|
+
this.token = '';
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
//# sourceMappingURL=notion.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"notion.js","sourceRoot":"","sources":["../../../src/data/connectors/notion.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AACnD,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAGjC,MAAM,OAAO,eAAe;IACjB,IAAI,GAAG,QAAQ,CAAC;IACjB,KAAK,GAAG,EAAE,CAAC;IAEnB,KAAK,CAAC,OAAO,CAAC,WAA8B;QAC1C,IAAI,CAAC,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC;IACjC,CAAC;IAED,KAAK,CAAC,IAAI,CAAC,SAAiB;QAC1B,SAAS,CAAC,SAAS,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QAC1C,MAAM,MAAM,GAAe,EAAE,KAAK,EAAE,CAAC,EAAE,OAAO,EAAE,CAAC,EAAE,OAAO,EAAE,CAAC,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC;QAE5E,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,kCAAkC,EAAE;gBAC1D,MAAM,EAAE,MAAM;gBACd,OAAO,EAAE;oBACP,aAAa,EAAE,UAAU,IAAI,CAAC,KAAK,EAAE;oBACrC,gBAAgB,EAAE,YAAY;oBAC9B,cAAc,EAAE,kBAAkB;iBACnC;gBACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,MAAM,EAAE,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,CAAC;aACvF,CAAC,CAAC;YACH,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC;gBAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,eAAe,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC;gBAAC,OAAO,MAAM,CAAC;YAAC,CAAC;YAEhF,MAAM,IAAI,GAAG,CAAC,MAAM,GAAG,CAAC,IAAI,EAAE,CAAQ,CAAC;YACvC,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,OAAO,IAAI,EAAE,EAAE,CAAC;gBACtC,MAAM,KAAK,GAAG,IAAI,CAAC,UAAU,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,EAAE,UAAU;uBACvD,IAAI,CAAC,UAAU,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,EAAE,UAAU;uBAC7C,IAAI,CAAC,EAAE,CAAC;gBACb,MAAM,QAAQ,GAAG,KAAK,CAAC,OAAO,CAAC,8BAA8B,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBACjF,MAAM,EAAE,GAAG,KAAK,KAAK,aAAa,IAAI,CAAC,EAAE,gBAAgB,IAAI,CAAC,YAAY,gBAAgB,IAAI,CAAC,gBAAgB,YAAY,IAAI,CAAC,GAAG,IAAI,CAAC;gBACxI,aAAa,CAAC,IAAI,CAAC,SAAS,EAAE,GAAG,QAAQ,KAAK,CAAC,EAAE,EAAE,CAAC,CAAC;gBACrD,MAAM,CAAC,KAAK,EAAE,CAAC;YACjB,CAAC;QACH,CAAC;QAAC,OAAO,GAAQ,EAAE,CAAC;YAClB,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QAClC,CAAC;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,KAAK,CAAC,UAAU;QACd,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC;IAClB,CAAC;CACF"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import { Connector, SyncResult } from './connector.js';
|
|
2
|
+
/**
|
|
3
|
+
* WeCom (企业微信) connector.
|
|
4
|
+
* Uses API: https://developer.work.weixin.qq.com/document/path/91039
|
|
5
|
+
*
|
|
6
|
+
* Credentials: { corpId, corpSecret }
|
|
7
|
+
*/
|
|
8
|
+
export declare class WeComConnector implements Connector {
|
|
9
|
+
readonly type = "wecom";
|
|
10
|
+
private token;
|
|
11
|
+
connect(credentials: {
|
|
12
|
+
corpId: string;
|
|
13
|
+
corpSecret: string;
|
|
14
|
+
}): Promise<void>;
|
|
15
|
+
sync(targetDir: string): Promise<SyncResult>;
|
|
16
|
+
disconnect(): Promise<void>;
|
|
17
|
+
private api;
|
|
18
|
+
}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { mkdirSync, writeFileSync } from 'node:fs';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
/**
|
|
4
|
+
* WeCom (企业微信) connector.
|
|
5
|
+
* Uses API: https://developer.work.weixin.qq.com/document/path/91039
|
|
6
|
+
*
|
|
7
|
+
* Credentials: { corpId, corpSecret }
|
|
8
|
+
*/
|
|
9
|
+
export class WeComConnector {
|
|
10
|
+
type = 'wecom';
|
|
11
|
+
token = '';
|
|
12
|
+
async connect(credentials) {
|
|
13
|
+
const res = await fetch(`https://qyapi.weixin.qq.com/cgi-bin/gettoken?corpid=${credentials.corpId}&corpsecret=${credentials.corpSecret}`);
|
|
14
|
+
const data = await res.json();
|
|
15
|
+
if (data.errcode !== 0)
|
|
16
|
+
throw new Error(`WeCom auth failed: ${data.errmsg}`);
|
|
17
|
+
this.token = data.access_token;
|
|
18
|
+
}
|
|
19
|
+
async sync(targetDir) {
|
|
20
|
+
mkdirSync(targetDir, { recursive: true });
|
|
21
|
+
const result = { added: 0, updated: 0, deleted: 0, errors: [] };
|
|
22
|
+
if (!this.token) {
|
|
23
|
+
result.errors.push('Not connected');
|
|
24
|
+
return result;
|
|
25
|
+
}
|
|
26
|
+
try {
|
|
27
|
+
// Get department list
|
|
28
|
+
const deptRes = await this.api('/cgi-bin/department/list');
|
|
29
|
+
const depts = deptRes.department || [];
|
|
30
|
+
const md = [
|
|
31
|
+
'# WeCom Organization',
|
|
32
|
+
'',
|
|
33
|
+
'## Departments',
|
|
34
|
+
'',
|
|
35
|
+
...depts.map((d) => `- ${d.name} (ID: ${d.id}, Parent: ${d.parentid})`),
|
|
36
|
+
].join('\n');
|
|
37
|
+
writeFileSync(join(targetDir, 'organization.md'), md);
|
|
38
|
+
result.added++;
|
|
39
|
+
// Get members per department (top-level only to avoid rate limits)
|
|
40
|
+
for (const dept of depts.slice(0, 5)) {
|
|
41
|
+
try {
|
|
42
|
+
const memberRes = await this.api(`/cgi-bin/user/simplelist?department_id=${dept.id}&fetch_child=0`);
|
|
43
|
+
const users = memberRes.userlist || [];
|
|
44
|
+
if (users.length === 0)
|
|
45
|
+
continue;
|
|
46
|
+
const memberMd = [
|
|
47
|
+
`# ${dept.name}`,
|
|
48
|
+
'',
|
|
49
|
+
...users.map((u) => `- ${u.name} (${u.userid})`),
|
|
50
|
+
].join('\n');
|
|
51
|
+
const safeName = (dept.name || dept.id).toString().replace(/[^a-zA-Z0-9\u4e00-\u9fff_-]/g, '_');
|
|
52
|
+
writeFileSync(join(targetDir, `dept_${safeName}.md`), memberMd);
|
|
53
|
+
result.added++;
|
|
54
|
+
}
|
|
55
|
+
catch { }
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
catch (err) {
|
|
59
|
+
result.errors.push(err.message);
|
|
60
|
+
}
|
|
61
|
+
return result;
|
|
62
|
+
}
|
|
63
|
+
async disconnect() {
|
|
64
|
+
this.token = '';
|
|
65
|
+
}
|
|
66
|
+
async api(path) {
|
|
67
|
+
const sep = path.includes('?') ? '&' : '?';
|
|
68
|
+
const res = await fetch(`https://qyapi.weixin.qq.com${path}${sep}access_token=${this.token}`);
|
|
69
|
+
if (!res.ok)
|
|
70
|
+
throw new Error(`WeCom API ${res.status}`);
|
|
71
|
+
return res.json();
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
//# sourceMappingURL=wecom.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"wecom.js","sourceRoot":"","sources":["../../../src/data/connectors/wecom.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
export declare class KeyStore {
|
|
2
|
+
constructor();
|
|
3
|
+
save(type: string, credentials: any): Promise<void>;
|
|
4
|
+
load(type: string): Promise<any>;
|
|
5
|
+
loadFromPath(credPath: string): Promise<any>;
|
|
6
|
+
has(type: string): boolean;
|
|
7
|
+
list(): string[];
|
|
8
|
+
delete(type: string): Promise<boolean>;
|
|
9
|
+
/**
|
|
10
|
+
* Build env vars from all stored credentials for openclaw injection.
|
|
11
|
+
*/
|
|
12
|
+
buildEnvVars(dataSources: Record<string, {
|
|
13
|
+
enabled: boolean;
|
|
14
|
+
credentials: string;
|
|
15
|
+
}>): Promise<Record<string, string>>;
|
|
16
|
+
}
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
import { readFileSync, writeFileSync, existsSync, mkdirSync, readdirSync } from 'node:fs';
|
|
2
|
+
import { homedir } from 'node:os';
|
|
3
|
+
import { join } from 'node:path';
|
|
4
|
+
import { createCipheriv, createDecipheriv, randomBytes, scryptSync } from 'node:crypto';
|
|
5
|
+
const CREDENTIALS_DIR = join(homedir(), '.openme', 'credentials');
|
|
6
|
+
const ALGO = 'aes-256-gcm';
|
|
7
|
+
function deriveKey() {
|
|
8
|
+
// Use machine-specific seed; in production, use a proper master key
|
|
9
|
+
const seed = `openme-${homedir()}-${process.env.USER || 'default'}`;
|
|
10
|
+
return scryptSync(seed, 'openme-salt', 32);
|
|
11
|
+
}
|
|
12
|
+
export class KeyStore {
|
|
13
|
+
constructor() {
|
|
14
|
+
mkdirSync(CREDENTIALS_DIR, { recursive: true });
|
|
15
|
+
}
|
|
16
|
+
async save(type, credentials) {
|
|
17
|
+
const key = deriveKey();
|
|
18
|
+
const iv = randomBytes(16);
|
|
19
|
+
const cipher = createCipheriv(ALGO, key, iv);
|
|
20
|
+
const plain = JSON.stringify(credentials);
|
|
21
|
+
const encrypted = Buffer.concat([cipher.update(plain, 'utf-8'), cipher.final()]);
|
|
22
|
+
const tag = cipher.getAuthTag();
|
|
23
|
+
const payload = JSON.stringify({
|
|
24
|
+
iv: iv.toString('hex'),
|
|
25
|
+
tag: tag.toString('hex'),
|
|
26
|
+
data: encrypted.toString('hex'),
|
|
27
|
+
});
|
|
28
|
+
writeFileSync(join(CREDENTIALS_DIR, `${type}.json`), payload);
|
|
29
|
+
}
|
|
30
|
+
async load(type) {
|
|
31
|
+
const filePath = join(CREDENTIALS_DIR, `${type}.json`);
|
|
32
|
+
if (!existsSync(filePath))
|
|
33
|
+
throw new Error(`No credentials for ${type}`);
|
|
34
|
+
const raw = JSON.parse(readFileSync(filePath, 'utf-8'));
|
|
35
|
+
const key = deriveKey();
|
|
36
|
+
const decipher = createDecipheriv(ALGO, key, Buffer.from(raw.iv, 'hex'));
|
|
37
|
+
decipher.setAuthTag(Buffer.from(raw.tag, 'hex'));
|
|
38
|
+
const decrypted = Buffer.concat([decipher.update(Buffer.from(raw.data, 'hex')), decipher.final()]);
|
|
39
|
+
return JSON.parse(decrypted.toString('utf-8'));
|
|
40
|
+
}
|
|
41
|
+
async loadFromPath(credPath) {
|
|
42
|
+
const resolved = credPath.replace(/^~/, homedir());
|
|
43
|
+
if (!existsSync(resolved))
|
|
44
|
+
throw new Error(`Credentials file not found: ${resolved}`);
|
|
45
|
+
const raw = JSON.parse(readFileSync(resolved, 'utf-8'));
|
|
46
|
+
// If encrypted format
|
|
47
|
+
if (raw.iv && raw.tag && raw.data) {
|
|
48
|
+
const key = deriveKey();
|
|
49
|
+
const decipher = createDecipheriv(ALGO, key, Buffer.from(raw.iv, 'hex'));
|
|
50
|
+
decipher.setAuthTag(Buffer.from(raw.tag, 'hex'));
|
|
51
|
+
const decrypted = Buffer.concat([decipher.update(Buffer.from(raw.data, 'hex')), decipher.final()]);
|
|
52
|
+
return JSON.parse(decrypted.toString('utf-8'));
|
|
53
|
+
}
|
|
54
|
+
// Plain JSON (for dev/migration)
|
|
55
|
+
return raw;
|
|
56
|
+
}
|
|
57
|
+
has(type) {
|
|
58
|
+
return existsSync(join(CREDENTIALS_DIR, `${type}.json`));
|
|
59
|
+
}
|
|
60
|
+
list() {
|
|
61
|
+
try {
|
|
62
|
+
return readdirSync(CREDENTIALS_DIR)
|
|
63
|
+
.filter((f) => f.endsWith('.json'))
|
|
64
|
+
.map((f) => f.replace('.json', ''));
|
|
65
|
+
}
|
|
66
|
+
catch {
|
|
67
|
+
return [];
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
async delete(type) {
|
|
71
|
+
const filePath = join(CREDENTIALS_DIR, `${type}.json`);
|
|
72
|
+
if (!existsSync(filePath))
|
|
73
|
+
return false;
|
|
74
|
+
const { unlinkSync } = await import('node:fs');
|
|
75
|
+
unlinkSync(filePath);
|
|
76
|
+
return true;
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Build env vars from all stored credentials for openclaw injection.
|
|
80
|
+
*/
|
|
81
|
+
async buildEnvVars(dataSources) {
|
|
82
|
+
const env = {};
|
|
83
|
+
const envMap = {
|
|
84
|
+
github: (c) => ({ GITHUB_TOKEN: c.token }),
|
|
85
|
+
notion: (c) => ({ NOTION_TOKEN: c.token }),
|
|
86
|
+
email: (c) => ({ IMAP_HOST: c.host, IMAP_USER: c.user, IMAP_PASS: c.pass }),
|
|
87
|
+
feishu: (c) => ({ FEISHU_APP_ID: c.appId, FEISHU_APP_SECRET: c.appSecret }),
|
|
88
|
+
dingtalk: (c) => ({ DINGTALK_APP_KEY: c.appKey, DINGTALK_APP_SECRET: c.appSecret }),
|
|
89
|
+
wecom: (c) => ({ WECOM_CORP_ID: c.corpId, WECOM_CORP_SECRET: c.corpSecret }),
|
|
90
|
+
baidupan: (c) => ({ BAIDUPAN_ACCESS_TOKEN: c.accessToken }),
|
|
91
|
+
};
|
|
92
|
+
for (const [type, ds] of Object.entries(dataSources)) {
|
|
93
|
+
if (!ds.enabled)
|
|
94
|
+
continue;
|
|
95
|
+
try {
|
|
96
|
+
const creds = await this.loadFromPath(ds.credentials);
|
|
97
|
+
const mapper = envMap[type];
|
|
98
|
+
if (mapper)
|
|
99
|
+
Object.assign(env, mapper(creds));
|
|
100
|
+
}
|
|
101
|
+
catch { }
|
|
102
|
+
}
|
|
103
|
+
return env;
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
//# sourceMappingURL=store.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"store.js","sourceRoot":"","sources":["../../../src/data/keys/store.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export interface UserProfile {
|
|
2
|
+
skills: string[];
|
|
3
|
+
interests: string[];
|
|
4
|
+
recentTopics: string[];
|
|
5
|
+
}
|
|
6
|
+
export declare class ProfileBuilder {
|
|
7
|
+
build(workspaceDir: string): Promise<UserProfile>;
|
|
8
|
+
injectToWorkspace(profile: UserProfile, workspaceDir: string): Promise<void>;
|
|
9
|
+
private extractFromDir;
|
|
10
|
+
}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { readdirSync, readFileSync, writeFileSync, existsSync } from 'node:fs';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
export class ProfileBuilder {
|
|
4
|
+
async build(workspaceDir) {
|
|
5
|
+
const skills = this.extractFromDir(join(workspaceDir, 'data/code'), /Language:\s*(.+)/);
|
|
6
|
+
const interests = this.extractFromDir(join(workspaceDir, 'data/notes'), /^#\s+(.+)/m);
|
|
7
|
+
const recentTopics = this.extractFromDir(join(workspaceDir, 'data/email'), /^#\s+(.+)/m);
|
|
8
|
+
return {
|
|
9
|
+
skills: [...new Set(skills)].slice(0, 10),
|
|
10
|
+
interests: [...new Set(interests)].slice(0, 10),
|
|
11
|
+
recentTopics: [...new Set(recentTopics)].slice(0, 10),
|
|
12
|
+
};
|
|
13
|
+
}
|
|
14
|
+
async injectToWorkspace(profile, workspaceDir) {
|
|
15
|
+
const md = [
|
|
16
|
+
'# 用户画像',
|
|
17
|
+
'',
|
|
18
|
+
'## 技术技能',
|
|
19
|
+
...profile.skills.map((s) => `- ${s}`),
|
|
20
|
+
'',
|
|
21
|
+
'## 兴趣领域',
|
|
22
|
+
...profile.interests.map((s) => `- ${s}`),
|
|
23
|
+
'',
|
|
24
|
+
'## 近期关注',
|
|
25
|
+
...profile.recentTopics.map((s) => `- ${s}`),
|
|
26
|
+
'',
|
|
27
|
+
`> 自动生成于 ${new Date().toISOString()}`,
|
|
28
|
+
].join('\n');
|
|
29
|
+
writeFileSync(join(workspaceDir, 'USER.md'), md);
|
|
30
|
+
console.log('[openme] Profile injected → USER.md');
|
|
31
|
+
}
|
|
32
|
+
extractFromDir(dir, pattern) {
|
|
33
|
+
if (!existsSync(dir))
|
|
34
|
+
return [];
|
|
35
|
+
const results = [];
|
|
36
|
+
try {
|
|
37
|
+
for (const file of readdirSync(dir).filter((f) => f.endsWith('.md')).slice(0, 50)) {
|
|
38
|
+
const content = readFileSync(join(dir, file), 'utf-8');
|
|
39
|
+
const match = content.match(pattern);
|
|
40
|
+
if (match?.[1])
|
|
41
|
+
results.push(match[1].trim());
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
catch { }
|
|
45
|
+
return results;
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
//# sourceMappingURL=builder.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"builder.js","sourceRoot":"","sources":["../../../src/data/profile/builder.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,YAAY,EAAE,aAAa,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AAC/E,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAQjC,MAAM,OAAO,cAAc;IACzB,KAAK,CAAC,KAAK,CAAC,YAAoB;QAC9B,MAAM,MAAM,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,YAAY,EAAE,WAAW,CAAC,EAAE,kBAAkB,CAAC,CAAC;QACxF,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,YAAY,EAAE,YAAY,CAAC,EAAE,YAAY,CAAC,CAAC;QACtF,MAAM,YAAY,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,YAAY,EAAE,YAAY,CAAC,EAAE,YAAY,CAAC,CAAC;QAEzF,OAAO;YACL,MAAM,EAAE,CAAC,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC;YACzC,SAAS,EAAE,CAAC,GAAG,IAAI,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC;YAC/C,YAAY,EAAE,CAAC,GAAG,IAAI,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC;SACtD,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,iBAAiB,CAAC,OAAoB,EAAE,YAAoB;QAChE,MAAM,EAAE,GAAG;YACT,QAAQ;YACR,EAAE;YACF,SAAS;YACT,GAAG,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC;YACtC,EAAE;YACF,SAAS;YACT,GAAG,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC;YACzC,EAAE;YACF,SAAS;YACT,GAAG,OAAO,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC;YAC5C,EAAE;YACF,WAAW,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE;SACtC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAEb,aAAa,CAAC,IAAI,CAAC,YAAY,EAAE,SAAS,CAAC,EAAE,EAAE,CAAC,CAAC;QACjD,OAAO,CAAC,GAAG,CAAC,qCAAqC,CAAC,CAAC;IACrD,CAAC;IAEO,cAAc,CAAC,GAAW,EAAE,OAAe;QACjD,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC;YAAE,OAAO,EAAE,CAAC;QAChC,MAAM,OAAO,GAAa,EAAE,CAAC;QAC7B,IAAI,CAAC;YACH,KAAK,MAAM,IAAI,IAAI,WAAW,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,EAAE,CAAC;gBAClF,MAAM,OAAO,GAAG,YAAY,CAAC,IAAI,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,OAAO,CAAC,CAAC;gBACvD,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;gBACrC,IAAI,KAAK,EAAE,CAAC,CAAC,CAAC;oBAAE,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;YAChD,CAAC;QACH,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;QACV,OAAO,OAAO,CAAC;IACjB,CAAC;CACF"}
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
import { mkdirSync, writeFileSync } from 'node:fs';
|
|
2
|
+
import { join } from 'node:path';
|
|
3
|
+
const TEMPLATES = {
|
|
4
|
+
github: [
|
|
5
|
+
{
|
|
6
|
+
name: 'github-push',
|
|
7
|
+
description: 'Push files to a GitHub repository',
|
|
8
|
+
skillMd: `# github-push\n\nPush files from workspace to a GitHub repository.\n\n## Parameters\n- repo: owner/repo\n- path: file path in repo\n- content: file content\n- message: commit message\n`,
|
|
9
|
+
runJs: `#!/usr/bin/env node
|
|
10
|
+
const { GITHUB_TOKEN } = process.env;
|
|
11
|
+
const args = JSON.parse(process.argv[2] || '{}');
|
|
12
|
+
const { repo, path, content, message } = args;
|
|
13
|
+
const body = JSON.stringify({ message, content: Buffer.from(content).toString('base64') });
|
|
14
|
+
fetch(\`https://api.github.com/repos/\${repo}/contents/\${path}\`, {
|
|
15
|
+
method: 'PUT', headers: { Authorization: \`Bearer \${GITHUB_TOKEN}\`, 'Content-Type': 'application/json' }, body
|
|
16
|
+
}).then(r => r.json()).then(d => console.log(JSON.stringify({ ok: true, url: d.content?.html_url })))
|
|
17
|
+
.catch(e => console.error(JSON.stringify({ ok: false, error: e.message })));
|
|
18
|
+
`,
|
|
19
|
+
},
|
|
20
|
+
],
|
|
21
|
+
notion: [
|
|
22
|
+
{
|
|
23
|
+
name: 'notion-save',
|
|
24
|
+
description: 'Create a page in Notion',
|
|
25
|
+
skillMd: `# notion-save\n\nCreate a new page in Notion.\n\n## Parameters\n- title: page title\n- content: page content (markdown)\n- parentId: parent page/database ID (optional)\n`,
|
|
26
|
+
runJs: `#!/usr/bin/env node
|
|
27
|
+
const { NOTION_TOKEN } = process.env;
|
|
28
|
+
const args = JSON.parse(process.argv[2] || '{}');
|
|
29
|
+
const { title, content, parentId } = args;
|
|
30
|
+
const body = JSON.stringify({
|
|
31
|
+
parent: parentId ? { page_id: parentId } : { type: 'workspace', workspace: true },
|
|
32
|
+
properties: { title: { title: [{ text: { content: title } }] } },
|
|
33
|
+
children: [{ object: 'block', type: 'paragraph', paragraph: { rich_text: [{ text: { content } }] } }]
|
|
34
|
+
});
|
|
35
|
+
fetch('https://api.notion.com/v1/pages', {
|
|
36
|
+
method: 'POST', headers: { Authorization: \`Bearer \${NOTION_TOKEN}\`, 'Notion-Version': '2022-06-28', 'Content-Type': 'application/json' }, body
|
|
37
|
+
}).then(r => r.json()).then(d => console.log(JSON.stringify({ ok: true, url: d.url })))
|
|
38
|
+
.catch(e => console.error(JSON.stringify({ ok: false, error: e.message })));
|
|
39
|
+
`,
|
|
40
|
+
},
|
|
41
|
+
],
|
|
42
|
+
email: [
|
|
43
|
+
{
|
|
44
|
+
name: 'email-send',
|
|
45
|
+
description: 'Send an email',
|
|
46
|
+
skillMd: `# email-send\n\nSend an email via SMTP.\n\n## Parameters\n- to: recipient email\n- subject: email subject\n- body: email body\n`,
|
|
47
|
+
runJs: `#!/usr/bin/env node
|
|
48
|
+
// Requires nodemailer or similar — stub implementation
|
|
49
|
+
const args = JSON.parse(process.argv[2] || '{}');
|
|
50
|
+
console.log(JSON.stringify({ ok: false, error: 'email-send requires nodemailer — not yet implemented', args }));
|
|
51
|
+
`,
|
|
52
|
+
},
|
|
53
|
+
],
|
|
54
|
+
};
|
|
55
|
+
export class SkillGenerator {
|
|
56
|
+
generate(enabledSources, workspaceDir) {
|
|
57
|
+
const skillsDir = join(workspaceDir, 'skills');
|
|
58
|
+
for (const type of enabledSources) {
|
|
59
|
+
const templates = TEMPLATES[type];
|
|
60
|
+
if (!templates)
|
|
61
|
+
continue;
|
|
62
|
+
for (const tpl of templates) {
|
|
63
|
+
const dir = join(skillsDir, tpl.name);
|
|
64
|
+
mkdirSync(dir, { recursive: true });
|
|
65
|
+
writeFileSync(join(dir, 'SKILL.md'), tpl.skillMd);
|
|
66
|
+
writeFileSync(join(dir, 'run.js'), tpl.runJs);
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
console.log(`[openme] Generated skills for: ${enabledSources.join(', ')}`);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
//# sourceMappingURL=generator.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"generator.js","sourceRoot":"","sources":["../../../src/data/skills/generator.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AACnD,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AASjC,MAAM,SAAS,GAAoC;IACjD,MAAM,EAAE;QACN;YACE,IAAI,EAAE,aAAa;YACnB,WAAW,EAAE,mCAAmC;YAChD,OAAO,EAAE,0LAA0L;YACnM,KAAK,EAAE;;;;;;;;;CASZ;SACI;KACF;IACD,MAAM,EAAE;QACN;YACE,IAAI,EAAE,aAAa;YACnB,WAAW,EAAE,yBAAyB;YACtC,OAAO,EAAE,2KAA2K;YACpL,KAAK,EAAE;;;;;;;;;;;;;CAaZ;SACI;KACF;IACD,KAAK,EAAE;QACL;YACE,IAAI,EAAE,YAAY;YAClB,WAAW,EAAE,eAAe;YAC5B,OAAO,EAAE,iIAAiI;YAC1I,KAAK,EAAE;;;;CAIZ;SACI;KACF;CACF,CAAC;AAEF,MAAM,OAAO,cAAc;IACzB,QAAQ,CAAC,cAAwB,EAAE,YAAoB;QACrD,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,EAAE,QAAQ,CAAC,CAAC;QAE/C,KAAK,MAAM,IAAI,IAAI,cAAc,EAAE,CAAC;YAClC,MAAM,SAAS,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;YAClC,IAAI,CAAC,SAAS;gBAAE,SAAS;YAEzB,KAAK,MAAM,GAAG,IAAI,SAAS,EAAE,CAAC;gBAC5B,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,EAAE,GAAG,CAAC,IAAI,CAAC,CAAC;gBACtC,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;gBACpC,aAAa,CAAC,IAAI,CAAC,GAAG,EAAE,UAAU,CAAC,EAAE,GAAG,CAAC,OAAO,CAAC,CAAC;gBAClD,aAAa,CAAC,IAAI,CAAC,GAAG,EAAE,QAAQ,CAAC,EAAE,GAAG,CAAC,KAAK,CAAC,CAAC;YAChD,CAAC;QACH,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,kCAAkC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAC7E,CAAC;CACF"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
interface DataSourceEntry {
|
|
2
|
+
enabled: boolean;
|
|
3
|
+
credentials: string;
|
|
4
|
+
syncInterval?: string;
|
|
5
|
+
syncTo: string;
|
|
6
|
+
}
|
|
7
|
+
export declare class SyncScheduler {
|
|
8
|
+
private timers;
|
|
9
|
+
start(dataSources: Record<string, DataSourceEntry>, workspaceDir: string, loadCredentials: (path: string) => Promise<any>): Promise<void>;
|
|
10
|
+
stop(): void;
|
|
11
|
+
}
|
|
12
|
+
export {};
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { join } from 'node:path';
|
|
2
|
+
import { createConnector } from '../connectors/index.js';
|
|
3
|
+
function parseInterval(s) {
|
|
4
|
+
if (!s)
|
|
5
|
+
return 3600_000;
|
|
6
|
+
const m = s.match(/^(\d+)(m|h|d)$/);
|
|
7
|
+
if (!m)
|
|
8
|
+
return 3600_000;
|
|
9
|
+
const n = parseInt(m[1]);
|
|
10
|
+
const unit = m[2];
|
|
11
|
+
if (unit === 'm')
|
|
12
|
+
return n * 60_000;
|
|
13
|
+
if (unit === 'h')
|
|
14
|
+
return n * 3600_000;
|
|
15
|
+
return n * 86400_000;
|
|
16
|
+
}
|
|
17
|
+
export class SyncScheduler {
|
|
18
|
+
timers = [];
|
|
19
|
+
async start(dataSources, workspaceDir, loadCredentials) {
|
|
20
|
+
for (const [type, ds] of Object.entries(dataSources)) {
|
|
21
|
+
if (!ds.enabled)
|
|
22
|
+
continue;
|
|
23
|
+
const connector = createConnector(type);
|
|
24
|
+
if (!connector) {
|
|
25
|
+
console.log(`[openme] No connector for ${type}, skipping`);
|
|
26
|
+
continue;
|
|
27
|
+
}
|
|
28
|
+
const targetDir = join(workspaceDir, ds.syncTo);
|
|
29
|
+
const syncOne = async () => {
|
|
30
|
+
try {
|
|
31
|
+
const creds = await loadCredentials(ds.credentials);
|
|
32
|
+
await connector.connect(creds);
|
|
33
|
+
const result = await connector.sync(targetDir);
|
|
34
|
+
console.log(`[openme] Synced ${type}: +${result.added} ~${result.updated} -${result.deleted}${result.errors.length ? ` (${result.errors.length} errors)` : ''}`);
|
|
35
|
+
await connector.disconnect();
|
|
36
|
+
}
|
|
37
|
+
catch (err) {
|
|
38
|
+
console.error(`[openme] Sync error for ${type}: ${err.message}`);
|
|
39
|
+
}
|
|
40
|
+
};
|
|
41
|
+
// Sync immediately, then on interval
|
|
42
|
+
await syncOne();
|
|
43
|
+
this.timers.push(setInterval(syncOne, parseInterval(ds.syncInterval)));
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
stop() {
|
|
47
|
+
this.timers.forEach(clearInterval);
|
|
48
|
+
this.timers = [];
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
//# sourceMappingURL=scheduler.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"scheduler.js","sourceRoot":"","sources":["../../../src/data/sync/scheduler.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AASzD,SAAS,aAAa,CAAC,CAAU;IAC/B,IAAI,CAAC,CAAC;QAAE,OAAO,QAAQ,CAAC;IACxB,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC;IACpC,IAAI,CAAC,CAAC;QAAE,OAAO,QAAQ,CAAC;IACxB,MAAM,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACzB,MAAM,IAAI,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;IAClB,IAAI,IAAI,KAAK,GAAG;QAAE,OAAO,CAAC,GAAG,MAAM,CAAC;IACpC,IAAI,IAAI,KAAK,GAAG;QAAE,OAAO,CAAC,GAAG,QAAQ,CAAC;IACtC,OAAO,CAAC,GAAG,SAAS,CAAC;AACvB,CAAC;AAED,MAAM,OAAO,aAAa;IAChB,MAAM,GAAqB,EAAE,CAAC;IAEtC,KAAK,CAAC,KAAK,CAAC,WAA4C,EAAE,YAAoB,EAAE,eAA+C;QAC7H,KAAK,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC,EAAE,CAAC;YACrD,IAAI,CAAC,EAAE,CAAC,OAAO;gBAAE,SAAS;YAE1B,MAAM,SAAS,GAAG,eAAe,CAAC,IAAI,CAAC,CAAC;YACxC,IAAI,CAAC,SAAS,EAAE,CAAC;gBACf,OAAO,CAAC,GAAG,CAAC,6BAA6B,IAAI,YAAY,CAAC,CAAC;gBAC3D,SAAS;YACX,CAAC;YAED,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC,MAAM,CAAC,CAAC;YAChD,MAAM,OAAO,GAAG,KAAK,IAAI,EAAE;gBACzB,IAAI,CAAC;oBACH,MAAM,KAAK,GAAG,MAAM,eAAe,CAAC,EAAE,CAAC,WAAW,CAAC,CAAC;oBACpD,MAAM,SAAS,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;oBAC/B,MAAM,MAAM,GAAG,MAAM,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;oBAC/C,OAAO,CAAC,GAAG,CAAC,mBAAmB,IAAI,MAAM,MAAM,CAAC,KAAK,KAAK,MAAM,CAAC,OAAO,KAAK,MAAM,CAAC,OAAO,GAAG,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,MAAM,CAAC,MAAM,CAAC,MAAM,UAAU,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;oBACjK,MAAM,SAAS,CAAC,UAAU,EAAE,CAAC;gBAC/B,CAAC;gBAAC,OAAO,GAAQ,EAAE,CAAC;oBAClB,OAAO,CAAC,KAAK,CAAC,2BAA2B,IAAI,KAAK,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC;gBACnE,CAAC;YACH,CAAC,CAAC;YAEF,qCAAqC;YACrC,MAAM,OAAO,EAAE,CAAC;YAChB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,aAAa,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;QACzE,CAAC;IACH,CAAC;IAED,IAAI;QACF,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;QACnC,IAAI,CAAC,MAAM,GAAG,EAAE,CAAC;IACnB,CAAC;CACF"}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { execSync } from 'node:child_process';
|
|
2
|
+
export function detectDep(name) {
|
|
3
|
+
try {
|
|
4
|
+
const path = execSync(`which ${name}`, { encoding: 'utf-8', timeout: 5000 }).trim();
|
|
5
|
+
let version;
|
|
6
|
+
try {
|
|
7
|
+
version = execSync(`${name} --version`, { encoding: 'utf-8', timeout: 5000 }).trim().split('\n')[0];
|
|
8
|
+
}
|
|
9
|
+
catch { }
|
|
10
|
+
return { name, installed: true, version, path };
|
|
11
|
+
}
|
|
12
|
+
catch {
|
|
13
|
+
return { name, installed: false };
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
export function detectAllDeps() {
|
|
17
|
+
return ['openclaw', 'ttyd', 'node'].map(detectDep);
|
|
18
|
+
}
|
|
19
|
+
//# sourceMappingURL=detector.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"detector.js","sourceRoot":"","sources":["../../src/deps/detector.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAC;AAS9C,MAAM,UAAU,SAAS,CAAC,IAAY;IACpC,IAAI,CAAC;QACH,MAAM,IAAI,GAAG,QAAQ,CAAC,SAAS,IAAI,EAAE,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;QACpF,IAAI,OAA2B,CAAC;QAChC,IAAI,CAAC;YACH,OAAO,GAAG,QAAQ,CAAC,GAAG,IAAI,YAAY,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;QACtG,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;QACV,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC;IAClD,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC;IACpC,CAAC;AACH,CAAC;AAED,MAAM,UAAU,aAAa;IAC3B,OAAO,CAAC,UAAU,EAAE,MAAM,EAAE,MAAM,CAAC,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;AACrD,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare function installDep(name: string): boolean;
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { execSync } from 'node:child_process';
|
|
2
|
+
import { detectOS, detectPkgManager } from './platform.js';
|
|
3
|
+
import { detectDep } from './detector.js';
|
|
4
|
+
const INSTALL_CMDS = {
|
|
5
|
+
ttyd: {
|
|
6
|
+
brew: 'brew install ttyd',
|
|
7
|
+
apt: 'sudo apt-get install -y ttyd',
|
|
8
|
+
yum: 'sudo yum install -y ttyd',
|
|
9
|
+
},
|
|
10
|
+
openclaw: {
|
|
11
|
+
brew: 'npm i -g openclaw',
|
|
12
|
+
apt: 'npm i -g openclaw',
|
|
13
|
+
yum: 'npm i -g openclaw',
|
|
14
|
+
},
|
|
15
|
+
};
|
|
16
|
+
export function installDep(name) {
|
|
17
|
+
const status = detectDep(name);
|
|
18
|
+
if (status.installed) {
|
|
19
|
+
console.log(`[openme] ${name} already installed (${status.version || 'unknown version'})`);
|
|
20
|
+
return true;
|
|
21
|
+
}
|
|
22
|
+
const pm = detectPkgManager();
|
|
23
|
+
const cmd = INSTALL_CMDS[name]?.[pm];
|
|
24
|
+
if (!cmd) {
|
|
25
|
+
console.error(`[openme] Cannot auto-install ${name} on ${detectOS()} (${pm}). Please install manually.`);
|
|
26
|
+
return false;
|
|
27
|
+
}
|
|
28
|
+
console.log(`[openme] Installing ${name}: ${cmd}`);
|
|
29
|
+
try {
|
|
30
|
+
execSync(cmd, { stdio: 'inherit', timeout: 120_000 });
|
|
31
|
+
return detectDep(name).installed;
|
|
32
|
+
}
|
|
33
|
+
catch (err) {
|
|
34
|
+
console.error(`[openme] Failed to install ${name}: ${err.message}`);
|
|
35
|
+
return false;
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
//# sourceMappingURL=installer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"installer.js","sourceRoot":"","sources":["../../src/deps/installer.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAC;AAC9C,OAAO,EAAE,QAAQ,EAAE,gBAAgB,EAAE,MAAM,eAAe,CAAC;AAC3D,OAAO,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAE1C,MAAM,YAAY,GAA2C;IAC3D,IAAI,EAAE;QACJ,IAAI,EAAE,mBAAmB;QACzB,GAAG,EAAE,8BAA8B;QACnC,GAAG,EAAE,0BAA0B;KAChC;IACD,QAAQ,EAAE;QACR,IAAI,EAAE,mBAAmB;QACzB,GAAG,EAAE,mBAAmB;QACxB,GAAG,EAAE,mBAAmB;KACzB;CACF,CAAC;AAEF,MAAM,UAAU,UAAU,CAAC,IAAY;IACrC,MAAM,MAAM,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;IAC/B,IAAI,MAAM,CAAC,SAAS,EAAE,CAAC;QACrB,OAAO,CAAC,GAAG,CAAC,YAAY,IAAI,uBAAuB,MAAM,CAAC,OAAO,IAAI,iBAAiB,GAAG,CAAC,CAAC;QAC3F,OAAO,IAAI,CAAC;IACd,CAAC;IAED,MAAM,EAAE,GAAG,gBAAgB,EAAE,CAAC;IAC9B,MAAM,GAAG,GAAG,YAAY,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;IACrC,IAAI,CAAC,GAAG,EAAE,CAAC;QACT,OAAO,CAAC,KAAK,CAAC,gCAAgC,IAAI,OAAO,QAAQ,EAAE,KAAK,EAAE,6BAA6B,CAAC,CAAC;QACzG,OAAO,KAAK,CAAC;IACf,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,uBAAuB,IAAI,KAAK,GAAG,EAAE,CAAC,CAAC;IACnD,IAAI,CAAC;QACH,QAAQ,CAAC,GAAG,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC,CAAC;QACtD,OAAO,SAAS,CAAC,IAAI,CAAC,CAAC,SAAS,CAAC;IACnC,CAAC;IAAC,OAAO,GAAQ,EAAE,CAAC;QAClB,OAAO,CAAC,KAAK,CAAC,8BAA8B,IAAI,KAAK,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC;QACpE,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { execSync } from 'node:child_process';
|
|
2
|
+
import { platform } from 'node:os';
|
|
3
|
+
export function detectOS() {
|
|
4
|
+
const p = platform();
|
|
5
|
+
if (p === 'darwin')
|
|
6
|
+
return 'macos';
|
|
7
|
+
if (p === 'win32')
|
|
8
|
+
return 'windows';
|
|
9
|
+
return 'linux';
|
|
10
|
+
}
|
|
11
|
+
export function detectPkgManager() {
|
|
12
|
+
const os = detectOS();
|
|
13
|
+
if (os === 'macos')
|
|
14
|
+
return 'brew';
|
|
15
|
+
for (const pm of ['apt', 'yum']) {
|
|
16
|
+
try {
|
|
17
|
+
execSync(`which ${pm}`, { stdio: 'ignore' });
|
|
18
|
+
return pm;
|
|
19
|
+
}
|
|
20
|
+
catch { }
|
|
21
|
+
}
|
|
22
|
+
return 'unknown';
|
|
23
|
+
}
|
|
24
|
+
//# sourceMappingURL=platform.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"platform.js","sourceRoot":"","sources":["../../src/deps/platform.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAC;AAC9C,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAKnC,MAAM,UAAU,QAAQ;IACtB,MAAM,CAAC,GAAG,QAAQ,EAAE,CAAC;IACrB,IAAI,CAAC,KAAK,QAAQ;QAAE,OAAO,OAAO,CAAC;IACnC,IAAI,CAAC,KAAK,OAAO;QAAE,OAAO,SAAS,CAAC;IACpC,OAAO,OAAO,CAAC;AACjB,CAAC;AAED,MAAM,UAAU,gBAAgB;IAC9B,MAAM,EAAE,GAAG,QAAQ,EAAE,CAAC;IACtB,IAAI,EAAE,KAAK,OAAO;QAAE,OAAO,MAAM,CAAC;IAClC,KAAK,MAAM,EAAE,IAAI,CAAC,KAAK,EAAE,KAAK,CAAU,EAAE,CAAC;QACzC,IAAI,CAAC;YACH,QAAQ,CAAC,SAAS,EAAE,EAAE,EAAE,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC,CAAC;YAC7C,OAAO,EAAE,CAAC;QACZ,CAAC;QAAC,MAAM,CAAC,CAAA,CAAC;IACZ,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC"}
|