cdp-tunnel 1.1.0 → 1.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/cli/index.js CHANGED
@@ -358,64 +358,83 @@ program
358
358
  program
359
359
  .command('update')
360
360
  .description('自动更新 cdp-tunnel 并重启服务')
361
- .option('-p, --port <port>', '重启时指定端口', parseInt)
362
- .option('-w, --watchdog', '重启时启用看门狗')
361
+ .option('-p, --port <port>', '指定端口', parseInt)
362
+ .option('-w, --watchdog', '启用看门狗')
363
363
  .action(async (options) => {
364
364
  const config = getConfig();
365
365
  const wasRunning = isServerRunning();
366
366
  const savedPort = options.port || config.port;
367
367
  const savedWatchdog = options.watchdog;
368
-
369
- console.log('');
370
- log('cyan', '⬆ 正在检查更新...');
371
-
368
+
372
369
  try {
373
- const beforeVersion = execSync('npm view cdp-tunnel version', { encoding: 'utf8' }).trim();
374
- const localVersion = require(path.join(__dirname, '..', 'package.json')).version;
370
+ log('cyan', '🔍 检查更新...');
371
+ let latestVersion;
372
+ try {
373
+ latestVersion = execSync('npm view cdp-tunnel version', {
374
+ encoding: 'utf8',
375
+ timeout: 30000
376
+ }).trim();
377
+ } catch (err) {
378
+ log('red', '❌ 无法连接 npm registry: ' + err.message);
379
+ process.exit(1);
380
+ }
381
+
382
+ const localVersion = JSON.parse(fs.readFileSync(
383
+ path.join(__dirname, '..', 'package.json'), 'utf8'
384
+ )).version;
385
+
375
386
  log('gray', ' 当前版本: ' + localVersion);
376
- log('gray', ' 最新版本: ' + beforeVersion);
377
-
387
+ log('gray', ' 最新版本: ' + latestVersion);
388
+
389
+ if (localVersion === latestVersion) {
390
+ log('green', '✅ 已是最新版本 (' + localVersion + ')');
391
+ if (wasRunning) {
392
+ log('cyan', ' 服务器仍在运行中');
393
+ }
394
+ process.exit(0);
395
+ }
396
+
378
397
  if (wasRunning) {
379
- log('yellow', ' 正在停止服务器...');
380
- try {
381
- const pid = getServerPid();
382
- process.kill(pid, 'SIGTERM');
383
- fs.unlinkSync(PID_FILE);
398
+ log('yellow', '⏸ 停止服务器...');
399
+ const pid = getServerPid();
400
+ if (pid) {
401
+ try { process.kill(pid, 'SIGTERM'); } catch {}
384
402
  await new Promise(r => setTimeout(r, 1500));
385
- log('green', ' ✓ 服务器已停止');
386
- } catch (e) {
387
- log('yellow', ' ⚠ 停止服务器失败: ' + e.message);
388
403
  }
389
404
  }
390
-
391
- log('cyan', ' 正在更新...');
392
- execSync('npm update -g cdp-tunnel', { stdio: 'inherit' });
393
-
394
- const afterVersion = require(path.join(__dirname, '..', 'package.json')).version;
395
- if (afterVersion !== localVersion) {
396
- log('green', ' ✓ 已更新: ' + localVersion + ' → ' + afterVersion);
405
+
406
+ log('cyan', '📦 更新中 (' + localVersion + ' → ' + latestVersion + ')...');
407
+ try {
408
+ execSync('npm update -g cdp-tunnel', {
409
+ stdio: 'inherit',
410
+ timeout: 120000
411
+ });
412
+ } catch (err) {
413
+ log('red', '❌ 更新失败: ' + err.message);
414
+ process.exit(1);
415
+ }
416
+
417
+ const newVersion = JSON.parse(fs.readFileSync(
418
+ path.join(__dirname, '..', 'package.json'), 'utf8'
419
+ )).version;
420
+
421
+ if (newVersion !== localVersion) {
422
+ log('green', '✅ 已更新: ' + localVersion + ' → ' + newVersion);
397
423
  } else {
398
- log('green', ' 已是最新版本');
424
+ log('yellow', ' 版本未变化,可能需要手动更新');
399
425
  }
400
-
401
- ensureConfigDir();
402
- cleanupLogFile();
403
- startServer(savedPort, savedWatchdog);
404
- log('green', ' ✓ 服务器已重启 (端口: ' + savedPort + ')');
405
- console.log('');
406
- } catch (e) {
407
- log('red', '✗ 更新失败: ' + e.message);
408
- console.log('');
426
+
409
427
  if (wasRunning) {
410
- log('yellow', '正在尝试恢复服务器...');
411
- try {
412
- ensureConfigDir();
413
- startServer(savedPort, savedWatchdog);
414
- log('green', ' 服务器已恢复');
415
- } catch (re) {
416
- log('red', '✗ 恢复失败,请手动启动: cdp-tunnel start');
417
- }
428
+ log('cyan', '🔄 重启服务器...');
429
+ startServer(savedPort, savedWatchdog, config.autoRestart);
430
+ log('green', '✅ 服务器已重启');
431
+ } else {
432
+ log('cyan', ' 运行 cdp-tunnel start 启动服务器');
418
433
  }
434
+
435
+ process.exit(0);
436
+ } catch (err) {
437
+ log('red', '❌ 更新出错: ' + err.message);
419
438
  process.exit(1);
420
439
  }
421
440
  });
@@ -576,4 +595,19 @@ program
576
595
  }
577
596
  });
578
597
 
598
+ program.addHelpText('after', `
599
+
600
+ 常用命令:
601
+ $ cdp-tunnel start 启动服务(自动启动 Chrome)
602
+ $ cdp-tunnel start --auto-restart Chrome 断连时自动重启
603
+ $ cdp-tunnel start --watchdog 服务崩溃时自动重启
604
+ $ cdp-tunnel status 查看状态
605
+ $ cdp-tunnel update 检查并更新
606
+ $ cdp-tunnel extension 安装 Chrome 扩展
607
+
608
+ 快速开始:
609
+ $ npm install -g cdp-tunnel
610
+ $ cdp-tunnel start # 一行命令搞定!
611
+ `);
612
+
579
613
  program.parse();
@@ -216,6 +216,18 @@ String.prototype.hashCode = function() {
216
216
  var sessionId = CDPUtils.generateSessionId();
217
217
  State.mapSession(sessionId, tabId, targetId);
218
218
 
219
+ Config.getAutoMute(function(enabled) {
220
+ if (enabled) {
221
+ chrome.tabs.update(tabId, { muted: true }, function() {
222
+ if (chrome.runtime.lastError) {
223
+ Logger.error('[TabMute] Failed to mute tab ' + tabId + ':', chrome.runtime.lastError.message);
224
+ } else {
225
+ Logger.info('[TabMute] Tab muted:', tabId);
226
+ }
227
+ });
228
+ }
229
+ });
230
+
219
231
  // 将标签页添加到CDP组(添加延迟等待)
220
232
  setTimeout(function() {
221
233
  // 获取openerTabId对应的clientId
@@ -1,4 +1,17 @@
1
1
  var SpecialHandler = (function() {
2
+ function muteTabIfNeeded(tabId) {
3
+ Config.getAutoMute(function(enabled) {
4
+ if (!enabled) return;
5
+ chrome.tabs.update(tabId, { muted: true }, function() {
6
+ if (chrome.runtime.lastError) {
7
+ Logger.error('[TabMute] Failed to mute tab ' + tabId + ':', chrome.runtime.lastError.message);
8
+ } else {
9
+ Logger.info('[TabMute] Tab muted:', tabId);
10
+ }
11
+ });
12
+ });
13
+ }
14
+
2
15
  function targetSetAutoAttach(context) {
3
16
  var params = context.params;
4
17
  State.setAutoAttachConfig({
@@ -116,6 +129,8 @@ var SpecialHandler = (function() {
116
129
  function addTabToAutomationGroup(tabId, clientId) {
117
130
  Logger.info('[TabGroup] Starting addTabToAutomationGroup for tabId:', tabId, 'clientId:', clientId);
118
131
 
132
+ muteTabIfNeeded(tabId);
133
+
119
134
  var groupName;
120
135
  var groupClientId = clientId;
121
136
 
@@ -458,6 +458,35 @@
458
458
  font-size: 12px;
459
459
  }
460
460
 
461
+ .toggle-switch {
462
+ position: relative;
463
+ display: inline-block;
464
+ width: 40px;
465
+ height: 22px;
466
+ }
467
+ .toggle-switch input { opacity: 0; width: 0; height: 0; }
468
+ .toggle-slider {
469
+ position: absolute;
470
+ cursor: pointer;
471
+ top: 0; left: 0; right: 0; bottom: 0;
472
+ background-color: #ccc;
473
+ transition: .3s;
474
+ border-radius: 22px;
475
+ }
476
+ .toggle-slider:before {
477
+ position: absolute;
478
+ content: "";
479
+ height: 16px;
480
+ width: 16px;
481
+ left: 3px;
482
+ bottom: 3px;
483
+ background-color: white;
484
+ transition: .3s;
485
+ border-radius: 50%;
486
+ }
487
+ input:checked + .toggle-slider { background-color: #4CAF50; }
488
+ input:checked + .toggle-slider:before { transform: translateX(18px); }
489
+
461
490
  .divider {
462
491
  height: 1px;
463
492
  background: #e5e7eb;
@@ -721,6 +750,16 @@
721
750
  </div>
722
751
  <button class="btn btn-primary btn-block" id="connectBtn">🔗 保存并连接</button>
723
752
 
753
+ <div class="config-item" style="display: flex; justify-content: space-between; align-items: center; margin-top: 12px;">
754
+ <label style="font-size: 13px; color: #666;">
755
+ 🔇 自动静音新标签页
756
+ </label>
757
+ <label class="toggle-switch">
758
+ <input type="checkbox" id="autoMuteToggle" checked>
759
+ <span class="toggle-slider"></span>
760
+ </label>
761
+ </div>
762
+
724
763
  <div class="divider"></div>
725
764
 
726
765
  <div class="usage-guide">
@@ -258,8 +258,16 @@
258
258
  elements.serverAddress.value = result.wsAddress;
259
259
  }
260
260
  });
261
+ chrome.storage.local.get(['autoMute'], function(result) {
262
+ document.getElementById('autoMuteToggle').checked = result.autoMute !== false;
263
+ });
261
264
  }
262
265
 
266
+ document.getElementById('autoMuteToggle').addEventListener('change', function(e) {
267
+ chrome.storage.local.set({ autoMute: e.target.checked });
268
+ addLog(e.target.checked ? '🔇 自动静音已开启' : '🔊 自动静音已关闭', 'action');
269
+ });
270
+
263
271
  fetchState();
264
272
 
265
273
  setInterval(fetchState, 5000);
@@ -16,5 +16,14 @@ var Config = {
16
16
  chrome.storage.local.get(['wsAddress'], function(result) {
17
17
  callback(result.wsAddress || Config.WS_URL);
18
18
  });
19
+ },
20
+ AUTO_MUTE: true,
21
+ getAutoMute: function(callback) {
22
+ chrome.storage.local.get(['autoMute'], function(result) {
23
+ callback(result.autoMute !== false);
24
+ });
25
+ },
26
+ setAutoMute: function(enabled, callback) {
27
+ chrome.storage.local.set({ autoMute: enabled }, callback);
19
28
  }
20
29
  };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "cdp-tunnel",
3
- "version": "1.1.0",
3
+ "version": "1.3.0",
4
4
  "description": "Chrome Extension CDP Proxy - 通过 Chrome 扩展将 chrome.debugger API 暴露为 WebSocket 端点",
5
5
  "main": "server/proxy-server.js",
6
6
  "bin": "./cli/index.js",