@apholdings/jensen-code 0.0.3 → 0.0.5

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 (166) hide show
  1. package/dist/cli/args.d.ts.map +1 -1
  2. package/dist/cli/args.js +6 -6
  3. package/dist/cli/args.js.map +1 -1
  4. package/dist/config.d.ts +6 -5
  5. package/dist/config.d.ts.map +1 -1
  6. package/dist/config.js +32 -25
  7. package/dist/config.js.map +1 -1
  8. package/dist/core/agent-session.d.ts +1 -0
  9. package/dist/core/agent-session.d.ts.map +1 -1
  10. package/dist/core/agent-session.js +25 -0
  11. package/dist/core/agent-session.js.map +1 -1
  12. package/dist/core/extensions/loader.d.ts.map +1 -1
  13. package/dist/core/extensions/loader.js +1 -1
  14. package/dist/core/extensions/loader.js.map +1 -1
  15. package/dist/core/footer-data-provider.d.ts +4 -1
  16. package/dist/core/footer-data-provider.d.ts.map +1 -1
  17. package/dist/core/footer-data-provider.js +25 -11
  18. package/dist/core/footer-data-provider.js.map +1 -1
  19. package/dist/index.d.ts +1 -1
  20. package/dist/index.d.ts.map +1 -1
  21. package/dist/index.js +1 -1
  22. package/dist/index.js.map +1 -1
  23. package/dist/modes/interactive/components/custom-editor.d.ts +1 -0
  24. package/dist/modes/interactive/components/custom-editor.d.ts.map +1 -1
  25. package/dist/modes/interactive/components/custom-editor.js +5 -0
  26. package/dist/modes/interactive/components/custom-editor.js.map +1 -1
  27. package/dist/modes/interactive/components/footer.d.ts +0 -2
  28. package/dist/modes/interactive/components/footer.d.ts.map +1 -1
  29. package/dist/modes/interactive/components/footer.js +8 -146
  30. package/dist/modes/interactive/components/footer.js.map +1 -1
  31. package/dist/modes/interactive/components/header.d.ts +9 -3
  32. package/dist/modes/interactive/components/header.d.ts.map +1 -1
  33. package/dist/modes/interactive/components/header.js +125 -196
  34. package/dist/modes/interactive/components/header.js.map +1 -1
  35. package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
  36. package/dist/modes/interactive/components/tool-execution.js +1 -2
  37. package/dist/modes/interactive/components/tool-execution.js.map +1 -1
  38. package/dist/modes/interactive/interactive-mode.d.ts +23 -4
  39. package/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
  40. package/dist/modes/interactive/interactive-mode.js +657 -243
  41. package/dist/modes/interactive/interactive-mode.js.map +1 -1
  42. package/dist/modes/interactive/theme/theme.d.ts.map +1 -1
  43. package/dist/modes/interactive/theme/theme.js +2 -0
  44. package/dist/modes/interactive/theme/theme.js.map +1 -1
  45. package/dist/utils/frontmatter.d.ts.map +1 -1
  46. package/dist/utils/frontmatter.js +8 -4
  47. package/dist/utils/frontmatter.js.map +1 -1
  48. package/dist/utils/tools-manager.d.ts.map +1 -1
  49. package/dist/utils/tools-manager.js +2 -2
  50. package/dist/utils/tools-manager.js.map +1 -1
  51. package/examples/extensions/osgrep.ts +643 -0
  52. package/examples/extensions/subagent/agents.ts +150 -38
  53. package/examples/extensions/subagent/index.ts +634 -514
  54. package/package.json +4 -3
  55. package/examples/README.md +0 -25
  56. package/examples/extensions/README.md +0 -206
  57. package/examples/extensions/antigravity-image-gen.ts +0 -416
  58. package/examples/extensions/auto-commit-on-exit.ts +0 -50
  59. package/examples/extensions/bash-spawn-hook.ts +0 -31
  60. package/examples/extensions/bookmark.ts +0 -51
  61. package/examples/extensions/built-in-tool-renderer.ts +0 -247
  62. package/examples/extensions/claude-rules.ts +0 -87
  63. package/examples/extensions/commands.ts +0 -73
  64. package/examples/extensions/confirm-destructive.ts +0 -60
  65. package/examples/extensions/custom-compaction.ts +0 -115
  66. package/examples/extensions/custom-footer.ts +0 -65
  67. package/examples/extensions/custom-header.ts +0 -74
  68. package/examples/extensions/custom-provider-anthropic/index.ts +0 -605
  69. package/examples/extensions/custom-provider-anthropic/package-lock.json +0 -24
  70. package/examples/extensions/custom-provider-anthropic/package.json +0 -19
  71. package/examples/extensions/custom-provider-gitlab-duo/index.ts +0 -350
  72. package/examples/extensions/custom-provider-gitlab-duo/package.json +0 -16
  73. package/examples/extensions/custom-provider-gitlab-duo/test.ts +0 -82
  74. package/examples/extensions/custom-provider-qwen-cli/index.ts +0 -346
  75. package/examples/extensions/custom-provider-qwen-cli/package.json +0 -16
  76. package/examples/extensions/dirty-repo-guard.ts +0 -57
  77. package/examples/extensions/doom-overlay/README.md +0 -46
  78. package/examples/extensions/doom-overlay/doom/build/doom.js +0 -21
  79. package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
  80. package/examples/extensions/doom-overlay/doom/build.sh +0 -152
  81. package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +0 -72
  82. package/examples/extensions/doom-overlay/doom-component.ts +0 -132
  83. package/examples/extensions/doom-overlay/doom-engine.ts +0 -173
  84. package/examples/extensions/doom-overlay/doom-keys.ts +0 -104
  85. package/examples/extensions/doom-overlay/index.ts +0 -75
  86. package/examples/extensions/doom-overlay/wad-finder.ts +0 -51
  87. package/examples/extensions/dynamic-resources/SKILL.md +0 -8
  88. package/examples/extensions/dynamic-resources/dynamic.json +0 -79
  89. package/examples/extensions/dynamic-resources/dynamic.md +0 -5
  90. package/examples/extensions/dynamic-resources/index.ts +0 -16
  91. package/examples/extensions/dynamic-tools.ts +0 -75
  92. package/examples/extensions/event-bus.ts +0 -44
  93. package/examples/extensions/file-trigger.ts +0 -42
  94. package/examples/extensions/git-checkpoint.ts +0 -54
  95. package/examples/extensions/handoff.ts +0 -151
  96. package/examples/extensions/hello.ts +0 -26
  97. package/examples/extensions/inline-bash.ts +0 -95
  98. package/examples/extensions/input-transform.ts +0 -44
  99. package/examples/extensions/interactive-shell.ts +0 -197
  100. package/examples/extensions/mac-system-theme.ts +0 -48
  101. package/examples/extensions/message-renderer.ts +0 -60
  102. package/examples/extensions/minimal-mode.ts +0 -427
  103. package/examples/extensions/modal-editor.ts +0 -86
  104. package/examples/extensions/model-status.ts +0 -32
  105. package/examples/extensions/notify.ts +0 -56
  106. package/examples/extensions/overlay-qa-tests.ts +0 -1349
  107. package/examples/extensions/overlay-test.ts +0 -151
  108. package/examples/extensions/permission-gate.ts +0 -35
  109. package/examples/extensions/pirate.ts +0 -48
  110. package/examples/extensions/plan-mode/README.md +0 -65
  111. package/examples/extensions/plan-mode/index.ts +0 -341
  112. package/examples/extensions/plan-mode/utils.ts +0 -168
  113. package/examples/extensions/preset.ts +0 -399
  114. package/examples/extensions/protected-paths.ts +0 -31
  115. package/examples/extensions/provider-payload.ts +0 -15
  116. package/examples/extensions/qna.ts +0 -120
  117. package/examples/extensions/question.ts +0 -265
  118. package/examples/extensions/questionnaire.ts +0 -428
  119. package/examples/extensions/rainbow-editor.ts +0 -89
  120. package/examples/extensions/reload-runtime.ts +0 -38
  121. package/examples/extensions/rpc-demo.ts +0 -125
  122. package/examples/extensions/sandbox/index.ts +0 -319
  123. package/examples/extensions/sandbox/package-lock.json +0 -92
  124. package/examples/extensions/sandbox/package.json +0 -19
  125. package/examples/extensions/send-user-message.ts +0 -98
  126. package/examples/extensions/session-name.ts +0 -28
  127. package/examples/extensions/shutdown-command.ts +0 -64
  128. package/examples/extensions/snake.ts +0 -344
  129. package/examples/extensions/space-invaders.ts +0 -561
  130. package/examples/extensions/ssh.ts +0 -221
  131. package/examples/extensions/status-line.ts +0 -41
  132. package/examples/extensions/subagent/README.md +0 -172
  133. package/examples/extensions/subagent/agents/planner.md +0 -37
  134. package/examples/extensions/subagent/agents/reviewer.md +0 -35
  135. package/examples/extensions/subagent/agents/scout.md +0 -50
  136. package/examples/extensions/subagent/agents/worker.md +0 -24
  137. package/examples/extensions/subagent/prompts/implement-and-review.md +0 -10
  138. package/examples/extensions/subagent/prompts/implement.md +0 -10
  139. package/examples/extensions/subagent/prompts/scout-and-plan.md +0 -9
  140. package/examples/extensions/summarize.ts +0 -196
  141. package/examples/extensions/system-prompt-header.ts +0 -18
  142. package/examples/extensions/timed-confirm.ts +0 -71
  143. package/examples/extensions/titlebar-spinner.ts +0 -59
  144. package/examples/extensions/todo.ts +0 -300
  145. package/examples/extensions/tool-override.ts +0 -144
  146. package/examples/extensions/tools.ts +0 -147
  147. package/examples/extensions/trigger-compact.ts +0 -41
  148. package/examples/extensions/truncated-tool.ts +0 -193
  149. package/examples/extensions/widget-placement.ts +0 -18
  150. package/examples/extensions/with-deps/index.ts +0 -33
  151. package/examples/extensions/with-deps/package-lock.json +0 -31
  152. package/examples/extensions/with-deps/package.json +0 -22
  153. package/examples/rpc-extension-ui.ts +0 -632
  154. package/examples/sdk/01-minimal.ts +0 -23
  155. package/examples/sdk/02-custom-model.ts +0 -50
  156. package/examples/sdk/03-custom-prompt.ts +0 -56
  157. package/examples/sdk/04-skills.ts +0 -47
  158. package/examples/sdk/05-tools.ts +0 -57
  159. package/examples/sdk/06-extensions.ts +0 -89
  160. package/examples/sdk/07-context-files.ts +0 -41
  161. package/examples/sdk/08-prompt-templates.ts +0 -48
  162. package/examples/sdk/09-api-keys-and-oauth.ts +0 -49
  163. package/examples/sdk/10-settings.ts +0 -52
  164. package/examples/sdk/11-sessions.ts +0 -49
  165. package/examples/sdk/12-full-control.ts +0 -83
  166. package/examples/sdk/README.md +0 -145
@@ -1,350 +0,0 @@
1
- /**
2
- * GitLab Duo Provider Extension
3
- *
4
- * Provides access to GitLab Duo AI models (Claude and GPT) through GitLab's AI Gateway.
5
- * Delegates to pi-ai's built-in Anthropic and OpenAI streaming implementations.
6
- *
7
- * Usage:
8
- * pi -e ./packages/coding-agent/examples/extensions/custom-provider-gitlab-duo
9
- * # Then /login gitlab-duo, or set GITLAB_TOKEN=glpat-...
10
- */
11
-
12
- import {
13
- type Api,
14
- type AssistantMessageEventStream,
15
- type Context,
16
- createAssistantMessageEventStream,
17
- type Model,
18
- type OAuthCredentials,
19
- type OAuthLoginCallbacks,
20
- type SimpleStreamOptions,
21
- streamSimpleAnthropic,
22
- streamSimpleOpenAIResponses,
23
- } from "@apholdings/jensen-ai";
24
- import type { ExtensionAPI } from "@apholdings/jensen-code";
25
-
26
- // =============================================================================
27
- // Constants
28
- // =============================================================================
29
-
30
- const GITLAB_COM_URL = "https://gitlab.com";
31
- const AI_GATEWAY_URL = "https://cloud.gitlab.com";
32
- const ANTHROPIC_PROXY_URL = `${AI_GATEWAY_URL}/ai/v1/proxy/anthropic/`;
33
- const OPENAI_PROXY_URL = `${AI_GATEWAY_URL}/ai/v1/proxy/openai/v1`;
34
-
35
- const BUNDLED_CLIENT_ID = "da4edff2e6ebd2bc3208611e2768bc1c1dd7be791dc5ff26ca34ca9ee44f7d4b";
36
- const OAUTH_SCOPES = ["api"];
37
- const REDIRECT_URI = "http://127.0.0.1:8080/callback";
38
- const DIRECT_ACCESS_TTL = 25 * 60 * 1000;
39
-
40
- // =============================================================================
41
- // Models - exported for use by tests
42
- // =============================================================================
43
-
44
- type Backend = "anthropic" | "openai";
45
-
46
- interface GitLabModel {
47
- id: string;
48
- name: string;
49
- backend: Backend;
50
- baseUrl: string;
51
- reasoning: boolean;
52
- input: ("text" | "image")[];
53
- cost: { input: number; output: number; cacheRead: number; cacheWrite: number };
54
- contextWindow: number;
55
- maxTokens: number;
56
- }
57
-
58
- export const MODELS: GitLabModel[] = [
59
- // Anthropic
60
- {
61
- id: "claude-opus-4-5-20251101",
62
- name: "Claude Opus 4.5",
63
- backend: "anthropic",
64
- baseUrl: ANTHROPIC_PROXY_URL,
65
- reasoning: true,
66
- input: ["text", "image"],
67
- cost: { input: 15, output: 75, cacheRead: 1.5, cacheWrite: 18.75 },
68
- contextWindow: 200000,
69
- maxTokens: 32000,
70
- },
71
- {
72
- id: "claude-sonnet-4-5-20250929",
73
- name: "Claude Sonnet 4.5",
74
- backend: "anthropic",
75
- baseUrl: ANTHROPIC_PROXY_URL,
76
- reasoning: true,
77
- input: ["text", "image"],
78
- cost: { input: 3, output: 15, cacheRead: 0.3, cacheWrite: 3.75 },
79
- contextWindow: 200000,
80
- maxTokens: 16384,
81
- },
82
- {
83
- id: "claude-haiku-4-5-20251001",
84
- name: "Claude Haiku 4.5",
85
- backend: "anthropic",
86
- baseUrl: ANTHROPIC_PROXY_URL,
87
- reasoning: true,
88
- input: ["text", "image"],
89
- cost: { input: 1, output: 5, cacheRead: 0.1, cacheWrite: 1.25 },
90
- contextWindow: 200000,
91
- maxTokens: 8192,
92
- },
93
- // OpenAI (all use Responses API)
94
- {
95
- id: "gpt-5.1-2025-11-13",
96
- name: "GPT-5.1",
97
- backend: "openai",
98
- baseUrl: OPENAI_PROXY_URL,
99
- reasoning: true,
100
- input: ["text", "image"],
101
- cost: { input: 2.5, output: 10, cacheRead: 0, cacheWrite: 0 },
102
- contextWindow: 128000,
103
- maxTokens: 16384,
104
- },
105
- {
106
- id: "gpt-5-mini-2025-08-07",
107
- name: "GPT-5 Mini",
108
- backend: "openai",
109
- baseUrl: OPENAI_PROXY_URL,
110
- reasoning: true,
111
- input: ["text", "image"],
112
- cost: { input: 0.15, output: 0.6, cacheRead: 0, cacheWrite: 0 },
113
- contextWindow: 128000,
114
- maxTokens: 16384,
115
- },
116
- {
117
- id: "gpt-5-codex",
118
- name: "GPT-5 Codex",
119
- backend: "openai",
120
- baseUrl: OPENAI_PROXY_URL,
121
- reasoning: true,
122
- input: ["text", "image"],
123
- cost: { input: 2.5, output: 10, cacheRead: 0, cacheWrite: 0 },
124
- contextWindow: 128000,
125
- maxTokens: 16384,
126
- },
127
- ];
128
-
129
- const MODEL_MAP = new Map(MODELS.map((m) => [m.id, m]));
130
-
131
- // =============================================================================
132
- // Direct Access Token Cache
133
- // =============================================================================
134
-
135
- interface DirectAccessToken {
136
- token: string;
137
- headers: Record<string, string>;
138
- expiresAt: number;
139
- }
140
-
141
- let cachedDirectAccess: DirectAccessToken | null = null;
142
-
143
- async function getDirectAccessToken(gitlabAccessToken: string): Promise<DirectAccessToken> {
144
- const now = Date.now();
145
- if (cachedDirectAccess && cachedDirectAccess.expiresAt > now) {
146
- return cachedDirectAccess;
147
- }
148
-
149
- const response = await fetch(`${GITLAB_COM_URL}/api/v4/ai/third_party_agents/direct_access`, {
150
- method: "POST",
151
- headers: { Authorization: `Bearer ${gitlabAccessToken}`, "Content-Type": "application/json" },
152
- body: JSON.stringify({ feature_flags: { DuoAgentPlatformNext: true } }),
153
- });
154
-
155
- if (!response.ok) {
156
- const errorText = await response.text();
157
- if (response.status === 403) {
158
- throw new Error(
159
- `GitLab Duo access denied. Ensure GitLab Duo is enabled for your account. Error: ${errorText}`,
160
- );
161
- }
162
- throw new Error(`Failed to get direct access token: ${response.status} ${errorText}`);
163
- }
164
-
165
- const data = (await response.json()) as { token: string; headers: Record<string, string> };
166
- cachedDirectAccess = { token: data.token, headers: data.headers, expiresAt: now + DIRECT_ACCESS_TTL };
167
- return cachedDirectAccess;
168
- }
169
-
170
- function invalidateDirectAccessToken() {
171
- cachedDirectAccess = null;
172
- }
173
-
174
- // =============================================================================
175
- // OAuth
176
- // =============================================================================
177
-
178
- async function generatePKCE(): Promise<{ verifier: string; challenge: string }> {
179
- const array = new Uint8Array(32);
180
- crypto.getRandomValues(array);
181
- const verifier = btoa(String.fromCharCode(...array))
182
- .replace(/\+/g, "-")
183
- .replace(/\//g, "_")
184
- .replace(/=+$/, "");
185
- const hash = await crypto.subtle.digest("SHA-256", new TextEncoder().encode(verifier));
186
- const challenge = btoa(String.fromCharCode(...new Uint8Array(hash)))
187
- .replace(/\+/g, "-")
188
- .replace(/\//g, "_")
189
- .replace(/=+$/, "");
190
- return { verifier, challenge };
191
- }
192
-
193
- async function loginGitLab(callbacks: OAuthLoginCallbacks): Promise<OAuthCredentials> {
194
- const { verifier, challenge } = await generatePKCE();
195
- const authParams = new URLSearchParams({
196
- client_id: BUNDLED_CLIENT_ID,
197
- redirect_uri: REDIRECT_URI,
198
- response_type: "code",
199
- scope: OAUTH_SCOPES.join(" "),
200
- code_challenge: challenge,
201
- code_challenge_method: "S256",
202
- state: crypto.randomUUID(),
203
- });
204
-
205
- callbacks.onAuth({ url: `${GITLAB_COM_URL}/oauth/authorize?${authParams.toString()}` });
206
- const callbackUrl = await callbacks.onPrompt({ message: "Paste the callback URL:" });
207
- const code = new URL(callbackUrl).searchParams.get("code");
208
- if (!code) throw new Error("No authorization code found in callback URL");
209
-
210
- const tokenResponse = await fetch(`${GITLAB_COM_URL}/oauth/token`, {
211
- method: "POST",
212
- headers: { "Content-Type": "application/x-www-form-urlencoded" },
213
- body: new URLSearchParams({
214
- client_id: BUNDLED_CLIENT_ID,
215
- grant_type: "authorization_code",
216
- code,
217
- code_verifier: verifier,
218
- redirect_uri: REDIRECT_URI,
219
- }).toString(),
220
- });
221
-
222
- if (!tokenResponse.ok) throw new Error(`Token exchange failed: ${await tokenResponse.text()}`);
223
- const data = (await tokenResponse.json()) as {
224
- access_token: string;
225
- refresh_token: string;
226
- expires_in: number;
227
- created_at: number;
228
- };
229
- invalidateDirectAccessToken();
230
- return {
231
- refresh: data.refresh_token,
232
- access: data.access_token,
233
- expires: (data.created_at + data.expires_in) * 1000 - 5 * 60 * 1000,
234
- };
235
- }
236
-
237
- async function refreshGitLabToken(credentials: OAuthCredentials): Promise<OAuthCredentials> {
238
- const response = await fetch(`${GITLAB_COM_URL}/oauth/token`, {
239
- method: "POST",
240
- headers: { "Content-Type": "application/x-www-form-urlencoded" },
241
- body: new URLSearchParams({
242
- client_id: BUNDLED_CLIENT_ID,
243
- grant_type: "refresh_token",
244
- refresh_token: credentials.refresh,
245
- }).toString(),
246
- });
247
- if (!response.ok) throw new Error(`Token refresh failed: ${await response.text()}`);
248
- const data = (await response.json()) as {
249
- access_token: string;
250
- refresh_token: string;
251
- expires_in: number;
252
- created_at: number;
253
- };
254
- invalidateDirectAccessToken();
255
- return {
256
- refresh: data.refresh_token,
257
- access: data.access_token,
258
- expires: (data.created_at + data.expires_in) * 1000 - 5 * 60 * 1000,
259
- };
260
- }
261
-
262
- // =============================================================================
263
- // Stream Function
264
- // =============================================================================
265
-
266
- export function streamGitLabDuo(
267
- model: Model<Api>,
268
- context: Context,
269
- options?: SimpleStreamOptions,
270
- ): AssistantMessageEventStream {
271
- const stream = createAssistantMessageEventStream();
272
-
273
- (async () => {
274
- try {
275
- const gitlabAccessToken = options?.apiKey;
276
- if (!gitlabAccessToken) throw new Error("No GitLab access token. Run /login gitlab-duo or set GITLAB_TOKEN");
277
-
278
- const cfg = MODEL_MAP.get(model.id);
279
- if (!cfg) throw new Error(`Unknown model: ${model.id}`);
280
-
281
- const directAccess = await getDirectAccessToken(gitlabAccessToken);
282
- const modelWithBaseUrl = { ...model, baseUrl: cfg.baseUrl };
283
- const headers = { ...directAccess.headers, Authorization: `Bearer ${directAccess.token}` };
284
- const streamOptions = { ...options, apiKey: "gitlab-duo", headers };
285
-
286
- const innerStream =
287
- cfg.backend === "anthropic"
288
- ? streamSimpleAnthropic(modelWithBaseUrl as Model<"anthropic-messages">, context, streamOptions)
289
- : streamSimpleOpenAIResponses(modelWithBaseUrl as Model<"openai-responses">, context, streamOptions);
290
-
291
- for await (const event of innerStream) stream.push(event);
292
- stream.end();
293
- } catch (error) {
294
- stream.push({
295
- type: "error",
296
- reason: "error",
297
- error: {
298
- role: "assistant",
299
- content: [],
300
- api: model.api,
301
- provider: model.provider,
302
- model: model.id,
303
- usage: {
304
- input: 0,
305
- output: 0,
306
- cacheRead: 0,
307
- cacheWrite: 0,
308
- totalTokens: 0,
309
- cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0, total: 0 },
310
- },
311
- stopReason: "error",
312
- errorMessage: error instanceof Error ? error.message : String(error),
313
- timestamp: Date.now(),
314
- },
315
- });
316
- stream.end();
317
- }
318
- })();
319
-
320
- return stream;
321
- }
322
-
323
- // =============================================================================
324
- // Extension Entry Point
325
- // =============================================================================
326
-
327
- export default function (pi: ExtensionAPI) {
328
- pi.registerProvider("gitlab-duo", {
329
- baseUrl: AI_GATEWAY_URL,
330
- apiKey: "GITLAB_TOKEN",
331
- api: "gitlab-duo-api",
332
- models: MODELS.map(({ id, name, reasoning, input, cost, contextWindow, maxTokens }) => ({
333
- id,
334
- name,
335
- reasoning,
336
- input,
337
- cost,
338
- contextWindow,
339
- maxTokens,
340
- })),
341
- oauth: {
342
- name: "GitLab Duo",
343
- login: loginGitLab,
344
- refreshToken: refreshGitLabToken,
345
- getApiKey: (cred) => cred.access,
346
- },
347
- streamSimple: streamGitLabDuo,
348
- });
349
- }
350
-
@@ -1,16 +0,0 @@
1
- {
2
- "name": "pi-extension-custom-provider-gitlab-duo",
3
- "private": true,
4
- "version": "1.8.1",
5
- "type": "module",
6
- "scripts": {
7
- "clean": "echo 'nothing to clean'",
8
- "build": "echo 'nothing to build'",
9
- "check": "echo 'nothing to check'"
10
- },
11
- "pi": {
12
- "extensions": [
13
- "./index.ts"
14
- ]
15
- }
16
- }
@@ -1,82 +0,0 @@
1
- /**
2
- * Test script for GitLab Duo extension
3
- * Run: npx tsx test.ts [model-id] [--thinking]
4
- *
5
- * Examples:
6
- * npx tsx test.ts # Test default (claude-sonnet-4-5-20250929)
7
- * npx tsx test.ts gpt-5-codex # Test GPT-5 Codex
8
- * npx tsx test.ts claude-sonnet-4-5-20250929 --thinking
9
- */
10
-
11
- import { type Api, type Context, type Model, registerApiProvider, streamSimple } from "@apholdings/jensen-ai";
12
- import { readFileSync } from "fs";
13
- import { homedir } from "os";
14
- import { join } from "path";
15
- import { MODELS, streamGitLabDuo } from "./index.js";
16
-
17
- const MODEL_MAP = new Map(MODELS.map((m) => [m.id, m]));
18
-
19
- async function main() {
20
- const modelId = process.argv[2] || "claude-sonnet-4-5-20250929";
21
- const useThinking = process.argv.includes("--thinking");
22
-
23
- const cfg = MODEL_MAP.get(modelId);
24
- if (!cfg) {
25
- console.error(`Unknown model: ${modelId}`);
26
- console.error("Available:", MODELS.map((m) => m.id).join(", "));
27
- process.exit(1);
28
- }
29
-
30
- // Read auth
31
- const authPath = join(homedir(), ".pi", "agent", "auth.json");
32
- const authData = JSON.parse(readFileSync(authPath, "utf-8"));
33
- const gitlabCred = authData["gitlab-duo"];
34
- if (!gitlabCred?.access) {
35
- console.error("No gitlab-duo credentials. Run /login gitlab-duo first.");
36
- process.exit(1);
37
- }
38
-
39
- // Register provider
40
- registerApiProvider({
41
- api: "gitlab-duo-api" as Api,
42
- stream: streamGitLabDuo,
43
- streamSimple: streamGitLabDuo,
44
- });
45
-
46
- // Create model
47
- const model: Model<Api> = {
48
- id: cfg.id,
49
- name: cfg.name,
50
- api: "gitlab-duo-api" as Api,
51
- provider: "gitlab-duo",
52
- baseUrl: cfg.baseUrl,
53
- reasoning: cfg.reasoning,
54
- input: cfg.input,
55
- cost: cfg.cost,
56
- contextWindow: cfg.contextWindow,
57
- maxTokens: cfg.maxTokens,
58
- };
59
-
60
- const context: Context = {
61
- messages: [{ role: "user", content: "Say hello in exactly 3 words.", timestamp: Date.now() }],
62
- };
63
-
64
- console.log(`Model: ${model.id}, Backend: ${cfg.backend}, Thinking: ${useThinking}`);
65
-
66
- const stream = streamSimple(model, context, {
67
- apiKey: gitlabCred.access,
68
- maxTokens: 100,
69
- reasoning: useThinking ? "low" : undefined,
70
- });
71
-
72
- for await (const event of stream) {
73
- if (event.type === "thinking_start") console.log("[Thinking]");
74
- else if (event.type === "thinking_delta") process.stdout.write(event.delta);
75
- else if (event.type === "thinking_end") console.log("\n[/Thinking]\n");
76
- else if (event.type === "text_delta") process.stdout.write(event.delta);
77
- else if (event.type === "error") console.error("\nError:", event.error.errorMessage);
78
- else if (event.type === "done") console.log("\n\nDone!", event.reason, event.message.usage);
79
- }
80
- }
81
-
82
- main().catch(console.error);