@f2a/openclaw-adapter 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 (51) hide show
  1. package/README.md +510 -0
  2. package/dist/announcement-queue.d.ts +71 -0
  3. package/dist/announcement-queue.d.ts.map +1 -0
  4. package/dist/announcement-queue.js +181 -0
  5. package/dist/announcement-queue.js.map +1 -0
  6. package/dist/capability-detector.d.ts +21 -0
  7. package/dist/capability-detector.d.ts.map +1 -0
  8. package/dist/capability-detector.js +177 -0
  9. package/dist/capability-detector.js.map +1 -0
  10. package/dist/connector.d.ts +62 -0
  11. package/dist/connector.d.ts.map +1 -0
  12. package/dist/connector.js +1063 -0
  13. package/dist/connector.js.map +1 -0
  14. package/dist/index.d.ts +16 -0
  15. package/dist/index.d.ts.map +1 -0
  16. package/dist/index.js +44 -0
  17. package/dist/index.js.map +1 -0
  18. package/dist/network-client.d.ts +48 -0
  19. package/dist/network-client.d.ts.map +1 -0
  20. package/dist/network-client.js +104 -0
  21. package/dist/network-client.js.map +1 -0
  22. package/dist/node-manager.d.ts +50 -0
  23. package/dist/node-manager.d.ts.map +1 -0
  24. package/dist/node-manager.js +187 -0
  25. package/dist/node-manager.js.map +1 -0
  26. package/dist/plugin.d.ts +16 -0
  27. package/dist/plugin.d.ts.map +1 -0
  28. package/dist/plugin.js +116 -0
  29. package/dist/plugin.js.map +1 -0
  30. package/dist/reputation.d.ts +72 -0
  31. package/dist/reputation.d.ts.map +1 -0
  32. package/dist/reputation.js +215 -0
  33. package/dist/reputation.js.map +1 -0
  34. package/dist/task-guard.d.ts +77 -0
  35. package/dist/task-guard.d.ts.map +1 -0
  36. package/dist/task-guard.js +330 -0
  37. package/dist/task-guard.js.map +1 -0
  38. package/dist/task-queue.d.ts +71 -0
  39. package/dist/task-queue.d.ts.map +1 -0
  40. package/dist/task-queue.js +126 -0
  41. package/dist/task-queue.js.map +1 -0
  42. package/dist/types.d.ts +248 -0
  43. package/dist/types.d.ts.map +1 -0
  44. package/dist/types.js +6 -0
  45. package/dist/types.js.map +1 -0
  46. package/dist/webhook-server.d.ts +44 -0
  47. package/dist/webhook-server.d.ts.map +1 -0
  48. package/dist/webhook-server.js +119 -0
  49. package/dist/webhook-server.js.map +1 -0
  50. package/openclaw.plugin.json +106 -0
  51. package/package.json +40 -0
package/dist/plugin.js ADDED
@@ -0,0 +1,116 @@
1
+ "use strict";
2
+ /**
3
+ * F2A OpenClaw Adapter Plugin
4
+ * OpenClaw 插件标准入口
5
+ */
6
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
7
+ if (k2 === undefined) k2 = k;
8
+ var desc = Object.getOwnPropertyDescriptor(m, k);
9
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
10
+ desc = { enumerable: true, get: function() { return m[k]; } };
11
+ }
12
+ Object.defineProperty(o, k2, desc);
13
+ }) : (function(o, m, k, k2) {
14
+ if (k2 === undefined) k2 = k;
15
+ o[k2] = m[k];
16
+ }));
17
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
18
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
19
+ };
20
+ Object.defineProperty(exports, "__esModule", { value: true });
21
+ exports.TaskGuard = exports.AnnouncementQueue = exports.TaskQueue = exports.F2AOpenClawAdapter = void 0;
22
+ exports.default = register;
23
+ const connector_js_1 = require("./connector.js");
24
+ /**
25
+ * OpenClaw 插件注册函数
26
+ * 这是 OpenClaw 加载插件时调用的入口
27
+ */
28
+ async function register(api) {
29
+ const plugin = new connector_js_1.F2AOpenClawAdapter();
30
+ // 从 OpenClaw 配置中获取插件配置
31
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
32
+ const pluginsConfig = api.config?.plugins;
33
+ const config = pluginsConfig?.entries?.['f2a-openclaw-adapter']?.config || {};
34
+ // 将 API 引用传递给插件(用于触发心跳等操作)
35
+ const fullConfig = {
36
+ ...config,
37
+ _api: api
38
+ };
39
+ // 初始化插件 - 等待完成后再注册工具
40
+ try {
41
+ await plugin.initialize(fullConfig);
42
+ api.logger?.info('[F2A Adapter] 初始化完成');
43
+ }
44
+ catch (error) {
45
+ api.logger?.error(`[F2A Adapter] 初始化失败: ${error.message}`);
46
+ // 清理已分配的资源,避免孤儿进程和端口占用
47
+ try {
48
+ api.logger?.info('[F2A Adapter] 正在清理资源...');
49
+ await plugin.shutdown?.();
50
+ }
51
+ catch (shutdownError) {
52
+ api.logger?.warn(`[F2A Adapter] 清理资源时出错: ${shutdownError.message}`);
53
+ }
54
+ // 抛出错误让 OpenClaw 知道插件加载失败
55
+ throw new Error(`F2A Adapter 初始化失败: ${error.message}`);
56
+ }
57
+ // 初始化完成后注册所有工具
58
+ // OpenClaw 的 registerTool 需要 execute 方法
59
+ const tools = plugin.getTools();
60
+ for (const tool of tools) {
61
+ api.registerTool?.({
62
+ name: tool.name,
63
+ description: tool.description,
64
+ parameters: tool.parameters,
65
+ // OpenClaw 使用 execute 而不是 handler
66
+ async execute(_id, params) {
67
+ try {
68
+ // 构造一个模拟的 SessionContext
69
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
70
+ const workspace = api.config?.agents?.defaults?.workspace || '.';
71
+ const mockContext = {
72
+ sessionId: _id,
73
+ workspace,
74
+ toJSON: () => ({})
75
+ };
76
+ const result = await tool.handler(params, mockContext);
77
+ // 将 ToolResult 转换为 OpenClaw 期望的格式
78
+ if (typeof result === 'string') {
79
+ return { content: [{ type: 'text', text: result }] };
80
+ }
81
+ if (result?.content) {
82
+ return { content: [{ type: 'text', text: result.content }] };
83
+ }
84
+ return { content: [{ type: 'text', text: JSON.stringify(result) }] };
85
+ }
86
+ catch (error) {
87
+ api.logger?.error(`[F2A Adapter] 工具 ${tool.name} 执行失败: ${error.message}`);
88
+ throw error;
89
+ }
90
+ }
91
+ });
92
+ }
93
+ // 注册后台服务(用于清理资源)
94
+ api.registerService?.({
95
+ id: 'f2a-adapter-service',
96
+ start: () => {
97
+ api.logger?.info('[F2A Adapter] 服务已启动');
98
+ },
99
+ stop: async () => {
100
+ api.logger?.info('[F2A Adapter] 正在停止服务...');
101
+ await plugin.shutdown?.();
102
+ }
103
+ });
104
+ api.logger?.info(`[F2A Adapter] 已注册 ${tools.length} 个工具`);
105
+ }
106
+ // 重新导出主要类,供外部使用
107
+ var connector_js_2 = require("./connector.js");
108
+ Object.defineProperty(exports, "F2AOpenClawAdapter", { enumerable: true, get: function () { return connector_js_2.F2AOpenClawAdapter; } });
109
+ __exportStar(require("./types.js"), exports);
110
+ var task_queue_js_1 = require("./task-queue.js");
111
+ Object.defineProperty(exports, "TaskQueue", { enumerable: true, get: function () { return task_queue_js_1.TaskQueue; } });
112
+ var announcement_queue_js_1 = require("./announcement-queue.js");
113
+ Object.defineProperty(exports, "AnnouncementQueue", { enumerable: true, get: function () { return announcement_queue_js_1.AnnouncementQueue; } });
114
+ var task_guard_js_1 = require("./task-guard.js");
115
+ Object.defineProperty(exports, "TaskGuard", { enumerable: true, get: function () { return task_guard_js_1.TaskGuard; } });
116
+ //# sourceMappingURL=plugin.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"plugin.js","sourceRoot":"","sources":["../src/plugin.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,72 @@
1
+ /**
2
+ * 信誉系统
3
+ * 管理 Peer 的信誉分数
4
+ */
5
+ import type { ReputationEntry, ReputationConfig } from './types.js';
6
+ export declare class ReputationSystem {
7
+ private config;
8
+ private entries;
9
+ private dataPath;
10
+ constructor(config: ReputationConfig, dataDir: string);
11
+ /**
12
+ * 获取 Peer 信誉
13
+ */
14
+ getReputation(peerId: string): ReputationEntry;
15
+ /**
16
+ * 记录任务成功
17
+ */
18
+ recordSuccess(peerId: string, taskId: string, latency: number): void;
19
+ /**
20
+ * 记录任务失败
21
+ */
22
+ recordFailure(peerId: string, taskId: string, reason?: string): void;
23
+ /**
24
+ * 记录任务拒绝
25
+ */
26
+ recordRejection(peerId: string, taskId: string, reason?: string): void;
27
+ /**
28
+ * 记录超时
29
+ */
30
+ recordTimeout(peerId: string, taskId: string): void;
31
+ /**
32
+ * 记录恶意行为
33
+ */
34
+ recordMalicious(peerId: string, reason: string): void;
35
+ /**
36
+ * 检查是否允许服务
37
+ */
38
+ isAllowed(peerId: string): boolean;
39
+ /**
40
+ * 获取所有信誉记录
41
+ */
42
+ getAllReputations(): ReputationEntry[];
43
+ /**
44
+ * 清理过期记录
45
+ */
46
+ cleanup(maxAgeDays?: number): void;
47
+ /**
48
+ * 获取高信誉节点
49
+ */
50
+ getHighReputationNodes(minScore: number): ReputationEntry[];
51
+ /**
52
+ * 创建默认条目
53
+ */
54
+ private createDefaultEntry;
55
+ /**
56
+ * 更新平均响应时间
57
+ */
58
+ private updateAvgResponseTime;
59
+ /**
60
+ * 修剪历史记录
61
+ */
62
+ private trimHistory;
63
+ /**
64
+ * 加载数据
65
+ */
66
+ private load;
67
+ /**
68
+ * 保存数据
69
+ */
70
+ private save;
71
+ }
72
+ //# sourceMappingURL=reputation.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"reputation.d.ts","sourceRoot":"","sources":["../src/reputation.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAIH,OAAO,KAAK,EACV,eAAe,EAEf,gBAAgB,EAEjB,MAAM,YAAY,CAAC;AAEpB,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAmB;IACjC,OAAO,CAAC,OAAO,CAA2C;IAC1D,OAAO,CAAC,QAAQ,CAAS;gBAEb,MAAM,EAAE,gBAAgB,EAAE,OAAO,EAAE,MAAM;IAMrD;;OAEG;IACH,aAAa,CAAC,MAAM,EAAE,MAAM,GAAG,eAAe;IAO9C;;OAEG;IACH,aAAa,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,IAAI;IAoBpE;;OAEG;IACH,aAAa,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,GAAG,IAAI;IAoBpE;;OAEG;IACH,eAAe,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,GAAG,IAAI;IAmBtE;;OAEG;IACH,aAAa,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,IAAI;IAkBnD;;OAEG;IACH,eAAe,CAAC,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,IAAI;IAiBrD;;OAEG;IACH,SAAS,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO;IAOlC;;OAEG;IACH,iBAAiB,IAAI,eAAe,EAAE;IAItC;;OAEG;IACH,OAAO,CAAC,UAAU,GAAE,MAAW,GAAG,IAAI;IAYtC;;OAEG;IACH,sBAAsB,CAAC,QAAQ,EAAE,MAAM,GAAG,eAAe,EAAE;IAK3D;;OAEG;IACH,OAAO,CAAC,kBAAkB;IAa1B;;OAEG;IACH,OAAO,CAAC,qBAAqB;IAQ7B;;OAEG;IACH,OAAO,CAAC,WAAW;IAMnB;;OAEG;IACH,OAAO,CAAC,IAAI;IAcZ;;OAEG;IACH,OAAO,CAAC,IAAI;CAQb"}
@@ -0,0 +1,215 @@
1
+ "use strict";
2
+ /**
3
+ * 信誉系统
4
+ * 管理 Peer 的信誉分数
5
+ */
6
+ Object.defineProperty(exports, "__esModule", { value: true });
7
+ exports.ReputationSystem = void 0;
8
+ const fs_1 = require("fs");
9
+ const path_1 = require("path");
10
+ class ReputationSystem {
11
+ config;
12
+ entries = new Map();
13
+ dataPath;
14
+ constructor(config, dataDir) {
15
+ this.config = config;
16
+ this.dataPath = (0, path_1.join)(dataDir, 'reputation.json');
17
+ this.load();
18
+ }
19
+ /**
20
+ * 获取 Peer 信誉
21
+ */
22
+ getReputation(peerId) {
23
+ if (!this.entries.has(peerId)) {
24
+ this.entries.set(peerId, this.createDefaultEntry(peerId));
25
+ }
26
+ return this.entries.get(peerId);
27
+ }
28
+ /**
29
+ * 记录任务成功
30
+ */
31
+ recordSuccess(peerId, taskId, latency) {
32
+ const entry = this.getReputation(peerId);
33
+ entry.successfulTasks++;
34
+ entry.totalTasks++;
35
+ entry.score = Math.min(100, entry.score + 10);
36
+ entry.avgResponseTime = this.updateAvgResponseTime(entry, latency);
37
+ entry.lastInteraction = Date.now();
38
+ entry.history.push({
39
+ type: 'task_success',
40
+ taskId,
41
+ delta: +10,
42
+ timestamp: Date.now()
43
+ });
44
+ this.trimHistory(entry);
45
+ this.save();
46
+ }
47
+ /**
48
+ * 记录任务失败
49
+ */
50
+ recordFailure(peerId, taskId, reason) {
51
+ const entry = this.getReputation(peerId);
52
+ entry.failedTasks++;
53
+ entry.totalTasks++;
54
+ entry.score = Math.max(0, entry.score - 20);
55
+ entry.lastInteraction = Date.now();
56
+ entry.history.push({
57
+ type: 'task_failure',
58
+ taskId,
59
+ delta: -20,
60
+ timestamp: Date.now(),
61
+ reason
62
+ });
63
+ this.trimHistory(entry);
64
+ this.save();
65
+ }
66
+ /**
67
+ * 记录任务拒绝
68
+ */
69
+ recordRejection(peerId, taskId, reason) {
70
+ const entry = this.getReputation(peerId);
71
+ entry.totalTasks++;
72
+ entry.score = Math.max(0, entry.score - 5);
73
+ entry.lastInteraction = Date.now();
74
+ entry.history.push({
75
+ type: 'task_rejected',
76
+ taskId,
77
+ delta: -5,
78
+ timestamp: Date.now(),
79
+ reason
80
+ });
81
+ this.trimHistory(entry);
82
+ this.save();
83
+ }
84
+ /**
85
+ * 记录超时
86
+ */
87
+ recordTimeout(peerId, taskId) {
88
+ const entry = this.getReputation(peerId);
89
+ entry.totalTasks++;
90
+ entry.score = Math.max(0, entry.score - 15);
91
+ entry.lastInteraction = Date.now();
92
+ entry.history.push({
93
+ type: 'timeout',
94
+ taskId,
95
+ delta: -15,
96
+ timestamp: Date.now()
97
+ });
98
+ this.trimHistory(entry);
99
+ this.save();
100
+ }
101
+ /**
102
+ * 记录恶意行为
103
+ */
104
+ recordMalicious(peerId, reason) {
105
+ const entry = this.getReputation(peerId);
106
+ entry.score = Math.max(0, entry.score - 50);
107
+ entry.lastInteraction = Date.now();
108
+ entry.history.push({
109
+ type: 'malicious',
110
+ delta: -50,
111
+ timestamp: Date.now(),
112
+ reason
113
+ });
114
+ this.trimHistory(entry);
115
+ this.save();
116
+ }
117
+ /**
118
+ * 检查是否允许服务
119
+ */
120
+ isAllowed(peerId) {
121
+ if (!this.config.enabled)
122
+ return true;
123
+ const entry = this.getReputation(peerId);
124
+ return entry.score >= this.config.minScoreForService;
125
+ }
126
+ /**
127
+ * 获取所有信誉记录
128
+ */
129
+ getAllReputations() {
130
+ return Array.from(this.entries.values());
131
+ }
132
+ /**
133
+ * 清理过期记录
134
+ */
135
+ cleanup(maxAgeDays = 30) {
136
+ const cutoff = Date.now() - (maxAgeDays * 24 * 60 * 60 * 1000);
137
+ for (const [peerId, entry] of this.entries) {
138
+ if (entry.lastInteraction < cutoff && entry.totalTasks === 0) {
139
+ this.entries.delete(peerId);
140
+ }
141
+ }
142
+ this.save();
143
+ }
144
+ /**
145
+ * 获取高信誉节点
146
+ */
147
+ getHighReputationNodes(minScore) {
148
+ return Array.from(this.entries.values())
149
+ .filter(entry => entry.score >= minScore);
150
+ }
151
+ /**
152
+ * 创建默认条目
153
+ */
154
+ createDefaultEntry(peerId) {
155
+ return {
156
+ peerId,
157
+ score: this.config.initialScore,
158
+ successfulTasks: 0,
159
+ failedTasks: 0,
160
+ totalTasks: 0,
161
+ avgResponseTime: 0,
162
+ lastInteraction: 0,
163
+ history: []
164
+ };
165
+ }
166
+ /**
167
+ * 更新平均响应时间
168
+ */
169
+ updateAvgResponseTime(entry, newLatency) {
170
+ if (entry.avgResponseTime === 0) {
171
+ return newLatency;
172
+ }
173
+ // 指数移动平均
174
+ return entry.avgResponseTime * 0.7 + newLatency * 0.3;
175
+ }
176
+ /**
177
+ * 修剪历史记录
178
+ */
179
+ trimHistory(entry, maxSize = 100) {
180
+ if (entry.history.length > maxSize) {
181
+ entry.history = entry.history.slice(-maxSize);
182
+ }
183
+ }
184
+ /**
185
+ * 加载数据
186
+ */
187
+ load() {
188
+ if ((0, fs_1.existsSync)(this.dataPath)) {
189
+ try {
190
+ const data = JSON.parse((0, fs_1.readFileSync)(this.dataPath, 'utf-8'));
191
+ for (const entry of data) {
192
+ this.entries.set(entry.peerId, entry);
193
+ }
194
+ console.log(`[F2A Reputation] 加载了 ${this.entries.size} 条记录`);
195
+ }
196
+ catch (e) {
197
+ console.error('[F2A Reputation] 加载失败:', e);
198
+ }
199
+ }
200
+ }
201
+ /**
202
+ * 保存数据
203
+ */
204
+ save() {
205
+ try {
206
+ const data = Array.from(this.entries.values());
207
+ (0, fs_1.writeFileSync)(this.dataPath, JSON.stringify(data, null, 2));
208
+ }
209
+ catch (e) {
210
+ console.error('[F2A Reputation] 保存失败:', e);
211
+ }
212
+ }
213
+ }
214
+ exports.ReputationSystem = ReputationSystem;
215
+ //# sourceMappingURL=reputation.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"reputation.js","sourceRoot":"","sources":["../src/reputation.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,77 @@
1
+ /**
2
+ * F2A Task Guard
3
+ * 轻量级任务安全检查和评审
4
+ */
5
+ import type { TaskRequest, TaskAnnouncement, ReputationEntry } from './types.js';
6
+ export interface TaskGuardRule {
7
+ id: string;
8
+ name: string;
9
+ description: string;
10
+ enabled: boolean;
11
+ severity: 'info' | 'warn' | 'block';
12
+ check: (task: TaskRequest | TaskAnnouncement, context: TaskGuardContext) => TaskGuardResult;
13
+ }
14
+ export interface TaskGuardContext {
15
+ requesterReputation?: ReputationEntry;
16
+ isWhitelisted: boolean;
17
+ isBlacklisted: boolean;
18
+ recentTaskCount: number;
19
+ config: TaskGuardConfig;
20
+ }
21
+ export interface TaskGuardConfig {
22
+ enabled: boolean;
23
+ requireConfirmationForDangerous: boolean;
24
+ maxTasksPerMinute: number;
25
+ blockedKeywords: string[];
26
+ dangerousPatterns: RegExp[];
27
+ minReputationForDangerous: number;
28
+ }
29
+ export interface TaskGuardResult {
30
+ passed: boolean;
31
+ severity: 'info' | 'warn' | 'block';
32
+ ruleId: string;
33
+ message: string;
34
+ details?: Record<string, unknown>;
35
+ }
36
+ export interface TaskGuardReport {
37
+ taskId: string;
38
+ passed: boolean;
39
+ results: TaskGuardResult[];
40
+ warnings: TaskGuardResult[];
41
+ blocks: TaskGuardResult[];
42
+ requiresConfirmation: boolean;
43
+ timestamp: number;
44
+ }
45
+ export declare const DEFAULT_TASK_GUARD_CONFIG: TaskGuardConfig;
46
+ export declare class TaskGuard {
47
+ private config;
48
+ private rules;
49
+ private recentTasks;
50
+ constructor(config?: Partial<TaskGuardConfig>);
51
+ /**
52
+ * 检查任务
53
+ */
54
+ check(task: TaskRequest | TaskAnnouncement, context?: Partial<TaskGuardContext>): TaskGuardReport;
55
+ /**
56
+ * 快速检查(只返回是否通过)
57
+ */
58
+ quickCheck(task: TaskRequest | TaskAnnouncement, context?: Partial<TaskGuardContext>): boolean;
59
+ /**
60
+ * 添加自定义规则
61
+ */
62
+ addRule(rule: TaskGuardRule): void;
63
+ /**
64
+ * 启用/禁用规则
65
+ */
66
+ setRuleEnabled(ruleId: string, enabled: boolean): void;
67
+ /**
68
+ * 更新配置
69
+ */
70
+ updateConfig(config: Partial<TaskGuardConfig>): void;
71
+ private createDefaultRules;
72
+ private isDangerousTask;
73
+ private getRecentTaskCount;
74
+ private recordTask;
75
+ }
76
+ export declare const taskGuard: TaskGuard;
77
+ //# sourceMappingURL=task-guard.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"task-guard.d.ts","sourceRoot":"","sources":["../src/task-guard.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,KAAK,EAAE,WAAW,EAAE,gBAAgB,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAEjF,MAAM,WAAW,aAAa;IAC5B,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,OAAO,EAAE,OAAO,CAAC;IACjB,QAAQ,EAAE,MAAM,GAAG,MAAM,GAAG,OAAO,CAAC;IACpC,KAAK,EAAE,CAAC,IAAI,EAAE,WAAW,GAAG,gBAAgB,EAAE,OAAO,EAAE,gBAAgB,KAAK,eAAe,CAAC;CAC7F;AAED,MAAM,WAAW,gBAAgB;IAC/B,mBAAmB,CAAC,EAAE,eAAe,CAAC;IACtC,aAAa,EAAE,OAAO,CAAC;IACvB,aAAa,EAAE,OAAO,CAAC;IACvB,eAAe,EAAE,MAAM,CAAC;IACxB,MAAM,EAAE,eAAe,CAAC;CACzB;AAED,MAAM,WAAW,eAAe;IAC9B,OAAO,EAAE,OAAO,CAAC;IACjB,+BAA+B,EAAE,OAAO,CAAC;IACzC,iBAAiB,EAAE,MAAM,CAAC;IAC1B,eAAe,EAAE,MAAM,EAAE,CAAC;IAC1B,iBAAiB,EAAE,MAAM,EAAE,CAAC;IAC5B,yBAAyB,EAAE,MAAM,CAAC;CACnC;AAED,MAAM,WAAW,eAAe;IAC9B,MAAM,EAAE,OAAO,CAAC;IAChB,QAAQ,EAAE,MAAM,GAAG,MAAM,GAAG,OAAO,CAAC;IACpC,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CACnC;AAED,MAAM,WAAW,eAAe;IAC9B,MAAM,EAAE,MAAM,CAAC;IACf,MAAM,EAAE,OAAO,CAAC;IAChB,OAAO,EAAE,eAAe,EAAE,CAAC;IAC3B,QAAQ,EAAE,eAAe,EAAE,CAAC;IAC5B,MAAM,EAAE,eAAe,EAAE,CAAC;IAC1B,oBAAoB,EAAE,OAAO,CAAC;IAC9B,SAAS,EAAE,MAAM,CAAC;CACnB;AAGD,eAAO,MAAM,yBAAyB,EAAE,eAoBvC,CAAC;AAEF,qBAAa,SAAS;IACpB,OAAO,CAAC,MAAM,CAAkB;IAChC,OAAO,CAAC,KAAK,CAAkB;IAC/B,OAAO,CAAC,WAAW,CAAoC;gBAE3C,MAAM,GAAE,OAAO,CAAC,eAAe,CAAM;IAKjD;;OAEG;IACH,KAAK,CACH,IAAI,EAAE,WAAW,GAAG,gBAAgB,EACpC,OAAO,GAAE,OAAO,CAAC,gBAAgB,CAAM,GACtC,eAAe;IAkDlB;;OAEG;IACH,UAAU,CACR,IAAI,EAAE,WAAW,GAAG,gBAAgB,EACpC,OAAO,CAAC,EAAE,OAAO,CAAC,gBAAgB,CAAC,GAClC,OAAO;IAKV;;OAEG;IACH,OAAO,CAAC,IAAI,EAAE,aAAa,GAAG,IAAI;IAIlC;;OAEG;IACH,cAAc,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,GAAG,IAAI;IAOtD;;OAEG;IACH,YAAY,CAAC,MAAM,EAAE,OAAO,CAAC,eAAe,CAAC,GAAG,IAAI;IAMpD,OAAO,CAAC,kBAAkB;IA+M1B,OAAO,CAAC,eAAe;IAgBvB,OAAO,CAAC,kBAAkB;IAa1B,OAAO,CAAC,UAAU;CAKnB;AAGD,eAAO,MAAM,SAAS,WAAkB,CAAC"}