@wendongfly/myhi 1.3.33 → 1.3.35

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/dist/admin.html CHANGED
@@ -116,6 +116,22 @@
116
116
  <div class="msg" id="pwd-msg"></div>
117
117
  </div>
118
118
 
119
+ <!-- 大模型网关 -->
120
+ <div class="card">
121
+ <h3>大模型网关</h3>
122
+ <div style="font-size:0.75rem;color:#6e7681;margin-bottom:0.6rem;line-height:1.5">
123
+ 配置后新建的 Claude 会话自动注入 <code style="background:#161b22;padding:0.1em 0.3em;border-radius:3px">ANTHROPIC_BASE_URL</code> / <code style="background:#161b22;padding:0.1em 0.3em;border-radius:3px">ANTHROPIC_API_KEY</code>
124
+ </div>
125
+ <div class="inp-row">
126
+ <input type="text" class="inp" id="gw-url" placeholder="网关 URL(如 https://gateway.example.com)">
127
+ </div>
128
+ <div class="inp-row" style="margin-top:0.4rem">
129
+ <input type="password" class="inp" id="gw-key" placeholder="Token / API Key(留空保持不变)">
130
+ <button class="btn-sm" onclick="saveGateway()">保存</button>
131
+ </div>
132
+ <div class="msg" id="gw-msg"></div>
133
+ </div>
134
+
119
135
  <div class="uptime" id="uptime"></div>
120
136
  </div>
121
137
 
@@ -147,6 +163,43 @@ async function showAdmin() {
147
163
  document.getElementById('pg-admin').style.display = '';
148
164
  await refresh();
149
165
  setInterval(refresh, 10000);
166
+ loadGateway();
167
+ }
168
+
169
+ async function loadGateway() {
170
+ try {
171
+ const r = await fetch('/api/gateway', { headers: H() });
172
+ if (!r.ok) return;
173
+ const d = await r.json();
174
+ document.getElementById('gw-url').value = d.baseUrl || '';
175
+ document.getElementById('gw-key').placeholder = d.hasKey ? `已设置(${d.apiKey}),留空保持不变` : 'Token / API Key';
176
+ } catch {}
177
+ }
178
+
179
+ async function saveGateway() {
180
+ const baseUrl = document.getElementById('gw-url').value.trim();
181
+ const apiKey = document.getElementById('gw-key').value.trim();
182
+ const msg = document.getElementById('gw-msg');
183
+ try {
184
+ const r = await fetch('/api/gateway', {
185
+ method: 'POST',
186
+ headers: H(),
187
+ body: JSON.stringify({ baseUrl, apiKey }),
188
+ });
189
+ const d = await r.json();
190
+ if (d.ok) {
191
+ msg.className = 'msg msg-ok';
192
+ msg.textContent = '已保存,新建 Claude 会话自动生效';
193
+ document.getElementById('gw-key').value = '';
194
+ loadGateway();
195
+ } else {
196
+ msg.className = 'msg msg-err';
197
+ msg.textContent = d.error || '失败';
198
+ }
199
+ } catch {
200
+ msg.className = 'msg msg-err';
201
+ msg.textContent = '连接失败';
202
+ }
150
203
  }
151
204
 
152
205
  async function refresh() {
package/dist/index.html CHANGED
@@ -419,6 +419,7 @@
419
419
  <div style="display:flex;align-items:center;gap:0.6rem">
420
420
  <span id="user-name" style="font-size:0.8rem;color:var(--muted)"></span>
421
421
  <button id="usage-btn" onclick="showUsage()" style="background:none;border:1px solid var(--border);color:var(--muted);font-size:0.75rem;padding:0.3rem 0.6rem;border-radius:6px;cursor:pointer;width:auto">用量</button>
422
+ <button id="settings-btn" onclick="openGatewaySheet()" style="display:none;background:none;border:1px solid var(--border);color:var(--muted);font-size:0.75rem;padding:0.3rem 0.6rem;border-radius:6px;cursor:pointer;width:auto">设置</button>
422
423
  <button id="logout-btn" onclick="doLogout()" style="display:none;background:none;border:1px solid var(--border);color:var(--muted);font-size:0.75rem;padding:0.3rem 0.6rem;border-radius:6px;cursor:pointer;width:auto">退出</button>
423
424
  <div id="conn-dot"></div>
424
425
  </div>
@@ -437,6 +438,32 @@
437
438
  </div>
438
439
  </div>
439
440
 
441
+ <!-- Gateway settings sheet -->
442
+ <div id="gw-backdrop" onclick="closeGatewaySheet()" style="display:none;position:fixed;inset:0;background:rgba(0,0,0,0.55);z-index:40"></div>
443
+ <style>#gw-backdrop.open { display:block; } #gw-sheet { position:fixed; bottom:-100%; left:0; right:0; background:var(--surface); border-top-left-radius:18px; border-top-right-radius:18px; padding:1rem 1.1rem 1.4rem; z-index:41; transition:bottom 0.25s ease; max-width:500px; margin:0 auto; } #gw-sheet.open { bottom:0; }</style>
444
+ <div id="gw-sheet">
445
+ <div class="sheet-handle"></div>
446
+ <div class="sheet-title">大模型网关配置</div>
447
+ <div style="font-size:0.78rem;color:var(--muted);margin-bottom:0.8rem;line-height:1.5">
448
+ 配置后,新建的 Claude 会话将自动使用此网关。<br>
449
+ 对应环境变量:<code style="background:var(--bg);padding:0.1em 0.3em;border-radius:3px">ANTHROPIC_BASE_URL</code> / <code style="background:var(--bg);padding:0.1em 0.3em;border-radius:3px">ANTHROPIC_API_KEY</code>
450
+ </div>
451
+ <div class="field">
452
+ <label>网关 URL</label>
453
+ <input id="gw-url" type="text" placeholder="https://your-gateway.example.com" autocomplete="off" spellcheck="false">
454
+ </div>
455
+ <div class="field">
456
+ <label>Token <span style="font-weight:400;text-transform:none">(API Key)</span></label>
457
+ <div style="display:flex;gap:0.5rem">
458
+ <input id="gw-key" type="password" placeholder="sk-ant-... 或留空保持不变" autocomplete="off" spellcheck="false" style="flex:1">
459
+ <button onclick="clearGatewayKey()" type="button" style="flex-shrink:0;background:var(--bg);border:1px solid var(--border);border-radius:8px;padding:0 0.75rem;color:var(--muted);font-size:0.75rem;cursor:pointer">清空</button>
460
+ </div>
461
+ </div>
462
+ <button onclick="saveGateway()" style="width:100%;margin-top:0.5rem">保存</button>
463
+ <button onclick="closeGatewaySheet()" style="width:100%;background:var(--bg);color:var(--muted);border:1px solid var(--border);border-radius:10px;padding:0.7rem;font-size:0.85rem;cursor:pointer;margin-top:0.5rem">取消</button>
464
+ <div id="gw-msg" style="text-align:center;font-size:0.78rem;color:var(--muted);margin-top:0.6rem;min-height:1.2em"></div>
465
+ </div>
466
+
440
467
  <!-- FAB -->
441
468
  <button id="fab" onclick="openSheet()">+</button>
442
469
 
@@ -523,6 +550,7 @@
523
550
 
524
551
  // 加载当前用户信息(独占模式显示用户名、退出按钮、锁定目录)
525
552
  let _userDir = null;
553
+ let _isAdmin = false;
526
554
  fetch('/api/me').then(r => r.json()).then(data => {
527
555
  if ((data.exclusive || data.hasUsers) && data.name) {
528
556
  document.getElementById('user-name').textContent = data.name;
@@ -535,8 +563,63 @@
535
563
  document.getElementById('drive-bar').style.display = 'none';
536
564
  renderRecentDirs();
537
565
  }
566
+ _isAdmin = data.role === 'admin';
567
+ if (_isAdmin) document.getElementById('settings-btn').style.display = '';
538
568
  }).catch(() => {});
539
569
 
570
+ // ── 大模型网关配置 ─────────────────────────────────
571
+ window.openGatewaySheet = async function() {
572
+ document.getElementById('gw-backdrop').classList.add('open');
573
+ document.getElementById('gw-sheet').classList.add('open');
574
+ document.getElementById('gw-msg').textContent = '';
575
+ try {
576
+ const r = await fetch('/api/gateway');
577
+ if (!r.ok) throw new Error('权限不足');
578
+ const d = await r.json();
579
+ document.getElementById('gw-url').value = d.baseUrl || '';
580
+ document.getElementById('gw-key').value = d.apiKey || '';
581
+ document.getElementById('gw-key').dataset.masked = d.hasKey ? '1' : '';
582
+ } catch (e) {
583
+ document.getElementById('gw-msg').textContent = '加载失败: ' + e.message;
584
+ }
585
+ };
586
+ window.closeGatewaySheet = function() {
587
+ document.getElementById('gw-backdrop').classList.remove('open');
588
+ document.getElementById('gw-sheet').classList.remove('open');
589
+ };
590
+ window.saveGateway = async function() {
591
+ const baseUrl = document.getElementById('gw-url').value.trim();
592
+ let apiKey = document.getElementById('gw-key').value.trim();
593
+ // 如果用户没改(还是掩码),后端会保留原值
594
+ const msg = document.getElementById('gw-msg');
595
+ msg.textContent = '保存中...';
596
+ try {
597
+ const r = await fetch('/api/gateway', {
598
+ method: 'POST',
599
+ headers: { 'Content-Type': 'application/json' },
600
+ body: JSON.stringify({ baseUrl, apiKey }),
601
+ });
602
+ const d = await r.json();
603
+ if (d.ok) {
604
+ msg.style.color = '#3fb950';
605
+ msg.textContent = '已保存。下次新建 Claude 会话自动生效';
606
+ setTimeout(() => closeGatewaySheet(), 1200);
607
+ } else {
608
+ msg.style.color = '#f85149';
609
+ msg.textContent = '保存失败: ' + (d.error || '未知错误');
610
+ }
611
+ } catch (e) {
612
+ msg.style.color = '#f85149';
613
+ msg.textContent = '保存失败: ' + e.message;
614
+ }
615
+ };
616
+ window.clearGatewayKey = function() {
617
+ const inp = document.getElementById('gw-key');
618
+ inp.value = '';
619
+ inp.dataset.masked = '';
620
+ inp.focus();
621
+ };
622
+
540
623
  async function showUsage() {
541
624
  try {
542
625
  const res = await fetch('/api/usage');