scitex 2.8.1__py3-none-any.whl → 2.10.2__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 (415) hide show
  1. scitex/__init__.py +15 -7
  2. scitex/__version__.py +1 -2
  3. scitex/_install_guide.py +250 -0
  4. scitex/_optional_deps.py +206 -39
  5. scitex/ai/_gen_ai/_Groq.py +2 -4
  6. scitex/ai/_gen_ai/_OpenAI.py +5 -2
  7. scitex/ai/_gen_ai/_Perplexity.py +20 -6
  8. scitex/audio/__init__.py +24 -15
  9. scitex/audio/_cross_process_lock.py +139 -0
  10. scitex/audio/_mcp_handlers.py +256 -0
  11. scitex/audio/_mcp_tool_schemas.py +203 -0
  12. scitex/audio/engines/elevenlabs_engine.py +5 -2
  13. scitex/audio/mcp_server.py +98 -457
  14. scitex/bridge/__init__.py +30 -19
  15. scitex/bridge/_figrecipe.py +245 -0
  16. scitex/bridge/_helpers.py +2 -1
  17. scitex/bridge/_plt_vis.py +23 -10
  18. scitex/bridge/_stats_plt.py +18 -5
  19. scitex/bridge/_stats_vis.py +16 -2
  20. scitex/browser/__init__.py +84 -44
  21. scitex/browser/automation/__init__.py +5 -1
  22. scitex/browser/core/BrowserMixin.py +17 -4
  23. scitex/browser/core/__init__.py +11 -2
  24. scitex/browser/remote/CaptchaHandler.py +1 -1
  25. scitex/browser/remote/ZenRowsAPIClient.py +1 -1
  26. scitex/capture/grid.py +487 -0
  27. scitex/capture/mcp_handlers.py +401 -0
  28. scitex/capture/mcp_tool_defs.py +192 -0
  29. scitex/capture/mcp_tools.py +241 -0
  30. scitex/capture/mcp_utils.py +30 -0
  31. scitex/cli/convert.py +421 -0
  32. scitex/cli/main.py +6 -4
  33. scitex/datetime/__init__.py +46 -0
  34. scitex/datetime/_linspace.py +100 -0
  35. scitex/datetime/_normalize_timestamp.py +306 -0
  36. scitex/db/_delete_duplicates.py +4 -4
  37. scitex/db/_sqlite3/_delete_duplicates.py +11 -2
  38. scitex/dev/plt/__init__.py +61 -62
  39. scitex/dev/plt/demo_plotters/__init__.py +0 -0
  40. scitex/dev/plt/demo_plotters/plot_mpl_axhline.py +28 -0
  41. scitex/dev/plt/demo_plotters/plot_mpl_axhspan.py +28 -0
  42. scitex/dev/plt/demo_plotters/plot_mpl_axvline.py +28 -0
  43. scitex/dev/plt/demo_plotters/plot_mpl_axvspan.py +28 -0
  44. scitex/dev/plt/demo_plotters/plot_mpl_bar.py +29 -0
  45. scitex/dev/plt/demo_plotters/plot_mpl_barh.py +29 -0
  46. scitex/dev/plt/demo_plotters/plot_mpl_boxplot.py +28 -0
  47. scitex/dev/plt/demo_plotters/plot_mpl_contour.py +31 -0
  48. scitex/dev/plt/demo_plotters/plot_mpl_contourf.py +31 -0
  49. scitex/dev/plt/demo_plotters/plot_mpl_errorbar.py +30 -0
  50. scitex/dev/plt/demo_plotters/plot_mpl_eventplot.py +28 -0
  51. scitex/dev/plt/demo_plotters/plot_mpl_fill.py +30 -0
  52. scitex/dev/plt/demo_plotters/plot_mpl_fill_between.py +31 -0
  53. scitex/dev/plt/demo_plotters/plot_mpl_hexbin.py +28 -0
  54. scitex/dev/plt/demo_plotters/plot_mpl_hist.py +28 -0
  55. scitex/dev/plt/demo_plotters/plot_mpl_hist2d.py +28 -0
  56. scitex/dev/plt/demo_plotters/plot_mpl_imshow.py +29 -0
  57. scitex/dev/plt/demo_plotters/plot_mpl_pcolormesh.py +31 -0
  58. scitex/dev/plt/demo_plotters/plot_mpl_pie.py +29 -0
  59. scitex/dev/plt/demo_plotters/plot_mpl_plot.py +29 -0
  60. scitex/dev/plt/demo_plotters/plot_mpl_quiver.py +31 -0
  61. scitex/dev/plt/demo_plotters/plot_mpl_scatter.py +28 -0
  62. scitex/dev/plt/demo_plotters/plot_mpl_stackplot.py +31 -0
  63. scitex/dev/plt/demo_plotters/plot_mpl_stem.py +29 -0
  64. scitex/dev/plt/demo_plotters/plot_mpl_step.py +29 -0
  65. scitex/dev/plt/demo_plotters/plot_mpl_violinplot.py +28 -0
  66. scitex/dev/plt/demo_plotters/plot_sns_barplot.py +29 -0
  67. scitex/dev/plt/demo_plotters/plot_sns_boxplot.py +29 -0
  68. scitex/dev/plt/demo_plotters/plot_sns_heatmap.py +28 -0
  69. scitex/dev/plt/demo_plotters/plot_sns_histplot.py +29 -0
  70. scitex/dev/plt/demo_plotters/plot_sns_kdeplot.py +29 -0
  71. scitex/dev/plt/demo_plotters/plot_sns_lineplot.py +31 -0
  72. scitex/dev/plt/demo_plotters/plot_sns_scatterplot.py +29 -0
  73. scitex/dev/plt/demo_plotters/plot_sns_stripplot.py +29 -0
  74. scitex/dev/plt/demo_plotters/plot_sns_swarmplot.py +29 -0
  75. scitex/dev/plt/demo_plotters/plot_sns_violinplot.py +29 -0
  76. scitex/dev/plt/demo_plotters/plot_stx_bar.py +29 -0
  77. scitex/dev/plt/demo_plotters/plot_stx_barh.py +29 -0
  78. scitex/dev/plt/demo_plotters/plot_stx_box.py +28 -0
  79. scitex/dev/plt/demo_plotters/plot_stx_boxplot.py +28 -0
  80. scitex/dev/plt/demo_plotters/plot_stx_conf_mat.py +28 -0
  81. scitex/dev/plt/demo_plotters/plot_stx_contour.py +31 -0
  82. scitex/dev/plt/demo_plotters/plot_stx_ecdf.py +28 -0
  83. scitex/dev/plt/demo_plotters/plot_stx_errorbar.py +30 -0
  84. scitex/dev/plt/demo_plotters/plot_stx_fill_between.py +31 -0
  85. scitex/dev/plt/demo_plotters/plot_stx_fillv.py +28 -0
  86. scitex/dev/plt/demo_plotters/plot_stx_heatmap.py +28 -0
  87. scitex/dev/plt/demo_plotters/plot_stx_image.py +28 -0
  88. scitex/dev/plt/demo_plotters/plot_stx_imshow.py +28 -0
  89. scitex/dev/plt/demo_plotters/plot_stx_joyplot.py +28 -0
  90. scitex/dev/plt/demo_plotters/plot_stx_kde.py +28 -0
  91. scitex/dev/plt/demo_plotters/plot_stx_line.py +28 -0
  92. scitex/dev/plt/demo_plotters/plot_stx_mean_ci.py +28 -0
  93. scitex/dev/plt/demo_plotters/plot_stx_mean_std.py +28 -0
  94. scitex/dev/plt/demo_plotters/plot_stx_median_iqr.py +28 -0
  95. scitex/dev/plt/demo_plotters/plot_stx_raster.py +28 -0
  96. scitex/dev/plt/demo_plotters/plot_stx_rectangle.py +28 -0
  97. scitex/dev/plt/demo_plotters/plot_stx_scatter.py +29 -0
  98. scitex/dev/plt/demo_plotters/plot_stx_shaded_line.py +29 -0
  99. scitex/dev/plt/demo_plotters/plot_stx_violin.py +28 -0
  100. scitex/dev/plt/demo_plotters/plot_stx_violinplot.py +28 -0
  101. scitex/dev/plt/mpl/get_dir_ax.py +46 -0
  102. scitex/dev/plt/mpl/get_signatures.py +176 -0
  103. scitex/dev/plt/mpl/get_signatures_details.py +522 -0
  104. scitex/dict/_pop_keys.py +1 -7
  105. scitex/dsp/__init__.py +15 -10
  106. scitex/dsp/add_noise.py +5 -2
  107. scitex/dsp/example.py +35 -22
  108. scitex/dsp/filt.py +8 -3
  109. scitex/dsp/reference.py +3 -2
  110. scitex/dsp/utils/__init__.py +2 -1
  111. scitex/dsp/utils/_differential_bandpass_filters.py +14 -4
  112. scitex/dt/__init__.py +39 -2
  113. scitex/errors.py +82 -521
  114. scitex/fig/__init__.py +4 -4
  115. scitex/fig/editor/edit/panel_loader.py +1 -1
  116. scitex/fig/io/_bundle.py +7 -7
  117. scitex/fts/README.md +262 -0
  118. scitex/fts/TODO.md +66 -0
  119. scitex/fts/__init__.py +90 -0
  120. scitex/fts/_bundle/README_IN_BUNDLE.md +102 -0
  121. scitex/fts/_bundle/_FTS.py +657 -0
  122. scitex/fts/_bundle/__init__.py +38 -0
  123. scitex/fts/_bundle/_children.py +216 -0
  124. scitex/fts/_bundle/_conversion/__init__.py +15 -0
  125. scitex/fts/_bundle/_conversion/_bundle2dict.py +44 -0
  126. scitex/fts/_bundle/_conversion/_dict2bundle.py +50 -0
  127. scitex/fts/_bundle/_dataclasses/_Axes.py +57 -0
  128. scitex/fts/_bundle/_dataclasses/_BBox.py +54 -0
  129. scitex/fts/_bundle/_dataclasses/_ColumnDef.py +72 -0
  130. scitex/fts/_bundle/_dataclasses/_DataFormat.py +40 -0
  131. scitex/fts/_bundle/_dataclasses/_DataInfo.py +135 -0
  132. scitex/fts/_bundle/_dataclasses/_DataSource.py +44 -0
  133. scitex/fts/_bundle/_dataclasses/_Node.py +319 -0
  134. scitex/fts/_bundle/_dataclasses/_NodeRefs.py +45 -0
  135. scitex/fts/_bundle/_dataclasses/_SizeMM.py +38 -0
  136. scitex/fts/_bundle/_dataclasses/__init__.py +35 -0
  137. scitex/fts/_bundle/_extractors/__init__.py +32 -0
  138. scitex/fts/_bundle/_extractors/_extract_bar.py +131 -0
  139. scitex/fts/_bundle/_extractors/_extract_line.py +71 -0
  140. scitex/fts/_bundle/_extractors/_extract_scatter.py +79 -0
  141. scitex/fts/_bundle/_loader.py +134 -0
  142. scitex/fts/_bundle/_mpl_helpers.py +389 -0
  143. scitex/fts/_bundle/_saver.py +269 -0
  144. scitex/fts/_bundle/_storage.py +200 -0
  145. scitex/fts/_bundle/_utils/__init__.py +55 -0
  146. scitex/fts/_bundle/_utils/_const.py +26 -0
  147. scitex/fts/_bundle/_utils/_errors.py +73 -0
  148. scitex/fts/_bundle/_utils/_generate.py +21 -0
  149. scitex/fts/_bundle/_utils/_types.py +76 -0
  150. scitex/fts/_bundle/_validation.py +434 -0
  151. scitex/fts/_bundle/_zipbundle.py +165 -0
  152. scitex/fts/_fig/__init__.py +22 -0
  153. scitex/fts/_fig/_backend/__init__.py +53 -0
  154. scitex/fts/_fig/_backend/_export.py +165 -0
  155. scitex/fts/_fig/_backend/_parser.py +188 -0
  156. scitex/fts/_fig/_backend/_render.py +538 -0
  157. scitex/fts/_fig/_composite.py +345 -0
  158. scitex/fts/_fig/_dataclasses/_ChannelEncoding.py +46 -0
  159. scitex/fts/_fig/_dataclasses/_Encoding.py +82 -0
  160. scitex/fts/_fig/_dataclasses/_Theme.py +441 -0
  161. scitex/fts/_fig/_dataclasses/_TraceEncoding.py +52 -0
  162. scitex/fts/_fig/_dataclasses/__init__.py +47 -0
  163. scitex/fts/_fig/_editor/__init__.py +14 -0
  164. scitex/fts/_fig/_editor/_cui/__init__.py +33 -0
  165. scitex/fts/_fig/_editor/_cui/_backend_detector.py +39 -0
  166. scitex/fts/_fig/_editor/_cui/_bundle_resolver.py +366 -0
  167. scitex/fts/_fig/_editor/_cui/_editor_launcher.py +175 -0
  168. scitex/fts/_fig/_editor/_cui/_manual_handler.py +52 -0
  169. scitex/fts/_fig/_editor/_cui/_panel_loader.py +246 -0
  170. scitex/fts/_fig/_editor/_cui/_path_resolver.py +66 -0
  171. scitex/fts/_fig/_editor/_defaults.py +300 -0
  172. scitex/fts/_fig/_editor/_gui/__init__.py +11 -0
  173. scitex/fts/_fig/_editor/_gui/_flask_editor/__init__.py +20 -0
  174. scitex/fts/_fig/_editor/_gui/_flask_editor/_bbox.py +1339 -0
  175. scitex/fts/_fig/_editor/_gui/_flask_editor/_core.py +1688 -0
  176. scitex/fts/_fig/_editor/_gui/_flask_editor/_plotter.py +664 -0
  177. scitex/fts/_fig/_editor/_gui/_flask_editor/_renderer.py +853 -0
  178. scitex/fts/_fig/_editor/_gui/_flask_editor/_utils.py +79 -0
  179. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/base/reset.css +41 -0
  180. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/base/typography.css +16 -0
  181. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/base/variables.css +85 -0
  182. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/buttons.css +217 -0
  183. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/context-menu.css +93 -0
  184. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/dropdown.css +57 -0
  185. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/forms.css +112 -0
  186. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/modal.css +59 -0
  187. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/sections.css +212 -0
  188. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/canvas.css +176 -0
  189. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/element-inspector.css +190 -0
  190. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/loading.css +59 -0
  191. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/overlay.css +45 -0
  192. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/panel-grid.css +95 -0
  193. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/selection.css +101 -0
  194. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/statistics.css +138 -0
  195. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/index.css +31 -0
  196. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/layout/container.css +7 -0
  197. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/layout/controls.css +56 -0
  198. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/layout/preview.css +78 -0
  199. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/alignment/axis.js +314 -0
  200. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/alignment/basic.js +107 -0
  201. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/alignment/distribute.js +54 -0
  202. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/canvas.js +172 -0
  203. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/dragging.js +258 -0
  204. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/resize.js +48 -0
  205. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/selection.js +71 -0
  206. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/core/api.js +288 -0
  207. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/core/state.js +143 -0
  208. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/core/utils.js +245 -0
  209. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/dev/element-inspector.js +992 -0
  210. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/bbox.js +339 -0
  211. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/element-drag.js +286 -0
  212. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/overlay.js +371 -0
  213. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/preview.js +293 -0
  214. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/main.js +426 -0
  215. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/shortcuts/context-menu.js +152 -0
  216. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/shortcuts/keyboard.js +265 -0
  217. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/controls.js +184 -0
  218. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/download.js +57 -0
  219. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/help.js +100 -0
  220. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/theme.js +34 -0
  221. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/__init__.py +124 -0
  222. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/_html.py +851 -0
  223. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/_scripts.py +4932 -0
  224. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/_styles.py +1657 -0
  225. scitex/fts/_fig/_editor/_gui/_flask_editor.py +36 -0
  226. scitex/fts/_fig/_models/_Annotations.py +115 -0
  227. scitex/fts/_fig/_models/_Axes.py +152 -0
  228. scitex/fts/_fig/_models/_Figure.py +138 -0
  229. scitex/fts/_fig/_models/_Guides.py +104 -0
  230. scitex/fts/_fig/_models/_Plot.py +123 -0
  231. scitex/fts/_fig/_models/_Styles.py +245 -0
  232. scitex/fts/_fig/_models/__init__.py +80 -0
  233. scitex/fts/_fig/_models/_plot_types/__init__.py +156 -0
  234. scitex/fts/_fig/_models/_plot_types/_bar.py +43 -0
  235. scitex/fts/_fig/_models/_plot_types/_box.py +38 -0
  236. scitex/fts/_fig/_models/_plot_types/_distribution.py +36 -0
  237. scitex/fts/_fig/_models/_plot_types/_errorbar.py +60 -0
  238. scitex/fts/_fig/_models/_plot_types/_histogram.py +30 -0
  239. scitex/fts/_fig/_models/_plot_types/_image.py +61 -0
  240. scitex/fts/_fig/_models/_plot_types/_line.py +57 -0
  241. scitex/fts/_fig/_models/_plot_types/_scatter.py +30 -0
  242. scitex/fts/_fig/_models/_plot_types/_seaborn.py +121 -0
  243. scitex/fts/_fig/_models/_plot_types/_violin.py +36 -0
  244. scitex/fts/_fig/_utils/__init__.py +129 -0
  245. scitex/fts/_fig/_utils/_auto_layout.py +127 -0
  246. scitex/fts/_fig/_utils/_calc_bounds.py +111 -0
  247. scitex/fts/_fig/_utils/_const_sizes.py +48 -0
  248. scitex/fts/_fig/_utils/_convert_coords.py +77 -0
  249. scitex/fts/_fig/_utils/_get_template.py +178 -0
  250. scitex/fts/_fig/_utils/_normalize.py +73 -0
  251. scitex/fts/_fig/_utils/_plot_layout.py +397 -0
  252. scitex/fts/_fig/_utils/_validate.py +197 -0
  253. scitex/fts/_kinds/__init__.py +45 -0
  254. scitex/fts/_kinds/_figure/__init__.py +19 -0
  255. scitex/fts/_kinds/_figure/_composite.py +345 -0
  256. scitex/fts/_kinds/_plot/__init__.py +25 -0
  257. scitex/fts/_kinds/_plot/_backend/__init__.py +53 -0
  258. scitex/fts/_kinds/_plot/_backend/_export.py +165 -0
  259. scitex/fts/_kinds/_plot/_backend/_parser.py +188 -0
  260. scitex/fts/_kinds/_plot/_backend/_render.py +538 -0
  261. scitex/fts/_kinds/_plot/_dataclasses/_ChannelEncoding.py +46 -0
  262. scitex/fts/_kinds/_plot/_dataclasses/_Encoding.py +82 -0
  263. scitex/fts/_kinds/_plot/_dataclasses/_Theme.py +441 -0
  264. scitex/fts/_kinds/_plot/_dataclasses/_TraceEncoding.py +52 -0
  265. scitex/fts/_kinds/_plot/_dataclasses/__init__.py +47 -0
  266. scitex/fts/_kinds/_plot/_models/_Annotations.py +115 -0
  267. scitex/fts/_kinds/_plot/_models/_Axes.py +152 -0
  268. scitex/fts/_kinds/_plot/_models/_Figure.py +138 -0
  269. scitex/fts/_kinds/_plot/_models/_Guides.py +104 -0
  270. scitex/fts/_kinds/_plot/_models/_Plot.py +123 -0
  271. scitex/fts/_kinds/_plot/_models/_Styles.py +245 -0
  272. scitex/fts/_kinds/_plot/_models/__init__.py +80 -0
  273. scitex/fts/_kinds/_plot/_models/_plot_types/__init__.py +156 -0
  274. scitex/fts/_kinds/_plot/_models/_plot_types/_bar.py +43 -0
  275. scitex/fts/_kinds/_plot/_models/_plot_types/_box.py +38 -0
  276. scitex/fts/_kinds/_plot/_models/_plot_types/_distribution.py +36 -0
  277. scitex/fts/_kinds/_plot/_models/_plot_types/_errorbar.py +60 -0
  278. scitex/fts/_kinds/_plot/_models/_plot_types/_histogram.py +30 -0
  279. scitex/fts/_kinds/_plot/_models/_plot_types/_image.py +61 -0
  280. scitex/fts/_kinds/_plot/_models/_plot_types/_line.py +57 -0
  281. scitex/fts/_kinds/_plot/_models/_plot_types/_scatter.py +30 -0
  282. scitex/fts/_kinds/_plot/_models/_plot_types/_seaborn.py +121 -0
  283. scitex/fts/_kinds/_plot/_models/_plot_types/_violin.py +36 -0
  284. scitex/fts/_kinds/_plot/_utils/__init__.py +129 -0
  285. scitex/fts/_kinds/_plot/_utils/_auto_layout.py +127 -0
  286. scitex/fts/_kinds/_plot/_utils/_calc_bounds.py +111 -0
  287. scitex/fts/_kinds/_plot/_utils/_const_sizes.py +48 -0
  288. scitex/fts/_kinds/_plot/_utils/_convert_coords.py +77 -0
  289. scitex/fts/_kinds/_plot/_utils/_get_template.py +178 -0
  290. scitex/fts/_kinds/_plot/_utils/_normalize.py +73 -0
  291. scitex/fts/_kinds/_plot/_utils/_plot_layout.py +397 -0
  292. scitex/fts/_kinds/_plot/_utils/_validate.py +197 -0
  293. scitex/fts/_kinds/_shape/__init__.py +141 -0
  294. scitex/fts/_kinds/_stats/__init__.py +56 -0
  295. scitex/fts/_kinds/_stats/_dataclasses/_Stats.py +423 -0
  296. scitex/fts/_kinds/_stats/_dataclasses/__init__.py +48 -0
  297. scitex/fts/_kinds/_table/__init__.py +72 -0
  298. scitex/fts/_kinds/_table/_latex/__init__.py +93 -0
  299. scitex/fts/_kinds/_table/_latex/_editor/__init__.py +11 -0
  300. scitex/fts/_kinds/_table/_latex/_editor/_app.py +725 -0
  301. scitex/fts/_kinds/_table/_latex/_export.py +279 -0
  302. scitex/fts/_kinds/_table/_latex/_figure_exporter.py +153 -0
  303. scitex/fts/_kinds/_table/_latex/_stats_formatter.py +274 -0
  304. scitex/fts/_kinds/_table/_latex/_table_exporter.py +362 -0
  305. scitex/fts/_kinds/_table/_latex/_utils.py +369 -0
  306. scitex/fts/_kinds/_table/_latex/_validator.py +445 -0
  307. scitex/fts/_kinds/_text/__init__.py +77 -0
  308. scitex/fts/_schemas/data_info.schema.json +75 -0
  309. scitex/fts/_schemas/encoding.schema.json +90 -0
  310. scitex/fts/_schemas/node.schema.json +145 -0
  311. scitex/fts/_schemas/render_manifest.schema.json +62 -0
  312. scitex/fts/_schemas/stats.schema.json +132 -0
  313. scitex/fts/_schemas/theme.schema.json +141 -0
  314. scitex/fts/_stats/__init__.py +48 -0
  315. scitex/fts/_stats/_dataclasses/_Stats.py +423 -0
  316. scitex/fts/_stats/_dataclasses/__init__.py +48 -0
  317. scitex/fts/_tables/__init__.py +65 -0
  318. scitex/fts/_tables/_latex/__init__.py +93 -0
  319. scitex/fts/_tables/_latex/_editor/__init__.py +11 -0
  320. scitex/fts/_tables/_latex/_editor/_app.py +725 -0
  321. scitex/fts/_tables/_latex/_export.py +279 -0
  322. scitex/fts/_tables/_latex/_figure_exporter.py +153 -0
  323. scitex/fts/_tables/_latex/_stats_formatter.py +274 -0
  324. scitex/fts/_tables/_latex/_table_exporter.py +362 -0
  325. scitex/fts/_tables/_latex/_utils.py +369 -0
  326. scitex/fts/_tables/_latex/_validator.py +445 -0
  327. scitex/gen/__init__.py +66 -25
  328. scitex/gen/misc.py +28 -0
  329. scitex/io/__init__.py +47 -32
  330. scitex/io/_load.py +87 -36
  331. scitex/io/_load_modules/__init__.py +10 -7
  332. scitex/io/_load_modules/_pandas.py +6 -1
  333. scitex/io/_save.py +299 -1556
  334. scitex/io/_save_modules/__init__.py +76 -19
  335. scitex/io/_save_modules/_figure_utils.py +90 -0
  336. scitex/io/_save_modules/_image_csv.py +497 -0
  337. scitex/io/_save_modules/_legends.py +91 -0
  338. scitex/io/_save_modules/_pltz_bundle.py +356 -0
  339. scitex/io/_save_modules/_pltz_stx.py +536 -0
  340. scitex/io/_save_modules/_stx_bundle.py +104 -0
  341. scitex/io/_save_modules/_symlink.py +96 -0
  342. scitex/io/_save_modules/_yaml.py +1 -1
  343. scitex/io/_save_modules/_zarr.py +64 -18
  344. scitex/io/bundle/README.md +212 -0
  345. scitex/io/bundle/__init__.py +110 -0
  346. scitex/io/{_bundle.py → bundle/_core.py} +168 -97
  347. scitex/io/bundle/_nested.py +713 -0
  348. scitex/io/bundle/_types.py +74 -0
  349. scitex/io/{_zip_bundle.py → bundle/_zip.py} +93 -45
  350. scitex/io/utils/h5_to_zarr.py +1 -1
  351. scitex/logging/__init__.py +108 -13
  352. scitex/logging/_errors.py +508 -0
  353. scitex/logging/_formatters.py +30 -6
  354. scitex/logging/_warnings.py +261 -0
  355. scitex/plt/__init__.py +4 -1
  356. scitex/plt/_figrecipe.py +236 -0
  357. scitex/plt/_subplots/_AxisWrapper.py +6 -0
  358. scitex/plt/_subplots/_AxisWrapperMixins/_UnitAwareMixin.py +112 -1
  359. scitex/plt/_subplots/_FigWrapper.py +15 -0
  360. scitex/plt/_subplots/_SubplotsWrapper.py +125 -489
  361. scitex/plt/_subplots/_export_as_csv.py +11 -0
  362. scitex/plt/_subplots/_export_as_csv_formatters/__init__.py +2 -0
  363. scitex/plt/_subplots/_export_as_csv_formatters/_format_pcolormesh.py +66 -0
  364. scitex/plt/_subplots/_export_as_csv_formatters/_format_stackplot.py +62 -0
  365. scitex/plt/_subplots/_export_as_csv_formatters/test_formatters.py +208 -0
  366. scitex/plt/_subplots/_fonts.py +71 -0
  367. scitex/plt/_subplots/_mm_layout.py +282 -0
  368. scitex/plt/gallery/__init__.py +99 -2
  369. scitex/plt/styles/_plot_postprocess.py +3 -1
  370. scitex/plt/utils/_configure_mpl.py +16 -19
  371. scitex/repro/_RandomStateManager.py +13 -8
  372. scitex/resource/__init__.py +19 -1
  373. scitex/resource/_utils/_get_env_info.py +13 -25
  374. scitex/schema/__init__.py +149 -160
  375. scitex/schema/_encoding.py +273 -0
  376. scitex/schema/_figure_elements.py +406 -0
  377. scitex/schema/_theme.py +360 -0
  378. scitex/schema/_validation.py +0 -98
  379. scitex/scholar/__init__.py +56 -14
  380. scitex/scholar/auth/ScholarAuthManager.py +1 -1
  381. scitex/scholar/auth/__init__.py +11 -2
  382. scitex/scholar/auth/providers/BaseAuthenticator.py +1 -1
  383. scitex/scholar/auth/providers/EZProxyAuthenticator.py +1 -1
  384. scitex/scholar/auth/providers/OpenAthensAuthenticator.py +1 -1
  385. scitex/scholar/auth/providers/ShibbolethAuthenticator.py +1 -1
  386. scitex/scholar/config/ScholarConfig.py +1 -1
  387. scitex/scholar/core/Scholar.py +1 -1
  388. scitex/session/_decorator.py +18 -16
  389. scitex/session/_lifecycle.py +9 -11
  390. scitex/session/template.py +9 -8
  391. scitex/sh/test_sh.py +72 -0
  392. scitex/sh/test_sh_simple.py +61 -0
  393. scitex/stats/__init__.py +221 -97
  394. scitex/stats/_schema.py +21 -22
  395. scitex/stats/descriptive/_circular.py +212 -351
  396. scitex/stats/descriptive/_describe.py +81 -132
  397. scitex/stats/descriptive/_nan.py +205 -433
  398. scitex/stats/descriptive/_real.py +127 -141
  399. scitex/str/_format_plot_text.py +5 -5
  400. scitex/str/_latex.py +26 -84
  401. scitex/str/_latex_fallback.py +53 -47
  402. scitex/web/_search_pubmed.py +5 -4
  403. scitex/writer/tests/test_diff_between.py +451 -0
  404. scitex/writer/tests/test_document_section.py +311 -0
  405. scitex/writer/tests/test_document_workflow.py +393 -0
  406. scitex/writer/tests/test_writer.py +361 -0
  407. scitex/writer/tests/test_writer_integration.py +303 -0
  408. {scitex-2.8.1.dist-info → scitex-2.10.2.dist-info}/METADATA +368 -183
  409. {scitex-2.8.1.dist-info → scitex-2.10.2.dist-info}/RECORD +412 -97
  410. scitex/scholar/docs/to_claude/guidelines/examples/mgmt/ARCHITECTURE_EXAMPLE.md +0 -905
  411. scitex/scholar/docs/to_claude/guidelines/examples/mgmt/BULLETIN_BOARD_EXAMPLE.md +0 -99
  412. scitex/scholar/docs/to_claude/guidelines/examples/mgmt/PROJECT_DESCRIPTION_EXAMPLE.md +0 -96
  413. {scitex-2.8.1.dist-info → scitex-2.10.2.dist-info}/WHEEL +0 -0
  414. {scitex-2.8.1.dist-info → scitex-2.10.2.dist-info}/entry_points.txt +0 -0
  415. {scitex-2.8.1.dist-info → scitex-2.10.2.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,401 @@
1
+ #!/usr/bin/env python3
2
+ """MCP tool handlers for SciTeX Capture."""
3
+
4
+ import asyncio
5
+ import base64
6
+ from datetime import datetime
7
+ from pathlib import Path
8
+
9
+ from scitex import capture
10
+
11
+ from .mcp_utils import get_capture_dir
12
+
13
+
14
+ class CaptureHandlers:
15
+ """Handlers for capture-related tools."""
16
+
17
+ @staticmethod
18
+ async def capture_screenshot(
19
+ message=None,
20
+ monitor_id=0,
21
+ all=False,
22
+ app=None,
23
+ url=None,
24
+ quality=85,
25
+ return_base64=False,
26
+ grid_overlay=False,
27
+ cursor_overlay=False,
28
+ grid_spacing=100,
29
+ ):
30
+ """Capture screenshot with optional overlays."""
31
+ try:
32
+ loop = asyncio.get_event_loop()
33
+
34
+ def do_capture():
35
+ return capture.snap(
36
+ message=message,
37
+ quality=quality,
38
+ monitor_id=monitor_id,
39
+ all=all,
40
+ app=app,
41
+ url=url,
42
+ verbose=True,
43
+ )
44
+
45
+ path = await loop.run_in_executor(None, do_capture)
46
+
47
+ if not path:
48
+ return {"success": False, "error": "Failed to capture"}
49
+
50
+ # Apply overlays if requested
51
+ if grid_overlay or cursor_overlay:
52
+ from .grid import draw_cursor_overlay, draw_grid_overlay
53
+
54
+ # Determine capture mode for coordinate mapping
55
+ capture_mode = "all" if all else str(monitor_id)
56
+
57
+ if grid_overlay:
58
+ path = draw_grid_overlay(path, grid_spacing=grid_spacing)
59
+ if cursor_overlay:
60
+ path = draw_cursor_overlay(
61
+ path, output_path=path, capture_mode=capture_mode
62
+ )
63
+
64
+ category = "stderr" if "-stderr.jpg" in path else "stdout"
65
+ result = {
66
+ "success": True,
67
+ "path": path,
68
+ "category": category,
69
+ "message": f"Screenshot saved to {path}",
70
+ "timestamp": datetime.now().isoformat(),
71
+ }
72
+
73
+ if return_base64 and path:
74
+ with open(path, "rb") as f:
75
+ result["base64"] = base64.b64encode(f.read()).decode()
76
+
77
+ return result
78
+ except Exception as e:
79
+ return {"success": False, "error": str(e)}
80
+
81
+ @staticmethod
82
+ async def capture_window(window_handle: int, output_path=None, quality=85):
83
+ """Capture window by handle."""
84
+ try:
85
+ loop = asyncio.get_event_loop()
86
+ path = await loop.run_in_executor(
87
+ None, capture.capture_window, window_handle, output_path
88
+ )
89
+ if path:
90
+ return {"success": True, "path": path, "window_handle": window_handle}
91
+ return {
92
+ "success": False,
93
+ "error": f"Failed to capture window {window_handle}",
94
+ }
95
+ except Exception as e:
96
+ return {"success": False, "error": str(e)}
97
+
98
+
99
+ class MonitoringHandlers:
100
+ """Handlers for monitoring tools."""
101
+
102
+ def __init__(self):
103
+ self.monitoring_active = False
104
+ self.monitoring_worker = None
105
+
106
+ async def start_monitoring(
107
+ self,
108
+ interval=1.0,
109
+ monitor_id=0,
110
+ capture_all=False,
111
+ output_dir=None,
112
+ quality=60,
113
+ verbose=True,
114
+ ):
115
+ """Start continuous monitoring."""
116
+ if self.monitoring_active:
117
+ return {"success": False, "message": "Already active"}
118
+
119
+ try:
120
+ loop = asyncio.get_event_loop()
121
+
122
+ def start():
123
+ return capture.start_monitor(
124
+ output_dir=output_dir or "~/.scitex/capture/",
125
+ interval=interval,
126
+ jpeg=True,
127
+ quality=quality,
128
+ verbose=verbose,
129
+ monitor_id=monitor_id,
130
+ capture_all=capture_all,
131
+ )
132
+
133
+ self.monitoring_worker = await loop.run_in_executor(None, start)
134
+ self.monitoring_active = True
135
+
136
+ return {
137
+ "success": True,
138
+ "message": f"Started monitoring with {interval}s interval",
139
+ "interval": interval,
140
+ "monitor_id": monitor_id,
141
+ }
142
+ except Exception as e:
143
+ return {"success": False, "error": str(e)}
144
+
145
+ async def stop_monitoring(self):
146
+ """Stop monitoring."""
147
+ if not self.monitoring_active:
148
+ return {"success": False, "message": "Not active"}
149
+
150
+ try:
151
+ loop = asyncio.get_event_loop()
152
+ await loop.run_in_executor(None, capture.stop)
153
+
154
+ stats = {}
155
+ if self.monitoring_worker:
156
+ stats = {
157
+ "screenshots_taken": self.monitoring_worker.screenshot_count,
158
+ "session_id": self.monitoring_worker.session_id,
159
+ }
160
+
161
+ self.monitoring_active = False
162
+ self.monitoring_worker = None
163
+
164
+ return {"success": True, "message": "Stopped", **stats}
165
+ except Exception as e:
166
+ return {"success": False, "error": str(e)}
167
+
168
+ async def get_status(self):
169
+ """Get monitoring status."""
170
+ status = {"active": self.monitoring_active, "cache_dir": str(get_capture_dir())}
171
+
172
+ if self.monitoring_active and self.monitoring_worker:
173
+ status.update(
174
+ {
175
+ "screenshots_taken": self.monitoring_worker.screenshot_count,
176
+ "session_id": self.monitoring_worker.session_id,
177
+ }
178
+ )
179
+
180
+ cache_dir = get_capture_dir()
181
+ if cache_dir.exists():
182
+ total_size = sum(f.stat().st_size for f in cache_dir.glob("*.jpg"))
183
+ status["cache_size_mb"] = round(total_size / (1024 * 1024), 2)
184
+ status["screenshot_count"] = len(list(cache_dir.glob("*.jpg")))
185
+
186
+ return status
187
+
188
+
189
+ class UtilityHandlers:
190
+ """Handlers for utility tools."""
191
+
192
+ @staticmethod
193
+ async def analyze_screenshot(path: str):
194
+ """Analyze screenshot."""
195
+ try:
196
+ from .utils import _detect_category
197
+
198
+ loop = asyncio.get_event_loop()
199
+ category = await loop.run_in_executor(None, _detect_category, path)
200
+
201
+ path_obj = Path(path)
202
+ if not path_obj.exists():
203
+ return {"success": False, "error": f"Not found: {path}"}
204
+
205
+ return {
206
+ "success": True,
207
+ "path": path,
208
+ "category": category,
209
+ "is_error": category == "stderr",
210
+ "size_kb": round(path_obj.stat().st_size / 1024, 2),
211
+ }
212
+ except Exception as e:
213
+ return {"success": False, "error": str(e)}
214
+
215
+ @staticmethod
216
+ async def list_recent_screenshots(limit=10, category="all"):
217
+ """List recent screenshots."""
218
+ try:
219
+ cache_dir = get_capture_dir()
220
+ if not cache_dir.exists():
221
+ return {"success": True, "screenshots": []}
222
+
223
+ screenshots = list(cache_dir.glob("*.jpg"))
224
+ if category == "stdout":
225
+ screenshots = [s for s in screenshots if "-stdout.jpg" in s.name]
226
+ elif category == "stderr":
227
+ screenshots = [s for s in screenshots if "-stderr.jpg" in s.name]
228
+
229
+ screenshots.sort(key=lambda p: p.stat().st_mtime, reverse=True)
230
+ screenshots = screenshots[:limit]
231
+
232
+ result_list = []
233
+ for s in screenshots:
234
+ cat = "stderr" if "-stderr.jpg" in s.name else "stdout"
235
+ result_list.append(
236
+ {
237
+ "filename": s.name,
238
+ "path": str(s),
239
+ "category": cat,
240
+ "size_kb": round(s.stat().st_size / 1024, 2),
241
+ }
242
+ )
243
+
244
+ return {
245
+ "success": True,
246
+ "screenshots": result_list,
247
+ "count": len(result_list),
248
+ }
249
+ except Exception as e:
250
+ return {"success": False, "error": str(e)}
251
+
252
+ @staticmethod
253
+ async def clear_cache(max_size_gb=1.0, clear_all=False):
254
+ """Clear cache."""
255
+ try:
256
+ cache_dir = get_capture_dir()
257
+ if not cache_dir.exists():
258
+ return {"success": True, "message": "Cache does not exist"}
259
+
260
+ if clear_all:
261
+ removed = sum(1 for s in cache_dir.glob("*.jpg") if s.unlink() or True)
262
+ return {"success": True, "removed_count": removed}
263
+ else:
264
+ from .utils import _manage_cache_size
265
+
266
+ loop = asyncio.get_event_loop()
267
+ await loop.run_in_executor(
268
+ None, _manage_cache_size, cache_dir, max_size_gb
269
+ )
270
+ return {"success": True, "max_size_gb": max_size_gb}
271
+ except Exception as e:
272
+ return {"success": False, "error": str(e)}
273
+
274
+ @staticmethod
275
+ async def get_info():
276
+ """Get system info."""
277
+ try:
278
+ loop = asyncio.get_event_loop()
279
+ info = await loop.run_in_executor(None, capture.get_info)
280
+ return {
281
+ "success": True,
282
+ "monitors": info.get("Monitors", {}),
283
+ "windows": info.get("Windows", {}),
284
+ }
285
+ except Exception as e:
286
+ return {"success": False, "error": str(e)}
287
+
288
+ @staticmethod
289
+ async def list_windows():
290
+ """List windows."""
291
+ try:
292
+ loop = asyncio.get_event_loop()
293
+ info = await loop.run_in_executor(None, capture.get_info)
294
+ windows = info.get("Windows", {}).get("Details", [])
295
+ formatted = [
296
+ {
297
+ "handle": w.get("Handle"),
298
+ "title": w.get("Title"),
299
+ "process": w.get("ProcessName"),
300
+ }
301
+ for w in windows
302
+ ]
303
+ return {"success": True, "windows": formatted, "count": len(formatted)}
304
+ except Exception as e:
305
+ return {"success": False, "error": str(e)}
306
+
307
+
308
+ class GifHandlers:
309
+ """Handlers for GIF tools."""
310
+
311
+ @staticmethod
312
+ async def create_gif(
313
+ session_id=None,
314
+ image_paths=None,
315
+ pattern=None,
316
+ output_path=None,
317
+ duration=0.5,
318
+ optimize=True,
319
+ max_frames=None,
320
+ ):
321
+ """Create GIF from screenshots."""
322
+ try:
323
+ from .gif import GifCreator
324
+
325
+ creator = GifCreator()
326
+ loop = asyncio.get_event_loop()
327
+
328
+ if session_id:
329
+ if session_id == "latest":
330
+ result = await loop.run_in_executor(
331
+ None,
332
+ creator.create_gif_from_recent_session,
333
+ "~/.scitex/capture",
334
+ duration,
335
+ optimize,
336
+ max_frames,
337
+ )
338
+ else:
339
+ result = await loop.run_in_executor(
340
+ None,
341
+ creator.create_gif_from_session,
342
+ session_id,
343
+ output_path,
344
+ "~/.scitex/capture",
345
+ duration,
346
+ optimize,
347
+ max_frames,
348
+ )
349
+ elif image_paths:
350
+ if not output_path:
351
+ output_path = (
352
+ f"~/.scitex/capture/gif_{datetime.now():%Y%m%d_%H%M%S}.gif"
353
+ )
354
+ result = await loop.run_in_executor(
355
+ None,
356
+ creator.create_gif_from_files,
357
+ image_paths,
358
+ output_path,
359
+ duration,
360
+ optimize,
361
+ )
362
+ elif pattern:
363
+ result = await loop.run_in_executor(
364
+ None,
365
+ creator.create_gif_from_pattern,
366
+ pattern,
367
+ output_path,
368
+ duration,
369
+ optimize,
370
+ max_frames,
371
+ )
372
+ else:
373
+ return {
374
+ "success": False,
375
+ "error": "Specify session_id, image_paths, or pattern",
376
+ }
377
+
378
+ if result:
379
+ return {"success": True, "path": result, "duration": duration}
380
+ return {"success": False, "error": "No images found"}
381
+ except Exception as e:
382
+ return {"success": False, "error": str(e)}
383
+
384
+ @staticmethod
385
+ async def list_sessions(limit=10):
386
+ """List sessions."""
387
+ try:
388
+ from .gif import GifCreator
389
+
390
+ creator = GifCreator()
391
+ loop = asyncio.get_event_loop()
392
+ sessions = await loop.run_in_executor(
393
+ None, creator.get_recent_sessions, "~/.scitex/capture"
394
+ )
395
+ return {
396
+ "success": True,
397
+ "sessions": sessions[:limit],
398
+ "count": min(len(sessions), limit),
399
+ }
400
+ except Exception as e:
401
+ return {"success": False, "error": str(e)}
@@ -0,0 +1,192 @@
1
+ #!/usr/bin/env python3
2
+ """MCP tool definitions for SciTeX Capture."""
3
+
4
+ import mcp.types as types
5
+
6
+
7
+ def get_capture_tools():
8
+ """Return capture-related tool definitions."""
9
+ return [
10
+ types.Tool(
11
+ name="capture_screenshot",
12
+ description="Capture screenshot with optional grid/cursor overlay",
13
+ inputSchema={
14
+ "type": "object",
15
+ "properties": {
16
+ "message": {
17
+ "type": "string",
18
+ "description": "Message for filename",
19
+ },
20
+ "monitor_id": {"type": "integer", "default": 0},
21
+ "all": {
22
+ "type": "boolean",
23
+ "default": False,
24
+ "description": "All monitors",
25
+ },
26
+ "app": {"type": "string", "description": "App name to capture"},
27
+ "url": {"type": "string", "description": "URL to capture"},
28
+ "quality": {
29
+ "type": "integer",
30
+ "minimum": 1,
31
+ "maximum": 100,
32
+ "default": 85,
33
+ },
34
+ "return_base64": {"type": "boolean", "default": False},
35
+ "grid_overlay": {
36
+ "type": "boolean",
37
+ "default": False,
38
+ "description": "Add grid",
39
+ },
40
+ "cursor_overlay": {
41
+ "type": "boolean",
42
+ "default": False,
43
+ "description": "Add cursor",
44
+ },
45
+ "grid_spacing": {"type": "integer", "default": 25},
46
+ },
47
+ },
48
+ ),
49
+ types.Tool(
50
+ name="capture_window",
51
+ description="Capture a specific window by handle",
52
+ inputSchema={
53
+ "type": "object",
54
+ "properties": {
55
+ "window_handle": {
56
+ "type": "integer",
57
+ "description": "Window handle",
58
+ },
59
+ "output_path": {"type": "string"},
60
+ "quality": {"type": "integer", "default": 85},
61
+ },
62
+ "required": ["window_handle"],
63
+ },
64
+ ),
65
+ ]
66
+
67
+
68
+ def get_monitoring_tools():
69
+ """Return monitoring tool definitions."""
70
+ return [
71
+ types.Tool(
72
+ name="start_monitoring",
73
+ description="Start continuous screenshot monitoring",
74
+ inputSchema={
75
+ "type": "object",
76
+ "properties": {
77
+ "interval": {"type": "number", "minimum": 0.1, "default": 1.0},
78
+ "monitor_id": {"type": "integer", "default": 0},
79
+ "capture_all": {"type": "boolean", "default": False},
80
+ "output_dir": {"type": "string"},
81
+ "quality": {"type": "integer", "default": 60},
82
+ "verbose": {"type": "boolean", "default": True},
83
+ },
84
+ },
85
+ ),
86
+ types.Tool(
87
+ name="stop_monitoring",
88
+ description="Stop continuous monitoring",
89
+ inputSchema={"type": "object", "properties": {}},
90
+ ),
91
+ types.Tool(
92
+ name="get_monitoring_status",
93
+ description="Get monitoring status",
94
+ inputSchema={"type": "object", "properties": {}},
95
+ ),
96
+ ]
97
+
98
+
99
+ def get_utility_tools():
100
+ """Return utility tool definitions."""
101
+ return [
102
+ types.Tool(
103
+ name="analyze_screenshot",
104
+ description="Analyze screenshot for errors",
105
+ inputSchema={
106
+ "type": "object",
107
+ "properties": {"path": {"type": "string"}},
108
+ "required": ["path"],
109
+ },
110
+ ),
111
+ types.Tool(
112
+ name="list_recent_screenshots",
113
+ description="List recent screenshots",
114
+ inputSchema={
115
+ "type": "object",
116
+ "properties": {
117
+ "limit": {
118
+ "type": "integer",
119
+ "default": 10,
120
+ "minimum": 1,
121
+ "maximum": 100,
122
+ },
123
+ "category": {
124
+ "type": "string",
125
+ "enum": ["stdout", "stderr", "all"],
126
+ "default": "all",
127
+ },
128
+ },
129
+ },
130
+ ),
131
+ types.Tool(
132
+ name="clear_cache",
133
+ description="Clear screenshot cache",
134
+ inputSchema={
135
+ "type": "object",
136
+ "properties": {
137
+ "max_size_gb": {"type": "number", "minimum": 0.001, "default": 1.0},
138
+ "clear_all": {"type": "boolean", "default": False},
139
+ },
140
+ },
141
+ ),
142
+ types.Tool(
143
+ name="get_info",
144
+ description="Get monitor and window info",
145
+ inputSchema={"type": "object", "properties": {}},
146
+ ),
147
+ types.Tool(
148
+ name="list_windows",
149
+ description="List visible windows",
150
+ inputSchema={"type": "object", "properties": {}},
151
+ ),
152
+ ]
153
+
154
+
155
+ def get_gif_tools():
156
+ """Return GIF tool definitions."""
157
+ return [
158
+ types.Tool(
159
+ name="create_gif",
160
+ description="Create GIF from screenshots",
161
+ inputSchema={
162
+ "type": "object",
163
+ "properties": {
164
+ "session_id": {"type": "string"},
165
+ "image_paths": {"type": "array", "items": {"type": "string"}},
166
+ "pattern": {"type": "string"},
167
+ "output_path": {"type": "string"},
168
+ "duration": {"type": "number", "default": 0.5},
169
+ "optimize": {"type": "boolean", "default": True},
170
+ "max_frames": {"type": "integer", "minimum": 1, "maximum": 100},
171
+ },
172
+ },
173
+ ),
174
+ types.Tool(
175
+ name="list_sessions",
176
+ description="List monitoring sessions",
177
+ inputSchema={
178
+ "type": "object",
179
+ "properties": {"limit": {"type": "integer", "default": 10}},
180
+ },
181
+ ),
182
+ ]
183
+
184
+
185
+ def get_all_tools():
186
+ """Return all tool definitions."""
187
+ return (
188
+ get_capture_tools()
189
+ + get_monitoring_tools()
190
+ + get_utility_tools()
191
+ + get_gif_tools()
192
+ )