wormclaude 1.0.119 → 1.0.121

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 (232) hide show
  1. package/dist/theme.js +1 -1
  2. package/dist/tui.js +6 -1
  3. package/package.json +1 -1
  4. package/skills/build-mcp-app/SKILL.md +0 -393
  5. package/skills/build-mcp-app/references/abuse-protection.md +0 -60
  6. package/skills/build-mcp-app/references/apps-sdk-messages.md +0 -227
  7. package/skills/build-mcp-app/references/directory-checklist.md +0 -18
  8. package/skills/build-mcp-app/references/iframe-sandbox.md +0 -164
  9. package/skills/build-mcp-app/references/payload-budgeting.md +0 -54
  10. package/skills/build-mcp-app/references/widget-templates.md +0 -249
  11. package/skills/build-mcp-server/SKILL.md +0 -222
  12. package/skills/build-mcp-server/references/auth.md +0 -108
  13. package/skills/build-mcp-server/references/deploy-cloudflare-workers.md +0 -106
  14. package/skills/build-mcp-server/references/elicitation.md +0 -129
  15. package/skills/build-mcp-server/references/remote-http-scaffold.md +0 -211
  16. package/skills/build-mcp-server/references/resources-and-prompts.md +0 -122
  17. package/skills/build-mcp-server/references/server-capabilities.md +0 -164
  18. package/skills/build-mcp-server/references/tool-design.md +0 -189
  19. package/skills/build-mcp-server/references/versions.md +0 -25
  20. package/skills/build-mcpb/SKILL.md +0 -200
  21. package/skills/build-mcpb/references/local-security.md +0 -149
  22. package/skills/build-mcpb/references/manifest-schema.md +0 -156
  23. package/skills/docx/script/__init__.py +0 -1
  24. package/skills/docx/script/accept_chages.py +0 -135
  25. package/skills/docx/script/comment.py +0 -318
  26. package/skills/docx/script/office/helpers/__init__.py +0 -0
  27. package/skills/docx/script/office/helpers/merge_runs.py +0 -199
  28. package/skills/docx/script/office/helpers/simplify_redlines.py +0 -197
  29. package/skills/docx/script/office/pack.py +0 -159
  30. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +0 -1499
  31. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +0 -146
  32. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +0 -1085
  33. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +0 -11
  34. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +0 -3081
  35. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +0 -23
  36. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +0 -185
  37. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +0 -287
  38. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/pml.xsd +0 -1676
  39. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +0 -28
  40. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +0 -144
  41. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +0 -174
  42. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +0 -25
  43. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +0 -18
  44. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +0 -59
  45. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +0 -56
  46. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +0 -195
  47. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +0 -582
  48. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +0 -25
  49. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/sml.xsd +0 -4439
  50. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +0 -570
  51. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +0 -509
  52. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +0 -12
  53. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +0 -108
  54. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +0 -96
  55. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/wml.xsd +0 -3646
  56. package/skills/docx/script/office/schemas/ISO-IEC29500-4_2016/xml.xsd +0 -116
  57. package/skills/docx/script/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +0 -42
  58. package/skills/docx/script/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +0 -50
  59. package/skills/docx/script/office/schemas/ecma/fouth-edition/opc-digSig.xsd +0 -49
  60. package/skills/docx/script/office/schemas/ecma/fouth-edition/opc-relationships.xsd +0 -33
  61. package/skills/docx/script/office/schemas/mce/mc.xsd +0 -75
  62. package/skills/docx/script/office/schemas/microsoft/wml-2010.xsd +0 -560
  63. package/skills/docx/script/office/schemas/microsoft/wml-2012.xsd +0 -67
  64. package/skills/docx/script/office/schemas/microsoft/wml-2018.xsd +0 -14
  65. package/skills/docx/script/office/schemas/microsoft/wml-cex-2018.xsd +0 -20
  66. package/skills/docx/script/office/schemas/microsoft/wml-cid-2016.xsd +0 -13
  67. package/skills/docx/script/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +0 -4
  68. package/skills/docx/script/office/schemas/microsoft/wml-symex-2015.xsd +0 -8
  69. package/skills/docx/script/office/soffice.py +0 -183
  70. package/skills/docx/script/office/unpack.py +0 -132
  71. package/skills/docx/script/office/validate.py +0 -117
  72. package/skills/docx/script/office/validators/__init__.py +0 -15
  73. package/skills/docx/script/office/validators/base.py +0 -851
  74. package/skills/docx/script/office/validators/docx.py +0 -446
  75. package/skills/docx/script/office/validators/pptx.py +0 -275
  76. package/skills/docx/script/office/validators/redlining.py +0 -247
  77. package/skills/docx/script/templates/comments.xml +0 -3
  78. package/skills/docx/script/templates/commentsExtended.xml +0 -3
  79. package/skills/docx/script/templates/commentsExtensible.xml +0 -3
  80. package/skills/docx/script/templates/commentsIds.xml +0 -3
  81. package/skills/docx/script/templates/people.xml +0 -3
  82. package/skills/docx/skill.md +0 -593
  83. package/skills/explain.md +0 -14
  84. package/skills/frontend-design/SKILL.md +0 -42
  85. package/skills/pdf/FORMS.md +0 -294
  86. package/skills/pdf/REFERENCE.md +0 -612
  87. package/skills/pdf/SKILL.md +0 -314
  88. package/skills/pdf/scripts/check_bounding_boxes.py +0 -65
  89. package/skills/pdf/scripts/check_fillable_fields.py +0 -11
  90. package/skills/pdf/scripts/convert_pdf_to_images.py +0 -33
  91. package/skills/pdf/scripts/create_validation_image.py +0 -37
  92. package/skills/pdf/scripts/extract_form_field_info.py +0 -122
  93. package/skills/pdf/scripts/extract_form_structure.py +0 -115
  94. package/skills/pdf/scripts/fill_fillable_fields.py +0 -98
  95. package/skills/pdf/scripts/fill_pdf_form_with_annotations.py +0 -107
  96. package/skills/playground/SKILL.md +0 -77
  97. package/skills/playground/templates/code-map.md +0 -158
  98. package/skills/playground/templates/concept-map.md +0 -73
  99. package/skills/playground/templates/data-explorer.md +0 -67
  100. package/skills/playground/templates/design-playground.md +0 -67
  101. package/skills/playground/templates/diff-review.md +0 -179
  102. package/skills/playground/templates/document-critique.md +0 -171
  103. package/skills/pptx/SKILL.md +0 -230
  104. package/skills/pptx/editing.md +0 -205
  105. package/skills/pptx/pptxgenjs.md +0 -437
  106. package/skills/pptx/scripts/__init__.py +0 -0
  107. package/skills/pptx/scripts/add_slide.py +0 -195
  108. package/skills/pptx/scripts/clean.py +0 -286
  109. package/skills/pptx/scripts/office/helpers/__init__.py +0 -0
  110. package/skills/pptx/scripts/office/helpers/merge_runs.py +0 -199
  111. package/skills/pptx/scripts/office/helpers/simplify_redlines.py +0 -197
  112. package/skills/pptx/scripts/office/pack.py +0 -159
  113. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +0 -1499
  114. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +0 -146
  115. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +0 -1085
  116. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +0 -11
  117. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +0 -3081
  118. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +0 -23
  119. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +0 -185
  120. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +0 -287
  121. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +0 -1676
  122. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +0 -28
  123. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +0 -144
  124. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +0 -174
  125. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +0 -25
  126. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +0 -18
  127. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +0 -59
  128. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +0 -56
  129. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +0 -195
  130. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +0 -582
  131. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +0 -25
  132. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +0 -4439
  133. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +0 -570
  134. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +0 -509
  135. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +0 -12
  136. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +0 -108
  137. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +0 -96
  138. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +0 -3646
  139. package/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +0 -116
  140. package/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +0 -42
  141. package/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +0 -50
  142. package/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd +0 -49
  143. package/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd +0 -33
  144. package/skills/pptx/scripts/office/schemas/mce/mc.xsd +0 -75
  145. package/skills/pptx/scripts/office/schemas/microsoft/wml-2010.xsd +0 -560
  146. package/skills/pptx/scripts/office/schemas/microsoft/wml-2012.xsd +0 -67
  147. package/skills/pptx/scripts/office/schemas/microsoft/wml-2018.xsd +0 -14
  148. package/skills/pptx/scripts/office/schemas/microsoft/wml-cex-2018.xsd +0 -20
  149. package/skills/pptx/scripts/office/schemas/microsoft/wml-cid-2016.xsd +0 -13
  150. package/skills/pptx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +0 -4
  151. package/skills/pptx/scripts/office/schemas/microsoft/wml-symex-2015.xsd +0 -8
  152. package/skills/pptx/scripts/office/soffice.py +0 -183
  153. package/skills/pptx/scripts/office/unpack.py +0 -132
  154. package/skills/pptx/scripts/office/validate.py +0 -117
  155. package/skills/pptx/scripts/office/validators/__init__.py +0 -15
  156. package/skills/pptx/scripts/office/validators/base.py +0 -851
  157. package/skills/pptx/scripts/office/validators/docx.py +0 -446
  158. package/skills/pptx/scripts/office/validators/pptx.py +0 -275
  159. package/skills/pptx/scripts/office/validators/redlining.py +0 -247
  160. package/skills/pptx/scripts/thumbnail.py +0 -289
  161. package/skills/recon.md +0 -16
  162. package/skills/security-audit/SKILL.md +0 -26
  163. package/skills/talent-creator/SKILL.md +0 -486
  164. package/skills/talent-creator/agents/analyzer.md +0 -274
  165. package/skills/talent-creator/agents/comparator.md +0 -202
  166. package/skills/talent-creator/agents/grader.md +0 -223
  167. package/skills/talent-creator/assets/eval_review.html +0 -146
  168. package/skills/talent-creator/eval-viewer/generate_review.py +0 -471
  169. package/skills/talent-creator/eval-viewer/viewer.html +0 -1325
  170. package/skills/talent-creator/references/schemas.md +0 -430
  171. package/skills/talent-creator/scripts/__init__.py +0 -0
  172. package/skills/talent-creator/scripts/aggregate_benchmark.py +0 -401
  173. package/skills/talent-creator/scripts/generate_report.py +0 -326
  174. package/skills/talent-creator/scripts/improve_description.py +0 -247
  175. package/skills/talent-creator/scripts/package_skill.py +0 -136
  176. package/skills/talent-creator/scripts/quick_validate.py +0 -146
  177. package/skills/talent-creator/scripts/run_eval.py +0 -310
  178. package/skills/talent-creator/scripts/run_loop.py +0 -328
  179. package/skills/talent-creator/scripts/utils.py +0 -47
  180. package/skills/xlsx/SKILL.md +0 -300
  181. package/skills/xlsx/scripts/office/helpers/__init__.py +0 -0
  182. package/skills/xlsx/scripts/office/helpers/merge_runs.py +0 -199
  183. package/skills/xlsx/scripts/office/helpers/simplify_redlines.py +0 -197
  184. package/skills/xlsx/scripts/office/pack.py +0 -159
  185. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +0 -1499
  186. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +0 -146
  187. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +0 -1085
  188. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +0 -11
  189. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +0 -3081
  190. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +0 -23
  191. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +0 -185
  192. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +0 -287
  193. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +0 -1676
  194. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +0 -28
  195. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +0 -144
  196. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +0 -174
  197. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +0 -25
  198. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +0 -18
  199. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +0 -59
  200. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +0 -56
  201. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +0 -195
  202. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +0 -582
  203. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +0 -25
  204. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +0 -4439
  205. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +0 -570
  206. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +0 -509
  207. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +0 -12
  208. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +0 -108
  209. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +0 -96
  210. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +0 -3646
  211. package/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +0 -116
  212. package/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +0 -42
  213. package/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +0 -50
  214. package/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd +0 -49
  215. package/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd +0 -33
  216. package/skills/xlsx/scripts/office/schemas/mce/mc.xsd +0 -75
  217. package/skills/xlsx/scripts/office/schemas/microsoft/wml-2010.xsd +0 -560
  218. package/skills/xlsx/scripts/office/schemas/microsoft/wml-2012.xsd +0 -67
  219. package/skills/xlsx/scripts/office/schemas/microsoft/wml-2018.xsd +0 -14
  220. package/skills/xlsx/scripts/office/schemas/microsoft/wml-cex-2018.xsd +0 -20
  221. package/skills/xlsx/scripts/office/schemas/microsoft/wml-cid-2016.xsd +0 -13
  222. package/skills/xlsx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +0 -4
  223. package/skills/xlsx/scripts/office/schemas/microsoft/wml-symex-2015.xsd +0 -8
  224. package/skills/xlsx/scripts/office/soffice.py +0 -183
  225. package/skills/xlsx/scripts/office/unpack.py +0 -132
  226. package/skills/xlsx/scripts/office/validate.py +0 -117
  227. package/skills/xlsx/scripts/office/validators/__init__.py +0 -15
  228. package/skills/xlsx/scripts/office/validators/base.py +0 -851
  229. package/skills/xlsx/scripts/office/validators/docx.py +0 -446
  230. package/skills/xlsx/scripts/office/validators/pptx.py +0 -275
  231. package/skills/xlsx/scripts/office/validators/redlining.py +0 -247
  232. package/skills/xlsx/scripts/recalc.py +0 -184
@@ -1,471 +0,0 @@
1
- #!/usr/bin/env python3
2
- """Generate and serve a review page for eval results.
3
-
4
- Reads the workspace directory, discovers runs (directories with outputs/),
5
- embeds all output data into a self-contained HTML page, and serves it via
6
- a tiny HTTP server. Feedback auto-saves to feedback.json in the workspace.
7
-
8
- Usage:
9
- python generate_review.py <workspace-path> [--port PORT] [--skill-name NAME]
10
- python generate_review.py <workspace-path> --previous-feedback /path/to/old/feedback.json
11
-
12
- No dependencies beyond the Python stdlib are required.
13
- """
14
-
15
- import argparse
16
- import base64
17
- import json
18
- import mimetypes
19
- import os
20
- import re
21
- import signal
22
- import subprocess
23
- import sys
24
- import time
25
- import webbrowser
26
- from functools import partial
27
- from http.server import HTTPServer, BaseHTTPRequestHandler
28
- from pathlib import Path
29
-
30
- # Files to exclude from output listings
31
- METADATA_FILES = {"transcript.md", "user_notes.md", "metrics.json"}
32
-
33
- # Extensions we render as inline text
34
- TEXT_EXTENSIONS = {
35
- ".txt", ".md", ".json", ".csv", ".py", ".js", ".ts", ".tsx", ".jsx",
36
- ".yaml", ".yml", ".xml", ".html", ".css", ".sh", ".rb", ".go", ".rs",
37
- ".java", ".c", ".cpp", ".h", ".hpp", ".sql", ".r", ".toml",
38
- }
39
-
40
- # Extensions we render as inline images
41
- IMAGE_EXTENSIONS = {".png", ".jpg", ".jpeg", ".gif", ".svg", ".webp"}
42
-
43
- # MIME type overrides for common types
44
- MIME_OVERRIDES = {
45
- ".svg": "image/svg+xml",
46
- ".xlsx": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
47
- ".docx": "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
48
- ".pptx": "application/vnd.openxmlformats-officedocument.presentationml.presentation",
49
- }
50
-
51
-
52
- def get_mime_type(path: Path) -> str:
53
- ext = path.suffix.lower()
54
- if ext in MIME_OVERRIDES:
55
- return MIME_OVERRIDES[ext]
56
- mime, _ = mimetypes.guess_type(str(path))
57
- return mime or "application/octet-stream"
58
-
59
-
60
- def find_runs(workspace: Path) -> list[dict]:
61
- """Recursively find directories that contain an outputs/ subdirectory."""
62
- runs: list[dict] = []
63
- _find_runs_recursive(workspace, workspace, runs)
64
- runs.sort(key=lambda r: (r.get("eval_id", float("inf")), r["id"]))
65
- return runs
66
-
67
-
68
- def _find_runs_recursive(root: Path, current: Path, runs: list[dict]) -> None:
69
- if not current.is_dir():
70
- return
71
-
72
- outputs_dir = current / "outputs"
73
- if outputs_dir.is_dir():
74
- run = build_run(root, current)
75
- if run:
76
- runs.append(run)
77
- return
78
-
79
- skip = {"node_modules", ".git", "__pycache__", "skill", "inputs"}
80
- for child in sorted(current.iterdir()):
81
- if child.is_dir() and child.name not in skip:
82
- _find_runs_recursive(root, child, runs)
83
-
84
-
85
- def build_run(root: Path, run_dir: Path) -> dict | None:
86
- """Build a run dict with prompt, outputs, and grading data."""
87
- prompt = ""
88
- eval_id = None
89
-
90
- # Try eval_metadata.json
91
- for candidate in [run_dir / "eval_metadata.json", run_dir.parent / "eval_metadata.json"]:
92
- if candidate.exists():
93
- try:
94
- metadata = json.loads(candidate.read_text())
95
- prompt = metadata.get("prompt", "")
96
- eval_id = metadata.get("eval_id")
97
- except (json.JSONDecodeError, OSError):
98
- pass
99
- if prompt:
100
- break
101
-
102
- # Fall back to transcript.md
103
- if not prompt:
104
- for candidate in [run_dir / "transcript.md", run_dir / "outputs" / "transcript.md"]:
105
- if candidate.exists():
106
- try:
107
- text = candidate.read_text()
108
- match = re.search(r"## Eval Prompt\n\n([\s\S]*?)(?=\n##|$)", text)
109
- if match:
110
- prompt = match.group(1).strip()
111
- except OSError:
112
- pass
113
- if prompt:
114
- break
115
-
116
- if not prompt:
117
- prompt = "(No prompt found)"
118
-
119
- run_id = str(run_dir.relative_to(root)).replace("/", "-").replace("\\", "-")
120
-
121
- # Collect output files
122
- outputs_dir = run_dir / "outputs"
123
- output_files: list[dict] = []
124
- if outputs_dir.is_dir():
125
- for f in sorted(outputs_dir.iterdir()):
126
- if f.is_file() and f.name not in METADATA_FILES:
127
- output_files.append(embed_file(f))
128
-
129
- # Load grading if present
130
- grading = None
131
- for candidate in [run_dir / "grading.json", run_dir.parent / "grading.json"]:
132
- if candidate.exists():
133
- try:
134
- grading = json.loads(candidate.read_text())
135
- except (json.JSONDecodeError, OSError):
136
- pass
137
- if grading:
138
- break
139
-
140
- return {
141
- "id": run_id,
142
- "prompt": prompt,
143
- "eval_id": eval_id,
144
- "outputs": output_files,
145
- "grading": grading,
146
- }
147
-
148
-
149
- def embed_file(path: Path) -> dict:
150
- """Read a file and return an embedded representation."""
151
- ext = path.suffix.lower()
152
- mime = get_mime_type(path)
153
-
154
- if ext in TEXT_EXTENSIONS:
155
- try:
156
- content = path.read_text(errors="replace")
157
- except OSError:
158
- content = "(Error reading file)"
159
- return {
160
- "name": path.name,
161
- "type": "text",
162
- "content": content,
163
- }
164
- elif ext in IMAGE_EXTENSIONS:
165
- try:
166
- raw = path.read_bytes()
167
- b64 = base64.b64encode(raw).decode("ascii")
168
- except OSError:
169
- return {"name": path.name, "type": "error", "content": "(Error reading file)"}
170
- return {
171
- "name": path.name,
172
- "type": "image",
173
- "mime": mime,
174
- "data_uri": f"data:{mime};base64,{b64}",
175
- }
176
- elif ext == ".pdf":
177
- try:
178
- raw = path.read_bytes()
179
- b64 = base64.b64encode(raw).decode("ascii")
180
- except OSError:
181
- return {"name": path.name, "type": "error", "content": "(Error reading file)"}
182
- return {
183
- "name": path.name,
184
- "type": "pdf",
185
- "data_uri": f"data:{mime};base64,{b64}",
186
- }
187
- elif ext == ".xlsx":
188
- try:
189
- raw = path.read_bytes()
190
- b64 = base64.b64encode(raw).decode("ascii")
191
- except OSError:
192
- return {"name": path.name, "type": "error", "content": "(Error reading file)"}
193
- return {
194
- "name": path.name,
195
- "type": "xlsx",
196
- "data_b64": b64,
197
- }
198
- else:
199
- # Binary / unknown — base64 download link
200
- try:
201
- raw = path.read_bytes()
202
- b64 = base64.b64encode(raw).decode("ascii")
203
- except OSError:
204
- return {"name": path.name, "type": "error", "content": "(Error reading file)"}
205
- return {
206
- "name": path.name,
207
- "type": "binary",
208
- "mime": mime,
209
- "data_uri": f"data:{mime};base64,{b64}",
210
- }
211
-
212
-
213
- def load_previous_iteration(workspace: Path) -> dict[str, dict]:
214
- """Load previous iteration's feedback and outputs.
215
-
216
- Returns a map of run_id -> {"feedback": str, "outputs": list[dict]}.
217
- """
218
- result: dict[str, dict] = {}
219
-
220
- # Load feedback
221
- feedback_map: dict[str, str] = {}
222
- feedback_path = workspace / "feedback.json"
223
- if feedback_path.exists():
224
- try:
225
- data = json.loads(feedback_path.read_text())
226
- feedback_map = {
227
- r["run_id"]: r["feedback"]
228
- for r in data.get("reviews", [])
229
- if r.get("feedback", "").strip()
230
- }
231
- except (json.JSONDecodeError, OSError, KeyError):
232
- pass
233
-
234
- # Load runs (to get outputs)
235
- prev_runs = find_runs(workspace)
236
- for run in prev_runs:
237
- result[run["id"]] = {
238
- "feedback": feedback_map.get(run["id"], ""),
239
- "outputs": run.get("outputs", []),
240
- }
241
-
242
- # Also add feedback for run_ids that had feedback but no matching run
243
- for run_id, fb in feedback_map.items():
244
- if run_id not in result:
245
- result[run_id] = {"feedback": fb, "outputs": []}
246
-
247
- return result
248
-
249
-
250
- def generate_html(
251
- runs: list[dict],
252
- skill_name: str,
253
- previous: dict[str, dict] | None = None,
254
- benchmark: dict | None = None,
255
- ) -> str:
256
- """Generate the complete standalone HTML page with embedded data."""
257
- template_path = Path(__file__).parent / "viewer.html"
258
- template = template_path.read_text()
259
-
260
- # Build previous_feedback and previous_outputs maps for the template
261
- previous_feedback: dict[str, str] = {}
262
- previous_outputs: dict[str, list[dict]] = {}
263
- if previous:
264
- for run_id, data in previous.items():
265
- if data.get("feedback"):
266
- previous_feedback[run_id] = data["feedback"]
267
- if data.get("outputs"):
268
- previous_outputs[run_id] = data["outputs"]
269
-
270
- embedded = {
271
- "skill_name": skill_name,
272
- "runs": runs,
273
- "previous_feedback": previous_feedback,
274
- "previous_outputs": previous_outputs,
275
- }
276
- if benchmark:
277
- embedded["benchmark"] = benchmark
278
-
279
- data_json = json.dumps(embedded)
280
-
281
- return template.replace("/*__EMBEDDED_DATA__*/", f"const EMBEDDED_DATA = {data_json};")
282
-
283
-
284
- # ---------------------------------------------------------------------------
285
- # HTTP server (stdlib only, zero dependencies)
286
- # ---------------------------------------------------------------------------
287
-
288
- def _kill_port(port: int) -> None:
289
- """Kill any process listening on the given port."""
290
- try:
291
- result = subprocess.run(
292
- ["lsof", "-ti", f":{port}"],
293
- capture_output=True, text=True, timeout=5,
294
- )
295
- for pid_str in result.stdout.strip().split("\n"):
296
- if pid_str.strip():
297
- try:
298
- os.kill(int(pid_str.strip()), signal.SIGTERM)
299
- except (ProcessLookupError, ValueError):
300
- pass
301
- if result.stdout.strip():
302
- time.sleep(0.5)
303
- except subprocess.TimeoutExpired:
304
- pass
305
- except FileNotFoundError:
306
- print("Note: lsof not found, cannot check if port is in use", file=sys.stderr)
307
-
308
- class ReviewHandler(BaseHTTPRequestHandler):
309
- """Serves the review HTML and handles feedback saves.
310
-
311
- Regenerates the HTML on each page load so that refreshing the browser
312
- picks up new eval outputs without restarting the server.
313
- """
314
-
315
- def __init__(
316
- self,
317
- workspace: Path,
318
- skill_name: str,
319
- feedback_path: Path,
320
- previous: dict[str, dict],
321
- benchmark_path: Path | None,
322
- *args,
323
- **kwargs,
324
- ):
325
- self.workspace = workspace
326
- self.skill_name = skill_name
327
- self.feedback_path = feedback_path
328
- self.previous = previous
329
- self.benchmark_path = benchmark_path
330
- super().__init__(*args, **kwargs)
331
-
332
- def do_GET(self) -> None:
333
- if self.path == "/" or self.path == "/index.html":
334
- # Regenerate HTML on each request (re-scans workspace for new outputs)
335
- runs = find_runs(self.workspace)
336
- benchmark = None
337
- if self.benchmark_path and self.benchmark_path.exists():
338
- try:
339
- benchmark = json.loads(self.benchmark_path.read_text())
340
- except (json.JSONDecodeError, OSError):
341
- pass
342
- html = generate_html(runs, self.skill_name, self.previous, benchmark)
343
- content = html.encode("utf-8")
344
- self.send_response(200)
345
- self.send_header("Content-Type", "text/html; charset=utf-8")
346
- self.send_header("Content-Length", str(len(content)))
347
- self.end_headers()
348
- self.wfile.write(content)
349
- elif self.path == "/api/feedback":
350
- data = b"{}"
351
- if self.feedback_path.exists():
352
- data = self.feedback_path.read_bytes()
353
- self.send_response(200)
354
- self.send_header("Content-Type", "application/json")
355
- self.send_header("Content-Length", str(len(data)))
356
- self.end_headers()
357
- self.wfile.write(data)
358
- else:
359
- self.send_error(404)
360
-
361
- def do_POST(self) -> None:
362
- if self.path == "/api/feedback":
363
- length = int(self.headers.get("Content-Length", 0))
364
- body = self.rfile.read(length)
365
- try:
366
- data = json.loads(body)
367
- if not isinstance(data, dict) or "reviews" not in data:
368
- raise ValueError("Expected JSON object with 'reviews' key")
369
- self.feedback_path.write_text(json.dumps(data, indent=2) + "\n")
370
- resp = b'{"ok":true}'
371
- self.send_response(200)
372
- except (json.JSONDecodeError, OSError, ValueError) as e:
373
- resp = json.dumps({"error": str(e)}).encode()
374
- self.send_response(500)
375
- self.send_header("Content-Type", "application/json")
376
- self.send_header("Content-Length", str(len(resp)))
377
- self.end_headers()
378
- self.wfile.write(resp)
379
- else:
380
- self.send_error(404)
381
-
382
- def log_message(self, format: str, *args: object) -> None:
383
- # Suppress request logging to keep terminal clean
384
- pass
385
-
386
-
387
- def main() -> None:
388
- parser = argparse.ArgumentParser(description="Generate and serve eval review")
389
- parser.add_argument("workspace", type=Path, help="Path to workspace directory")
390
- parser.add_argument("--port", "-p", type=int, default=3117, help="Server port (default: 3117)")
391
- parser.add_argument("--skill-name", "-n", type=str, default=None, help="Skill name for header")
392
- parser.add_argument(
393
- "--previous-workspace", type=Path, default=None,
394
- help="Path to previous iteration's workspace (shows old outputs and feedback as context)",
395
- )
396
- parser.add_argument(
397
- "--benchmark", type=Path, default=None,
398
- help="Path to benchmark.json to show in the Benchmark tab",
399
- )
400
- parser.add_argument(
401
- "--static", "-s", type=Path, default=None,
402
- help="Write standalone HTML to this path instead of starting a server",
403
- )
404
- args = parser.parse_args()
405
-
406
- workspace = args.workspace.resolve()
407
- if not workspace.is_dir():
408
- print(f"Error: {workspace} is not a directory", file=sys.stderr)
409
- sys.exit(1)
410
-
411
- runs = find_runs(workspace)
412
- if not runs:
413
- print(f"No runs found in {workspace}", file=sys.stderr)
414
- sys.exit(1)
415
-
416
- skill_name = args.skill_name or workspace.name.replace("-workspace", "")
417
- feedback_path = workspace / "feedback.json"
418
-
419
- previous: dict[str, dict] = {}
420
- if args.previous_workspace:
421
- previous = load_previous_iteration(args.previous_workspace.resolve())
422
-
423
- benchmark_path = args.benchmark.resolve() if args.benchmark else None
424
- benchmark = None
425
- if benchmark_path and benchmark_path.exists():
426
- try:
427
- benchmark = json.loads(benchmark_path.read_text())
428
- except (json.JSONDecodeError, OSError):
429
- pass
430
-
431
- if args.static:
432
- html = generate_html(runs, skill_name, previous, benchmark)
433
- args.static.parent.mkdir(parents=True, exist_ok=True)
434
- args.static.write_text(html)
435
- print(f"\n Static viewer written to: {args.static}\n")
436
- sys.exit(0)
437
-
438
- # Kill any existing process on the target port
439
- port = args.port
440
- _kill_port(port)
441
- handler = partial(ReviewHandler, workspace, skill_name, feedback_path, previous, benchmark_path)
442
- try:
443
- server = HTTPServer(("127.0.0.1", port), handler)
444
- except OSError:
445
- # Port still in use after kill attempt — find a free one
446
- server = HTTPServer(("127.0.0.1", 0), handler)
447
- port = server.server_address[1]
448
-
449
- url = f"http://localhost:{port}"
450
- print(f"\n Eval Viewer")
451
- print(f" ─────────────────────────────────")
452
- print(f" URL: {url}")
453
- print(f" Workspace: {workspace}")
454
- print(f" Feedback: {feedback_path}")
455
- if previous:
456
- print(f" Previous: {args.previous_workspace} ({len(previous)} runs)")
457
- if benchmark_path:
458
- print(f" Benchmark: {benchmark_path}")
459
- print(f"\n Press Ctrl+C to stop.\n")
460
-
461
- webbrowser.open(url)
462
-
463
- try:
464
- server.serve_forever()
465
- except KeyboardInterrupt:
466
- print("\nStopped.")
467
- server.server_close()
468
-
469
-
470
- if __name__ == "__main__":
471
- main()