pythinker-code 0.8.0__py3-none-any.whl

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 (790) hide show
  1. pythinker_code/CHANGELOG.md +60 -0
  2. pythinker_code/__init__.py +0 -0
  3. pythinker_code/__main__.py +97 -0
  4. pythinker_code/acp/AGENTS.md +93 -0
  5. pythinker_code/acp/__init__.py +13 -0
  6. pythinker_code/acp/convert.py +128 -0
  7. pythinker_code/acp/host.py +301 -0
  8. pythinker_code/acp/mcp.py +46 -0
  9. pythinker_code/acp/server.py +497 -0
  10. pythinker_code/acp/session.py +502 -0
  11. pythinker_code/acp/tools.py +174 -0
  12. pythinker_code/acp/types.py +13 -0
  13. pythinker_code/acp/version.py +45 -0
  14. pythinker_code/agents/default/agent.yaml +55 -0
  15. pythinker_code/agents/default/code_reviewer.yaml +47 -0
  16. pythinker_code/agents/default/coder.yaml +42 -0
  17. pythinker_code/agents/default/debugger.yaml +35 -0
  18. pythinker_code/agents/default/explore.yaml +59 -0
  19. pythinker_code/agents/default/implementer.yaml +46 -0
  20. pythinker_code/agents/default/plan.yaml +42 -0
  21. pythinker_code/agents/default/review.yaml +47 -0
  22. pythinker_code/agents/default/security_reviewer.yaml +37 -0
  23. pythinker_code/agents/default/system.md +192 -0
  24. pythinker_code/agents/default/verifier.yaml +46 -0
  25. pythinker_code/agents/okabe/agent.yaml +22 -0
  26. pythinker_code/agentspec.py +163 -0
  27. pythinker_code/app.py +847 -0
  28. pythinker_code/approval_runtime/__init__.py +29 -0
  29. pythinker_code/approval_runtime/models.py +42 -0
  30. pythinker_code/approval_runtime/runtime.py +235 -0
  31. pythinker_code/auth/__init__.py +25 -0
  32. pythinker_code/auth/anthropic_direct.py +207 -0
  33. pythinker_code/auth/deepseek.py +192 -0
  34. pythinker_code/auth/github_feedback.py +228 -0
  35. pythinker_code/auth/lm_studio.py +418 -0
  36. pythinker_code/auth/minimax.py +203 -0
  37. pythinker_code/auth/oauth.py +1145 -0
  38. pythinker_code/auth/ollama.py +293 -0
  39. pythinker_code/auth/openai.py +783 -0
  40. pythinker_code/auth/opencode_go.py +203 -0
  41. pythinker_code/auth/openrouter.py +225 -0
  42. pythinker_code/auth/platforms.py +475 -0
  43. pythinker_code/background/__init__.py +36 -0
  44. pythinker_code/background/agent_runner.py +231 -0
  45. pythinker_code/background/ids.py +19 -0
  46. pythinker_code/background/manager.py +668 -0
  47. pythinker_code/background/models.py +118 -0
  48. pythinker_code/background/store.py +243 -0
  49. pythinker_code/background/summary.py +66 -0
  50. pythinker_code/background/worker.py +209 -0
  51. pythinker_code/cli/__init__.py +1326 -0
  52. pythinker_code/cli/__main__.py +19 -0
  53. pythinker_code/cli/_lazy_group.py +268 -0
  54. pythinker_code/cli/debug.py +11 -0
  55. pythinker_code/cli/export.py +322 -0
  56. pythinker_code/cli/info.py +62 -0
  57. pythinker_code/cli/mcp.py +362 -0
  58. pythinker_code/cli/plugin.py +351 -0
  59. pythinker_code/cli/review.py +74 -0
  60. pythinker_code/cli/secscan.py +11 -0
  61. pythinker_code/cli/security_scan.py +35 -0
  62. pythinker_code/cli/toad.py +74 -0
  63. pythinker_code/cli/update.py +26 -0
  64. pythinker_code/cli/vis.py +38 -0
  65. pythinker_code/cli/web.py +80 -0
  66. pythinker_code/config.py +511 -0
  67. pythinker_code/constant.py +33 -0
  68. pythinker_code/events.py +106 -0
  69. pythinker_code/exception.py +43 -0
  70. pythinker_code/extensions.py +151 -0
  71. pythinker_code/hooks/__init__.py +4 -0
  72. pythinker_code/hooks/config.py +34 -0
  73. pythinker_code/hooks/engine.py +383 -0
  74. pythinker_code/hooks/events.py +190 -0
  75. pythinker_code/hooks/runner.py +92 -0
  76. pythinker_code/llm.py +441 -0
  77. pythinker_code/metadata.py +79 -0
  78. pythinker_code/notifications/__init__.py +33 -0
  79. pythinker_code/notifications/llm.py +77 -0
  80. pythinker_code/notifications/manager.py +145 -0
  81. pythinker_code/notifications/models.py +50 -0
  82. pythinker_code/notifications/notifier.py +41 -0
  83. pythinker_code/notifications/store.py +118 -0
  84. pythinker_code/notifications/wire.py +21 -0
  85. pythinker_code/plugin/__init__.py +124 -0
  86. pythinker_code/plugin/manager.py +166 -0
  87. pythinker_code/plugin/tool.py +173 -0
  88. pythinker_code/prompt_templates.py +181 -0
  89. pythinker_code/prompts/__init__.py +6 -0
  90. pythinker_code/prompts/compact.md +73 -0
  91. pythinker_code/prompts/init.md +21 -0
  92. pythinker_code/py.typed +0 -0
  93. pythinker_code/session.py +319 -0
  94. pythinker_code/session_fork.py +325 -0
  95. pythinker_code/session_state.py +132 -0
  96. pythinker_code/share.py +14 -0
  97. pythinker_code/skill/__init__.py +727 -0
  98. pythinker_code/skill/flow/__init__.py +99 -0
  99. pythinker_code/skill/flow/d2.py +482 -0
  100. pythinker_code/skill/flow/mermaid.py +266 -0
  101. pythinker_code/skills/pythinker-code-help/SKILL.md +54 -0
  102. pythinker_code/skills/skill-creator/SKILL.md +367 -0
  103. pythinker_code/soul/__init__.py +304 -0
  104. pythinker_code/soul/agent.py +552 -0
  105. pythinker_code/soul/approval.py +267 -0
  106. pythinker_code/soul/btw.py +220 -0
  107. pythinker_code/soul/compaction.py +189 -0
  108. pythinker_code/soul/context.py +339 -0
  109. pythinker_code/soul/denwarenji.py +39 -0
  110. pythinker_code/soul/dynamic_injection.py +84 -0
  111. pythinker_code/soul/dynamic_injections/__init__.py +0 -0
  112. pythinker_code/soul/dynamic_injections/auto_mode.py +72 -0
  113. pythinker_code/soul/dynamic_injections/plan_mode.py +239 -0
  114. pythinker_code/soul/message.py +92 -0
  115. pythinker_code/soul/permission.py +368 -0
  116. pythinker_code/soul/pythinkersoul.py +1763 -0
  117. pythinker_code/soul/slash.py +340 -0
  118. pythinker_code/soul/toolset.py +826 -0
  119. pythinker_code/subagents/__init__.py +21 -0
  120. pythinker_code/subagents/builder.py +43 -0
  121. pythinker_code/subagents/core.py +86 -0
  122. pythinker_code/subagents/discovery.py +234 -0
  123. pythinker_code/subagents/git_context.py +172 -0
  124. pythinker_code/subagents/models.py +56 -0
  125. pythinker_code/subagents/output.py +71 -0
  126. pythinker_code/subagents/registry.py +28 -0
  127. pythinker_code/subagents/runner.py +442 -0
  128. pythinker_code/subagents/store.py +200 -0
  129. pythinker_code/telemetry/__init__.py +217 -0
  130. pythinker_code/telemetry/config.py +113 -0
  131. pythinker_code/telemetry/crash.py +191 -0
  132. pythinker_code/telemetry/errors.py +113 -0
  133. pythinker_code/telemetry/metrics.py +208 -0
  134. pythinker_code/telemetry/otel.py +303 -0
  135. pythinker_code/telemetry/sentry.py +212 -0
  136. pythinker_code/telemetry/sink.py +189 -0
  137. pythinker_code/tools/AGENTS.md +6 -0
  138. pythinker_code/tools/__init__.py +105 -0
  139. pythinker_code/tools/agent/__init__.py +326 -0
  140. pythinker_code/tools/agent/description.md +65 -0
  141. pythinker_code/tools/ask_user/__init__.py +162 -0
  142. pythinker_code/tools/ask_user/description.md +19 -0
  143. pythinker_code/tools/background/__init__.py +318 -0
  144. pythinker_code/tools/background/list.md +10 -0
  145. pythinker_code/tools/background/output.md +11 -0
  146. pythinker_code/tools/background/stop.md +8 -0
  147. pythinker_code/tools/display.py +46 -0
  148. pythinker_code/tools/dmail/__init__.py +38 -0
  149. pythinker_code/tools/dmail/dmail.md +17 -0
  150. pythinker_code/tools/file/__init__.py +31 -0
  151. pythinker_code/tools/file/glob.md +17 -0
  152. pythinker_code/tools/file/glob.py +163 -0
  153. pythinker_code/tools/file/grep.md +6 -0
  154. pythinker_code/tools/file/grep_local.py +904 -0
  155. pythinker_code/tools/file/plan_mode.py +45 -0
  156. pythinker_code/tools/file/read.md +16 -0
  157. pythinker_code/tools/file/read.py +303 -0
  158. pythinker_code/tools/file/read_media.md +24 -0
  159. pythinker_code/tools/file/read_media.py +220 -0
  160. pythinker_code/tools/file/replace.md +7 -0
  161. pythinker_code/tools/file/replace.py +204 -0
  162. pythinker_code/tools/file/utils.py +257 -0
  163. pythinker_code/tools/file/write.md +5 -0
  164. pythinker_code/tools/file/write.py +186 -0
  165. pythinker_code/tools/plan/__init__.py +362 -0
  166. pythinker_code/tools/plan/description.md +29 -0
  167. pythinker_code/tools/plan/enter.py +193 -0
  168. pythinker_code/tools/plan/enter_description.md +35 -0
  169. pythinker_code/tools/plan/handoff.py +69 -0
  170. pythinker_code/tools/plan/heroes.py +277 -0
  171. pythinker_code/tools/shell/__init__.py +263 -0
  172. pythinker_code/tools/shell/bash.md +35 -0
  173. pythinker_code/tools/shell/powershell.md +30 -0
  174. pythinker_code/tools/test.py +55 -0
  175. pythinker_code/tools/think/__init__.py +21 -0
  176. pythinker_code/tools/think/think.md +1 -0
  177. pythinker_code/tools/todo/__init__.py +168 -0
  178. pythinker_code/tools/todo/set_todo_list.md +23 -0
  179. pythinker_code/tools/utils.py +200 -0
  180. pythinker_code/tools/web/__init__.py +4 -0
  181. pythinker_code/tools/web/fetch.md +1 -0
  182. pythinker_code/tools/web/fetch.py +261 -0
  183. pythinker_code/tools/web/search.md +1 -0
  184. pythinker_code/tools/web/search.py +163 -0
  185. pythinker_code/ui/__init__.py +0 -0
  186. pythinker_code/ui/acp/__init__.py +99 -0
  187. pythinker_code/ui/print/__init__.py +474 -0
  188. pythinker_code/ui/print/visualize.py +185 -0
  189. pythinker_code/ui/shell/__init__.py +1806 -0
  190. pythinker_code/ui/shell/components/__init__.py +110 -0
  191. pythinker_code/ui/shell/components/base.py +25 -0
  192. pythinker_code/ui/shell/components/bash_execution.py +249 -0
  193. pythinker_code/ui/shell/components/bordered_loader.py +62 -0
  194. pythinker_code/ui/shell/components/diff.py +308 -0
  195. pythinker_code/ui/shell/components/footer.py +231 -0
  196. pythinker_code/ui/shell/components/key_hints.py +27 -0
  197. pythinker_code/ui/shell/components/messages.py +152 -0
  198. pythinker_code/ui/shell/components/render_utils.py +198 -0
  199. pythinker_code/ui/shell/components/settings_list.py +369 -0
  200. pythinker_code/ui/shell/components/special_messages.py +125 -0
  201. pythinker_code/ui/shell/components/tool_execution.py +261 -0
  202. pythinker_code/ui/shell/console.py +109 -0
  203. pythinker_code/ui/shell/debug.py +190 -0
  204. pythinker_code/ui/shell/echo.py +30 -0
  205. pythinker_code/ui/shell/export_import.py +117 -0
  206. pythinker_code/ui/shell/keyboard.py +300 -0
  207. pythinker_code/ui/shell/keymap.py +84 -0
  208. pythinker_code/ui/shell/mcp_status.py +112 -0
  209. pythinker_code/ui/shell/model_picker.py +318 -0
  210. pythinker_code/ui/shell/oauth.py +273 -0
  211. pythinker_code/ui/shell/placeholders.py +578 -0
  212. pythinker_code/ui/shell/prompt.py +2888 -0
  213. pythinker_code/ui/shell/replay.py +215 -0
  214. pythinker_code/ui/shell/selector.py +364 -0
  215. pythinker_code/ui/shell/selectors/__init__.py +38 -0
  216. pythinker_code/ui/shell/selectors/extension.py +37 -0
  217. pythinker_code/ui/shell/selectors/oauth.py +63 -0
  218. pythinker_code/ui/shell/selectors/settings.py +349 -0
  219. pythinker_code/ui/shell/selectors/show_images.py +29 -0
  220. pythinker_code/ui/shell/selectors/theme.py +28 -0
  221. pythinker_code/ui/shell/selectors/thinking.py +42 -0
  222. pythinker_code/ui/shell/session_picker.py +227 -0
  223. pythinker_code/ui/shell/setup.py +212 -0
  224. pythinker_code/ui/shell/slash.py +1433 -0
  225. pythinker_code/ui/shell/spinner_words.py +222 -0
  226. pythinker_code/ui/shell/startup.py +32 -0
  227. pythinker_code/ui/shell/task_browser.py +486 -0
  228. pythinker_code/ui/shell/tool_renderers/__init__.py +197 -0
  229. pythinker_code/ui/shell/tool_renderers/_render_utils.py +168 -0
  230. pythinker_code/ui/shell/tool_renderers/agent.py +140 -0
  231. pythinker_code/ui/shell/tool_renderers/ask_user.py +93 -0
  232. pythinker_code/ui/shell/tool_renderers/background.py +144 -0
  233. pythinker_code/ui/shell/tool_renderers/bash.py +103 -0
  234. pythinker_code/ui/shell/tool_renderers/edit.py +163 -0
  235. pythinker_code/ui/shell/tool_renderers/find.py +81 -0
  236. pythinker_code/ui/shell/tool_renderers/generic.py +60 -0
  237. pythinker_code/ui/shell/tool_renderers/grep.py +98 -0
  238. pythinker_code/ui/shell/tool_renderers/plan.py +98 -0
  239. pythinker_code/ui/shell/tool_renderers/read.py +103 -0
  240. pythinker_code/ui/shell/tool_renderers/think.py +66 -0
  241. pythinker_code/ui/shell/tool_renderers/todo.py +164 -0
  242. pythinker_code/ui/shell/tool_renderers/web.py +128 -0
  243. pythinker_code/ui/shell/tool_renderers/write.py +102 -0
  244. pythinker_code/ui/shell/update.py +352 -0
  245. pythinker_code/ui/shell/usage.py +291 -0
  246. pythinker_code/ui/shell/usage_adapters/__init__.py +50 -0
  247. pythinker_code/ui/shell/usage_adapters/anthropic_admin.py +233 -0
  248. pythinker_code/ui/shell/usage_adapters/base.py +72 -0
  249. pythinker_code/ui/shell/usage_adapters/deepseek.py +137 -0
  250. pythinker_code/ui/shell/usage_adapters/minimax.py +236 -0
  251. pythinker_code/ui/shell/usage_adapters/openai_admin.py +225 -0
  252. pythinker_code/ui/shell/usage_adapters/openai_chatgpt.py +241 -0
  253. pythinker_code/ui/shell/usage_adapters/opencode_go.py +232 -0
  254. pythinker_code/ui/shell/usage_adapters/openrouter.py +105 -0
  255. pythinker_code/ui/shell/usage_adapters/pythinker.py +189 -0
  256. pythinker_code/ui/shell/usage_adapters/pythinker_ai.py +50 -0
  257. pythinker_code/ui/shell/usage_render.py +150 -0
  258. pythinker_code/ui/shell/visualize/__init__.py +165 -0
  259. pythinker_code/ui/shell/visualize/_approval_panel.py +539 -0
  260. pythinker_code/ui/shell/visualize/_blocks.py +802 -0
  261. pythinker_code/ui/shell/visualize/_btw_panel.py +227 -0
  262. pythinker_code/ui/shell/visualize/_input_router.py +48 -0
  263. pythinker_code/ui/shell/visualize/_interactive.py +531 -0
  264. pythinker_code/ui/shell/visualize/_live_view.py +891 -0
  265. pythinker_code/ui/shell/visualize/_question_panel.py +586 -0
  266. pythinker_code/ui/shell/visualize/_worklog.py +245 -0
  267. pythinker_code/ui/theme.py +395 -0
  268. pythinker_code/ui/tui_config.py +82 -0
  269. pythinker_code/usage_ratelimit_cache.py +175 -0
  270. pythinker_code/utils/__init__.py +0 -0
  271. pythinker_code/utils/aiohttp.py +24 -0
  272. pythinker_code/utils/aioqueue.py +72 -0
  273. pythinker_code/utils/broadcast.py +38 -0
  274. pythinker_code/utils/changelog.py +108 -0
  275. pythinker_code/utils/clipboard.py +246 -0
  276. pythinker_code/utils/datetime.py +64 -0
  277. pythinker_code/utils/diff.py +135 -0
  278. pythinker_code/utils/editor.py +91 -0
  279. pythinker_code/utils/environment.py +73 -0
  280. pythinker_code/utils/envvar.py +22 -0
  281. pythinker_code/utils/export.py +696 -0
  282. pythinker_code/utils/file_filter.py +375 -0
  283. pythinker_code/utils/frontmatter.py +70 -0
  284. pythinker_code/utils/io.py +27 -0
  285. pythinker_code/utils/logging.py +146 -0
  286. pythinker_code/utils/media_tags.py +29 -0
  287. pythinker_code/utils/message.py +24 -0
  288. pythinker_code/utils/path.py +199 -0
  289. pythinker_code/utils/proctitle.py +33 -0
  290. pythinker_code/utils/proxy.py +31 -0
  291. pythinker_code/utils/pyinstaller.py +45 -0
  292. pythinker_code/utils/rich/__init__.py +33 -0
  293. pythinker_code/utils/rich/columns.py +99 -0
  294. pythinker_code/utils/rich/diff_render.py +481 -0
  295. pythinker_code/utils/rich/markdown.py +935 -0
  296. pythinker_code/utils/rich/markdown_sample.md +108 -0
  297. pythinker_code/utils/rich/markdown_sample_short.md +2 -0
  298. pythinker_code/utils/rich/syntax.py +114 -0
  299. pythinker_code/utils/sensitive.py +54 -0
  300. pythinker_code/utils/server.py +121 -0
  301. pythinker_code/utils/signals.py +43 -0
  302. pythinker_code/utils/slashcmd.py +124 -0
  303. pythinker_code/utils/string.py +41 -0
  304. pythinker_code/utils/subprocess_env.py +83 -0
  305. pythinker_code/utils/term.py +168 -0
  306. pythinker_code/utils/typing.py +20 -0
  307. pythinker_code/vis/__init__.py +0 -0
  308. pythinker_code/vis/api/__init__.py +5 -0
  309. pythinker_code/vis/api/sessions.py +714 -0
  310. pythinker_code/vis/api/statistics.py +209 -0
  311. pythinker_code/vis/api/system.py +19 -0
  312. pythinker_code/vis/app.py +199 -0
  313. pythinker_code/vis/static/assets/highlighted-body-B3W2YXNL-CY1rtwrX.js +1 -0
  314. pythinker_code/vis/static/assets/index-DSRInNnm.css +1 -0
  315. pythinker_code/vis/static/assets/index-DgmTI2M_.js +185 -0
  316. pythinker_code/vis/static/assets/inter-cyrillic-ext-wght-normal-BOeWTOD4.woff2 +0 -0
  317. pythinker_code/vis/static/assets/inter-cyrillic-wght-normal-DqGufNeO.woff2 +0 -0
  318. pythinker_code/vis/static/assets/inter-greek-ext-wght-normal-DlzME5K_.woff2 +0 -0
  319. pythinker_code/vis/static/assets/inter-greek-wght-normal-CkhJZR-_.woff2 +0 -0
  320. pythinker_code/vis/static/assets/inter-latin-ext-wght-normal-DO1Apj_S.woff2 +0 -0
  321. pythinker_code/vis/static/assets/inter-latin-wght-normal-Dx4kXJAl.woff2 +0 -0
  322. pythinker_code/vis/static/assets/inter-vietnamese-wght-normal-CBcvBZtf.woff2 +0 -0
  323. pythinker_code/vis/static/index.html +17 -0
  324. pythinker_code/web/__init__.py +5 -0
  325. pythinker_code/web/api/__init__.py +15 -0
  326. pythinker_code/web/api/config.py +217 -0
  327. pythinker_code/web/api/open_in.py +233 -0
  328. pythinker_code/web/api/sessions.py +1256 -0
  329. pythinker_code/web/app.py +449 -0
  330. pythinker_code/web/auth.py +191 -0
  331. pythinker_code/web/models.py +98 -0
  332. pythinker_code/web/runner/__init__.py +5 -0
  333. pythinker_code/web/runner/messages.py +57 -0
  334. pythinker_code/web/runner/process.py +754 -0
  335. pythinker_code/web/runner/worker.py +97 -0
  336. pythinker_code/web/static/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2 +0 -0
  337. pythinker_code/web/static/assets/KaTeX_AMS-Regular-DMm9YOAa.woff +0 -0
  338. pythinker_code/web/static/assets/KaTeX_AMS-Regular-DRggAlZN.ttf +0 -0
  339. pythinker_code/web/static/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf +0 -0
  340. pythinker_code/web/static/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff +0 -0
  341. pythinker_code/web/static/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2 +0 -0
  342. pythinker_code/web/static/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff +0 -0
  343. pythinker_code/web/static/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2 +0 -0
  344. pythinker_code/web/static/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf +0 -0
  345. pythinker_code/web/static/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf +0 -0
  346. pythinker_code/web/static/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff +0 -0
  347. pythinker_code/web/static/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2 +0 -0
  348. pythinker_code/web/static/assets/KaTeX_Fraktur-Regular-CB_wures.ttf +0 -0
  349. pythinker_code/web/static/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2 +0 -0
  350. pythinker_code/web/static/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff +0 -0
  351. pythinker_code/web/static/assets/KaTeX_Main-Bold-Cx986IdX.woff2 +0 -0
  352. pythinker_code/web/static/assets/KaTeX_Main-Bold-Jm3AIy58.woff +0 -0
  353. pythinker_code/web/static/assets/KaTeX_Main-Bold-waoOVXN0.ttf +0 -0
  354. pythinker_code/web/static/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2 +0 -0
  355. pythinker_code/web/static/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf +0 -0
  356. pythinker_code/web/static/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff +0 -0
  357. pythinker_code/web/static/assets/KaTeX_Main-Italic-3WenGoN9.ttf +0 -0
  358. pythinker_code/web/static/assets/KaTeX_Main-Italic-BMLOBm91.woff +0 -0
  359. pythinker_code/web/static/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2 +0 -0
  360. pythinker_code/web/static/assets/KaTeX_Main-Regular-B22Nviop.woff2 +0 -0
  361. pythinker_code/web/static/assets/KaTeX_Main-Regular-Dr94JaBh.woff +0 -0
  362. pythinker_code/web/static/assets/KaTeX_Main-Regular-ypZvNtVU.ttf +0 -0
  363. pythinker_code/web/static/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf +0 -0
  364. pythinker_code/web/static/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2 +0 -0
  365. pythinker_code/web/static/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff +0 -0
  366. pythinker_code/web/static/assets/KaTeX_Math-Italic-DA0__PXp.woff +0 -0
  367. pythinker_code/web/static/assets/KaTeX_Math-Italic-flOr_0UB.ttf +0 -0
  368. pythinker_code/web/static/assets/KaTeX_Math-Italic-t53AETM-.woff2 +0 -0
  369. pythinker_code/web/static/assets/KaTeX_SansSerif-Bold-CFMepnvq.ttf +0 -0
  370. pythinker_code/web/static/assets/KaTeX_SansSerif-Bold-D1sUS0GD.woff2 +0 -0
  371. pythinker_code/web/static/assets/KaTeX_SansSerif-Bold-DbIhKOiC.woff +0 -0
  372. pythinker_code/web/static/assets/KaTeX_SansSerif-Italic-C3H0VqGB.woff2 +0 -0
  373. pythinker_code/web/static/assets/KaTeX_SansSerif-Italic-DN2j7dab.woff +0 -0
  374. pythinker_code/web/static/assets/KaTeX_SansSerif-Italic-YYjJ1zSn.ttf +0 -0
  375. pythinker_code/web/static/assets/KaTeX_SansSerif-Regular-BNo7hRIc.ttf +0 -0
  376. pythinker_code/web/static/assets/KaTeX_SansSerif-Regular-CS6fqUqJ.woff +0 -0
  377. pythinker_code/web/static/assets/KaTeX_SansSerif-Regular-DDBCnlJ7.woff2 +0 -0
  378. pythinker_code/web/static/assets/KaTeX_Script-Regular-C5JkGWo-.ttf +0 -0
  379. pythinker_code/web/static/assets/KaTeX_Script-Regular-D3wIWfF6.woff2 +0 -0
  380. pythinker_code/web/static/assets/KaTeX_Script-Regular-D5yQViql.woff +0 -0
  381. pythinker_code/web/static/assets/KaTeX_Size1-Regular-C195tn64.woff +0 -0
  382. pythinker_code/web/static/assets/KaTeX_Size1-Regular-Dbsnue_I.ttf +0 -0
  383. pythinker_code/web/static/assets/KaTeX_Size1-Regular-mCD8mA8B.woff2 +0 -0
  384. pythinker_code/web/static/assets/KaTeX_Size2-Regular-B7gKUWhC.ttf +0 -0
  385. pythinker_code/web/static/assets/KaTeX_Size2-Regular-Dy4dx90m.woff2 +0 -0
  386. pythinker_code/web/static/assets/KaTeX_Size2-Regular-oD1tc_U0.woff +0 -0
  387. pythinker_code/web/static/assets/KaTeX_Size3-Regular-CTq5MqoE.woff +0 -0
  388. pythinker_code/web/static/assets/KaTeX_Size3-Regular-DgpXs0kz.ttf +0 -0
  389. pythinker_code/web/static/assets/KaTeX_Size4-Regular-BF-4gkZK.woff +0 -0
  390. pythinker_code/web/static/assets/KaTeX_Size4-Regular-DWFBv043.ttf +0 -0
  391. pythinker_code/web/static/assets/KaTeX_Size4-Regular-Dl5lxZxV.woff2 +0 -0
  392. pythinker_code/web/static/assets/KaTeX_Typewriter-Regular-C0xS9mPB.woff +0 -0
  393. pythinker_code/web/static/assets/KaTeX_Typewriter-Regular-CO6r4hn1.woff2 +0 -0
  394. pythinker_code/web/static/assets/KaTeX_Typewriter-Regular-D3Ib7_Hf.ttf +0 -0
  395. pythinker_code/web/static/assets/_baseUniq-DpSMr1jx.js +1 -0
  396. pythinker_code/web/static/assets/abap-BdImnpbu.js +1 -0
  397. pythinker_code/web/static/assets/actionscript-3-CfeIJUat.js +1 -0
  398. pythinker_code/web/static/assets/ada-bCR0ucgS.js +1 -0
  399. pythinker_code/web/static/assets/andromeeda-C-Jbm3Hp.js +1 -0
  400. pythinker_code/web/static/assets/angular-html-CU67Zn6k.js +1 -0
  401. pythinker_code/web/static/assets/angular-ts-BwZT4LLn.js +1 -0
  402. pythinker_code/web/static/assets/apache-Pmp26Uib.js +1 -0
  403. pythinker_code/web/static/assets/apex-D8_7TLub.js +1 -0
  404. pythinker_code/web/static/assets/apl-dKokRX4l.js +1 -0
  405. pythinker_code/web/static/assets/applescript-Co6uUVPk.js +1 -0
  406. pythinker_code/web/static/assets/ara-BRHolxvo.js +1 -0
  407. pythinker_code/web/static/assets/arc-DpsahJyV.js +1 -0
  408. pythinker_code/web/static/assets/architectureDiagram-VXUJARFQ-DqiRv9Eg.js +36 -0
  409. pythinker_code/web/static/assets/asciidoc-Dv7Oe6Be.js +1 -0
  410. pythinker_code/web/static/assets/asm-D_Q5rh1f.js +1 -0
  411. pythinker_code/web/static/assets/astro-CbQHKStN.js +1 -0
  412. pythinker_code/web/static/assets/aurora-x-D-2ljcwZ.js +1 -0
  413. pythinker_code/web/static/assets/awk-DMzUqQB5.js +1 -0
  414. pythinker_code/web/static/assets/ayu-dark-CmMr59Fi.js +1 -0
  415. pythinker_code/web/static/assets/ballerina-BFfxhgS-.js +1 -0
  416. pythinker_code/web/static/assets/bat-BkioyH1T.js +1 -0
  417. pythinker_code/web/static/assets/beancount-k_qm7-4y.js +1 -0
  418. pythinker_code/web/static/assets/berry-uYugtg8r.js +1 -0
  419. pythinker_code/web/static/assets/bibtex-CHM0blh-.js +1 -0
  420. pythinker_code/web/static/assets/bicep-Bmn6On1c.js +1 -0
  421. pythinker_code/web/static/assets/blade-D4QpJJKB.js +1 -0
  422. pythinker_code/web/static/assets/blockDiagram-VD42YOAC-WgtUvqbp.js +122 -0
  423. pythinker_code/web/static/assets/bsl-BO_Y6i37.js +1 -0
  424. pythinker_code/web/static/assets/c-BIGW1oBm.js +1 -0
  425. pythinker_code/web/static/assets/c3-VCDPK7BO.js +1 -0
  426. pythinker_code/web/static/assets/c4Diagram-YG6GDRKO-rK0RPuZd.js +10 -0
  427. pythinker_code/web/static/assets/cadence-Bv_4Rxtq.js +1 -0
  428. pythinker_code/web/static/assets/cairo-KRGpt6FW.js +1 -0
  429. pythinker_code/web/static/assets/catppuccin-frappe-DFWUc33u.js +1 -0
  430. pythinker_code/web/static/assets/catppuccin-latte-C9dUb6Cb.js +1 -0
  431. pythinker_code/web/static/assets/catppuccin-macchiato-DQyhUUbL.js +1 -0
  432. pythinker_code/web/static/assets/catppuccin-mocha-D87Tk5Gz.js +1 -0
  433. pythinker_code/web/static/assets/channel-B0rlvkH-.js +1 -0
  434. pythinker_code/web/static/assets/chunk-4BX2VUAB-DIkMuLV-.js +1 -0
  435. pythinker_code/web/static/assets/chunk-55IACEB6-CORdm4k4.js +1 -0
  436. pythinker_code/web/static/assets/chunk-B4BG7PRW-D9xDhwHO.js +165 -0
  437. pythinker_code/web/static/assets/chunk-DI55MBZ5-BDmF9Bh-.js +220 -0
  438. pythinker_code/web/static/assets/chunk-FMBD7UC4-BCse_HmM.js +15 -0
  439. pythinker_code/web/static/assets/chunk-QN33PNHL-DCpBmTzA.js +1 -0
  440. pythinker_code/web/static/assets/chunk-QZHKN3VN-BqLuqobw.js +1 -0
  441. pythinker_code/web/static/assets/chunk-TZMSLE5B-8K2ogOKS.js +1 -0
  442. pythinker_code/web/static/assets/clarity-D53aC0YG.js +1 -0
  443. pythinker_code/web/static/assets/classDiagram-2ON5EDUG-D_ZHSii2.js +1 -0
  444. pythinker_code/web/static/assets/classDiagram-v2-WZHVMYZB-D_ZHSii2.js +1 -0
  445. pythinker_code/web/static/assets/clojure-P80f7IUj.js +1 -0
  446. pythinker_code/web/static/assets/clone-GSXejyY1.js +1 -0
  447. pythinker_code/web/static/assets/cmake-D1j8_8rp.js +1 -0
  448. pythinker_code/web/static/assets/cobol-nwyudZeR.js +1 -0
  449. pythinker_code/web/static/assets/code-block-IT6T5CEO-DWTFYA28.js +2 -0
  450. pythinker_code/web/static/assets/codeowners-Bp6g37R7.js +1 -0
  451. pythinker_code/web/static/assets/codeql-DsOJ9woJ.js +1 -0
  452. pythinker_code/web/static/assets/coffee-Ch7k5sss.js +1 -0
  453. pythinker_code/web/static/assets/common-lisp-Cg-RD9OK.js +1 -0
  454. pythinker_code/web/static/assets/coq-DkFqJrB1.js +1 -0
  455. pythinker_code/web/static/assets/cose-bilkent-S5V4N54A-BRI7ES-N.js +1 -0
  456. pythinker_code/web/static/assets/cpp-CofmeUqb.js +1 -0
  457. pythinker_code/web/static/assets/crystal-tKQVLTB8.js +1 -0
  458. pythinker_code/web/static/assets/csharp-K5feNrxe.js +1 -0
  459. pythinker_code/web/static/assets/css-DPfMkruS.js +1 -0
  460. pythinker_code/web/static/assets/csv-fuZLfV_i.js +1 -0
  461. pythinker_code/web/static/assets/cue-D82EKSYY.js +1 -0
  462. pythinker_code/web/static/assets/cypher-COkxafJQ.js +1 -0
  463. pythinker_code/web/static/assets/cytoscape.esm-B6BxUuKW.js +321 -0
  464. pythinker_code/web/static/assets/d-85-TOEBH.js +1 -0
  465. pythinker_code/web/static/assets/dagre-6UL2VRFP-Ci5GdWfi.js +4 -0
  466. pythinker_code/web/static/assets/dark-plus-C3mMm8J8.js +1 -0
  467. pythinker_code/web/static/assets/dart-CF10PKvl.js +1 -0
  468. pythinker_code/web/static/assets/dax-CEL-wOlO.js +1 -0
  469. pythinker_code/web/static/assets/defaultLocale-DX6XiGOO.js +1 -0
  470. pythinker_code/web/static/assets/desktop-BmXAJ9_W.js +1 -0
  471. pythinker_code/web/static/assets/diagram-PSM6KHXK-0hhAylV4.js +24 -0
  472. pythinker_code/web/static/assets/diagram-QEK2KX5R-8fxgaW6d.js +43 -0
  473. pythinker_code/web/static/assets/diagram-S2PKOQOG-FRr0_atE.js +24 -0
  474. pythinker_code/web/static/assets/diff-D97Zzqfu.js +1 -0
  475. pythinker_code/web/static/assets/docker-BcOcwvcX.js +1 -0
  476. pythinker_code/web/static/assets/dotenv-Da5cRb03.js +1 -0
  477. pythinker_code/web/static/assets/dracula-BzJJZx-M.js +1 -0
  478. pythinker_code/web/static/assets/dracula-soft-BXkSAIEj.js +1 -0
  479. pythinker_code/web/static/assets/dream-maker-BtqSS_iP.js +1 -0
  480. pythinker_code/web/static/assets/edge-BkV0erSs.js +1 -0
  481. pythinker_code/web/static/assets/elixir-CDX3lj18.js +1 -0
  482. pythinker_code/web/static/assets/elm-DbKCFpqz.js +1 -0
  483. pythinker_code/web/static/assets/emacs-lisp-C9XAeP06.js +1 -0
  484. pythinker_code/web/static/assets/erDiagram-Q2GNP2WA-B3T-hJUM.js +60 -0
  485. pythinker_code/web/static/assets/erb-BOJIQeun.js +1 -0
  486. pythinker_code/web/static/assets/erlang-DsQrWhSR.js +1 -0
  487. pythinker_code/web/static/assets/everforest-dark-BgDCqdQA.js +1 -0
  488. pythinker_code/web/static/assets/everforest-light-C8M2exoo.js +1 -0
  489. pythinker_code/web/static/assets/fennel-BYunw83y.js +1 -0
  490. pythinker_code/web/static/assets/fish-BvzEVeQv.js +1 -0
  491. pythinker_code/web/static/assets/flowDiagram-NV44I4VS-D0S3u7ot.js +162 -0
  492. pythinker_code/web/static/assets/fluent-C4IJs8-o.js +1 -0
  493. pythinker_code/web/static/assets/fortran-fixed-form-CkoXwp7k.js +1 -0
  494. pythinker_code/web/static/assets/fortran-free-form-BxgE0vQu.js +1 -0
  495. pythinker_code/web/static/assets/fsharp-CXgrBDvD.js +1 -0
  496. pythinker_code/web/static/assets/ganttDiagram-JELNMOA3-CHrN2a23.js +267 -0
  497. pythinker_code/web/static/assets/gdresource-B7Tvp0Sc.js +1 -0
  498. pythinker_code/web/static/assets/gdscript-DTMYz4Jt.js +1 -0
  499. pythinker_code/web/static/assets/gdshader-DkwncUOv.js +1 -0
  500. pythinker_code/web/static/assets/genie-D0YGMca9.js +1 -0
  501. pythinker_code/web/static/assets/gherkin-DyxjwDmM.js +1 -0
  502. pythinker_code/web/static/assets/git-commit-F4YmCXRG.js +1 -0
  503. pythinker_code/web/static/assets/git-rebase-r7XF79zn.js +1 -0
  504. pythinker_code/web/static/assets/gitGraphDiagram-NY62KEGX-CfcXZWg0.js +65 -0
  505. pythinker_code/web/static/assets/github-dark-DHJKELXO.js +1 -0
  506. pythinker_code/web/static/assets/github-dark-default-Cuk6v7N8.js +1 -0
  507. pythinker_code/web/static/assets/github-dark-dimmed-DH5Ifo-i.js +1 -0
  508. pythinker_code/web/static/assets/github-dark-high-contrast-E3gJ1_iC.js +1 -0
  509. pythinker_code/web/static/assets/github-light-DAi9KRSo.js +1 -0
  510. pythinker_code/web/static/assets/github-light-default-D7oLnXFd.js +1 -0
  511. pythinker_code/web/static/assets/github-light-high-contrast-BfjtVDDH.js +1 -0
  512. pythinker_code/web/static/assets/gleam-BspZqrRM.js +1 -0
  513. pythinker_code/web/static/assets/glimmer-js-Rg0-pVw9.js +1 -0
  514. pythinker_code/web/static/assets/glimmer-ts-U6CK756n.js +1 -0
  515. pythinker_code/web/static/assets/glsl-DplSGwfg.js +1 -0
  516. pythinker_code/web/static/assets/gn-n2N0HUVH.js +1 -0
  517. pythinker_code/web/static/assets/gnuplot-DdkO51Og.js +1 -0
  518. pythinker_code/web/static/assets/go-Dn2_MT6a.js +1 -0
  519. pythinker_code/web/static/assets/graph-8jMJwCqE.js +1 -0
  520. pythinker_code/web/static/assets/graphql-ChdNCCLP.js +1 -0
  521. pythinker_code/web/static/assets/groovy-gcz8RCvz.js +1 -0
  522. pythinker_code/web/static/assets/gruvbox-dark-hard-CFHQjOhq.js +1 -0
  523. pythinker_code/web/static/assets/gruvbox-dark-medium-GsRaNv29.js +1 -0
  524. pythinker_code/web/static/assets/gruvbox-dark-soft-CVdnzihN.js +1 -0
  525. pythinker_code/web/static/assets/gruvbox-light-hard-CH1njM8p.js +1 -0
  526. pythinker_code/web/static/assets/gruvbox-light-medium-DRw_LuNl.js +1 -0
  527. pythinker_code/web/static/assets/gruvbox-light-soft-hJgmCMqR.js +1 -0
  528. pythinker_code/web/static/assets/hack-CaT9iCJl.js +1 -0
  529. pythinker_code/web/static/assets/haml-B8DHNrY2.js +1 -0
  530. pythinker_code/web/static/assets/handlebars-BL8al0AC.js +1 -0
  531. pythinker_code/web/static/assets/haskell-Df6bDoY_.js +1 -0
  532. pythinker_code/web/static/assets/haxe-CzTSHFRz.js +1 -0
  533. pythinker_code/web/static/assets/hcl-BWvSN4gD.js +1 -0
  534. pythinker_code/web/static/assets/hjson-D5-asLiD.js +1 -0
  535. pythinker_code/web/static/assets/hlsl-D3lLCCz7.js +1 -0
  536. pythinker_code/web/static/assets/houston-DnULxvSX.js +1 -0
  537. pythinker_code/web/static/assets/html-GMplVEZG.js +1 -0
  538. pythinker_code/web/static/assets/html-derivative-BFtXZ54Q.js +1 -0
  539. pythinker_code/web/static/assets/http-jrhK8wxY.js +1 -0
  540. pythinker_code/web/static/assets/hurl-irOxFIW8.js +1 -0
  541. pythinker_code/web/static/assets/hxml-Bvhsp5Yf.js +1 -0
  542. pythinker_code/web/static/assets/hy-DFXneXwc.js +1 -0
  543. pythinker_code/web/static/assets/imba-DGztddWO.js +1 -0
  544. pythinker_code/web/static/assets/index-BXrFnzMy.js +153 -0
  545. pythinker_code/web/static/assets/index-BpoLgcEt.js +1 -0
  546. pythinker_code/web/static/assets/index-BrfQJnRD.js +5 -0
  547. pythinker_code/web/static/assets/index-C4gFzubz.js +2 -0
  548. pythinker_code/web/static/assets/index-CzV_vCfu.css +1 -0
  549. pythinker_code/web/static/assets/index-DI2oedCt.js +19 -0
  550. pythinker_code/web/static/assets/infoDiagram-WHAUD3N6-DdxonBf3.js +2 -0
  551. pythinker_code/web/static/assets/ini-BEwlwnbL.js +1 -0
  552. pythinker_code/web/static/assets/init-Gi6I4Gst.js +1 -0
  553. pythinker_code/web/static/assets/inter-cyrillic-ext-wght-normal-BOeWTOD4.woff2 +0 -0
  554. pythinker_code/web/static/assets/inter-cyrillic-wght-normal-DqGufNeO.woff2 +0 -0
  555. pythinker_code/web/static/assets/inter-greek-ext-wght-normal-DlzME5K_.woff2 +0 -0
  556. pythinker_code/web/static/assets/inter-greek-wght-normal-CkhJZR-_.woff2 +0 -0
  557. pythinker_code/web/static/assets/inter-latin-ext-wght-normal-DO1Apj_S.woff2 +0 -0
  558. pythinker_code/web/static/assets/inter-latin-wght-normal-Dx4kXJAl.woff2 +0 -0
  559. pythinker_code/web/static/assets/inter-vietnamese-wght-normal-CBcvBZtf.woff2 +0 -0
  560. pythinker_code/web/static/assets/java-CylS5w8V.js +1 -0
  561. pythinker_code/web/static/assets/javascript-wDzz0qaB.js +1 -0
  562. pythinker_code/web/static/assets/jinja-4LBKfQ-Z.js +1 -0
  563. pythinker_code/web/static/assets/jison-wvAkD_A8.js +1 -0
  564. pythinker_code/web/static/assets/journeyDiagram-XKPGCS4Q-BXf4aQei.js +139 -0
  565. pythinker_code/web/static/assets/json-Cp-IABpG.js +1 -0
  566. pythinker_code/web/static/assets/json5-C9tS-k6U.js +1 -0
  567. pythinker_code/web/static/assets/jsonc-Des-eS-w.js +1 -0
  568. pythinker_code/web/static/assets/jsonl-DcaNXYhu.js +1 -0
  569. pythinker_code/web/static/assets/jsonnet-DFQXde-d.js +1 -0
  570. pythinker_code/web/static/assets/jssm-C2t-YnRu.js +1 -0
  571. pythinker_code/web/static/assets/jsx-g9-lgVsj.js +1 -0
  572. pythinker_code/web/static/assets/julia-CxzCAyBv.js +1 -0
  573. pythinker_code/web/static/assets/kanagawa-dragon-CkXjmgJE.js +1 -0
  574. pythinker_code/web/static/assets/kanagawa-lotus-CfQXZHmo.js +1 -0
  575. pythinker_code/web/static/assets/kanagawa-wave-DWedfzmr.js +1 -0
  576. pythinker_code/web/static/assets/kanban-definition-3W4ZIXB7-DLpPPOu8.js +89 -0
  577. pythinker_code/web/static/assets/katex-D2lIc1rk.css +1 -0
  578. pythinker_code/web/static/assets/kdl-DV7GczEv.js +1 -0
  579. pythinker_code/web/static/assets/kotlin-BdnUsdx6.js +1 -0
  580. pythinker_code/web/static/assets/kusto-DZf3V79B.js +1 -0
  581. pythinker_code/web/static/assets/laserwave-DUszq2jm.js +1 -0
  582. pythinker_code/web/static/assets/latex-B4uzh10-.js +1 -0
  583. pythinker_code/web/static/assets/layout-DH73UoAH.js +1 -0
  584. pythinker_code/web/static/assets/lean-BZvkOJ9d.js +1 -0
  585. pythinker_code/web/static/assets/less-B1dDrJ26.js +1 -0
  586. pythinker_code/web/static/assets/light-plus-B7mTdjB0.js +1 -0
  587. pythinker_code/web/static/assets/linear-bAer2-sK.js +1 -0
  588. pythinker_code/web/static/assets/liquid-DYVedYrR.js +1 -0
  589. pythinker_code/web/static/assets/llvm-BtvRca6l.js +1 -0
  590. pythinker_code/web/static/assets/log-2UxHyX5q.js +1 -0
  591. pythinker_code/web/static/assets/logo-BtOb2qkB.js +1 -0
  592. pythinker_code/web/static/assets/lua-BbnMAYS6.js +1 -0
  593. pythinker_code/web/static/assets/luau-C-HG3fhB.js +1 -0
  594. pythinker_code/web/static/assets/make-CHLpvVh8.js +1 -0
  595. pythinker_code/web/static/assets/markdown-Cvjx9yec.js +1 -0
  596. pythinker_code/web/static/assets/marko-DZsq8hO1.js +1 -0
  597. pythinker_code/web/static/assets/material-theme-D5KoaKCx.js +1 -0
  598. pythinker_code/web/static/assets/material-theme-darker-BfHTSMKl.js +1 -0
  599. pythinker_code/web/static/assets/material-theme-lighter-B0m2ddpp.js +1 -0
  600. pythinker_code/web/static/assets/material-theme-ocean-CyktbL80.js +1 -0
  601. pythinker_code/web/static/assets/material-theme-palenight-Csfq5Kiy.js +1 -0
  602. pythinker_code/web/static/assets/matlab-D7o27uSR.js +1 -0
  603. pythinker_code/web/static/assets/mdc-DUICxH0z.js +1 -0
  604. pythinker_code/web/static/assets/mdx-Cmh6b_Ma.js +1 -0
  605. pythinker_code/web/static/assets/mermaid-VLURNSYL-B2P5VJ9v.css +1 -0
  606. pythinker_code/web/static/assets/mermaid-VLURNSYL-CuqbwKXv.js +465 -0
  607. pythinker_code/web/static/assets/mermaid-mWjccvbQ.js +1 -0
  608. pythinker_code/web/static/assets/mermaid.core-Nx-rTKiV.js +191 -0
  609. pythinker_code/web/static/assets/min-DbfD8Ywu.js +1 -0
  610. pythinker_code/web/static/assets/min-dark-CafNBF8u.js +1 -0
  611. pythinker_code/web/static/assets/min-light-CTRr51gU.js +1 -0
  612. pythinker_code/web/static/assets/mindmap-definition-VGOIOE7T-C6l761Ue.js +68 -0
  613. pythinker_code/web/static/assets/mipsasm-CKIfxQSi.js +1 -0
  614. pythinker_code/web/static/assets/mojo-B93PlW-d.js +1 -0
  615. pythinker_code/web/static/assets/monokai-D4h5O-jR.js +1 -0
  616. pythinker_code/web/static/assets/moonbit-Ba13S78F.js +1 -0
  617. pythinker_code/web/static/assets/move-Bu9oaDYs.js +1 -0
  618. pythinker_code/web/static/assets/narrat-DRg8JJMk.js +1 -0
  619. pythinker_code/web/static/assets/nextflow-BrzmwbiE.js +1 -0
  620. pythinker_code/web/static/assets/nginx-DknmC5AR.js +1 -0
  621. pythinker_code/web/static/assets/night-owl-C39BiMTA.js +1 -0
  622. pythinker_code/web/static/assets/nim-CVrawwO9.js +1 -0
  623. pythinker_code/web/static/assets/nix-CwoSXNpI.js +1 -0
  624. pythinker_code/web/static/assets/nord-Ddv68eIx.js +1 -0
  625. pythinker_code/web/static/assets/nushell-C-sUppwS.js +1 -0
  626. pythinker_code/web/static/assets/objective-c-DXmwc3jG.js +1 -0
  627. pythinker_code/web/static/assets/objective-cpp-CLxacb5B.js +1 -0
  628. pythinker_code/web/static/assets/ocaml-C0hk2d4L.js +1 -0
  629. pythinker_code/web/static/assets/one-dark-pro-DVMEJ2y_.js +1 -0
  630. pythinker_code/web/static/assets/one-light-PoHY5YXO.js +1 -0
  631. pythinker_code/web/static/assets/openscad-C4EeE6gA.js +1 -0
  632. pythinker_code/web/static/assets/ordinal-Cboi1Yqb.js +1 -0
  633. pythinker_code/web/static/assets/pascal-D93ZcfNL.js +1 -0
  634. pythinker_code/web/static/assets/perl-C0TMdlhV.js +1 -0
  635. pythinker_code/web/static/assets/php-CDn_0X-4.js +1 -0
  636. pythinker_code/web/static/assets/pieDiagram-ADFJNKIX-fNg41mT9.js +30 -0
  637. pythinker_code/web/static/assets/pkl-u5AG7uiY.js +1 -0
  638. pythinker_code/web/static/assets/plastic-3e1v2bzS.js +1 -0
  639. pythinker_code/web/static/assets/plsql-ChMvpjG-.js +1 -0
  640. pythinker_code/web/static/assets/po-BTJTHyun.js +1 -0
  641. pythinker_code/web/static/assets/poimandres-CS3Unz2-.js +1 -0
  642. pythinker_code/web/static/assets/polar-C0HS_06l.js +1 -0
  643. pythinker_code/web/static/assets/postcss-CXtECtnM.js +1 -0
  644. pythinker_code/web/static/assets/powerquery-CEu0bR-o.js +1 -0
  645. pythinker_code/web/static/assets/powershell-Dpen1YoG.js +1 -0
  646. pythinker_code/web/static/assets/prisma-Dd19v3D-.js +1 -0
  647. pythinker_code/web/static/assets/prolog-CbFg5uaA.js +1 -0
  648. pythinker_code/web/static/assets/proto-C7zT0LnQ.js +1 -0
  649. pythinker_code/web/static/assets/pug-CGlum2m_.js +1 -0
  650. pythinker_code/web/static/assets/puppet-BMWR74SV.js +1 -0
  651. pythinker_code/web/static/assets/purescript-CklMAg4u.js +1 -0
  652. pythinker_code/web/static/assets/python-B6aJPvgy.js +1 -0
  653. pythinker_code/web/static/assets/qml-3beO22l8.js +1 -0
  654. pythinker_code/web/static/assets/qmldir-C8lEn-DE.js +1 -0
  655. pythinker_code/web/static/assets/qss-IeuSbFQv.js +1 -0
  656. pythinker_code/web/static/assets/quadrantDiagram-AYHSOK5B-DJz3Kx87.js +7 -0
  657. pythinker_code/web/static/assets/r-Dspwwk_N.js +1 -0
  658. pythinker_code/web/static/assets/racket-BqYA7rlc.js +1 -0
  659. pythinker_code/web/static/assets/raku-DXvB9xmW.js +1 -0
  660. pythinker_code/web/static/assets/razor-C1TweQQi.js +1 -0
  661. pythinker_code/web/static/assets/red-bN70gL4F.js +1 -0
  662. pythinker_code/web/static/assets/reg-C-SQnVFl.js +1 -0
  663. pythinker_code/web/static/assets/regexp-CDVJQ6XC.js +1 -0
  664. pythinker_code/web/static/assets/rel-C3B-1QV4.js +1 -0
  665. pythinker_code/web/static/assets/requirementDiagram-UZGBJVZJ-B4SbrfE9.js +64 -0
  666. pythinker_code/web/static/assets/riscv-BM1_JUlF.js +1 -0
  667. pythinker_code/web/static/assets/rose-pine-dawn-DHQR4-dF.js +1 -0
  668. pythinker_code/web/static/assets/rose-pine-moon-D4_iv3hh.js +1 -0
  669. pythinker_code/web/static/assets/rose-pine-qdsjHGoJ.js +1 -0
  670. pythinker_code/web/static/assets/rosmsg-BJDFO7_C.js +1 -0
  671. pythinker_code/web/static/assets/rst-B0xPkSld.js +1 -0
  672. pythinker_code/web/static/assets/ruby-BvKwtOVI.js +1 -0
  673. pythinker_code/web/static/assets/rust-B1yitclQ.js +1 -0
  674. pythinker_code/web/static/assets/sankeyDiagram-TZEHDZUN-CoSUjLAG.js +10 -0
  675. pythinker_code/web/static/assets/sas-cz2c8ADy.js +1 -0
  676. pythinker_code/web/static/assets/sass-Cj5Yp3dK.js +1 -0
  677. pythinker_code/web/static/assets/scala-C151Ov-r.js +1 -0
  678. pythinker_code/web/static/assets/scheme-C98Dy4si.js +1 -0
  679. pythinker_code/web/static/assets/scss-OYdSNvt2.js +1 -0
  680. pythinker_code/web/static/assets/sdbl-DVxCFoDh.js +1 -0
  681. pythinker_code/web/static/assets/sequenceDiagram-WL72ISMW-PjhBNHi3.js +145 -0
  682. pythinker_code/web/static/assets/shaderlab-Dg9Lc6iA.js +1 -0
  683. pythinker_code/web/static/assets/shellscript-Yzrsuije.js +1 -0
  684. pythinker_code/web/static/assets/shellsession-BADoaaVG.js +1 -0
  685. pythinker_code/web/static/assets/slack-dark-BthQWCQV.js +1 -0
  686. pythinker_code/web/static/assets/slack-ochin-DqwNpetd.js +1 -0
  687. pythinker_code/web/static/assets/smalltalk-BERRCDM3.js +1 -0
  688. pythinker_code/web/static/assets/snazzy-light-Bw305WKR.js +1 -0
  689. pythinker_code/web/static/assets/solarized-dark-DXbdFlpD.js +1 -0
  690. pythinker_code/web/static/assets/solarized-light-L9t79GZl.js +1 -0
  691. pythinker_code/web/static/assets/solidity-rGO070M0.js +1 -0
  692. pythinker_code/web/static/assets/soy-Brmx7dQM.js +1 -0
  693. pythinker_code/web/static/assets/sparql-rVzFXLq3.js +1 -0
  694. pythinker_code/web/static/assets/splunk-BtCnVYZw.js +1 -0
  695. pythinker_code/web/static/assets/sql-BLtJtn59.js +1 -0
  696. pythinker_code/web/static/assets/ssh-config-_ykCGR6B.js +1 -0
  697. pythinker_code/web/static/assets/stata-BH5u7GGu.js +1 -0
  698. pythinker_code/web/static/assets/stateDiagram-FKZM4ZOC-DOwESt8-.js +1 -0
  699. pythinker_code/web/static/assets/stateDiagram-v2-4FDKWEC3-yl3OHWiP.js +1 -0
  700. pythinker_code/web/static/assets/stylus-BEDo0Tqx.js +1 -0
  701. pythinker_code/web/static/assets/svelte-zxCyuUbr.js +1 -0
  702. pythinker_code/web/static/assets/swift-Dg5xB15N.js +1 -0
  703. pythinker_code/web/static/assets/synthwave-84-CbfX1IO0.js +1 -0
  704. pythinker_code/web/static/assets/system-verilog-CnnmHF94.js +1 -0
  705. pythinker_code/web/static/assets/systemd-4A_iFExJ.js +1 -0
  706. pythinker_code/web/static/assets/talonscript-CkByrt1z.js +1 -0
  707. pythinker_code/web/static/assets/tasl-QIJgUcNo.js +1 -0
  708. pythinker_code/web/static/assets/tcl-dwOrl1Do.js +1 -0
  709. pythinker_code/web/static/assets/templ-W15q3VgB.js +1 -0
  710. pythinker_code/web/static/assets/terraform-BETggiCN.js +1 -0
  711. pythinker_code/web/static/assets/tex-CvyZ59Mk.js +1 -0
  712. pythinker_code/web/static/assets/timeline-definition-IT6M3QCI-CkCLnAgi.js +61 -0
  713. pythinker_code/web/static/assets/tokyo-night-hegEt444.js +1 -0
  714. pythinker_code/web/static/assets/toml-vGWfd6FD.js +1 -0
  715. pythinker_code/web/static/assets/treemap-KMMF4GRG-CZS5XwTf.js +128 -0
  716. pythinker_code/web/static/assets/ts-tags-zn1MmPIZ.js +1 -0
  717. pythinker_code/web/static/assets/tsv-B_m7g4N7.js +1 -0
  718. pythinker_code/web/static/assets/tsx-COt5Ahok.js +1 -0
  719. pythinker_code/web/static/assets/turtle-BsS91CYL.js +1 -0
  720. pythinker_code/web/static/assets/twig-CO9l9SDP.js +1 -0
  721. pythinker_code/web/static/assets/typescript-BPQ3VLAy.js +1 -0
  722. pythinker_code/web/static/assets/typespec-BGHnOYBU.js +1 -0
  723. pythinker_code/web/static/assets/typst-DHCkPAjA.js +1 -0
  724. pythinker_code/web/static/assets/v-BcVCzyr7.js +1 -0
  725. pythinker_code/web/static/assets/vala-CsfeWuGM.js +1 -0
  726. pythinker_code/web/static/assets/vb-D17OF-Vu.js +1 -0
  727. pythinker_code/web/static/assets/verilog-BQ8w6xss.js +1 -0
  728. pythinker_code/web/static/assets/vesper-DU1UobuO.js +1 -0
  729. pythinker_code/web/static/assets/vhdl-CeAyd5Ju.js +1 -0
  730. pythinker_code/web/static/assets/viml-CJc9bBzg.js +1 -0
  731. pythinker_code/web/static/assets/vitesse-black-Bkuqu6BP.js +1 -0
  732. pythinker_code/web/static/assets/vitesse-dark-D0r3Knsf.js +1 -0
  733. pythinker_code/web/static/assets/vitesse-light-CVO1_9PV.js +1 -0
  734. pythinker_code/web/static/assets/vue-DN_0RTcg.js +1 -0
  735. pythinker_code/web/static/assets/vue-html-AaS7Mt5G.js +1 -0
  736. pythinker_code/web/static/assets/vue-vine-CQOfvN7w.js +1 -0
  737. pythinker_code/web/static/assets/vyper-CDx5xZoG.js +1 -0
  738. pythinker_code/web/static/assets/wasm-CG6Dc4jp.js +1 -0
  739. pythinker_code/web/static/assets/wasm-MzD3tlZU.js +1 -0
  740. pythinker_code/web/static/assets/wenyan-BV7otONQ.js +1 -0
  741. pythinker_code/web/static/assets/wgsl-Dx-B1_4e.js +1 -0
  742. pythinker_code/web/static/assets/wikitext-BhOHFoWU.js +1 -0
  743. pythinker_code/web/static/assets/wit-5i3qLPDT.js +1 -0
  744. pythinker_code/web/static/assets/wolfram-lXgVvXCa.js +1 -0
  745. pythinker_code/web/static/assets/xml-sdJ4AIDG.js +1 -0
  746. pythinker_code/web/static/assets/xsl-CtQFsRM5.js +1 -0
  747. pythinker_code/web/static/assets/xychartDiagram-PRI3JC2R-DkqqHNLh.js +7 -0
  748. pythinker_code/web/static/assets/yaml-Buea-lGh.js +1 -0
  749. pythinker_code/web/static/assets/zenscript-DVFEvuxE.js +1 -0
  750. pythinker_code/web/static/assets/zig-VOosw3JB.js +1 -0
  751. pythinker_code/web/static/brand/apple-touch-icon.png +0 -0
  752. pythinker_code/web/static/brand/arctecture.webp +0 -0
  753. pythinker_code/web/static/brand/bimi-logo.svg +46 -0
  754. pythinker_code/web/static/brand/favicon.ico +0 -0
  755. pythinker_code/web/static/brand/fonts/dm-sans-latin-ext.woff2 +0 -0
  756. pythinker_code/web/static/brand/fonts/dm-sans-latin.woff2 +0 -0
  757. pythinker_code/web/static/brand/fonts/instrument-sans-latin-ext.woff2 +0 -0
  758. pythinker_code/web/static/brand/fonts/instrument-sans-latin.woff2 +0 -0
  759. pythinker_code/web/static/brand/fonts/instrument-serif-latin-ext.woff2 +0 -0
  760. pythinker_code/web/static/brand/fonts/instrument-serif-latin.woff2 +0 -0
  761. pythinker_code/web/static/brand/fonts/libre-baskerville-italic-latin-ext.woff2 +0 -0
  762. pythinker_code/web/static/brand/fonts/libre-baskerville-italic-latin.woff2 +0 -0
  763. pythinker_code/web/static/brand/fonts/libre-baskerville-latin-ext.woff2 +0 -0
  764. pythinker_code/web/static/brand/fonts/libre-baskerville-latin.woff2 +0 -0
  765. pythinker_code/web/static/brand/fonts/roboto-latin-ext.woff2 +0 -0
  766. pythinker_code/web/static/brand/fonts/roboto-latin.woff2 +0 -0
  767. pythinker_code/web/static/brand/icon-192.png +0 -0
  768. pythinker_code/web/static/brand/icon-512.png +0 -0
  769. pythinker_code/web/static/brand/icon.svg +1 -0
  770. pythinker_code/web/static/brand/logo.png +0 -0
  771. pythinker_code/web/static/brand/pythinker_animated.svg +79 -0
  772. pythinker_code/web/static/brand/robots.txt +4 -0
  773. pythinker_code/web/static/index.html +15 -0
  774. pythinker_code/web/static/logo.png +0 -0
  775. pythinker_code/web/store/__init__.py +1 -0
  776. pythinker_code/web/store/sessions.py +433 -0
  777. pythinker_code/wire/__init__.py +148 -0
  778. pythinker_code/wire/file.py +151 -0
  779. pythinker_code/wire/jsonrpc.py +263 -0
  780. pythinker_code/wire/protocol.py +2 -0
  781. pythinker_code/wire/root_hub.py +27 -0
  782. pythinker_code/wire/serde.py +26 -0
  783. pythinker_code/wire/server.py +1072 -0
  784. pythinker_code/wire/types.py +698 -0
  785. pythinker_code-0.8.0.dist-info/METADATA +706 -0
  786. pythinker_code-0.8.0.dist-info/RECORD +790 -0
  787. pythinker_code-0.8.0.dist-info/WHEEL +4 -0
  788. pythinker_code-0.8.0.dist-info/entry_points.txt +4 -0
  789. pythinker_code-0.8.0.dist-info/licenses/LICENSE +202 -0
  790. pythinker_code-0.8.0.dist-info/licenses/NOTICE +14 -0
@@ -0,0 +1,935 @@
1
+ # This file is modified from https://github.com/Textualize/rich/blob/4d6d631a3d2deddf8405522d4b8c976a6d35726c/rich/markdown.py
2
+ # pyright: standard
3
+
4
+ from __future__ import annotations
5
+
6
+ import sys
7
+ from collections.abc import Iterable, Mapping
8
+ from typing import ClassVar, get_args
9
+
10
+ from markdown_it import MarkdownIt
11
+ from markdown_it.token import Token
12
+ from rich import box
13
+ from rich._loop import loop_first
14
+ from rich._stack import Stack
15
+ from rich.console import Console, ConsoleOptions, JustifyMethod, RenderResult
16
+ from rich.containers import Renderables
17
+ from rich.jupyter import JupyterMixin
18
+ from rich.rule import Rule
19
+ from rich.segment import Segment
20
+ from rich.style import Style, StyleStack
21
+ from rich.syntax import Syntax, SyntaxTheme
22
+ from rich.table import Table
23
+ from rich.text import Text, TextType
24
+
25
+ from pythinker_code.utils.rich.syntax import PYTHINKER_ANSI_THEME_NAME, resolve_code_theme
26
+
27
+ LIST_INDENT_WIDTH = 2
28
+
29
+ _FALLBACK_STYLES: Mapping[str, Style] = {
30
+ "markdown.paragraph": Style(),
31
+ "markdown.h1": Style(color="bright_white", bold=True),
32
+ "markdown.h1.underline": Style(color="bright_white", bold=True),
33
+ "markdown.h2": Style(color="white", bold=True, underline=True),
34
+ "markdown.h3": Style(bold=True),
35
+ "markdown.h4": Style(bold=True),
36
+ "markdown.h5": Style(bold=True),
37
+ "markdown.h6": Style(dim=True, italic=True),
38
+ "markdown.code": Style(color="bright_cyan", bold=True),
39
+ "markdown.code_block": Style(color="bright_cyan"),
40
+ "markdown.item": Style(),
41
+ "markdown.item.bullet": Style(),
42
+ "markdown.item.number": Style(),
43
+ "markdown.em": Style(italic=True),
44
+ "markdown.strong": Style(bold=True),
45
+ "markdown.s": Style(strike=True),
46
+ "markdown.link": Style(color="bright_blue", underline=True),
47
+ "markdown.link_url": Style(color="cyan", underline=True),
48
+ "markdown.block_quote": Style(),
49
+ "markdown.hr": Style(color="grey58"),
50
+ }
51
+
52
+
53
+ def _strip_background(text: Text) -> Text:
54
+ """Return a copy of ``text`` with all background colors removed."""
55
+
56
+ clean = Text(
57
+ text.plain,
58
+ justify=text.justify,
59
+ overflow=text.overflow,
60
+ no_wrap=text.no_wrap,
61
+ end=text.end,
62
+ tab_size=text.tab_size,
63
+ )
64
+
65
+ if text.style:
66
+ base_style = text.style
67
+ if not isinstance(base_style, Style):
68
+ base_style = Style.parse(str(base_style))
69
+ base_style = base_style.copy()
70
+ if base_style._bgcolor is not None:
71
+ base_style._bgcolor = None
72
+ clean.stylize(base_style, 0, len(clean))
73
+
74
+ for span in text.spans:
75
+ style = span.style
76
+ if style is None:
77
+ continue
78
+ new_style = Style.parse(str(style)) if not isinstance(style, Style) else style.copy()
79
+ if new_style._bgcolor is not None:
80
+ new_style._bgcolor = None
81
+ clean.stylize(new_style, span.start, span.end)
82
+
83
+ return clean
84
+
85
+
86
+ class MarkdownElement:
87
+ new_line: ClassVar[bool] = True
88
+
89
+ @classmethod
90
+ def create(cls, markdown: Markdown, token: Token) -> MarkdownElement:
91
+ """Factory to create markdown element,
92
+
93
+ Args:
94
+ markdown (Markdown): The parent Markdown object.
95
+ token (Token): A node from markdown-it.
96
+
97
+ Returns:
98
+ MarkdownElement: A new markdown element
99
+ """
100
+ return cls()
101
+
102
+ def on_enter(self, context: MarkdownContext) -> None:
103
+ """Called when the node is entered.
104
+
105
+ Args:
106
+ context (MarkdownContext): The markdown context.
107
+ """
108
+
109
+ def on_text(self, context: MarkdownContext, text: TextType) -> None:
110
+ """Called when text is parsed.
111
+
112
+ Args:
113
+ context (MarkdownContext): The markdown context.
114
+ """
115
+
116
+ def on_leave(self, context: MarkdownContext) -> None:
117
+ """Called when the parser leaves the element.
118
+
119
+ Args:
120
+ context (MarkdownContext): [description]
121
+ """
122
+
123
+ def on_child_close(self, context: MarkdownContext, child: MarkdownElement) -> bool:
124
+ """Called when a child element is closed.
125
+
126
+ This method allows a parent element to take over rendering of its children.
127
+
128
+ Args:
129
+ context (MarkdownContext): The markdown context.
130
+ child (MarkdownElement): The child markdown element.
131
+
132
+ Returns:
133
+ bool: Return True to render the element, or False to not render the element.
134
+ """
135
+ return True
136
+
137
+ def __rich_console__(self, console: Console, options: ConsoleOptions) -> RenderResult:
138
+ return ()
139
+
140
+
141
+ class UnknownElement(MarkdownElement):
142
+ """An unknown element.
143
+
144
+ Hopefully there will be no unknown elements, and we will have a MarkdownElement for
145
+ everything in the document.
146
+
147
+ """
148
+
149
+
150
+ class TextElement(MarkdownElement):
151
+ """Base class for elements that render text."""
152
+
153
+ style_name = "none"
154
+
155
+ def on_enter(self, context: MarkdownContext) -> None:
156
+ self.style = context.enter_style(self.style_name)
157
+ self.text = Text(justify="left")
158
+
159
+ def on_text(self, context: MarkdownContext, text: TextType) -> None:
160
+ self.text.append(text, context.current_style if isinstance(text, str) else None)
161
+
162
+ def on_leave(self, context: MarkdownContext) -> None:
163
+ context.leave_style()
164
+
165
+
166
+ class Paragraph(TextElement):
167
+ """A Paragraph."""
168
+
169
+ style_name = "markdown.paragraph"
170
+ justify: JustifyMethod
171
+
172
+ @classmethod
173
+ def create(cls, markdown: Markdown, token: Token) -> Paragraph:
174
+ return cls(justify=markdown.justify or "left")
175
+
176
+ def __init__(self, justify: JustifyMethod) -> None:
177
+ self.justify = justify
178
+
179
+ def __rich_console__(self, console: Console, options: ConsoleOptions) -> RenderResult:
180
+ self.text.justify = self.justify
181
+ yield self.text
182
+
183
+
184
+ class Heading(TextElement):
185
+ """A heading."""
186
+
187
+ @classmethod
188
+ def create(cls, markdown: Markdown, token: Token) -> Heading:
189
+ return cls(token.tag)
190
+
191
+ def on_enter(self, context: MarkdownContext) -> None:
192
+ self.text = Text()
193
+ context.enter_style(self.style_name)
194
+
195
+ def __init__(self, tag: str) -> None:
196
+ self.tag = tag
197
+ self.style_name = f"markdown.{tag}"
198
+ super().__init__()
199
+
200
+ def __rich_console__(self, console: Console, options: ConsoleOptions) -> RenderResult:
201
+ text = self.text
202
+ text.justify = "left"
203
+ width = max(1, text.cell_len)
204
+
205
+ if self.tag == "h1":
206
+ underline = Text("═" * width)
207
+ underline.stylize("markdown.h1.underline")
208
+ yield text
209
+ yield underline
210
+ else:
211
+ yield text
212
+
213
+
214
+ class CodeBlock(TextElement):
215
+ """A code block with syntax highlighting."""
216
+
217
+ style_name = "markdown.code_block"
218
+
219
+ @classmethod
220
+ def create(cls, markdown: Markdown, token: Token) -> CodeBlock:
221
+ node_info = token.info or ""
222
+ lexer_name = node_info.partition(" ")[0]
223
+ return cls(lexer_name or "text", markdown.code_theme)
224
+
225
+ def __init__(self, lexer_name: str, theme: str | SyntaxTheme) -> None:
226
+ self.lexer_name = lexer_name
227
+ self.theme = theme
228
+
229
+ def __rich_console__(self, console: Console, options: ConsoleOptions) -> RenderResult:
230
+ code = str(self.text).rstrip()
231
+ syntax = Syntax(
232
+ code,
233
+ self.lexer_name,
234
+ theme=self.theme,
235
+ word_wrap=True,
236
+ background_color=None,
237
+ padding=0,
238
+ )
239
+ highlighted = syntax.highlight(code)
240
+ highlighted.rstrip()
241
+ stripped = _strip_background(highlighted)
242
+ stripped.rstrip()
243
+ yield stripped
244
+
245
+
246
+ class BlockQuote(TextElement):
247
+ """A block quote."""
248
+
249
+ style_name = "markdown.block_quote"
250
+
251
+ def __init__(self) -> None:
252
+ self.elements: Renderables = Renderables()
253
+
254
+ def on_child_close(self, context: MarkdownContext, child: MarkdownElement) -> bool:
255
+ self.elements.append(child)
256
+ return False
257
+
258
+ def __rich_console__(self, console: Console, options: ConsoleOptions) -> RenderResult:
259
+ render_options = options.update(width=options.max_width - 4)
260
+ style = self.style.without_color
261
+ lines = console.render_lines(self.elements, render_options, style=style)
262
+ new_line = Segment("\n")
263
+ padding = Segment("▌ ", style)
264
+ for line in lines:
265
+ yield padding
266
+ yield from line
267
+ yield new_line
268
+
269
+
270
+ class HorizontalRule(MarkdownElement):
271
+ """A horizontal rule to divide sections."""
272
+
273
+ new_line = False
274
+
275
+ def __rich_console__(self, console: Console, options: ConsoleOptions) -> RenderResult:
276
+ style = _FALLBACK_STYLES["markdown.hr"].copy()
277
+ yield Rule(style=style)
278
+
279
+
280
+ class TableElement(MarkdownElement):
281
+ """MarkdownElement corresponding to `table_open`."""
282
+
283
+ def __init__(self) -> None:
284
+ self.header: TableHeaderElement | None = None
285
+ self.body: TableBodyElement | None = None
286
+
287
+ def on_child_close(self, context: MarkdownContext, child: MarkdownElement) -> bool:
288
+ if isinstance(child, TableHeaderElement):
289
+ self.header = child
290
+ elif isinstance(child, TableBodyElement):
291
+ self.body = child
292
+ else:
293
+ raise RuntimeError("Couldn't process markdown table.")
294
+ return False
295
+
296
+ def __rich_console__(self, console: Console, options: ConsoleOptions) -> RenderResult:
297
+ headers = (
298
+ [column.content.plain.strip() for column in self.header.row.cells]
299
+ if self.header is not None and self.header.row is not None
300
+ else []
301
+ )
302
+ rows = [row.cells for row in self.body.rows] if self.body is not None else []
303
+
304
+ if headers and rows and _table_should_render_as_records(headers, rows):
305
+ for row_index, row in enumerate(rows, start=1):
306
+ yield _record_title(row_index, row)
307
+ for header, cell in zip(headers[1:], row[1:], strict=False):
308
+ value = _cell_plain(cell.content)
309
+ if not value:
310
+ continue
311
+ line = Text(" ")
312
+ line.append(f"{header}: ", style="bold")
313
+ line.append_text(cell.content)
314
+ yield line
315
+ return
316
+
317
+ table = Table(box=box.SIMPLE_HEAVY, show_edge=False)
318
+
319
+ if self.header is not None and self.header.row is not None:
320
+ for column in self.header.row.cells:
321
+ table.add_column(column.content)
322
+
323
+ if self.body is not None:
324
+ for row in self.body.rows:
325
+ row_content = [element.content for element in row.cells]
326
+ table.add_row(*row_content)
327
+
328
+ yield table
329
+
330
+
331
+ def _cell_plain(cell: Text) -> str:
332
+ return cell.plain.strip()
333
+
334
+
335
+ def _table_should_render_as_records(headers: list[str], rows: list[list[TableDataElement]]) -> bool:
336
+ if len(headers) >= 4:
337
+ return True
338
+ return any(len(_cell_plain(cell.content)) > 48 for row in rows for cell in row)
339
+
340
+
341
+ def _record_title(row_index: int, row: list[TableDataElement]) -> Text:
342
+ title = _cell_plain(row[0].content) if row else "Row"
343
+ return Text(f"{row_index}. {title}", style="bold")
344
+
345
+
346
+ class TableHeaderElement(MarkdownElement):
347
+ """MarkdownElement corresponding to `thead_open` and `thead_close`."""
348
+
349
+ def __init__(self) -> None:
350
+ self.row: TableRowElement | None = None
351
+
352
+ def on_child_close(self, context: MarkdownContext, child: MarkdownElement) -> bool:
353
+ assert isinstance(child, TableRowElement)
354
+ self.row = child
355
+ return False
356
+
357
+
358
+ class TableBodyElement(MarkdownElement):
359
+ """MarkdownElement corresponding to `tbody_open` and `tbody_close`."""
360
+
361
+ def __init__(self) -> None:
362
+ self.rows: list[TableRowElement] = []
363
+
364
+ def on_child_close(self, context: MarkdownContext, child: MarkdownElement) -> bool:
365
+ assert isinstance(child, TableRowElement)
366
+ self.rows.append(child)
367
+ return False
368
+
369
+
370
+ class TableRowElement(MarkdownElement):
371
+ """MarkdownElement corresponding to `tr_open` and `tr_close`."""
372
+
373
+ def __init__(self) -> None:
374
+ self.cells: list[TableDataElement] = []
375
+
376
+ def on_child_close(self, context: MarkdownContext, child: MarkdownElement) -> bool:
377
+ assert isinstance(child, TableDataElement)
378
+ self.cells.append(child)
379
+ return False
380
+
381
+
382
+ class TableDataElement(MarkdownElement):
383
+ """MarkdownElement corresponding to `td_open` and `td_close`
384
+ and `th_open` and `th_close`."""
385
+
386
+ @classmethod
387
+ def create(cls, markdown: Markdown, token: Token) -> MarkdownElement:
388
+ style = str(token.attrs.get("style")) or ""
389
+
390
+ justify: JustifyMethod
391
+ if "text-align:right" in style:
392
+ justify = "right"
393
+ elif "text-align:center" in style:
394
+ justify = "center"
395
+ elif "text-align:left" in style:
396
+ justify = "left"
397
+ else:
398
+ justify = "default"
399
+
400
+ assert justify in get_args(JustifyMethod)
401
+ return cls(justify=justify)
402
+
403
+ def __init__(self, justify: JustifyMethod) -> None:
404
+ self.content: Text = Text("", justify=justify)
405
+ self.justify = justify
406
+
407
+ def on_text(self, context: MarkdownContext, text: TextType) -> None:
408
+ text = Text(text) if isinstance(text, str) else text
409
+ text.stylize(context.current_style)
410
+ self.content.append_text(text)
411
+
412
+
413
+ class ListElement(MarkdownElement):
414
+ """A list element."""
415
+
416
+ @classmethod
417
+ def create(cls, markdown: Markdown, token: Token) -> ListElement:
418
+ return cls(token.type, int(token.attrs.get("start", 1)))
419
+
420
+ def __init__(self, list_type: str, list_start: int | None) -> None:
421
+ self.items: list[ListItem] = []
422
+ self.list_type = list_type
423
+ self.list_start = list_start
424
+
425
+ def on_child_close(self, context: MarkdownContext, child: MarkdownElement) -> bool:
426
+ assert isinstance(child, ListItem)
427
+ self.items.append(child)
428
+ return False
429
+
430
+ def __rich_console__(self, console: Console, options: ConsoleOptions) -> RenderResult:
431
+ if self.list_type == "bullet_list_open":
432
+ for item in self.items:
433
+ yield from item.render_bullet(console, options)
434
+ else:
435
+ number = 1 if self.list_start is None else self.list_start
436
+ last_number = number + len(self.items)
437
+ for index, item in enumerate(self.items):
438
+ yield from item.render_number(console, options, number + index, last_number)
439
+
440
+
441
+ class ListItem(TextElement):
442
+ """An item in a list."""
443
+
444
+ style_name = "markdown.item"
445
+
446
+ @staticmethod
447
+ def _line_starts_with_list_marker(text: str) -> bool:
448
+ stripped = text.lstrip()
449
+ if not stripped:
450
+ return False
451
+ if stripped.startswith(("• ", "- ", "* ")):
452
+ return True
453
+ index = 0
454
+ while index < len(stripped) and stripped[index].isdigit():
455
+ index += 1
456
+ if index == 0 or index >= len(stripped):
457
+ return False
458
+ marker = stripped[index]
459
+ has_space = index + 1 < len(stripped) and stripped[index + 1] == " "
460
+ return marker in {".", ")"} and has_space
461
+
462
+ @classmethod
463
+ def create(cls, markdown: Markdown, token: Token) -> MarkdownElement:
464
+ # `list_item_open` levels grow by 2 for each nested list depth.
465
+ depth = max(0, (token.level - 1) // 2)
466
+ return cls(indent=depth)
467
+
468
+ def __init__(self, indent: int = 0) -> None:
469
+ self.indent = indent
470
+ self.elements: Renderables = Renderables()
471
+
472
+ def on_child_close(self, context: MarkdownContext, child: MarkdownElement) -> bool:
473
+ self.elements.append(child)
474
+ return False
475
+
476
+ def render_bullet(self, console: Console, options: ConsoleOptions) -> RenderResult:
477
+ lines = console.render_lines(self.elements, options, style=self.style)
478
+ indent_padding_len = LIST_INDENT_WIDTH * self.indent
479
+ indent_text = " " * indent_padding_len
480
+ bullet = Segment("• ")
481
+ new_line = Segment("\n")
482
+ bullet_width = len(bullet.text)
483
+ for first, line in loop_first(lines):
484
+ if first:
485
+ if indent_text:
486
+ yield Segment(indent_text)
487
+ yield bullet
488
+ else:
489
+ plain = "".join(segment.text for segment in line)
490
+ if self._line_starts_with_list_marker(plain):
491
+ prefix = ""
492
+ else:
493
+ existing = len(plain) - len(plain.lstrip(" "))
494
+ target = indent_padding_len + bullet_width
495
+ missing = max(0, target - existing)
496
+ prefix = " " * missing
497
+ if prefix:
498
+ yield Segment(prefix)
499
+ yield from line
500
+ yield new_line
501
+
502
+ def render_number(
503
+ self, console: Console, options: ConsoleOptions, number: int, last_number: int
504
+ ) -> RenderResult:
505
+ lines = console.render_lines(self.elements, options, style=self.style)
506
+ new_line = Segment("\n")
507
+ indent_padding_len = LIST_INDENT_WIDTH * self.indent
508
+ indent_text = " " * indent_padding_len
509
+ numeral_text = f"{number}. "
510
+ numeral = Segment(numeral_text)
511
+ numeral_width = len(numeral_text)
512
+ for first, line in loop_first(lines):
513
+ if first:
514
+ if indent_text:
515
+ yield Segment(indent_text)
516
+ yield numeral
517
+ else:
518
+ plain = "".join(segment.text for segment in line)
519
+ if self._line_starts_with_list_marker(plain):
520
+ prefix = ""
521
+ else:
522
+ existing = len(plain) - len(plain.lstrip(" "))
523
+ target = indent_padding_len + numeral_width
524
+ missing = max(0, target - existing)
525
+ prefix = " " * missing
526
+ if prefix:
527
+ yield Segment(prefix)
528
+ yield from line
529
+ yield new_line
530
+
531
+
532
+ class Link(TextElement):
533
+ @classmethod
534
+ def create(cls, markdown: Markdown, token: Token) -> MarkdownElement:
535
+ url = token.attrs.get("href", "#")
536
+ return cls(token.content, str(url))
537
+
538
+ def __init__(self, text: str, href: str):
539
+ self.text = Text(text)
540
+ self.href = href
541
+
542
+
543
+ class ImageItem(TextElement):
544
+ """Renders a placeholder for an image."""
545
+
546
+ new_line = False
547
+
548
+ @classmethod
549
+ def create(cls, markdown: Markdown, token: Token) -> MarkdownElement:
550
+ """Factory to create markdown element,
551
+
552
+ Args:
553
+ markdown (Markdown): The parent Markdown object.
554
+ token (Any): A token from markdown-it.
555
+
556
+ Returns:
557
+ MarkdownElement: A new markdown element
558
+ """
559
+ return cls(str(token.attrs.get("src", "")), markdown.hyperlinks)
560
+
561
+ def __init__(self, destination: str, hyperlinks: bool) -> None:
562
+ self.destination = destination
563
+ self.hyperlinks = hyperlinks
564
+ self.link: str | None = None
565
+ super().__init__()
566
+
567
+ def on_enter(self, context: MarkdownContext) -> None:
568
+ self.link = context.current_style.link
569
+ self.text = Text(justify="left")
570
+ super().on_enter(context)
571
+
572
+ def __rich_console__(self, console: Console, options: ConsoleOptions) -> RenderResult:
573
+ link_style = Style(link=self.link or self.destination or None)
574
+ title = self.text or Text(self.destination.strip("/").rsplit("/", 1)[-1])
575
+ if self.hyperlinks:
576
+ title.stylize(link_style)
577
+ text = Text.assemble("🌆 ", title, " ", end="")
578
+ yield text
579
+
580
+
581
+ class MarkdownContext:
582
+ """Manages the console render state."""
583
+
584
+ def __init__(
585
+ self,
586
+ console: Console,
587
+ options: ConsoleOptions,
588
+ style: Style,
589
+ fallback_styles: Mapping[str, Style],
590
+ inline_code_lexer: str | None = None,
591
+ inline_code_theme: str | SyntaxTheme = PYTHINKER_ANSI_THEME_NAME,
592
+ ) -> None:
593
+ self.console = console
594
+ self.options = options
595
+ self.style_stack: StyleStack = StyleStack(style)
596
+ self.stack: Stack[MarkdownElement] = Stack()
597
+ self._fallback_styles = fallback_styles
598
+
599
+ self._syntax: Syntax | None = None
600
+ if inline_code_lexer is not None:
601
+ self._syntax = Syntax("", inline_code_lexer, theme=inline_code_theme)
602
+
603
+ @property
604
+ def current_style(self) -> Style:
605
+ """Current style which is the product of all styles on the stack."""
606
+ return self.style_stack.current
607
+
608
+ def on_text(self, text: str, node_type: str) -> None:
609
+ """Called when the parser visits text."""
610
+ if node_type in {"fence", "code_inline"} and self._syntax is not None:
611
+ highlighted = self._syntax.highlight(text)
612
+ highlighted.rstrip()
613
+ stripped = _strip_background(highlighted)
614
+ combined = Text.assemble(stripped, style=self.style_stack.current)
615
+ self.stack.top.on_text(self, combined)
616
+ else:
617
+ self.stack.top.on_text(self, text)
618
+
619
+ def enter_style(self, style_name: str | Style) -> Style:
620
+ """Enter a style context."""
621
+ if isinstance(style_name, Style):
622
+ style = style_name
623
+ else:
624
+ fallback = self._fallback_styles.get(style_name, Style())
625
+ style = self.console.get_style(style_name, default=fallback)
626
+ style = fallback + style
627
+ style = style.copy()
628
+ if isinstance(style_name, str) and style_name == "markdown.block_quote":
629
+ style = style.without_color
630
+ if (
631
+ isinstance(style_name, str)
632
+ and style_name in {"markdown.code", "markdown.code_block"}
633
+ and style._bgcolor is not None
634
+ ):
635
+ style._bgcolor = None
636
+ self.style_stack.push(style)
637
+ return self.current_style
638
+
639
+ def leave_style(self) -> Style:
640
+ """Leave a style context."""
641
+ style = self.style_stack.pop()
642
+ return style
643
+
644
+
645
+ class Markdown(JupyterMixin):
646
+ """A Markdown renderable.
647
+
648
+ Args:
649
+ markup (str): A string containing markdown.
650
+ code_theme (str, optional): Pygments theme for code blocks. Defaults to "pythinker-ansi".
651
+ See https://pygments.org/styles/ for code themes.
652
+ justify (JustifyMethod, optional): Justify value for paragraphs. Defaults to None.
653
+ style (Union[str, Style], optional): Optional style to apply to markdown.
654
+ hyperlinks (bool, optional): Enable hyperlinks. Defaults to ``True``.
655
+ inline_code_lexer: (str, optional): Lexer to use if inline code highlighting is
656
+ enabled. Defaults to None.
657
+ inline_code_theme: (Optional[str], optional): Pygments theme for inline code
658
+ highlighting, or None for no highlighting. Defaults to None.
659
+ """
660
+
661
+ elements: ClassVar[dict[str, type[MarkdownElement]]] = {
662
+ "paragraph_open": Paragraph,
663
+ "heading_open": Heading,
664
+ "fence": CodeBlock,
665
+ "code_block": CodeBlock,
666
+ "blockquote_open": BlockQuote,
667
+ "hr": HorizontalRule,
668
+ "bullet_list_open": ListElement,
669
+ "ordered_list_open": ListElement,
670
+ "list_item_open": ListItem,
671
+ "image": ImageItem,
672
+ "table_open": TableElement,
673
+ "tbody_open": TableBodyElement,
674
+ "thead_open": TableHeaderElement,
675
+ "tr_open": TableRowElement,
676
+ "td_open": TableDataElement,
677
+ "th_open": TableDataElement,
678
+ }
679
+
680
+ inlines = {"em", "strong", "code", "s"}
681
+
682
+ def __init__(
683
+ self,
684
+ markup: str,
685
+ code_theme: str = PYTHINKER_ANSI_THEME_NAME,
686
+ justify: JustifyMethod | None = None,
687
+ style: str | Style = "none",
688
+ hyperlinks: bool = True,
689
+ inline_code_lexer: str | None = None,
690
+ inline_code_theme: str | None = None,
691
+ ) -> None:
692
+ parser = MarkdownIt().enable("strikethrough").enable("table")
693
+ self.markup = markup
694
+ self.parsed = parser.parse(markup)
695
+ self.code_theme = resolve_code_theme(code_theme)
696
+ self.justify: JustifyMethod | None = justify
697
+ self.style = style
698
+ self.hyperlinks = hyperlinks
699
+ self.inline_code_lexer = inline_code_lexer
700
+ self.inline_code_theme = resolve_code_theme(inline_code_theme or code_theme)
701
+
702
+ def _flatten_tokens(self, tokens: Iterable[Token]) -> Iterable[Token]:
703
+ """Flattens the token stream."""
704
+ for token in tokens:
705
+ is_fence = token.type == "fence"
706
+ is_image = token.tag == "img"
707
+ if token.children and not (is_image or is_fence):
708
+ yield from self._flatten_tokens(token.children)
709
+ else:
710
+ yield token
711
+
712
+ def __rich_console__(self, console: Console, options: ConsoleOptions) -> RenderResult:
713
+ """Render markdown to the console."""
714
+ style = console.get_style(self.style, default="none")
715
+ options = options.update(height=None)
716
+ context = MarkdownContext(
717
+ console,
718
+ options,
719
+ style,
720
+ _FALLBACK_STYLES,
721
+ inline_code_lexer=self.inline_code_lexer,
722
+ inline_code_theme=self.inline_code_theme,
723
+ )
724
+ tokens = self.parsed
725
+ inline_style_tags = self.inlines
726
+ new_line = False
727
+ _new_line_segment = Segment.line()
728
+ render_started = False
729
+
730
+ for token in self._flatten_tokens(tokens):
731
+ node_type = token.type
732
+ tag = token.tag
733
+
734
+ entering = token.nesting == 1
735
+ exiting = token.nesting == -1
736
+ self_closing = token.nesting == 0
737
+
738
+ if node_type in {"text", "html_inline", "html_block"}:
739
+ # Render HTML tokens as plain text so safeword markup stays visible.
740
+ if context.stack:
741
+ context.on_text(token.content, node_type)
742
+ else:
743
+ # Orphan text/html blocks can appear outside any element (e.g. <analysis>).
744
+ paragraph = Paragraph(justify=self.justify or "left")
745
+ paragraph.on_enter(context)
746
+ paragraph.on_text(context, token.content)
747
+ paragraph.on_leave(context)
748
+ if new_line and render_started:
749
+ yield _new_line_segment
750
+ rendered = console.render(paragraph, context.options)
751
+ for segment in rendered:
752
+ render_started = True
753
+ yield segment
754
+ new_line = paragraph.new_line
755
+ elif node_type == "hardbreak":
756
+ context.on_text("\n", node_type)
757
+ elif node_type == "softbreak":
758
+ context.on_text(" ", node_type)
759
+ elif node_type == "link_open":
760
+ href = str(token.attrs.get("href", ""))
761
+ if self.hyperlinks:
762
+ link_style = console.get_style("markdown.link_url", default="none")
763
+ link_style += Style(link=href)
764
+ context.enter_style(link_style)
765
+ else:
766
+ context.stack.push(Link.create(self, token))
767
+ elif node_type == "link_close":
768
+ if self.hyperlinks:
769
+ context.leave_style()
770
+ else:
771
+ element = context.stack.pop()
772
+ assert isinstance(element, Link)
773
+ link_style = console.get_style("markdown.link", default="none")
774
+ context.enter_style(link_style)
775
+ context.on_text(element.text.plain, node_type)
776
+ context.leave_style()
777
+ context.on_text(" (", node_type)
778
+ link_url_style = console.get_style("markdown.link_url", default="none")
779
+ context.enter_style(link_url_style)
780
+ context.on_text(element.href, node_type)
781
+ context.leave_style()
782
+ context.on_text(")", node_type)
783
+ elif tag in inline_style_tags and node_type != "fence" and node_type != "code_block":
784
+ if entering:
785
+ # If it's an opening inline token e.g. strong, em, etc.
786
+ # Then we move into a style context i.e. push to stack.
787
+ context.enter_style(f"markdown.{tag}")
788
+ elif exiting:
789
+ # If it's a closing inline style, then we pop the style
790
+ # off of the stack, to move out of the context of it...
791
+ context.leave_style()
792
+ else:
793
+ # If it's a self-closing inline style e.g. `code_inline`
794
+ context.enter_style(f"markdown.{tag}")
795
+ if token.content:
796
+ context.on_text(token.content, node_type)
797
+ context.leave_style()
798
+ else:
799
+ # Map the markdown tag -> MarkdownElement renderable
800
+ element_class = self.elements.get(token.type) or UnknownElement
801
+ element = element_class.create(self, token)
802
+
803
+ if entering or self_closing:
804
+ context.stack.push(element)
805
+ element.on_enter(context)
806
+
807
+ if exiting: # CLOSING tag
808
+ element = context.stack.pop()
809
+
810
+ should_render = not context.stack or (
811
+ context.stack and context.stack.top.on_child_close(context, element)
812
+ )
813
+
814
+ if should_render:
815
+ if new_line and render_started:
816
+ yield _new_line_segment
817
+
818
+ rendered = console.render(element, context.options)
819
+ for segment in rendered:
820
+ render_started = True
821
+ yield segment
822
+ elif self_closing: # SELF-CLOSING tags (e.g. text, code, image)
823
+ context.stack.pop()
824
+ text = token.content
825
+ if text is not None:
826
+ element.on_text(context, text)
827
+
828
+ should_render = (
829
+ not context.stack
830
+ or context.stack
831
+ and context.stack.top.on_child_close(context, element)
832
+ )
833
+ if should_render:
834
+ if new_line and node_type != "inline" and render_started:
835
+ yield _new_line_segment
836
+ rendered = console.render(element, context.options)
837
+ for segment in rendered:
838
+ render_started = True
839
+ yield segment
840
+
841
+ if exiting or self_closing:
842
+ element.on_leave(context)
843
+ new_line = element.new_line
844
+
845
+
846
+ if __name__ == "__main__":
847
+ import argparse
848
+ import sys
849
+
850
+ parser = argparse.ArgumentParser(description="Render Markdown to the console with Rich")
851
+ parser.add_argument(
852
+ "path",
853
+ metavar="PATH",
854
+ help="path to markdown file, or - for stdin",
855
+ )
856
+ parser.add_argument(
857
+ "-c",
858
+ "--force-color",
859
+ dest="force_color",
860
+ action="store_true",
861
+ default=None,
862
+ help="force color for non-terminals",
863
+ )
864
+ parser.add_argument(
865
+ "-t",
866
+ "--code-theme",
867
+ dest="code_theme",
868
+ default=PYTHINKER_ANSI_THEME_NAME,
869
+ help='code theme (pygments name or "pythinker-ansi")',
870
+ )
871
+ parser.add_argument(
872
+ "-i",
873
+ "--inline-code-lexer",
874
+ dest="inline_code_lexer",
875
+ default=None,
876
+ help="inline_code_lexer",
877
+ )
878
+ parser.add_argument(
879
+ "-y",
880
+ "--hyperlinks",
881
+ dest="hyperlinks",
882
+ action="store_true",
883
+ help="enable hyperlinks",
884
+ )
885
+ parser.add_argument(
886
+ "-w",
887
+ "--width",
888
+ type=int,
889
+ dest="width",
890
+ default=None,
891
+ help="width of output (default will auto-detect)",
892
+ )
893
+ parser.add_argument(
894
+ "-j",
895
+ "--justify",
896
+ dest="justify",
897
+ action="store_true",
898
+ help="enable full text justify",
899
+ )
900
+ parser.add_argument(
901
+ "-p",
902
+ "--page",
903
+ dest="page",
904
+ action="store_true",
905
+ help="use pager to scroll output",
906
+ )
907
+ args = parser.parse_args()
908
+
909
+ from rich.console import Console
910
+
911
+ if args.path == "-":
912
+ markdown_body = sys.stdin.read()
913
+ else:
914
+ with open(args.path, encoding="utf-8") as markdown_file:
915
+ markdown_body = markdown_file.read()
916
+
917
+ markdown = Markdown(
918
+ markdown_body,
919
+ justify="full" if args.justify else "left",
920
+ code_theme=args.code_theme,
921
+ hyperlinks=args.hyperlinks,
922
+ inline_code_lexer=args.inline_code_lexer,
923
+ )
924
+ if args.page:
925
+ import io
926
+ import pydoc
927
+
928
+ fileio = io.StringIO()
929
+ console = Console(file=fileio, force_terminal=args.force_color, width=args.width)
930
+ console.print(markdown)
931
+ pydoc.pager(fileio.getvalue())
932
+
933
+ else:
934
+ console = Console(force_terminal=args.force_color, width=args.width, record=True)
935
+ console.print(markdown)