opc-agent 1.4.0 → 1.4.1

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 (192) hide show
  1. package/CHANGELOG.md +69 -23
  2. package/CONTRIBUTING.md +60 -21
  3. package/README.md +358 -235
  4. package/README.zh-CN.md +415 -415
  5. package/dist/channels/slack.js +10 -93
  6. package/dist/channels/web.d.ts +0 -10
  7. package/dist/channels/web.js +2 -33
  8. package/dist/cli.js +60 -255
  9. package/dist/core/dashboard.d.ts +35 -0
  10. package/dist/core/dashboard.js +157 -0
  11. package/dist/core/fast-mode.d.ts +27 -0
  12. package/dist/core/fast-mode.js +59 -0
  13. package/dist/core/priority.d.ts +52 -0
  14. package/dist/core/priority.js +102 -0
  15. package/dist/core/runtime.d.ts +0 -4
  16. package/dist/core/runtime.js +0 -27
  17. package/dist/deploy/hermes.js +22 -22
  18. package/dist/deploy/openclaw.js +40 -31
  19. package/dist/index.d.ts +14 -3
  20. package/dist/index.js +20 -6
  21. package/dist/memory/cloud-storage.d.ts +40 -0
  22. package/dist/memory/cloud-storage.js +211 -0
  23. package/dist/providers/index.d.ts +1 -1
  24. package/dist/providers/index.js +1 -7
  25. package/dist/schema/oad.d.ts +2 -1
  26. package/dist/templates/code-reviewer.d.ts +8 -0
  27. package/dist/templates/code-reviewer.js +9 -5
  28. package/dist/templates/customer-service.d.ts +8 -0
  29. package/dist/templates/customer-service.js +6 -2
  30. package/dist/templates/data-analyst.d.ts +8 -0
  31. package/dist/templates/data-analyst.js +9 -5
  32. package/dist/templates/knowledge-base.d.ts +8 -0
  33. package/dist/templates/knowledge-base.js +6 -2
  34. package/dist/templates/sales-assistant.d.ts +8 -0
  35. package/dist/templates/sales-assistant.js +8 -4
  36. package/dist/templates/teacher.d.ts +8 -0
  37. package/dist/templates/teacher.js +10 -6
  38. package/docs/.vitepress/config.ts +103 -103
  39. package/docs/api/cli.md +48 -48
  40. package/docs/api/oad-schema.md +64 -64
  41. package/docs/api/sdk.md +80 -80
  42. package/docs/guide/concepts.md +51 -51
  43. package/docs/guide/configuration.md +79 -79
  44. package/docs/guide/deployment.md +42 -42
  45. package/docs/guide/getting-started.md +44 -44
  46. package/docs/guide/templates.md +28 -28
  47. package/docs/guide/testing.md +84 -84
  48. package/docs/index.md +27 -27
  49. package/docs/zh/api/cli.md +54 -54
  50. package/docs/zh/api/oad-schema.md +87 -87
  51. package/docs/zh/api/sdk.md +102 -102
  52. package/docs/zh/guide/concepts.md +104 -104
  53. package/docs/zh/guide/configuration.md +135 -135
  54. package/docs/zh/guide/deployment.md +81 -81
  55. package/docs/zh/guide/getting-started.md +82 -82
  56. package/docs/zh/guide/templates.md +84 -84
  57. package/docs/zh/guide/testing.md +88 -88
  58. package/docs/zh/index.md +27 -27
  59. package/examples/customer-service-demo/README.md +90 -90
  60. package/examples/customer-service-demo/oad.yaml +107 -107
  61. package/package.json +1 -1
  62. package/src/analytics/index.ts +66 -66
  63. package/src/channels/discord.ts +192 -192
  64. package/src/channels/email.ts +177 -177
  65. package/src/channels/feishu.ts +236 -236
  66. package/src/channels/index.ts +15 -15
  67. package/src/channels/slack.ts +160 -217
  68. package/src/channels/telegram.ts +90 -90
  69. package/src/channels/voice.ts +106 -106
  70. package/src/channels/web.ts +2 -38
  71. package/src/channels/webhook.ts +199 -199
  72. package/src/channels/websocket.ts +87 -87
  73. package/src/channels/wechat.ts +149 -149
  74. package/src/cli.ts +58 -282
  75. package/src/core/a2a.ts +143 -143
  76. package/src/core/agent.ts +152 -152
  77. package/src/core/analytics-engine.ts +186 -186
  78. package/src/core/auth.ts +57 -57
  79. package/src/core/cache.ts +141 -141
  80. package/src/core/compose.ts +77 -77
  81. package/src/core/config.ts +14 -14
  82. package/src/core/dashboard.ts +219 -0
  83. package/src/core/errors.ts +148 -148
  84. package/src/core/fast-mode.ts +75 -0
  85. package/src/core/hitl.ts +138 -138
  86. package/src/core/logger.ts +57 -57
  87. package/src/core/orchestrator.ts +215 -215
  88. package/src/core/performance.ts +187 -187
  89. package/src/core/priority.ts +140 -0
  90. package/src/core/rate-limiter.ts +128 -128
  91. package/src/core/room.ts +109 -109
  92. package/src/core/runtime.ts +152 -183
  93. package/src/core/sandbox.ts +101 -101
  94. package/src/core/security.ts +171 -171
  95. package/src/core/types.ts +68 -68
  96. package/src/core/versioning.ts +106 -106
  97. package/src/core/watch.ts +178 -178
  98. package/src/core/workflow.ts +235 -235
  99. package/src/deploy/hermes.ts +156 -156
  100. package/src/deploy/openclaw.ts +200 -190
  101. package/src/dtv/data.ts +29 -0
  102. package/src/dtv/trust.ts +43 -0
  103. package/src/dtv/value.ts +47 -0
  104. package/src/i18n/index.ts +216 -216
  105. package/src/index.ts +16 -3
  106. package/src/marketplace/index.ts +223 -0
  107. package/src/memory/cloud-storage.ts +217 -0
  108. package/src/memory/deepbrain.ts +108 -108
  109. package/src/memory/index.ts +34 -34
  110. package/src/plugins/index.ts +208 -208
  111. package/src/providers/index.ts +1 -9
  112. package/src/schema/oad.ts +155 -154
  113. package/src/skills/base.ts +16 -16
  114. package/src/skills/document.ts +100 -100
  115. package/src/skills/http.ts +35 -35
  116. package/src/skills/index.ts +27 -27
  117. package/src/skills/scheduler.ts +80 -80
  118. package/src/skills/webhook-trigger.ts +59 -59
  119. package/src/templates/code-reviewer.ts +34 -30
  120. package/src/templates/customer-service.ts +80 -76
  121. package/src/templates/data-analyst.ts +70 -66
  122. package/src/templates/executive-assistant.ts +71 -71
  123. package/src/templates/financial-advisor.ts +60 -60
  124. package/src/templates/knowledge-base.ts +31 -27
  125. package/src/templates/legal-assistant.ts +71 -71
  126. package/src/templates/sales-assistant.ts +79 -75
  127. package/src/templates/teacher.ts +79 -75
  128. package/src/testing/index.ts +181 -181
  129. package/src/tools/calculator.ts +73 -73
  130. package/src/tools/datetime.ts +149 -149
  131. package/src/tools/json-transform.ts +187 -187
  132. package/src/tools/mcp.ts +76 -76
  133. package/src/tools/text-analysis.ts +116 -116
  134. package/templates/Dockerfile +15 -15
  135. package/templates/code-reviewer/README.md +27 -27
  136. package/templates/code-reviewer/oad.yaml +41 -41
  137. package/templates/customer-service/README.md +22 -22
  138. package/templates/customer-service/oad.yaml +36 -36
  139. package/templates/docker-compose.yml +21 -21
  140. package/templates/ecommerce-assistant/README.md +45 -45
  141. package/templates/ecommerce-assistant/oad.yaml +47 -47
  142. package/templates/knowledge-base/README.md +28 -28
  143. package/templates/knowledge-base/oad.yaml +38 -38
  144. package/templates/sales-assistant/README.md +26 -26
  145. package/templates/sales-assistant/oad.yaml +43 -43
  146. package/templates/tech-support/README.md +43 -43
  147. package/templates/tech-support/oad.yaml +45 -45
  148. package/tests/a2a.test.ts +66 -66
  149. package/tests/agent.test.ts +72 -72
  150. package/tests/analytics.test.ts +50 -50
  151. package/tests/channel.test.ts +39 -39
  152. package/tests/e2e.test.ts +134 -134
  153. package/tests/errors.test.ts +83 -83
  154. package/tests/hitl.test.ts +71 -71
  155. package/tests/i18n.test.ts +41 -41
  156. package/tests/mcp.test.ts +54 -54
  157. package/tests/oad.test.ts +68 -68
  158. package/tests/performance.test.ts +115 -115
  159. package/tests/plugin.test.ts +74 -74
  160. package/tests/room.test.ts +106 -106
  161. package/tests/runtime.test.ts +42 -42
  162. package/tests/sandbox.test.ts +46 -46
  163. package/tests/security.test.ts +60 -60
  164. package/tests/templates.test.ts +77 -77
  165. package/tests/v070.test.ts +76 -76
  166. package/tests/versioning.test.ts +75 -75
  167. package/tests/voice.test.ts +61 -61
  168. package/tests/webhook.test.ts +29 -29
  169. package/tests/workflow.test.ts +143 -143
  170. package/tsconfig.json +19 -19
  171. package/vitest.config.ts +9 -9
  172. package/.github/ISSUE_TEMPLATE/bug_report.md +0 -20
  173. package/.github/ISSUE_TEMPLATE/feature_request.md +0 -14
  174. package/.github/PULL_REQUEST_TEMPLATE.md +0 -13
  175. package/.github/workflows/ci.yml +0 -24
  176. package/dist/traces/index.d.ts +0 -49
  177. package/dist/traces/index.js +0 -102
  178. package/examples/README.md +0 -22
  179. package/examples/basic-agent.ts +0 -90
  180. package/examples/brain-integration.ts +0 -71
  181. package/examples/multi-channel.ts +0 -74
  182. package/src/traces/index.ts +0 -132
  183. package/test-agent/Dockerfile +0 -9
  184. package/test-agent/README.md +0 -50
  185. package/test-agent/agent.yaml +0 -23
  186. package/test-agent/docker-compose.yml +0 -11
  187. package/test-agent/oad.yaml +0 -31
  188. package/test-agent/package-lock.json +0 -1492
  189. package/test-agent/package.json +0 -18
  190. package/test-agent/src/index.ts +0 -24
  191. package/test-agent/src/skills/echo.ts +0 -15
  192. package/test-agent/tsconfig.json +0 -25
@@ -1,187 +1,187 @@
1
- import { Logger } from '../core/logger';
2
-
3
- // ── Connection Pool ─────────────────────────────────────────
4
-
5
- export interface PooledConnection {
6
- id: string;
7
- provider: string;
8
- createdAt: number;
9
- lastUsedAt: number;
10
- inUse: boolean;
11
- }
12
-
13
- export class ConnectionPool {
14
- private pool: Map<string, PooledConnection[]> = new Map();
15
- private maxPerProvider: number;
16
- private ttlMs: number;
17
- private logger = new Logger('connection-pool');
18
-
19
- constructor(maxPerProvider = 5, ttlMs = 300000) {
20
- this.maxPerProvider = maxPerProvider;
21
- this.ttlMs = ttlMs;
22
- }
23
-
24
- acquire(provider: string): PooledConnection {
25
- const connections = this.pool.get(provider) ?? [];
26
-
27
- // Cleanup expired
28
- const now = Date.now();
29
- const active = connections.filter(c => now - c.createdAt < this.ttlMs);
30
-
31
- // Find available
32
- const available = active.find(c => !c.inUse);
33
- if (available) {
34
- available.inUse = true;
35
- available.lastUsedAt = now;
36
- return available;
37
- }
38
-
39
- // Create new if under limit
40
- if (active.length < this.maxPerProvider) {
41
- const conn: PooledConnection = {
42
- id: `conn_${now}_${Math.random().toString(36).slice(2, 8)}`,
43
- provider,
44
- createdAt: now,
45
- lastUsedAt: now,
46
- inUse: true,
47
- };
48
- active.push(conn);
49
- this.pool.set(provider, active);
50
- return conn;
51
- }
52
-
53
- // Wait for one (return oldest used for now)
54
- const oldest = active.sort((a, b) => a.lastUsedAt - b.lastUsedAt)[0];
55
- oldest.inUse = true;
56
- oldest.lastUsedAt = now;
57
- return oldest;
58
- }
59
-
60
- release(id: string): void {
61
- for (const connections of this.pool.values()) {
62
- const conn = connections.find(c => c.id === id);
63
- if (conn) {
64
- conn.inUse = false;
65
- return;
66
- }
67
- }
68
- }
69
-
70
- getStats(): Record<string, { total: number; inUse: number }> {
71
- const stats: Record<string, { total: number; inUse: number }> = {};
72
- for (const [provider, connections] of this.pool) {
73
- stats[provider] = {
74
- total: connections.length,
75
- inUse: connections.filter(c => c.inUse).length,
76
- };
77
- }
78
- return stats;
79
- }
80
-
81
- drain(): void {
82
- this.pool.clear();
83
- }
84
- }
85
-
86
- // ── Request Batcher ─────────────────────────────────────────
87
-
88
- export interface BatchRequest<T> {
89
- payload: T;
90
- resolve: (result: unknown) => void;
91
- reject: (error: Error) => void;
92
- }
93
-
94
- export class RequestBatcher<T> {
95
- private queue: BatchRequest<T>[] = [];
96
- private timer: ReturnType<typeof setTimeout> | null = null;
97
- private maxBatchSize: number;
98
- private delayMs: number;
99
- private processor: (batch: T[]) => Promise<unknown[]>;
100
- private logger = new Logger('batcher');
101
-
102
- constructor(
103
- processor: (batch: T[]) => Promise<unknown[]>,
104
- maxBatchSize = 10,
105
- delayMs = 50,
106
- ) {
107
- this.processor = processor;
108
- this.maxBatchSize = maxBatchSize;
109
- this.delayMs = delayMs;
110
- }
111
-
112
- add(payload: T): Promise<unknown> {
113
- return new Promise((resolve, reject) => {
114
- this.queue.push({ payload, resolve, reject });
115
-
116
- if (this.queue.length >= this.maxBatchSize) {
117
- this.flush();
118
- } else if (!this.timer) {
119
- this.timer = setTimeout(() => this.flush(), this.delayMs);
120
- }
121
- });
122
- }
123
-
124
- async flush(): Promise<void> {
125
- if (this.timer) {
126
- clearTimeout(this.timer);
127
- this.timer = null;
128
- }
129
-
130
- if (this.queue.length === 0) return;
131
-
132
- const batch = this.queue.splice(0, this.maxBatchSize);
133
- try {
134
- const results = await this.processor(batch.map(b => b.payload));
135
- batch.forEach((req, i) => req.resolve(results[i]));
136
- } catch (err) {
137
- batch.forEach(req => req.reject(err as Error));
138
- }
139
- }
140
-
141
- get pending(): number {
142
- return this.queue.length;
143
- }
144
- }
145
-
146
- // ── Lazy Loader ─────────────────────────────────────────────
147
-
148
- export class LazyLoader<T> {
149
- private cache: Map<string, T> = new Map();
150
- private loaders: Map<string, () => Promise<T>> = new Map();
151
-
152
- register(name: string, loader: () => Promise<T>): void {
153
- this.loaders.set(name, loader);
154
- }
155
-
156
- async get(name: string): Promise<T> {
157
- const cached = this.cache.get(name);
158
- if (cached) return cached;
159
-
160
- const loader = this.loaders.get(name);
161
- if (!loader) throw new Error(`No loader registered for "${name}"`);
162
-
163
- const instance = await loader();
164
- this.cache.set(name, instance);
165
- return instance;
166
- }
167
-
168
- isLoaded(name: string): boolean {
169
- return this.cache.has(name);
170
- }
171
-
172
- evict(name: string): void {
173
- this.cache.delete(name);
174
- }
175
-
176
- clear(): void {
177
- this.cache.clear();
178
- }
179
-
180
- get loadedCount(): number {
181
- return this.cache.size;
182
- }
183
-
184
- get registeredCount(): number {
185
- return this.loaders.size;
186
- }
187
- }
1
+ import { Logger } from '../core/logger';
2
+
3
+ // ── Connection Pool ─────────────────────────────────────────
4
+
5
+ export interface PooledConnection {
6
+ id: string;
7
+ provider: string;
8
+ createdAt: number;
9
+ lastUsedAt: number;
10
+ inUse: boolean;
11
+ }
12
+
13
+ export class ConnectionPool {
14
+ private pool: Map<string, PooledConnection[]> = new Map();
15
+ private maxPerProvider: number;
16
+ private ttlMs: number;
17
+ private logger = new Logger('connection-pool');
18
+
19
+ constructor(maxPerProvider = 5, ttlMs = 300000) {
20
+ this.maxPerProvider = maxPerProvider;
21
+ this.ttlMs = ttlMs;
22
+ }
23
+
24
+ acquire(provider: string): PooledConnection {
25
+ const connections = this.pool.get(provider) ?? [];
26
+
27
+ // Cleanup expired
28
+ const now = Date.now();
29
+ const active = connections.filter(c => now - c.createdAt < this.ttlMs);
30
+
31
+ // Find available
32
+ const available = active.find(c => !c.inUse);
33
+ if (available) {
34
+ available.inUse = true;
35
+ available.lastUsedAt = now;
36
+ return available;
37
+ }
38
+
39
+ // Create new if under limit
40
+ if (active.length < this.maxPerProvider) {
41
+ const conn: PooledConnection = {
42
+ id: `conn_${now}_${Math.random().toString(36).slice(2, 8)}`,
43
+ provider,
44
+ createdAt: now,
45
+ lastUsedAt: now,
46
+ inUse: true,
47
+ };
48
+ active.push(conn);
49
+ this.pool.set(provider, active);
50
+ return conn;
51
+ }
52
+
53
+ // Wait for one (return oldest used for now)
54
+ const oldest = active.sort((a, b) => a.lastUsedAt - b.lastUsedAt)[0];
55
+ oldest.inUse = true;
56
+ oldest.lastUsedAt = now;
57
+ return oldest;
58
+ }
59
+
60
+ release(id: string): void {
61
+ for (const connections of this.pool.values()) {
62
+ const conn = connections.find(c => c.id === id);
63
+ if (conn) {
64
+ conn.inUse = false;
65
+ return;
66
+ }
67
+ }
68
+ }
69
+
70
+ getStats(): Record<string, { total: number; inUse: number }> {
71
+ const stats: Record<string, { total: number; inUse: number }> = {};
72
+ for (const [provider, connections] of this.pool) {
73
+ stats[provider] = {
74
+ total: connections.length,
75
+ inUse: connections.filter(c => c.inUse).length,
76
+ };
77
+ }
78
+ return stats;
79
+ }
80
+
81
+ drain(): void {
82
+ this.pool.clear();
83
+ }
84
+ }
85
+
86
+ // ── Request Batcher ─────────────────────────────────────────
87
+
88
+ export interface BatchRequest<T> {
89
+ payload: T;
90
+ resolve: (result: unknown) => void;
91
+ reject: (error: Error) => void;
92
+ }
93
+
94
+ export class RequestBatcher<T> {
95
+ private queue: BatchRequest<T>[] = [];
96
+ private timer: ReturnType<typeof setTimeout> | null = null;
97
+ private maxBatchSize: number;
98
+ private delayMs: number;
99
+ private processor: (batch: T[]) => Promise<unknown[]>;
100
+ private logger = new Logger('batcher');
101
+
102
+ constructor(
103
+ processor: (batch: T[]) => Promise<unknown[]>,
104
+ maxBatchSize = 10,
105
+ delayMs = 50,
106
+ ) {
107
+ this.processor = processor;
108
+ this.maxBatchSize = maxBatchSize;
109
+ this.delayMs = delayMs;
110
+ }
111
+
112
+ add(payload: T): Promise<unknown> {
113
+ return new Promise((resolve, reject) => {
114
+ this.queue.push({ payload, resolve, reject });
115
+
116
+ if (this.queue.length >= this.maxBatchSize) {
117
+ this.flush();
118
+ } else if (!this.timer) {
119
+ this.timer = setTimeout(() => this.flush(), this.delayMs);
120
+ }
121
+ });
122
+ }
123
+
124
+ async flush(): Promise<void> {
125
+ if (this.timer) {
126
+ clearTimeout(this.timer);
127
+ this.timer = null;
128
+ }
129
+
130
+ if (this.queue.length === 0) return;
131
+
132
+ const batch = this.queue.splice(0, this.maxBatchSize);
133
+ try {
134
+ const results = await this.processor(batch.map(b => b.payload));
135
+ batch.forEach((req, i) => req.resolve(results[i]));
136
+ } catch (err) {
137
+ batch.forEach(req => req.reject(err as Error));
138
+ }
139
+ }
140
+
141
+ get pending(): number {
142
+ return this.queue.length;
143
+ }
144
+ }
145
+
146
+ // ── Lazy Loader ─────────────────────────────────────────────
147
+
148
+ export class LazyLoader<T> {
149
+ private cache: Map<string, T> = new Map();
150
+ private loaders: Map<string, () => Promise<T>> = new Map();
151
+
152
+ register(name: string, loader: () => Promise<T>): void {
153
+ this.loaders.set(name, loader);
154
+ }
155
+
156
+ async get(name: string): Promise<T> {
157
+ const cached = this.cache.get(name);
158
+ if (cached) return cached;
159
+
160
+ const loader = this.loaders.get(name);
161
+ if (!loader) throw new Error(`No loader registered for "${name}"`);
162
+
163
+ const instance = await loader();
164
+ this.cache.set(name, instance);
165
+ return instance;
166
+ }
167
+
168
+ isLoaded(name: string): boolean {
169
+ return this.cache.has(name);
170
+ }
171
+
172
+ evict(name: string): void {
173
+ this.cache.delete(name);
174
+ }
175
+
176
+ clear(): void {
177
+ this.cache.clear();
178
+ }
179
+
180
+ get loadedCount(): number {
181
+ return this.cache.size;
182
+ }
183
+
184
+ get registeredCount(): number {
185
+ return this.loaders.size;
186
+ }
187
+ }
@@ -0,0 +1,140 @@
1
+ // ─── Priority / Fast Mode ────────────────────────────────────
2
+ // Route requests through provider priority tiers for lower latency.
3
+ // Toggle via config or runtime command.
4
+
5
+ export interface PriorityConfig {
6
+ /** Enable priority mode (default: false) */
7
+ enabled: boolean;
8
+ /** Provider-specific priority settings */
9
+ providers?: PriorityProviderConfig[];
10
+ /** Default priority tier */
11
+ defaultTier?: PriorityTier;
12
+ }
13
+
14
+ export type PriorityTier = 'standard' | 'fast' | 'batch';
15
+
16
+ export interface PriorityProviderConfig {
17
+ provider: string;
18
+ tier: PriorityTier;
19
+ /** Custom endpoint override for priority routing */
20
+ endpoint?: string;
21
+ /** Supported models for this tier */
22
+ models?: string[];
23
+ }
24
+
25
+ interface PriorityHeaders {
26
+ [key: string]: string;
27
+ }
28
+
29
+ // Known priority-capable providers and their routing
30
+ const PROVIDER_PRIORITY_MAP: Record<string, {
31
+ headerKey: string;
32
+ headerValue: Record<PriorityTier, string>;
33
+ supportedModels: string[];
34
+ }> = {
35
+ openai: {
36
+ headerKey: 'X-OpenAI-Processing-Priority',
37
+ headerValue: { fast: 'priority', standard: 'auto', batch: 'batch' },
38
+ supportedModels: ['gpt-5', 'gpt-5.4', 'gpt-4.1', 'codex-*', 'o3-*', 'o4-mini*'],
39
+ },
40
+ anthropic: {
41
+ headerKey: 'anthropic-priority',
42
+ headerValue: { fast: 'high', standard: 'normal', batch: 'low' },
43
+ supportedModels: ['claude-opus-*', 'claude-sonnet-*', 'claude-4*'],
44
+ },
45
+ google: {
46
+ headerKey: 'X-Goog-Priority',
47
+ headerValue: { fast: 'high', standard: 'normal', batch: 'low' },
48
+ supportedModels: ['gemini-2.5-*', 'gemini-3-*'],
49
+ },
50
+ };
51
+
52
+ export class PriorityRouter {
53
+ private config: PriorityConfig;
54
+ private runtimeTier: PriorityTier;
55
+
56
+ constructor(config: PriorityConfig) {
57
+ this.config = config;
58
+ this.runtimeTier = config.defaultTier ?? 'standard';
59
+ }
60
+
61
+ /** Toggle fast mode on/off at runtime */
62
+ toggle(): PriorityTier {
63
+ this.runtimeTier = this.runtimeTier === 'fast' ? 'standard' : 'fast';
64
+ return this.runtimeTier;
65
+ }
66
+
67
+ /** Set specific tier */
68
+ setTier(tier: PriorityTier): void {
69
+ this.runtimeTier = tier;
70
+ }
71
+
72
+ /** Get current tier */
73
+ getTier(): PriorityTier {
74
+ return this.runtimeTier;
75
+ }
76
+
77
+ /** Check if fast mode is active */
78
+ isFast(): boolean {
79
+ return this.runtimeTier === 'fast';
80
+ }
81
+
82
+ /**
83
+ * Get priority headers for a provider + model combination.
84
+ * Returns empty object if provider doesn't support priority or model isn't eligible.
85
+ */
86
+ getHeaders(provider: string, model: string): PriorityHeaders {
87
+ if (!this.config.enabled) return {};
88
+
89
+ const tier = this.getEffectiveTier(provider);
90
+ if (tier === 'standard') return {};
91
+
92
+ const providerMap = PROVIDER_PRIORITY_MAP[provider.toLowerCase()];
93
+ if (!providerMap) return {};
94
+
95
+ // Check model eligibility
96
+ if (!this.isModelEligible(providerMap.supportedModels, model)) return {};
97
+
98
+ return { [providerMap.headerKey]: providerMap.headerValue[tier] };
99
+ }
100
+
101
+ /**
102
+ * Get effective endpoint for a provider, allowing priority-specific routing.
103
+ */
104
+ getEndpoint(provider: string, defaultEndpoint: string): string {
105
+ const providerConfig = this.config.providers?.find(
106
+ (p) => p.provider.toLowerCase() === provider.toLowerCase()
107
+ );
108
+ if (providerConfig?.endpoint && this.runtimeTier === 'fast') {
109
+ return providerConfig.endpoint;
110
+ }
111
+ return defaultEndpoint;
112
+ }
113
+
114
+ private getEffectiveTier(provider: string): PriorityTier {
115
+ // Check provider-specific override first
116
+ const providerConfig = this.config.providers?.find(
117
+ (p) => p.provider.toLowerCase() === provider.toLowerCase()
118
+ );
119
+ if (providerConfig) return providerConfig.tier;
120
+ return this.runtimeTier;
121
+ }
122
+
123
+ private isModelEligible(patterns: string[], model: string): boolean {
124
+ return patterns.some((pattern) => {
125
+ if (pattern.endsWith('*')) {
126
+ return model.startsWith(pattern.slice(0, -1));
127
+ }
128
+ return model === pattern;
129
+ });
130
+ }
131
+
132
+ /** Status summary for dashboard / CLI */
133
+ status(): { tier: PriorityTier; enabled: boolean; providers: string[] } {
134
+ return {
135
+ tier: this.runtimeTier,
136
+ enabled: this.config.enabled,
137
+ providers: Object.keys(PROVIDER_PRIORITY_MAP),
138
+ };
139
+ }
140
+ }