flex-mcp 1.0.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 (131) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +154 -0
  3. package/dist/api/auth.d.ts +6 -0
  4. package/dist/api/auth.d.ts.map +1 -0
  5. package/dist/api/auth.js +111 -0
  6. package/dist/api/auth.js.map +1 -0
  7. package/dist/api/configs.d.ts +6 -0
  8. package/dist/api/configs.d.ts.map +1 -0
  9. package/dist/api/configs.js +693 -0
  10. package/dist/api/configs.js.map +1 -0
  11. package/dist/api/index.d.ts +6 -0
  12. package/dist/api/index.d.ts.map +1 -0
  13. package/dist/api/index.js +24 -0
  14. package/dist/api/index.js.map +1 -0
  15. package/dist/api/mcp.d.ts +7 -0
  16. package/dist/api/mcp.d.ts.map +1 -0
  17. package/dist/api/mcp.js +46 -0
  18. package/dist/api/mcp.js.map +1 -0
  19. package/dist/api/middleware.d.ts +25 -0
  20. package/dist/api/middleware.d.ts.map +1 -0
  21. package/dist/api/middleware.js +65 -0
  22. package/dist/api/middleware.js.map +1 -0
  23. package/dist/api/prompts.d.ts +6 -0
  24. package/dist/api/prompts.d.ts.map +1 -0
  25. package/dist/api/prompts.js +260 -0
  26. package/dist/api/prompts.js.map +1 -0
  27. package/dist/api/resources.d.ts +6 -0
  28. package/dist/api/resources.d.ts.map +1 -0
  29. package/dist/api/resources.js +159 -0
  30. package/dist/api/resources.js.map +1 -0
  31. package/dist/api/tables.d.ts +15 -0
  32. package/dist/api/tables.d.ts.map +1 -0
  33. package/dist/api/tables.js +702 -0
  34. package/dist/api/tables.js.map +1 -0
  35. package/dist/api/users.d.ts +6 -0
  36. package/dist/api/users.d.ts.map +1 -0
  37. package/dist/api/users.js +96 -0
  38. package/dist/api/users.js.map +1 -0
  39. package/dist/auth/index.d.ts +15 -0
  40. package/dist/auth/index.d.ts.map +1 -0
  41. package/dist/auth/index.js +126 -0
  42. package/dist/auth/index.js.map +1 -0
  43. package/dist/cli.d.ts +7 -0
  44. package/dist/cli.d.ts.map +1 -0
  45. package/dist/cli.js +288 -0
  46. package/dist/cli.js.map +1 -0
  47. package/dist/config.d.ts +21 -0
  48. package/dist/config.d.ts.map +1 -0
  49. package/dist/config.js +54 -0
  50. package/dist/config.js.map +1 -0
  51. package/dist/db/index.d.ts +11 -0
  52. package/dist/db/index.d.ts.map +1 -0
  53. package/dist/db/index.js +164 -0
  54. package/dist/db/index.js.map +1 -0
  55. package/dist/db/schema.d.ts +646 -0
  56. package/dist/db/schema.d.ts.map +1 -0
  57. package/dist/db/schema.js +73 -0
  58. package/dist/db/schema.js.map +1 -0
  59. package/dist/index.d.ts +5 -0
  60. package/dist/index.d.ts.map +1 -0
  61. package/dist/index.js +153 -0
  62. package/dist/index.js.map +1 -0
  63. package/dist/mcp/aggregator.d.ts +78 -0
  64. package/dist/mcp/aggregator.d.ts.map +1 -0
  65. package/dist/mcp/aggregator.js +1266 -0
  66. package/dist/mcp/aggregator.js.map +1 -0
  67. package/dist/mcp/component-loader.d.ts +17 -0
  68. package/dist/mcp/component-loader.d.ts.map +1 -0
  69. package/dist/mcp/component-loader.js +131 -0
  70. package/dist/mcp/component-loader.js.map +1 -0
  71. package/dist/mcp/index.d.ts +7 -0
  72. package/dist/mcp/index.d.ts.map +1 -0
  73. package/dist/mcp/index.js +107 -0
  74. package/dist/mcp/index.js.map +1 -0
  75. package/dist/mcp/mcp-client.d.ts +53 -0
  76. package/dist/mcp/mcp-client.d.ts.map +1 -0
  77. package/dist/mcp/mcp-client.js +418 -0
  78. package/dist/mcp/mcp-client.js.map +1 -0
  79. package/dist/mcp/post-sse-transport.d.ts +52 -0
  80. package/dist/mcp/post-sse-transport.d.ts.map +1 -0
  81. package/dist/mcp/post-sse-transport.js +375 -0
  82. package/dist/mcp/post-sse-transport.js.map +1 -0
  83. package/dist/mcp/service.d.ts +49 -0
  84. package/dist/mcp/service.d.ts.map +1 -0
  85. package/dist/mcp/service.js +358 -0
  86. package/dist/mcp/service.js.map +1 -0
  87. package/dist/mcp/tool-sync.d.ts +27 -0
  88. package/dist/mcp/tool-sync.d.ts.map +1 -0
  89. package/dist/mcp/tool-sync.js +200 -0
  90. package/dist/mcp/tool-sync.js.map +1 -0
  91. package/dist/script/compiler.d.ts +15 -0
  92. package/dist/script/compiler.d.ts.map +1 -0
  93. package/dist/script/compiler.js +163 -0
  94. package/dist/script/compiler.js.map +1 -0
  95. package/dist/script/context.d.ts +11 -0
  96. package/dist/script/context.d.ts.map +1 -0
  97. package/dist/script/context.js +786 -0
  98. package/dist/script/context.js.map +1 -0
  99. package/dist/script/executor.d.ts +43 -0
  100. package/dist/script/executor.d.ts.map +1 -0
  101. package/dist/script/executor.js +126 -0
  102. package/dist/script/executor.js.map +1 -0
  103. package/dist/static.d.ts +21 -0
  104. package/dist/static.d.ts.map +1 -0
  105. package/dist/static.js +95 -0
  106. package/dist/static.js.map +1 -0
  107. package/dist/types/index.d.ts +337 -0
  108. package/dist/types/index.d.ts.map +1 -0
  109. package/dist/types/index.js +5 -0
  110. package/dist/types/index.js.map +1 -0
  111. package/dist/utils/id.d.ts +5 -0
  112. package/dist/utils/id.d.ts.map +1 -0
  113. package/dist/utils/id.js +8 -0
  114. package/dist/utils/id.js.map +1 -0
  115. package/dist/utils/logger.d.ts +21 -0
  116. package/dist/utils/logger.d.ts.map +1 -0
  117. package/dist/utils/logger.js +31 -0
  118. package/dist/utils/logger.js.map +1 -0
  119. package/dist/utils/prompt-template.d.ts +15 -0
  120. package/dist/utils/prompt-template.d.ts.map +1 -0
  121. package/dist/utils/prompt-template.js +82 -0
  122. package/dist/utils/prompt-template.js.map +1 -0
  123. package/dist/utils/telemetry.d.ts +45 -0
  124. package/dist/utils/telemetry.d.ts.map +1 -0
  125. package/dist/utils/telemetry.js +245 -0
  126. package/dist/utils/telemetry.js.map +1 -0
  127. package/dist/utils/variables.d.ts +24 -0
  128. package/dist/utils/variables.d.ts.map +1 -0
  129. package/dist/utils/variables.js +45 -0
  130. package/dist/utils/variables.js.map +1 -0
  131. package/package.json +70 -0
@@ -0,0 +1,73 @@
1
+ /**
2
+ * 数据库 Schema
3
+ */
4
+ import { sqliteTable, text, integer } from 'drizzle-orm/sqlite-core';
5
+ // 用户表
6
+ export const users = sqliteTable('users', {
7
+ id: text('id').primaryKey(),
8
+ email: text('email').notNull().unique(),
9
+ apiKey: text('api_key').notNull().unique(),
10
+ role: text('role', { enum: ['admin', 'user'] }).notNull().default('user'),
11
+ createdAt: integer('created_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
12
+ updatedAt: integer('updated_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
13
+ });
14
+ // MCP 配置表
15
+ export const mcpConfigs = sqliteTable('mcp_configs', {
16
+ id: text('id').primaryKey(),
17
+ userId: text('user_id').notNull().references(() => users.id, { onDelete: 'cascade' }),
18
+ name: text('name').notNull(),
19
+ components: text('components', { mode: 'json' }).notNull(), // JSON 数组
20
+ customTools: text('custom_tools', { mode: 'json' }).notNull(), // JSON 数组
21
+ variables: text('variables', { mode: 'json' }).notNull(), // JSON 数组
22
+ initScript: text('init_script'), // 初始化脚本
23
+ enableExploreTable: integer('enable_explore_table', { mode: 'boolean' }).default(true), // 是否启用 explore_table 工具
24
+ createdAt: integer('created_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
25
+ updatedAt: integer('updated_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
26
+ });
27
+ // 表定义表(存储表的结构信息,每个用户独立)
28
+ export const tableDefinitions = sqliteTable('table_definitions', {
29
+ id: text('id').primaryKey(),
30
+ userId: text('user_id').notNull().references(() => users.id, { onDelete: 'cascade' }), // 用户ID,实现隔离
31
+ name: text('name').notNull(), // 表名(同一用户内唯一)
32
+ description: text('description'), // 表描述
33
+ columns: text('columns', { mode: 'json' }).notNull(), // 列定义 JSON 数组,每个列包含 {name, type, nullable, defaultValue}
34
+ allowExplore: integer('allow_explore', { mode: 'boolean' }).default(true), // 是否允许 AI 探索此表(默认为 true)
35
+ createdAt: integer('created_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
36
+ updatedAt: integer('updated_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
37
+ });
38
+ // 表数据表(存储表的实际数据,使用 JSON 存储行数据)
39
+ export const tableData = sqliteTable('table_data', {
40
+ id: text('id').primaryKey(),
41
+ tableId: text('table_id').notNull().references(() => tableDefinitions.id, { onDelete: 'cascade' }),
42
+ rowData: text('row_data', { mode: 'json' }).notNull(), // 行数据 JSON 对象
43
+ createdAt: integer('created_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
44
+ updatedAt: integer('updated_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
45
+ });
46
+ // 提示词表(每个用户独立)
47
+ export const prompts = sqliteTable('prompts', {
48
+ id: text('id').primaryKey(),
49
+ userId: text('user_id').notNull().references(() => users.id, { onDelete: 'cascade' }),
50
+ name: text('name').notNull(), // 提示词名称(同一用户内唯一)
51
+ description: text('description'), // 提示词描述
52
+ arguments: text('arguments', { mode: 'json' }).notNull(), // 参数定义 JSON 数组
53
+ messages: text('messages', { mode: 'json' }).notNull(), // 消息模板 JSON 数组
54
+ createdAt: integer('created_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
55
+ updatedAt: integer('updated_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
56
+ });
57
+ // MCP 组件工具表(存储从 MCP 组件同步的工具配置)
58
+ export const mcpComponentTools = sqliteTable('mcp_component_tools', {
59
+ id: text('id').primaryKey(),
60
+ userId: text('user_id').notNull().references(() => users.id, { onDelete: 'cascade' }),
61
+ configId: text('config_id').notNull().references(() => mcpConfigs.id, { onDelete: 'cascade' }), // 关联到 MCP 配置
62
+ componentId: text('component_id').notNull(), // 组件 ID(关联到 mcpConfigs.components 中的某个组件)
63
+ componentName: text('component_name').notNull(), // 组件名称
64
+ toolName: text('tool_name').notNull(), // 工具名称(带命名空间前缀,如 "namespace.toolName")
65
+ originalToolName: text('original_tool_name').notNull(), // 原始工具名称(不带命名空间)
66
+ description: text('description').notNull(), // 工具描述
67
+ inputSchema: text('input_schema', { mode: 'json' }).notNull(), // 工具的输入 schema
68
+ disabled: integer('disabled', { mode: 'boolean' }).notNull().default(false), // 是否禁用
69
+ toolPatches: text('tool_patches', { mode: 'json' }), // 工具补丁配置(用于编辑 schema)
70
+ createdAt: integer('created_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
71
+ updatedAt: integer('updated_at', { mode: 'timestamp' }).notNull().$defaultFn(() => new Date()),
72
+ });
73
+ //# sourceMappingURL=schema.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.js","sourceRoot":"","sources":["../../src/db/schema.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * 服务器入口文件
3
+ */
4
+ import 'dotenv/config';
5
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,eAAe,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,153 @@
1
+ /**
2
+ * 服务器入口文件
3
+ */
4
+ import 'dotenv/config';
5
+ import { serve } from '@hono/node-server';
6
+ import { Hono } from 'hono';
7
+ import { cors } from 'hono/cors';
8
+ import { initDatabase } from './db/index.js';
9
+ import { apiRouter } from './api/index.js';
10
+ import { mcpRouter } from './mcp/index.js';
11
+ import { hasWebFiles, readStaticFile, getMimeType } from './static.js';
12
+ import { reportStartup } from './utils/telemetry.js';
13
+ const app = new Hono();
14
+ // 日志中间件
15
+ app.use('*', async (c, next) => {
16
+ const start = Date.now();
17
+ await next();
18
+ const ms = Date.now() - start;
19
+ console.log(`${c.req.method} ${c.req.url} - ${c.res.status} (${ms}ms)`);
20
+ });
21
+ // CORS 配置
22
+ app.use('*', cors({
23
+ origin: process.env.CORS_ORIGIN || '*',
24
+ credentials: true,
25
+ }));
26
+ // 健康检查
27
+ app.get('/health', (c) => {
28
+ return c.json({ status: 'ok' });
29
+ });
30
+ // API 路由
31
+ app.route('/api', apiRouter);
32
+ // MCP 路由
33
+ app.route('/mcp', mcpRouter);
34
+ // 静态文件服务(Web 界面)
35
+ // 注意:必须在 API 和 MCP 路由之后,作为 fallback
36
+ let hasWeb = false;
37
+ if (hasWebFiles()) {
38
+ hasWeb = true;
39
+ // 提供静态文件(必须在最后,作为 fallback)
40
+ app.get('*', async (c) => {
41
+ const url = new URL(c.req.url);
42
+ const pathname = url.pathname;
43
+ // API 和 MCP 路径不处理(应该已经被前面的路由处理了)
44
+ if (pathname.startsWith('/api') || pathname.startsWith('/mcp') || pathname === '/health') {
45
+ return c.notFound();
46
+ }
47
+ // 根路径返回 index.html
48
+ if (pathname === '/' || pathname === '') {
49
+ const indexFile = readStaticFile('index.html');
50
+ if (indexFile) {
51
+ return c.html(indexFile.toString('utf-8'));
52
+ }
53
+ }
54
+ // 尝试读取静态文件(去掉开头的 /)
55
+ const filePath = pathname.startsWith('/') ? pathname.substring(1) : pathname;
56
+ const file = readStaticFile(filePath);
57
+ if (file) {
58
+ const mimeType = getMimeType(filePath);
59
+ // 根据文件类型返回不同格式
60
+ if (mimeType.startsWith('text/') || mimeType.includes('json') || mimeType.includes('javascript') || mimeType.includes('css')) {
61
+ return c.text(file.toString('utf-8'), 200, { 'Content-Type': mimeType });
62
+ }
63
+ else {
64
+ return new Response(file, {
65
+ status: 200,
66
+ headers: { 'Content-Type': mimeType },
67
+ });
68
+ }
69
+ }
70
+ // SPA 路由:所有其他路径返回 index.html
71
+ const indexFile = readStaticFile('index.html');
72
+ if (indexFile) {
73
+ return c.html(indexFile.toString('utf-8'));
74
+ }
75
+ return c.notFound();
76
+ });
77
+ }
78
+ else {
79
+ // 如果 web 未构建,提供提示页面
80
+ app.get('/', (c) => {
81
+ return c.html(`
82
+ <!DOCTYPE html>
83
+ <html lang="zh-CN">
84
+ <head>
85
+ <meta charset="UTF-8">
86
+ <meta name="viewport" content="width=device-width, initial-scale=1.0">
87
+ <title>Meta MCP Server</title>
88
+ <style>
89
+ body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; padding: 40px; text-align: center; background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); color: white; min-height: 100vh; margin: 0; }
90
+ .container { max-width: 600px; margin: 0 auto; background: rgba(255,255,255,0.1); padding: 40px; border-radius: 20px; backdrop-filter: blur(10px); }
91
+ h1 { margin: 0 0 20px 0; font-size: 2.5em; }
92
+ p { margin: 15px 0; line-height: 1.6; }
93
+ code { background: rgba(0,0,0,0.3); padding: 4px 8px; border-radius: 4px; font-family: 'Courier New', monospace; }
94
+ a { color: #fff; text-decoration: underline; }
95
+ .links { margin-top: 30px; }
96
+ .links a { display: inline-block; margin: 10px; padding: 10px 20px; background: rgba(255,255,255,0.2); border-radius: 5px; text-decoration: none; }
97
+ </style>
98
+ </head>
99
+ <body>
100
+ <div class="container">
101
+ <h1>🚀 Meta MCP Server</h1>
102
+ <p>服务器运行正常!</p>
103
+ <p>Web 界面未构建,请选择以下方式之一:</p>
104
+ <div style="text-align: left; margin: 20px 0;">
105
+ <p><strong>方式 1(开发模式):</strong></p>
106
+ <p>在项目根目录运行:<code>pnpm --filter web dev</code></p>
107
+ <p>然后访问: <a href="http://localhost:5173">http://localhost:5173</a></p>
108
+ <p style="margin-top: 20px;"><strong>方式 2(生产模式):</strong></p>
109
+ <p>构建前端:<code>pnpm --filter web build</code></p>
110
+ <p>然后重启服务器,Web 界面将在 <code>http://localhost:3000</code> 可用</p>
111
+ </div>
112
+ <div class="links">
113
+ <a href="/health">健康检查</a>
114
+ <a href="/api">API 端点</a>
115
+ </div>
116
+ </div>
117
+ </body>
118
+ </html>
119
+ `);
120
+ });
121
+ }
122
+ // 启动服务器
123
+ const port = parseInt(process.env.PORT || '3000');
124
+ const host = process.env.HOST || 'localhost';
125
+ async function start() {
126
+ // 初始化数据库
127
+ await initDatabase();
128
+ // 启动服务器
129
+ serve({
130
+ fetch: app.fetch,
131
+ port,
132
+ hostname: host,
133
+ });
134
+ console.log(`🚀 Server running at http://${host}:${port}`);
135
+ console.log(`📊 API: http://${host}:${port}/api`);
136
+ console.log(`🔌 MCP: http://${host}:${port}/mcp`);
137
+ if (hasWeb) {
138
+ console.log(`🌐 Web UI: http://${host}:${port}`);
139
+ console.log(` - 用户登录: http://${host}:${port}/user/login`);
140
+ console.log(` - 管理员登录: http://${host}:${port}/admin/login`);
141
+ }
142
+ else {
143
+ console.log(`💡 Web UI: 运行 'pnpm --filter web dev' 启动开发服务器 (http://localhost:5173)`);
144
+ console.log(` 或构建前端: 'pnpm --filter web build' 然后重启服务器`);
145
+ }
146
+ // 上报服务器启动统计(延迟执行,不阻塞启动)
147
+ reportStartup();
148
+ }
149
+ start().catch((error) => {
150
+ console.error('Failed to start server:', error);
151
+ process.exit(1);
152
+ });
153
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,78 @@
1
+ /**
2
+ * MCP 聚合器
3
+ * 聚合多个 MCP 组件,管理工具和资源
4
+ */
5
+ import type { MCPConfig, MCPTool, MCPPrompt } from '../types/index.js';
6
+ export declare class MCPAggregator {
7
+ private config;
8
+ private scriptExecutor;
9
+ private components;
10
+ private failedComponents;
11
+ private variableMap;
12
+ private mcpClientConfig;
13
+ private isInitialized;
14
+ private userId;
15
+ readonly configUpdatedAt: Date;
16
+ constructor(config: MCPConfig, mcpClientConfig?: {
17
+ headers: Record<string, string>;
18
+ config: Record<string, any>;
19
+ } | null, userId?: string);
20
+ /**
21
+ * 检查是否已初始化
22
+ */
23
+ get initialized(): boolean;
24
+ /**
25
+ * 初始化聚合器
26
+ * 注意:不再在初始化时加载组件,组件将在调用工具时按需加载(延迟加载)
27
+ */
28
+ initialize(): Promise<void>;
29
+ /**
30
+ * 按需加载组件(延迟加载)
31
+ */
32
+ private ensureComponentLoaded;
33
+ /**
34
+ * 生成 explore_table 工具的描述(基于数据库中的表定义)
35
+ */
36
+ private generateExploreTableDescription;
37
+ /**
38
+ * 生成 explore_table 工具的执行脚本
39
+ */
40
+ private generateExploreTableScript;
41
+ /**
42
+ * 获取所有工具(从数据库读取,不实时查询 MCP 组件)
43
+ * 每次调用时都从数据库重新加载最新配置,确保返回最新的工具列表
44
+ */
45
+ getAllTools(): Promise<MCPTool[]>;
46
+ /**
47
+ * 获取所有提示
48
+ */
49
+ getAllPrompts(): Promise<MCPPrompt[]>;
50
+ /**
51
+ * 替换消息模板中的变量(支持条件逻辑)
52
+ */
53
+ private replaceVariables;
54
+ /**
55
+ * 获取提示内容
56
+ * @param promptName 提示名称
57
+ * @param args 可选参数,用于填充 prompt 模板。如果 prompt 不需要参数,可以不传或传空对象 {}
58
+ */
59
+ getPrompt(promptName: string, args?: Record<string, any>): Promise<any>;
60
+ /**
61
+ * 从数据库检查工具是否被禁用
62
+ */
63
+ private isToolDisabledInDB;
64
+ /**
65
+ * 调用工具(实时转发到 MCP 组件)
66
+ */
67
+ callTool(toolName: string, args: any, allowDisabled?: boolean): Promise<any>;
68
+ /**
69
+ * 处理 MCP 协议请求
70
+ */
71
+ handleRequest(request: any): Promise<any>;
72
+ /**
73
+ * 关闭聚合器
74
+ * 确保所有资源正确释放
75
+ */
76
+ close(): Promise<void>;
77
+ }
78
+ //# sourceMappingURL=aggregator.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"aggregator.d.ts","sourceRoot":"","sources":["../../src/mcp/aggregator.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,KAAK,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAgB,MAAM,mBAAmB,CAAC;AAWrF,qBAAa,aAAa;IACxB,OAAO,CAAC,MAAM,CAAY;IAC1B,OAAO,CAAC,cAAc,CAAiB;IACvC,OAAO,CAAC,UAAU,CAA2C;IAC7D,OAAO,CAAC,gBAAgB,CAAkC;IAC1D,OAAO,CAAC,WAAW,CAAyB;IAC5C,OAAO,CAAC,eAAe,CAA0E;IACjG,OAAO,CAAC,aAAa,CAAkB;IACvC,OAAO,CAAC,MAAM,CAAS;IACvB,SAAgB,eAAe,EAAE,IAAI,CAAC;gBAE1B,MAAM,EAAE,SAAS,EAAE,eAAe,CAAC,EAAE;QAAE,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;KAAE,GAAG,IAAI,EAAE,MAAM,CAAC,EAAE,MAAM;IASzI;;OAEG;IACH,IAAI,WAAW,IAAI,OAAO,CAEzB;IAED;;;OAGG;IACG,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAyBjC;;OAEG;YACW,qBAAqB;IAyDnC;;OAEG;YACW,+BAA+B;IAmD7C;;OAEG;IACH,OAAO,CAAC,0BAA0B;IAkOlC;;;OAGG;IACG,WAAW,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;IA0QvC;;OAEG;IACG,aAAa,IAAI,OAAO,CAAC,SAAS,EAAE,CAAC;IAmE3C;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAIxB;;;;OAIG;IACG,SAAS,CAAC,UAAU,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,OAAO,CAAC,GAAG,CAAC;IA0G7E;;OAEG;YACW,kBAAkB;IAoBhC;;OAEG;IACG,QAAQ,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,EAAE,GAAG,EAAE,aAAa,GAAE,OAAe,GAAG,OAAO,CAAC,GAAG,CAAC;IAuRzF;;OAEG;IACG,aAAa,CAAC,OAAO,EAAE,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC;IA6K/C;;;OAGG;IACG,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;CAmC7B"}