mednotes-opencode 0.1.0

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 (430) hide show
  1. package/.opencode/agents/med-chat-triager.md +204 -0
  2. package/.opencode/agents/med-flashcard-maker.md +63 -0
  3. package/.opencode/agents/med-knowledge-architect.md +230 -0
  4. package/.opencode/agents/med-link-graph-curator.md +177 -0
  5. package/.opencode/agents/med-publish-guard.md +62 -0
  6. package/.opencode/commands/flashcards.md +25 -0
  7. package/.opencode/commands/mednotes/create.md +25 -0
  8. package/.opencode/commands/mednotes/enrich.md +27 -0
  9. package/.opencode/commands/mednotes/fix-wiki.md +27 -0
  10. package/.opencode/commands/mednotes/history.md +22 -0
  11. package/.opencode/commands/mednotes/link-body.md +25 -0
  12. package/.opencode/commands/mednotes/link-related.md +27 -0
  13. package/.opencode/commands/mednotes/link.md +27 -0
  14. package/.opencode/commands/mednotes/pdf-library.md +27 -0
  15. package/.opencode/commands/mednotes/process-chats.md +23 -0
  16. package/.opencode/commands/mednotes/setup.md +21 -0
  17. package/.opencode/commands/mednotes/status.md +27 -0
  18. package/.opencode/commands/mednotes/telemetry.md +27 -0
  19. package/.opencode/commands/report.md +26 -0
  20. package/.opencode/mednotes/AGENTS.md +57 -0
  21. package/.opencode/mednotes/agents/med-chat-triager.md +197 -0
  22. package/.opencode/mednotes/agents/med-flashcard-maker.md +56 -0
  23. package/.opencode/mednotes/agents/med-knowledge-architect.md +224 -0
  24. package/.opencode/mednotes/agents/med-link-graph-curator.md +171 -0
  25. package/.opencode/mednotes/agents/med-publish-guard.md +55 -0
  26. package/.opencode/mednotes/contracts/.gitkeep +1 -0
  27. package/.opencode/mednotes/contracts/agents.json +116 -0
  28. package/.opencode/mednotes/contracts/opencode-plugin.json +70 -0
  29. package/.opencode/mednotes/docs/agent-prompt-hardening.md +567 -0
  30. package/.opencode/mednotes/docs/agent-role-contracts.md +94 -0
  31. package/.opencode/mednotes/docs/anki-mcp-twenty-rules.md +214 -0
  32. package/.opencode/mednotes/docs/anki-templates/README.md +39 -0
  33. package/.opencode/mednotes/docs/anki-templates/cloze.back.html +23 -0
  34. package/.opencode/mednotes/docs/anki-templates/cloze.front.html +14 -0
  35. package/.opencode/mednotes/docs/anki-templates/qa.back.html +24 -0
  36. package/.opencode/mednotes/docs/anki-templates/qa.front.html +14 -0
  37. package/.opencode/mednotes/docs/anki-templates/style.css +182 -0
  38. package/.opencode/mednotes/docs/atomicity-splitting-policy.md +113 -0
  39. package/.opencode/mednotes/docs/extension-docs.md +40 -0
  40. package/.opencode/mednotes/docs/flashcard-ingestion.md +278 -0
  41. package/.opencode/mednotes/docs/knowledge-architect.md +208 -0
  42. package/.opencode/mednotes/docs/merge-policy.md +110 -0
  43. package/.opencode/mednotes/docs/public-vocabulary.md +104 -0
  44. package/.opencode/mednotes/docs/semantic-linker.md +141 -0
  45. package/.opencode/mednotes/docs/taxonomy-policy.md +90 -0
  46. package/.opencode/mednotes/docs/triage-policy.md +187 -0
  47. package/.opencode/mednotes/docs/vault-version-control.md +758 -0
  48. package/.opencode/mednotes/docs/vocabulary-db-recovery.md +58 -0
  49. package/.opencode/mednotes/docs/workflow-output-contract.md +779 -0
  50. package/.opencode/mednotes/hooks/hooks.json +79 -0
  51. package/.opencode/mednotes/package-lock.json +6361 -0
  52. package/.opencode/mednotes/package.json +15 -0
  53. package/.opencode/mednotes/pyproject.toml +48 -0
  54. package/.opencode/mednotes/scripts/bootstrap_windows_python_uv.cmd +13 -0
  55. package/.opencode/mednotes/scripts/bootstrap_windows_python_uv.ps1 +172 -0
  56. package/.opencode/mednotes/scripts/enrich_notes.py +23 -0
  57. package/.opencode/mednotes/scripts/full_reset_windows_python_uv.cmd +13 -0
  58. package/.opencode/mednotes/scripts/hooks/antigravity_hook_status.mjs +212 -0
  59. package/.opencode/mednotes/scripts/hooks/mednotes_hook/adapters/antigravity.mjs +169 -0
  60. package/.opencode/mednotes/scripts/hooks/mednotes_hook/adapters/harness_payload.mjs +103 -0
  61. package/.opencode/mednotes/scripts/hooks/mednotes_hook/adapters/opencode_plugin.mjs +341 -0
  62. package/.opencode/mednotes/scripts/hooks/mednotes_hook/adapters/opencode_user_config_sync.mjs +177 -0
  63. package/.opencode/mednotes/scripts/hooks/mednotes_hook/anki_preflight.mjs +214 -0
  64. package/.opencode/mednotes/scripts/hooks/mednotes_hook/cli.mjs +143 -0
  65. package/.opencode/mednotes/scripts/hooks/mednotes_hook/diagnostics.mjs +11 -0
  66. package/.opencode/mednotes/scripts/hooks/mednotes_hook/domain/agent_directive_core.mjs +160 -0
  67. package/.opencode/mednotes/scripts/hooks/mednotes_hook/fsm_directive.mjs +1470 -0
  68. package/.opencode/mednotes/scripts/hooks/mednotes_hook/hook_errors.mjs +120 -0
  69. package/.opencode/mednotes/scripts/hooks/mednotes_hook/retention.mjs +114 -0
  70. package/.opencode/mednotes/scripts/hooks/mednotes_hook/runtime.mjs +174 -0
  71. package/.opencode/mednotes/scripts/hooks/mednotes_hook/telemetry_capture.mjs +511 -0
  72. package/.opencode/mednotes/scripts/hooks/mednotes_hook/vault_guard.mjs +624 -0
  73. package/.opencode/mednotes/scripts/hooks/mednotes_hook.mjs +5 -0
  74. package/.opencode/mednotes/scripts/mednotes/_runtime_paths.py +24 -0
  75. package/.opencode/mednotes/scripts/mednotes/anki_model_validator.py +18 -0
  76. package/.opencode/mednotes/scripts/mednotes/capture_extension_diff.py +1562 -0
  77. package/.opencode/mednotes/scripts/mednotes/feedback_report.py +16 -0
  78. package/.opencode/mednotes/scripts/mednotes/flashcard_index.py +18 -0
  79. package/.opencode/mednotes/scripts/mednotes/flashcard_pipeline.py +18 -0
  80. package/.opencode/mednotes/scripts/mednotes/flashcard_report.py +18 -0
  81. package/.opencode/mednotes/scripts/mednotes/flashcard_sources.py +18 -0
  82. package/.opencode/mednotes/scripts/mednotes/obsidian/README.md +6 -0
  83. package/.opencode/mednotes/scripts/mednotes/obsidian_note_utils.py +20 -0
  84. package/.opencode/mednotes/scripts/mednotes/pdf_library/cli.py +16 -0
  85. package/.opencode/mednotes/scripts/mednotes/project_fsm.py +229 -0
  86. package/.opencode/mednotes/scripts/mednotes/setup_telemetry_email.py +404 -0
  87. package/.opencode/mednotes/scripts/mednotes/sync_anki_twenty_rules.py +18 -0
  88. package/.opencode/mednotes/scripts/mednotes/sync_opencode_user_config.py +36 -0
  89. package/.opencode/mednotes/scripts/mednotes/wiki/cli.py +20 -0
  90. package/.opencode/mednotes/scripts/mednotes/wiki_graph.py +18 -0
  91. package/.opencode/mednotes/scripts/mednotes/wiki_tree.py +134 -0
  92. package/.opencode/mednotes/scripts/reset_windows_python_uv.ps1 +625 -0
  93. package/.opencode/mednotes/scripts/run_python.mjs +109 -0
  94. package/.opencode/mednotes/scripts/vault/vault_commit.ps1 +19 -0
  95. package/.opencode/mednotes/scripts/vault/vault_commit.sh +18 -0
  96. package/.opencode/mednotes/scripts/vault/vault_git.ps1 +19 -0
  97. package/.opencode/mednotes/scripts/vault/vault_git.py +3107 -0
  98. package/.opencode/mednotes/scripts/vault/vault_git.sh +18 -0
  99. package/.opencode/mednotes/scripts/vault/vault_precommit.ps1 +19 -0
  100. package/.opencode/mednotes/scripts/vault/vault_precommit.sh +18 -0
  101. package/.opencode/mednotes/skills/THIRD_PARTY_NOTICES.md +45 -0
  102. package/.opencode/mednotes/skills/create-medical-flashcards/SKILL.md +113 -0
  103. package/.opencode/mednotes/skills/create-medical-note/SKILL.md +90 -0
  104. package/.opencode/mednotes/skills/enrich-medical-note/SKILL.md +120 -0
  105. package/.opencode/mednotes/skills/fix-medical-wiki/SKILL.md +559 -0
  106. package/.opencode/mednotes/skills/link-medical-wiki/SKILL.md +224 -0
  107. package/.opencode/mednotes/skills/obsidian-cli/SKILL.md +118 -0
  108. package/.opencode/mednotes/skills/obsidian-markdown/SKILL.md +207 -0
  109. package/.opencode/mednotes/skills/obsidian-markdown/references/CALLOUTS.md +58 -0
  110. package/.opencode/mednotes/skills/obsidian-markdown/references/EMBEDS.md +63 -0
  111. package/.opencode/mednotes/skills/obsidian-markdown/references/PROPERTIES.md +61 -0
  112. package/.opencode/mednotes/skills/obsidian-ops/SKILL.md +136 -0
  113. package/.opencode/mednotes/skills/pdf-library/SKILL.md +45 -0
  114. package/.opencode/mednotes/skills/process-medical-chats/SKILL.md +246 -0
  115. package/.opencode/mednotes/skills/workflow-report/SKILL.md +100 -0
  116. package/.opencode/mednotes/src/mednotes/__init__.py +5 -0
  117. package/.opencode/mednotes/src/mednotes/domains/__init__.py +5 -0
  118. package/.opencode/mednotes/src/mednotes/domains/flashcards/README.md +26 -0
  119. package/.opencode/mednotes/src/mednotes/domains/flashcards/__init__.py +2 -0
  120. package/.opencode/mednotes/src/mednotes/domains/flashcards/build_demo_apkg.py +177 -0
  121. package/.opencode/mednotes/src/mednotes/domains/flashcards/contracts.py +385 -0
  122. package/.opencode/mednotes/src/mednotes/domains/flashcards/flashcards_machine.py +522 -0
  123. package/.opencode/mednotes/src/mednotes/domains/flashcards/fsm.py +817 -0
  124. package/.opencode/mednotes/src/mednotes/domains/flashcards/index.py +630 -0
  125. package/.opencode/mednotes/src/mednotes/domains/flashcards/install_models.py +445 -0
  126. package/.opencode/mednotes/src/mednotes/domains/flashcards/model.py +359 -0
  127. package/.opencode/mednotes/src/mednotes/domains/flashcards/obsidian_links.py +135 -0
  128. package/.opencode/mednotes/src/mednotes/domains/flashcards/obsidian_note_utils.py +546 -0
  129. package/.opencode/mednotes/src/mednotes/domains/flashcards/pipeline.py +580 -0
  130. package/.opencode/mednotes/src/mednotes/domains/flashcards/report.py +510 -0
  131. package/.opencode/mednotes/src/mednotes/domains/flashcards/sources.py +682 -0
  132. package/.opencode/mednotes/src/mednotes/domains/flashcards/sync_rules.py +184 -0
  133. package/.opencode/mednotes/src/mednotes/domains/history/__init__.py +1 -0
  134. package/.opencode/mednotes/src/mednotes/domains/history/history_fsm.py +852 -0
  135. package/.opencode/mednotes/src/mednotes/domains/history/history_machine.py +453 -0
  136. package/.opencode/mednotes/src/mednotes/domains/setup/__init__.py +7 -0
  137. package/.opencode/mednotes/src/mednotes/domains/setup/setup_fsm.py +808 -0
  138. package/.opencode/mednotes/src/mednotes/domains/setup/setup_machine.py +973 -0
  139. package/.opencode/mednotes/src/mednotes/domains/wiki/README.md +64 -0
  140. package/.opencode/mednotes/src/mednotes/domains/wiki/__init__.py +1 -0
  141. package/.opencode/mednotes/src/mednotes/domains/wiki/api.py +668 -0
  142. package/.opencode/mednotes/src/mednotes/domains/wiki/batch_state.py +102 -0
  143. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/__init__.py +1 -0
  144. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/atomicity/__init__.py +1 -0
  145. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/atomicity/atomicity.py +877 -0
  146. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/body_link/__init__.py +1 -0
  147. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/body_link/body_linker.py +1562 -0
  148. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/effects/__init__.py +1 -0
  149. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/effects/effect_adapters.py +949 -0
  150. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/effects/fix_wiki_runtime_adapters.py +433 -0
  151. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/graph/__init__.py +1 -0
  152. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/graph/coverage.py +413 -0
  153. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/graph/graph.py +396 -0
  154. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/graph/graph_fixes.py +161 -0
  155. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/hygiene/__init__.py +1 -0
  156. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/hygiene/hygiene.py +483 -0
  157. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/__init__.py +2 -0
  158. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/anchors.py +185 -0
  159. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/core/__init__.py +0 -0
  160. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/core/cache.py +223 -0
  161. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/core/config.py +131 -0
  162. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/core/download.py +224 -0
  163. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/core/frontmatter.py +59 -0
  164. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/core/insert.py +227 -0
  165. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/core/local_import.py +54 -0
  166. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/sources/__init__.py +42 -0
  167. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/sources/web_profiles.py +99 -0
  168. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/sources/web_search.py +203 -0
  169. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/illustrate/sources/wikimedia.py +102 -0
  170. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/markdown/__init__.py +1 -0
  171. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/markdown/markdown_db_adapter.mjs +434 -0
  172. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/markdown/markdown_node_runtime.py +274 -0
  173. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/markdown/markdown_query.py +227 -0
  174. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/__init__.py +1 -0
  175. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/artifacts.py +605 -0
  176. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/canonical_merge.py +277 -0
  177. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/markdown_zones.py +85 -0
  178. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/meaning_planner.py +307 -0
  179. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/note_iter.py +67 -0
  180. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/note_merge.py +278 -0
  181. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/note_plan.py +409 -0
  182. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/note_policy.py +22 -0
  183. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/note_style/__init__.py +79 -0
  184. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/note_style/fixes.py +264 -0
  185. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/note_style/frontmatter.py +435 -0
  186. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/note_style/models.py +208 -0
  187. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/note_style/prompts.py +37 -0
  188. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/note_style/tables.py +236 -0
  189. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/note_style/validate.py +404 -0
  190. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/provenance.py +478 -0
  191. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/raw_chats.py +273 -0
  192. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/notes/sources_backfill.py +235 -0
  193. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/__init__.py +10 -0
  194. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/anchors.py +16 -0
  195. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/captions.py +47 -0
  196. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/cli.py +179 -0
  197. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/cloud.py +52 -0
  198. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/config.py +196 -0
  199. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/context_packets.py +76 -0
  200. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/db.py +81 -0
  201. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/doctor.py +102 -0
  202. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/figure_ids.py +42 -0
  203. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/ingest.py +326 -0
  204. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/insert.py +316 -0
  205. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/mentions.py +57 -0
  206. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/ocr.py +71 -0
  207. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/paths.py +35 -0
  208. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/pdf_engine.py +77 -0
  209. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/schema.py +155 -0
  210. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/search.py +188 -0
  211. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/tui/__init__.py +1 -0
  212. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/tui/app.py +89 -0
  213. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/tui/image_backend.py +29 -0
  214. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/pdf/tui/state.py +65 -0
  215. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/publish/__init__.py +1 -0
  216. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/publish/publish.py +1139 -0
  217. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/publish/publish_receipts.py +365 -0
  218. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/publish/publish_recovery.py +240 -0
  219. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/quality/__init__.py +1 -0
  220. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/quality/agent_behavior_corpus.py +2069 -0
  221. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/quality/agent_report_validation.py +4448 -0
  222. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/quality/agent_run_audit.py +852 -0
  223. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/quality/architect_prompt_eval.py +341 -0
  224. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/quality/body_linker_eval.py +240 -0
  225. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/quality/curator_output_validation.py +175 -0
  226. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/quality/curator_prompt_eval.py +865 -0
  227. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/quality/triager_prompt_eval.py +1295 -0
  228. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/related_notes/__init__.py +1 -0
  229. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/related_notes/related_notes.py +1920 -0
  230. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/related_notes/related_notes_headless.py +1186 -0
  231. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/specialist/__init__.py +1 -0
  232. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/specialist/plan_attestation.py +148 -0
  233. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/specialist/specialist_receipts.py +360 -0
  234. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/specialist/specialist_runtime.py +52 -0
  235. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/specialist/specialist_task_runner.py +2470 -0
  236. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/style/__init__.py +1 -0
  237. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/style/style.py +1952 -0
  238. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/subagents/__init__.py +1 -0
  239. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/subagents/agents.py +1767 -0
  240. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/__init__.py +1 -0
  241. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/alias_projection.py +331 -0
  242. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/link_terms.py +151 -0
  243. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/llm_disambiguation.py +182 -0
  244. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/taxonomy/__init__.py +116 -0
  245. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/taxonomy/audit.py +201 -0
  246. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/taxonomy/migration.py +314 -0
  247. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/taxonomy/normalize.py +72 -0
  248. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/taxonomy/policy.py +135 -0
  249. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/taxonomy/resolve.py +413 -0
  250. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/taxonomy/schema.py +157 -0
  251. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/taxonomy/status.py +137 -0
  252. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/vocabulary_bootstrap.py +509 -0
  253. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/vocabulary_curator_batch.py +1115 -0
  254. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/vocabulary_ingestion.py +632 -0
  255. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/vocabulary_map.py +930 -0
  256. package/.opencode/mednotes/src/mednotes/domains/wiki/capabilities/vocabulary/vocabulary_recovery.py +1388 -0
  257. package/.opencode/mednotes/src/mednotes/domains/wiki/cli.py +6665 -0
  258. package/.opencode/mednotes/src/mednotes/domains/wiki/common.py +69 -0
  259. package/.opencode/mednotes/src/mednotes/domains/wiki/config.py +210 -0
  260. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/__init__.py +74 -0
  261. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/agent_report.py +242 -0
  262. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/agent_run_audit.py +196 -0
  263. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/agents.py +601 -0
  264. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/curator.py +256 -0
  265. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/effect_payloads.py +519 -0
  266. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/happy_path.py +190 -0
  267. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/link_git.py +110 -0
  268. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/link_runtime_artifact.py +52 -0
  269. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/note_plan.py +75 -0
  270. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/paths.py +114 -0
  271. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/public_report.py +53 -0
  272. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/publish.py +111 -0
  273. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/raw_coverage.py +217 -0
  274. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/related_notes.py +136 -0
  275. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/related_notes_headless.py +153 -0
  276. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/related_notes_runtime.py +395 -0
  277. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/schema_registry.py +637 -0
  278. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/specialist.py +432 -0
  279. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/status.py +62 -0
  280. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/style_rewrite.py +568 -0
  281. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/vocabulary_ingestion.py +223 -0
  282. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/workflow_blockers.py +510 -0
  283. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/workflow_guardrails.py +637 -0
  284. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/workflow_outcomes.py +121 -0
  285. package/.opencode/mednotes/src/mednotes/domains/wiki/contracts/workflow_receipts.py +100 -0
  286. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/__init__.py +1 -0
  287. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/__init__.py +1 -0
  288. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/__main__.py +4 -0
  289. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/cli.py +275 -0
  290. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/__init__.py +2 -0
  291. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/candidates.py +193 -0
  292. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/cli.py +189 -0
  293. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/gemini.py +220 -0
  294. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/inputs.py +120 -0
  295. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/models.py +34 -0
  296. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/parsing.py +48 -0
  297. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/prompts.py +216 -0
  298. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/quality.py +54 -0
  299. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/reporting.py +24 -0
  300. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/runner.py +433 -0
  301. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/utils.py +39 -0
  302. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/enrich/workflow/vault_guard_bridge.py +17 -0
  303. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/__init__.py +1 -0
  304. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/fix_wiki_context_packets.py +454 -0
  305. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/fix_wiki_decision_projection.py +133 -0
  306. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/fix_wiki_effects.py +1260 -0
  307. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/fix_wiki_fsm.py +2768 -0
  308. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/fix_wiki_machine.py +1588 -0
  309. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/fix_wiki_plan.py +306 -0
  310. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/fix_wiki_primary_objective.py +316 -0
  311. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/fix_wiki_problem.py +153 -0
  312. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/fix_wiki_receipt_evidence.py +306 -0
  313. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/fix_wiki_states.py +290 -0
  314. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/fix_wiki_user_report.py +342 -0
  315. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/fix_wiki/health.py +6332 -0
  316. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link/__init__.py +1 -0
  317. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link/link_fsm.py +1119 -0
  318. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link/link_git.py +638 -0
  319. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link/link_machine.py +1106 -0
  320. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link/link_retry_governance.py +374 -0
  321. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link/link_runtime_result.py +485 -0
  322. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link/link_triggers.py +183 -0
  323. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link/linking.py +2758 -0
  324. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link/reference_repair.py +718 -0
  325. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link/related_notes_fsm.py +1855 -0
  326. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link_related/__init__.py +1 -0
  327. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/link_related/link_related_machine.py +834 -0
  328. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/process_chats/__init__.py +1 -0
  329. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/process_chats/process_chats_fsm.py +1592 -0
  330. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/process_chats/process_chats_machine.py +3097 -0
  331. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/process_chats/process_chats_primary_objective.py +28 -0
  332. package/.opencode/mednotes/src/mednotes/domains/wiki/flows/process_chats/process_chats_runtime_result.py +185 -0
  333. package/.opencode/mednotes/src/mednotes/domains/wiki/performance.py +97 -0
  334. package/.opencode/mednotes/src/mednotes/kernel/__init__.py +6 -0
  335. package/.opencode/mednotes/src/mednotes/kernel/agent_directive.py +336 -0
  336. package/.opencode/mednotes/src/mednotes/kernel/base.py +51 -0
  337. package/.opencode/mednotes/src/mednotes/kernel/blockers.py +39 -0
  338. package/.opencode/mednotes/src/mednotes/kernel/effect_executor.py +55 -0
  339. package/.opencode/mednotes/src/mednotes/kernel/effect_intent.py +69 -0
  340. package/.opencode/mednotes/src/mednotes/kernel/effects.py +160 -0
  341. package/.opencode/mednotes/src/mednotes/kernel/errors.py +38 -0
  342. package/.opencode/mednotes/src/mednotes/kernel/fsm_event.py +35 -0
  343. package/.opencode/mednotes/src/mednotes/kernel/fsm_model.py +55 -0
  344. package/.opencode/mednotes/src/mednotes/kernel/fsm_transition_result.py +75 -0
  345. package/.opencode/mednotes/src/mednotes/kernel/guardrails.py +188 -0
  346. package/.opencode/mednotes/src/mednotes/kernel/progress.py +319 -0
  347. package/.opencode/mednotes/src/mednotes/kernel/public_report.py +346 -0
  348. package/.opencode/mednotes/src/mednotes/kernel/state_machine.py +164 -0
  349. package/.opencode/mednotes/src/mednotes/kernel/workflow.py +619 -0
  350. package/.opencode/mednotes/src/mednotes/platform/__init__.py +5 -0
  351. package/.opencode/mednotes/src/mednotes/platform/backup_policy.py +382 -0
  352. package/.opencode/mednotes/src/mednotes/platform/feedback/__init__.py +62 -0
  353. package/.opencode/mednotes/src/mednotes/platform/feedback/cli.py +275 -0
  354. package/.opencode/mednotes/src/mednotes/platform/feedback/contracts.py +83 -0
  355. package/.opencode/mednotes/src/mednotes/platform/feedback/core.py +4168 -0
  356. package/.opencode/mednotes/src/mednotes/platform/feedback/integrity.py +989 -0
  357. package/.opencode/mednotes/src/mednotes/platform/feedback/operational_contract.py +2293 -0
  358. package/.opencode/mednotes/src/mednotes/platform/feedback/telemetry.py +875 -0
  359. package/.opencode/mednotes/src/mednotes/platform/feedback/telemetry_config.py +65 -0
  360. package/.opencode/mednotes/src/mednotes/platform/opencode_runtime_config.py +182 -0
  361. package/.opencode/mednotes/src/mednotes/platform/paths/__init__.py +1560 -0
  362. package/.opencode/mednotes/src/mednotes/platform/secrets.py +89 -0
  363. package/.opencode/mednotes/src/mednotes/platform/user_config.py +103 -0
  364. package/.opencode/mednotes/src/mednotes/platform/vault_guard.py +214 -0
  365. package/.opencode/mednotes/uv.lock +932 -0
  366. package/.opencode/mednotes.generated.json +395 -0
  367. package/.opencode/opencode.json +31 -0
  368. package/.opencode/plugins/mednotes-fsm.mjs +7 -0
  369. package/.opencode/plugins/mednotes_hook/adapters/antigravity.mjs +169 -0
  370. package/.opencode/plugins/mednotes_hook/adapters/harness_payload.mjs +103 -0
  371. package/.opencode/plugins/mednotes_hook/adapters/opencode_plugin.mjs +341 -0
  372. package/.opencode/plugins/mednotes_hook/adapters/opencode_user_config_sync.mjs +177 -0
  373. package/.opencode/plugins/mednotes_hook/anki_preflight.mjs +214 -0
  374. package/.opencode/plugins/mednotes_hook/cli.mjs +143 -0
  375. package/.opencode/plugins/mednotes_hook/diagnostics.mjs +11 -0
  376. package/.opencode/plugins/mednotes_hook/domain/agent_directive_core.mjs +160 -0
  377. package/.opencode/plugins/mednotes_hook/fsm_directive.mjs +1470 -0
  378. package/.opencode/plugins/mednotes_hook/hook_errors.mjs +120 -0
  379. package/.opencode/plugins/mednotes_hook/retention.mjs +114 -0
  380. package/.opencode/plugins/mednotes_hook/runtime.mjs +174 -0
  381. package/.opencode/plugins/mednotes_hook/telemetry_capture.mjs +511 -0
  382. package/.opencode/plugins/mednotes_hook/vault_guard.mjs +624 -0
  383. package/AGENTS.md +57 -0
  384. package/README.md +194 -0
  385. package/adapters/antigravity/agents.json +80 -0
  386. package/adapters/antigravity/templates/med-chat-triager.md +214 -0
  387. package/adapters/antigravity/templates/med-flashcard-maker.md +72 -0
  388. package/adapters/antigravity/templates/med-knowledge-architect.md +241 -0
  389. package/adapters/antigravity/templates/med-link-graph-curator.md +187 -0
  390. package/adapters/antigravity/templates/med-publish-guard.md +71 -0
  391. package/adapters/gemini-cli/gemini-extension.json +14 -0
  392. package/adapters/gemini-cli/package.json +15 -0
  393. package/adapters/gemini-cli/pyproject.toml +48 -0
  394. package/bin/mednotes-opencode.mjs +155 -0
  395. package/contracts/agents.json +116 -0
  396. package/core/agents/med-chat-triager.md +197 -0
  397. package/core/agents/med-flashcard-maker.md +56 -0
  398. package/core/agents/med-knowledge-architect.md +224 -0
  399. package/core/agents/med-link-graph-curator.md +171 -0
  400. package/core/agents/med-publish-guard.md +55 -0
  401. package/core/commands/flashcards.toml +22 -0
  402. package/core/commands/mednotes/create.toml +22 -0
  403. package/core/commands/mednotes/enrich.toml +24 -0
  404. package/core/commands/mednotes/fix-wiki.toml +24 -0
  405. package/core/commands/mednotes/history.toml +19 -0
  406. package/core/commands/mednotes/link-body.toml +22 -0
  407. package/core/commands/mednotes/link-related.toml +24 -0
  408. package/core/commands/mednotes/link.toml +24 -0
  409. package/core/commands/mednotes/pdf-library.toml +24 -0
  410. package/core/commands/mednotes/process-chats.toml +20 -0
  411. package/core/commands/mednotes/setup.toml +18 -0
  412. package/core/commands/mednotes/status.toml +24 -0
  413. package/core/commands/mednotes/telemetry.toml +24 -0
  414. package/core/commands/report.toml +23 -0
  415. package/core/skills/THIRD_PARTY_NOTICES.md +45 -0
  416. package/core/skills/create-medical-flashcards/SKILL.md +113 -0
  417. package/core/skills/create-medical-note/SKILL.md +90 -0
  418. package/core/skills/enrich-medical-note/SKILL.md +120 -0
  419. package/core/skills/fix-medical-wiki/SKILL.md +559 -0
  420. package/core/skills/link-medical-wiki/SKILL.md +224 -0
  421. package/core/skills/obsidian-cli/SKILL.md +118 -0
  422. package/core/skills/obsidian-markdown/SKILL.md +207 -0
  423. package/core/skills/obsidian-markdown/references/CALLOUTS.md +58 -0
  424. package/core/skills/obsidian-markdown/references/EMBEDS.md +63 -0
  425. package/core/skills/obsidian-markdown/references/PROPERTIES.md +61 -0
  426. package/core/skills/obsidian-ops/SKILL.md +136 -0
  427. package/core/skills/pdf-library/SKILL.md +45 -0
  428. package/core/skills/process-medical-chats/SKILL.md +246 -0
  429. package/core/skills/workflow-report/SKILL.md +100 -0
  430. package/package.json +45 -0
@@ -0,0 +1,188 @@
1
+ from __future__ import annotations
2
+
3
+ from pydantic import ConfigDict, Field, StrictBool, StrictInt, StrictStr
4
+
5
+ from mednotes.kernel.base import ContractModel, JsonObject, JsonObjectAdapter, JsonValue
6
+
7
+ CONTRACT_GAP_MISSING_NEXT_ACTION = "contract_gap.missing_next_action"
8
+
9
+ BLOCKING_STATUSES_REQUIRING_NEXT_ACTION = {
10
+ "blocked",
11
+ "failed",
12
+ "error",
13
+ "needs_review",
14
+ }
15
+
16
+ NONBLOCKING_TERMINAL_STATUSES = {
17
+ "applied",
18
+ "completed",
19
+ "completed_with_warnings",
20
+ "diagnosis_ready",
21
+ "published",
22
+ "ready",
23
+ }
24
+
25
+
26
+ class OperationalErrorContext(ContractModel):
27
+ phase: StrictStr = Field(min_length=1)
28
+ blocked_reason: StrictStr = Field(min_length=1)
29
+ root_cause: StrictStr = Field(min_length=1)
30
+ affected_artifact: StrictStr = Field(min_length=1)
31
+ error_summary: StrictStr = Field(min_length=1)
32
+ suggested_fix: StrictStr = Field(min_length=1)
33
+ next_action: StrictStr = Field(min_length=1)
34
+ retry_scope: StrictStr = Field(min_length=1)
35
+ human_decision_required: StrictBool = False
36
+ affected_items: list[StrictStr] = Field(default_factory=list)
37
+ missing_inputs: list[StrictStr] = Field(default_factory=list)
38
+ max_attempts: int | None = None
39
+ details: JsonValue = None
40
+
41
+
42
+ class WorkflowGuardrailPayloadFields(ContractModel):
43
+ # This model is a typed decision view over larger workflow payloads. It must
44
+ # ignore unrelated keys so feedback, FSM results, and legacy evidence
45
+ # payloads can be inspected without becoming this helper's full schema.
46
+ model_config = ConfigDict(extra="ignore", populate_by_name=True, validate_assignment=True)
47
+
48
+ phase: StrictStr = ""
49
+ status: StrictStr = ""
50
+ blocked_reason: StrictStr = ""
51
+ next_action: StrictStr = ""
52
+ next_command: StrictStr = ""
53
+ required_inputs: list[StrictStr] = Field(default_factory=list)
54
+ human_decision_required: StrictBool = False
55
+ human_decision_packet: JsonObject | None = None
56
+ human_decision_packets: list[JsonObject] = Field(default_factory=list)
57
+ error_context: JsonObject | None = None
58
+ diagnostic_context: JsonObject = Field(default_factory=dict)
59
+ affected_artifact: StrictStr = ""
60
+ error: StrictStr = ""
61
+ message: StrictStr = ""
62
+ blocked: StrictBool | None = None
63
+ ok: StrictBool | None = None
64
+ parse_error: JsonValue = None
65
+ blocker_count: StrictInt | None = None
66
+ error_count: StrictInt | None = None
67
+
68
+
69
+ def _payload_value(payload: JsonObject, key: str) -> object:
70
+ if key not in payload:
71
+ return None
72
+ return payload[key]
73
+
74
+
75
+ def _optional_string(payload: JsonObject, key: str) -> object:
76
+ value = _payload_value(payload, key)
77
+ if value is None:
78
+ return ""
79
+ return value if isinstance(value, str) else value
80
+
81
+
82
+ def _optional_display_string(payload: JsonObject, key: str) -> str:
83
+ value = _payload_value(payload, key)
84
+ return value if isinstance(value, str) else ""
85
+
86
+
87
+ def _optional_bool(payload: JsonObject, key: str) -> object:
88
+ value = _payload_value(payload, key)
89
+ if value is None:
90
+ return None
91
+ return value if isinstance(value, bool) else value
92
+
93
+
94
+ def _optional_int(payload: JsonObject, key: str) -> object:
95
+ value = _payload_value(payload, key)
96
+ if value is None:
97
+ return None
98
+ return value if isinstance(value, int) and not isinstance(value, bool) else value
99
+
100
+
101
+ def _string_list(payload: JsonObject, key: str) -> list[str]:
102
+ value = _payload_value(payload, key)
103
+ if not isinstance(value, list):
104
+ return []
105
+ return [item for item in value if isinstance(item, str)]
106
+
107
+
108
+ def _json_object_or_none(payload: JsonObject, key: str) -> JsonObject | None:
109
+ value = _payload_value(payload, key)
110
+ return value if isinstance(value, dict) else None
111
+
112
+
113
+ def _json_object_list(payload: JsonObject, key: str) -> list[JsonObject]:
114
+ value = _payload_value(payload, key)
115
+ if not isinstance(value, list):
116
+ return []
117
+ return [item for item in value if isinstance(item, dict)]
118
+
119
+
120
+ def _optional_json_value(payload: JsonObject, key: str) -> JsonValue:
121
+ if key not in payload:
122
+ return None
123
+ return JsonObjectAdapter.validate_python({"value": payload[key]})["value"]
124
+
125
+
126
+ def workflow_guardrail_payload_fields(payload: JsonObject) -> WorkflowGuardrailPayloadFields:
127
+ """Extract only the fields that can drive generic recovery guardrails."""
128
+
129
+ raw_fields: JsonObject = {
130
+ "phase": _optional_display_string(payload, "phase"),
131
+ "status": _optional_string(payload, "status"),
132
+ "blocked_reason": _optional_string(payload, "blocked_reason"),
133
+ "next_action": _optional_string(payload, "next_action"),
134
+ "next_command": _optional_string(payload, "next_command"),
135
+ "required_inputs": _string_list(payload, "required_inputs"),
136
+ "human_decision_required": _optional_bool(payload, "human_decision_required") or False,
137
+ "human_decision_packet": _json_object_or_none(payload, "human_decision_packet"),
138
+ "human_decision_packets": _json_object_list(payload, "human_decision_packets"),
139
+ "error_context": _json_object_or_none(payload, "error_context"),
140
+ "diagnostic_context": _json_object_or_none(payload, "diagnostic_context") or {},
141
+ "affected_artifact": _optional_string(payload, "affected_artifact"),
142
+ "error": _optional_string(payload, "error"),
143
+ "message": _optional_string(payload, "message"),
144
+ "blocked": _optional_bool(payload, "blocked"),
145
+ "ok": _optional_bool(payload, "ok"),
146
+ "parse_error": _optional_json_value(payload, "parse_error"),
147
+ "blocker_count": _optional_int(payload, "blocker_count"),
148
+ "error_count": _optional_int(payload, "error_count"),
149
+ }
150
+ return WorkflowGuardrailPayloadFields.model_validate(raw_fields)
151
+
152
+
153
+ def validate_error_context(payload: JsonObject) -> OperationalErrorContext:
154
+ return OperationalErrorContext.model_validate(payload)
155
+
156
+
157
+ def default_contract_next_action(*, workflow: str = "", command: str = "") -> str:
158
+ """Public recovery text for blocked payloads missing an actionable route."""
159
+
160
+ route = workflow or command or "workflow oficial"
161
+ return (
162
+ f"Pare antes de mutar. Reexecute a rota oficial de {route} e reporte este "
163
+ "contract_gap ao mantenedor se o payload continuar sem next_action."
164
+ )
165
+
166
+
167
+ def blocked_payload_requires_next_action(payload: JsonObject) -> bool:
168
+ """Return whether an operational payload must expose a recovery route."""
169
+
170
+ fields = workflow_guardrail_payload_fields(payload)
171
+ if fields.status in BLOCKING_STATUSES_REQUIRING_NEXT_ACTION or fields.blocked_reason:
172
+ return True
173
+ if fields.status in NONBLOCKING_TERMINAL_STATUSES:
174
+ return False
175
+ if fields.blocked is True or fields.ok is False:
176
+ return True
177
+ if fields.error or fields.parse_error:
178
+ return True
179
+ return (fields.blocker_count or 0) > 0 or (fields.error_count or 0) > 0
180
+
181
+
182
+ def needs_next_action_hardening(payload: JsonObject) -> bool:
183
+ """Return whether a blocked payload needs contract-gap hardening."""
184
+
185
+ if not blocked_payload_requires_next_action(payload):
186
+ return False
187
+ fields = workflow_guardrail_payload_fields(payload)
188
+ return not (fields.next_action or fields.next_command).strip()
@@ -0,0 +1,319 @@
1
+ from __future__ import annotations
2
+
3
+ from enum import StrEnum
4
+
5
+ from pydantic import Field, field_validator, model_validator
6
+ from pydantic_core.core_schema import ValidationInfo
7
+
8
+ from mednotes.kernel.base import ContractModel, JsonObject
9
+
10
+
11
+ class WorkflowProgressStatus(StrEnum):
12
+ IDLE = "idle"
13
+ RUNNING = "running"
14
+ WAITING_AGENT = "waiting_agent"
15
+ WAITING_EXTERNAL = "waiting_external"
16
+ WAITING_HUMAN = "waiting_human"
17
+ BLOCKED = "blocked"
18
+ FAILED = "failed"
19
+ COMPLETED = "completed"
20
+ COMPLETED_WITH_WARNINGS = "completed_with_warnings"
21
+
22
+
23
+ class ProgressMode(StrEnum):
24
+ DETERMINATE = "determinate"
25
+ INDETERMINATE = "indeterminate"
26
+
27
+
28
+ class WorkflowProgressEventType(StrEnum):
29
+ WORKFLOW_STARTED = "workflow_started"
30
+ STATE_ENTERED = "state_entered"
31
+ STEP_STARTED = "step_started"
32
+ ITEM_PROCESSED = "item_processed"
33
+ CACHE_HIT = "cache_hit"
34
+ API_CALL_STARTED = "api_call_started"
35
+ API_CALL_THROTTLED = "api_call_throttled"
36
+ EXTERNAL_WAIT_STARTED = "external_wait_started"
37
+ RESOURCE_MUTATED = "resource_mutated"
38
+ DECISION_EMITTED = "decision_emitted"
39
+ VALIDATION_COMPLETED = "validation_completed"
40
+ WORKFLOW_COMPLETED = "workflow_completed"
41
+ WORKFLOW_FAILED = "workflow_failed"
42
+
43
+
44
+ class WorkflowProgressCounts(ContractModel):
45
+ planned_items: int = Field(default=0, ge=0)
46
+ processed_items: int = Field(default=0, ge=0)
47
+ cache_hits: int = Field(default=0, ge=0)
48
+ api_calls: int = Field(default=0, ge=0)
49
+ api_failures: int = Field(default=0, ge=0)
50
+ warnings: int = Field(default=0, ge=0)
51
+ mutated_files: int = Field(default=0, ge=0)
52
+ written_files: int = Field(default=0, ge=0)
53
+ remaining_items: int = Field(default=0, ge=0)
54
+ blocked_items: int = Field(default=0, ge=0)
55
+ deferred_items: int = Field(default=0, ge=0)
56
+
57
+ def plus(self, other: WorkflowProgressCounts) -> WorkflowProgressCounts:
58
+ return WorkflowProgressCounts(
59
+ planned_items=self.planned_items + other.planned_items,
60
+ processed_items=self.processed_items + other.processed_items,
61
+ cache_hits=self.cache_hits + other.cache_hits,
62
+ api_calls=self.api_calls + other.api_calls,
63
+ api_failures=self.api_failures + other.api_failures,
64
+ warnings=self.warnings + other.warnings,
65
+ mutated_files=self.mutated_files + other.mutated_files,
66
+ written_files=self.written_files + other.written_files,
67
+ remaining_items=self.remaining_items + other.remaining_items,
68
+ blocked_items=self.blocked_items + other.blocked_items,
69
+ deferred_items=self.deferred_items + other.deferred_items,
70
+ )
71
+
72
+
73
+ class WorkflowProgressEvent(ContractModel):
74
+ workflow: str = Field(min_length=1)
75
+ run_id: str = Field(min_length=1)
76
+ state: str = Field(min_length=1)
77
+ phase: str = Field(min_length=1)
78
+ event_type: WorkflowProgressEventType
79
+ message: str = ""
80
+ status: WorkflowProgressStatus = WorkflowProgressStatus.RUNNING
81
+ current: int = Field(default=0, ge=0)
82
+ total: int = Field(default=0, ge=0)
83
+ counts: WorkflowProgressCounts = Field(default_factory=WorkflowProgressCounts)
84
+ resume_action: str = ""
85
+ resume_supported: bool = False
86
+ can_continue_now: bool = True
87
+ user_action: str = ""
88
+ decision: JsonObject | None = None
89
+ technical_context: JsonObject = Field(default_factory=dict)
90
+
91
+ @field_validator("workflow", "run_id", "state", "phase")
92
+ @classmethod
93
+ def _required_text(cls, value: str, info: ValidationInfo) -> str:
94
+ cleaned = value.strip()
95
+ if not cleaned:
96
+ raise ValueError(f"{info.field_name} must be non-empty")
97
+ return cleaned
98
+
99
+
100
+ class WorkflowProgressState(ContractModel):
101
+ workflow: str = Field(min_length=1)
102
+ run_id: str = Field(min_length=1)
103
+ state: str = Field(min_length=1)
104
+ phase: str = Field(min_length=1)
105
+ event_type: WorkflowProgressEventType
106
+ message: str = ""
107
+ status: WorkflowProgressStatus = WorkflowProgressStatus.RUNNING
108
+ current: int = Field(default=0, ge=0)
109
+ total: int = Field(default=0, ge=0)
110
+ counts: WorkflowProgressCounts = Field(default_factory=WorkflowProgressCounts)
111
+ resume_action: str = ""
112
+ resume_supported: bool = False
113
+ can_continue_now: bool = True
114
+ user_action: str = ""
115
+ decision: JsonObject | None = None
116
+ technical_context: JsonObject = Field(default_factory=dict)
117
+
118
+ @field_validator("workflow", "run_id", "state", "phase")
119
+ @classmethod
120
+ def _required_text(cls, value: str, info: ValidationInfo) -> str:
121
+ cleaned = value.strip()
122
+ if not cleaned:
123
+ raise ValueError(f"{info.field_name} must be non-empty")
124
+ return cleaned
125
+
126
+
127
+ class WorkflowProgressViewModel(ContractModel):
128
+ workflow: str = Field(min_length=1)
129
+ run_id: str = Field(min_length=1)
130
+ state: str = Field(min_length=1)
131
+ phase: str = Field(min_length=1)
132
+ status: WorkflowProgressStatus
133
+ mode: ProgressMode
134
+ percent: int = Field(ge=0, le=100)
135
+ terminal: bool
136
+ successful: bool
137
+ current: int = Field(default=0, ge=0)
138
+ total: int = Field(default=0, ge=0)
139
+ count_label: str = ""
140
+ message: str = ""
141
+ user_action: str = ""
142
+ resume_action: str = ""
143
+ resume_supported: bool = False
144
+ can_continue_now: bool = True
145
+ counts: WorkflowProgressCounts = Field(default_factory=WorkflowProgressCounts)
146
+ decision: JsonObject | None = None
147
+ technical_context: JsonObject = Field(default_factory=dict)
148
+
149
+ @field_validator("workflow", "run_id", "state", "phase")
150
+ @classmethod
151
+ def _required_text(cls, value: str, info: ValidationInfo) -> str:
152
+ cleaned = value.strip()
153
+ if not cleaned:
154
+ raise ValueError(f"{info.field_name} must be non-empty")
155
+ return cleaned
156
+
157
+ @model_validator(mode="after")
158
+ def _successful_requires_completed_status(self) -> WorkflowProgressViewModel:
159
+ if self.successful and self.status not in _SUCCESS_STATUSES:
160
+ raise ValueError("successful requires completed status")
161
+ if self.status == WorkflowProgressStatus.WAITING_EXTERNAL:
162
+ object.__setattr__(self, "can_continue_now", False)
163
+ return self
164
+
165
+
166
+ _TERMINAL_STATUSES = {
167
+ WorkflowProgressStatus.BLOCKED,
168
+ WorkflowProgressStatus.FAILED,
169
+ WorkflowProgressStatus.COMPLETED,
170
+ WorkflowProgressStatus.COMPLETED_WITH_WARNINGS,
171
+ }
172
+ _SUCCESS_STATUSES = {
173
+ WorkflowProgressStatus.COMPLETED,
174
+ WorkflowProgressStatus.COMPLETED_WITH_WARNINGS,
175
+ }
176
+
177
+
178
+ def fold_progress_events(events: list[WorkflowProgressEvent]) -> WorkflowProgressState:
179
+ if not events:
180
+ raise ValueError("progress events must be non-empty")
181
+
182
+ expected_workflow = events[0].workflow
183
+ expected_run_id = events[0].run_id
184
+ counts = WorkflowProgressCounts()
185
+ current = 0
186
+ total = 0
187
+ for event in events:
188
+ if event.workflow != expected_workflow:
189
+ raise ValueError("progress events must share the same workflow")
190
+ if event.run_id != expected_run_id:
191
+ raise ValueError("progress events must share the same run_id")
192
+ counts = counts.plus(event.counts)
193
+ current = max(current, event.current)
194
+ total = max(total, event.total)
195
+
196
+ last = events[-1]
197
+ return WorkflowProgressState(
198
+ workflow=last.workflow,
199
+ run_id=last.run_id,
200
+ state=last.state,
201
+ phase=last.phase,
202
+ event_type=last.event_type,
203
+ message=last.message,
204
+ status=last.status,
205
+ current=current,
206
+ total=total,
207
+ counts=counts,
208
+ resume_action=last.resume_action,
209
+ resume_supported=last.resume_supported,
210
+ can_continue_now=last.can_continue_now,
211
+ user_action=last.user_action,
212
+ decision=last.decision,
213
+ technical_context=last.technical_context,
214
+ )
215
+
216
+
217
+ def build_progress_view_model(state: WorkflowProgressState) -> WorkflowProgressViewModel:
218
+ mode = ProgressMode.DETERMINATE if state.total > 0 else ProgressMode.INDETERMINATE
219
+ successful = state.status in _SUCCESS_STATUSES
220
+ terminal = state.status in _TERMINAL_STATUSES
221
+ percent = _progress_percent(state.current, state.total, successful=successful)
222
+ count_label = f"{state.current} de {state.total}" if state.total > 0 else ""
223
+ user_action = _user_action_for(state)
224
+
225
+ return WorkflowProgressViewModel(
226
+ workflow=state.workflow,
227
+ run_id=state.run_id,
228
+ state=state.state,
229
+ phase=state.phase,
230
+ status=state.status,
231
+ mode=mode,
232
+ percent=percent,
233
+ terminal=terminal,
234
+ successful=successful,
235
+ current=state.current,
236
+ total=state.total,
237
+ count_label=count_label,
238
+ message=state.message,
239
+ user_action=user_action,
240
+ resume_action=state.resume_action,
241
+ resume_supported=state.resume_supported,
242
+ can_continue_now=state.can_continue_now,
243
+ counts=state.counts,
244
+ decision=state.decision,
245
+ technical_context=state.technical_context,
246
+ )
247
+
248
+
249
+ def progress_state_from_view_model(view_model: WorkflowProgressViewModel) -> WorkflowProgressState:
250
+ """Rehydrate the internal progress state from the public view model.
251
+
252
+ FSM result models hide ``progress_state`` from public schemas/payloads. When
253
+ a public payload is revalidated, this helper rebuilds the private state used
254
+ by internal drift validators without making it a second public source of
255
+ truth.
256
+ """
257
+
258
+ return WorkflowProgressState(
259
+ workflow=view_model.workflow,
260
+ run_id=view_model.run_id,
261
+ state=view_model.state,
262
+ phase=view_model.phase,
263
+ event_type=_event_type_for_view_status(view_model.status),
264
+ message=view_model.message,
265
+ status=view_model.status,
266
+ current=view_model.current,
267
+ total=view_model.total,
268
+ counts=view_model.counts,
269
+ resume_action=view_model.resume_action,
270
+ resume_supported=view_model.resume_supported,
271
+ can_continue_now=view_model.can_continue_now,
272
+ user_action=view_model.user_action,
273
+ decision=view_model.decision,
274
+ technical_context=view_model.technical_context,
275
+ )
276
+
277
+
278
+ def _event_type_for_view_status(status: WorkflowProgressStatus) -> WorkflowProgressEventType:
279
+ """Choose a coherent private event type for revalidated public payloads."""
280
+
281
+ match status:
282
+ case WorkflowProgressStatus.WAITING_EXTERNAL:
283
+ return WorkflowProgressEventType.EXTERNAL_WAIT_STARTED
284
+ case WorkflowProgressStatus.WAITING_HUMAN:
285
+ return WorkflowProgressEventType.DECISION_EMITTED
286
+ case WorkflowProgressStatus.FAILED:
287
+ return WorkflowProgressEventType.WORKFLOW_FAILED
288
+ case WorkflowProgressStatus.COMPLETED | WorkflowProgressStatus.COMPLETED_WITH_WARNINGS:
289
+ return WorkflowProgressEventType.WORKFLOW_COMPLETED
290
+ case WorkflowProgressStatus.BLOCKED:
291
+ return WorkflowProgressEventType.VALIDATION_COMPLETED
292
+ case WorkflowProgressStatus.RUNNING:
293
+ return WorkflowProgressEventType.STATE_ENTERED
294
+ case WorkflowProgressStatus.WAITING_AGENT:
295
+ return WorkflowProgressEventType.STEP_STARTED
296
+ case WorkflowProgressStatus.IDLE:
297
+ return WorkflowProgressEventType.WORKFLOW_STARTED
298
+
299
+
300
+ def _progress_percent(current: int, total: int, *, successful: bool) -> int:
301
+ if successful:
302
+ return 100
303
+ if total <= 0:
304
+ return 0
305
+ return min(99, max(0, int((max(0, current) / total) * 100)))
306
+
307
+
308
+ def _user_action_for(state: WorkflowProgressState) -> str:
309
+ if state.user_action.strip():
310
+ return state.user_action.strip()
311
+ if state.status == WorkflowProgressStatus.WAITING_AGENT:
312
+ if state.resume_action:
313
+ return state.resume_action
314
+ return "Continue pela etapa assistida indicada pelo workflow antes de concluir."
315
+ if state.status == WorkflowProgressStatus.WAITING_EXTERNAL:
316
+ if state.resume_action:
317
+ return "Aguarde a condicao externa e retome pela acao oficial quando ela estiver disponivel."
318
+ return "Aguarde a condicao externa antes de retomar pela rota oficial."
319
+ return ""