vite-plugin-opencode-assistant 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +282 -0
- package/dist/client.d.ts +0 -0
- package/dist/client.js +1549 -0
- package/dist/constants.d.ts +76 -0
- package/dist/constants.js +77 -0
- package/dist/index.d.ts +23 -0
- package/dist/index.js +448 -0
- package/dist/injector.d.ts +20 -0
- package/dist/injector.js +24 -0
- package/dist/plugins/page-context.d.ts +20 -0
- package/dist/plugins/page-context.js +121 -0
- package/dist/types.d.ts +130 -0
- package/dist/types.js +2 -0
- package/dist/utils.d.ts +42 -0
- package/dist/utils.js +156 -0
- package/dist/web.d.ts +18 -0
- package/dist/web.js +95 -0
- package/package.json +54 -0
package/dist/index.js
ADDED
|
@@ -0,0 +1,448 @@
|
|
|
1
|
+
import path from 'path';
|
|
2
|
+
import fs from 'fs';
|
|
3
|
+
import http from 'http';
|
|
4
|
+
import { startOpenCodeWeb } from './web';
|
|
5
|
+
import { injectWidget } from './injector';
|
|
6
|
+
import { checkOpenCodeInstalled, findAvailablePort, killProcessOnPort, checkOpenCodeProcess } from './utils';
|
|
7
|
+
import { DEFAULT_CONFIG, DEFAULT_RETRIES, RETRY_DELAY, PROCESS_KILL_DELAY, LOG_PREFIX, WIDGET_SCRIPT_PATH, CONTEXT_API_PATH, START_API_PATH, SESSIONS_API_PATH, } from './constants';
|
|
8
|
+
/**
|
|
9
|
+
* OpenCode Vite 插件
|
|
10
|
+
* @param options - 插件配置选项
|
|
11
|
+
* @returns Vite 插件实例
|
|
12
|
+
* @example
|
|
13
|
+
* ```ts
|
|
14
|
+
* // vite.config.ts
|
|
15
|
+
* import opencode from 'vite-plugin-opencode'
|
|
16
|
+
*
|
|
17
|
+
* export default {
|
|
18
|
+
* plugins: [
|
|
19
|
+
* opencode({
|
|
20
|
+
* webPort: 4097,
|
|
21
|
+
* position: 'bottom-right',
|
|
22
|
+
* theme: 'auto'
|
|
23
|
+
* })
|
|
24
|
+
* ]
|
|
25
|
+
* }
|
|
26
|
+
* ```
|
|
27
|
+
*/
|
|
28
|
+
export default function opencodePlugin(options = {}) {
|
|
29
|
+
let webProcess = null;
|
|
30
|
+
let sessionUrl = null;
|
|
31
|
+
let actualWebPort = DEFAULT_CONFIG.webPort;
|
|
32
|
+
let isStarted = false;
|
|
33
|
+
let startPromise = null;
|
|
34
|
+
let pageContext = { url: '', title: '' };
|
|
35
|
+
const config = { ...DEFAULT_CONFIG, ...options };
|
|
36
|
+
/**
|
|
37
|
+
* 输出日志(仅在 verbose 模式下)
|
|
38
|
+
*/
|
|
39
|
+
function log(message, ...args) {
|
|
40
|
+
if (config.verbose) {
|
|
41
|
+
console.log(`${LOG_PREFIX} ${message}`, ...args);
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* 输出错误日志
|
|
46
|
+
*/
|
|
47
|
+
function logError(message, ...args) {
|
|
48
|
+
console.error(`${LOG_PREFIX} ${message}`, ...args);
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Base64 编码字符串
|
|
52
|
+
*/
|
|
53
|
+
function base64Encode(str) {
|
|
54
|
+
return Buffer.from(str).toString('base64');
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* 延迟执行
|
|
58
|
+
*/
|
|
59
|
+
function sleep(ms) {
|
|
60
|
+
return new Promise(resolve => setTimeout(resolve, ms));
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* 创建 HTTP 请求 Promise
|
|
64
|
+
*/
|
|
65
|
+
function createHttpRequest(options, body) {
|
|
66
|
+
return new Promise((resolve, reject) => {
|
|
67
|
+
const req = http.request(options, (res) => {
|
|
68
|
+
let data = '';
|
|
69
|
+
res.on('data', chunk => data += chunk);
|
|
70
|
+
res.on('end', () => {
|
|
71
|
+
try {
|
|
72
|
+
resolve(JSON.parse(data));
|
|
73
|
+
}
|
|
74
|
+
catch {
|
|
75
|
+
reject(new Error(`JSON parse error: ${data.substring(0, 100)}`));
|
|
76
|
+
}
|
|
77
|
+
});
|
|
78
|
+
});
|
|
79
|
+
req.on('error', reject);
|
|
80
|
+
if (body)
|
|
81
|
+
req.write(body);
|
|
82
|
+
req.end();
|
|
83
|
+
});
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* 获取会话列表
|
|
87
|
+
*/
|
|
88
|
+
async function getSessions(retries = DEFAULT_RETRIES) {
|
|
89
|
+
let lastError = null;
|
|
90
|
+
for (let i = 0; i < retries; i++) {
|
|
91
|
+
try {
|
|
92
|
+
return await createHttpRequest({
|
|
93
|
+
hostname: config.hostname,
|
|
94
|
+
port: actualWebPort,
|
|
95
|
+
path: '/session',
|
|
96
|
+
});
|
|
97
|
+
}
|
|
98
|
+
catch (e) {
|
|
99
|
+
lastError = e instanceof Error ? e : new Error(String(e));
|
|
100
|
+
if (i < retries - 1)
|
|
101
|
+
await sleep(RETRY_DELAY);
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
throw lastError;
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* 创建新会话
|
|
108
|
+
*/
|
|
109
|
+
async function createSession(retries = DEFAULT_RETRIES) {
|
|
110
|
+
let lastError = null;
|
|
111
|
+
for (let i = 0; i < retries; i++) {
|
|
112
|
+
try {
|
|
113
|
+
return await createHttpRequest({
|
|
114
|
+
hostname: config.hostname,
|
|
115
|
+
port: actualWebPort,
|
|
116
|
+
path: '/session',
|
|
117
|
+
method: 'POST',
|
|
118
|
+
});
|
|
119
|
+
}
|
|
120
|
+
catch (e) {
|
|
121
|
+
lastError = e instanceof Error ? e : new Error(String(e));
|
|
122
|
+
if (i < retries - 1)
|
|
123
|
+
await sleep(RETRY_DELAY);
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
throw lastError;
|
|
127
|
+
}
|
|
128
|
+
/**
|
|
129
|
+
* 删除会话
|
|
130
|
+
*/
|
|
131
|
+
async function deleteSession(sessionId, retries = DEFAULT_RETRIES) {
|
|
132
|
+
let lastError = null;
|
|
133
|
+
for (let i = 0; i < retries; i++) {
|
|
134
|
+
try {
|
|
135
|
+
await createHttpRequest({
|
|
136
|
+
hostname: config.hostname,
|
|
137
|
+
port: actualWebPort,
|
|
138
|
+
path: `/session/${sessionId}`,
|
|
139
|
+
method: 'DELETE',
|
|
140
|
+
});
|
|
141
|
+
return;
|
|
142
|
+
}
|
|
143
|
+
catch (e) {
|
|
144
|
+
lastError = e instanceof Error ? e : new Error(String(e));
|
|
145
|
+
if (i < retries - 1)
|
|
146
|
+
await sleep(RETRY_DELAY);
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
throw lastError;
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* 获取或创建会话
|
|
153
|
+
*/
|
|
154
|
+
async function getOrCreateSession() {
|
|
155
|
+
const projectDir = process.cwd();
|
|
156
|
+
log('Getting sessions...');
|
|
157
|
+
const sessions = await getSessions();
|
|
158
|
+
log(`Found ${sessions.length} sessions`);
|
|
159
|
+
const matchingSession = sessions.find(s => s.directory === projectDir);
|
|
160
|
+
if (matchingSession) {
|
|
161
|
+
return `http://${config.hostname}:${actualWebPort}/${base64Encode(projectDir)}/session/${matchingSession.id}`;
|
|
162
|
+
}
|
|
163
|
+
log('Creating new session...');
|
|
164
|
+
const newSession = await createSession();
|
|
165
|
+
return `http://${config.hostname}:${actualWebPort}/${base64Encode(projectDir)}/session/${newSession.id}`;
|
|
166
|
+
}
|
|
167
|
+
/**
|
|
168
|
+
* 设置 OpenCode 插件
|
|
169
|
+
*/
|
|
170
|
+
function setupOpenCodePlugin() {
|
|
171
|
+
const projectDir = process.cwd();
|
|
172
|
+
const cacheDir = path.join(projectDir, 'node_modules', '.cache', 'opencode');
|
|
173
|
+
const pluginsDir = path.join(cacheDir, 'plugins');
|
|
174
|
+
if (!fs.existsSync(pluginsDir)) {
|
|
175
|
+
fs.mkdirSync(pluginsDir, { recursive: true });
|
|
176
|
+
}
|
|
177
|
+
const pluginSourcePath = path.join(__dirname, 'plugins', 'page-context.js');
|
|
178
|
+
const pluginTargetPath = path.join(pluginsDir, 'page-context.js');
|
|
179
|
+
if (fs.existsSync(pluginSourcePath)) {
|
|
180
|
+
fs.copyFileSync(pluginSourcePath, pluginTargetPath);
|
|
181
|
+
log(`Plugin installed to ${pluginTargetPath}`);
|
|
182
|
+
}
|
|
183
|
+
else {
|
|
184
|
+
console.warn(`${LOG_PREFIX} Plugin source not found: ${pluginSourcePath}`);
|
|
185
|
+
}
|
|
186
|
+
return cacheDir;
|
|
187
|
+
}
|
|
188
|
+
/**
|
|
189
|
+
* 启动服务
|
|
190
|
+
*/
|
|
191
|
+
async function startServices(corsOrigins, contextApiUrl) {
|
|
192
|
+
if (isStarted)
|
|
193
|
+
return;
|
|
194
|
+
if (startPromise)
|
|
195
|
+
return startPromise;
|
|
196
|
+
startPromise = (async () => {
|
|
197
|
+
log('Starting OpenCode services...');
|
|
198
|
+
if (!await checkOpenCodeInstalled()) {
|
|
199
|
+
logError(`OpenCode is not installed!
|
|
200
|
+
|
|
201
|
+
Please install OpenCode first:
|
|
202
|
+
|
|
203
|
+
# Using Homebrew (macOS)
|
|
204
|
+
brew install opencode-ai/tap/opencode
|
|
205
|
+
|
|
206
|
+
# Or using the install script
|
|
207
|
+
curl -fsSL https://opencode.ai/install | bash
|
|
208
|
+
`);
|
|
209
|
+
return;
|
|
210
|
+
}
|
|
211
|
+
actualWebPort = await findAvailablePort(config.webPort, config.hostname);
|
|
212
|
+
if (actualWebPort !== config.webPort) {
|
|
213
|
+
log(`Port ${config.webPort} is in use, using ${actualWebPort} instead`);
|
|
214
|
+
}
|
|
215
|
+
const existingProcess = await checkOpenCodeProcess(actualWebPort);
|
|
216
|
+
if (existingProcess) {
|
|
217
|
+
log(`Found existing OpenCode process on port ${actualWebPort}`);
|
|
218
|
+
}
|
|
219
|
+
else {
|
|
220
|
+
const killed = await killProcessOnPort(actualWebPort, config.hostname);
|
|
221
|
+
if (killed) {
|
|
222
|
+
log(`Killed stale process on port ${actualWebPort}`);
|
|
223
|
+
await sleep(PROCESS_KILL_DELAY);
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
const configDir = setupOpenCodePlugin();
|
|
227
|
+
if (!existingProcess) {
|
|
228
|
+
webProcess = await startOpenCodeWeb({
|
|
229
|
+
port: actualWebPort,
|
|
230
|
+
hostname: config.hostname,
|
|
231
|
+
serverUrl: '',
|
|
232
|
+
cwd: process.cwd(),
|
|
233
|
+
configDir,
|
|
234
|
+
corsOrigins,
|
|
235
|
+
contextApiUrl,
|
|
236
|
+
});
|
|
237
|
+
log(`OpenCode Web started at http://${config.hostname}:${actualWebPort}`);
|
|
238
|
+
}
|
|
239
|
+
try {
|
|
240
|
+
sessionUrl = await getOrCreateSession();
|
|
241
|
+
log(`Session URL: ${sessionUrl}`);
|
|
242
|
+
}
|
|
243
|
+
catch (e) {
|
|
244
|
+
console.warn(`${LOG_PREFIX} Failed to get/create session:`, e);
|
|
245
|
+
}
|
|
246
|
+
isStarted = true;
|
|
247
|
+
})();
|
|
248
|
+
return startPromise;
|
|
249
|
+
}
|
|
250
|
+
/**
|
|
251
|
+
* 停止服务
|
|
252
|
+
*/
|
|
253
|
+
async function stopServices() {
|
|
254
|
+
log('Stopping OpenCode services...');
|
|
255
|
+
if (webProcess) {
|
|
256
|
+
webProcess.kill('SIGTERM');
|
|
257
|
+
await new Promise(resolve => {
|
|
258
|
+
webProcess?.on('exit', () => resolve());
|
|
259
|
+
setTimeout(() => {
|
|
260
|
+
webProcess?.kill('SIGKILL');
|
|
261
|
+
resolve();
|
|
262
|
+
}, 3000);
|
|
263
|
+
});
|
|
264
|
+
webProcess = null;
|
|
265
|
+
}
|
|
266
|
+
if (isStarted) {
|
|
267
|
+
await killProcessOnPort(actualWebPort, config.hostname);
|
|
268
|
+
}
|
|
269
|
+
isStarted = false;
|
|
270
|
+
startPromise = null;
|
|
271
|
+
}
|
|
272
|
+
/**
|
|
273
|
+
* 处理上下文 API 请求
|
|
274
|
+
*/
|
|
275
|
+
function handleContextRequest(req, res) {
|
|
276
|
+
res.setHeader('Content-Type', 'application/json');
|
|
277
|
+
res.setHeader('Access-Control-Allow-Origin', '*');
|
|
278
|
+
res.setHeader('Access-Control-Allow-Methods', 'GET, POST, DELETE, OPTIONS');
|
|
279
|
+
res.setHeader('Access-Control-Allow-Headers', 'Content-Type');
|
|
280
|
+
if (req.method === 'OPTIONS') {
|
|
281
|
+
res.writeHead(200);
|
|
282
|
+
res.end();
|
|
283
|
+
return;
|
|
284
|
+
}
|
|
285
|
+
if (req.method === 'GET') {
|
|
286
|
+
res.writeHead(200);
|
|
287
|
+
res.end(JSON.stringify(pageContext));
|
|
288
|
+
return;
|
|
289
|
+
}
|
|
290
|
+
if (req.method === 'DELETE') {
|
|
291
|
+
pageContext.selectedElements = [];
|
|
292
|
+
log('Selected elements cleared');
|
|
293
|
+
res.writeHead(200);
|
|
294
|
+
res.end(JSON.stringify({ success: true }));
|
|
295
|
+
return;
|
|
296
|
+
}
|
|
297
|
+
if (req.method === 'POST') {
|
|
298
|
+
let body = '';
|
|
299
|
+
req.on('data', chunk => body += chunk);
|
|
300
|
+
req.on('end', () => {
|
|
301
|
+
try {
|
|
302
|
+
const data = JSON.parse(body);
|
|
303
|
+
pageContext = {
|
|
304
|
+
url: data.url || '',
|
|
305
|
+
title: data.title || '',
|
|
306
|
+
selectedElements: data.selectedElements || [],
|
|
307
|
+
};
|
|
308
|
+
log(`Context updated: ${pageContext.url}`, data.selectedElements?.length ? `elements: ${data.selectedElements.length}` : '');
|
|
309
|
+
res.writeHead(200);
|
|
310
|
+
res.end(JSON.stringify({ success: true }));
|
|
311
|
+
}
|
|
312
|
+
catch {
|
|
313
|
+
res.writeHead(400);
|
|
314
|
+
res.end(JSON.stringify({ error: 'Invalid JSON' }));
|
|
315
|
+
}
|
|
316
|
+
});
|
|
317
|
+
return;
|
|
318
|
+
}
|
|
319
|
+
res.writeHead(405);
|
|
320
|
+
res.end(JSON.stringify({ error: 'Method not allowed' }));
|
|
321
|
+
}
|
|
322
|
+
return {
|
|
323
|
+
name: 'vite-plugin-opencode',
|
|
324
|
+
async configureServer(server) {
|
|
325
|
+
if (!config.enabled)
|
|
326
|
+
return;
|
|
327
|
+
const vitePort = server.config.server.port || 5173;
|
|
328
|
+
const viteHost = server.config.server.host || 'localhost';
|
|
329
|
+
const viteOrigin = `http://${viteHost}:${vitePort}`;
|
|
330
|
+
const contextApiUrl = `http://${viteHost}:${vitePort}${CONTEXT_API_PATH}`;
|
|
331
|
+
if (!config.lazy) {
|
|
332
|
+
await startServices([viteOrigin], contextApiUrl);
|
|
333
|
+
}
|
|
334
|
+
server.middlewares.use(WIDGET_SCRIPT_PATH, async (_req, res) => {
|
|
335
|
+
if (config.lazy && !isStarted) {
|
|
336
|
+
await startServices([viteOrigin], contextApiUrl);
|
|
337
|
+
}
|
|
338
|
+
const widgetPath = path.join(__dirname, 'client.js');
|
|
339
|
+
if (fs.existsSync(widgetPath)) {
|
|
340
|
+
res.setHeader('Content-Type', 'application/javascript');
|
|
341
|
+
res.setHeader('Access-Control-Allow-Origin', '*');
|
|
342
|
+
fs.createReadStream(widgetPath).pipe(res);
|
|
343
|
+
}
|
|
344
|
+
else {
|
|
345
|
+
res.writeHead(404);
|
|
346
|
+
res.end('Widget script not found');
|
|
347
|
+
}
|
|
348
|
+
});
|
|
349
|
+
server.middlewares.use(CONTEXT_API_PATH, async (req, res) => {
|
|
350
|
+
if (config.lazy && !isStarted) {
|
|
351
|
+
await startServices([viteOrigin], contextApiUrl);
|
|
352
|
+
}
|
|
353
|
+
handleContextRequest(req, res);
|
|
354
|
+
});
|
|
355
|
+
server.middlewares.use(START_API_PATH, async (_req, res) => {
|
|
356
|
+
res.setHeader('Content-Type', 'application/json');
|
|
357
|
+
res.setHeader('Access-Control-Allow-Origin', '*');
|
|
358
|
+
if (config.lazy && !isStarted) {
|
|
359
|
+
try {
|
|
360
|
+
await startServices([viteOrigin], contextApiUrl);
|
|
361
|
+
res.writeHead(200);
|
|
362
|
+
res.end(JSON.stringify({ success: true, sessionUrl }));
|
|
363
|
+
}
|
|
364
|
+
catch (e) {
|
|
365
|
+
res.writeHead(500);
|
|
366
|
+
res.end(JSON.stringify({ error: String(e) }));
|
|
367
|
+
}
|
|
368
|
+
}
|
|
369
|
+
else {
|
|
370
|
+
res.writeHead(200);
|
|
371
|
+
res.end(JSON.stringify({ success: true, sessionUrl }));
|
|
372
|
+
}
|
|
373
|
+
});
|
|
374
|
+
server.middlewares.use(SESSIONS_API_PATH, async (req, res) => {
|
|
375
|
+
res.setHeader('Content-Type', 'application/json');
|
|
376
|
+
res.setHeader('Access-Control-Allow-Origin', '*');
|
|
377
|
+
res.setHeader('Access-Control-Allow-Methods', 'GET, POST, DELETE, OPTIONS');
|
|
378
|
+
res.setHeader('Access-Control-Allow-Headers', 'Content-Type');
|
|
379
|
+
if (req.method === 'OPTIONS') {
|
|
380
|
+
res.writeHead(200);
|
|
381
|
+
res.end();
|
|
382
|
+
return;
|
|
383
|
+
}
|
|
384
|
+
if (config.lazy && !isStarted) {
|
|
385
|
+
await startServices([viteOrigin], contextApiUrl);
|
|
386
|
+
}
|
|
387
|
+
try {
|
|
388
|
+
if (req.method === 'GET') {
|
|
389
|
+
const sessions = await getSessions();
|
|
390
|
+
res.writeHead(200);
|
|
391
|
+
res.end(JSON.stringify(sessions));
|
|
392
|
+
}
|
|
393
|
+
else if (req.method === 'POST') {
|
|
394
|
+
const newSession = await createSession();
|
|
395
|
+
res.writeHead(200);
|
|
396
|
+
res.end(JSON.stringify(newSession));
|
|
397
|
+
}
|
|
398
|
+
else if (req.method === 'DELETE') {
|
|
399
|
+
const url = new URL(req.url || '', `http://${req.headers.host}`);
|
|
400
|
+
const sessionId = url.searchParams.get('id');
|
|
401
|
+
if (!sessionId) {
|
|
402
|
+
res.writeHead(400);
|
|
403
|
+
res.end(JSON.stringify({ error: 'Session ID is required' }));
|
|
404
|
+
return;
|
|
405
|
+
}
|
|
406
|
+
await deleteSession(sessionId);
|
|
407
|
+
res.writeHead(200);
|
|
408
|
+
res.end(JSON.stringify({ success: true }));
|
|
409
|
+
}
|
|
410
|
+
else {
|
|
411
|
+
res.writeHead(405);
|
|
412
|
+
res.end(JSON.stringify({ error: 'Method not allowed' }));
|
|
413
|
+
}
|
|
414
|
+
}
|
|
415
|
+
catch (e) {
|
|
416
|
+
res.writeHead(500);
|
|
417
|
+
res.end(JSON.stringify({ error: String(e) }));
|
|
418
|
+
}
|
|
419
|
+
});
|
|
420
|
+
server.httpServer?.on('close', stopServices);
|
|
421
|
+
const cleanup = async () => {
|
|
422
|
+
await stopServices();
|
|
423
|
+
process.exit(0);
|
|
424
|
+
};
|
|
425
|
+
process.on('SIGINT', cleanup);
|
|
426
|
+
process.on('SIGTERM', cleanup);
|
|
427
|
+
process.on('exit', () => {
|
|
428
|
+
webProcess?.kill('SIGKILL');
|
|
429
|
+
});
|
|
430
|
+
},
|
|
431
|
+
transformIndexHtml(html) {
|
|
432
|
+
const widget = injectWidget({
|
|
433
|
+
webUrl: `http://${config.hostname}:${actualWebPort}`,
|
|
434
|
+
serverUrl: `http://${config.hostname}:${actualWebPort}`,
|
|
435
|
+
position: config.position,
|
|
436
|
+
theme: config.theme,
|
|
437
|
+
open: config.open,
|
|
438
|
+
autoReload: config.autoReload,
|
|
439
|
+
cwd: process.cwd(),
|
|
440
|
+
sessionUrl: sessionUrl || undefined,
|
|
441
|
+
lazy: config.lazy,
|
|
442
|
+
hotkey: config.hotkey,
|
|
443
|
+
});
|
|
444
|
+
return html.replace('</body>', `${widget}</body>`);
|
|
445
|
+
},
|
|
446
|
+
};
|
|
447
|
+
}
|
|
448
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Plugin, ViteDevServer } from 'vite'\nimport { spawn, ChildProcess } from 'child_process'\nimport path from 'path'\nimport fs from 'fs'\nimport http from 'http'\nimport { startOpenCodeWeb } from './web'\nimport { injectWidget } from './injector'\nimport { checkOpenCodeInstalled, findAvailablePort, killProcessOnPort, checkOpenCodeProcess } from './utils'\nimport { OpenCodeOptions, SessionInfo, PageContext } from './types'\nimport {\n  DEFAULT_CONFIG,\n  DEFAULT_RETRIES,\n  RETRY_DELAY,\n  PROCESS_KILL_DELAY,\n  LOG_PREFIX,\n  WIDGET_SCRIPT_PATH,\n  CONTEXT_API_PATH,\n  START_API_PATH,\n  SESSIONS_API_PATH,\n} from './constants'\n\n/**\n * OpenCode Vite 插件\n * @param options - 插件配置选项\n * @returns Vite 插件实例\n * @example\n * ```ts\n * // vite.config.ts\n * import opencode from 'vite-plugin-opencode'\n *\n * export default {\n *   plugins: [\n *     opencode({\n *       webPort: 4097,\n *       position: 'bottom-right',\n *       theme: 'auto'\n *     })\n *   ]\n * }\n * ```\n */\nexport default function opencodePlugin(options: OpenCodeOptions = {}): Plugin {\n  let webProcess: ChildProcess | null = null\n  let sessionUrl: string | null = null\n  let actualWebPort: number = DEFAULT_CONFIG.webPort\n  let isStarted = false\n  let startPromise: Promise<void> | null = null\n  let pageContext: PageContext = { url: '', title: '' }\n\n  const config = { ...DEFAULT_CONFIG, ...options }\n\n  /**\n   * 输出日志（仅在 verbose 模式下）\n   */\n  function log(message: string, ...args: unknown[]): void {\n    if (config.verbose) {\n      console.log(`${LOG_PREFIX} ${message}`, ...args)\n    }\n  }\n\n  /**\n   * 输出错误日志\n   */\n  function logError(message: string, ...args: unknown[]): void {\n    console.error(`${LOG_PREFIX} ${message}`, ...args)\n  }\n\n  /**\n   * Base64 编码字符串\n   */\n  function base64Encode(str: string): string {\n    return Buffer.from(str).toString('base64')\n  }\n\n  /**\n   * 延迟执行\n   */\n  function sleep(ms: number): Promise<void> {\n    return new Promise(resolve => setTimeout(resolve, ms))\n  }\n\n  /**\n   * 创建 HTTP 请求 Promise\n   */\n  function createHttpRequest<T>(options: http.RequestOptions, body?: string): Promise<T> {\n    return new Promise((resolve, reject) => {\n      const req = http.request(options, (res) => {\n        let data = ''\n        res.on('data', chunk => data += chunk)\n        res.on('end', () => {\n          try {\n            resolve(JSON.parse(data))\n          } catch {\n            reject(new Error(`JSON parse error: ${data.substring(0, 100)}`))\n          }\n        })\n      })\n      req.on('error', reject)\n      if (body) req.write(body)\n      req.end()\n    })\n  }\n\n  /**\n   * 获取会话列表\n   */\n  async function getSessions(retries = DEFAULT_RETRIES): Promise<SessionInfo[]> {\n    let lastError: Error | null = null\n\n    for (let i = 0; i < retries; i++) {\n      try {\n        return await createHttpRequest<SessionInfo[]>({\n          hostname: config.hostname,\n          port: actualWebPort,\n          path: '/session',\n        })\n      } catch (e) {\n        lastError = e instanceof Error ? e : new Error(String(e))\n        if (i < retries - 1) await sleep(RETRY_DELAY)\n      }\n    }\n\n    throw lastError\n  }\n\n  /**\n   * 创建新会话\n   */\n  async function createSession(retries = DEFAULT_RETRIES): Promise<SessionInfo> {\n    let lastError: Error | null = null\n\n    for (let i = 0; i < retries; i++) {\n      try {\n        return await createHttpRequest<SessionInfo>({\n          hostname: config.hostname,\n          port: actualWebPort,\n          path: '/session',\n          method: 'POST',\n        })\n      } catch (e) {\n        lastError = e instanceof Error ? e : new Error(String(e))\n        if (i < retries - 1) await sleep(RETRY_DELAY)\n      }\n    }\n\n    throw lastError\n  }\n\n  /**\n   * 删除会话\n   */\n  async function deleteSession(sessionId: string, retries = DEFAULT_RETRIES): Promise<void> {\n    let lastError: Error | null = null\n\n    for (let i = 0; i < retries; i++) {\n      try {\n        await createHttpRequest<void>({\n          hostname: config.hostname,\n          port: actualWebPort,\n          path: `/session/${sessionId}`,\n          method: 'DELETE',\n        })\n        return\n      } catch (e) {\n        lastError = e instanceof Error ? e : new Error(String(e))\n        if (i < retries - 1) await sleep(RETRY_DELAY)\n      }\n    }\n\n    throw lastError\n  }\n\n  /**\n   * 获取或创建会话\n   */\n  async function getOrCreateSession(): Promise<string> {\n    const projectDir = process.cwd()\n    log('Getting sessions...')\n    const sessions = await getSessions()\n    log(`Found ${sessions.length} sessions`)\n\n    const matchingSession = sessions.find(s => s.directory === projectDir)\n\n    if (matchingSession) {\n      return `http://${config.hostname}:${actualWebPort}/${base64Encode(projectDir)}/session/${matchingSession.id}`\n    }\n\n    log('Creating new session...')\n    const newSession = await createSession()\n    return `http://${config.hostname}:${actualWebPort}/${base64Encode(projectDir)}/session/${newSession.id}`\n  }\n\n  /**\n   * 设置 OpenCode 插件\n   */\n  function setupOpenCodePlugin(): string {\n    const projectDir = process.cwd()\n    const cacheDir = path.join(projectDir, 'node_modules', '.cache', 'opencode')\n    const pluginsDir = path.join(cacheDir, 'plugins')\n\n    if (!fs.existsSync(pluginsDir)) {\n      fs.mkdirSync(pluginsDir, { recursive: true })\n    }\n\n    const pluginSourcePath = path.join(__dirname, 'plugins', 'page-context.js')\n    const pluginTargetPath = path.join(pluginsDir, 'page-context.js')\n\n    if (fs.existsSync(pluginSourcePath)) {\n      fs.copyFileSync(pluginSourcePath, pluginTargetPath)\n      log(`Plugin installed to ${pluginTargetPath}`)\n    } else {\n      console.warn(`${LOG_PREFIX} Plugin source not found: ${pluginSourcePath}`)\n    }\n\n    return cacheDir\n  }\n\n  /**\n   * 启动服务\n   */\n  async function startServices(corsOrigins?: string[], contextApiUrl?: string): Promise<void> {\n    if (isStarted) return\n    if (startPromise) return startPromise\n\n    startPromise = (async () => {\n      log('Starting OpenCode services...')\n\n      if (!await checkOpenCodeInstalled()) {\n        logError(`OpenCode is not installed!\n\nPlease install OpenCode first:\n\n  # Using Homebrew (macOS)\n  brew install opencode-ai/tap/opencode\n\n  # Or using the install script\n  curl -fsSL https://opencode.ai/install | bash\n        `)\n        return\n      }\n\n      actualWebPort = await findAvailablePort(config.webPort, config.hostname)\n      if (actualWebPort !== config.webPort) {\n        log(`Port ${config.webPort} is in use, using ${actualWebPort} instead`)\n      }\n\n      const existingProcess = await checkOpenCodeProcess(actualWebPort)\n      if (existingProcess) {\n        log(`Found existing OpenCode process on port ${actualWebPort}`)\n      } else {\n        const killed = await killProcessOnPort(actualWebPort, config.hostname)\n        if (killed) {\n          log(`Killed stale process on port ${actualWebPort}`)\n          await sleep(PROCESS_KILL_DELAY)\n        }\n      }\n\n      const configDir = setupOpenCodePlugin()\n\n      if (!existingProcess) {\n        webProcess = await startOpenCodeWeb({\n          port: actualWebPort,\n          hostname: config.hostname,\n          serverUrl: '',\n          cwd: process.cwd(),\n          configDir,\n          corsOrigins,\n          contextApiUrl,\n        })\n        log(`OpenCode Web started at http://${config.hostname}:${actualWebPort}`)\n      }\n\n      try {\n        sessionUrl = await getOrCreateSession()\n        log(`Session URL: ${sessionUrl}`)\n      } catch (e) {\n        console.warn(`${LOG_PREFIX} Failed to get/create session:`, e)\n      }\n\n      isStarted = true\n    })()\n\n    return startPromise\n  }\n\n  /**\n   * 停止服务\n   */\n  async function stopServices(): Promise<void> {\n    log('Stopping OpenCode services...')\n    \n    if (webProcess) {\n      webProcess.kill('SIGTERM')\n      await new Promise<void>(resolve => {\n        webProcess?.on('exit', () => resolve())\n        setTimeout(() => {\n          webProcess?.kill('SIGKILL')\n          resolve()\n        }, 3000)\n      })\n      webProcess = null\n    }\n    \n    if (isStarted) {\n      await killProcessOnPort(actualWebPort, config.hostname)\n    }\n    \n    isStarted = false\n    startPromise = null\n  }\n\n  /**\n   * 处理上下文 API 请求\n   */\n  function handleContextRequest(req: http.IncomingMessage, res: http.ServerResponse): void {\n    res.setHeader('Content-Type', 'application/json')\n    res.setHeader('Access-Control-Allow-Origin', '*')\n    res.setHeader('Access-Control-Allow-Methods', 'GET, POST, DELETE, OPTIONS')\n    res.setHeader('Access-Control-Allow-Headers', 'Content-Type')\n\n    if (req.method === 'OPTIONS') {\n      res.writeHead(200)\n      res.end()\n      return\n    }\n\n    if (req.method === 'GET') {\n      res.writeHead(200)\n      res.end(JSON.stringify(pageContext))\n      return\n    }\n\n    if (req.method === 'DELETE') {\n      pageContext.selectedElements = []\n      log('Selected elements cleared')\n      res.writeHead(200)\n      res.end(JSON.stringify({ success: true }))\n      return\n    }\n\n    if (req.method === 'POST') {\n      let body = ''\n      req.on('data', chunk => body += chunk)\n      req.on('end', () => {\n        try {\n          const data = JSON.parse(body)\n          pageContext = {\n            url: data.url || '',\n            title: data.title || '',\n            selectedElements: data.selectedElements || [],\n          }\n          log(`Context updated: ${pageContext.url}`, data.selectedElements?.length ? `elements: ${data.selectedElements.length}` : '')\n          res.writeHead(200)\n          res.end(JSON.stringify({ success: true }))\n        } catch {\n          res.writeHead(400)\n          res.end(JSON.stringify({ error: 'Invalid JSON' }))\n        }\n      })\n      return\n    }\n\n    res.writeHead(405)\n    res.end(JSON.stringify({ error: 'Method not allowed' }))\n  }\n\n  return {\n    name: 'vite-plugin-opencode',\n\n    async configureServer(server: ViteDevServer) {\n      if (!config.enabled) return\n\n      const vitePort = server.config.server.port || 5173\n      const viteHost = server.config.server.host || 'localhost'\n      const viteOrigin = `http://${viteHost}:${vitePort}`\n      const contextApiUrl = `http://${viteHost}:${vitePort}${CONTEXT_API_PATH}`\n\n      if (!config.lazy) {\n        await startServices([viteOrigin], contextApiUrl)\n      }\n\n      server.middlewares.use(WIDGET_SCRIPT_PATH, async (_req, res) => {\n        if (config.lazy && !isStarted) {\n          await startServices([viteOrigin], contextApiUrl)\n        }\n\n        const widgetPath = path.join(__dirname, 'client.js')\n        if (fs.existsSync(widgetPath)) {\n          res.setHeader('Content-Type', 'application/javascript')\n          res.setHeader('Access-Control-Allow-Origin', '*')\n          fs.createReadStream(widgetPath).pipe(res)\n        } else {\n          res.writeHead(404)\n          res.end('Widget script not found')\n        }\n      })\n\n      server.middlewares.use(CONTEXT_API_PATH, async (req, res) => {\n        if (config.lazy && !isStarted) {\n          await startServices([viteOrigin], contextApiUrl)\n        }\n        handleContextRequest(req as http.IncomingMessage, res)\n      })\n\n      server.middlewares.use(START_API_PATH, async (_req, res) => {\n        res.setHeader('Content-Type', 'application/json')\n        res.setHeader('Access-Control-Allow-Origin', '*')\n\n        if (config.lazy && !isStarted) {\n          try {\n            await startServices([viteOrigin], contextApiUrl)\n            res.writeHead(200)\n            res.end(JSON.stringify({ success: true, sessionUrl }))\n          } catch (e) {\n            res.writeHead(500)\n            res.end(JSON.stringify({ error: String(e) }))\n          }\n        } else {\n          res.writeHead(200)\n          res.end(JSON.stringify({ success: true, sessionUrl }))\n        }\n      })\n\n      server.middlewares.use(SESSIONS_API_PATH, async (req, res) => {\n        res.setHeader('Content-Type', 'application/json')\n        res.setHeader('Access-Control-Allow-Origin', '*')\n        res.setHeader('Access-Control-Allow-Methods', 'GET, POST, DELETE, OPTIONS')\n        res.setHeader('Access-Control-Allow-Headers', 'Content-Type')\n\n        if (req.method === 'OPTIONS') {\n          res.writeHead(200)\n          res.end()\n          return\n        }\n\n        if (config.lazy && !isStarted) {\n          await startServices([viteOrigin], contextApiUrl)\n        }\n\n        try {\n          if (req.method === 'GET') {\n            const sessions = await getSessions()\n            res.writeHead(200)\n            res.end(JSON.stringify(sessions))\n          } else if (req.method === 'POST') {\n            const newSession = await createSession()\n            res.writeHead(200)\n            res.end(JSON.stringify(newSession))\n          } else if (req.method === 'DELETE') {\n            const url = new URL(req.url || '', `http://${req.headers.host}`)\n            const sessionId = url.searchParams.get('id')\n            \n            if (!sessionId) {\n              res.writeHead(400)\n              res.end(JSON.stringify({ error: 'Session ID is required' }))\n              return\n            }\n            \n            await deleteSession(sessionId)\n            res.writeHead(200)\n            res.end(JSON.stringify({ success: true }))\n          } else {\n            res.writeHead(405)\n            res.end(JSON.stringify({ error: 'Method not allowed' }))\n          }\n        } catch (e) {\n          res.writeHead(500)\n          res.end(JSON.stringify({ error: String(e) }))\n        }\n      })\n\n      server.httpServer?.on('close', stopServices)\n      \n      const cleanup = async () => {\n        await stopServices()\n        process.exit(0)\n      }\n      \n      process.on('SIGINT', cleanup)\n      process.on('SIGTERM', cleanup)\n      process.on('exit', () => {\n        webProcess?.kill('SIGKILL')\n      })\n    },\n\n    transformIndexHtml(html) {\n      const widget = injectWidget({\n        webUrl: `http://${config.hostname}:${actualWebPort}`,\n        serverUrl: `http://${config.hostname}:${actualWebPort}`,\n        position: config.position,\n        theme: config.theme,\n        open: config.open,\n        autoReload: config.autoReload,\n        cwd: process.cwd(),\n        sessionUrl: sessionUrl || undefined,\n        lazy: config.lazy,\n        hotkey: config.hotkey,\n      })\n      return html.replace('</body>', `${widget}</body>`)\n    },\n  }\n}\n"]}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { WidgetOptions } from './types';
|
|
2
|
+
/**
|
|
3
|
+
* 生成挂件注入脚本标签
|
|
4
|
+
* @param options - 挂件配置选项
|
|
5
|
+
* @returns HTML 脚本标签字符串
|
|
6
|
+
* @example
|
|
7
|
+
* ```ts
|
|
8
|
+
* const widget = injectWidget({
|
|
9
|
+
* webUrl: 'http://127.0.0.1:4097',
|
|
10
|
+
* serverUrl: 'http://127.0.0.1:4097',
|
|
11
|
+
* position: 'bottom-right',
|
|
12
|
+
* theme: 'auto',
|
|
13
|
+
* open: false,
|
|
14
|
+
* autoReload: true,
|
|
15
|
+
* cwd: process.cwd()
|
|
16
|
+
* })
|
|
17
|
+
* // 返回: '<script src="/__opencode_widget__.js" data-opencode-config="..."></script>'
|
|
18
|
+
* ```
|
|
19
|
+
*/
|
|
20
|
+
export declare function injectWidget(options: WidgetOptions): string;
|
package/dist/injector.js
ADDED
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { WIDGET_SCRIPT_PATH, CONFIG_DATA_ATTR } from './constants';
|
|
2
|
+
/**
|
|
3
|
+
* 生成挂件注入脚本标签
|
|
4
|
+
* @param options - 挂件配置选项
|
|
5
|
+
* @returns HTML 脚本标签字符串
|
|
6
|
+
* @example
|
|
7
|
+
* ```ts
|
|
8
|
+
* const widget = injectWidget({
|
|
9
|
+
* webUrl: 'http://127.0.0.1:4097',
|
|
10
|
+
* serverUrl: 'http://127.0.0.1:4097',
|
|
11
|
+
* position: 'bottom-right',
|
|
12
|
+
* theme: 'auto',
|
|
13
|
+
* open: false,
|
|
14
|
+
* autoReload: true,
|
|
15
|
+
* cwd: process.cwd()
|
|
16
|
+
* })
|
|
17
|
+
* // 返回: '<script src="/__opencode_widget__.js" data-opencode-config="..."></script>'
|
|
18
|
+
* ```
|
|
19
|
+
*/
|
|
20
|
+
export function injectWidget(options) {
|
|
21
|
+
const configBase64 = Buffer.from(JSON.stringify(options)).toString('base64');
|
|
22
|
+
return `<script src="${WIDGET_SCRIPT_PATH}" ${CONFIG_DATA_ATTR}="${configBase64}"></script>`;
|
|
23
|
+
}
|
|
24
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5qZWN0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5qZWN0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLGtCQUFrQixFQUFFLGdCQUFnQixFQUFFLE1BQU0sYUFBYSxDQUFBO0FBRWxFOzs7Ozs7Ozs7Ozs7Ozs7OztHQWlCRztBQUNILE1BQU0sVUFBVSxZQUFZLENBQUMsT0FBc0I7SUFDakQsTUFBTSxZQUFZLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsUUFBUSxDQUFDLFFBQVEsQ0FBQyxDQUFBO0lBQzVFLE9BQU8sZ0JBQWdCLGtCQUFrQixLQUFLLGdCQUFnQixLQUFLLFlBQVksYUFBYSxDQUFBO0FBQzlGLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBXaWRnZXRPcHRpb25zIH0gZnJvbSAnLi90eXBlcydcbmltcG9ydCB7IFdJREdFVF9TQ1JJUFRfUEFUSCwgQ09ORklHX0RBVEFfQVRUUiB9IGZyb20gJy4vY29uc3RhbnRzJ1xuXG4vKipcbiAqIOeUn+aIkOaMguS7tuazqOWFpeiEmuacrOagh+etvlxuICogQHBhcmFtIG9wdGlvbnMgLSDmjILku7bphY3nva7pgInpoblcbiAqIEByZXR1cm5zIEhUTUwg6ISa5pys5qCH562+5a2X56ym5LiyXG4gKiBAZXhhbXBsZVxuICogYGBgdHNcbiAqIGNvbnN0IHdpZGdldCA9IGluamVjdFdpZGdldCh7XG4gKiAgIHdlYlVybDogJ2h0dHA6Ly8xMjcuMC4wLjE6NDA5NycsXG4gKiAgIHNlcnZlclVybDogJ2h0dHA6Ly8xMjcuMC4wLjE6NDA5NycsXG4gKiAgIHBvc2l0aW9uOiAnYm90dG9tLXJpZ2h0JyxcbiAqICAgdGhlbWU6ICdhdXRvJyxcbiAqICAgb3BlbjogZmFsc2UsXG4gKiAgIGF1dG9SZWxvYWQ6IHRydWUsXG4gKiAgIGN3ZDogcHJvY2Vzcy5jd2QoKVxuICogfSlcbiAqIC8vIOi/lOWbnjogJzxzY3JpcHQgc3JjPVwiL19fb3BlbmNvZGVfd2lkZ2V0X18uanNcIiBkYXRhLW9wZW5jb2RlLWNvbmZpZz1cIi4uLlwiPjwvc2NyaXB0PidcbiAqIGBgYFxuICovXG5leHBvcnQgZnVuY3Rpb24gaW5qZWN0V2lkZ2V0KG9wdGlvbnM6IFdpZGdldE9wdGlvbnMpOiBzdHJpbmcge1xuICBjb25zdCBjb25maWdCYXNlNjQgPSBCdWZmZXIuZnJvbShKU09OLnN0cmluZ2lmeShvcHRpb25zKSkudG9TdHJpbmcoJ2Jhc2U2NCcpXG4gIHJldHVybiBgPHNjcmlwdCBzcmM9XCIke1dJREdFVF9TQ1JJUFRfUEFUSH1cIiAke0NPTkZJR19EQVRBX0FUVFJ9PVwiJHtjb25maWdCYXNlNjR9XCI+PC9zY3JpcHQ+YFxufVxuIl19
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview OpenCode 页面上下文插件
|
|
3
|
+
* @description 用于将页面上下文信息注入到 AI 对话中
|
|
4
|
+
*/
|
|
5
|
+
import type { Plugin } from "@opencode-ai/plugin";
|
|
6
|
+
/**
|
|
7
|
+
* OpenCode 页面上下文插件
|
|
8
|
+
* @returns 插件钩子
|
|
9
|
+
* @example
|
|
10
|
+
* ```ts
|
|
11
|
+
* // 在 opencode 配置中使用
|
|
12
|
+
* import { PageContextPlugin } from './plugins/page-context'
|
|
13
|
+
*
|
|
14
|
+
* export default {
|
|
15
|
+
* plugins: [PageContextPlugin]
|
|
16
|
+
* }
|
|
17
|
+
* ```
|
|
18
|
+
*/
|
|
19
|
+
export declare const PageContextPlugin: Plugin;
|
|
20
|
+
export default PageContextPlugin;
|