llm-simple-router 0.11.30 → 0.12.0-dev.7cf31a2

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 (151) hide show
  1. package/config/recommended-providers.json +24 -12
  2. package/config/recommended-retry-rules.json +128 -11
  3. package/dist/admin/providers.js +1 -1
  4. package/dist/admin/quick-setup.js +61 -6
  5. package/dist/admin/usage.js +19 -12
  6. package/dist/config/recommended-providers.json +24 -12
  7. package/dist/config/recommended-retry-rules.json +128 -11
  8. package/dist/config/recommended.d.ts +1 -0
  9. package/dist/core/concurrency/adaptive-controller.d.ts +2 -0
  10. package/dist/core/concurrency/adaptive-controller.js +32 -27
  11. package/dist/core/concurrency/types.d.ts +0 -1
  12. package/dist/core/monitor/types.d.ts +2 -0
  13. package/dist/db/index.js +42 -0
  14. package/dist/proxy/orchestration/orchestrator.js +2 -0
  15. package/frontend-dist/assets/AuthLayout-AvTL8TD9.js +1 -0
  16. package/frontend-dist/assets/Card-_-Rsez1Z.js +1 -0
  17. package/frontend-dist/assets/CardContent-CXB0aTQX.js +1 -0
  18. package/frontend-dist/assets/{CardTitle-Cs6WQYrl.js → CardTitle-DEgSNH9O.js} +1 -1
  19. package/frontend-dist/assets/CascadingModelSelect-vGUnj1X-.js +1 -0
  20. package/frontend-dist/assets/{Checkbox-ljNZhtRQ.js → Checkbox-Bwhfqxux.js} +1 -1
  21. package/frontend-dist/assets/{CollapsibleContent-Yk-lxVPc.js → CollapsibleContent-DDzm_Bxn.js} +1 -1
  22. package/frontend-dist/assets/CollapsibleTrigger-CWUxEdhm.js +1 -0
  23. package/frontend-dist/assets/ConcurrencyControl-DXIrmfyy.js +1 -0
  24. package/frontend-dist/assets/Dashboard-Gzd7h17f.js +3 -0
  25. package/frontend-dist/assets/{Input-BGoap26P.js → Input-Qz28j7do.js} +1 -1
  26. package/frontend-dist/assets/{Label-CR9CcQmY.js → Label-CgLYbqNp.js} +1 -1
  27. package/frontend-dist/assets/Login-Dds3B1xs.js +1 -0
  28. package/frontend-dist/assets/Logs-OknLq4IX.js +1 -0
  29. package/frontend-dist/assets/ModelMappings-BBOSIVlU.js +1 -0
  30. package/frontend-dist/assets/Monitor-BN6VeuSD.js +1 -0
  31. package/frontend-dist/assets/Providers-C6v2RKW1.js +1 -0
  32. package/frontend-dist/assets/ProxyEnhancement-Bs_ocCbv.js +1 -0
  33. package/frontend-dist/assets/QuickSetup-CG1X-HWo.js +1 -0
  34. package/frontend-dist/assets/RetryRules-DCXYuPl-.js +1 -0
  35. package/frontend-dist/assets/RouterKeys-4hYMVOwN.js +1 -0
  36. package/frontend-dist/assets/RouterKeys-DEMKBc7g.css +1 -0
  37. package/frontend-dist/assets/{RovingFocusItem-C31XRuKO.js → RovingFocusItem-CCbZ7_un.js} +1 -1
  38. package/frontend-dist/assets/Schedules-C0Kf-gvS.js +1 -0
  39. package/frontend-dist/assets/Settings-u7_Vhsqi.js +6 -0
  40. package/frontend-dist/assets/Setup-BXxAhB7r.js +1 -0
  41. package/frontend-dist/assets/Skeleton-QesVeTCW.js +1 -0
  42. package/frontend-dist/assets/{Switch-Dxml8U-u.js → Switch-D2xEGYqH.js} +1 -1
  43. package/frontend-dist/assets/TableHeader-CHaThXAU.js +1 -0
  44. package/frontend-dist/assets/TabsTrigger-Cl1kx8BK.js +1 -0
  45. package/frontend-dist/assets/TooltipTrigger-8hiSB2Rk.js +1 -0
  46. package/frontend-dist/assets/UnifiedRequestDialog-Cm6Ao45M.js +3 -0
  47. package/frontend-dist/assets/UnifiedRequestDialog-DaZodSzm.css +1 -0
  48. package/frontend-dist/assets/{VisuallyHiddenInput-D7dv0owV.js → VisuallyHiddenInput-Lv6BsAIS.js} +1 -1
  49. package/frontend-dist/assets/_plugin-vue_export-helper-BH2M0_bF.js +1 -0
  50. package/frontend-dist/assets/arrow-down-C7Be80wb.js +1 -0
  51. package/frontend-dist/assets/badge-C82CIl1p.js +1 -0
  52. package/frontend-dist/assets/{button-CiJBAzrS.js → button-DXlWUlLe.js} +2 -2
  53. package/frontend-dist/assets/chevron-right-DJE8HF-i.js +1 -0
  54. package/frontend-dist/assets/common-ZLJwmI0c.js +1 -0
  55. package/frontend-dist/assets/common-qa3O5YJX.js +1 -0
  56. package/frontend-dist/assets/dashboard-BlCt9ud6.js +1 -0
  57. package/frontend-dist/assets/dashboard-CqwGLWoK.js +1 -0
  58. package/frontend-dist/assets/dialog-DN2mvc9h.js +1 -0
  59. package/frontend-dist/assets/image-8GxBtf66.js +1 -0
  60. package/frontend-dist/assets/index-BcYqgllr.css +1 -0
  61. package/frontend-dist/assets/index-FYAc4GSw.js +3 -0
  62. package/frontend-dist/assets/logs-BI_TLeBe.js +1 -0
  63. package/frontend-dist/assets/logs-DZyb4-Ew.js +1 -0
  64. package/frontend-dist/assets/mappings-Bw39-I5f.js +1 -0
  65. package/frontend-dist/assets/mappings-DBDiQL8p.js +1 -0
  66. package/frontend-dist/assets/model-patches-Dr6eEIuK.js +1 -0
  67. package/frontend-dist/assets/monitor-Crmj9XSn.js +1 -0
  68. package/frontend-dist/assets/monitor-D5lXaCoi.js +1 -0
  69. package/frontend-dist/assets/pencil-BzLz9HuE.js +1 -0
  70. package/frontend-dist/assets/plus-DghQKiGV.js +1 -0
  71. package/frontend-dist/assets/providers-Cy6EVDp5.js +1 -0
  72. package/frontend-dist/assets/providers-DTIaw_44.js +1 -0
  73. package/frontend-dist/assets/proxyEnhancement-BEbUoM8Y.js +3 -0
  74. package/frontend-dist/assets/proxyEnhancement-rdWlqMlw.js +3 -0
  75. package/frontend-dist/assets/quickSetup-CjxI1qgV.js +1 -0
  76. package/frontend-dist/assets/quickSetup-zSIVom2e.js +1 -0
  77. package/frontend-dist/assets/requestDetail-BgwV24CW.js +1 -0
  78. package/frontend-dist/assets/requestDetail-zegFIYZs.js +1 -0
  79. package/frontend-dist/assets/routerKeys-B2_9kz1O.js +1 -0
  80. package/frontend-dist/assets/routerKeys-C579ZS8v.js +1 -0
  81. package/frontend-dist/assets/{schedules-ByzpVhl2.js → schedules-DeR-EOsE.js} +1 -1
  82. package/frontend-dist/assets/schedules-u1et4hyF.js +1 -0
  83. package/frontend-dist/assets/search-BvelBk-g.js +1 -0
  84. package/frontend-dist/assets/{sparkles-DcNGAfEa.js → sparkles-B2aVxqID.js} +1 -1
  85. package/frontend-dist/assets/transform-domain-BGh_oMvU.js +1 -0
  86. package/frontend-dist/assets/{trash-2-C_XOOsld.js → trash-2-BFS2nrZH.js} +1 -1
  87. package/frontend-dist/assets/useClipboard-D_fR9Erj.js +1 -0
  88. package/frontend-dist/assets/useLogRetention-CLvwmbya.js +1 -0
  89. package/frontend-dist/assets/useProviderGroups-DYlN-TXY.js +1 -0
  90. package/frontend-dist/index.html +4 -4
  91. package/package.json +1 -1
  92. package/frontend-dist/assets/CardContent-BU3fjxiM.js +0 -1
  93. package/frontend-dist/assets/CascadingModelSelect-B_n7DYfi.js +0 -1
  94. package/frontend-dist/assets/CollapsibleTrigger-8_A4LNtD.js +0 -1
  95. package/frontend-dist/assets/Dashboard-C5oqC_3e.js +0 -3
  96. package/frontend-dist/assets/Login-CM-J-SVi.js +0 -1
  97. package/frontend-dist/assets/Logs-CSZi8Fkp.js +0 -1
  98. package/frontend-dist/assets/MappingEntryEditor-Dng2wCdx.js +0 -1
  99. package/frontend-dist/assets/ModelMappings-_ONUTtBQ.js +0 -1
  100. package/frontend-dist/assets/Monitor-o28u2RV9.js +0 -1
  101. package/frontend-dist/assets/Providers-awOg4hbb.js +0 -1
  102. package/frontend-dist/assets/ProxyEnhancement-DV-dfZLk.js +0 -1
  103. package/frontend-dist/assets/QuickSetup-XvyghGod.js +0 -1
  104. package/frontend-dist/assets/RetryRules-EnfS-FTJ.js +0 -1
  105. package/frontend-dist/assets/RouterKeys-BzcUK5x_.js +0 -1
  106. package/frontend-dist/assets/Schedules-BuP-3qX8.js +0 -1
  107. package/frontend-dist/assets/Settings-hBsvQQWG.js +0 -6
  108. package/frontend-dist/assets/Setup-CukhOjX3.js +0 -1
  109. package/frontend-dist/assets/TooltipTrigger-BITX1gT4.js +0 -1
  110. package/frontend-dist/assets/TransformRulesForm-DFW-oDll.js +0 -1
  111. package/frontend-dist/assets/UnifiedRequestDialog-C3sWdkTe.css +0 -1
  112. package/frontend-dist/assets/UnifiedRequestDialog-DChZe4do.js +0 -3
  113. package/frontend-dist/assets/common-BCe8fK5N.js +0 -1
  114. package/frontend-dist/assets/common-CdzdKRsV.js +0 -1
  115. package/frontend-dist/assets/copy-C035pTD9.js +0 -1
  116. package/frontend-dist/assets/dashboard-C0XEgNnb.js +0 -1
  117. package/frontend-dist/assets/dashboard-uBv6TkGn.js +0 -1
  118. package/frontend-dist/assets/dialog-QrJZom9g.js +0 -1
  119. package/frontend-dist/assets/index-B_tA9Si9.css +0 -1
  120. package/frontend-dist/assets/index-bv08h-oy.js +0 -3
  121. package/frontend-dist/assets/logs-C3X3MZjQ.js +0 -1
  122. package/frontend-dist/assets/logs-HY8sUOwb.js +0 -1
  123. package/frontend-dist/assets/mappings-B6NCfNio.js +0 -1
  124. package/frontend-dist/assets/mappings-yo-RxHK6.js +0 -1
  125. package/frontend-dist/assets/model-patches-1yALrtK3.js +0 -1
  126. package/frontend-dist/assets/monitor-CcfwHbJF.js +0 -1
  127. package/frontend-dist/assets/monitor-XNX6LZGC.js +0 -1
  128. package/frontend-dist/assets/plus-7SwZDVUb.js +0 -1
  129. package/frontend-dist/assets/providers-BaiCysMg.js +0 -1
  130. package/frontend-dist/assets/providers-DsC5Qd6m.js +0 -1
  131. package/frontend-dist/assets/proxyEnhancement-BC3eVKDp.js +0 -3
  132. package/frontend-dist/assets/proxyEnhancement-Dq47MAAI.js +0 -3
  133. package/frontend-dist/assets/quickSetup-C8jt6VzA.js +0 -1
  134. package/frontend-dist/assets/quickSetup-tMWsXvo3.js +0 -1
  135. package/frontend-dist/assets/requestDetail-C5px2Mru.js +0 -1
  136. package/frontend-dist/assets/requestDetail-Tozq3jwE.js +0 -1
  137. package/frontend-dist/assets/routerKeys-C4oCPJrT.js +0 -1
  138. package/frontend-dist/assets/routerKeys-CyZ4L-1h.js +0 -1
  139. package/frontend-dist/assets/schedules-tSjne6S-.js +0 -1
  140. package/frontend-dist/assets/useClipboard-DaHSM2Ce.js +0 -1
  141. package/frontend-dist/assets/useLogRetention-D7MYPV9z.js +0 -1
  142. /package/frontend-dist/assets/{login-DXMVxesL.js → login-BjRWJuuH.js} +0 -0
  143. /package/frontend-dist/assets/{login-BozjQrft.js → login-Drdlb6If.js} +0 -0
  144. /package/frontend-dist/assets/{retryRules-DGg26acb.js → retryRules-KY-2u2K-.js} +0 -0
  145. /package/frontend-dist/assets/{retryRules-CR3EIOTr.js → retryRules-v9HTZUGQ.js} +0 -0
  146. /package/frontend-dist/assets/{settings-DQqli04G.js → settings-av7tA8wt.js} +0 -0
  147. /package/frontend-dist/assets/{settings-CveTb9z9.js → settings-dzu7sg6b.js} +0 -0
  148. /package/frontend-dist/assets/{setup-BKw9vkV3.js → setup-B0bSyAH4.js} +0 -0
  149. /package/frontend-dist/assets/{setup-D9yk-VSM.js → setup-Bw-U-YJa.js} +0 -0
  150. /package/frontend-dist/assets/{sidebar-T-fjy0Pb.js → sidebar-BxTRaJDZ.js} +0 -0
  151. /package/frontend-dist/assets/{sidebar-HuUg5Wsk.js → sidebar-Dl1-HY8Q.js} +0 -0
@@ -1,13 +1,130 @@
1
1
  [
2
- { "name": "429 Too Many Requests", "status_code": 429, "body_pattern": ".*", "retry_strategy": "exponential", "retry_delay_ms": 5000, "max_retries": 10, "max_delay_ms": 60000, "providers": [] },
3
- { "name": "503 Service Unavailable", "status_code": 503, "body_pattern": ".*", "retry_strategy": "exponential", "retry_delay_ms": 5000, "max_retries": 10, "max_delay_ms": 60000, "providers": [] },
4
- { "name": "ZAI 网络错误 (code 1234)", "status_code": 400, "body_pattern": "\"type\"\\s*:\\s*\"error\".*\"code\"\\s*:\\s*\"1234\"", "retry_strategy": "exponential", "retry_delay_ms": 5000, "max_retries": 10, "max_delay_ms": 60000, "providers": ["智谱"] },
5
- { "name": "ZAI 临时不可用", "status_code": 400, "body_pattern": "\"type\"\\s*:\\s*\"error\".*请稍后重试", "retry_strategy": "exponential", "retry_delay_ms": 5000, "max_retries": 10, "max_delay_ms": 60000, "providers": ["智谱"] },
6
- { "name": "ZAI 操作失败 (code 500)", "status_code": 400, "body_pattern": "\"type\"\\s*:\\s*\"error\".*\"code\"\\s*:\\s*\"500\"", "retry_strategy": "exponential", "retry_delay_ms": 5000, "max_retries": 10, "max_delay_ms": 60000, "providers": ["智谱"] },
7
- { "name": "ZAI 速率限制 (HTTP 200, code 1302)", "status_code": 200, "body_pattern": "\"error\".*\"code\"\\s*:\\s*\"1302\"", "retry_strategy": "exponential", "retry_delay_ms": 5000, "max_retries": 10, "max_delay_ms": 60000, "providers": ["智谱"] },
8
- { "name": "ZAI SSE 错误 (HTTP 200, code 500)", "status_code": 200, "body_pattern": "\"error\".*\"code\"\\s*:\\s*\"500\"", "retry_strategy": "exponential", "retry_delay_ms": 5000, "max_retries": 10, "max_delay_ms": 60000, "providers": ["智谱"] },
9
- { "name": "ZAI SSE 错误 (HTTP 200, code 1234)", "status_code": 200, "body_pattern": "\"error\".*\"code\"\\s*:\\s*\"1234\"", "retry_strategy": "exponential", "retry_delay_ms": 5000, "max_retries": 10, "max_delay_ms": 60000, "providers": ["智谱"] },
10
- { "name": "ZAI 模型过载 (HTTP 200, code 1305)", "status_code": 200, "body_pattern": "\"error\".*\"code\"\\s*:\\s*\"1305\"", "retry_strategy": "exponential", "retry_delay_ms": 5000, "max_retries": 10, "max_delay_ms": 60000, "providers": ["智谱"] },
11
- { "name": "KIMI 401 认证错误", "status_code": 401, "body_pattern": ".*authentication_error.*", "retry_strategy": "exponential", "retry_delay_ms": 5000, "max_retries": 3, "max_delay_ms": 60000, "providers": ["月之暗面"] },
12
- { "name": "DeepSeek 并发限流 (429)", "status_code": 429, "body_pattern": "Too many requests.*concurrency", "retry_strategy": "exponential", "retry_delay_ms": 2000, "max_retries": 5, "max_delay_ms": 120000, "providers": ["DeepSeek", "OpenCode"] }
2
+ {
3
+ "name": "429 Too Many Requests",
4
+ "status_code": 429,
5
+ "body_pattern": ".*",
6
+ "retry_strategy": "exponential",
7
+ "retry_delay_ms": 5000,
8
+ "max_retries": 10,
9
+ "max_delay_ms": 60000,
10
+ "providers": []
11
+ },
12
+ {
13
+ "name": "503 Service Unavailable",
14
+ "status_code": 503,
15
+ "body_pattern": ".*",
16
+ "retry_strategy": "exponential",
17
+ "retry_delay_ms": 5000,
18
+ "max_retries": 10,
19
+ "max_delay_ms": 60000,
20
+ "providers": []
21
+ },
22
+ {
23
+ "name": "ZAI 网络错误 (code 1234)",
24
+ "status_code": 400,
25
+ "body_pattern": "\"type\"\\s*:\\s*\"error\".*\"code\"\\s*:\\s*\"1234\"",
26
+ "retry_strategy": "exponential",
27
+ "retry_delay_ms": 5000,
28
+ "max_retries": 10,
29
+ "max_delay_ms": 60000,
30
+ "providers": [
31
+ "zhipu"
32
+ ]
33
+ },
34
+ {
35
+ "name": "ZAI 临时不可用",
36
+ "status_code": 400,
37
+ "body_pattern": "\"type\"\\s*:\\s*\"error\".*请稍后重试",
38
+ "retry_strategy": "exponential",
39
+ "retry_delay_ms": 5000,
40
+ "max_retries": 10,
41
+ "max_delay_ms": 60000,
42
+ "providers": [
43
+ "zhipu"
44
+ ]
45
+ },
46
+ {
47
+ "name": "ZAI 操作失败 (code 500)",
48
+ "status_code": 400,
49
+ "body_pattern": "\"type\"\\s*:\\s*\"error\".*\"code\"\\s*:\\s*\"500\"",
50
+ "retry_strategy": "exponential",
51
+ "retry_delay_ms": 5000,
52
+ "max_retries": 10,
53
+ "max_delay_ms": 60000,
54
+ "providers": [
55
+ "zhipu"
56
+ ]
57
+ },
58
+ {
59
+ "name": "ZAI 速率限制 (HTTP 200, code 1302)",
60
+ "status_code": 200,
61
+ "body_pattern": "\"error\".*\"code\"\\s*:\\s*\"1302\"",
62
+ "retry_strategy": "exponential",
63
+ "retry_delay_ms": 5000,
64
+ "max_retries": 10,
65
+ "max_delay_ms": 60000,
66
+ "providers": [
67
+ "zhipu"
68
+ ]
69
+ },
70
+ {
71
+ "name": "ZAI SSE 错误 (HTTP 200, code 500)",
72
+ "status_code": 200,
73
+ "body_pattern": "\"error\".*\"code\"\\s*:\\s*\"500\"",
74
+ "retry_strategy": "exponential",
75
+ "retry_delay_ms": 5000,
76
+ "max_retries": 10,
77
+ "max_delay_ms": 60000,
78
+ "providers": [
79
+ "zhipu"
80
+ ]
81
+ },
82
+ {
83
+ "name": "ZAI SSE 错误 (HTTP 200, code 1234)",
84
+ "status_code": 200,
85
+ "body_pattern": "\"error\".*\"code\"\\s*:\\s*\"1234\"",
86
+ "retry_strategy": "exponential",
87
+ "retry_delay_ms": 5000,
88
+ "max_retries": 10,
89
+ "max_delay_ms": 60000,
90
+ "providers": [
91
+ "zhipu"
92
+ ]
93
+ },
94
+ {
95
+ "name": "ZAI 模型过载 (HTTP 200, code 1305)",
96
+ "status_code": 200,
97
+ "body_pattern": "\"error\".*\"code\"\\s*:\\s*\"1305\"",
98
+ "retry_strategy": "exponential",
99
+ "retry_delay_ms": 5000,
100
+ "max_retries": 10,
101
+ "max_delay_ms": 60000,
102
+ "providers": [
103
+ "zhipu"
104
+ ]
105
+ },
106
+ {
107
+ "name": "KIMI 401 认证错误",
108
+ "status_code": 401,
109
+ "body_pattern": ".*authentication_error.*",
110
+ "retry_strategy": "exponential",
111
+ "retry_delay_ms": 5000,
112
+ "max_retries": 3,
113
+ "max_delay_ms": 60000,
114
+ "providers": [
115
+ "moonshot"
116
+ ]
117
+ },
118
+ {
119
+ "name": "DeepSeek 并发限流 (429)",
120
+ "status_code": 429,
121
+ "body_pattern": "Too many requests.*concurrency",
122
+ "retry_strategy": "exponential",
123
+ "retry_delay_ms": 2000,
124
+ "max_retries": 5,
125
+ "max_delay_ms": 120000,
126
+ "providers": [
127
+ "opencode"
128
+ ]
129
+ }
13
130
  ]
@@ -14,6 +14,7 @@ export interface ProviderPreset {
14
14
  }
15
15
  export interface ProviderGroup {
16
16
  group: string;
17
+ shortname: string;
17
18
  presets: ProviderPreset[];
18
19
  }
19
20
  export interface RecommendedRetryRule {
@@ -18,6 +18,8 @@ export declare class AdaptiveController {
18
18
  onRequestComplete(providerId: string, result: AdaptiveResult): void;
19
19
  getStatus(providerId: string): AdaptiveState | undefined;
20
20
  syncProvider(providerId: string, p: ProviderConcurrencyParams): void;
21
+ /** 将 max_concurrency 钳制到 >= ADAPTIVE_MIN,防止 NaN/undefined 泄漏到 deriveProfile */
22
+ private clampMax;
21
23
  /** 根据当前位置和容量推导行为参数,实现水位梯度控制 */
22
24
  private deriveProfile;
23
25
  private transitionSuccess;
@@ -9,10 +9,10 @@ const CLIMB_LEVEL_WEIGHT = 2;
9
9
  const DROP_BASE = 5;
10
10
  const DROP_CAPACITY_WEIGHT = 2;
11
11
  const DROP_LEVEL_WEIGHT = 2;
12
- const KEEP_RATIO_MIN = 0.5;
13
12
  const COOLDOWN_BASE_MS = 10_000;
14
13
  const COOLDOWN_LEVEL_MS = 10_000;
15
- const SAFE_ZONE_DIVISOR = 2;
14
+ /** 满额时保留半数成功计数器的除数 */
15
+ const AT_MAX_COUNTER_HALVE_DIVISOR = 2;
16
16
  export class AdaptiveController {
17
17
  semaphoreControl;
18
18
  logger;
@@ -22,16 +22,15 @@ export class AdaptiveController {
22
22
  this.logger = logger;
23
23
  }
24
24
  init(providerId, config, semParams) {
25
- const initialLimit = config.max;
25
+ const max = this.clampMax(config.max);
26
26
  this.entries.set(providerId, {
27
27
  state: {
28
- currentLimit: initialLimit,
29
- limitReached: false,
28
+ currentLimit: max,
30
29
  consecutiveSuccesses: 0,
31
30
  consecutiveFailures: 0,
32
31
  cooldownUntil: 0,
33
32
  },
34
- max: config.max,
33
+ max,
35
34
  queueTimeoutMs: semParams.queueTimeoutMs,
36
35
  maxQueueSize: semParams.maxQueueSize,
37
36
  });
@@ -63,10 +62,11 @@ export class AdaptiveController {
63
62
  if (p.adaptive_enabled) {
64
63
  const existing = this.entries.get(providerId);
65
64
  if (existing) {
66
- existing.max = p.max_concurrency;
65
+ const max = this.clampMax(p.max_concurrency);
66
+ existing.max = max;
67
67
  existing.queueTimeoutMs = p.queue_timeout_ms;
68
68
  existing.maxQueueSize = p.max_queue_size;
69
- existing.state.currentLimit = Math.min(Math.max(existing.state.currentLimit, ADAPTIVE_MIN), existing.max);
69
+ existing.state.currentLimit = Math.min(Math.max(existing.state.currentLimit, ADAPTIVE_MIN), max);
70
70
  this.syncToSemaphore(providerId);
71
71
  }
72
72
  else {
@@ -85,6 +85,11 @@ export class AdaptiveController {
85
85
  });
86
86
  }
87
87
  }
88
+ /** 将 max_concurrency 钳制到 >= ADAPTIVE_MIN,防止 NaN/undefined 泄漏到 deriveProfile */
89
+ clampMax(value) {
90
+ const n = Number(value);
91
+ return Number.isFinite(n) && n >= ADAPTIVE_MIN ? n : ADAPTIVE_MIN;
92
+ }
88
93
  /** 根据当前位置和容量推导行为参数,实现水位梯度控制 */
89
94
  deriveProfile(currentLimit, max) {
90
95
  const level = Math.min(1, currentLimit / max);
@@ -92,33 +97,27 @@ export class AdaptiveController {
92
97
  return {
93
98
  climbThreshold: Math.max(CLIMB_BASE, Math.round(CLIMB_BASE + capacity * CLIMB_CAPACITY_WEIGHT + level * CLIMB_LEVEL_WEIGHT)),
94
99
  dropThreshold: Math.max(1, Math.round(DROP_BASE - capacity * DROP_CAPACITY_WEIGHT - level * DROP_LEVEL_WEIGHT)),
95
- keepRatio: currentLimit > 1 ? 1 - 1 / currentLimit : KEEP_RATIO_MIN,
96
100
  cooldownMs: Math.round(COOLDOWN_BASE_MS + level * COOLDOWN_LEVEL_MS),
97
101
  };
98
102
  }
99
103
  transitionSuccess(providerId, entry, result) {
100
104
  const s = entry.state;
101
- // 冷却期内不累计成功计数
102
- if (Date.now() < s.cooldownUntil)
103
- return;
105
+ // V3: 冷却期不阻止成功累积(翻转语义:冷却期只阻止下降)
104
106
  s.consecutiveSuccesses++;
105
107
  s.consecutiveFailures = 0;
106
- // 利用率信号:请求排过队说明 limit 被实际触及
107
- if (result.wasQueued) {
108
- s.limitReached = true;
109
- }
110
108
  const profile = this.deriveProfile(s.currentLimit, entry.max);
111
109
  if (s.consecutiveSuccesses >= profile.climbThreshold) {
112
- // 利用率门控:安全区(limit <= max/2) 或 limitReached 才爬升
113
- const safeZone = s.currentLimit <= Math.floor(entry.max / SAFE_ZONE_DIVISOR);
114
- if (safeZone || s.limitReached) {
115
- const prevLimit = s.currentLimit;
116
- s.currentLimit = Math.min(s.currentLimit + 1, entry.max);
110
+ const prevLimit = s.currentLimit;
111
+ s.currentLimit = Math.min(s.currentLimit + 1, entry.max);
112
+ // 满额时保留半数计数器,避免反复从 0 开始爬升
113
+ if (s.currentLimit === entry.max) {
114
+ s.consecutiveSuccesses = Math.floor(s.consecutiveSuccesses / AT_MAX_COUNTER_HALVE_DIVISOR);
115
+ this.logger?.debug?.({ providerId, prevLimit, newLimit: s.currentLimit, action: "at_max_counter_cycle" }, "Adaptive: at max, counters preserved");
116
+ }
117
+ else {
118
+ s.consecutiveSuccesses = 0;
117
119
  this.logger?.info?.({ providerId, requestId: result.requestId, prevLimit, newLimit: s.currentLimit, action: "limit_increased" }, "Adaptive: limit increased by 1");
118
120
  }
119
- // 无论是否爬升,都重置计数周期
120
- s.consecutiveSuccesses = 0;
121
- s.limitReached = false;
122
121
  this.syncToSemaphore(providerId);
123
122
  }
124
123
  }
@@ -135,24 +134,30 @@ export class AdaptiveController {
135
134
  return;
136
135
  }
137
136
  const s = entry.state;
137
+ // V3: 冷却期前置检查——阻止进一步下降,但不重置成功计数
138
+ if (Date.now() < s.cooldownUntil) {
139
+ this.logger?.debug?.({ providerId, statusCode, action: "failure_blocked_by_cooldown" }, "Adaptive: failure blocked by cooldown");
140
+ return;
141
+ }
138
142
  s.consecutiveFailures++;
139
143
  s.consecutiveSuccesses = 0;
140
144
  if (statusCode === RATE_LIMIT_STATUS) {
141
- // 429 和信号量错误:丢 1 + 冷却
145
+ // 429: 固定 -1 + 冷却期
142
146
  const profile = this.deriveProfile(s.currentLimit, entry.max);
143
147
  const prevLimit = s.currentLimit;
144
- s.currentLimit = Math.max(Math.floor(s.currentLimit * profile.keepRatio), ADAPTIVE_MIN);
148
+ s.currentLimit = Math.max(s.currentLimit - 1, ADAPTIVE_MIN);
145
149
  s.cooldownUntil = Date.now() + profile.cooldownMs;
146
150
  s.consecutiveFailures = 0;
147
151
  this.syncToSemaphore(providerId);
148
152
  this.logger?.warn?.({ providerId, requestId: result.requestId, prevLimit, newLimit: s.currentLimit, cooldownMs: profile.cooldownMs, statusCode, action: "rate_limit_backoff" }, "Adaptive: 429/semaphore, lost 1 slot and entered cooldown");
149
153
  }
150
154
  else {
151
- // 5xx / 网络错误(statusCode=undefined):连续失败退避
155
+ // 5xx / 网络错误(statusCode=undefined):连续失败退避 + 冷却期
152
156
  const profile = this.deriveProfile(s.currentLimit, entry.max);
153
157
  if (s.consecutiveFailures >= profile.dropThreshold) {
154
158
  const prevLimit = s.currentLimit;
155
159
  s.currentLimit = Math.max(s.currentLimit - 1, ADAPTIVE_MIN);
160
+ s.cooldownUntil = Date.now() + profile.cooldownMs;
156
161
  s.consecutiveFailures = 0;
157
162
  this.syncToSemaphore(providerId);
158
163
  this.logger?.warn?.({ providerId, requestId: result.requestId, prevLimit, newLimit: s.currentLimit, statusCode, retryRuleMatched: result.retryRuleMatched ?? false, action: "failure_backoff" }, "Adaptive: sustained failures, decreased concurrency");
@@ -7,7 +7,6 @@ export interface ConcurrencyConfig {
7
7
  /** Internal state of adaptive concurrency for a provider. */
8
8
  export interface AdaptiveState {
9
9
  currentLimit: number;
10
- limitReached: boolean;
11
10
  consecutiveSuccesses: number;
12
11
  consecutiveFailures: number;
13
12
  cooldownUntil: number;
@@ -47,6 +47,8 @@ export interface AttemptSnapshot {
47
47
  error: string | null;
48
48
  latencyMs: number;
49
49
  providerId: string;
50
+ model?: string;
51
+ apiType?: string;
50
52
  }
51
53
  export interface StreamMetricsSnapshot {
52
54
  inputTokens: number | null;
package/dist/db/index.js CHANGED
@@ -137,6 +137,48 @@ function runApplicationMigrations(db) {
137
137
  }
138
138
  db.prepare("INSERT OR REPLACE INTO settings (key, value) VALUES (?, ?)").run(markerKey, "done");
139
139
  })();
140
+ // 050: patches 名称标准化 — 连字符 → 下划线 + 去重
141
+ const patchMarker = "app_migration_050_patch_name_normalize";
142
+ const patchDone = db.prepare("SELECT value FROM settings WHERE key = ?").get(patchMarker);
143
+ if (!patchDone) {
144
+ const providers = db.prepare("SELECT id, models FROM providers").all();
145
+ const update = db.prepare("UPDATE providers SET models = ? WHERE id = ?");
146
+ db.transaction(() => {
147
+ let patched = 0;
148
+ for (const p of providers) {
149
+ try {
150
+ // eslint-disable-next-line taste/no-raw-json-parse-models -- 迁移代码需要操作原始 JSON 结构
151
+ const raw = JSON.parse(p.models);
152
+ if (!Array.isArray(raw))
153
+ continue;
154
+ let changed = false;
155
+ for (const entry of raw) {
156
+ if (typeof entry !== "object" || entry === null)
157
+ continue;
158
+ const patches = entry.patches;
159
+ if (!Array.isArray(patches))
160
+ continue;
161
+ const normalized = patches.map((v) => typeof v === "string" ? v.replace(/-/g, "_") : v);
162
+ const deduped = [...new Set(normalized)];
163
+ if (deduped.length !== patches.length ||
164
+ deduped.some((v, i) => v !== patches[i])) {
165
+ entry.patches = deduped;
166
+ changed = true;
167
+ }
168
+ }
169
+ if (changed) {
170
+ update.run(JSON.stringify(raw), p.id);
171
+ patched++;
172
+ }
173
+ }
174
+ catch { /* JSON parse failed — skip */ } // eslint-disable-line taste/no-silent-catch
175
+ }
176
+ if (patched > 0) {
177
+ console.log(`[migration-050] Normalized patch names in ${patched} provider(s)`);
178
+ }
179
+ db.prepare("INSERT OR REPLACE INTO settings (key, value) VALUES (?, ?)").run(patchMarker, "done");
180
+ })();
181
+ }
140
182
  }
141
183
  // --- Re-export from per-table modules ---
142
184
  export { getActiveProviders, getAllProviders, getProviderById, getActiveProviderByName, getActiveProvidersWithModels, createProvider, updateProvider, deleteProvider, PROVIDER_CONCURRENCY_DEFAULTS, } from "./providers.js";
@@ -83,6 +83,8 @@ export class ProxyOrchestrator {
83
83
  error: a.error,
84
84
  latencyMs: a.latencyMs,
85
85
  providerId: a.target.provider_id,
86
+ model: a.target.backend_model,
87
+ apiType: apiType,
86
88
  })));
87
89
  const { status, statusCode } = this.extractTrackStatus(result);
88
90
  // 如果有重试尝试(非 throw 类型),说明 resilience 层的重试规则匹配了,
@@ -0,0 +1 @@
1
+ import{Gt as e,Q as t,Xt as n,Z as r,et as i,gt as a,it as o,kt as s,n as c,rt as l,yt as u}from"./button-DXlWUlLe.js";import{lt as d,r as f,ut as p}from"./index-FYAc4GSw.js";import{t as m}from"./Card-_-Rsez1Z.js";import{t as h}from"./CardContent-CXB0aTQX.js";var g={class:`min-h-screen flex items-center justify-center bg-background relative`},_={class:`text-center mb-6`},v={class:`text-sm text-muted-foreground mt-1`},y=o({__name:`AuthLayout`,props:{subtitle:{}},setup(o){let{isDark:y,toggleTheme:b}=f();return(f,x)=>(a(),i(`div`,g,[l(e(c),{variant:`ghost`,size:`icon`,class:`absolute top-4 right-4 text-muted-foreground hover:text-foreground`,onClick:e(b)},{default:s(()=>[e(y)?(a(),t(e(d),{key:1,class:`w-4 h-4`})):(a(),t(e(p),{key:0,class:`w-4 h-4`}))]),_:1},8,[`onClick`]),l(e(m),{class:`w-full max-w-sm shadow-lg`},{default:s(()=>[l(e(h),{class:`pt-6`},{default:s(()=>[r(`div`,_,[x[0]||=r(`div`,{class:`w-12 h-12 bg-primary rounded-lg mx-auto mb-3 flex items-center justify-center`},[r(`svg`,{class:`w-7 h-7 text-white`,fill:`none`,stroke:`currentColor`,viewBox:`0 0 24 24`},[r(`path`,{"stroke-linecap":`round`,"stroke-linejoin":`round`,"stroke-width":`2`,d:`M8 9l3 3-3 3m5 0h3M5 20h14a2 2 0 002-2V6a2 2 0 00-2-2H5a2 2 0 00-2 2v12a2 2 0 002 2z`})])],-1),x[1]||=r(`h1`,{class:`text-xl font-semibold text-foreground`},` LLM Simple Router `,-1),r(`p`,v,n(o.subtitle),1)]),u(f.$slots,`default`)]),_:3})]),_:3})]))}});export{y as t};
@@ -0,0 +1 @@
1
+ import{Gt as e,et as t,gt as n,it as r,qt as i,r as a,yt as o}from"./button-DXlWUlLe.js";var s=[`data-size`,`data-flush`],c=r({__name:`Card`,props:{class:{type:[Boolean,null,String,Object,Array]},size:{default:`default`},flush:{type:Boolean,default:!1}},setup(r){let c=r;return(l,u)=>(n(),t(`div`,{"data-slot":`card`,"data-size":r.size,"data-flush":r.flush||void 0,class:i(e(a)(`border-border bg-card text-card-foreground rounded-lg border text-sm group/card flex flex-col`,r.flush?`overflow-hidden`:`overflow-hidden gap-4 py-4 has-data-[slot=card-footer]:pb-0 has-[>img:first-child]:pt-0 *:[img:first-child]:rounded-t-lg *:[img:last-child]:rounded-b-lg`,`data-[size=sm]:gap-3 data-[size=sm]:py-3 data-[size=sm]:has-data-[slot=card-footer]:pb-0`,c.class))},[o(l.$slots,`default`)],10,s))}});export{c as t};
@@ -0,0 +1 @@
1
+ import{Gt as e,et as t,gt as n,it as r,qt as i,r as a,yt as o}from"./button-DXlWUlLe.js";var s=r({__name:`CardContent`,props:{class:{type:[Boolean,null,String,Object,Array]}},setup(r){let s=r;return(r,c)=>(n(),t(`div`,{"data-slot":`card-content`,class:i(e(a)(`px-4 group-data-[size=sm]/card:px-3`,s.class))},[o(r.$slots,`default`)],2))}});export{s as t};
@@ -1 +1 @@
1
- import{Gt as e,et as t,gt as n,it as r,qt as i,r as a,yt as o}from"./button-CiJBAzrS.js";var s=r({__name:`CardHeader`,props:{class:{type:[Boolean,null,String,Object,Array]}},setup(r){let s=r;return(r,c)=>(n(),t(`div`,{"data-slot":`card-header`,class:i(e(a)(`gap-1 rounded-t-xl px-4 group-data-[size=sm]/card:px-3 [.border-b]:pb-4 group-data-[size=sm]/card:[.border-b]:pb-3 group/card-header @container/card-header grid auto-rows-min items-start has-data-[slot=card-action]:grid-cols-[1fr_auto] has-data-[slot=card-description]:grid-rows-[auto_auto]`,s.class))},[o(r.$slots,`default`)],2))}}),c=r({__name:`CardTitle`,props:{class:{type:[Boolean,null,String,Object,Array]}},setup(r){let s=r;return(r,c)=>(n(),t(`div`,{"data-slot":`card-title`,class:i(e(a)(`text-base leading-snug font-medium group-data-[size=sm]/card:text-sm cn-font-heading`,s.class))},[o(r.$slots,`default`)],2))}});export{s as n,c as t};
1
+ import{Gt as e,et as t,gt as n,it as r,qt as i,r as a,yt as o}from"./button-DXlWUlLe.js";var s=r({__name:`CardHeader`,props:{class:{type:[Boolean,null,String,Object,Array]}},setup(r){let s=r;return(r,c)=>(n(),t(`div`,{"data-slot":`card-header`,class:i(e(a)(`gap-1 rounded-t-xl px-4 group-data-[size=sm]/card:px-3 [.border-b]:pb-4 group-data-[size=sm]/card:[.border-b]:pb-3 group/card-header @container/card-header grid auto-rows-min items-start has-data-[slot=card-action]:grid-cols-[1fr_auto] has-data-[slot=card-description]:grid-rows-[auto_auto]`,s.class))},[o(r.$slots,`default`)],2))}}),c=r({__name:`CardTitle`,props:{class:{type:[Boolean,null,String,Object,Array]}},setup(r){let s=r;return(r,c)=>(n(),t(`div`,{"data-slot":`card-title`,class:i(e(a)(`text-base leading-snug font-medium group-data-[size=sm]/card:text-sm cn-font-heading`,s.class))},[o(r.$slots,`default`)],2))}});export{s as n,c as t};
@@ -0,0 +1 @@
1
+ import{$ as e,Gt as t,H as n,J as r,Lt as i,Q as a,X as o,Xt as s,Z as c,et as l,gt as u,it as d,kt as f,qt as p,rt as m,vt as h}from"./button-DXlWUlLe.js";import{t as g}from"./chevron-right-DJE8HF-i.js";import{b as _,ht as v,vt as y,x as b,y as x}from"./index-FYAc4GSw.js";var S=[`onMouseenter`],C={class:`truncate max-w-40`},w={key:0,class:`ml-1 text-[10px] px-1 py-px rounded bg-emerald-500/15 text-emerald-400 shrink-0`},T=[`onMouseenter`],E=[`onClick`],D={class:`truncate`},O={key:0,class:`shrink-0 text-xs text-muted-foreground`},k={key:0,class:`px-2 py-1.5 text-sm text-muted-foreground`},A=d({__name:`CascadingSelect`,props:{groups:{},modelValue:{},placeholder:{default:``},compact:{type:Boolean,default:!1},dashed:{type:Boolean,default:!1}},emits:[`update:modelValue`],setup(d,{emit:y}){let{t:A}=n(),j=d,M=o(()=>j.placeholder||A(`common.selectPlaceholder`)),N=y,P=i(!1),F=i(null),I=o(()=>{if(!j.modelValue)return``;let e=j.groups.find(e=>e.key===j.modelValue.groupKey);if(!e)return``;let t=e.options.find(e=>e.value===j.modelValue.value);return t?`${e.label} / ${t.label}`:``});function L(e,t){N(`update:modelValue`,{groupKey:e,value:t}),P.value=!1}function R(e){P.value=e,e||(F.value=null)}return(n,i)=>(u(),a(t(b),{open:P.value,"onUpdate:open":R},{default:f(()=>[m(t(x),{"as-child":``},{default:f(()=>[c(`div`,{class:p([`flex w-full items-center justify-between rounded-md bg-background ring-offset-background cursor-pointer hover:bg-accent hover:text-accent-foreground`,[d.compact?`h-7 text-xs px-2 py-1 rounded-md`:`h-10 text-sm px-3 py-2 border border-border rounded-md`,!d.compact&&d.dashed?`border-dashed border-primary/20`:``,{"ring-2 ring-ring ring-offset-2":P.value}]])},[c(`span`,{class:p([`truncate`,d.modelValue?`text-foreground`:`text-muted-foreground`])},s(I.value||M.value),3),m(t(v),{class:`h-4 w-4 shrink-0 opacity-50`})],2)]),_:1}),m(t(_),{align:`start`,"side-offset":4,class:`z-[200] w-auto min-w-56 overflow-visible p-1`},{default:f(()=>[(u(!0),l(r,null,h(d.groups,n=>(u(),l(`div`,{key:n.key,class:p([`relative flex cursor-pointer items-center justify-between rounded-sm px-2 py-1.5 text-sm hover:bg-accent hover:text-accent-foreground`,{"bg-accent text-accent-foreground z-10":F.value===n.key}]),onMouseenter:e=>F.value=n.key},[c(`span`,C,s(n.label),1),n.badge?(u(),l(`span`,w,s(n.badge),1)):e(``,!0),m(t(g),{class:`ml-1 h-4 w-4 shrink-0 opacity-50`}),F.value===n.key&&n.options.length>0?(u(),l(`div`,{key:1,class:`absolute left-full top-0 ml-0.5 min-w-48 rounded-md bg-popover p-1 text-popover-foreground shadow-md`,onMouseenter:e=>F.value=n.key},[(u(!0),l(r,null,h(n.options,t=>(u(),l(`div`,{key:t.value,class:p([`flex cursor-pointer items-center justify-between gap-2 rounded-sm px-2 py-1.5 text-sm hover:bg-accent hover:text-accent-foreground`,{"bg-accent text-accent-foreground":d.modelValue?.groupKey===n.key&&d.modelValue?.value===t.value}]),onClick:e=>L(n.key,t.value)},[c(`span`,D,s(t.label),1),t.tag?(u(),l(`span`,O,s(t.tag),1)):e(``,!0)],10,E))),128))],40,T)):e(``,!0)],42,S))),128)),d.groups.length===0?(u(),l(`div`,k,s(t(A)(`common.noOptions`)),1)):e(``,!0)]),_:1})]),_:1},8,[`open`]))}}),j=d({__name:`CascadingModelSelect`,props:{providers:{},modelValue:{},placeholder:{default:``},compact:{type:Boolean,default:!1},dashed:{type:Boolean,default:!1}},emits:[`update:modelValue`],setup(e,{emit:t}){let{t:r}=n(),i=e,s=o(()=>i.placeholder||r(`mappings.selectProviderModel`)),c=t,l=o(()=>i.providers.map(e=>({key:e.provider.id,label:e.provider.name,badge:e.isNew?r(`common.new`):void 0,options:e.models.map(e=>({value:e.name,label:e.name,tag:y(e.contextWindow)}))}))),d=o(()=>i.modelValue?{groupKey:i.modelValue.provider_id,value:i.modelValue.model}:void 0);function f(e){c(`update:modelValue`,{provider_id:e.groupKey,model:e.value})}return(t,n)=>(u(),a(A,{groups:l.value,"model-value":d.value,placeholder:s.value,compact:e.compact,dashed:e.dashed,"onUpdate:modelValue":f},null,8,[`groups`,`model-value`,`placeholder`,`compact`,`dashed`]))}});export{j as t};
@@ -1 +1 @@
1
- import{$ as e,Gt as t,Jt as n,K as r,Q as i,X as a,dt as o,gt as s,i as c,it as l,kt as u,m as d,o as f,ot as p,q as m,r as h,rt as g,x as _,xt as v,yt as y}from"./button-CiJBAzrS.js";import{t as b}from"./VisuallyHiddenInput-D7dv0owV.js";import{t as x}from"./RovingFocusItem-C31XRuKO.js";import{J as S,K as C,R as w,U as T,V as E,X as D,pt as O}from"./index-bv08h-oy.js";function k(e,t){return C(e)?!1:Array.isArray(e)?e.some(e=>D(e,t)):D(e,t)}var[A,j]=S(`CheckboxGroupRoot`);function M(e){return e===`indeterminate`}function N(e){return M(e)?`indeterminate`:e?`checked`:`unchecked`}var[P,F]=S(`CheckboxRoot`),I=l({inheritAttrs:!1,__name:`CheckboxRoot`,props:{defaultValue:{type:null,required:!1},modelValue:{type:null,required:!1,default:void 0},disabled:{type:Boolean,required:!1},value:{type:null,required:!1,default:`on`},id:{type:String,required:!1},trueValue:{type:null,required:!1,default:()=>!0},falseValue:{type:null,required:!1,default:()=>!1},asChild:{type:Boolean,required:!1},as:{type:null,required:!1,default:`button`},name:{type:String,required:!1},required:{type:Boolean,required:!1}},emits:[`update:modelValue`],setup(n,{emit:l}){let p=n,h=l,{forwardRef:g,currentElement:_}=f(),S=A(null),w=d(p,`modelValue`,h,{defaultValue:p.defaultValue??p.falseValue,passive:p.modelValue===void 0}),E=a(()=>S?.disabled.value||p.disabled),O=a(()=>D(w.value,p.trueValue)),j=a(()=>C(S?.modelValue.value)?w.value===`indeterminate`?`indeterminate`:O.value:k(S.modelValue.value,p.value));function P(){if(C(S?.modelValue.value))w.value===`indeterminate`?w.value=p.trueValue:w.value=O.value?p.falseValue:p.trueValue;else{let e=[...S.modelValue.value||[]];if(k(e,p.value)){let t=e.findIndex(e=>D(e,p.value));e.splice(t,1)}else e.push(p.value);S.modelValue.value=e}}let I=T(_),L=a(()=>p.id&&_.value?document.querySelector(`[for="${p.id}"]`)?.innerText:void 0);return F({disabled:E,state:j}),(n,a)=>(s(),i(v(t(S)?.rovingFocus.value?t(x):t(c)),o(n.$attrs,{id:n.id,ref:t(g),role:`checkbox`,"as-child":n.asChild,as:n.as,type:n.as===`button`?`button`:void 0,"aria-checked":t(M)(j.value)?`mixed`:j.value,"aria-required":n.required,"aria-label":n.$attrs[`aria-label`]||L.value,"data-state":t(N)(j.value),"data-disabled":E.value?``:void 0,disabled:E.value,focusable:t(S)?.rovingFocus.value?!E.value:void 0,onKeydown:r(m(()=>{},[`prevent`]),[`enter`]),onClick:P}),{default:u(()=>[y(n.$slots,`default`,{modelValue:t(w),state:j.value}),t(I)&&n.name&&!t(S)?(s(),i(t(b),{key:0,type:`checkbox`,checked:!!j.value,name:n.name,value:n.value,disabled:E.value,required:n.required},null,8,[`checked`,`name`,`value`,`disabled`,`required`])):e(`v-if`,!0)]),_:3},16,[`id`,`as-child`,`as`,`type`,`aria-checked`,`aria-required`,`aria-label`,`data-state`,`data-disabled`,`disabled`,`focusable`,`onKeydown`]))}}),L=l({__name:`CheckboxIndicator`,props:{forceMount:{type:Boolean,required:!1},asChild:{type:Boolean,required:!1},as:{type:null,required:!1,default:`span`}},setup(e){let{forwardRef:n}=f(),r=P();return(e,a)=>(s(),i(t(w),{present:e.forceMount||t(M)(t(r).state.value)||t(r).state.value===!0},{default:u(()=>[g(t(c),o({ref:t(n),"data-state":t(N)(t(r).state.value),"data-disabled":t(r).disabled.value?``:void 0,style:{pointerEvents:`none`},"as-child":e.asChild,as:e.as},e.$attrs),{default:u(()=>[y(e.$slots,`default`)]),_:3},16,[`data-state`,`data-disabled`,`as-child`,`as`])]),_:3},8,[`present`]))}}),R=l({__name:`Checkbox`,props:{defaultValue:{},modelValue:{},disabled:{type:Boolean},value:{},id:{},trueValue:{},falseValue:{},asChild:{type:Boolean},as:{},name:{},required:{type:Boolean},class:{type:[Boolean,null,String,Object,Array]}},emits:[`update:modelValue`],setup(e,{emit:r}){let a=e,c=r,l=E(_(a,`class`),c);return(e,r)=>(s(),i(t(I),o({"data-slot":`checkbox`},t(l),{class:t(h)(`border-input dark:bg-input/30 data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground dark:data-[state=checked]:bg-primary data-[state=checked]:border-primary aria-invalid:aria-checked:border-primary aria-invalid:border-destructive dark:aria-invalid:border-destructive/50 focus-visible:border-ring focus-visible:ring-ring/50 aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 flex size-4 items-center justify-center rounded-md border transition-colors group-has-disabled/field:opacity-50 focus-visible:ring-3 aria-invalid:ring-3 peer relative shrink-0 outline-none after:absolute after:-inset-x-3 after:-inset-y-2 disabled:cursor-not-allowed disabled:opacity-50`,a.class)}),{default:u(r=>[g(t(L),{"data-slot":`checkbox-indicator`,class:`[&>svg]:size-3.5 grid place-content-center text-current transition-none`},{default:u(()=>[y(e.$slots,`default`,n(p(r)),()=>[g(t(O))])]),_:2},1024)]),_:3},16,[`class`]))}});export{R as t};
1
+ import{$ as e,Gt as t,Jt as n,K as r,Q as i,X as a,dt as o,gt as s,i as c,it as l,kt as u,m as d,o as f,ot as p,q as m,r as h,rt as g,x as _,xt as v,yt as y}from"./button-DXlWUlLe.js";import{t as b}from"./VisuallyHiddenInput-Lv6BsAIS.js";import{t as x}from"./RovingFocusItem-CCbZ7_un.js";import{H as S,W as C,Y as w,Z as T,gt as E,q as D,z as O}from"./index-FYAc4GSw.js";function k(e,t){return D(e)?!1:Array.isArray(e)?e.some(e=>T(e,t)):T(e,t)}var[A,j]=w(`CheckboxGroupRoot`);function M(e){return e===`indeterminate`}function N(e){return M(e)?`indeterminate`:e?`checked`:`unchecked`}var[P,F]=w(`CheckboxRoot`),I=l({inheritAttrs:!1,__name:`CheckboxRoot`,props:{defaultValue:{type:null,required:!1},modelValue:{type:null,required:!1,default:void 0},disabled:{type:Boolean,required:!1},value:{type:null,required:!1,default:`on`},id:{type:String,required:!1},trueValue:{type:null,required:!1,default:()=>!0},falseValue:{type:null,required:!1,default:()=>!1},asChild:{type:Boolean,required:!1},as:{type:null,required:!1,default:`button`},name:{type:String,required:!1},required:{type:Boolean,required:!1}},emits:[`update:modelValue`],setup(n,{emit:l}){let p=n,h=l,{forwardRef:g,currentElement:_}=f(),S=A(null),w=d(p,`modelValue`,h,{defaultValue:p.defaultValue??p.falseValue,passive:p.modelValue===void 0}),E=a(()=>S?.disabled.value||p.disabled),O=a(()=>T(w.value,p.trueValue)),j=a(()=>D(S?.modelValue.value)?w.value===`indeterminate`?`indeterminate`:O.value:k(S.modelValue.value,p.value));function P(){if(D(S?.modelValue.value))w.value===`indeterminate`?w.value=p.trueValue:w.value=O.value?p.falseValue:p.trueValue;else{let e=[...S.modelValue.value||[]];if(k(e,p.value)){let t=e.findIndex(e=>T(e,p.value));e.splice(t,1)}else e.push(p.value);S.modelValue.value=e}}let I=C(_),L=a(()=>p.id&&_.value?document.querySelector(`[for="${p.id}"]`)?.innerText:void 0);return F({disabled:E,state:j}),(n,a)=>(s(),i(v(t(S)?.rovingFocus.value?t(x):t(c)),o(n.$attrs,{id:n.id,ref:t(g),role:`checkbox`,"as-child":n.asChild,as:n.as,type:n.as===`button`?`button`:void 0,"aria-checked":t(M)(j.value)?`mixed`:j.value,"aria-required":n.required,"aria-label":n.$attrs[`aria-label`]||L.value,"data-state":t(N)(j.value),"data-disabled":E.value?``:void 0,disabled:E.value,focusable:t(S)?.rovingFocus.value?!E.value:void 0,onKeydown:r(m(()=>{},[`prevent`]),[`enter`]),onClick:P}),{default:u(()=>[y(n.$slots,`default`,{modelValue:t(w),state:j.value}),t(I)&&n.name&&!t(S)?(s(),i(t(b),{key:0,type:`checkbox`,checked:!!j.value,name:n.name,value:n.value,disabled:E.value,required:n.required},null,8,[`checked`,`name`,`value`,`disabled`,`required`])):e(`v-if`,!0)]),_:3},16,[`id`,`as-child`,`as`,`type`,`aria-checked`,`aria-required`,`aria-label`,`data-state`,`data-disabled`,`disabled`,`focusable`,`onKeydown`]))}}),L=l({__name:`CheckboxIndicator`,props:{forceMount:{type:Boolean,required:!1},asChild:{type:Boolean,required:!1},as:{type:null,required:!1,default:`span`}},setup(e){let{forwardRef:n}=f(),r=P();return(e,a)=>(s(),i(t(O),{present:e.forceMount||t(M)(t(r).state.value)||t(r).state.value===!0},{default:u(()=>[g(t(c),o({ref:t(n),"data-state":t(N)(t(r).state.value),"data-disabled":t(r).disabled.value?``:void 0,style:{pointerEvents:`none`},"as-child":e.asChild,as:e.as},e.$attrs),{default:u(()=>[y(e.$slots,`default`)]),_:3},16,[`data-state`,`data-disabled`,`as-child`,`as`])]),_:3},8,[`present`]))}}),R=l({__name:`Checkbox`,props:{defaultValue:{},modelValue:{},disabled:{type:Boolean},value:{},id:{},trueValue:{},falseValue:{},asChild:{type:Boolean},as:{},name:{},required:{type:Boolean},class:{type:[Boolean,null,String,Object,Array]}},emits:[`update:modelValue`],setup(e,{emit:r}){let a=e,c=r,l=S(_(a,`class`),c);return(e,r)=>(s(),i(t(I),o({"data-slot":`checkbox`},t(l),{class:t(h)(`border-input dark:bg-input/30 data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground dark:data-[state=checked]:bg-primary data-[state=checked]:border-primary aria-invalid:aria-checked:border-primary aria-invalid:border-destructive dark:aria-invalid:border-destructive/50 focus-visible:border-ring focus-visible:ring-ring/50 aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 flex size-4 items-center justify-center rounded-md border transition-colors group-has-disabled/field:opacity-50 focus-visible:ring-3 aria-invalid:ring-3 peer relative shrink-0 outline-none after:absolute after:-inset-x-3 after:-inset-y-2 disabled:cursor-not-allowed disabled:opacity-50`,a.class)}),{default:u(r=>[g(t(L),{"data-slot":`checkbox-indicator`,class:`[&>svg]:size-3.5 grid place-content-center text-current transition-none`},{default:u(()=>[y(e.$slots,`default`,n(p(r)),()=>[g(t(E))])]),_:2},1024)]),_:3},16,[`class`]))}});export{R as t};
@@ -1 +1 @@
1
- import{$ as e,Et as t,Gt as n,Ht as r,Jt as i,Lt as a,Q as o,X as s,d as c,dt as l,ft as u,gt as d,i as f,it as p,kt as m,m as h,mt as g,o as _,ot as v,rt as y,yt as b}from"./button-CiJBAzrS.js";import{B as x,J as S,R as C,V as w}from"./index-bv08h-oy.js";var[T,E]=S(`CollapsibleRoot`),D=p({__name:`CollapsibleRoot`,props:{defaultOpen:{type:Boolean,required:!1,default:!1},open:{type:Boolean,required:!1,default:void 0},disabled:{type:Boolean,required:!1},unmountOnHide:{type:Boolean,required:!1,default:!0},asChild:{type:Boolean,required:!1},as:{type:null,required:!1}},emits:[`update:open`],setup(e,{expose:t,emit:i}){let a=e,s=h(a,`open`,i,{defaultValue:a.defaultOpen,passive:a.open===void 0}),{disabled:c,unmountOnHide:l}=r(a);return E({contentId:``,disabled:c,open:s,unmountOnHide:l,onOpenToggle:()=>{c.value||(s.value=!s.value)}}),t({open:s}),_(),(e,t)=>(d(),o(n(f),{as:e.as,"as-child":a.asChild,"data-state":n(s)?`open`:`closed`,"data-disabled":n(c)?``:void 0},{default:m(()=>[b(e.$slots,`default`,{open:n(s)})]),_:3},8,[`as`,`as-child`,`data-state`,`data-disabled`]))}}),O=p({inheritAttrs:!1,__name:`CollapsibleContent`,props:{forceMount:{type:Boolean,required:!1},asChild:{type:Boolean,required:!1},as:{type:null,required:!1}},emits:[`contentFound`],setup(r,{emit:i}){let p=r,h=i,v=T();v.contentId||=x(void 0,`reka-collapsible-content`);let S=a(),{forwardRef:w,currentElement:E}=_(),D=a(0),O=a(0),k=s(()=>v.open.value),A=a(k.value),j=a();t(()=>[k.value,S.value?.present],async()=>{await u();let e=E.value;if(!e)return;j.value=j.value||{transitionDuration:e.style.transitionDuration,animationName:e.style.animationName},e.style.transitionDuration=`0s`,e.style.animationName=`none`;let t=e.getBoundingClientRect();O.value=t.height,D.value=t.width,A.value||(e.style.transitionDuration=j.value.transitionDuration,e.style.animationName=j.value.animationName)},{immediate:!0});let M=s(()=>A.value&&v.open.value);return g(()=>{requestAnimationFrame(()=>{A.value=!1})}),c(E,`beforematch`,e=>{requestAnimationFrame(()=>{v.onOpenToggle(),h(`contentFound`)})}),(t,r)=>(d(),o(n(C),{ref_key:`presentRef`,ref:S,present:t.forceMount||n(v).open.value,"force-mount":!0},{default:m(({present:r})=>[y(n(f),l(t.$attrs,{id:n(v).contentId,ref:n(w),"as-child":p.asChild,as:t.as,hidden:r?void 0:n(v).unmountOnHide.value?``:`until-found`,"data-state":M.value?void 0:n(v).open.value?`open`:`closed`,"data-disabled":n(v).disabled?.value?``:void 0,style:{"--reka-collapsible-content-height":`${O.value}px`,"--reka-collapsible-content-width":`${D.value}px`}}),{default:m(()=>[!n(v).unmountOnHide.value||r?b(t.$slots,`default`,{key:0}):e(`v-if`,!0)]),_:2},1040,[`id`,`as-child`,`as`,`hidden`,`data-state`,`data-disabled`,`style`])]),_:3},8,[`present`]))}}),k=p({__name:`Collapsible`,props:{defaultOpen:{type:Boolean},open:{type:Boolean},disabled:{type:Boolean},unmountOnHide:{type:Boolean},asChild:{type:Boolean},as:{}},emits:[`update:open`],setup(e,{emit:t}){let r=w(e,t);return(e,t)=>(d(),o(n(D),l({"data-slot":`collapsible`},n(r)),{default:m(t=>[b(e.$slots,`default`,i(v(t)))]),_:3},16))}}),A=p({__name:`CollapsibleContent`,props:{forceMount:{type:Boolean},asChild:{type:Boolean},as:{}},setup(e){let t=e;return(e,r)=>(d(),o(n(O),l({"data-slot":`collapsible-content`},t),{default:m(()=>[b(e.$slots,`default`)]),_:3},16))}});export{k as n,T as r,A as t};
1
+ import{$ as e,Et as t,Gt as n,Ht as r,Jt as i,Lt as a,Q as o,X as s,d as c,dt as l,ft as u,gt as d,i as f,it as p,kt as m,m as h,mt as g,o as _,ot as v,rt as y,yt as b}from"./button-DXlWUlLe.js";import{H as x,V as S,Y as C,z as w}from"./index-FYAc4GSw.js";var[T,E]=C(`CollapsibleRoot`),D=p({__name:`CollapsibleRoot`,props:{defaultOpen:{type:Boolean,required:!1,default:!1},open:{type:Boolean,required:!1,default:void 0},disabled:{type:Boolean,required:!1},unmountOnHide:{type:Boolean,required:!1,default:!0},asChild:{type:Boolean,required:!1},as:{type:null,required:!1}},emits:[`update:open`],setup(e,{expose:t,emit:i}){let a=e,s=h(a,`open`,i,{defaultValue:a.defaultOpen,passive:a.open===void 0}),{disabled:c,unmountOnHide:l}=r(a);return E({contentId:``,disabled:c,open:s,unmountOnHide:l,onOpenToggle:()=>{c.value||(s.value=!s.value)}}),t({open:s}),_(),(e,t)=>(d(),o(n(f),{as:e.as,"as-child":a.asChild,"data-state":n(s)?`open`:`closed`,"data-disabled":n(c)?``:void 0},{default:m(()=>[b(e.$slots,`default`,{open:n(s)})]),_:3},8,[`as`,`as-child`,`data-state`,`data-disabled`]))}}),O=p({inheritAttrs:!1,__name:`CollapsibleContent`,props:{forceMount:{type:Boolean,required:!1},asChild:{type:Boolean,required:!1},as:{type:null,required:!1}},emits:[`contentFound`],setup(r,{emit:i}){let p=r,h=i,v=T();v.contentId||=S(void 0,`reka-collapsible-content`);let x=a(),{forwardRef:C,currentElement:E}=_(),D=a(0),O=a(0),k=s(()=>v.open.value),A=a(k.value),j=a();t(()=>[k.value,x.value?.present],async()=>{await u();let e=E.value;if(!e)return;j.value=j.value||{transitionDuration:e.style.transitionDuration,animationName:e.style.animationName},e.style.transitionDuration=`0s`,e.style.animationName=`none`;let t=e.getBoundingClientRect();O.value=t.height,D.value=t.width,A.value||(e.style.transitionDuration=j.value.transitionDuration,e.style.animationName=j.value.animationName)},{immediate:!0});let M=s(()=>A.value&&v.open.value);return g(()=>{requestAnimationFrame(()=>{A.value=!1})}),c(E,`beforematch`,e=>{requestAnimationFrame(()=>{v.onOpenToggle(),h(`contentFound`)})}),(t,r)=>(d(),o(n(w),{ref_key:`presentRef`,ref:x,present:t.forceMount||n(v).open.value,"force-mount":!0},{default:m(({present:r})=>[y(n(f),l(t.$attrs,{id:n(v).contentId,ref:n(C),"as-child":p.asChild,as:t.as,hidden:r?void 0:n(v).unmountOnHide.value?``:`until-found`,"data-state":M.value?void 0:n(v).open.value?`open`:`closed`,"data-disabled":n(v).disabled?.value?``:void 0,style:{"--reka-collapsible-content-height":`${O.value}px`,"--reka-collapsible-content-width":`${D.value}px`}}),{default:m(()=>[!n(v).unmountOnHide.value||r?b(t.$slots,`default`,{key:0}):e(`v-if`,!0)]),_:2},1040,[`id`,`as-child`,`as`,`hidden`,`data-state`,`data-disabled`,`style`])]),_:3},8,[`present`]))}}),k=p({__name:`Collapsible`,props:{defaultOpen:{type:Boolean},open:{type:Boolean},disabled:{type:Boolean},unmountOnHide:{type:Boolean},asChild:{type:Boolean},as:{}},emits:[`update:open`],setup(e,{emit:t}){let r=x(e,t);return(e,t)=>(d(),o(n(D),l({"data-slot":`collapsible`},n(r)),{default:m(t=>[b(e.$slots,`default`,i(v(t)))]),_:3},16))}}),A=p({__name:`CollapsibleContent`,props:{forceMount:{type:Boolean},asChild:{type:Boolean},as:{}},setup(e){let t=e;return(e,r)=>(d(),o(n(O),l({"data-slot":`collapsible-content`},t),{default:m(()=>[b(e.$slots,`default`)]),_:3},16))}});export{k as n,T as r,A as t};
@@ -0,0 +1 @@
1
+ import{Gt as e,Q as t,dt as n,gt as r,i,it as a,kt as o,o as s,yt as c}from"./button-DXlWUlLe.js";import{r as l}from"./CollapsibleContent-DDzm_Bxn.js";var u=a({__name:`CollapsibleTrigger`,props:{asChild:{type:Boolean,required:!1},as:{type:null,required:!1,default:`button`}},setup(n){let a=n;s();let u=l();return(n,s)=>(r(),t(e(i),{type:n.as===`button`?`button`:void 0,as:n.as,"as-child":a.asChild,"aria-controls":e(u).contentId,"aria-expanded":e(u).open.value,"data-state":e(u).open.value?`open`:`closed`,"data-disabled":e(u).disabled?.value?``:void 0,disabled:e(u).disabled?.value,onClick:e(u).onOpenToggle},{default:o(()=>[c(n.$slots,`default`)]),_:3},8,[`type`,`as`,`as-child`,`aria-controls`,`aria-expanded`,`data-state`,`data-disabled`,`disabled`,`onClick`]))}}),d=a({__name:`CollapsibleTrigger`,props:{asChild:{type:Boolean},as:{}},setup(i){let a=i;return(i,s)=>(r(),t(e(u),n({"data-slot":`collapsible-trigger`},a),{default:o(()=>[c(i.$slots,`default`)]),_:3},16))}});export{d as t};
@@ -0,0 +1 @@
1
+ import{$ as e,Gt as t,H as n,J as r,K as i,L as a,Lt as o,Q as s,X as c,Xt as l,Z as u,et as d,gt as f,it as p,kt as m,n as h,nt as g,q as _,qt as v,rt as y,vt as b}from"./button-DXlWUlLe.js";import{t as x}from"./plus-DghQKiGV.js";import{t as S}from"./trash-2-BFS2nrZH.js";import{t as C}from"./Label-CgLYbqNp.js";import{t as ee}from"./Switch-D2xEGYqH.js";import{_ as w,ct as te,g as T,h as E,ht as ne,m as D,ot as re,v as O}from"./index-FYAc4GSw.js";import{t as ie}from"./badge-C82CIl1p.js";import{t as k}from"./Input-Qz28j7do.js";import{t as A}from"./CascadingModelSelect-vGUnj1X-.js";import{r as j}from"./transform-domain-BGh_oMvU.js";var M=a(`arrow-right`,[[`path`,{d:`M5 12h14`,key:`1ays0h`}],[`path`,{d:`m12 5 7 7-7 7`,key:`xquz4c`}]]),N=a(`grid-3x3`,[[`rect`,{width:`18`,height:`18`,x:`3`,y:`3`,rx:`2`,key:`afitv7`}],[`path`,{d:`M3 9h18`,key:`1pudct`}],[`path`,{d:`M3 15h18`,key:`5xshup`}],[`path`,{d:`M9 3v18`,key:`fh3hqa`}],[`path`,{d:`M15 3v18`,key:`14nvp0`}]]),ae=[`title`],oe={class:`flex items-center gap-1 min-w-0 flex-1 flex-wrap`},se={key:0,class:`inline-flex items-center gap-1 px-1.5 py-0.5 rounded text-xs font-mono bg-primary/10 border border-border text-primary whitespace-nowrap`},P={key:0,class:`text-[10px] opacity-60`},F={class:`inline-flex items-center gap-1 px-1.5 py-0.5 rounded text-xs font-mono bg-muted/30 border border-border text-muted-foreground whitespace-nowrap`},I={class:`text-[10px] font-semibold`},L={class:`inline-flex items-center gap-1 px-1.5 py-0.5 rounded text-xs font-mono border border-dashed border-border text-primary/60 whitespace-nowrap`},R={class:`inline-flex items-center gap-1 px-1.5 py-0.5 rounded text-xs font-mono border border-dashed border-border text-blue-500/60 whitespace-nowrap`},z={class:`text-[10px] text-muted-foreground/30 hidden sm:inline shrink-0 whitespace-nowrap`},B={key:1,class:`space-y-2`},V={class:`flex items-center gap-2`},H={class:`text-[10px] text-muted-foreground/60 shrink-0 w-10 uppercase tracking-wider font-medium`},U={key:1,class:`flex-1 flex items-center gap-1.5`},W={class:`pt-0.5`},G={class:`flex items-center gap-1 mb-1`},K={class:`text-[10px] font-medium text-muted-foreground/70 uppercase tracking-wider`},ce={class:`flex items-center gap-2`},le={class:`flex-1`},ue={key:0,class:`flex items-center gap-1 pl-7 py-0.5`},de={class:`text-[9px] text-orange-400/50`},fe={class:`pt-2 mt-1 border-t border-dashed border-primary/15`},pe={class:`flex items-center gap-1.5 mb-1`},me={class:`text-[10px] font-medium text-primary/60 uppercase tracking-wider`},he={class:`text-[9px] text-primary/40 hidden sm:inline`},ge={key:0,class:`flex items-center gap-2`},_e={class:`flex-1`},ve={class:`pt-2 mt-1 border-t border-dashed border-blue-500/15`},ye={class:`flex items-center gap-1.5 mb-1`},be={class:`text-[10px] font-medium text-blue-500/60 uppercase tracking-wider`},xe={key:0,class:`flex items-center gap-2`},Se={class:`text-xs shrink-0 w-5 text-center px-0.5 py-0.5 rounded bg-blue-500/6 text-blue-500/60`},Ce={class:`flex-1`},we={key:2,class:`mt-2 px-2.5 py-2 rounded border-[1px] border-orange-400/25 bg-orange-400/5 flex gap-2`},Te={class:`text-[10px] leading-relaxed space-y-0.5`},Ee={class:`text-orange-400 font-medium`},De={class:`text-muted-foreground/60`},Oe={class:`text-muted-foreground/40`},ke=6,q=p({__name:`MappingEntryEditor`,props:{entry:{},providerGroups:{},expanded:{type:Boolean},editableClientModel:{type:Boolean,default:!1}},emits:[`update:targets`,`update:clientModel`,`update:multimodalFallback`,`toggle-active`,`remove`,`expand`],setup(a,{emit:p}){let{t:C}=n(),w=a,T=p,E=o(!1),D=o(``);function O(){w.editableClientModel&&(D.value=w.entry.clientModel,E.value=!0)}function j(){let e=D.value.trim();e&&e!==w.entry.clientModel&&T(`update:clientModel`,e),E.value=!1}function q(){E.value=!1}function J(e){return e===`__new__`?null:w.providerGroups.find(t=>t.provider.id===e)?.provider.name??e.slice(0,ke)}function Y(){let e=w.providerGroups[0];T(`update:targets`,[...w.entry.targets,{backend_model:e?.models[0]?.name??``,provider_id:e?.provider.id??``}])}function Ae(e){w.entry.targets.length<=1||T(`update:targets`,w.entry.targets.filter((t,n)=>n!==e))}function je(e,t){let n=[...w.entry.targets];n[e]={...n[e],provider_id:t.provider_id,backend_model:t.model},T(`update:targets`,n)}function Me(){let e=w.providerGroups[0];T(`update:targets`,w.entry.targets.map((t,n)=>n===0?{...t,overflow_provider_id:e?.provider.id??``,overflow_model:e?.models[0]?.name??``}:t))}function X(e){T(`update:targets`,w.entry.targets.map((t,n)=>{if(n===0){if(e)return{...t,overflow_provider_id:e.provider_id,overflow_model:e.model};let{overflow_provider_id:n,overflow_model:r,...i}=t;return i}return t}))}function Z(){let e=w.providerGroups[0];T(`update:multimodalFallback`,{provider_id:e?.provider.id??``,backend_model:``})}function Ne(e){T(`update:multimodalFallback`,{provider_id:e.provider_id,backend_model:e.model})}function Pe(){T(`update:multimodalFallback`,void 0)}let Fe=c(()=>{switch(w.entry.tag){case`def`:return`bg-primary/15 text-primary`;case`auto`:return`bg-green-500/15 text-green-500`;case`cust`:return`bg-blue-500/15 text-blue-500`;case`existing`:return`bg-muted/40 text-muted-foreground`;default:return`bg-muted/40 text-muted-foreground`}}),Ie=c(()=>{switch(w.entry.tag){case`def`:return C(`providers.shared.tagDefault`);case`auto`:return C(`providers.shared.tagAuto`);case`cust`:return C(`providers.shared.tagCustom`);case`existing`:return C(`providers.shared.tagExisting`);default:return``}}),Q=c(()=>!!w.entry.targets[0]?.overflow_model),$=c(()=>!!w.entry.multimodalFallback?.backend_model);return(n,o)=>(f(),d(`div`,null,[a.expanded?(f(),d(`div`,B,[u(`div`,V,[u(`span`,H,l(t(C)(`mappings.editor.clientLabel`)),1),E.value?(f(),d(`div`,{key:0,class:`flex-1 flex items-center gap-1`,onClick:o[5]||=_(()=>{},[`stop`])},[y(t(k),{modelValue:D.value,"onUpdate:modelValue":o[4]||=e=>D.value=e,class:`h-7 flex-1 text-xs font-mono border-[1px] border-border`,placeholder:t(C)(`mappings.editor.clientInputPlaceholder`),onKeydown:[i(_(j,[`prevent`]),[`enter`]),i(_(q,[`prevent`]),[`escape`])],onBlur:j,autofocus:``},null,8,[`modelValue`,`placeholder`,`onKeydown`])])):(f(),d(`div`,U,[u(`span`,{class:v([`font-mono text-xs font-semibold text-foreground`,{"cursor-pointer hover:text-primary transition-colors":a.editableClientModel}]),onClick:O},l(a.entry.clientModel),3),a.editableClientModel?(f(),s(t(h),{key:0,variant:`ghost`,size:`xs`,class:`text-[10px] text-muted-foreground/40`,onClick:_(O,[`stop`])},{default:m(()=>[g(l(t(C)(`mappings.editor.edit`)),1)]),_:1})):e(``,!0)]))]),u(`div`,W,[u(`div`,G,[y(t(re),{class:`size-3 text-muted-foreground/50`}),u(`span`,K,l(t(C)(`mappings.editor.failoverChain`)),1)]),(f(!0),d(r,null,b(a.entry.targets,(n,r)=>(f(),d(`div`,{key:r,class:`space-y-0`},[u(`div`,ce,[u(`span`,{class:v([`text-xs font-medium shrink-0 w-5 text-center px-0.5 py-0.5 rounded`,r===0?`bg-primary/10 text-primary`:`bg-muted/40 text-muted-foreground`])},l(r===0?`①`:r===1?`②`:r===2?`③`:`${r+1}`),3),u(`div`,le,[y(A,{providers:a.providerGroups,"model-value":{provider_id:n.provider_id,model:n.backend_model},compact:``,placeholder:t(C)(`providers.shared.selectModel`),"onUpdate:modelValue":e=>je(r,e)},null,8,[`providers`,`model-value`,`placeholder`,`onUpdate:modelValue`])]),r>0&&a.entry.targets.length>1?(f(),s(t(h),{key:0,variant:`ghost`,size:`icon-xs`,class:`shrink-0 text-muted-foreground/30 hover:text-destructive`,onClick:e=>Ae(r)},{default:m(()=>[y(t(S),{class:`size-3`})]),_:1},8,[`onClick`])):e(``,!0)]),r<a.entry.targets.length-1?(f(),d(`div`,ue,[o[13]||=u(`div`,{class:`w-px h-1.5 bg-orange-400/30`},null,-1),u(`span`,de,l(t(C)(`mappings.editor.failoverOnError`)),1)])):e(``,!0)]))),128)),y(t(h),{variant:`ghost`,class:`flex items-center justify-center gap-1 w-full py-1 mt-0.5 text-xs text-muted-foreground/40 border-[1px] border-dashed border-border rounded hover:text-primary hover:border-primary/40`,onClick:Y},{default:m(()=>[y(t(x),{class:`size-3`}),g(` `+l(t(C)(`mappings.addBackup`)),1)]),_:1})]),u(`div`,fe,[u(`div`,pe,[y(t(ne),{class:`size-3 text-primary/50`}),u(`span`,me,l(t(C)(`mappings.editor.contextOverflow`)),1),u(`span`,he,l(t(C)(`mappings.editor.contextOverflowHint`)),1)]),Q.value?(f(),d(`div`,ge,[o[14]||=u(`span`,{class:`text-xs shrink-0 w-5 text-center px-0.5 py-0.5 rounded bg-primary/6 text-primary/60`},`↓`,-1),u(`div`,_e,[y(A,{providers:a.providerGroups,"model-value":a.entry.targets[0]?.overflow_provider_id&&a.entry.targets[0]?.overflow_model?{provider_id:a.entry.targets[0].overflow_provider_id,model:a.entry.targets[0].overflow_model}:void 0,compact:``,dashed:``,placeholder:t(C)(`providers.shared.overflowPlaceholder`),"onUpdate:modelValue":o[6]||=e=>X(e)},null,8,[`providers`,`model-value`,`placeholder`])]),y(t(h),{variant:`ghost`,size:`icon-xs`,class:`shrink-0 text-muted-foreground/30 hover:text-destructive`,onClick:o[7]||=e=>X(void 0)},{default:m(()=>[y(t(S),{class:`size-3`})]),_:1})])):(f(),s(t(h),{key:1,variant:`ghost`,size:`xs`,class:`text-xs text-muted-foreground/50 hover:text-primary/60`,onClick:Me},{default:m(()=>[y(t(x),{class:`size-3`}),g(` `+l(t(C)(`providers.shared.addOverflow`)),1)]),_:1}))]),u(`div`,ve,[u(`div`,ye,[y(t(N),{class:`size-3 text-blue-500/50`}),u(`span`,be,l(t(C)(`mappings.editor.multimodalFallback`)),1),$.value?(f(),s(t(ie),{key:0,variant:`outline`,class:`text-[9px] px-1.5 py-0 leading-none text-primary/60 border-[1px] border-primary/20`},{default:m(()=>[g(l(t(C)(`mappings.multimodalFallback.configured`)),1)]),_:1})):e(``,!0)]),$.value?(f(),d(`div`,xe,[u(`span`,Se,[y(t(N),{class:`size-2.5`})]),u(`div`,Ce,[y(A,{providers:a.providerGroups,"model-value":{provider_id:a.entry.multimodalFallback.provider_id,model:a.entry.multimodalFallback.backend_model},compact:``,dashed:``,placeholder:t(C)(`mappings.multimodalFallback.selectProviderModel`),"onUpdate:modelValue":o[8]||=e=>Ne(e)},null,8,[`providers`,`model-value`,`placeholder`])]),y(t(h),{variant:`ghost`,size:`icon-xs`,class:`shrink-0 text-muted-foreground/30 hover:text-destructive`,onClick:Pe},{default:m(()=>[y(t(S),{class:`size-3`})]),_:1})])):(f(),s(t(h),{key:1,variant:`ghost`,size:`xs`,class:`text-xs text-muted-foreground/50 hover:text-primary/60`,onClick:Z},{default:m(()=>[y(t(x),{class:`size-3`}),g(` `+l(t(C)(`mappings.multimodalFallback.add`)),1)]),_:1})),$.value?(f(),d(`div`,we,[y(t(te),{class:`size-3.5 text-orange-400 shrink-0 mt-0.5`}),u(`div`,Te,[u(`div`,Ee,l(t(C)(`mappings.multimodalFallback.sessionLockWarning`)),1),u(`div`,De,l(t(C)(`mappings.multimodalFallback.sessionLockReason`)),1),u(`div`,Oe,l(t(C)(`mappings.multimodalFallback.costSuggestion`)),1)])])):e(``,!0)])])):(f(),d(`div`,{key:0,class:`flex items-center gap-2 cursor-pointer select-none min-h-[28px]`,onClick:o[3]||=e=>T(`expand`)},[u(`span`,{class:`font-mono text-xs font-semibold text-foreground min-w-[90px] truncate shrink-0`,title:a.entry.clientModel},l(a.entry.clientModel),9,ae),y(t(M),{class:`size-3 text-muted-foreground/30 shrink-0`}),u(`div`,oe,[a.entry.targets.length>0?(f(),d(`span`,se,[o[9]||=u(`span`,{class:`text-[10px] font-semibold`},`①`,-1),g(` `+l(a.entry.targets[0].backend_model)+` `,1),J(a.entry.targets[0].provider_id)?(f(),d(`span`,P,l(J(a.entry.targets[0].provider_id)),1)):e(``,!0)])):e(``,!0),(f(!0),d(r,null,b(a.entry.targets.slice(1),(e,t)=>(f(),d(r,{key:`fo-`+t},[o[10]||=u(`span`,{class:`text-orange-400/40 text-xs shrink-0`},`|`,-1),u(`span`,F,[u(`span`,I,l(t===0?`②`:t===1?`③`:`${t+2}`),1),g(` `+l(e.backend_model),1)])],64))),128)),Q.value?(f(),d(r,{key:1},[o[11]||=u(`span`,{class:`text-muted-foreground/30 text-xs shrink-0`},`|`,-1),u(`span`,L,` ↓ `+l(a.entry.targets[0].overflow_model),1)],64)):e(``,!0),$.value?(f(),d(r,{key:2},[o[12]||=u(`span`,{class:`text-muted-foreground/30 text-xs shrink-0`},`|`,-1),u(`span`,R,[y(t(N),{class:`size-2.5`}),g(` `+l(a.entry.multimodalFallback.backend_model),1)])],64)):e(``,!0)]),u(`span`,{class:v([`text-[10px] px-1.5 py-px rounded font-medium leading-none shrink-0`,Fe.value])},l(Ie.value),3),u(`div`,{class:`shrink-0 flex items-center`,onClick:o[1]||=_(()=>{},[`stop`])},[y(t(ee),{"model-value":a.entry.active,"onUpdate:modelValue":o[0]||=e=>T(`toggle-active`),class:`scale-75`},null,8,[`model-value`])]),y(t(h),{variant:`ghost`,size:`icon-xs`,class:`shrink-0 text-muted-foreground/20 hover:text-destructive`,onClick:o[2]||=_(e=>T(`remove`),[`stop`])},{default:m(()=>[y(t(S),{class:`size-3`})]),_:1}),u(`span`,z,l(t(C)(`mappings.editor.clickToEdit`)),1)]))]))}}),J={key:1,class:`text-[10px] text-muted-foreground leading-snug`},Y=p({__name:`ConcurrencyControl`,props:{modelValue:{},compact:{type:Boolean,default:!1}},emits:[`update:modelValue`],setup(i,{emit:a}){let{t:o}=n(),s=i,p=a;function h(e){p(`update:modelValue`,{...s.modelValue,...e})}let _=c(()=>s.modelValue.mode??j.mode),b=c(()=>s.modelValue.max_concurrency??j.max_concurrency),x=c(()=>s.modelValue.queue_timeout_ms??j.queue_timeout_ms),S=c(()=>s.modelValue.max_queue_size??j.max_queue_size);return(n,a)=>(f(),d(`div`,{class:v(i.compact?`space-y-2`:`flex items-end gap-2 flex-wrap`)},[u(`div`,{class:v([i.compact?``:`w-36`,`space-y-1`])},[y(t(C),{class:`text-xs text-muted-foreground`},{default:m(()=>[g(l(t(o)(`providers.concurrency.mode`)),1)]),_:1}),y(t(O),{"model-value":_.value,"onUpdate:modelValue":a[0]||=e=>h({mode:e})},{default:m(()=>[y(t(E),{class:`w-full text-xs data-[size=default]:h-7`},{default:m(()=>[y(t(D))]),_:1}),y(t(w),null,{default:m(()=>[y(t(T),{value:`auto`},{default:m(()=>[g(l(t(o)(`providers.concurrency.autoAdaptive`)),1)]),_:1}),y(t(T),{value:`manual`},{default:m(()=>[g(l(t(o)(`providers.concurrency.manual`)),1)]),_:1}),y(t(T),{value:`none`},{default:m(()=>[g(l(t(o)(`providers.concurrency.none`)),1)]),_:1})]),_:1})]),_:1},8,[`model-value`])],2),_.value===`none`?e(``,!0):(f(),d(r,{key:0},[u(`div`,{class:v([i.compact?``:`w-24`,`space-y-1`])},[y(t(C),{class:`text-xs text-muted-foreground`},{default:m(()=>[g(l(t(o)(`providers.concurrency.maxConcurrency`)),1)]),_:1}),y(t(k),{"model-value":b.value,type:`number`,min:`1`,max:`100`,class:`text-xs md:text-xs md:text-xs h-7`,"onUpdate:modelValue":a[1]||=e=>h({max_concurrency:Number(e)})},null,8,[`model-value`])],2),u(`div`,{class:v([i.compact?``:`w-30`,`space-y-1`])},[y(t(C),{class:`text-xs text-muted-foreground`},{default:m(()=>[g(l(t(o)(`providers.concurrency.queueTimeout`)),1)]),_:1}),y(t(k),{"model-value":x.value,type:`number`,min:`0`,placeholder:t(o)(`providers.shared.queueTimeoutPlaceholder`),class:`font-mono text-xs md:text-xs md:text-xs h-7`,"onUpdate:modelValue":a[2]||=e=>h({queue_timeout_ms:Number(e)})},null,8,[`model-value`,`placeholder`])],2),u(`div`,{class:v([i.compact?``:`w-30`,`space-y-1`])},[y(t(C),{class:`text-xs text-muted-foreground`},{default:m(()=>[g(l(t(o)(`providers.concurrency.maxQueueSize`)),1)]),_:1}),y(t(k),{"model-value":S.value,type:`number`,min:`1`,max:`1000`,class:`text-xs md:text-xs md:text-xs h-7`,"onUpdate:modelValue":a[3]||=e=>h({max_queue_size:Number(e)})},null,8,[`model-value`])],2)],64)),_.value===`auto`&&!i.compact?(f(),d(`div`,J,l(t(o)(`providers.shared.autoHint`)),1)):e(``,!0)],2))}});export{q as n,M as r,Y as t};