opc-agent 1.2.1 → 1.3.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 (152) hide show
  1. package/CONTRIBUTING.md +75 -75
  2. package/README.md +235 -358
  3. package/README.zh-CN.md +415 -415
  4. package/dist/channels/web.js +256 -256
  5. package/dist/core/knowledge.d.ts +5 -0
  6. package/dist/core/knowledge.js +39 -2
  7. package/dist/deploy/hermes.js +22 -22
  8. package/dist/deploy/openclaw.js +31 -31
  9. package/dist/index.d.ts +0 -4
  10. package/dist/index.js +1 -7
  11. package/dist/providers/index.d.ts +1 -1
  12. package/dist/providers/index.js +158 -14
  13. package/dist/schema/oad.d.ts +3 -3
  14. package/dist/templates/code-reviewer.js +5 -5
  15. package/dist/templates/customer-service.js +2 -2
  16. package/dist/templates/data-analyst.js +5 -5
  17. package/dist/templates/knowledge-base.js +2 -2
  18. package/dist/templates/sales-assistant.js +4 -4
  19. package/dist/templates/teacher.js +6 -6
  20. package/docs/.vitepress/config.ts +103 -103
  21. package/docs/api/cli.md +48 -48
  22. package/docs/api/oad-schema.md +64 -64
  23. package/docs/api/sdk.md +80 -80
  24. package/docs/guide/concepts.md +51 -51
  25. package/docs/guide/configuration.md +79 -79
  26. package/docs/guide/deployment.md +42 -42
  27. package/docs/guide/getting-started.md +44 -44
  28. package/docs/guide/templates.md +28 -28
  29. package/docs/guide/testing.md +84 -84
  30. package/docs/index.md +27 -27
  31. package/docs/zh/api/cli.md +54 -54
  32. package/docs/zh/api/oad-schema.md +87 -87
  33. package/docs/zh/api/sdk.md +102 -102
  34. package/docs/zh/guide/concepts.md +104 -104
  35. package/docs/zh/guide/configuration.md +135 -135
  36. package/docs/zh/guide/deployment.md +81 -81
  37. package/docs/zh/guide/getting-started.md +82 -82
  38. package/docs/zh/guide/templates.md +84 -84
  39. package/docs/zh/guide/testing.md +88 -88
  40. package/docs/zh/index.md +27 -27
  41. package/examples/customer-service-demo/README.md +90 -90
  42. package/examples/customer-service-demo/oad.yaml +107 -107
  43. package/package.json +50 -50
  44. package/src/analytics/index.ts +66 -66
  45. package/src/channels/discord.ts +192 -192
  46. package/src/channels/email.ts +177 -177
  47. package/src/channels/feishu.ts +236 -236
  48. package/src/channels/index.ts +15 -15
  49. package/src/channels/slack.ts +160 -160
  50. package/src/channels/telegram.ts +90 -90
  51. package/src/channels/voice.ts +106 -106
  52. package/src/channels/webhook.ts +199 -199
  53. package/src/channels/websocket.ts +87 -87
  54. package/src/channels/wechat.ts +149 -149
  55. package/src/cli.ts +119 -1
  56. package/src/core/a2a.ts +143 -143
  57. package/src/core/agent.ts +152 -152
  58. package/src/core/analytics-engine.ts +186 -186
  59. package/src/core/auth.ts +57 -57
  60. package/src/core/cache.ts +141 -141
  61. package/src/core/compose.ts +77 -77
  62. package/src/core/config.ts +14 -14
  63. package/src/core/errors.ts +148 -148
  64. package/src/core/hitl.ts +138 -138
  65. package/src/core/logger.ts +57 -57
  66. package/src/core/orchestrator.ts +215 -215
  67. package/src/core/performance.ts +187 -187
  68. package/src/core/rate-limiter.ts +128 -128
  69. package/src/core/room.ts +109 -109
  70. package/src/core/runtime.ts +152 -152
  71. package/src/core/sandbox.ts +101 -101
  72. package/src/core/security.ts +171 -171
  73. package/src/core/types.ts +68 -68
  74. package/src/core/versioning.ts +106 -106
  75. package/src/core/watch.ts +178 -178
  76. package/src/core/workflow.ts +235 -235
  77. package/src/deploy/hermes.ts +156 -156
  78. package/src/deploy/openclaw.ts +200 -200
  79. package/src/i18n/index.ts +216 -216
  80. package/src/index.ts +6 -2
  81. package/src/memory/deepbrain.ts +108 -108
  82. package/src/memory/index.ts +34 -34
  83. package/src/plugins/index.ts +208 -208
  84. package/src/schema/oad.ts +154 -155
  85. package/src/skills/base.ts +16 -16
  86. package/src/skills/document.ts +100 -100
  87. package/src/skills/http.ts +35 -35
  88. package/src/skills/index.ts +27 -27
  89. package/src/skills/scheduler.ts +80 -80
  90. package/src/skills/webhook-trigger.ts +59 -59
  91. package/src/templates/code-reviewer.ts +30 -34
  92. package/src/templates/customer-service.ts +76 -80
  93. package/src/templates/data-analyst.ts +66 -70
  94. package/src/templates/executive-assistant.ts +71 -71
  95. package/src/templates/financial-advisor.ts +60 -60
  96. package/src/templates/knowledge-base.ts +27 -31
  97. package/src/templates/legal-assistant.ts +71 -71
  98. package/src/templates/sales-assistant.ts +75 -79
  99. package/src/templates/teacher.ts +75 -79
  100. package/src/testing/index.ts +181 -181
  101. package/src/tools/calculator.ts +73 -73
  102. package/src/tools/datetime.ts +149 -149
  103. package/src/tools/json-transform.ts +187 -187
  104. package/src/tools/mcp.ts +76 -76
  105. package/src/tools/text-analysis.ts +116 -116
  106. package/src/traces/index.ts +132 -0
  107. package/templates/Dockerfile +15 -15
  108. package/templates/code-reviewer/README.md +27 -27
  109. package/templates/code-reviewer/oad.yaml +41 -41
  110. package/templates/customer-service/README.md +22 -22
  111. package/templates/customer-service/oad.yaml +36 -36
  112. package/templates/docker-compose.yml +21 -21
  113. package/templates/ecommerce-assistant/README.md +45 -45
  114. package/templates/ecommerce-assistant/oad.yaml +47 -47
  115. package/templates/knowledge-base/README.md +28 -28
  116. package/templates/knowledge-base/oad.yaml +38 -38
  117. package/templates/sales-assistant/README.md +26 -26
  118. package/templates/sales-assistant/oad.yaml +43 -43
  119. package/templates/tech-support/README.md +43 -43
  120. package/templates/tech-support/oad.yaml +45 -45
  121. package/tests/a2a.test.ts +66 -66
  122. package/tests/agent.test.ts +72 -72
  123. package/tests/analytics.test.ts +50 -50
  124. package/tests/channel.test.ts +39 -39
  125. package/tests/e2e.test.ts +134 -134
  126. package/tests/errors.test.ts +83 -83
  127. package/tests/hitl.test.ts +71 -71
  128. package/tests/i18n.test.ts +41 -41
  129. package/tests/mcp.test.ts +54 -54
  130. package/tests/oad.test.ts +68 -68
  131. package/tests/performance.test.ts +115 -115
  132. package/tests/plugin.test.ts +74 -74
  133. package/tests/room.test.ts +106 -106
  134. package/tests/runtime.test.ts +42 -42
  135. package/tests/sandbox.test.ts +46 -46
  136. package/tests/security.test.ts +60 -60
  137. package/tests/templates.test.ts +77 -77
  138. package/tests/v070.test.ts +76 -76
  139. package/tests/versioning.test.ts +75 -75
  140. package/tests/voice.test.ts +61 -61
  141. package/tests/webhook.test.ts +29 -29
  142. package/tests/workflow.test.ts +143 -143
  143. package/tsconfig.json +19 -19
  144. package/vitest.config.ts +9 -9
  145. package/dist/core/streaming.d.ts +0 -56
  146. package/dist/core/streaming.js +0 -160
  147. package/dist/tools/gateway.d.ts +0 -28
  148. package/dist/tools/gateway.js +0 -177
  149. package/src/dtv/data.ts +0 -29
  150. package/src/dtv/trust.ts +0 -43
  151. package/src/dtv/value.ts +0 -47
  152. package/src/marketplace/index.ts +0 -223
package/src/core/cache.ts CHANGED
@@ -1,141 +1,141 @@
1
- /**
2
- * Caching Layer - Cache LLM responses with configurable TTL.
3
- * Hash-based key from input messages + system prompt.
4
- */
5
- import * as fs from 'fs';
6
- import * as path from 'path';
7
- import * as crypto from 'crypto';
8
-
9
- export interface CacheEntry {
10
- key: string;
11
- value: string;
12
- createdAt: number;
13
- ttlMs: number;
14
- hits: number;
15
- }
16
-
17
- export interface CacheConfig {
18
- enabled: boolean;
19
- ttlMs: number; // default TTL
20
- maxEntries: number;
21
- dataDir: string;
22
- }
23
-
24
- export class LLMCache {
25
- private cache: Map<string, CacheEntry> = new Map();
26
- private config: CacheConfig;
27
- private filePath: string;
28
- private stats = { hits: 0, misses: 0, evictions: 0 };
29
-
30
- constructor(config?: Partial<CacheConfig>) {
31
- this.config = {
32
- enabled: config?.enabled ?? true,
33
- ttlMs: config?.ttlMs ?? 3600_000, // 1 hour default
34
- maxEntries: config?.maxEntries ?? 1000,
35
- dataDir: config?.dataDir ?? '.',
36
- };
37
- this.filePath = path.join(this.config.dataDir, 'data', 'cache.json');
38
- this.load();
39
- }
40
-
41
- private load(): void {
42
- try {
43
- if (fs.existsSync(this.filePath)) {
44
- const raw = fs.readFileSync(this.filePath, 'utf-8');
45
- const entries: CacheEntry[] = JSON.parse(raw);
46
- for (const entry of entries) {
47
- if (!this.isExpired(entry)) {
48
- this.cache.set(entry.key, entry);
49
- }
50
- }
51
- }
52
- } catch {
53
- // ignore
54
- }
55
- }
56
-
57
- private save(): void {
58
- const dir = path.dirname(this.filePath);
59
- if (!fs.existsSync(dir)) fs.mkdirSync(dir, { recursive: true });
60
- const entries = Array.from(this.cache.values());
61
- fs.writeFileSync(this.filePath, JSON.stringify(entries));
62
- }
63
-
64
- private isExpired(entry: CacheEntry): boolean {
65
- return Date.now() - entry.createdAt > entry.ttlMs;
66
- }
67
-
68
- /**
69
- * Generate a cache key from messages and system prompt.
70
- */
71
- static makeKey(messages: Array<{ role: string; content: string }>, systemPrompt?: string): string {
72
- const payload = JSON.stringify({ systemPrompt, messages: messages.map(m => ({ role: m.role, content: m.content })) });
73
- return crypto.createHash('sha256').update(payload).digest('hex').slice(0, 16);
74
- }
75
-
76
- /**
77
- * Get a cached response. Returns null if not found or expired.
78
- */
79
- get(key: string): string | null {
80
- if (!this.config.enabled) return null;
81
- const entry = this.cache.get(key);
82
- if (!entry || this.isExpired(entry)) {
83
- if (entry) {
84
- this.cache.delete(key);
85
- this.stats.evictions++;
86
- }
87
- this.stats.misses++;
88
- return null;
89
- }
90
- entry.hits++;
91
- this.stats.hits++;
92
- return entry.value;
93
- }
94
-
95
- /**
96
- * Set a cached response.
97
- */
98
- set(key: string, value: string, ttlMs?: number): void {
99
- if (!this.config.enabled) return;
100
-
101
- // Evict oldest if at capacity
102
- if (this.cache.size >= this.config.maxEntries) {
103
- let oldestKey: string | null = null;
104
- let oldestTime = Infinity;
105
- for (const [k, v] of this.cache) {
106
- if (v.createdAt < oldestTime) {
107
- oldestTime = v.createdAt;
108
- oldestKey = k;
109
- }
110
- }
111
- if (oldestKey) {
112
- this.cache.delete(oldestKey);
113
- this.stats.evictions++;
114
- }
115
- }
116
-
117
- this.cache.set(key, {
118
- key,
119
- value,
120
- createdAt: Date.now(),
121
- ttlMs: ttlMs ?? this.config.ttlMs,
122
- hits: 0,
123
- });
124
- this.save();
125
- }
126
-
127
- getStats(): { hits: number; misses: number; evictions: number; size: number; hitRate: string } {
128
- const total = this.stats.hits + this.stats.misses;
129
- return {
130
- ...this.stats,
131
- size: this.cache.size,
132
- hitRate: total > 0 ? `${((this.stats.hits / total) * 100).toFixed(1)}%` : '0%',
133
- };
134
- }
135
-
136
- clear(): void {
137
- this.cache.clear();
138
- this.stats = { hits: 0, misses: 0, evictions: 0 };
139
- this.save();
140
- }
141
- }
1
+ /**
2
+ * Caching Layer - Cache LLM responses with configurable TTL.
3
+ * Hash-based key from input messages + system prompt.
4
+ */
5
+ import * as fs from 'fs';
6
+ import * as path from 'path';
7
+ import * as crypto from 'crypto';
8
+
9
+ export interface CacheEntry {
10
+ key: string;
11
+ value: string;
12
+ createdAt: number;
13
+ ttlMs: number;
14
+ hits: number;
15
+ }
16
+
17
+ export interface CacheConfig {
18
+ enabled: boolean;
19
+ ttlMs: number; // default TTL
20
+ maxEntries: number;
21
+ dataDir: string;
22
+ }
23
+
24
+ export class LLMCache {
25
+ private cache: Map<string, CacheEntry> = new Map();
26
+ private config: CacheConfig;
27
+ private filePath: string;
28
+ private stats = { hits: 0, misses: 0, evictions: 0 };
29
+
30
+ constructor(config?: Partial<CacheConfig>) {
31
+ this.config = {
32
+ enabled: config?.enabled ?? true,
33
+ ttlMs: config?.ttlMs ?? 3600_000, // 1 hour default
34
+ maxEntries: config?.maxEntries ?? 1000,
35
+ dataDir: config?.dataDir ?? '.',
36
+ };
37
+ this.filePath = path.join(this.config.dataDir, 'data', 'cache.json');
38
+ this.load();
39
+ }
40
+
41
+ private load(): void {
42
+ try {
43
+ if (fs.existsSync(this.filePath)) {
44
+ const raw = fs.readFileSync(this.filePath, 'utf-8');
45
+ const entries: CacheEntry[] = JSON.parse(raw);
46
+ for (const entry of entries) {
47
+ if (!this.isExpired(entry)) {
48
+ this.cache.set(entry.key, entry);
49
+ }
50
+ }
51
+ }
52
+ } catch {
53
+ // ignore
54
+ }
55
+ }
56
+
57
+ private save(): void {
58
+ const dir = path.dirname(this.filePath);
59
+ if (!fs.existsSync(dir)) fs.mkdirSync(dir, { recursive: true });
60
+ const entries = Array.from(this.cache.values());
61
+ fs.writeFileSync(this.filePath, JSON.stringify(entries));
62
+ }
63
+
64
+ private isExpired(entry: CacheEntry): boolean {
65
+ return Date.now() - entry.createdAt > entry.ttlMs;
66
+ }
67
+
68
+ /**
69
+ * Generate a cache key from messages and system prompt.
70
+ */
71
+ static makeKey(messages: Array<{ role: string; content: string }>, systemPrompt?: string): string {
72
+ const payload = JSON.stringify({ systemPrompt, messages: messages.map(m => ({ role: m.role, content: m.content })) });
73
+ return crypto.createHash('sha256').update(payload).digest('hex').slice(0, 16);
74
+ }
75
+
76
+ /**
77
+ * Get a cached response. Returns null if not found or expired.
78
+ */
79
+ get(key: string): string | null {
80
+ if (!this.config.enabled) return null;
81
+ const entry = this.cache.get(key);
82
+ if (!entry || this.isExpired(entry)) {
83
+ if (entry) {
84
+ this.cache.delete(key);
85
+ this.stats.evictions++;
86
+ }
87
+ this.stats.misses++;
88
+ return null;
89
+ }
90
+ entry.hits++;
91
+ this.stats.hits++;
92
+ return entry.value;
93
+ }
94
+
95
+ /**
96
+ * Set a cached response.
97
+ */
98
+ set(key: string, value: string, ttlMs?: number): void {
99
+ if (!this.config.enabled) return;
100
+
101
+ // Evict oldest if at capacity
102
+ if (this.cache.size >= this.config.maxEntries) {
103
+ let oldestKey: string | null = null;
104
+ let oldestTime = Infinity;
105
+ for (const [k, v] of this.cache) {
106
+ if (v.createdAt < oldestTime) {
107
+ oldestTime = v.createdAt;
108
+ oldestKey = k;
109
+ }
110
+ }
111
+ if (oldestKey) {
112
+ this.cache.delete(oldestKey);
113
+ this.stats.evictions++;
114
+ }
115
+ }
116
+
117
+ this.cache.set(key, {
118
+ key,
119
+ value,
120
+ createdAt: Date.now(),
121
+ ttlMs: ttlMs ?? this.config.ttlMs,
122
+ hits: 0,
123
+ });
124
+ this.save();
125
+ }
126
+
127
+ getStats(): { hits: number; misses: number; evictions: number; size: number; hitRate: string } {
128
+ const total = this.stats.hits + this.stats.misses;
129
+ return {
130
+ ...this.stats,
131
+ size: this.cache.size,
132
+ hitRate: total > 0 ? `${((this.stats.hits / total) * 100).toFixed(1)}%` : '0%',
133
+ };
134
+ }
135
+
136
+ clear(): void {
137
+ this.cache.clear();
138
+ this.stats = { hits: 0, misses: 0, evictions: 0 };
139
+ this.save();
140
+ }
141
+ }
@@ -1,77 +1,77 @@
1
- import type { AgentContext, Message } from './types';
2
-
3
- /**
4
- * Agent Composition — v0.8.0
5
- * Combine multiple agents into a pipeline: Agent A output → Agent B input.
6
- * Configurable in OAD: `compose: [agent-a, agent-b]`
7
- */
8
-
9
- export type AgentHandler = (context: AgentContext, message: Message) => Promise<Message>;
10
-
11
- export interface ComposableAgent {
12
- id: string;
13
- name: string;
14
- handler: AgentHandler;
15
- }
16
-
17
- export interface ComposeOptions {
18
- /** Stop pipeline if any agent returns empty content */
19
- stopOnEmpty?: boolean;
20
- /** Transform output between agents */
21
- transform?: (output: Message, nextAgentId: string) => Message;
22
- /** Timeout per agent in ms */
23
- timeoutMs?: number;
24
- }
25
-
26
- export class AgentPipeline {
27
- private agents: ComposableAgent[] = [];
28
- private options: ComposeOptions;
29
-
30
- constructor(agents: ComposableAgent[], options: ComposeOptions = {}) {
31
- this.agents = agents;
32
- this.options = options;
33
- }
34
-
35
- /** Run the pipeline sequentially: each agent's output becomes the next agent's input */
36
- async execute(context: AgentContext, initialMessage: Message): Promise<Message> {
37
- let currentMessage = initialMessage;
38
-
39
- for (const agent of this.agents) {
40
- if (this.options.stopOnEmpty && !currentMessage.content.trim()) {
41
- break;
42
- }
43
-
44
- // Apply transform if provided
45
- if (this.options.transform) {
46
- currentMessage = this.options.transform(currentMessage, agent.id);
47
- }
48
-
49
- if (this.options.timeoutMs) {
50
- const result = await Promise.race([
51
- agent.handler(context, currentMessage),
52
- new Promise<never>((_, reject) =>
53
- setTimeout(() => reject(new Error(`Agent ${agent.id} timed out`)), this.options.timeoutMs)
54
- ),
55
- ]);
56
- currentMessage = result;
57
- } else {
58
- currentMessage = await agent.handler(context, currentMessage);
59
- }
60
- }
61
-
62
- return currentMessage;
63
- }
64
-
65
- /** Get the pipeline agent IDs in order */
66
- getAgentIds(): string[] {
67
- return this.agents.map((a) => a.id);
68
- }
69
- }
70
-
71
- /**
72
- * Create a pipeline from an array of composable agents.
73
- * Usage in OAD: `compose: [agent-a, agent-b, agent-c]`
74
- */
75
- export function compose(agents: ComposableAgent[], options?: ComposeOptions): AgentPipeline {
76
- return new AgentPipeline(agents, options);
77
- }
1
+ import type { AgentContext, Message } from './types';
2
+
3
+ /**
4
+ * Agent Composition — v0.8.0
5
+ * Combine multiple agents into a pipeline: Agent A output → Agent B input.
6
+ * Configurable in OAD: `compose: [agent-a, agent-b]`
7
+ */
8
+
9
+ export type AgentHandler = (context: AgentContext, message: Message) => Promise<Message>;
10
+
11
+ export interface ComposableAgent {
12
+ id: string;
13
+ name: string;
14
+ handler: AgentHandler;
15
+ }
16
+
17
+ export interface ComposeOptions {
18
+ /** Stop pipeline if any agent returns empty content */
19
+ stopOnEmpty?: boolean;
20
+ /** Transform output between agents */
21
+ transform?: (output: Message, nextAgentId: string) => Message;
22
+ /** Timeout per agent in ms */
23
+ timeoutMs?: number;
24
+ }
25
+
26
+ export class AgentPipeline {
27
+ private agents: ComposableAgent[] = [];
28
+ private options: ComposeOptions;
29
+
30
+ constructor(agents: ComposableAgent[], options: ComposeOptions = {}) {
31
+ this.agents = agents;
32
+ this.options = options;
33
+ }
34
+
35
+ /** Run the pipeline sequentially: each agent's output becomes the next agent's input */
36
+ async execute(context: AgentContext, initialMessage: Message): Promise<Message> {
37
+ let currentMessage = initialMessage;
38
+
39
+ for (const agent of this.agents) {
40
+ if (this.options.stopOnEmpty && !currentMessage.content.trim()) {
41
+ break;
42
+ }
43
+
44
+ // Apply transform if provided
45
+ if (this.options.transform) {
46
+ currentMessage = this.options.transform(currentMessage, agent.id);
47
+ }
48
+
49
+ if (this.options.timeoutMs) {
50
+ const result = await Promise.race([
51
+ agent.handler(context, currentMessage),
52
+ new Promise<never>((_, reject) =>
53
+ setTimeout(() => reject(new Error(`Agent ${agent.id} timed out`)), this.options.timeoutMs)
54
+ ),
55
+ ]);
56
+ currentMessage = result;
57
+ } else {
58
+ currentMessage = await agent.handler(context, currentMessage);
59
+ }
60
+ }
61
+
62
+ return currentMessage;
63
+ }
64
+
65
+ /** Get the pipeline agent IDs in order */
66
+ getAgentIds(): string[] {
67
+ return this.agents.map((a) => a.id);
68
+ }
69
+ }
70
+
71
+ /**
72
+ * Create a pipeline from an array of composable agents.
73
+ * Usage in OAD: `compose: [agent-a, agent-b, agent-c]`
74
+ */
75
+ export function compose(agents: ComposableAgent[], options?: ComposeOptions): AgentPipeline {
76
+ return new AgentPipeline(agents, options);
77
+ }
@@ -1,14 +1,14 @@
1
- import * as fs from 'fs';
2
- import * as yaml from 'js-yaml';
3
- import { OADSchema, type OADDocument } from '../schema/oad';
4
-
5
- export function loadOAD(filePath: string): OADDocument {
6
- const raw = fs.readFileSync(filePath, 'utf-8');
7
- const ext = filePath.split('.').pop()?.toLowerCase();
8
- const data = (ext === 'yaml' || ext === 'yml') ? yaml.load(raw) : JSON.parse(raw);
9
- return OADSchema.parse(data);
10
- }
11
-
12
- export function validateOAD(data: unknown): OADDocument {
13
- return OADSchema.parse(data);
14
- }
1
+ import * as fs from 'fs';
2
+ import * as yaml from 'js-yaml';
3
+ import { OADSchema, type OADDocument } from '../schema/oad';
4
+
5
+ export function loadOAD(filePath: string): OADDocument {
6
+ const raw = fs.readFileSync(filePath, 'utf-8');
7
+ const ext = filePath.split('.').pop()?.toLowerCase();
8
+ const data = (ext === 'yaml' || ext === 'yml') ? yaml.load(raw) : JSON.parse(raw);
9
+ return OADSchema.parse(data);
10
+ }
11
+
12
+ export function validateOAD(data: unknown): OADDocument {
13
+ return OADSchema.parse(data);
14
+ }