mcp-rubber-duck 1.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 (184) hide show
  1. package/.dockerignore +19 -0
  2. package/.env.desktop.example +145 -0
  3. package/.env.example +45 -0
  4. package/.env.pi.example +106 -0
  5. package/.env.template +165 -0
  6. package/.eslintrc.json +40 -0
  7. package/.github/ISSUE_TEMPLATE/bug_report.md +65 -0
  8. package/.github/ISSUE_TEMPLATE/feature_request.md +58 -0
  9. package/.github/ISSUE_TEMPLATE/question.md +67 -0
  10. package/.github/pull_request_template.md +111 -0
  11. package/.github/workflows/docker-build.yml +138 -0
  12. package/.github/workflows/release.yml +182 -0
  13. package/.github/workflows/security.yml +141 -0
  14. package/.github/workflows/semantic-release.yml +89 -0
  15. package/.prettierrc +10 -0
  16. package/.releaserc.json +66 -0
  17. package/CHANGELOG.md +95 -0
  18. package/CONTRIBUTING.md +242 -0
  19. package/Dockerfile +62 -0
  20. package/LICENSE +21 -0
  21. package/README.md +803 -0
  22. package/audit-ci.json +8 -0
  23. package/config/claude_desktop.json +14 -0
  24. package/config/config.example.json +91 -0
  25. package/dist/config/config.d.ts +51 -0
  26. package/dist/config/config.d.ts.map +1 -0
  27. package/dist/config/config.js +301 -0
  28. package/dist/config/config.js.map +1 -0
  29. package/dist/config/types.d.ts +356 -0
  30. package/dist/config/types.d.ts.map +1 -0
  31. package/dist/config/types.js +41 -0
  32. package/dist/config/types.js.map +1 -0
  33. package/dist/index.d.ts +3 -0
  34. package/dist/index.d.ts.map +1 -0
  35. package/dist/index.js +109 -0
  36. package/dist/index.js.map +1 -0
  37. package/dist/providers/duck-provider-enhanced.d.ts +29 -0
  38. package/dist/providers/duck-provider-enhanced.d.ts.map +1 -0
  39. package/dist/providers/duck-provider-enhanced.js +230 -0
  40. package/dist/providers/duck-provider-enhanced.js.map +1 -0
  41. package/dist/providers/enhanced-manager.d.ts +54 -0
  42. package/dist/providers/enhanced-manager.d.ts.map +1 -0
  43. package/dist/providers/enhanced-manager.js +217 -0
  44. package/dist/providers/enhanced-manager.js.map +1 -0
  45. package/dist/providers/manager.d.ts +28 -0
  46. package/dist/providers/manager.d.ts.map +1 -0
  47. package/dist/providers/manager.js +204 -0
  48. package/dist/providers/manager.js.map +1 -0
  49. package/dist/providers/provider.d.ts +29 -0
  50. package/dist/providers/provider.d.ts.map +1 -0
  51. package/dist/providers/provider.js +179 -0
  52. package/dist/providers/provider.js.map +1 -0
  53. package/dist/providers/types.d.ts +69 -0
  54. package/dist/providers/types.d.ts.map +1 -0
  55. package/dist/providers/types.js +2 -0
  56. package/dist/providers/types.js.map +1 -0
  57. package/dist/server.d.ts +24 -0
  58. package/dist/server.d.ts.map +1 -0
  59. package/dist/server.js +501 -0
  60. package/dist/server.js.map +1 -0
  61. package/dist/services/approval.d.ts +44 -0
  62. package/dist/services/approval.d.ts.map +1 -0
  63. package/dist/services/approval.js +159 -0
  64. package/dist/services/approval.js.map +1 -0
  65. package/dist/services/cache.d.ts +21 -0
  66. package/dist/services/cache.d.ts.map +1 -0
  67. package/dist/services/cache.js +63 -0
  68. package/dist/services/cache.js.map +1 -0
  69. package/dist/services/conversation.d.ts +24 -0
  70. package/dist/services/conversation.d.ts.map +1 -0
  71. package/dist/services/conversation.js +108 -0
  72. package/dist/services/conversation.js.map +1 -0
  73. package/dist/services/function-bridge.d.ts +41 -0
  74. package/dist/services/function-bridge.d.ts.map +1 -0
  75. package/dist/services/function-bridge.js +259 -0
  76. package/dist/services/function-bridge.js.map +1 -0
  77. package/dist/services/health.d.ts +17 -0
  78. package/dist/services/health.d.ts.map +1 -0
  79. package/dist/services/health.js +77 -0
  80. package/dist/services/health.js.map +1 -0
  81. package/dist/services/mcp-client-manager.d.ts +49 -0
  82. package/dist/services/mcp-client-manager.d.ts.map +1 -0
  83. package/dist/services/mcp-client-manager.js +279 -0
  84. package/dist/services/mcp-client-manager.js.map +1 -0
  85. package/dist/tools/approve-mcp-request.d.ts +9 -0
  86. package/dist/tools/approve-mcp-request.d.ts.map +1 -0
  87. package/dist/tools/approve-mcp-request.js +111 -0
  88. package/dist/tools/approve-mcp-request.js.map +1 -0
  89. package/dist/tools/ask-duck.d.ts +9 -0
  90. package/dist/tools/ask-duck.d.ts.map +1 -0
  91. package/dist/tools/ask-duck.js +43 -0
  92. package/dist/tools/ask-duck.js.map +1 -0
  93. package/dist/tools/chat-duck.d.ts +9 -0
  94. package/dist/tools/chat-duck.d.ts.map +1 -0
  95. package/dist/tools/chat-duck.js +57 -0
  96. package/dist/tools/chat-duck.js.map +1 -0
  97. package/dist/tools/clear-conversations.d.ts +8 -0
  98. package/dist/tools/clear-conversations.d.ts.map +1 -0
  99. package/dist/tools/clear-conversations.js +17 -0
  100. package/dist/tools/clear-conversations.js.map +1 -0
  101. package/dist/tools/compare-ducks.d.ts +8 -0
  102. package/dist/tools/compare-ducks.d.ts.map +1 -0
  103. package/dist/tools/compare-ducks.js +49 -0
  104. package/dist/tools/compare-ducks.js.map +1 -0
  105. package/dist/tools/duck-council.d.ts +8 -0
  106. package/dist/tools/duck-council.d.ts.map +1 -0
  107. package/dist/tools/duck-council.js +69 -0
  108. package/dist/tools/duck-council.js.map +1 -0
  109. package/dist/tools/get-pending-approvals.d.ts +15 -0
  110. package/dist/tools/get-pending-approvals.d.ts.map +1 -0
  111. package/dist/tools/get-pending-approvals.js +74 -0
  112. package/dist/tools/get-pending-approvals.js.map +1 -0
  113. package/dist/tools/list-ducks.d.ts +9 -0
  114. package/dist/tools/list-ducks.d.ts.map +1 -0
  115. package/dist/tools/list-ducks.js +47 -0
  116. package/dist/tools/list-ducks.js.map +1 -0
  117. package/dist/tools/list-models.d.ts +8 -0
  118. package/dist/tools/list-models.d.ts.map +1 -0
  119. package/dist/tools/list-models.js +72 -0
  120. package/dist/tools/list-models.js.map +1 -0
  121. package/dist/tools/mcp-status.d.ts +17 -0
  122. package/dist/tools/mcp-status.d.ts.map +1 -0
  123. package/dist/tools/mcp-status.js +100 -0
  124. package/dist/tools/mcp-status.js.map +1 -0
  125. package/dist/utils/ascii-art.d.ts +19 -0
  126. package/dist/utils/ascii-art.d.ts.map +1 -0
  127. package/dist/utils/ascii-art.js +73 -0
  128. package/dist/utils/ascii-art.js.map +1 -0
  129. package/dist/utils/logger.d.ts +3 -0
  130. package/dist/utils/logger.d.ts.map +1 -0
  131. package/dist/utils/logger.js +86 -0
  132. package/dist/utils/logger.js.map +1 -0
  133. package/dist/utils/safe-logger.d.ts +23 -0
  134. package/dist/utils/safe-logger.d.ts.map +1 -0
  135. package/dist/utils/safe-logger.js +145 -0
  136. package/dist/utils/safe-logger.js.map +1 -0
  137. package/docker-compose.yml +161 -0
  138. package/jest.config.js +26 -0
  139. package/package.json +65 -0
  140. package/scripts/build-multiarch.sh +290 -0
  141. package/scripts/deploy-raspbian.sh +410 -0
  142. package/scripts/deploy.sh +322 -0
  143. package/scripts/gh-deploy.sh +343 -0
  144. package/scripts/setup-docker-raspbian.sh +530 -0
  145. package/server.json +8 -0
  146. package/src/config/config.ts +357 -0
  147. package/src/config/types.ts +89 -0
  148. package/src/index.ts +114 -0
  149. package/src/providers/duck-provider-enhanced.ts +294 -0
  150. package/src/providers/enhanced-manager.ts +290 -0
  151. package/src/providers/manager.ts +257 -0
  152. package/src/providers/provider.ts +207 -0
  153. package/src/providers/types.ts +78 -0
  154. package/src/server.ts +603 -0
  155. package/src/services/approval.ts +225 -0
  156. package/src/services/cache.ts +79 -0
  157. package/src/services/conversation.ts +146 -0
  158. package/src/services/function-bridge.ts +329 -0
  159. package/src/services/health.ts +107 -0
  160. package/src/services/mcp-client-manager.ts +362 -0
  161. package/src/tools/approve-mcp-request.ts +126 -0
  162. package/src/tools/ask-duck.ts +74 -0
  163. package/src/tools/chat-duck.ts +82 -0
  164. package/src/tools/clear-conversations.ts +24 -0
  165. package/src/tools/compare-ducks.ts +67 -0
  166. package/src/tools/duck-council.ts +88 -0
  167. package/src/tools/get-pending-approvals.ts +90 -0
  168. package/src/tools/list-ducks.ts +65 -0
  169. package/src/tools/list-models.ts +101 -0
  170. package/src/tools/mcp-status.ts +117 -0
  171. package/src/utils/ascii-art.ts +85 -0
  172. package/src/utils/logger.ts +116 -0
  173. package/src/utils/safe-logger.ts +165 -0
  174. package/systemd/mcp-rubber-duck-with-ollama.service +55 -0
  175. package/systemd/mcp-rubber-duck.service +58 -0
  176. package/test-functionality.js +147 -0
  177. package/test-mcp-interface.js +221 -0
  178. package/tests/ascii-art.test.ts +36 -0
  179. package/tests/config.test.ts +239 -0
  180. package/tests/conversation.test.ts +308 -0
  181. package/tests/mcp-bridge.test.ts +291 -0
  182. package/tests/providers.test.ts +269 -0
  183. package/tests/tools/clear-conversations.test.ts +163 -0
  184. package/tsconfig.json +26 -0
@@ -0,0 +1,204 @@
1
+ import { DuckProvider } from './provider.js';
2
+ import { logger } from '../utils/logger.js';
3
+ import { getRandomDuckMessage } from '../utils/ascii-art.js';
4
+ export class ProviderManager {
5
+ providers = new Map();
6
+ healthStatus = new Map();
7
+ configManager;
8
+ defaultProvider;
9
+ constructor(configManager) {
10
+ this.configManager = configManager;
11
+ this.initializeProviders();
12
+ }
13
+ initializeProviders() {
14
+ const config = this.configManager.getConfig();
15
+ const allProviders = config.providers;
16
+ for (const [name, providerConfig] of Object.entries(allProviders)) {
17
+ try {
18
+ const provider = new DuckProvider(name, providerConfig.nickname, {
19
+ apiKey: providerConfig.api_key,
20
+ baseURL: providerConfig.base_url,
21
+ model: providerConfig.default_model,
22
+ availableModels: providerConfig.models,
23
+ temperature: providerConfig.temperature,
24
+ timeout: providerConfig.timeout,
25
+ maxRetries: providerConfig.max_retries,
26
+ systemPrompt: providerConfig.system_prompt,
27
+ });
28
+ this.providers.set(name, provider);
29
+ logger.info(`Initialized provider: ${name} (${providerConfig.nickname})`);
30
+ }
31
+ catch (error) {
32
+ logger.error(`Failed to initialize provider ${name}:`, error);
33
+ }
34
+ }
35
+ this.defaultProvider = config.default_provider;
36
+ if (this.providers.size === 0) {
37
+ throw new Error('No providers could be initialized');
38
+ }
39
+ }
40
+ async checkHealth(providerName) {
41
+ const results = [];
42
+ const providersToCheck = providerName
43
+ ? [this.providers.get(providerName)].filter(Boolean)
44
+ : Array.from(this.providers.values());
45
+ for (const provider of providersToCheck) {
46
+ if (!provider)
47
+ continue;
48
+ const startTime = Date.now();
49
+ try {
50
+ const healthy = await provider.healthCheck();
51
+ const health = {
52
+ provider: provider.name,
53
+ healthy,
54
+ latency: Date.now() - startTime,
55
+ lastCheck: new Date(),
56
+ };
57
+ this.healthStatus.set(provider.name, health);
58
+ results.push(health);
59
+ }
60
+ catch (error) {
61
+ const health = {
62
+ provider: provider.name,
63
+ healthy: false,
64
+ lastCheck: new Date(),
65
+ error: error instanceof Error ? error.message : String(error),
66
+ };
67
+ this.healthStatus.set(provider.name, health);
68
+ results.push(health);
69
+ }
70
+ }
71
+ return results;
72
+ }
73
+ async askDuck(providerName, prompt, options) {
74
+ const provider = this.getProvider(providerName);
75
+ const startTime = Date.now();
76
+ try {
77
+ const response = await provider.chat({
78
+ messages: [{ role: 'user', content: prompt, timestamp: new Date() }],
79
+ ...options,
80
+ });
81
+ return {
82
+ provider: provider.name,
83
+ nickname: provider.nickname,
84
+ model: response.model,
85
+ content: response.content,
86
+ usage: response.usage ? {
87
+ prompt_tokens: response.usage.promptTokens,
88
+ completion_tokens: response.usage.completionTokens,
89
+ total_tokens: response.usage.totalTokens,
90
+ promptTokens: response.usage.promptTokens,
91
+ completionTokens: response.usage.completionTokens,
92
+ totalTokens: response.usage.totalTokens,
93
+ } : undefined,
94
+ latency: Date.now() - startTime,
95
+ cached: false,
96
+ };
97
+ }
98
+ catch (error) {
99
+ // Try failover if enabled
100
+ if (this.configManager.getConfig().enable_failover && providerName === undefined) {
101
+ const errorMessage = error instanceof Error ? error.message : String(error);
102
+ logger.warn(`Primary provider failed, attempting failover: ${errorMessage}`);
103
+ return this.askWithFailover(prompt, options, provider.name);
104
+ }
105
+ throw error;
106
+ }
107
+ }
108
+ async compareDucks(prompt, providerNames, options) {
109
+ const providersToUse = providerNames
110
+ ? providerNames.map(name => this.providers.get(name)).filter(Boolean)
111
+ : Array.from(this.providers.values());
112
+ if (providersToUse.length === 0) {
113
+ throw new Error('No valid providers specified');
114
+ }
115
+ const promises = providersToUse.map(provider => provider ? this.askDuck(provider.name, prompt, options).catch(error => ({
116
+ provider: provider.name,
117
+ nickname: provider.nickname,
118
+ model: '',
119
+ content: `Error: ${error instanceof Error ? error.message : String(error)}`,
120
+ latency: 0,
121
+ cached: false,
122
+ })) : Promise.resolve({
123
+ provider: 'unknown',
124
+ nickname: 'Unknown',
125
+ model: '',
126
+ content: 'Error: Invalid provider',
127
+ latency: 0,
128
+ cached: false,
129
+ }));
130
+ return Promise.all(promises);
131
+ }
132
+ async duckCouncil(prompt, options) {
133
+ return this.compareDucks(prompt, undefined, options);
134
+ }
135
+ async askWithFailover(prompt, options, failedProvider) {
136
+ const availableProviders = Array.from(this.providers.keys()).filter(name => name !== failedProvider);
137
+ for (const providerName of availableProviders) {
138
+ try {
139
+ logger.info(`${getRandomDuckMessage('failover')} Trying ${providerName}...`);
140
+ return await this.askDuck(providerName, prompt, options);
141
+ }
142
+ catch (error) {
143
+ logger.warn(`Failover to ${providerName} failed:`, error);
144
+ continue;
145
+ }
146
+ }
147
+ throw new Error('All ducks have flown away! No providers available.');
148
+ }
149
+ getProvider(name) {
150
+ const providerName = name || this.defaultProvider;
151
+ if (!providerName) {
152
+ throw new Error('No provider specified and no default provider configured');
153
+ }
154
+ const provider = this.providers.get(providerName);
155
+ if (!provider) {
156
+ throw new Error(`Duck "${providerName}" not found in the pond`);
157
+ }
158
+ return provider;
159
+ }
160
+ getAllProviders() {
161
+ return Array.from(this.providers.entries()).map(([name, provider]) => ({
162
+ name,
163
+ info: provider.getInfo(),
164
+ health: this.healthStatus.get(name),
165
+ }));
166
+ }
167
+ getProviderNames() {
168
+ return Array.from(this.providers.keys());
169
+ }
170
+ async getAvailableModels(providerName) {
171
+ const provider = this.providers.get(providerName);
172
+ if (!provider) {
173
+ throw new Error(`Provider ${providerName} not found`);
174
+ }
175
+ return provider.listModels();
176
+ }
177
+ async getAllModels() {
178
+ const allModels = new Map();
179
+ for (const [name, provider] of this.providers) {
180
+ try {
181
+ const models = await provider.listModels();
182
+ allModels.set(name, models);
183
+ }
184
+ catch (error) {
185
+ logger.error(`Failed to get models for ${name}:`, error);
186
+ allModels.set(name, []);
187
+ }
188
+ }
189
+ return allModels;
190
+ }
191
+ validateModel(providerName, modelId) {
192
+ const provider = this.providers.get(providerName);
193
+ if (!provider) {
194
+ return false;
195
+ }
196
+ const info = provider.getInfo();
197
+ if (info.availableModels) {
198
+ return info.availableModels.includes(modelId);
199
+ }
200
+ // If no models list, accept any model (let the API validate)
201
+ return true;
202
+ }
203
+ }
204
+ //# sourceMappingURL=manager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"manager.js","sourceRoot":"","sources":["../../src/providers/manager.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,eAAe,CAAC;AAI7C,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAC5C,OAAO,EAAE,oBAAoB,EAAE,MAAM,uBAAuB,CAAC;AAE7D,MAAM,OAAO,eAAe;IAClB,SAAS,GAA8B,IAAI,GAAG,EAAE,CAAC;IACjD,YAAY,GAAgC,IAAI,GAAG,EAAE,CAAC;IACpD,aAAa,CAAgB;IAC/B,eAAe,CAAU;IAEjC,YAAY,aAA4B;QACtC,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;QACnC,IAAI,CAAC,mBAAmB,EAAE,CAAC;IAC7B,CAAC;IAEO,mBAAmB;QACzB,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC,SAAS,EAAE,CAAC;QAC9C,MAAM,YAAY,GAAG,MAAM,CAAC,SAAS,CAAC;QAEtC,KAAK,MAAM,CAAC,IAAI,EAAE,cAAc,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC;YAClE,IAAI,CAAC;gBACH,MAAM,QAAQ,GAAG,IAAI,YAAY,CAAC,IAAI,EAAE,cAAc,CAAC,QAAQ,EAAE;oBAC/D,MAAM,EAAE,cAAc,CAAC,OAAO;oBAC9B,OAAO,EAAE,cAAc,CAAC,QAAQ;oBAChC,KAAK,EAAE,cAAc,CAAC,aAAa;oBACnC,eAAe,EAAE,cAAc,CAAC,MAAM;oBACtC,WAAW,EAAE,cAAc,CAAC,WAAW;oBACvC,OAAO,EAAE,cAAc,CAAC,OAAO;oBAC/B,UAAU,EAAE,cAAc,CAAC,WAAW;oBACtC,YAAY,EAAE,cAAc,CAAC,aAAa;iBAC3C,CAAC,CAAC;gBAEH,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;gBACnC,MAAM,CAAC,IAAI,CAAC,yBAAyB,IAAI,KAAK,cAAc,CAAC,QAAQ,GAAG,CAAC,CAAC;YAC5E,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,MAAM,CAAC,KAAK,CAAC,iCAAiC,IAAI,GAAG,EAAE,KAAK,CAAC,CAAC;YAChE,CAAC;QACH,CAAC;QAED,IAAI,CAAC,eAAe,GAAG,MAAM,CAAC,gBAAgB,CAAC;QAE/C,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,KAAK,CAAC,EAAE,CAAC;YAC9B,MAAM,IAAI,KAAK,CAAC,mCAAmC,CAAC,CAAC;QACvD,CAAC;IACH,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,YAAqB;QACrC,MAAM,OAAO,GAAqB,EAAE,CAAC;QACrC,MAAM,gBAAgB,GAAG,YAAY;YACnC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC;YACpD,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,CAAC,CAAC;QAExC,KAAK,MAAM,QAAQ,IAAI,gBAAgB,EAAE,CAAC;YACxC,IAAI,CAAC,QAAQ;gBAAE,SAAS;YAExB,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;YAC7B,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,MAAM,QAAQ,CAAC,WAAW,EAAE,CAAC;gBAC7C,MAAM,MAAM,GAAmB;oBAC7B,QAAQ,EAAE,QAAQ,CAAC,IAAI;oBACvB,OAAO;oBACP,OAAO,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS;oBAC/B,SAAS,EAAE,IAAI,IAAI,EAAE;iBACtB,CAAC;gBAEF,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;gBAC7C,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YACvB,CAAC;YAAC,OAAO,KAAc,EAAE,CAAC;gBACxB,MAAM,MAAM,GAAmB;oBAC7B,QAAQ,EAAE,QAAQ,CAAC,IAAI;oBACvB,OAAO,EAAE,KAAK;oBACd,SAAS,EAAE,IAAI,IAAI,EAAE;oBACrB,KAAK,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;iBAC9D,CAAC;gBAEF,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;gBAC7C,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YACvB,CAAC;QACH,CAAC;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,KAAK,CAAC,OAAO,CACX,YAAgC,EAChC,MAAc,EACd,OAA8B;QAE9B,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC;QAChD,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAE7B,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC;gBACnC,QAAQ,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,IAAI,IAAI,EAAE,EAAE,CAAC;gBACpE,GAAG,OAAO;aACX,CAAC,CAAC;YAEH,OAAO;gBACL,QAAQ,EAAE,QAAQ,CAAC,IAAI;gBACvB,QAAQ,EAAE,QAAQ,CAAC,QAAQ;gBAC3B,KAAK,EAAE,QAAQ,CAAC,KAAK;gBACrB,OAAO,EAAE,QAAQ,CAAC,OAAO;gBACzB,KAAK,EAAE,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;oBACtB,aAAa,EAAE,QAAQ,CAAC,KAAK,CAAC,YAAY;oBAC1C,iBAAiB,EAAE,QAAQ,CAAC,KAAK,CAAC,gBAAgB;oBAClD,YAAY,EAAE,QAAQ,CAAC,KAAK,CAAC,WAAW;oBACxC,YAAY,EAAE,QAAQ,CAAC,KAAK,CAAC,YAAY;oBACzC,gBAAgB,EAAE,QAAQ,CAAC,KAAK,CAAC,gBAAgB;oBACjD,WAAW,EAAE,QAAQ,CAAC,KAAK,CAAC,WAAW;iBACxC,CAAC,CAAC,CAAC,SAAS;gBACb,OAAO,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS;gBAC/B,MAAM,EAAE,KAAK;aACd,CAAC;QACJ,CAAC;QAAC,OAAO,KAAc,EAAE,CAAC;YACxB,0BAA0B;YAC1B,IAAI,IAAI,CAAC,aAAa,CAAC,SAAS,EAAE,CAAC,eAAe,IAAI,YAAY,KAAK,SAAS,EAAE,CAAC;gBACjF,MAAM,YAAY,GAAG,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBAC5E,MAAM,CAAC,IAAI,CAAC,iDAAiD,YAAY,EAAE,CAAC,CAAC;gBAC7E,OAAO,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,OAAO,EAAE,QAAQ,CAAC,IAAI,CAAC,CAAC;YAC9D,CAAC;YACD,MAAM,KAAK,CAAC;QACd,CAAC;IACH,CAAC;IAED,KAAK,CAAC,YAAY,CAChB,MAAc,EACd,aAAwB,EACxB,OAA8B;QAE9B,MAAM,cAAc,GAAG,aAAa;YAClC,CAAC,CAAC,aAAa,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC;YACrE,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,CAAC,CAAC;QAExC,IAAI,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAChC,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC,CAAC;QAClD,CAAC;QAED,MAAM,QAAQ,GAAG,cAAc,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAC7C,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;YACtE,QAAQ,EAAE,QAAQ,CAAC,IAAI;YACvB,QAAQ,EAAE,QAAQ,CAAC,QAAQ;YAC3B,KAAK,EAAE,EAAE;YACT,OAAO,EAAE,UAAU,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE;YAC3E,OAAO,EAAE,CAAC;YACV,MAAM,EAAE,KAAK;SACd,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC;YACpB,QAAQ,EAAE,SAAS;YACnB,QAAQ,EAAE,SAAS;YACnB,KAAK,EAAE,EAAE;YACT,OAAO,EAAE,yBAAyB;YAClC,OAAO,EAAE,CAAC;YACV,MAAM,EAAE,KAAK;SACd,CAAC,CACH,CAAC;QAEF,OAAO,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IAC/B,CAAC;IAED,KAAK,CAAC,WAAW,CACf,MAAc,EACd,OAA8B;QAE9B,OAAO,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,SAAS,EAAE,OAAO,CAAC,CAAC;IACvD,CAAC;IAEO,KAAK,CAAC,eAAe,CAC3B,MAAc,EACd,OAAyC,EACzC,cAAsB;QAEtB,MAAM,kBAAkB,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CACjE,IAAI,CAAC,EAAE,CAAC,IAAI,KAAK,cAAc,CAChC,CAAC;QAEF,KAAK,MAAM,YAAY,IAAI,kBAAkB,EAAE,CAAC;YAC9C,IAAI,CAAC;gBACH,MAAM,CAAC,IAAI,CAAC,GAAG,oBAAoB,CAAC,UAAU,CAAC,WAAW,YAAY,KAAK,CAAC,CAAC;gBAC7E,OAAO,MAAM,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,MAAM,EAAE,OAAO,CAAC,CAAC;YAC3D,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,MAAM,CAAC,IAAI,CAAC,eAAe,YAAY,UAAU,EAAE,KAAK,CAAC,CAAC;gBAC1D,SAAS;YACX,CAAC;QACH,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,oDAAoD,CAAC,CAAC;IACxE,CAAC;IAED,WAAW,CAAC,IAAa;QACvB,MAAM,YAAY,GAAG,IAAI,IAAI,IAAI,CAAC,eAAe,CAAC;QAElD,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,MAAM,IAAI,KAAK,CAAC,0DAA0D,CAAC,CAAC;QAC9E,CAAC;QAED,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;QAElD,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,MAAM,IAAI,KAAK,CAAC,SAAS,YAAY,yBAAyB,CAAC,CAAC;QAClE,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED,eAAe;QACb,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,EAAE,QAAQ,CAAC,EAAE,EAAE,CAAC,CAAC;YACrE,IAAI;YACJ,IAAI,EAAE,QAAQ,CAAC,OAAO,EAAE;YACxB,MAAM,EAAE,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,IAAI,CAAC;SACpC,CAAC,CAAC,CAAC;IACN,CAAC;IAED,gBAAgB;QACd,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC;IAC3C,CAAC;IAED,KAAK,CAAC,kBAAkB,CAAC,YAAoB;QAC3C,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;QAClD,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,MAAM,IAAI,KAAK,CAAC,YAAY,YAAY,YAAY,CAAC,CAAC;QACxD,CAAC;QACD,OAAO,QAAQ,CAAC,UAAU,EAAE,CAAC;IAC/B,CAAC;IAED,KAAK,CAAC,YAAY;QAChB,MAAM,SAAS,GAAG,IAAI,GAAG,EAAuB,CAAC;QAEjD,KAAK,MAAM,CAAC,IAAI,EAAE,QAAQ,CAAC,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YAC9C,IAAI,CAAC;gBACH,MAAM,MAAM,GAAG,MAAM,QAAQ,CAAC,UAAU,EAAE,CAAC;gBAC3C,SAAS,CAAC,GAAG,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;YAC9B,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,MAAM,CAAC,KAAK,CAAC,4BAA4B,IAAI,GAAG,EAAE,KAAK,CAAC,CAAC;gBACzD,SAAS,CAAC,GAAG,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;YAC1B,CAAC;QACH,CAAC;QAED,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,aAAa,CAAC,YAAoB,EAAE,OAAe;QACjD,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;QAClD,IAAI,CAAC,QAAQ,EAAE,CAAC;YACd,OAAO,KAAK,CAAC;QACf,CAAC;QAED,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,EAAE,CAAC;QAChC,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YACzB,OAAO,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;QAChD,CAAC;QAED,6DAA6D;QAC7D,OAAO,IAAI,CAAC;IACd,CAAC;CACF"}
@@ -0,0 +1,29 @@
1
+ import OpenAI from 'openai';
2
+ import { ChatOptions, ChatResponse, ProviderOptions, StreamChunk, ModelInfo, OpenAIChatParams, OpenAIChatResponse } from './types.js';
3
+ import { ConversationMessage } from '../config/types.js';
4
+ export declare class DuckProvider {
5
+ protected client: OpenAI;
6
+ protected options: ProviderOptions;
7
+ name: string;
8
+ nickname: string;
9
+ constructor(name: string, nickname: string, options: ProviderOptions);
10
+ protected supportsTemperature(model: string): boolean;
11
+ chat(options: ChatOptions): Promise<ChatResponse>;
12
+ protected createChatCompletion(baseParams: Partial<OpenAIChatParams>): Promise<OpenAIChatResponse>;
13
+ chatStream(options: ChatOptions): AsyncGenerator<StreamChunk>;
14
+ healthCheck(): Promise<boolean>;
15
+ protected prepareMessages(messages: ConversationMessage[], systemPrompt?: string): Array<{
16
+ role: string;
17
+ content: string;
18
+ }>;
19
+ listModels(): Promise<ModelInfo[]>;
20
+ getInfo(): {
21
+ name: string;
22
+ nickname: string;
23
+ model: string;
24
+ availableModels: string[] | undefined;
25
+ baseURL: string;
26
+ hasApiKey: boolean;
27
+ };
28
+ }
29
+ //# sourceMappingURL=provider.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"provider.d.ts","sourceRoot":"","sources":["../../src/providers/provider.ts"],"names":[],"mappings":"AAAA,OAAO,MAAM,MAAM,QAAQ,CAAC;AAC5B,OAAO,EAAE,WAAW,EAAE,YAAY,EAAE,eAAe,EAAE,WAAW,EAAE,SAAS,EAAE,gBAAgB,EAAE,kBAAkB,EAAiB,MAAM,YAAY,CAAC;AACrJ,OAAO,EAAE,mBAAmB,EAAE,MAAM,oBAAoB,CAAC;AAGzD,qBAAa,YAAY;IACvB,SAAS,CAAC,MAAM,EAAE,MAAM,CAAC;IACzB,SAAS,CAAC,OAAO,EAAE,eAAe,CAAC;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,MAAM,CAAC;gBAEZ,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,eAAe;IAapE,SAAS,CAAC,mBAAmB,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO;IAU/C,IAAI,CAAC,OAAO,EAAE,WAAW,GAAG,OAAO,CAAC,YAAY,CAAC;cAoCvC,oBAAoB,CAAC,UAAU,EAAE,OAAO,CAAC,gBAAgB,CAAC,GAAG,OAAO,CAAC,kBAAkB,CAAC;IAKjG,UAAU,CAAC,OAAO,EAAE,WAAW,GAAG,cAAc,CAAC,WAAW,CAAC;IA6B9D,WAAW,IAAI,OAAO,CAAC,OAAO,CAAC;IAkCrC,SAAS,CAAC,eAAe,CACvB,QAAQ,EAAE,mBAAmB,EAAE,EAC/B,YAAY,CAAC,EAAE,MAAM,GACpB,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAA;KAAE,CAAC;IAoBrC,UAAU,IAAI,OAAO,CAAC,SAAS,EAAE,CAAC;IAmCxC,OAAO;;;;;;;;CAUR"}
@@ -0,0 +1,179 @@
1
+ import OpenAI from 'openai';
2
+ import { logger } from '../utils/logger.js';
3
+ export class DuckProvider {
4
+ client;
5
+ options;
6
+ name;
7
+ nickname;
8
+ constructor(name, nickname, options) {
9
+ this.name = name;
10
+ this.nickname = nickname;
11
+ this.options = options;
12
+ this.client = new OpenAI({
13
+ apiKey: options.apiKey || 'not-needed',
14
+ baseURL: options.baseURL,
15
+ timeout: options.timeout || 300000,
16
+ maxRetries: options.maxRetries || 3,
17
+ });
18
+ }
19
+ supportsTemperature(model) {
20
+ // Reasoning models don't support temperature parameter
21
+ return !model.startsWith('o1') &&
22
+ !model.includes('o1-') &&
23
+ !model.startsWith('o3') &&
24
+ !model.includes('o3-') &&
25
+ !model.startsWith('gpt-5') &&
26
+ !model.includes('gpt-5');
27
+ }
28
+ async chat(options) {
29
+ try {
30
+ const messages = this.prepareMessages(options.messages, options.systemPrompt);
31
+ const modelToUse = options.model || this.options.model;
32
+ const baseParams = {
33
+ model: modelToUse,
34
+ messages: messages,
35
+ stream: false,
36
+ };
37
+ // Only add temperature if the model supports it
38
+ if (this.supportsTemperature(modelToUse)) {
39
+ baseParams.temperature = options.temperature ?? this.options.temperature ?? 0.7;
40
+ }
41
+ const response = await this.createChatCompletion(baseParams);
42
+ const choice = response.choices[0];
43
+ return {
44
+ content: choice.message?.content || '',
45
+ usage: response.usage ? {
46
+ promptTokens: response.usage.prompt_tokens,
47
+ completionTokens: response.usage.completion_tokens,
48
+ totalTokens: response.usage.total_tokens,
49
+ } : undefined,
50
+ model: modelToUse, // Return the requested model, not the resolved one
51
+ finishReason: choice.finish_reason || undefined,
52
+ };
53
+ }
54
+ catch (error) {
55
+ logger.error(`Provider ${this.name} chat error:`, error);
56
+ const errorMessage = error instanceof Error ? error.message : String(error);
57
+ throw new Error(`Duck ${this.nickname} couldn't respond: ${errorMessage}`);
58
+ }
59
+ }
60
+ async createChatCompletion(baseParams) {
61
+ const params = { ...baseParams };
62
+ return await this.client.chat.completions.create(params);
63
+ }
64
+ async *chatStream(options) {
65
+ try {
66
+ const messages = this.prepareMessages(options.messages, options.systemPrompt);
67
+ const modelToUse = options.model || this.options.model;
68
+ const streamParams = {
69
+ model: modelToUse,
70
+ messages: messages,
71
+ stream: true,
72
+ ...(this.supportsTemperature(modelToUse) && {
73
+ temperature: options.temperature ?? this.options.temperature ?? 0.7
74
+ })
75
+ };
76
+ const stream = await this.client.chat.completions.create(streamParams);
77
+ for await (const chunk of stream) {
78
+ const content = chunk.choices[0]?.delta?.content || '';
79
+ const done = chunk.choices[0]?.finish_reason !== null;
80
+ yield { content, done };
81
+ }
82
+ }
83
+ catch (error) {
84
+ logger.error(`Provider ${this.name} stream error:`, error);
85
+ const errorMessage = error instanceof Error ? error.message : String(error);
86
+ throw new Error(`Duck ${this.nickname} stream failed: ${errorMessage}`);
87
+ }
88
+ }
89
+ async healthCheck() {
90
+ try {
91
+ const baseParams = {
92
+ model: this.options.model,
93
+ messages: [{ role: 'user', content: 'Say "healthy"' }],
94
+ stream: false,
95
+ };
96
+ // Only add temperature if the model supports it
97
+ if (this.supportsTemperature(this.options.model)) {
98
+ baseParams.temperature = 0.5;
99
+ }
100
+ // Health check without token limits
101
+ const response = await this.createChatCompletion(baseParams);
102
+ const content = response.choices[0]?.message?.content;
103
+ const hasContent = !!content;
104
+ if (!hasContent) {
105
+ logger.warn(`Health check for ${this.name}: No content in response`, {
106
+ response: JSON.stringify(response, null, 2)
107
+ });
108
+ }
109
+ else {
110
+ logger.debug(`Health check for ${this.name} succeeded with response: ${content}`);
111
+ }
112
+ return hasContent;
113
+ }
114
+ catch (error) {
115
+ logger.warn(`Health check failed for ${this.name}:`, error);
116
+ return false;
117
+ }
118
+ }
119
+ prepareMessages(messages, systemPrompt) {
120
+ const prepared = [];
121
+ // Add system prompt if provided
122
+ const prompt = systemPrompt || this.options.systemPrompt;
123
+ if (prompt) {
124
+ prepared.push({ role: 'system', content: prompt });
125
+ }
126
+ // Add conversation messages
127
+ for (const msg of messages) {
128
+ prepared.push({
129
+ role: msg.role,
130
+ content: msg.content,
131
+ });
132
+ }
133
+ return prepared;
134
+ }
135
+ async listModels() {
136
+ try {
137
+ // Try to fetch models from the API
138
+ const response = await this.client.models.list();
139
+ const models = [];
140
+ for await (const model of response) {
141
+ models.push({
142
+ id: model.id,
143
+ created: model.created,
144
+ owned_by: model.owned_by,
145
+ object: model.object,
146
+ });
147
+ }
148
+ logger.debug(`Fetched ${models.length} models from ${this.name}`);
149
+ return models;
150
+ }
151
+ catch (error) {
152
+ const errorMessage = error instanceof Error ? error.message : String(error);
153
+ logger.warn(`Failed to fetch models from ${this.name}: ${errorMessage}`);
154
+ // Fall back to configured models
155
+ if (this.options.availableModels && this.options.availableModels.length > 0) {
156
+ return this.options.availableModels.map(id => ({
157
+ id,
158
+ description: 'Configured model (not fetched from API)',
159
+ }));
160
+ }
161
+ // Last fallback: return just the default model
162
+ return [{
163
+ id: this.options.model,
164
+ description: 'Default configured model',
165
+ }];
166
+ }
167
+ }
168
+ getInfo() {
169
+ return {
170
+ name: this.name,
171
+ nickname: this.nickname,
172
+ model: this.options.model,
173
+ availableModels: this.options.availableModels,
174
+ baseURL: this.options.baseURL,
175
+ hasApiKey: !!this.options.apiKey,
176
+ };
177
+ }
178
+ }
179
+ //# sourceMappingURL=provider.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"provider.js","sourceRoot":"","sources":["../../src/providers/provider.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,69 @@
1
+ import { ConversationMessage } from '../config/types.js';
2
+ import { FunctionDefinition } from '../services/function-bridge.js';
3
+ import type { ChatCompletion, ChatCompletionMessageParam, ChatCompletionMessageToolCall, ChatCompletionCreateParamsNonStreaming } from 'openai/resources/chat/completions';
4
+ export interface ModelInfo {
5
+ id: string;
6
+ created?: number;
7
+ owned_by?: string;
8
+ object?: string;
9
+ context_window?: number;
10
+ description?: string;
11
+ }
12
+ export interface ProviderOptions {
13
+ apiKey?: string;
14
+ baseURL: string;
15
+ model: string;
16
+ availableModels?: string[];
17
+ temperature?: number;
18
+ timeout?: number;
19
+ maxRetries?: number;
20
+ systemPrompt?: string;
21
+ }
22
+ export interface ToolCall {
23
+ id: string;
24
+ type: 'function';
25
+ function: {
26
+ name: string;
27
+ arguments: string;
28
+ };
29
+ }
30
+ export interface ChatOptions {
31
+ messages: ConversationMessage[];
32
+ model?: string;
33
+ stream?: boolean;
34
+ temperature?: number;
35
+ systemPrompt?: string;
36
+ tools?: FunctionDefinition[];
37
+ toolChoice?: 'auto' | 'none' | {
38
+ type: 'function';
39
+ function: {
40
+ name: string;
41
+ };
42
+ };
43
+ }
44
+ export interface ChatResponse {
45
+ content: string;
46
+ usage?: {
47
+ promptTokens: number;
48
+ completionTokens: number;
49
+ totalTokens: number;
50
+ };
51
+ model: string;
52
+ finishReason?: string;
53
+ toolCalls?: ToolCall[];
54
+ }
55
+ export interface StreamChunk {
56
+ content: string;
57
+ done: boolean;
58
+ }
59
+ export type OpenAIChatParams = ChatCompletionCreateParamsNonStreaming;
60
+ export type OpenAIMessage = ChatCompletionMessageParam;
61
+ export type OpenAIChatResponse = ChatCompletion;
62
+ export type OpenAIToolCall = ChatCompletionMessageToolCall;
63
+ export interface MCPResult {
64
+ id: string;
65
+ success: boolean;
66
+ data?: unknown;
67
+ error?: string;
68
+ }
69
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/providers/types.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,mBAAmB,EAAE,MAAM,oBAAoB,CAAC;AACzD,OAAO,EAAE,kBAAkB,EAAE,MAAM,gCAAgC,CAAC;AACpE,OAAO,KAAK,EACV,cAAc,EACd,0BAA0B,EAC1B,6BAA6B,EAC7B,sCAAsC,EACvC,MAAM,mCAAmC,CAAC;AAE3C,MAAM,WAAW,SAAS;IACxB,EAAE,EAAE,MAAM,CAAC;IACX,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED,MAAM,WAAW,eAAe;IAC9B,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,EAAE,MAAM,CAAC;IACd,eAAe,CAAC,EAAE,MAAM,EAAE,CAAC;IAC3B,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,YAAY,CAAC,EAAE,MAAM,CAAC;CACvB;AAED,MAAM,WAAW,QAAQ;IACvB,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,UAAU,CAAC;IACjB,QAAQ,EAAE;QACR,IAAI,EAAE,MAAM,CAAC;QACb,SAAS,EAAE,MAAM,CAAC;KACnB,CAAC;CACH;AAED,MAAM,WAAW,WAAW;IAC1B,QAAQ,EAAE,mBAAmB,EAAE,CAAC;IAChC,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,KAAK,CAAC,EAAE,kBAAkB,EAAE,CAAC;IAC7B,UAAU,CAAC,EAAE,MAAM,GAAG,MAAM,GAAG;QAAE,IAAI,EAAE,UAAU,CAAC;QAAC,QAAQ,EAAE;YAAE,IAAI,EAAE,MAAM,CAAA;SAAE,CAAA;KAAE,CAAC;CACjF;AAED,MAAM,WAAW,YAAY;IAC3B,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,CAAC,EAAE;QACN,YAAY,EAAE,MAAM,CAAC;QACrB,gBAAgB,EAAE,MAAM,CAAC;QACzB,WAAW,EAAE,MAAM,CAAC;KACrB,CAAC;IACF,KAAK,EAAE,MAAM,CAAC;IACd,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,SAAS,CAAC,EAAE,QAAQ,EAAE,CAAC;CACxB;AAED,MAAM,WAAW,WAAW;IAC1B,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,OAAO,CAAC;CACf;AAGD,MAAM,MAAM,gBAAgB,GAAG,sCAAsC,CAAC;AACtE,MAAM,MAAM,aAAa,GAAG,0BAA0B,CAAC;AACvD,MAAM,MAAM,kBAAkB,GAAG,cAAc,CAAC;AAChD,MAAM,MAAM,cAAc,GAAG,6BAA6B,CAAC;AAG3D,MAAM,WAAW,SAAS;IACxB,EAAE,EAAE,MAAM,CAAC;IACX,OAAO,EAAE,OAAO,CAAC;IACjB,IAAI,CAAC,EAAE,OAAO,CAAC;IACf,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/providers/types.ts"],"names":[],"mappings":""}
@@ -0,0 +1,24 @@
1
+ export declare class RubberDuckServer {
2
+ private server;
3
+ private configManager;
4
+ private providerManager;
5
+ private enhancedProviderManager?;
6
+ private conversationManager;
7
+ private cache;
8
+ private healthMonitor;
9
+ private mcpClientManager?;
10
+ private approvalService?;
11
+ private functionBridge?;
12
+ private mcpEnabled;
13
+ constructor();
14
+ private initializeMCPBridge;
15
+ private setupHandlers;
16
+ private handleAskDuckWithMCP;
17
+ private handleCompareDucksWithMCP;
18
+ private handleDuckCouncilWithMCP;
19
+ private formatEnhancedDuckResponse;
20
+ private getTools;
21
+ start(): Promise<void>;
22
+ stop(): Promise<void>;
23
+ }
24
+ //# sourceMappingURL=server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAmCA,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,aAAa,CAAgB;IACrC,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,uBAAuB,CAAC,CAA0B;IAC1D,OAAO,CAAC,mBAAmB,CAAsB;IACjD,OAAO,CAAC,KAAK,CAAgB;IAC7B,OAAO,CAAC,aAAa,CAAgB;IAGrC,OAAO,CAAC,gBAAgB,CAAC,CAAmB;IAC5C,OAAO,CAAC,eAAe,CAAC,CAAkB;IAC1C,OAAO,CAAC,cAAc,CAAC,CAAiB;IACxC,OAAO,CAAC,UAAU,CAAkB;;IA4BpC,OAAO,CAAC,mBAAmB;IA2C3B,OAAO,CAAC,aAAa;YA8FP,oBAAoB;YAwCpB,yBAAyB;YA6BzB,wBAAwB;IAwBtC,OAAO,CAAC,0BAA0B;IAyClC,OAAO,CAAC,QAAQ;IAqMV,KAAK;IAwCL,IAAI;CAkBX"}