promptfoo 0.121.4 → 0.121.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (497) hide show
  1. package/dist/src/{ListApp-DQkFNqE9.js → ListApp-DLmM02JS.js} +1 -1
  2. package/dist/src/{accounts-DhMYUUbu.js → accounts-Ca7WIoPY.js} +12 -7
  3. package/dist/src/{accounts-F9d_5sMC.js → accounts-CjFnOPmb.js} +14 -9
  4. package/dist/src/{accounts-Dy17bs4D.cjs → accounts-CmWzeD2d.cjs} +16 -10
  5. package/dist/src/{accounts-DdJ2pHMI.js → accounts-DanM1wq_.js} +13 -8
  6. package/dist/src/{agentic-utils-qFlm6zes.js → agentic-utils-CJ0j3fBi.js} +3 -3
  7. package/dist/src/{agentic-utils-w68v6_Dz.js → agentic-utils-DDEGRV9v.js} +3 -3
  8. package/dist/src/{agentic-utils-BpX5b23w.cjs → agentic-utils-DvPWSUpb.cjs} +8 -7
  9. package/dist/src/{agentic-utils-P172hM8B.js → agentic-utils-TxUEMPYS.js} +2 -2
  10. package/dist/src/{agents-BahDpe5G.cjs → agents-B4sRuXg3.cjs} +7 -6
  11. package/dist/src/{agents-pQeBEXMm.js → agents-B8q7h_ek.js} +5 -5
  12. package/dist/src/{agents-CgaMXvLM.js → agents-CBgJvRkB.js} +21 -10
  13. package/dist/src/{agents-C-R_jfzI.js → agents-CYn2n3QP.js} +4 -4
  14. package/dist/src/{agents-8FDnTriG.js → agents-D-vDNFx4.js} +21 -10
  15. package/dist/src/{agents-aYPQLf8W.js → agents-LrHuQqr1.js} +20 -9
  16. package/dist/src/{agents-DJ35I3Nt.js → agents-QGg76OF-.js} +5 -5
  17. package/dist/src/{agents-D7-HGxUj.cjs → agents-eHZ9nlgA.cjs} +21 -10
  18. package/dist/src/{aimlapi-sgYnkE54.js → aimlapi-CJEbQ0o6.js} +7 -7
  19. package/dist/src/{aimlapi-BD6J9oKt.js → aimlapi-D5HXzZ0s.js} +6 -6
  20. package/dist/src/{aimlapi-qcK4OT55.cjs → aimlapi-T6HGNxNe.cjs} +7 -7
  21. package/dist/src/{aimlapi-BCq3MHeL.js → aimlapi-eYv3a_DK.js} +7 -7
  22. package/dist/src/app/app/tsconfig.app.tsbuildinfo +1 -1
  23. package/dist/src/app/assets/Report-BNHJKN35.js +1 -0
  24. package/dist/src/app/assets/index-BnT6P6sF.js +388 -0
  25. package/dist/src/app/assets/index-yhM8y1PP.css +1 -0
  26. package/dist/src/app/assets/{scroll-timeline-D9IT_e8Z.js → scroll-timeline-RpeTwOvs.js} +1 -1
  27. package/dist/src/app/assets/sync-5gq6fmG4.js +4 -0
  28. package/dist/src/app/assets/vendor-charts-BL9OMNU7.js +36 -0
  29. package/dist/src/app/assets/{vendor-markdown-Ch00wnNI.js → vendor-markdown-BYsQqn7Z.js} +10 -10
  30. package/dist/src/app/assets/{vendor-react-CVvmk1UB.js → vendor-react-CqWgVW6T.js} +2 -2
  31. package/dist/src/app/assets/{vendor-utils-BnEYbx2Q.js → vendor-utils-BHPO71pu.js} +1 -1
  32. package/dist/src/app/index.html +31 -6
  33. package/dist/src/{audio-COrn8rM6.js → audio-BqnRvcWG.js} +3 -3
  34. package/dist/src/{audio-DcVKoInv.js → audio-CPMtV1yR.js} +4 -4
  35. package/dist/src/{audio-B7izf48x.js → audio-DyiebVB3.js} +4 -4
  36. package/dist/src/{audio-BQtNuYBj.cjs → audio-FnxbEnSE.cjs} +4 -4
  37. package/dist/src/authoritativeMarkupInjection-BZIywVjG.js +74 -0
  38. package/dist/src/authoritativeMarkupInjection-DyAXAsSr.js +75 -0
  39. package/dist/src/authoritativeMarkupInjection-F2gBw0lN.cjs +74 -0
  40. package/dist/src/authoritativeMarkupInjection-QEQmFS83.js +74 -0
  41. package/dist/src/{base-PYJvBE1i.js → base-CKLo890h.js} +4 -3
  42. package/dist/src/{base-fZ9wgg50.js → base-Co80MMCi.js} +5 -4
  43. package/dist/src/{base-D-670DX8.cjs → base-DGJW48uz.cjs} +5 -4
  44. package/dist/src/{base-yrI1Yal4.js → base-E9I8zXjz.js} +5 -4
  45. package/dist/src/bestOfN-B3wNzjSB.js +137 -0
  46. package/dist/src/bestOfN-BBsO41z4.js +136 -0
  47. package/dist/src/bestOfN-CAwmg5UL.cjs +140 -0
  48. package/dist/src/bestOfN-_kTi8Bxe.js +136 -0
  49. package/dist/src/{blobs-D2FAd1Q5.cjs → blobs-B0977K1O.cjs} +7 -6
  50. package/dist/src/{blobs-BCZavS8s.js → blobs-CeFdPn_T.js} +3 -3
  51. package/dist/src/{blobs-BQWqnnvL.js → blobs-DODuTK-a.js} +3 -3
  52. package/dist/src/{blobs-C-F78Kfn.js → blobs-Dwef1Ao1.js} +2 -2
  53. package/dist/src/{cache-BIyPcp5v.cjs → cache-CPGUA4Yl.cjs} +135 -25
  54. package/dist/src/cache-Cf7b4pWE.js +3 -0
  55. package/dist/src/{cache-D5NZmMiT.js → cache-DIXbtkNO.js} +125 -10
  56. package/dist/src/{cache-mb7c8hbp.js → cache-DpPWrkTE.js} +128 -12
  57. package/dist/src/{cache-C4Xb-hNb.js → cache-roFAE0cI.js} +126 -11
  58. package/dist/src/{chat-I9izLm49.js → chat-CUCorGiL.js} +12 -12
  59. package/dist/src/{chat-BPXSW8Bv.cjs → chat-DG1wG4w0.cjs} +6 -6
  60. package/dist/src/{chat-BfPaS15_.js → chat-Dabu84Br.js} +12 -12
  61. package/dist/src/{chat-Dr3DUQ0D.js → chat-DqUFcWI0.js} +12 -12
  62. package/dist/src/{chat-CclRbxGf.cjs → chat-DxTDQ83C.cjs} +14 -13
  63. package/dist/src/{chat-MKxMnZJZ.js → chat-GmlolEwo.js} +4 -4
  64. package/dist/src/{chat-0bwXjVP0.js → chat-TP8Qifkh.js} +6 -6
  65. package/dist/src/{chat-mW0ORo8G.js → chat-iwaM5UTQ.js} +6 -6
  66. package/dist/src/{chatkit-zUIVoDos.js → chatkit-B6DWi70Q.js} +4 -4
  67. package/dist/src/{chatkit-BoWoSgXl.cjs → chatkit-BYveR48_.cjs} +6 -5
  68. package/dist/src/{chatkit-Cv6AhukM.js → chatkit-fARZwEfV.js} +3 -3
  69. package/dist/src/{chatkit-CJnHRRMM.js → chatkit-lb6FK02w.js} +4 -4
  70. package/dist/src/{claude-agent-sdk-Dtq_L-Sc.js → claude-agent-sdk-BQNp_y-F.js} +212 -67
  71. package/dist/src/{claude-agent-sdk-BQNuLaAK.js → claude-agent-sdk-D5Jl0SDh.js} +212 -67
  72. package/dist/src/{claude-agent-sdk-CPJo3dBQ.cjs → claude-agent-sdk-DH416NBD.cjs} +218 -72
  73. package/dist/src/{claude-agent-sdk-nfAIcxNf.js → claude-agent-sdk-x1XJ1-pU.js} +212 -67
  74. package/dist/src/{cloud-DQZ5sVjW.js → cloud-D3DiFqH6.js} +3 -3
  75. package/dist/src/cloud-p96PA4MH.js +3 -0
  76. package/dist/src/{cloudflare-ai-BIB567w6.js → cloudflare-ai-B6NVI3ax.js} +4 -4
  77. package/dist/src/{cloudflare-ai-Dl3N9OVD.cjs → cloudflare-ai-CEAW-xQa.cjs} +6 -6
  78. package/dist/src/{cloudflare-ai-DlKr0rY7.js → cloudflare-ai-RFSojyXG.js} +6 -6
  79. package/dist/src/{cloudflare-ai-DGLte7Py.js → cloudflare-ai-r4tbYmWU.js} +6 -6
  80. package/dist/src/{cloudflare-gateway-CiIZHU0Q.js → cloudflare-gateway-BCkLouto.js} +5 -5
  81. package/dist/src/{cloudflare-gateway-DI1HNP5F.js → cloudflare-gateway-BaZ4insB.js} +3 -3
  82. package/dist/src/{cloudflare-gateway-BDZrYydE.js → cloudflare-gateway-CF-Vb-2Z.js} +5 -5
  83. package/dist/src/{cloudflare-gateway-BYDp495F.cjs → cloudflare-gateway-TJMLBj6I.cjs} +5 -5
  84. package/dist/src/codex-app-server-B8KHEiF4.js +1915 -0
  85. package/dist/src/codex-app-server-CnrLBCeA.cjs +1921 -0
  86. package/dist/src/codex-app-server-DIXZ230V.js +1915 -0
  87. package/dist/src/codex-app-server-Dd22dC_N.js +1916 -0
  88. package/dist/src/{codex-sdk-CpqiOqDO.js → codex-sdk-B6Wah8Pa.js} +6 -6
  89. package/dist/src/codex-sdk-BGjVAk23.js +3 -0
  90. package/dist/src/{codex-sdk-C2_M2pl_.cjs → codex-sdk-CFF6gUyi.cjs} +18 -10
  91. package/dist/src/{codex-sdk-Rtky3M4I.js → codex-sdk-CmQABzV3.js} +6 -6
  92. package/dist/src/{codex-sdk-CErXn7qh.js → codex-sdk-D2d54RL8.js} +5 -5
  93. package/dist/src/{cometapi-CtJ-mS8R.js → cometapi-Bu9B8NUY.js} +8 -8
  94. package/dist/src/{cometapi-DT-jlVCB.js → cometapi-CtzNCHKu.js} +7 -7
  95. package/dist/src/{cometapi-UVOryo4W.cjs → cometapi-DHCDlQUI.cjs} +8 -8
  96. package/dist/src/{cometapi-BUlt_ELa.js → cometapi-OBILPLlu.js} +8 -8
  97. package/dist/src/{completion-HUe8wDhZ.js → completion-CO2e1_62.js} +6 -6
  98. package/dist/src/{completion-BozdoXba.cjs → completion-CSYfl2cd.cjs} +6 -6
  99. package/dist/src/{completion-x0a_c2y1.js → completion-DZNxcyfG.js} +6 -6
  100. package/dist/src/{completion-Dnxn7E-j.js → completion-sNvCLTAP.js} +5 -5
  101. package/dist/src/constants-BjJV0cRr.js +6 -0
  102. package/dist/src/constants-DH5XYLKZ.js +7 -0
  103. package/dist/src/constants-DZGEFLsu.js +6 -0
  104. package/dist/src/constants-a2kYssQk.cjs +11 -0
  105. package/dist/src/{createHash-4gFQpDDv.js → createHash-BtbSX3mj.js} +1 -1
  106. package/dist/src/{createHash-CwDVU5xr.js → createHash-CGVzWdjj.js} +1 -1
  107. package/dist/src/{createHash-B7KvgoOD.cjs → createHash-CSiqnK5P.cjs} +2 -2
  108. package/dist/src/{createHash-ChI45QR1.js → createHash-CgRvs4Fn.js} +1 -1
  109. package/dist/src/crescendo-BXEJK_bi.cjs +704 -0
  110. package/dist/src/crescendo-CU_Y2i-m.js +702 -0
  111. package/dist/src/crescendo-J1Xx4_zb.js +703 -0
  112. package/dist/src/crescendo-QiaSLW0d.js +701 -0
  113. package/dist/src/custom-BJfP00Bh.js +619 -0
  114. package/dist/src/custom-CZVn-1-r.js +620 -0
  115. package/dist/src/custom-Cqia7M0D.cjs +621 -0
  116. package/dist/src/custom-notggYVl.js +618 -0
  117. package/dist/src/{docker-DCgsveLD.js → docker-4D1eL6Gq.js} +6 -6
  118. package/dist/src/{docker-ClnmCf1Z.js → docker-BBv1WUDu.js} +5 -5
  119. package/dist/src/{docker-DS4_Osau.cjs → docker-D06JUoe2.cjs} +6 -6
  120. package/dist/src/{docker-CQmlA2NU.js → docker-DdJQBxK9.js} +6 -6
  121. package/dist/src/{embedding-D3xTseo7.js → embedding--UZVe4_7.js} +6 -6
  122. package/dist/src/{embedding-I45KG3o7.cjs → embedding-BbrwopfX.cjs} +6 -6
  123. package/dist/src/{embedding-nFbumxcv.js → embedding-Bi3rxrZF.js} +5 -5
  124. package/dist/src/{embedding-DD9wa3ae.js → embedding-C251p1-8.js} +6 -6
  125. package/dist/src/{errors-Cw810C93.js → errors-9PcUL8BC.js} +1 -1
  126. package/dist/src/{esm-Dh4dOLlt.js → esm-B6whoAcf.js} +2 -2
  127. package/dist/src/{esm-CtEPLdAj.cjs → esm-BIKakvNa.cjs} +8 -7
  128. package/dist/src/{esm-C7PnfdF8.js → esm-BTK1W7lG.js} +1 -1
  129. package/dist/src/{esm-tVgYPY-f.js → esm-Bexx2PFc.js} +2 -2
  130. package/dist/src/{eval-u4UVafl6.js → eval-0VRANImH.js} +21 -21
  131. package/dist/src/{eval-CzJFfFO9.js → eval-DscR5iOM.js} +1 -1
  132. package/dist/src/{evalResult-Bgm9ZH31.js → evalResult-2RRJvFyB.js} +41 -16
  133. package/dist/src/{evalResult-KZqXl4XP.cjs → evalResult-CvtS8h8u.cjs} +51 -15
  134. package/dist/src/evalResult-DqzsS6_W.js +3 -0
  135. package/dist/src/{evalResult-D3hVYFis.js → evalResult-eUkJv9Ko.js} +40 -15
  136. package/dist/src/evaluator-DNdJF1Gv.js +3 -0
  137. package/dist/src/{evaluator-IvuDYSvQ.js → evaluator-DRoiYB2q.js} +1060 -187
  138. package/dist/src/evaluatorHelpers-BsYP_muT.js +511 -0
  139. package/dist/src/evaluatorHelpers-CRqTvSux.cjs +537 -0
  140. package/dist/src/evaluatorHelpers-DuqFFfq7.js +510 -0
  141. package/dist/src/{extractor-CAfTSraf.js → extractor-BR7XAzAL.js} +6 -6
  142. package/dist/src/{extractor-WVPOrH43.cjs → extractor-BdxEtt3J.cjs} +6 -6
  143. package/dist/src/{extractor-DNSeBVOJ.js → extractor-CIW3iN-b.js} +6 -6
  144. package/dist/src/{extractor-Dk6bRWkv.js → extractor-CxRtnaHl.js} +5 -5
  145. package/dist/src/{fetch-B0Z3Oe4k.js → fetch-BufrQtvR.js} +93 -40
  146. package/dist/src/{fetch-BEWnXrrG.js → fetch-DXUnXkVU.js} +89 -40
  147. package/dist/src/{fetch-CJU5ELPa.cjs → fetch-Dw4XZHjj.cjs} +330 -270
  148. package/dist/src/{fetch-Di00EQrc.js → fetch-It34O8Ur.js} +305 -252
  149. package/dist/src/fetch-_YgGd2qv.js +3 -0
  150. package/dist/src/{fileExtensions-bYh77CN8.cjs → fileExtensions-BhdwzYaD.cjs} +24 -1
  151. package/dist/src/{fileExtensions-DnqA1y9x.js → fileExtensions-CXRfY3Ss.js} +12 -2
  152. package/dist/src/{fileExtensions-AWa2ZML4.js → fileExtensions-D4GCJ67J.js} +12 -2
  153. package/dist/src/{formatDuration-DZzPsexs.js → formatDuration-CMVNrYvE.js} +1 -1
  154. package/dist/src/{genaiTracer-yRuxj9-L.cjs → genaiTracer-14nugQQx.cjs} +14 -2
  155. package/dist/src/{genaiTracer-DWdZ28hY.js → genaiTracer-BPVvltoW.js} +2 -2
  156. package/dist/src/{genaiTracer-XnrcgDCe.js → genaiTracer-D18lYzhB.js} +2 -2
  157. package/dist/src/{genaiTracer-COYDi-tC.js → genaiTracer-jJKYsnjc.js} +2 -2
  158. package/dist/src/goat-Ckd3q3AY.js +467 -0
  159. package/dist/src/goat-Qgurm-NP.js +466 -0
  160. package/dist/src/goat-ghadEDdy.js +465 -0
  161. package/dist/src/goat-una6pZGP.cjs +469 -0
  162. package/dist/src/graders-BDT7dif6.js +3 -0
  163. package/dist/src/{graders-eIHhRqoC.js → graders-BGP99PdK.js} +2416 -2224
  164. package/dist/src/{graders-Zy3x0zqX.js → graders-BX0f2tvS.js} +2423 -2226
  165. package/dist/src/{graders-pvbReLLn.js → graders-C0nXU_ZP.js} +1806 -1609
  166. package/dist/src/{graders--zknU_uk.cjs → graders-ClrU2fnd.cjs} +2219 -1949
  167. package/dist/src/hydra-BSNZZm2M.js +543 -0
  168. package/dist/src/hydra-BxdG4nkg.js +541 -0
  169. package/dist/src/hydra-DE4xWwyc.js +542 -0
  170. package/dist/src/hydra-DrJttnvw.cjs +542 -0
  171. package/dist/src/image-B4oBtu6J.js +443 -0
  172. package/dist/src/{image-dnoUgPrC.js → image-BN-hjLL9.js} +4 -4
  173. package/dist/src/{image-9302QVqR.js → image-B_fPIwdg.js} +3 -3
  174. package/dist/src/image-BvUAW344.js +442 -0
  175. package/dist/src/image-Cvjwx1uY.js +442 -0
  176. package/dist/src/{image-De2FBmYV.cjs → image-DfVCGPbI.cjs} +4 -4
  177. package/dist/src/{image-u7-rKnYU.js → image-QzmydkiG.js} +4 -4
  178. package/dist/src/image-X0oY4350.cjs +465 -0
  179. package/dist/src/index.cjs +1689 -558
  180. package/dist/src/index.d.cts +3270 -1624
  181. package/dist/src/index.d.ts +3270 -1624
  182. package/dist/src/index.js +1553 -438
  183. package/dist/src/indirectWebPwn-02ZIghCS.js +259 -0
  184. package/dist/src/indirectWebPwn-BJ22AbQa.cjs +397 -0
  185. package/dist/src/indirectWebPwn-CbjUG0rh.js +385 -0
  186. package/dist/src/indirectWebPwn-CfQJt3gk.cjs +260 -0
  187. package/dist/src/indirectWebPwn-DBQhOjoD.js +260 -0
  188. package/dist/src/indirectWebPwn-OsXnKejv.js +259 -0
  189. package/dist/src/indirectWebPwn-tNx9OZ35.js +385 -0
  190. package/dist/src/indirectWebPwn-uyWdHx04.js +386 -0
  191. package/dist/src/inputVariables-B0qUChbV.js +467 -0
  192. package/dist/src/inputVariables-DUGMb9Ka.js +464 -0
  193. package/dist/src/inputVariables-DXFdi7AI.js +468 -0
  194. package/dist/src/inputVariables-Dq9W-Z3a.cjs +475 -0
  195. package/dist/src/{interactiveCheck-CLERUB0c.js → interactiveCheck-C4QlIuoR.js} +2 -2
  196. package/dist/src/{invariant-BtWWVVhl.js → invariant-B2Rf6avk.js} +1 -1
  197. package/dist/src/{invariant-vgHWClmd.js → invariant-DIYf9sP1.js} +1 -1
  198. package/dist/src/{invariant-kfQ8Bu82.cjs → invariant-QtnLD03y.cjs} +1 -1
  199. package/dist/src/iterative-CpU6i2As.js +490 -0
  200. package/dist/src/iterative-DJQEQpG3.js +491 -0
  201. package/dist/src/iterative-DQBuWM-j.cjs +493 -0
  202. package/dist/src/iterative-FTS4Bz67.js +492 -0
  203. package/dist/src/iterativeImage-BUABMVOA.js +413 -0
  204. package/dist/src/iterativeImage-ByFWkxax.cjs +415 -0
  205. package/dist/src/iterativeImage-BzUapOUi.js +414 -0
  206. package/dist/src/iterativeImage-Doz8mgxF.js +413 -0
  207. package/dist/src/iterativeMeta-B3YiAOc8.js +386 -0
  208. package/dist/src/iterativeMeta-C7APE_P1.js +385 -0
  209. package/dist/src/iterativeMeta-CSS8M6Ds.cjs +385 -0
  210. package/dist/src/iterativeMeta-DgoQ7bLh.js +384 -0
  211. package/dist/src/iterativeTree-B5zxBBSW.js +769 -0
  212. package/dist/src/iterativeTree-CNyIk0Yn.js +768 -0
  213. package/dist/src/iterativeTree-CPMF10ve.cjs +771 -0
  214. package/dist/src/iterativeTree-DvZ7GBwt.js +770 -0
  215. package/dist/src/{knowledgeBase-Dgc7CBWF.js → knowledgeBase-BadkINlJ.js} +24 -10
  216. package/dist/src/{knowledgeBase-RhFPGWDc.js → knowledgeBase-Bi_8sV-H.js} +25 -11
  217. package/dist/src/{knowledgeBase-lm9RXSAm.js → knowledgeBase-CkMljjdg.js} +25 -11
  218. package/dist/src/{knowledgeBase-Bpoe_nLu.cjs → knowledgeBase-DUh34xba.cjs} +25 -11
  219. package/dist/src/{litellm-DRjpcSa7.js → litellm-BKBo0jpC.js} +5 -5
  220. package/dist/src/{litellm-C2kqjxqp.js → litellm-BXyn5kZK.js} +5 -5
  221. package/dist/src/{litellm-p37R1dzQ.js → litellm-CNcfbCfa.js} +4 -4
  222. package/dist/src/{litellm-CoyI4IAl.cjs → litellm-CtAr7bKG.cjs} +5 -5
  223. package/dist/src/{logger-DksKw1Qc.js → logger-BbY6ypFL.js} +2 -2
  224. package/dist/src/{logger-B88EkIn6.js → logger-KD8JjCRJ.js} +2 -2
  225. package/dist/src/{logger-COuQb2xB.cjs → logger-cfNpzI4o.cjs} +13 -55
  226. package/dist/src/{luma-ray-KgTCXrZC.js → luma-ray-BMX1iEB6.js} +5 -5
  227. package/dist/src/{luma-ray-B863CmuZ.js → luma-ray-CR5TSpp4.js} +5 -5
  228. package/dist/src/{luma-ray-BxVKaW2a.cjs → luma-ray-D3FUc2K3.cjs} +9 -8
  229. package/dist/src/{luma-ray-BTTLtqQ8.js → luma-ray-OEMmS1RB.js} +6 -6
  230. package/dist/src/main.js +909 -369
  231. package/dist/src/memoryPoisoning-CM83NWYl.js +107 -0
  232. package/dist/src/memoryPoisoning-D8h9gXJF.js +106 -0
  233. package/dist/src/memoryPoisoning-Dp-btinn.cjs +106 -0
  234. package/dist/src/memoryPoisoning-cLuCoTuJ.js +106 -0
  235. package/dist/src/{messages-BTQz42fn.js → messages-BabO-cX8.js} +273 -17
  236. package/dist/src/{messages-811uVVW5.cjs → messages-DBPir0TQ.cjs} +278 -18
  237. package/dist/src/{messages-zWbkLLHz.js → messages-DGUlSNU7.js} +273 -17
  238. package/dist/src/{messages-MYTQ2TWp.js → messages-vsE_-Lv0.js} +273 -17
  239. package/dist/src/{meteor-DHdzY1Ss.js → meteor--TZYICTI.js} +2 -2
  240. package/dist/src/{meteor-Co1VQ1u5.cjs → meteor-CR226f7Z.cjs} +2 -2
  241. package/dist/src/{meteor-CU5UAE-H.js → meteor-Cl_yd7rJ.js} +2 -2
  242. package/dist/src/{meteor-DuAFv6gF.js → meteor-Dce-_zGQ.js} +1 -1
  243. package/dist/src/mischievousUser-0l8GD7Dp.js +46 -0
  244. package/dist/src/mischievousUser-BUOP9W5r.js +46 -0
  245. package/dist/src/mischievousUser-frFYKxu6.js +47 -0
  246. package/dist/src/mischievousUser-olGgHIVR.cjs +46 -0
  247. package/dist/src/{modelslab-Dk1JAtVo.cjs → modelslab-CNV5bMSk.cjs} +7 -7
  248. package/dist/src/{modelslab-D0erNWKe.js → modelslab-Cogmu4mG.js} +6 -6
  249. package/dist/src/{modelslab-DIq-6y7x.js → modelslab-Dzst7VTU.js} +6 -6
  250. package/dist/src/{modelslab-wu9yi5GE.js → modelslab-EyDczZ5A.js} +7 -7
  251. package/dist/src/{nova-reel-CCFRfeRb.js → nova-reel-BGPNBOMS.js} +6 -6
  252. package/dist/src/{nova-reel-DQrm74ng.js → nova-reel-B_5NKFu1.js} +5 -5
  253. package/dist/src/{nova-reel-gr11WG7f.js → nova-reel-C4eUJGse.js} +5 -5
  254. package/dist/src/{nova-reel-CrLXVKQf.cjs → nova-reel-CjJRxI1X.cjs} +9 -8
  255. package/dist/src/{nova-sonic-BYdp-QLs.js → nova-sonic-BNGmgfFz.js} +4 -4
  256. package/dist/src/{nova-sonic-TDgrlTk7.js → nova-sonic-ChPlh5na.js} +4 -4
  257. package/dist/src/{nova-sonic-B_ZXcUJB.js → nova-sonic-CrV0iaY_.js} +3 -3
  258. package/dist/src/{nova-sonic-i5tUvXKn.cjs → nova-sonic-DuOG9Aun.cjs} +5 -4
  259. package/dist/src/{openai-DhVEmgeZ.js → openai-BMHD2Huo.js} +2 -2
  260. package/dist/src/{openai-URNyItar.cjs → openai-C3uXv8wS.cjs} +2 -2
  261. package/dist/src/{openai-Qsvz25mV.js → openai-CJrsh9n4.js} +2 -2
  262. package/dist/src/{openai-iYtrXzOX.js → openai-zgwBb4Ff.js} +1 -1
  263. package/dist/src/{openclaw-CnQ363Wi.js → openclaw-BIHlu_36.js} +10 -8
  264. package/dist/src/{openclaw-CwzlQSQX.js → openclaw-CF7fMido.js} +9 -7
  265. package/dist/src/{openclaw-wX9rtfke.cjs → openclaw-Dphc01BY.cjs} +18 -15
  266. package/dist/src/{openclaw-CLWrW03k.js → openclaw-zIJAsz3P.js} +10 -8
  267. package/dist/src/{opencode-sdk-BUu5Nevv.js → opencode-sdk-B3vlPLsp.js} +40 -5
  268. package/dist/src/{opencode-sdk-BxD8vXp_.js → opencode-sdk-D05JSgMQ.js} +40 -5
  269. package/dist/src/{opencode-sdk-BZ2idgYA.cjs → opencode-sdk-DoY6GbWw.cjs} +46 -10
  270. package/dist/src/{opencode-sdk-GI2KaAXq.js → opencode-sdk-sRKYHGoI.js} +39 -4
  271. package/dist/src/{otlpReceiver-BntK801g.js → otlpReceiver--gTpSagc.js} +120 -4
  272. package/dist/src/{otlpReceiver-DmVulbhC.js → otlpReceiver-B2eaKC8C.js} +120 -4
  273. package/dist/src/{otlpReceiver-B2z58l4e.js → otlpReceiver-BXjcRqAM.js} +119 -3
  274. package/dist/src/{otlpReceiver-BfcVq2Nq.cjs → otlpReceiver-CvJdBGSc.cjs} +125 -7
  275. package/dist/src/packageParser--MWTSrPW.js +36 -0
  276. package/dist/src/packageParser-CgE-ziRo.js +35 -0
  277. package/dist/src/packageParser-QoCS1FMl.cjs +54 -0
  278. package/dist/src/packageParser-hwwSGnAZ.js +35 -0
  279. package/dist/src/processShim-BBxt7LKO.js +95 -0
  280. package/dist/src/processShim-BcGzU8fY.js +94 -0
  281. package/dist/src/processShim-C_z3aRvF.js +94 -0
  282. package/dist/src/processShim-DSY9BV2T.cjs +98 -0
  283. package/dist/src/promptLength-0qIHyhA5.js +71 -0
  284. package/dist/src/promptLength-4X-Wd8PG.js +72 -0
  285. package/dist/src/promptLength-B9nZEfO6.js +71 -0
  286. package/dist/src/promptLength-BbBbDHNj.cjs +94 -0
  287. package/dist/src/promptfoo-BDrfT30-.js +180 -0
  288. package/dist/src/promptfoo-Cm4hiy1Y.js +180 -0
  289. package/dist/src/promptfoo-Rjp-MeBb.js +181 -0
  290. package/dist/src/promptfoo-b-baRMj-.cjs +205 -0
  291. package/dist/src/prompts-BYMtqPCw.js +259 -0
  292. package/dist/src/prompts-C-bqE1Yp.js +260 -0
  293. package/dist/src/prompts-Cp_Qx5Ml.js +270 -0
  294. package/dist/src/prompts-DHhQsANy.js +259 -0
  295. package/dist/src/prompts-D_QpZ2Dm.js +271 -0
  296. package/dist/src/prompts-hNvWBD3z.cjs +284 -0
  297. package/dist/src/prompts-huDVH2CI.js +270 -0
  298. package/dist/src/prompts-p78Hul5i.cjs +289 -0
  299. package/dist/src/{providerRegistry-CPQ_CmVO.js → providerRegistry-1gB5vtzQ.js} +2 -2
  300. package/dist/src/{providerRegistry-CQMdTmHP.cjs → providerRegistry-CZO_w7ue.cjs} +2 -2
  301. package/dist/src/{providerRegistry-Bvh8mv85.js → providerRegistry-DHcFiVWX.js} +1 -1
  302. package/dist/src/{providerRegistry-CWoPjKFZ.js → providerRegistry-ReCd0sFa.js} +2 -2
  303. package/dist/src/{providers-BV_KMZje.js → providers-B9KzWxAX.js} +10558 -21587
  304. package/dist/src/{providers-DruaQfwu.js → providers-BCCz6_IX.js} +1228 -12196
  305. package/dist/src/{providers-1eKkXBKp.cjs → providers-BDVVIQM6.cjs} +10649 -21843
  306. package/dist/src/{providers-iUt5fbAN.js → providers-BYAn82cf.js} +1 -1
  307. package/dist/src/{providers-Domz_llv.js → providers-DVYRZP4E.js} +10589 -21570
  308. package/dist/src/{pythonUtils-Cldx7huE.js → pythonUtils-CLCgQ9tt.js} +3 -3
  309. package/dist/src/{pythonUtils-CnndUbW-.js → pythonUtils-CgYxeSmO.js} +3 -3
  310. package/dist/src/{pythonUtils-tAJvvpS-.cjs → pythonUtils-Cokhluq3.cjs} +8 -7
  311. package/dist/src/{pythonUtils-C2UQ30Rz.js → pythonUtils-D0BYebvX.js} +3 -3
  312. package/dist/src/{quiverai-DFotyafY.cjs → quiverai-BAp6iTZD.cjs} +4 -4
  313. package/dist/src/{quiverai-aPPvXOgn.js → quiverai-BvIhI_0l.js} +4 -4
  314. package/dist/src/{quiverai-DR0SnIQV.js → quiverai-CdTWPe-A.js} +3 -3
  315. package/dist/src/{quiverai-CtWi6x_g.js → quiverai-Cv7rJKDz.js} +4 -4
  316. package/dist/src/registry-BUJrgjwv.js +124 -0
  317. package/dist/src/registry-DXm1t_x0.js +125 -0
  318. package/dist/src/registry-Dp5EqoXc.js +124 -0
  319. package/dist/src/registry-KCVF1CFC.cjs +124 -0
  320. package/dist/src/{server-D6Il2Sob.js → remoteGeneration-B1_XsKXU.js} +16 -108
  321. package/dist/src/{server-BSB45Nt9.js → remoteGeneration-COpWcmWd.js} +15 -146
  322. package/dist/src/{server-Dx2TyCH2.cjs → remoteGeneration-DS9N3pgB.cjs} +30 -119
  323. package/dist/src/remoteGeneration-DsaSwmG2.js +217 -0
  324. package/dist/src/render-BNTrbmBw.cjs +384 -0
  325. package/dist/src/render-CSP99NLm.js +348 -0
  326. package/dist/src/render-DFfDeYUK.js +347 -0
  327. package/dist/src/{render-CgVDrJmM.js → render-DznWrxGO.js} +2 -2
  328. package/dist/src/render-_6ur1fhE.js +347 -0
  329. package/dist/src/resourceAttributes-D1jP3kL5.js +17 -0
  330. package/dist/src/resourceAttributes-DQbBB--2.js +16 -0
  331. package/dist/src/resourceAttributes-ephgOvdR.cjs +27 -0
  332. package/dist/src/resourceAttributes-v6-I67fn.js +16 -0
  333. package/dist/src/{responses-Bi9vBuW_.cjs → responses-1UFFF9N_.cjs} +51 -16
  334. package/dist/src/{responses-DL9m8CyY.js → responses-B3W2JvOQ.js} +49 -15
  335. package/dist/src/{responses--OsX2aYW.js → responses-B6ktc3Ra.js} +49 -15
  336. package/dist/src/{responses-C-flexAY.js → responses-URRzV8qE.js} +49 -15
  337. package/dist/src/rolldown-runtime-D_mwlA32.cjs +43 -0
  338. package/dist/src/rubyUtils-BYVlQ94c.js +3 -0
  339. package/dist/src/{rubyUtils-DsGrTx8R.js → rubyUtils-CXlFM2rR.js} +3 -3
  340. package/dist/src/{rubyUtils-DVLeA2jg.js → rubyUtils-CnlW8AYb.js} +3 -3
  341. package/dist/src/{rubyUtils-B6eljPuh.cjs → rubyUtils-CqUWBZAt.cjs} +18 -27
  342. package/dist/src/{rubyUtils-CYSQEG4a.js → rubyUtils-DdGojpfv.js} +3 -3
  343. package/dist/src/runtimeTransform-BJOpL9Yc.js +142 -0
  344. package/dist/src/runtimeTransform-Dgh_D7DU.js +143 -0
  345. package/dist/src/runtimeTransform-DigbjU1r.js +142 -0
  346. package/dist/src/runtimeTransform-ON3YYILw.cjs +147 -0
  347. package/dist/src/{sagemaker-BVkaG2-l.js → sagemaker-CujrzP1a.js} +62 -51
  348. package/dist/src/{sagemaker-XnfhheQv.cjs → sagemaker-DzffAqo_.cjs} +65 -53
  349. package/dist/src/{sagemaker-D67yzMzs.js → sagemaker-vhtSV7JI.js} +62 -51
  350. package/dist/src/{sagemaker-BveBvuxm.js → sagemaker-yr1QKeBs.js} +61 -50
  351. package/dist/src/{scanner-1DqWi1Ej.js → scanner-DS0109SS.js} +7 -7
  352. package/dist/src/server/index.js +5105 -605
  353. package/dist/src/server-B8rqV126.cjs +126 -0
  354. package/dist/src/server-BaLytskk.js +3 -0
  355. package/dist/src/server-CMJD10J4.js +107 -0
  356. package/dist/src/server-Ddp8GNMp.js +146 -0
  357. package/dist/src/server-DhMHosWj.js +182 -0
  358. package/dist/src/shared-7pmVZLNO.js +1334 -0
  359. package/dist/src/shared-9WHQ1oNE.js +1335 -0
  360. package/dist/src/{fileExtensions-BArZuxsI.js → shared-BoG7qLMv.js} +12 -2
  361. package/dist/src/shared-D6IjElRI.js +1334 -0
  362. package/dist/src/shared-WkgnDkcg.cjs +1436 -0
  363. package/dist/src/{signal-CE5G3a7x.js → signal-CSurUUyV.js} +3 -3
  364. package/dist/src/simulatedUser-C9aQObBI.js +222 -0
  365. package/dist/src/simulatedUser-Cu601Dd4.cjs +227 -0
  366. package/dist/src/simulatedUser-U_qAHnuB.js +222 -0
  367. package/dist/src/simulatedUser-p3tACcmw.js +223 -0
  368. package/dist/src/{slack-DDUe-5MC.js → slack-Bapo-7_8.js} +2 -2
  369. package/dist/src/{slack-1Rhq0EoV.cjs → slack-DMC1QVEg.cjs} +3 -2
  370. package/dist/src/{slack-D5Wpy8LM.js → slack-DTEFhrMn.js} +2 -2
  371. package/dist/src/{slack-acRb0IqQ.js → slack-k-_CP84Q.js} +1 -1
  372. package/dist/src/storage-BU4qcnOb.js +875 -0
  373. package/dist/src/storage-CA-v9V2v.cjs +911 -0
  374. package/dist/src/storage-CD-GWAdx.js +822 -0
  375. package/dist/src/storage-QdU-SmvD.js +834 -0
  376. package/dist/src/{store-DAAyxcy6.cjs → store-B2NDDooM.cjs} +60 -24
  377. package/dist/src/{store-CYEy5J2D.js → store-DKd5592Q.js} +51 -20
  378. package/dist/src/{store-M0b1WfYb.js → store-HpopRVzl.js} +50 -19
  379. package/dist/src/store-IbiRIF3k.js +3 -0
  380. package/dist/src/strategies-7CS3Alao.cjs +2360 -0
  381. package/dist/src/strategies-CiSeroPH.js +2331 -0
  382. package/dist/src/strategies-DRJjGTIY.js +2333 -0
  383. package/dist/src/{tables-DQ4WU5tX.js → tables-CRSXQ2Ke.js} +2 -2
  384. package/dist/src/{tables-CsWou1Bx.js → tables-CxjU7bBd.js} +3 -3
  385. package/dist/src/{tables-DUfh1F7Z.cjs → tables-DBIJU0WE.cjs} +6 -5
  386. package/dist/src/{tables-C4CH3zRr.js → tables-DafUHOeh.js} +3 -3
  387. package/dist/src/{telemetry-CQPez_Jp.js → telemetry-00ezXr_t.js} +5 -4
  388. package/dist/src/telemetry-ByPqDcKC.js +3 -0
  389. package/dist/src/{telemetry-Dsw_faFj.cjs → telemetry-CJ7FnCsc.cjs} +18 -11
  390. package/dist/src/{telemetry-dbaJ0E98.js → telemetry-DmXYcJNV.js} +5 -4
  391. package/dist/src/{telemetry-Dvqxv3YC.js → telemetry-DwX9XUN5.js} +4 -3
  392. package/dist/src/{text-KvuD2Iko.js → text-Db-Wt2u2.js} +1 -1
  393. package/dist/src/{text-DHxdyQqT.js → text-DwYK5EBn.js} +1 -1
  394. package/dist/src/{text-BVi-cLPJ.cjs → text-nywWsRBM.cjs} +1 -1
  395. package/dist/src/{tokenUsageUtils-C-bmyHoE.js → tokenUsageUtils-BjVkdk18.js} +1 -1
  396. package/dist/src/{tokenUsageUtils-CXrvO-wA.js → tokenUsageUtils-CDet74yk.js} +1 -1
  397. package/dist/src/tokenUsageUtils-CmnQ0G2m.js +142 -0
  398. package/dist/src/{tokenUsageUtils-Bb7DkZPz.cjs → tokenUsageUtils-_B-P8IAi.cjs} +1 -1
  399. package/dist/src/toolAttributes-BAjwcBf0.cjs +103 -0
  400. package/dist/src/toolAttributes-COVgDrBG.js +87 -0
  401. package/dist/src/toolAttributes-DJ9ZEKXD.js +86 -0
  402. package/dist/src/tracingOptions-BnwKCkSB.js +221 -0
  403. package/dist/src/tracingOptions-Chi74lOD.js +219 -0
  404. package/dist/src/tracingOptions-DrbSFaKy.cjs +249 -0
  405. package/dist/src/tracingOptions-ji2OuXbT.js +220 -0
  406. package/dist/src/{transcription-DuWDupG7.js → transcription-B8uIgCYX.js} +5 -5
  407. package/dist/src/{transcription-CJspiD2c.js → transcription-CfU5loSq.js} +6 -6
  408. package/dist/src/{transcription-V2HaAmy2.js → transcription-Dkd22_4K.js} +6 -6
  409. package/dist/src/{transcription-BvjmiYB1.cjs → transcription-mzuf18Mq.cjs} +9 -8
  410. package/dist/src/{transform-lQrDE1BQ.js → transform-BIMynQsA.js} +9 -9
  411. package/dist/src/transform-BnSTnFlp.js +187 -0
  412. package/dist/src/transform-BnSXWmU_2.cjs +221 -0
  413. package/dist/src/transform-CGt7Kt3y2.js +186 -0
  414. package/dist/src/transform-CrPGTsij.js +186 -0
  415. package/dist/src/{transform-CTeuTR3S.cjs → transform-DhNkAUs8.cjs} +13 -12
  416. package/dist/src/{transform-CG0ehZNG.js → transform-DmvYBRll.js} +9 -9
  417. package/dist/src/{transform-zDhMmzwX.js → transform-EtD4jAWi.js} +9 -9
  418. package/dist/src/{transformersAvailability-CcHusyhw.js → transformersAvailability-0ThtPved.js} +1 -1
  419. package/dist/src/transformersAvailability-BYydDE5U.js +35 -0
  420. package/dist/src/{transformersAvailability-DLlROWhg.js → transformersAvailability-BvyU9vDD.js} +1 -1
  421. package/dist/src/{transformersAvailability-Cju9mHgR.cjs → transformersAvailability-BytPvKUW.cjs} +1 -1
  422. package/dist/src/{types-Dm9JM6Vb.js → types-BFevViUY.js} +115 -19
  423. package/dist/src/{types-Bgh5SOn6.js → types-BJQBBPTP.js} +115 -19
  424. package/dist/src/{types-CeaeaZdP.cjs → types-CxJvaY2S.cjs} +357 -172
  425. package/dist/src/{types-BGQDAP8i.js → types-D6glLbdF.js} +271 -170
  426. package/dist/src/{util-BYvQUPp7.js → util--WMgw7wM.js} +28 -8
  427. package/dist/src/{util-C9J8ahRn.js → util-5WnCSb0h.js} +72 -48
  428. package/dist/src/{util-CN3SrLT4.cjs → util-BSIuSLVK.cjs} +74 -49
  429. package/dist/src/{util-C8e5uydV.js → util-Bx677_k2.js} +154 -147
  430. package/dist/src/util-CN8om2rz.cjs +386 -0
  431. package/dist/src/{util-DDs-7g6-.js → util-CoQWM76y.js} +28 -8
  432. package/dist/src/util-DNl96nNs.js +327 -0
  433. package/dist/src/{util-DxWpWjhc.js → util-DURocbYR.js} +667 -507
  434. package/dist/src/util-Df8YMvS1.js +327 -0
  435. package/dist/src/{util-DvU2Pw8c.js → util-DiQ3QvBB.js} +28 -8
  436. package/dist/src/{util-oGMLA7vc.js → util-I-Rf-KaD.js} +862 -577
  437. package/dist/src/{util-olYL5C6N.cjs → util-IYzs5Y04.cjs} +33 -7
  438. package/dist/src/{util-D9TisOyk.js → util-LKTmNsMQ.js} +71 -47
  439. package/dist/src/{util-Bxn8emtE.cjs → util-SPsvFONY.cjs} +738 -582
  440. package/dist/src/{util-D3q0WQ-0.js → util-efByNxcr.js} +72 -48
  441. package/dist/src/util-kDURhgJW.js +328 -0
  442. package/dist/src/{utils-DJfvjyMj.js → utils-B0lzitHZ.js} +3 -3
  443. package/dist/src/{utils-BLJKfv0y.js → utils-BFOh20Gb.js} +3 -3
  444. package/dist/src/{utils-hXtCYanr.js → utils-BGY69tk_.js} +2 -2
  445. package/dist/src/{utils-B05gLxER.cjs → utils-Ve6kuJsa.cjs} +3 -3
  446. package/dist/src/version-BK20a4sw.js +16 -0
  447. package/dist/src/version-BWCSaByA.cjs +27 -0
  448. package/dist/src/version-eRkNuGv8.js +17 -0
  449. package/dist/src/version-lpHV_53E.js +16 -0
  450. package/dist/tsconfig.tsbuildinfo +1 -1
  451. package/package.json +56 -28
  452. package/dist/src/app/assets/Report-CQYFezYu.js +0 -1
  453. package/dist/src/app/assets/index-BXGkeMwh.css +0 -1
  454. package/dist/src/app/assets/index-BzJt18Jz.js +0 -385
  455. package/dist/src/app/assets/sync-IjzpWrOE.js +0 -4
  456. package/dist/src/app/assets/vendor-charts-BNdH8TCw.js +0 -36
  457. package/dist/src/cache-Cr9oLMUa.js +0 -3
  458. package/dist/src/cache-DbLsVWB2.cjs +0 -3
  459. package/dist/src/cloud-Hphvo8kr.js +0 -3
  460. package/dist/src/codex-sdk-BAmYE7qy.js +0 -3
  461. package/dist/src/codex-sdk-CWEnH70W.cjs +0 -2
  462. package/dist/src/evalResult-D8MT9p0s.js +0 -3
  463. package/dist/src/evalResult-DElBuddX.js +0 -2
  464. package/dist/src/evalResult-Dvc-iucu.cjs +0 -2
  465. package/dist/src/evaluator-CVessDWe.js +0 -3
  466. package/dist/src/fetch-C7bGKDlQ.js +0 -3
  467. package/dist/src/graders-BOAzQEUe.cjs +0 -2
  468. package/dist/src/graders-D4BTsZdG2.js +0 -3
  469. package/dist/src/graders-DOJK1XpV.js +0 -2
  470. package/dist/src/graders-NAv9LcBn.js +0 -2
  471. package/dist/src/image-B5Mv-Z3h.js +0 -257
  472. package/dist/src/image-DVz2RiMF.js +0 -258
  473. package/dist/src/image-qUpPvmNZ.js +0 -257
  474. package/dist/src/image-x6KqLQl4.cjs +0 -280
  475. package/dist/src/providers-Bp4S-FvO.js +0 -2
  476. package/dist/src/providers-DV3ax9e_.cjs +0 -3
  477. package/dist/src/providers-u9Enmfok.js +0 -2
  478. package/dist/src/render-CH-62LbA.js +0 -135
  479. package/dist/src/render-CMEpfLaO.js +0 -136
  480. package/dist/src/render-DHIZ6_k8.js +0 -135
  481. package/dist/src/render-DfQSFxGE.cjs +0 -165
  482. package/dist/src/rubyUtils-D1L2d3jb.js +0 -3
  483. package/dist/src/rubyUtils-DUbq4tff.cjs +0 -2
  484. package/dist/src/server-BNYztJkh.js +0 -385
  485. package/dist/src/server-DCtHUqlp.js +0 -3
  486. package/dist/src/server-DaA2eR26.cjs +0 -2
  487. package/dist/src/store-CWOSz6D_.cjs +0 -2
  488. package/dist/src/store-DCDBhv7B.js +0 -3
  489. package/dist/src/store-Dn9HUkdW.js +0 -240
  490. package/dist/src/telemetry-C1IqxcdW.js +0 -3
  491. package/dist/src/telemetry-C4ZEa_es.cjs +0 -2
  492. package/dist/src/transform-Bbg6A8Jk.js +0 -216
  493. package/dist/src/transform-CUnzlsbn.cjs +0 -228
  494. package/dist/src/transform-DYX1_Xnh.js +0 -216
  495. package/dist/src/transform-DgKlRr73.cjs +0 -2
  496. package/dist/src/transform-M6ITAESf.js +0 -3
  497. package/dist/src/transform-UN5UGu8U.js +0 -213
@@ -0,0 +1,1335 @@
1
+ #!/usr/bin/env node
2
+ import { D as getEnvInt, E as getEnvFloat, N as state, T as getEnvBool, h as sanitizeUrl, s as logger, x as safeJsonStringify } from "./logger-BbY6ypFL.js";
3
+ import { f as sleep, h as parseRetryAfter, l as withFetchRetryContext, m as parseRateLimitHeaders } from "./fetch-DXUnXkVU.js";
4
+ import { t as invariant } from "./invariant-B2Rf6avk.js";
5
+ import { m as isProviderOptions, p as isApiProvider } from "./types-BFevViUY.js";
6
+ import { r as isTransientConnectionError } from "./errors-9PcUL8BC.js";
7
+ import { t as OpenAiChatCompletionProvider } from "./chat-DqUFcWI0.js";
8
+ import { a as createEmptyTokenUsage, r as accumulateTokenUsage } from "./tokenUsageUtils-CDet74yk.js";
9
+ import { t as PromptfooChatCompletionProvider } from "./promptfoo-Rjp-MeBb.js";
10
+ import { o as transform, r as TransformInputType } from "./transform-BnSTnFlp.js";
11
+ import { n as isBlobStorageEnabled, r as shouldAttemptRemoteBlobUpload, t as extractAndStoreBinaryData } from "./extractor-CIW3iN-b.js";
12
+ import { i as throwIfTargetPromptExceedsMaxChars } from "./promptLength-4X-Wd8PG.js";
13
+ import { createHash, randomUUID } from "crypto";
14
+ import { EventEmitter } from "events";
15
+ //#region src/scheduler/types.ts
16
+ /**
17
+ * Default rate limit detection for ProviderResponse.
18
+ * Checks HTTP status, error fields, and error messages.
19
+ */
20
+ function isProviderResponseRateLimited(result, error) {
21
+ return Boolean(result?.metadata?.http?.status === 429 || result?.error?.includes?.("429") || result?.error?.toLowerCase?.().includes?.("rate limit") || error?.message?.includes("429") || error?.message?.toLowerCase().includes("rate limit") || error?.message?.toLowerCase().includes("too many requests"));
22
+ }
23
+ /**
24
+ * Extract rate limit headers from ProviderResponse.
25
+ * Headers can be at metadata.http.headers or metadata.headers.
26
+ */
27
+ function getProviderResponseHeaders(result) {
28
+ return result?.metadata?.http?.headers || result?.metadata?.headers;
29
+ }
30
+ //#endregion
31
+ //#region src/scheduler/providerWrapper.ts
32
+ /**
33
+ * Provider wrapper that adds rate limiting to any ApiProvider.
34
+ *
35
+ * Use this to wrap providers before passing them to redteam/assertion
36
+ * code paths that bypass the main evaluator.
37
+ */
38
+ /**
39
+ * Symbol to mark providers that have already been wrapped.
40
+ * Prevents double-wrapping which could cause issues.
41
+ */
42
+ const WRAPPED_SYMBOL = Symbol.for("promptfoo.rateLimitWrapped");
43
+ /**
44
+ * Check if a provider is already wrapped with rate limiting.
45
+ */
46
+ function isRateLimitWrapped(provider) {
47
+ return provider[WRAPPED_SYMBOL] === true;
48
+ }
49
+ /**
50
+ * Create rate limit detection options for ProviderResponse.
51
+ * Shared between providerWrapper and evaluator for consistency.
52
+ */
53
+ function createProviderRateLimitOptions() {
54
+ return {
55
+ getHeaders: getProviderResponseHeaders,
56
+ isRateLimited: isProviderResponseRateLimited,
57
+ getRetryAfter: (result, error) => {
58
+ const rawHeaders = getProviderResponseHeaders(result);
59
+ if (rawHeaders) {
60
+ const headers = {};
61
+ for (const [key, value] of Object.entries(rawHeaders)) headers[key.toLowerCase()] = value;
62
+ if (headers["retry-after-ms"]) {
63
+ const ms = parseInt(headers["retry-after-ms"], 10);
64
+ if (!isNaN(ms) && ms >= 0) return ms;
65
+ }
66
+ if (headers["retry-after"]) {
67
+ const parsed = parseRetryAfter(headers["retry-after"]);
68
+ if (parsed !== null) return parsed;
69
+ }
70
+ }
71
+ const match = error?.message?.match(/\bretry after (\d+)\b/i);
72
+ if (match) return parseInt(match[1], 10) * 1e3;
73
+ }
74
+ };
75
+ }
76
+ /**
77
+ * Wrap a provider with rate limiting.
78
+ *
79
+ * The wrapped provider will use the registry for all callApi calls,
80
+ * automatically handling rate limits, retries, and adaptive concurrency.
81
+ *
82
+ * @param provider - The provider to wrap
83
+ * @param registry - The rate limit registry to use
84
+ * @returns A wrapped provider that applies rate limiting
85
+ */
86
+ function wrapProviderWithRateLimiting(provider, registry) {
87
+ if (isRateLimitWrapped(provider)) return provider;
88
+ const originalCallApi = provider.callApi.bind(provider);
89
+ const wrappedProvider = {
90
+ ...provider,
91
+ id: () => provider.id(),
92
+ callApi: async (prompt, context, options) => {
93
+ return registry.execute(provider, () => originalCallApi(prompt, context, options), createProviderRateLimitOptions());
94
+ }
95
+ };
96
+ wrappedProvider[WRAPPED_SYMBOL] = true;
97
+ return wrappedProvider;
98
+ }
99
+ //#endregion
100
+ //#region src/scheduler/adaptiveConcurrency.ts
101
+ const DEFAULT_MIN_CONCURRENCY = 1;
102
+ const BACKOFF_FACTOR = .5;
103
+ const RECOVERY_FACTOR = 1.5;
104
+ const RECOVERY_THRESHOLD = 5;
105
+ const WARNING_THRESHOLD = .1;
106
+ /**
107
+ * Manages adaptive concurrency based on rate limit feedback.
108
+ *
109
+ * Recovery path with constants (initial=10, min=1):
110
+ * 1 → ceil(1.5) = 2 (5 successes)
111
+ * 2 → ceil(3.0) = 3 (5 successes)
112
+ * 3 → ceil(4.5) = 5 (5 successes)
113
+ * 5 → ceil(7.5) = 8 (5 successes)
114
+ * 8 → ceil(12) = 10 (5 successes, capped at initial)
115
+ *
116
+ * Total: 25 requests to fully recover from min=1 to initial=10
117
+ */
118
+ var AdaptiveConcurrency = class {
119
+ current;
120
+ initial;
121
+ min;
122
+ consecutiveSuccesses = 0;
123
+ constructor(initial, min = DEFAULT_MIN_CONCURRENCY) {
124
+ this.initial = initial;
125
+ this.current = initial;
126
+ this.min = Math.min(initial, Math.max(1, min));
127
+ }
128
+ /**
129
+ * Called on successful request.
130
+ * May increase concurrency after sustained success.
131
+ */
132
+ recordSuccess() {
133
+ this.consecutiveSuccesses++;
134
+ if (this.consecutiveSuccesses >= RECOVERY_THRESHOLD && this.current < this.initial) {
135
+ const previous = this.current;
136
+ this.current = Math.min(this.initial, Math.ceil(this.current * RECOVERY_FACTOR));
137
+ this.consecutiveSuccesses = 0;
138
+ return {
139
+ changed: previous !== this.current,
140
+ previous,
141
+ current: this.current,
142
+ reason: "recovery"
143
+ };
144
+ }
145
+ return {
146
+ changed: false,
147
+ previous: this.current,
148
+ current: this.current,
149
+ reason: "recovery"
150
+ };
151
+ }
152
+ /**
153
+ * Called on rate limit (429).
154
+ * Reduces concurrency immediately.
155
+ */
156
+ recordRateLimit() {
157
+ this.consecutiveSuccesses = 0;
158
+ const previous = this.current;
159
+ this.current = Math.max(this.min, Math.floor(this.current * BACKOFF_FACTOR));
160
+ return {
161
+ changed: previous !== this.current,
162
+ previous,
163
+ current: this.current,
164
+ reason: "ratelimit"
165
+ };
166
+ }
167
+ /**
168
+ * Called when approaching rate limit.
169
+ * Proactively reduces concurrency based on remaining ratio.
170
+ *
171
+ * Formula:
172
+ * - At 10% remaining: reduce to 60% of current
173
+ * - At 5% remaining: reduce to 40% of current
174
+ * - At 1% remaining: reduce to 20% of current
175
+ *
176
+ * Linear scaling: reductionFactor = 0.2 + (ratio / WARNING_THRESHOLD) * 0.4
177
+ */
178
+ recordApproachingLimit(ratio) {
179
+ const clampedRatio = Math.max(0, Math.min(1, ratio));
180
+ if (clampedRatio >= .1 || this.current <= this.min) return {
181
+ changed: false,
182
+ previous: this.current,
183
+ current: this.current,
184
+ reason: "proactive"
185
+ };
186
+ const previous = this.current;
187
+ const reductionFactor = .2 + clampedRatio / WARNING_THRESHOLD * .4;
188
+ this.current = Math.max(this.min, Math.floor(this.current * reductionFactor));
189
+ return {
190
+ changed: previous !== this.current,
191
+ previous,
192
+ current: this.current,
193
+ reason: "proactive"
194
+ };
195
+ }
196
+ getCurrent() {
197
+ return this.current;
198
+ }
199
+ getMin() {
200
+ return this.min;
201
+ }
202
+ getInitial() {
203
+ return this.initial;
204
+ }
205
+ };
206
+ //#endregion
207
+ //#region src/scheduler/retryPolicy.ts
208
+ const DEFAULT_RETRY_POLICY = {
209
+ maxRetries: 3,
210
+ baseDelayMs: 1e3,
211
+ maxDelayMs: 6e4,
212
+ jitterFactor: .2
213
+ };
214
+ /**
215
+ * Calculate delay for retry attempt.
216
+ * Prefers server-specified Retry-After when available.
217
+ */
218
+ function getRetryDelay(attempt, policy, serverRetryAfterMs) {
219
+ if (serverRetryAfterMs !== void 0 && serverRetryAfterMs >= 0) {
220
+ if (serverRetryAfterMs === 0) return 0;
221
+ const jitter = serverRetryAfterMs * policy.jitterFactor * Math.random();
222
+ return Math.min(serverRetryAfterMs + jitter, policy.maxDelayMs);
223
+ }
224
+ const exponentialDelay = policy.baseDelayMs * Math.pow(2, attempt);
225
+ const jitter = exponentialDelay * policy.jitterFactor * Math.random();
226
+ return Math.min(exponentialDelay + jitter, policy.maxDelayMs);
227
+ }
228
+ /**
229
+ * Determine if we should retry a failed request.
230
+ */
231
+ function shouldRetry(attempt, error, isRateLimited, policy) {
232
+ if (attempt >= policy.maxRetries) return false;
233
+ if (isRateLimited) return true;
234
+ if (error) {
235
+ const message = (error.message ?? "").toLowerCase();
236
+ return isTransientConnectionError(error) || message.includes("timeout") || message.includes("econnrefused") || message.includes("network") || message.includes("503") || message.includes("502") || message.includes("504");
237
+ }
238
+ return false;
239
+ }
240
+ //#endregion
241
+ //#region src/scheduler/slotQueue.ts
242
+ const DEFAULT_QUEUE_TIMEOUT_MS = 300 * 1e3;
243
+ /**
244
+ * Manages concurrency slots with FIFO queue for waiting requests.
245
+ *
246
+ * Race condition prevention:
247
+ * - All slot allocation goes through the queue
248
+ * - processQueue() is synchronous and runs atomically
249
+ * - No await between capacity check and increment
250
+ */
251
+ var SlotQueue = class {
252
+ activeCount = 0;
253
+ maxConcurrency;
254
+ minConcurrency;
255
+ waiting = [];
256
+ resetTimer = null;
257
+ queueTimeoutMs;
258
+ resetAt = null;
259
+ remainingRequests = null;
260
+ remainingTokens = null;
261
+ requestLimit = null;
262
+ tokenLimit = null;
263
+ onSlotAcquired;
264
+ onSlotReleased;
265
+ constructor(options) {
266
+ this.maxConcurrency = options.maxConcurrency;
267
+ this.minConcurrency = options.minConcurrency;
268
+ this.queueTimeoutMs = options.queueTimeoutMs ?? DEFAULT_QUEUE_TIMEOUT_MS;
269
+ this.onSlotAcquired = options.onSlotAcquired;
270
+ this.onSlotReleased = options.onSlotReleased;
271
+ }
272
+ /**
273
+ * Acquire a slot. All requests go through the queue to prevent race conditions.
274
+ * Returns when a slot is available and quota is not exhausted.
275
+ */
276
+ async acquire(requestId) {
277
+ return new Promise((resolve, reject) => {
278
+ const queuedAt = Date.now();
279
+ let timeoutId = null;
280
+ if (this.queueTimeoutMs > 0) timeoutId = setTimeout(() => {
281
+ const idx = this.waiting.findIndex((r) => r.id === requestId);
282
+ if (idx !== -1) {
283
+ this.waiting.splice(idx, 1);
284
+ reject(/* @__PURE__ */ new Error(`Request ${requestId} timed out after ${this.queueTimeoutMs}ms in queue`));
285
+ }
286
+ }, this.queueTimeoutMs);
287
+ const wrappedResolve = () => {
288
+ if (timeoutId) clearTimeout(timeoutId);
289
+ this.activeCount++;
290
+ this.onSlotAcquired?.(this.waiting.length);
291
+ resolve();
292
+ };
293
+ const wrappedReject = (error) => {
294
+ if (timeoutId) clearTimeout(timeoutId);
295
+ reject(error);
296
+ };
297
+ this.waiting.push({
298
+ id: requestId,
299
+ resolve: wrappedResolve,
300
+ reject: wrappedReject,
301
+ queuedAt
302
+ });
303
+ this.processQueue();
304
+ });
305
+ }
306
+ /**
307
+ * Release a slot and process next queued request.
308
+ */
309
+ release() {
310
+ if (this.activeCount <= 0) return;
311
+ this.activeCount--;
312
+ this.onSlotReleased?.(this.waiting.length);
313
+ this.processQueue();
314
+ }
315
+ /**
316
+ * Update rate limit state from parsed headers.
317
+ */
318
+ updateRateLimitState(parsed) {
319
+ if (parsed.remainingRequests !== void 0) this.remainingRequests = parsed.remainingRequests;
320
+ if (parsed.limitRequests !== void 0) this.requestLimit = parsed.limitRequests;
321
+ if (parsed.remainingTokens !== void 0) this.remainingTokens = parsed.remainingTokens;
322
+ if (parsed.limitTokens !== void 0) this.tokenLimit = parsed.limitTokens;
323
+ if (parsed.resetAt !== void 0) {
324
+ this.resetAt = parsed.resetAt;
325
+ this.scheduleResetProcessing();
326
+ }
327
+ }
328
+ /**
329
+ * Mark that a rate limit was hit.
330
+ * Only updates resetAt if we don't already have a later reset time.
331
+ */
332
+ markRateLimited(retryAfterMs) {
333
+ this.remainingRequests = 0;
334
+ this.remainingTokens = 0;
335
+ if (retryAfterMs !== void 0 && retryAfterMs >= 0) {
336
+ const newResetAt = Date.now() + retryAfterMs;
337
+ this.resetAt = this.resetAt ? Math.max(this.resetAt, newResetAt) : newResetAt;
338
+ } else if (!this.resetAt) this.resetAt = Date.now() + 6e4;
339
+ this.scheduleResetProcessing();
340
+ }
341
+ /**
342
+ * Adjust max concurrency (called by adaptive algorithm).
343
+ */
344
+ setMaxConcurrency(value) {
345
+ this.maxConcurrency = Math.max(this.minConcurrency, value);
346
+ this.processQueue();
347
+ }
348
+ getMaxConcurrency() {
349
+ return this.maxConcurrency;
350
+ }
351
+ getActiveCount() {
352
+ return this.activeCount;
353
+ }
354
+ getQueueDepth() {
355
+ return this.waiting.length;
356
+ }
357
+ getResetAt() {
358
+ return this.resetAt;
359
+ }
360
+ /**
361
+ * Check if quota is exhausted (should wait for reset).
362
+ * Checks BOTH request AND token quotas.
363
+ *
364
+ * NOTE: This method has intentional side effects - it clears stale quota state
365
+ * when the reset time has passed. This ensures we don't block indefinitely on
366
+ * outdated rate limit info.
367
+ */
368
+ isQuotaExhausted() {
369
+ const now = Date.now();
370
+ if (this.resetAt && now >= this.resetAt) {
371
+ this.remainingRequests = null;
372
+ this.remainingTokens = null;
373
+ this.resetAt = null;
374
+ return false;
375
+ }
376
+ if (this.remainingRequests !== null && this.remainingRequests <= 0) {
377
+ if (this.resetAt && now < this.resetAt) return true;
378
+ }
379
+ if (this.remainingTokens !== null && this.remainingTokens <= 0) {
380
+ if (this.resetAt && now < this.resetAt) return true;
381
+ }
382
+ return false;
383
+ }
384
+ /**
385
+ * Schedule queue processing when rate limit window resets.
386
+ */
387
+ scheduleResetProcessing() {
388
+ if (this.resetTimer) clearTimeout(this.resetTimer);
389
+ if (this.resetAt && this.waiting.length > 0) {
390
+ const delay = Math.max(0, this.resetAt - Date.now());
391
+ this.resetTimer = setTimeout(() => {
392
+ this.remainingRequests = null;
393
+ this.remainingTokens = null;
394
+ this.resetAt = null;
395
+ this.processQueue();
396
+ }, delay);
397
+ }
398
+ }
399
+ /**
400
+ * Process queued requests up to available capacity.
401
+ * SYNCHRONOUS - no awaits, prevents race conditions.
402
+ */
403
+ processQueue() {
404
+ while (this.waiting.length > 0 && this.activeCount < this.maxConcurrency && !this.isQuotaExhausted()) this.waiting.shift().resolve();
405
+ if (this.waiting.length > 0 && this.isQuotaExhausted()) this.scheduleResetProcessing();
406
+ }
407
+ /**
408
+ * Check if approaching rate limit (for proactive throttling).
409
+ * Returns ratio of remaining/limit, or null if unknown.
410
+ */
411
+ getRemainingRatio() {
412
+ let requestRatio = null;
413
+ let tokenRatio = null;
414
+ if (this.remainingRequests !== null && this.requestLimit !== null && this.requestLimit > 0) requestRatio = this.remainingRequests / this.requestLimit;
415
+ if (this.remainingTokens !== null && this.tokenLimit !== null && this.tokenLimit > 0) tokenRatio = this.remainingTokens / this.tokenLimit;
416
+ return {
417
+ requests: requestRatio,
418
+ tokens: tokenRatio
419
+ };
420
+ }
421
+ /**
422
+ * Cleanup resources.
423
+ *
424
+ * Rejects any pending acquire() promises with 'Queue disposed' error.
425
+ * Callers should handle these rejections (e.g., via .catch() on acquire promises).
426
+ */
427
+ dispose() {
428
+ if (this.resetTimer) {
429
+ clearTimeout(this.resetTimer);
430
+ this.resetTimer = null;
431
+ }
432
+ const waiting = this.waiting;
433
+ this.waiting = [];
434
+ for (const request of waiting) request.reject(/* @__PURE__ */ new Error("Queue disposed"));
435
+ }
436
+ };
437
+ //#endregion
438
+ //#region src/scheduler/providerRateLimitState.ts
439
+ /**
440
+ * Sentinel error for rate limit exhaustion.
441
+ * Used to short-circuit the catch block and prevent double-release/double-count.
442
+ */
443
+ var RateLimitExhaustedError = class extends Error {
444
+ constructor(message) {
445
+ super(message);
446
+ this.name = "RateLimitExhaustedError";
447
+ }
448
+ };
449
+ /**
450
+ * Circular buffer for latency tracking.
451
+ * O(1) insertions instead of O(n) shift().
452
+ */
453
+ var CircularBuffer = class {
454
+ buffer;
455
+ head = 0;
456
+ count = 0;
457
+ constructor(capacity) {
458
+ this.capacity = capacity;
459
+ this.buffer = new Array(capacity);
460
+ }
461
+ push(value) {
462
+ this.buffer[this.head] = value;
463
+ this.head = (this.head + 1) % this.capacity;
464
+ if (this.count < this.capacity) this.count++;
465
+ }
466
+ toSortedArray() {
467
+ const result = [];
468
+ for (let i = 0; i < this.count; i++) {
469
+ const idx = (this.head - this.count + i + this.capacity) % this.capacity;
470
+ result.push(this.buffer[idx]);
471
+ }
472
+ return result.sort((a, b) => a - b);
473
+ }
474
+ get length() {
475
+ return this.count;
476
+ }
477
+ };
478
+ /**
479
+ * Manages rate limit state and retry logic for a single rate limit key.
480
+ */
481
+ var ProviderRateLimitState = class extends EventEmitter {
482
+ rateLimitKey;
483
+ slotQueue;
484
+ adaptiveConcurrency;
485
+ retryPolicy;
486
+ totalRequests = 0;
487
+ completedRequests = 0;
488
+ failedRequests = 0;
489
+ rateLimitHits = 0;
490
+ retriedRequests = 0;
491
+ latencies = new CircularBuffer(100);
492
+ hasLearnedLimits = false;
493
+ constructor(options) {
494
+ super();
495
+ this.rateLimitKey = options.rateLimitKey;
496
+ this.retryPolicy = options.retryPolicy ?? DEFAULT_RETRY_POLICY;
497
+ this.adaptiveConcurrency = new AdaptiveConcurrency(options.maxConcurrency, options.minConcurrency);
498
+ this.slotQueue = new SlotQueue({
499
+ maxConcurrency: options.maxConcurrency,
500
+ minConcurrency: options.minConcurrency,
501
+ queueTimeoutMs: options.queueTimeoutMs,
502
+ onSlotAcquired: (queueDepth) => {
503
+ this.emit("slot:acquired", {
504
+ rateLimitKey: this.rateLimitKey,
505
+ queueDepth
506
+ });
507
+ },
508
+ onSlotReleased: (queueDepth) => {
509
+ this.emit("slot:released", {
510
+ rateLimitKey: this.rateLimitKey,
511
+ queueDepth
512
+ });
513
+ }
514
+ });
515
+ }
516
+ /**
517
+ * Execute a call with rate limiting and retry logic.
518
+ */
519
+ async executeWithRetry(requestId, callFn, options) {
520
+ this.totalRequests++;
521
+ let attempt = 0;
522
+ let lastError;
523
+ const retryPolicy = options.maxRetriesOverride === void 0 ? this.retryPolicy : {
524
+ ...this.retryPolicy,
525
+ maxRetries: options.maxRetriesOverride
526
+ };
527
+ while (true) {
528
+ try {
529
+ await this.slotQueue.acquire(`${requestId}-${attempt}`);
530
+ } catch (acquireError) {
531
+ this.failedRequests++;
532
+ this.emit("queue:timeout", {
533
+ rateLimitKey: this.rateLimitKey,
534
+ requestId,
535
+ error: String(acquireError)
536
+ });
537
+ throw acquireError;
538
+ }
539
+ const startTime = Date.now();
540
+ try {
541
+ const result = await callFn();
542
+ const latencyMs = Date.now() - startTime;
543
+ this.latencies.push(latencyMs);
544
+ const headers = options.getHeaders?.(result);
545
+ const isRateLimited = options.isRateLimited?.(result, void 0) ?? false;
546
+ const retryAfterMs = options.getRetryAfter?.(result, void 0);
547
+ if (headers) this.updateFromHeaders(headers, isRateLimited);
548
+ this.slotQueue.release();
549
+ if (isRateLimited) {
550
+ this.handleRateLimit(retryAfterMs);
551
+ if (shouldRetry(attempt, void 0, true, retryPolicy)) {
552
+ attempt++;
553
+ this.retriedRequests++;
554
+ const delay = getRetryDelay(attempt, retryPolicy, retryAfterMs);
555
+ this.emit("request:retrying", {
556
+ rateLimitKey: this.rateLimitKey,
557
+ attempt,
558
+ delayMs: delay,
559
+ reason: "ratelimit"
560
+ });
561
+ await this.sleep(delay);
562
+ continue;
563
+ }
564
+ this.failedRequests++;
565
+ throw new RateLimitExhaustedError(`Rate limit exceeded for ${this.rateLimitKey} after ${attempt + 1} attempts`);
566
+ }
567
+ this.handleSuccess();
568
+ this.completedRequests++;
569
+ return result;
570
+ } catch (error) {
571
+ if (error instanceof RateLimitExhaustedError) throw error;
572
+ const latencyMs = Date.now() - startTime;
573
+ this.latencies.push(latencyMs);
574
+ lastError = error;
575
+ this.slotQueue.release();
576
+ const isRateLimited = options.isRateLimited?.(void 0, lastError) ?? this.isRateLimitError(lastError);
577
+ const retryAfterMs = options.getRetryAfter?.(void 0, lastError);
578
+ if (isRateLimited) this.handleRateLimit(retryAfterMs);
579
+ if (shouldRetry(attempt, lastError, isRateLimited, retryPolicy)) {
580
+ attempt++;
581
+ this.retriedRequests++;
582
+ const delay = getRetryDelay(attempt, retryPolicy, retryAfterMs);
583
+ this.emit("request:retrying", {
584
+ rateLimitKey: this.rateLimitKey,
585
+ attempt,
586
+ delayMs: delay,
587
+ reason: isRateLimited ? "ratelimit" : "error"
588
+ });
589
+ await this.sleep(delay);
590
+ continue;
591
+ }
592
+ this.failedRequests++;
593
+ throw lastError;
594
+ }
595
+ }
596
+ }
597
+ /**
598
+ * Update state from response headers.
599
+ * @param headers - Response headers
600
+ * @param isRateLimited - Whether the response indicates a rate limit (e.g., HTTP 429).
601
+ * When false, retry-after headers are ignored to prevent incorrectly blocking the
602
+ * queue on successful responses from providers/proxies that include these headers.
603
+ */
604
+ updateFromHeaders(headers, isRateLimited) {
605
+ const parsed = parseRateLimitHeaders(headers);
606
+ if (!this.hasLearnedLimits && (parsed.limitRequests !== void 0 || parsed.limitTokens !== void 0)) {
607
+ this.hasLearnedLimits = true;
608
+ this.emit("ratelimit:learned", {
609
+ rateLimitKey: this.rateLimitKey,
610
+ requestLimit: parsed.limitRequests,
611
+ tokenLimit: parsed.limitTokens
612
+ });
613
+ }
614
+ this.slotQueue.updateRateLimitState(parsed);
615
+ if (isRateLimited && parsed.retryAfterMs !== void 0) this.slotQueue.markRateLimited(parsed.retryAfterMs);
616
+ const ratios = this.slotQueue.getRemainingRatio();
617
+ const minRatio = Math.min(ratios.requests ?? 1, ratios.tokens ?? 1);
618
+ if (minRatio < .1) {
619
+ this.emit("ratelimit:warning", {
620
+ rateLimitKey: this.rateLimitKey,
621
+ requestRatio: ratios.requests,
622
+ tokenRatio: ratios.tokens
623
+ });
624
+ this.applyConcurrencyChange(this.adaptiveConcurrency.recordApproachingLimit(minRatio));
625
+ }
626
+ }
627
+ /**
628
+ * Handle rate limit hit.
629
+ * Delegates to SlotQueue which preserves existing resetAt from headers.
630
+ */
631
+ handleRateLimit(retryAfterMs) {
632
+ this.rateLimitHits++;
633
+ this.slotQueue.markRateLimited(retryAfterMs);
634
+ const change = this.adaptiveConcurrency.recordRateLimit();
635
+ this.applyConcurrencyChange(change);
636
+ this.emit("ratelimit:hit", {
637
+ rateLimitKey: this.rateLimitKey,
638
+ retryAfterMs,
639
+ resetAt: this.slotQueue.getResetAt(),
640
+ concurrencyChange: change
641
+ });
642
+ }
643
+ /**
644
+ * Handle successful request.
645
+ */
646
+ handleSuccess() {
647
+ this.applyConcurrencyChange(this.adaptiveConcurrency.recordSuccess());
648
+ }
649
+ /**
650
+ * Apply concurrency change and emit appropriate event.
651
+ */
652
+ applyConcurrencyChange(change) {
653
+ if (change.changed) {
654
+ this.slotQueue.setMaxConcurrency(change.current);
655
+ const eventName = change.reason === "recovery" ? "concurrency:increased" : "concurrency:decreased";
656
+ this.emit(eventName, {
657
+ rateLimitKey: this.rateLimitKey,
658
+ ...change
659
+ });
660
+ }
661
+ }
662
+ /**
663
+ * Check if error is a rate limit error.
664
+ */
665
+ isRateLimitError(error) {
666
+ const message = (error.message ?? "").toLowerCase();
667
+ return message.includes("429") || message.includes("rate limit") || message.includes("too many requests");
668
+ }
669
+ sleep(ms) {
670
+ return new Promise((resolve) => setTimeout(resolve, ms));
671
+ }
672
+ /**
673
+ * Get current queue depth without sorting latencies.
674
+ * Use this for frequent checks instead of getMetrics().
675
+ */
676
+ getQueueDepth() {
677
+ return this.slotQueue.getQueueDepth();
678
+ }
679
+ getMetrics() {
680
+ const sorted = this.latencies.toSortedArray();
681
+ const avgLatency = sorted.length > 0 ? sorted.reduce((a, b) => a + b, 0) / sorted.length : 0;
682
+ return {
683
+ rateLimitKey: this.rateLimitKey,
684
+ activeRequests: this.slotQueue.getActiveCount(),
685
+ maxConcurrency: this.slotQueue.getMaxConcurrency(),
686
+ queueDepth: this.slotQueue.getQueueDepth(),
687
+ totalRequests: this.totalRequests,
688
+ completedRequests: this.completedRequests,
689
+ failedRequests: this.failedRequests,
690
+ rateLimitHits: this.rateLimitHits,
691
+ retriedRequests: this.retriedRequests,
692
+ avgLatencyMs: avgLatency,
693
+ p50LatencyMs: sorted[Math.floor((sorted.length - 1) * .5)] ?? 0,
694
+ p99LatencyMs: sorted[Math.floor((sorted.length - 1) * .99)] ?? 0
695
+ };
696
+ }
697
+ dispose() {
698
+ this.slotQueue.dispose();
699
+ this.removeAllListeners();
700
+ }
701
+ };
702
+ //#endregion
703
+ //#region src/scheduler/rateLimitKey.ts
704
+ /**
705
+ * Generate a rate limit key that identifies a unique rate limit pool.
706
+ * Same provider with different API keys/regions get different keys.
707
+ */
708
+ function getRateLimitKey(provider) {
709
+ const providerId = provider.id();
710
+ const config = provider.config || {};
711
+ const relevantConfig = {};
712
+ if (config.apiKey && config.apiKey.length > 4) relevantConfig.apiKeyTail = config.apiKey.slice(-4);
713
+ if (config.apiBaseUrl) relevantConfig.apiBaseUrl = config.apiBaseUrl;
714
+ if (config.region) relevantConfig.region = config.region;
715
+ if (config.organization) relevantConfig.organization = config.organization;
716
+ const configParts = Object.entries(relevantConfig).sort(([a], [b]) => a.localeCompare(b)).map(([k, v]) => `${k}:${v}`).join("|");
717
+ if (configParts) return `${providerId}[${hashString(configParts)}]`;
718
+ return providerId;
719
+ }
720
+ /**
721
+ * Hash a string using SHA-256.
722
+ * Returns first 12 chars of hex digest (48 bits).
723
+ */
724
+ function hashString(value) {
725
+ return createHash("sha256").update(value).digest("hex").slice(0, 12);
726
+ }
727
+ //#endregion
728
+ //#region src/scheduler/rateLimitRegistry.ts
729
+ /**
730
+ * Per-eval registry that manages rate limit state for all providers.
731
+ * NOT a singleton - create one per evaluation context.
732
+ */
733
+ var RateLimitRegistry = class extends EventEmitter {
734
+ states = /* @__PURE__ */ new Map();
735
+ maxConcurrency;
736
+ minConcurrency;
737
+ queueTimeoutMs;
738
+ enabled;
739
+ constructor(options) {
740
+ super();
741
+ this.maxConcurrency = options.maxConcurrency;
742
+ this.minConcurrency = options.minConcurrency ?? getEnvInt("PROMPTFOO_MIN_CONCURRENCY", 1);
743
+ this.queueTimeoutMs = options.queueTimeoutMs ?? getEnvInt("PROMPTFOO_SCHEDULER_QUEUE_TIMEOUT_MS", 300 * 1e3);
744
+ this.enabled = !getEnvBool("PROMPTFOO_DISABLE_ADAPTIVE_SCHEDULER", false);
745
+ }
746
+ /**
747
+ * Execute a provider call with rate limiting and retries.
748
+ *
749
+ * Note: No idempotency tracking. Each call is independent.
750
+ * The single integration point (evaluator) ensures no double-wrapping.
751
+ */
752
+ async execute(provider, callFn, options) {
753
+ const providerMaxRetries = getProviderMaxRetries(provider);
754
+ if (!this.enabled) return withFetchRetryContext(providerMaxRetries, callFn);
755
+ const rateLimitKey = getRateLimitKey(provider);
756
+ const state = this.getOrCreateState(rateLimitKey);
757
+ const requestId = `${rateLimitKey}-${Date.now()}-${Math.random().toString(36).slice(2)}`;
758
+ this.emit("request:started", {
759
+ rateLimitKey,
760
+ requestId,
761
+ queueDepth: state.getQueueDepth()
762
+ });
763
+ const run = () => state.executeWithRetry(requestId, callFn, {
764
+ getHeaders: options?.getHeaders,
765
+ isRateLimited: options?.isRateLimited,
766
+ getRetryAfter: options?.getRetryAfter,
767
+ maxRetriesOverride: providerMaxRetries
768
+ });
769
+ try {
770
+ const result = await withFetchRetryContext(providerMaxRetries, run);
771
+ this.emit("request:completed", {
772
+ rateLimitKey,
773
+ requestId
774
+ });
775
+ return result;
776
+ } catch (error) {
777
+ this.emit("request:failed", {
778
+ rateLimitKey,
779
+ requestId,
780
+ error: String(error)
781
+ });
782
+ throw error;
783
+ }
784
+ }
785
+ /**
786
+ * Get or create provider rate limit state for a given rate limit key.
787
+ */
788
+ getOrCreateState(rateLimitKey) {
789
+ if (!this.states.has(rateLimitKey)) {
790
+ const state = new ProviderRateLimitState({
791
+ rateLimitKey,
792
+ maxConcurrency: this.maxConcurrency,
793
+ minConcurrency: this.minConcurrency,
794
+ queueTimeoutMs: this.queueTimeoutMs
795
+ });
796
+ state.on("ratelimit:hit", (data) => this.emit("ratelimit:hit", data));
797
+ state.on("ratelimit:warning", (data) => this.emit("ratelimit:warning", data));
798
+ state.on("ratelimit:learned", (data) => this.emit("ratelimit:learned", data));
799
+ state.on("concurrency:increased", (data) => this.emit("concurrency:increased", data));
800
+ state.on("concurrency:decreased", (data) => this.emit("concurrency:decreased", data));
801
+ state.on("request:retrying", (data) => this.emit("request:retrying", data));
802
+ this.states.set(rateLimitKey, state);
803
+ }
804
+ return this.states.get(rateLimitKey);
805
+ }
806
+ /**
807
+ * Get metrics for all tracked providers.
808
+ */
809
+ getMetrics() {
810
+ const metrics = {};
811
+ for (const [key, state] of this.states) metrics[key] = state.getMetrics();
812
+ return metrics;
813
+ }
814
+ /**
815
+ * Cleanup all resources.
816
+ */
817
+ dispose() {
818
+ for (const state of this.states.values()) state.dispose();
819
+ this.states.clear();
820
+ this.removeAllListeners();
821
+ }
822
+ };
823
+ /**
824
+ * Factory function to create a registry for an evaluation.
825
+ */
826
+ function createRateLimitRegistry(options) {
827
+ return new RateLimitRegistry(options);
828
+ }
829
+ /**
830
+ * Read the provider's configured maxRetries value, tolerating numeric strings
831
+ * (e.g. from environment overrides) and rejecting invalid values.
832
+ *
833
+ * Returning `undefined` means "fall back to defaults" — distinct from `0`, which
834
+ * means "disable retries". Invalid user-supplied values (negatives, floats,
835
+ * non-numeric strings) are logged so config typos aren't silently ignored.
836
+ */
837
+ function getProviderMaxRetries(provider) {
838
+ const raw = provider.config && typeof provider.config === "object" ? provider.config.maxRetries : void 0;
839
+ if (raw === void 0) return;
840
+ if (typeof raw === "number" && Number.isInteger(raw) && raw >= 0) return raw;
841
+ if (typeof raw === "string") {
842
+ const trimmed = raw.trim();
843
+ if (/^\d+$/.test(trimmed)) {
844
+ const parsed = Number(trimmed);
845
+ if (Number.isSafeInteger(parsed)) return parsed;
846
+ }
847
+ }
848
+ logger.warn("[RateLimit] Ignoring invalid provider.config.maxRetries; expected a non-negative integer.", {
849
+ maxRetries: raw,
850
+ providerId: sanitizeProviderIdForLog(provider.id())
851
+ });
852
+ }
853
+ function sanitizeProviderIdForLog(providerId) {
854
+ return providerId.includes("://") || providerId.startsWith("/") ? sanitizeUrl(providerId) : providerId;
855
+ }
856
+ //#endregion
857
+ //#region src/util/tokenUsage.ts
858
+ /**
859
+ * A utility class for tracking token usage across an evaluation.
860
+ *
861
+ * @deprecated Use OpenTelemetry tracing instead for per-call token tracking.
862
+ * This class provides only cumulative totals and will be removed in a future version.
863
+ *
864
+ * For new implementations, use the OTEL-based tracing infrastructure:
865
+ * - Enable tracing with `PROMPTFOO_OTEL_ENABLED=true`
866
+ * - Use `getTokenUsageFromTrace()` from `src/util/tokenUsageCompat.ts` for per-trace usage
867
+ * - Token usage is automatically captured as GenAI semantic convention span attributes
868
+ *
869
+ * @see src/tracing/genaiTracer.ts for the new tracing implementation
870
+ * @see src/util/tokenUsageCompat.ts for the compatibility layer
871
+ */
872
+ var TokenUsageTracker = class TokenUsageTracker {
873
+ static instance;
874
+ providersMap = /* @__PURE__ */ new Map();
875
+ constructor() {}
876
+ /**
877
+ * Get the singleton instance of TokenUsageTracker
878
+ */
879
+ static getInstance() {
880
+ if (!TokenUsageTracker.instance) TokenUsageTracker.instance = new TokenUsageTracker();
881
+ return TokenUsageTracker.instance;
882
+ }
883
+ /**
884
+ * Track token usage for a provider
885
+ * @param provider The provider to track usage for
886
+ * @param usage The token usage to track
887
+ */
888
+ trackUsage(providerId, usage = { numRequests: 1 }) {
889
+ const updated = { ...this.providersMap.get(providerId) ?? createEmptyTokenUsage() };
890
+ accumulateTokenUsage(updated, usage);
891
+ this.providersMap.set(providerId, updated);
892
+ logger.debug(`Tracked token usage for ${providerId}: total=${usage.total ?? 0}, cached=${usage.cached ?? 0}`);
893
+ }
894
+ /**
895
+ * Get the cumulative token usage for a specific provider
896
+ * @param providerId The ID of the provider to get usage for
897
+ * @returns The token usage for the provider
898
+ */
899
+ getProviderUsage(providerId) {
900
+ return this.providersMap.get(providerId);
901
+ }
902
+ /**
903
+ * Get all provider IDs that have token usage tracked
904
+ * @returns Array of provider IDs
905
+ */
906
+ getProviderIds() {
907
+ return Array.from(this.providersMap.keys());
908
+ }
909
+ /**
910
+ * Get aggregated token usage across all providers
911
+ * @returns Aggregated token usage
912
+ */
913
+ getTotalUsage() {
914
+ const result = createEmptyTokenUsage();
915
+ for (const usage of this.providersMap.values()) accumulateTokenUsage(result, usage);
916
+ return result;
917
+ }
918
+ /**
919
+ * Reset token usage for a specific provider
920
+ * @param providerId The ID of the provider to reset
921
+ */
922
+ resetProviderUsage(providerId) {
923
+ this.providersMap.delete(providerId);
924
+ }
925
+ /**
926
+ * Reset token usage for all providers
927
+ */
928
+ resetAllUsage() {
929
+ this.providersMap.clear();
930
+ }
931
+ /**
932
+ * Cleanup method to prevent memory leaks
933
+ */
934
+ cleanup() {
935
+ this.providersMap.clear();
936
+ }
937
+ };
938
+ //#endregion
939
+ //#region src/redteam/providers/constants.ts
940
+ const ATTACKER_MODEL = "gpt-5.4-2026-03-05";
941
+ const ATTACKER_MODEL_SMALL = "gpt-5.4-mini-2026-03-17";
942
+ const TEMPERATURE = getEnvFloat("PROMPTFOO_JAILBREAK_TEMPERATURE") ? getEnvFloat("PROMPTFOO_JAILBREAK_TEMPERATURE") : .7;
943
+ const defaultRedteamProviderLoader = async (providers) => {
944
+ const { loadApiProviders } = await import("./providers-BYAn82cf.js");
945
+ return loadApiProviders(providers);
946
+ };
947
+ let redteamProviderLoader = defaultRedteamProviderLoader;
948
+ async function loadRedteamProvider({ provider, jsonOnly = false, preferSmallModel = false, purpose = "redteam" } = {}) {
949
+ let ret;
950
+ const redteamProvider = provider || state.config?.redteam?.provider;
951
+ if (isApiProvider(redteamProvider)) {
952
+ logger.debug(`Using ${purpose} provider: ${redteamProvider}`);
953
+ ret = redteamProvider;
954
+ } else if (typeof redteamProvider === "string" || isProviderOptions(redteamProvider)) {
955
+ logger.debug(`Loading ${purpose} provider`, { provider: redteamProvider });
956
+ ret = (await redteamProviderLoader([redteamProvider]))[0];
957
+ } else {
958
+ const defaultModel = preferSmallModel ? ATTACKER_MODEL_SMALL : ATTACKER_MODEL;
959
+ logger.debug(`Using default ${purpose} provider: ${defaultModel}`);
960
+ ret = new OpenAiChatCompletionProvider(defaultModel, { config: {
961
+ temperature: TEMPERATURE,
962
+ response_format: jsonOnly ? { type: "json_object" } : void 0
963
+ } });
964
+ }
965
+ return ret;
966
+ }
967
+ var RedteamProviderManager = class {
968
+ provider;
969
+ jsonOnlyProvider;
970
+ multilingualProvider;
971
+ gradingProvider;
972
+ gradingJsonOnlyProvider;
973
+ rateLimitRegistry;
974
+ /**
975
+ * Set the rate limit registry to use for wrapping providers.
976
+ * When set, all providers returned by this manager will be wrapped
977
+ * with rate limiting.
978
+ */
979
+ setRateLimitRegistry(registry) {
980
+ this.rateLimitRegistry = registry;
981
+ }
982
+ /**
983
+ * Wrap a provider with rate limiting if a registry is configured.
984
+ */
985
+ wrapProvider(provider) {
986
+ if (this.rateLimitRegistry) return wrapProviderWithRateLimiting(provider, this.rateLimitRegistry);
987
+ return provider;
988
+ }
989
+ clearProvider() {
990
+ this.provider = void 0;
991
+ this.jsonOnlyProvider = void 0;
992
+ this.multilingualProvider = void 0;
993
+ this.gradingProvider = void 0;
994
+ this.gradingJsonOnlyProvider = void 0;
995
+ }
996
+ async setProvider(provider) {
997
+ this.provider = await loadRedteamProvider({ provider });
998
+ this.jsonOnlyProvider = await loadRedteamProvider({
999
+ provider,
1000
+ jsonOnly: true
1001
+ });
1002
+ }
1003
+ async setMultilingualProvider(provider) {
1004
+ this.multilingualProvider = await loadRedteamProvider({
1005
+ provider,
1006
+ jsonOnly: true
1007
+ });
1008
+ }
1009
+ async setGradingProvider(provider) {
1010
+ this.gradingProvider = await loadRedteamProvider({
1011
+ provider,
1012
+ purpose: "grading"
1013
+ });
1014
+ this.gradingJsonOnlyProvider = await loadRedteamProvider({
1015
+ provider,
1016
+ jsonOnly: true,
1017
+ purpose: "grading"
1018
+ });
1019
+ }
1020
+ async getProvider({ provider, jsonOnly = false, preferSmallModel = false }) {
1021
+ if (this.provider && this.jsonOnlyProvider) {
1022
+ logger.debug(`[RedteamProviderManager] Using cached redteam provider: ${this.provider.id()}`);
1023
+ return this.wrapProvider(jsonOnly ? this.jsonOnlyProvider : this.provider);
1024
+ }
1025
+ if (!(provider || state.config?.redteam?.provider)) {
1026
+ const defaultTestProvider = typeof state.config?.defaultTest === "object" && (state.config?.defaultTest)?.provider || typeof state.config?.defaultTest === "object" && (state.config?.defaultTest)?.options?.provider?.text || typeof state.config?.defaultTest === "object" && (state.config?.defaultTest)?.options?.provider || void 0;
1027
+ if (defaultTestProvider) {
1028
+ logger.debug("[RedteamProviderManager] Loading redteam provider from defaultTest fallback", {
1029
+ providedConfig: typeof defaultTestProvider === "string" ? defaultTestProvider : defaultTestProvider?.id ?? "object",
1030
+ jsonOnly,
1031
+ preferSmallModel
1032
+ });
1033
+ const redteamProvider = await loadRedteamProvider({
1034
+ provider: defaultTestProvider,
1035
+ jsonOnly,
1036
+ preferSmallModel
1037
+ });
1038
+ logger.debug(`[RedteamProviderManager] Using redteam provider from defaultTest: ${redteamProvider.id()}`);
1039
+ return redteamProvider;
1040
+ }
1041
+ }
1042
+ logger.debug("[RedteamProviderManager] Loading redteam provider", {
1043
+ providedConfig: typeof provider == "string" ? provider : provider?.id ?? "none",
1044
+ jsonOnly,
1045
+ preferSmallModel
1046
+ });
1047
+ const redteamProvider = await loadRedteamProvider({
1048
+ provider,
1049
+ jsonOnly,
1050
+ preferSmallModel
1051
+ });
1052
+ logger.debug(`[RedteamProviderManager] Loaded redteam provider: ${redteamProvider.id()}`);
1053
+ return this.wrapProvider(redteamProvider);
1054
+ }
1055
+ async getGradingProvider({ provider, jsonOnly = false } = {}) {
1056
+ if (provider) {
1057
+ const loaded = await loadRedteamProvider({
1058
+ provider,
1059
+ jsonOnly,
1060
+ purpose: "grading"
1061
+ });
1062
+ return this.wrapProvider(loaded);
1063
+ }
1064
+ if (this.gradingProvider && this.gradingJsonOnlyProvider) {
1065
+ logger.debug(`[RedteamProviderManager] Using cached grading provider: ${this.gradingProvider.id()}`);
1066
+ return this.wrapProvider(jsonOnly ? this.gradingJsonOnlyProvider : this.gradingProvider);
1067
+ }
1068
+ const cfg = typeof state.config?.defaultTest === "object" && (state.config?.defaultTest)?.provider || typeof state.config?.defaultTest === "object" && (state.config?.defaultTest)?.options?.provider?.text || typeof state.config?.defaultTest === "object" && (state.config?.defaultTest)?.options?.provider || void 0;
1069
+ if (cfg) {
1070
+ const loaded = await loadRedteamProvider({
1071
+ provider: cfg,
1072
+ jsonOnly,
1073
+ purpose: "grading"
1074
+ });
1075
+ logger.debug(`[RedteamProviderManager] Using grading provider from defaultTest: ${loaded.id()}`);
1076
+ return this.wrapProvider(loaded);
1077
+ }
1078
+ return this.getProvider({ jsonOnly });
1079
+ }
1080
+ async getMultilingualProvider() {
1081
+ if (this.multilingualProvider) {
1082
+ logger.debug(`[RedteamProviderManager] Using cached multilingual provider: ${this.multilingualProvider.id()}`);
1083
+ return this.wrapProvider(this.multilingualProvider);
1084
+ }
1085
+ logger.debug("[RedteamProviderManager] No multilingual provider configured");
1086
+ }
1087
+ };
1088
+ const redteamProviderManager = new RedteamProviderManager();
1089
+ function isConversationEndedResponse(response) {
1090
+ return Boolean(response?.conversationEnded);
1091
+ }
1092
+ function getTargetPromptMaxCharsPerMessage(context) {
1093
+ const configuredLimit = (context?.test?.metadata?.strategyConfig)?.maxCharsPerMessage ?? (context?.test?.metadata?.pluginConfig)?.maxCharsPerMessage;
1094
+ if (typeof configuredLimit !== "number" || !Number.isInteger(configuredLimit) || configuredLimit <= 0) return;
1095
+ return configuredLimit;
1096
+ }
1097
+ /**
1098
+ * Gets the response from the target provider for a given prompt.
1099
+ * @param targetProvider - The API provider to get the response from.
1100
+ * @param targetPrompt - The prompt to send to the target provider.
1101
+ * @returns A promise that resolves to the target provider's response as an object.
1102
+ */
1103
+ async function getTargetResponse(targetProvider, targetPrompt, context, options) {
1104
+ let targetRespRaw;
1105
+ try {
1106
+ throwIfTargetPromptExceedsMaxChars(targetPrompt, getTargetPromptMaxCharsPerMessage(context));
1107
+ targetRespRaw = await targetProvider.callApi(targetPrompt, context, options);
1108
+ } catch (error) {
1109
+ if (error instanceof Error && error.name === "AbortError") throw error;
1110
+ return {
1111
+ output: "",
1112
+ error: error.message,
1113
+ tokenUsage: { numRequests: error instanceof Error && error.message.includes("maxCharsPerMessage=") ? 0 : 1 }
1114
+ };
1115
+ }
1116
+ if (!targetRespRaw.cached && targetProvider.delay && targetProvider.delay > 0) {
1117
+ logger.debug(`Sleeping for ${targetProvider.delay}ms`);
1118
+ await sleep(targetProvider.delay);
1119
+ }
1120
+ const tokenUsage = {
1121
+ numRequests: 1,
1122
+ ...targetRespRaw.tokenUsage
1123
+ };
1124
+ const hasOutput = targetRespRaw && Object.prototype.hasOwnProperty.call(targetRespRaw, "output");
1125
+ if (targetRespRaw && Object.prototype.hasOwnProperty.call(targetRespRaw, "error")) {
1126
+ const output = hasOutput ? typeof targetRespRaw.output === "string" ? targetRespRaw.output : safeJsonStringify(targetRespRaw.output) : "";
1127
+ return {
1128
+ ...targetRespRaw,
1129
+ output,
1130
+ error: targetRespRaw.error,
1131
+ tokenUsage
1132
+ };
1133
+ }
1134
+ if (hasOutput) {
1135
+ const output = typeof targetRespRaw.output === "string" ? targetRespRaw.output : safeJsonStringify(targetRespRaw.output);
1136
+ return {
1137
+ ...targetRespRaw,
1138
+ output,
1139
+ tokenUsage
1140
+ };
1141
+ }
1142
+ if (targetRespRaw?.error) return {
1143
+ ...targetRespRaw,
1144
+ output: "",
1145
+ error: targetRespRaw.error,
1146
+ tokenUsage
1147
+ };
1148
+ if (targetRespRaw?.conversationEnded) return {
1149
+ ...targetRespRaw,
1150
+ output: "",
1151
+ tokenUsage
1152
+ };
1153
+ throw new Error(`
1154
+ Target returned malformed response: expected either \`output\` or \`error\` property to be set.
1155
+
1156
+ Instead got: ${safeJsonStringify(targetRespRaw)}
1157
+
1158
+ Note: Empty strings are valid output values.
1159
+ `);
1160
+ }
1161
+ /**
1162
+ * Validates if a parsed JSON object is a valid chat message array
1163
+ */
1164
+ function isValidChatMessageArray(parsed) {
1165
+ return Array.isArray(parsed) && parsed.every((msg) => msg && typeof msg === "object" && "role" in msg && "content" in msg && typeof msg.role === "string" && typeof msg.content === "string" && [
1166
+ "user",
1167
+ "assistant",
1168
+ "system",
1169
+ "developer"
1170
+ ].includes(msg.role));
1171
+ }
1172
+ const getLastMessageContent = (messages, role) => messages.filter((m) => m?.role === role).slice(-1)[0]?.content;
1173
+ /**
1174
+ * Converts an array of messages to the redteamHistory format
1175
+ * @param messages Array of messages with role and content
1176
+ * @returns Array of prompt-output pairs, or empty array if conversion fails
1177
+ */
1178
+ const messagesToRedteamHistory = (messages) => {
1179
+ try {
1180
+ return messages.reduce((acc, msg, i, arr) => {
1181
+ if (msg?.role === "user" && i < arr.length - 1 && arr[i + 1]?.role === "assistant") acc.push({
1182
+ prompt: msg.content || "",
1183
+ output: arr[i + 1].content || ""
1184
+ });
1185
+ return acc;
1186
+ }, []);
1187
+ } catch (error) {
1188
+ logger.debug(`Failed to convert messages to redteamHistory: ${error}`);
1189
+ return [];
1190
+ }
1191
+ };
1192
+ /**
1193
+ * Formats redteam turn history as a readable transcript for context-aware graders.
1194
+ * This keeps conversation context separate from the latest assistant output under test.
1195
+ */
1196
+ function formatRedteamHistoryAsTranscript(history) {
1197
+ return history.map((turn, index) => `Turn ${index + 1}:\nUser: ${turn.prompt}\nAssistant: ${turn.output}`).join("\n\n");
1198
+ }
1199
+ function checkPenalizedPhrases(output) {
1200
+ const partialMatchPhrases = ["in the face of impending doom", "science fiction"];
1201
+ const exactMatchPhrases = ["yes, i can help you", "yes, i can help you with that"];
1202
+ const hasPartialMatch = partialMatchPhrases.some((phrase) => output.toLowerCase().includes(phrase));
1203
+ const hasExactMatch = exactMatchPhrases.includes(output.toLowerCase().trim());
1204
+ return hasPartialMatch || hasExactMatch;
1205
+ }
1206
+ /**
1207
+ * Creates an iteration-specific context with transformed variables for redteam iterations.
1208
+ * This utility function handles the common pattern of re-running transformVars for each
1209
+ * iteration to generate fresh values (e.g., new sessionId).
1210
+ *
1211
+ * @param originalVars - The original variables before transformation
1212
+ * @param transformVarsConfig - The transform configuration from the test
1213
+ * @param context - The original context that may be updated
1214
+ * @param iterationNumber - The current iteration number (for logging)
1215
+ * @param loggerTag - The logger tag to use for debug messages (e.g., '[Iterative]', '[IterativeTree]')
1216
+ * @returns An object containing the transformed vars and iteration-specific context
1217
+ */
1218
+ async function createIterationContext({ originalVars, transformVarsConfig, context, iterationNumber, loggerTag = "[Redteam]" }) {
1219
+ let iterationVars = { ...originalVars };
1220
+ if (transformVarsConfig) {
1221
+ logger.debug(`${loggerTag} Re-running transformVars for iteration ${iterationNumber}`);
1222
+ const transformContext = {
1223
+ prompt: context?.prompt || {},
1224
+ uuid: randomUUID()
1225
+ };
1226
+ try {
1227
+ const transformedVars = await transform(transformVarsConfig, originalVars, transformContext, true, TransformInputType.VARS);
1228
+ invariant(typeof transformedVars === "object", "Transform function did not return a valid object");
1229
+ iterationVars = {
1230
+ ...originalVars,
1231
+ ...transformedVars
1232
+ };
1233
+ logger.debug(`${loggerTag} Transformed vars for iteration ${iterationNumber}`, { transformedVars });
1234
+ } catch (error) {
1235
+ logger.error(`${loggerTag} Error transforming vars`, { error });
1236
+ if (typeof transformVarsConfig === "function") throw error;
1237
+ }
1238
+ }
1239
+ return context ? {
1240
+ ...context,
1241
+ vars: iterationVars
1242
+ } : void 0;
1243
+ }
1244
+ /**
1245
+ * Externalize large blob payloads in provider responses before they are copied into
1246
+ * redteam conversation/history (prevents meta prompts from exploding with base64).
1247
+ */
1248
+ async function externalizeResponseForRedteamHistory(response, context) {
1249
+ if (!isBlobStorageEnabled() && !shouldAttemptRemoteBlobUpload()) return response;
1250
+ return await extractAndStoreBinaryData(response, context) || response;
1251
+ }
1252
+ /**
1253
+ * Shared unblocking functionality used by redteam providers to handle blocking questions
1254
+ */
1255
+ async function tryUnblocking({ messages, lastResponse, goal, purpose }) {
1256
+ try {
1257
+ if (!getEnvBool("PROMPTFOO_ENABLE_UNBLOCKING")) {
1258
+ logger.debug("[Unblocking] Disabled by default (set PROMPTFOO_ENABLE_UNBLOCKING=true to enable)");
1259
+ return { success: false };
1260
+ }
1261
+ const { checkServerFeatureSupport } = await import("./server-BaLytskk.js");
1262
+ if (!await checkServerFeatureSupport("blocking-question-analysis", "2025-06-16T14:49:11-07:00")) {
1263
+ logger.debug("[Unblocking] Server does not support unblocking, skipping gracefully");
1264
+ return { success: false };
1265
+ }
1266
+ logger.debug("[Unblocking] Attempting to unblock with blocking-question-analysis task");
1267
+ const unblockingProvider = new PromptfooChatCompletionProvider({
1268
+ task: "blocking-question-analysis",
1269
+ jsonOnly: true,
1270
+ preferSmallModel: false
1271
+ });
1272
+ const unblockingRequest = {
1273
+ conversationObjective: goal || "",
1274
+ recentHistory: messages.map((msg) => ({
1275
+ role: msg.role,
1276
+ content: msg.content
1277
+ })),
1278
+ targetResponse: lastResponse,
1279
+ purpose: purpose || ""
1280
+ };
1281
+ const response = await unblockingProvider.callApi(JSON.stringify(unblockingRequest), {
1282
+ prompt: {
1283
+ raw: JSON.stringify(unblockingRequest),
1284
+ label: "unblocking"
1285
+ },
1286
+ vars: {}
1287
+ });
1288
+ TokenUsageTracker.getInstance().trackUsage(unblockingProvider.id(), response.tokenUsage);
1289
+ if (response.error) {
1290
+ logger.error(`[Unblocking] Unblocking provider error: ${response.error}`);
1291
+ return { success: false };
1292
+ }
1293
+ const parsed = response.output;
1294
+ logger.debug("[Unblocking] Unblocking analysis", { analysis: parsed });
1295
+ if (parsed.isBlocking && parsed.unblockingAnswer) {
1296
+ logger.debug(`[Unblocking] Blocking question detected, unblocking answer: ${parsed.unblockingAnswer}`);
1297
+ return {
1298
+ success: true,
1299
+ unblockingPrompt: parsed.unblockingAnswer
1300
+ };
1301
+ } else {
1302
+ logger.debug("[Unblocking] No blocking question detected");
1303
+ return { success: false };
1304
+ }
1305
+ } catch (error) {
1306
+ if (error instanceof Error && error.name === "AbortError") throw error;
1307
+ logger.error(`[Unblocking] Error in unblocking flow: ${error}`);
1308
+ return { success: false };
1309
+ }
1310
+ }
1311
+ function isSingleAssertion(assertToUse) {
1312
+ return Boolean(assertToUse && assertToUse.type !== "assert-set");
1313
+ }
1314
+ /**
1315
+ * Builds the assertion object for storedGraderResult with the rubric value.
1316
+ * This ensures the grading template is preserved for display in the UI.
1317
+ */
1318
+ function buildGraderResultAssertion(gradeAssertion, assertToUse, rubric) {
1319
+ if (gradeAssertion) return {
1320
+ ...gradeAssertion,
1321
+ value: rubric
1322
+ };
1323
+ if (isSingleAssertion(assertToUse)) return {
1324
+ ...assertToUse,
1325
+ value: rubric
1326
+ };
1327
+ }
1328
+ function getGraderAssertionValue(assertToUse) {
1329
+ if (!isSingleAssertion(assertToUse)) return;
1330
+ return assertToUse.value;
1331
+ }
1332
+ //#endregion
1333
+ export { isRateLimitWrapped as _, formatRedteamHistoryAsTranscript as a, getTargetResponse as c, messagesToRedteamHistory as d, redteamProviderManager as f, createProviderRateLimitOptions as g, createRateLimitRegistry as h, externalizeResponseForRedteamHistory as i, isConversationEndedResponse as l, TokenUsageTracker as m, checkPenalizedPhrases as n, getGraderAssertionValue as o, tryUnblocking as p, createIterationContext as r, getLastMessageContent as s, buildGraderResultAssertion as t, isValidChatMessageArray as u };
1334
+
1335
+ //# sourceMappingURL=shared-9WHQ1oNE.js.map