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.
- package/README.md +2 -0
- package/dist/src/{ListApp-Du7YVwj5.js → ListApp-DQkFNqE9.js} +1 -1
- package/dist/src/{accounts-B0pgC1oV.js → accounts-DdJ2pHMI.js} +5 -5
- package/dist/src/{accounts-CiBLOnA7.js → accounts-DhMYUUbu.js} +5 -5
- package/dist/src/{accounts-gtkH-5KX.cjs → accounts-Dy17bs4D.cjs} +5 -5
- package/dist/src/{accounts-Bm2D8Db9.js → accounts-F9d_5sMC.js} +6 -6
- package/dist/src/{cometapi-C4xSqeID.cjs → agentic-utils-BpX5b23w.cjs} +24 -62
- package/dist/src/{cometapi-CUQq3H_a.js → agentic-utils-P172hM8B.js} +4 -61
- package/dist/src/agentic-utils-qFlm6zes.js +153 -0
- package/dist/src/{agentic-utils-DS1g3GLF.js → agentic-utils-w68v6_Dz.js} +3 -3
- package/dist/src/{agents-CmvBq8LV.js → agents-8FDnTriG.js} +6 -7
- package/dist/src/{agents-DbRtpYxR.cjs → agents-BahDpe5G.cjs} +255 -20
- package/dist/src/{agents-DgF2zDag.js → agents-C-R_jfzI.js} +255 -20
- package/dist/src/{agents-9qiOy0ho.js → agents-CgaMXvLM.js} +5 -7
- package/dist/src/{agents-Di9DKPzn.cjs → agents-D7-HGxUj.cjs} +5 -7
- package/dist/src/{agents-CBr9A01V.js → agents-DJ35I3Nt.js} +255 -20
- package/dist/src/{agents-cLXA8a_8.js → agents-aYPQLf8W.js} +5 -9
- package/dist/src/{agents-D__IdAlg.js → agents-pQeBEXMm.js} +255 -21
- package/dist/src/{aimlapi-B4rcnZgv.js → aimlapi-BCq3MHeL.js} +8 -14
- package/dist/src/{aimlapi-DHJU_kcV.js → aimlapi-BD6J9oKt.js} +7 -14
- package/dist/src/{aimlapi-BvlNH0gr.cjs → aimlapi-qcK4OT55.cjs} +8 -15
- package/dist/src/{aimlapi-CnkC2HqE.js → aimlapi-sgYnkE54.js} +7 -16
- package/dist/src/app/app/tsconfig.app.tsbuildinfo +1 -0
- package/dist/src/app/assets/Report-CQYFezYu.js +1 -0
- package/dist/src/app/assets/index-BXGkeMwh.css +1 -0
- package/dist/src/app/assets/index-BzJt18Jz.js +385 -0
- package/dist/src/app/assets/rolldown-runtime-COnpUsM8.js +1 -0
- package/dist/src/app/assets/scroll-timeline-D9IT_e8Z.js +1 -0
- package/dist/src/app/assets/sync-IjzpWrOE.js +4 -0
- package/dist/src/app/assets/vendor-charts-BNdH8TCw.js +36 -0
- package/dist/src/app/assets/vendor-markdown-Ch00wnNI.js +29 -0
- package/dist/src/app/assets/vendor-react-CVvmk1UB.js +9 -0
- package/dist/src/app/assets/vendor-utils-BnEYbx2Q.js +37 -0
- package/dist/src/app/index.html +7 -7
- package/dist/src/{audio-Dz3z7s3J.js → audio-B7izf48x.js} +4 -5
- package/dist/src/{audio-CGMyULza.cjs → audio-BQtNuYBj.cjs} +4 -4
- package/dist/src/{audio-Bkv46et0.js → audio-COrn8rM6.js} +4 -4
- package/dist/src/{audio-ClI_AFre.js → audio-DcVKoInv.js} +4 -4
- package/dist/src/{base-CGrhspbK.cjs → base-D-670DX8.cjs} +3 -3
- package/dist/src/{base-CpjcHe4e.js → base-PYJvBE1i.js} +3 -3
- package/dist/src/{base-Dy1V8--Z.js → base-fZ9wgg50.js} +3 -3
- package/dist/src/{base-DLKtKMFh.js → base-yrI1Yal4.js} +3 -3
- package/dist/src/{blobs-BDbfYdrJ.js → blobs-BCZavS8s.js} +4 -4
- package/dist/src/{blobs-CMHN0Qcz.js → blobs-BQWqnnvL.js} +4 -4
- package/dist/src/{blobs-CBO20krR.js → blobs-C-F78Kfn.js} +3 -3
- package/dist/src/{blobs-D23XLin-.cjs → blobs-D2FAd1Q5.cjs} +3 -3
- package/dist/src/cache-BIyPcp5v.cjs +376 -0
- package/dist/src/cache-C4Xb-hNb.js +310 -0
- package/dist/src/cache-Cr9oLMUa.js +3 -0
- package/dist/src/cache-D5NZmMiT.js +310 -0
- package/dist/src/cache-DbLsVWB2.cjs +3 -0
- package/dist/src/cache-mb7c8hbp.js +280 -0
- package/dist/src/{chat-C2jrdPMx.js → chat-0bwXjVP0.js} +3 -13
- package/dist/src/{chat-C1Qst7jL.cjs → chat-BPXSW8Bv.cjs} +3 -13
- package/dist/src/{chat-DJIw17u0.js → chat-BfPaS15_.js} +68 -42
- package/dist/src/{chat-CgF-J-Jj.cjs → chat-CclRbxGf.cjs} +68 -42
- package/dist/src/{chat-BiKyneZl.js → chat-Dr3DUQ0D.js} +68 -42
- package/dist/src/{chat-DqxYYtWA.js → chat-I9izLm49.js} +67 -41
- package/dist/src/{chat-CzkrVDfz.js → chat-MKxMnZJZ.js} +3 -13
- package/dist/src/{chat-qmatte1u.js → chat-mW0ORo8G.js} +3 -14
- package/dist/src/{chatkit-DKyPi1Gs.cjs → chatkit-BoWoSgXl.cjs} +4 -4
- package/dist/src/{chatkit-65VXf5SR.js → chatkit-CJnHRRMM.js} +4 -4
- package/dist/src/{chatkit-Be-Q-a9F.js → chatkit-Cv6AhukM.js} +4 -4
- package/dist/src/{chatkit-BxFvW8KY.js → chatkit-zUIVoDos.js} +4 -4
- package/dist/src/{claude-agent-sdk-Apiy0iaz.js → claude-agent-sdk-BQNuLaAK.js} +41 -18
- package/dist/src/{claude-agent-sdk-D9Z5Pr9X.cjs → claude-agent-sdk-CPJo3dBQ.cjs} +45 -22
- package/dist/src/{claude-agent-sdk-D2bJee9S.js → claude-agent-sdk-Dtq_L-Sc.js} +40 -17
- package/dist/src/{claude-agent-sdk-DfCoW0E6.js → claude-agent-sdk-nfAIcxNf.js} +42 -20
- package/dist/src/{cloud-C0dlstV_.js → cloud-DQZ5sVjW.js} +25 -13
- package/dist/src/cloud-Hphvo8kr.js +3 -0
- package/dist/src/{cloudflare-ai-8TDxHR0x.js → cloudflare-ai-BIB567w6.js} +5 -14
- package/dist/src/{cloudflare-ai-g7PB6VHR.js → cloudflare-ai-DGLte7Py.js} +5 -14
- package/dist/src/{cloudflare-ai-CknbZ5LJ.cjs → cloudflare-ai-Dl3N9OVD.cjs} +6 -15
- package/dist/src/{cloudflare-ai-BxAGvfju.js → cloudflare-ai-DlKr0rY7.js} +5 -15
- package/dist/src/{cloudflare-gateway-B9HWA5wf.js → cloudflare-gateway-BDZrYydE.js} +4 -16
- package/dist/src/{cloudflare-gateway-BSnDmHYo.cjs → cloudflare-gateway-BYDp495F.cjs} +4 -15
- package/dist/src/{cloudflare-gateway-CKDb4dJ8.js → cloudflare-gateway-CiIZHU0Q.js} +5 -164
- package/dist/src/{cloudflare-gateway-CP9QEWYS.js → cloudflare-gateway-DI1HNP5F.js} +4 -15
- package/dist/src/codex-sdk-BAmYE7qy.js +3 -0
- package/dist/src/codex-sdk-C2_M2pl_.cjs +1172 -0
- package/dist/src/codex-sdk-CErXn7qh.js +1165 -0
- package/dist/src/codex-sdk-CWEnH70W.cjs +2 -0
- package/dist/src/codex-sdk-CpqiOqDO.js +1164 -0
- package/dist/src/codex-sdk-Rtky3M4I.js +1165 -0
- package/dist/src/{cometapi-BL9yvj_f.js → cometapi-BUlt_ELa.js} +8 -15
- package/dist/src/{cometapi-DFNiKmSz.js → cometapi-CtJ-mS8R.js} +8 -16
- package/dist/src/cometapi-DT-jlVCB.js +55 -0
- package/dist/src/cometapi-UVOryo4W.cjs +55 -0
- package/dist/src/{completion-CM6oK8PS.cjs → completion-BozdoXba.cjs} +7 -62
- package/dist/src/{completion-5MzrpJxT.js → completion-Dnxn7E-j.js} +8 -57
- package/dist/src/{completion-qRoZAYRB.js → completion-HUe8wDhZ.js} +8 -57
- package/dist/src/{completion-DZ083F31.js → completion-x0a_c2y1.js} +8 -57
- package/dist/src/{createHash-CTQmL3G2.js → createHash-4gFQpDDv.js} +3 -3
- package/dist/src/{createHash-CfZSc0b4.cjs → createHash-B7KvgoOD.cjs} +4 -4
- package/dist/src/{createHash-Da8fMwqB.js → createHash-ChI45QR1.js} +3 -3
- package/dist/src/{createHash-DmPQkvBh.js → createHash-CwDVU5xr.js} +3 -3
- package/dist/src/{docker-ExVyLp0S.js → docker-CQmlA2NU.js} +7 -14
- package/dist/src/{docker-Bb5dcxr8.js → docker-ClnmCf1Z.js} +6 -14
- package/dist/src/{docker-BvfL2BrW.js → docker-DCgsveLD.js} +6 -16
- package/dist/src/{docker-DcF2pRrj.cjs → docker-DS4_Osau.cjs} +7 -15
- package/dist/src/embedding-D3xTseo7.js +59 -0
- package/dist/src/embedding-DD9wa3ae.js +58 -0
- package/dist/src/embedding-I45KG3o7.cjs +63 -0
- package/dist/src/embedding-nFbumxcv.js +58 -0
- package/dist/src/entrypoint.js +69 -6
- package/dist/src/{errors-P6ll7XSJ.js → errors-Cw810C93.js} +1 -1
- package/dist/src/{esm-CaIwzWR5.js → esm-C7PnfdF8.js} +16 -7
- package/dist/src/{esm-CnNt7sI4.cjs → esm-CtEPLdAj.cjs} +15 -6
- package/dist/src/{esm-Cd1AjG1D.js → esm-Dh4dOLlt.js} +15 -6
- package/dist/src/{esm-C03C-mv3.js → esm-tVgYPY-f.js} +27 -18
- package/dist/src/eval-CzJFfFO9.js +3 -0
- package/dist/src/{eval-Dg2nG4v2.js → eval-u4UVafl6.js} +49 -20
- package/dist/src/{evalResult-BDMqrapS.js → evalResult-Bgm9ZH31.js} +7 -7
- package/dist/src/{evalResult-BBRNtX4I.js → evalResult-D3hVYFis.js} +7 -7
- package/dist/src/evalResult-D8MT9p0s.js +3 -0
- package/dist/src/evalResult-DElBuddX.js +2 -0
- package/dist/src/evalResult-Dvc-iucu.cjs +2 -0
- package/dist/src/{evalResult-fuaI8HkH.cjs → evalResult-KZqXl4XP.cjs} +7 -7
- package/dist/src/evaluator-CVessDWe.js +3 -0
- package/dist/src/{evaluator-BhoWwp5b.js → evaluator-IvuDYSvQ.js} +3080 -1254
- package/dist/src/{extractor-D25qpmGX.js → extractor-CAfTSraf.js} +6 -6
- package/dist/src/{extractor-DReVID0K.js → extractor-DNSeBVOJ.js} +6 -6
- package/dist/src/{extractor-C0EVHewb.js → extractor-Dk6bRWkv.js} +6 -6
- package/dist/src/{extractor-pYLLi3wS.cjs → extractor-WVPOrH43.cjs} +6 -6
- package/dist/src/{fetch-HaqdX7U1.js → fetch-B0Z3Oe4k.js} +218 -55
- package/dist/src/{fetch-Dxpd4_sr.js → fetch-BEWnXrrG.js} +195 -45
- package/dist/src/fetch-C7bGKDlQ.js +3 -0
- package/dist/src/{fetch-BPkYtG8K.cjs → fetch-CJU5ELPa.cjs} +223 -48
- package/dist/src/{fetch-Cwxnd8zz.js → fetch-Di00EQrc.js} +218 -55
- package/dist/src/{fileExtensions-Ds-foDzt.js → fileExtensions-AWa2ZML4.js} +1 -1
- package/dist/src/{fileExtensions-LcDYkU4v.js → fileExtensions-BArZuxsI.js} +1 -1
- package/dist/src/{formatDuration-DgBVMN65.js → formatDuration-DZzPsexs.js} +1 -1
- package/dist/src/{genaiTracer-D3fD9dNV.js → genaiTracer-COYDi-tC.js} +6 -2
- package/dist/src/{genaiTracer-C1rxGO8Q.js → genaiTracer-DWdZ28hY.js} +6 -2
- package/dist/src/{genaiTracer-70Z8BIuV.js → genaiTracer-XnrcgDCe.js} +6 -2
- package/dist/src/{genaiTracer-DN4dQywX.cjs → genaiTracer-yRuxj9-L.cjs} +7 -3
- package/dist/src/golang/wrapper.go +1 -1
- package/dist/src/{graders-DU49_J8Y.cjs → graders--zknU_uk.cjs} +5747 -3206
- package/dist/src/graders-BOAzQEUe.cjs +2 -0
- package/dist/src/graders-D4BTsZdG2.js +3 -0
- package/dist/src/graders-DOJK1XpV.js +2 -0
- package/dist/src/graders-NAv9LcBn.js +2 -0
- package/dist/src/{graders-BTeBGqjJ.js → graders-Zy3x0zqX.js} +5727 -3218
- package/dist/src/{graders-Bj_Odv7c.js → graders-eIHhRqoC.js} +5719 -3210
- package/dist/src/{graders-DP7KFFo-.js → graders-pvbReLLn.js} +5728 -3219
- package/dist/src/{image-B0h9VEMc.js → image-9302QVqR.js} +4 -4
- package/dist/src/{image-CHfWvljl.js → image-B5Mv-Z3h.js} +8 -8
- package/dist/src/{image-DS-o-0ph.js → image-DVz2RiMF.js} +8 -8
- package/dist/src/{image-C1madmKh.cjs → image-De2FBmYV.cjs} +4 -4
- package/dist/src/{image-B02ogr_b.js → image-dnoUgPrC.js} +4 -5
- package/dist/src/{image-Dpxa1Jt6.js → image-qUpPvmNZ.js} +8 -8
- package/dist/src/{image-Bb4vWQLM.js → image-u7-rKnYU.js} +4 -4
- package/dist/src/{image-BLmROtN3.cjs → image-x6KqLQl4.cjs} +8 -8
- package/dist/src/index.cjs +3728 -1482
- package/dist/src/index.d.cts +3232 -79
- package/dist/src/index.d.ts +3232 -79
- package/dist/src/index.js +3735 -1490
- package/dist/src/{interactiveCheck-BgLZUIt3.js → interactiveCheck-CLERUB0c.js} +2 -2
- package/dist/src/{knowledgeBase-DOO_BM9b.cjs → knowledgeBase-Bpoe_nLu.cjs} +6 -8
- package/dist/src/{knowledgeBase-D33Ty2l6.js → knowledgeBase-Dgc7CBWF.js} +6 -8
- package/dist/src/{knowledgeBase-B3OoKIej.js → knowledgeBase-RhFPGWDc.js} +6 -8
- package/dist/src/{knowledgeBase-CYTLHOt1.js → knowledgeBase-lm9RXSAm.js} +6 -9
- package/dist/src/{litellm-NbjknEh6.js → litellm-C2kqjxqp.js} +6 -14
- package/dist/src/{litellm-I_hbp_dc.cjs → litellm-CoyI4IAl.cjs} +6 -15
- package/dist/src/{litellm-TrljxD9G.js → litellm-DRjpcSa7.js} +5 -14
- package/dist/src/{litellm-AaeZcZQF.js → litellm-p37R1dzQ.js} +5 -16
- package/dist/src/{logger-DLcq4dWf.js → logger-B88EkIn6.js} +48 -23
- package/dist/src/{logger-Cp1GPUjj.cjs → logger-COuQb2xB.cjs} +77 -22
- package/dist/src/{logger-CT3IKMKA.js → logger-Ct2S6Yx-.js} +48 -23
- package/dist/src/{logger-KkObSCzq.js → logger-DksKw1Qc.js} +48 -23
- package/dist/src/{luma-ray-f6I2fft-.js → luma-ray-B863CmuZ.js} +6 -10
- package/dist/src/{luma-ray-DDsjcgZZ.js → luma-ray-BTTLtqQ8.js} +7 -10
- package/dist/src/{luma-ray-Due0n7di.cjs → luma-ray-BxVKaW2a.cjs} +6 -10
- package/dist/src/{luma-ray-BS2_tY8L.js → luma-ray-KgTCXrZC.js} +6 -12
- package/dist/src/main.d.ts +1 -26
- package/dist/src/main.js +1011 -548
- package/dist/src/{messages-Bs1kC7P4.cjs → messages-811uVVW5.cjs} +74 -19
- package/dist/src/{messages-BS17jdMx.js → messages-BTQz42fn.js} +74 -19
- package/dist/src/{messages-ZJk778GH.js → messages-MYTQ2TWp.js} +74 -19
- package/dist/src/{messages-D0lx5qK7.js → messages-zWbkLLHz.js} +74 -19
- package/dist/src/{meteor-D-SotUw9.js → meteor-CU5UAE-H.js} +1 -1
- package/dist/src/{meteor-DLZZ3osF.cjs → meteor-Co1VQ1u5.cjs} +1 -1
- package/dist/src/{meteor-44VjEACX.js → meteor-DHdzY1Ss.js} +1 -1
- package/dist/src/{meteor-DUiCJRC-.js → meteor-DuAFv6gF.js} +1 -1
- package/dist/src/{modelslab-Bmni6skY.js → modelslab-D0erNWKe.js} +7 -10
- package/dist/src/{modelslab-DRb74SP4.js → modelslab-DIq-6y7x.js} +7 -10
- package/dist/src/{modelslab-CoUX6Jc_.cjs → modelslab-Dk1JAtVo.cjs} +7 -10
- package/dist/src/{modelslab-Bx9IrZfS.js → modelslab-wu9yi5GE.js} +7 -11
- package/dist/src/{nova-reel-BfPq-0Yk.js → nova-reel-CCFRfeRb.js} +7 -10
- package/dist/src/{nova-reel-C_QM18Xn.cjs → nova-reel-CrLXVKQf.cjs} +6 -10
- package/dist/src/{nova-reel-bgjxilYW.js → nova-reel-DQrm74ng.js} +6 -10
- package/dist/src/{nova-reel-D_W1tjMH.js → nova-reel-gr11WG7f.js} +6 -12
- package/dist/src/{nova-sonic-DIGQNR07.js → nova-sonic-BYdp-QLs.js} +5 -7
- package/dist/src/{nova-sonic-CFb5GYhg.js → nova-sonic-B_ZXcUJB.js} +4 -7
- package/dist/src/{nova-sonic-De1HW5fD.js → nova-sonic-TDgrlTk7.js} +4 -9
- package/dist/src/{nova-sonic-zfcljeRp.cjs → nova-sonic-i5tUvXKn.cjs} +4 -7
- package/dist/src/{openai-DElQ-fPX.js → openai-DhVEmgeZ.js} +6 -3
- package/dist/src/{openai-DhbB7eWK.js → openai-Qsvz25mV.js} +6 -3
- package/dist/src/{openai-Cuif0GEt.cjs → openai-URNyItar.cjs} +6 -3
- package/dist/src/{openai-j-sE2O7r.js → openai-iYtrXzOX.js} +6 -3
- package/dist/src/openclaw-CLWrW03k.js +1200 -0
- package/dist/src/openclaw-CnQ363Wi.js +1199 -0
- package/dist/src/openclaw-CwzlQSQX.js +1199 -0
- package/dist/src/openclaw-wX9rtfke.cjs +1205 -0
- package/dist/src/{opencode-sdk-B3CWY9h_.js → opencode-sdk-BUu5Nevv.js} +12 -14
- package/dist/src/{opencode-sdk-BL764Jdi.cjs → opencode-sdk-BZ2idgYA.cjs} +16 -18
- package/dist/src/{opencode-sdk-0j6rTWNb.js → opencode-sdk-BxD8vXp_.js} +12 -15
- package/dist/src/{opencode-sdk-C2y6UkP2.js → opencode-sdk-GI2KaAXq.js} +12 -14
- package/dist/src/{otlpReceiver-C99PPb48.js → otlpReceiver-B2z58l4e.js} +154 -98
- package/dist/src/{otlpReceiver-CGq6LspY.cjs → otlpReceiver-BfcVq2Nq.cjs} +154 -98
- package/dist/src/{otlpReceiver-D89fR-rC.js → otlpReceiver-BntK801g.js} +154 -98
- package/dist/src/{otlpReceiver-CdNBdbsk.js → otlpReceiver-DmVulbhC.js} +154 -98
- package/dist/src/{providerRegistry-CD8MEar9.js → providerRegistry-Bvh8mv85.js} +2 -2
- package/dist/src/{providerRegistry-DM8rZYol.js → providerRegistry-CPQ_CmVO.js} +2 -2
- package/dist/src/{providerRegistry-Civky8Ar.cjs → providerRegistry-CQMdTmHP.cjs} +2 -2
- package/dist/src/{providerRegistry-B0RUOLI_.js → providerRegistry-CWoPjKFZ.js} +2 -2
- package/dist/src/{providers-CgKOSgTR.cjs → providers-1eKkXBKp.cjs} +1435 -930
- package/dist/src/{providers-BlqUifFg.js → providers-BV_KMZje.js} +1419 -944
- package/dist/src/providers-Bp4S-FvO.js +2 -0
- package/dist/src/providers-DV3ax9e_.cjs +3 -0
- package/dist/src/{providers-D8lF1sqW.js → providers-Domz_llv.js} +1427 -952
- package/dist/src/{providers-Dk_6ocUX.js → providers-DruaQfwu.js} +1424 -949
- package/dist/src/providers-iUt5fbAN.js +3 -0
- package/dist/src/providers-u9Enmfok.js +2 -0
- package/dist/src/python/persistent_wrapper.py +0 -5
- package/dist/src/{pythonUtils-D6fwaDSg.js → pythonUtils-C2UQ30Rz.js} +4 -4
- package/dist/src/{pythonUtils-D5nxkQ0P.js → pythonUtils-Cldx7huE.js} +4 -4
- package/dist/src/{pythonUtils-C3py6GC1.js → pythonUtils-CnndUbW-.js} +3 -3
- package/dist/src/{pythonUtils-CTU3Y3lw.cjs → pythonUtils-tAJvvpS-.cjs} +3 -3
- package/dist/src/{quiverai-CIaELU_m.js → quiverai-CtWi6x_g.js} +4 -4
- package/dist/src/{quiverai-PdShCPox.cjs → quiverai-DFotyafY.cjs} +4 -4
- package/dist/src/{quiverai-BbOUOn2L.js → quiverai-DR0SnIQV.js} +4 -4
- package/dist/src/{quiverai-uH-dcTIr.js → quiverai-aPPvXOgn.js} +4 -5
- package/dist/src/render-CH-62LbA.js +135 -0
- package/dist/src/render-CMEpfLaO.js +136 -0
- package/dist/src/{render-Drod8m7K.js → render-CgVDrJmM.js} +2 -3
- package/dist/src/render-DHIZ6_k8.js +135 -0
- package/dist/src/render-DfQSFxGE.cjs +165 -0
- package/dist/src/{responses-DIR9Ud3j.js → responses--OsX2aYW.js} +23 -14
- package/dist/src/{responses-D8SBTL64.cjs → responses-Bi9vBuW_.cjs} +24 -15
- package/dist/src/{responses-CB2jwoAr.js → responses-C-flexAY.js} +24 -15
- package/dist/src/{responses-WNGNYe3K.js → responses-DL9m8CyY.js} +24 -15
- package/dist/src/{rubyUtils-DhCAlxZr.cjs → rubyUtils-B6eljPuh.cjs} +3 -3
- package/dist/src/{rubyUtils-BcuGX77l.js → rubyUtils-CYSQEG4a.js} +3 -3
- package/dist/src/rubyUtils-D1L2d3jb.js +3 -0
- package/dist/src/rubyUtils-DUbq4tff.cjs +2 -0
- package/dist/src/{rubyUtils-BUVePouc.js → rubyUtils-DVLeA2jg.js} +3 -3
- package/dist/src/{rubyUtils-Boc4HZzX.js → rubyUtils-DsGrTx8R.js} +3 -3
- package/dist/src/{sagemaker-CNBxx5CJ.js → sagemaker-BVkaG2-l.js} +14 -18
- package/dist/src/{sagemaker-CemTFp2h.js → sagemaker-BveBvuxm.js} +14 -18
- package/dist/src/{sagemaker-YSyBXQQh.js → sagemaker-D67yzMzs.js} +14 -19
- package/dist/src/{sagemaker-Cl28mZU2.cjs → sagemaker-XnfhheQv.cjs} +14 -18
- package/dist/src/{scanner-BsBlNXNn.js → scanner-1DqWi1Ej.js} +130 -35
- package/dist/src/server/golang/wrapper.go +1 -1
- package/dist/src/server/index.js +3757 -1511
- package/dist/src/server/python/persistent_wrapper.py +0 -5
- package/dist/src/{server-CqzrVGpF.js → server-BNYztJkh.js} +128 -9
- package/dist/src/{server-CuxBbeSY.js → server-BSB45Nt9.js} +127 -8
- package/dist/src/{server-VWgWb00X.js → server-D6Il2Sob.js} +126 -7
- package/dist/src/server-DCtHUqlp.js +3 -0
- package/dist/src/server-DaA2eR26.cjs +2 -0
- package/dist/src/{server-C_7Ax-hA.cjs → server-Dx2TyCH2.cjs} +140 -6
- package/dist/src/{signal-4U3mfRvL.js → signal-CE5G3a7x.js} +3 -3
- package/dist/src/{slack-BmVAVGaK.cjs → slack-1Rhq0EoV.cjs} +2 -2
- package/dist/src/{slack-DCUPTzS2.js → slack-D5Wpy8LM.js} +2 -2
- package/dist/src/{slack-DXMKtA-f.js → slack-DDUe-5MC.js} +2 -2
- package/dist/src/{slack-DOdy_kyv.js → slack-acRb0IqQ.js} +2 -2
- package/dist/src/store-CWOSz6D_.cjs +2 -0
- package/dist/src/{store-Dim__MDd.js → store-CYEy5J2D.js} +17 -5
- package/dist/src/{store-DLlFCC4h.cjs → store-DAAyxcy6.cjs} +17 -5
- package/dist/src/store-DCDBhv7B.js +3 -0
- package/dist/src/{store-CXGFv4aR.js → store-Dn9HUkdW.js} +17 -5
- package/dist/src/{store-DXilxTl-.js → store-M0b1WfYb.js} +17 -5
- package/dist/src/{tables-gftXzE9I.js → tables-C4CH3zRr.js} +3 -3
- package/dist/src/{tables-DLJPUdUE.js → tables-CsWou1Bx.js} +3 -3
- package/dist/src/{tables-6YKwjN9-.js → tables-DQ4WU5tX.js} +3 -3
- package/dist/src/{tables-DPi7wKeM.cjs → tables-DUfh1F7Z.cjs} +3 -3
- package/dist/src/telemetry-C1IqxcdW.js +3 -0
- package/dist/src/telemetry-C4ZEa_es.cjs +2 -0
- package/dist/src/{telemetry-CMrFgtPB.js → telemetry-CQPez_Jp.js} +4 -4
- package/dist/src/{telemetry-DaX14Chu.cjs → telemetry-Dsw_faFj.cjs} +4 -4
- package/dist/src/{telemetry-Dthj_BbD.js → telemetry-Dvqxv3YC.js} +4 -4
- package/dist/src/{telemetry-Cps3mIU-.js → telemetry-dbaJ0E98.js} +4 -4
- package/dist/src/{text-CW1cyrwj.cjs → text-BVi-cLPJ.cjs} +1 -1
- package/dist/src/{text-B_UCRPp2.js → text-CZr46tp_.js} +1 -1
- package/dist/src/{text-TIv0QYnd.js → text-DHxdyQqT.js} +1 -1
- package/dist/src/{text-Db-Wt2u2.js → text-KvuD2Iko.js} +1 -1
- package/dist/src/{tokenUsageUtils-bVa1ga6f.cjs → tokenUsageUtils-Bb7DkZPz.cjs} +7 -3
- package/dist/src/{tokenUsageUtils-NYT-WKS6.js → tokenUsageUtils-C-bmyHoE.js} +7 -3
- package/dist/src/{tokenUsageUtils-DflFMjS0.js → tokenUsageUtils-CXrvO-wA.js} +7 -3
- package/dist/src/{transcription-NLVG9MT1.cjs → transcription-BvjmiYB1.cjs} +12 -16
- package/dist/src/{transcription-BNYURcXg.js → transcription-CJspiD2c.js} +11 -14
- package/dist/src/{transcription-B_OdaHp7.js → transcription-DuWDupG7.js} +10 -14
- package/dist/src/{transcription-s6A-bNrZ.js → transcription-V2HaAmy2.js} +10 -16
- package/dist/src/{transform-DECvGmzp.js → transform-Bbg6A8Jk.js} +4 -4
- package/dist/src/{transform-vNucnNr0.js → transform-CG0ehZNG.js} +11 -7
- package/dist/src/{transform-DuHvhZpj.cjs → transform-CTeuTR3S.cjs} +31 -9
- package/dist/src/{transform-CzK1Q0zl.cjs → transform-CUnzlsbn.cjs} +4 -4
- package/dist/src/{transform-aa6tmVpZ.js → transform-DYX1_Xnh.js} +5 -5
- package/dist/src/transform-DgKlRr73.cjs +2 -0
- package/dist/src/transform-M6ITAESf.js +3 -0
- package/dist/src/{transform-DilY9wbS.js → transform-UN5UGu8U.js} +5 -5
- package/dist/src/{transform-uAytVuyX.js → transform-lQrDE1BQ.js} +11 -7
- package/dist/src/{transform-D5HsjduX.js → transform-zDhMmzwX.js} +11 -7
- package/dist/src/{transformersAvailability-CEVM2GNQ.js → transformersAvailability-CcHusyhw.js} +1 -1
- package/dist/src/{transformersAvailability-CwayUSlh.cjs → transformersAvailability-Cju9mHgR.cjs} +1 -1
- package/dist/src/{transformersAvailability-D6c6ROpT.js → transformersAvailability-DLlROWhg.js} +1 -1
- package/dist/src/{types-DmyIJ-sR.js → types-BGQDAP8i.js} +357 -22
- package/dist/src/{types-CzW2QFyi.js → types-Bgh5SOn6.js} +358 -24
- package/dist/src/{types-C_7nyzr1.cjs → types-CeaeaZdP.cjs} +393 -22
- package/dist/src/{types-Cbd8uOMq.js → types-Dm9JM6Vb.js} +368 -23
- package/dist/src/{util-BHGHw5G1.js → util-BYvQUPp7.js} +138 -36
- package/dist/src/{util-B9vlHIIh.cjs → util-Bxn8emtE.cjs} +15 -168
- package/dist/src/{util-ZzmqNPlg.js → util-C8e5uydV.js} +19 -142
- package/dist/src/{util-CMy69ZgQ.js → util-C9J8ahRn.js} +18 -4
- package/dist/src/{util-BzMcevZc.cjs → util-CN3SrLT4.cjs} +18 -4
- package/dist/src/{util-BV4XUC0n.js → util-D3q0WQ-0.js} +18 -4
- package/dist/src/{util-Dnmk2mBQ.js → util-D9TisOyk.js} +18 -4
- package/dist/src/{util-B3xGByQh.js → util-DDs-7g6-.js} +138 -36
- package/dist/src/{util-Bv6uGDfH.js → util-DvU2Pw8c.js} +138 -36
- package/dist/src/{util-C1CeHl-P.js → util-DxWpWjhc.js} +13 -136
- package/dist/src/{util-BRYkYPTd.js → util-oGMLA7vc.js} +17 -140
- package/dist/src/{util-DGNOS1db.cjs → util-olYL5C6N.cjs} +143 -35
- package/dist/src/{utils-Cz9qXqII.cjs → utils-B05gLxER.cjs} +6 -4
- package/dist/src/{utils-f2-Moju7.js → utils-BLJKfv0y.js} +6 -4
- package/dist/src/{utils-dLokC-eR.js → utils-DJfvjyMj.js} +6 -4
- package/dist/src/{utils-XiOAgly5.js → utils-hXtCYanr.js} +6 -4
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +66 -57
- package/dist/src/app/assets/index-4LKxG2CG.js +0 -439
- package/dist/src/app/assets/index-C3zcsZFQ.css +0 -1
- package/dist/src/app/assets/scroll-timeline-BdJZVXlz.js +0 -1
- package/dist/src/app/assets/sync-9qqYcY-B.js +0 -4
- package/dist/src/app/assets/vendor-charts-BnDWwBlI.js +0 -36
- package/dist/src/app/assets/vendor-markdown-0tekx3KX.js +0 -29
- package/dist/src/app/assets/vendor-react-AtKqiNEf.js +0 -4
- package/dist/src/app/assets/vendor-syntax-D06x6TQF.js +0 -2
- package/dist/src/app/assets/vendor-utils-BvMHZmO7.js +0 -37
- package/dist/src/app/tsconfig.app.tsbuildinfo +0 -1
- package/dist/src/cache-BVeDlD87.js +0 -726
- package/dist/src/cache-C4Nxf52C.js +0 -756
- package/dist/src/cache-CeUpFm3M.cjs +0 -5
- package/dist/src/cache-Dh5WtQps.cjs +0 -816
- package/dist/src/cache-i1P6crbO.js +0 -756
- package/dist/src/cache-n-RCJ-hL.js +0 -6
- package/dist/src/cloud-BBh91EUK.js +0 -4
- package/dist/src/codex-sdk-C6UMlxwV.js +0 -665
- package/dist/src/codex-sdk-DUwKWezN.js +0 -665
- package/dist/src/codex-sdk-GGAw0qbD.js +0 -666
- package/dist/src/codex-sdk-fAO0c3yA.cjs +0 -669
- package/dist/src/eval-B3r2CVXr.js +0 -15
- package/dist/src/evalResult-5xwYnECe.js +0 -12
- package/dist/src/evalResult-71lY93Kj.cjs +0 -10
- package/dist/src/evalResult-Dx5P5cIv.js +0 -10
- package/dist/src/evaluator-Jx6bRZV6.js +0 -36
- package/dist/src/fetch-BxNb_Lp3.js +0 -5
- package/dist/src/graders-B_pgMLS2.js +0 -34
- package/dist/src/graders-DErokPDO.cjs +0 -32
- package/dist/src/graders-DR_uNe54.js +0 -32
- package/dist/src/graders-w3176Wz-.js +0 -32
- package/dist/src/openclaw-CSugPYAr.cjs +0 -586
- package/dist/src/openclaw-DiSz3I5L.js +0 -582
- package/dist/src/openclaw-DuvJKEW5.js +0 -580
- package/dist/src/openclaw-tiVYRtr-.js +0 -580
- package/dist/src/providers-B7V0njNs.js +0 -32
- package/dist/src/providers-BEwbhv0X.js +0 -30
- package/dist/src/providers-CH3C7zf7.js +0 -30
- package/dist/src/providers-zyB6k_38.cjs +0 -31
- package/dist/src/rubyUtils-BUHu6PhO.js +0 -5
- package/dist/src/rubyUtils-CP42kMvq.cjs +0 -4
- package/dist/src/server-DA4Cyrrq.js +0 -7
- package/dist/src/server-Dulb-4-K.cjs +0 -5
- package/dist/src/store-CXS-Q_91.js +0 -6
- package/dist/src/store-eYkaKMwq.cjs +0 -5
- package/dist/src/telemetry-BpMfhthR.cjs +0 -5
- package/dist/src/telemetry-Dw38hanS.js +0 -7
- package/dist/src/tokenUsageUtils-BDGe-iyI.js +0 -138
- package/dist/src/transform-DTGDnAzW.js +0 -6
- 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
|