@danmoisan/drm-copilot-mcp 0.0.1

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 (380) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +50 -0
  3. package/out/mcp-server.js +17323 -0
  4. package/package.json +36 -0
  5. package/resources/claude-customizations/.claude/agent-memory/orchestrator/MEMORY.md +3 -0
  6. package/resources/claude-customizations/.claude/agent-memory/orchestrator/feedback_repo_root_is_source_of_truth.md +11 -0
  7. package/resources/claude-customizations/.claude/agent-memory/orchestrator/feedback_vsce_verify_package_location.md +19 -0
  8. package/resources/claude-customizations/.claude/agent-memory/orchestrator/project_extension_location.md +11 -0
  9. package/resources/claude-customizations/.claude/agent-memory/prd-feature/MEMORY.md +1 -0
  10. package/resources/claude-customizations/.claude/agent-memory/prd-feature/project_push_down_pattern.md +13 -0
  11. package/resources/claude-customizations/.claude/agent-memory/task-researcher/MEMORY.md +3 -0
  12. package/resources/claude-customizations/.claude/agent-memory/task-researcher/project_push_down_claude_dir.md +11 -0
  13. package/resources/claude-customizations/.claude/agents/atomic-executor.md +135 -0
  14. package/resources/claude-customizations/.claude/agents/atomic-planner.md +71 -0
  15. package/resources/claude-customizations/.claude/agents/csharp-typed-engineer.md +69 -0
  16. package/resources/claude-customizations/.claude/agents/epic-review.md +40 -0
  17. package/resources/claude-customizations/.claude/agents/feature-review.md +136 -0
  18. package/resources/claude-customizations/.claude/agents/orchestrator.md +83 -0
  19. package/resources/claude-customizations/.claude/agents/powershell-typed-engineer.md +80 -0
  20. package/resources/claude-customizations/.claude/agents/prd-feature.md +42 -0
  21. package/resources/claude-customizations/.claude/agents/python-typed-engineer.md +72 -0
  22. package/resources/claude-customizations/.claude/agents/staged-review.md +41 -0
  23. package/resources/claude-customizations/.claude/agents/status-updater.md +41 -0
  24. package/resources/claude-customizations/.claude/agents/task-researcher.md +81 -0
  25. package/resources/claude-customizations/.claude/agents/typescript-engineer.md +24 -0
  26. package/resources/claude-customizations/.claude/hooks/check-powershell-test-purity.ps1 +111 -0
  27. package/resources/claude-customizations/.claude/hooks/check-python-test-purity.ps1 +146 -0
  28. package/resources/claude-customizations/.claude/hooks/enforce-evidence-locations.ps1 +150 -0
  29. package/resources/claude-customizations/.claude/hooks/enforce-powershell-batch-budget.ps1 +238 -0
  30. package/resources/claude-customizations/.claude/hooks/enforce-promotion-mcp-only.ps1 +147 -0
  31. package/resources/claude-customizations/.claude/hooks/enforce-python-batch-budget.ps1 +235 -0
  32. package/resources/claude-customizations/.claude/hooks/validate-bash.ps1 +69 -0
  33. package/resources/claude-customizations/.claude/hooks/validate-executor-output.ps1 +296 -0
  34. package/resources/claude-customizations/.claude/hooks/validate-feature-review-coverage.ps1 +389 -0
  35. package/resources/claude-customizations/.claude/hooks/validate-orchestrator-output.ps1 +141 -0
  36. package/resources/claude-customizations/.claude/hooks/validate-planner-output.ps1 +288 -0
  37. package/resources/claude-customizations/.claude/hooks/validate-required-artifact-output.ps1 +171 -0
  38. package/resources/claude-customizations/.claude/hooks/validate-task-researcher-output.ps1 +142 -0
  39. package/resources/claude-customizations/.claude/rules/csharp.md +62 -0
  40. package/resources/claude-customizations/.claude/rules/general-code-change.md +71 -0
  41. package/resources/claude-customizations/.claude/rules/general-unit-test.md +60 -0
  42. package/resources/claude-customizations/.claude/rules/powershell.md +97 -0
  43. package/resources/claude-customizations/.claude/rules/python-suppressions.md +143 -0
  44. package/resources/claude-customizations/.claude/rules/python.md +99 -0
  45. package/resources/claude-customizations/.claude/rules/self-explanatory-code-commenting.md +97 -0
  46. package/resources/claude-customizations/.claude/rules/tonality.md +80 -0
  47. package/resources/claude-customizations/.claude/rules/typescript-suppressions.md +66 -0
  48. package/resources/claude-customizations/.claude/rules/typescript.md +45 -0
  49. package/resources/claude-customizations/.claude/settings.json +144 -0
  50. package/resources/claude-customizations/.claude/skills/acceptance-criteria-tracking/SKILL.md +102 -0
  51. package/resources/claude-customizations/.claude/skills/atomic-plan-contract/SKILL.md +189 -0
  52. package/resources/claude-customizations/.claude/skills/commit-message/SKILL.md +65 -0
  53. package/resources/claude-customizations/.claude/skills/csharp-change-budget-router/SKILL.md +90 -0
  54. package/resources/claude-customizations/.claude/skills/csharp-orchestration-state-machine/SKILL.md +58 -0
  55. package/resources/claude-customizations/.claude/skills/csharp-qa-gate/SKILL.md +77 -0
  56. package/resources/claude-customizations/.claude/skills/evidence-and-timestamp-conventions/SKILL.md +164 -0
  57. package/resources/claude-customizations/.claude/skills/execute-hard-lock/SKILL.md +82 -0
  58. package/resources/claude-customizations/.claude/skills/feature-promotion-lifecycle/SKILL.md +115 -0
  59. package/resources/claude-customizations/.claude/skills/feature-review-workflow/SKILL.md +167 -0
  60. package/resources/claude-customizations/.claude/skills/fill-feature-docs/SKILL.md +22 -0
  61. package/resources/claude-customizations/.claude/skills/invoke-csharp-engineer/SKILL.md +64 -0
  62. package/resources/claude-customizations/.claude/skills/invoke-powershell-engineer/SKILL.md +65 -0
  63. package/resources/claude-customizations/.claude/skills/invoke-python-engineer/SKILL.md +64 -0
  64. package/resources/claude-customizations/.claude/skills/make-skill-template/SKILL.md +147 -0
  65. package/resources/claude-customizations/.claude/skills/orchestrate/SKILL.md +132 -0
  66. package/resources/claude-customizations/.claude/skills/policy-audit-template-usage/SKILL.md +49 -0
  67. package/resources/claude-customizations/.claude/skills/policy-compliance-order/SKILL.md +40 -0
  68. package/resources/claude-customizations/.claude/skills/powershell-change-budget-router/SKILL.md +49 -0
  69. package/resources/claude-customizations/.claude/skills/powershell-orchestration-state-machine/SKILL.md +58 -0
  70. package/resources/claude-customizations/.claude/skills/powershell-qa-gate/SKILL.md +77 -0
  71. package/resources/claude-customizations/.claude/skills/pr-author/SKILL.md +50 -0
  72. package/resources/claude-customizations/.claude/skills/pr-base-branch-merge-base/SKILL.md +56 -0
  73. package/resources/claude-customizations/.claude/skills/pr-context-artifacts/SKILL.md +30 -0
  74. package/resources/claude-customizations/.claude/skills/python-change-budget-router/SKILL.md +79 -0
  75. package/resources/claude-customizations/.claude/skills/python-qa-gate/SKILL.md +77 -0
  76. package/resources/claude-customizations/.claude/skills/remediation-handoff-atomic-planner/SKILL.md +40 -0
  77. package/resources/claude-customizations/.claude/skills/research-issue/SKILL.md +67 -0
  78. package/resources/claude-customizations/.claude/skills/review-epic/SKILL.md +21 -0
  79. package/resources/claude-customizations/.claude/skills/review-feature/SKILL.md +25 -0
  80. package/resources/claude-customizations/.claude/skills/review-staged/SKILL.md +21 -0
  81. package/resources/claude-customizations/.claude/skills/skill-canonical-location-audit/SKILL.md +49 -0
  82. package/resources/claude-customizations/.claude/skills/translate-copilot-to-claude/SKILL.md +295 -0
  83. package/resources/claude-customizations/.claude/skills/update-status/SKILL.md +21 -0
  84. package/resources/claude-dir-customizations/.mcp.json +8 -0
  85. package/resources/codex-and-agents-customizations/.agents/README.md +86 -0
  86. package/resources/codex-and-agents-customizations/.agents/skills/README.md +49 -0
  87. package/resources/codex-and-agents-customizations/.agents/skills/acceptance-criteria-tracking/SKILL.md +107 -0
  88. package/resources/codex-and-agents-customizations/.agents/skills/atomic-executor/SKILL.md +73 -0
  89. package/resources/codex-and-agents-customizations/.agents/skills/atomic-plan-contract/SKILL.md +194 -0
  90. package/resources/codex-and-agents-customizations/.agents/skills/atomic-planner/SKILL.md +87 -0
  91. package/resources/codex-and-agents-customizations/.agents/skills/commit-message/SKILL.md +70 -0
  92. package/resources/codex-and-agents-customizations/.agents/skills/commit-message-conventions/SKILL.md +95 -0
  93. package/resources/codex-and-agents-customizations/.agents/skills/csharp/SKILL.md +67 -0
  94. package/resources/codex-and-agents-customizations/.agents/skills/csharp-change-budget-router/SKILL.md +94 -0
  95. package/resources/codex-and-agents-customizations/.agents/skills/csharp-orchestration-state-machine/SKILL.md +64 -0
  96. package/resources/codex-and-agents-customizations/.agents/skills/csharp-qa-gate/SKILL.md +82 -0
  97. package/resources/codex-and-agents-customizations/.agents/skills/evidence-and-timestamp-conventions/SKILL.md +168 -0
  98. package/resources/codex-and-agents-customizations/.agents/skills/execute-hard-lock/SKILL.md +88 -0
  99. package/resources/codex-and-agents-customizations/.agents/skills/feature-promotion-lifecycle/SKILL.md +129 -0
  100. package/resources/codex-and-agents-customizations/.agents/skills/feature-review/SKILL.md +106 -0
  101. package/resources/codex-and-agents-customizations/.agents/skills/feature-review-workflow/SKILL.md +181 -0
  102. package/resources/codex-and-agents-customizations/.agents/skills/fill-feature-docs/SKILL.md +27 -0
  103. package/resources/codex-and-agents-customizations/.agents/skills/invoke-csharp-engineer/SKILL.md +73 -0
  104. package/resources/codex-and-agents-customizations/.agents/skills/invoke-powershell-engineer/SKILL.md +74 -0
  105. package/resources/codex-and-agents-customizations/.agents/skills/invoke-python-engineer/SKILL.md +73 -0
  106. package/resources/codex-and-agents-customizations/.agents/skills/make-skill-template/SKILL.md +152 -0
  107. package/resources/codex-and-agents-customizations/.agents/skills/orchestrate/SKILL.md +143 -0
  108. package/resources/codex-and-agents-customizations/.agents/skills/orchestrator-workflow/SKILL.md +317 -0
  109. package/resources/codex-and-agents-customizations/.agents/skills/policy-audit-template-usage/SKILL.md +53 -0
  110. package/resources/codex-and-agents-customizations/.agents/skills/policy-compliance-order/SKILL.md +49 -0
  111. package/resources/codex-and-agents-customizations/.agents/skills/powershell/SKILL.md +102 -0
  112. package/resources/codex-and-agents-customizations/.agents/skills/powershell-change-budget-router/SKILL.md +53 -0
  113. package/resources/codex-and-agents-customizations/.agents/skills/powershell-orchestration-state-machine/SKILL.md +64 -0
  114. package/resources/codex-and-agents-customizations/.agents/skills/powershell-qa-gate/SKILL.md +83 -0
  115. package/resources/codex-and-agents-customizations/.agents/skills/pr-author/SKILL.md +55 -0
  116. package/resources/codex-and-agents-customizations/.agents/skills/pr-authoring/SKILL.md +124 -0
  117. package/resources/codex-and-agents-customizations/.agents/skills/pr-base-branch-merge-base/SKILL.md +60 -0
  118. package/resources/codex-and-agents-customizations/.agents/skills/pr-context-artifacts/SKILL.md +34 -0
  119. package/resources/codex-and-agents-customizations/.agents/skills/python/SKILL.md +104 -0
  120. package/resources/codex-and-agents-customizations/.agents/skills/python-change-budget-router/SKILL.md +84 -0
  121. package/resources/codex-and-agents-customizations/.agents/skills/python-qa-gate/SKILL.md +82 -0
  122. package/resources/codex-and-agents-customizations/.agents/skills/python-suppressions/SKILL.md +148 -0
  123. package/resources/codex-and-agents-customizations/.agents/skills/remediation-handoff-atomic-planner/SKILL.md +49 -0
  124. package/resources/codex-and-agents-customizations/.agents/skills/repo-automation-adapter/SKILL.md +142 -0
  125. package/resources/codex-and-agents-customizations/.agents/skills/repo-automation-adapter/agents/openai.yaml +5 -0
  126. package/resources/codex-and-agents-customizations/.agents/skills/research-issue/SKILL.md +72 -0
  127. package/resources/codex-and-agents-customizations/.agents/skills/review-epic/SKILL.md +26 -0
  128. package/resources/codex-and-agents-customizations/.agents/skills/review-feature/SKILL.md +30 -0
  129. package/resources/codex-and-agents-customizations/.agents/skills/review-staged/SKILL.md +26 -0
  130. package/resources/codex-and-agents-customizations/.agents/skills/self-explanatory-code-commenting/SKILL.md +102 -0
  131. package/resources/codex-and-agents-customizations/.agents/skills/skill-canonical-location-audit/SKILL.md +52 -0
  132. package/resources/codex-and-agents-customizations/.agents/skills/translate-copilot-to-claude/SKILL.md +317 -0
  133. package/resources/codex-and-agents-customizations/.agents/skills/typescript/SKILL.md +50 -0
  134. package/resources/codex-and-agents-customizations/.agents/skills/typescript-suppressions/SKILL.md +71 -0
  135. package/resources/codex-and-agents-customizations/.agents/skills/update-status/SKILL.md +26 -0
  136. package/resources/codex-and-agents-customizations/.codex/agents/5.1-beast-adjusted.toml +23 -0
  137. package/resources/codex-and-agents-customizations/.codex/agents/5.1-thinking-beast-mode-adjusted.toml +23 -0
  138. package/resources/codex-and-agents-customizations/.codex/agents/api-architect.toml +23 -0
  139. package/resources/codex-and-agents-customizations/.codex/agents/atomic-executor.toml +151 -0
  140. package/resources/codex-and-agents-customizations/.codex/agents/atomic-planner.toml +93 -0
  141. package/resources/codex-and-agents-customizations/.codex/agents/atomic-planning.toml +24 -0
  142. package/resources/codex-and-agents-customizations/.codex/agents/commentary-remediation.toml +23 -0
  143. package/resources/codex-and-agents-customizations/.codex/agents/commit-steward.toml +20 -0
  144. package/resources/codex-and-agents-customizations/.codex/agents/csharp-atomic-executor.toml +24 -0
  145. package/resources/codex-and-agents-customizations/.codex/agents/csharp-atomic-planning.toml +25 -0
  146. package/resources/codex-and-agents-customizations/.codex/agents/csharp-orchestrator.toml +56 -0
  147. package/resources/codex-and-agents-customizations/.codex/agents/csharp-typed-engineer.toml +97 -0
  148. package/resources/codex-and-agents-customizations/.codex/agents/epic-review.toml +52 -0
  149. package/resources/codex-and-agents-customizations/.codex/agents/expert-nextjs-developer.toml +23 -0
  150. package/resources/codex-and-agents-customizations/.codex/agents/expert-react-frontend-engineer.toml +23 -0
  151. package/resources/codex-and-agents-customizations/.codex/agents/feature-review.toml +149 -0
  152. package/resources/codex-and-agents-customizations/.codex/agents/feature-reviewer.toml +60 -0
  153. package/resources/codex-and-agents-customizations/.codex/agents/gpt-5-beast-mode.toml +23 -0
  154. package/resources/codex-and-agents-customizations/.codex/agents/hlbpa.toml +23 -0
  155. package/resources/codex-and-agents-customizations/.codex/agents/mentor.toml +23 -0
  156. package/resources/codex-and-agents-customizations/.codex/agents/orchestrator.toml +121 -0
  157. package/resources/codex-and-agents-customizations/.codex/agents/powershell-atomic-executor.toml +24 -0
  158. package/resources/codex-and-agents-customizations/.codex/agents/powershell-atomic-planning.toml +25 -0
  159. package/resources/codex-and-agents-customizations/.codex/agents/powershell-di-unit-test-engineer.toml +24 -0
  160. package/resources/codex-and-agents-customizations/.codex/agents/powershell-orchestrator.toml +56 -0
  161. package/resources/codex-and-agents-customizations/.codex/agents/powershell-typed-engineer.toml +108 -0
  162. package/resources/codex-and-agents-customizations/.codex/agents/pr-author.toml +26 -0
  163. package/resources/codex-and-agents-customizations/.codex/agents/prd-feature.toml +53 -0
  164. package/resources/codex-and-agents-customizations/.codex/agents/prd.toml +23 -0
  165. package/resources/codex-and-agents-customizations/.codex/agents/pytest-unit-test-coding.toml +24 -0
  166. package/resources/codex-and-agents-customizations/.codex/agents/python-atomic-executor.toml +24 -0
  167. package/resources/codex-and-agents-customizations/.codex/agents/python-atomic-planning.toml +25 -0
  168. package/resources/codex-and-agents-customizations/.codex/agents/python-execution-only-typed.toml +24 -0
  169. package/resources/codex-and-agents-customizations/.codex/agents/python-orchestrator.toml +54 -0
  170. package/resources/codex-and-agents-customizations/.codex/agents/python-typed-engineer.toml +100 -0
  171. package/resources/codex-and-agents-customizations/.codex/agents/staged-review.toml +53 -0
  172. package/resources/codex-and-agents-customizations/.codex/agents/status-updater.toml +53 -0
  173. package/resources/codex-and-agents-customizations/.codex/agents/task-researcher.toml +103 -0
  174. package/resources/codex-and-agents-customizations/.codex/agents/tdd-green.toml +23 -0
  175. package/resources/codex-and-agents-customizations/.codex/agents/tdd-red.toml +23 -0
  176. package/resources/codex-and-agents-customizations/.codex/agents/tdd-refactor.toml +23 -0
  177. package/resources/codex-and-agents-customizations/.codex/agents/typescript-engineer.toml +48 -0
  178. package/resources/codex-and-agents-customizations/.codex/agents/voidbeast-gpt41enhanced.toml +23 -0
  179. package/resources/codex-and-agents-customizations/.codex/codex-web-setup.plan.md +26 -0
  180. package/resources/codex-and-agents-customizations/.codex/codex-web-setup.sh +384 -0
  181. package/resources/codex-and-agents-customizations/.codex/config.toml +137 -0
  182. package/resources/codex-and-agents-customizations/.codex/hooks/check-powershell-test-purity.ps1 +113 -0
  183. package/resources/codex-and-agents-customizations/.codex/hooks/check-python-test-purity.ps1 +149 -0
  184. package/resources/codex-and-agents-customizations/.codex/hooks/enforce-evidence-locations.ps1 +153 -0
  185. package/resources/codex-and-agents-customizations/.codex/hooks/enforce-powershell-batch-budget.ps1 +241 -0
  186. package/resources/codex-and-agents-customizations/.codex/hooks/enforce-promotion-mcp-only.ps1 +150 -0
  187. package/resources/codex-and-agents-customizations/.codex/hooks/enforce-python-batch-budget.ps1 +238 -0
  188. package/resources/codex-and-agents-customizations/.codex/hooks/validate-bash.ps1 +72 -0
  189. package/resources/codex-and-agents-customizations/.codex/hooks/validate-feature-review-coverage.ps1 +265 -0
  190. package/resources/codex-and-agents-customizations/.codex/prompts/feature-review-remediate.md +10 -0
  191. package/resources/codex-and-agents-customizations/.codex/prompts/generate-commit-message-repo.md +11 -0
  192. package/resources/codex-and-agents-customizations/.codex/prompts/generate-pr.md +15 -0
  193. package/resources/codex-and-agents-customizations/.codex/prompts/orchestrate-work.md +22 -0
  194. package/resources/codex-and-agents-customizations/AGENTS.md +317 -0
  195. package/resources/customizations/.github/agents/5.1-Beast-adjusted.agent.md +181 -0
  196. package/resources/customizations/.github/agents/5.1-Thinking-Beast-Mode-adjusted.agent.md +361 -0
  197. package/resources/customizations/.github/agents/Powershell DI Unit Test Engineer.agent.md +192 -0
  198. package/resources/customizations/.github/agents/api-architect.agent.md +40 -0
  199. package/resources/customizations/.github/agents/atomic_executor.agent.md +251 -0
  200. package/resources/customizations/.github/agents/atomic_planning.agent.md +658 -0
  201. package/resources/customizations/.github/agents/commentary-remediation.agent.md +35 -0
  202. package/resources/customizations/.github/agents/commit-steward.agent.md +200 -0
  203. package/resources/customizations/.github/agents/csharp-atomic-executor.agent.md +288 -0
  204. package/resources/customizations/.github/agents/csharp-atomic-planning.agent.md +354 -0
  205. package/resources/customizations/.github/agents/csharp-orchestrator.agent.md +375 -0
  206. package/resources/customizations/.github/agents/csharp-typed-engineer.agent.md +285 -0
  207. package/resources/customizations/.github/agents/epic-review.agent.md +374 -0
  208. package/resources/customizations/.github/agents/expert-nextjs-developer.agent.md +477 -0
  209. package/resources/customizations/.github/agents/expert-react-frontend-engineer.agent.md +739 -0
  210. package/resources/customizations/.github/agents/feature-review.agent.md +49 -0
  211. package/resources/customizations/.github/agents/gpt-5-beast-mode.agent.md +116 -0
  212. package/resources/customizations/.github/agents/hlbpa.agent.md +219 -0
  213. package/resources/customizations/.github/agents/mentor.agent.md +32 -0
  214. package/resources/customizations/.github/agents/orchestrator.agent.md +449 -0
  215. package/resources/customizations/.github/agents/powershell-atomic-executor.agent.md +287 -0
  216. package/resources/customizations/.github/agents/powershell-atomic-planning.agent.md +647 -0
  217. package/resources/customizations/.github/agents/powershell-orchestrator.agent.md +382 -0
  218. package/resources/customizations/.github/agents/powershell-typed-engineer.agent.md +293 -0
  219. package/resources/customizations/.github/agents/pr-author.agent.md +138 -0
  220. package/resources/customizations/.github/agents/prd-feature.agent.md +52 -0
  221. package/resources/customizations/.github/agents/prd.agent.md +202 -0
  222. package/resources/customizations/.github/agents/pytest-unit-test-coding.agent.md +202 -0
  223. package/resources/customizations/.github/agents/python-atomic-executor.agent.md +289 -0
  224. package/resources/customizations/.github/agents/python-atomic-planning.agent.md +429 -0
  225. package/resources/customizations/.github/agents/python-execution-only-typed.agent.md +217 -0
  226. package/resources/customizations/.github/agents/python-orchestrator.agent.md +380 -0
  227. package/resources/customizations/.github/agents/python-typed-engineer.agent.md +271 -0
  228. package/resources/customizations/.github/agents/staged-review.agent.md +246 -0
  229. package/resources/customizations/.github/agents/status_updater.agent.md +279 -0
  230. package/resources/customizations/.github/agents/task-researcher.agent.md +298 -0
  231. package/resources/customizations/.github/agents/tdd-green.agent.md +60 -0
  232. package/resources/customizations/.github/agents/tdd-red.agent.md +66 -0
  233. package/resources/customizations/.github/agents/tdd-refactor.agent.md +94 -0
  234. package/resources/customizations/.github/agents/typescript-engineer.agent.md +167 -0
  235. package/resources/customizations/.github/agents/voidbeast-gpt41enhanced.agent.md +230 -0
  236. package/resources/customizations/.github/codex/execute-hard-lock.prompt.md +105 -0
  237. package/resources/customizations/.github/codex/resume-hard-lock.prompt.md +92 -0
  238. package/resources/customizations/.github/copilot-instructions.md +7 -0
  239. package/resources/customizations/.github/instructions/csharp-code-change.instructions.md +184 -0
  240. package/resources/customizations/.github/instructions/csharp-unit-test.instructions.md +52 -0
  241. package/resources/customizations/.github/instructions/general-code-change.instructions.md +290 -0
  242. package/resources/customizations/.github/instructions/general-unit-test.instructions.md +106 -0
  243. package/resources/customizations/.github/instructions/github-actions-ci-cd-best-practices.instructions.md +607 -0
  244. package/resources/customizations/.github/instructions/github-actions.instructions.md +23 -0
  245. package/resources/customizations/.github/instructions/powershell-code-change.instructions.md +81 -0
  246. package/resources/customizations/.github/instructions/powershell-unit-test.instructions.md +69 -0
  247. package/resources/customizations/.github/instructions/python-code-change.instructions.md +232 -0
  248. package/resources/customizations/.github/instructions/python-suppressions.instructions.md +609 -0
  249. package/resources/customizations/.github/instructions/python-unit-test.instructions.md +71 -0
  250. package/resources/customizations/.github/instructions/self-explanatory-code-commenting.instructions.md +238 -0
  251. package/resources/customizations/.github/instructions/tonality.instructions.md +133 -0
  252. package/resources/customizations/.github/instructions/typescript-code-change.instructions.md +203 -0
  253. package/resources/customizations/.github/instructions/typescript-suppressions.instructions.md +157 -0
  254. package/resources/customizations/.github/instructions/typescript-unit-test.instructions.md +112 -0
  255. package/resources/customizations/.github/prompts/add-educational-comments.prompt.md +129 -0
  256. package/resources/customizations/.github/prompts/breakdown-bug-prd.prompt.md +29 -0
  257. package/resources/customizations/.github/prompts/breakdown-epic-arch.prompt.md +66 -0
  258. package/resources/customizations/.github/prompts/breakdown-epic-pm.prompt.md +58 -0
  259. package/resources/customizations/.github/prompts/breakdown-feature-implementation.prompt.md +128 -0
  260. package/resources/customizations/.github/prompts/breakdown-feature-prd.prompt.md +61 -0
  261. package/resources/customizations/.github/prompts/code-exemplars-blueprint-generator.prompt.md +126 -0
  262. package/resources/customizations/.github/prompts/drafts/create-github-issues-feature-from-implementation-plan.prompt.md +28 -0
  263. package/resources/customizations/.github/prompts/drafts/create-implementation-plan.prompt.md +158 -0
  264. package/resources/customizations/.github/prompts/drafts/create-technical-spike.prompt.md +231 -0
  265. package/resources/customizations/.github/prompts/drafts/potential-feature-prd.prompt.md +19 -0
  266. package/resources/customizations/.github/prompts/drafts/update-implementation-plan.prompt.md +158 -0
  267. package/resources/customizations/.github/prompts/execute-plan-template.md +21 -0
  268. package/resources/customizations/.github/prompts/export-chat.prompt.md +7 -0
  269. package/resources/customizations/.github/prompts/fillout-prd-feature.prompt.md +46 -0
  270. package/resources/customizations/.github/prompts/generate-atomic-plan.prompt.md +96 -0
  271. package/resources/customizations/.github/prompts/generate-commit-message-repo.prompt.md +108 -0
  272. package/resources/customizations/.github/prompts/generate-pr.prompt.md +151 -0
  273. package/resources/customizations/.github/prompts/javascript-typescript-jest.prompt.md +44 -0
  274. package/resources/customizations/.github/prompts/orchestrate-csharp-work.prompt.md +66 -0
  275. package/resources/customizations/.github/prompts/orchestrate-powershell-work.prompt.md +50 -0
  276. package/resources/customizations/.github/prompts/orchestrate-python-work.prompt.md +50 -0
  277. package/resources/customizations/.github/prompts/orchestrate-work.prompt.md +66 -0
  278. package/resources/customizations/.github/prompts/remediate-comments.prompt.md +53 -0
  279. package/resources/customizations/.github/prompts/research-issue.prompt.md +125 -0
  280. package/resources/customizations/.github/prompts/review-epic.prompt.md +94 -0
  281. package/resources/customizations/.github/prompts/review-feature.prompt.md +130 -0
  282. package/resources/customizations/.github/prompts/review-staged.prompt.md +43 -0
  283. package/resources/customizations/.github/prompts/update_status.prompt.md +68 -0
  284. package/resources/customizations/.github/skills/README.md +26 -0
  285. package/resources/customizations/.github/skills/acceptance-criteria-tracking/SKILL.md +102 -0
  286. package/resources/customizations/.github/skills/atomic-plan-contract/SKILL.md +174 -0
  287. package/resources/customizations/.github/skills/csharp-change-budget-router/SKILL.md +48 -0
  288. package/resources/customizations/.github/skills/csharp-orchestration-state-machine/SKILL.md +57 -0
  289. package/resources/customizations/.github/skills/evidence-and-timestamp-conventions/SKILL.md +135 -0
  290. package/resources/customizations/.github/skills/feature-promotion-lifecycle/SKILL.md +121 -0
  291. package/resources/customizations/.github/skills/feature-review-workflow/SKILL.md +153 -0
  292. package/resources/customizations/.github/skills/make-skill-template/SKILL.md +147 -0
  293. package/resources/customizations/.github/skills/policy-audit-template-usage/SKILL.md +48 -0
  294. package/resources/customizations/.github/skills/policy-compliance-order/SKILL.md +37 -0
  295. package/resources/customizations/.github/skills/powershell-change-budget-router/SKILL.md +48 -0
  296. package/resources/customizations/.github/skills/powershell-orchestration-state-machine/SKILL.md +57 -0
  297. package/resources/customizations/.github/skills/pr-base-branch-merge-base/SKILL.md +55 -0
  298. package/resources/customizations/.github/skills/pr-context-artifacts/SKILL.md +29 -0
  299. package/resources/customizations/.github/skills/remediation-handoff-atomic-planner/SKILL.md +39 -0
  300. package/resources/customizations/.github/skills/skill-canonical-location-audit/SKILL.md +48 -0
  301. package/resources/feature-templates/bug/plan.yyyy-MM-ddTHH-mm.md +44 -0
  302. package/resources/feature-templates/bug/potential_bug.md +59 -0
  303. package/resources/feature-templates/bug/spec.md +99 -0
  304. package/resources/feature-templates/epic/initiative.md +43 -0
  305. package/resources/feature-templates/feature/plan.yyyy-MM-ddTHH-mm.md +53 -0
  306. package/resources/feature-templates/feature/spec.md +66 -0
  307. package/resources/feature-templates/feature/user-story.md +42 -0
  308. package/resources/feature-templates/potential/template.md +33 -0
  309. package/resources/feature-templates/refactor/plan.yyyy-MM-ddTHH-mm.md +52 -0
  310. package/resources/feature-templates/refactor/spec.md +69 -0
  311. package/resources/powershell/PoshQC/PoshQC.Analyzer.psm1 +254 -0
  312. package/resources/powershell/PoshQC/PoshQC.FileDiscovery.psm1 +138 -0
  313. package/resources/powershell/PoshQC/PoshQC.Testing.psm1 +409 -0
  314. package/resources/powershell/PoshQC/PoshQC.psd1 +31 -0
  315. package/resources/powershell/PoshQC/PoshQC.psm1 +101 -0
  316. package/resources/powershell/PoshQC/README.md +80 -0
  317. package/resources/powershell/PoshQC/settings/pester.runsettings.psd1 +59 -0
  318. package/resources/powershell/PoshQC/settings/pssa.settings.psd1 +55 -0
  319. package/resources/scripts/dev_tools/__init__.py +0 -0
  320. package/resources/scripts/dev_tools/agentic_sync.py +819 -0
  321. package/resources/scripts/dev_tools/codex_native_converter/__init__.py +11 -0
  322. package/resources/scripts/dev_tools/codex_native_converter/__main__.py +6 -0
  323. package/resources/scripts/dev_tools/codex_native_converter/cli.py +11 -0
  324. package/resources/scripts/dev_tools/new_active_feature_folder.py +79 -0
  325. package/resources/scripts/dev_tools/new_active_feature_folder_docs.py +268 -0
  326. package/resources/scripts/dev_tools/new_active_feature_folder_flow.py +366 -0
  327. package/resources/scripts/dev_tools/new_active_feature_folder_io.py +306 -0
  328. package/resources/scripts/dev_tools/new_active_feature_folder_markdown.py +252 -0
  329. package/resources/scripts/dev_tools/new_active_feature_folder_models.py +136 -0
  330. package/resources/scripts/dev_tools/new_potential_bug_entry.py +465 -0
  331. package/resources/scripts/dev_tools/potential_to_issue.py +421 -0
  332. package/resources/scripts/dev_tools/potential_to_issue_content.py +212 -0
  333. package/resources/scripts/dev_tools/pr_context/__init__.py +0 -0
  334. package/resources/scripts/dev_tools/pr_context/collector.py +619 -0
  335. package/resources/scripts/dev_tools/pr_context/feature_docs.py +349 -0
  336. package/resources/scripts/dev_tools/pr_context/git.py +153 -0
  337. package/resources/scripts/dev_tools/pr_context/github.py +549 -0
  338. package/resources/scripts/dev_tools/pr_context/models.py +198 -0
  339. package/resources/scripts/dev_tools/pr_context/render.py +342 -0
  340. package/resources/scripts/dev_tools/pr_context/render_feature_excerpts.py +256 -0
  341. package/resources/scripts/dev_tools/pr_context/render_pr_helpers.py +291 -0
  342. package/resources/scripts/dev_tools/pr_context/summary_helpers.py +386 -0
  343. package/resources/scripts/dev_tools/pr_context/verification_evidence.py +171 -0
  344. package/resources/scripts/dev_tools/prompt_mode_contract.py +152 -0
  345. package/resources/scripts/dev_tools/push_down_claude_customizations.py +188 -0
  346. package/resources/scripts/dev_tools/push_down_codex_and_agents_customizations.py +139 -0
  347. package/resources/scripts/dev_tools/push_down_copilot_customizations.py +504 -0
  348. package/resources/scripts/dev_tools/push_down_copilot_customizations_filesystem.py +217 -0
  349. package/resources/scripts/dev_tools/push_down_copilot_customizations_rewrites.py +293 -0
  350. package/resources/scripts/dev_tools/resolve_file_prompt.py +457 -0
  351. package/resources/scripts/dev_tools/resolve_hard_lock_prompt.py +444 -0
  352. package/resources/scripts/dev_tools/validate_orchestration_artifacts.py +554 -0
  353. package/resources/templates/codex_native_converter.py +35 -0
  354. package/resources/templates/collect_commit_context.py +212 -0
  355. package/resources/templates/collect_pr_context.py +74 -0
  356. package/resources/templates/hello_pwsh.ps1 +3 -0
  357. package/resources/templates/hello_python.py +11 -0
  358. package/resources/templates/link-parent-child.ps1 +480 -0
  359. package/resources/templates/new-claude-worktree-session.ps1 +232 -0
  360. package/resources/templates/new-potential-entry.ps1 +187 -0
  361. package/resources/templates/new_active_feature_folder.py +67 -0
  362. package/resources/templates/new_potential_bug_entry.py +54 -0
  363. package/resources/templates/policy_audit/AGENTS.md +117 -0
  364. package/resources/templates/policy_audit/code-review.yyyy-MM-ddTHH-mm.md +165 -0
  365. package/resources/templates/policy_audit/feature-audit.yyyy-MM-ddTHH-mm.md +124 -0
  366. package/resources/templates/policy_audit/policy-audit.yyyy-MM-ddTHH-mm.md +649 -0
  367. package/resources/templates/potential_to_issue.py +55 -0
  368. package/resources/templates/push_down_claude_customizations.py +188 -0
  369. package/resources/templates/push_down_codex_and_agents_customizations.py +95 -0
  370. package/resources/templates/push_down_copilot_customizations.py +124 -0
  371. package/resources/templates/resolve_atomic_plan_prompt.py +75 -0
  372. package/resources/templates/resolve_hard_lock_prompt.py +65 -0
  373. package/resources/templates/run-poshqc-analyze-autofix.ps1 +16 -0
  374. package/resources/templates/run-poshqc-analyze.ps1 +26 -0
  375. package/resources/templates/run-poshqc-format.ps1 +26 -0
  376. package/resources/templates/run-poshqc-suite.ps1 +24 -0
  377. package/resources/templates/run-poshqc-test.ps1 +32 -0
  378. package/resources/templates/sync-agents-from-instructions.ps1 +400 -0
  379. package/resources/templates/validate_orchestration_artifacts.py +55 -0
  380. package/resources/templates/vscode-cli.helpers.ps1 +63 -0
@@ -0,0 +1,549 @@
1
+ """GitHub CLI helpers for PR context collection."""
2
+
3
+ from __future__ import annotations
4
+
5
+ import base64
6
+ import json
7
+ import shutil
8
+ from typing import TYPE_CHECKING, cast
9
+
10
+ from .models import IssueDetails, PullRequestDetails, find_user_story_link
11
+
12
+ if TYPE_CHECKING:
13
+ from collections.abc import Sequence
14
+ from pathlib import Path
15
+
16
+ from .git import CommandRunner
17
+
18
+
19
+ class GhClient:
20
+ """GitHub CLI wrapper used for entity classification and content fetches."""
21
+
22
+ def __init__(
23
+ self, runner: CommandRunner, cwd: Path, gh_path: str | None = None
24
+ ) -> None:
25
+ self._runner = runner
26
+ self._cwd = cwd
27
+ self._gh_path = gh_path or shutil.which("gh")
28
+ self._repo_cache: str | None = None
29
+ self._availability_error: str | None = None
30
+ self._available = False
31
+ self._status_message: str | None = None
32
+ self._hydrate_availability()
33
+
34
+ @property
35
+ def available(self) -> bool:
36
+ return self._available
37
+
38
+ def ensure_available(self) -> None:
39
+ if not self._available:
40
+ message = self._availability_error or "GitHub CLI is unavailable."
41
+ raise RuntimeError(message)
42
+
43
+ @property
44
+ def status_message(self) -> str | None:
45
+ return self._status_message
46
+
47
+ def _hydrate_availability(self) -> None:
48
+ if not self._gh_path:
49
+ self._availability_error = "GitHub CLI (gh) is not installed. Install from https://cli.github.com/."
50
+ return
51
+
52
+ status = self._runner.run(
53
+ [self._gh_path, "auth", "status"], cwd=self._cwd, allow_error=True
54
+ )
55
+ if status.code != 0:
56
+ stderr = status.stderr or status.stdout or "unknown error"
57
+ self._availability_error = (
58
+ "GitHub CLI is installed but not authenticated. Run 'gh auth login' "
59
+ "to authenticate."
60
+ )
61
+ if stderr:
62
+ self._availability_error += f" Details: {stderr.strip()}"
63
+ return
64
+
65
+ repo = self._repo_name()
66
+ if not repo:
67
+ self._availability_error = (
68
+ "GitHub CLI is authenticated but failed to resolve repository. "
69
+ "Ensure network access and repository permissions."
70
+ )
71
+ return
72
+
73
+ self._available = True
74
+ self._availability_error = None
75
+ self._status_message = f"GitHub CLI authenticated for {repo}"
76
+
77
+ def _repo_name(self) -> str | None:
78
+ if self._repo_cache:
79
+ return self._repo_cache
80
+ if not self._gh_path:
81
+ return None
82
+
83
+ result = self._runner.run(
84
+ [self._gh_path, "repo", "view", "--json", "nameWithOwner"],
85
+ cwd=self._cwd,
86
+ allow_error=True,
87
+ )
88
+ if result.code != 0 or not result.stdout:
89
+ return None
90
+
91
+ try:
92
+ payload = json.loads(result.stdout)
93
+ except json.JSONDecodeError:
94
+ return None
95
+
96
+ if not isinstance(payload, dict):
97
+ return None
98
+
99
+ payload_dict = cast("dict[str, object]", payload)
100
+ name_raw = payload_dict.get("nameWithOwner")
101
+ if isinstance(name_raw, str):
102
+ self._repo_cache = name_raw
103
+ return name_raw
104
+ return None
105
+
106
+ def _request_json(
107
+ self,
108
+ args: Sequence[str],
109
+ *,
110
+ context: str,
111
+ allow_not_found: bool = False,
112
+ ) -> object:
113
+ result = self._runner.run(args, cwd=self._cwd, allow_error=True)
114
+ if result.code != 0:
115
+ message = result.stderr or result.stdout or "unknown error"
116
+ if allow_not_found and ("404" in message or "Not Found" in message):
117
+ return None
118
+ raise RuntimeError(f"{context} failed ({result.code}): {message.strip()}")
119
+ try:
120
+ return json.loads(result.stdout)
121
+ except json.JSONDecodeError as exc: # pragma: no cover - defensive
122
+ raise RuntimeError(f"{context} returned invalid JSON") from exc
123
+
124
+ def classify_entity(self, number: str) -> str | None:
125
+ """Return "issue" or "pull" if determinable, otherwise None."""
126
+
127
+ self.ensure_available()
128
+ repo = self._repo_name()
129
+ if not repo:
130
+ return None
131
+
132
+ api_path = f"repos/{repo}/issues/{number}"
133
+ try:
134
+ payload = self._request_json(
135
+ [self._gh_path or "gh", "api", api_path],
136
+ context="Classify entity",
137
+ allow_not_found=True,
138
+ )
139
+ except RuntimeError:
140
+ return None
141
+
142
+ if payload is None:
143
+ return None
144
+ if isinstance(payload, dict) and "pull_request" in payload:
145
+ return "pull"
146
+ return "issue"
147
+
148
+ def closing_issues(self, pr_number: str | None = None) -> list[str]:
149
+ self.ensure_available()
150
+ repo = self._repo_name()
151
+ if not repo:
152
+ return []
153
+
154
+ args = [
155
+ self._gh_path or "gh",
156
+ "pr",
157
+ "view",
158
+ "--json",
159
+ "closingIssuesReferences,number",
160
+ ]
161
+ if pr_number:
162
+ args.insert(3, pr_number)
163
+
164
+ payload = self._request_json(args, context="Fetch closing issues")
165
+
166
+ if not isinstance(payload, dict):
167
+ return []
168
+
169
+ payload_dict = cast("dict[str, object]", payload)
170
+ issues_raw: object | None = payload_dict.get("closingIssuesReferences")
171
+ numbers: list[str] = []
172
+ if isinstance(issues_raw, list):
173
+ issues_list = cast("list[object]", issues_raw)
174
+ for entry in issues_list:
175
+ if not isinstance(entry, dict):
176
+ continue
177
+
178
+ entry_dict = cast("dict[str, object]", entry)
179
+ number = entry_dict.get("number")
180
+ if isinstance(number, int):
181
+ numbers.append(f"#{number}")
182
+
183
+ return sorted(set(numbers))
184
+
185
+ def issue_details(self, number: str) -> IssueDetails:
186
+ self.ensure_available()
187
+ repo = self._repo_name()
188
+ if not repo:
189
+ raise RuntimeError("Unable to resolve repository for issue lookup.")
190
+
191
+ payload = self._request_json(
192
+ [self._gh_path or "gh", "api", f"repos/{repo}/issues/{number}"],
193
+ context="Fetch issue",
194
+ )
195
+ if not isinstance(payload, dict):
196
+ raise RuntimeError("Unexpected issue payload format.")
197
+
198
+ payload_dict = cast("dict[str, object]", payload)
199
+
200
+ title_raw = payload_dict.get("title")
201
+ title = title_raw if isinstance(title_raw, str) else ""
202
+ state_raw = payload_dict.get("state")
203
+ state = state_raw if isinstance(state_raw, str) else ""
204
+
205
+ body_raw = payload_dict.get("body")
206
+ body = body_raw if isinstance(body_raw, str) else ""
207
+
208
+ labels_raw = payload_dict.get("labels")
209
+ labels: list[str] = []
210
+ if isinstance(labels_raw, list):
211
+ for entry in cast("list[object]", labels_raw):
212
+ if isinstance(entry, dict):
213
+ entry_dict = cast("dict[str, object]", entry)
214
+ name_raw = entry_dict.get("name")
215
+ if isinstance(name_raw, str):
216
+ labels.append(name_raw)
217
+
218
+ assignees_raw = payload_dict.get("assignees")
219
+ assignees: list[str] = []
220
+ if isinstance(assignees_raw, list):
221
+ for entry in cast("list[object]", assignees_raw):
222
+ if isinstance(entry, dict):
223
+ entry_dict = cast("dict[str, object]", entry)
224
+ login_raw = entry_dict.get("login")
225
+ if isinstance(login_raw, str):
226
+ assignees.append(login_raw)
227
+
228
+ user_raw = payload_dict.get("user")
229
+ author = ""
230
+ if isinstance(user_raw, dict):
231
+ user_dict = cast("dict[str, object]", user_raw)
232
+ login_raw = user_dict.get("login")
233
+ author = login_raw if isinstance(login_raw, str) else ""
234
+
235
+ created_raw = payload_dict.get("created_at")
236
+ updated_raw = payload_dict.get("updated_at")
237
+ created_at = created_raw if isinstance(created_raw, str) else ""
238
+ updated_at = updated_raw if isinstance(updated_raw, str) else ""
239
+
240
+ comments_url_raw = payload_dict.get("comments_url")
241
+ comments_url = comments_url_raw if isinstance(comments_url_raw, str) else None
242
+
243
+ comments: list[str] = []
244
+ if comments_url:
245
+ comment_payload = self._request_json(
246
+ [self._gh_path or "gh", "api", f"{comments_url}?per_page=50"],
247
+ context="Fetch issue comments",
248
+ )
249
+ if isinstance(comment_payload, list):
250
+ comment_entries = cast("list[object]", comment_payload)
251
+ for entry in comment_entries:
252
+ if not isinstance(entry, dict):
253
+ continue
254
+ entry_dict = cast("dict[str, object]", entry)
255
+ user_raw = entry_dict.get("user")
256
+ login = ""
257
+ if isinstance(user_raw, dict):
258
+ user_dict = cast("dict[str, object]", user_raw)
259
+ login_raw = user_dict.get("login")
260
+ if isinstance(login_raw, str):
261
+ login = login_raw
262
+ created_raw = entry_dict.get("created_at")
263
+ created = created_raw if isinstance(created_raw, str) else ""
264
+ entry_body_raw = entry_dict.get("body")
265
+ entry_body = (
266
+ entry_body_raw if isinstance(entry_body_raw, str) else ""
267
+ )
268
+ prefix = f"{login}" if login else "(unknown)"
269
+ if created:
270
+ prefix = f"{prefix} at {created}"
271
+ comments.append(f"{prefix}: {entry_body}".strip())
272
+
273
+ story_path = find_user_story_link(body)
274
+ story_content = None
275
+ if story_path:
276
+ local_path = self._cwd / story_path
277
+ if local_path.exists():
278
+ story_content = local_path.read_text(encoding="utf-8")
279
+ else:
280
+ story_content = self.fetch_repo_file(story_path)
281
+
282
+ return IssueDetails(
283
+ number=f"#{number}",
284
+ title=title or "(no title)",
285
+ state=state or "(unknown)",
286
+ labels=labels,
287
+ assignees=assignees,
288
+ author=author or "(unknown)",
289
+ created_at=created_at,
290
+ updated_at=updated_at,
291
+ body=body or "(no body)",
292
+ comments=comments,
293
+ user_story_path=story_path,
294
+ user_story_content=story_content,
295
+ )
296
+
297
+ def pr_details(self, number: str) -> PullRequestDetails:
298
+ self.ensure_available()
299
+ repo = self._repo_name()
300
+ if not repo:
301
+ raise RuntimeError("Unable to resolve repository for PR lookup.")
302
+
303
+ payload = self._request_json(
304
+ [
305
+ self._gh_path or "gh",
306
+ "pr",
307
+ "view",
308
+ number,
309
+ "--json",
310
+ (
311
+ "number,title,body,state,author,baseRefName,headRefName,"
312
+ "createdAt,updatedAt,mergedAt,labels,assignees,"
313
+ "closingIssuesReferences,files"
314
+ ),
315
+ ],
316
+ context="Fetch pull request",
317
+ )
318
+ if not isinstance(payload, dict):
319
+ raise RuntimeError("Unexpected pull request payload format.")
320
+
321
+ payload_dict = cast("dict[str, object]", payload)
322
+
323
+ title_raw = payload_dict.get("title")
324
+ title = title_raw if isinstance(title_raw, str) else ""
325
+ state_raw = payload_dict.get("state")
326
+ state = state_raw if isinstance(state_raw, str) else ""
327
+
328
+ body_raw = payload_dict.get("body")
329
+ body = body_raw if isinstance(body_raw, str) else ""
330
+ closing: list[str] = []
331
+ closing_raw = payload_dict.get("closingIssuesReferences")
332
+ if isinstance(closing_raw, list):
333
+ for entry in cast("list[object]", closing_raw):
334
+ if isinstance(entry, dict):
335
+ entry_dict = cast("dict[str, object]", entry)
336
+ num_raw = entry_dict.get("number")
337
+ if isinstance(num_raw, int):
338
+ closing.append(f"#{num_raw}")
339
+
340
+ labels_raw = payload_dict.get("labels")
341
+ labels: list[str] = []
342
+ if isinstance(labels_raw, list):
343
+ for entry in cast("list[object]", labels_raw):
344
+ if isinstance(entry, dict):
345
+ entry_dict = cast("dict[str, object]", entry)
346
+ name_raw = entry_dict.get("name")
347
+ if isinstance(name_raw, str):
348
+ labels.append(name_raw)
349
+
350
+ assignees_raw = payload_dict.get("assignees")
351
+ assignees: list[str] = []
352
+ if isinstance(assignees_raw, list):
353
+ for entry in cast("list[object]", assignees_raw):
354
+ if isinstance(entry, dict):
355
+ entry_dict = cast("dict[str, object]", entry)
356
+ login_raw = entry_dict.get("login")
357
+ if isinstance(login_raw, str):
358
+ assignees.append(login_raw)
359
+
360
+ author_raw = payload_dict.get("author")
361
+ author_login = ""
362
+ if isinstance(author_raw, dict):
363
+ author_dict = cast("dict[str, object]", author_raw)
364
+ author_login_raw = author_dict.get("login")
365
+ author_login = author_login_raw if isinstance(author_login_raw, str) else ""
366
+
367
+ base_ref_raw = payload_dict.get("baseRefName")
368
+ head_ref_raw = payload_dict.get("headRefName")
369
+ created_raw = payload_dict.get("createdAt")
370
+ updated_raw = payload_dict.get("updatedAt")
371
+ merged_raw = payload_dict.get("mergedAt")
372
+ files_raw = payload_dict.get("files")
373
+ files_changed: list[str] = []
374
+ if isinstance(files_raw, list):
375
+ for entry in cast("list[object]", files_raw):
376
+ if isinstance(entry, dict):
377
+ entry_dict = cast("dict[str, object]", entry)
378
+ path_raw = entry_dict.get("path")
379
+ if isinstance(path_raw, str):
380
+ files_changed.append(path_raw)
381
+
382
+ return PullRequestDetails(
383
+ number=f"#{number}",
384
+ title=title or "(no title)",
385
+ state=state or "(unknown)",
386
+ author=author_login or "(unknown)",
387
+ base_ref=base_ref_raw if isinstance(base_ref_raw, str) else "(unknown)",
388
+ head_ref=head_ref_raw if isinstance(head_ref_raw, str) else "(unknown)",
389
+ created_at=created_raw if isinstance(created_raw, str) else "",
390
+ updated_at=updated_raw if isinstance(updated_raw, str) else "",
391
+ merged_at=merged_raw if isinstance(merged_raw, str) else None,
392
+ labels=labels,
393
+ assignees=assignees,
394
+ body=body or "(no body)",
395
+ closing_issues=closing,
396
+ files_changed=files_changed,
397
+ )
398
+
399
+ def current_pr(self) -> PullRequestDetails | None:
400
+ if not self._available:
401
+ return None
402
+
403
+ args = [
404
+ self._gh_path or "gh",
405
+ "pr",
406
+ "view",
407
+ "--json",
408
+ (
409
+ "number,title,body,state,author,baseRefName,headRefName,"
410
+ "createdAt,updatedAt,mergedAt,labels,assignees,closingIssuesReferences"
411
+ ),
412
+ ]
413
+ result = self._runner.run(args, cwd=self._cwd, allow_error=True)
414
+ if result.code != 0 or not result.stdout:
415
+ return None
416
+
417
+ try:
418
+ payload = json.loads(result.stdout)
419
+ except json.JSONDecodeError:
420
+ return None
421
+
422
+ if not isinstance(payload, dict):
423
+ return None
424
+
425
+ payload_dict = cast("dict[str, object]", payload)
426
+
427
+ closing_raw = payload_dict.get("closingIssuesReferences")
428
+ closing: list[str] = []
429
+ if isinstance(closing_raw, list):
430
+ closing_entries = cast("list[object]", closing_raw)
431
+ for entry in closing_entries:
432
+ if not isinstance(entry, dict):
433
+ continue
434
+ entry_dict = cast("dict[str, object]", entry)
435
+ number = entry_dict.get("number")
436
+ if isinstance(number, int):
437
+ closing.append(f"#{number}")
438
+
439
+ title_raw = payload_dict.get("title")
440
+ title = title_raw if isinstance(title_raw, str) else ""
441
+ body_raw = payload_dict.get("body")
442
+ body = body_raw if isinstance(body_raw, str) else ""
443
+ number_raw = payload_dict.get("number")
444
+ number = number_raw if isinstance(number_raw, int) else None
445
+ if number is None:
446
+ return None
447
+
448
+ state_raw = payload_dict.get("state")
449
+ state = state_raw if isinstance(state_raw, str) else ""
450
+ author_raw = payload_dict.get("author")
451
+ author_login = ""
452
+ if isinstance(author_raw, dict):
453
+ author_dict = cast("dict[str, object]", author_raw)
454
+ author_login_raw = author_dict.get("login")
455
+ author_login = author_login_raw if isinstance(author_login_raw, str) else ""
456
+ base_ref_raw = payload_dict.get("baseRefName")
457
+ head_ref_raw = payload_dict.get("headRefName")
458
+ created_raw = payload_dict.get("createdAt")
459
+ updated_raw = payload_dict.get("updatedAt")
460
+ merged_raw = payload_dict.get("mergedAt")
461
+ labels_raw = payload_dict.get("labels")
462
+ labels: list[str] = []
463
+ if isinstance(labels_raw, list):
464
+ for entry in cast("list[object]", labels_raw):
465
+ if isinstance(entry, dict):
466
+ entry_dict = cast("dict[str, object]", entry)
467
+ name_raw = entry_dict.get("name")
468
+ if isinstance(name_raw, str):
469
+ labels.append(name_raw)
470
+ assignees_raw = payload_dict.get("assignees")
471
+ assignees: list[str] = []
472
+ if isinstance(assignees_raw, list):
473
+ for entry in cast("list[object]", assignees_raw):
474
+ if isinstance(entry, dict):
475
+ entry_dict = cast("dict[str, object]", entry)
476
+ login_raw = entry_dict.get("login")
477
+ if isinstance(login_raw, str):
478
+ assignees.append(login_raw)
479
+
480
+ return PullRequestDetails(
481
+ number=f"#{number}",
482
+ title=title or "(no title)",
483
+ state=state or "(unknown)",
484
+ author=author_login or "(unknown)",
485
+ base_ref=base_ref_raw if isinstance(base_ref_raw, str) else "(unknown)",
486
+ head_ref=head_ref_raw if isinstance(head_ref_raw, str) else "(unknown)",
487
+ created_at=created_raw if isinstance(created_raw, str) else "",
488
+ updated_at=updated_raw if isinstance(updated_raw, str) else "",
489
+ merged_at=merged_raw if isinstance(merged_raw, str) else None,
490
+ labels=labels,
491
+ assignees=assignees,
492
+ body=body or "(no body)",
493
+ closing_issues=sorted(set(closing)),
494
+ files_changed=[],
495
+ )
496
+
497
+ def fetch_repo_file(self, path: str) -> str | None:
498
+ self.ensure_available()
499
+ repo = self._repo_name()
500
+ if not repo:
501
+ return None
502
+
503
+ api_path = f"repos/{repo}/contents/{path.lstrip('/')}"
504
+ try:
505
+ payload = self._request_json(
506
+ [self._gh_path or "gh", "api", api_path], context="Fetch file"
507
+ )
508
+ except RuntimeError:
509
+ return None
510
+
511
+ if not isinstance(payload, dict):
512
+ return None
513
+
514
+ payload_dict = cast("dict[str, object]", payload)
515
+
516
+ content_raw = payload_dict.get("content")
517
+ if not isinstance(content_raw, str):
518
+ return None
519
+
520
+ try:
521
+ decoded = base64.b64decode(content_raw)
522
+ except (ValueError, TypeError): # pragma: no cover - defensive
523
+ return None
524
+ return decoded.decode("utf-8", errors="replace")
525
+
526
+ def ci_status(self, head_sha: str) -> tuple[str | None, list[str]]:
527
+ self.ensure_available()
528
+ args = [
529
+ self._gh_path or "gh",
530
+ "run",
531
+ "list",
532
+ "--commit",
533
+ head_sha,
534
+ "--limit",
535
+ "1",
536
+ "--json",
537
+ "conclusion,status,name,headSha",
538
+ ]
539
+ payload = self._request_json(args, context="Fetch CI status")
540
+ if not isinstance(payload, list) or not payload:
541
+ return None, []
542
+ runs = cast("list[object]", payload)
543
+ first_obj = runs[0]
544
+ if not isinstance(first_obj, dict):
545
+ return None, []
546
+ first = cast("dict[str, object]", first_obj)
547
+ status_raw = first.get("status") or first.get("conclusion")
548
+ status = status_raw if isinstance(status_raw, str) else None
549
+ return status, []