superacli 1.1.5 → 1.1.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 (1081) hide show
  1. package/README.md +77 -53
  2. package/__tests__/azd-plugin.test.js +109 -0
  3. package/__tests__/config.test.js +4 -3
  4. package/__tests__/discover.test.js +59 -0
  5. package/__tests__/goose-plugin.test.js +149 -0
  6. package/__tests__/help-json.test.js +2 -0
  7. package/__tests__/openhands-plugin.test.js +106 -0
  8. package/__tests__/plugin-cocoindex-code-uninstall.test.js +19 -0
  9. package/__tests__/plugin-cocoindex-code.test.js +37 -0
  10. package/__tests__/plugin-install-guidance.test.js +81 -0
  11. package/__tests__/plugins-registry.test.js +44 -0
  12. package/__tests__/plugins-store.test.js +40 -5
  13. package/__tests__/process-adapter.test.js +50 -1
  14. package/__tests__/resend-plugin.test.js +22 -1
  15. package/__tests__/server-app.test.js +1 -0
  16. package/__tests__/server-routes-commands.test.js +20 -2
  17. package/__tests__/server-routes-plugins.test.js +130 -0
  18. package/__tests__/skills.test.js +26 -0
  19. package/__tests__/squirrelscan-plugin.test.js +129 -0
  20. package/__tests__/uipath-plugin.test.js +104 -0
  21. package/__tests__/uipathcli-plugin.test.js +95 -0
  22. package/cli/adapters/mcp.js +2 -0
  23. package/cli/adapters/process.js +49 -2
  24. package/cli/config.js +240 -3
  25. package/cli/discover.js +157 -0
  26. package/cli/help-json.js +16 -1
  27. package/cli/plugin-install-guidance.js +92 -37
  28. package/cli/plugins-manager.js +1 -0
  29. package/cli/plugins-registry.js +74 -8
  30. package/cli/plugins-store.js +78 -17
  31. package/cli/skills-mcp.js +1 -1
  32. package/cli/skills.js +39 -2
  33. package/cli/supercli.js +87 -11
  34. package/docs/feature-gaps.md +8 -8
  35. package/docs/features/azd-uipath-plugins.md +43 -0
  36. package/docs/features/server-plugins.md +62 -0
  37. package/docs/features/skills.md +9 -5
  38. package/docs/{supported-harnesses.md → plugins-available.md} +4 -3
  39. package/docs/{plugin-harness-guide.md → plugins-how-to.md} +1 -1
  40. package/docs/plugins.md +26 -20
  41. package/docs/server-plugins-usage-guide.md +182 -0
  42. package/docs/skills-catalog.md +12 -10
  43. package/package.json +1 -1
  44. package/plugins/agent-browser/README.md +69 -0
  45. package/plugins/agent-browser/plugin.json +111 -0
  46. package/plugins/agent-browser/skills/quickstart/SKILL.md +66 -0
  47. package/plugins/aider/README.md +53 -0
  48. package/plugins/aider/plugin.json +105 -0
  49. package/plugins/aider/scripts/aider-wrapper.js +243 -0
  50. package/plugins/aider/scripts/setup-aider.js +37 -0
  51. package/plugins/aider/skills/dry-run-review.md +24 -0
  52. package/plugins/aider/skills/model-and-provider.md +24 -0
  53. package/plugins/aider/skills/one-shot-edits.md +30 -0
  54. package/plugins/aider/skills/quickstart/SKILL.md +51 -0
  55. package/plugins/azd/README.md +28 -0
  56. package/plugins/azd/plugin.json +87 -0
  57. package/plugins/azd/skills/quickstart/SKILL.md +41 -0
  58. package/plugins/blogwatcher/README.md +3 -3
  59. package/plugins/boxlite/Dockerfile +9 -0
  60. package/plugins/boxlite/README.md +62 -0
  61. package/plugins/boxlite/plugin.json +201 -0
  62. package/plugins/boxlite/scripts/run-boxlite.js +106 -0
  63. package/plugins/boxlite/skills/quickstart/SKILL.md +40 -0
  64. package/plugins/cass/plugin.json +150 -0
  65. package/plugins/cass/scripts/setup-cass.js +47 -0
  66. package/plugins/cass/skills/quickstart/SKILL.md +46 -0
  67. package/plugins/clever/README.md +46 -0
  68. package/plugins/clever/plugin.json +119 -0
  69. package/plugins/clever/scripts/setup-clever.js +28 -0
  70. package/plugins/clever/skills/auth-and-profile.md +29 -0
  71. package/plugins/clever/skills/passthrough-safety.md +21 -0
  72. package/plugins/clever/skills/quickstart/SKILL.md +45 -0
  73. package/plugins/clever/skills/resource-inventory.md +24 -0
  74. package/plugins/clix/README.md +4 -4
  75. package/plugins/cocoindex-code/README.md +64 -0
  76. package/plugins/cocoindex-code/plugin.json +81 -0
  77. package/plugins/cocoindex-code/scripts/__pycache__/query.cpython-310.pyc +0 -0
  78. package/plugins/cocoindex-code/scripts/__pycache__/query.cpython-311.pyc +0 -0
  79. package/plugins/cocoindex-code/scripts/post-install.js +61 -0
  80. package/plugins/cocoindex-code/scripts/post-uninstall.js +25 -0
  81. package/plugins/cocoindex-code/scripts/query.py +88 -0
  82. package/plugins/cocoindex-code/scripts/run-query.js +50 -0
  83. package/plugins/cocoindex-code/skills/quickstart/SKILL.md +73 -0
  84. package/plugins/copilot/README.md +24 -0
  85. package/plugins/copilot/plugin.json +80 -0
  86. package/plugins/copilot/skills/quickstart/SKILL.md +44 -0
  87. package/plugins/gemini/README.md +24 -0
  88. package/plugins/gemini/plugin.json +98 -0
  89. package/plugins/gemini/skills/quickstart/SKILL.md +44 -0
  90. package/plugins/gifcap/plugin.json +119 -0
  91. package/plugins/gifcap/scripts/setup-gifcap.js +44 -0
  92. package/plugins/gifcap/skills/quickstart/SKILL.md +34 -0
  93. package/plugins/gifcap/test-record-quiet.gif +0 -0
  94. package/plugins/gifcap/test-record.gif +0 -0
  95. package/plugins/goose/README.md +36 -0
  96. package/plugins/goose/plugin.json +183 -0
  97. package/plugins/goose/skills/quickstart/SKILL.md +44 -0
  98. package/plugins/json-server/README.md +58 -0
  99. package/plugins/json-server/plugin.json +113 -0
  100. package/plugins/json-server/skills/quickstart/SKILL.md +57 -0
  101. package/plugins/lightpanda/README.md +145 -0
  102. package/plugins/lightpanda/package-lock.json +1375 -0
  103. package/plugins/lightpanda/package.json +12 -0
  104. package/plugins/lightpanda/plugin.json +116 -0
  105. package/plugins/lightpanda/scripts/lightpanda-contacts.js +494 -0
  106. package/plugins/lightpanda/scripts/lightpanda-generic-extract.js +403 -0
  107. package/plugins/lightpanda/scripts/lightpanda-wrapper.js +480 -0
  108. package/plugins/lightpanda/scripts/setup-lightpanda.js +39 -0
  109. package/plugins/lightpanda/skills/contact-discovery.md +51 -0
  110. package/plugins/lightpanda/skills/generic-extraction.md +66 -0
  111. package/plugins/lightpanda/skills/quickstart/SKILL.md +103 -0
  112. package/plugins/lightpanda/skills/resilient-navigation.md +42 -0
  113. package/plugins/monty/README.md +2 -2
  114. package/plugins/nullclaw/README.md +3 -3
  115. package/plugins/offline-ai/README.md +23 -0
  116. package/plugins/offline-ai/plugin.json +82 -0
  117. package/plugins/offline-ai/skills/quickstart/SKILL.md +43 -0
  118. package/plugins/openhands/README.md +25 -0
  119. package/plugins/openhands/plugin.json +116 -0
  120. package/plugins/openhands/skills/quickstart/SKILL.md +26 -0
  121. package/plugins/plandex/README.md +25 -0
  122. package/plugins/plandex/plugin.json +130 -0
  123. package/plugins/plandex/skills/quickstart/SKILL.md +50 -0
  124. package/plugins/plugins.json +190 -2
  125. package/plugins/resend/plugin.json +279 -2
  126. package/plugins/resend/skills/quickstart/SKILL.md +32 -13
  127. package/plugins/squirrelscan/Dockerfile +5 -0
  128. package/plugins/squirrelscan/README.md +47 -0
  129. package/plugins/squirrelscan/plugin.json +493 -0
  130. package/plugins/squirrelscan/scripts/post-install.js +33 -0
  131. package/plugins/squirrelscan/scripts/post-uninstall.js +25 -0
  132. package/plugins/squirrelscan/scripts/run-squirrel.js +73 -0
  133. package/plugins/squirrelscan/skills/audit-workflow/SKILL.md +33 -0
  134. package/plugins/squirrelscan/skills/publish-report/SKILL.md +33 -0
  135. package/plugins/squirrelscan/skills/quickstart/SKILL.md +41 -0
  136. package/plugins/uipath/README.md +27 -0
  137. package/plugins/uipath/plugin.json +86 -0
  138. package/plugins/uipath/skills/quickstart/SKILL.md +47 -0
  139. package/plugins/uipathcli/README.md +28 -0
  140. package/plugins/uipathcli/plugin.json +120 -0
  141. package/plugins/uipathcli/scripts/run-uipath-cli.js +49 -0
  142. package/plugins/uipathcli/skills/quickstart/SKILL.md +22 -0
  143. package/plugins/xurl/README.md +4 -4
  144. package/server/app.js +5 -2
  145. package/server/public/app.js +3 -0
  146. package/server/routes/commands.js +95 -12
  147. package/server/routes/plugins.js +262 -0
  148. package/server/services/pluginsService.js +303 -0
  149. package/server/views/command-edit.ejs +196 -14
  150. package/server/views/partials/head.ejs +1 -0
  151. package/server/views/plugins.ejs +264 -0
  152. package/tests/test-plugins-registry.js +30 -0
  153. package/tests/test-resend-smoke.sh +7 -3
  154. package/.beads/.br_history/issues.20260308_200823_636718328.jsonl +0 -20
  155. package/.beads/.br_history/issues.20260308_200823_636718328.jsonl.meta.json +0 -1
  156. package/.beads/.br_history/issues.20260308_200827_033159453.jsonl +0 -21
  157. package/.beads/.br_history/issues.20260308_200827_033159453.jsonl.meta.json +0 -1
  158. package/.beads/.br_history/issues.20260308_200829_595900053.jsonl +0 -22
  159. package/.beads/.br_history/issues.20260308_200829_595900053.jsonl.meta.json +0 -1
  160. package/.beads/.br_history/issues.20260308_200834_079930100.jsonl +0 -23
  161. package/.beads/.br_history/issues.20260308_200834_079930100.jsonl.meta.json +0 -1
  162. package/.beads/.br_history/issues.20260308_200858_370924996.jsonl +0 -24
  163. package/.beads/.br_history/issues.20260308_200858_370924996.jsonl.meta.json +0 -1
  164. package/.beads/.br_history/issues.20260308_201031_019730855.jsonl +0 -24
  165. package/.beads/.br_history/issues.20260308_201031_019730855.jsonl.meta.json +0 -1
  166. package/.beads/.br_history/issues.20260308_201031_578974884.jsonl +0 -24
  167. package/.beads/.br_history/issues.20260308_201031_578974884.jsonl.meta.json +0 -1
  168. package/.beads/.br_history/issues.20260308_201054_780345548.jsonl +0 -24
  169. package/.beads/.br_history/issues.20260308_201054_780345548.jsonl.meta.json +0 -1
  170. package/.beads/.br_history/issues.20260308_201054_896980019.jsonl +0 -24
  171. package/.beads/.br_history/issues.20260308_201054_896980019.jsonl.meta.json +0 -1
  172. package/.beads/.br_history/issues.20260308_201128_599819688.jsonl +0 -24
  173. package/.beads/.br_history/issues.20260308_201128_599819688.jsonl.meta.json +0 -1
  174. package/.beads/.br_history/issues.20260308_201128_710221699.jsonl +0 -24
  175. package/.beads/.br_history/issues.20260308_201128_710221699.jsonl.meta.json +0 -1
  176. package/.beads/.br_history/issues.20260308_201204_745649213.jsonl +0 -24
  177. package/.beads/.br_history/issues.20260308_201204_745649213.jsonl.meta.json +0 -1
  178. package/.beads/.br_history/issues.20260308_201338_908436144.jsonl +0 -24
  179. package/.beads/.br_history/issues.20260308_201338_908436144.jsonl.meta.json +0 -1
  180. package/.beads/.br_history/issues.20260308_201344_734860714.jsonl +0 -25
  181. package/.beads/.br_history/issues.20260308_201344_734860714.jsonl.meta.json +0 -1
  182. package/.beads/.br_history/issues.20260308_201630_819282295.jsonl +0 -25
  183. package/.beads/.br_history/issues.20260308_201630_819282295.jsonl.meta.json +0 -1
  184. package/.beads/.br_history/issues.20260308_203538_054279699.jsonl +0 -25
  185. package/.beads/.br_history/issues.20260308_203538_054279699.jsonl.meta.json +0 -1
  186. package/.beads/.br_history/issues.20260308_203547_597113070.jsonl +0 -26
  187. package/.beads/.br_history/issues.20260308_203547_597113070.jsonl.meta.json +0 -1
  188. package/.beads/.br_history/issues.20260308_203547_775139216.jsonl +0 -27
  189. package/.beads/.br_history/issues.20260308_203547_775139216.jsonl.meta.json +0 -1
  190. package/.beads/.br_history/issues.20260308_203547_950724773.jsonl +0 -28
  191. package/.beads/.br_history/issues.20260308_203547_950724773.jsonl.meta.json +0 -1
  192. package/.beads/.br_history/issues.20260308_203548_107684523.jsonl +0 -29
  193. package/.beads/.br_history/issues.20260308_203548_107684523.jsonl.meta.json +0 -1
  194. package/.beads/.br_history/issues.20260308_203548_310389993.jsonl +0 -30
  195. package/.beads/.br_history/issues.20260308_203548_310389993.jsonl.meta.json +0 -1
  196. package/.beads/.br_history/issues.20260308_203825_953337320.jsonl +0 -31
  197. package/.beads/.br_history/issues.20260308_203825_953337320.jsonl.meta.json +0 -1
  198. package/.beads/.br_history/issues.20260308_204056_071377736.jsonl +0 -32
  199. package/.beads/.br_history/issues.20260308_204056_071377736.jsonl.meta.json +0 -1
  200. package/.beads/.br_history/issues.20260308_205141_517616844.jsonl +0 -32
  201. package/.beads/.br_history/issues.20260308_205141_517616844.jsonl.meta.json +0 -1
  202. package/.beads/.br_history/issues.20260308_205141_648994024.jsonl +0 -32
  203. package/.beads/.br_history/issues.20260308_205141_648994024.jsonl.meta.json +0 -1
  204. package/.beads/.br_history/issues.20260308_205141_867598036.jsonl +0 -32
  205. package/.beads/.br_history/issues.20260308_205141_867598036.jsonl.meta.json +0 -1
  206. package/.beads/.br_history/issues.20260308_205142_094157355.jsonl +0 -32
  207. package/.beads/.br_history/issues.20260308_205142_094157355.jsonl.meta.json +0 -1
  208. package/.beads/.br_history/issues.20260308_205142_327315677.jsonl +0 -32
  209. package/.beads/.br_history/issues.20260308_205142_327315677.jsonl.meta.json +0 -1
  210. package/.beads/.br_history/issues.20260308_205142_545563822.jsonl +0 -32
  211. package/.beads/.br_history/issues.20260308_205142_545563822.jsonl.meta.json +0 -1
  212. package/.beads/.br_history/issues.20260308_205213_061989333.jsonl +0 -32
  213. package/.beads/.br_history/issues.20260308_205213_061989333.jsonl.meta.json +0 -1
  214. package/.beads/.br_history/issues.20260308_205213_181103364.jsonl +0 -32
  215. package/.beads/.br_history/issues.20260308_205213_181103364.jsonl.meta.json +0 -1
  216. package/.beads/.br_history/issues.20260308_205213_408872234.jsonl +0 -32
  217. package/.beads/.br_history/issues.20260308_205213_408872234.jsonl.meta.json +0 -1
  218. package/.beads/.br_history/issues.20260308_205213_616681652.jsonl +0 -32
  219. package/.beads/.br_history/issues.20260308_205213_616681652.jsonl.meta.json +0 -1
  220. package/.beads/.br_history/issues.20260308_205213_821507069.jsonl +0 -32
  221. package/.beads/.br_history/issues.20260308_205213_821507069.jsonl.meta.json +0 -1
  222. package/.beads/.br_history/issues.20260308_205214_026661112.jsonl +0 -32
  223. package/.beads/.br_history/issues.20260308_205214_026661112.jsonl.meta.json +0 -1
  224. package/.beads/.br_history/issues.20260308_205454_955250554.jsonl +0 -32
  225. package/.beads/.br_history/issues.20260308_205454_955250554.jsonl.meta.json +0 -1
  226. package/.beads/.br_history/issues.20260308_205556_337800392.jsonl +0 -33
  227. package/.beads/.br_history/issues.20260308_205556_337800392.jsonl.meta.json +0 -1
  228. package/.beads/.br_history/issues.20260308_205824_274686694.jsonl +0 -33
  229. package/.beads/.br_history/issues.20260308_205824_274686694.jsonl.meta.json +0 -1
  230. package/.beads/.br_history/issues.20260308_210240_583768328.jsonl +0 -34
  231. package/.beads/.br_history/issues.20260308_210240_583768328.jsonl.meta.json +0 -1
  232. package/.beads/.br_history/issues.20260308_212223_641541494.jsonl +0 -34
  233. package/.beads/.br_history/issues.20260308_212223_641541494.jsonl.meta.json +0 -1
  234. package/.beads/.br_history/issues.20260308_212227_735550996.jsonl +0 -35
  235. package/.beads/.br_history/issues.20260308_212227_735550996.jsonl.meta.json +0 -1
  236. package/.beads/.br_history/issues.20260308_212232_547298548.jsonl +0 -36
  237. package/.beads/.br_history/issues.20260308_212232_547298548.jsonl.meta.json +0 -1
  238. package/.beads/.br_history/issues.20260308_212528_843628125.jsonl +0 -37
  239. package/.beads/.br_history/issues.20260308_212528_843628125.jsonl.meta.json +0 -1
  240. package/.beads/.br_history/issues.20260308_212529_094530502.jsonl +0 -38
  241. package/.beads/.br_history/issues.20260308_212529_094530502.jsonl.meta.json +0 -1
  242. package/.beads/.br_history/issues.20260308_212529_331000853.jsonl +0 -39
  243. package/.beads/.br_history/issues.20260308_212529_331000853.jsonl.meta.json +0 -1
  244. package/.beads/.br_history/issues.20260308_212529_587925652.jsonl +0 -40
  245. package/.beads/.br_history/issues.20260308_212529_587925652.jsonl.meta.json +0 -1
  246. package/.beads/.br_history/issues.20260308_212804_927764103.jsonl +0 -41
  247. package/.beads/.br_history/issues.20260308_212804_927764103.jsonl.meta.json +0 -1
  248. package/.beads/.br_history/issues.20260308_212805_153673453.jsonl +0 -42
  249. package/.beads/.br_history/issues.20260308_212805_153673453.jsonl.meta.json +0 -1
  250. package/.beads/.br_history/issues.20260308_212805_415982363.jsonl +0 -43
  251. package/.beads/.br_history/issues.20260308_212805_415982363.jsonl.meta.json +0 -1
  252. package/.beads/.br_history/issues.20260308_212805_657497741.jsonl +0 -44
  253. package/.beads/.br_history/issues.20260308_212805_657497741.jsonl.meta.json +0 -1
  254. package/.beads/.br_history/issues.20260308_212805_952838724.jsonl +0 -45
  255. package/.beads/.br_history/issues.20260308_212805_952838724.jsonl.meta.json +0 -1
  256. package/.beads/.br_history/issues.20260308_212806_325433779.jsonl +0 -46
  257. package/.beads/.br_history/issues.20260308_212806_325433779.jsonl.meta.json +0 -1
  258. package/.beads/.br_history/issues.20260308_212806_584685598.jsonl +0 -47
  259. package/.beads/.br_history/issues.20260308_212806_584685598.jsonl.meta.json +0 -1
  260. package/.beads/.br_history/issues.20260308_212806_827817208.jsonl +0 -48
  261. package/.beads/.br_history/issues.20260308_212806_827817208.jsonl.meta.json +0 -1
  262. package/.beads/.br_history/issues.20260308_212807_111320451.jsonl +0 -49
  263. package/.beads/.br_history/issues.20260308_212807_111320451.jsonl.meta.json +0 -1
  264. package/.beads/.br_history/issues.20260308_212807_409545536.jsonl +0 -50
  265. package/.beads/.br_history/issues.20260308_212807_409545536.jsonl.meta.json +0 -1
  266. package/.beads/.br_history/issues.20260308_212807_625063294.jsonl +0 -51
  267. package/.beads/.br_history/issues.20260308_212807_625063294.jsonl.meta.json +0 -1
  268. package/.beads/.br_history/issues.20260308_212807_843906551.jsonl +0 -52
  269. package/.beads/.br_history/issues.20260308_212807_843906551.jsonl.meta.json +0 -1
  270. package/.beads/.br_history/issues.20260308_212808_100304073.jsonl +0 -53
  271. package/.beads/.br_history/issues.20260308_212808_100304073.jsonl.meta.json +0 -1
  272. package/.beads/.br_history/issues.20260308_212808_324723976.jsonl +0 -54
  273. package/.beads/.br_history/issues.20260308_212808_324723976.jsonl.meta.json +0 -1
  274. package/.beads/.br_history/issues.20260308_212808_557513104.jsonl +0 -55
  275. package/.beads/.br_history/issues.20260308_212808_557513104.jsonl.meta.json +0 -1
  276. package/.beads/.br_history/issues.20260308_212808_788048322.jsonl +0 -56
  277. package/.beads/.br_history/issues.20260308_212808_788048322.jsonl.meta.json +0 -1
  278. package/.beads/.br_history/issues.20260308_213702_613249728.jsonl +0 -57
  279. package/.beads/.br_history/issues.20260308_213702_613249728.jsonl.meta.json +0 -1
  280. package/.beads/.br_history/issues.20260308_213715_115792063.jsonl +0 -57
  281. package/.beads/.br_history/issues.20260308_213715_115792063.jsonl.meta.json +0 -1
  282. package/.beads/.br_history/issues.20260308_213715_462220666.jsonl +0 -57
  283. package/.beads/.br_history/issues.20260308_213715_462220666.jsonl.meta.json +0 -1
  284. package/.beads/.br_history/issues.20260308_213727_191258923.jsonl +0 -57
  285. package/.beads/.br_history/issues.20260308_213727_191258923.jsonl.meta.json +0 -1
  286. package/.beads/.br_history/issues.20260308_213727_684383652.jsonl +0 -57
  287. package/.beads/.br_history/issues.20260308_213727_684383652.jsonl.meta.json +0 -1
  288. package/.beads/.br_history/issues.20260308_213735_751882991.jsonl +0 -57
  289. package/.beads/.br_history/issues.20260308_213735_751882991.jsonl.meta.json +0 -1
  290. package/.beads/.br_history/issues.20260308_222052_279844960.jsonl +0 -57
  291. package/.beads/.br_history/issues.20260308_222052_279844960.jsonl.meta.json +0 -1
  292. package/.beads/.br_history/issues.20260308_222056_873282114.jsonl +0 -57
  293. package/.beads/.br_history/issues.20260308_222056_873282114.jsonl.meta.json +0 -1
  294. package/.beads/.br_history/issues.20260308_222103_402410761.jsonl +0 -57
  295. package/.beads/.br_history/issues.20260308_222103_402410761.jsonl.meta.json +0 -1
  296. package/.beads/.br_history/issues.20260308_235202_180577215.jsonl +0 -57
  297. package/.beads/.br_history/issues.20260308_235202_180577215.jsonl.meta.json +0 -1
  298. package/.beads/.br_history/issues.20260308_235202_387414163.jsonl +0 -57
  299. package/.beads/.br_history/issues.20260308_235202_387414163.jsonl.meta.json +0 -1
  300. package/.beads/.br_history/issues.20260308_235202_564422794.jsonl +0 -57
  301. package/.beads/.br_history/issues.20260308_235202_564422794.jsonl.meta.json +0 -1
  302. package/.beads/.br_history/issues.20260308_235202_742600597.jsonl +0 -57
  303. package/.beads/.br_history/issues.20260308_235202_742600597.jsonl.meta.json +0 -1
  304. package/.beads/.br_history/issues.20260308_235208_133360069.jsonl +0 -57
  305. package/.beads/.br_history/issues.20260308_235208_133360069.jsonl.meta.json +0 -1
  306. package/.beads/.br_history/issues.20260308_235505_473406307.jsonl +0 -57
  307. package/.beads/.br_history/issues.20260308_235505_473406307.jsonl.meta.json +0 -1
  308. package/.beads/.br_history/issues.20260308_235505_662360489.jsonl +0 -57
  309. package/.beads/.br_history/issues.20260308_235505_662360489.jsonl.meta.json +0 -1
  310. package/.beads/.br_history/issues.20260308_235505_843935624.jsonl +0 -57
  311. package/.beads/.br_history/issues.20260308_235505_843935624.jsonl.meta.json +0 -1
  312. package/.beads/.br_history/issues.20260308_235506_044530221.jsonl +0 -57
  313. package/.beads/.br_history/issues.20260308_235506_044530221.jsonl.meta.json +0 -1
  314. package/.beads/.br_history/issues.20260309_002618_115728731.jsonl +0 -57
  315. package/.beads/.br_history/issues.20260309_002618_115728731.jsonl.meta.json +0 -1
  316. package/.beads/.br_history/issues.20260309_003748_878174586.jsonl +0 -57
  317. package/.beads/.br_history/issues.20260309_003748_878174586.jsonl.meta.json +0 -1
  318. package/.beads/.br_history/issues.20260309_004057_868755623.jsonl +0 -57
  319. package/.beads/.br_history/issues.20260309_004057_868755623.jsonl.meta.json +0 -1
  320. package/.beads/.br_history/issues.20260309_004058_512842163.jsonl +0 -57
  321. package/.beads/.br_history/issues.20260309_004058_512842163.jsonl.meta.json +0 -1
  322. package/.beads/.br_history/issues.20260309_004058_994445226.jsonl +0 -57
  323. package/.beads/.br_history/issues.20260309_004058_994445226.jsonl.meta.json +0 -1
  324. package/.beads/.br_history/issues.20260309_004059_475988596.jsonl +0 -57
  325. package/.beads/.br_history/issues.20260309_004059_475988596.jsonl.meta.json +0 -1
  326. package/.beads/.br_history/issues.20260309_161902_566857851.jsonl +0 -57
  327. package/.beads/.br_history/issues.20260309_161902_566857851.jsonl.meta.json +0 -1
  328. package/.beads/.br_history/issues.20260309_170512_277017739.jsonl +0 -57
  329. package/.beads/.br_history/issues.20260309_170512_277017739.jsonl.meta.json +0 -1
  330. package/.beads/.br_history/issues.20260309_170512_477876921.jsonl +0 -57
  331. package/.beads/.br_history/issues.20260309_170512_477876921.jsonl.meta.json +0 -1
  332. package/.beads/.br_history/issues.20260309_170512_664382701.jsonl +0 -57
  333. package/.beads/.br_history/issues.20260309_170512_664382701.jsonl.meta.json +0 -1
  334. package/.beads/.br_history/issues.20260309_170512_859400333.jsonl +0 -57
  335. package/.beads/.br_history/issues.20260309_170512_859400333.jsonl.meta.json +0 -1
  336. package/.beads/.br_history/issues.20260309_212326_082771164.jsonl +0 -57
  337. package/.beads/.br_history/issues.20260309_212326_082771164.jsonl.meta.json +0 -1
  338. package/.beads/.br_history/issues.20260309_212326_245619716.jsonl +0 -58
  339. package/.beads/.br_history/issues.20260309_212326_245619716.jsonl.meta.json +0 -1
  340. package/.beads/.br_history/issues.20260309_212326_403198317.jsonl +0 -59
  341. package/.beads/.br_history/issues.20260309_212326_403198317.jsonl.meta.json +0 -1
  342. package/.beads/.br_history/issues.20260309_212332_539197678.jsonl +0 -60
  343. package/.beads/.br_history/issues.20260309_212332_539197678.jsonl.meta.json +0 -1
  344. package/.beads/.br_history/issues.20260309_212332_731373599.jsonl +0 -60
  345. package/.beads/.br_history/issues.20260309_212332_731373599.jsonl.meta.json +0 -1
  346. package/.beads/.br_history/issues.20260309_212332_928710953.jsonl +0 -60
  347. package/.beads/.br_history/issues.20260309_212332_928710953.jsonl.meta.json +0 -1
  348. package/.beads/.br_history/issues.20260309_213021_341505240.jsonl +0 -60
  349. package/.beads/.br_history/issues.20260309_213021_341505240.jsonl.meta.json +0 -1
  350. package/.beads/.br_history/issues.20260309_213022_023136934.jsonl +0 -60
  351. package/.beads/.br_history/issues.20260309_213022_023136934.jsonl.meta.json +0 -1
  352. package/.beads/.br_history/issues.20260309_213022_400050719.jsonl +0 -60
  353. package/.beads/.br_history/issues.20260309_213022_400050719.jsonl.meta.json +0 -1
  354. package/.beads/config.yaml +0 -4
  355. package/.beads/issues.jsonl +0 -60
  356. package/.beads/metadata.json +0 -4
  357. package/docs/mcp-cheatsheet.md +0 -324
  358. package/docs/visual-overview.md +0 -21
  359. package/ref-monty/.cargo/config.toml +0 -3
  360. package/ref-monty/.claude/settings.json +0 -60
  361. package/ref-monty/.claude/skills/fastmod/SKILL.md +0 -22
  362. package/ref-monty/.claude/skills/python-playground/SKILL.md +0 -47
  363. package/ref-monty/.codecov.yml +0 -12
  364. package/ref-monty/.github/actions/build-pgo-wheel/action.yml +0 -72
  365. package/ref-monty/.github/workflows/ci.yml +0 -776
  366. package/ref-monty/.github/workflows/codspeed.yml +0 -45
  367. package/ref-monty/.github/workflows/init-npm-packages.yml +0 -82
  368. package/ref-monty/.pre-commit-config.yaml +0 -47
  369. package/ref-monty/.python-version +0 -1
  370. package/ref-monty/.rustfmt.toml +0 -4
  371. package/ref-monty/.zed/settings.json +0 -11
  372. package/ref-monty/CLAUDE.md +0 -535
  373. package/ref-monty/Cargo.lock +0 -3798
  374. package/ref-monty/Cargo.toml +0 -87
  375. package/ref-monty/LICENSE +0 -21
  376. package/ref-monty/Makefile +0 -216
  377. package/ref-monty/README.md +0 -430
  378. package/ref-monty/RELEASING.md +0 -47
  379. package/ref-monty/crates/fuzz/Cargo.toml +0 -30
  380. package/ref-monty/crates/fuzz/fuzz_targets/string_input_panic.rs +0 -37
  381. package/ref-monty/crates/fuzz/fuzz_targets/tokens_input_panic.rs +0 -552
  382. package/ref-monty/crates/monty/Cargo.toml +0 -68
  383. package/ref-monty/crates/monty/benches/main.rs +0 -247
  384. package/ref-monty/crates/monty/build.rs +0 -10
  385. package/ref-monty/crates/monty/src/args.rs +0 -733
  386. package/ref-monty/crates/monty/src/asyncio.rs +0 -179
  387. package/ref-monty/crates/monty/src/builtins/abs.rs +0 -55
  388. package/ref-monty/crates/monty/src/builtins/all.rs +0 -30
  389. package/ref-monty/crates/monty/src/builtins/any.rs +0 -30
  390. package/ref-monty/crates/monty/src/builtins/bin.rs +0 -59
  391. package/ref-monty/crates/monty/src/builtins/chr.rs +0 -46
  392. package/ref-monty/crates/monty/src/builtins/divmod.rs +0 -164
  393. package/ref-monty/crates/monty/src/builtins/enumerate.rs +0 -52
  394. package/ref-monty/crates/monty/src/builtins/filter.rs +0 -67
  395. package/ref-monty/crates/monty/src/builtins/getattr.rs +0 -65
  396. package/ref-monty/crates/monty/src/builtins/hash.rs +0 -28
  397. package/ref-monty/crates/monty/src/builtins/hex.rs +0 -58
  398. package/ref-monty/crates/monty/src/builtins/id.rs +0 -24
  399. package/ref-monty/crates/monty/src/builtins/isinstance.rs +0 -68
  400. package/ref-monty/crates/monty/src/builtins/len.rs +0 -25
  401. package/ref-monty/crates/monty/src/builtins/map.rs +0 -98
  402. package/ref-monty/crates/monty/src/builtins/min_max.rs +0 -113
  403. package/ref-monty/crates/monty/src/builtins/mod.rs +0 -246
  404. package/ref-monty/crates/monty/src/builtins/next.rs +0 -21
  405. package/ref-monty/crates/monty/src/builtins/oct.rs +0 -59
  406. package/ref-monty/crates/monty/src/builtins/ord.rs +0 -67
  407. package/ref-monty/crates/monty/src/builtins/pow.rs +0 -365
  408. package/ref-monty/crates/monty/src/builtins/print.rs +0 -141
  409. package/ref-monty/crates/monty/src/builtins/repr.rs +0 -16
  410. package/ref-monty/crates/monty/src/builtins/reversed.rs +0 -28
  411. package/ref-monty/crates/monty/src/builtins/round.rs +0 -174
  412. package/ref-monty/crates/monty/src/builtins/sorted.rs +0 -151
  413. package/ref-monty/crates/monty/src/builtins/sum.rs +0 -66
  414. package/ref-monty/crates/monty/src/builtins/type_.rs +0 -16
  415. package/ref-monty/crates/monty/src/builtins/zip.rs +0 -77
  416. package/ref-monty/crates/monty/src/bytecode/builder.rs +0 -699
  417. package/ref-monty/crates/monty/src/bytecode/code.rs +0 -310
  418. package/ref-monty/crates/monty/src/bytecode/compiler.rs +0 -3206
  419. package/ref-monty/crates/monty/src/bytecode/mod.rs +0 -24
  420. package/ref-monty/crates/monty/src/bytecode/op.rs +0 -617
  421. package/ref-monty/crates/monty/src/bytecode/vm/async_exec.rs +0 -1058
  422. package/ref-monty/crates/monty/src/bytecode/vm/attr.rs +0 -63
  423. package/ref-monty/crates/monty/src/bytecode/vm/binary.rs +0 -487
  424. package/ref-monty/crates/monty/src/bytecode/vm/call.rs +0 -767
  425. package/ref-monty/crates/monty/src/bytecode/vm/collections.rs +0 -741
  426. package/ref-monty/crates/monty/src/bytecode/vm/compare.rs +0 -147
  427. package/ref-monty/crates/monty/src/bytecode/vm/exceptions.rs +0 -297
  428. package/ref-monty/crates/monty/src/bytecode/vm/format.rs +0 -132
  429. package/ref-monty/crates/monty/src/bytecode/vm/mod.rs +0 -1958
  430. package/ref-monty/crates/monty/src/bytecode/vm/scheduler.rs +0 -620
  431. package/ref-monty/crates/monty/src/exception_private.rs +0 -1513
  432. package/ref-monty/crates/monty/src/exception_public.rs +0 -346
  433. package/ref-monty/crates/monty/src/expressions.rs +0 -694
  434. package/ref-monty/crates/monty/src/fstring.rs +0 -854
  435. package/ref-monty/crates/monty/src/function.rs +0 -119
  436. package/ref-monty/crates/monty/src/heap.rs +0 -1073
  437. package/ref-monty/crates/monty/src/heap_data.rs +0 -985
  438. package/ref-monty/crates/monty/src/heap_traits.rs +0 -312
  439. package/ref-monty/crates/monty/src/intern.rs +0 -837
  440. package/ref-monty/crates/monty/src/io.rs +0 -106
  441. package/ref-monty/crates/monty/src/lib.rs +0 -52
  442. package/ref-monty/crates/monty/src/modules/asyncio.rs +0 -144
  443. package/ref-monty/crates/monty/src/modules/math.rs +0 -1453
  444. package/ref-monty/crates/monty/src/modules/mod.rs +0 -120
  445. package/ref-monty/crates/monty/src/modules/os.rs +0 -116
  446. package/ref-monty/crates/monty/src/modules/pathlib.rs +0 -33
  447. package/ref-monty/crates/monty/src/modules/re.rs +0 -606
  448. package/ref-monty/crates/monty/src/modules/sys.rs +0 -60
  449. package/ref-monty/crates/monty/src/modules/typing.rs +0 -70
  450. package/ref-monty/crates/monty/src/namespace.rs +0 -21
  451. package/ref-monty/crates/monty/src/object.rs +0 -1040
  452. package/ref-monty/crates/monty/src/os.rs +0 -215
  453. package/ref-monty/crates/monty/src/parse.rs +0 -1730
  454. package/ref-monty/crates/monty/src/prepare.rs +0 -3015
  455. package/ref-monty/crates/monty/src/repl.rs +0 -1109
  456. package/ref-monty/crates/monty/src/resource.rs +0 -559
  457. package/ref-monty/crates/monty/src/run.rs +0 -457
  458. package/ref-monty/crates/monty/src/run_progress.rs +0 -821
  459. package/ref-monty/crates/monty/src/signature.rs +0 -651
  460. package/ref-monty/crates/monty/src/sorting.rs +0 -100
  461. package/ref-monty/crates/monty/src/types/bytes.rs +0 -2356
  462. package/ref-monty/crates/monty/src/types/dataclass.rs +0 -345
  463. package/ref-monty/crates/monty/src/types/dict.rs +0 -879
  464. package/ref-monty/crates/monty/src/types/dict_view.rs +0 -619
  465. package/ref-monty/crates/monty/src/types/iter.rs +0 -799
  466. package/ref-monty/crates/monty/src/types/list.rs +0 -929
  467. package/ref-monty/crates/monty/src/types/long_int.rs +0 -211
  468. package/ref-monty/crates/monty/src/types/mod.rs +0 -48
  469. package/ref-monty/crates/monty/src/types/module.rs +0 -146
  470. package/ref-monty/crates/monty/src/types/namedtuple.rs +0 -261
  471. package/ref-monty/crates/monty/src/types/path.rs +0 -596
  472. package/ref-monty/crates/monty/src/types/property.rs +0 -35
  473. package/ref-monty/crates/monty/src/types/py_trait.rs +0 -322
  474. package/ref-monty/crates/monty/src/types/range.rs +0 -285
  475. package/ref-monty/crates/monty/src/types/re_match.rs +0 -522
  476. package/ref-monty/crates/monty/src/types/re_pattern.rs +0 -726
  477. package/ref-monty/crates/monty/src/types/set.rs +0 -1373
  478. package/ref-monty/crates/monty/src/types/slice.rs +0 -257
  479. package/ref-monty/crates/monty/src/types/str.rs +0 -2051
  480. package/ref-monty/crates/monty/src/types/tuple.rs +0 -376
  481. package/ref-monty/crates/monty/src/types/type.rs +0 -407
  482. package/ref-monty/crates/monty/src/value.rs +0 -2558
  483. package/ref-monty/crates/monty/test_cases/args__dict_get_no_args.py +0 -3
  484. package/ref-monty/crates/monty/test_cases/args__dict_get_too_many.py +0 -3
  485. package/ref-monty/crates/monty/test_cases/args__dict_items_with_args.py +0 -3
  486. package/ref-monty/crates/monty/test_cases/args__dict_keys_with_args.py +0 -3
  487. package/ref-monty/crates/monty/test_cases/args__dict_pop_no_args.py +0 -3
  488. package/ref-monty/crates/monty/test_cases/args__dict_pop_too_many.py +0 -3
  489. package/ref-monty/crates/monty/test_cases/args__dict_values_with_args.py +0 -3
  490. package/ref-monty/crates/monty/test_cases/args__id_too_many.py +0 -2
  491. package/ref-monty/crates/monty/test_cases/args__len_no_args.py +0 -2
  492. package/ref-monty/crates/monty/test_cases/args__len_too_many.py +0 -2
  493. package/ref-monty/crates/monty/test_cases/args__len_type_error_int.py +0 -9
  494. package/ref-monty/crates/monty/test_cases/args__len_type_error_none.py +0 -9
  495. package/ref-monty/crates/monty/test_cases/args__list_append_no_args.py +0 -3
  496. package/ref-monty/crates/monty/test_cases/args__list_append_too_many.py +0 -3
  497. package/ref-monty/crates/monty/test_cases/args__list_insert_too_few.py +0 -3
  498. package/ref-monty/crates/monty/test_cases/args__list_insert_too_many.py +0 -3
  499. package/ref-monty/crates/monty/test_cases/args__repr_no_args.py +0 -2
  500. package/ref-monty/crates/monty/test_cases/arith__div_zero_float.py +0 -2
  501. package/ref-monty/crates/monty/test_cases/arith__div_zero_int.py +0 -2
  502. package/ref-monty/crates/monty/test_cases/arith__floordiv_zero_float.py +0 -2
  503. package/ref-monty/crates/monty/test_cases/arith__floordiv_zero_int.py +0 -2
  504. package/ref-monty/crates/monty/test_cases/arith__pow_zero_neg.py +0 -2
  505. package/ref-monty/crates/monty/test_cases/arith__pow_zero_neg_builtin.py +0 -9
  506. package/ref-monty/crates/monty/test_cases/assert__expr_fail.py +0 -2
  507. package/ref-monty/crates/monty/test_cases/assert__fail.py +0 -2
  508. package/ref-monty/crates/monty/test_cases/assert__fail_msg.py +0 -2
  509. package/ref-monty/crates/monty/test_cases/assert__fn_fail.py +0 -3
  510. package/ref-monty/crates/monty/test_cases/assert__ops.py +0 -11
  511. package/ref-monty/crates/monty/test_cases/async__asyncio_run.py +0 -47
  512. package/ref-monty/crates/monty/test_cases/async__basic.py +0 -10
  513. package/ref-monty/crates/monty/test_cases/async__closure.py +0 -14
  514. package/ref-monty/crates/monty/test_cases/async__double_await_coroutine.py +0 -16
  515. package/ref-monty/crates/monty/test_cases/async__exception.py +0 -10
  516. package/ref-monty/crates/monty/test_cases/async__ext_call.py +0 -73
  517. package/ref-monty/crates/monty/test_cases/async__gather_all.py +0 -85
  518. package/ref-monty/crates/monty/test_cases/async__nested_await.py +0 -15
  519. package/ref-monty/crates/monty/test_cases/async__nested_gather_ext.py +0 -37
  520. package/ref-monty/crates/monty/test_cases/async__not_awaitable.py +0 -10
  521. package/ref-monty/crates/monty/test_cases/async__not_imported.py +0 -14
  522. package/ref-monty/crates/monty/test_cases/async__recursion_depth_isolation.py +0 -27
  523. package/ref-monty/crates/monty/test_cases/async__return_types.py +0 -31
  524. package/ref-monty/crates/monty/test_cases/async__sequential.py +0 -16
  525. package/ref-monty/crates/monty/test_cases/async__traceback.py +0 -19
  526. package/ref-monty/crates/monty/test_cases/async__with_args.py +0 -14
  527. package/ref-monty/crates/monty/test_cases/attr__get_int_error.py +0 -9
  528. package/ref-monty/crates/monty/test_cases/attr__get_list_error.py +0 -9
  529. package/ref-monty/crates/monty/test_cases/attr__set_frozen_nonfield.py +0 -12
  530. package/ref-monty/crates/monty/test_cases/attr__set_int_error.py +0 -10
  531. package/ref-monty/crates/monty/test_cases/attr__set_list_error.py +0 -10
  532. package/ref-monty/crates/monty/test_cases/bench__kitchen_sink.py +0 -68
  533. package/ref-monty/crates/monty/test_cases/bool__ops.py +0 -20
  534. package/ref-monty/crates/monty/test_cases/builtin__add_type_error.py +0 -2
  535. package/ref-monty/crates/monty/test_cases/builtin__filter.py +0 -62
  536. package/ref-monty/crates/monty/test_cases/builtin__filter_not_iterable.py +0 -11
  537. package/ref-monty/crates/monty/test_cases/builtin__getattr.py +0 -84
  538. package/ref-monty/crates/monty/test_cases/builtin__iter_funcs.py +0 -42
  539. package/ref-monty/crates/monty/test_cases/builtin__iter_next.py +0 -66
  540. package/ref-monty/crates/monty/test_cases/builtin__map.py +0 -74
  541. package/ref-monty/crates/monty/test_cases/builtin__map_not_iterable.py +0 -11
  542. package/ref-monty/crates/monty/test_cases/builtin__math_funcs.py +0 -154
  543. package/ref-monty/crates/monty/test_cases/builtin__more_iter_funcs.py +0 -148
  544. package/ref-monty/crates/monty/test_cases/builtin__next_stop_iteration.py +0 -10
  545. package/ref-monty/crates/monty/test_cases/builtin__print_invalid_kwarg.py +0 -9
  546. package/ref-monty/crates/monty/test_cases/builtin__print_kwargs.py +0 -12
  547. package/ref-monty/crates/monty/test_cases/builtin__repr.py +0 -3
  548. package/ref-monty/crates/monty/test_cases/builtin__string_funcs.py +0 -73
  549. package/ref-monty/crates/monty/test_cases/bytes__decode_invalid_utf8.py +0 -18
  550. package/ref-monty/crates/monty/test_cases/bytes__endswith_str_error.py +0 -10
  551. package/ref-monty/crates/monty/test_cases/bytes__getitem_index_error.py +0 -10
  552. package/ref-monty/crates/monty/test_cases/bytes__index_start_gt_end.py +0 -10
  553. package/ref-monty/crates/monty/test_cases/bytes__methods.py +0 -394
  554. package/ref-monty/crates/monty/test_cases/bytes__negative_count.py +0 -9
  555. package/ref-monty/crates/monty/test_cases/bytes__ops.py +0 -90
  556. package/ref-monty/crates/monty/test_cases/bytes__startswith_str_error.py +0 -10
  557. package/ref-monty/crates/monty/test_cases/call_object.py +0 -3
  558. package/ref-monty/crates/monty/test_cases/chain_comparison__all.py +0 -79
  559. package/ref-monty/crates/monty/test_cases/closure__param_shadows_outer.py +0 -81
  560. package/ref-monty/crates/monty/test_cases/closure__pep448.py +0 -203
  561. package/ref-monty/crates/monty/test_cases/closure__undefined_nonlocal.py +0 -13
  562. package/ref-monty/crates/monty/test_cases/compare__mixed_types.py +0 -120
  563. package/ref-monty/crates/monty/test_cases/comprehension__all.py +0 -208
  564. package/ref-monty/crates/monty/test_cases/comprehension__scope.py +0 -7
  565. package/ref-monty/crates/monty/test_cases/comprehension__unbound_local.py +0 -14
  566. package/ref-monty/crates/monty/test_cases/dataclass__basic.py +0 -238
  567. package/ref-monty/crates/monty/test_cases/dataclass__call_field_error.py +0 -12
  568. package/ref-monty/crates/monty/test_cases/dataclass__frozen_set_error.py +0 -12
  569. package/ref-monty/crates/monty/test_cases/dataclass__get_missing_attr_error.py +0 -11
  570. package/ref-monty/crates/monty/test_cases/dict__get_unhashable_key.py +0 -3
  571. package/ref-monty/crates/monty/test_cases/dict__literal_unhashable_key.py +0 -2
  572. package/ref-monty/crates/monty/test_cases/dict__method_pop_missing_error.py +0 -3
  573. package/ref-monty/crates/monty/test_cases/dict__methods.py +0 -151
  574. package/ref-monty/crates/monty/test_cases/dict__ops.py +0 -133
  575. package/ref-monty/crates/monty/test_cases/dict__pop_unhashable_key.py +0 -4
  576. package/ref-monty/crates/monty/test_cases/dict__popitem_empty.py +0 -9
  577. package/ref-monty/crates/monty/test_cases/dict__subscript_missing_key.py +0 -3
  578. package/ref-monty/crates/monty/test_cases/dict__unhashable_dict_key.py +0 -2
  579. package/ref-monty/crates/monty/test_cases/dict__unhashable_list_key.py +0 -2
  580. package/ref-monty/crates/monty/test_cases/dict__unpack_type_error.py +0 -2
  581. package/ref-monty/crates/monty/test_cases/dict__views.py +0 -165
  582. package/ref-monty/crates/monty/test_cases/edge__all.py +0 -26
  583. package/ref-monty/crates/monty/test_cases/edge__float_int_mod.py +0 -2
  584. package/ref-monty/crates/monty/test_cases/edge__int_float_mod.py +0 -2
  585. package/ref-monty/crates/monty/test_cases/exc__args.py +0 -16
  586. package/ref-monty/crates/monty/test_cases/exc__str.py +0 -15
  587. package/ref-monty/crates/monty/test_cases/execute_ok__all.py +0 -54
  588. package/ref-monty/crates/monty/test_cases/execute_raise__error_instance_str.py +0 -2
  589. package/ref-monty/crates/monty/test_cases/execute_raise__error_no_args.py +0 -2
  590. package/ref-monty/crates/monty/test_cases/execute_raise__error_string_arg.py +0 -2
  591. package/ref-monty/crates/monty/test_cases/execute_raise__error_string_arg_quotes.py +0 -2
  592. package/ref-monty/crates/monty/test_cases/execute_raise__error_type.py +0 -2
  593. package/ref-monty/crates/monty/test_cases/execute_raise__raise_instance_via_var.py +0 -4
  594. package/ref-monty/crates/monty/test_cases/execute_raise__raise_list.py +0 -2
  595. package/ref-monty/crates/monty/test_cases/execute_raise__raise_number.py +0 -2
  596. package/ref-monty/crates/monty/test_cases/execute_raise__raise_type_call_via_var.py +0 -4
  597. package/ref-monty/crates/monty/test_cases/execute_raise__raise_type_direct.py +0 -3
  598. package/ref-monty/crates/monty/test_cases/execute_raise__raise_type_via_var.py +0 -4
  599. package/ref-monty/crates/monty/test_cases/ext_call__arg_side_effect_bug.py +0 -22
  600. package/ref-monty/crates/monty/test_cases/ext_call__augmented.py +0 -17
  601. package/ref-monty/crates/monty/test_cases/ext_call__augmented_refcount_bug.py +0 -7
  602. package/ref-monty/crates/monty/test_cases/ext_call__bare_raise_after_resume.py +0 -34
  603. package/ref-monty/crates/monty/test_cases/ext_call__basic.py +0 -99
  604. package/ref-monty/crates/monty/test_cases/ext_call__boolean.py +0 -37
  605. package/ref-monty/crates/monty/test_cases/ext_call__boolean_side_effect_hang.py +0 -17
  606. package/ref-monty/crates/monty/test_cases/ext_call__closure_bug.py +0 -16
  607. package/ref-monty/crates/monty/test_cases/ext_call__comparison.py +0 -26
  608. package/ref-monty/crates/monty/test_cases/ext_call__deep_call_stack.py +0 -18
  609. package/ref-monty/crates/monty/test_cases/ext_call__elif.py +0 -171
  610. package/ref-monty/crates/monty/test_cases/ext_call__exc.py +0 -4
  611. package/ref-monty/crates/monty/test_cases/ext_call__exc_deep_stack.py +0 -39
  612. package/ref-monty/crates/monty/test_cases/ext_call__exc_in_function.py +0 -17
  613. package/ref-monty/crates/monty/test_cases/ext_call__exc_nested_functions.py +0 -31
  614. package/ref-monty/crates/monty/test_cases/ext_call__ext_exc.py +0 -171
  615. package/ref-monty/crates/monty/test_cases/ext_call__for.py +0 -114
  616. package/ref-monty/crates/monty/test_cases/ext_call__fstring.py +0 -12
  617. package/ref-monty/crates/monty/test_cases/ext_call__if.py +0 -135
  618. package/ref-monty/crates/monty/test_cases/ext_call__if_condition.py +0 -37
  619. package/ref-monty/crates/monty/test_cases/ext_call__in_closure.py +0 -14
  620. package/ref-monty/crates/monty/test_cases/ext_call__in_function.py +0 -40
  621. package/ref-monty/crates/monty/test_cases/ext_call__in_function_simple.py +0 -7
  622. package/ref-monty/crates/monty/test_cases/ext_call__literals.py +0 -17
  623. package/ref-monty/crates/monty/test_cases/ext_call__multi_in_func.py +0 -32
  624. package/ref-monty/crates/monty/test_cases/ext_call__name_lookup.py +0 -69
  625. package/ref-monty/crates/monty/test_cases/ext_call__name_lookup_undefined.py +0 -4
  626. package/ref-monty/crates/monty/test_cases/ext_call__nested_calls.py +0 -14
  627. package/ref-monty/crates/monty/test_cases/ext_call__recursion_bug.py +0 -19
  628. package/ref-monty/crates/monty/test_cases/ext_call__return.py +0 -28
  629. package/ref-monty/crates/monty/test_cases/ext_call__side_effects.py +0 -25
  630. package/ref-monty/crates/monty/test_cases/ext_call__subscript.py +0 -7
  631. package/ref-monty/crates/monty/test_cases/ext_call__ternary.py +0 -28
  632. package/ref-monty/crates/monty/test_cases/ext_call__try.py +0 -280
  633. package/ref-monty/crates/monty/test_cases/ext_call__try_simple.py +0 -10
  634. package/ref-monty/crates/monty/test_cases/ext_call__unary.py +0 -13
  635. package/ref-monty/crates/monty/test_cases/frozenset__ops.py +0 -178
  636. package/ref-monty/crates/monty/test_cases/fstring__all.py +0 -236
  637. package/ref-monty/crates/monty/test_cases/fstring__error_eq_align_on_str.py +0 -3
  638. package/ref-monty/crates/monty/test_cases/fstring__error_float_f_on_str.py +0 -3
  639. package/ref-monty/crates/monty/test_cases/fstring__error_int_d_on_float.py +0 -3
  640. package/ref-monty/crates/monty/test_cases/fstring__error_int_d_on_str.py +0 -3
  641. package/ref-monty/crates/monty/test_cases/fstring__error_invalid_spec.py +0 -4
  642. package/ref-monty/crates/monty/test_cases/fstring__error_invalid_spec_dynamic.py +0 -4
  643. package/ref-monty/crates/monty/test_cases/fstring__error_invalid_spec_str.py +0 -4
  644. package/ref-monty/crates/monty/test_cases/fstring__error_str_s_on_int.py +0 -3
  645. package/ref-monty/crates/monty/test_cases/function__call_duplicate_kwargs.py +0 -6
  646. package/ref-monty/crates/monty/test_cases/function__call_unpack.py +0 -42
  647. package/ref-monty/crates/monty/test_cases/function__defaults.py +0 -117
  648. package/ref-monty/crates/monty/test_cases/function__err_duplicate_arg.py +0 -7
  649. package/ref-monty/crates/monty/test_cases/function__err_duplicate_first_arg.py +0 -7
  650. package/ref-monty/crates/monty/test_cases/function__err_duplicate_kwarg_cleanup.py +0 -9
  651. package/ref-monty/crates/monty/test_cases/function__err_kwonly_as_positional.py +0 -7
  652. package/ref-monty/crates/monty/test_cases/function__err_missing_all_posonly.py +0 -7
  653. package/ref-monty/crates/monty/test_cases/function__err_missing_heap_cleanup.py +0 -9
  654. package/ref-monty/crates/monty/test_cases/function__err_missing_kwonly.py +0 -7
  655. package/ref-monty/crates/monty/test_cases/function__err_missing_posonly_with_kwarg.py +0 -7
  656. package/ref-monty/crates/monty/test_cases/function__err_missing_with_posonly.py +0 -7
  657. package/ref-monty/crates/monty/test_cases/function__err_posonly_as_kwarg.py +0 -7
  658. package/ref-monty/crates/monty/test_cases/function__err_posonly_first_as_kwarg.py +0 -7
  659. package/ref-monty/crates/monty/test_cases/function__err_too_many_posonly.py +0 -7
  660. package/ref-monty/crates/monty/test_cases/function__err_too_many_with_kwonly.py +0 -7
  661. package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg.py +0 -7
  662. package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg_cleanup.py +0 -9
  663. package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg_quote.py +0 -13
  664. package/ref-monty/crates/monty/test_cases/function__err_unexpected_kwarg_simple.py +0 -7
  665. package/ref-monty/crates/monty/test_cases/function__err_unpack_duplicate_arg.py +0 -6
  666. package/ref-monty/crates/monty/test_cases/function__err_unpack_duplicate_heap.py +0 -8
  667. package/ref-monty/crates/monty/test_cases/function__err_unpack_int.py +0 -6
  668. package/ref-monty/crates/monty/test_cases/function__err_unpack_nonstring_key.py +0 -6
  669. package/ref-monty/crates/monty/test_cases/function__err_unpack_not_mapping.py +0 -6
  670. package/ref-monty/crates/monty/test_cases/function__kwargs_unpacking.py +0 -173
  671. package/ref-monty/crates/monty/test_cases/function__ops.py +0 -294
  672. package/ref-monty/crates/monty/test_cases/function__return_none.py +0 -42
  673. package/ref-monty/crates/monty/test_cases/function__signatures.py +0 -47
  674. package/ref-monty/crates/monty/test_cases/function__too_few_args_all.py +0 -6
  675. package/ref-monty/crates/monty/test_cases/function__too_few_args_one.py +0 -6
  676. package/ref-monty/crates/monty/test_cases/function__too_few_args_two.py +0 -6
  677. package/ref-monty/crates/monty/test_cases/function__too_many_args_one.py +0 -6
  678. package/ref-monty/crates/monty/test_cases/function__too_many_args_two.py +0 -6
  679. package/ref-monty/crates/monty/test_cases/function__too_many_args_zero.py +0 -6
  680. package/ref-monty/crates/monty/test_cases/global__error_assigned_before.py +0 -7
  681. package/ref-monty/crates/monty/test_cases/global__ops.py +0 -163
  682. package/ref-monty/crates/monty/test_cases/hash__dict_unhashable.py +0 -2
  683. package/ref-monty/crates/monty/test_cases/hash__list_unhashable.py +0 -2
  684. package/ref-monty/crates/monty/test_cases/hash__ops.py +0 -153
  685. package/ref-monty/crates/monty/test_cases/id__bytes_literals_distinct.py +0 -3
  686. package/ref-monty/crates/monty/test_cases/id__int_copy_distinct.py +0 -5
  687. package/ref-monty/crates/monty/test_cases/id__is_number_is_number.py +0 -3
  688. package/ref-monty/crates/monty/test_cases/id__non_overlapping_lifetimes_distinct_types.py +0 -10
  689. package/ref-monty/crates/monty/test_cases/id__non_overlapping_lifetimes_same_types.py +0 -6
  690. package/ref-monty/crates/monty/test_cases/id__ops.py +0 -97
  691. package/ref-monty/crates/monty/test_cases/id__str_literals_same.py +0 -3
  692. package/ref-monty/crates/monty/test_cases/if__elif_else.py +0 -207
  693. package/ref-monty/crates/monty/test_cases/if__raise_elif.py +0 -11
  694. package/ref-monty/crates/monty/test_cases/if__raise_else.py +0 -13
  695. package/ref-monty/crates/monty/test_cases/if__raise_if.py +0 -9
  696. package/ref-monty/crates/monty/test_cases/if__raise_in_elif_condition.py +0 -18
  697. package/ref-monty/crates/monty/test_cases/if__raise_in_if_condition.py +0 -16
  698. package/ref-monty/crates/monty/test_cases/if_else_expr__all.py +0 -55
  699. package/ref-monty/crates/monty/test_cases/import__error_cannot_import.py +0 -9
  700. package/ref-monty/crates/monty/test_cases/import__error_module_not_found.py +0 -9
  701. package/ref-monty/crates/monty/test_cases/import__local_scope.py +0 -68
  702. package/ref-monty/crates/monty/test_cases/import__os.py +0 -25
  703. package/ref-monty/crates/monty/test_cases/import__relative_error.py +0 -9
  704. package/ref-monty/crates/monty/test_cases/import__relative_no_module_error.py +0 -9
  705. package/ref-monty/crates/monty/test_cases/import__runtime_error_when_executed.py +0 -14
  706. package/ref-monty/crates/monty/test_cases/import__star_error.py +0 -11
  707. package/ref-monty/crates/monty/test_cases/import__sys.py +0 -47
  708. package/ref-monty/crates/monty/test_cases/import__sys_monty.py +0 -28
  709. package/ref-monty/crates/monty/test_cases/import__type_checking_guard.py +0 -37
  710. package/ref-monty/crates/monty/test_cases/import__typing.py +0 -25
  711. package/ref-monty/crates/monty/test_cases/import__typing_type_ignore.py +0 -4
  712. package/ref-monty/crates/monty/test_cases/int__bigint.py +0 -467
  713. package/ref-monty/crates/monty/test_cases/int__bigint_errors.py +0 -260
  714. package/ref-monty/crates/monty/test_cases/int__ops.py +0 -219
  715. package/ref-monty/crates/monty/test_cases/int__overflow_division.py +0 -84
  716. package/ref-monty/crates/monty/test_cases/is_variant__all.py +0 -36
  717. package/ref-monty/crates/monty/test_cases/isinstance__arg2_list_error.py +0 -2
  718. package/ref-monty/crates/monty/test_cases/isinstance__arg2_type_error.py +0 -2
  719. package/ref-monty/crates/monty/test_cases/iter__dict_mutation.py +0 -4
  720. package/ref-monty/crates/monty/test_cases/iter__for.py +0 -243
  721. package/ref-monty/crates/monty/test_cases/iter__for_loop_unpacking.py +0 -66
  722. package/ref-monty/crates/monty/test_cases/iter__generator_expr.py +0 -20
  723. package/ref-monty/crates/monty/test_cases/iter__generator_expr_type.py +0 -7
  724. package/ref-monty/crates/monty/test_cases/iter__not_iterable.py +0 -3
  725. package/ref-monty/crates/monty/test_cases/lambda__all.py +0 -145
  726. package/ref-monty/crates/monty/test_cases/list__extend_not_iterable.py +0 -7
  727. package/ref-monty/crates/monty/test_cases/list__getitem_out_of_bounds.py +0 -3
  728. package/ref-monty/crates/monty/test_cases/list__index_not_found.py +0 -9
  729. package/ref-monty/crates/monty/test_cases/list__index_start_gt_end.py +0 -10
  730. package/ref-monty/crates/monty/test_cases/list__ops.py +0 -473
  731. package/ref-monty/crates/monty/test_cases/list__pop_empty.py +0 -9
  732. package/ref-monty/crates/monty/test_cases/list__pop_out_of_range.py +0 -9
  733. package/ref-monty/crates/monty/test_cases/list__pop_type_error.py +0 -9
  734. package/ref-monty/crates/monty/test_cases/list__remove_not_found.py +0 -9
  735. package/ref-monty/crates/monty/test_cases/list__setitem_dict_index.py +0 -13
  736. package/ref-monty/crates/monty/test_cases/list__setitem_huge_int_index.py +0 -13
  737. package/ref-monty/crates/monty/test_cases/list__setitem_index_error.py +0 -10
  738. package/ref-monty/crates/monty/test_cases/list__setitem_type_error.py +0 -10
  739. package/ref-monty/crates/monty/test_cases/list__unpack_type_error.py +0 -2
  740. package/ref-monty/crates/monty/test_cases/longint__index_error.py +0 -3
  741. package/ref-monty/crates/monty/test_cases/longint__repeat_error.py +0 -3
  742. package/ref-monty/crates/monty/test_cases/loop__break_continue.py +0 -113
  743. package/ref-monty/crates/monty/test_cases/loop__break_finally.py +0 -69
  744. package/ref-monty/crates/monty/test_cases/loop__break_in_function_error.py +0 -13
  745. package/ref-monty/crates/monty/test_cases/loop__break_in_if_error.py +0 -11
  746. package/ref-monty/crates/monty/test_cases/loop__break_nested_except_clears.py +0 -55
  747. package/ref-monty/crates/monty/test_cases/loop__break_outside_error.py +0 -9
  748. package/ref-monty/crates/monty/test_cases/loop__continue_finally.py +0 -81
  749. package/ref-monty/crates/monty/test_cases/loop__continue_in_function_error.py +0 -13
  750. package/ref-monty/crates/monty/test_cases/loop__continue_in_if_error.py +0 -11
  751. package/ref-monty/crates/monty/test_cases/loop__continue_nested_except_clears.py +0 -60
  752. package/ref-monty/crates/monty/test_cases/loop__continue_outside_error.py +0 -9
  753. package/ref-monty/crates/monty/test_cases/math__acos_domain_error.py +0 -11
  754. package/ref-monty/crates/monty/test_cases/math__acosh_domain_error.py +0 -11
  755. package/ref-monty/crates/monty/test_cases/math__asin_domain_error.py +0 -11
  756. package/ref-monty/crates/monty/test_cases/math__atanh_domain_error.py +0 -11
  757. package/ref-monty/crates/monty/test_cases/math__cos_inf_error.py +0 -11
  758. package/ref-monty/crates/monty/test_cases/math__cosh_overflow_error.py +0 -11
  759. package/ref-monty/crates/monty/test_cases/math__exp_overflow_error.py +0 -11
  760. package/ref-monty/crates/monty/test_cases/math__factorial_float_error.py +0 -11
  761. package/ref-monty/crates/monty/test_cases/math__factorial_negative_error.py +0 -11
  762. package/ref-monty/crates/monty/test_cases/math__floor_inf_error.py +0 -11
  763. package/ref-monty/crates/monty/test_cases/math__floor_nan_error.py +0 -11
  764. package/ref-monty/crates/monty/test_cases/math__floor_str_error.py +0 -11
  765. package/ref-monty/crates/monty/test_cases/math__fmod_inf_error.py +0 -11
  766. package/ref-monty/crates/monty/test_cases/math__gamma_neg_int_error.py +0 -11
  767. package/ref-monty/crates/monty/test_cases/math__gcd_float_error.py +0 -11
  768. package/ref-monty/crates/monty/test_cases/math__isqrt_negative_error.py +0 -11
  769. package/ref-monty/crates/monty/test_cases/math__ldexp_overflow_error.py +0 -11
  770. package/ref-monty/crates/monty/test_cases/math__log1p_domain_error.py +0 -11
  771. package/ref-monty/crates/monty/test_cases/math__log_base1_error.py +0 -11
  772. package/ref-monty/crates/monty/test_cases/math__log_zero_error.py +0 -11
  773. package/ref-monty/crates/monty/test_cases/math__module.py +0 -1432
  774. package/ref-monty/crates/monty/test_cases/math__pow_domain_error.py +0 -11
  775. package/ref-monty/crates/monty/test_cases/math__sin_inf_error.py +0 -11
  776. package/ref-monty/crates/monty/test_cases/math__sqrt_negative_error.py +0 -11
  777. package/ref-monty/crates/monty/test_cases/math__tan_inf_error.py +0 -11
  778. package/ref-monty/crates/monty/test_cases/math__trunc_str_error.py +0 -11
  779. package/ref-monty/crates/monty/test_cases/method__args_kwargs_unpacking.py +0 -259
  780. package/ref-monty/crates/monty/test_cases/name_error__unbound_local_func.py +0 -19
  781. package/ref-monty/crates/monty/test_cases/name_error__unbound_local_module.py +0 -12
  782. package/ref-monty/crates/monty/test_cases/name_error__undefined_call_chained.py +0 -9
  783. package/ref-monty/crates/monty/test_cases/name_error__undefined_call_in_expr.py +0 -9
  784. package/ref-monty/crates/monty/test_cases/name_error__undefined_call_in_function.py +0 -16
  785. package/ref-monty/crates/monty/test_cases/name_error__undefined_call_with_args.py +0 -9
  786. package/ref-monty/crates/monty/test_cases/name_error__undefined_global.py +0 -10
  787. package/ref-monty/crates/monty/test_cases/namedtuple__missing_attr.py +0 -11
  788. package/ref-monty/crates/monty/test_cases/namedtuple__ops.py +0 -34
  789. package/ref-monty/crates/monty/test_cases/nonlocal__error_module_level.py +0 -3
  790. package/ref-monty/crates/monty/test_cases/nonlocal__ops.py +0 -353
  791. package/ref-monty/crates/monty/test_cases/os__environ.py +0 -40
  792. package/ref-monty/crates/monty/test_cases/os__getenv_key_list_error.py +0 -5
  793. package/ref-monty/crates/monty/test_cases/os__getenv_key_type_error.py +0 -5
  794. package/ref-monty/crates/monty/test_cases/parse_error__complex.py +0 -3
  795. package/ref-monty/crates/monty/test_cases/pathlib__import.py +0 -11
  796. package/ref-monty/crates/monty/test_cases/pathlib__os.py +0 -136
  797. package/ref-monty/crates/monty/test_cases/pathlib__os_read_error.py +0 -12
  798. package/ref-monty/crates/monty/test_cases/pathlib__pure.py +0 -81
  799. package/ref-monty/crates/monty/test_cases/pyobject__cycle_dict_self.py +0 -5
  800. package/ref-monty/crates/monty/test_cases/pyobject__cycle_list_dict.py +0 -6
  801. package/ref-monty/crates/monty/test_cases/pyobject__cycle_list_self.py +0 -5
  802. package/ref-monty/crates/monty/test_cases/pyobject__cycle_multiple_refs.py +0 -6
  803. package/ref-monty/crates/monty/test_cases/range__error_no_args.py +0 -2
  804. package/ref-monty/crates/monty/test_cases/range__error_step_zero.py +0 -2
  805. package/ref-monty/crates/monty/test_cases/range__error_too_many_args.py +0 -2
  806. package/ref-monty/crates/monty/test_cases/range__getitem_index_error.py +0 -10
  807. package/ref-monty/crates/monty/test_cases/range__ops.py +0 -236
  808. package/ref-monty/crates/monty/test_cases/re__basic.py +0 -756
  809. package/ref-monty/crates/monty/test_cases/re__grouping.py +0 -241
  810. package/ref-monty/crates/monty/test_cases/re__match.py +0 -148
  811. package/ref-monty/crates/monty/test_cases/recursion__deep_drop.py +0 -26
  812. package/ref-monty/crates/monty/test_cases/recursion__deep_eq.py +0 -23
  813. package/ref-monty/crates/monty/test_cases/recursion__deep_hash.py +0 -46
  814. package/ref-monty/crates/monty/test_cases/recursion__deep_repr.py +0 -12
  815. package/ref-monty/crates/monty/test_cases/recursion__function_depth.py +0 -13
  816. package/ref-monty/crates/monty/test_cases/refcount__cycle_mutual_reference.py +0 -18
  817. package/ref-monty/crates/monty/test_cases/refcount__cycle_self_reference.py +0 -12
  818. package/ref-monty/crates/monty/test_cases/refcount__dict_basic.py +0 -5
  819. package/ref-monty/crates/monty/test_cases/refcount__dict_get.py +0 -5
  820. package/ref-monty/crates/monty/test_cases/refcount__dict_keys_and.py +0 -14
  821. package/ref-monty/crates/monty/test_cases/refcount__dict_overwrite.py +0 -6
  822. package/ref-monty/crates/monty/test_cases/refcount__gather_cleanup.py +0 -16
  823. package/ref-monty/crates/monty/test_cases/refcount__gather_exception.py +0 -18
  824. package/ref-monty/crates/monty/test_cases/refcount__gather_nested_cancel.py +0 -25
  825. package/ref-monty/crates/monty/test_cases/refcount__immediate_skipped.py +0 -4
  826. package/ref-monty/crates/monty/test_cases/refcount__kwargs_unpacking.py +0 -27
  827. package/ref-monty/crates/monty/test_cases/refcount__list_append_multiple.py +0 -6
  828. package/ref-monty/crates/monty/test_cases/refcount__list_append_ref.py +0 -5
  829. package/ref-monty/crates/monty/test_cases/refcount__list_concat.py +0 -5
  830. package/ref-monty/crates/monty/test_cases/refcount__list_getitem.py +0 -5
  831. package/ref-monty/crates/monty/test_cases/refcount__list_iadd.py +0 -5
  832. package/ref-monty/crates/monty/test_cases/refcount__nested_list.py +0 -4
  833. package/ref-monty/crates/monty/test_cases/refcount__re_pattern_sub_error_paths.py +0 -37
  834. package/ref-monty/crates/monty/test_cases/refcount__re_search_match.py +0 -34
  835. package/ref-monty/crates/monty/test_cases/refcount__re_sub_error_paths.py +0 -31
  836. package/ref-monty/crates/monty/test_cases/refcount__shared_reference.py +0 -4
  837. package/ref-monty/crates/monty/test_cases/refcount__single_list.py +0 -3
  838. package/ref-monty/crates/monty/test_cases/repr__cycle_detection.py +0 -24
  839. package/ref-monty/crates/monty/test_cases/set__ops.py +0 -191
  840. package/ref-monty/crates/monty/test_cases/set__review_bugs.py +0 -35
  841. package/ref-monty/crates/monty/test_cases/set__unpack_type_error.py +0 -2
  842. package/ref-monty/crates/monty/test_cases/slice__invalid_indices.py +0 -2
  843. package/ref-monty/crates/monty/test_cases/slice__kwargs.py +0 -9
  844. package/ref-monty/crates/monty/test_cases/slice__no_args.py +0 -9
  845. package/ref-monty/crates/monty/test_cases/slice__ops.py +0 -149
  846. package/ref-monty/crates/monty/test_cases/slice__step_zero.py +0 -9
  847. package/ref-monty/crates/monty/test_cases/slice__step_zero_bytes.py +0 -9
  848. package/ref-monty/crates/monty/test_cases/slice__step_zero_range.py +0 -9
  849. package/ref-monty/crates/monty/test_cases/slice__step_zero_str.py +0 -9
  850. package/ref-monty/crates/monty/test_cases/slice__step_zero_tuple.py +0 -9
  851. package/ref-monty/crates/monty/test_cases/slice__too_many_args.py +0 -9
  852. package/ref-monty/crates/monty/test_cases/str__getitem_index_error.py +0 -10
  853. package/ref-monty/crates/monty/test_cases/str__index_not_found.py +0 -9
  854. package/ref-monty/crates/monty/test_cases/str__join_no_args.py +0 -9
  855. package/ref-monty/crates/monty/test_cases/str__join_non_string.py +0 -9
  856. package/ref-monty/crates/monty/test_cases/str__join_not_iterable.py +0 -9
  857. package/ref-monty/crates/monty/test_cases/str__join_too_many_args.py +0 -9
  858. package/ref-monty/crates/monty/test_cases/str__methods.py +0 -327
  859. package/ref-monty/crates/monty/test_cases/str__ops.py +0 -162
  860. package/ref-monty/crates/monty/test_cases/str__partition_empty.py +0 -9
  861. package/ref-monty/crates/monty/test_cases/str__rsplit_empty_sep.py +0 -9
  862. package/ref-monty/crates/monty/test_cases/str__split_empty_sep.py +0 -9
  863. package/ref-monty/crates/monty/test_cases/sys__types.py +0 -7
  864. package/ref-monty/crates/monty/test_cases/traceback__division_error.py +0 -30
  865. package/ref-monty/crates/monty/test_cases/traceback__index_error.py +0 -17
  866. package/ref-monty/crates/monty/test_cases/traceback__insert_as_int.py +0 -10
  867. package/ref-monty/crates/monty/test_cases/traceback__nested_call.py +0 -29
  868. package/ref-monty/crates/monty/test_cases/traceback__nonlocal_module_scope.py +0 -10
  869. package/ref-monty/crates/monty/test_cases/traceback__nonlocal_unbound.py +0 -24
  870. package/ref-monty/crates/monty/test_cases/traceback__range_as_int.py +0 -9
  871. package/ref-monty/crates/monty/test_cases/traceback__recursion_error.py +0 -23
  872. package/ref-monty/crates/monty/test_cases/traceback__set_mutation.py +0 -11
  873. package/ref-monty/crates/monty/test_cases/traceback__undefined_attr_call.py +0 -16
  874. package/ref-monty/crates/monty/test_cases/traceback__undefined_call.py +0 -16
  875. package/ref-monty/crates/monty/test_cases/traceback__undefined_raise.py +0 -16
  876. package/ref-monty/crates/monty/test_cases/try_except__all.py +0 -472
  877. package/ref-monty/crates/monty/test_cases/try_except__bare_raise_no_context.py +0 -2
  878. package/ref-monty/crates/monty/test_cases/try_except__invalid_type.py +0 -5
  879. package/ref-monty/crates/monty/test_cases/tuple__getitem_out_of_bounds.py +0 -3
  880. package/ref-monty/crates/monty/test_cases/tuple__index_not_found.py +0 -9
  881. package/ref-monty/crates/monty/test_cases/tuple__index_start_gt_end.py +0 -10
  882. package/ref-monty/crates/monty/test_cases/tuple__methods.py +0 -19
  883. package/ref-monty/crates/monty/test_cases/tuple__ops.py +0 -133
  884. package/ref-monty/crates/monty/test_cases/tuple__unpack_type_error.py +0 -2
  885. package/ref-monty/crates/monty/test_cases/type__builtin_attr_error.py +0 -9
  886. package/ref-monty/crates/monty/test_cases/type__bytes_negative.py +0 -2
  887. package/ref-monty/crates/monty/test_cases/type__cell_not_builtin.py +0 -9
  888. package/ref-monty/crates/monty/test_cases/type__exception_attr_error.py +0 -11
  889. package/ref-monty/crates/monty/test_cases/type__float_conversion_error.py +0 -2
  890. package/ref-monty/crates/monty/test_cases/type__float_repr_both_quotes.py +0 -9
  891. package/ref-monty/crates/monty/test_cases/type__float_repr_newline.py +0 -9
  892. package/ref-monty/crates/monty/test_cases/type__float_repr_single_quote.py +0 -9
  893. package/ref-monty/crates/monty/test_cases/type__int_conversion_error.py +0 -2
  894. package/ref-monty/crates/monty/test_cases/type__list_not_iterable.py +0 -2
  895. package/ref-monty/crates/monty/test_cases/type__non_builtin_name_error.py +0 -9
  896. package/ref-monty/crates/monty/test_cases/type__ops.py +0 -200
  897. package/ref-monty/crates/monty/test_cases/type__shadow_exc.py +0 -3
  898. package/ref-monty/crates/monty/test_cases/type__shadow_int.py +0 -9
  899. package/ref-monty/crates/monty/test_cases/type__shadow_len.py +0 -3
  900. package/ref-monty/crates/monty/test_cases/type__tuple_not_iterable.py +0 -2
  901. package/ref-monty/crates/monty/test_cases/type_error__int_add_list.py +0 -2
  902. package/ref-monty/crates/monty/test_cases/type_error__int_div_str.py +0 -2
  903. package/ref-monty/crates/monty/test_cases/type_error__int_floordiv_str.py +0 -2
  904. package/ref-monty/crates/monty/test_cases/type_error__int_iadd_str.py +0 -3
  905. package/ref-monty/crates/monty/test_cases/type_error__int_mod_str.py +0 -2
  906. package/ref-monty/crates/monty/test_cases/type_error__int_pow_str.py +0 -2
  907. package/ref-monty/crates/monty/test_cases/type_error__int_sub_str.py +0 -2
  908. package/ref-monty/crates/monty/test_cases/type_error__list_add_int.py +0 -2
  909. package/ref-monty/crates/monty/test_cases/type_error__list_add_str.py +0 -2
  910. package/ref-monty/crates/monty/test_cases/type_error__list_iadd_int.py +0 -6
  911. package/ref-monty/crates/monty/test_cases/type_error__str_add_int.py +0 -2
  912. package/ref-monty/crates/monty/test_cases/type_error__str_iadd_int.py +0 -3
  913. package/ref-monty/crates/monty/test_cases/type_error__unary_invert_str.py +0 -3
  914. package/ref-monty/crates/monty/test_cases/type_error__unary_minus_str.py +0 -4
  915. package/ref-monty/crates/monty/test_cases/type_error__unary_neg_str.py +0 -3
  916. package/ref-monty/crates/monty/test_cases/type_error__unary_plus_str.py +0 -4
  917. package/ref-monty/crates/monty/test_cases/typing__types.py +0 -24
  918. package/ref-monty/crates/monty/test_cases/unpack__nested.py +0 -48
  919. package/ref-monty/crates/monty/test_cases/unpack__non_sequence.py +0 -9
  920. package/ref-monty/crates/monty/test_cases/unpack__not_enough.py +0 -9
  921. package/ref-monty/crates/monty/test_cases/unpack__ops.py +0 -153
  922. package/ref-monty/crates/monty/test_cases/unpack__star_not_enough.py +0 -9
  923. package/ref-monty/crates/monty/test_cases/unpack__too_many.py +0 -9
  924. package/ref-monty/crates/monty/test_cases/version__cpython.py +0 -4
  925. package/ref-monty/crates/monty/test_cases/walrus__all.py +0 -178
  926. package/ref-monty/crates/monty/test_cases/while__all.py +0 -206
  927. package/ref-monty/crates/monty/tests/asyncio.rs +0 -764
  928. package/ref-monty/crates/monty/tests/binary_serde.rs +0 -185
  929. package/ref-monty/crates/monty/tests/bytecode_limits.rs +0 -248
  930. package/ref-monty/crates/monty/tests/datatest_runner.rs +0 -2029
  931. package/ref-monty/crates/monty/tests/inputs.rs +0 -420
  932. package/ref-monty/crates/monty/tests/json_serde.rs +0 -250
  933. package/ref-monty/crates/monty/tests/main.rs +0 -71
  934. package/ref-monty/crates/monty/tests/math_module.rs +0 -114
  935. package/ref-monty/crates/monty/tests/name_lookup.rs +0 -482
  936. package/ref-monty/crates/monty/tests/os_tests.rs +0 -459
  937. package/ref-monty/crates/monty/tests/parse_errors.rs +0 -441
  938. package/ref-monty/crates/monty/tests/print_writer.rs +0 -238
  939. package/ref-monty/crates/monty/tests/py_object.rs +0 -121
  940. package/ref-monty/crates/monty/tests/regex.rs +0 -90
  941. package/ref-monty/crates/monty/tests/repl.rs +0 -344
  942. package/ref-monty/crates/monty/tests/resource_limits.rs +0 -1826
  943. package/ref-monty/crates/monty/tests/try_from.rs +0 -167
  944. package/ref-monty/crates/monty-cli/Cargo.toml +0 -25
  945. package/ref-monty/crates/monty-cli/src/main.rs +0 -541
  946. package/ref-monty/crates/monty-js/.cargo/config.toml +0 -2
  947. package/ref-monty/crates/monty-js/.prettierignore +0 -8
  948. package/ref-monty/crates/monty-js/Cargo.toml +0 -32
  949. package/ref-monty/crates/monty-js/README.md +0 -207
  950. package/ref-monty/crates/monty-js/__test__/async.spec.ts +0 -350
  951. package/ref-monty/crates/monty-js/__test__/basic.spec.ts +0 -114
  952. package/ref-monty/crates/monty-js/__test__/exceptions.spec.ts +0 -427
  953. package/ref-monty/crates/monty-js/__test__/external.spec.ts +0 -354
  954. package/ref-monty/crates/monty-js/__test__/inputs.spec.ts +0 -143
  955. package/ref-monty/crates/monty-js/__test__/limits.spec.ts +0 -162
  956. package/ref-monty/crates/monty-js/__test__/package.json +0 -3
  957. package/ref-monty/crates/monty-js/__test__/print.spec.ts +0 -229
  958. package/ref-monty/crates/monty-js/__test__/repl.spec.ts +0 -34
  959. package/ref-monty/crates/monty-js/__test__/serialize.spec.ts +0 -205
  960. package/ref-monty/crates/monty-js/__test__/start.spec.ts +0 -443
  961. package/ref-monty/crates/monty-js/__test__/type_check.spec.ts +0 -147
  962. package/ref-monty/crates/monty-js/__test__/types.spec.ts +0 -319
  963. package/ref-monty/crates/monty-js/build.rs +0 -61
  964. package/ref-monty/crates/monty-js/index-header.d.ts +0 -3
  965. package/ref-monty/crates/monty-js/package-lock.json +0 -4694
  966. package/ref-monty/crates/monty-js/package.json +0 -100
  967. package/ref-monty/crates/monty-js/scripts/smoke-test.sh +0 -69
  968. package/ref-monty/crates/monty-js/smoke-test/package.json +0 -17
  969. package/ref-monty/crates/monty-js/smoke-test/test.ts +0 -171
  970. package/ref-monty/crates/monty-js/smoke-test/tsconfig.json +0 -11
  971. package/ref-monty/crates/monty-js/src/convert.rs +0 -648
  972. package/ref-monty/crates/monty-js/src/exceptions.rs +0 -293
  973. package/ref-monty/crates/monty-js/src/lib.rs +0 -41
  974. package/ref-monty/crates/monty-js/src/limits.rs +0 -53
  975. package/ref-monty/crates/monty-js/src/monty_cls.rs +0 -1407
  976. package/ref-monty/crates/monty-js/tsconfig.json +0 -17
  977. package/ref-monty/crates/monty-js/wrapper.ts +0 -701
  978. package/ref-monty/crates/monty-python/Cargo.toml +0 -38
  979. package/ref-monty/crates/monty-python/README.md +0 -134
  980. package/ref-monty/crates/monty-python/build.rs +0 -4
  981. package/ref-monty/crates/monty-python/example.py +0 -40
  982. package/ref-monty/crates/monty-python/exercise.py +0 -46
  983. package/ref-monty/crates/monty-python/pyproject.toml +0 -57
  984. package/ref-monty/crates/monty-python/python/pydantic_monty/__init__.py +0 -281
  985. package/ref-monty/crates/monty-python/python/pydantic_monty/_monty.pyi +0 -677
  986. package/ref-monty/crates/monty-python/python/pydantic_monty/os_access.py +0 -933
  987. package/ref-monty/crates/monty-python/python/pydantic_monty/py.typed +0 -0
  988. package/ref-monty/crates/monty-python/src/convert.rs +0 -273
  989. package/ref-monty/crates/monty-python/src/dataclass.rs +0 -461
  990. package/ref-monty/crates/monty-python/src/exceptions.rs +0 -557
  991. package/ref-monty/crates/monty-python/src/external.rs +0 -165
  992. package/ref-monty/crates/monty-python/src/lib.rs +0 -77
  993. package/ref-monty/crates/monty-python/src/limits.rs +0 -142
  994. package/ref-monty/crates/monty-python/src/monty_cls.rs +0 -1650
  995. package/ref-monty/crates/monty-python/src/repl.rs +0 -470
  996. package/ref-monty/crates/monty-python/src/serialization.rs +0 -761
  997. package/ref-monty/crates/monty-python/tests/test_async.py +0 -1201
  998. package/ref-monty/crates/monty-python/tests/test_basic.py +0 -66
  999. package/ref-monty/crates/monty-python/tests/test_dataclasses.py +0 -971
  1000. package/ref-monty/crates/monty-python/tests/test_exceptions.py +0 -361
  1001. package/ref-monty/crates/monty-python/tests/test_external.py +0 -367
  1002. package/ref-monty/crates/monty-python/tests/test_inputs.py +0 -126
  1003. package/ref-monty/crates/monty-python/tests/test_limits.py +0 -257
  1004. package/ref-monty/crates/monty-python/tests/test_os_access.py +0 -1286
  1005. package/ref-monty/crates/monty-python/tests/test_os_access_compat.py +0 -731
  1006. package/ref-monty/crates/monty-python/tests/test_os_access_raw.py +0 -483
  1007. package/ref-monty/crates/monty-python/tests/test_os_calls.py +0 -819
  1008. package/ref-monty/crates/monty-python/tests/test_print.py +0 -208
  1009. package/ref-monty/crates/monty-python/tests/test_re.py +0 -170
  1010. package/ref-monty/crates/monty-python/tests/test_readme_examples.py +0 -20
  1011. package/ref-monty/crates/monty-python/tests/test_repl.py +0 -749
  1012. package/ref-monty/crates/monty-python/tests/test_serialize.py +0 -284
  1013. package/ref-monty/crates/monty-python/tests/test_start.py +0 -346
  1014. package/ref-monty/crates/monty-python/tests/test_threading.py +0 -163
  1015. package/ref-monty/crates/monty-python/tests/test_type_check.py +0 -344
  1016. package/ref-monty/crates/monty-python/tests/test_types.py +0 -553
  1017. package/ref-monty/crates/monty-type-checking/Cargo.toml +0 -32
  1018. package/ref-monty/crates/monty-type-checking/src/db.rs +0 -116
  1019. package/ref-monty/crates/monty-type-checking/src/lib.rs +0 -4
  1020. package/ref-monty/crates/monty-type-checking/src/type_check.rs +0 -280
  1021. package/ref-monty/crates/monty-type-checking/tests/bad_types.py +0 -109
  1022. package/ref-monty/crates/monty-type-checking/tests/bad_types_output.txt +0 -21
  1023. package/ref-monty/crates/monty-type-checking/tests/good_types.py +0 -475
  1024. package/ref-monty/crates/monty-type-checking/tests/main.rs +0 -205
  1025. package/ref-monty/crates/monty-type-checking/tests/reveal_types.py +0 -56
  1026. package/ref-monty/crates/monty-type-checking/tests/reveal_types_output.txt +0 -41
  1027. package/ref-monty/crates/monty-typeshed/Cargo.toml +0 -29
  1028. package/ref-monty/crates/monty-typeshed/README.md +0 -11
  1029. package/ref-monty/crates/monty-typeshed/build.rs +0 -101
  1030. package/ref-monty/crates/monty-typeshed/custom/README.md +0 -1
  1031. package/ref-monty/crates/monty-typeshed/custom/asyncio.pyi +0 -138
  1032. package/ref-monty/crates/monty-typeshed/custom/os.pyi +0 -87
  1033. package/ref-monty/crates/monty-typeshed/custom/sys.pyi +0 -33
  1034. package/ref-monty/crates/monty-typeshed/src/lib.rs +0 -56
  1035. package/ref-monty/crates/monty-typeshed/update.py +0 -321
  1036. package/ref-monty/crates/monty-typeshed/vendor/typeshed/source_commit.txt +0 -1
  1037. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/VERSIONS +0 -20
  1038. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/_collections_abc.pyi +0 -105
  1039. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/_typeshed/__init__.pyi +0 -394
  1040. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/asyncio.pyi +0 -138
  1041. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/builtins.pyi +0 -1434
  1042. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/collections/__init__.pyi +0 -527
  1043. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/collections/abc.pyi +0 -2
  1044. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/dataclasses.pyi +0 -502
  1045. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/enum.pyi +0 -376
  1046. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/math.pyi +0 -149
  1047. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/os.pyi +0 -87
  1048. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/pathlib/__init__.pyi +0 -395
  1049. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/pathlib/types.pyi +0 -8
  1050. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/re.pyi +0 -337
  1051. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/sys.pyi +0 -33
  1052. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/types.pyi +0 -741
  1053. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/typing.pyi +0 -1217
  1054. package/ref-monty/crates/monty-typeshed/vendor/typeshed/stdlib/typing_extensions.pyi +0 -716
  1055. package/ref-monty/docs/usage-guide.md +0 -117
  1056. package/ref-monty/examples/README.md +0 -3
  1057. package/ref-monty/examples/expense_analysis/README.md +0 -3
  1058. package/ref-monty/examples/expense_analysis/data.py +0 -124
  1059. package/ref-monty/examples/expense_analysis/main.py +0 -115
  1060. package/ref-monty/examples/sql_playground/README.md +0 -20
  1061. package/ref-monty/examples/sql_playground/external_functions.py +0 -129
  1062. package/ref-monty/examples/sql_playground/main.py +0 -81
  1063. package/ref-monty/examples/sql_playground/sandbox_code.py +0 -82
  1064. package/ref-monty/examples/sql_playground/type_stubs.pyi +0 -14
  1065. package/ref-monty/examples/web_scraper/README.md +0 -15
  1066. package/ref-monty/examples/web_scraper/browser.py +0 -56
  1067. package/ref-monty/examples/web_scraper/example_code.py +0 -59
  1068. package/ref-monty/examples/web_scraper/external_functions.py +0 -324
  1069. package/ref-monty/examples/web_scraper/main.py +0 -193
  1070. package/ref-monty/examples/web_scraper/sub_agent.py +0 -79
  1071. package/ref-monty/monty-npm.md +0 -235
  1072. package/ref-monty/pyproject.toml +0 -162
  1073. package/ref-monty/scripts/check_imports.py +0 -91
  1074. package/ref-monty/scripts/codecov_diff.py +0 -412
  1075. package/ref-monty/scripts/complete_tests.py +0 -146
  1076. package/ref-monty/scripts/flamegraph_to_text.py +0 -208
  1077. package/ref-monty/scripts/iter_test_methods.py +0 -540
  1078. package/ref-monty/scripts/run_traceback.py +0 -180
  1079. package/ref-monty/scripts/startup_performance.py +0 -130
  1080. package/ref-monty/uv.lock +0 -1779
  1081. /package/docs/{plugin-examples.md → plugins-examples.md} +0 -0
@@ -1,1730 +0,0 @@
1
- use std::{borrow::Cow, fmt};
2
-
3
- use num_bigint::BigInt;
4
- use ruff_python_ast::{
5
- self as ast, BoolOp, CmpOp, ConversionFlag as RuffConversionFlag, ElifElseClause, Expr as AstExpr,
6
- InterpolatedStringElement, Keyword, Number, Operator as AstOperator, ParameterWithDefault, Stmt, UnaryOp,
7
- name::Name,
8
- };
9
- use ruff_python_parser::parse_module;
10
- use ruff_text_size::{Ranged, TextRange};
11
-
12
- use crate::{
13
- StackFrame,
14
- args::{ArgExprs, CallArg, CallKwarg, Kwarg},
15
- exception_private::ExcType,
16
- exception_public::{CodeLoc, MontyException},
17
- expressions::{
18
- Callable, CmpOperator, Comprehension, DictItem, Expr, ExprLoc, Identifier, Literal, Node, Operator,
19
- SequenceItem, UnpackTarget,
20
- },
21
- fstring::{ConversionFlag, FStringPart, FormatSpec},
22
- intern::{InternerBuilder, StringId},
23
- value::EitherStr,
24
- };
25
-
26
- /// Maximum nesting depth for AST structures during parsing.
27
- /// Matches CPython's limit of ~200 for nested parentheses.
28
- /// This prevents stack overflow from deeply nested structures like `((((x,),),),)`.
29
- #[cfg(not(debug_assertions))]
30
- pub const MAX_NESTING_DEPTH: u16 = 200;
31
- /// In debug builds, we use a lower limit because stack frames are much larger
32
- /// (no inlining, debug info, etc.). The limit is set conservatively to prevent
33
- /// stack overflow while still catching the error before the recursion limit.
34
- #[cfg(debug_assertions)]
35
- pub const MAX_NESTING_DEPTH: u16 = 35;
36
-
37
- /// A parameter in a function signature with optional default value.
38
- #[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
39
- pub struct ParsedParam {
40
- /// The parameter name.
41
- pub name: StringId,
42
- /// The default value expression (evaluated at definition time).
43
- pub default: Option<ExprLoc>,
44
- }
45
-
46
- /// A parsed function signature with all parameter types.
47
- ///
48
- /// This intermediate representation captures the structure of Python function
49
- /// parameters before name resolution. Default value expressions are stored
50
- /// as unevaluated AST and will be evaluated during the prepare phase.
51
- #[derive(Debug, Clone, Default, serde::Serialize, serde::Deserialize)]
52
- pub struct ParsedSignature {
53
- /// Positional-only parameters (before `/`).
54
- pub pos_args: Vec<ParsedParam>,
55
- /// Positional-or-keyword parameters.
56
- pub args: Vec<ParsedParam>,
57
- /// Variable positional parameter (`*args`).
58
- pub var_args: Option<StringId>,
59
- /// Keyword-only parameters (after `*` or `*args`).
60
- pub kwargs: Vec<ParsedParam>,
61
- /// Variable keyword parameter (`**kwargs`).
62
- pub var_kwargs: Option<StringId>,
63
- }
64
-
65
- impl ParsedSignature {
66
- /// Returns an iterator over all parameter names in the signature.
67
- ///
68
- /// Order: pos_args, args, var_args, kwargs, var_kwargs
69
- pub fn param_names(&self) -> impl Iterator<Item = StringId> + '_ {
70
- self.pos_args
71
- .iter()
72
- .map(|p| p.name)
73
- .chain(self.args.iter().map(|p| p.name))
74
- .chain(self.var_args.iter().copied())
75
- .chain(self.kwargs.iter().map(|p| p.name))
76
- .chain(self.var_kwargs.iter().copied())
77
- }
78
- }
79
-
80
- /// A raw (unprepared) function definition from the parser.
81
- ///
82
- /// Contains the function name, signature, and body as parsed AST nodes.
83
- /// During the prepare phase, this is transformed into `PreparedFunctionDef`
84
- /// with resolved names and scope information.
85
- #[derive(Debug, Clone)]
86
- pub struct RawFunctionDef {
87
- /// The function name identifier (not yet resolved to a namespace index).
88
- pub name: Identifier,
89
- /// The parsed function signature with parameter names and default expressions.
90
- pub signature: ParsedSignature,
91
- /// The unprepared function body (names not yet resolved).
92
- pub body: Vec<ParseNode>,
93
- /// Whether this is an async function (`async def`).
94
- pub is_async: bool,
95
- }
96
-
97
- /// Type alias for parsed AST nodes (output of the parser).
98
- ///
99
- /// This uses `Node<RawFunctionDef>` where function definitions contain their
100
- /// full unprepared body. After the prepare phase, this becomes `PreparedNode`
101
- /// (aka `Node<PreparedFunctionDef>`).
102
- pub type ParseNode = Node<RawFunctionDef>;
103
-
104
- #[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
105
- pub struct Try<N> {
106
- pub body: Vec<N>,
107
- pub handlers: Vec<ExceptHandler<N>>,
108
- pub or_else: Vec<N>,
109
- pub finally: Vec<N>,
110
- }
111
-
112
- /// A parsed exception handler (except clause).
113
- ///
114
- /// Represents `except ExcType as name:` or bare `except:` clauses.
115
- /// The exception type and variable binding are both optional.
116
- #[derive(Debug, Clone, serde::Serialize, serde::Deserialize)]
117
- pub struct ExceptHandler<N> {
118
- /// Exception type(s) to catch. None = bare except (catches all).
119
- pub exc_type: Option<ExprLoc>,
120
- /// Variable name for `except X as e:`. None = no binding.
121
- pub name: Option<Identifier>,
122
- /// Handler body statements.
123
- pub body: Vec<N>,
124
- }
125
-
126
- /// Result of parsing: the AST nodes and the string interner with all interned names.
127
- #[derive(Debug)]
128
- pub struct ParseResult {
129
- pub nodes: Vec<ParseNode>,
130
- pub interner: InternerBuilder,
131
- }
132
-
133
- pub(crate) fn parse(code: &str, filename: &str) -> Result<ParseResult, ParseError> {
134
- parse_with_interner(code, filename, InternerBuilder::new(code))
135
- }
136
-
137
- /// Parses code using a caller-provided interner seed.
138
- ///
139
- /// This enables incremental compilation flows (e.g. REPL) where existing
140
- /// interned IDs must remain stable across parse invocations.
141
- pub(crate) fn parse_with_interner(
142
- code: &str,
143
- filename: &str,
144
- interner: InternerBuilder,
145
- ) -> Result<ParseResult, ParseError> {
146
- let mut parser = Parser::new(code, filename, interner);
147
- let parsed = parse_module(code).map_err(|e| ParseError::syntax(e.to_string(), parser.convert_range(e.range())))?;
148
- let module = parsed.into_syntax();
149
- let nodes = parser.parse_statements(module.body)?;
150
- Ok(ParseResult {
151
- nodes,
152
- interner: parser.interner,
153
- })
154
- }
155
-
156
- /// Parser for converting ruff AST to Monty's intermediate ParseNode representation.
157
- ///
158
- /// Holds references to the source code and owns a string interner for names.
159
- /// The filename is interned once at construction and reused for all CodeRanges.
160
- pub struct Parser<'a> {
161
- line_ends: Vec<usize>,
162
- code: &'a str,
163
- /// Interned filename ID, used for all CodeRanges created by this parser.
164
- filename_id: StringId,
165
- /// String interner for names (variables, functions, etc).
166
- pub interner: InternerBuilder,
167
- /// Remaining nesting depth budget for recursive structures.
168
- /// Starts at MAX_NESTING_DEPTH and decrements on each nested level.
169
- /// When it reaches zero, we return a "too many nested parentheses" error.
170
- depth_remaining: u16,
171
- }
172
-
173
- impl<'a> Parser<'a> {
174
- fn new(code: &'a str, filename: &'a str, mut interner: InternerBuilder) -> Self {
175
- // Position of each line in the source code, to convert indexes to line number and column number
176
- let mut line_ends = vec![];
177
- for (i, c) in code.chars().enumerate() {
178
- if c == '\n' {
179
- line_ends.push(i);
180
- }
181
- }
182
- let filename_id = interner.intern(filename);
183
- Self {
184
- line_ends,
185
- code,
186
- filename_id,
187
- interner,
188
- depth_remaining: MAX_NESTING_DEPTH,
189
- }
190
- }
191
-
192
- fn parse_statements(&mut self, statements: Vec<Stmt>) -> Result<Vec<ParseNode>, ParseError> {
193
- statements.into_iter().map(|f| self.parse_statement(f)).collect()
194
- }
195
-
196
- fn parse_elif_else_clauses(&mut self, clauses: Vec<ElifElseClause>) -> Result<Vec<ParseNode>, ParseError> {
197
- let mut tail: Vec<ParseNode> = Vec::new();
198
- for clause in clauses.into_iter().rev() {
199
- match clause.test {
200
- Some(test) => {
201
- let test = self.parse_expression(test)?;
202
- let body = self.parse_statements(clause.body)?;
203
- let or_else = tail;
204
- let nested = Node::If { test, body, or_else };
205
- tail = vec![nested];
206
- }
207
- None => {
208
- tail = self.parse_statements(clause.body)?;
209
- }
210
- }
211
- }
212
- Ok(tail)
213
- }
214
-
215
- /// Parses an exception handler (except clause).
216
- ///
217
- /// Handles `except:`, `except ExcType:`, and `except ExcType as name:` forms.
218
- fn parse_except_handler(
219
- &mut self,
220
- handler: ruff_python_ast::ExceptHandler,
221
- ) -> Result<ExceptHandler<ParseNode>, ParseError> {
222
- let ruff_python_ast::ExceptHandler::ExceptHandler(h) = handler;
223
- let exc_type = match h.type_ {
224
- Some(expr) => Some(self.parse_expression(*expr)?),
225
- None => None,
226
- };
227
- let name = h.name.map(|n| self.identifier(&n.id, n.range));
228
- let body = self.parse_statements(h.body)?;
229
- Ok(ExceptHandler { exc_type, name, body })
230
- }
231
-
232
- fn parse_statement(&mut self, statement: Stmt) -> Result<ParseNode, ParseError> {
233
- self.decr_depth_remaining(|| statement.range())?;
234
- let result = self.parse_statement_impl(statement);
235
- self.depth_remaining += 1;
236
- result
237
- }
238
-
239
- fn parse_statement_impl(&mut self, statement: Stmt) -> Result<ParseNode, ParseError> {
240
- match statement {
241
- Stmt::FunctionDef(function) => {
242
- let params = &function.parameters;
243
-
244
- // Parse positional-only parameters (before /)
245
- let pos_args = self.parse_params_with_defaults(&params.posonlyargs)?;
246
-
247
- // Parse positional-or-keyword parameters
248
- let args = self.parse_params_with_defaults(&params.args)?;
249
-
250
- // Parse *args
251
- let var_args = params.vararg.as_ref().map(|p| self.interner.intern(&p.name.id));
252
-
253
- // Parse keyword-only parameters (after * or *args)
254
- let kwargs = self.parse_params_with_defaults(&params.kwonlyargs)?;
255
-
256
- // Parse **kwargs
257
- let var_kwargs = params.kwarg.as_ref().map(|p| self.interner.intern(&p.name.id));
258
-
259
- let signature = ParsedSignature {
260
- pos_args,
261
- args,
262
- var_args,
263
- kwargs,
264
- var_kwargs,
265
- };
266
-
267
- let name = self.identifier(&function.name.id, function.name.range);
268
- // Parse function body recursively
269
- let body = self.parse_statements(function.body)?;
270
- let is_async = function.is_async;
271
-
272
- Ok(Node::FunctionDef(RawFunctionDef {
273
- name,
274
- signature,
275
- body,
276
- is_async,
277
- }))
278
- }
279
- Stmt::ClassDef(c) => Err(ParseError::not_implemented(
280
- "class definitions",
281
- self.convert_range(c.range),
282
- )),
283
- Stmt::Return(ast::StmtReturn { value, .. }) => match value {
284
- Some(value) => Ok(Node::Return(self.parse_expression(*value)?)),
285
- None => Ok(Node::ReturnNone),
286
- },
287
- Stmt::Delete(d) => Err(ParseError::not_implemented(
288
- "the 'del' statement",
289
- self.convert_range(d.range),
290
- )),
291
- Stmt::TypeAlias(t) => Err(ParseError::not_implemented("type aliases", self.convert_range(t.range))),
292
- Stmt::Assign(ast::StmtAssign {
293
- targets, value, range, ..
294
- }) => self.parse_assignment(first(targets, self.convert_range(range))?, *value),
295
- Stmt::AugAssign(ast::StmtAugAssign { target, op, value, .. }) => {
296
- let op = convert_op(op);
297
- let value = self.parse_expression(*value)?;
298
- match *target {
299
- AstExpr::Subscript(ast::ExprSubscript {
300
- value: object,
301
- slice,
302
- range,
303
- ..
304
- }) => Ok(Node::SubscriptOpAssign {
305
- target: self.parse_identifier(*object)?,
306
- index: self.parse_expression(*slice)?,
307
- op,
308
- object: value,
309
- target_position: self.convert_range(range),
310
- }),
311
- other => Ok(Node::OpAssign {
312
- target: self.parse_identifier(other)?,
313
- op,
314
- object: value,
315
- }),
316
- }
317
- }
318
- Stmt::AnnAssign(ast::StmtAnnAssign { target, value, .. }) => match value {
319
- Some(value) => self.parse_assignment(*target, *value),
320
- None => Ok(Node::Pass),
321
- },
322
- Stmt::For(ast::StmtFor {
323
- is_async,
324
- target,
325
- iter,
326
- body,
327
- orelse,
328
- range,
329
- ..
330
- }) => {
331
- if is_async {
332
- return Err(ParseError::not_implemented(
333
- "async for loops",
334
- self.convert_range(range),
335
- ));
336
- }
337
- Ok(Node::For {
338
- target: self.parse_unpack_target(*target)?,
339
- iter: self.parse_expression(*iter)?,
340
- body: self.parse_statements(body)?,
341
- or_else: self.parse_statements(orelse)?,
342
- })
343
- }
344
- Stmt::While(ast::StmtWhile { test, body, orelse, .. }) => Ok(Node::While {
345
- test: self.parse_expression(*test)?,
346
- body: self.parse_statements(body)?,
347
- or_else: self.parse_statements(orelse)?,
348
- }),
349
- Stmt::If(ast::StmtIf {
350
- test,
351
- body,
352
- elif_else_clauses,
353
- ..
354
- }) => {
355
- let test = self.parse_expression(*test)?;
356
- let body = self.parse_statements(body)?;
357
- let or_else = self.parse_elif_else_clauses(elif_else_clauses)?;
358
- Ok(Node::If { test, body, or_else })
359
- }
360
- Stmt::With(ast::StmtWith { is_async, range, .. }) => {
361
- if is_async {
362
- Err(ParseError::not_implemented(
363
- "async context managers (async with)",
364
- self.convert_range(range),
365
- ))
366
- } else {
367
- Err(ParseError::not_implemented(
368
- "context managers (with statements)",
369
- self.convert_range(range),
370
- ))
371
- }
372
- }
373
- Stmt::Match(m) => Err(ParseError::not_implemented(
374
- "pattern matching (match statements)",
375
- self.convert_range(m.range),
376
- )),
377
- Stmt::Raise(ast::StmtRaise { exc, .. }) => {
378
- // TODO add cause to Node::Raise
379
- let expr = match exc {
380
- Some(expr) => Some(self.parse_expression(*expr)?),
381
- None => None,
382
- };
383
- Ok(Node::Raise(expr))
384
- }
385
- Stmt::Try(ast::StmtTry {
386
- body,
387
- handlers,
388
- orelse,
389
- finalbody,
390
- is_star,
391
- range,
392
- ..
393
- }) => {
394
- if is_star {
395
- Err(ParseError::not_implemented(
396
- "exception groups (try*/except*)",
397
- self.convert_range(range),
398
- ))
399
- } else {
400
- let body = self.parse_statements(body)?;
401
- let handlers = handlers
402
- .into_iter()
403
- .map(|h| self.parse_except_handler(h))
404
- .collect::<Result<Vec<_>, _>>()?;
405
- let or_else = self.parse_statements(orelse)?;
406
- let finally = self.parse_statements(finalbody)?;
407
- Ok(Node::Try(Try {
408
- body,
409
- handlers,
410
- or_else,
411
- finally,
412
- }))
413
- }
414
- }
415
- Stmt::Assert(ast::StmtAssert { test, msg, .. }) => {
416
- let test = self.parse_expression(*test)?;
417
- let msg = match msg {
418
- Some(m) => Some(self.parse_expression(*m)?),
419
- None => None,
420
- };
421
- Ok(Node::Assert { test, msg })
422
- }
423
- Stmt::Import(ast::StmtImport { names, range, .. }) => {
424
- // We only support single module imports (e.g., `import sys`)
425
- // Multi-module imports (e.g., `import sys, os`) are not supported
426
- let position = self.convert_range(range);
427
- if names.len() != 1 {
428
- return Err(ParseError::not_implemented("multi-module import statements", position));
429
- }
430
- let alias_node = &names[0];
431
- let module_name = self.interner.intern(&alias_node.name);
432
- // The binding name is the alias if present, otherwise the module name
433
- let binding_name = alias_node
434
- .asname
435
- .as_ref()
436
- .map_or(module_name, |n| self.interner.intern(&n.id));
437
- // Create an unresolved identifier (namespace slot will be set during prepare)
438
- let binding = Identifier::new(binding_name, position);
439
- Ok(Node::Import { module_name, binding })
440
- }
441
- Stmt::ImportFrom(ast::StmtImportFrom {
442
- module,
443
- names,
444
- level,
445
- range,
446
- ..
447
- }) => {
448
- let position = self.convert_range(range);
449
- // We only support absolute imports (level 0)
450
- if level != 0 {
451
- return Err(ParseError::import_error(
452
- "attempted relative import with no known parent package",
453
- position,
454
- ));
455
- }
456
- // Module name is required for absolute imports
457
- let module_name = match module {
458
- Some(m) => self.interner.intern(&m),
459
- None => {
460
- return Err(ParseError::import_error(
461
- "attempted relative import with no known parent package",
462
- position,
463
- ));
464
- }
465
- };
466
- // Parse the imported names
467
- let names = names
468
- .iter()
469
- .map(|alias| {
470
- // Check for star import which is not supported
471
- if alias.name.as_str() == "*" {
472
- return Err(ParseError::not_supported(
473
- "Wildcard imports (`from ... import *`) are not supported",
474
- position,
475
- ));
476
- }
477
- let name = self.interner.intern(&alias.name);
478
- // The binding name is the alias if provided, otherwise the import name
479
- let binding_name = alias.asname.as_ref().map_or(name, |n| self.interner.intern(&n.id));
480
- // Create an unresolved identifier (namespace slot will be set during prepare)
481
- let binding = Identifier::new(binding_name, position);
482
- Ok((name, binding))
483
- })
484
- .collect::<Result<Vec<_>, _>>()?;
485
- Ok(Node::ImportFrom {
486
- module_name,
487
- names,
488
- position,
489
- })
490
- }
491
- Stmt::Global(ast::StmtGlobal { names, range, .. }) => {
492
- let names = names
493
- .iter()
494
- .map(|id| self.interner.intern(&self.code[id.range]))
495
- .collect();
496
- Ok(Node::Global {
497
- position: self.convert_range(range),
498
- names,
499
- })
500
- }
501
- Stmt::Nonlocal(ast::StmtNonlocal { names, range, .. }) => {
502
- let names = names
503
- .iter()
504
- .map(|id| self.interner.intern(&self.code[id.range]))
505
- .collect();
506
- Ok(Node::Nonlocal {
507
- position: self.convert_range(range),
508
- names,
509
- })
510
- }
511
- Stmt::Expr(ast::StmtExpr { value, .. }) => self.parse_expression(*value).map(Node::Expr),
512
- Stmt::Pass(_) => Ok(Node::Pass),
513
- Stmt::Break(b) => Ok(Node::Break {
514
- position: self.convert_range(b.range),
515
- }),
516
- Stmt::Continue(c) => Ok(Node::Continue {
517
- position: self.convert_range(c.range),
518
- }),
519
- Stmt::IpyEscapeCommand(i) => Err(ParseError::not_implemented(
520
- "IPython escape commands",
521
- self.convert_range(i.range),
522
- )),
523
- }
524
- }
525
-
526
- /// `lhs = rhs` -> `lhs, rhs`
527
- /// Handles simple assignments (x = value), subscript assignments (dict[key] = value),
528
- /// attribute assignments (obj.attr = value), and tuple unpacking (a, b = value)
529
- fn parse_assignment(&mut self, lhs: AstExpr, rhs: AstExpr) -> Result<ParseNode, ParseError> {
530
- match lhs {
531
- // Subscript assignment like dict[key] = value
532
- AstExpr::Subscript(ast::ExprSubscript {
533
- value, slice, range, ..
534
- }) => Ok(Node::SubscriptAssign {
535
- target: self.parse_identifier(*value)?,
536
- index: self.parse_expression(*slice)?,
537
- value: self.parse_expression(rhs)?,
538
- target_position: self.convert_range(range),
539
- }),
540
- // Attribute assignment like obj.attr = value (supports chained like a.b.c = value)
541
- AstExpr::Attribute(ast::ExprAttribute { value, attr, range, .. }) => Ok(Node::AttrAssign {
542
- object: self.parse_expression(*value)?,
543
- attr: EitherStr::Interned(self.interner.intern(attr.id())),
544
- target_position: self.convert_range(range),
545
- value: self.parse_expression(rhs)?,
546
- }),
547
- // Tuple unpacking like a, b = value or (a, b), c = nested
548
- AstExpr::Tuple(ast::ExprTuple { elts, range, .. }) => {
549
- let targets_position = self.convert_range(range);
550
- let targets = elts
551
- .into_iter()
552
- .map(|e| self.parse_unpack_target(e)) // Use parse_unpack_target for recursion
553
- .collect::<Result<Vec<_>, _>>()?;
554
- Ok(Node::UnpackAssign {
555
- targets,
556
- targets_position,
557
- object: self.parse_expression(rhs)?,
558
- })
559
- }
560
- // List unpacking like [a, b] = value or [a, *rest] = value
561
- AstExpr::List(ast::ExprList { elts, range, .. }) => {
562
- let targets_position = self.convert_range(range);
563
- let targets = elts
564
- .into_iter()
565
- .map(|e| self.parse_unpack_target(e))
566
- .collect::<Result<Vec<_>, _>>()?;
567
- Ok(Node::UnpackAssign {
568
- targets,
569
- targets_position,
570
- object: self.parse_expression(rhs)?,
571
- })
572
- }
573
- // Simple identifier assignment like x = value
574
- _ => Ok(Node::Assign {
575
- target: self.parse_identifier(lhs)?,
576
- object: self.parse_expression(rhs)?,
577
- }),
578
- }
579
- }
580
-
581
- /// Parses an expression from the ruff AST into Monty's ExprLoc representation.
582
- ///
583
- /// Includes depth tracking to prevent stack overflow from deeply nested structures.
584
- /// Matches CPython's limit of 200 for nested parentheses.
585
- fn parse_expression(&mut self, expression: AstExpr) -> Result<ExprLoc, ParseError> {
586
- self.decr_depth_remaining(|| expression.range())?;
587
- let result = self.parse_expression_impl(expression);
588
- self.depth_remaining += 1;
589
- result
590
- }
591
-
592
- fn parse_expression_impl(&mut self, expression: AstExpr) -> Result<ExprLoc, ParseError> {
593
- match expression {
594
- AstExpr::BoolOp(ast::ExprBoolOp { op, values, range, .. }) => {
595
- // Handle chained boolean operations like `a and b and c` by right-folding
596
- // into nested binary operations: `a and (b and c)`
597
- let rust_op = convert_bool_op(op);
598
- let position = self.convert_range(range);
599
- let mut values_iter = values.into_iter().rev();
600
-
601
- // Start with the rightmost value
602
- let last_value = values_iter.next().expect("Expected at least one value in boolean op");
603
- let mut result = self.parse_expression(last_value)?;
604
-
605
- // Fold from right to left
606
- for value in values_iter {
607
- let left = Box::new(self.parse_expression(value)?);
608
- result = ExprLoc::new(
609
- position,
610
- Expr::Op {
611
- left,
612
- op: rust_op.clone(),
613
- right: Box::new(result),
614
- },
615
- );
616
- }
617
- Ok(result)
618
- }
619
- AstExpr::Named(ast::ExprNamed {
620
- target, value, range, ..
621
- }) => {
622
- let target_ident = self.parse_identifier(*target)?;
623
- let value_expr = self.parse_expression(*value)?;
624
- Ok(ExprLoc::new(
625
- self.convert_range(range),
626
- Expr::Named {
627
- target: target_ident,
628
- value: Box::new(value_expr),
629
- },
630
- ))
631
- }
632
- AstExpr::BinOp(ast::ExprBinOp {
633
- left, op, right, range, ..
634
- }) => {
635
- let left = Box::new(self.parse_expression(*left)?);
636
- let right = Box::new(self.parse_expression(*right)?);
637
- Ok(ExprLoc {
638
- position: self.convert_range(range),
639
- expr: Expr::Op {
640
- left,
641
- op: convert_op(op),
642
- right,
643
- },
644
- })
645
- }
646
- AstExpr::UnaryOp(ast::ExprUnaryOp { op, operand, range, .. }) => match op {
647
- UnaryOp::Not => {
648
- let operand = Box::new(self.parse_expression(*operand)?);
649
- Ok(ExprLoc::new(self.convert_range(range), Expr::Not(operand)))
650
- }
651
- UnaryOp::USub => {
652
- let operand = Box::new(self.parse_expression(*operand)?);
653
- Ok(ExprLoc::new(self.convert_range(range), Expr::UnaryMinus(operand)))
654
- }
655
- UnaryOp::UAdd => {
656
- let operand = Box::new(self.parse_expression(*operand)?);
657
- Ok(ExprLoc::new(self.convert_range(range), Expr::UnaryPlus(operand)))
658
- }
659
- UnaryOp::Invert => {
660
- let operand = Box::new(self.parse_expression(*operand)?);
661
- Ok(ExprLoc::new(self.convert_range(range), Expr::UnaryInvert(operand)))
662
- }
663
- },
664
- AstExpr::Lambda(ast::ExprLambda {
665
- parameters,
666
- body,
667
- range,
668
- ..
669
- }) => {
670
- let position = self.convert_range(range);
671
-
672
- // Intern the lambda name
673
- let name_id = self.interner.intern("<lambda>");
674
-
675
- // Parse lambda parameters (similar to function parameters)
676
- let signature = if let Some(params) = parameters {
677
- // Parse positional-only parameters (before /)
678
- let pos_args = self.parse_params_with_defaults(&params.posonlyargs)?;
679
-
680
- // Parse positional-or-keyword parameters
681
- let args = self.parse_params_with_defaults(&params.args)?;
682
-
683
- // Parse *args
684
- let var_args = params.vararg.as_ref().map(|p| self.interner.intern(&p.name.id));
685
-
686
- // Parse keyword-only parameters (after * or *args)
687
- let kwargs = self.parse_params_with_defaults(&params.kwonlyargs)?;
688
-
689
- // Parse **kwargs
690
- let var_kwargs = params.kwarg.as_ref().map(|p| self.interner.intern(&p.name.id));
691
-
692
- ParsedSignature {
693
- pos_args,
694
- args,
695
- var_args,
696
- kwargs,
697
- var_kwargs,
698
- }
699
- } else {
700
- // No parameters (e.g., `lambda: 42`)
701
- ParsedSignature::default()
702
- };
703
-
704
- // Parse the body expression
705
- let body = Box::new(self.parse_expression(*body)?);
706
-
707
- Ok(ExprLoc::new(
708
- position,
709
- Expr::LambdaRaw {
710
- name_id,
711
- signature,
712
- body,
713
- },
714
- ))
715
- }
716
- AstExpr::If(ast::ExprIf {
717
- test,
718
- body,
719
- orelse,
720
- range,
721
- ..
722
- }) => Ok(ExprLoc::new(
723
- self.convert_range(range),
724
- Expr::IfElse {
725
- test: Box::new(self.parse_expression(*test)?),
726
- body: Box::new(self.parse_expression(*body)?),
727
- orelse: Box::new(self.parse_expression(*orelse)?),
728
- },
729
- )),
730
- AstExpr::Dict(ast::ExprDict { items, range, .. }) => {
731
- let position = self.convert_range(range);
732
- let mut dict_items = Vec::new();
733
- for ast::DictItem { key, value } in items {
734
- // key is Option<Expr> - None represents ** unpacking (PEP 448)
735
- if let Some(key_expr_ast) = key {
736
- let key_expr = self.parse_expression(key_expr_ast)?;
737
- let value_expr = self.parse_expression(value)?;
738
- dict_items.push(DictItem::Pair(key_expr, value_expr));
739
- } else {
740
- // **expr unpack in a dict literal: later keys silently win
741
- let unpack_expr = self.parse_expression(value)?;
742
- dict_items.push(DictItem::Unpack(unpack_expr));
743
- }
744
- }
745
- Ok(ExprLoc::new(position, Expr::Dict(dict_items)))
746
- }
747
- AstExpr::Set(ast::ExprSet { elts, range, .. }) => {
748
- let mut items = Vec::new();
749
- for e in elts {
750
- items.push(self.parse_sequence_item(e)?);
751
- }
752
- Ok(ExprLoc::new(self.convert_range(range), Expr::Set(items)))
753
- }
754
- AstExpr::ListComp(ast::ExprListComp {
755
- elt, generators, range, ..
756
- }) => {
757
- let elt = Box::new(self.parse_expression(*elt)?);
758
- let generators = self.parse_comprehension_generators(generators)?;
759
- Ok(ExprLoc::new(
760
- self.convert_range(range),
761
- Expr::ListComp { elt, generators },
762
- ))
763
- }
764
- AstExpr::SetComp(ast::ExprSetComp {
765
- elt, generators, range, ..
766
- }) => {
767
- let elt = Box::new(self.parse_expression(*elt)?);
768
- let generators = self.parse_comprehension_generators(generators)?;
769
- Ok(ExprLoc::new(
770
- self.convert_range(range),
771
- Expr::SetComp { elt, generators },
772
- ))
773
- }
774
- AstExpr::DictComp(ast::ExprDictComp {
775
- key,
776
- value,
777
- generators,
778
- range,
779
- ..
780
- }) => {
781
- let key = Box::new(self.parse_expression(*key)?);
782
- let value = Box::new(self.parse_expression(*value)?);
783
- let generators = self.parse_comprehension_generators(generators)?;
784
- Ok(ExprLoc::new(
785
- self.convert_range(range),
786
- Expr::DictComp { key, value, generators },
787
- ))
788
- }
789
- AstExpr::Generator(ast::ExprGenerator {
790
- elt, generators, range, ..
791
- }) => {
792
- // TODO: When proper generators are implemented, this should produce
793
- // Expr::Generator instead of Expr::ListComp. Currently we treat generator
794
- // expressions as list comprehensions since we don't have generator support.
795
- let elt = Box::new(self.parse_expression(*elt)?);
796
- let generators = self.parse_comprehension_generators(generators)?;
797
- Ok(ExprLoc::new(
798
- self.convert_range(range),
799
- Expr::ListComp { elt, generators },
800
- ))
801
- }
802
- AstExpr::Await(a) => {
803
- let value = self.parse_expression(*a.value)?;
804
- Ok(ExprLoc::new(self.convert_range(a.range), Expr::Await(Box::new(value))))
805
- }
806
- AstExpr::Yield(y) => Err(ParseError::not_implemented(
807
- "yield expressions",
808
- self.convert_range(y.range),
809
- )),
810
- AstExpr::YieldFrom(y) => Err(ParseError::not_implemented(
811
- "yield from expressions",
812
- self.convert_range(y.range),
813
- )),
814
- AstExpr::Compare(ast::ExprCompare {
815
- left,
816
- ops,
817
- comparators,
818
- range,
819
- ..
820
- }) => {
821
- let position = self.convert_range(range);
822
- let ops_vec = ops.into_vec();
823
- let comparators_vec = comparators.into_vec();
824
-
825
- // Simple case: single comparison (most common)
826
- if ops_vec.len() == 1 {
827
- return Ok(ExprLoc::new(
828
- position,
829
- Expr::CmpOp {
830
- left: Box::new(self.parse_expression(*left)?),
831
- op: convert_compare_op(ops_vec.into_iter().next().unwrap()),
832
- right: Box::new(self.parse_expression(comparators_vec.into_iter().next().unwrap())?),
833
- },
834
- ));
835
- }
836
-
837
- // Chain comparison: transform to nested And expressions
838
- self.parse_chain_comparison(*left, ops_vec, comparators_vec, position)
839
- }
840
- AstExpr::Call(ast::ExprCall {
841
- func, arguments, range, ..
842
- }) => {
843
- let position = self.convert_range(range);
844
- let ast::Arguments { args, keywords, .. } = arguments;
845
- let args_vec = args.into_vec();
846
- let keywords_vec = keywords.into_vec();
847
-
848
- // Detect whether we need the generalized path (PEP 448):
849
- // - multiple *args unpacks, OR
850
- // - positional argument after *args, OR
851
- // - multiple **kwargs unpacks
852
- let needs_generalized = Self::needs_generalized_call(&args_vec, &keywords_vec);
853
-
854
- let args = if needs_generalized {
855
- self.parse_generalized_call_args(args_vec, keywords_vec)?
856
- } else {
857
- self.parse_simple_call_args(args_vec, keywords_vec)?
858
- };
859
- match *func {
860
- AstExpr::Name(ast::ExprName { id, range, .. }) => {
861
- // Always create Callable::Name — builtin resolution happens in
862
- // the prepare phase with scope awareness, so local assignments
863
- // can shadow builtins.
864
- let ident = self.identifier(&id, range);
865
- let callable = Callable::Name(ident);
866
- Ok(ExprLoc::new(
867
- position,
868
- Expr::Call {
869
- callable,
870
- args: Box::new(args),
871
- },
872
- ))
873
- }
874
- AstExpr::Attribute(ast::ExprAttribute { value, attr, .. }) => {
875
- let object = Box::new(self.parse_expression(*value)?);
876
- Ok(ExprLoc::new(
877
- position,
878
- Expr::AttrCall {
879
- object,
880
- attr: EitherStr::Interned(self.interner.intern(attr.id())),
881
- args: Box::new(args),
882
- },
883
- ))
884
- }
885
- other => {
886
- // Handle arbitrary expression as callable (e.g., lambda calls)
887
- let callable = Box::new(self.parse_expression(other)?);
888
- Ok(ExprLoc::new(
889
- position,
890
- Expr::IndirectCall {
891
- callable,
892
- args: Box::new(args),
893
- },
894
- ))
895
- }
896
- }
897
- }
898
- AstExpr::FString(ast::ExprFString { value, range, .. }) => self.parse_fstring(&value, range),
899
- AstExpr::TString(t) => Err(ParseError::not_implemented(
900
- "template strings (t-strings)",
901
- self.convert_range(t.range),
902
- )),
903
- AstExpr::StringLiteral(ast::ExprStringLiteral { value, range, .. }) => {
904
- let string_id = self.interner.intern(&value.to_string());
905
- Ok(ExprLoc::new(
906
- self.convert_range(range),
907
- Expr::Literal(Literal::Str(string_id)),
908
- ))
909
- }
910
- AstExpr::BytesLiteral(ast::ExprBytesLiteral { value, range, .. }) => {
911
- let bytes: Cow<'_, [u8]> = Cow::from(&value);
912
- let bytes_id = self.interner.intern_bytes(&bytes);
913
- Ok(ExprLoc::new(
914
- self.convert_range(range),
915
- Expr::Literal(Literal::Bytes(bytes_id)),
916
- ))
917
- }
918
- AstExpr::NumberLiteral(ast::ExprNumberLiteral { value, range, .. }) => {
919
- let position = self.convert_range(range);
920
- let const_value = match value {
921
- Number::Int(i) => {
922
- if let Some(i) = i.as_i64() {
923
- Literal::Int(i)
924
- } else {
925
- // Integer too large for i64, parse string representation as BigInt
926
- // Handles radix prefixes (0x, 0o, 0b) and underscores
927
- let bi = parse_int_literal(&i.to_string())
928
- .ok_or_else(|| ParseError::syntax(format!("invalid integer literal: {i}"), position))?;
929
- let long_int_id = self.interner.intern_long_int(bi);
930
- Literal::LongInt(long_int_id)
931
- }
932
- }
933
- Number::Float(f) => Literal::Float(f),
934
- Number::Complex { .. } => return Err(ParseError::not_implemented("complex constants", position)),
935
- };
936
- Ok(ExprLoc::new(position, Expr::Literal(const_value)))
937
- }
938
- AstExpr::BooleanLiteral(ast::ExprBooleanLiteral { value, range, .. }) => Ok(ExprLoc::new(
939
- self.convert_range(range),
940
- Expr::Literal(Literal::Bool(value)),
941
- )),
942
- AstExpr::NoneLiteral(ast::ExprNoneLiteral { range, .. }) => {
943
- Ok(ExprLoc::new(self.convert_range(range), Expr::Literal(Literal::None)))
944
- }
945
- AstExpr::EllipsisLiteral(ast::ExprEllipsisLiteral { range, .. }) => Ok(ExprLoc::new(
946
- self.convert_range(range),
947
- Expr::Literal(Literal::Ellipsis),
948
- )),
949
- AstExpr::Attribute(ast::ExprAttribute { value, attr, range, .. }) => {
950
- let object = Box::new(self.parse_expression(*value)?);
951
- let position = self.convert_range(range);
952
- Ok(ExprLoc::new(
953
- position,
954
- Expr::AttrGet {
955
- object,
956
- attr: EitherStr::Interned(self.interner.intern(attr.id())),
957
- },
958
- ))
959
- }
960
- AstExpr::Subscript(ast::ExprSubscript {
961
- value, slice, range, ..
962
- }) => {
963
- let object = Box::new(self.parse_expression(*value)?);
964
- let index = Box::new(self.parse_expression(*slice)?);
965
- Ok(ExprLoc::new(
966
- self.convert_range(range),
967
- Expr::Subscript { object, index },
968
- ))
969
- }
970
- AstExpr::Starred(s) => Err(ParseError::not_implemented(
971
- "starred expressions (*expr)",
972
- self.convert_range(s.range),
973
- )),
974
- AstExpr::Name(ast::ExprName { id, range, .. }) => {
975
- let position = self.convert_range(range);
976
- // Always create Expr::Name — builtin resolution happens in the prepare
977
- // phase with scope awareness, so local assignments can shadow builtins.
978
- let expr = Expr::Name(self.identifier(&id, range));
979
- Ok(ExprLoc::new(position, expr))
980
- }
981
- AstExpr::List(ast::ExprList { elts, range, .. }) => {
982
- let mut items = Vec::new();
983
- for e in elts {
984
- items.push(self.parse_sequence_item(e)?);
985
- }
986
- Ok(ExprLoc::new(self.convert_range(range), Expr::List(items)))
987
- }
988
- AstExpr::Tuple(ast::ExprTuple { elts, range, .. }) => {
989
- let mut items = Vec::new();
990
- for e in elts {
991
- items.push(self.parse_sequence_item(e)?);
992
- }
993
- Ok(ExprLoc::new(self.convert_range(range), Expr::Tuple(items)))
994
- }
995
- AstExpr::Slice(ast::ExprSlice {
996
- lower,
997
- upper,
998
- step,
999
- range,
1000
- ..
1001
- }) => {
1002
- let lower = lower.map(|e| self.parse_expression(*e)).transpose()?;
1003
- let upper = upper.map(|e| self.parse_expression(*e)).transpose()?;
1004
- let step = step.map(|e| self.parse_expression(*e)).transpose()?;
1005
- Ok(ExprLoc::new(
1006
- self.convert_range(range),
1007
- Expr::Slice {
1008
- lower: lower.map(Box::new),
1009
- upper: upper.map(Box::new),
1010
- step: step.map(Box::new),
1011
- },
1012
- ))
1013
- }
1014
- AstExpr::IpyEscapeCommand(i) => Err(ParseError::not_implemented(
1015
- "IPython escape commands",
1016
- self.convert_range(i.range),
1017
- )),
1018
- }
1019
- }
1020
-
1021
- /// Converts an AST expression into a `SequenceItem` for list/tuple/set literals.
1022
- ///
1023
- /// A `Starred` node becomes `SequenceItem::Unpack`; all other expressions
1024
- /// become `SequenceItem::Value`. This is the entry point for PEP 448 unpack
1025
- /// handling in collection literals.
1026
- fn parse_sequence_item(&mut self, expr: AstExpr) -> Result<SequenceItem, ParseError> {
1027
- if let AstExpr::Starred(ast::ExprStarred { value, .. }) = expr {
1028
- Ok(SequenceItem::Unpack(self.parse_expression(*value)?))
1029
- } else {
1030
- Ok(SequenceItem::Value(self.parse_expression(expr)?))
1031
- }
1032
- }
1033
-
1034
- /// Detects whether a function call needs the generalized `GeneralizedCall` path.
1035
- ///
1036
- /// Returns `true` when the call has:
1037
- /// - More than one `*unpack` among positional args, OR
1038
- /// - A plain positional arg following a `*unpack`, OR
1039
- /// - More than one `**unpack` among keyword args.
1040
- ///
1041
- /// In all these cases the simple `ArgsKargs` representation is insufficient
1042
- /// and `parse_generalized_call_args` must be used instead.
1043
- fn needs_generalized_call(args: &[AstExpr], keywords: &[Keyword]) -> bool {
1044
- let mut seen_star = false;
1045
- for arg in args {
1046
- match arg {
1047
- AstExpr::Starred(_) => {
1048
- if seen_star {
1049
- return true; // second *unpack
1050
- }
1051
- seen_star = true;
1052
- }
1053
- _ => {
1054
- if seen_star {
1055
- return true; // positional after *unpack
1056
- }
1057
- }
1058
- }
1059
- }
1060
- // Multiple **kwargs unpacks?
1061
- keywords.iter().filter(|k| k.arg.is_none()).count() > 1
1062
- }
1063
-
1064
- /// Parses function call args for the simple case (at most one * and one **).
1065
- ///
1066
- /// Returns `ArgExprs::new_with_var_kwargs(...)` as before, preserving the
1067
- /// fast path for the vast majority of function calls.
1068
- fn parse_simple_call_args(
1069
- &mut self,
1070
- args_vec: Vec<AstExpr>,
1071
- keywords_vec: Vec<Keyword>,
1072
- ) -> Result<ArgExprs, ParseError> {
1073
- let mut positional_args = Vec::new();
1074
- let mut var_args_expr: Option<ExprLoc> = None;
1075
-
1076
- for arg_expr in args_vec {
1077
- match arg_expr {
1078
- AstExpr::Starred(ast::ExprStarred { value, .. }) => {
1079
- var_args_expr = Some(self.parse_expression(*value)?);
1080
- }
1081
- other => {
1082
- positional_args.push(self.parse_expression(other)?);
1083
- }
1084
- }
1085
- }
1086
- let (kwargs, var_kwargs) = self.parse_keywords(keywords_vec)?;
1087
- Ok(ArgExprs::new_with_var_kwargs(
1088
- positional_args,
1089
- var_args_expr,
1090
- kwargs,
1091
- var_kwargs,
1092
- ))
1093
- }
1094
-
1095
- /// Parses function call args for the PEP 448 generalized case.
1096
- ///
1097
- /// Builds `Vec<CallArg>` and `Vec<CallKwarg>` preserving the full order of
1098
- /// positional and keyword arguments so the compiler can emit correct
1099
- /// `ListAppend`/`ListExtend`/`DictMerge` sequences.
1100
- fn parse_generalized_call_args(
1101
- &mut self,
1102
- args_vec: Vec<AstExpr>,
1103
- keywords_vec: Vec<Keyword>,
1104
- ) -> Result<ArgExprs, ParseError> {
1105
- let mut call_args = Vec::new();
1106
- for arg_expr in args_vec {
1107
- match arg_expr {
1108
- AstExpr::Starred(ast::ExprStarred { value, .. }) => {
1109
- call_args.push(CallArg::Unpack(self.parse_expression(*value)?));
1110
- }
1111
- other => {
1112
- call_args.push(CallArg::Value(self.parse_expression(other)?));
1113
- }
1114
- }
1115
- }
1116
-
1117
- let mut call_kwargs = Vec::new();
1118
- for kwarg in keywords_vec {
1119
- if let Some(key) = kwarg.arg {
1120
- let key_ident = self.identifier(&key.id, key.range);
1121
- let value = self.parse_expression(kwarg.value)?;
1122
- call_kwargs.push(CallKwarg::Named(Kwarg { key: key_ident, value }));
1123
- } else {
1124
- let unpack_expr = self.parse_expression(kwarg.value)?;
1125
- call_kwargs.push(CallKwarg::Unpack(unpack_expr));
1126
- }
1127
- }
1128
-
1129
- Ok(ArgExprs::new_generalized(call_args, call_kwargs))
1130
- }
1131
-
1132
- /// Parses keyword arguments, separating regular kwargs from var_kwargs (`**expr`).
1133
- ///
1134
- /// Returns `(kwargs, var_kwargs)` where kwargs is a vec of named keyword arguments
1135
- /// and var_kwargs is an optional expression for `**expr` unpacking.
1136
- fn parse_keywords(&mut self, keywords: Vec<Keyword>) -> Result<(Vec<Kwarg>, Option<ExprLoc>), ParseError> {
1137
- let mut kwargs = Vec::new();
1138
- let mut var_kwargs = None;
1139
-
1140
- for kwarg in keywords {
1141
- if let Some(key) = kwarg.arg {
1142
- // Regular kwarg: key=value
1143
- let key = self.identifier(&key.id, key.range);
1144
- let value = self.parse_expression(kwarg.value)?;
1145
- kwargs.push(Kwarg { key, value });
1146
- } else {
1147
- // Var kwargs: **expr
1148
- if var_kwargs.is_some() {
1149
- return Err(ParseError::not_implemented(
1150
- "multiple **kwargs unpacking",
1151
- self.convert_range(kwarg.range),
1152
- ));
1153
- }
1154
- var_kwargs = Some(self.parse_expression(kwarg.value)?);
1155
- }
1156
- }
1157
-
1158
- Ok((kwargs, var_kwargs))
1159
- }
1160
-
1161
- fn parse_identifier(&mut self, ast: AstExpr) -> Result<Identifier, ParseError> {
1162
- match ast {
1163
- AstExpr::Name(ast::ExprName { id, range, .. }) => Ok(self.identifier(&id, range)),
1164
- other => Err(ParseError::syntax(
1165
- format!("Expected name, got {other:?}"),
1166
- self.convert_range(other.range()),
1167
- )),
1168
- }
1169
- }
1170
-
1171
- /// Parses a chain comparison expression like `a < b < c < d`.
1172
- ///
1173
- /// Chain comparisons evaluate each intermediate value only once and short-circuit
1174
- /// on the first false result. This creates an `Expr::ChainCmp` node which is
1175
- /// compiled to bytecode using stack manipulation (Dup, Rot) rather than
1176
- /// temporary variables, avoiding namespace pollution.
1177
- fn parse_chain_comparison(
1178
- &mut self,
1179
- left: AstExpr,
1180
- ops: Vec<CmpOp>,
1181
- comparators: Vec<AstExpr>,
1182
- position: CodeRange,
1183
- ) -> Result<ExprLoc, ParseError> {
1184
- let left_expr = self.parse_expression(left)?;
1185
- let comparisons = ops
1186
- .into_iter()
1187
- .zip(comparators)
1188
- .map(|(op, cmp)| Ok((convert_compare_op(op), self.parse_expression(cmp)?)))
1189
- .collect::<Result<Vec<_>, ParseError>>()?;
1190
-
1191
- Ok(ExprLoc::new(
1192
- position,
1193
- Expr::ChainCmp {
1194
- left: Box::new(left_expr),
1195
- comparisons,
1196
- },
1197
- ))
1198
- }
1199
-
1200
- /// Parses an unpack target - either a single identifier or a nested tuple.
1201
- ///
1202
- /// Handles patterns like `a` (single variable), `a, b` (flat tuple), or `(a, b), c` (nested).
1203
- /// Includes depth tracking to prevent stack overflow from deeply nested structures.
1204
- fn parse_unpack_target(&mut self, ast: AstExpr) -> Result<UnpackTarget, ParseError> {
1205
- self.decr_depth_remaining(|| ast.range())?;
1206
- let result = self.parse_unpack_target_impl(ast);
1207
- self.depth_remaining += 1;
1208
- result
1209
- }
1210
-
1211
- fn parse_unpack_target_impl(&mut self, ast: AstExpr) -> Result<UnpackTarget, ParseError> {
1212
- match ast {
1213
- AstExpr::Name(ast::ExprName { id, range, .. }) => Ok(UnpackTarget::Name(self.identifier(&id, range))),
1214
- AstExpr::Tuple(ast::ExprTuple { elts, range, .. }) => {
1215
- let position = self.convert_range(range);
1216
- let targets = elts
1217
- .into_iter()
1218
- .map(|e| self.parse_unpack_target(e)) // Recursive call for nested tuples
1219
- .collect::<Result<Vec<_>, _>>()?;
1220
- if targets.is_empty() {
1221
- return Err(ParseError::syntax("empty tuple in unpack target", position));
1222
- }
1223
- // Validate at most one starred target
1224
- let starred_count = targets.iter().filter(|t| matches!(t, UnpackTarget::Starred(_))).count();
1225
- if starred_count > 1 {
1226
- return Err(ParseError::syntax(
1227
- "multiple starred expressions in assignment",
1228
- position,
1229
- ));
1230
- }
1231
- Ok(UnpackTarget::Tuple { targets, position })
1232
- }
1233
- AstExpr::Starred(ast::ExprStarred { value, range, .. }) => {
1234
- // Starred target must be a simple name
1235
- match *value {
1236
- AstExpr::Name(ast::ExprName { id, range, .. }) => {
1237
- Ok(UnpackTarget::Starred(self.identifier(&id, range)))
1238
- }
1239
- _ => Err(ParseError::syntax(
1240
- "starred assignment target must be a name",
1241
- self.convert_range(range),
1242
- )),
1243
- }
1244
- }
1245
- AstExpr::List(ast::ExprList { elts, range, .. }) => {
1246
- // List unpacking target [a, b, *rest] - same as tuple
1247
- let position = self.convert_range(range);
1248
- let targets = elts
1249
- .into_iter()
1250
- .map(|e| self.parse_unpack_target(e))
1251
- .collect::<Result<Vec<_>, _>>()?;
1252
- if targets.is_empty() {
1253
- return Err(ParseError::syntax("empty list in unpack target", position));
1254
- }
1255
- // Validate at most one starred target
1256
- let starred_count = targets.iter().filter(|t| matches!(t, UnpackTarget::Starred(_))).count();
1257
- if starred_count > 1 {
1258
- return Err(ParseError::syntax(
1259
- "multiple starred expressions in assignment",
1260
- position,
1261
- ));
1262
- }
1263
- Ok(UnpackTarget::Tuple { targets, position })
1264
- }
1265
- other => Err(ParseError::syntax(
1266
- format!("invalid unpacking target: {other:?}"),
1267
- self.convert_range(other.range()),
1268
- )),
1269
- }
1270
- }
1271
-
1272
- fn identifier(&mut self, id: &Name, range: TextRange) -> Identifier {
1273
- let string_id = self.interner.intern(id);
1274
- Identifier::new(string_id, self.convert_range(range))
1275
- }
1276
-
1277
- /// Parses function parameters with optional default values.
1278
- ///
1279
- /// Handles parameters like `a`, `b=10`, `c=None` by extracting the parameter
1280
- /// name and parsing any default expression. Default expressions are stored
1281
- /// as unevaluated AST and will be evaluated during the prepare phase.
1282
- fn parse_params_with_defaults(&mut self, params: &[ParameterWithDefault]) -> Result<Vec<ParsedParam>, ParseError> {
1283
- params
1284
- .iter()
1285
- .map(|p| {
1286
- let name = self.interner.intern(&p.parameter.name.id);
1287
- let default = match &p.default {
1288
- Some(expr) => Some(self.parse_expression((**expr).clone())?),
1289
- None => None,
1290
- };
1291
- Ok(ParsedParam { name, default })
1292
- })
1293
- .collect()
1294
- }
1295
-
1296
- /// Parses comprehension generators (the `for ... in ... if ...` clauses).
1297
- ///
1298
- /// Each generator represents one `for` clause with zero or more `if` filters.
1299
- /// Multiple generators create nested iteration. Supports both single identifiers
1300
- /// (`for x in ...`) and tuple unpacking (`for x, y in ...`).
1301
- fn parse_comprehension_generators(
1302
- &mut self,
1303
- generators: Vec<ast::Comprehension>,
1304
- ) -> Result<Vec<Comprehension>, ParseError> {
1305
- generators
1306
- .into_iter()
1307
- .map(|comp| {
1308
- if comp.is_async {
1309
- return Err(ParseError::not_implemented(
1310
- "async comprehensions",
1311
- self.convert_range(comp.range),
1312
- ));
1313
- }
1314
- let target = self.parse_unpack_target(comp.target)?;
1315
- let iter = self.parse_expression(comp.iter)?;
1316
- let ifs = comp
1317
- .ifs
1318
- .into_iter()
1319
- .map(|cond| self.parse_expression(cond))
1320
- .collect::<Result<Vec<_>, _>>()?;
1321
- Ok(Comprehension { target, iter, ifs })
1322
- })
1323
- .collect()
1324
- }
1325
-
1326
- /// Parses an f-string value into expression parts.
1327
- ///
1328
- /// F-strings in ruff AST are represented as `FStringValue` containing
1329
- /// `FStringPart`s, which can be either literal strings or `FString`
1330
- /// interpolated sections. Each `FString` contains `InterpolatedStringElements`.
1331
- fn parse_fstring(&mut self, value: &ast::FStringValue, range: TextRange) -> Result<ExprLoc, ParseError> {
1332
- let mut parts = Vec::new();
1333
-
1334
- for fstring_part in value {
1335
- match fstring_part {
1336
- ast::FStringPart::Literal(lit) => {
1337
- // Literal string segment - intern for use at runtime
1338
- let processed = lit.value.to_string();
1339
- if !processed.is_empty() {
1340
- let string_id = self.interner.intern(&processed);
1341
- parts.push(FStringPart::Literal(string_id));
1342
- }
1343
- }
1344
- ast::FStringPart::FString(fstring) => {
1345
- // Interpolated f-string section
1346
- for element in &fstring.elements {
1347
- let part = self.parse_fstring_element(element)?;
1348
- parts.push(part);
1349
- }
1350
- }
1351
- }
1352
- }
1353
-
1354
- // Optimization: if only one literal part, return as simple string literal
1355
- if parts.len() == 1
1356
- && let FStringPart::Literal(string_id) = parts[0]
1357
- {
1358
- return Ok(ExprLoc::new(
1359
- self.convert_range(range),
1360
- Expr::Literal(Literal::Str(string_id)),
1361
- ));
1362
- }
1363
-
1364
- Ok(ExprLoc::new(self.convert_range(range), Expr::FString(parts)))
1365
- }
1366
-
1367
- /// Parses a single f-string element (literal or interpolation).
1368
- fn parse_fstring_element(&mut self, element: &InterpolatedStringElement) -> Result<FStringPart, ParseError> {
1369
- match element {
1370
- InterpolatedStringElement::Literal(lit) => {
1371
- // Intern the literal string for use at runtime
1372
- let processed = lit.value.to_string();
1373
- let string_id = self.interner.intern(&processed);
1374
- Ok(FStringPart::Literal(string_id))
1375
- }
1376
- InterpolatedStringElement::Interpolation(interp) => {
1377
- let expr = Box::new(self.parse_expression((*interp.expression).clone())?);
1378
- let conversion = convert_conversion_flag(interp.conversion);
1379
- let format_spec = match &interp.format_spec {
1380
- Some(spec) => Some(self.parse_format_spec(spec)?),
1381
- None => None,
1382
- };
1383
- // Extract debug prefix for `=` specifier (e.g., f'{a=}' -> "a=")
1384
- let debug_prefix = interp.debug_text.as_ref().map(|dt| {
1385
- let expr_text = &self.code[interp.expression.range()];
1386
- self.interner
1387
- .intern(&format!("{}{}{}", dt.leading, expr_text, dt.trailing))
1388
- });
1389
- Ok(FStringPart::Interpolation {
1390
- expr,
1391
- conversion,
1392
- format_spec,
1393
- debug_prefix,
1394
- })
1395
- }
1396
- }
1397
- }
1398
-
1399
- /// Parses a format specification, which may contain nested interpolations.
1400
- ///
1401
- /// For static specs (no interpolations), parses the format string into a
1402
- /// `ParsedFormatSpec` at parse time to avoid runtime parsing overhead.
1403
- fn parse_format_spec(&mut self, spec: &ast::InterpolatedStringFormatSpec) -> Result<FormatSpec, ParseError> {
1404
- let mut parts = Vec::new();
1405
- let mut has_interpolation = false;
1406
-
1407
- for element in &spec.elements {
1408
- match element {
1409
- InterpolatedStringElement::Literal(lit) => {
1410
- // Intern the literal string
1411
- let processed = lit.value.to_string();
1412
- let string_id = self.interner.intern(&processed);
1413
- parts.push(FStringPart::Literal(string_id));
1414
- }
1415
- InterpolatedStringElement::Interpolation(interp) => {
1416
- has_interpolation = true;
1417
- let expr = Box::new(self.parse_expression((*interp.expression).clone())?);
1418
- let conversion = convert_conversion_flag(interp.conversion);
1419
- // Format specs within format specs are not allowed in Python,
1420
- // and debug_prefix doesn't apply to nested interpolations
1421
- parts.push(FStringPart::Interpolation {
1422
- expr,
1423
- conversion,
1424
- format_spec: None,
1425
- debug_prefix: None,
1426
- });
1427
- }
1428
- }
1429
- }
1430
-
1431
- if has_interpolation {
1432
- Ok(FormatSpec::Dynamic(parts))
1433
- } else {
1434
- // Combine all literal parts into a single static string and parse at parse time
1435
- let static_spec: String = parts
1436
- .into_iter()
1437
- .filter_map(|p| {
1438
- if let FStringPart::Literal(string_id) = p {
1439
- Some(self.interner.get_str(string_id).to_owned())
1440
- } else {
1441
- None
1442
- }
1443
- })
1444
- .collect();
1445
- let parsed = static_spec.parse().map_err(|spec_str| {
1446
- ParseError::syntax(
1447
- format!("Invalid format specifier '{spec_str}'"),
1448
- self.convert_range(spec.range),
1449
- )
1450
- })?;
1451
- Ok(FormatSpec::Static(parsed))
1452
- }
1453
- }
1454
-
1455
- fn convert_range(&self, range: TextRange) -> CodeRange {
1456
- let start = range.start().into();
1457
- let (start_line_no, start_line_start, _) = self.index_to_position(start);
1458
- let start = CodeLoc::new(start_line_no, start - start_line_start);
1459
-
1460
- let end = range.end().into();
1461
- let (end_line_no, end_line_start, _) = self.index_to_position(end);
1462
- let end = CodeLoc::new(end_line_no, end - end_line_start);
1463
-
1464
- // Store line number for single-line ranges, None for multi-line
1465
- let preview_line = if start_line_no == end_line_no {
1466
- Some(u32::try_from(start_line_no).expect("line number exceeds u32"))
1467
- } else {
1468
- None
1469
- };
1470
-
1471
- CodeRange::new(self.filename_id, start, end, preview_line)
1472
- }
1473
-
1474
- fn index_to_position(&self, index: usize) -> (usize, usize, Option<usize>) {
1475
- let mut line_start = 0;
1476
- for (line_no, line_end) in self.line_ends.iter().enumerate() {
1477
- if index <= *line_end {
1478
- return (line_no, line_start, Some(*line_end));
1479
- }
1480
- line_start = *line_end + 1;
1481
- }
1482
- // Content after the last newline (file without trailing newline)
1483
- // line_ends.len() gives the correct 0-indexed line number
1484
- (self.line_ends.len(), line_start, None)
1485
- }
1486
-
1487
- /// Decrements the depth remaining for nested parentheses.
1488
- /// Returns an error if the depth remaining goes to zero.
1489
- fn decr_depth_remaining(&mut self, get_range: impl FnOnce() -> TextRange) -> Result<(), ParseError> {
1490
- if let Some(depth_remaining) = self.depth_remaining.checked_sub(1) {
1491
- self.depth_remaining = depth_remaining;
1492
- Ok(())
1493
- } else {
1494
- let position = self.convert_range(get_range());
1495
- Err(ParseError::syntax("too many nested parentheses", position))
1496
- }
1497
- }
1498
- }
1499
-
1500
- fn first<T: fmt::Debug>(v: Vec<T>, position: CodeRange) -> Result<T, ParseError> {
1501
- if v.len() == 1 {
1502
- v.into_iter()
1503
- .next()
1504
- .ok_or_else(|| ParseError::syntax("Expected 1 element, got 0", position))
1505
- } else {
1506
- Err(ParseError::syntax(
1507
- format!("Expected 1 element, got {} (raw: {v:?})", v.len()),
1508
- position,
1509
- ))
1510
- }
1511
- }
1512
-
1513
- fn convert_op(op: AstOperator) -> Operator {
1514
- match op {
1515
- AstOperator::Add => Operator::Add,
1516
- AstOperator::Sub => Operator::Sub,
1517
- AstOperator::Mult => Operator::Mult,
1518
- AstOperator::MatMult => Operator::MatMult,
1519
- AstOperator::Div => Operator::Div,
1520
- AstOperator::Mod => Operator::Mod,
1521
- AstOperator::Pow => Operator::Pow,
1522
- AstOperator::LShift => Operator::LShift,
1523
- AstOperator::RShift => Operator::RShift,
1524
- AstOperator::BitOr => Operator::BitOr,
1525
- AstOperator::BitXor => Operator::BitXor,
1526
- AstOperator::BitAnd => Operator::BitAnd,
1527
- AstOperator::FloorDiv => Operator::FloorDiv,
1528
- }
1529
- }
1530
-
1531
- fn convert_bool_op(op: BoolOp) -> Operator {
1532
- match op {
1533
- BoolOp::And => Operator::And,
1534
- BoolOp::Or => Operator::Or,
1535
- }
1536
- }
1537
-
1538
- fn convert_compare_op(op: CmpOp) -> CmpOperator {
1539
- match op {
1540
- CmpOp::Eq => CmpOperator::Eq,
1541
- CmpOp::NotEq => CmpOperator::NotEq,
1542
- CmpOp::Lt => CmpOperator::Lt,
1543
- CmpOp::LtE => CmpOperator::LtE,
1544
- CmpOp::Gt => CmpOperator::Gt,
1545
- CmpOp::GtE => CmpOperator::GtE,
1546
- CmpOp::Is => CmpOperator::Is,
1547
- CmpOp::IsNot => CmpOperator::IsNot,
1548
- CmpOp::In => CmpOperator::In,
1549
- CmpOp::NotIn => CmpOperator::NotIn,
1550
- }
1551
- }
1552
-
1553
- /// Converts ruff's ConversionFlag to our ConversionFlag.
1554
- fn convert_conversion_flag(flag: RuffConversionFlag) -> ConversionFlag {
1555
- match flag {
1556
- RuffConversionFlag::None => ConversionFlag::None,
1557
- RuffConversionFlag::Str => ConversionFlag::Str,
1558
- RuffConversionFlag::Repr => ConversionFlag::Repr,
1559
- RuffConversionFlag::Ascii => ConversionFlag::Ascii,
1560
- }
1561
- }
1562
-
1563
- /// Source code location information for error reporting.
1564
- ///
1565
- /// Contains filename (as StringId), line/column positions, and optionally a line number for
1566
- /// extracting the preview line from source during traceback formatting.
1567
- ///
1568
- /// To display the filename, the caller must provide access to the string storage.
1569
- #[derive(Clone, Copy, Default, Eq, PartialEq, Hash, serde::Serialize, serde::Deserialize)]
1570
- pub struct CodeRange {
1571
- /// Interned filename ID - look up in Interns to get the actual string.
1572
- pub filename: StringId,
1573
- /// Line number (0-indexed) for extracting preview from source. None if range spans multiple lines.
1574
- preview_line: Option<u32>,
1575
- start: CodeLoc,
1576
- end: CodeLoc,
1577
- }
1578
-
1579
- /// Custom Debug implementation to make displaying code much less verbose.
1580
- impl fmt::Debug for CodeRange {
1581
- fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
1582
- write!(
1583
- f,
1584
- "CodeRange{{filename: {:?}, start: {:?}, end: {:?}}}",
1585
- self.filename, self.start, self.end
1586
- )
1587
- }
1588
- }
1589
-
1590
- impl CodeRange {
1591
- fn new(filename: StringId, start: CodeLoc, end: CodeLoc, preview_line: Option<u32>) -> Self {
1592
- Self {
1593
- filename,
1594
- preview_line,
1595
- start,
1596
- end,
1597
- }
1598
- }
1599
-
1600
- /// Returns the start position.
1601
- #[must_use]
1602
- pub fn start(&self) -> CodeLoc {
1603
- self.start
1604
- }
1605
-
1606
- /// Returns the end position.
1607
- #[must_use]
1608
- pub fn end(&self) -> CodeLoc {
1609
- self.end
1610
- }
1611
-
1612
- /// Returns the preview line number (0-indexed) if available.
1613
- #[must_use]
1614
- pub fn preview_line_number(&self) -> Option<u32> {
1615
- self.preview_line
1616
- }
1617
- }
1618
-
1619
- /// Errors that can occur during parsing or preparation of Python code.
1620
- #[derive(Debug, Clone)]
1621
- pub enum ParseError {
1622
- /// Error in syntax
1623
- Syntax {
1624
- msg: Cow<'static, str>,
1625
- position: CodeRange,
1626
- },
1627
- /// Missing feature from Monty, we hope to implement in the future.
1628
- /// Message gets prefixed with "The monty syntax parser does not yet support ".
1629
- NotImplemented {
1630
- msg: Cow<'static, str>,
1631
- position: CodeRange,
1632
- },
1633
- /// Missing feature with a custom full message (no prefix added).
1634
- NotSupported {
1635
- msg: Cow<'static, str>,
1636
- position: CodeRange,
1637
- },
1638
- /// Import error (e.g., relative imports without a package).
1639
- Import {
1640
- msg: Cow<'static, str>,
1641
- position: CodeRange,
1642
- },
1643
- }
1644
-
1645
- impl ParseError {
1646
- fn not_implemented(msg: impl Into<Cow<'static, str>>, position: CodeRange) -> Self {
1647
- Self::NotImplemented {
1648
- msg: msg.into(),
1649
- position,
1650
- }
1651
- }
1652
-
1653
- fn not_supported(msg: impl Into<Cow<'static, str>>, position: CodeRange) -> Self {
1654
- Self::NotSupported {
1655
- msg: msg.into(),
1656
- position,
1657
- }
1658
- }
1659
-
1660
- fn import_error(msg: impl Into<Cow<'static, str>>, position: CodeRange) -> Self {
1661
- Self::Import {
1662
- msg: msg.into(),
1663
- position,
1664
- }
1665
- }
1666
-
1667
- pub(crate) fn syntax(msg: impl Into<Cow<'static, str>>, position: CodeRange) -> Self {
1668
- Self::Syntax {
1669
- msg: msg.into(),
1670
- position,
1671
- }
1672
- }
1673
- }
1674
-
1675
- impl ParseError {
1676
- pub fn into_python_exc(self, filename: &str, source: &str) -> MontyException {
1677
- match self {
1678
- Self::Syntax { msg, position } => MontyException::new_full(
1679
- ExcType::SyntaxError,
1680
- Some(msg.into_owned()),
1681
- vec![StackFrame::from_position_syntax_error(position, filename, source)],
1682
- ),
1683
- Self::NotImplemented { msg, position } => MontyException::new_full(
1684
- ExcType::NotImplementedError,
1685
- Some(format!("The monty syntax parser does not yet support {msg}")),
1686
- vec![StackFrame::from_position(position, filename, source)],
1687
- ),
1688
- Self::NotSupported { msg, position } => MontyException::new_full(
1689
- ExcType::NotImplementedError,
1690
- Some(msg.into_owned()),
1691
- vec![StackFrame::from_position(position, filename, source)],
1692
- ),
1693
- Self::Import { msg, position } => MontyException::new_full(
1694
- ExcType::ImportError,
1695
- Some(msg.into_owned()),
1696
- vec![StackFrame::from_position_no_caret(position, filename, source)],
1697
- ),
1698
- }
1699
- }
1700
- }
1701
-
1702
- /// Parses an integer literal string into a `BigInt`, handling radix prefixes and underscores.
1703
- ///
1704
- /// Supports Python integer literal formats:
1705
- /// - Decimal: `123`, `1_000_000`
1706
- /// - Hexadecimal: `0x1a2b`, `0X1A2B`
1707
- /// - Octal: `0o777`, `0O777`
1708
- /// - Binary: `0b1010`, `0B1010`
1709
- ///
1710
- /// Returns `None` if the string cannot be parsed.
1711
- fn parse_int_literal(s: &str) -> Option<BigInt> {
1712
- // Remove underscores (Python allows them as digit separators)
1713
- let cleaned: String = s.chars().filter(|c| *c != '_').collect();
1714
- let cleaned = cleaned.as_str();
1715
-
1716
- // Detect radix from prefix
1717
- if cleaned.len() >= 2 {
1718
- let prefix = &cleaned[..2];
1719
- let digits = &cleaned[2..];
1720
- match prefix.to_ascii_lowercase().as_str() {
1721
- "0x" => return BigInt::parse_bytes(digits.as_bytes(), 16),
1722
- "0o" => return BigInt::parse_bytes(digits.as_bytes(), 8),
1723
- "0b" => return BigInt::parse_bytes(digits.as_bytes(), 2),
1724
- _ => {}
1725
- }
1726
- }
1727
-
1728
- // Default to decimal
1729
- cleaned.parse::<BigInt>().ok()
1730
- }