promptfoo 0.121.2 → 0.121.4

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 (379) hide show
  1. package/README.md +2 -0
  2. package/dist/src/{ListApp-Du7YVwj5.js → ListApp-DQkFNqE9.js} +1 -1
  3. package/dist/src/{accounts-B0pgC1oV.js → accounts-DdJ2pHMI.js} +5 -5
  4. package/dist/src/{accounts-CiBLOnA7.js → accounts-DhMYUUbu.js} +5 -5
  5. package/dist/src/{accounts-gtkH-5KX.cjs → accounts-Dy17bs4D.cjs} +5 -5
  6. package/dist/src/{accounts-Bm2D8Db9.js → accounts-F9d_5sMC.js} +6 -6
  7. package/dist/src/{cometapi-C4xSqeID.cjs → agentic-utils-BpX5b23w.cjs} +24 -62
  8. package/dist/src/{cometapi-CUQq3H_a.js → agentic-utils-P172hM8B.js} +4 -61
  9. package/dist/src/agentic-utils-qFlm6zes.js +153 -0
  10. package/dist/src/{agentic-utils-DS1g3GLF.js → agentic-utils-w68v6_Dz.js} +3 -3
  11. package/dist/src/{agents-CmvBq8LV.js → agents-8FDnTriG.js} +6 -7
  12. package/dist/src/{agents-DbRtpYxR.cjs → agents-BahDpe5G.cjs} +255 -20
  13. package/dist/src/{agents-DgF2zDag.js → agents-C-R_jfzI.js} +255 -20
  14. package/dist/src/{agents-9qiOy0ho.js → agents-CgaMXvLM.js} +5 -7
  15. package/dist/src/{agents-Di9DKPzn.cjs → agents-D7-HGxUj.cjs} +5 -7
  16. package/dist/src/{agents-CBr9A01V.js → agents-DJ35I3Nt.js} +255 -20
  17. package/dist/src/{agents-cLXA8a_8.js → agents-aYPQLf8W.js} +5 -9
  18. package/dist/src/{agents-D__IdAlg.js → agents-pQeBEXMm.js} +255 -21
  19. package/dist/src/{aimlapi-B4rcnZgv.js → aimlapi-BCq3MHeL.js} +8 -14
  20. package/dist/src/{aimlapi-DHJU_kcV.js → aimlapi-BD6J9oKt.js} +7 -14
  21. package/dist/src/{aimlapi-BvlNH0gr.cjs → aimlapi-qcK4OT55.cjs} +8 -15
  22. package/dist/src/{aimlapi-CnkC2HqE.js → aimlapi-sgYnkE54.js} +7 -16
  23. package/dist/src/app/app/tsconfig.app.tsbuildinfo +1 -0
  24. package/dist/src/app/assets/Report-CQYFezYu.js +1 -0
  25. package/dist/src/app/assets/index-BXGkeMwh.css +1 -0
  26. package/dist/src/app/assets/index-BzJt18Jz.js +385 -0
  27. package/dist/src/app/assets/rolldown-runtime-COnpUsM8.js +1 -0
  28. package/dist/src/app/assets/scroll-timeline-D9IT_e8Z.js +1 -0
  29. package/dist/src/app/assets/sync-IjzpWrOE.js +4 -0
  30. package/dist/src/app/assets/vendor-charts-BNdH8TCw.js +36 -0
  31. package/dist/src/app/assets/vendor-markdown-Ch00wnNI.js +29 -0
  32. package/dist/src/app/assets/vendor-react-CVvmk1UB.js +9 -0
  33. package/dist/src/app/assets/vendor-utils-BnEYbx2Q.js +37 -0
  34. package/dist/src/app/index.html +7 -7
  35. package/dist/src/{audio-Dz3z7s3J.js → audio-B7izf48x.js} +4 -5
  36. package/dist/src/{audio-CGMyULza.cjs → audio-BQtNuYBj.cjs} +4 -4
  37. package/dist/src/{audio-Bkv46et0.js → audio-COrn8rM6.js} +4 -4
  38. package/dist/src/{audio-ClI_AFre.js → audio-DcVKoInv.js} +4 -4
  39. package/dist/src/{base-CGrhspbK.cjs → base-D-670DX8.cjs} +3 -3
  40. package/dist/src/{base-CpjcHe4e.js → base-PYJvBE1i.js} +3 -3
  41. package/dist/src/{base-Dy1V8--Z.js → base-fZ9wgg50.js} +3 -3
  42. package/dist/src/{base-DLKtKMFh.js → base-yrI1Yal4.js} +3 -3
  43. package/dist/src/{blobs-BDbfYdrJ.js → blobs-BCZavS8s.js} +4 -4
  44. package/dist/src/{blobs-CMHN0Qcz.js → blobs-BQWqnnvL.js} +4 -4
  45. package/dist/src/{blobs-CBO20krR.js → blobs-C-F78Kfn.js} +3 -3
  46. package/dist/src/{blobs-D23XLin-.cjs → blobs-D2FAd1Q5.cjs} +3 -3
  47. package/dist/src/cache-BIyPcp5v.cjs +376 -0
  48. package/dist/src/cache-C4Xb-hNb.js +310 -0
  49. package/dist/src/cache-Cr9oLMUa.js +3 -0
  50. package/dist/src/cache-D5NZmMiT.js +310 -0
  51. package/dist/src/cache-DbLsVWB2.cjs +3 -0
  52. package/dist/src/cache-mb7c8hbp.js +280 -0
  53. package/dist/src/{chat-C2jrdPMx.js → chat-0bwXjVP0.js} +3 -13
  54. package/dist/src/{chat-C1Qst7jL.cjs → chat-BPXSW8Bv.cjs} +3 -13
  55. package/dist/src/{chat-DJIw17u0.js → chat-BfPaS15_.js} +68 -42
  56. package/dist/src/{chat-CgF-J-Jj.cjs → chat-CclRbxGf.cjs} +68 -42
  57. package/dist/src/{chat-BiKyneZl.js → chat-Dr3DUQ0D.js} +68 -42
  58. package/dist/src/{chat-DqxYYtWA.js → chat-I9izLm49.js} +67 -41
  59. package/dist/src/{chat-CzkrVDfz.js → chat-MKxMnZJZ.js} +3 -13
  60. package/dist/src/{chat-qmatte1u.js → chat-mW0ORo8G.js} +3 -14
  61. package/dist/src/{chatkit-DKyPi1Gs.cjs → chatkit-BoWoSgXl.cjs} +4 -4
  62. package/dist/src/{chatkit-65VXf5SR.js → chatkit-CJnHRRMM.js} +4 -4
  63. package/dist/src/{chatkit-Be-Q-a9F.js → chatkit-Cv6AhukM.js} +4 -4
  64. package/dist/src/{chatkit-BxFvW8KY.js → chatkit-zUIVoDos.js} +4 -4
  65. package/dist/src/{claude-agent-sdk-Apiy0iaz.js → claude-agent-sdk-BQNuLaAK.js} +41 -18
  66. package/dist/src/{claude-agent-sdk-D9Z5Pr9X.cjs → claude-agent-sdk-CPJo3dBQ.cjs} +45 -22
  67. package/dist/src/{claude-agent-sdk-D2bJee9S.js → claude-agent-sdk-Dtq_L-Sc.js} +40 -17
  68. package/dist/src/{claude-agent-sdk-DfCoW0E6.js → claude-agent-sdk-nfAIcxNf.js} +42 -20
  69. package/dist/src/{cloud-C0dlstV_.js → cloud-DQZ5sVjW.js} +25 -13
  70. package/dist/src/cloud-Hphvo8kr.js +3 -0
  71. package/dist/src/{cloudflare-ai-8TDxHR0x.js → cloudflare-ai-BIB567w6.js} +5 -14
  72. package/dist/src/{cloudflare-ai-g7PB6VHR.js → cloudflare-ai-DGLte7Py.js} +5 -14
  73. package/dist/src/{cloudflare-ai-CknbZ5LJ.cjs → cloudflare-ai-Dl3N9OVD.cjs} +6 -15
  74. package/dist/src/{cloudflare-ai-BxAGvfju.js → cloudflare-ai-DlKr0rY7.js} +5 -15
  75. package/dist/src/{cloudflare-gateway-B9HWA5wf.js → cloudflare-gateway-BDZrYydE.js} +4 -16
  76. package/dist/src/{cloudflare-gateway-BSnDmHYo.cjs → cloudflare-gateway-BYDp495F.cjs} +4 -15
  77. package/dist/src/{cloudflare-gateway-CKDb4dJ8.js → cloudflare-gateway-CiIZHU0Q.js} +5 -164
  78. package/dist/src/{cloudflare-gateway-CP9QEWYS.js → cloudflare-gateway-DI1HNP5F.js} +4 -15
  79. package/dist/src/codex-sdk-BAmYE7qy.js +3 -0
  80. package/dist/src/codex-sdk-C2_M2pl_.cjs +1172 -0
  81. package/dist/src/codex-sdk-CErXn7qh.js +1165 -0
  82. package/dist/src/codex-sdk-CWEnH70W.cjs +2 -0
  83. package/dist/src/codex-sdk-CpqiOqDO.js +1164 -0
  84. package/dist/src/codex-sdk-Rtky3M4I.js +1165 -0
  85. package/dist/src/{cometapi-BL9yvj_f.js → cometapi-BUlt_ELa.js} +8 -15
  86. package/dist/src/{cometapi-DFNiKmSz.js → cometapi-CtJ-mS8R.js} +8 -16
  87. package/dist/src/cometapi-DT-jlVCB.js +55 -0
  88. package/dist/src/cometapi-UVOryo4W.cjs +55 -0
  89. package/dist/src/{completion-CM6oK8PS.cjs → completion-BozdoXba.cjs} +7 -62
  90. package/dist/src/{completion-5MzrpJxT.js → completion-Dnxn7E-j.js} +8 -57
  91. package/dist/src/{completion-qRoZAYRB.js → completion-HUe8wDhZ.js} +8 -57
  92. package/dist/src/{completion-DZ083F31.js → completion-x0a_c2y1.js} +8 -57
  93. package/dist/src/{createHash-CTQmL3G2.js → createHash-4gFQpDDv.js} +3 -3
  94. package/dist/src/{createHash-CfZSc0b4.cjs → createHash-B7KvgoOD.cjs} +4 -4
  95. package/dist/src/{createHash-Da8fMwqB.js → createHash-ChI45QR1.js} +3 -3
  96. package/dist/src/{createHash-DmPQkvBh.js → createHash-CwDVU5xr.js} +3 -3
  97. package/dist/src/{docker-ExVyLp0S.js → docker-CQmlA2NU.js} +7 -14
  98. package/dist/src/{docker-Bb5dcxr8.js → docker-ClnmCf1Z.js} +6 -14
  99. package/dist/src/{docker-BvfL2BrW.js → docker-DCgsveLD.js} +6 -16
  100. package/dist/src/{docker-DcF2pRrj.cjs → docker-DS4_Osau.cjs} +7 -15
  101. package/dist/src/embedding-D3xTseo7.js +59 -0
  102. package/dist/src/embedding-DD9wa3ae.js +58 -0
  103. package/dist/src/embedding-I45KG3o7.cjs +63 -0
  104. package/dist/src/embedding-nFbumxcv.js +58 -0
  105. package/dist/src/entrypoint.js +69 -6
  106. package/dist/src/{errors-P6ll7XSJ.js → errors-Cw810C93.js} +1 -1
  107. package/dist/src/{esm-CaIwzWR5.js → esm-C7PnfdF8.js} +16 -7
  108. package/dist/src/{esm-CnNt7sI4.cjs → esm-CtEPLdAj.cjs} +15 -6
  109. package/dist/src/{esm-Cd1AjG1D.js → esm-Dh4dOLlt.js} +15 -6
  110. package/dist/src/{esm-C03C-mv3.js → esm-tVgYPY-f.js} +27 -18
  111. package/dist/src/eval-CzJFfFO9.js +3 -0
  112. package/dist/src/{eval-Dg2nG4v2.js → eval-u4UVafl6.js} +49 -20
  113. package/dist/src/{evalResult-BDMqrapS.js → evalResult-Bgm9ZH31.js} +7 -7
  114. package/dist/src/{evalResult-BBRNtX4I.js → evalResult-D3hVYFis.js} +7 -7
  115. package/dist/src/evalResult-D8MT9p0s.js +3 -0
  116. package/dist/src/evalResult-DElBuddX.js +2 -0
  117. package/dist/src/evalResult-Dvc-iucu.cjs +2 -0
  118. package/dist/src/{evalResult-fuaI8HkH.cjs → evalResult-KZqXl4XP.cjs} +7 -7
  119. package/dist/src/evaluator-CVessDWe.js +3 -0
  120. package/dist/src/{evaluator-BhoWwp5b.js → evaluator-IvuDYSvQ.js} +3080 -1254
  121. package/dist/src/{extractor-D25qpmGX.js → extractor-CAfTSraf.js} +6 -6
  122. package/dist/src/{extractor-DReVID0K.js → extractor-DNSeBVOJ.js} +6 -6
  123. package/dist/src/{extractor-C0EVHewb.js → extractor-Dk6bRWkv.js} +6 -6
  124. package/dist/src/{extractor-pYLLi3wS.cjs → extractor-WVPOrH43.cjs} +6 -6
  125. package/dist/src/{fetch-HaqdX7U1.js → fetch-B0Z3Oe4k.js} +218 -55
  126. package/dist/src/{fetch-Dxpd4_sr.js → fetch-BEWnXrrG.js} +195 -45
  127. package/dist/src/fetch-C7bGKDlQ.js +3 -0
  128. package/dist/src/{fetch-BPkYtG8K.cjs → fetch-CJU5ELPa.cjs} +223 -48
  129. package/dist/src/{fetch-Cwxnd8zz.js → fetch-Di00EQrc.js} +218 -55
  130. package/dist/src/{fileExtensions-Ds-foDzt.js → fileExtensions-AWa2ZML4.js} +1 -1
  131. package/dist/src/{fileExtensions-LcDYkU4v.js → fileExtensions-BArZuxsI.js} +1 -1
  132. package/dist/src/{formatDuration-DgBVMN65.js → formatDuration-DZzPsexs.js} +1 -1
  133. package/dist/src/{genaiTracer-D3fD9dNV.js → genaiTracer-COYDi-tC.js} +6 -2
  134. package/dist/src/{genaiTracer-C1rxGO8Q.js → genaiTracer-DWdZ28hY.js} +6 -2
  135. package/dist/src/{genaiTracer-70Z8BIuV.js → genaiTracer-XnrcgDCe.js} +6 -2
  136. package/dist/src/{genaiTracer-DN4dQywX.cjs → genaiTracer-yRuxj9-L.cjs} +7 -3
  137. package/dist/src/golang/wrapper.go +1 -1
  138. package/dist/src/{graders-DU49_J8Y.cjs → graders--zknU_uk.cjs} +5747 -3206
  139. package/dist/src/graders-BOAzQEUe.cjs +2 -0
  140. package/dist/src/graders-D4BTsZdG2.js +3 -0
  141. package/dist/src/graders-DOJK1XpV.js +2 -0
  142. package/dist/src/graders-NAv9LcBn.js +2 -0
  143. package/dist/src/{graders-BTeBGqjJ.js → graders-Zy3x0zqX.js} +5727 -3218
  144. package/dist/src/{graders-Bj_Odv7c.js → graders-eIHhRqoC.js} +5719 -3210
  145. package/dist/src/{graders-DP7KFFo-.js → graders-pvbReLLn.js} +5728 -3219
  146. package/dist/src/{image-B0h9VEMc.js → image-9302QVqR.js} +4 -4
  147. package/dist/src/{image-CHfWvljl.js → image-B5Mv-Z3h.js} +8 -8
  148. package/dist/src/{image-DS-o-0ph.js → image-DVz2RiMF.js} +8 -8
  149. package/dist/src/{image-C1madmKh.cjs → image-De2FBmYV.cjs} +4 -4
  150. package/dist/src/{image-B02ogr_b.js → image-dnoUgPrC.js} +4 -5
  151. package/dist/src/{image-Dpxa1Jt6.js → image-qUpPvmNZ.js} +8 -8
  152. package/dist/src/{image-Bb4vWQLM.js → image-u7-rKnYU.js} +4 -4
  153. package/dist/src/{image-BLmROtN3.cjs → image-x6KqLQl4.cjs} +8 -8
  154. package/dist/src/index.cjs +3728 -1482
  155. package/dist/src/index.d.cts +3232 -79
  156. package/dist/src/index.d.ts +3232 -79
  157. package/dist/src/index.js +3735 -1490
  158. package/dist/src/{interactiveCheck-BgLZUIt3.js → interactiveCheck-CLERUB0c.js} +2 -2
  159. package/dist/src/{knowledgeBase-DOO_BM9b.cjs → knowledgeBase-Bpoe_nLu.cjs} +6 -8
  160. package/dist/src/{knowledgeBase-D33Ty2l6.js → knowledgeBase-Dgc7CBWF.js} +6 -8
  161. package/dist/src/{knowledgeBase-B3OoKIej.js → knowledgeBase-RhFPGWDc.js} +6 -8
  162. package/dist/src/{knowledgeBase-CYTLHOt1.js → knowledgeBase-lm9RXSAm.js} +6 -9
  163. package/dist/src/{litellm-NbjknEh6.js → litellm-C2kqjxqp.js} +6 -14
  164. package/dist/src/{litellm-I_hbp_dc.cjs → litellm-CoyI4IAl.cjs} +6 -15
  165. package/dist/src/{litellm-TrljxD9G.js → litellm-DRjpcSa7.js} +5 -14
  166. package/dist/src/{litellm-AaeZcZQF.js → litellm-p37R1dzQ.js} +5 -16
  167. package/dist/src/{logger-DLcq4dWf.js → logger-B88EkIn6.js} +48 -23
  168. package/dist/src/{logger-Cp1GPUjj.cjs → logger-COuQb2xB.cjs} +77 -22
  169. package/dist/src/{logger-CT3IKMKA.js → logger-Ct2S6Yx-.js} +48 -23
  170. package/dist/src/{logger-KkObSCzq.js → logger-DksKw1Qc.js} +48 -23
  171. package/dist/src/{luma-ray-f6I2fft-.js → luma-ray-B863CmuZ.js} +6 -10
  172. package/dist/src/{luma-ray-DDsjcgZZ.js → luma-ray-BTTLtqQ8.js} +7 -10
  173. package/dist/src/{luma-ray-Due0n7di.cjs → luma-ray-BxVKaW2a.cjs} +6 -10
  174. package/dist/src/{luma-ray-BS2_tY8L.js → luma-ray-KgTCXrZC.js} +6 -12
  175. package/dist/src/main.d.ts +1 -26
  176. package/dist/src/main.js +1011 -548
  177. package/dist/src/{messages-Bs1kC7P4.cjs → messages-811uVVW5.cjs} +74 -19
  178. package/dist/src/{messages-BS17jdMx.js → messages-BTQz42fn.js} +74 -19
  179. package/dist/src/{messages-ZJk778GH.js → messages-MYTQ2TWp.js} +74 -19
  180. package/dist/src/{messages-D0lx5qK7.js → messages-zWbkLLHz.js} +74 -19
  181. package/dist/src/{meteor-D-SotUw9.js → meteor-CU5UAE-H.js} +1 -1
  182. package/dist/src/{meteor-DLZZ3osF.cjs → meteor-Co1VQ1u5.cjs} +1 -1
  183. package/dist/src/{meteor-44VjEACX.js → meteor-DHdzY1Ss.js} +1 -1
  184. package/dist/src/{meteor-DUiCJRC-.js → meteor-DuAFv6gF.js} +1 -1
  185. package/dist/src/{modelslab-Bmni6skY.js → modelslab-D0erNWKe.js} +7 -10
  186. package/dist/src/{modelslab-DRb74SP4.js → modelslab-DIq-6y7x.js} +7 -10
  187. package/dist/src/{modelslab-CoUX6Jc_.cjs → modelslab-Dk1JAtVo.cjs} +7 -10
  188. package/dist/src/{modelslab-Bx9IrZfS.js → modelslab-wu9yi5GE.js} +7 -11
  189. package/dist/src/{nova-reel-BfPq-0Yk.js → nova-reel-CCFRfeRb.js} +7 -10
  190. package/dist/src/{nova-reel-C_QM18Xn.cjs → nova-reel-CrLXVKQf.cjs} +6 -10
  191. package/dist/src/{nova-reel-bgjxilYW.js → nova-reel-DQrm74ng.js} +6 -10
  192. package/dist/src/{nova-reel-D_W1tjMH.js → nova-reel-gr11WG7f.js} +6 -12
  193. package/dist/src/{nova-sonic-DIGQNR07.js → nova-sonic-BYdp-QLs.js} +5 -7
  194. package/dist/src/{nova-sonic-CFb5GYhg.js → nova-sonic-B_ZXcUJB.js} +4 -7
  195. package/dist/src/{nova-sonic-De1HW5fD.js → nova-sonic-TDgrlTk7.js} +4 -9
  196. package/dist/src/{nova-sonic-zfcljeRp.cjs → nova-sonic-i5tUvXKn.cjs} +4 -7
  197. package/dist/src/{openai-DElQ-fPX.js → openai-DhVEmgeZ.js} +6 -3
  198. package/dist/src/{openai-DhbB7eWK.js → openai-Qsvz25mV.js} +6 -3
  199. package/dist/src/{openai-Cuif0GEt.cjs → openai-URNyItar.cjs} +6 -3
  200. package/dist/src/{openai-j-sE2O7r.js → openai-iYtrXzOX.js} +6 -3
  201. package/dist/src/openclaw-CLWrW03k.js +1200 -0
  202. package/dist/src/openclaw-CnQ363Wi.js +1199 -0
  203. package/dist/src/openclaw-CwzlQSQX.js +1199 -0
  204. package/dist/src/openclaw-wX9rtfke.cjs +1205 -0
  205. package/dist/src/{opencode-sdk-B3CWY9h_.js → opencode-sdk-BUu5Nevv.js} +12 -14
  206. package/dist/src/{opencode-sdk-BL764Jdi.cjs → opencode-sdk-BZ2idgYA.cjs} +16 -18
  207. package/dist/src/{opencode-sdk-0j6rTWNb.js → opencode-sdk-BxD8vXp_.js} +12 -15
  208. package/dist/src/{opencode-sdk-C2y6UkP2.js → opencode-sdk-GI2KaAXq.js} +12 -14
  209. package/dist/src/{otlpReceiver-C99PPb48.js → otlpReceiver-B2z58l4e.js} +154 -98
  210. package/dist/src/{otlpReceiver-CGq6LspY.cjs → otlpReceiver-BfcVq2Nq.cjs} +154 -98
  211. package/dist/src/{otlpReceiver-D89fR-rC.js → otlpReceiver-BntK801g.js} +154 -98
  212. package/dist/src/{otlpReceiver-CdNBdbsk.js → otlpReceiver-DmVulbhC.js} +154 -98
  213. package/dist/src/{providerRegistry-CD8MEar9.js → providerRegistry-Bvh8mv85.js} +2 -2
  214. package/dist/src/{providerRegistry-DM8rZYol.js → providerRegistry-CPQ_CmVO.js} +2 -2
  215. package/dist/src/{providerRegistry-Civky8Ar.cjs → providerRegistry-CQMdTmHP.cjs} +2 -2
  216. package/dist/src/{providerRegistry-B0RUOLI_.js → providerRegistry-CWoPjKFZ.js} +2 -2
  217. package/dist/src/{providers-CgKOSgTR.cjs → providers-1eKkXBKp.cjs} +1435 -930
  218. package/dist/src/{providers-BlqUifFg.js → providers-BV_KMZje.js} +1419 -944
  219. package/dist/src/providers-Bp4S-FvO.js +2 -0
  220. package/dist/src/providers-DV3ax9e_.cjs +3 -0
  221. package/dist/src/{providers-D8lF1sqW.js → providers-Domz_llv.js} +1427 -952
  222. package/dist/src/{providers-Dk_6ocUX.js → providers-DruaQfwu.js} +1424 -949
  223. package/dist/src/providers-iUt5fbAN.js +3 -0
  224. package/dist/src/providers-u9Enmfok.js +2 -0
  225. package/dist/src/python/persistent_wrapper.py +0 -5
  226. package/dist/src/{pythonUtils-D6fwaDSg.js → pythonUtils-C2UQ30Rz.js} +4 -4
  227. package/dist/src/{pythonUtils-D5nxkQ0P.js → pythonUtils-Cldx7huE.js} +4 -4
  228. package/dist/src/{pythonUtils-C3py6GC1.js → pythonUtils-CnndUbW-.js} +3 -3
  229. package/dist/src/{pythonUtils-CTU3Y3lw.cjs → pythonUtils-tAJvvpS-.cjs} +3 -3
  230. package/dist/src/{quiverai-CIaELU_m.js → quiverai-CtWi6x_g.js} +4 -4
  231. package/dist/src/{quiverai-PdShCPox.cjs → quiverai-DFotyafY.cjs} +4 -4
  232. package/dist/src/{quiverai-BbOUOn2L.js → quiverai-DR0SnIQV.js} +4 -4
  233. package/dist/src/{quiverai-uH-dcTIr.js → quiverai-aPPvXOgn.js} +4 -5
  234. package/dist/src/render-CH-62LbA.js +135 -0
  235. package/dist/src/render-CMEpfLaO.js +136 -0
  236. package/dist/src/{render-Drod8m7K.js → render-CgVDrJmM.js} +2 -3
  237. package/dist/src/render-DHIZ6_k8.js +135 -0
  238. package/dist/src/render-DfQSFxGE.cjs +165 -0
  239. package/dist/src/{responses-DIR9Ud3j.js → responses--OsX2aYW.js} +23 -14
  240. package/dist/src/{responses-D8SBTL64.cjs → responses-Bi9vBuW_.cjs} +24 -15
  241. package/dist/src/{responses-CB2jwoAr.js → responses-C-flexAY.js} +24 -15
  242. package/dist/src/{responses-WNGNYe3K.js → responses-DL9m8CyY.js} +24 -15
  243. package/dist/src/{rubyUtils-DhCAlxZr.cjs → rubyUtils-B6eljPuh.cjs} +3 -3
  244. package/dist/src/{rubyUtils-BcuGX77l.js → rubyUtils-CYSQEG4a.js} +3 -3
  245. package/dist/src/rubyUtils-D1L2d3jb.js +3 -0
  246. package/dist/src/rubyUtils-DUbq4tff.cjs +2 -0
  247. package/dist/src/{rubyUtils-BUVePouc.js → rubyUtils-DVLeA2jg.js} +3 -3
  248. package/dist/src/{rubyUtils-Boc4HZzX.js → rubyUtils-DsGrTx8R.js} +3 -3
  249. package/dist/src/{sagemaker-CNBxx5CJ.js → sagemaker-BVkaG2-l.js} +14 -18
  250. package/dist/src/{sagemaker-CemTFp2h.js → sagemaker-BveBvuxm.js} +14 -18
  251. package/dist/src/{sagemaker-YSyBXQQh.js → sagemaker-D67yzMzs.js} +14 -19
  252. package/dist/src/{sagemaker-Cl28mZU2.cjs → sagemaker-XnfhheQv.cjs} +14 -18
  253. package/dist/src/{scanner-BsBlNXNn.js → scanner-1DqWi1Ej.js} +130 -35
  254. package/dist/src/server/golang/wrapper.go +1 -1
  255. package/dist/src/server/index.js +3757 -1511
  256. package/dist/src/server/python/persistent_wrapper.py +0 -5
  257. package/dist/src/{server-CqzrVGpF.js → server-BNYztJkh.js} +128 -9
  258. package/dist/src/{server-CuxBbeSY.js → server-BSB45Nt9.js} +127 -8
  259. package/dist/src/{server-VWgWb00X.js → server-D6Il2Sob.js} +126 -7
  260. package/dist/src/server-DCtHUqlp.js +3 -0
  261. package/dist/src/server-DaA2eR26.cjs +2 -0
  262. package/dist/src/{server-C_7Ax-hA.cjs → server-Dx2TyCH2.cjs} +140 -6
  263. package/dist/src/{signal-4U3mfRvL.js → signal-CE5G3a7x.js} +3 -3
  264. package/dist/src/{slack-BmVAVGaK.cjs → slack-1Rhq0EoV.cjs} +2 -2
  265. package/dist/src/{slack-DCUPTzS2.js → slack-D5Wpy8LM.js} +2 -2
  266. package/dist/src/{slack-DXMKtA-f.js → slack-DDUe-5MC.js} +2 -2
  267. package/dist/src/{slack-DOdy_kyv.js → slack-acRb0IqQ.js} +2 -2
  268. package/dist/src/store-CWOSz6D_.cjs +2 -0
  269. package/dist/src/{store-Dim__MDd.js → store-CYEy5J2D.js} +17 -5
  270. package/dist/src/{store-DLlFCC4h.cjs → store-DAAyxcy6.cjs} +17 -5
  271. package/dist/src/store-DCDBhv7B.js +3 -0
  272. package/dist/src/{store-CXGFv4aR.js → store-Dn9HUkdW.js} +17 -5
  273. package/dist/src/{store-DXilxTl-.js → store-M0b1WfYb.js} +17 -5
  274. package/dist/src/{tables-gftXzE9I.js → tables-C4CH3zRr.js} +3 -3
  275. package/dist/src/{tables-DLJPUdUE.js → tables-CsWou1Bx.js} +3 -3
  276. package/dist/src/{tables-6YKwjN9-.js → tables-DQ4WU5tX.js} +3 -3
  277. package/dist/src/{tables-DPi7wKeM.cjs → tables-DUfh1F7Z.cjs} +3 -3
  278. package/dist/src/telemetry-C1IqxcdW.js +3 -0
  279. package/dist/src/telemetry-C4ZEa_es.cjs +2 -0
  280. package/dist/src/{telemetry-CMrFgtPB.js → telemetry-CQPez_Jp.js} +4 -4
  281. package/dist/src/{telemetry-DaX14Chu.cjs → telemetry-Dsw_faFj.cjs} +4 -4
  282. package/dist/src/{telemetry-Dthj_BbD.js → telemetry-Dvqxv3YC.js} +4 -4
  283. package/dist/src/{telemetry-Cps3mIU-.js → telemetry-dbaJ0E98.js} +4 -4
  284. package/dist/src/{text-CW1cyrwj.cjs → text-BVi-cLPJ.cjs} +1 -1
  285. package/dist/src/{text-B_UCRPp2.js → text-CZr46tp_.js} +1 -1
  286. package/dist/src/{text-TIv0QYnd.js → text-DHxdyQqT.js} +1 -1
  287. package/dist/src/{text-Db-Wt2u2.js → text-KvuD2Iko.js} +1 -1
  288. package/dist/src/{tokenUsageUtils-bVa1ga6f.cjs → tokenUsageUtils-Bb7DkZPz.cjs} +7 -3
  289. package/dist/src/{tokenUsageUtils-NYT-WKS6.js → tokenUsageUtils-C-bmyHoE.js} +7 -3
  290. package/dist/src/{tokenUsageUtils-DflFMjS0.js → tokenUsageUtils-CXrvO-wA.js} +7 -3
  291. package/dist/src/{transcription-NLVG9MT1.cjs → transcription-BvjmiYB1.cjs} +12 -16
  292. package/dist/src/{transcription-BNYURcXg.js → transcription-CJspiD2c.js} +11 -14
  293. package/dist/src/{transcription-B_OdaHp7.js → transcription-DuWDupG7.js} +10 -14
  294. package/dist/src/{transcription-s6A-bNrZ.js → transcription-V2HaAmy2.js} +10 -16
  295. package/dist/src/{transform-DECvGmzp.js → transform-Bbg6A8Jk.js} +4 -4
  296. package/dist/src/{transform-vNucnNr0.js → transform-CG0ehZNG.js} +11 -7
  297. package/dist/src/{transform-DuHvhZpj.cjs → transform-CTeuTR3S.cjs} +31 -9
  298. package/dist/src/{transform-CzK1Q0zl.cjs → transform-CUnzlsbn.cjs} +4 -4
  299. package/dist/src/{transform-aa6tmVpZ.js → transform-DYX1_Xnh.js} +5 -5
  300. package/dist/src/transform-DgKlRr73.cjs +2 -0
  301. package/dist/src/transform-M6ITAESf.js +3 -0
  302. package/dist/src/{transform-DilY9wbS.js → transform-UN5UGu8U.js} +5 -5
  303. package/dist/src/{transform-uAytVuyX.js → transform-lQrDE1BQ.js} +11 -7
  304. package/dist/src/{transform-D5HsjduX.js → transform-zDhMmzwX.js} +11 -7
  305. package/dist/src/{transformersAvailability-CEVM2GNQ.js → transformersAvailability-CcHusyhw.js} +1 -1
  306. package/dist/src/{transformersAvailability-CwayUSlh.cjs → transformersAvailability-Cju9mHgR.cjs} +1 -1
  307. package/dist/src/{transformersAvailability-D6c6ROpT.js → transformersAvailability-DLlROWhg.js} +1 -1
  308. package/dist/src/{types-DmyIJ-sR.js → types-BGQDAP8i.js} +357 -22
  309. package/dist/src/{types-CzW2QFyi.js → types-Bgh5SOn6.js} +358 -24
  310. package/dist/src/{types-C_7nyzr1.cjs → types-CeaeaZdP.cjs} +393 -22
  311. package/dist/src/{types-Cbd8uOMq.js → types-Dm9JM6Vb.js} +368 -23
  312. package/dist/src/{util-BHGHw5G1.js → util-BYvQUPp7.js} +138 -36
  313. package/dist/src/{util-B9vlHIIh.cjs → util-Bxn8emtE.cjs} +15 -168
  314. package/dist/src/{util-ZzmqNPlg.js → util-C8e5uydV.js} +19 -142
  315. package/dist/src/{util-CMy69ZgQ.js → util-C9J8ahRn.js} +18 -4
  316. package/dist/src/{util-BzMcevZc.cjs → util-CN3SrLT4.cjs} +18 -4
  317. package/dist/src/{util-BV4XUC0n.js → util-D3q0WQ-0.js} +18 -4
  318. package/dist/src/{util-Dnmk2mBQ.js → util-D9TisOyk.js} +18 -4
  319. package/dist/src/{util-B3xGByQh.js → util-DDs-7g6-.js} +138 -36
  320. package/dist/src/{util-Bv6uGDfH.js → util-DvU2Pw8c.js} +138 -36
  321. package/dist/src/{util-C1CeHl-P.js → util-DxWpWjhc.js} +13 -136
  322. package/dist/src/{util-BRYkYPTd.js → util-oGMLA7vc.js} +17 -140
  323. package/dist/src/{util-DGNOS1db.cjs → util-olYL5C6N.cjs} +143 -35
  324. package/dist/src/{utils-Cz9qXqII.cjs → utils-B05gLxER.cjs} +6 -4
  325. package/dist/src/{utils-f2-Moju7.js → utils-BLJKfv0y.js} +6 -4
  326. package/dist/src/{utils-dLokC-eR.js → utils-DJfvjyMj.js} +6 -4
  327. package/dist/src/{utils-XiOAgly5.js → utils-hXtCYanr.js} +6 -4
  328. package/dist/tsconfig.tsbuildinfo +1 -1
  329. package/package.json +66 -57
  330. package/dist/src/app/assets/index-4LKxG2CG.js +0 -439
  331. package/dist/src/app/assets/index-C3zcsZFQ.css +0 -1
  332. package/dist/src/app/assets/scroll-timeline-BdJZVXlz.js +0 -1
  333. package/dist/src/app/assets/sync-9qqYcY-B.js +0 -4
  334. package/dist/src/app/assets/vendor-charts-BnDWwBlI.js +0 -36
  335. package/dist/src/app/assets/vendor-markdown-0tekx3KX.js +0 -29
  336. package/dist/src/app/assets/vendor-react-AtKqiNEf.js +0 -4
  337. package/dist/src/app/assets/vendor-syntax-D06x6TQF.js +0 -2
  338. package/dist/src/app/assets/vendor-utils-BvMHZmO7.js +0 -37
  339. package/dist/src/app/tsconfig.app.tsbuildinfo +0 -1
  340. package/dist/src/cache-BVeDlD87.js +0 -726
  341. package/dist/src/cache-C4Nxf52C.js +0 -756
  342. package/dist/src/cache-CeUpFm3M.cjs +0 -5
  343. package/dist/src/cache-Dh5WtQps.cjs +0 -816
  344. package/dist/src/cache-i1P6crbO.js +0 -756
  345. package/dist/src/cache-n-RCJ-hL.js +0 -6
  346. package/dist/src/cloud-BBh91EUK.js +0 -4
  347. package/dist/src/codex-sdk-C6UMlxwV.js +0 -665
  348. package/dist/src/codex-sdk-DUwKWezN.js +0 -665
  349. package/dist/src/codex-sdk-GGAw0qbD.js +0 -666
  350. package/dist/src/codex-sdk-fAO0c3yA.cjs +0 -669
  351. package/dist/src/eval-B3r2CVXr.js +0 -15
  352. package/dist/src/evalResult-5xwYnECe.js +0 -12
  353. package/dist/src/evalResult-71lY93Kj.cjs +0 -10
  354. package/dist/src/evalResult-Dx5P5cIv.js +0 -10
  355. package/dist/src/evaluator-Jx6bRZV6.js +0 -36
  356. package/dist/src/fetch-BxNb_Lp3.js +0 -5
  357. package/dist/src/graders-B_pgMLS2.js +0 -34
  358. package/dist/src/graders-DErokPDO.cjs +0 -32
  359. package/dist/src/graders-DR_uNe54.js +0 -32
  360. package/dist/src/graders-w3176Wz-.js +0 -32
  361. package/dist/src/openclaw-CSugPYAr.cjs +0 -586
  362. package/dist/src/openclaw-DiSz3I5L.js +0 -582
  363. package/dist/src/openclaw-DuvJKEW5.js +0 -580
  364. package/dist/src/openclaw-tiVYRtr-.js +0 -580
  365. package/dist/src/providers-B7V0njNs.js +0 -32
  366. package/dist/src/providers-BEwbhv0X.js +0 -30
  367. package/dist/src/providers-CH3C7zf7.js +0 -30
  368. package/dist/src/providers-zyB6k_38.cjs +0 -31
  369. package/dist/src/rubyUtils-BUHu6PhO.js +0 -5
  370. package/dist/src/rubyUtils-CP42kMvq.cjs +0 -4
  371. package/dist/src/server-DA4Cyrrq.js +0 -7
  372. package/dist/src/server-Dulb-4-K.cjs +0 -5
  373. package/dist/src/store-CXS-Q_91.js +0 -6
  374. package/dist/src/store-eYkaKMwq.cjs +0 -5
  375. package/dist/src/telemetry-BpMfhthR.cjs +0 -5
  376. package/dist/src/telemetry-Dw38hanS.js +0 -7
  377. package/dist/src/tokenUsageUtils-BDGe-iyI.js +0 -138
  378. package/dist/src/transform-DTGDnAzW.js +0 -6
  379. package/dist/src/transform-m3qNw4KP.cjs +0 -5
@@ -0,0 +1,1199 @@
1
+ import { T as getEnvString, a as logger, x as getConfigDirectoryPath } from "./logger-Ct2S6Yx-.js";
2
+ import { F as VERSION, h as REQUEST_TIMEOUT_MS, t as fetchWithProxy } from "./fetch-Di00EQrc.js";
3
+ import { n as sha256 } from "./createHash-4gFQpDDv.js";
4
+ import { t as OpenAiChatCompletionProvider } from "./chat-I9izLm49.js";
5
+ import { t as OpenAiResponsesProvider } from "./responses--OsX2aYW.js";
6
+ import { t as OpenAiEmbeddingProvider } from "./embedding-nFbumxcv.js";
7
+ import fs from "fs";
8
+ import path from "path";
9
+ import os from "os";
10
+ import crypto from "crypto";
11
+ import WebSocket from "ws";
12
+ import JSON5 from "json5";
13
+ //#region src/providers/openclaw/device-auth.ts
14
+ const DEFAULT_CLIENT_DIR = "openclaw";
15
+ const DEFAULT_DEVICE_IDENTITY_FILE = "device-identity.json";
16
+ const DEFAULT_DEVICE_AUTH_FILE = "device-auth.json";
17
+ function defaultOpenClawClientPath(fileName) {
18
+ return path.join(getConfigDirectoryPath(true), DEFAULT_CLIENT_DIR, fileName);
19
+ }
20
+ function getDefaultOpenClawDeviceIdentityPath() {
21
+ return defaultOpenClawClientPath(DEFAULT_DEVICE_IDENTITY_FILE);
22
+ }
23
+ function getDefaultOpenClawDeviceAuthPath() {
24
+ return defaultOpenClawClientPath(DEFAULT_DEVICE_AUTH_FILE);
25
+ }
26
+ function ensureParentDirectory(filePath) {
27
+ fs.mkdirSync(path.dirname(filePath), {
28
+ recursive: true,
29
+ mode: 448
30
+ });
31
+ }
32
+ function writeJsonSecure(filePath, value) {
33
+ ensureParentDirectory(filePath);
34
+ fs.writeFileSync(filePath, `${JSON.stringify(value, null, 2)}\n`, {
35
+ encoding: "utf-8",
36
+ mode: 384
37
+ });
38
+ fs.chmodSync(filePath, 384);
39
+ }
40
+ function parseJsonObject(raw) {
41
+ try {
42
+ const parsed = JSON.parse(raw);
43
+ return parsed && typeof parsed === "object" && !Array.isArray(parsed) ? parsed : void 0;
44
+ } catch (err) {
45
+ logger.debug("[OpenClaw Device Auth] Failed to parse JSON", {
46
+ err,
47
+ rawLength: raw.length,
48
+ rawPreview: raw.slice(0, 100)
49
+ });
50
+ return;
51
+ }
52
+ }
53
+ function normalizeDeviceMetadataValue(value) {
54
+ const trimmed = value?.trim();
55
+ if (!trimmed) return "";
56
+ return trimmed.replace(/[A-Z]/g, (letter) => letter.toLowerCase());
57
+ }
58
+ function normalizeScopes$1(scopes) {
59
+ const seen = /* @__PURE__ */ new Set();
60
+ for (const scope of scopes || []) {
61
+ const trimmed = scope.trim();
62
+ if (trimmed) seen.add(trimmed);
63
+ }
64
+ if (seen.has("operator.admin")) {
65
+ seen.add("operator.read");
66
+ seen.add("operator.write");
67
+ } else if (seen.has("operator.write")) seen.add("operator.read");
68
+ return Array.from(seen).sort();
69
+ }
70
+ function publicKeyBase64UrlFromPem(publicKeyPem) {
71
+ const publicKey = crypto.createPublicKey(publicKeyPem);
72
+ const jwk = publicKey.export({ format: "jwk" });
73
+ if (typeof jwk.x === "string" && jwk.x.trim()) return jwk.x;
74
+ const der = publicKey.export({
75
+ type: "spki",
76
+ format: "der"
77
+ });
78
+ return Buffer.from(der.subarray(Math.max(0, der.length - 32))).toString("base64url");
79
+ }
80
+ function deriveDeviceId(publicKeyPem) {
81
+ return sha256(Buffer.from(publicKeyBase64UrlFromPem(publicKeyPem), "base64url"));
82
+ }
83
+ function isValidIdentity(identity) {
84
+ try {
85
+ const publicKeyFromPrivateKey = crypto.createPublicKey(identity.privateKeyPem).export({
86
+ type: "spki",
87
+ format: "pem"
88
+ });
89
+ return identity.deviceId === deriveDeviceId(identity.publicKeyPem) && publicKeyBase64UrlFromPem(identity.publicKeyPem) === publicKeyBase64UrlFromPem(publicKeyFromPrivateKey);
90
+ } catch {
91
+ return false;
92
+ }
93
+ }
94
+ function parseIdentity(raw) {
95
+ const parsed = parseJsonObject(raw);
96
+ const identity = {
97
+ deviceId: typeof parsed?.deviceId === "string" ? parsed.deviceId : "",
98
+ publicKeyPem: typeof parsed?.publicKeyPem === "string" ? parsed.publicKeyPem : "",
99
+ privateKeyPem: typeof parsed?.privateKeyPem === "string" ? parsed.privateKeyPem : ""
100
+ };
101
+ return isValidIdentity(identity) ? identity : void 0;
102
+ }
103
+ function createDeviceIdentity() {
104
+ const keyPair = crypto.generateKeyPairSync("ed25519", {
105
+ publicKeyEncoding: {
106
+ type: "spki",
107
+ format: "pem"
108
+ },
109
+ privateKeyEncoding: {
110
+ type: "pkcs8",
111
+ format: "pem"
112
+ }
113
+ });
114
+ const publicKeyPem = keyPair.publicKey;
115
+ return {
116
+ deviceId: deriveDeviceId(publicKeyPem),
117
+ publicKeyPem,
118
+ privateKeyPem: keyPair.privateKey
119
+ };
120
+ }
121
+ function loadOrCreateOpenClawDeviceIdentity(filePath) {
122
+ const identityPath = filePath || getDefaultOpenClawDeviceIdentityPath();
123
+ try {
124
+ if (fs.existsSync(identityPath)) {
125
+ const identity = parseIdentity(fs.readFileSync(identityPath, "utf-8"));
126
+ if (identity) return identity;
127
+ }
128
+ } catch (err) {
129
+ logger.warn("[OpenClaw Device Auth] Failed to read device identity; generating new identity", {
130
+ err,
131
+ identityPath
132
+ });
133
+ }
134
+ const identity = createDeviceIdentity();
135
+ try {
136
+ writeJsonSecure(identityPath, identity);
137
+ } catch (err) {
138
+ logger.warn("[OpenClaw Device Auth] Failed to persist device identity", {
139
+ err,
140
+ identityPath
141
+ });
142
+ }
143
+ return identity;
144
+ }
145
+ function buildOpenClawDeviceAuthPayloadV3(params) {
146
+ return [
147
+ "v3",
148
+ params.deviceId,
149
+ params.clientId,
150
+ params.clientMode,
151
+ params.role,
152
+ params.scopes.join(","),
153
+ String(params.signedAtMs),
154
+ params.token ?? "",
155
+ params.nonce,
156
+ normalizeDeviceMetadataValue(params.platform),
157
+ normalizeDeviceMetadataValue(params.deviceFamily)
158
+ ].join("|");
159
+ }
160
+ function signOpenClawDevicePayload(privateKeyPem, payload) {
161
+ return crypto.sign(null, Buffer.from(payload), privateKeyPem).toString("base64url");
162
+ }
163
+ function buildSignedOpenClawDevice(params) {
164
+ const signedAt = params.nowMs ?? Date.now();
165
+ const payload = buildOpenClawDeviceAuthPayloadV3({
166
+ deviceId: params.identity.deviceId,
167
+ clientId: params.clientId,
168
+ clientMode: params.clientMode,
169
+ role: params.role,
170
+ scopes: params.scopes,
171
+ signedAtMs: signedAt,
172
+ token: params.token,
173
+ nonce: params.nonce,
174
+ platform: params.platform,
175
+ deviceFamily: params.deviceFamily
176
+ });
177
+ return {
178
+ id: params.identity.deviceId,
179
+ publicKey: publicKeyBase64UrlFromPem(params.identity.publicKeyPem),
180
+ signature: signOpenClawDevicePayload(params.identity.privateKeyPem, payload),
181
+ signedAt,
182
+ nonce: params.nonce
183
+ };
184
+ }
185
+ function parseDeviceAuthStore(raw) {
186
+ const parsed = parseJsonObject(raw);
187
+ if (!parsed || parsed.version !== 1 || typeof parsed.deviceId !== "string") return;
188
+ const tokens = {};
189
+ const rawTokens = parsed.tokens && typeof parsed.tokens === "object" && !Array.isArray(parsed.tokens) ? parsed.tokens : {};
190
+ for (const [role, tokenRecord] of Object.entries(rawTokens)) {
191
+ if (!tokenRecord || typeof tokenRecord !== "object" || Array.isArray(tokenRecord)) continue;
192
+ const record = tokenRecord;
193
+ if (typeof record.token !== "string" || !record.token.trim()) continue;
194
+ tokens[role] = {
195
+ token: record.token,
196
+ role: typeof record.role === "string" ? record.role : role,
197
+ scopes: Array.isArray(record.scopes) ? normalizeScopes$1(record.scopes.filter((scope) => typeof scope === "string")) : [],
198
+ updatedAtMs: typeof record.updatedAtMs === "number" ? record.updatedAtMs : 0
199
+ };
200
+ }
201
+ return {
202
+ version: 1,
203
+ deviceId: parsed.deviceId,
204
+ tokens
205
+ };
206
+ }
207
+ function readDeviceAuthStore(filePath) {
208
+ try {
209
+ if (!fs.existsSync(filePath)) return;
210
+ return parseDeviceAuthStore(fs.readFileSync(filePath, "utf-8"));
211
+ } catch (err) {
212
+ logger.debug("[OpenClaw Device Auth] Failed to read device auth store", {
213
+ err,
214
+ filePath
215
+ });
216
+ return;
217
+ }
218
+ }
219
+ function loadOpenClawDeviceAuthToken(params) {
220
+ const store = readDeviceAuthStore(params.filePath || getDefaultOpenClawDeviceAuthPath());
221
+ if (!store || store.deviceId !== params.deviceId) return;
222
+ const role = params.role.trim();
223
+ const tokenRecord = store.tokens[role];
224
+ const token = tokenRecord?.token.trim();
225
+ return token ? {
226
+ ...tokenRecord,
227
+ token
228
+ } : void 0;
229
+ }
230
+ function storeOpenClawDeviceAuthToken(params) {
231
+ const token = params.token.trim();
232
+ const role = params.role.trim();
233
+ if (!token || !role) return;
234
+ const authPath = params.filePath || getDefaultOpenClawDeviceAuthPath();
235
+ const existing = readDeviceAuthStore(authPath);
236
+ const store = existing && existing.deviceId === params.deviceId ? existing : {
237
+ version: 1,
238
+ deviceId: params.deviceId,
239
+ tokens: {}
240
+ };
241
+ store.tokens[role] = {
242
+ token,
243
+ role,
244
+ scopes: normalizeScopes$1(params.scopes),
245
+ updatedAtMs: Date.now()
246
+ };
247
+ try {
248
+ writeJsonSecure(authPath, store);
249
+ } catch (err) {
250
+ logger.warn("[OpenClaw Device Auth] Failed to persist device auth token", {
251
+ err,
252
+ authPath
253
+ });
254
+ }
255
+ }
256
+ function clearOpenClawDeviceAuthToken(params) {
257
+ const authPath = params.filePath || getDefaultOpenClawDeviceAuthPath();
258
+ const store = readDeviceAuthStore(authPath);
259
+ if (!store || store.deviceId !== params.deviceId) return;
260
+ delete store.tokens[params.role.trim()];
261
+ try {
262
+ writeJsonSecure(authPath, store);
263
+ } catch (err) {
264
+ logger.warn("[OpenClaw Device Auth] Failed to clear device auth token", {
265
+ err,
266
+ authPath
267
+ });
268
+ }
269
+ }
270
+ //#endregion
271
+ //#region src/providers/openclaw/shared.ts
272
+ const DEFAULT_GATEWAY_PORT = 18789;
273
+ const DEFAULT_GATEWAY_HOST = "127.0.0.1";
274
+ /**
275
+ * Cached config to avoid re-reading the file multiple times during provider init.
276
+ */
277
+ let cachedConfig;
278
+ let cachedConfigPath;
279
+ function resolveConfigPath(env) {
280
+ return env?.OPENCLAW_CONFIG_PATH || getEnvString("OPENCLAW_CONFIG_PATH") || path.join(os.homedir(), ".openclaw", "openclaw.json");
281
+ }
282
+ function normalizeGatewayUrl(url, transport) {
283
+ const trimmed = url.trim();
284
+ if (!trimmed) return;
285
+ if (transport === "http") {
286
+ if (trimmed.startsWith("wss://")) return `https://${trimmed.slice(6)}`;
287
+ if (trimmed.startsWith("ws://")) return `http://${trimmed.slice(5)}`;
288
+ return trimmed;
289
+ }
290
+ if (trimmed.startsWith("https://")) return `wss://${trimmed.slice(8)}`;
291
+ if (trimmed.startsWith("http://")) return `ws://${trimmed.slice(7)}`;
292
+ return trimmed;
293
+ }
294
+ function resolveGatewayHost(gatewayConfig) {
295
+ const bind = gatewayConfig?.bind?.trim();
296
+ const customBindHost = gatewayConfig?.customBindHost?.trim();
297
+ if (!bind || bind === "auto" || bind === "loopback" || bind === "lan" || bind === "tailnet" || bind === "0.0.0.0" || bind === "::" || bind === "127.0.0.1" || bind === "localhost" || bind === "::1") return DEFAULT_GATEWAY_HOST;
298
+ if (bind === "custom") {
299
+ if (!customBindHost || customBindHost === "0.0.0.0" || customBindHost === "::" || customBindHost === "127.0.0.1" || customBindHost === "localhost" || customBindHost === "::1") return DEFAULT_GATEWAY_HOST;
300
+ return customBindHost;
301
+ }
302
+ return bind;
303
+ }
304
+ function buildLocalGatewayUrl(gatewayConfig, transport, portOverride) {
305
+ const scheme = transport === "ws" ? gatewayConfig?.tls?.enabled ? "wss" : "ws" : gatewayConfig?.tls?.enabled ? "https" : "http";
306
+ const port = portOverride ?? gatewayConfig?.port ?? 18789;
307
+ return `${scheme}://${resolveGatewayHost(gatewayConfig)}:${port}`;
308
+ }
309
+ function resolveGatewayUrlFromConfig(openclawConfig, transport, portOverride) {
310
+ const gatewayConfig = openclawConfig?.gateway;
311
+ if (!gatewayConfig) return;
312
+ if (gatewayConfig.mode === "remote") {
313
+ const remoteUrl = normalizeGatewayUrl(gatewayConfig.remote?.url ?? "", transport);
314
+ if (remoteUrl) return remoteUrl;
315
+ }
316
+ return buildLocalGatewayUrl(gatewayConfig, transport, portOverride);
317
+ }
318
+ function resolveGatewayPortOverride(env) {
319
+ const trimmedPort = (env?.OPENCLAW_GATEWAY_PORT || getEnvString("OPENCLAW_GATEWAY_PORT"))?.trim();
320
+ if (!trimmedPort || !/^\d+$/.test(trimmedPort)) return;
321
+ const parsedPort = Number(trimmedPort);
322
+ return Number.isInteger(parsedPort) && parsedPort > 0 && parsedPort <= 65535 ? parsedPort : void 0;
323
+ }
324
+ function toAuthSecret(kind, value) {
325
+ const trimmed = value?.trim();
326
+ return trimmed ? {
327
+ kind,
328
+ value: trimmed
329
+ } : void 0;
330
+ }
331
+ function resolveAuthSecretFromConfig(openclawConfig) {
332
+ const gatewayConfig = openclawConfig?.gateway;
333
+ const authMode = gatewayConfig?.auth?.mode?.trim();
334
+ const preferRemoteCredentials = gatewayConfig?.mode === "remote";
335
+ const localToken = toAuthSecret("token", gatewayConfig?.auth?.token);
336
+ const localPassword = toAuthSecret("password", gatewayConfig?.auth?.password);
337
+ const remoteToken = toAuthSecret("token", gatewayConfig?.remote?.token);
338
+ const remotePassword = toAuthSecret("password", gatewayConfig?.remote?.password);
339
+ if (preferRemoteCredentials) {
340
+ if (authMode === "password") return remotePassword || localPassword || remoteToken || localToken;
341
+ if (authMode === "token") return remoteToken || localToken || remotePassword || localPassword;
342
+ return remoteToken || remotePassword || localToken || localPassword;
343
+ }
344
+ if (authMode === "password") return localPassword || remotePassword || localToken || remoteToken;
345
+ if (authMode === "token") return localToken || remoteToken || localPassword || remotePassword;
346
+ return localToken || localPassword || remoteToken || remotePassword;
347
+ }
348
+ /**
349
+ * Read and parse the active OpenClaw configuration file.
350
+ * Results are cached based on file modification time.
351
+ * Returns undefined if the file doesn't exist or can't be parsed.
352
+ */
353
+ function readOpenClawConfig(env) {
354
+ const configPath = resolveConfigPath(env);
355
+ try {
356
+ if (!fs.existsSync(configPath)) return;
357
+ const mtime = fs.statSync(configPath).mtimeMs;
358
+ if (cachedConfig && cachedConfigPath === configPath && cachedConfig.mtime === mtime) return cachedConfig.config;
359
+ const raw = fs.readFileSync(configPath, "utf-8");
360
+ const config = JSON5.parse(raw);
361
+ cachedConfig = {
362
+ config,
363
+ mtime
364
+ };
365
+ cachedConfigPath = configPath;
366
+ return config;
367
+ } catch (err) {
368
+ logger.warn(`Failed to read OpenClaw config at ${configPath}`, { err });
369
+ return;
370
+ }
371
+ }
372
+ /**
373
+ * Auto-detect the OpenClaw gateway URL from config, env overrides, or the active config file.
374
+ */
375
+ function resolveGatewayUrl(config, env) {
376
+ return resolveGatewayTransportUrl(config, env, "http");
377
+ }
378
+ function resolveGatewayWsUrl(config, env) {
379
+ return resolveGatewayTransportUrl(config, env, "ws");
380
+ }
381
+ function resolveGatewayTransportUrl(config, env, transport) {
382
+ const configUrl = config?.gateway_url?.trim();
383
+ if (configUrl) return normalizeGatewayUrl(configUrl, transport) || configUrl;
384
+ const trimmedEnvUrl = (env?.OPENCLAW_GATEWAY_URL || getEnvString("OPENCLAW_GATEWAY_URL") || env?.CLAWDBOT_GATEWAY_URL || getEnvString("CLAWDBOT_GATEWAY_URL"))?.trim();
385
+ if (trimmedEnvUrl) return normalizeGatewayUrl(trimmedEnvUrl, transport) || trimmedEnvUrl;
386
+ const portOverride = resolveGatewayPortOverride(env);
387
+ const resolvedUrl = resolveGatewayUrlFromConfig(readOpenClawConfig(env), transport, portOverride);
388
+ if (resolvedUrl) return resolvedUrl;
389
+ return `${transport === "ws" ? "ws" : "http"}://${DEFAULT_GATEWAY_HOST}:${portOverride ?? 18789}`;
390
+ }
391
+ function resolveAuthSecret(config, env) {
392
+ if (config?.auth_token) return {
393
+ kind: "token",
394
+ value: config.auth_token
395
+ };
396
+ if (config?.auth_password) return {
397
+ kind: "password",
398
+ value: config.auth_password
399
+ };
400
+ const envToken = env?.OPENCLAW_GATEWAY_TOKEN || getEnvString("OPENCLAW_GATEWAY_TOKEN") || env?.CLAWDBOT_GATEWAY_TOKEN || getEnvString("CLAWDBOT_GATEWAY_TOKEN");
401
+ if (envToken) return {
402
+ kind: "token",
403
+ value: envToken
404
+ };
405
+ const envPassword = env?.OPENCLAW_GATEWAY_PASSWORD || getEnvString("OPENCLAW_GATEWAY_PASSWORD") || env?.CLAWDBOT_GATEWAY_PASSWORD || getEnvString("CLAWDBOT_GATEWAY_PASSWORD");
406
+ if (envPassword) return {
407
+ kind: "password",
408
+ value: envPassword
409
+ };
410
+ return resolveAuthSecretFromConfig(readOpenClawConfig(env));
411
+ }
412
+ /**
413
+ * Auto-detect the OpenClaw gateway bearer secret from config, env overrides, or the active
414
+ * config file. OpenClaw accepts either a token or password as the HTTP bearer secret.
415
+ */
416
+ function resolveAuthToken(config, env) {
417
+ return resolveAuthSecret(config, env)?.value;
418
+ }
419
+ /**
420
+ * Build the canonical OpenClaw model id for OpenAI-compatible endpoints.
421
+ */
422
+ function buildOpenClawModelName(agentId) {
423
+ const trimmedAgentId = agentId.trim();
424
+ if (!trimmedAgentId || trimmedAgentId === "default" || trimmedAgentId === "openclaw") return "openclaw/default";
425
+ if (trimmedAgentId.startsWith("openclaw/")) {
426
+ const targetAgentId = trimmedAgentId.slice(9).trim();
427
+ return targetAgentId ? `openclaw/${targetAgentId}` : "openclaw/default";
428
+ }
429
+ if (trimmedAgentId.startsWith("openclaw:")) {
430
+ const targetAgentId = trimmedAgentId.slice(9).trim();
431
+ return targetAgentId ? `openclaw/${targetAgentId}` : "openclaw/default";
432
+ }
433
+ if (trimmedAgentId.startsWith("agent:")) {
434
+ const targetAgentId = trimmedAgentId.slice(6).trim();
435
+ return targetAgentId ? `openclaw/${targetAgentId}` : "openclaw/default";
436
+ }
437
+ return `openclaw/${trimmedAgentId}`;
438
+ }
439
+ function normalizeHeaderValue(value) {
440
+ return value?.trim() || void 0;
441
+ }
442
+ /**
443
+ * Build OpenClaw request context headers shared by HTTP-compatible endpoints.
444
+ */
445
+ function buildOpenClawContextHeaders(config) {
446
+ const headers = {};
447
+ const backendModel = normalizeHeaderValue(config?.backend_model || config?.model_override);
448
+ const messageChannel = normalizeHeaderValue(config?.message_channel);
449
+ const accountId = normalizeHeaderValue(config?.account_id);
450
+ const scopes = config?.scopes?.map((scope) => scope.trim()).filter(Boolean).join(",");
451
+ if (backendModel) headers["x-openclaw-model"] = backendModel;
452
+ if (messageChannel) headers["x-openclaw-message-channel"] = messageChannel;
453
+ if (accountId) headers["x-openclaw-account-id"] = accountId;
454
+ if (scopes) headers["x-openclaw-scopes"] = scopes;
455
+ return headers;
456
+ }
457
+ /**
458
+ * Build common OpenClaw headers for agent-id, session-key, and request context.
459
+ * Note: thinking_level is only supported by the WS Agent provider and is
460
+ * passed as an RPC param there, not as an HTTP header.
461
+ */
462
+ function buildOpenClawHeaders(agentId, config) {
463
+ const headers = { "x-openclaw-agent-id": agentId };
464
+ if (config?.session_key) headers["x-openclaw-session-key"] = config.session_key;
465
+ return {
466
+ ...headers,
467
+ ...buildOpenClawContextHeaders(config)
468
+ };
469
+ }
470
+ /**
471
+ * Build provider options for OpenAI-compatible OpenClaw providers (chat, responses).
472
+ * Resolves gateway URL, auth token, and merges OpenClaw-specific headers.
473
+ */
474
+ function buildOpenClawProviderOptions(agentId, providerOptions) {
475
+ const config = providerOptions.config || {};
476
+ const env = providerOptions.env;
477
+ const gatewayUrl = resolveGatewayUrl(config, env);
478
+ const authToken = resolveAuthToken(config, env);
479
+ return {
480
+ ...providerOptions,
481
+ config: {
482
+ ...config,
483
+ apiBaseUrl: `${gatewayUrl}/v1`,
484
+ ...authToken && { apiKey: authToken },
485
+ apiKeyRequired: false,
486
+ headers: {
487
+ ...config.headers,
488
+ ...buildOpenClawHeaders(agentId, config)
489
+ }
490
+ }
491
+ };
492
+ }
493
+ //#endregion
494
+ //#region src/providers/openclaw/agent.ts
495
+ const OPENCLAW_PROTOCOL_VERSION = 3;
496
+ const CLIENT_ID = "gateway-client";
497
+ const CLIENT_MODE = "cli";
498
+ const CLIENT_ROLE = "operator";
499
+ const DEFAULT_SCOPES = ["operator.read", "operator.write"];
500
+ function normalizeScopes(scopes) {
501
+ const seen = /* @__PURE__ */ new Set();
502
+ const normalized = [];
503
+ for (const scope of scopes || []) {
504
+ const trimmed = scope.trim();
505
+ if (trimmed && !seen.has(trimmed)) {
506
+ seen.add(trimmed);
507
+ normalized.push(trimmed);
508
+ }
509
+ }
510
+ return normalized;
511
+ }
512
+ function getStringArray(value) {
513
+ if (!Array.isArray(value)) return;
514
+ const strings = value.filter((item) => typeof item === "string");
515
+ return strings.length > 0 ? strings : void 0;
516
+ }
517
+ function stripRetryMarker(result) {
518
+ const { retryWithDeviceToken: _retryWithDeviceToken, ...providerResponse } = result;
519
+ return providerResponse;
520
+ }
521
+ function buildOpenClawAgentSessionKey(agentId, sessionKey) {
522
+ const trimmedSessionKey = sessionKey.trim();
523
+ if (!trimmedSessionKey || trimmedSessionKey.toLowerCase().startsWith("agent:")) return trimmedSessionKey;
524
+ const trimmedAgentId = agentId.trim();
525
+ if (!trimmedAgentId || trimmedAgentId.toLowerCase() === "main") return trimmedSessionKey;
526
+ return `agent:${trimmedAgentId}:${trimmedSessionKey}`;
527
+ }
528
+ /**
529
+ * OpenClaw WebSocket Agent Provider
530
+ *
531
+ * Custom provider that uses the native OpenClaw WS RPC protocol to invoke agents.
532
+ * Supports full streaming with event accumulation.
533
+ *
534
+ * Protocol flow:
535
+ * 1. Open WS connection to gateway
536
+ * 2. Receive connect.challenge event → send signed connect request
537
+ * 3. Receive connect response → persist device token and send agent request
538
+ * 4. Receive agent accepted response → send agent.wait
539
+ * 5. Accumulate streaming "agent" events (assistant text/delta, lifecycle errors, error streams)
540
+ * 6. Resolve on agent.wait response
541
+ *
542
+ * Usage:
543
+ * openclaw:agent - default agent (main)
544
+ * openclaw:agent:main - explicit agent ID
545
+ * openclaw:agent:my-agent - custom agent ID
546
+ */
547
+ var OpenClawAgentProvider = class {
548
+ agentId;
549
+ gatewayUrl;
550
+ authKind;
551
+ authSecret;
552
+ openclawConfig;
553
+ timeoutMs;
554
+ scopes;
555
+ hasExplicitScopes;
556
+ activeConnections = /* @__PURE__ */ new Set();
557
+ constructor(agentId, providerOptions = {}) {
558
+ this.agentId = agentId;
559
+ this.openclawConfig = providerOptions.config || {};
560
+ const env = providerOptions.env;
561
+ this.gatewayUrl = resolveGatewayWsUrl(this.openclawConfig, env);
562
+ const authSecret = resolveAuthSecret(this.openclawConfig, env);
563
+ this.authKind = authSecret?.kind;
564
+ this.authSecret = authSecret?.value;
565
+ this.timeoutMs = this.openclawConfig.timeoutMs ?? REQUEST_TIMEOUT_MS;
566
+ this.scopes = normalizeScopes(this.openclawConfig.scopes);
567
+ this.hasExplicitScopes = this.scopes.length > 0;
568
+ if (!this.hasExplicitScopes) this.scopes = [...DEFAULT_SCOPES];
569
+ }
570
+ id() {
571
+ return `openclaw:agent:${this.agentId}`;
572
+ }
573
+ toString() {
574
+ return `[OpenClaw Agent Provider ${this.agentId}]`;
575
+ }
576
+ toJSON() {
577
+ return { provider: this.id() };
578
+ }
579
+ async cleanup() {
580
+ for (const ws of this.activeConnections) ws.close();
581
+ this.activeConnections.clear();
582
+ }
583
+ async callApi(prompt) {
584
+ const sessionKey = buildOpenClawAgentSessionKey(this.agentId, this.openclawConfig.session_key || `promptfoo-${crypto.randomUUID()}`);
585
+ const firstResult = await this.callApiOnce(prompt, sessionKey);
586
+ if (firstResult.retryWithDeviceToken) {
587
+ const retryDeviceToken = this.resolveDeviceTokenForRetry();
588
+ if (retryDeviceToken) return stripRetryMarker(await this.callApiOnce(prompt, sessionKey, retryDeviceToken));
589
+ }
590
+ return stripRetryMarker(firstResult);
591
+ }
592
+ callApiOnce(prompt, sessionKey, retryDeviceToken) {
593
+ return new Promise((resolve) => {
594
+ const wsHeaders = this.buildWebSocketHeaders();
595
+ const ws = wsHeaders ? new WebSocket(this.gatewayUrl, { headers: wsHeaders }) : new WebSocket(this.gatewayUrl);
596
+ this.activeConnections.add(ws);
597
+ let resolved = false;
598
+ let timeout;
599
+ const finish = (result, closeSocket = true) => {
600
+ if (resolved) return;
601
+ resolved = true;
602
+ clearTimeout(timeout);
603
+ this.activeConnections.delete(ws);
604
+ if (closeSocket) ws.close();
605
+ resolve(result);
606
+ };
607
+ const state = {
608
+ agentRequestId: crypto.randomUUID(),
609
+ waitRequestId: crypto.randomUUID(),
610
+ idempotencyKey: crypto.randomUUID(),
611
+ lastText: "",
612
+ lastError: "",
613
+ connected: false,
614
+ prompt,
615
+ sessionKey,
616
+ ws,
617
+ finish,
618
+ retryDeviceToken
619
+ };
620
+ timeout = setTimeout(() => {
621
+ finish({ error: `OpenClaw agent request timed out after ${this.timeoutMs}ms` });
622
+ }, this.timeoutMs);
623
+ ws.on("error", (err) => {
624
+ finish({ error: `OpenClaw WebSocket error: ${err.message}` });
625
+ });
626
+ ws.on("close", () => {
627
+ this.activeConnections.delete(ws);
628
+ if (!resolved) finish({ error: "OpenClaw WebSocket connection closed unexpectedly" }, false);
629
+ });
630
+ ws.on("message", (data) => {
631
+ const frame = this.parseFrame(data);
632
+ if (frame) this.handleFrame(frame, state);
633
+ });
634
+ });
635
+ }
636
+ parseFrame(data) {
637
+ try {
638
+ const frame = JSON.parse(data.toString());
639
+ logger.debug("[OpenClaw Agent] Frame received", {
640
+ type: frame.type,
641
+ event: frame.event,
642
+ id: frame.id,
643
+ ok: frame.ok,
644
+ payloadStatus: frame.payload && typeof frame.payload.status === "string" ? frame.payload.status : void 0,
645
+ payloadStream: frame.payload && typeof frame.payload.stream === "string" ? frame.payload.stream : void 0
646
+ });
647
+ return frame;
648
+ } catch {
649
+ logger.debug("[OpenClaw Agent] Failed to parse WS frame");
650
+ return;
651
+ }
652
+ }
653
+ handleFrame(frame, state) {
654
+ try {
655
+ if (frame.type === "event" && frame.event === "connect.challenge") {
656
+ this.handleConnectChallenge(frame, state);
657
+ return;
658
+ }
659
+ if (frame.type === "res" && !state.connected) {
660
+ this.handleConnectResponse(frame, state);
661
+ return;
662
+ }
663
+ if (frame.type === "res" && frame.id === state.agentRequestId) {
664
+ this.handleAgentAccepted(frame, state);
665
+ return;
666
+ }
667
+ if (frame.type === "event" && frame.event === "agent") {
668
+ this.handleAgentEvent(frame, state);
669
+ return;
670
+ }
671
+ if (frame.type === "res" && frame.id === state.waitRequestId) this.handleAgentWaitResponse(frame, state);
672
+ } catch (err) {
673
+ state.finish({ error: `OpenClaw WebSocket error: ${err instanceof Error ? err.message : String(err)}` });
674
+ }
675
+ }
676
+ handleConnectChallenge(frame, state) {
677
+ state.connectAuthState = this.buildConnectAuthState(state.retryDeviceToken);
678
+ this.sendJson(state.ws, {
679
+ type: "req",
680
+ id: crypto.randomUUID(),
681
+ method: "connect",
682
+ params: this.buildConnectParams(frame, state.connectAuthState)
683
+ });
684
+ }
685
+ handleConnectResponse(frame, state) {
686
+ if (!frame.ok) {
687
+ this.clearStoredDeviceTokenOnFailure(state.connectAuthState, frame.error);
688
+ state.finish({
689
+ error: this.formatConnectError(frame.error),
690
+ retryWithDeviceToken: !state.retryDeviceToken && this.shouldRetryConnectWithDeviceToken(frame.error)
691
+ });
692
+ return;
693
+ }
694
+ state.connected = true;
695
+ this.storeDeviceTokenFromHello(frame.payload, state.connectAuthState);
696
+ this.sendJson(state.ws, {
697
+ type: "req",
698
+ id: state.agentRequestId,
699
+ method: "agent",
700
+ params: this.buildAgentRequestParams(state)
701
+ });
702
+ }
703
+ buildAgentRequestParams(state) {
704
+ return {
705
+ message: state.prompt,
706
+ agentId: this.agentId,
707
+ idempotencyKey: state.idempotencyKey,
708
+ sessionKey: state.sessionKey,
709
+ ...this.openclawConfig.message_channel && { channel: this.openclawConfig.message_channel },
710
+ ...this.openclawConfig.account_id && { accountId: this.openclawConfig.account_id },
711
+ ...this.openclawConfig.thinking_level && { thinking: this.openclawConfig.thinking_level },
712
+ ...this.openclawConfig.extra_system_prompt && { extraSystemPrompt: this.openclawConfig.extra_system_prompt }
713
+ };
714
+ }
715
+ handleAgentAccepted(frame, state) {
716
+ if (!frame.ok) {
717
+ state.finish({ error: `OpenClaw agent error: ${frame.error?.message || "unknown error"}` });
718
+ return;
719
+ }
720
+ const payload = frame.payload;
721
+ state.runId = typeof payload?.runId === "string" && payload.runId.trim() ? payload.runId : void 0;
722
+ if (!state.runId) {
723
+ logger.warn("[OpenClaw Agent] Missing runId in accepted response", {
724
+ agentId: this.agentId,
725
+ payload
726
+ });
727
+ state.finish({ error: "OpenClaw agent error: gateway accepted request without a runId" });
728
+ return;
729
+ }
730
+ this.sendJson(state.ws, {
731
+ type: "req",
732
+ id: state.waitRequestId,
733
+ method: "agent.wait",
734
+ params: {
735
+ runId: state.runId,
736
+ timeoutMs: this.timeoutMs
737
+ }
738
+ });
739
+ }
740
+ handleAgentEvent(frame, state) {
741
+ const payload = frame.payload;
742
+ if (payload?.runId && state.runId && payload.runId !== state.runId) return;
743
+ if (payload?.stream === "lifecycle" && payload.data?.phase === "error") {
744
+ state.lastError = payload.data.error || "OpenClaw agent lifecycle error";
745
+ return;
746
+ }
747
+ if (payload?.stream === "error") {
748
+ state.lastError = payload.data?.error || payload.data?.reason || "OpenClaw agent stream error";
749
+ return;
750
+ }
751
+ if (payload?.stream !== "assistant") return;
752
+ if (typeof payload?.data?.text === "string") {
753
+ state.lastText = payload.data.text;
754
+ state.lastError = "";
755
+ return;
756
+ }
757
+ if (typeof payload?.data?.delta === "string") {
758
+ state.lastText += payload.data.delta;
759
+ state.lastError = "";
760
+ }
761
+ }
762
+ handleAgentWaitResponse(frame, state) {
763
+ if (!frame.ok) {
764
+ state.finish({ error: `OpenClaw agent error: ${frame.error?.message || "unknown error"}` });
765
+ return;
766
+ }
767
+ const payload = frame.payload;
768
+ if (payload?.status === "error") {
769
+ state.finish({ error: `OpenClaw agent error: ${payload.error || "unknown error"}` });
770
+ return;
771
+ }
772
+ if (payload?.status === "timeout") {
773
+ state.finish({ error: `OpenClaw agent error: timed out waiting for run ${state.runId}` });
774
+ return;
775
+ }
776
+ const finalText = typeof payload?.output === "string" ? payload.output : typeof payload?.text === "string" ? payload.text : void 0;
777
+ if (state.lastText || finalText) {
778
+ state.finish({ output: state.lastText || finalText });
779
+ return;
780
+ }
781
+ if (state.lastError) {
782
+ state.finish({ error: `OpenClaw agent error: ${state.lastError}` });
783
+ return;
784
+ }
785
+ state.finish({ output: "No output from agent" });
786
+ }
787
+ sendJson(ws, payload) {
788
+ ws.send(JSON.stringify(payload));
789
+ }
790
+ buildWebSocketHeaders() {
791
+ const headers = {
792
+ ...this.openclawConfig.headers || {},
793
+ ...this.openclawConfig.ws_headers || {}
794
+ };
795
+ return Object.keys(headers).length > 0 ? headers : void 0;
796
+ }
797
+ buildConnectAuthState(retryDeviceToken) {
798
+ const deviceIdentity = this.loadDeviceIdentity();
799
+ const storedOrConfiguredDeviceToken = deviceIdentity ? this.resolveDeviceTokenForConnect(deviceIdentity) : void 0;
800
+ const deviceToken = retryDeviceToken || storedOrConfiguredDeviceToken;
801
+ if (retryDeviceToken) return {
802
+ auth: { deviceToken: retryDeviceToken.token },
803
+ deviceIdentity,
804
+ deviceTokenSource: retryDeviceToken.source,
805
+ kind: "deviceToken",
806
+ scopes: this.resolveScopesForDeviceToken(retryDeviceToken),
807
+ signatureToken: retryDeviceToken.token
808
+ };
809
+ if (this.authSecret && this.authKind) return {
810
+ auth: this.authKind === "password" ? { password: this.authSecret } : { token: this.authSecret },
811
+ deviceIdentity,
812
+ kind: this.authKind,
813
+ scopes: this.scopes,
814
+ signatureToken: this.authKind === "token" ? this.authSecret : null
815
+ };
816
+ if (deviceToken) return {
817
+ auth: { deviceToken: deviceToken.token },
818
+ deviceIdentity,
819
+ deviceTokenSource: deviceToken.source,
820
+ kind: "deviceToken",
821
+ scopes: this.resolveScopesForDeviceToken(deviceToken),
822
+ signatureToken: deviceToken.token
823
+ };
824
+ return {
825
+ deviceIdentity,
826
+ kind: "none",
827
+ scopes: this.scopes,
828
+ signatureToken: null
829
+ };
830
+ }
831
+ buildConnectParams(frame, authState) {
832
+ const nonce = this.extractChallengeNonce(frame);
833
+ const params = {
834
+ minProtocol: OPENCLAW_PROTOCOL_VERSION,
835
+ maxProtocol: OPENCLAW_PROTOCOL_VERSION,
836
+ client: {
837
+ id: CLIENT_ID,
838
+ displayName: "promptfoo",
839
+ version: VERSION,
840
+ platform: process.platform,
841
+ ...this.openclawConfig.device_family && { deviceFamily: this.openclawConfig.device_family },
842
+ mode: CLIENT_MODE
843
+ },
844
+ role: CLIENT_ROLE,
845
+ scopes: authState.scopes,
846
+ caps: [],
847
+ commands: [],
848
+ permissions: {},
849
+ ...authState.auth && { auth: authState.auth }
850
+ };
851
+ const device = this.buildSignedDevice(authState, nonce);
852
+ if (device) params.device = device;
853
+ return params;
854
+ }
855
+ buildSignedDevice(authState, nonce) {
856
+ if (!authState.deviceIdentity || this.openclawConfig.disable_device_auth) return;
857
+ try {
858
+ return buildSignedOpenClawDevice({
859
+ identity: authState.deviceIdentity,
860
+ clientId: CLIENT_ID,
861
+ clientMode: CLIENT_MODE,
862
+ role: CLIENT_ROLE,
863
+ scopes: authState.scopes,
864
+ nonce,
865
+ token: authState.signatureToken,
866
+ platform: process.platform,
867
+ deviceFamily: this.openclawConfig.device_family
868
+ });
869
+ } catch (err) {
870
+ logger.warn("[OpenClaw Agent] Failed to sign device identity; connecting without device auth", { err });
871
+ return;
872
+ }
873
+ }
874
+ loadDeviceIdentity() {
875
+ if (this.openclawConfig.disable_device_auth) return;
876
+ try {
877
+ return loadOrCreateOpenClawDeviceIdentity(this.openclawConfig.device_identity_path);
878
+ } catch (err) {
879
+ logger.warn("[OpenClaw Agent] Failed to load device identity", { err });
880
+ return;
881
+ }
882
+ }
883
+ resolveDeviceTokenForConnect(deviceIdentity) {
884
+ const configDeviceToken = this.openclawConfig.device_token?.trim();
885
+ if (configDeviceToken) return {
886
+ token: configDeviceToken,
887
+ scopes: this.scopes,
888
+ source: "config"
889
+ };
890
+ const storedDeviceToken = loadOpenClawDeviceAuthToken({
891
+ deviceId: deviceIdentity.deviceId,
892
+ role: CLIENT_ROLE,
893
+ filePath: this.openclawConfig.device_auth_path
894
+ });
895
+ if (!storedDeviceToken?.token) return;
896
+ return {
897
+ token: storedDeviceToken.token,
898
+ scopes: this.hasExplicitScopes ? this.scopes : storedDeviceToken.scopes,
899
+ source: "stored"
900
+ };
901
+ }
902
+ resolveDeviceTokenForRetry() {
903
+ const identity = this.loadDeviceIdentity();
904
+ if (!identity) return;
905
+ const deviceToken = this.resolveDeviceTokenForConnect(identity);
906
+ return deviceToken ? {
907
+ ...deviceToken,
908
+ source: "retry"
909
+ } : void 0;
910
+ }
911
+ resolveScopesForDeviceToken(deviceToken) {
912
+ if (this.hasExplicitScopes) return this.scopes;
913
+ return deviceToken.scopes && deviceToken.scopes.length > 0 ? deviceToken.scopes : this.scopes;
914
+ }
915
+ storeDeviceTokenFromHello(payload, authState) {
916
+ if (!authState?.deviceIdentity) return;
917
+ const authPayload = payload?.auth && typeof payload.auth === "object" && !Array.isArray(payload.auth) ? payload.auth : void 0;
918
+ const deviceToken = typeof authPayload?.deviceToken === "string" ? authPayload.deviceToken : "";
919
+ if (!deviceToken.trim()) return;
920
+ const scopes = getStringArray(authPayload?.scopes) || authState.scopes;
921
+ const role = typeof authPayload?.role === "string" && authPayload.role.trim() ? authPayload.role : CLIENT_ROLE;
922
+ storeOpenClawDeviceAuthToken({
923
+ deviceId: authState.deviceIdentity.deviceId,
924
+ role,
925
+ token: deviceToken,
926
+ scopes,
927
+ filePath: this.openclawConfig.device_auth_path
928
+ });
929
+ }
930
+ clearStoredDeviceTokenOnFailure(authState, error) {
931
+ if (!authState?.deviceIdentity || authState.kind !== "deviceToken" || authState.deviceTokenSource === "config") return;
932
+ const detailCode = this.getConnectErrorDetailCode(error);
933
+ const topLevelCode = error?.code;
934
+ if (detailCode !== "AUTH_DEVICE_TOKEN_MISMATCH" && topLevelCode !== "AUTH_DEVICE_TOKEN_MISMATCH") return;
935
+ clearOpenClawDeviceAuthToken({
936
+ deviceId: authState.deviceIdentity.deviceId,
937
+ role: CLIENT_ROLE,
938
+ filePath: this.openclawConfig.device_auth_path
939
+ });
940
+ }
941
+ extractChallengeNonce(frame) {
942
+ const nonce = frame.payload?.nonce;
943
+ return typeof nonce === "string" ? nonce : "";
944
+ }
945
+ getConnectErrorDetailCode(error) {
946
+ return error?.details?.code || error?.code;
947
+ }
948
+ shouldRetryConnectWithDeviceToken(error) {
949
+ const detailCode = this.getConnectErrorDetailCode(error);
950
+ const recommendedNextStep = error?.details?.recommendedNextStep;
951
+ return detailCode === "AUTH_TOKEN_MISMATCH" || error?.details?.canRetryWithDeviceToken === true || recommendedNextStep === "retry_with_device_token";
952
+ }
953
+ formatConnectError(error) {
954
+ const message = error?.message || "unknown error";
955
+ const detailCode = this.getConnectErrorDetailCode(error);
956
+ return detailCode && !message.includes(detailCode) ? `OpenClaw connect failed: ${message} (${detailCode})` : `OpenClaw connect failed: ${message}`;
957
+ }
958
+ };
959
+ //#endregion
960
+ //#region src/providers/openclaw/chat.ts
961
+ /**
962
+ * OpenClaw chat provider extends OpenAI chat completion provider.
963
+ *
964
+ * OpenClaw exposes an OpenAI-compatible HTTP API at /v1/chat/completions.
965
+ * This provider auto-detects gateway URL and bearer auth from:
966
+ * 1. Explicit config (gateway_url, auth_token, auth_password)
967
+ * 2. Environment variables (OPENCLAW_GATEWAY_URL, OPENCLAW_GATEWAY_TOKEN, OPENCLAW_GATEWAY_PASSWORD)
968
+ * 3. The active OpenClaw config file (OPENCLAW_CONFIG_PATH or ~/.openclaw/openclaw.json),
969
+ * including gateway.remote.url and gateway.tls.enabled
970
+ *
971
+ * Usage:
972
+ * openclaw - default agent (main)
973
+ * openclaw:main - specific agent
974
+ * openclaw:coding-agent - named agent
975
+ */
976
+ var OpenClawChatProvider = class extends OpenAiChatCompletionProvider {
977
+ agentId;
978
+ constructor(agentId, providerOptions = {}) {
979
+ super(buildOpenClawModelName(agentId), buildOpenClawProviderOptions(agentId, providerOptions));
980
+ this.agentId = agentId;
981
+ }
982
+ id() {
983
+ return `openclaw:${this.agentId}`;
984
+ }
985
+ toString() {
986
+ return `[OpenClaw Provider ${this.agentId}]`;
987
+ }
988
+ getApiUrlDefault() {
989
+ return `http://${DEFAULT_GATEWAY_HOST}:${DEFAULT_GATEWAY_PORT}/v1`;
990
+ }
991
+ getApiKey() {
992
+ return this.config.apiKey;
993
+ }
994
+ getApiUrl() {
995
+ return this.config.apiBaseUrl || this.getApiUrlDefault();
996
+ }
997
+ };
998
+ //#endregion
999
+ //#region src/providers/openclaw/embedding.ts
1000
+ /**
1001
+ * OpenClaw embedding provider extends the OpenAI-compatible embeddings provider.
1002
+ *
1003
+ * The OpenClaw gateway accepts agent-target model ids at /v1/embeddings and
1004
+ * optionally accepts x-openclaw-model as the backend embedding-model override.
1005
+ */
1006
+ var OpenClawEmbeddingProvider = class extends OpenAiEmbeddingProvider {
1007
+ agentId;
1008
+ constructor(agentId, providerOptions = {}) {
1009
+ super(buildOpenClawModelName(agentId), buildOpenClawProviderOptions(agentId, providerOptions));
1010
+ this.agentId = agentId;
1011
+ }
1012
+ id() {
1013
+ return `openclaw:embedding:${this.agentId}`;
1014
+ }
1015
+ toString() {
1016
+ return `[OpenClaw Embedding Provider ${this.agentId}]`;
1017
+ }
1018
+ getApiUrlDefault() {
1019
+ return `http://${DEFAULT_GATEWAY_HOST}:${DEFAULT_GATEWAY_PORT}/v1`;
1020
+ }
1021
+ getApiKey() {
1022
+ return this.config.apiKey;
1023
+ }
1024
+ getApiUrl() {
1025
+ return this.config.apiBaseUrl || this.getApiUrlDefault();
1026
+ }
1027
+ };
1028
+ //#endregion
1029
+ //#region src/providers/openclaw/responses.ts
1030
+ /**
1031
+ * OpenClaw Responses API Provider
1032
+ *
1033
+ * Extends OpenAI Responses API provider with OpenClaw-specific configuration.
1034
+ * Routes through the OpenClaw gateway's /v1/responses endpoint.
1035
+ *
1036
+ * Requires `gateway.http.endpoints.responses.enabled=true` in OpenClaw config.
1037
+ *
1038
+ * Usage:
1039
+ * openclaw:responses - default agent (main)
1040
+ * openclaw:responses:main - explicit agent ID
1041
+ * openclaw:responses:X - custom agent ID
1042
+ */
1043
+ var OpenClawResponsesProvider = class extends OpenAiResponsesProvider {
1044
+ agentId;
1045
+ constructor(agentId, providerOptions = {}) {
1046
+ super(buildOpenClawModelName(agentId), buildOpenClawProviderOptions(agentId, providerOptions));
1047
+ this.agentId = agentId;
1048
+ }
1049
+ id() {
1050
+ return `openclaw:responses:${this.agentId}`;
1051
+ }
1052
+ toString() {
1053
+ return `[OpenClaw Responses Provider ${this.agentId}]`;
1054
+ }
1055
+ getApiUrlDefault() {
1056
+ return `http://${DEFAULT_GATEWAY_HOST}:${DEFAULT_GATEWAY_PORT}/v1`;
1057
+ }
1058
+ getApiKey() {
1059
+ return this.config.apiKey;
1060
+ }
1061
+ getApiUrl() {
1062
+ return this.config.apiBaseUrl || this.getApiUrlDefault();
1063
+ }
1064
+ async getOpenAiBody(prompt, context, callApiOptions) {
1065
+ const result = await super.getOpenAiBody(prompt, context, callApiOptions);
1066
+ if ("text" in result.body) delete result.body.text;
1067
+ return result;
1068
+ }
1069
+ };
1070
+ //#endregion
1071
+ //#region src/providers/openclaw/tools.ts
1072
+ /**
1073
+ * OpenClaw Tool Invoke Provider
1074
+ *
1075
+ * Simple HTTP provider for direct tool invocation via POST /tools/invoke.
1076
+ * The tool name is extracted from the provider path:
1077
+ * openclaw:tools:sessions_list → tool="sessions_list"
1078
+ *
1079
+ * The prompt is parsed as JSON for tool arguments. If it's not valid JSON,
1080
+ * it's passed as a single `input` argument.
1081
+ *
1082
+ * Usage:
1083
+ * openclaw:tools:sessions_list - invoke the sessions_list tool
1084
+ * openclaw:tools:session_status - invoke the session_status tool
1085
+ *
1086
+ * Optional config:
1087
+ * action - tool sub-action, forwarded as body.action
1088
+ * dry_run - dry-run hint, forwarded as body.dryRun
1089
+ */
1090
+ var OpenClawToolInvokeProvider = class {
1091
+ toolName;
1092
+ gatewayUrl;
1093
+ authToken;
1094
+ openclawConfig;
1095
+ timeoutMs;
1096
+ constructor(toolName, providerOptions = {}) {
1097
+ this.toolName = toolName;
1098
+ this.openclawConfig = providerOptions.config || {};
1099
+ const env = providerOptions.env;
1100
+ this.gatewayUrl = resolveGatewayUrl(this.openclawConfig, env);
1101
+ this.authToken = resolveAuthToken(this.openclawConfig, env);
1102
+ this.timeoutMs = this.openclawConfig.timeoutMs ?? REQUEST_TIMEOUT_MS;
1103
+ }
1104
+ id() {
1105
+ return `openclaw:tools:${this.toolName}`;
1106
+ }
1107
+ toString() {
1108
+ return `[OpenClaw Tool Provider ${this.toolName}]`;
1109
+ }
1110
+ toJSON() {
1111
+ return { provider: this.id() };
1112
+ }
1113
+ async callApi(prompt) {
1114
+ let args;
1115
+ try {
1116
+ args = JSON.parse(prompt);
1117
+ } catch {
1118
+ args = { input: prompt };
1119
+ }
1120
+ const url = `${this.gatewayUrl}/tools/invoke`;
1121
+ const headers = {
1122
+ "Content-Type": "application/json",
1123
+ ...this.openclawConfig.headers || {},
1124
+ ...buildOpenClawContextHeaders(this.openclawConfig)
1125
+ };
1126
+ if (this.authToken) headers["Authorization"] = `Bearer ${this.authToken}`;
1127
+ const body = {
1128
+ tool: this.toolName,
1129
+ ...this.openclawConfig.action && { action: this.openclawConfig.action },
1130
+ args,
1131
+ ...typeof this.openclawConfig.dry_run === "boolean" && { dryRun: this.openclawConfig.dry_run },
1132
+ ...this.openclawConfig.session_key && { sessionKey: this.openclawConfig.session_key }
1133
+ };
1134
+ logger.debug(`[OpenClaw Tool] POST ${url}`, {
1135
+ tool: this.toolName,
1136
+ args
1137
+ });
1138
+ try {
1139
+ const controller = new AbortController();
1140
+ const fetchTimeout = setTimeout(() => controller.abort(), this.timeoutMs);
1141
+ const response = await fetchWithProxy(url, {
1142
+ method: "POST",
1143
+ headers,
1144
+ body: JSON.stringify(body),
1145
+ signal: controller.signal
1146
+ }).finally(() => clearTimeout(fetchTimeout));
1147
+ if (!response.ok) {
1148
+ const text = await response.text();
1149
+ return { error: `OpenClaw tool invoke failed (${response.status}): ${text}` };
1150
+ }
1151
+ const data = await response.json();
1152
+ if (data.ok) return { output: typeof data.result === "string" ? data.result : JSON.stringify(data.result) };
1153
+ if (typeof data.error === "string") return { error: data.error };
1154
+ return { error: data.error?.message || data.error?.type || data.error?.code || "Unknown tool error" };
1155
+ } catch (err) {
1156
+ return { error: `OpenClaw tool invoke error: ${err instanceof Error ? err.message : String(err)}` };
1157
+ }
1158
+ }
1159
+ };
1160
+ //#endregion
1161
+ //#region src/providers/openclaw/index.ts
1162
+ /**
1163
+ * Create an OpenClaw provider from a provider path string.
1164
+ *
1165
+ * Routing:
1166
+ * openclaw → OpenClawChatProvider('main')
1167
+ * openclaw:main → OpenClawChatProvider('main')
1168
+ * openclaw:my-agent → OpenClawChatProvider('my-agent')
1169
+ * openclaw:responses → OpenClawResponsesProvider('main')
1170
+ * openclaw:responses:X → OpenClawResponsesProvider('X')
1171
+ * openclaw:embedding → OpenClawEmbeddingProvider('main')
1172
+ * openclaw:embedding:X → OpenClawEmbeddingProvider('X')
1173
+ * openclaw:embeddings → OpenClawEmbeddingProvider('main')
1174
+ * openclaw:embeddings:X → OpenClawEmbeddingProvider('X')
1175
+ * openclaw:agent → OpenClawAgentProvider('main')
1176
+ * openclaw:agent:X → OpenClawAgentProvider('X')
1177
+ * openclaw:tools:sessions_list → OpenClawToolInvokeProvider('sessions_list')
1178
+ */
1179
+ function createOpenClawProvider(providerPath, providerOptions = {}, env) {
1180
+ const splits = providerPath.split(":");
1181
+ const keyword = splits[1];
1182
+ const opts = {
1183
+ ...providerOptions,
1184
+ env
1185
+ };
1186
+ if (keyword === "responses") return new OpenClawResponsesProvider(splits[2] || "main", opts);
1187
+ if (keyword === "embedding" || keyword === "embeddings") return new OpenClawEmbeddingProvider(splits[2] || "main", opts);
1188
+ if (keyword === "agent") return new OpenClawAgentProvider(splits[2] || "main", opts);
1189
+ if (keyword === "tools") {
1190
+ const toolName = splits.slice(2).join(":");
1191
+ if (!toolName) throw new Error("OpenClaw tools provider requires a tool name: openclaw:tools:<tool-name>");
1192
+ return new OpenClawToolInvokeProvider(toolName, opts);
1193
+ }
1194
+ return new OpenClawChatProvider(splits.length > 1 ? splits.slice(1).join(":") : "main", opts);
1195
+ }
1196
+ //#endregion
1197
+ export { createOpenClawProvider };
1198
+
1199
+ //# sourceMappingURL=openclaw-CwzlQSQX.js.map