@f2a/network 0.1.2

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 (234) hide show
  1. package/.github/workflows/ci.yml +113 -0
  2. package/.github/workflows/publish.yml +60 -0
  3. package/LICENSE +21 -0
  4. package/MONOREPO.md +58 -0
  5. package/README.md +280 -0
  6. package/SKILL.md +137 -0
  7. package/dist/adapters/openclaw.d.ts +103 -0
  8. package/dist/adapters/openclaw.d.ts.map +1 -0
  9. package/dist/adapters/openclaw.js +297 -0
  10. package/dist/adapters/openclaw.js.map +1 -0
  11. package/dist/cli/commands.d.ts +17 -0
  12. package/dist/cli/commands.d.ts.map +1 -0
  13. package/dist/cli/commands.js +107 -0
  14. package/dist/cli/commands.js.map +1 -0
  15. package/dist/cli/index.d.ts +6 -0
  16. package/dist/cli/index.d.ts.map +1 -0
  17. package/dist/cli/index.js +203 -0
  18. package/dist/cli/index.js.map +1 -0
  19. package/dist/core/autonomous-economy.d.ts +136 -0
  20. package/dist/core/autonomous-economy.d.ts.map +1 -0
  21. package/dist/core/autonomous-economy.js +255 -0
  22. package/dist/core/autonomous-economy.js.map +1 -0
  23. package/dist/core/connection-manager.d.ts +80 -0
  24. package/dist/core/connection-manager.d.ts.map +1 -0
  25. package/dist/core/connection-manager.js +235 -0
  26. package/dist/core/connection-manager.js.map +1 -0
  27. package/dist/core/connection-manager.test.d.ts +2 -0
  28. package/dist/core/connection-manager.test.d.ts.map +1 -0
  29. package/dist/core/connection-manager.test.js +52 -0
  30. package/dist/core/connection-manager.test.js.map +1 -0
  31. package/dist/core/e2ee-crypto.d.ts +90 -0
  32. package/dist/core/e2ee-crypto.d.ts.map +1 -0
  33. package/dist/core/e2ee-crypto.js +190 -0
  34. package/dist/core/e2ee-crypto.js.map +1 -0
  35. package/dist/core/f2a.d.ts +126 -0
  36. package/dist/core/f2a.d.ts.map +1 -0
  37. package/dist/core/f2a.js +425 -0
  38. package/dist/core/f2a.js.map +1 -0
  39. package/dist/core/identity.d.ts +47 -0
  40. package/dist/core/identity.d.ts.map +1 -0
  41. package/dist/core/identity.js +130 -0
  42. package/dist/core/identity.js.map +1 -0
  43. package/dist/core/identity.test.d.ts +2 -0
  44. package/dist/core/identity.test.d.ts.map +1 -0
  45. package/dist/core/identity.test.js +43 -0
  46. package/dist/core/identity.test.js.map +1 -0
  47. package/dist/core/p2p-network.d.ts +242 -0
  48. package/dist/core/p2p-network.d.ts.map +1 -0
  49. package/dist/core/p2p-network.js +1182 -0
  50. package/dist/core/p2p-network.js.map +1 -0
  51. package/dist/core/reputation-security.d.ts +168 -0
  52. package/dist/core/reputation-security.d.ts.map +1 -0
  53. package/dist/core/reputation-security.js +369 -0
  54. package/dist/core/reputation-security.js.map +1 -0
  55. package/dist/core/reputation.d.ts +179 -0
  56. package/dist/core/reputation.d.ts.map +1 -0
  57. package/dist/core/reputation.js +472 -0
  58. package/dist/core/reputation.js.map +1 -0
  59. package/dist/core/review-committee.d.ts +130 -0
  60. package/dist/core/review-committee.d.ts.map +1 -0
  61. package/dist/core/review-committee.js +251 -0
  62. package/dist/core/review-committee.js.map +1 -0
  63. package/dist/core/serverless.d.ts +155 -0
  64. package/dist/core/serverless.d.ts.map +1 -0
  65. package/dist/core/serverless.js +615 -0
  66. package/dist/core/serverless.js.map +1 -0
  67. package/dist/core/token-manager.d.ts +42 -0
  68. package/dist/core/token-manager.d.ts.map +1 -0
  69. package/dist/core/token-manager.js +122 -0
  70. package/dist/core/token-manager.js.map +1 -0
  71. package/dist/daemon/control-server.d.ts +55 -0
  72. package/dist/daemon/control-server.d.ts.map +1 -0
  73. package/dist/daemon/control-server.js +262 -0
  74. package/dist/daemon/control-server.js.map +1 -0
  75. package/dist/daemon/index.d.ts +35 -0
  76. package/dist/daemon/index.d.ts.map +1 -0
  77. package/dist/daemon/index.js +69 -0
  78. package/dist/daemon/index.js.map +1 -0
  79. package/dist/daemon/main.d.ts +6 -0
  80. package/dist/daemon/main.d.ts.map +1 -0
  81. package/dist/daemon/main.js +38 -0
  82. package/dist/daemon/main.js.map +1 -0
  83. package/dist/daemon/start.d.ts +6 -0
  84. package/dist/daemon/start.d.ts.map +1 -0
  85. package/dist/daemon/start.js +25 -0
  86. package/dist/daemon/start.js.map +1 -0
  87. package/dist/daemon/webhook.d.ts +30 -0
  88. package/dist/daemon/webhook.d.ts.map +1 -0
  89. package/dist/daemon/webhook.js +86 -0
  90. package/dist/daemon/webhook.js.map +1 -0
  91. package/dist/daemon/webhook.test.d.ts +2 -0
  92. package/dist/daemon/webhook.test.d.ts.map +1 -0
  93. package/dist/daemon/webhook.test.js +24 -0
  94. package/dist/daemon/webhook.test.js.map +1 -0
  95. package/dist/index.d.ts +24 -0
  96. package/dist/index.d.ts.map +1 -0
  97. package/dist/index.js +25 -0
  98. package/dist/index.js.map +1 -0
  99. package/dist/protocol/messages.d.ts +739 -0
  100. package/dist/protocol/messages.d.ts.map +1 -0
  101. package/dist/protocol/messages.js +188 -0
  102. package/dist/protocol/messages.js.map +1 -0
  103. package/dist/protocol/messages.test.d.ts +2 -0
  104. package/dist/protocol/messages.test.d.ts.map +1 -0
  105. package/dist/protocol/messages.test.js +55 -0
  106. package/dist/protocol/messages.test.js.map +1 -0
  107. package/dist/types/index.d.ts +247 -0
  108. package/dist/types/index.d.ts.map +1 -0
  109. package/dist/types/index.js +10 -0
  110. package/dist/types/index.js.map +1 -0
  111. package/dist/types/result.d.ts +28 -0
  112. package/dist/types/result.d.ts.map +1 -0
  113. package/dist/types/result.js +16 -0
  114. package/dist/types/result.js.map +1 -0
  115. package/dist/utils/benchmark.d.ts +67 -0
  116. package/dist/utils/benchmark.d.ts.map +1 -0
  117. package/dist/utils/benchmark.js +179 -0
  118. package/dist/utils/benchmark.js.map +1 -0
  119. package/dist/utils/logger.d.ts +105 -0
  120. package/dist/utils/logger.d.ts.map +1 -0
  121. package/dist/utils/logger.js +275 -0
  122. package/dist/utils/logger.js.map +1 -0
  123. package/dist/utils/middleware.d.ts +85 -0
  124. package/dist/utils/middleware.d.ts.map +1 -0
  125. package/dist/utils/middleware.js +173 -0
  126. package/dist/utils/middleware.js.map +1 -0
  127. package/dist/utils/rate-limiter.d.ts +71 -0
  128. package/dist/utils/rate-limiter.d.ts.map +1 -0
  129. package/dist/utils/rate-limiter.js +160 -0
  130. package/dist/utils/rate-limiter.js.map +1 -0
  131. package/dist/utils/signature.d.ts +57 -0
  132. package/dist/utils/signature.d.ts.map +1 -0
  133. package/dist/utils/signature.js +102 -0
  134. package/dist/utils/signature.js.map +1 -0
  135. package/dist/utils/validation.d.ts +504 -0
  136. package/dist/utils/validation.d.ts.map +1 -0
  137. package/dist/utils/validation.js +159 -0
  138. package/dist/utils/validation.js.map +1 -0
  139. package/docs/F2A-PROTOCOL.md +61 -0
  140. package/docs/MOBILE_BOOTSTRAP_DESIGN.md +126 -0
  141. package/docs/a2a-lessons.md +316 -0
  142. package/docs/middleware-guide.md +448 -0
  143. package/docs/readme-update-checklist.md +90 -0
  144. package/docs/reputation-guide.md +396 -0
  145. package/docs/rfcs/001-reputation-system.md +712 -0
  146. package/docs/security-design.md +247 -0
  147. package/install.sh +231 -0
  148. package/package.json +64 -0
  149. package/packages/openclaw-adapter/README.md +510 -0
  150. package/packages/openclaw-adapter/openclaw.plugin.json +106 -0
  151. package/packages/openclaw-adapter/package.json +40 -0
  152. package/packages/openclaw-adapter/src/announcement-queue.test.ts +449 -0
  153. package/packages/openclaw-adapter/src/announcement-queue.ts +403 -0
  154. package/packages/openclaw-adapter/src/capability-detector.test.ts +99 -0
  155. package/packages/openclaw-adapter/src/capability-detector.ts +183 -0
  156. package/packages/openclaw-adapter/src/claim-handlers.test.ts +974 -0
  157. package/packages/openclaw-adapter/src/claim-handlers.ts +482 -0
  158. package/packages/openclaw-adapter/src/connector.business.test.ts +583 -0
  159. package/packages/openclaw-adapter/src/connector.ts +795 -0
  160. package/packages/openclaw-adapter/src/index.test.ts +82 -0
  161. package/packages/openclaw-adapter/src/index.ts +18 -0
  162. package/packages/openclaw-adapter/src/integration.e2e.test.ts +829 -0
  163. package/packages/openclaw-adapter/src/logger.ts +51 -0
  164. package/packages/openclaw-adapter/src/network-client.test.ts +266 -0
  165. package/packages/openclaw-adapter/src/network-client.ts +251 -0
  166. package/packages/openclaw-adapter/src/network-recovery.test.ts +465 -0
  167. package/packages/openclaw-adapter/src/node-manager.test.ts +136 -0
  168. package/packages/openclaw-adapter/src/node-manager.ts +429 -0
  169. package/packages/openclaw-adapter/src/plugin.test.ts +439 -0
  170. package/packages/openclaw-adapter/src/plugin.ts +104 -0
  171. package/packages/openclaw-adapter/src/reputation.test.ts +221 -0
  172. package/packages/openclaw-adapter/src/reputation.ts +368 -0
  173. package/packages/openclaw-adapter/src/task-guard.test.ts +502 -0
  174. package/packages/openclaw-adapter/src/task-guard.ts +860 -0
  175. package/packages/openclaw-adapter/src/task-queue.concurrency.test.ts +462 -0
  176. package/packages/openclaw-adapter/src/task-queue.edge-cases.test.ts +284 -0
  177. package/packages/openclaw-adapter/src/task-queue.persistence.test.ts +408 -0
  178. package/packages/openclaw-adapter/src/task-queue.ts +668 -0
  179. package/packages/openclaw-adapter/src/tool-handlers.test.ts +906 -0
  180. package/packages/openclaw-adapter/src/tool-handlers.ts +574 -0
  181. package/packages/openclaw-adapter/src/types.ts +361 -0
  182. package/packages/openclaw-adapter/src/webhook-pusher.test.ts +188 -0
  183. package/packages/openclaw-adapter/src/webhook-pusher.ts +220 -0
  184. package/packages/openclaw-adapter/src/webhook-server.test.ts +580 -0
  185. package/packages/openclaw-adapter/src/webhook-server.ts +202 -0
  186. package/packages/openclaw-adapter/tsconfig.json +20 -0
  187. package/src/cli/commands.test.ts +157 -0
  188. package/src/cli/commands.ts +129 -0
  189. package/src/cli/index.test.ts +77 -0
  190. package/src/cli/index.ts +234 -0
  191. package/src/core/autonomous-economy.test.ts +291 -0
  192. package/src/core/autonomous-economy.ts +428 -0
  193. package/src/core/e2ee-crypto.test.ts +125 -0
  194. package/src/core/e2ee-crypto.ts +246 -0
  195. package/src/core/f2a.test.ts +269 -0
  196. package/src/core/f2a.ts +618 -0
  197. package/src/core/p2p-network.test.ts +199 -0
  198. package/src/core/p2p-network.ts +1432 -0
  199. package/src/core/reputation-security.test.ts +403 -0
  200. package/src/core/reputation-security.ts +562 -0
  201. package/src/core/reputation.test.ts +260 -0
  202. package/src/core/reputation.ts +576 -0
  203. package/src/core/review-committee.test.ts +380 -0
  204. package/src/core/review-committee.ts +401 -0
  205. package/src/core/token-manager.test.ts +133 -0
  206. package/src/core/token-manager.ts +140 -0
  207. package/src/daemon/control-server.test.ts +216 -0
  208. package/src/daemon/control-server.ts +292 -0
  209. package/src/daemon/index.test.ts +85 -0
  210. package/src/daemon/index.ts +89 -0
  211. package/src/daemon/main.ts +44 -0
  212. package/src/daemon/start.ts +29 -0
  213. package/src/daemon/webhook.test.ts +68 -0
  214. package/src/daemon/webhook.ts +105 -0
  215. package/src/index.test.ts +436 -0
  216. package/src/index.ts +72 -0
  217. package/src/types/index.test.ts +87 -0
  218. package/src/types/index.ts +341 -0
  219. package/src/types/result.ts +68 -0
  220. package/src/utils/benchmark.ts +237 -0
  221. package/src/utils/logger.ts +331 -0
  222. package/src/utils/middleware.ts +229 -0
  223. package/src/utils/rate-limiter.ts +207 -0
  224. package/src/utils/signature.ts +136 -0
  225. package/src/utils/validation.ts +186 -0
  226. package/tests/docker/Dockerfile.node +23 -0
  227. package/tests/docker/Dockerfile.runner +18 -0
  228. package/tests/docker/docker-compose.test.yml +73 -0
  229. package/tests/integration/message-passing.test.ts +109 -0
  230. package/tests/integration/multi-node.test.ts +92 -0
  231. package/tests/integration/p2p-connection.test.ts +83 -0
  232. package/tests/integration/test-config.ts +32 -0
  233. package/tsconfig.json +21 -0
  234. package/vitest.config.ts +26 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"token-manager.d.ts","sourceRoot":"","sources":["../../src/core/token-manager.ts"],"names":[],"mappings":"AAMA;;;GAGG;AACH,qBAAa,YAAY;IACvB,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,KAAK,CAAuB;IACpC,OAAO,CAAC,MAAM,CAAS;gBAEX,OAAO,CAAC,EAAE,MAAM;IAa5B;;;OAGG;IACH,QAAQ,IAAI,MAAM;IAqClB;;;OAGG;IACH,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,SAAS,GAAG,OAAO;IAqB/C;;OAEG;IACH,aAAa,CAAC,UAAU,EAAE;QAAE,EAAE,CAAC,EAAE,MAAM,CAAC;QAAC,MAAM,CAAC,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,OAAO,CAAA;KAAE,GAAG,IAAI;IAcnF;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAK3B;;OAEG;IACH,OAAO,CAAC,SAAS;IAQjB;;OAEG;IACH,YAAY,IAAI,MAAM;CAGvB;AAGD,eAAO,MAAM,mBAAmB,cAAqB,CAAC"}
@@ -0,0 +1,122 @@
1
+ import { existsSync, readFileSync, writeFileSync, mkdirSync, appendFileSync } from 'fs';
2
+ import { join, dirname } from 'path';
3
+ import { randomBytes, timingSafeEqual } from 'crypto';
4
+ import { homedir } from 'os';
5
+ import { Logger } from '../utils/logger.js';
6
+ /**
7
+ * Token 管理器
8
+ * 负责生成、存储和验证 F2A 控制 Token
9
+ */
10
+ export class TokenManager {
11
+ tokenPath;
12
+ token = null;
13
+ logger;
14
+ constructor(dataDir) {
15
+ this.logger = new Logger({ component: 'TokenManager' });
16
+ // 默认存储在用户主目录的 .f2a 文件夹
17
+ const baseDir = dataDir || join(homedir(), '.f2a');
18
+ this.tokenPath = join(baseDir, 'control-token');
19
+ // 确保目录存在
20
+ const dir = join(baseDir);
21
+ if (!existsSync(dir)) {
22
+ mkdirSync(dir, { recursive: true });
23
+ }
24
+ }
25
+ /**
26
+ * 获取或生成 Token
27
+ * 优先从环境变量读取,其次从文件读取,最后生成新的
28
+ */
29
+ getToken() {
30
+ // 1. 优先使用环境变量
31
+ const envToken = process.env.F2A_CONTROL_TOKEN;
32
+ if (envToken) {
33
+ // 检查是否为不安全默认值
34
+ if (envToken === 'f2a-default-token') {
35
+ this.logger.error('F2A_CONTROL_TOKEN is using the insecure default value!');
36
+ this.logger.error('Please set a secure token: export F2A_CONTROL_TOKEN=$(openssl rand -hex 32)');
37
+ throw new Error('Insecure token detected. F2A_CONTROL_TOKEN cannot use the default value "f2a-default-token". ' +
38
+ 'Please set a secure token: export F2A_CONTROL_TOKEN=$(openssl rand -hex 32)');
39
+ }
40
+ this.token = envToken;
41
+ return envToken;
42
+ }
43
+ // 2. 从文件读取
44
+ if (existsSync(this.tokenPath)) {
45
+ const fileToken = readFileSync(this.tokenPath, 'utf-8').trim();
46
+ if (fileToken) {
47
+ this.token = fileToken;
48
+ return fileToken;
49
+ }
50
+ }
51
+ // 3. 生成新的随机 Token
52
+ const newToken = this.generateSecureToken();
53
+ this.saveToken(newToken);
54
+ this.token = newToken;
55
+ this.logger.info('Generated new control token', { path: this.tokenPath });
56
+ this.logger.info('To use a custom token, set F2A_CONTROL_TOKEN environment variable');
57
+ return newToken;
58
+ }
59
+ /**
60
+ * 验证 Token 是否有效
61
+ * 使用 timingSafeEqual 防止时序攻击
62
+ */
63
+ verifyToken(token) {
64
+ if (!token)
65
+ return false;
66
+ const expectedToken = this.getToken();
67
+ // 两个 token 长度必须相同
68
+ if (token.length !== expectedToken.length) {
69
+ return false;
70
+ }
71
+ // 使用 timingSafeEqual 防止时序攻击
72
+ try {
73
+ return timingSafeEqual(Buffer.from(token, 'utf-8'), Buffer.from(expectedToken, 'utf-8'));
74
+ }
75
+ catch {
76
+ return false;
77
+ }
78
+ }
79
+ /**
80
+ * 记录 Token 使用审计日志
81
+ */
82
+ logTokenUsage(clientInfo) {
83
+ const auditPath = join(dirname(this.tokenPath), 'token-audit.log');
84
+ const entry = {
85
+ timestamp: new Date().toISOString(),
86
+ ...clientInfo
87
+ };
88
+ try {
89
+ appendFileSync(auditPath, JSON.stringify(entry) + '\n', { mode: 0o600 });
90
+ }
91
+ catch (error) {
92
+ this.logger.error('Failed to write audit log', { error });
93
+ }
94
+ }
95
+ /**
96
+ * 生成安全的随机 Token
97
+ */
98
+ generateSecureToken() {
99
+ // 生成 32 字节 (64 字符) 的十六进制随机字符串
100
+ return 'f2a-' + randomBytes(32).toString('hex');
101
+ }
102
+ /**
103
+ * 保存 Token 到文件
104
+ */
105
+ saveToken(token) {
106
+ try {
107
+ writeFileSync(this.tokenPath, token, { mode: 0o600 }); // 仅所有者可读写
108
+ }
109
+ catch (error) {
110
+ this.logger.error('Failed to save token', { error });
111
+ }
112
+ }
113
+ /**
114
+ * 获取 Token 文件路径
115
+ */
116
+ getTokenPath() {
117
+ return this.tokenPath;
118
+ }
119
+ }
120
+ // 单例导出
121
+ export const defaultTokenManager = new TokenManager();
122
+ //# sourceMappingURL=token-manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"token-manager.js","sourceRoot":"","sources":["../../src/core/token-manager.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,55 @@
1
+ /**
2
+ * HTTP 控制服务器
3
+ * 接收 CLI 命令 - P2P 版本
4
+ */
5
+ import { F2A } from '../core/f2a.js';
6
+ import { TokenManager } from '../core/token-manager.js';
7
+ export interface ControlServerOptions {
8
+ port: number;
9
+ token?: string;
10
+ /** 允许的 CORS 来源列表,默认为 ['http://localhost'] */
11
+ allowedOrigins?: string[];
12
+ }
13
+ export declare class ControlServer {
14
+ private server?;
15
+ private f2a;
16
+ private port;
17
+ private tokenManager;
18
+ private logger;
19
+ private rateLimiter;
20
+ private allowedOrigins;
21
+ constructor(f2a: F2A, port: number, tokenManager?: TokenManager, options?: ControlServerOptions);
22
+ /**
23
+ * 启动控制服务器
24
+ */
25
+ start(): Promise<void>;
26
+ /**
27
+ * 停止控制服务器
28
+ */
29
+ stop(): void;
30
+ /**
31
+ * 从 Authorization header 提取 Bearer token
32
+ */
33
+ private extractBearerToken;
34
+ /**
35
+ * 处理请求
36
+ */
37
+ private handleRequest;
38
+ /**
39
+ * 处理命令
40
+ */
41
+ private processCommand;
42
+ /**
43
+ * 获取状态
44
+ */
45
+ private handleStatus;
46
+ /**
47
+ * 获取已连接的 Peers
48
+ */
49
+ private handlePeers;
50
+ /**
51
+ * 发现 Agents
52
+ */
53
+ private handleDiscover;
54
+ }
55
+ //# sourceMappingURL=control-server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"control-server.d.ts","sourceRoot":"","sources":["../../src/daemon/control-server.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAGH,OAAO,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AACrC,OAAO,EAAE,YAAY,EAAE,MAAM,0BAA0B,CAAC;AAIxD,MAAM,WAAW,oBAAoB;IACnC,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,6CAA6C;IAC7C,cAAc,CAAC,EAAE,MAAM,EAAE,CAAC;CAC3B;AAKD,qBAAa,aAAa;IACxB,OAAO,CAAC,MAAM,CAAC,CAAS;IACxB,OAAO,CAAC,GAAG,CAAM;IACjB,OAAO,CAAC,IAAI,CAAS;IACrB,OAAO,CAAC,YAAY,CAAe;IACnC,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,WAAW,CAAc;IACjC,OAAO,CAAC,cAAc,CAAW;gBAErB,GAAG,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,EAAE,YAAY,CAAC,EAAE,YAAY,EAAE,OAAO,CAAC,EAAE,oBAAoB;IAW/F;;OAEG;IACH,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAetB;;OAEG;IACH,IAAI,IAAI,IAAI;IAUZ;;OAEG;IACH,OAAO,CAAC,kBAAkB;IAM1B;;OAEG;IACH,OAAO,CAAC,aAAa;IAgIrB;;OAEG;IACH,OAAO,CAAC,cAAc;IAgCtB;;OAEG;IACH,OAAO,CAAC,YAAY;IASpB;;OAEG;IACH,OAAO,CAAC,WAAW;IASnB;;OAEG;YACW,cAAc;CAiB7B"}
@@ -0,0 +1,262 @@
1
+ /**
2
+ * HTTP 控制服务器
3
+ * 接收 CLI 命令 - P2P 版本
4
+ */
5
+ import { createServer } from 'http';
6
+ import { TokenManager } from '../core/token-manager.js';
7
+ import { Logger } from '../utils/logger.js';
8
+ import { RateLimiter } from '../utils/rate-limiter.js';
9
+ /** 默认允许的 CORS 来源 */
10
+ const DEFAULT_ALLOWED_ORIGINS = ['http://localhost'];
11
+ export class ControlServer {
12
+ server;
13
+ f2a;
14
+ port;
15
+ tokenManager;
16
+ logger;
17
+ rateLimiter;
18
+ allowedOrigins;
19
+ constructor(f2a, port, tokenManager, options) {
20
+ this.f2a = f2a;
21
+ this.port = port;
22
+ this.tokenManager = tokenManager || new TokenManager();
23
+ this.logger = new Logger({ component: 'ControlServer' });
24
+ // 速率限制: 每分钟最多 60 个请求
25
+ this.rateLimiter = new RateLimiter({ maxRequests: 60, windowMs: 60000 });
26
+ // CORS 配置:优先使用传入的 allowedOrigins,否则使用默认值
27
+ this.allowedOrigins = options?.allowedOrigins ?? DEFAULT_ALLOWED_ORIGINS;
28
+ }
29
+ /**
30
+ * 启动控制服务器
31
+ */
32
+ start() {
33
+ return new Promise((resolve, reject) => {
34
+ this.server = createServer((req, res) => {
35
+ this.handleRequest(req, res);
36
+ });
37
+ this.server.on('error', reject);
38
+ this.server.listen(this.port, () => {
39
+ this.logger.info('Listening', { port: this.port });
40
+ resolve();
41
+ });
42
+ });
43
+ }
44
+ /**
45
+ * 停止控制服务器
46
+ */
47
+ stop() {
48
+ if (this.server) {
49
+ this.server.close();
50
+ this.server = undefined;
51
+ }
52
+ // 清理速率限制器资源
53
+ this.rateLimiter.stop();
54
+ this.logger.info('Stopped');
55
+ }
56
+ /**
57
+ * 从 Authorization header 提取 Bearer token
58
+ */
59
+ extractBearerToken(authHeader) {
60
+ if (!authHeader)
61
+ return undefined;
62
+ const match = authHeader.match(/^Bearer\s+(.+)$/i);
63
+ return match ? match[1] : undefined;
64
+ }
65
+ /**
66
+ * 处理请求
67
+ */
68
+ handleRequest(req, res) {
69
+ // 设置 CORS - 使用配置的允许来源
70
+ const origin = req.headers.origin;
71
+ const allowOrigin = origin && this.allowedOrigins.includes(origin)
72
+ ? origin
73
+ : this.allowedOrigins[0];
74
+ res.setHeader('Access-Control-Allow-Origin', allowOrigin);
75
+ res.setHeader('Access-Control-Allow-Methods', 'GET, POST, OPTIONS');
76
+ res.setHeader('Access-Control-Allow-Headers', 'Content-Type, X-F2A-Token');
77
+ if (req.method === 'OPTIONS') {
78
+ res.writeHead(200);
79
+ res.end();
80
+ return;
81
+ }
82
+ // 健康检查端点 (不需要认证)
83
+ if (req.method === 'GET' && req.url === '/health') {
84
+ res.writeHead(200);
85
+ res.end(JSON.stringify({ status: 'ok', peerId: this.f2a.peerId }));
86
+ return;
87
+ }
88
+ // GET /status - 获取状态 (需要认证)
89
+ if (req.method === 'GET' && req.url === '/status') {
90
+ const clientIp = req.socket.remoteAddress || 'unknown';
91
+ if (!this.rateLimiter.allowRequest(clientIp)) {
92
+ res.writeHead(429);
93
+ res.end(JSON.stringify({ success: false, error: 'Too many requests' }));
94
+ return;
95
+ }
96
+ // 支持 X-F2A-Token 或 Authorization: Bearer xxx
97
+ const token = req.headers['x-f2a-token']
98
+ || this.extractBearerToken(req.headers.authorization);
99
+ if (!this.tokenManager.verifyToken(token)) {
100
+ res.writeHead(401);
101
+ res.end(JSON.stringify({ success: false, error: 'Unauthorized' }));
102
+ return;
103
+ }
104
+ res.writeHead(200);
105
+ res.end(JSON.stringify({
106
+ success: true,
107
+ peerId: this.f2a.peerId,
108
+ multiaddrs: this.f2a.agentInfo.multiaddrs || []
109
+ }));
110
+ return;
111
+ }
112
+ // GET /peers - 获取已知的 Peers (需要认证)
113
+ if (req.method === 'GET' && req.url === '/peers') {
114
+ const clientIp = req.socket.remoteAddress || 'unknown';
115
+ if (!this.rateLimiter.allowRequest(clientIp)) {
116
+ res.writeHead(429);
117
+ res.end(JSON.stringify({ success: false, error: 'Too many requests' }));
118
+ return;
119
+ }
120
+ // 支持 X-F2A-Token 或 Authorization: Bearer xxx
121
+ const token = req.headers['x-f2a-token']
122
+ || this.extractBearerToken(req.headers.authorization);
123
+ if (!this.tokenManager.verifyToken(token)) {
124
+ res.writeHead(401);
125
+ res.end(JSON.stringify({ success: false, error: 'Unauthorized' }));
126
+ return;
127
+ }
128
+ // 返回所有已知的节点(包括已断开但已发现的)
129
+ const peers = this.f2a.getAllPeers();
130
+ res.writeHead(200);
131
+ res.end(JSON.stringify(peers));
132
+ return;
133
+ }
134
+ if (req.method !== 'POST') {
135
+ res.writeHead(405);
136
+ res.end(JSON.stringify({
137
+ success: false,
138
+ error: 'Method not allowed',
139
+ code: 'METHOD_NOT_ALLOWED'
140
+ }));
141
+ return;
142
+ }
143
+ const clientIp = req.socket.remoteAddress || 'unknown';
144
+ if (!this.rateLimiter.allowRequest(clientIp)) {
145
+ this.logger.warn('Rate limit exceeded', { clientIp });
146
+ res.writeHead(429);
147
+ res.end(JSON.stringify({
148
+ success: false,
149
+ error: 'Too many requests',
150
+ code: 'RATE_LIMIT_EXCEEDED'
151
+ }));
152
+ return;
153
+ }
154
+ // 验证 Token
155
+ const token = req.headers['x-f2a-token'];
156
+ if (!this.tokenManager.verifyToken(token)) {
157
+ // 记录失败的验证尝试
158
+ this.tokenManager.logTokenUsage({
159
+ ip: clientIp,
160
+ action: 'auth',
161
+ success: false
162
+ });
163
+ this.logger.warn('Unauthorized request', { clientIp });
164
+ res.writeHead(401);
165
+ res.end(JSON.stringify({
166
+ success: false,
167
+ error: 'Unauthorized: Invalid or missing token',
168
+ code: 'UNAUTHORIZED'
169
+ }));
170
+ return;
171
+ }
172
+ // 记录成功的验证
173
+ this.tokenManager.logTokenUsage({
174
+ ip: clientIp,
175
+ action: 'auth',
176
+ success: true
177
+ });
178
+ let body = '';
179
+ req.on('data', chunk => body += chunk);
180
+ req.on('end', () => {
181
+ this.processCommand(body, res);
182
+ });
183
+ }
184
+ /**
185
+ * 处理命令
186
+ */
187
+ processCommand(body, res) {
188
+ try {
189
+ const command = JSON.parse(body);
190
+ switch (command.action) {
191
+ case 'status':
192
+ this.handleStatus(res);
193
+ break;
194
+ case 'peers':
195
+ this.handlePeers(res);
196
+ break;
197
+ case 'discover':
198
+ this.handleDiscover(command.capability, res);
199
+ break;
200
+ default:
201
+ res.writeHead(400);
202
+ res.end(JSON.stringify({
203
+ success: false,
204
+ error: 'Unknown action',
205
+ code: 'UNKNOWN_ACTION'
206
+ }));
207
+ }
208
+ }
209
+ catch {
210
+ res.writeHead(400);
211
+ res.end(JSON.stringify({
212
+ success: false,
213
+ error: 'Invalid JSON',
214
+ code: 'INVALID_JSON'
215
+ }));
216
+ }
217
+ }
218
+ /**
219
+ * 获取状态
220
+ */
221
+ handleStatus(res) {
222
+ res.writeHead(200);
223
+ res.end(JSON.stringify({
224
+ success: true,
225
+ peerId: this.f2a.peerId,
226
+ agentInfo: this.f2a.agentInfo
227
+ }));
228
+ }
229
+ /**
230
+ * 获取已连接的 Peers
231
+ */
232
+ handlePeers(res) {
233
+ const peers = this.f2a.getConnectedPeers();
234
+ res.writeHead(200);
235
+ res.end(JSON.stringify({
236
+ success: true,
237
+ peers
238
+ }));
239
+ }
240
+ /**
241
+ * 发现 Agents
242
+ */
243
+ async handleDiscover(capability, res) {
244
+ try {
245
+ const agents = await this.f2a.discoverAgents(capability);
246
+ res.writeHead(200);
247
+ res.end(JSON.stringify({
248
+ success: true,
249
+ agents
250
+ }));
251
+ }
252
+ catch (error) {
253
+ res.writeHead(500);
254
+ res.end(JSON.stringify({
255
+ success: false,
256
+ error: error instanceof Error ? error.message : String(error),
257
+ code: 'DISCOVER_FAILED'
258
+ }));
259
+ }
260
+ }
261
+ }
262
+ //# sourceMappingURL=control-server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"control-server.js","sourceRoot":"","sources":["../../src/daemon/control-server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,35 @@
1
+ /**
2
+ * F2A Daemon
3
+ * 后台服务主入口 - P2P 版本
4
+ */
5
+ import { F2A } from '../core/f2a.js';
6
+ import { F2AOptions, WebhookConfig } from '../types/index.js';
7
+ export interface DaemonOptions extends F2AOptions {
8
+ webhook?: WebhookConfig;
9
+ controlPort?: number;
10
+ }
11
+ export declare class F2ADaemon {
12
+ private options;
13
+ private f2a?;
14
+ private controlServer?;
15
+ private running;
16
+ constructor(options?: DaemonOptions);
17
+ /**
18
+ * 启动 Daemon
19
+ */
20
+ start(): Promise<void>;
21
+ /**
22
+ * 停止 Daemon
23
+ */
24
+ stop(): Promise<void>;
25
+ /**
26
+ * 获取 F2A 实例
27
+ */
28
+ getF2A(): F2A | undefined;
29
+ /**
30
+ * 是否运行中
31
+ */
32
+ isRunning(): boolean;
33
+ }
34
+ export default F2ADaemon;
35
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/daemon/index.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AAErC,OAAO,EAAE,UAAU,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAE9D,MAAM,WAAW,aAAc,SAAQ,UAAU;IAC/C,OAAO,CAAC,EAAE,aAAa,CAAC;IACxB,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED,qBAAa,SAAS;IACpB,OAAO,CAAC,OAAO,CAAgB;IAC/B,OAAO,CAAC,GAAG,CAAC,CAAM;IAClB,OAAO,CAAC,aAAa,CAAC,CAAgB;IACtC,OAAO,CAAC,OAAO,CAAkB;gBAErB,OAAO,GAAE,aAAkB;IAOvC;;OAEG;IACG,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IA2B5B;;OAEG;IACG,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAY3B;;OAEG;IACH,MAAM,IAAI,GAAG,GAAG,SAAS;IAIzB;;OAEG;IACH,SAAS,IAAI,OAAO;CAGrB;AAGD,eAAe,SAAS,CAAC"}
@@ -0,0 +1,69 @@
1
+ /**
2
+ * F2A Daemon
3
+ * 后台服务主入口 - P2P 版本
4
+ */
5
+ import { F2A } from '../core/f2a.js';
6
+ import { ControlServer } from './control-server.js';
7
+ export class F2ADaemon {
8
+ options;
9
+ f2a;
10
+ controlServer;
11
+ running = false;
12
+ constructor(options = {}) {
13
+ this.options = {
14
+ controlPort: 9001,
15
+ ...options
16
+ };
17
+ }
18
+ /**
19
+ * 启动 Daemon
20
+ */
21
+ async start() {
22
+ if (this.running) {
23
+ throw new Error('Daemon already running');
24
+ }
25
+ console.log('[Daemon] Starting F2A Daemon...');
26
+ // 创建并启动 F2A
27
+ this.f2a = await F2A.create(this.options);
28
+ const result = await this.f2a.start();
29
+ if (!result.success) {
30
+ const errorData = result.error;
31
+ const errorMsg = typeof errorData === 'string'
32
+ ? errorData
33
+ : JSON.stringify(errorData);
34
+ throw new Error(`Failed to start F2A: ${errorMsg}`);
35
+ }
36
+ // 启动控制服务器
37
+ this.controlServer = new ControlServer(this.f2a, this.options.controlPort);
38
+ await this.controlServer.start();
39
+ this.running = true;
40
+ console.log(`[Daemon] F2A Daemon started with peerId: ${this.f2a.peerId.slice(0, 16)}...`);
41
+ }
42
+ /**
43
+ * 停止 Daemon
44
+ */
45
+ async stop() {
46
+ if (!this.running)
47
+ return;
48
+ console.log('[Daemon] Stopping F2A Daemon...');
49
+ await this.controlServer?.stop();
50
+ await this.f2a?.stop();
51
+ this.running = false;
52
+ console.log('[Daemon] F2A Daemon stopped');
53
+ }
54
+ /**
55
+ * 获取 F2A 实例
56
+ */
57
+ getF2A() {
58
+ return this.f2a;
59
+ }
60
+ /**
61
+ * 是否运行中
62
+ */
63
+ isRunning() {
64
+ return this.running;
65
+ }
66
+ }
67
+ // 默认导出
68
+ export default F2ADaemon;
69
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/daemon/index.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AACrC,OAAO,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAQpD,MAAM,OAAO,SAAS;IACZ,OAAO,CAAgB;IACvB,GAAG,CAAO;IACV,aAAa,CAAiB;IAC9B,OAAO,GAAY,KAAK,CAAC;IAEjC,YAAY,UAAyB,EAAE;QACrC,IAAI,CAAC,OAAO,GAAG;YACb,WAAW,EAAE,IAAI;YACjB,GAAG,OAAO;SACX,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,KAAK;QACT,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,MAAM,IAAI,KAAK,CAAC,wBAAwB,CAAC,CAAC;QAC5C,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,iCAAiC,CAAC,CAAC;QAE/C,YAAY;QACZ,IAAI,CAAC,GAAG,GAAG,MAAM,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC1C,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,CAAC;QAEtC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACpB,MAAM,SAAS,GAAI,MAA6B,CAAC,KAAK,CAAC;YACvD,MAAM,QAAQ,GAAG,OAAO,SAAS,KAAK,QAAQ;gBAC5C,CAAC,CAAC,SAAS;gBACX,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;YAC9B,MAAM,IAAI,KAAK,CAAC,wBAAwB,QAAQ,EAAE,CAAC,CAAC;QACtD,CAAC;QAED,UAAU;QACV,IAAI,CAAC,aAAa,GAAG,IAAI,aAAa,CAAC,IAAI,CAAC,GAAG,EAAE,IAAI,CAAC,OAAO,CAAC,WAAY,CAAC,CAAC;QAC5E,MAAM,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;QAEjC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,OAAO,CAAC,GAAG,CAAC,4CAA4C,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC;IAC7F,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,IAAI;QACR,IAAI,CAAC,IAAI,CAAC,OAAO;YAAE,OAAO;QAE1B,OAAO,CAAC,GAAG,CAAC,iCAAiC,CAAC,CAAC;QAE/C,MAAM,IAAI,CAAC,aAAa,EAAE,IAAI,EAAE,CAAC;QACjC,MAAM,IAAI,CAAC,GAAG,EAAE,IAAI,EAAE,CAAC;QAEvB,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;QACrB,OAAO,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAC;IAC7C,CAAC;IAED;;OAEG;IACH,MAAM;QACJ,OAAO,IAAI,CAAC,GAAG,CAAC;IAClB,CAAC;IAED;;OAEG;IACH,SAAS;QACP,OAAO,IAAI,CAAC,OAAO,CAAC;IACtB,CAAC;CACF;AAED,OAAO;AACP,eAAe,SAAS,CAAC"}
@@ -0,0 +1,6 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * F2A Daemon 入口
4
+ */
5
+ export {};
6
+ //# sourceMappingURL=main.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"main.d.ts","sourceRoot":"","sources":["../../src/daemon/main.ts"],"names":[],"mappings":";AACA;;GAEG"}
@@ -0,0 +1,38 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * F2A Daemon 入口
4
+ */
5
+ import { F2ADaemon } from './index.js';
6
+ // 解析引导节点地址
7
+ const bootstrapPeers = process.env.BOOTSTRAP_PEERS
8
+ ? process.env.BOOTSTRAP_PEERS.split(',')
9
+ : undefined;
10
+ // P2P 端口(默认 0 = 随机分配)
11
+ const p2pPort = parseInt(process.env.F2A_P2P_PORT || '0');
12
+ const daemon = new F2ADaemon({
13
+ controlPort: parseInt(process.env.F2A_CONTROL_PORT || '9001'),
14
+ network: {
15
+ listenPort: p2pPort,
16
+ bootstrapPeers,
17
+ },
18
+ });
19
+ // 处理信号
20
+ process.on('SIGINT', async () => {
21
+ console.log('[Daemon] Received SIGINT, shutting down...');
22
+ await daemon.stop();
23
+ process.exit(0);
24
+ });
25
+ process.on('SIGTERM', async () => {
26
+ console.log('[Daemon] Received SIGTERM, shutting down...');
27
+ await daemon.stop();
28
+ process.exit(0);
29
+ });
30
+ // 启动
31
+ daemon.start().catch((error) => {
32
+ console.error('[Daemon] Failed to start:', error instanceof Error ? error.message : String(error));
33
+ if (error instanceof Error && error.stack) {
34
+ console.error('[Daemon] Stack trace:', error.stack);
35
+ }
36
+ process.exit(1);
37
+ });
38
+ //# sourceMappingURL=main.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"main.js","sourceRoot":"","sources":["../../src/daemon/main.ts"],"names":[],"mappings":";AACA;;GAEG;AAEH,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAEvC,WAAW;AACX,MAAM,cAAc,GAAG,OAAO,CAAC,GAAG,CAAC,eAAe;IAChD,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,KAAK,CAAC,GAAG,CAAC;IACxC,CAAC,CAAC,SAAS,CAAC;AAEd,sBAAsB;AACtB,MAAM,OAAO,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,YAAY,IAAI,GAAG,CAAC,CAAC;AAE1D,MAAM,MAAM,GAAG,IAAI,SAAS,CAAC;IAC3B,WAAW,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,gBAAgB,IAAI,MAAM,CAAC;IAC7D,OAAO,EAAE;QACP,UAAU,EAAE,OAAO;QACnB,cAAc;KACf;CACF,CAAC,CAAC;AAEH,OAAO;AACP,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,KAAK,IAAI,EAAE;IAC9B,OAAO,CAAC,GAAG,CAAC,4CAA4C,CAAC,CAAC;IAC1D,MAAM,MAAM,CAAC,IAAI,EAAE,CAAC;IACpB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC;AAEH,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,KAAK,IAAI,EAAE;IAC/B,OAAO,CAAC,GAAG,CAAC,6CAA6C,CAAC,CAAC;IAC3D,MAAM,MAAM,CAAC,IAAI,EAAE,CAAC;IACpB,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC;AAEH,KAAK;AACL,MAAM,CAAC,KAAK,EAAE,CAAC,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE;IAC7B,OAAO,CAAC,KAAK,CAAC,2BAA2B,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;IACnG,IAAI,KAAK,YAAY,KAAK,IAAI,KAAK,CAAC,KAAK,EAAE,CAAC;QAC1C,OAAO,CAAC,KAAK,CAAC,uBAAuB,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC;IACtD,CAAC;IACD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;AAClB,CAAC,CAAC,CAAC"}
@@ -0,0 +1,6 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * F2A Daemon 入口
4
+ */
5
+ export {};
6
+ //# sourceMappingURL=start.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"start.d.ts","sourceRoot":"","sources":["../../src/daemon/start.ts"],"names":[],"mappings":";AACA;;GAEG"}