ai-zero-token 2.0.6 → 2.0.7

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 (30) hide show
  1. package/CHANGELOG.md +9 -1
  2. package/README.md +5 -5
  3. package/README.zh-CN.md +5 -5
  4. package/admin-ui/dist/assets/accounts-D3tsDc3k.js +4 -0
  5. package/admin-ui/dist/assets/{docs--eK_2fzC.js → docs-BO-aSEzh.js} +1 -1
  6. package/admin-ui/dist/assets/{image-bed-7wBZ1GhS.js → image-bed-Dql7Vqd9.js} +1 -1
  7. package/admin-ui/dist/assets/{index-CdFYy5j6.js → index-CCiBaGwU.js} +3 -3
  8. package/admin-ui/dist/assets/{launch-BiD1Khtg.js → launch-DXLo-NIM.js} +1 -1
  9. package/admin-ui/dist/assets/{logs-BdoKDqh2.js → logs-Cwn8-rDu.js} +1 -1
  10. package/admin-ui/dist/assets/{network-detect-BvKns5nQ.js → network-detect-vzWfL-Tz.js} +1 -1
  11. package/admin-ui/dist/assets/{overview-wm6M45fu.js → overview-B_yad8ge.js} +1 -1
  12. package/admin-ui/dist/assets/{profiles-DMOjJORP.js → profiles-C5SmQvju.js} +1 -1
  13. package/admin-ui/dist/assets/{settings-DOOu7Kd8.js → settings-BdRWcKJb.js} +1 -1
  14. package/admin-ui/dist/assets/{tester-NrARmlis.js → tester-BKoMSoCz.js} +1 -1
  15. package/admin-ui/dist/assets/{usage-CdWRVMDV.js → usage-B-qQxXzQ.js} +1 -1
  16. package/admin-ui/dist/index.html +2 -2
  17. package/dist/cli/commands/help.js +1 -1
  18. package/dist/cli/commands/models.js +3 -2
  19. package/dist/core/context.js +1 -1
  20. package/dist/core/models/openai-codex-models.js +106 -1
  21. package/dist/core/providers/http-client.js +142 -12
  22. package/dist/core/services/auth-service.js +104 -7
  23. package/dist/core/services/chat-service.js +16 -18
  24. package/dist/core/services/model-service.js +22 -8
  25. package/dist/server/app.js +179 -23
  26. package/dist/server/index.js +1 -1
  27. package/docs/API_USAGE.md +1 -1
  28. package/docs/DESKTOP_RELEASE.md +12 -1
  29. package/package.json +1 -1
  30. package/admin-ui/dist/assets/accounts-bCDKXGg9.js +0 -4
@@ -1 +1 @@
1
- import{t as e}from"./jsx-runtime-DqpGtLhh.js";import{t}from"./activity-D21-Xrc4.js";import{t as n}from"./server-BrjJPb9D.js";import{t as r}from"./zap-B4_oDbCp.js";import{m as i}from"./profiles-DMOjJORP.js";import{C as a,h as o,v as s,y as c}from"./index-CdFYy5j6.js";import{t as l}from"./InfoRow-0ULI9iI3.js";var u=e(),d=`https://github.com/fchangjun/AI-Zero-Token/releases`;function f(e){return(0,u.jsxs)(`section`,{className:`launch-page`,children:[(0,u.jsxs)(`div`,{className:`launch-copy`,children:[(0,u.jsxs)(`div`,{className:`launch-identity`,children:[(0,u.jsx)(`div`,{className:`launch-app-icon`,children:(0,u.jsx)(`img`,{src:a,alt:``})}),(0,u.jsxs)(`div`,{children:[(0,u.jsx)(`span`,{className:`badge brand`,children:`Desktop Gateway`}),(0,u.jsx)(`h2`,{children:`AI Zero Token`}),(0,u.jsx)(`p`,{children:`本地 OpenAI 兼容网关、账号池和接口测试工作台。`})]})]}),(0,u.jsxs)(`div`,{className:`launch-metrics`,children:[(0,u.jsxs)(`div`,{children:[(0,u.jsx)(o,{size:18}),(0,u.jsx)(`span`,{children:`账号池`}),(0,u.jsx)(`strong`,{children:e.config?.status.profileCount||0})]}),(0,u.jsxs)(`div`,{children:[(0,u.jsx)(n,{size:18}),(0,u.jsx)(`span`,{children:`网关`}),(0,u.jsx)(`strong`,{children:e.config?.status.loggedIn?`运行中`:`待登录`})]}),(0,u.jsxs)(`div`,{children:[(0,u.jsx)(t,{size:18}),(0,u.jsx)(`span`,{children:`模型`}),(0,u.jsx)(`strong`,{children:e.config?.settings.defaultModel||`-`})]})]}),(0,u.jsxs)(`div`,{className:`launch-actions`,children:[(0,u.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:()=>e.onRoute(`accounts`),children:[(0,u.jsx)(o,{size:16}),`管理账号`]}),(0,u.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:()=>e.onRoute(`tester`),children:[(0,u.jsx)(r,{size:16}),`测试接口`]}),(0,u.jsxs)(`a`,{className:`btn-secondary`,href:d,target:`_blank`,rel:`noreferrer`,children:[(0,u.jsx)(c,{size:16}),`桌面版下载`]}),(0,u.jsxs)(`a`,{className:`btn-secondary`,href:d,target:`_blank`,rel:`noreferrer`,title:`桌面版安装包发布后会放在这里`,children:[(0,u.jsx)(s,{size:16}),`发布页`]})]}),(0,u.jsx)(`p`,{className:`launch-download-note`,children:`当前还没有公开安装包时,这里会先打开 GitHub Releases 发布页。`}),(0,u.jsxs)(`div`,{className:`launch-status`,children:[(0,u.jsx)(l,{label:`当前账号`,value:i(e.activeProfile,e.showEmails)}),(0,u.jsx)(l,{label:`服务状态`,value:e.config?.status.loggedIn?`已登录并运行`:`等待登录`}),(0,u.jsx)(l,{label:`Base URL`,value:e.config?.baseUrl||`http://127.0.0.1:8787/v1`,code:!0}),(0,u.jsx)(l,{label:`同步消息`,value:e.status})]})]}),(0,u.jsx)(`div`,{className:`launch-visual`,children:(0,u.jsxs)(`div`,{className:`launch-visual-stage`,children:[(0,u.jsx)(`img`,{className:`launch-visual-mark`,src:a,alt:`AI Zero Token 图标`}),(0,u.jsx)(`img`,{className:`launch-visual-dashboard`,src:e.visualSrc||`data:image/svg+xml,%3csvg%20width='920'%20height='600'%20viewBox='0%200%20920%20600'%20fill='none'%20xmlns='http://www.w3.org/2000/svg'%3e%3crect%20width='920'%20height='600'%20rx='36'%20fill='%23F8FAFC'/%3e%3crect%20x='56'%20y='54'%20width='808'%20height='492'%20rx='24'%20fill='%23FFFFFF'%20stroke='%23D8E0EA'/%3e%3crect%20x='56'%20y='54'%20width='190'%20height='492'%20rx='24'%20fill='%23111827'/%3e%3cpath%20d='M246%2054h594c13.255%200%2024%2010.745%2024%2024v40H246V54Z'%20fill='%23F8FAFC'/%3e%3ccircle%20cx='88'%20cy='86'%20r='7'%20fill='%23EF4444'/%3e%3ccircle%20cx='112'%20cy='86'%20r='7'%20fill='%23F59E0B'/%3e%3ccircle%20cx='136'%20cy='86'%20r='7'%20fill='%2322C55E'/%3e%3crect%20x='86'%20y='148'%20width='108'%20height='14'%20rx='7'%20fill='%23E5E7EB'%20fill-opacity='.75'/%3e%3crect%20x='86'%20y='194'%20width='128'%20height='34'%20rx='10'%20fill='%232563EB'/%3e%3crect%20x='86'%20y='244'%20width='112'%20height='34'%20rx='10'%20fill='%231F2937'/%3e%3crect%20x='86'%20y='294'%20width='124'%20height='34'%20rx='10'%20fill='%231F2937'/%3e%3crect%20x='86'%20y='344'%20width='98'%20height='34'%20rx='10'%20fill='%231F2937'/%3e%3crect%20x='278'%20y='150'%20width='250'%20height='96'%20rx='18'%20fill='%23EFF6FF'%20stroke='%23BFDBFE'/%3e%3crect%20x='558'%20y='150'%20width='250'%20height='96'%20rx='18'%20fill='%23F0FDF4'%20stroke='%23BBF7D0'/%3e%3crect%20x='278'%20y='280'%20width='530'%20height='210'%20rx='18'%20fill='%23FFFFFF'%20stroke='%23D8E0EA'/%3e%3cpath%20d='M318%20430c43-68%2089-70%20137-8%2039%2049%2081%2037%20126-36%2038-62%2081-79%20129-21'%20stroke='%232563EB'%20stroke-width='8'%20stroke-linecap='round'/%3e%3cpath%20d='M318%20386c42-35%2083-33%20123%206%2045%2045%2092%2040%20141-16%2038-43%2080-49%20126-18'%20stroke='%2322C55E'%20stroke-width='8'%20stroke-linecap='round'/%3e%3crect%20x='318'%20y='316'%20width='118'%20height='13'%20rx='6.5'%20fill='%23111827'/%3e%3crect%20x='318'%20y='342'%20width='194'%20height='10'%20rx='5'%20fill='%23CBD5E1'/%3e%3crect%20x='596'%20y='182'%20width='80'%20height='12'%20rx='6'%20fill='%23111827'/%3e%3crect%20x='596'%20y='207'%20width='150'%20height='10'%20rx='5'%20fill='%2386EFAC'/%3e%3crect%20x='316'%20y='182'%20width='80'%20height='12'%20rx='6'%20fill='%23111827'/%3e%3crect%20x='316'%20y='207'%20width='150'%20height='10'%20rx='5'%20fill='%2393C5FD'/%3e%3ccircle%20cx='776'%20cy='198'%20r='16'%20fill='%2322C55E'/%3e%3cpath%20d='M768%20198l6%206%2011-13'%20stroke='%23FFFFFF'%20stroke-width='5'%20stroke-linecap='round'%20stroke-linejoin='round'/%3e%3cpath%20d='M478%20186l15%2020-15%2020-15-20%2015-20Z'%20fill='%232563EB'/%3e%3c/svg%3e`,alt:`AI Zero Token 桌面端启动页配图`})]})})]})}export{f as LaunchPage};
1
+ import{t as e}from"./jsx-runtime-DqpGtLhh.js";import{t}from"./activity-D21-Xrc4.js";import{t as n}from"./server-BrjJPb9D.js";import{t as r}from"./zap-B4_oDbCp.js";import{m as i}from"./profiles-C5SmQvju.js";import{C as a,h as o,v as s,y as c}from"./index-CCiBaGwU.js";import{t as l}from"./InfoRow-0ULI9iI3.js";var u=e(),d=`https://github.com/fchangjun/AI-Zero-Token/releases`;function f(e){return(0,u.jsxs)(`section`,{className:`launch-page`,children:[(0,u.jsxs)(`div`,{className:`launch-copy`,children:[(0,u.jsxs)(`div`,{className:`launch-identity`,children:[(0,u.jsx)(`div`,{className:`launch-app-icon`,children:(0,u.jsx)(`img`,{src:a,alt:``})}),(0,u.jsxs)(`div`,{children:[(0,u.jsx)(`span`,{className:`badge brand`,children:`Desktop Gateway`}),(0,u.jsx)(`h2`,{children:`AI Zero Token`}),(0,u.jsx)(`p`,{children:`本地 OpenAI 兼容网关、账号池和接口测试工作台。`})]})]}),(0,u.jsxs)(`div`,{className:`launch-metrics`,children:[(0,u.jsxs)(`div`,{children:[(0,u.jsx)(o,{size:18}),(0,u.jsx)(`span`,{children:`账号池`}),(0,u.jsx)(`strong`,{children:e.config?.status.profileCount||0})]}),(0,u.jsxs)(`div`,{children:[(0,u.jsx)(n,{size:18}),(0,u.jsx)(`span`,{children:`网关`}),(0,u.jsx)(`strong`,{children:e.config?.status.loggedIn?`运行中`:`待登录`})]}),(0,u.jsxs)(`div`,{children:[(0,u.jsx)(t,{size:18}),(0,u.jsx)(`span`,{children:`模型`}),(0,u.jsx)(`strong`,{children:e.config?.settings.defaultModel||`-`})]})]}),(0,u.jsxs)(`div`,{className:`launch-actions`,children:[(0,u.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:()=>e.onRoute(`accounts`),children:[(0,u.jsx)(o,{size:16}),`管理账号`]}),(0,u.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:()=>e.onRoute(`tester`),children:[(0,u.jsx)(r,{size:16}),`测试接口`]}),(0,u.jsxs)(`a`,{className:`btn-secondary`,href:d,target:`_blank`,rel:`noreferrer`,children:[(0,u.jsx)(c,{size:16}),`桌面版下载`]}),(0,u.jsxs)(`a`,{className:`btn-secondary`,href:d,target:`_blank`,rel:`noreferrer`,title:`桌面版安装包发布后会放在这里`,children:[(0,u.jsx)(s,{size:16}),`发布页`]})]}),(0,u.jsx)(`p`,{className:`launch-download-note`,children:`当前还没有公开安装包时,这里会先打开 GitHub Releases 发布页。`}),(0,u.jsxs)(`div`,{className:`launch-status`,children:[(0,u.jsx)(l,{label:`当前账号`,value:i(e.activeProfile,e.showEmails)}),(0,u.jsx)(l,{label:`服务状态`,value:e.config?.status.loggedIn?`已登录并运行`:`等待登录`}),(0,u.jsx)(l,{label:`Base URL`,value:e.config?.baseUrl||`http://127.0.0.1:8787/v1`,code:!0}),(0,u.jsx)(l,{label:`同步消息`,value:e.status})]})]}),(0,u.jsx)(`div`,{className:`launch-visual`,children:(0,u.jsxs)(`div`,{className:`launch-visual-stage`,children:[(0,u.jsx)(`img`,{className:`launch-visual-mark`,src:a,alt:`AI Zero Token 图标`}),(0,u.jsx)(`img`,{className:`launch-visual-dashboard`,src:e.visualSrc||`data:image/svg+xml,%3csvg%20width='920'%20height='600'%20viewBox='0%200%20920%20600'%20fill='none'%20xmlns='http://www.w3.org/2000/svg'%3e%3crect%20width='920'%20height='600'%20rx='36'%20fill='%23F8FAFC'/%3e%3crect%20x='56'%20y='54'%20width='808'%20height='492'%20rx='24'%20fill='%23FFFFFF'%20stroke='%23D8E0EA'/%3e%3crect%20x='56'%20y='54'%20width='190'%20height='492'%20rx='24'%20fill='%23111827'/%3e%3cpath%20d='M246%2054h594c13.255%200%2024%2010.745%2024%2024v40H246V54Z'%20fill='%23F8FAFC'/%3e%3ccircle%20cx='88'%20cy='86'%20r='7'%20fill='%23EF4444'/%3e%3ccircle%20cx='112'%20cy='86'%20r='7'%20fill='%23F59E0B'/%3e%3ccircle%20cx='136'%20cy='86'%20r='7'%20fill='%2322C55E'/%3e%3crect%20x='86'%20y='148'%20width='108'%20height='14'%20rx='7'%20fill='%23E5E7EB'%20fill-opacity='.75'/%3e%3crect%20x='86'%20y='194'%20width='128'%20height='34'%20rx='10'%20fill='%232563EB'/%3e%3crect%20x='86'%20y='244'%20width='112'%20height='34'%20rx='10'%20fill='%231F2937'/%3e%3crect%20x='86'%20y='294'%20width='124'%20height='34'%20rx='10'%20fill='%231F2937'/%3e%3crect%20x='86'%20y='344'%20width='98'%20height='34'%20rx='10'%20fill='%231F2937'/%3e%3crect%20x='278'%20y='150'%20width='250'%20height='96'%20rx='18'%20fill='%23EFF6FF'%20stroke='%23BFDBFE'/%3e%3crect%20x='558'%20y='150'%20width='250'%20height='96'%20rx='18'%20fill='%23F0FDF4'%20stroke='%23BBF7D0'/%3e%3crect%20x='278'%20y='280'%20width='530'%20height='210'%20rx='18'%20fill='%23FFFFFF'%20stroke='%23D8E0EA'/%3e%3cpath%20d='M318%20430c43-68%2089-70%20137-8%2039%2049%2081%2037%20126-36%2038-62%2081-79%20129-21'%20stroke='%232563EB'%20stroke-width='8'%20stroke-linecap='round'/%3e%3cpath%20d='M318%20386c42-35%2083-33%20123%206%2045%2045%2092%2040%20141-16%2038-43%2080-49%20126-18'%20stroke='%2322C55E'%20stroke-width='8'%20stroke-linecap='round'/%3e%3crect%20x='318'%20y='316'%20width='118'%20height='13'%20rx='6.5'%20fill='%23111827'/%3e%3crect%20x='318'%20y='342'%20width='194'%20height='10'%20rx='5'%20fill='%23CBD5E1'/%3e%3crect%20x='596'%20y='182'%20width='80'%20height='12'%20rx='6'%20fill='%23111827'/%3e%3crect%20x='596'%20y='207'%20width='150'%20height='10'%20rx='5'%20fill='%2386EFAC'/%3e%3crect%20x='316'%20y='182'%20width='80'%20height='12'%20rx='6'%20fill='%23111827'/%3e%3crect%20x='316'%20y='207'%20width='150'%20height='10'%20rx='5'%20fill='%2393C5FD'/%3e%3ccircle%20cx='776'%20cy='198'%20r='16'%20fill='%2322C55E'/%3e%3cpath%20d='M768%20198l6%206%2011-13'%20stroke='%23FFFFFF'%20stroke-width='5'%20stroke-linecap='round'%20stroke-linejoin='round'/%3e%3cpath%20d='M478%20186l15%2020-15%2020-15-20%2015-20Z'%20fill='%232563EB'/%3e%3c/svg%3e`,alt:`AI Zero Token 桌面端启动页配图`})]})})]})}export{f as LaunchPage};
@@ -1 +1 @@
1
- import{a as e,n as t,r as n,t as r}from"./jsx-runtime-DqpGtLhh.js";import{t as i}from"./search-B2hz41D3.js";import{T as a,w as o,x as s}from"./profiles-DMOjJORP.js";import{b as c,n as l}from"./index-CdFYy5j6.js";var u=t(`funnel`,[[`path`,{d:`M10 20a1 1 0 0 0 .553.895l2 1A1 1 0 0 0 14 21v-7a2 2 0 0 1 .517-1.341L21.74 4.67A1 1 0 0 0 21 3H3a1 1 0 0 0-.742 1.67l7.225 7.989A2 2 0 0 1 10 14z`,key:`sc7q7i`}]]),d=e(n(),1),f=r();function p(e){let[t,n]=(0,d.useState)(``),[r,p]=(0,d.useState)(`all`),[m,h]=(0,d.useState)(`all`),[g,_]=(0,d.useState)(`all`),[v,y]=(0,d.useState)(null);(0,d.useEffect)(()=>{if(e.logs.length===0){y(null);return}(!v||!e.logs.some(e=>e.id===v))&&y(e.logs[0].id)},[e.logs,v]);let b=(0,d.useMemo)(()=>Array.from(new Set(e.logs.map(e=>e.source||`管理页`))),[e.logs]),x=(0,d.useMemo)(()=>Array.from(new Set(e.logs.map(e=>e.method))),[e.logs]),S=(0,d.useMemo)(()=>{let n=t.trim().toLowerCase();return e.logs.filter(e=>{let t=[e.time,e.method,e.endpoint,e.account,e.model,e.statusCode,e.durationMs,e.source].join(` `).toLowerCase();return!(n&&!t.includes(n)||r!==`all`&&e.method!==r||m!==`all`&&(e.source||`管理页`)!==m||g===`ok`&&e.statusCode>=400||g===`error`&&e.statusCode<400)})},[r,e.logs,t,m,g]),C=S.find(e=>e.id===v)||S[0]||null;function w(){C&&l(o(C)).then(e=>{}).catch(()=>void 0)}return(0,f.jsxs)(`section`,{className:`log-table-wrap`,id:`logs`,children:[(0,f.jsx)(`div`,{className:`section-head compact`,children:(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`h2`,{children:`请求日志`}),(0,f.jsx)(`p`,{children:`记录网关最近收到的 API 请求,详情为安全摘要。`})]})}),(0,f.jsxs)(`div`,{className:`log-toolbar`,children:[(0,f.jsxs)(`label`,{className:`search-box log-search`,children:[(0,f.jsx)(i,{size:16}),(0,f.jsx)(`input`,{value:t,onChange:e=>n(e.target.value),placeholder:`搜索时间、接口、账号、模型或状态`})]}),(0,f.jsxs)(`label`,{className:`filter-chip`,children:[(0,f.jsx)(u,{size:14}),(0,f.jsxs)(`select`,{value:r,onChange:e=>p(e.target.value),children:[(0,f.jsx)(`option`,{value:`all`,children:`全部方法`}),x.map(e=>(0,f.jsx)(`option`,{value:e,children:e},e))]})]}),(0,f.jsx)(`label`,{className:`filter-chip`,children:(0,f.jsxs)(`select`,{value:m,onChange:e=>h(e.target.value),children:[(0,f.jsx)(`option`,{value:`all`,children:`全部来源`}),b.map(e=>(0,f.jsx)(`option`,{value:e,children:e},e))]})}),(0,f.jsx)(`label`,{className:`filter-chip`,children:(0,f.jsxs)(`select`,{value:g,onChange:e=>_(e.target.value),children:[(0,f.jsx)(`option`,{value:`all`,children:`全部状态`}),(0,f.jsx)(`option`,{value:`ok`,children:`成功`}),(0,f.jsx)(`option`,{value:`error`,children:`失败`})]})})]}),(0,f.jsx)(`div`,{className:`table-scroller`,children:(0,f.jsxs)(`table`,{children:[(0,f.jsx)(`thead`,{children:(0,f.jsxs)(`tr`,{children:[(0,f.jsx)(`th`,{children:`时间`}),(0,f.jsx)(`th`,{children:`方法`}),(0,f.jsx)(`th`,{children:`接口`}),(0,f.jsx)(`th`,{children:`账号`}),(0,f.jsx)(`th`,{children:`模型`}),(0,f.jsx)(`th`,{children:`状态`}),(0,f.jsx)(`th`,{children:`耗时`}),(0,f.jsx)(`th`,{children:`来源`})]})}),(0,f.jsx)(`tbody`,{children:S.length===0?(0,f.jsx)(`tr`,{children:(0,f.jsx)(`td`,{colSpan:8,children:`最近 API 请求会在这里显示。`})}):S.map(e=>(0,f.jsxs)(`tr`,{className:e.id===C?.id?`is-selected`:``,onClick:()=>y(e.id),children:[(0,f.jsx)(`td`,{children:a(e.time)}),(0,f.jsx)(`td`,{children:(0,f.jsx)(`span`,{className:`method-pill method-${e.method.toLowerCase()}`,children:e.method})}),(0,f.jsx)(`td`,{children:(0,f.jsx)(`code`,{children:e.endpoint})}),(0,f.jsx)(`td`,{children:e.account}),(0,f.jsx)(`td`,{children:e.model}),(0,f.jsx)(`td`,{children:(0,f.jsx)(`span`,{className:`status-pill ${e.statusCode>=400?`is-error`:`is-ok`}`,children:e.statusCode})}),(0,f.jsx)(`td`,{children:s(e.durationMs)}),(0,f.jsx)(`td`,{children:e.source||`管理页`})]},e.id))})]})}),(0,f.jsxs)(`div`,{className:`table-footer`,children:[`当前展示 `,S.length,` 条请求记录,最近总计 `,e.logs.length,` 条。`]}),C&&(0,f.jsxs)(`div`,{className:`log-detail-panel`,children:[(0,f.jsxs)(`div`,{className:`log-detail-head`,children:[(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`h3`,{children:`日志详情`}),(0,f.jsxs)(`p`,{children:[a(C.time),` · `,C.method,` `,C.endpoint]})]}),(0,f.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:w,children:[(0,f.jsx)(c,{size:16}),`复制详情`]})]}),(0,f.jsxs)(`div`,{className:`log-detail-grid`,children:[(0,f.jsxs)(`div`,{className:`log-detail-meta`,children:[(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`span`,{children:`账号`}),(0,f.jsx)(`strong`,{children:C.account})]}),(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`span`,{children:`模型`}),(0,f.jsx)(`strong`,{children:C.model})]}),(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`span`,{children:`状态`}),(0,f.jsx)(`strong`,{children:C.statusCode})]}),(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`span`,{children:`耗时`}),(0,f.jsx)(`strong`,{children:s(C.durationMs)})]}),(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`span`,{children:`来源`}),(0,f.jsx)(`strong`,{children:C.source||`管理页`})]})]}),(0,f.jsx)(`pre`,{className:`pre log-detail-pre`,children:o(C)})]})]})]})}function m(e){return(0,f.jsx)(p,{logs:e.logs})}export{m as LogsPage};
1
+ import{a as e,n as t,r as n,t as r}from"./jsx-runtime-DqpGtLhh.js";import{t as i}from"./search-B2hz41D3.js";import{T as a,w as o,x as s}from"./profiles-C5SmQvju.js";import{b as c,n as l}from"./index-CCiBaGwU.js";var u=t(`funnel`,[[`path`,{d:`M10 20a1 1 0 0 0 .553.895l2 1A1 1 0 0 0 14 21v-7a2 2 0 0 1 .517-1.341L21.74 4.67A1 1 0 0 0 21 3H3a1 1 0 0 0-.742 1.67l7.225 7.989A2 2 0 0 1 10 14z`,key:`sc7q7i`}]]),d=e(n(),1),f=r();function p(e){let[t,n]=(0,d.useState)(``),[r,p]=(0,d.useState)(`all`),[m,h]=(0,d.useState)(`all`),[g,_]=(0,d.useState)(`all`),[v,y]=(0,d.useState)(null);(0,d.useEffect)(()=>{if(e.logs.length===0){y(null);return}(!v||!e.logs.some(e=>e.id===v))&&y(e.logs[0].id)},[e.logs,v]);let b=(0,d.useMemo)(()=>Array.from(new Set(e.logs.map(e=>e.source||`管理页`))),[e.logs]),x=(0,d.useMemo)(()=>Array.from(new Set(e.logs.map(e=>e.method))),[e.logs]),S=(0,d.useMemo)(()=>{let n=t.trim().toLowerCase();return e.logs.filter(e=>{let t=[e.time,e.method,e.endpoint,e.account,e.model,e.statusCode,e.durationMs,e.source].join(` `).toLowerCase();return!(n&&!t.includes(n)||r!==`all`&&e.method!==r||m!==`all`&&(e.source||`管理页`)!==m||g===`ok`&&e.statusCode>=400||g===`error`&&e.statusCode<400)})},[r,e.logs,t,m,g]),C=S.find(e=>e.id===v)||S[0]||null;function w(){C&&l(o(C)).then(e=>{}).catch(()=>void 0)}return(0,f.jsxs)(`section`,{className:`log-table-wrap`,id:`logs`,children:[(0,f.jsx)(`div`,{className:`section-head compact`,children:(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`h2`,{children:`请求日志`}),(0,f.jsx)(`p`,{children:`记录网关最近收到的 API 请求,详情为安全摘要。`})]})}),(0,f.jsxs)(`div`,{className:`log-toolbar`,children:[(0,f.jsxs)(`label`,{className:`search-box log-search`,children:[(0,f.jsx)(i,{size:16}),(0,f.jsx)(`input`,{value:t,onChange:e=>n(e.target.value),placeholder:`搜索时间、接口、账号、模型或状态`})]}),(0,f.jsxs)(`label`,{className:`filter-chip`,children:[(0,f.jsx)(u,{size:14}),(0,f.jsxs)(`select`,{value:r,onChange:e=>p(e.target.value),children:[(0,f.jsx)(`option`,{value:`all`,children:`全部方法`}),x.map(e=>(0,f.jsx)(`option`,{value:e,children:e},e))]})]}),(0,f.jsx)(`label`,{className:`filter-chip`,children:(0,f.jsxs)(`select`,{value:m,onChange:e=>h(e.target.value),children:[(0,f.jsx)(`option`,{value:`all`,children:`全部来源`}),b.map(e=>(0,f.jsx)(`option`,{value:e,children:e},e))]})}),(0,f.jsx)(`label`,{className:`filter-chip`,children:(0,f.jsxs)(`select`,{value:g,onChange:e=>_(e.target.value),children:[(0,f.jsx)(`option`,{value:`all`,children:`全部状态`}),(0,f.jsx)(`option`,{value:`ok`,children:`成功`}),(0,f.jsx)(`option`,{value:`error`,children:`失败`})]})})]}),(0,f.jsx)(`div`,{className:`table-scroller`,children:(0,f.jsxs)(`table`,{children:[(0,f.jsx)(`thead`,{children:(0,f.jsxs)(`tr`,{children:[(0,f.jsx)(`th`,{children:`时间`}),(0,f.jsx)(`th`,{children:`方法`}),(0,f.jsx)(`th`,{children:`接口`}),(0,f.jsx)(`th`,{children:`账号`}),(0,f.jsx)(`th`,{children:`模型`}),(0,f.jsx)(`th`,{children:`状态`}),(0,f.jsx)(`th`,{children:`耗时`}),(0,f.jsx)(`th`,{children:`来源`})]})}),(0,f.jsx)(`tbody`,{children:S.length===0?(0,f.jsx)(`tr`,{children:(0,f.jsx)(`td`,{colSpan:8,children:`最近 API 请求会在这里显示。`})}):S.map(e=>(0,f.jsxs)(`tr`,{className:e.id===C?.id?`is-selected`:``,onClick:()=>y(e.id),children:[(0,f.jsx)(`td`,{children:a(e.time)}),(0,f.jsx)(`td`,{children:(0,f.jsx)(`span`,{className:`method-pill method-${e.method.toLowerCase()}`,children:e.method})}),(0,f.jsx)(`td`,{children:(0,f.jsx)(`code`,{children:e.endpoint})}),(0,f.jsx)(`td`,{children:e.account}),(0,f.jsx)(`td`,{children:e.model}),(0,f.jsx)(`td`,{children:(0,f.jsx)(`span`,{className:`status-pill ${e.statusCode>=400?`is-error`:`is-ok`}`,children:e.statusCode})}),(0,f.jsx)(`td`,{children:s(e.durationMs)}),(0,f.jsx)(`td`,{children:e.source||`管理页`})]},e.id))})]})}),(0,f.jsxs)(`div`,{className:`table-footer`,children:[`当前展示 `,S.length,` 条请求记录,最近总计 `,e.logs.length,` 条。`]}),C&&(0,f.jsxs)(`div`,{className:`log-detail-panel`,children:[(0,f.jsxs)(`div`,{className:`log-detail-head`,children:[(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`h3`,{children:`日志详情`}),(0,f.jsxs)(`p`,{children:[a(C.time),` · `,C.method,` `,C.endpoint]})]}),(0,f.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:w,children:[(0,f.jsx)(c,{size:16}),`复制详情`]})]}),(0,f.jsxs)(`div`,{className:`log-detail-grid`,children:[(0,f.jsxs)(`div`,{className:`log-detail-meta`,children:[(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`span`,{children:`账号`}),(0,f.jsx)(`strong`,{children:C.account})]}),(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`span`,{children:`模型`}),(0,f.jsx)(`strong`,{children:C.model})]}),(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`span`,{children:`状态`}),(0,f.jsx)(`strong`,{children:C.statusCode})]}),(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`span`,{children:`耗时`}),(0,f.jsx)(`strong`,{children:s(C.durationMs)})]}),(0,f.jsxs)(`div`,{children:[(0,f.jsx)(`span`,{children:`来源`}),(0,f.jsx)(`strong`,{children:C.source||`管理页`})]})]}),(0,f.jsx)(`pre`,{className:`pre log-detail-pre`,children:o(C)})]})]})]})}function m(e){return(0,f.jsx)(p,{logs:e.logs})}export{m as LogsPage};
@@ -1 +1 @@
1
- import{a as e,n as t,r as n,t as r}from"./jsx-runtime-DqpGtLhh.js";import{t as i}from"./activity-D21-Xrc4.js";import{t as a}from"./circle-check-ZYtn9GqY.js";import{t as o}from"./clock-3-BzDANsVk.js";import{t as s}from"./refresh-cw-CAAH2rqe.js";import{t as c}from"./server-BrjJPb9D.js";import{g as l,m as u,p as ee}from"./index-CdFYy5j6.js";var d=t(`circle-x`,[[`circle`,{cx:`12`,cy:`12`,r:`10`,key:`1mglay`}],[`path`,{d:`m15 9-6 6`,key:`1uzhvr`}],[`path`,{d:`m9 9 6 6`,key:`z0biqf`}]]),f=t(`map-pin`,[[`path`,{d:`M20 10c0 4.993-5.539 10.193-7.399 11.799a1 1 0 0 1-1.202 0C9.539 20.193 4 14.993 4 10a8 8 0 0 1 16 0`,key:`1r0f0z`}],[`circle`,{cx:`12`,cy:`10`,r:`3`,key:`ilqhr7`}]]),p=t(`triangle-alert`,[[`path`,{d:`m21.73 18-8-14a2 2 0 0 0-3.48 0l-8 14A2 2 0 0 0 4 21h16a2 2 0 0 0 1.73-3`,key:`wmoenq`}],[`path`,{d:`M12 9v4`,key:`juzpu7`}],[`path`,{d:`M12 17h.01`,key:`p32p05`}]]),m=e(n(),1),h=`data:image/svg+xml,%3csvg%20role='img'%20viewBox='0%200%2024%2024'%20xmlns='http://www.w3.org/2000/svg'%3e%3ctitle%3eClaude%3c/title%3e%3cpath%20d='m4.7144%2015.9555%204.7174-2.6471.079-.2307-.079-.1275h-.2307l-.7893-.0486-2.6956-.0729-2.3375-.0971-2.2646-.1214-.5707-.1215-.5343-.7042.0546-.3522.4797-.3218.686.0608%201.5179.1032%202.2767.1578%201.6514.0972%202.4468.255h.3886l.0546-.1579-.1336-.0971-.1032-.0972L6.973%209.8356l-2.55-1.6879-1.3356-.9714-.7225-.4918-.3643-.4614-.1578-1.0078.6557-.7225.8803.0607.2246.0607.8925.686%201.9064%201.4754%202.4893%201.8336.3643.3035.1457-.1032.0182-.0728-.164-.2733-1.3539-2.4467-1.445-2.4893-.6435-1.032-.17-.6194c-.0607-.255-.1032-.4674-.1032-.7285L6.287.1335%206.6997%200l.9957.1336.419.3642.6192%201.4147%201.0018%202.2282%201.5543%203.0296.4553.8985.2429.8318.091.255h.1579v-.1457l.1275-1.706.2368-2.0947.2307-2.6957.0789-.7589.3764-.9107.7468-.4918.5828.2793.4797.686-.0668.4433-.2853%201.8517-.5586%202.9021-.3643%201.9429h.2125l.2429-.2429.9835-1.3053%201.6514-2.0643.7286-.8196.85-.9046.5464-.4311h1.0321l.759%201.1293-.34%201.1657-1.0625%201.3478-.8804%201.1414-1.2628%201.7-.7893%201.36.0729.1093.1882-.0183%202.8535-.607%201.5421-.2794%201.8396-.3157.8318.3886.091.3946-.3278.8075-1.967.4857-2.3072.4614-3.4364.8136-.0425.0304.0486.0607%201.5482.1457.6618.0364h1.621l3.0175.2247.7892.522.4736.6376-.079.4857-1.2142.6193-1.6393-.3886-3.825-.9107-1.3113-.3279h-.1822v.1093l1.0929%201.0686%202.0035%201.8092%202.5075%202.3314.1275.5768-.3218.4554-.34-.0486-2.2039-1.6575-.85-.7468-1.9246-1.621h-.1275v.17l.4432.6496%202.3436%203.5214.1214%201.0807-.17.3521-.6071.2125-.6679-.1214-1.3721-1.9246L14.38%2017.959l-1.1414-1.9428-.1397.079-.674%207.2552-.3156.3703-.7286.2793-.6071-.4614-.3218-.7468.3218-1.4753.3886-1.9246.3157-1.53.2853-1.9004.17-.6314-.0121-.0425-.1397.0182-1.4328%201.9672-2.1796%202.9446-1.7243%201.8456-.4128.164-.7164-.3704.0667-.6618.4008-.5889%202.386-3.0357%201.4389-1.882.929-1.0868-.0062-.1579h-.0546l-6.3385%204.1164-1.1293.1457-.4857-.4554.0608-.7467.2307-.2429%201.9064-1.3114Z'/%3e%3c/svg%3e`,g=`data:image/svg+xml,%3csvg%20role='img'%20viewBox='0%200%2024%2024'%20xmlns='http://www.w3.org/2000/svg'%3e%3ctitle%3eOpenAI%3c/title%3e%3cpath%20d='M22.2819%209.8211a5.9847%205.9847%200%200%200-.5157-4.9108%206.0462%206.0462%200%200%200-6.5098-2.9A6.0651%206.0651%200%200%200%204.9807%204.1818a5.9847%205.9847%200%200%200-3.9977%202.9%206.0462%206.0462%200%200%200%20.7427%207.0966%205.98%205.98%200%200%200%20.511%204.9107%206.051%206.051%200%200%200%206.5146%202.9001A5.9847%205.9847%200%200%200%2013.2599%2024a6.0557%206.0557%200%200%200%205.7718-4.2058%205.9894%205.9894%200%200%200%203.9977-2.9001%206.0557%206.0557%200%200%200-.7475-7.0729zm-9.022%2012.6081a4.4755%204.4755%200%200%201-2.8764-1.0408l.1419-.0804%204.7783-2.7582a.7948.7948%200%200%200%20.3927-.6813v-6.7369l2.02%201.1686a.071.071%200%200%201%20.038.052v5.5826a4.504%204.504%200%200%201-4.4945%204.4944zm-9.6607-4.1254a4.4708%204.4708%200%200%201-.5346-3.0137l.142.0852%204.783%202.7582a.7712.7712%200%200%200%20.7806%200l5.8428-3.3685v2.3324a.0804.0804%200%200%201-.0332.0615L9.74%2019.9502a4.4992%204.4992%200%200%201-6.1408-1.6464zM2.3408%207.8956a4.485%204.485%200%200%201%202.3655-1.9728V11.6a.7664.7664%200%200%200%20.3879.6765l5.8144%203.3543-2.0201%201.1685a.0757.0757%200%200%201-.071%200l-4.8303-2.7865A4.504%204.504%200%200%201%202.3408%207.872zm16.5963%203.8558L13.1038%208.364%2015.1192%207.2a.0757.0757%200%200%201%20.071%200l4.8303%202.7913a4.4944%204.4944%200%200%201-.6765%208.1042v-5.6772a.79.79%200%200%200-.407-.667zm2.0107-3.0231l-.142-.0852-4.7735-2.7818a.7759.7759%200%200%200-.7854%200L9.409%209.2297V6.8974a.0662.0662%200%200%201%20.0284-.0615l4.8303-2.7866a4.4992%204.4992%200%200%201%206.6802%204.66zM8.3065%2012.863l-2.02-1.1638a.0804.0804%200%200%201-.038-.0567V6.0742a4.4992%204.4992%200%200%201%207.3757-3.4537l-.142.0805L8.704%205.459a.7948.7948%200%200%200-.3927.6813zm1.0976-2.3654l2.602-1.4998%202.6069%201.4998v2.9994l-2.5974%201.4997-2.6067-1.4997Z'/%3e%3c/svg%3e`,_=`data:image/svg+xml,%3csvg%20role='img'%20viewBox='0%200%2024%2024'%20xmlns='http://www.w3.org/2000/svg'%3e%3ctitle%3eGoogle%3c/title%3e%3cpath%20d='M12.48%2010.92v3.28h7.84c-.24%201.84-.853%203.187-1.787%204.133-1.147%201.147-2.933%202.4-6.053%202.4-4.827%200-8.6-3.893-8.6-8.72s3.773-8.72%208.6-8.72c2.6%200%204.507%201.027%205.907%202.347l2.307-2.307C18.747%201.44%2016.133%200%2012.48%200%205.867%200%20.307%205.387.307%2012s5.56%2012%2012.173%2012c3.573%200%206.267-1.173%208.373-3.36%202.16-2.16%202.84-5.213%202.84-7.667%200-.76-.053-1.467-.173-2.053H12.48z'/%3e%3c/svg%3e`,v=`data:image/svg+xml,%3csvg%20role='img'%20viewBox='0%200%2024%2024'%20xmlns='http://www.w3.org/2000/svg'%3e%3ctitle%3eX%3c/title%3e%3cpath%20d='M14.234%2010.162%2022.977%200h-2.072l-7.591%208.824L7.251%200H.258l9.168%2013.343L.258%2024H2.33l8.016-9.318L16.749%2024h6.993zm-2.837%203.299-.929-1.329L3.076%201.56h3.182l5.965%208.532.929%201.329%207.754%2011.09h-3.182z'/%3e%3c/svg%3e`,y=`data:image/svg+xml,%3csvg%20role='img'%20viewBox='0%200%2024%2024'%20xmlns='http://www.w3.org/2000/svg'%3e%3ctitle%3eYouTube%3c/title%3e%3cpath%20d='M23.498%206.186a3.016%203.016%200%200%200-2.122-2.136C19.505%203.545%2012%203.545%2012%203.545s-7.505%200-9.377.505A3.017%203.017%200%200%200%20.502%206.186C0%208.07%200%2012%200%2012s0%203.93.502%205.814a3.016%203.016%200%200%200%202.122%202.136c1.871.505%209.376.505%209.376.505s7.505%200%209.377-.505a3.015%203.015%200%200%200%202.122-2.136C24%2015.93%2024%2012%2024%2012s0-3.93-.502-5.814zM9.545%2015.568V8.432L15.818%2012l-6.273%203.568z'/%3e%3c/svg%3e`,b=r(),x={claude:h,chatgpt:g,google:_,x:v,youtube:y};function S(e){return`tone-${e}`}function C(e){try{return new URL(e).hostname.replace(/^www\./i,``)}catch{return e}}function w(e){return/^(10|127|169\.254|172\.(1[6-9]|2\d|3[0-1])|192\.168)\./.test(e)||/^100\.(6[4-9]|[7-9]\d|1\d\d|2[0-3]\d|24[0-7])\./.test(e)||/^198\.(18|19)\./.test(e)||/^192\.0\.2\./.test(e)||/^198\.51\.100\./.test(e)||/^203\.0\.113\./.test(e)||/^fc00:/i.test(e)||/^fd00:/i.test(e)||/^fe80:/i.test(e)||/^::1$/.test(e)}function T(e){let t=e.match(/Electron\/([\d.]+)/i);if(t)return`Electron ${t[1]}`;let n=e.match(/Chrome\/([\d.]+)/i);if(n&&/Safari\//i.test(e))return`Chrome ${n[1].split(`.`)[0]}`;let r=e.match(/Version\/([\d.]+).*Safari\//i);if(r&&!/Chrome\//i.test(e))return`Safari ${r[1].split(`.`)[0]}`;let i=e.match(/Firefox\/([\d.]+)/i);return i?`Firefox ${i[1].split(`.`)[0]}`:e.slice(0,40)}async function E(){if(typeof RTCPeerConnection>`u`)return{status:`不支持`,detail:`当前环境不支持 WebRTC。`,tone:`orange`,candidates:[]};let e=new Set,t=new RTCPeerConnection({iceServers:[]}),n=!1,r=new Promise(r=>{t.onicecandidate=t=>{t.candidate?.candidate&&e.add(t.candidate.candidate),t.candidate||(n=!0,r())}});try{t.createDataChannel(`probe`);let e=await t.createOffer();await t.setLocalDescription(e),await Promise.race([r,new Promise(e=>window.setTimeout(e,2400))])}catch{}finally{t.close()}let i=[...e],a=i.some(e=>/\btyp srflx\b/.test(e)||/\btyp relay\b/.test(e)),o=i.some(e=>/\btyp host\b/.test(e));return a?{status:`需要留意`,detail:`收集到 ${i.length} 个候选地址,其中包含公网回显特征。`,tone:`red`,candidates:i}:o||n?{status:`本地候选为主`,detail:`收集到 ${i.length} 个候选地址,主要来自本地网段。`,tone:`orange`,candidates:i}:{status:`暂未发现异常`,detail:`未收集到可识别的候选地址。`,tone:`green`,candidates:i}}async function D(){let[e]=await Promise.all([E()]);return{timezone:Intl.DateTimeFormat().resolvedOptions().timeZone||`-`,language:navigator.language||`-`,browser:T(navigator.userAgent||``),webrtc:e}}function O(e){return e.length===0?`slate`:e.some(e=>w(e))?`orange`:`green`}function k(e){return e.length===0?`未读取`:e.some(e=>w(e))?`内网 / 企业 DNS`:`公网 DNS`}function A(e){return{total:e.length,reachable:e.filter(e=>e.status===`可达`).length,limited:e.filter(e=>e.status===`受限`).length,unavailable:e.filter(e=>e.status===`不可用`).length}}function j(e,t){return!e||t.total===0?{label:`正在判断海外访问`,detail:`正在检查 ChatGPT、Claude、Google、YouTube 和 X 的连通性。`,tone:`blue`}:t.unavailable>0?{label:`海外访问受阻`,detail:`${t.unavailable} 个海外应用不可用,优先检查代理出口、DNS 或上游链路。`,tone:`red`}:t.limited>0?{label:`部分应用受限`,detail:`${t.reachable}/${t.total} 个海外应用可达,其余应用返回限制状态。`,tone:`orange`}:{label:`海外应用可访问`,detail:`${t.total} 个目标应用均已连通,当前出口可用于海外反向代理场景。`,tone:`green`}}function te(e,t){if(!e&&!t)return{label:`检测中`,detail:`正在采集公网出口、DNS、WebRTC、代理和平台可达性。`,tone:`blue`};if(!e||!t)return{label:`部分结果`,detail:`已有项目返回,剩余项目会保留失败原因,方便继续排查。`,tone:`orange`};let n=e.publicIpv4.available&&!!e.publicIpv4.ip,r=e.publicIpv6.available&&!!e.publicIpv6.ip,i=[e.platforms.some(e=>e.tone===`red`),t.webrtc.tone===`red`,!n&&!r],a=[e.platforms.some(e=>e.tone===`orange`),t.webrtc.tone===`orange`,O(e.dns.servers)!==`green`,!n,!e.publicIpv6.available];return i.some(Boolean)?{label:`建议先处理`,detail:`有明显阻塞项,先把出口或浏览器回显风险排掉。`,tone:`red`}:a.some(Boolean)?{label:`需要留意`,detail:`结果可用,但有几项信息值得再看一眼。`,tone:`orange`}:{label:`状态正常`,detail:`出口、DNS、WebRTC 和常用平台状态都比较稳定。`,tone:`green`}}function M(e){return(0,b.jsx)(`span`,{className:`network-chip ${e.tone}`,children:e.children})}function N(e){return(0,b.jsxs)(`span`,{className:`access-badge ${e.tone}`,children:[e.tone===`green`?(0,b.jsx)(a,{size:13}):e.tone===`red`?(0,b.jsx)(d,{size:13}):(0,b.jsx)(p,{size:13}),e.children]})}function P(e){let t=e.platformKey.toLowerCase(),n=x[t];return(0,b.jsx)(`span`,{className:`platform-icon ${e.tone} platform-${t}`,children:n?(0,b.jsx)(`img`,{alt:``,src:n}):e.icon})}function F(e){return(0,b.jsxs)(`div`,{className:`network-metric ${S(e.tone)}`,children:[(0,b.jsx)(`span`,{children:e.label}),(0,b.jsx)(`strong`,{children:e.value}),e.detail?(0,b.jsx)(`p`,{children:e.detail}):null]})}function I(e){let t=e.icon;return(0,b.jsxs)(`div`,{className:`signal-card ${S(e.tone)}`,children:[(0,b.jsx)(`div`,{className:`signal-card-icon`,children:(0,b.jsx)(t,{size:17})}),(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`span`,{children:e.label}),(0,b.jsx)(`strong`,{children:e.value}),(0,b.jsx)(`p`,{children:e.detail})]})]})}function L(e){return(0,b.jsxs)(`div`,{className:`network-block ${S(e.tone)}`,children:[(0,b.jsx)(`span`,{children:e.label}),(0,b.jsx)(`strong`,{children:e.value}),(0,b.jsx)(`p`,{children:e.detail})]})}function R(e){let{item:t}=e;return(0,b.jsxs)(`div`,{className:`platform-access-row ${t.tone}`,children:[(0,b.jsxs)(`div`,{className:`platform-access-main`,children:[(0,b.jsx)(P,{icon:t.label.slice(0,1),tone:t.tone,platformKey:t.key}),(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`strong`,{children:t.label}),(0,b.jsx)(`span`,{children:C(t.url)})]})]}),(0,b.jsxs)(`div`,{className:`platform-access-result`,children:[(0,b.jsx)(N,{tone:t.tone,children:t.status}),(0,b.jsx)(`span`,{children:t.httpStatus?`HTTP ${t.httpStatus}`:`连接结果`})]}),(0,b.jsxs)(`div`,{className:`platform-access-latency`,children:[(0,b.jsx)(`strong`,{children:z(t.elapsedMs)}),(0,b.jsx)(`span`,{children:t.httpStatus?`${t.httpStatus} 响应`:`等待结果`})]}),(0,b.jsx)(`div`,{className:`platform-access-detail`,children:(0,b.jsx)(`span`,{children:t.detail})})]})}function z(e){return typeof e==`number`?`${e} ms`:`-`}function B(e){return e instanceof DOMException&&e.name===`AbortError`?`网络探测超过 18 秒,已停止等待。`:e instanceof Error?e.message:String(e)}function V(){let[e,t]=(0,m.useState)(null),[n,r]=(0,m.useState)(null),[a,d]=(0,m.useState)(!0),[p,h]=(0,m.useState)(`正在采集真实网络状态...`),[g,_]=(0,m.useState)(null);async function v(){d(!0),_(null),h(`正在采集真实网络状态...`);let e=new AbortController,n=window.setTimeout(()=>e.abort(),18e3);try{let[n,i]=await Promise.allSettled([ee(`/_gateway/admin/network-detect`,{signal:e.signal}),D()]);n.status===`fulfilled`&&t(n.value),i.status===`fulfilled`&&r(i.value);let a=[];n.status===`rejected`&&a.push(`网络探测: ${B(n.reason)}`),i.status===`rejected`&&a.push(`本地环境: ${B(i.reason)}`);let o=n.status===`fulfilled`&&i.status===`fulfilled`?`检测完成。`:n.status===`fulfilled`||i.status===`fulfilled`?`部分结果已更新。`:`检测失败。`;_(a.length>0?a.join(` · `):null),h(o)}catch(e){_(B(e)),h(`检测失败。`)}finally{window.clearTimeout(n),d(!1)}}(0,m.useEffect)(()=>{v().catch(()=>void 0)},[]);let y=(0,m.useMemo)(()=>e?new Intl.DateTimeFormat(`zh-CN`,{hour:`2-digit`,minute:`2-digit`,second:`2-digit`,hour12:!1}).format(new Date(e.checkedAt)):`--:--:--`,[e]),x=te(e,n),S=O(e?.dns.servers||[]),C=k(e?.dns.servers||[]),T=e?.dns.servers?.length?e.dns.servers.slice(0,2).join(` / `):`-`,E=e?.dns.detail||`尚未检测到系统 DNS。`,N=A(e?.platforms||[]),P=e?e.publicIpv4.available&&e.publicIpv4.ip?`green`:`orange`:`blue`,V=e?e.publicIpv6.available?`green`:`orange`:`blue`,H=e?e.publicIpv4.available&&e.publicIpv4.ip?e.publicIpv4.ip:`未获取到`:a?`检测中`:`未读取`,U=e?.publicIpv4.detail||`正在采集公网 IPv4 出口。`,W=e?.proxy.enabled?`代理已启用`:e?`直连`:`待检测`,G=e?.proxy.enabled?e.proxy.url||`代理地址未读取`:`检测请求按当前网关配置发起。`,K=e?e.proxy.enabled?`orange`:`green`:`blue`,q=j(e,N),J=N.total>0?Math.round(N.reachable/N.total*100):0,Y=e?`${J}%`:`--`,X=N.total>0?String(N.total):`5`,Z=e?.publicIpv4.countryName?`${e.publicIpv4.countryName}${e.publicIpv4.colo?` · ${e.publicIpv4.colo}`:``}`:e?.publicIpv4.countryCode||`未知出口`,ne=e?.publicIpv4.available?`${e.publicIpv4.source} · ${z(e.publicIpv4.elapsedMs)}`:U,Q=e?.platforms.find(e=>e.tone===`red`)??e?.platforms.find(e=>e.tone===`orange`),re=Q?`${Q.label}: ${Q.detail}`:e?`海外目标应用均有响应。`:`等待海外目标应用响应。`,$=n?.webrtc.tone===`red`?`red`:n?.webrtc.tone===`orange`||S===`orange`?`orange`:S===`slate`||!n?`blue`:`green`;return(0,b.jsxs)(`section`,{className:`network-page`,children:[(0,b.jsxs)(`section`,{className:`network-command-center ${q.tone}`,children:[(0,b.jsxs)(`div`,{className:`network-command-copy`,children:[(0,b.jsxs)(`div`,{className:`network-eyebrow`,children:[(0,b.jsx)(u,{size:14}),(0,b.jsx)(`span`,{children:`海外应用访问诊断`})]}),(0,b.jsx)(`h2`,{children:q.label}),(0,b.jsx)(`p`,{children:q.detail}),(0,b.jsxs)(`div`,{className:`network-command-status`,children:[(0,b.jsx)(M,{tone:x.tone,children:x.label}),(0,b.jsxs)(`span`,{children:[(0,b.jsx)(o,{size:13}),y]}),(0,b.jsx)(`span`,{className:`network-toolbar-status`,children:p})]})]}),(0,b.jsxs)(`div`,{className:`access-score-card`,children:[(0,b.jsx)(`span`,{children:`海外可达率`}),(0,b.jsx)(`strong`,{children:Y}),(0,b.jsx)(`div`,{className:`access-score-bar`,"aria-hidden":`true`,children:(0,b.jsx)(`i`,{style:{width:`${J}%`}})}),(0,b.jsxs)(`p`,{children:[N.reachable,`/`,X,` 个目标应用可达`]}),(0,b.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:()=>v().catch(()=>void 0),disabled:a,children:[(0,b.jsx)(s,{size:16}),a?`检测中`:`重新检测`]})]})]}),g?(0,b.jsxs)(`p`,{className:`network-error`,children:[`检测提示: `,g]}):null,(0,b.jsxs)(`section`,{className:`signal-grid`,children:[(0,b.jsx)(I,{icon:i,label:`海外连通性`,value:`${N.reachable}/${X}`,detail:re,tone:q.tone}),(0,b.jsx)(I,{icon:f,label:`当前出口`,value:e?.publicIpv4.available?Z:H,detail:ne,tone:P}),(0,b.jsx)(I,{icon:c,label:`代理路径`,value:W,detail:G,tone:K}),(0,b.jsx)(I,{icon:l,label:`解析与浏览器`,value:n?`${C} · ${n.webrtc.status}`:`检测中`,detail:n?.webrtc.detail||E,tone:$})]}),(0,b.jsxs)(`section`,{className:`network-workspace`,children:[(0,b.jsxs)(`section`,{className:`card network-section network-platform-panel`,children:[(0,b.jsxs)(`div`,{className:`section-head compact`,children:[(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`h3`,{children:`海外应用矩阵`}),(0,b.jsx)(`p`,{children:`反代链路最常用目标的实时可达性。`})]}),(0,b.jsxs)(`div`,{className:`platform-summary-strip`,children:[(0,b.jsxs)(`div`,{className:`platform-summary-chip green`,children:[(0,b.jsx)(`strong`,{children:N.reachable}),(0,b.jsx)(`span`,{children:`可达`})]}),(0,b.jsxs)(`div`,{className:`platform-summary-chip orange`,children:[(0,b.jsx)(`strong`,{children:N.limited}),(0,b.jsx)(`span`,{children:`受限`})]}),(0,b.jsxs)(`div`,{className:`platform-summary-chip red`,children:[(0,b.jsx)(`strong`,{children:N.unavailable}),(0,b.jsx)(`span`,{children:`阻断`})]})]})]}),e?.platforms.length?(0,b.jsxs)(b.Fragment,{children:[(0,b.jsxs)(`div`,{className:`platform-access-header`,"aria-hidden":`true`,children:[(0,b.jsx)(`span`,{children:`应用`}),(0,b.jsx)(`span`,{children:`状态`}),(0,b.jsx)(`span`,{children:`耗时`}),(0,b.jsx)(`span`,{children:`说明`})]}),(0,b.jsx)(`div`,{className:`platform-access-list`,children:e.platforms.map(e=>(0,b.jsx)(R,{item:e},e.key))})]}):(0,b.jsx)(`div`,{className:`network-empty-state`,children:`平台探测结果还没有返回。`})]}),(0,b.jsxs)(`aside`,{className:`network-side`,children:[(0,b.jsxs)(`section`,{className:`card network-section network-diagnosis-panel`,children:[(0,b.jsx)(`div`,{className:`section-head compact`,children:(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`h3`,{children:`访问结论`}),(0,b.jsx)(`p`,{children:`用于判断当前环境是否适合海外反向代理。`})]})}),(0,b.jsxs)(`div`,{className:`network-list`,children:[(0,b.jsxs)(`div`,{className:`network-list-item`,children:[(0,b.jsx)(M,{tone:q.tone,children:q.label}),(0,b.jsx)(`p`,{children:q.detail})]}),(0,b.jsxs)(`div`,{className:`network-list-item`,children:[(0,b.jsx)(M,{tone:x.tone,children:x.label}),(0,b.jsx)(`p`,{children:x.detail})]}),(0,b.jsxs)(`div`,{className:`network-list-item`,children:[(0,b.jsx)(M,{tone:n?.webrtc.tone||`blue`,children:n?.webrtc.status||`检测中`}),(0,b.jsx)(`p`,{children:n?.webrtc.detail||`正在采集 WebRTC 候选。`})]})]})]}),(0,b.jsxs)(`section`,{className:`card network-section`,children:[(0,b.jsx)(`div`,{className:`section-head compact`,children:(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`h3`,{children:`出口与代理`}),(0,b.jsx)(`p`,{children:`反代请求实际经过的公网出口。`})]})}),(0,b.jsxs)(`div`,{className:`network-dual`,children:[(0,b.jsx)(L,{label:`IPv4 出口`,value:H,detail:e?e.publicIpv4.available?`${Z} · ${z(e.publicIpv4.elapsedMs)}`:`${e.publicIpv4.detail} · ${z(e.publicIpv4.elapsedMs)}`:`正在采集公网出口信息。`,tone:P}),(0,b.jsx)(L,{label:`IPv6 出口`,value:e?.publicIpv6.available?e.publicIpv6.ip||`-`:`未检测到`,detail:e?.publicIpv6.detail||`未检测到独立 IPv6 出口。`,tone:V})]}),(0,b.jsx)(`div`,{className:`network-list compact-list`,children:(0,b.jsxs)(`div`,{className:`network-list-item`,children:[(0,b.jsx)(M,{tone:K,children:W}),(0,b.jsx)(`p`,{children:G})]})})]})]})]}),(0,b.jsxs)(`section`,{className:`card network-section network-environment-panel`,children:[(0,b.jsx)(`div`,{className:`section-head compact`,children:(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`h3`,{children:`解析与本地指纹`}),(0,b.jsx)(`p`,{children:`DNS、WebRTC、时区和语言会影响海外应用的风控判断。`})]})}),(0,b.jsxs)(`div`,{className:`network-three-up`,children:[(0,b.jsx)(F,{label:`DNS`,value:T,detail:`${C} · ${e?.dns.source||`尚未获取`}`,tone:S}),(0,b.jsx)(F,{label:`WebRTC`,value:n?.webrtc.status||(a?`检测中`:`-`),detail:n?.webrtc.detail||`正在采集 WebRTC 候选。`,tone:n?.webrtc.tone||`blue`}),(0,b.jsx)(F,{label:`本地环境`,value:n?.timezone||`-`,detail:`${n?.language||`-`} · ${n?.browser||`-`}`,tone:$})]}),e?.dns.servers.length?(0,b.jsx)(`div`,{className:`dns-chip-row`,children:e.dns.servers.map(e=>(0,b.jsx)(`span`,{className:`dns-chip ${w(e)?`orange`:`green`}`,children:e},e))}):null]})]})}export{V as NetworkDetectPage};
1
+ import{a as e,n as t,r as n,t as r}from"./jsx-runtime-DqpGtLhh.js";import{t as i}from"./activity-D21-Xrc4.js";import{t as a}from"./circle-check-ZYtn9GqY.js";import{t as o}from"./clock-3-BzDANsVk.js";import{t as s}from"./refresh-cw-CAAH2rqe.js";import{t as c}from"./server-BrjJPb9D.js";import{g as l,m as u,p as ee}from"./index-CCiBaGwU.js";var d=t(`circle-x`,[[`circle`,{cx:`12`,cy:`12`,r:`10`,key:`1mglay`}],[`path`,{d:`m15 9-6 6`,key:`1uzhvr`}],[`path`,{d:`m9 9 6 6`,key:`z0biqf`}]]),f=t(`map-pin`,[[`path`,{d:`M20 10c0 4.993-5.539 10.193-7.399 11.799a1 1 0 0 1-1.202 0C9.539 20.193 4 14.993 4 10a8 8 0 0 1 16 0`,key:`1r0f0z`}],[`circle`,{cx:`12`,cy:`10`,r:`3`,key:`ilqhr7`}]]),p=t(`triangle-alert`,[[`path`,{d:`m21.73 18-8-14a2 2 0 0 0-3.48 0l-8 14A2 2 0 0 0 4 21h16a2 2 0 0 0 1.73-3`,key:`wmoenq`}],[`path`,{d:`M12 9v4`,key:`juzpu7`}],[`path`,{d:`M12 17h.01`,key:`p32p05`}]]),m=e(n(),1),h=`data:image/svg+xml,%3csvg%20role='img'%20viewBox='0%200%2024%2024'%20xmlns='http://www.w3.org/2000/svg'%3e%3ctitle%3eClaude%3c/title%3e%3cpath%20d='m4.7144%2015.9555%204.7174-2.6471.079-.2307-.079-.1275h-.2307l-.7893-.0486-2.6956-.0729-2.3375-.0971-2.2646-.1214-.5707-.1215-.5343-.7042.0546-.3522.4797-.3218.686.0608%201.5179.1032%202.2767.1578%201.6514.0972%202.4468.255h.3886l.0546-.1579-.1336-.0971-.1032-.0972L6.973%209.8356l-2.55-1.6879-1.3356-.9714-.7225-.4918-.3643-.4614-.1578-1.0078.6557-.7225.8803.0607.2246.0607.8925.686%201.9064%201.4754%202.4893%201.8336.3643.3035.1457-.1032.0182-.0728-.164-.2733-1.3539-2.4467-1.445-2.4893-.6435-1.032-.17-.6194c-.0607-.255-.1032-.4674-.1032-.7285L6.287.1335%206.6997%200l.9957.1336.419.3642.6192%201.4147%201.0018%202.2282%201.5543%203.0296.4553.8985.2429.8318.091.255h.1579v-.1457l.1275-1.706.2368-2.0947.2307-2.6957.0789-.7589.3764-.9107.7468-.4918.5828.2793.4797.686-.0668.4433-.2853%201.8517-.5586%202.9021-.3643%201.9429h.2125l.2429-.2429.9835-1.3053%201.6514-2.0643.7286-.8196.85-.9046.5464-.4311h1.0321l.759%201.1293-.34%201.1657-1.0625%201.3478-.8804%201.1414-1.2628%201.7-.7893%201.36.0729.1093.1882-.0183%202.8535-.607%201.5421-.2794%201.8396-.3157.8318.3886.091.3946-.3278.8075-1.967.4857-2.3072.4614-3.4364.8136-.0425.0304.0486.0607%201.5482.1457.6618.0364h1.621l3.0175.2247.7892.522.4736.6376-.079.4857-1.2142.6193-1.6393-.3886-3.825-.9107-1.3113-.3279h-.1822v.1093l1.0929%201.0686%202.0035%201.8092%202.5075%202.3314.1275.5768-.3218.4554-.34-.0486-2.2039-1.6575-.85-.7468-1.9246-1.621h-.1275v.17l.4432.6496%202.3436%203.5214.1214%201.0807-.17.3521-.6071.2125-.6679-.1214-1.3721-1.9246L14.38%2017.959l-1.1414-1.9428-.1397.079-.674%207.2552-.3156.3703-.7286.2793-.6071-.4614-.3218-.7468.3218-1.4753.3886-1.9246.3157-1.53.2853-1.9004.17-.6314-.0121-.0425-.1397.0182-1.4328%201.9672-2.1796%202.9446-1.7243%201.8456-.4128.164-.7164-.3704.0667-.6618.4008-.5889%202.386-3.0357%201.4389-1.882.929-1.0868-.0062-.1579h-.0546l-6.3385%204.1164-1.1293.1457-.4857-.4554.0608-.7467.2307-.2429%201.9064-1.3114Z'/%3e%3c/svg%3e`,g=`data:image/svg+xml,%3csvg%20role='img'%20viewBox='0%200%2024%2024'%20xmlns='http://www.w3.org/2000/svg'%3e%3ctitle%3eOpenAI%3c/title%3e%3cpath%20d='M22.2819%209.8211a5.9847%205.9847%200%200%200-.5157-4.9108%206.0462%206.0462%200%200%200-6.5098-2.9A6.0651%206.0651%200%200%200%204.9807%204.1818a5.9847%205.9847%200%200%200-3.9977%202.9%206.0462%206.0462%200%200%200%20.7427%207.0966%205.98%205.98%200%200%200%20.511%204.9107%206.051%206.051%200%200%200%206.5146%202.9001A5.9847%205.9847%200%200%200%2013.2599%2024a6.0557%206.0557%200%200%200%205.7718-4.2058%205.9894%205.9894%200%200%200%203.9977-2.9001%206.0557%206.0557%200%200%200-.7475-7.0729zm-9.022%2012.6081a4.4755%204.4755%200%200%201-2.8764-1.0408l.1419-.0804%204.7783-2.7582a.7948.7948%200%200%200%20.3927-.6813v-6.7369l2.02%201.1686a.071.071%200%200%201%20.038.052v5.5826a4.504%204.504%200%200%201-4.4945%204.4944zm-9.6607-4.1254a4.4708%204.4708%200%200%201-.5346-3.0137l.142.0852%204.783%202.7582a.7712.7712%200%200%200%20.7806%200l5.8428-3.3685v2.3324a.0804.0804%200%200%201-.0332.0615L9.74%2019.9502a4.4992%204.4992%200%200%201-6.1408-1.6464zM2.3408%207.8956a4.485%204.485%200%200%201%202.3655-1.9728V11.6a.7664.7664%200%200%200%20.3879.6765l5.8144%203.3543-2.0201%201.1685a.0757.0757%200%200%201-.071%200l-4.8303-2.7865A4.504%204.504%200%200%201%202.3408%207.872zm16.5963%203.8558L13.1038%208.364%2015.1192%207.2a.0757.0757%200%200%201%20.071%200l4.8303%202.7913a4.4944%204.4944%200%200%201-.6765%208.1042v-5.6772a.79.79%200%200%200-.407-.667zm2.0107-3.0231l-.142-.0852-4.7735-2.7818a.7759.7759%200%200%200-.7854%200L9.409%209.2297V6.8974a.0662.0662%200%200%201%20.0284-.0615l4.8303-2.7866a4.4992%204.4992%200%200%201%206.6802%204.66zM8.3065%2012.863l-2.02-1.1638a.0804.0804%200%200%201-.038-.0567V6.0742a4.4992%204.4992%200%200%201%207.3757-3.4537l-.142.0805L8.704%205.459a.7948.7948%200%200%200-.3927.6813zm1.0976-2.3654l2.602-1.4998%202.6069%201.4998v2.9994l-2.5974%201.4997-2.6067-1.4997Z'/%3e%3c/svg%3e`,_=`data:image/svg+xml,%3csvg%20role='img'%20viewBox='0%200%2024%2024'%20xmlns='http://www.w3.org/2000/svg'%3e%3ctitle%3eGoogle%3c/title%3e%3cpath%20d='M12.48%2010.92v3.28h7.84c-.24%201.84-.853%203.187-1.787%204.133-1.147%201.147-2.933%202.4-6.053%202.4-4.827%200-8.6-3.893-8.6-8.72s3.773-8.72%208.6-8.72c2.6%200%204.507%201.027%205.907%202.347l2.307-2.307C18.747%201.44%2016.133%200%2012.48%200%205.867%200%20.307%205.387.307%2012s5.56%2012%2012.173%2012c3.573%200%206.267-1.173%208.373-3.36%202.16-2.16%202.84-5.213%202.84-7.667%200-.76-.053-1.467-.173-2.053H12.48z'/%3e%3c/svg%3e`,v=`data:image/svg+xml,%3csvg%20role='img'%20viewBox='0%200%2024%2024'%20xmlns='http://www.w3.org/2000/svg'%3e%3ctitle%3eX%3c/title%3e%3cpath%20d='M14.234%2010.162%2022.977%200h-2.072l-7.591%208.824L7.251%200H.258l9.168%2013.343L.258%2024H2.33l8.016-9.318L16.749%2024h6.993zm-2.837%203.299-.929-1.329L3.076%201.56h3.182l5.965%208.532.929%201.329%207.754%2011.09h-3.182z'/%3e%3c/svg%3e`,y=`data:image/svg+xml,%3csvg%20role='img'%20viewBox='0%200%2024%2024'%20xmlns='http://www.w3.org/2000/svg'%3e%3ctitle%3eYouTube%3c/title%3e%3cpath%20d='M23.498%206.186a3.016%203.016%200%200%200-2.122-2.136C19.505%203.545%2012%203.545%2012%203.545s-7.505%200-9.377.505A3.017%203.017%200%200%200%20.502%206.186C0%208.07%200%2012%200%2012s0%203.93.502%205.814a3.016%203.016%200%200%200%202.122%202.136c1.871.505%209.376.505%209.376.505s7.505%200%209.377-.505a3.015%203.015%200%200%200%202.122-2.136C24%2015.93%2024%2012%2024%2012s0-3.93-.502-5.814zM9.545%2015.568V8.432L15.818%2012l-6.273%203.568z'/%3e%3c/svg%3e`,b=r(),x={claude:h,chatgpt:g,google:_,x:v,youtube:y};function S(e){return`tone-${e}`}function C(e){try{return new URL(e).hostname.replace(/^www\./i,``)}catch{return e}}function w(e){return/^(10|127|169\.254|172\.(1[6-9]|2\d|3[0-1])|192\.168)\./.test(e)||/^100\.(6[4-9]|[7-9]\d|1\d\d|2[0-3]\d|24[0-7])\./.test(e)||/^198\.(18|19)\./.test(e)||/^192\.0\.2\./.test(e)||/^198\.51\.100\./.test(e)||/^203\.0\.113\./.test(e)||/^fc00:/i.test(e)||/^fd00:/i.test(e)||/^fe80:/i.test(e)||/^::1$/.test(e)}function T(e){let t=e.match(/Electron\/([\d.]+)/i);if(t)return`Electron ${t[1]}`;let n=e.match(/Chrome\/([\d.]+)/i);if(n&&/Safari\//i.test(e))return`Chrome ${n[1].split(`.`)[0]}`;let r=e.match(/Version\/([\d.]+).*Safari\//i);if(r&&!/Chrome\//i.test(e))return`Safari ${r[1].split(`.`)[0]}`;let i=e.match(/Firefox\/([\d.]+)/i);return i?`Firefox ${i[1].split(`.`)[0]}`:e.slice(0,40)}async function E(){if(typeof RTCPeerConnection>`u`)return{status:`不支持`,detail:`当前环境不支持 WebRTC。`,tone:`orange`,candidates:[]};let e=new Set,t=new RTCPeerConnection({iceServers:[]}),n=!1,r=new Promise(r=>{t.onicecandidate=t=>{t.candidate?.candidate&&e.add(t.candidate.candidate),t.candidate||(n=!0,r())}});try{t.createDataChannel(`probe`);let e=await t.createOffer();await t.setLocalDescription(e),await Promise.race([r,new Promise(e=>window.setTimeout(e,2400))])}catch{}finally{t.close()}let i=[...e],a=i.some(e=>/\btyp srflx\b/.test(e)||/\btyp relay\b/.test(e)),o=i.some(e=>/\btyp host\b/.test(e));return a?{status:`需要留意`,detail:`收集到 ${i.length} 个候选地址,其中包含公网回显特征。`,tone:`red`,candidates:i}:o||n?{status:`本地候选为主`,detail:`收集到 ${i.length} 个候选地址,主要来自本地网段。`,tone:`orange`,candidates:i}:{status:`暂未发现异常`,detail:`未收集到可识别的候选地址。`,tone:`green`,candidates:i}}async function D(){let[e]=await Promise.all([E()]);return{timezone:Intl.DateTimeFormat().resolvedOptions().timeZone||`-`,language:navigator.language||`-`,browser:T(navigator.userAgent||``),webrtc:e}}function O(e){return e.length===0?`slate`:e.some(e=>w(e))?`orange`:`green`}function k(e){return e.length===0?`未读取`:e.some(e=>w(e))?`内网 / 企业 DNS`:`公网 DNS`}function A(e){return{total:e.length,reachable:e.filter(e=>e.status===`可达`).length,limited:e.filter(e=>e.status===`受限`).length,unavailable:e.filter(e=>e.status===`不可用`).length}}function j(e,t){return!e||t.total===0?{label:`正在判断海外访问`,detail:`正在检查 ChatGPT、Claude、Google、YouTube 和 X 的连通性。`,tone:`blue`}:t.unavailable>0?{label:`海外访问受阻`,detail:`${t.unavailable} 个海外应用不可用,优先检查代理出口、DNS 或上游链路。`,tone:`red`}:t.limited>0?{label:`部分应用受限`,detail:`${t.reachable}/${t.total} 个海外应用可达,其余应用返回限制状态。`,tone:`orange`}:{label:`海外应用可访问`,detail:`${t.total} 个目标应用均已连通,当前出口可用于海外反向代理场景。`,tone:`green`}}function te(e,t){if(!e&&!t)return{label:`检测中`,detail:`正在采集公网出口、DNS、WebRTC、代理和平台可达性。`,tone:`blue`};if(!e||!t)return{label:`部分结果`,detail:`已有项目返回,剩余项目会保留失败原因,方便继续排查。`,tone:`orange`};let n=e.publicIpv4.available&&!!e.publicIpv4.ip,r=e.publicIpv6.available&&!!e.publicIpv6.ip,i=[e.platforms.some(e=>e.tone===`red`),t.webrtc.tone===`red`,!n&&!r],a=[e.platforms.some(e=>e.tone===`orange`),t.webrtc.tone===`orange`,O(e.dns.servers)!==`green`,!n,!e.publicIpv6.available];return i.some(Boolean)?{label:`建议先处理`,detail:`有明显阻塞项,先把出口或浏览器回显风险排掉。`,tone:`red`}:a.some(Boolean)?{label:`需要留意`,detail:`结果可用,但有几项信息值得再看一眼。`,tone:`orange`}:{label:`状态正常`,detail:`出口、DNS、WebRTC 和常用平台状态都比较稳定。`,tone:`green`}}function M(e){return(0,b.jsx)(`span`,{className:`network-chip ${e.tone}`,children:e.children})}function N(e){return(0,b.jsxs)(`span`,{className:`access-badge ${e.tone}`,children:[e.tone===`green`?(0,b.jsx)(a,{size:13}):e.tone===`red`?(0,b.jsx)(d,{size:13}):(0,b.jsx)(p,{size:13}),e.children]})}function P(e){let t=e.platformKey.toLowerCase(),n=x[t];return(0,b.jsx)(`span`,{className:`platform-icon ${e.tone} platform-${t}`,children:n?(0,b.jsx)(`img`,{alt:``,src:n}):e.icon})}function F(e){return(0,b.jsxs)(`div`,{className:`network-metric ${S(e.tone)}`,children:[(0,b.jsx)(`span`,{children:e.label}),(0,b.jsx)(`strong`,{children:e.value}),e.detail?(0,b.jsx)(`p`,{children:e.detail}):null]})}function I(e){let t=e.icon;return(0,b.jsxs)(`div`,{className:`signal-card ${S(e.tone)}`,children:[(0,b.jsx)(`div`,{className:`signal-card-icon`,children:(0,b.jsx)(t,{size:17})}),(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`span`,{children:e.label}),(0,b.jsx)(`strong`,{children:e.value}),(0,b.jsx)(`p`,{children:e.detail})]})]})}function L(e){return(0,b.jsxs)(`div`,{className:`network-block ${S(e.tone)}`,children:[(0,b.jsx)(`span`,{children:e.label}),(0,b.jsx)(`strong`,{children:e.value}),(0,b.jsx)(`p`,{children:e.detail})]})}function R(e){let{item:t}=e;return(0,b.jsxs)(`div`,{className:`platform-access-row ${t.tone}`,children:[(0,b.jsxs)(`div`,{className:`platform-access-main`,children:[(0,b.jsx)(P,{icon:t.label.slice(0,1),tone:t.tone,platformKey:t.key}),(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`strong`,{children:t.label}),(0,b.jsx)(`span`,{children:C(t.url)})]})]}),(0,b.jsxs)(`div`,{className:`platform-access-result`,children:[(0,b.jsx)(N,{tone:t.tone,children:t.status}),(0,b.jsx)(`span`,{children:t.httpStatus?`HTTP ${t.httpStatus}`:`连接结果`})]}),(0,b.jsxs)(`div`,{className:`platform-access-latency`,children:[(0,b.jsx)(`strong`,{children:z(t.elapsedMs)}),(0,b.jsx)(`span`,{children:t.httpStatus?`${t.httpStatus} 响应`:`等待结果`})]}),(0,b.jsx)(`div`,{className:`platform-access-detail`,children:(0,b.jsx)(`span`,{children:t.detail})})]})}function z(e){return typeof e==`number`?`${e} ms`:`-`}function B(e){return e instanceof DOMException&&e.name===`AbortError`?`网络探测超过 18 秒,已停止等待。`:e instanceof Error?e.message:String(e)}function V(){let[e,t]=(0,m.useState)(null),[n,r]=(0,m.useState)(null),[a,d]=(0,m.useState)(!0),[p,h]=(0,m.useState)(`正在采集真实网络状态...`),[g,_]=(0,m.useState)(null);async function v(){d(!0),_(null),h(`正在采集真实网络状态...`);let e=new AbortController,n=window.setTimeout(()=>e.abort(),18e3);try{let[n,i]=await Promise.allSettled([ee(`/_gateway/admin/network-detect`,{signal:e.signal}),D()]);n.status===`fulfilled`&&t(n.value),i.status===`fulfilled`&&r(i.value);let a=[];n.status===`rejected`&&a.push(`网络探测: ${B(n.reason)}`),i.status===`rejected`&&a.push(`本地环境: ${B(i.reason)}`);let o=n.status===`fulfilled`&&i.status===`fulfilled`?`检测完成。`:n.status===`fulfilled`||i.status===`fulfilled`?`部分结果已更新。`:`检测失败。`;_(a.length>0?a.join(` · `):null),h(o)}catch(e){_(B(e)),h(`检测失败。`)}finally{window.clearTimeout(n),d(!1)}}(0,m.useEffect)(()=>{v().catch(()=>void 0)},[]);let y=(0,m.useMemo)(()=>e?new Intl.DateTimeFormat(`zh-CN`,{hour:`2-digit`,minute:`2-digit`,second:`2-digit`,hour12:!1}).format(new Date(e.checkedAt)):`--:--:--`,[e]),x=te(e,n),S=O(e?.dns.servers||[]),C=k(e?.dns.servers||[]),T=e?.dns.servers?.length?e.dns.servers.slice(0,2).join(` / `):`-`,E=e?.dns.detail||`尚未检测到系统 DNS。`,N=A(e?.platforms||[]),P=e?e.publicIpv4.available&&e.publicIpv4.ip?`green`:`orange`:`blue`,V=e?e.publicIpv6.available?`green`:`orange`:`blue`,H=e?e.publicIpv4.available&&e.publicIpv4.ip?e.publicIpv4.ip:`未获取到`:a?`检测中`:`未读取`,U=e?.publicIpv4.detail||`正在采集公网 IPv4 出口。`,W=e?.proxy.enabled?`代理已启用`:e?`直连`:`待检测`,G=e?.proxy.enabled?e.proxy.url||`代理地址未读取`:`检测请求按当前网关配置发起。`,K=e?e.proxy.enabled?`orange`:`green`:`blue`,q=j(e,N),J=N.total>0?Math.round(N.reachable/N.total*100):0,Y=e?`${J}%`:`--`,X=N.total>0?String(N.total):`5`,Z=e?.publicIpv4.countryName?`${e.publicIpv4.countryName}${e.publicIpv4.colo?` · ${e.publicIpv4.colo}`:``}`:e?.publicIpv4.countryCode||`未知出口`,ne=e?.publicIpv4.available?`${e.publicIpv4.source} · ${z(e.publicIpv4.elapsedMs)}`:U,Q=e?.platforms.find(e=>e.tone===`red`)??e?.platforms.find(e=>e.tone===`orange`),re=Q?`${Q.label}: ${Q.detail}`:e?`海外目标应用均有响应。`:`等待海外目标应用响应。`,$=n?.webrtc.tone===`red`?`red`:n?.webrtc.tone===`orange`||S===`orange`?`orange`:S===`slate`||!n?`blue`:`green`;return(0,b.jsxs)(`section`,{className:`network-page`,children:[(0,b.jsxs)(`section`,{className:`network-command-center ${q.tone}`,children:[(0,b.jsxs)(`div`,{className:`network-command-copy`,children:[(0,b.jsxs)(`div`,{className:`network-eyebrow`,children:[(0,b.jsx)(u,{size:14}),(0,b.jsx)(`span`,{children:`海外应用访问诊断`})]}),(0,b.jsx)(`h2`,{children:q.label}),(0,b.jsx)(`p`,{children:q.detail}),(0,b.jsxs)(`div`,{className:`network-command-status`,children:[(0,b.jsx)(M,{tone:x.tone,children:x.label}),(0,b.jsxs)(`span`,{children:[(0,b.jsx)(o,{size:13}),y]}),(0,b.jsx)(`span`,{className:`network-toolbar-status`,children:p})]})]}),(0,b.jsxs)(`div`,{className:`access-score-card`,children:[(0,b.jsx)(`span`,{children:`海外可达率`}),(0,b.jsx)(`strong`,{children:Y}),(0,b.jsx)(`div`,{className:`access-score-bar`,"aria-hidden":`true`,children:(0,b.jsx)(`i`,{style:{width:`${J}%`}})}),(0,b.jsxs)(`p`,{children:[N.reachable,`/`,X,` 个目标应用可达`]}),(0,b.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:()=>v().catch(()=>void 0),disabled:a,children:[(0,b.jsx)(s,{size:16}),a?`检测中`:`重新检测`]})]})]}),g?(0,b.jsxs)(`p`,{className:`network-error`,children:[`检测提示: `,g]}):null,(0,b.jsxs)(`section`,{className:`signal-grid`,children:[(0,b.jsx)(I,{icon:i,label:`海外连通性`,value:`${N.reachable}/${X}`,detail:re,tone:q.tone}),(0,b.jsx)(I,{icon:f,label:`当前出口`,value:e?.publicIpv4.available?Z:H,detail:ne,tone:P}),(0,b.jsx)(I,{icon:c,label:`代理路径`,value:W,detail:G,tone:K}),(0,b.jsx)(I,{icon:l,label:`解析与浏览器`,value:n?`${C} · ${n.webrtc.status}`:`检测中`,detail:n?.webrtc.detail||E,tone:$})]}),(0,b.jsxs)(`section`,{className:`network-workspace`,children:[(0,b.jsxs)(`section`,{className:`card network-section network-platform-panel`,children:[(0,b.jsxs)(`div`,{className:`section-head compact`,children:[(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`h3`,{children:`海外应用矩阵`}),(0,b.jsx)(`p`,{children:`反代链路最常用目标的实时可达性。`})]}),(0,b.jsxs)(`div`,{className:`platform-summary-strip`,children:[(0,b.jsxs)(`div`,{className:`platform-summary-chip green`,children:[(0,b.jsx)(`strong`,{children:N.reachable}),(0,b.jsx)(`span`,{children:`可达`})]}),(0,b.jsxs)(`div`,{className:`platform-summary-chip orange`,children:[(0,b.jsx)(`strong`,{children:N.limited}),(0,b.jsx)(`span`,{children:`受限`})]}),(0,b.jsxs)(`div`,{className:`platform-summary-chip red`,children:[(0,b.jsx)(`strong`,{children:N.unavailable}),(0,b.jsx)(`span`,{children:`阻断`})]})]})]}),e?.platforms.length?(0,b.jsxs)(b.Fragment,{children:[(0,b.jsxs)(`div`,{className:`platform-access-header`,"aria-hidden":`true`,children:[(0,b.jsx)(`span`,{children:`应用`}),(0,b.jsx)(`span`,{children:`状态`}),(0,b.jsx)(`span`,{children:`耗时`}),(0,b.jsx)(`span`,{children:`说明`})]}),(0,b.jsx)(`div`,{className:`platform-access-list`,children:e.platforms.map(e=>(0,b.jsx)(R,{item:e},e.key))})]}):(0,b.jsx)(`div`,{className:`network-empty-state`,children:`平台探测结果还没有返回。`})]}),(0,b.jsxs)(`aside`,{className:`network-side`,children:[(0,b.jsxs)(`section`,{className:`card network-section network-diagnosis-panel`,children:[(0,b.jsx)(`div`,{className:`section-head compact`,children:(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`h3`,{children:`访问结论`}),(0,b.jsx)(`p`,{children:`用于判断当前环境是否适合海外反向代理。`})]})}),(0,b.jsxs)(`div`,{className:`network-list`,children:[(0,b.jsxs)(`div`,{className:`network-list-item`,children:[(0,b.jsx)(M,{tone:q.tone,children:q.label}),(0,b.jsx)(`p`,{children:q.detail})]}),(0,b.jsxs)(`div`,{className:`network-list-item`,children:[(0,b.jsx)(M,{tone:x.tone,children:x.label}),(0,b.jsx)(`p`,{children:x.detail})]}),(0,b.jsxs)(`div`,{className:`network-list-item`,children:[(0,b.jsx)(M,{tone:n?.webrtc.tone||`blue`,children:n?.webrtc.status||`检测中`}),(0,b.jsx)(`p`,{children:n?.webrtc.detail||`正在采集 WebRTC 候选。`})]})]})]}),(0,b.jsxs)(`section`,{className:`card network-section`,children:[(0,b.jsx)(`div`,{className:`section-head compact`,children:(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`h3`,{children:`出口与代理`}),(0,b.jsx)(`p`,{children:`反代请求实际经过的公网出口。`})]})}),(0,b.jsxs)(`div`,{className:`network-dual`,children:[(0,b.jsx)(L,{label:`IPv4 出口`,value:H,detail:e?e.publicIpv4.available?`${Z} · ${z(e.publicIpv4.elapsedMs)}`:`${e.publicIpv4.detail} · ${z(e.publicIpv4.elapsedMs)}`:`正在采集公网出口信息。`,tone:P}),(0,b.jsx)(L,{label:`IPv6 出口`,value:e?.publicIpv6.available?e.publicIpv6.ip||`-`:`未检测到`,detail:e?.publicIpv6.detail||`未检测到独立 IPv6 出口。`,tone:V})]}),(0,b.jsx)(`div`,{className:`network-list compact-list`,children:(0,b.jsxs)(`div`,{className:`network-list-item`,children:[(0,b.jsx)(M,{tone:K,children:W}),(0,b.jsx)(`p`,{children:G})]})})]})]})]}),(0,b.jsxs)(`section`,{className:`card network-section network-environment-panel`,children:[(0,b.jsx)(`div`,{className:`section-head compact`,children:(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`h3`,{children:`解析与本地指纹`}),(0,b.jsx)(`p`,{children:`DNS、WebRTC、时区和语言会影响海外应用的风控判断。`})]})}),(0,b.jsxs)(`div`,{className:`network-three-up`,children:[(0,b.jsx)(F,{label:`DNS`,value:T,detail:`${C} · ${e?.dns.source||`尚未获取`}`,tone:S}),(0,b.jsx)(F,{label:`WebRTC`,value:n?.webrtc.status||(a?`检测中`:`-`),detail:n?.webrtc.detail||`正在采集 WebRTC 候选。`,tone:n?.webrtc.tone||`blue`}),(0,b.jsx)(F,{label:`本地环境`,value:n?.timezone||`-`,detail:`${n?.language||`-`} · ${n?.browser||`-`}`,tone:$})]}),e?.dns.servers.length?(0,b.jsx)(`div`,{className:`dns-chip-row`,children:e.dns.servers.map(e=>(0,b.jsx)(`span`,{className:`dns-chip ${w(e)?`orange`:`green`}`,children:e},e))}):null]})]})}export{V as NetworkDetectPage};
@@ -1 +1 @@
1
- import{a as e,r as t,t as n}from"./jsx-runtime-DqpGtLhh.js";import{t as r}from"./circle-check-ZYtn9GqY.js";import{t as i}from"./clock-3-BzDANsVk.js";import{t as a}from"./earth-DFdZaQIi.js";import{t as o}from"./zap-B4_oDbCp.js";import{c as s,l as c,m as l,x as u}from"./profiles-DMOjJORP.js";import{g as d,h as f}from"./index-CdFYy5j6.js";import{t as p}from"./InfoRow-0ULI9iI3.js";import{t as m}from"./StatCard-7TEzqn2i.js";var h=e(t(),1),g=n();function _(e){let t=l(e.apiProfile,e.showEmails),n=e.codexProfile?l(e.codexProfile,e.showEmails):e.codexEmail?e.showEmails?e.codexEmail:s(e.codexEmail):e.codexAccountId?e.showEmails?e.codexAccountId:c(e.codexAccountId):`未应用`;return(0,g.jsxs)(`div`,{className:`usage-summary`,children:[(0,g.jsxs)(`div`,{className:`usage-summary-row`,children:[(0,g.jsx)(`span`,{children:`网关:`}),(0,g.jsx)(`strong`,{children:t})]}),(0,g.jsxs)(`div`,{className:`usage-summary-row`,children:[(0,g.jsx)(`span`,{children:`Codex:`}),(0,g.jsx)(`strong`,{children:n})]})]})}function v(e){let t=e?.quota?.primaryUsedPercent;return typeof t!=`number`||!Number.isFinite(t)?0:Math.max(0,Math.min(100,Math.round(t)))}function y(e,t,n,r){let i=Array.isArray(e?.profiles)?e.profiles:[],a=620+(e?.profile?v(e.profile):42)*7+n*90,o=i.reduce((e,t,n)=>e+v(t)*(n+1),0),s=Math.max(1,r/60);return Array.from({length:12},(e,r)=>{let i=Math.sin((r+1+n)*(.65+s*.08))*(120+s*18),c=Math.cos((r+1)*(1.05+s*.06)+n)*(72+s*10),l=t[r]?t[r].durationMs*(n===0?.24:.12):0,u=a+i+c+l+o%280;return Math.max(220,Math.min(2200,Math.round(u)))})}function b(e,t,n,r){return e.map((i,a)=>{let o=t/(e.length-1)*a,s=n-i/r*(n-16)-8;return`${a===0?`M`:`L`}${o.toFixed(2)} ${s.toFixed(2)}`}).join(` `)}function x(e){let t=y(e.config,e.requestLogs,0,e.windowMinutes),n=y(e.config,e.requestLogs,1,e.windowMinutes).map(e=>Math.max(180,e-260)),r=Math.max(...t,...n,2e3),i=b(t,720,210,r),a=b(n,720,210,r),o=Date.now(),s=Math.max(10,Math.round(e.windowMinutes/6)),c=Array.from({length:6},(e,t)=>new Date(o-(5-t)*s*60*1e3).toLocaleTimeString(`zh-CN`,{hour12:!1,hour:`2-digit`,minute:`2-digit`}));return(0,g.jsxs)(`section`,{className:`trend-card`,"aria-label":`请求耗时趋势`,children:[(0,g.jsxs)(`div`,{className:`section-head compact`,children:[(0,g.jsxs)(`div`,{children:[(0,g.jsx)(`h3`,{children:`请求耗时趋势`}),(0,g.jsx)(`p`,{children:`基于最近调试请求和账号额度状态生成的本地趋势视图。`})]}),(0,g.jsxs)(`select`,{className:`control`,value:e.windowMinutes,onChange:t=>e.onWindow(Number(t.target.value)),children:[(0,g.jsx)(`option`,{value:60,children:`近 1 小时`}),(0,g.jsx)(`option`,{value:180,children:`近 3 小时`}),(0,g.jsx)(`option`,{value:720,children:`近 12 小时`})]})]}),(0,g.jsxs)(`div`,{className:`chart-wrap`,children:[(0,g.jsxs)(`div`,{className:`chart-legend`,children:[(0,g.jsxs)(`span`,{className:`legend-item`,children:[(0,g.jsx)(`span`,{className:`legend-swatch purple`}),`网关响应`]}),(0,g.jsxs)(`span`,{className:`legend-item`,children:[(0,g.jsx)(`span`,{className:`legend-swatch blue`}),`上游响应`]})]}),(0,g.jsxs)(`svg`,{className:`trend-svg`,viewBox:`0 0 720 210`,role:`img`,"aria-label":`请求耗时趋势折线图`,children:[(0,g.jsxs)(`defs`,{children:[(0,g.jsxs)(`linearGradient`,{id:`areaA`,x1:`0`,y1:`0`,x2:`0`,y2:`1`,children:[(0,g.jsx)(`stop`,{offset:`0%`,stopColor:`rgba(99,91,255,0.18)`}),(0,g.jsx)(`stop`,{offset:`100%`,stopColor:`rgba(99,91,255,0.02)`})]}),(0,g.jsxs)(`linearGradient`,{id:`areaB`,x1:`0`,y1:`0`,x2:`0`,y2:`1`,children:[(0,g.jsx)(`stop`,{offset:`0%`,stopColor:`rgba(59,130,246,0.16)`}),(0,g.jsx)(`stop`,{offset:`100%`,stopColor:`rgba(59,130,246,0.02)`})]})]}),[1,2,3,4].map(e=>(0,g.jsx)(`line`,{x1:`0`,y1:e*42,x2:720,y2:e*42,stroke:`#e2e8f0`,strokeWidth:`1`},e)),(0,g.jsx)(`path`,{d:`${i} L 720 210 L 0 210 Z`,fill:`url(#areaA)`,stroke:`none`}),(0,g.jsx)(`path`,{d:`${a} L 720 210 L 0 210 Z`,fill:`url(#areaB)`,stroke:`none`}),(0,g.jsx)(`path`,{d:i,fill:`none`,stroke:`#635bff`,strokeWidth:`2.4`,strokeLinecap:`round`}),(0,g.jsx)(`path`,{d:a,fill:`none`,stroke:`#3b82f6`,strokeWidth:`2.4`,strokeLinecap:`round`})]}),(0,g.jsx)(`div`,{className:`trend-labels`,children:c.map((e,t)=>(0,g.jsx)(`span`,{children:e},`${e}-${t}`))})]})]})}function S(e){return(0,g.jsxs)(`section`,{className:`card service-card endpoint-card`,children:[(0,g.jsx)(`div`,{className:`section-head compact`,children:(0,g.jsxs)(`div`,{children:[(0,g.jsx)(`h3`,{children:`网关信息`}),(0,g.jsx)(`p`,{children:`桌面端与 CLI 共享同一套本地服务。`})]})}),(0,g.jsxs)(`div`,{className:`service-list compact-grid`,children:[(0,g.jsx)(p,{label:`管理页`,value:e.config?.adminUrl||`-`,code:!0}),(0,g.jsx)(p,{label:`Base URL`,value:e.config?.baseUrl||`-`,code:!0}),(0,g.jsx)(p,{label:`默认模型`,value:e.config?.settings.defaultModel||`-`}),(0,g.jsx)(p,{label:`生图模型`,value:`gpt-image-2`}),(0,g.jsx)(p,{label:`兼容接口`,value:e.config?.supportedEndpoints.map(e=>e.path).join(`,`)||`-`}),(0,g.jsx)(p,{label:`令牌预览`,value:e.config?.profile?.accessTokenPreview||`未登录`,code:!0}),(0,g.jsx)(p,{label:`模型来源`,value:e.config?.modelCatalog.source||`-`})]})]})}function C(e){return new Intl.NumberFormat(`zh-CN`).format(Math.round(e||0))}function w(e){let[t,n]=(0,h.useState)(60),s=e.requestLogs.length?e.requestLogs.reduce((e,t)=>e+t.durationMs,0)/e.requestLogs.length:0,c=e.config?.usage?.today,l=c?.failureCount??0;return(0,g.jsxs)(g.Fragment,{children:[(0,g.jsxs)(`section`,{className:`summary-grid desktop-summary-grid overview-summary-grid`,children:[(0,g.jsx)(m,{icon:f,label:`账号总数`,value:String(e.config?.status.profileCount||0),detail:`已保存到本地账号池`,tone:`blue`}),(0,g.jsx)(m,{icon:a,label:`当前账号状态`,value:(0,g.jsx)(_,{apiProfile:e.activeProfile,codexProfile:e.codexProfile,codexEmail:e.codexEmail,codexAccountId:e.codexAccountId,showEmails:e.showEmails}),detail:e.config?.status.loggedIn||e.codexProfile?``:`需要先登录或导入账号`,tone:e.config?.status.loggedIn||e.codexProfile?`green`:`orange`,compact:!0}),(0,g.jsx)(m,{icon:o,label:`今日请求数`,value:C(c?.requestCount??e.requestLogs.length),detail:c?`${C(c.successCount)} 成功 / ${C(c.failureCount)} 失败`:`基于本页最近测试记录`,tone:`blue`}),(0,g.jsx)(m,{icon:i,label:`今日 token`,value:C(c?.totalTokens??0),detail:c?`未返回 token ${C(c.unknownTokenCount)} 次`:`统计最近 ${e.requestLogs.length} 次`,tone:`orange`}),(0,g.jsx)(m,{icon:d,label:`服务状态`,value:e.config?.status.loggedIn?`运行中`:`等待登录`,detail:`网关可转发请求`,tone:e.config?.status.loggedIn?`green`:`orange`}),(0,g.jsx)(m,{icon:r,label:`今日异常`,value:C(l),detail:`平均耗时 ${u(c?c.averageDurationMs:s)}`,tone:l>0?`orange`:`green`})]}),(0,g.jsxs)(`section`,{className:`overview-grid`,children:[(0,g.jsx)(x,{config:e.config,requestLogs:e.requestLogs,windowMinutes:t,onWindow:n}),(0,g.jsx)(S,{config:e.config})]})]})}export{w as OverviewPage};
1
+ import{a as e,r as t,t as n}from"./jsx-runtime-DqpGtLhh.js";import{t as r}from"./circle-check-ZYtn9GqY.js";import{t as i}from"./clock-3-BzDANsVk.js";import{t as a}from"./earth-DFdZaQIi.js";import{t as o}from"./zap-B4_oDbCp.js";import{c as s,l as c,m as l,x as u}from"./profiles-C5SmQvju.js";import{g as d,h as f}from"./index-CCiBaGwU.js";import{t as p}from"./InfoRow-0ULI9iI3.js";import{t as m}from"./StatCard-7TEzqn2i.js";var h=e(t(),1),g=n();function _(e){let t=l(e.apiProfile,e.showEmails),n=e.codexProfile?l(e.codexProfile,e.showEmails):e.codexEmail?e.showEmails?e.codexEmail:s(e.codexEmail):e.codexAccountId?e.showEmails?e.codexAccountId:c(e.codexAccountId):`未应用`;return(0,g.jsxs)(`div`,{className:`usage-summary`,children:[(0,g.jsxs)(`div`,{className:`usage-summary-row`,children:[(0,g.jsx)(`span`,{children:`网关:`}),(0,g.jsx)(`strong`,{children:t})]}),(0,g.jsxs)(`div`,{className:`usage-summary-row`,children:[(0,g.jsx)(`span`,{children:`Codex:`}),(0,g.jsx)(`strong`,{children:n})]})]})}function v(e){let t=e?.quota?.primaryUsedPercent;return typeof t!=`number`||!Number.isFinite(t)?0:Math.max(0,Math.min(100,Math.round(t)))}function y(e,t,n,r){let i=Array.isArray(e?.profiles)?e.profiles:[],a=620+(e?.profile?v(e.profile):42)*7+n*90,o=i.reduce((e,t,n)=>e+v(t)*(n+1),0),s=Math.max(1,r/60);return Array.from({length:12},(e,r)=>{let i=Math.sin((r+1+n)*(.65+s*.08))*(120+s*18),c=Math.cos((r+1)*(1.05+s*.06)+n)*(72+s*10),l=t[r]?t[r].durationMs*(n===0?.24:.12):0,u=a+i+c+l+o%280;return Math.max(220,Math.min(2200,Math.round(u)))})}function b(e,t,n,r){return e.map((i,a)=>{let o=t/(e.length-1)*a,s=n-i/r*(n-16)-8;return`${a===0?`M`:`L`}${o.toFixed(2)} ${s.toFixed(2)}`}).join(` `)}function x(e){let t=y(e.config,e.requestLogs,0,e.windowMinutes),n=y(e.config,e.requestLogs,1,e.windowMinutes).map(e=>Math.max(180,e-260)),r=Math.max(...t,...n,2e3),i=b(t,720,210,r),a=b(n,720,210,r),o=Date.now(),s=Math.max(10,Math.round(e.windowMinutes/6)),c=Array.from({length:6},(e,t)=>new Date(o-(5-t)*s*60*1e3).toLocaleTimeString(`zh-CN`,{hour12:!1,hour:`2-digit`,minute:`2-digit`}));return(0,g.jsxs)(`section`,{className:`trend-card`,"aria-label":`请求耗时趋势`,children:[(0,g.jsxs)(`div`,{className:`section-head compact`,children:[(0,g.jsxs)(`div`,{children:[(0,g.jsx)(`h3`,{children:`请求耗时趋势`}),(0,g.jsx)(`p`,{children:`基于最近调试请求和账号额度状态生成的本地趋势视图。`})]}),(0,g.jsxs)(`select`,{className:`control`,value:e.windowMinutes,onChange:t=>e.onWindow(Number(t.target.value)),children:[(0,g.jsx)(`option`,{value:60,children:`近 1 小时`}),(0,g.jsx)(`option`,{value:180,children:`近 3 小时`}),(0,g.jsx)(`option`,{value:720,children:`近 12 小时`})]})]}),(0,g.jsxs)(`div`,{className:`chart-wrap`,children:[(0,g.jsxs)(`div`,{className:`chart-legend`,children:[(0,g.jsxs)(`span`,{className:`legend-item`,children:[(0,g.jsx)(`span`,{className:`legend-swatch purple`}),`网关响应`]}),(0,g.jsxs)(`span`,{className:`legend-item`,children:[(0,g.jsx)(`span`,{className:`legend-swatch blue`}),`上游响应`]})]}),(0,g.jsxs)(`svg`,{className:`trend-svg`,viewBox:`0 0 720 210`,role:`img`,"aria-label":`请求耗时趋势折线图`,children:[(0,g.jsxs)(`defs`,{children:[(0,g.jsxs)(`linearGradient`,{id:`areaA`,x1:`0`,y1:`0`,x2:`0`,y2:`1`,children:[(0,g.jsx)(`stop`,{offset:`0%`,stopColor:`rgba(99,91,255,0.18)`}),(0,g.jsx)(`stop`,{offset:`100%`,stopColor:`rgba(99,91,255,0.02)`})]}),(0,g.jsxs)(`linearGradient`,{id:`areaB`,x1:`0`,y1:`0`,x2:`0`,y2:`1`,children:[(0,g.jsx)(`stop`,{offset:`0%`,stopColor:`rgba(59,130,246,0.16)`}),(0,g.jsx)(`stop`,{offset:`100%`,stopColor:`rgba(59,130,246,0.02)`})]})]}),[1,2,3,4].map(e=>(0,g.jsx)(`line`,{x1:`0`,y1:e*42,x2:720,y2:e*42,stroke:`#e2e8f0`,strokeWidth:`1`},e)),(0,g.jsx)(`path`,{d:`${i} L 720 210 L 0 210 Z`,fill:`url(#areaA)`,stroke:`none`}),(0,g.jsx)(`path`,{d:`${a} L 720 210 L 0 210 Z`,fill:`url(#areaB)`,stroke:`none`}),(0,g.jsx)(`path`,{d:i,fill:`none`,stroke:`#635bff`,strokeWidth:`2.4`,strokeLinecap:`round`}),(0,g.jsx)(`path`,{d:a,fill:`none`,stroke:`#3b82f6`,strokeWidth:`2.4`,strokeLinecap:`round`})]}),(0,g.jsx)(`div`,{className:`trend-labels`,children:c.map((e,t)=>(0,g.jsx)(`span`,{children:e},`${e}-${t}`))})]})]})}function S(e){return(0,g.jsxs)(`section`,{className:`card service-card endpoint-card`,children:[(0,g.jsx)(`div`,{className:`section-head compact`,children:(0,g.jsxs)(`div`,{children:[(0,g.jsx)(`h3`,{children:`网关信息`}),(0,g.jsx)(`p`,{children:`桌面端与 CLI 共享同一套本地服务。`})]})}),(0,g.jsxs)(`div`,{className:`service-list compact-grid`,children:[(0,g.jsx)(p,{label:`管理页`,value:e.config?.adminUrl||`-`,code:!0}),(0,g.jsx)(p,{label:`Base URL`,value:e.config?.baseUrl||`-`,code:!0}),(0,g.jsx)(p,{label:`默认模型`,value:e.config?.settings.defaultModel||`-`}),(0,g.jsx)(p,{label:`生图模型`,value:`gpt-image-2`}),(0,g.jsx)(p,{label:`兼容接口`,value:e.config?.supportedEndpoints.map(e=>e.path).join(`,`)||`-`}),(0,g.jsx)(p,{label:`令牌预览`,value:e.config?.profile?.accessTokenPreview||`未登录`,code:!0}),(0,g.jsx)(p,{label:`模型来源`,value:e.config?.modelCatalog.source||`-`})]})]})}function C(e){return new Intl.NumberFormat(`zh-CN`).format(Math.round(e||0))}function w(e){let[t,n]=(0,h.useState)(60),s=e.requestLogs.length?e.requestLogs.reduce((e,t)=>e+t.durationMs,0)/e.requestLogs.length:0,c=e.config?.usage?.today,l=c?.failureCount??0;return(0,g.jsxs)(g.Fragment,{children:[(0,g.jsxs)(`section`,{className:`summary-grid desktop-summary-grid overview-summary-grid`,children:[(0,g.jsx)(m,{icon:f,label:`账号总数`,value:String(e.config?.status.profileCount||0),detail:`已保存到本地账号池`,tone:`blue`}),(0,g.jsx)(m,{icon:a,label:`当前账号状态`,value:(0,g.jsx)(_,{apiProfile:e.activeProfile,codexProfile:e.codexProfile,codexEmail:e.codexEmail,codexAccountId:e.codexAccountId,showEmails:e.showEmails}),detail:e.config?.status.loggedIn||e.codexProfile?``:`需要先登录或导入账号`,tone:e.config?.status.loggedIn||e.codexProfile?`green`:`orange`,compact:!0}),(0,g.jsx)(m,{icon:o,label:`今日请求数`,value:C(c?.requestCount??e.requestLogs.length),detail:c?`${C(c.successCount)} 成功 / ${C(c.failureCount)} 失败`:`基于本页最近测试记录`,tone:`blue`}),(0,g.jsx)(m,{icon:i,label:`今日 token`,value:C(c?.totalTokens??0),detail:c?`未返回 token ${C(c.unknownTokenCount)} 次`:`统计最近 ${e.requestLogs.length} 次`,tone:`orange`}),(0,g.jsx)(m,{icon:d,label:`服务状态`,value:e.config?.status.loggedIn?`运行中`:`等待登录`,detail:`网关可转发请求`,tone:e.config?.status.loggedIn?`green`:`orange`}),(0,g.jsx)(m,{icon:r,label:`今日异常`,value:C(l),detail:`平均耗时 ${u(c?c.averageDurationMs:s)}`,tone:l>0?`orange`:`green`})]}),(0,g.jsxs)(`section`,{className:`overview-grid`,children:[(0,g.jsx)(x,{config:e.config,requestLogs:e.requestLogs,windowMinutes:t,onWindow:n}),(0,g.jsx)(S,{config:e.config})]})]})}export{w as OverviewPage};
@@ -1 +1 @@
1
- function e(e){return JSON.stringify(e,null,2)}function t(e){return e?new Intl.DateTimeFormat(`zh-CN`,{month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`}).format(new Date(e)):`-`}function n(e){if(!(typeof e!=`number`||!Number.isFinite(e)||e<=0))return e<0xe8d4a51000?e*1e3:e}function r(e){return e?new Intl.DateTimeFormat(`zh-CN`,{year:`numeric`,month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`,second:`2-digit`}).format(new Date(e)):`-`}function i(e){return Number.isFinite(e)?e<1e3?`${Math.round(e)} ms`:`${(e/1e3).toFixed(e<1e4?2:1)} s`:`-`}function a(e){if(!Number.isFinite(e)||e<=0)return`0 B`;let t=[`B`,`KiB`,`MiB`,`GiB`],n=Math.min(Math.floor(Math.log(e)/Math.log(1024)),t.length-1),r=e/1024**n;return`${r.toFixed(n===0?0:r>=10?1:2)} ${t[n]}`}function o(e){return typeof e!=`number`||!Number.isFinite(e)?0:Math.max(0,Math.min(100,Math.round(e)))}function s(e){return e?.quota?.planType||`unknown`}function c(e){let t=s(e).toLowerCase();return t.includes(`enterprise`)||t.includes(`business`)?60:t.includes(`team`)?50:t.includes(`pro`)||t.includes(`premium`)?40:t.includes(`plus`)?30:t.includes(`free`)?10:0}function l(e){let t=s(e).toLowerCase();return t.includes(`plus`)?`plus`:t.includes(`pro`)?`pro`:t.includes(`enterprise`)||t.includes(`business`)?`enterprise`:t.includes(`team`)?`team`:t.includes(`free`)?`free`:t.includes(`premium`)?`premium`:`unknown`}function u(e){let[t,n]=e.split(`@`);return n?`${t.slice(0,2)}${`*`.repeat(Math.max(3,Math.min(5,t.length)))}@${n}`:d(e)}function d(e){return e?e.length<=10?e:`${e.slice(0,4)}...${e.slice(-4)}`:`未提供`}function f(e,t){return e?e.email?t?e.email:u(e.email):t?e.accountId||e.profileId:d(e.accountId||e.profileId):`未登录`}function p(e){return(e.email||e.accountId||e.profileId||`A`).trim().slice(0,1).toUpperCase()}function m(e){return o(e.quota?.primaryUsedPercent)}function h(e){return 100-m(e)}function g(e){return o(e.quota?.secondaryUsedPercent)}function _(e){return e>=95?`red`:e>=75?`orange`:`blue`}function v(e,t){return e.isActive&&t?{className:`dual`,label:`API + Codex`}:e.isActive?{className:`api-only`,label:`API`}:t?{className:`codex-only`,label:`Codex`}:null}function y(e){return m(e)>=100||g(e)>=100}function b(e){return e.authStatus?.state===`token_invalidated`||e.authStatus?.state===`auth_error`}function x(e){return b(e)||!!(e.expiresAt&&e.expiresAt<=Date.now())}function S(e,t){return e.isActive||t&&e.accountId===t?0:x(e)?2:1}function C(e){let t=e.authStatus;return!t||t.state===`ok`?t?.checkedAt?`正常 · ${r(t.checkedAt)}`:`正常`:`${t.state===`token_invalidated`?`登录失效`:`认证异常`}${t.code||t.httpStatus?` (${t.code||t.httpStatus})`:``} · ${r(t.checkedAt)}`}function w(e){return e.authStatus?.state===`token_invalidated`?{key:`invalid`,label:`登录失效`,tone:`red`}:e.authStatus?.state===`auth_error`?{key:`invalid`,label:`认证异常`,tone:`red`}:e.expiresAt&&e.expiresAt<=Date.now()?{key:`expired`,label:`已过期`,tone:`red`}:y(e)?{key:`exhausted`,label:`额度耗尽`,tone:`orange`}:m(e)>=75||g(e)>=75?{key:`warning`,label:`即将耗尽`,tone:`orange`}:{key:`healthy`,label:`健康`,tone:`green`}}function T(e,t){let n=t===`primary`?e.quota?.primaryWindowMinutes:e.quota?.secondaryWindowMinutes;return n?n<60?`${n} 分钟重置`:n<1440?`${Math.round(n/60)} 小时重置`:`${Math.round(n/60/24)} 天重置`:t===`primary`?`主额度重置`:`周重置`}function E(e,r){let i=r===`primary`?e.quota?.primaryResetAt:e.quota?.secondaryResetAt,a=r===`primary`?e.quota?.primaryResetAfterSeconds:e.quota?.secondaryResetAfterSeconds,o=n(i);return o?t(o):typeof a==`number`&&a>0?t((n(e.quota?.capturedAt)||Date.now())+a*1e3):`-`}function D(e){return e?e.authStatus?.state===`token_invalidated`||e.authStatus?.state===`auth_error`?{ok:!1,detail:`账号认证已失效,请重新登录后再测试图片接口。`}:y(e)?{ok:!0,detail:`账号额度可能不足,实际以接口返回为准。`}:{ok:!0,detail:`当前账号可使用图片生成接口。`}:{ok:!1,detail:`登录后可测试图片接口。`}}export{r as C,a as S,t as T,T as _,D as a,v as b,u as c,m as d,w as f,_ as g,S as h,s as i,d as l,f as m,l as n,b as o,p,c as r,y as s,C as t,h as u,E as v,e as w,i as x,g as y};
1
+ function e(e){return JSON.stringify(e,null,2)}function t(e){return e?new Intl.DateTimeFormat(`zh-CN`,{month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`}).format(new Date(e)):`-`}function n(e){if(!(typeof e!=`number`||!Number.isFinite(e)||e<=0))return e<0xe8d4a51000?e*1e3:e}function r(e){return e?new Intl.DateTimeFormat(`zh-CN`,{year:`numeric`,month:`2-digit`,day:`2-digit`,hour:`2-digit`,minute:`2-digit`,second:`2-digit`}).format(new Date(e)):`-`}function i(e){return Number.isFinite(e)?e<1e3?`${Math.round(e)} ms`:`${(e/1e3).toFixed(e<1e4?2:1)} s`:`-`}function a(e){if(!Number.isFinite(e)||e<=0)return`0 B`;let t=[`B`,`KiB`,`MiB`,`GiB`],n=Math.min(Math.floor(Math.log(e)/Math.log(1024)),t.length-1),r=e/1024**n;return`${r.toFixed(n===0?0:r>=10?1:2)} ${t[n]}`}function o(e){return typeof e!=`number`||!Number.isFinite(e)?0:Math.max(0,Math.min(100,Math.round(e)))}function s(e){return e?.quota?.planType||`unknown`}function c(e){let t=s(e).toLowerCase();return t.includes(`enterprise`)||t.includes(`business`)?60:t.includes(`team`)?50:t.includes(`pro`)||t.includes(`premium`)?40:t.includes(`plus`)?30:t.includes(`free`)?10:0}function l(e){let t=s(e).toLowerCase();return t.includes(`plus`)?`plus`:t.includes(`pro`)?`pro`:t.includes(`enterprise`)||t.includes(`business`)?`enterprise`:t.includes(`team`)?`team`:t.includes(`free`)?`free`:t.includes(`premium`)?`premium`:`unknown`}function u(e){let[t,n]=e.split(`@`);return n?`${t.slice(0,2)}${`*`.repeat(Math.max(3,Math.min(5,t.length)))}@${n}`:d(e)}function d(e){return e?e.length<=10?e:`${e.slice(0,4)}...${e.slice(-4)}`:`未提供`}function f(e,t){return e?e.email?t?e.email:u(e.email):t?e.accountId||e.profileId:d(e.accountId||e.profileId):`未登录`}function p(e){return(e.email||e.accountId||e.profileId||`A`).trim().slice(0,1).toUpperCase()}function m(e){return o(e.quota?.primaryUsedPercent)}function h(e){return 100-m(e)}function g(e){return o(e.quota?.secondaryUsedPercent)}function _(e){return e>=95?`red`:e>=75?`orange`:`blue`}function v(e,t){return e.isActive&&t?{className:`dual`,label:`API + Codex`}:e.isActive?{className:`api-only`,label:`API`}:t?{className:`codex-only`,label:`Codex`}:null}function y(e){return m(e)>=100||g(e)>=100}function b(e){return e.authStatus?.state===`token_invalidated`||e.authStatus?.state===`auth_error`}function x(e){return b(e)||!!(e.expiresAt&&e.expiresAt<=Date.now())}function S(e,t){return e.isActive||t&&e.accountId===t?0:x(e)?2:1}function C(e){let t=e.authStatus;return!t||t.state===`ok`?t?.checkedAt?`正常 · ${r(t.checkedAt)}`:`正常`:`${t.state===`token_invalidated`?`登录失效`:`认证异常`}${t.code||t.httpStatus?` (${t.code||t.httpStatus})`:``} · ${r(t.checkedAt)}`}function w(e){return e.authStatus?.state===`token_invalidated`?{key:`invalid`,label:`登录失效`,tone:`red`}:e.authStatus?.state===`auth_error`?{key:`invalid`,label:`认证异常`,tone:`red`}:e.expiresAt&&e.expiresAt<=Date.now()?{key:`expired`,label:`已过期`,tone:`red`}:e.quota?.capturedAt?y(e)?{key:`exhausted`,label:`额度耗尽`,tone:`orange`}:m(e)>=75||g(e)>=75?{key:`warning`,label:`即将耗尽`,tone:`orange`}:{key:`healthy`,label:`健康`,tone:`green`}:{key:`unknown`,label:`待请求验证`,tone:`blue`}}function T(e,t){let n=t===`primary`?e.quota?.primaryWindowMinutes:e.quota?.secondaryWindowMinutes;return n?n<60?`${n} 分钟重置`:n<1440?`${Math.round(n/60)} 小时重置`:`${Math.round(n/60/24)} 天重置`:t===`primary`?`主额度重置`:`周重置`}function E(e,r){let i=r===`primary`?e.quota?.primaryResetAt:e.quota?.secondaryResetAt,a=r===`primary`?e.quota?.primaryResetAfterSeconds:e.quota?.secondaryResetAfterSeconds,o=n(i);return o?t(o):typeof a==`number`&&a>0?t((n(e.quota?.capturedAt)||Date.now())+a*1e3):`-`}function D(e){return e?e.authStatus?.state===`token_invalidated`||e.authStatus?.state===`auth_error`?{ok:!1,detail:`账号认证已失效,请重新登录后再测试图片接口。`}:y(e)?{ok:!0,detail:`账号额度可能不足,实际以接口返回为准。`}:{ok:!0,detail:`当前账号可使用图片生成接口。`}:{ok:!1,detail:`登录后可测试图片接口。`}}export{r as C,a as S,t as T,T as _,D as a,v as b,u as c,m as d,w as f,_ as g,S as h,s as i,d as l,f as m,l as n,b as o,p,c as r,y as s,C as t,h as u,E as v,e as w,i as x,g as y};
@@ -1,4 +1,4 @@
1
- import{a as e,n as t,r as n,t as r}from"./jsx-runtime-DqpGtLhh.js";import{t as i}from"./earth-DFdZaQIi.js";import{t as a}from"./refresh-cw-CAAH2rqe.js";import{t as o}from"./search-B2hz41D3.js";import{f as s,i as c,m as l,o as u,s as d,w as f}from"./profiles-DMOjJORP.js";import{_ as p,p as m,r as h}from"./index-CdFYy5j6.js";var g=t(`monitor-cog`,[[`path`,{d:`M12 17v4`,key:`1riwvh`}],[`path`,{d:`m14.305 7.53.923-.382`,key:`1mlnsw`}],[`path`,{d:`m15.228 4.852-.923-.383`,key:`82mpwg`}],[`path`,{d:`m16.852 3.228-.383-.924`,key:`ln4sir`}],[`path`,{d:`m16.852 8.772-.383.923`,key:`1dejw0`}],[`path`,{d:`m19.148 3.228.383-.924`,key:`192kgf`}],[`path`,{d:`m19.53 9.696-.382-.924`,key:`fiavlr`}],[`path`,{d:`m20.772 4.852.924-.383`,key:`1j8mgp`}],[`path`,{d:`m20.772 7.148.924.383`,key:`zix9be`}],[`path`,{d:`M22 13v2a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h7`,key:`1tnzv8`}],[`path`,{d:`M8 21h8`,key:`1ev6f3`}],[`circle`,{cx:`18`,cy:`6`,r:`3`,key:`1h7g24`}]]),ee=t(`plug-zap`,[[`path`,{d:`M6.3 20.3a2.4 2.4 0 0 0 3.4 0L12 18l-6-6-2.3 2.3a2.4 2.4 0 0 0 0 3.4Z`,key:`goz73y`}],[`path`,{d:`m2 22 3-3`,key:`19mgm9`}],[`path`,{d:`M7.5 13.5 10 11`,key:`7xgeeb`}],[`path`,{d:`M10.5 16.5 13 14`,key:`10btkg`}],[`path`,{d:`m18 3-4 4h6l-4 4`,key:`16psg9`}]]),te=t(`unplug`,[[`path`,{d:`m19 5 3-3`,key:`yk6iyv`}],[`path`,{d:`m2 22 3-3`,key:`19mgm9`}],[`path`,{d:`M6.3 20.3a2.4 2.4 0 0 0 3.4 0L12 18l-6-6-2.3 2.3a2.4 2.4 0 0 0 0 3.4Z`,key:`goz73y`}],[`path`,{d:`M7.5 13.5 10 11`,key:`7xgeeb`}],[`path`,{d:`M10.5 16.5 13 14`,key:`10btkg`}],[`path`,{d:`m12 6 6 6 2.3-2.3a2.4 2.4 0 0 0 0-3.4l-2.6-2.6a2.4 2.4 0 0 0-3.4 0Z`,key:`1snsnr`}]]),_=e(n(),1),v=r();function y(e){return e===`ai-zero-token`?`ai-zero-token`:`openai`}function b(e){return e===`openai`?`openai`:`AI Zero Token`}function x(e){return e===`openai`?`保留 Codex 原生历史`:`新的 provider 历史`}function ne(e){return e===`openai`?`openai_base_url`:`[model_providers.ai-zero-token]`}function S(e){let t=e.trim();if(!t)throw Error(`请填写 Codex 网关 URL。`);/^[A-Za-z][A-Za-z0-9+.-]*:\/\//.test(t)||(t=`http://${t}`);let n;try{n=new URL(t)}catch{throw Error(`Codex 网关 URL 格式错误,请填写 http(s) 地址或 IP:端口。`)}if(n.protocol!==`http:`&&n.protocol!==`https:`)throw Error(`Codex 网关 URL 只支持 http 或 https。`);n.hash=``,n.search=``;let r=n.pathname.replace(/\/+$/g,``);return!r||r===`/`||r===`/v1`?n.pathname=`/codex/v1`:r.endsWith(`/codex`)?n.pathname=`${r}/v1`:n.pathname=r,n.toString().replace(/\/+$/g,``)}function C(e){try{return S(e)}catch{return e.trim().replace(/\/+$/g,``)}}function w(e){return e?.codexBaseUrl||`http://127.0.0.1:8787/codex/v1`}function re(e){return{defaultModel:e.settings.defaultModel,proxyEnabled:e.settings.networkProxy.enabled,proxyUrl:e.settings.networkProxy.url,proxyNoProxy:e.settings.networkProxy.noProxy||`localhost,127.0.0.1,::1`,autoSwitchEnabled:e.settings.autoSwitch.enabled,autoSwitchExcludedProfileIds:e.settings.autoSwitch.excludedProfileIds||[],quotaSyncConcurrency:String(e.settings.runtime?.quotaSyncConcurrency||3),freeAccountWebGenerationEnabled:!!e.settings.image?.freeAccountWebGenerationEnabled,serverPort:String(e.settings.server.port||8787)}}function ie(e){return[l(e,!0),e.email||``,e.accountId,e.profileId,c(e)].join(` `).toLowerCase()}function T(e){let[t,n]=(0,_.useState)({defaultModel:``,proxyEnabled:!1,proxyUrl:``,proxyNoProxy:`localhost,127.0.0.1,::1`,autoSwitchEnabled:!1,autoSwitchExcludedProfileIds:[],quotaSyncConcurrency:`3`,freeAccountWebGenerationEnabled:!1,serverPort:`8787`}),[r,T]=(0,_.useState)(`local`),[E,D]=(0,_.useState)(`http://127.0.0.1:8787/codex/v1`),[O,ae]=(0,_.useState)(!1),[k,A]=(0,_.useState)(()=>new Set),[j,oe]=(0,_.useState)(``),[M,N]=(0,_.useState)(`openai`),[P,se]=(0,_.useState)(!1),F=k.size>0;(0,_.useEffect)(()=>{!e.config||F||n(re(e.config))},[e.config,F]),(0,_.useEffect)(()=>{if(!e.config||O)return;let t=w(e.config),n=e.config.codex.gatewayProvider?.baseUrl;D(n||t),T(n&&C(n)!==C(t)?`remote`:`local`)},[e.config,O]),(0,_.useEffect)(()=>{!e.config||P||N(y(e.config.codex.gatewayProvider?.providerId))},[e.config,P]);function I(e){n(t=>({...t,...e})),A(t=>{let n=new Set(t);for(let t of Object.keys(e))n.add(t);return n})}function ce(e,n){let r=new Set(t.autoSwitchExcludedProfileIds);n?r.add(e):r.delete(e),I({autoSwitchExcludedProfileIds:Array.from(r)})}function L(t){let n=w(e.config);ae(!0),T(t),t===`local`?D(n):E.trim()||D(n)}function le(){return r===`local`?w(e.config):E}function R(e){se(!0),N(e)}let z=(0,_.useMemo)(()=>new Set(t.autoSwitchExcludedProfileIds),[t.autoSwitchExcludedProfileIds]),B=(0,_.useMemo)(()=>{let t=j.trim().toLowerCase();return(e.config?.profiles||[]).filter(e=>!t||ie(e).includes(t))},[j,e.config?.profiles]),ue=e.config?.profiles.length||0,V=(e.config?.profiles||[]).filter(e=>z.has(e.profileId)).length,de=Math.max(0,ue-V);async function H(n){let r=(...e)=>e.some(e=>k.has(e)),i=Number.parseInt(t.serverPort,10);if(r(`serverPort`)&&(!Number.isInteger(i)||i<1||i>65535)){e.setStatus(`端口必须是 1 到 65535 之间的整数。`);return}let a=Number.parseInt(t.quotaSyncConcurrency,10);if(r(`quotaSyncConcurrency`)&&(!Number.isInteger(a)||a<1||a>32)){e.setStatus(`全局额度刷新并发数必须是 1 到 32 之间的整数。`);return}let o={};r(`defaultModel`)&&(o.defaultModel=t.defaultModel),r(`proxyEnabled`,`proxyUrl`,`proxyNoProxy`)&&(o.networkProxy={enabled:t.proxyEnabled,url:t.proxyUrl,noProxy:t.proxyNoProxy}),r(`autoSwitchEnabled`,`autoSwitchExcludedProfileIds`)&&(o.autoSwitch={},r(`autoSwitchEnabled`)&&(o.autoSwitch.enabled=t.autoSwitchEnabled),r(`autoSwitchExcludedProfileIds`)&&(o.autoSwitch.excludedProfileIds=t.autoSwitchExcludedProfileIds)),r(`quotaSyncConcurrency`)&&(o.runtime={quotaSyncConcurrency:a}),r(`freeAccountWebGenerationEnabled`)&&(o.image={freeAccountWebGenerationEnabled:t.freeAccountWebGenerationEnabled}),r(`serverPort`)&&(o.server={port:i});let s=n?.restart?`restart`:`settings`;e.setBusy(s);try{let t=await m(`/_gateway/admin/settings`,{method:`PUT`,headers:{"Content-Type":`application/json`},body:f(o)});e.setConfig(t),A(new Set),n?.restart?(e.setStatus(`设置已保存,正在重启本地网关...`),await m(`/_gateway/admin/restart`,{method:`POST`}),e.setStatus(`本地网关正在重启,页面会自动恢复。`)):e.setStatus(`设置已保存。`)}catch(t){e.setStatus(h(t))}finally{e.setBusy(null)}}async function fe(){e.setBusy(`proxy`);try{let n=await m(`/_gateway/admin/settings/proxy-test`,{method:`POST`,headers:{"Content-Type":`application/json`},body:f({networkProxy:{enabled:t.proxyEnabled,url:t.proxyUrl,noProxy:t.proxyNoProxy}})});e.setStatus(`代理测试通过: HTTP ${n.status},耗时 ${n.elapsedMs} ms。`)}catch(t){e.setStatus(`代理测试失败: ${h(t)}`)}finally{e.setBusy(null)}}async function pe(){e.setBusy(`models`);try{await m(`/_gateway/models/refresh`,{method:`POST`}),await e.refreshConfig({silent:!0}),e.setStatus(`Codex 模型列表已同步。`)}catch(t){e.setStatus(h(t))}finally{e.setBusy(null)}}async function U(t){if(t.config?.codexRestartSupported&&window.confirm(t.confirmMessage)){e.setStatus(t.restartingStatus);try{await m(`/_gateway/admin/desktop/restart-codex`,{method:`POST`}),e.setStatus(t.restartedStatus)}catch(n){e.setStatus(`${t.failedStatusPrefix}: ${h(n)}`)}return}e.setStatus(t.deferStatus)}async function me(){e.setBusy(`codex-provider`);try{let t=M,n=b(t),r=S(le()),i=e.config?.codex.gatewayProvider?.baseUrl,a=y(e.config?.codex.gatewayProvider?.providerId),o=!!(e.config?.codex.gatewayProvider?.active&&a!==t),s=!!(e.config?.codex.gatewayProvider?.active&&i&&C(i)!==r);if(e.config?.codex.gatewayProvider?.active&&!s&&!o){let r=await m(`/_gateway/admin/codex/remove-provider`,{method:`POST`,headers:{"Content-Type":`application/json`},body:f({providerId:t})});r.config&&e.setConfig(r.config),r.codexProvider.removed?await U({config:r.config??e.config,confirmMessage:`Codex ${n} 接管已解除,是否现在重启 Codex 客户端?\n\nCodex 通常在启动时读取本机 config.toml,重启后会回到原本的 Codex 配置。`,deferStatus:`已解除 ${n} 接管。重启 Codex 后会回到原本的 Codex 配置。`,restartingStatus:`正在重启 Codex 客户端...`,restartedStatus:`已解除 ${n} 接管,并已重启 Codex 客户端。`,failedStatusPrefix:`已解除 ${n} 接管,但重启 Codex 失败`}):e.setStatus(`未发现当前受管的 Codex provider 配置。`);return}let c=!!e.config?.codex.gatewayProvider?.active,l=await m(`/_gateway/admin/codex/configure-provider`,{method:`POST`,headers:{"Content-Type":`application/json`},body:f({baseUrl:r,providerId:t})});l.config&&e.setConfig(l.config);let u=l.codexProvider.historyMigration?.migratedCount||0,d=u>0?`,已迁移 ${u} 条历史记录`:``;await U({config:l.config??e.config,confirmMessage:t===`openai`?`Codex 接管将使用 openai 历史记录模式,是否现在重启 Codex 客户端?
1
+ import{a as e,n as t,r as n,t as r}from"./jsx-runtime-DqpGtLhh.js";import{t as i}from"./earth-DFdZaQIi.js";import{t as a}from"./refresh-cw-CAAH2rqe.js";import{t as o}from"./search-B2hz41D3.js";import{f as s,i as c,m as l,o as u,s as d,w as f}from"./profiles-C5SmQvju.js";import{_ as p,p as m,r as h}from"./index-CCiBaGwU.js";var g=t(`monitor-cog`,[[`path`,{d:`M12 17v4`,key:`1riwvh`}],[`path`,{d:`m14.305 7.53.923-.382`,key:`1mlnsw`}],[`path`,{d:`m15.228 4.852-.923-.383`,key:`82mpwg`}],[`path`,{d:`m16.852 3.228-.383-.924`,key:`ln4sir`}],[`path`,{d:`m16.852 8.772-.383.923`,key:`1dejw0`}],[`path`,{d:`m19.148 3.228.383-.924`,key:`192kgf`}],[`path`,{d:`m19.53 9.696-.382-.924`,key:`fiavlr`}],[`path`,{d:`m20.772 4.852.924-.383`,key:`1j8mgp`}],[`path`,{d:`m20.772 7.148.924.383`,key:`zix9be`}],[`path`,{d:`M22 13v2a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h7`,key:`1tnzv8`}],[`path`,{d:`M8 21h8`,key:`1ev6f3`}],[`circle`,{cx:`18`,cy:`6`,r:`3`,key:`1h7g24`}]]),ee=t(`plug-zap`,[[`path`,{d:`M6.3 20.3a2.4 2.4 0 0 0 3.4 0L12 18l-6-6-2.3 2.3a2.4 2.4 0 0 0 0 3.4Z`,key:`goz73y`}],[`path`,{d:`m2 22 3-3`,key:`19mgm9`}],[`path`,{d:`M7.5 13.5 10 11`,key:`7xgeeb`}],[`path`,{d:`M10.5 16.5 13 14`,key:`10btkg`}],[`path`,{d:`m18 3-4 4h6l-4 4`,key:`16psg9`}]]),te=t(`unplug`,[[`path`,{d:`m19 5 3-3`,key:`yk6iyv`}],[`path`,{d:`m2 22 3-3`,key:`19mgm9`}],[`path`,{d:`M6.3 20.3a2.4 2.4 0 0 0 3.4 0L12 18l-6-6-2.3 2.3a2.4 2.4 0 0 0 0 3.4Z`,key:`goz73y`}],[`path`,{d:`M7.5 13.5 10 11`,key:`7xgeeb`}],[`path`,{d:`M10.5 16.5 13 14`,key:`10btkg`}],[`path`,{d:`m12 6 6 6 2.3-2.3a2.4 2.4 0 0 0 0-3.4l-2.6-2.6a2.4 2.4 0 0 0-3.4 0Z`,key:`1snsnr`}]]),_=e(n(),1),v=r();function y(e){return e===`ai-zero-token`?`ai-zero-token`:`openai`}function b(e){return e===`openai`?`openai`:`AI Zero Token`}function x(e){return e===`openai`?`保留 Codex 原生历史`:`新的 provider 历史`}function ne(e){return e===`openai`?`openai_base_url`:`[model_providers.ai-zero-token]`}function S(e){let t=e.trim();if(!t)throw Error(`请填写 Codex 网关 URL。`);/^[A-Za-z][A-Za-z0-9+.-]*:\/\//.test(t)||(t=`http://${t}`);let n;try{n=new URL(t)}catch{throw Error(`Codex 网关 URL 格式错误,请填写 http(s) 地址或 IP:端口。`)}if(n.protocol!==`http:`&&n.protocol!==`https:`)throw Error(`Codex 网关 URL 只支持 http 或 https。`);n.hash=``,n.search=``;let r=n.pathname.replace(/\/+$/g,``);return!r||r===`/`||r===`/v1`?n.pathname=`/codex/v1`:r.endsWith(`/codex`)?n.pathname=`${r}/v1`:n.pathname=r,n.toString().replace(/\/+$/g,``)}function C(e){try{return S(e)}catch{return e.trim().replace(/\/+$/g,``)}}function w(e){return e?.codexBaseUrl||`http://127.0.0.1:8787/codex/v1`}function re(e){return{defaultModel:e.settings.defaultModel,proxyEnabled:e.settings.networkProxy.enabled,proxyUrl:e.settings.networkProxy.url,proxyNoProxy:e.settings.networkProxy.noProxy||`localhost,127.0.0.1,::1`,autoSwitchEnabled:e.settings.autoSwitch.enabled,autoSwitchExcludedProfileIds:e.settings.autoSwitch.excludedProfileIds||[],quotaSyncConcurrency:String(e.settings.runtime?.quotaSyncConcurrency||3),freeAccountWebGenerationEnabled:!!e.settings.image?.freeAccountWebGenerationEnabled,serverPort:String(e.settings.server.port||8787)}}function ie(e){return[l(e,!0),e.email||``,e.accountId,e.profileId,c(e)].join(` `).toLowerCase()}function T(e){let[t,n]=(0,_.useState)({defaultModel:``,proxyEnabled:!1,proxyUrl:``,proxyNoProxy:`localhost,127.0.0.1,::1`,autoSwitchEnabled:!1,autoSwitchExcludedProfileIds:[],quotaSyncConcurrency:`3`,freeAccountWebGenerationEnabled:!1,serverPort:`8787`}),[r,T]=(0,_.useState)(`local`),[E,D]=(0,_.useState)(`http://127.0.0.1:8787/codex/v1`),[O,ae]=(0,_.useState)(!1),[k,A]=(0,_.useState)(()=>new Set),[j,oe]=(0,_.useState)(``),[M,N]=(0,_.useState)(`openai`),[P,se]=(0,_.useState)(!1),F=k.size>0;(0,_.useEffect)(()=>{!e.config||F||n(re(e.config))},[e.config,F]),(0,_.useEffect)(()=>{if(!e.config||O)return;let t=w(e.config),n=e.config.codex.gatewayProvider?.baseUrl;D(n||t),T(n&&C(n)!==C(t)?`remote`:`local`)},[e.config,O]),(0,_.useEffect)(()=>{!e.config||P||N(y(e.config.codex.gatewayProvider?.providerId))},[e.config,P]);function I(e){n(t=>({...t,...e})),A(t=>{let n=new Set(t);for(let t of Object.keys(e))n.add(t);return n})}function ce(e,n){let r=new Set(t.autoSwitchExcludedProfileIds);n?r.add(e):r.delete(e),I({autoSwitchExcludedProfileIds:Array.from(r)})}function L(t){let n=w(e.config);ae(!0),T(t),t===`local`?D(n):E.trim()||D(n)}function le(){return r===`local`?w(e.config):E}function R(e){se(!0),N(e)}let z=(0,_.useMemo)(()=>new Set(t.autoSwitchExcludedProfileIds),[t.autoSwitchExcludedProfileIds]),B=(0,_.useMemo)(()=>{let t=j.trim().toLowerCase();return(e.config?.profiles||[]).filter(e=>!t||ie(e).includes(t))},[j,e.config?.profiles]),ue=e.config?.profiles.length||0,V=(e.config?.profiles||[]).filter(e=>z.has(e.profileId)).length,de=Math.max(0,ue-V);async function H(n){let r=(...e)=>e.some(e=>k.has(e)),i=Number.parseInt(t.serverPort,10);if(r(`serverPort`)&&(!Number.isInteger(i)||i<1||i>65535)){e.setStatus(`端口必须是 1 到 65535 之间的整数。`);return}let a=Number.parseInt(t.quotaSyncConcurrency,10);if(r(`quotaSyncConcurrency`)&&(!Number.isInteger(a)||a<1||a>32)){e.setStatus(`全局额度刷新并发数必须是 1 到 32 之间的整数。`);return}let o={};r(`defaultModel`)&&(o.defaultModel=t.defaultModel),r(`proxyEnabled`,`proxyUrl`,`proxyNoProxy`)&&(o.networkProxy={enabled:t.proxyEnabled,url:t.proxyUrl,noProxy:t.proxyNoProxy}),r(`autoSwitchEnabled`,`autoSwitchExcludedProfileIds`)&&(o.autoSwitch={},r(`autoSwitchEnabled`)&&(o.autoSwitch.enabled=t.autoSwitchEnabled),r(`autoSwitchExcludedProfileIds`)&&(o.autoSwitch.excludedProfileIds=t.autoSwitchExcludedProfileIds)),r(`quotaSyncConcurrency`)&&(o.runtime={quotaSyncConcurrency:a}),r(`freeAccountWebGenerationEnabled`)&&(o.image={freeAccountWebGenerationEnabled:t.freeAccountWebGenerationEnabled}),r(`serverPort`)&&(o.server={port:i});let s=n?.restart?`restart`:`settings`;e.setBusy(s);try{let t=await m(`/_gateway/admin/settings`,{method:`PUT`,headers:{"Content-Type":`application/json`},body:f(o)});e.setConfig(t),A(new Set),n?.restart?(e.setStatus(`设置已保存,正在重启本地网关...`),await m(`/_gateway/admin/restart`,{method:`POST`}),e.setStatus(`本地网关正在重启,页面会自动恢复。`)):e.setStatus(`设置已保存。`)}catch(t){e.setStatus(h(t))}finally{e.setBusy(null)}}async function fe(){e.setBusy(`proxy`);try{let n=await m(`/_gateway/admin/settings/proxy-test`,{method:`POST`,headers:{"Content-Type":`application/json`},body:f({networkProxy:{enabled:t.proxyEnabled,url:t.proxyUrl,noProxy:t.proxyNoProxy}})});e.setStatus(`代理测试通过: HTTP ${n.status},耗时 ${n.elapsedMs} ms。`)}catch(t){e.setStatus(`代理测试失败: ${h(t)}`)}finally{e.setBusy(null)}}async function pe(){e.setBusy(`models`);try{let t=await m(`/_gateway/models/refresh`,{method:`POST`});await e.refreshConfig({silent:!0});let n=t.catalog?.modelCount??0;e.setStatus(n>0?`Codex 模型列表已从网络同步,共 ${n} 个。`:`Codex 模型列表已从网络同步。`)}catch(t){e.setStatus(h(t))}finally{e.setBusy(null)}}async function U(t){if(t.config?.codexRestartSupported&&window.confirm(t.confirmMessage)){e.setStatus(t.restartingStatus);try{await m(`/_gateway/admin/desktop/restart-codex`,{method:`POST`}),e.setStatus(t.restartedStatus)}catch(n){e.setStatus(`${t.failedStatusPrefix}: ${h(n)}`)}return}e.setStatus(t.deferStatus)}async function me(){e.setBusy(`codex-provider`);try{let t=M,n=b(t),r=S(le()),i=e.config?.codex.gatewayProvider?.baseUrl,a=y(e.config?.codex.gatewayProvider?.providerId),o=!!(e.config?.codex.gatewayProvider?.active&&a!==t),s=!!(e.config?.codex.gatewayProvider?.active&&i&&C(i)!==r);if(e.config?.codex.gatewayProvider?.active&&!s&&!o){let r=await m(`/_gateway/admin/codex/remove-provider`,{method:`POST`,headers:{"Content-Type":`application/json`},body:f({providerId:t})});r.config&&e.setConfig(r.config),r.codexProvider.removed?await U({config:r.config??e.config,confirmMessage:`Codex ${n} 接管已解除,是否现在重启 Codex 客户端?\n\nCodex 通常在启动时读取本机 config.toml,重启后会回到原本的 Codex 配置。`,deferStatus:`已解除 ${n} 接管。重启 Codex 后会回到原本的 Codex 配置。`,restartingStatus:`正在重启 Codex 客户端...`,restartedStatus:`已解除 ${n} 接管,并已重启 Codex 客户端。`,failedStatusPrefix:`已解除 ${n} 接管,但重启 Codex 失败`}):e.setStatus(`未发现当前受管的 Codex provider 配置。`);return}let c=!!e.config?.codex.gatewayProvider?.active,l=await m(`/_gateway/admin/codex/configure-provider`,{method:`POST`,headers:{"Content-Type":`application/json`},body:f({baseUrl:r,providerId:t})});l.config&&e.setConfig(l.config);let u=l.codexProvider.historyMigration?.migratedCount||0,d=u>0?`,已迁移 ${u} 条历史记录`:``;await U({config:l.config??e.config,confirmMessage:t===`openai`?`Codex 接管将使用 openai 历史记录模式,是否现在重启 Codex 客户端?
2
2
 
3
3
  重启后请求仍会走 AI Zero Token 网关,历史记录会继续归在 Codex 原生 openai provider 下。`:`Codex 接管将切换到 AI Zero Token 新 provider,是否现在重启 Codex 客户端?
4
4
 
@@ -1,3 +1,3 @@
1
- import{a as e,n as t,r as n,t as r}from"./jsx-runtime-DqpGtLhh.js";import{t as i}from"./upload-CwXb7Q1b.js";import{t as a}from"./zap-B4_oDbCp.js";import{S as o,m as s,w as c,x as l}from"./profiles-DMOjJORP.js";import{_ as u,a as d,b as f,c as p,i as m,l as h,n as g,o as _,p as v,r as y,s as b,t as x,u as S}from"./index-CdFYy5j6.js";var C=t(`rotate-ccw`,[[`path`,{d:`M3 12a9 9 0 1 0 9-9 9.75 9.75 0 0 0-6.74 2.74L3 8`,key:`1357e3`}],[`path`,{d:`M3 3v5h5`,key:`1xhq8a`}]]),w=e(n(),1),T=r();function E(e){let t=e.endpoint.startsWith(`/v1/images/`);function n(t){let n=t.currentTarget.files?.[0];t.currentTarget.value=``,n&&e.onImageUpload(n,e.imageUploadMode).catch(()=>void 0)}return(0,T.jsxs)(`section`,{className:`card tester-card`,id:`tester`,children:[(0,T.jsxs)(`div`,{className:`section-head compact`,children:[(0,T.jsxs)(`div`,{children:[(0,T.jsx)(`h2`,{children:`快速测试`}),(0,T.jsx)(`p`,{children:`页面直接调用当前网关暴露的 OpenAI 风格接口。`})]}),(0,T.jsx)(`span`,{className:`badge brand`,children:e.busy===`test`?`请求中`:`准备就绪`})]}),(0,T.jsx)(`div`,{className:`tester-tabs`,children:e.endpoints.map(t=>(0,T.jsx)(`button`,{className:`tab-btn ${e.endpoint===t.path?`is-active`:``}`,type:`button`,onClick:()=>e.onEndpoint(t.path),children:S[t.path]||t.path},t.path))}),(0,T.jsxs)(`div`,{className:`tester-workbench`,children:[(0,T.jsxs)(`div`,{className:`tester-pane tester-request-pane`,children:[(0,T.jsxs)(`label`,{className:`field`,children:[(0,T.jsx)(`span`,{children:`接口`}),(0,T.jsx)(`select`,{className:`control`,value:e.endpoint,onChange:t=>e.onEndpoint(t.target.value),children:e.endpoints.map(e=>(0,T.jsxs)(`option`,{value:e.path,children:[e.method,` `,e.path]},e.path))})]}),(0,T.jsxs)(`div`,{className:`tester-copy-row tester-copy-row-top`,children:[(0,T.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:e.onCopyRequest,children:[(0,T.jsx)(f,{size:16}),`复制请求`]}),(0,T.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:e.onCopyResponse,children:[(0,T.jsx)(f,{size:16}),`复制响应`]}),(0,T.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:e.onCopyTiming,children:[(0,T.jsx)(f,{size:16}),`复制日志`]}),(0,T.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:e.onResetExample,children:[(0,T.jsx)(C,{size:16}),`重置示例`]})]}),(0,T.jsxs)(`label`,{className:`field tester-body-field`,children:[(0,T.jsx)(`span`,{children:`请求体 JSON`}),(0,T.jsx)(`textarea`,{className:`textarea tester-textarea`,value:e.requestBody,onChange:t=>e.onRequestBody(t.target.value),disabled:e.activeEndpoint.method===`GET`,spellCheck:!1})]}),e.endpoint===`/v1/images/edits`&&(0,T.jsxs)(`div`,{className:`edit-upload-row`,children:[(0,T.jsxs)(`div`,{className:`edit-upload-mode`,role:`group`,"aria-label":`图片写入方式`,children:[(0,T.jsx)(`span`,{children:`写入方式`}),(0,T.jsxs)(`div`,{className:`edit-upload-toggle`,children:[(0,T.jsx)(`button`,{className:`tab-btn ${e.imageUploadMode===`base64`?`is-active`:``}`,type:`button`,onClick:()=>e.onImageUploadMode(`base64`),children:`Base64`}),(0,T.jsx)(`button`,{className:`tab-btn ${e.imageUploadMode===`image-bed`?`is-active`:``}`,type:`button`,onClick:()=>e.onImageUploadMode(`image-bed`),children:`图床`})]})]}),(0,T.jsxs)(`label`,{className:`btn-secondary upload-btn`,title:e.imageUploadMode===`base64`?`上传图片并写入 base64 data URL`:`上传图片到图床并写入公网链接`,children:[e.imageUploadMode===`image-bed`&&e.busy===`image-bed-upload`?(0,T.jsx)(u,{className:`spin`,size:16}):(0,T.jsx)(i,{size:16}),e.imageUploadMode===`base64`?`上传并写入 Base64`:`上传并写入图床链接`,(0,T.jsx)(`input`,{type:`file`,accept:`image/*`,onChange:n})]}),(0,T.jsxs)(`span`,{children:[`目标字段:images[0].image_url · `,e.imageUploadMode===`base64`?`直接写入 data URL`:`先上传图床再写入公网链接`]})]}),(0,T.jsx)(`p`,{className:`hint`,children:t?e.capability.detail:e.activeEndpoint.description||`GET /v1/models 无需请求体。`}),(0,T.jsxs)(`div`,{className:`tester-actions-bar`,children:[(0,T.jsx)(`div`,{className:`tester-actions-group`,children:(0,T.jsx)(`div`,{className:`example-row`,children:p.map(t=>(0,T.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:()=>e.onEndpoint(t),disabled:!e.endpoints.some(e=>e.path===t),children:[`示例 `,S[t]||t]},t))})}),(0,T.jsxs)(`button`,{className:`btn-primary`,type:`button`,onClick:e.onRun,disabled:e.busy===`test`||t&&!e.config?.profile,children:[e.busy===`test`?(0,T.jsx)(u,{className:`spin`,size:16}):(0,T.jsx)(a,{size:16}),`发送请求`]})]})]}),(0,T.jsxs)(`div`,{className:`tester-pane tester-response-pane`,children:[(0,T.jsxs)(`div`,{className:`tester-result-head`,children:[(0,T.jsxs)(`div`,{className:`tester-result-tabs`,children:[(0,T.jsx)(`button`,{className:`tab-btn ${e.resultTab===`response`?`is-active`:``}`,type:`button`,onClick:()=>e.onResultTab(`response`),children:`响应 JSON`}),(0,T.jsx)(`button`,{className:`tab-btn ${e.resultTab===`timing`?`is-active`:``}`,type:`button`,onClick:()=>e.onResultTab(`timing`),children:`耗时日志`}),(0,T.jsx)(`button`,{className:`tab-btn ${e.resultTab===`preview`?`is-active`:``}`,type:`button`,onClick:()=>e.onResultTab(`preview`),children:`图片预览`})]}),(0,T.jsx)(`p`,{className:`status-inline`,children:e.status})]}),e.resultTab===`response`&&(0,T.jsx)(`pre`,{className:`pre`,children:e.responseBody}),e.resultTab===`timing`&&(0,T.jsx)(`pre`,{className:`pre`,children:e.timingBody}),e.resultTab===`preview`&&(0,T.jsx)(`div`,{className:`preview-panel`,children:e.previewImages.length===0?(0,T.jsx)(`div`,{className:`preview-empty`,children:`图片结果会显示在这里。点击缩略图可查看大图。`}):(0,T.jsx)(`div`,{className:`preview-grid`,children:e.previewImages.map(t=>(0,T.jsxs)(`figure`,{className:`preview-card`,children:[(0,T.jsx)(`button`,{type:`button`,onClick:()=>e.onPreview({src:t.src,meta:t.meta,filename:t.filename}),children:(0,T.jsx)(`img`,{src:t.src,alt:t.meta})}),(0,T.jsx)(`figcaption`,{children:t.meta}),(0,T.jsx)(`div`,{className:`preview-actions`,children:(0,T.jsx)(`a`,{href:t.src,download:t.filename,children:`下载`})})]},t.filename))})})]})]})]})}function D(e){let[t,n]=(0,w.useState)(`/v1/models`),[r,i]=(0,w.useState)(``),[a,u]=(0,w.useState)(`等待请求...`),[f,S]=(0,w.useState)(`等待请求...`),[C,D]=(0,w.useState)(`response`),[O,k]=(0,w.useState)([]),[A,j]=(0,w.useState)(`base64`),M=(0,w.useMemo)(()=>[...e.config?.supportedEndpoints||[]].filter(e=>p.includes(e.path)).sort(h),[e.config?.supportedEndpoints]),N=(0,w.useMemo)(()=>M.find(e=>e.path===t)||M[0]||{method:`GET`,path:`/v1/models`,description:``},[t,M]);(0,w.useEffect)(()=>{if(!e.config)return;let r=p.find(t=>e.config?.supportedEndpoints.some(e=>e.path===t)),a=e.config.supportedEndpoints.some(e=>e.path===t)?t:r||`/v1/models`;n(a),i(x(a,e.config.settings.defaultModel))},[e.config]);function P(t){n(t),i(x(t,e.config?.settings.defaultModel||`gpt-5.4`)),k([])}function F(){P(t)}function I(){g(r||x(t,e.config?.settings.defaultModel||`gpt-5.4`)).then(t=>e.setStatus(t?`请求体已复制。`:`请求体复制失败。`)).catch(()=>e.setStatus(`请求体复制失败。`))}function L(){g(a).then(t=>e.setStatus(t?`响应内容已复制。`:`响应内容复制失败。`)).catch(()=>e.setStatus(`响应内容复制失败。`))}function R(){g(f).then(t=>e.setStatus(t?`耗时日志已复制。`:`耗时日志复制失败。`)).catch(()=>e.setStatus(`耗时日志复制失败。`))}async function z(){let t=N,n=performance.now(),i=[];e.setBusy(`test`),D(`response`),u(`请求发送中...`),S(`请求发送中...`),k([]);try{let a=null,o={method:t.method,headers:{}};if(t.method!==`GET`){let e=performance.now();a=r.trim()?JSON.parse(r):{},i.push(`解析请求体: ${l(performance.now()-e)}`),o.headers[`Content-Type`]=`application/json`,o.body=c(a)}let d=performance.now(),f=await fetch(t.path,o);i.push(`等待响应头: ${l(performance.now()-d)}`);let p=performance.now(),h=await f.text();i.push(`读取响应体: ${l(performance.now()-p)}`);let g=performance.now(),_=h;try{_=h?JSON.parse(h):null}catch{_=h}i.push(`解析响应体: ${l(performance.now()-g)}`);let v=m(_);k(v),v.length>0&&D(`preview`),u(typeof _==`string`?_:c(b(_))),S([`${t.method} ${t.path}`,`HTTP 状态: ${f.status} ${f.statusText}`,...i].join(`
1
+ import{a as e,n as t,r as n,t as r}from"./jsx-runtime-DqpGtLhh.js";import{t as i}from"./upload-CwXb7Q1b.js";import{t as a}from"./zap-B4_oDbCp.js";import{S as o,m as s,w as c,x as l}from"./profiles-C5SmQvju.js";import{_ as u,a as d,b as f,c as p,i as m,l as h,n as g,o as _,p as v,r as y,s as b,t as x,u as S}from"./index-CCiBaGwU.js";var C=t(`rotate-ccw`,[[`path`,{d:`M3 12a9 9 0 1 0 9-9 9.75 9.75 0 0 0-6.74 2.74L3 8`,key:`1357e3`}],[`path`,{d:`M3 3v5h5`,key:`1xhq8a`}]]),w=e(n(),1),T=r();function E(e){let t=e.endpoint.startsWith(`/v1/images/`);function n(t){let n=t.currentTarget.files?.[0];t.currentTarget.value=``,n&&e.onImageUpload(n,e.imageUploadMode).catch(()=>void 0)}return(0,T.jsxs)(`section`,{className:`card tester-card`,id:`tester`,children:[(0,T.jsxs)(`div`,{className:`section-head compact`,children:[(0,T.jsxs)(`div`,{children:[(0,T.jsx)(`h2`,{children:`快速测试`}),(0,T.jsx)(`p`,{children:`页面直接调用当前网关暴露的 OpenAI 风格接口。`})]}),(0,T.jsx)(`span`,{className:`badge brand`,children:e.busy===`test`?`请求中`:`准备就绪`})]}),(0,T.jsx)(`div`,{className:`tester-tabs`,children:e.endpoints.map(t=>(0,T.jsx)(`button`,{className:`tab-btn ${e.endpoint===t.path?`is-active`:``}`,type:`button`,onClick:()=>e.onEndpoint(t.path),children:S[t.path]||t.path},t.path))}),(0,T.jsxs)(`div`,{className:`tester-workbench`,children:[(0,T.jsxs)(`div`,{className:`tester-pane tester-request-pane`,children:[(0,T.jsxs)(`label`,{className:`field`,children:[(0,T.jsx)(`span`,{children:`接口`}),(0,T.jsx)(`select`,{className:`control`,value:e.endpoint,onChange:t=>e.onEndpoint(t.target.value),children:e.endpoints.map(e=>(0,T.jsxs)(`option`,{value:e.path,children:[e.method,` `,e.path]},e.path))})]}),(0,T.jsxs)(`div`,{className:`tester-copy-row tester-copy-row-top`,children:[(0,T.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:e.onCopyRequest,children:[(0,T.jsx)(f,{size:16}),`复制请求`]}),(0,T.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:e.onCopyResponse,children:[(0,T.jsx)(f,{size:16}),`复制响应`]}),(0,T.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:e.onCopyTiming,children:[(0,T.jsx)(f,{size:16}),`复制日志`]}),(0,T.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:e.onResetExample,children:[(0,T.jsx)(C,{size:16}),`重置示例`]})]}),(0,T.jsxs)(`label`,{className:`field tester-body-field`,children:[(0,T.jsx)(`span`,{children:`请求体 JSON`}),(0,T.jsx)(`textarea`,{className:`textarea tester-textarea`,value:e.requestBody,onChange:t=>e.onRequestBody(t.target.value),disabled:e.activeEndpoint.method===`GET`,spellCheck:!1})]}),e.endpoint===`/v1/images/edits`&&(0,T.jsxs)(`div`,{className:`edit-upload-row`,children:[(0,T.jsxs)(`div`,{className:`edit-upload-mode`,role:`group`,"aria-label":`图片写入方式`,children:[(0,T.jsx)(`span`,{children:`写入方式`}),(0,T.jsxs)(`div`,{className:`edit-upload-toggle`,children:[(0,T.jsx)(`button`,{className:`tab-btn ${e.imageUploadMode===`base64`?`is-active`:``}`,type:`button`,onClick:()=>e.onImageUploadMode(`base64`),children:`Base64`}),(0,T.jsx)(`button`,{className:`tab-btn ${e.imageUploadMode===`image-bed`?`is-active`:``}`,type:`button`,onClick:()=>e.onImageUploadMode(`image-bed`),children:`图床`})]})]}),(0,T.jsxs)(`label`,{className:`btn-secondary upload-btn`,title:e.imageUploadMode===`base64`?`上传图片并写入 base64 data URL`:`上传图片到图床并写入公网链接`,children:[e.imageUploadMode===`image-bed`&&e.busy===`image-bed-upload`?(0,T.jsx)(u,{className:`spin`,size:16}):(0,T.jsx)(i,{size:16}),e.imageUploadMode===`base64`?`上传并写入 Base64`:`上传并写入图床链接`,(0,T.jsx)(`input`,{type:`file`,accept:`image/*`,onChange:n})]}),(0,T.jsxs)(`span`,{children:[`目标字段:images[0].image_url · `,e.imageUploadMode===`base64`?`直接写入 data URL`:`先上传图床再写入公网链接`]})]}),(0,T.jsx)(`p`,{className:`hint`,children:t?e.capability.detail:e.activeEndpoint.description||`GET /v1/models 无需请求体。`}),(0,T.jsxs)(`div`,{className:`tester-actions-bar`,children:[(0,T.jsx)(`div`,{className:`tester-actions-group`,children:(0,T.jsx)(`div`,{className:`example-row`,children:p.map(t=>(0,T.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:()=>e.onEndpoint(t),disabled:!e.endpoints.some(e=>e.path===t),children:[`示例 `,S[t]||t]},t))})}),(0,T.jsxs)(`button`,{className:`btn-primary`,type:`button`,onClick:e.onRun,disabled:e.busy===`test`||t&&!e.config?.profile,children:[e.busy===`test`?(0,T.jsx)(u,{className:`spin`,size:16}):(0,T.jsx)(a,{size:16}),`发送请求`]})]})]}),(0,T.jsxs)(`div`,{className:`tester-pane tester-response-pane`,children:[(0,T.jsxs)(`div`,{className:`tester-result-head`,children:[(0,T.jsxs)(`div`,{className:`tester-result-tabs`,children:[(0,T.jsx)(`button`,{className:`tab-btn ${e.resultTab===`response`?`is-active`:``}`,type:`button`,onClick:()=>e.onResultTab(`response`),children:`响应 JSON`}),(0,T.jsx)(`button`,{className:`tab-btn ${e.resultTab===`timing`?`is-active`:``}`,type:`button`,onClick:()=>e.onResultTab(`timing`),children:`耗时日志`}),(0,T.jsx)(`button`,{className:`tab-btn ${e.resultTab===`preview`?`is-active`:``}`,type:`button`,onClick:()=>e.onResultTab(`preview`),children:`图片预览`})]}),(0,T.jsx)(`p`,{className:`status-inline`,children:e.status})]}),e.resultTab===`response`&&(0,T.jsx)(`pre`,{className:`pre`,children:e.responseBody}),e.resultTab===`timing`&&(0,T.jsx)(`pre`,{className:`pre`,children:e.timingBody}),e.resultTab===`preview`&&(0,T.jsx)(`div`,{className:`preview-panel`,children:e.previewImages.length===0?(0,T.jsx)(`div`,{className:`preview-empty`,children:`图片结果会显示在这里。点击缩略图可查看大图。`}):(0,T.jsx)(`div`,{className:`preview-grid`,children:e.previewImages.map(t=>(0,T.jsxs)(`figure`,{className:`preview-card`,children:[(0,T.jsx)(`button`,{type:`button`,onClick:()=>e.onPreview({src:t.src,meta:t.meta,filename:t.filename}),children:(0,T.jsx)(`img`,{src:t.src,alt:t.meta})}),(0,T.jsx)(`figcaption`,{children:t.meta}),(0,T.jsx)(`div`,{className:`preview-actions`,children:(0,T.jsx)(`a`,{href:t.src,download:t.filename,children:`下载`})})]},t.filename))})})]})]})]})}function D(e){let[t,n]=(0,w.useState)(`/v1/models`),[r,i]=(0,w.useState)(``),[a,u]=(0,w.useState)(`等待请求...`),[f,S]=(0,w.useState)(`等待请求...`),[C,D]=(0,w.useState)(`response`),[O,k]=(0,w.useState)([]),[A,j]=(0,w.useState)(`base64`),M=(0,w.useMemo)(()=>[...e.config?.supportedEndpoints||[]].filter(e=>p.includes(e.path)).sort(h),[e.config?.supportedEndpoints]),N=(0,w.useMemo)(()=>M.find(e=>e.path===t)||M[0]||{method:`GET`,path:`/v1/models`,description:``},[t,M]);(0,w.useEffect)(()=>{if(!e.config)return;let r=p.find(t=>e.config?.supportedEndpoints.some(e=>e.path===t)),a=e.config.supportedEndpoints.some(e=>e.path===t)?t:r||`/v1/models`;n(a),i(x(a,e.config.settings.defaultModel))},[e.config]);function P(t){n(t),i(x(t,e.config?.settings.defaultModel||`gpt-5.4`)),k([])}function F(){P(t)}function I(){g(r||x(t,e.config?.settings.defaultModel||`gpt-5.4`)).then(t=>e.setStatus(t?`请求体已复制。`:`请求体复制失败。`)).catch(()=>e.setStatus(`请求体复制失败。`))}function L(){g(a).then(t=>e.setStatus(t?`响应内容已复制。`:`响应内容复制失败。`)).catch(()=>e.setStatus(`响应内容复制失败。`))}function R(){g(f).then(t=>e.setStatus(t?`耗时日志已复制。`:`耗时日志复制失败。`)).catch(()=>e.setStatus(`耗时日志复制失败。`))}async function z(){let t=N,n=performance.now(),i=[];e.setBusy(`test`),D(`response`),u(`请求发送中...`),S(`请求发送中...`),k([]);try{let a=null,o={method:t.method,headers:{}};if(t.method!==`GET`){let e=performance.now();a=r.trim()?JSON.parse(r):{},i.push(`解析请求体: ${l(performance.now()-e)}`),o.headers[`Content-Type`]=`application/json`,o.body=c(a)}let d=performance.now(),f=await fetch(t.path,o);i.push(`等待响应头: ${l(performance.now()-d)}`);let p=performance.now(),h=await f.text();i.push(`读取响应体: ${l(performance.now()-p)}`);let g=performance.now(),_=h;try{_=h?JSON.parse(h):null}catch{_=h}i.push(`解析响应体: ${l(performance.now()-g)}`);let v=m(_);k(v),v.length>0&&D(`preview`),u(typeof _==`string`?_:c(b(_))),S([`${t.method} ${t.path}`,`HTTP 状态: ${f.status} ${f.statusText}`,...i].join(`
2
2
  `)),e.setStatus(`${f.ok?`成功`:`失败`}: HTTP ${f.status} ${t.method} ${t.path}`),e.setRequestLogs(r=>[{id:crypto.randomUUID(),time:Date.now(),method:t.method,endpoint:t.path,account:s(e.config?.profile,e.showEmails),model:typeof a==`object`&&a&&`model`in a?String(a.model||e.config?.settings.defaultModel||`-`):e.config?.settings.defaultModel||`-`,statusCode:f.status,durationMs:performance.now()-n,source:`管理台`},...r].slice(0,20)),e.config?.profile&&e.refreshConfig({silent:!0}).catch(()=>void 0)}catch(n){let r=y(n);u(r),S([`${t.method} ${t.path}(失败)`,...i,`错误: ${r}`].join(`
3
3
  `)),e.setStatus(`请求失败。`)}finally{e.setBusy(null)}}async function B(t,n){if(!t.type.startsWith(`image/`)){e.setStatus(`请选择图片文件。`);return}try{if(n===`image-bed`){e.setBusy(`image-bed-upload`);let n=await _(t);i(d(r,(await v(`/_gateway/image-bed/upload`,{method:`POST`,headers:{"Content-Type":`application/json`},body:JSON.stringify({filename:t.name,dataUrl:n})})).url,e.config?.settings.defaultModel||`gpt-image-2`)),e.setStatus(`已上传到图床并写入公网链接(${t.name},${o(t.size)})。`);return}i(d(r,await _(t),e.config?.settings.defaultModel||`gpt-image-2`)),e.setStatus(`已将 ${t.name} 转成 base64 data URL,并写入请求体 images[0].image_url(${o(t.size)})。`)}catch(t){e.setStatus(`图片写入失败: ${y(t)}`)}finally{n===`image-bed`&&e.setBusy(null)}}return(0,T.jsx)(E,{config:e.config,endpoints:M,activeEndpoint:N,endpoint:t,requestBody:r,responseBody:a,timingBody:f,resultTab:C,status:e.status,busy:e.busy,previewImages:O,capability:e.capability,imageUploadMode:A,onEndpoint:P,onRequestBody:i,onResultTab:D,onRun:z,onResetExample:F,onCopyRequest:I,onCopyResponse:L,onCopyTiming:R,onImageUploadMode:j,onPreview:e.setPreviewImage,onImageUpload:B})}export{D as TesterPage};
@@ -1 +1 @@
1
- import{a as e,n as t,r as n,t as r}from"./jsx-runtime-DqpGtLhh.js";import{t as i}from"./clock-3-BzDANsVk.js";import{t as a}from"./refresh-cw-CAAH2rqe.js";import{t as o}from"./zap-B4_oDbCp.js";import{C as s,x as c}from"./profiles-DMOjJORP.js";import{S as l,_ as u,p as d,r as f}from"./index-CdFYy5j6.js";import{t as p}from"./StatCard-7TEzqn2i.js";var m=t(`camera`,[[`path`,{d:`M13.997 4a2 2 0 0 1 1.76 1.05l.486.9A2 2 0 0 0 18.003 7H20a2 2 0 0 1 2 2v9a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V9a2 2 0 0 1 2-2h1.997a2 2 0 0 0 1.759-1.048l.489-.904A2 2 0 0 1 10.004 4z`,key:`18u6gg`}],[`circle`,{cx:`12`,cy:`13`,r:`3`,key:`1vg3eu`}]]),h=t(`database`,[[`ellipse`,{cx:`12`,cy:`5`,rx:`9`,ry:`3`,key:`msslwz`}],[`path`,{d:`M3 5V19A9 3 0 0 0 21 19V5`,key:`1wlel7`}],[`path`,{d:`M3 12A9 3 0 0 0 21 12`,key:`mv7ke4`}]]),g=t(`sigma`,[[`path`,{d:`M18 7V5a1 1 0 0 0-1-1H6.5a.5.5 0 0 0-.4.8l4.5 6a2 2 0 0 1 0 2.4l-4.5 6a.5.5 0 0 0 .4.8H17a1 1 0 0 0 1-1v-2`,key:`wuwx1p`}]]),_=e(n(),1),v=r();function y(){return{requestCount:0,successCount:0,failureCount:0,inputTokens:0,outputTokens:0,totalTokens:0,unknownTokenCount:0,imageCount:0,totalDurationMs:0,averageDurationMs:0,p95DurationMs:0,durationBuckets:{}}}function b(e){return new Intl.NumberFormat(`zh-CN`).format(Math.round(e||0))}function x(e){return!Number.isFinite(e)||e<=0?`0`:e>=1e6?`${(e/1e6).toFixed(e>=1e7?1:2)}M`:e>=1e4?`${(e/1e3).toFixed(e>=1e5?0:1)}K`:b(e)}function S(e){return`${b(e.successCount)} 成功 / ${b(e.failureCount)} 失败`}function C(e){let t=e.aggregate??y();return(0,v.jsxs)(`article`,{className:`usage-scope-card`,children:[(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`span`,{children:e.title}),(0,v.jsxs)(`strong`,{children:[x(t.totalTokens),` 已知 token`]}),(0,v.jsx)(`p`,{children:e.detail})]}),(0,v.jsxs)(`dl`,{children:[(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`dt`,{children:`请求`}),(0,v.jsx)(`dd`,{children:b(t.requestCount)})]}),(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`dt`,{children:`图片`}),(0,v.jsx)(`dd`,{children:b(t.imageCount)})]}),(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`dt`,{children:`平均`}),(0,v.jsx)(`dd`,{children:c(t.averageDurationMs)})]}),(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`dt`,{children:`P95`}),(0,v.jsx)(`dd`,{children:c(t.p95DurationMs)})]})]}),(0,v.jsxs)(`p`,{className:`usage-scope-footer`,children:[S(t),` · 未返回用量的请求 `,b(t.unknownTokenCount),` 次`]})]})}function w(e){return(0,v.jsxs)(`section`,{className:`usage-table-card`,children:[(0,v.jsxs)(`div`,{className:`usage-table-head`,children:[(0,v.jsx)(`h3`,{children:e.title}),(0,v.jsxs)(`span`,{children:[e.rows.length,` 项`]})]}),e.rows.length===0?(0,v.jsx)(`div`,{className:`usage-empty`,children:e.empty}):(0,v.jsx)(`div`,{className:`usage-table-scroll`,children:(0,v.jsxs)(`table`,{className:`usage-table`,children:[(0,v.jsx)(`thead`,{children:(0,v.jsxs)(`tr`,{children:[(0,v.jsx)(`th`,{children:`维度`}),(0,v.jsx)(`th`,{children:`请求`}),(0,v.jsx)(`th`,{children:`失败`}),(0,v.jsx)(`th`,{children:`已知 token`}),(0,v.jsx)(`th`,{children:`未返回用量`}),(0,v.jsx)(`th`,{children:`图片`}),(0,v.jsx)(`th`,{children:`平均耗时`})]})}),(0,v.jsx)(`tbody`,{children:e.rows.map(e=>(0,v.jsxs)(`tr`,{children:[(0,v.jsx)(`td`,{children:e.label}),(0,v.jsx)(`td`,{children:b(e.aggregate.requestCount)}),(0,v.jsx)(`td`,{children:b(e.aggregate.failureCount)}),(0,v.jsx)(`td`,{children:x(e.aggregate.totalTokens)}),(0,v.jsx)(`td`,{children:b(e.aggregate.unknownTokenCount)}),(0,v.jsx)(`td`,{children:b(e.aggregate.imageCount)}),(0,v.jsx)(`td`,{children:c(e.aggregate.averageDurationMs)})]},e.key))})]})})]})}function T(e){return(0,v.jsxs)(`section`,{className:`usage-table-card usage-daily-card`,children:[(0,v.jsxs)(`div`,{className:`usage-table-head`,children:[(0,v.jsx)(`h3`,{children:`每日趋势`}),(0,v.jsxs)(`span`,{children:[`最近 `,e.rows.length,` 天`]})]}),e.rows.length===0?(0,v.jsx)(`div`,{className:`usage-empty`,children:`还没有历史用量。`}):(0,v.jsx)(`div`,{className:`usage-table-scroll`,children:(0,v.jsxs)(`table`,{className:`usage-table`,children:[(0,v.jsx)(`thead`,{children:(0,v.jsxs)(`tr`,{children:[(0,v.jsx)(`th`,{children:`日期`}),(0,v.jsx)(`th`,{children:`请求`}),(0,v.jsx)(`th`,{children:`成功`}),(0,v.jsx)(`th`,{children:`失败`}),(0,v.jsx)(`th`,{children:`已知输入`}),(0,v.jsx)(`th`,{children:`已知输出`}),(0,v.jsx)(`th`,{children:`已知总 token`}),(0,v.jsx)(`th`,{children:`未返回用量`}),(0,v.jsx)(`th`,{children:`图片`}),(0,v.jsx)(`th`,{children:`P95`})]})}),(0,v.jsx)(`tbody`,{children:e.rows.map(e=>(0,v.jsxs)(`tr`,{children:[(0,v.jsx)(`td`,{children:e.date}),(0,v.jsx)(`td`,{children:b(e.aggregate.requestCount)}),(0,v.jsx)(`td`,{children:b(e.aggregate.successCount)}),(0,v.jsx)(`td`,{children:b(e.aggregate.failureCount)}),(0,v.jsx)(`td`,{children:x(e.aggregate.inputTokens)}),(0,v.jsx)(`td`,{children:x(e.aggregate.outputTokens)}),(0,v.jsx)(`td`,{children:x(e.aggregate.totalTokens)}),(0,v.jsx)(`td`,{children:b(e.aggregate.unknownTokenCount)}),(0,v.jsx)(`td`,{children:b(e.aggregate.imageCount)}),(0,v.jsx)(`td`,{children:c(e.aggregate.p95DurationMs)})]},e.date))})]})})]})}function E(e){let[t,n]=(0,_.useState)(e.config?.usage??null),[r,S]=(0,_.useState)(!1);(0,_.useEffect)(()=>{n(e.config?.usage??null)},[e.config?.usage]);async function E(){S(!0);try{n(await d(`/_gateway/admin/usage`)),e.setStatus(`用量统计已刷新。`)}catch(t){e.setStatus(`用量统计刷新失败: ${f(t)}`)}finally{S(!1)}}let D=t,O=D?.lifetime??y(),k=D?.today??y(),A=D?.startup??y(),j=(0,_.useMemo)(()=>O.requestCount===0?`0%`:`${(O.failureCount/O.requestCount*100).toFixed(1)}%`,[O.failureCount,O.requestCount]);return(0,v.jsxs)(`section`,{className:`usage-page`,children:[(0,v.jsxs)(`div`,{className:`usage-actions`,children:[(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`span`,{children:`统计文件`}),(0,v.jsx)(`code`,{children:D?.storageDir||`~/.ai-zero-token/.state/usage`})]}),(0,v.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:()=>void E(),disabled:r,children:[r?(0,v.jsx)(u,{className:`spin`,size:16}):(0,v.jsx)(a,{size:16}),`刷新统计`]})]}),(0,v.jsxs)(`section`,{className:`summary-grid desktop-summary-grid usage-summary-grid`,children:[(0,v.jsx)(p,{icon:g,label:`历史已知 token`,value:x(O.totalTokens),detail:`${b(O.requestCount)} 次请求,${b(O.unknownTokenCount)} 次未返回用量`,tone:`blue`}),(0,v.jsx)(p,{icon:o,label:`今日已知 token`,value:x(k.totalTokens),detail:`${D?.todayDate||`今天`} · ${b(k.requestCount)} 次请求,${b(k.unknownTokenCount)} 次未返回用量`,tone:`green`}),(0,v.jsx)(p,{icon:l,label:`本次启动已知 token`,value:x(A.totalTokens),detail:`启动于 ${s(D?.startedAt)}`,tone:`brand`}),(0,v.jsx)(p,{icon:m,label:`图片张数`,value:b(O.imageCount),detail:`历史累计生成或编辑图片`,tone:`orange`}),(0,v.jsx)(p,{icon:i,label:`历史 P95`,value:c(O.p95DurationMs),detail:`平均 ${c(O.averageDurationMs)}`,tone:`orange`}),(0,v.jsx)(p,{icon:h,label:`失败率`,value:j,detail:`${b(O.failureCount)} 次失败`,tone:O.failureCount>0?`orange`:`green`})]}),(0,v.jsxs)(`section`,{className:`usage-scope-grid`,children:[(0,v.jsx)(C,{title:`今日用量`,detail:`当天 00:00 到现在`,aggregate:k}),(0,v.jsx)(C,{title:`本次启动`,detail:`当前网关进程启动后到现在`,aggregate:A}),(0,v.jsx)(C,{title:`历史累计`,detail:`从开始记录以来的全部累计`,aggregate:O})]}),(0,v.jsx)(T,{rows:D?.daily??[]}),(0,v.jsxs)(`section`,{className:`usage-dimension-grid`,children:[(0,v.jsx)(w,{title:`按账号`,rows:D?.byAccount??[],empty:`还没有账号维度数据。`}),(0,v.jsx)(w,{title:`按模型`,rows:D?.byModel??[],empty:`还没有模型维度数据。`}),(0,v.jsx)(w,{title:`按接口`,rows:D?.byEndpoint??[],empty:`还没有接口维度数据。`}),(0,v.jsx)(w,{title:`按错误`,rows:D?.byError??[],empty:`还没有错误记录。`}),(0,v.jsx)(w,{title:`按生图链路`,rows:D?.byImageRoute??[],empty:`还没有生图链路数据。`}),(0,v.jsx)(w,{title:`按来源`,rows:D?.bySource??[],empty:`还没有来源维度数据。`})]})]})}export{E as UsagePage};
1
+ import{a as e,n as t,r as n,t as r}from"./jsx-runtime-DqpGtLhh.js";import{t as i}from"./clock-3-BzDANsVk.js";import{t as a}from"./refresh-cw-CAAH2rqe.js";import{t as o}from"./zap-B4_oDbCp.js";import{C as s,x as c}from"./profiles-C5SmQvju.js";import{S as l,_ as u,p as d,r as f}from"./index-CCiBaGwU.js";import{t as p}from"./StatCard-7TEzqn2i.js";var m=t(`camera`,[[`path`,{d:`M13.997 4a2 2 0 0 1 1.76 1.05l.486.9A2 2 0 0 0 18.003 7H20a2 2 0 0 1 2 2v9a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V9a2 2 0 0 1 2-2h1.997a2 2 0 0 0 1.759-1.048l.489-.904A2 2 0 0 1 10.004 4z`,key:`18u6gg`}],[`circle`,{cx:`12`,cy:`13`,r:`3`,key:`1vg3eu`}]]),h=t(`database`,[[`ellipse`,{cx:`12`,cy:`5`,rx:`9`,ry:`3`,key:`msslwz`}],[`path`,{d:`M3 5V19A9 3 0 0 0 21 19V5`,key:`1wlel7`}],[`path`,{d:`M3 12A9 3 0 0 0 21 12`,key:`mv7ke4`}]]),g=t(`sigma`,[[`path`,{d:`M18 7V5a1 1 0 0 0-1-1H6.5a.5.5 0 0 0-.4.8l4.5 6a2 2 0 0 1 0 2.4l-4.5 6a.5.5 0 0 0 .4.8H17a1 1 0 0 0 1-1v-2`,key:`wuwx1p`}]]),_=e(n(),1),v=r();function y(){return{requestCount:0,successCount:0,failureCount:0,inputTokens:0,outputTokens:0,totalTokens:0,unknownTokenCount:0,imageCount:0,totalDurationMs:0,averageDurationMs:0,p95DurationMs:0,durationBuckets:{}}}function b(e){return new Intl.NumberFormat(`zh-CN`).format(Math.round(e||0))}function x(e){return!Number.isFinite(e)||e<=0?`0`:e>=1e6?`${(e/1e6).toFixed(e>=1e7?1:2)}M`:e>=1e4?`${(e/1e3).toFixed(e>=1e5?0:1)}K`:b(e)}function S(e){return`${b(e.successCount)} 成功 / ${b(e.failureCount)} 失败`}function C(e){let t=e.aggregate??y();return(0,v.jsxs)(`article`,{className:`usage-scope-card`,children:[(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`span`,{children:e.title}),(0,v.jsxs)(`strong`,{children:[x(t.totalTokens),` 已知 token`]}),(0,v.jsx)(`p`,{children:e.detail})]}),(0,v.jsxs)(`dl`,{children:[(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`dt`,{children:`请求`}),(0,v.jsx)(`dd`,{children:b(t.requestCount)})]}),(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`dt`,{children:`图片`}),(0,v.jsx)(`dd`,{children:b(t.imageCount)})]}),(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`dt`,{children:`平均`}),(0,v.jsx)(`dd`,{children:c(t.averageDurationMs)})]}),(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`dt`,{children:`P95`}),(0,v.jsx)(`dd`,{children:c(t.p95DurationMs)})]})]}),(0,v.jsxs)(`p`,{className:`usage-scope-footer`,children:[S(t),` · 未返回用量的请求 `,b(t.unknownTokenCount),` 次`]})]})}function w(e){return(0,v.jsxs)(`section`,{className:`usage-table-card`,children:[(0,v.jsxs)(`div`,{className:`usage-table-head`,children:[(0,v.jsx)(`h3`,{children:e.title}),(0,v.jsxs)(`span`,{children:[e.rows.length,` 项`]})]}),e.rows.length===0?(0,v.jsx)(`div`,{className:`usage-empty`,children:e.empty}):(0,v.jsx)(`div`,{className:`usage-table-scroll`,children:(0,v.jsxs)(`table`,{className:`usage-table`,children:[(0,v.jsx)(`thead`,{children:(0,v.jsxs)(`tr`,{children:[(0,v.jsx)(`th`,{children:`维度`}),(0,v.jsx)(`th`,{children:`请求`}),(0,v.jsx)(`th`,{children:`失败`}),(0,v.jsx)(`th`,{children:`已知 token`}),(0,v.jsx)(`th`,{children:`未返回用量`}),(0,v.jsx)(`th`,{children:`图片`}),(0,v.jsx)(`th`,{children:`平均耗时`})]})}),(0,v.jsx)(`tbody`,{children:e.rows.map(e=>(0,v.jsxs)(`tr`,{children:[(0,v.jsx)(`td`,{children:e.label}),(0,v.jsx)(`td`,{children:b(e.aggregate.requestCount)}),(0,v.jsx)(`td`,{children:b(e.aggregate.failureCount)}),(0,v.jsx)(`td`,{children:x(e.aggregate.totalTokens)}),(0,v.jsx)(`td`,{children:b(e.aggregate.unknownTokenCount)}),(0,v.jsx)(`td`,{children:b(e.aggregate.imageCount)}),(0,v.jsx)(`td`,{children:c(e.aggregate.averageDurationMs)})]},e.key))})]})})]})}function T(e){return(0,v.jsxs)(`section`,{className:`usage-table-card usage-daily-card`,children:[(0,v.jsxs)(`div`,{className:`usage-table-head`,children:[(0,v.jsx)(`h3`,{children:`每日趋势`}),(0,v.jsxs)(`span`,{children:[`最近 `,e.rows.length,` 天`]})]}),e.rows.length===0?(0,v.jsx)(`div`,{className:`usage-empty`,children:`还没有历史用量。`}):(0,v.jsx)(`div`,{className:`usage-table-scroll`,children:(0,v.jsxs)(`table`,{className:`usage-table`,children:[(0,v.jsx)(`thead`,{children:(0,v.jsxs)(`tr`,{children:[(0,v.jsx)(`th`,{children:`日期`}),(0,v.jsx)(`th`,{children:`请求`}),(0,v.jsx)(`th`,{children:`成功`}),(0,v.jsx)(`th`,{children:`失败`}),(0,v.jsx)(`th`,{children:`已知输入`}),(0,v.jsx)(`th`,{children:`已知输出`}),(0,v.jsx)(`th`,{children:`已知总 token`}),(0,v.jsx)(`th`,{children:`未返回用量`}),(0,v.jsx)(`th`,{children:`图片`}),(0,v.jsx)(`th`,{children:`P95`})]})}),(0,v.jsx)(`tbody`,{children:e.rows.map(e=>(0,v.jsxs)(`tr`,{children:[(0,v.jsx)(`td`,{children:e.date}),(0,v.jsx)(`td`,{children:b(e.aggregate.requestCount)}),(0,v.jsx)(`td`,{children:b(e.aggregate.successCount)}),(0,v.jsx)(`td`,{children:b(e.aggregate.failureCount)}),(0,v.jsx)(`td`,{children:x(e.aggregate.inputTokens)}),(0,v.jsx)(`td`,{children:x(e.aggregate.outputTokens)}),(0,v.jsx)(`td`,{children:x(e.aggregate.totalTokens)}),(0,v.jsx)(`td`,{children:b(e.aggregate.unknownTokenCount)}),(0,v.jsx)(`td`,{children:b(e.aggregate.imageCount)}),(0,v.jsx)(`td`,{children:c(e.aggregate.p95DurationMs)})]},e.date))})]})})]})}function E(e){let[t,n]=(0,_.useState)(e.config?.usage??null),[r,S]=(0,_.useState)(!1);(0,_.useEffect)(()=>{n(e.config?.usage??null)},[e.config?.usage]);async function E(){S(!0);try{n(await d(`/_gateway/admin/usage`)),e.setStatus(`用量统计已刷新。`)}catch(t){e.setStatus(`用量统计刷新失败: ${f(t)}`)}finally{S(!1)}}let D=t,O=D?.lifetime??y(),k=D?.today??y(),A=D?.startup??y(),j=(0,_.useMemo)(()=>O.requestCount===0?`0%`:`${(O.failureCount/O.requestCount*100).toFixed(1)}%`,[O.failureCount,O.requestCount]);return(0,v.jsxs)(`section`,{className:`usage-page`,children:[(0,v.jsxs)(`div`,{className:`usage-actions`,children:[(0,v.jsxs)(`div`,{children:[(0,v.jsx)(`span`,{children:`统计文件`}),(0,v.jsx)(`code`,{children:D?.storageDir||`~/.ai-zero-token/.state/usage`})]}),(0,v.jsxs)(`button`,{className:`btn-secondary`,type:`button`,onClick:()=>void E(),disabled:r,children:[r?(0,v.jsx)(u,{className:`spin`,size:16}):(0,v.jsx)(a,{size:16}),`刷新统计`]})]}),(0,v.jsxs)(`section`,{className:`summary-grid desktop-summary-grid usage-summary-grid`,children:[(0,v.jsx)(p,{icon:g,label:`历史已知 token`,value:x(O.totalTokens),detail:`${b(O.requestCount)} 次请求,${b(O.unknownTokenCount)} 次未返回用量`,tone:`blue`}),(0,v.jsx)(p,{icon:o,label:`今日已知 token`,value:x(k.totalTokens),detail:`${D?.todayDate||`今天`} · ${b(k.requestCount)} 次请求,${b(k.unknownTokenCount)} 次未返回用量`,tone:`green`}),(0,v.jsx)(p,{icon:l,label:`本次启动已知 token`,value:x(A.totalTokens),detail:`启动于 ${s(D?.startedAt)}`,tone:`brand`}),(0,v.jsx)(p,{icon:m,label:`图片张数`,value:b(O.imageCount),detail:`历史累计生成或编辑图片`,tone:`orange`}),(0,v.jsx)(p,{icon:i,label:`历史 P95`,value:c(O.p95DurationMs),detail:`平均 ${c(O.averageDurationMs)}`,tone:`orange`}),(0,v.jsx)(p,{icon:h,label:`失败率`,value:j,detail:`${b(O.failureCount)} 次失败`,tone:O.failureCount>0?`orange`:`green`})]}),(0,v.jsxs)(`section`,{className:`usage-scope-grid`,children:[(0,v.jsx)(C,{title:`今日用量`,detail:`当天 00:00 到现在`,aggregate:k}),(0,v.jsx)(C,{title:`本次启动`,detail:`当前网关进程启动后到现在`,aggregate:A}),(0,v.jsx)(C,{title:`历史累计`,detail:`从开始记录以来的全部累计`,aggregate:O})]}),(0,v.jsx)(T,{rows:D?.daily??[]}),(0,v.jsxs)(`section`,{className:`usage-dimension-grid`,children:[(0,v.jsx)(w,{title:`按账号`,rows:D?.byAccount??[],empty:`还没有账号维度数据。`}),(0,v.jsx)(w,{title:`按模型`,rows:D?.byModel??[],empty:`还没有模型维度数据。`}),(0,v.jsx)(w,{title:`按接口`,rows:D?.byEndpoint??[],empty:`还没有接口维度数据。`}),(0,v.jsx)(w,{title:`按错误`,rows:D?.byError??[],empty:`还没有错误记录。`}),(0,v.jsx)(w,{title:`按生图链路`,rows:D?.byImageRoute??[],empty:`还没有生图链路数据。`}),(0,v.jsx)(w,{title:`按来源`,rows:D?.bySource??[],empty:`还没有来源维度数据。`})]})]})}export{E as UsagePage};
@@ -5,9 +5,9 @@
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
6
  <link rel="icon" type="image/svg+xml" href="/assets/app-mark-nsRs4vo7.svg" />
7
7
  <title>AI Zero Token</title>
8
- <script type="module" crossorigin src="/assets/index-CdFYy5j6.js"></script>
8
+ <script type="module" crossorigin src="/assets/index-CCiBaGwU.js"></script>
9
9
  <link rel="modulepreload" crossorigin href="/assets/jsx-runtime-DqpGtLhh.js">
10
- <link rel="modulepreload" crossorigin href="/assets/profiles-DMOjJORP.js">
10
+ <link rel="modulepreload" crossorigin href="/assets/profiles-C5SmQvju.js">
11
11
  <link rel="stylesheet" crossorigin href="/assets/index-C22_3Mxq.css">
12
12
  </head>
13
13
  <body>
@@ -20,7 +20,7 @@ function printHelp() {
20
20
  \u8BF4\u660E:
21
21
 
22
22
  login \u8D70\u771F\u5B9E OpenAI Codex OAuth\uFF0C\u65B0\u589E\u5E76\u4FDD\u5B58\u4E00\u4E2A\u8D26\u53F7 profile
23
- models \u67E5\u770B\u5F53\u524D\u53EF\u7528\u6A21\u578B\u5217\u8868\uFF1B\u4F18\u5148\u8BFB\u53D6 ~/.codex/models_cache.json\uFF0C--refresh \u53EF\u624B\u52A8\u91CD\u8BFB
23
+ models \u67E5\u770B\u5F53\u524D\u53EF\u7528\u6A21\u578B\u5217\u8868\uFF1B\u4F18\u5148\u8BFB\u53D6 ~/.codex/models_cache.json\uFF0C--refresh \u4ECE Codex \u540E\u7AEF\u540C\u6B65
24
24
  profiles \u5BFC\u5165/\u5BFC\u51FA\u8D26\u53F7 JSON\uFF1B\u5BFC\u51FA\u6587\u4EF6\u5305\u542B\u5B8C\u6574 refresh token\uFF0C\u8BF7\u53EA\u5206\u4EAB\u7ED9\u53EF\u4FE1\u5BF9\u8C61
25
25
  status \u67E5\u770B\u5F53\u524D demo \u5F53\u524D\u6FC0\u6D3B\u8D26\u53F7\u3001\u8D26\u53F7\u6570\u91CF\u548C\u8FC7\u671F\u65F6\u95F4
26
26
  ask \u7528\u4FDD\u5B58\u7684 token \u8C03\u771F\u5B9E Codex Responses API
@@ -7,8 +7,9 @@ async function runModelsCommand(args = []) {
7
7
  models: await ctx.modelService.listModels(),
8
8
  catalog: await ctx.modelService.getCatalog()
9
9
  };
10
- console.log(refresh ? "\u5DF2\u91CD\u65B0\u8BFB\u53D6 Codex \u6A21\u578B\u5217\u8868:" : "\u5F53\u524D demo \u53EF\u7528\u6A21\u578B\u5217\u8868:");
11
- console.log(`- \u6765\u6E90: ${result.catalog.source === "codex-cache" ? "Codex \u672C\u5730\u7F13\u5B58" : "\u9879\u76EE\u5185\u7F6E\u56DE\u9000\u5217\u8868"}`);
10
+ console.log(refresh ? "\u5DF2\u4ECE\u7F51\u7EDC\u540C\u6B65 Codex \u6A21\u578B\u5217\u8868:" : "\u5F53\u524D demo \u53EF\u7528\u6A21\u578B\u5217\u8868:");
11
+ const sourceLabel = result.catalog.source === "codex-network" ? "Codex \u7F51\u7EDC\u540C\u6B65" : result.catalog.source === "codex-cache" ? "Codex \u672C\u5730\u7F13\u5B58" : "\u9879\u76EE\u5185\u7F6E\u56DE\u9000\u5217\u8868";
12
+ console.log(`- \u6765\u6E90: ${sourceLabel}`);
12
13
  console.log(`- \u8DEF\u5F84: ${result.catalog.cachePath}`);
13
14
  if (result.catalog.fetchedAt) {
14
15
  console.log(`- Codex \u66F4\u65B0\u65F6\u95F4: ${result.catalog.fetchedAt}`);
@@ -11,7 +11,7 @@ import { UsageService } from "./services/usage-service.js";
11
11
  function createGatewayContext() {
12
12
  const configService = new ConfigService();
13
13
  const authService = new AuthService(configService);
14
- const modelService = new ModelService(configService);
14
+ const modelService = new ModelService(configService, authService);
15
15
  const versionService = new VersionService();
16
16
  const usageService = new UsageService();
17
17
  const networkDetectService = new NetworkDetectService();
@@ -2,7 +2,11 @@
2
2
  import fs from "node:fs/promises";
3
3
  import os from "node:os";
4
4
  import path from "node:path";
5
+ import { requestText } from "../providers/http-client.js";
5
6
  const DEFAULT_CODEX_MODEL = "gpt-5.4";
7
+ const CODEX_MODELS_URL = process.env.CODEX_MODELS_URL || "https://chatgpt.com/backend-api/codex/models";
8
+ const CODEX_MODELS_REFRESH_TIMEOUT_MS = 6e4;
9
+ const DEFAULT_CODEX_MODELS_CLIENT_VERSION = "0.130.0";
6
10
  const CODEX_MODEL_INFOS = [
7
11
  { provider: "openai-codex", id: "gpt-5.4", name: "GPT-5.4", input: ["text", "image"], source: "static" },
8
12
  { provider: "openai-codex", id: "gpt-5.2", name: "GPT-5.2", input: ["text", "image"], source: "static" },
@@ -56,6 +60,66 @@ function normalizeCodexCacheEntry(entry) {
56
60
  source: "codex-cache"
57
61
  };
58
62
  }
63
+ function isRecord(value) {
64
+ return !!value && typeof value === "object" && !Array.isArray(value);
65
+ }
66
+ function normalizeNetworkModelsBody(value, headers, clientVersion) {
67
+ const body = isRecord(value) ? value : {};
68
+ const rawModels = Array.isArray(value) ? value : Array.isArray(body.models) ? body.models : Array.isArray(body.data) ? body.data : null;
69
+ if (!rawModels) {
70
+ throw new Error("Codex \u6A21\u578B\u5217\u8868\u54CD\u5E94\u7F3A\u5C11 models \u5B57\u6BB5\u3002");
71
+ }
72
+ const models = rawModels.filter(isRecord);
73
+ const cache = {
74
+ ...body,
75
+ fetched_at: typeof body.fetched_at === "string" ? body.fetched_at : (/* @__PURE__ */ new Date()).toISOString(),
76
+ client_version: typeof body.client_version === "string" ? body.client_version : clientVersion,
77
+ models
78
+ };
79
+ const responseEtag = headers.etag;
80
+ if (!cache.etag && responseEtag) {
81
+ cache.etag = responseEtag;
82
+ }
83
+ return cache;
84
+ }
85
+ async function getCodexModelsClientVersion(cachePath) {
86
+ const override = process.env.CODEX_MODELS_CLIENT_VERSION?.trim();
87
+ if (override) {
88
+ return override;
89
+ }
90
+ try {
91
+ const parsed = JSON.parse(await fs.readFile(cachePath, "utf8"));
92
+ if (typeof parsed.client_version === "string" && parsed.client_version.trim()) {
93
+ return parsed.client_version.trim();
94
+ }
95
+ } catch {
96
+ }
97
+ return DEFAULT_CODEX_MODELS_CLIENT_VERSION;
98
+ }
99
+ function buildCodexModelsUrl(clientVersion) {
100
+ const url = new URL(CODEX_MODELS_URL);
101
+ if (!url.searchParams.has("client_version")) {
102
+ url.searchParams.set("client_version", clientVersion);
103
+ }
104
+ return url.toString();
105
+ }
106
+ function buildCodexModelsRequestHeaders(profile) {
107
+ return {
108
+ Accept: "application/json",
109
+ Authorization: `Bearer ${profile.access}`,
110
+ "ChatGPT-Account-Id": profile.accountId,
111
+ "OpenAI-Beta": "responses=experimental",
112
+ Originator: "pi",
113
+ "User-Agent": "pi (bun demo)"
114
+ };
115
+ }
116
+ function createCodexModelsError(status, transport, body) {
117
+ const preview = body.trim().slice(0, 1e3);
118
+ const error = new Error(`\u540C\u6B65 Codex \u6A21\u578B\u5931\u8D25: HTTP ${status} via ${transport}${preview ? ` ${preview}` : ""}`);
119
+ error.upstreamStatus = status;
120
+ error.upstreamErrorMessage = preview;
121
+ return error;
122
+ }
59
123
  function dedupeModels(models) {
60
124
  const seen = /* @__PURE__ */ new Set();
61
125
  const next = [];
@@ -96,6 +160,46 @@ async function getCodexModelCatalog() {
96
160
  }
97
161
  };
98
162
  }
163
+ async function refreshCodexModelCatalogFromNetwork(profile) {
164
+ const cachePath = getCodexModelsCachePath();
165
+ const clientVersion = await getCodexModelsClientVersion(cachePath);
166
+ const response = await requestText({
167
+ method: "GET",
168
+ url: buildCodexModelsUrl(clientVersion),
169
+ headers: buildCodexModelsRequestHeaders(profile),
170
+ timeoutMs: CODEX_MODELS_REFRESH_TIMEOUT_MS
171
+ });
172
+ if (response.status < 200 || response.status >= 300) {
173
+ throw createCodexModelsError(response.status, response.transport, response.body);
174
+ }
175
+ let parsed;
176
+ try {
177
+ parsed = JSON.parse(response.body);
178
+ } catch {
179
+ throw new Error("\u540C\u6B65 Codex \u6A21\u578B\u5931\u8D25: \u4E0A\u6E38\u8FD4\u56DE\u7684\u6A21\u578B\u5217\u8868\u4E0D\u662F\u6709\u6548 JSON\u3002");
180
+ }
181
+ const cache = normalizeNetworkModelsBody(parsed, response.headers, clientVersion);
182
+ const models = dedupeModels((cache.models ?? []).map(normalizeCodexCacheEntry).filter(Boolean));
183
+ if (models.length === 0) {
184
+ throw new Error("\u540C\u6B65 Codex \u6A21\u578B\u5931\u8D25: \u4E0A\u6E38\u6A21\u578B\u5217\u8868\u4E3A\u7A7A\u6216\u6CA1\u6709\u53EF\u5C55\u793A\u6A21\u578B\u3002");
185
+ }
186
+ await fs.mkdir(path.dirname(cachePath), { recursive: true });
187
+ await fs.writeFile(cachePath, `${JSON.stringify(cache, null, 2)}
188
+ `, "utf8");
189
+ const networkModels = models.map((model) => ({
190
+ ...model,
191
+ source: "codex-network"
192
+ }));
193
+ return {
194
+ models: networkModels,
195
+ catalog: {
196
+ source: "codex-network",
197
+ cachePath,
198
+ fetchedAt: cache.fetched_at,
199
+ modelCount: networkModels.length
200
+ }
201
+ };
202
+ }
99
203
  async function hasCodexModel(model) {
100
204
  const { models } = await getCodexModelCatalog();
101
205
  return models.some((item) => item.id === model);
@@ -118,5 +222,6 @@ export {
118
222
  getCodexModelsCachePath,
119
223
  getPreferredCodexModel,
120
224
  hasCodexModel,
121
- isSupportedCodexModel
225
+ isSupportedCodexModel,
226
+ refreshCodexModelCatalogFromNetwork
122
227
  };