@mohanscodex/spectra-code 0.4.9 → 0.5.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 (861) hide show
  1. package/README.md +42 -2
  2. package/dist/package.json +8 -5
  3. package/dist/src/agents/definitions/build.d.ts +3 -0
  4. package/dist/src/agents/definitions/build.d.ts.map +1 -0
  5. package/dist/src/agents/definitions/build.js +51 -0
  6. package/dist/src/agents/definitions/build.js.map +1 -0
  7. package/dist/src/agents/definitions/debug.d.ts +3 -0
  8. package/dist/src/agents/definitions/debug.d.ts.map +1 -0
  9. package/dist/src/agents/definitions/debug.js +45 -0
  10. package/dist/src/agents/definitions/debug.js.map +1 -0
  11. package/dist/src/agents/definitions/explore.d.ts +3 -0
  12. package/dist/src/agents/definitions/explore.d.ts.map +1 -0
  13. package/dist/src/agents/definitions/explore.js +34 -0
  14. package/dist/src/agents/definitions/explore.js.map +1 -0
  15. package/dist/src/agents/definitions/index.d.ts +5 -0
  16. package/dist/src/agents/definitions/index.d.ts.map +1 -0
  17. package/dist/src/agents/definitions/index.js +19 -0
  18. package/dist/src/agents/definitions/index.js.map +1 -0
  19. package/dist/src/agents/definitions/plan.d.ts +3 -0
  20. package/dist/src/agents/definitions/plan.d.ts.map +1 -0
  21. package/dist/src/agents/definitions/plan.js +44 -0
  22. package/dist/src/agents/definitions/plan.js.map +1 -0
  23. package/dist/src/agents/definitions/title.d.ts +3 -0
  24. package/dist/src/agents/definitions/title.d.ts.map +1 -0
  25. package/dist/src/agents/definitions/title.js +34 -0
  26. package/dist/src/agents/definitions/title.js.map +1 -0
  27. package/dist/src/agents/index.d.ts +6 -0
  28. package/dist/src/agents/index.d.ts.map +1 -0
  29. package/dist/src/agents/index.js +10 -0
  30. package/dist/src/agents/index.js.map +1 -0
  31. package/dist/src/agents/types.d.ts +15 -0
  32. package/dist/src/agents/types.d.ts.map +1 -0
  33. package/dist/src/agents/types.js +2 -0
  34. package/dist/src/agents/types.js.map +1 -0
  35. package/dist/src/cli.js +2 -2
  36. package/dist/src/cli.js.map +1 -1
  37. package/dist/src/commands/session.js +1 -1
  38. package/dist/src/commands/session.js.map +1 -1
  39. package/dist/src/integrations/acp/server.js +1 -1
  40. package/dist/src/integrations/acp/server.js.map +1 -1
  41. package/dist/src/services/prompt-history.d.ts +32 -0
  42. package/dist/src/services/prompt-history.d.ts.map +1 -0
  43. package/dist/src/services/prompt-history.js +144 -0
  44. package/dist/src/services/prompt-history.js.map +1 -0
  45. package/dist/src/services/snapshot-manager.d.ts +32 -32
  46. package/dist/src/services/snapshot-manager.d.ts.map +1 -1
  47. package/dist/src/services/snapshot-manager.js +308 -155
  48. package/dist/src/services/snapshot-manager.js.map +1 -1
  49. package/dist/src/tools/edit.d.ts.map +1 -1
  50. package/dist/src/tools/edit.js +5 -22
  51. package/dist/src/tools/edit.js.map +1 -1
  52. package/dist/src/tools/glob.d.ts.map +1 -1
  53. package/dist/src/tools/glob.js +2 -1
  54. package/dist/src/tools/glob.js.map +1 -1
  55. package/dist/src/tools/grep.js +2 -2
  56. package/dist/src/tools/grep.js.map +1 -1
  57. package/dist/src/tools/index.d.ts +5 -0
  58. package/dist/src/tools/index.d.ts.map +1 -1
  59. package/dist/src/tools/index.js +30 -1
  60. package/dist/src/tools/index.js.map +1 -1
  61. package/dist/src/tools/task.js +1 -1
  62. package/dist/src/tools/task.js.map +1 -1
  63. package/dist/src/tools/write.d.ts.map +1 -1
  64. package/dist/src/tools/write.js +7 -4
  65. package/dist/src/tools/write.js.map +1 -1
  66. package/dist/src/tui/app-constants.js +1 -1
  67. package/dist/src/tui/app-constants.js.map +1 -1
  68. package/dist/src/tui/app.d.ts.map +1 -1
  69. package/dist/src/tui/app.js +35 -15
  70. package/dist/src/tui/app.js.map +1 -1
  71. package/dist/src/tui/commands.d.ts +15 -0
  72. package/dist/src/tui/commands.d.ts.map +1 -1
  73. package/dist/src/tui/commands.js +252 -1
  74. package/dist/src/tui/commands.js.map +1 -1
  75. package/dist/src/tui/components/chat-area.d.ts.map +1 -1
  76. package/dist/src/tui/components/chat-area.js +1 -1
  77. package/dist/src/tui/components/chat-area.js.map +1 -1
  78. package/dist/src/tui/components/message.d.ts.map +1 -1
  79. package/dist/src/tui/components/message.js +79 -4
  80. package/dist/src/tui/components/message.js.map +1 -1
  81. package/dist/src/tui/hooks/use-agent.d.ts.map +1 -1
  82. package/dist/src/tui/hooks/use-agent.js +17 -4
  83. package/dist/src/tui/hooks/use-agent.js.map +1 -1
  84. package/dist/src/tui/hooks/use-app-keyboard.d.ts +4 -5
  85. package/dist/src/tui/hooks/use-app-keyboard.d.ts.map +1 -1
  86. package/dist/src/tui/hooks/use-app-keyboard.js +23 -24
  87. package/dist/src/tui/hooks/use-app-keyboard.js.map +1 -1
  88. package/dist/src/tui/hooks/use-chat-submit.d.ts +2 -2
  89. package/dist/src/tui/hooks/use-chat-submit.d.ts.map +1 -1
  90. package/dist/src/tui/hooks/use-chat-submit.js +143 -20
  91. package/dist/src/tui/hooks/use-chat-submit.js.map +1 -1
  92. package/dist/src/tui/hooks/use-revert.d.ts +3 -5
  93. package/dist/src/tui/hooks/use-revert.d.ts.map +1 -1
  94. package/dist/src/tui/hooks/use-revert.js +83 -34
  95. package/dist/src/tui/hooks/use-revert.js.map +1 -1
  96. package/dist/src/tui/hooks/use-title-agent.d.ts +13 -0
  97. package/dist/src/tui/hooks/use-title-agent.d.ts.map +1 -0
  98. package/dist/src/tui/hooks/use-title-agent.js +50 -0
  99. package/dist/src/tui/hooks/use-title-agent.js.map +1 -0
  100. package/dist/src/tui/prompt-bar.d.ts.map +1 -1
  101. package/dist/src/tui/prompt-bar.js +27 -33
  102. package/dist/src/tui/prompt-bar.js.map +1 -1
  103. package/dist/src/tui/theme.d.ts +2 -0
  104. package/dist/src/tui/theme.d.ts.map +1 -1
  105. package/dist/src/tui/theme.js +2 -0
  106. package/dist/src/tui/theme.js.map +1 -1
  107. package/dist/src/tui/types.d.ts +5 -0
  108. package/dist/src/tui/types.d.ts.map +1 -1
  109. package/dist/src/tui/ui/agent-switcher.js +1 -1
  110. package/dist/src/tui/ui/agent-switcher.js.map +1 -1
  111. package/dist/src/tui/ui/cost-dialog.d.ts +14 -0
  112. package/dist/src/tui/ui/cost-dialog.d.ts.map +1 -0
  113. package/dist/src/tui/ui/cost-dialog.js +28 -0
  114. package/dist/src/tui/ui/cost-dialog.js.map +1 -0
  115. package/dist/src/tui/ui/debug-dialog.js +1 -1
  116. package/dist/src/tui/ui/debug-dialog.js.map +1 -1
  117. package/dist/src/tui/ui/session-list.js +1 -1
  118. package/dist/src/tui/ui/session-list.js.map +1 -1
  119. package/dist/src/tui/utils/terminal-title.d.ts +3 -0
  120. package/dist/src/tui/utils/terminal-title.d.ts.map +1 -0
  121. package/dist/src/tui/utils/terminal-title.js +14 -0
  122. package/dist/src/tui/utils/terminal-title.js.map +1 -0
  123. package/dist/src/tui/utils/version.d.ts +1 -1
  124. package/dist/src/tui/utils/version.d.ts.map +1 -1
  125. package/dist/src/tui/utils/version.js +4 -22
  126. package/dist/src/tui/utils/version.js.map +1 -1
  127. package/dist/src/utils/paths.d.ts +1 -0
  128. package/dist/src/utils/paths.d.ts.map +1 -1
  129. package/dist/src/utils/paths.js +13 -0
  130. package/dist/src/utils/paths.js.map +1 -1
  131. package/package.json +9 -6
  132. package/skills/REQUESTS.md +36 -0
  133. package/skills/architecture/ABOUT.md +20 -0
  134. package/skills/architecture/preserving-productive-tensions/SKILL.md +152 -0
  135. package/skills/aspnet-core/LICENSE.txt +202 -0
  136. package/skills/aspnet-core/SKILL.md +61 -0
  137. package/skills/aspnet-core/agents/openai.yaml +5 -0
  138. package/skills/aspnet-core/assets/dotnet-logo.png +0 -0
  139. package/skills/aspnet-core/references/_sections.md +40 -0
  140. package/skills/aspnet-core/references/apis-minimal-and-controllers.md +81 -0
  141. package/skills/aspnet-core/references/data-state-and-services.md +69 -0
  142. package/skills/aspnet-core/references/program-and-pipeline.md +103 -0
  143. package/skills/aspnet-core/references/realtime-grpc-and-background-work.md +58 -0
  144. package/skills/aspnet-core/references/security-and-identity.md +75 -0
  145. package/skills/aspnet-core/references/source-map.md +43 -0
  146. package/skills/aspnet-core/references/stack-selection.md +63 -0
  147. package/skills/aspnet-core/references/testing-performance-and-operations.md +92 -0
  148. package/skills/aspnet-core/references/ui-blazor.md +53 -0
  149. package/skills/aspnet-core/references/ui-mvc.md +56 -0
  150. package/skills/aspnet-core/references/ui-razor-pages.md +55 -0
  151. package/skills/aspnet-core/references/versioning-and-upgrades.md +51 -0
  152. package/skills/chatgpt-apps/LICENSE.txt +201 -0
  153. package/skills/chatgpt-apps/SKILL.md +320 -0
  154. package/skills/chatgpt-apps/agents/openai.yaml +13 -0
  155. package/skills/chatgpt-apps/references/app-archetypes.md +132 -0
  156. package/skills/chatgpt-apps/references/apps-sdk-docs-workflow.md +135 -0
  157. package/skills/chatgpt-apps/references/interactive-state-sync-patterns.md +113 -0
  158. package/skills/chatgpt-apps/references/repo-contract-and-validation.md +93 -0
  159. package/skills/chatgpt-apps/references/search-fetch-standard.md +67 -0
  160. package/skills/chatgpt-apps/references/upstream-example-workflow.md +79 -0
  161. package/skills/chatgpt-apps/references/window-openai-patterns.md +70 -0
  162. package/skills/chatgpt-apps/scripts/scaffold_node_ext_apps.mjs +606 -0
  163. package/skills/cloudflare-deploy/LICENSE.txt +201 -0
  164. package/skills/cloudflare-deploy/SKILL.md +224 -0
  165. package/skills/cloudflare-deploy/agents/openai.yaml +6 -0
  166. package/skills/cloudflare-deploy/assets/cloudflare-small.svg +3 -0
  167. package/skills/cloudflare-deploy/assets/cloudflare.png +0 -0
  168. package/skills/cloudflare-deploy/references/agents-sdk/README.md +89 -0
  169. package/skills/cloudflare-deploy/references/agents-sdk/api.md +190 -0
  170. package/skills/cloudflare-deploy/references/agents-sdk/configuration.md +182 -0
  171. package/skills/cloudflare-deploy/references/agents-sdk/gotchas.md +158 -0
  172. package/skills/cloudflare-deploy/references/agents-sdk/patterns.md +192 -0
  173. package/skills/cloudflare-deploy/references/ai-gateway/README.md +175 -0
  174. package/skills/cloudflare-deploy/references/ai-gateway/configuration.md +111 -0
  175. package/skills/cloudflare-deploy/references/ai-gateway/dynamic-routing.md +82 -0
  176. package/skills/cloudflare-deploy/references/ai-gateway/features.md +96 -0
  177. package/skills/cloudflare-deploy/references/ai-gateway/sdk-integration.md +114 -0
  178. package/skills/cloudflare-deploy/references/ai-gateway/troubleshooting.md +88 -0
  179. package/skills/cloudflare-deploy/references/ai-search/README.md +138 -0
  180. package/skills/cloudflare-deploy/references/ai-search/api.md +87 -0
  181. package/skills/cloudflare-deploy/references/ai-search/configuration.md +88 -0
  182. package/skills/cloudflare-deploy/references/ai-search/gotchas.md +81 -0
  183. package/skills/cloudflare-deploy/references/ai-search/patterns.md +85 -0
  184. package/skills/cloudflare-deploy/references/analytics-engine/README.md +92 -0
  185. package/skills/cloudflare-deploy/references/analytics-engine/api.md +112 -0
  186. package/skills/cloudflare-deploy/references/analytics-engine/configuration.md +112 -0
  187. package/skills/cloudflare-deploy/references/analytics-engine/gotchas.md +85 -0
  188. package/skills/cloudflare-deploy/references/analytics-engine/patterns.md +83 -0
  189. package/skills/cloudflare-deploy/references/api/README.md +65 -0
  190. package/skills/cloudflare-deploy/references/api/api.md +204 -0
  191. package/skills/cloudflare-deploy/references/api/configuration.md +160 -0
  192. package/skills/cloudflare-deploy/references/api/gotchas.md +225 -0
  193. package/skills/cloudflare-deploy/references/api/patterns.md +204 -0
  194. package/skills/cloudflare-deploy/references/api-shield/README.md +44 -0
  195. package/skills/cloudflare-deploy/references/api-shield/api.md +141 -0
  196. package/skills/cloudflare-deploy/references/api-shield/configuration.md +192 -0
  197. package/skills/cloudflare-deploy/references/api-shield/gotchas.md +125 -0
  198. package/skills/cloudflare-deploy/references/api-shield/patterns.md +180 -0
  199. package/skills/cloudflare-deploy/references/argo-smart-routing/README.md +90 -0
  200. package/skills/cloudflare-deploy/references/argo-smart-routing/api.md +240 -0
  201. package/skills/cloudflare-deploy/references/argo-smart-routing/configuration.md +197 -0
  202. package/skills/cloudflare-deploy/references/argo-smart-routing/gotchas.md +111 -0
  203. package/skills/cloudflare-deploy/references/argo-smart-routing/patterns.md +104 -0
  204. package/skills/cloudflare-deploy/references/bindings/README.md +122 -0
  205. package/skills/cloudflare-deploy/references/bindings/api.md +203 -0
  206. package/skills/cloudflare-deploy/references/bindings/configuration.md +188 -0
  207. package/skills/cloudflare-deploy/references/bindings/gotchas.md +208 -0
  208. package/skills/cloudflare-deploy/references/bindings/patterns.md +200 -0
  209. package/skills/cloudflare-deploy/references/bot-management/README.md +94 -0
  210. package/skills/cloudflare-deploy/references/bot-management/api.md +169 -0
  211. package/skills/cloudflare-deploy/references/bot-management/configuration.md +163 -0
  212. package/skills/cloudflare-deploy/references/bot-management/gotchas.md +114 -0
  213. package/skills/cloudflare-deploy/references/bot-management/patterns.md +182 -0
  214. package/skills/cloudflare-deploy/references/browser-rendering/README.md +78 -0
  215. package/skills/cloudflare-deploy/references/browser-rendering/api.md +108 -0
  216. package/skills/cloudflare-deploy/references/browser-rendering/configuration.md +78 -0
  217. package/skills/cloudflare-deploy/references/browser-rendering/gotchas.md +88 -0
  218. package/skills/cloudflare-deploy/references/browser-rendering/patterns.md +91 -0
  219. package/skills/cloudflare-deploy/references/c3/README.md +111 -0
  220. package/skills/cloudflare-deploy/references/c3/api.md +71 -0
  221. package/skills/cloudflare-deploy/references/c3/configuration.md +81 -0
  222. package/skills/cloudflare-deploy/references/c3/gotchas.md +92 -0
  223. package/skills/cloudflare-deploy/references/c3/patterns.md +82 -0
  224. package/skills/cloudflare-deploy/references/cache-reserve/README.md +147 -0
  225. package/skills/cloudflare-deploy/references/cache-reserve/api.md +194 -0
  226. package/skills/cloudflare-deploy/references/cache-reserve/configuration.md +169 -0
  227. package/skills/cloudflare-deploy/references/cache-reserve/gotchas.md +132 -0
  228. package/skills/cloudflare-deploy/references/cache-reserve/patterns.md +197 -0
  229. package/skills/cloudflare-deploy/references/containers/README.md +85 -0
  230. package/skills/cloudflare-deploy/references/containers/api.md +187 -0
  231. package/skills/cloudflare-deploy/references/containers/configuration.md +188 -0
  232. package/skills/cloudflare-deploy/references/containers/gotchas.md +178 -0
  233. package/skills/cloudflare-deploy/references/containers/patterns.md +202 -0
  234. package/skills/cloudflare-deploy/references/cron-triggers/README.md +99 -0
  235. package/skills/cloudflare-deploy/references/cron-triggers/api.md +196 -0
  236. package/skills/cloudflare-deploy/references/cron-triggers/configuration.md +180 -0
  237. package/skills/cloudflare-deploy/references/cron-triggers/gotchas.md +199 -0
  238. package/skills/cloudflare-deploy/references/cron-triggers/patterns.md +190 -0
  239. package/skills/cloudflare-deploy/references/d1/README.md +133 -0
  240. package/skills/cloudflare-deploy/references/d1/api.md +196 -0
  241. package/skills/cloudflare-deploy/references/d1/configuration.md +188 -0
  242. package/skills/cloudflare-deploy/references/d1/gotchas.md +98 -0
  243. package/skills/cloudflare-deploy/references/d1/patterns.md +189 -0
  244. package/skills/cloudflare-deploy/references/ddos/README.md +41 -0
  245. package/skills/cloudflare-deploy/references/ddos/api.md +164 -0
  246. package/skills/cloudflare-deploy/references/ddos/configuration.md +93 -0
  247. package/skills/cloudflare-deploy/references/ddos/gotchas.md +107 -0
  248. package/skills/cloudflare-deploy/references/ddos/patterns.md +174 -0
  249. package/skills/cloudflare-deploy/references/do-storage/README.md +75 -0
  250. package/skills/cloudflare-deploy/references/do-storage/api.md +102 -0
  251. package/skills/cloudflare-deploy/references/do-storage/configuration.md +112 -0
  252. package/skills/cloudflare-deploy/references/do-storage/gotchas.md +150 -0
  253. package/skills/cloudflare-deploy/references/do-storage/patterns.md +182 -0
  254. package/skills/cloudflare-deploy/references/do-storage/testing.md +183 -0
  255. package/skills/cloudflare-deploy/references/durable-objects/README.md +185 -0
  256. package/skills/cloudflare-deploy/references/durable-objects/api.md +187 -0
  257. package/skills/cloudflare-deploy/references/durable-objects/configuration.md +160 -0
  258. package/skills/cloudflare-deploy/references/durable-objects/gotchas.md +197 -0
  259. package/skills/cloudflare-deploy/references/durable-objects/patterns.md +201 -0
  260. package/skills/cloudflare-deploy/references/email-routing/README.md +89 -0
  261. package/skills/cloudflare-deploy/references/email-routing/api.md +195 -0
  262. package/skills/cloudflare-deploy/references/email-routing/configuration.md +186 -0
  263. package/skills/cloudflare-deploy/references/email-routing/gotchas.md +196 -0
  264. package/skills/cloudflare-deploy/references/email-routing/patterns.md +229 -0
  265. package/skills/cloudflare-deploy/references/email-workers/README.md +151 -0
  266. package/skills/cloudflare-deploy/references/email-workers/api.md +237 -0
  267. package/skills/cloudflare-deploy/references/email-workers/configuration.md +112 -0
  268. package/skills/cloudflare-deploy/references/email-workers/gotchas.md +125 -0
  269. package/skills/cloudflare-deploy/references/email-workers/patterns.md +102 -0
  270. package/skills/cloudflare-deploy/references/hyperdrive/README.md +82 -0
  271. package/skills/cloudflare-deploy/references/hyperdrive/api.md +143 -0
  272. package/skills/cloudflare-deploy/references/hyperdrive/configuration.md +159 -0
  273. package/skills/cloudflare-deploy/references/hyperdrive/gotchas.md +77 -0
  274. package/skills/cloudflare-deploy/references/hyperdrive/patterns.md +190 -0
  275. package/skills/cloudflare-deploy/references/images/README.md +61 -0
  276. package/skills/cloudflare-deploy/references/images/api.md +96 -0
  277. package/skills/cloudflare-deploy/references/images/configuration.md +211 -0
  278. package/skills/cloudflare-deploy/references/images/gotchas.md +99 -0
  279. package/skills/cloudflare-deploy/references/images/patterns.md +115 -0
  280. package/skills/cloudflare-deploy/references/kv/README.md +89 -0
  281. package/skills/cloudflare-deploy/references/kv/api.md +160 -0
  282. package/skills/cloudflare-deploy/references/kv/configuration.md +144 -0
  283. package/skills/cloudflare-deploy/references/kv/gotchas.md +131 -0
  284. package/skills/cloudflare-deploy/references/kv/patterns.md +196 -0
  285. package/skills/cloudflare-deploy/references/miniflare/README.md +105 -0
  286. package/skills/cloudflare-deploy/references/miniflare/api.md +187 -0
  287. package/skills/cloudflare-deploy/references/miniflare/configuration.md +173 -0
  288. package/skills/cloudflare-deploy/references/miniflare/gotchas.md +160 -0
  289. package/skills/cloudflare-deploy/references/miniflare/patterns.md +181 -0
  290. package/skills/cloudflare-deploy/references/network-interconnect/README.md +99 -0
  291. package/skills/cloudflare-deploy/references/network-interconnect/api.md +199 -0
  292. package/skills/cloudflare-deploy/references/network-interconnect/configuration.md +114 -0
  293. package/skills/cloudflare-deploy/references/network-interconnect/gotchas.md +165 -0
  294. package/skills/cloudflare-deploy/references/network-interconnect/patterns.md +166 -0
  295. package/skills/cloudflare-deploy/references/observability/README.md +87 -0
  296. package/skills/cloudflare-deploy/references/observability/api.md +164 -0
  297. package/skills/cloudflare-deploy/references/observability/configuration.md +169 -0
  298. package/skills/cloudflare-deploy/references/observability/gotchas.md +115 -0
  299. package/skills/cloudflare-deploy/references/observability/patterns.md +105 -0
  300. package/skills/cloudflare-deploy/references/pages/README.md +88 -0
  301. package/skills/cloudflare-deploy/references/pages/api.md +204 -0
  302. package/skills/cloudflare-deploy/references/pages/configuration.md +201 -0
  303. package/skills/cloudflare-deploy/references/pages/gotchas.md +203 -0
  304. package/skills/cloudflare-deploy/references/pages/patterns.md +204 -0
  305. package/skills/cloudflare-deploy/references/pages-functions/README.md +98 -0
  306. package/skills/cloudflare-deploy/references/pages-functions/api.md +143 -0
  307. package/skills/cloudflare-deploy/references/pages-functions/configuration.md +122 -0
  308. package/skills/cloudflare-deploy/references/pages-functions/gotchas.md +94 -0
  309. package/skills/cloudflare-deploy/references/pages-functions/patterns.md +137 -0
  310. package/skills/cloudflare-deploy/references/pipelines/README.md +105 -0
  311. package/skills/cloudflare-deploy/references/pipelines/api.md +208 -0
  312. package/skills/cloudflare-deploy/references/pipelines/configuration.md +98 -0
  313. package/skills/cloudflare-deploy/references/pipelines/gotchas.md +80 -0
  314. package/skills/cloudflare-deploy/references/pipelines/patterns.md +87 -0
  315. package/skills/cloudflare-deploy/references/pulumi/README.md +100 -0
  316. package/skills/cloudflare-deploy/references/pulumi/api.md +200 -0
  317. package/skills/cloudflare-deploy/references/pulumi/configuration.md +198 -0
  318. package/skills/cloudflare-deploy/references/pulumi/gotchas.md +181 -0
  319. package/skills/cloudflare-deploy/references/pulumi/patterns.md +191 -0
  320. package/skills/cloudflare-deploy/references/queues/README.md +96 -0
  321. package/skills/cloudflare-deploy/references/queues/api.md +206 -0
  322. package/skills/cloudflare-deploy/references/queues/configuration.md +144 -0
  323. package/skills/cloudflare-deploy/references/queues/gotchas.md +206 -0
  324. package/skills/cloudflare-deploy/references/queues/patterns.md +220 -0
  325. package/skills/cloudflare-deploy/references/r2/README.md +95 -0
  326. package/skills/cloudflare-deploy/references/r2/api.md +200 -0
  327. package/skills/cloudflare-deploy/references/r2/configuration.md +165 -0
  328. package/skills/cloudflare-deploy/references/r2/gotchas.md +190 -0
  329. package/skills/cloudflare-deploy/references/r2/patterns.md +193 -0
  330. package/skills/cloudflare-deploy/references/r2-data-catalog/README.md +149 -0
  331. package/skills/cloudflare-deploy/references/r2-data-catalog/api.md +199 -0
  332. package/skills/cloudflare-deploy/references/r2-data-catalog/configuration.md +198 -0
  333. package/skills/cloudflare-deploy/references/r2-data-catalog/gotchas.md +170 -0
  334. package/skills/cloudflare-deploy/references/r2-data-catalog/patterns.md +191 -0
  335. package/skills/cloudflare-deploy/references/r2-sql/README.md +128 -0
  336. package/skills/cloudflare-deploy/references/r2-sql/api.md +158 -0
  337. package/skills/cloudflare-deploy/references/r2-sql/configuration.md +147 -0
  338. package/skills/cloudflare-deploy/references/r2-sql/gotchas.md +212 -0
  339. package/skills/cloudflare-deploy/references/r2-sql/patterns.md +222 -0
  340. package/skills/cloudflare-deploy/references/realtime-sfu/README.md +65 -0
  341. package/skills/cloudflare-deploy/references/realtime-sfu/api.md +158 -0
  342. package/skills/cloudflare-deploy/references/realtime-sfu/configuration.md +137 -0
  343. package/skills/cloudflare-deploy/references/realtime-sfu/gotchas.md +133 -0
  344. package/skills/cloudflare-deploy/references/realtime-sfu/patterns.md +174 -0
  345. package/skills/cloudflare-deploy/references/realtimekit/README.md +113 -0
  346. package/skills/cloudflare-deploy/references/realtimekit/api.md +212 -0
  347. package/skills/cloudflare-deploy/references/realtimekit/configuration.md +203 -0
  348. package/skills/cloudflare-deploy/references/realtimekit/gotchas.md +169 -0
  349. package/skills/cloudflare-deploy/references/realtimekit/patterns.md +223 -0
  350. package/skills/cloudflare-deploy/references/sandbox/README.md +96 -0
  351. package/skills/cloudflare-deploy/references/sandbox/api.md +198 -0
  352. package/skills/cloudflare-deploy/references/sandbox/configuration.md +143 -0
  353. package/skills/cloudflare-deploy/references/sandbox/gotchas.md +194 -0
  354. package/skills/cloudflare-deploy/references/sandbox/patterns.md +201 -0
  355. package/skills/cloudflare-deploy/references/secrets-store/README.md +74 -0
  356. package/skills/cloudflare-deploy/references/secrets-store/api.md +200 -0
  357. package/skills/cloudflare-deploy/references/secrets-store/configuration.md +185 -0
  358. package/skills/cloudflare-deploy/references/secrets-store/gotchas.md +97 -0
  359. package/skills/cloudflare-deploy/references/secrets-store/patterns.md +207 -0
  360. package/skills/cloudflare-deploy/references/smart-placement/README.md +138 -0
  361. package/skills/cloudflare-deploy/references/smart-placement/api.md +183 -0
  362. package/skills/cloudflare-deploy/references/smart-placement/configuration.md +196 -0
  363. package/skills/cloudflare-deploy/references/smart-placement/gotchas.md +174 -0
  364. package/skills/cloudflare-deploy/references/smart-placement/patterns.md +183 -0
  365. package/skills/cloudflare-deploy/references/snippets/README.md +68 -0
  366. package/skills/cloudflare-deploy/references/snippets/api.md +198 -0
  367. package/skills/cloudflare-deploy/references/snippets/configuration.md +227 -0
  368. package/skills/cloudflare-deploy/references/snippets/gotchas.md +86 -0
  369. package/skills/cloudflare-deploy/references/snippets/patterns.md +135 -0
  370. package/skills/cloudflare-deploy/references/spectrum/README.md +52 -0
  371. package/skills/cloudflare-deploy/references/spectrum/api.md +181 -0
  372. package/skills/cloudflare-deploy/references/spectrum/configuration.md +194 -0
  373. package/skills/cloudflare-deploy/references/spectrum/gotchas.md +145 -0
  374. package/skills/cloudflare-deploy/references/spectrum/patterns.md +196 -0
  375. package/skills/cloudflare-deploy/references/static-assets/README.md +65 -0
  376. package/skills/cloudflare-deploy/references/static-assets/api.md +199 -0
  377. package/skills/cloudflare-deploy/references/static-assets/configuration.md +186 -0
  378. package/skills/cloudflare-deploy/references/static-assets/gotchas.md +162 -0
  379. package/skills/cloudflare-deploy/references/static-assets/patterns.md +189 -0
  380. package/skills/cloudflare-deploy/references/stream/README.md +114 -0
  381. package/skills/cloudflare-deploy/references/stream/api-live.md +195 -0
  382. package/skills/cloudflare-deploy/references/stream/api.md +199 -0
  383. package/skills/cloudflare-deploy/references/stream/configuration.md +141 -0
  384. package/skills/cloudflare-deploy/references/stream/gotchas.md +130 -0
  385. package/skills/cloudflare-deploy/references/stream/patterns.md +184 -0
  386. package/skills/cloudflare-deploy/references/tail-workers/README.md +89 -0
  387. package/skills/cloudflare-deploy/references/tail-workers/api.md +200 -0
  388. package/skills/cloudflare-deploy/references/tail-workers/configuration.md +176 -0
  389. package/skills/cloudflare-deploy/references/tail-workers/gotchas.md +192 -0
  390. package/skills/cloudflare-deploy/references/tail-workers/patterns.md +180 -0
  391. package/skills/cloudflare-deploy/references/terraform/README.md +102 -0
  392. package/skills/cloudflare-deploy/references/terraform/api.md +178 -0
  393. package/skills/cloudflare-deploy/references/terraform/configuration.md +197 -0
  394. package/skills/cloudflare-deploy/references/terraform/gotchas.md +150 -0
  395. package/skills/cloudflare-deploy/references/terraform/patterns.md +174 -0
  396. package/skills/cloudflare-deploy/references/tunnel/README.md +129 -0
  397. package/skills/cloudflare-deploy/references/tunnel/api.md +193 -0
  398. package/skills/cloudflare-deploy/references/tunnel/configuration.md +157 -0
  399. package/skills/cloudflare-deploy/references/tunnel/gotchas.md +147 -0
  400. package/skills/cloudflare-deploy/references/tunnel/networking.md +168 -0
  401. package/skills/cloudflare-deploy/references/tunnel/patterns.md +192 -0
  402. package/skills/cloudflare-deploy/references/turn/README.md +82 -0
  403. package/skills/cloudflare-deploy/references/turn/api.md +239 -0
  404. package/skills/cloudflare-deploy/references/turn/configuration.md +179 -0
  405. package/skills/cloudflare-deploy/references/turn/gotchas.md +231 -0
  406. package/skills/cloudflare-deploy/references/turn/patterns.md +213 -0
  407. package/skills/cloudflare-deploy/references/turnstile/README.md +99 -0
  408. package/skills/cloudflare-deploy/references/turnstile/api.md +240 -0
  409. package/skills/cloudflare-deploy/references/turnstile/configuration.md +222 -0
  410. package/skills/cloudflare-deploy/references/turnstile/gotchas.md +218 -0
  411. package/skills/cloudflare-deploy/references/turnstile/patterns.md +193 -0
  412. package/skills/cloudflare-deploy/references/vectorize/README.md +133 -0
  413. package/skills/cloudflare-deploy/references/vectorize/api.md +88 -0
  414. package/skills/cloudflare-deploy/references/vectorize/configuration.md +88 -0
  415. package/skills/cloudflare-deploy/references/vectorize/gotchas.md +76 -0
  416. package/skills/cloudflare-deploy/references/vectorize/patterns.md +90 -0
  417. package/skills/cloudflare-deploy/references/waf/README.md +113 -0
  418. package/skills/cloudflare-deploy/references/waf/api.md +202 -0
  419. package/skills/cloudflare-deploy/references/waf/configuration.md +203 -0
  420. package/skills/cloudflare-deploy/references/waf/gotchas.md +204 -0
  421. package/skills/cloudflare-deploy/references/waf/patterns.md +197 -0
  422. package/skills/cloudflare-deploy/references/web-analytics/README.md +140 -0
  423. package/skills/cloudflare-deploy/references/web-analytics/configuration.md +76 -0
  424. package/skills/cloudflare-deploy/references/web-analytics/gotchas.md +82 -0
  425. package/skills/cloudflare-deploy/references/web-analytics/integration.md +60 -0
  426. package/skills/cloudflare-deploy/references/web-analytics/patterns.md +91 -0
  427. package/skills/cloudflare-deploy/references/workerd/README.md +78 -0
  428. package/skills/cloudflare-deploy/references/workerd/api.md +185 -0
  429. package/skills/cloudflare-deploy/references/workerd/configuration.md +183 -0
  430. package/skills/cloudflare-deploy/references/workerd/gotchas.md +139 -0
  431. package/skills/cloudflare-deploy/references/workerd/patterns.md +192 -0
  432. package/skills/cloudflare-deploy/references/workers/README.md +108 -0
  433. package/skills/cloudflare-deploy/references/workers/api.md +195 -0
  434. package/skills/cloudflare-deploy/references/workers/configuration.md +185 -0
  435. package/skills/cloudflare-deploy/references/workers/frameworks.md +197 -0
  436. package/skills/cloudflare-deploy/references/workers/gotchas.md +136 -0
  437. package/skills/cloudflare-deploy/references/workers/patterns.md +198 -0
  438. package/skills/cloudflare-deploy/references/workers-ai/README.md +197 -0
  439. package/skills/cloudflare-deploy/references/workers-ai/api.md +112 -0
  440. package/skills/cloudflare-deploy/references/workers-ai/configuration.md +97 -0
  441. package/skills/cloudflare-deploy/references/workers-ai/gotchas.md +114 -0
  442. package/skills/cloudflare-deploy/references/workers-ai/patterns.md +120 -0
  443. package/skills/cloudflare-deploy/references/workers-for-platforms/README.md +89 -0
  444. package/skills/cloudflare-deploy/references/workers-for-platforms/api.md +196 -0
  445. package/skills/cloudflare-deploy/references/workers-for-platforms/configuration.md +167 -0
  446. package/skills/cloudflare-deploy/references/workers-for-platforms/gotchas.md +134 -0
  447. package/skills/cloudflare-deploy/references/workers-for-platforms/patterns.md +188 -0
  448. package/skills/cloudflare-deploy/references/workers-playground/README.md +127 -0
  449. package/skills/cloudflare-deploy/references/workers-playground/api.md +101 -0
  450. package/skills/cloudflare-deploy/references/workers-playground/configuration.md +163 -0
  451. package/skills/cloudflare-deploy/references/workers-playground/gotchas.md +88 -0
  452. package/skills/cloudflare-deploy/references/workers-playground/patterns.md +132 -0
  453. package/skills/cloudflare-deploy/references/workers-vpc/README.md +127 -0
  454. package/skills/cloudflare-deploy/references/workers-vpc/api.md +202 -0
  455. package/skills/cloudflare-deploy/references/workers-vpc/configuration.md +147 -0
  456. package/skills/cloudflare-deploy/references/workers-vpc/gotchas.md +167 -0
  457. package/skills/cloudflare-deploy/references/workers-vpc/patterns.md +209 -0
  458. package/skills/cloudflare-deploy/references/workflows/README.md +69 -0
  459. package/skills/cloudflare-deploy/references/workflows/api.md +185 -0
  460. package/skills/cloudflare-deploy/references/workflows/configuration.md +151 -0
  461. package/skills/cloudflare-deploy/references/workflows/gotchas.md +97 -0
  462. package/skills/cloudflare-deploy/references/workflows/patterns.md +175 -0
  463. package/skills/cloudflare-deploy/references/wrangler/README.md +141 -0
  464. package/skills/cloudflare-deploy/references/wrangler/api.md +188 -0
  465. package/skills/cloudflare-deploy/references/wrangler/auth.md +73 -0
  466. package/skills/cloudflare-deploy/references/wrangler/configuration.md +197 -0
  467. package/skills/cloudflare-deploy/references/wrangler/gotchas.md +197 -0
  468. package/skills/cloudflare-deploy/references/wrangler/patterns.md +209 -0
  469. package/skills/cloudflare-deploy/references/zaraz/IMPLEMENTATION_SUMMARY.md +121 -0
  470. package/skills/cloudflare-deploy/references/zaraz/README.md +111 -0
  471. package/skills/cloudflare-deploy/references/zaraz/api.md +112 -0
  472. package/skills/cloudflare-deploy/references/zaraz/configuration.md +90 -0
  473. package/skills/cloudflare-deploy/references/zaraz/gotchas.md +81 -0
  474. package/skills/cloudflare-deploy/references/zaraz/patterns.md +74 -0
  475. package/skills/collaboration/brainstorming/SKILL.md +75 -0
  476. package/skills/collaboration/dispatching-parallel-agents/SKILL.md +184 -0
  477. package/skills/collaboration/executing-plans/SKILL.md +78 -0
  478. package/skills/collaboration/finishing-a-development-branch/SKILL.md +202 -0
  479. package/skills/collaboration/phase-prompting/SKILL.md +292 -0
  480. package/skills/collaboration/receiving-code-review/SKILL.md +211 -0
  481. package/skills/collaboration/requesting-code-review/SKILL.md +107 -0
  482. package/skills/collaboration/requesting-code-review/code-reviewer.md +146 -0
  483. package/skills/collaboration/subagent-driven-development/SKILL.md +188 -0
  484. package/skills/collaboration/using-git-worktrees/SKILL.md +215 -0
  485. package/skills/collaboration/writing-plans/SKILL.md +118 -0
  486. package/skills/debugging/defense-in-depth/SKILL.md +130 -0
  487. package/skills/debugging/root-cause-tracing/SKILL.md +177 -0
  488. package/skills/debugging/root-cause-tracing/find-polluter.sh +63 -0
  489. package/skills/debugging/systematic-debugging/SKILL.md +295 -0
  490. package/skills/debugging/systematic-debugging/test-academic.md +14 -0
  491. package/skills/debugging/systematic-debugging/test-pressure-1.md +58 -0
  492. package/skills/debugging/systematic-debugging/test-pressure-2.md +68 -0
  493. package/skills/debugging/systematic-debugging/test-pressure-3.md +69 -0
  494. package/skills/debugging/verification-before-completion/SKILL.md +142 -0
  495. package/skills/develop-web-game/LICENSE.txt +201 -0
  496. package/skills/develop-web-game/SKILL.md +149 -0
  497. package/skills/develop-web-game/agents/openai.yaml +6 -0
  498. package/skills/develop-web-game/assets/game-small.svg +4 -0
  499. package/skills/develop-web-game/assets/game.png +0 -0
  500. package/skills/develop-web-game/references/action_payloads.json +7 -0
  501. package/skills/develop-web-game/scripts/web_game_playwright_client.js +356 -0
  502. package/skills/doc/LICENSE.txt +201 -0
  503. package/skills/doc/SKILL.md +80 -0
  504. package/skills/doc/agents/openai.yaml +6 -0
  505. package/skills/doc/assets/doc-small.svg +3 -0
  506. package/skills/doc/assets/doc.png +0 -0
  507. package/skills/doc/scripts/render_docx.py +296 -0
  508. package/skills/figma/LICENSE.txt +202 -0
  509. package/skills/figma/SKILL.md +42 -0
  510. package/skills/figma/agents/openai.yaml +14 -0
  511. package/skills/figma/assets/figma-small.svg +3 -0
  512. package/skills/figma/assets/figma.png +0 -0
  513. package/skills/figma/assets/icon.svg +28 -0
  514. package/skills/figma/references/figma-mcp-config.md +35 -0
  515. package/skills/figma/references/figma-tools-and-prompts.md +34 -0
  516. package/skills/figma-implement-design/LICENSE.txt +202 -0
  517. package/skills/figma-implement-design/SKILL.md +264 -0
  518. package/skills/figma-implement-design/agents/openai.yaml +14 -0
  519. package/skills/figma-implement-design/assets/figma-small.svg +3 -0
  520. package/skills/figma-implement-design/assets/figma.png +0 -0
  521. package/skills/figma-implement-design/assets/icon.svg +28 -0
  522. package/skills/gh-address-comments/LICENSE.txt +202 -0
  523. package/skills/gh-address-comments/SKILL.md +25 -0
  524. package/skills/gh-address-comments/agents/openai.yaml +6 -0
  525. package/skills/gh-address-comments/assets/github-small.svg +3 -0
  526. package/skills/gh-address-comments/assets/github.png +0 -0
  527. package/skills/gh-address-comments/scripts/fetch_comments.py +237 -0
  528. package/skills/gh-fix-ci/LICENSE.txt +201 -0
  529. package/skills/gh-fix-ci/SKILL.md +69 -0
  530. package/skills/gh-fix-ci/agents/openai.yaml +6 -0
  531. package/skills/gh-fix-ci/assets/github-small.svg +3 -0
  532. package/skills/gh-fix-ci/assets/github.png +0 -0
  533. package/skills/gh-fix-ci/scripts/inspect_pr_checks.py +509 -0
  534. package/skills/goal-driven-project-loop/SKILL.md +217 -0
  535. package/skills/goal-driven-project-loop/references/goal-contract-template.md +42 -0
  536. package/skills/imagegen/LICENSE.txt +201 -0
  537. package/skills/imagegen/SKILL.md +174 -0
  538. package/skills/imagegen/agents/openai.yaml +6 -0
  539. package/skills/imagegen/assets/imagegen-small.svg +5 -0
  540. package/skills/imagegen/assets/imagegen.png +0 -0
  541. package/skills/imagegen/references/cli.md +132 -0
  542. package/skills/imagegen/references/codex-network.md +28 -0
  543. package/skills/imagegen/references/image-api.md +36 -0
  544. package/skills/imagegen/references/prompting.md +81 -0
  545. package/skills/imagegen/references/sample-prompts.md +384 -0
  546. package/skills/imagegen/scripts/image_gen.py +876 -0
  547. package/skills/jupyter-notebook/LICENSE.txt +201 -0
  548. package/skills/jupyter-notebook/SKILL.md +107 -0
  549. package/skills/jupyter-notebook/agents/openai.yaml +6 -0
  550. package/skills/jupyter-notebook/assets/experiment-template.ipynb +110 -0
  551. package/skills/jupyter-notebook/assets/jupyter-small.svg +3 -0
  552. package/skills/jupyter-notebook/assets/jupyter.png +0 -0
  553. package/skills/jupyter-notebook/assets/tutorial-template.ipynb +107 -0
  554. package/skills/jupyter-notebook/references/experiment-patterns.md +10 -0
  555. package/skills/jupyter-notebook/references/notebook-structure.md +17 -0
  556. package/skills/jupyter-notebook/references/quality-checklist.md +11 -0
  557. package/skills/jupyter-notebook/references/tutorial-patterns.md +9 -0
  558. package/skills/jupyter-notebook/scripts/new_notebook.py +130 -0
  559. package/skills/linear/LICENSE.txt +202 -0
  560. package/skills/linear/SKILL.md +87 -0
  561. package/skills/linear/agents/openai.yaml +14 -0
  562. package/skills/linear/assets/linear-small.svg +5 -0
  563. package/skills/linear/assets/linear.png +0 -0
  564. package/skills/netlify-deploy/LICENSE.txt +201 -0
  565. package/skills/netlify-deploy/SKILL.md +247 -0
  566. package/skills/netlify-deploy/agents/openai.yaml +6 -0
  567. package/skills/netlify-deploy/assets/netlify-small.svg +11 -0
  568. package/skills/netlify-deploy/assets/netlify.png +0 -0
  569. package/skills/netlify-deploy/references/cli-commands.md +162 -0
  570. package/skills/netlify-deploy/references/deployment-patterns.md +303 -0
  571. package/skills/netlify-deploy/references/netlify-toml.md +259 -0
  572. package/skills/notion-knowledge-capture/LICENSE.txt +7 -0
  573. package/skills/notion-knowledge-capture/SKILL.md +56 -0
  574. package/skills/notion-knowledge-capture/agents/openai.yaml +14 -0
  575. package/skills/notion-knowledge-capture/assets/notion-small.svg +11 -0
  576. package/skills/notion-knowledge-capture/assets/notion.png +0 -0
  577. package/skills/notion-knowledge-capture/evaluations/README.md +95 -0
  578. package/skills/notion-knowledge-capture/evaluations/conversation-to-wiki.json +31 -0
  579. package/skills/notion-knowledge-capture/evaluations/decision-record.json +31 -0
  580. package/skills/notion-knowledge-capture/examples/conversation-to-faq.md +226 -0
  581. package/skills/notion-knowledge-capture/examples/decision-capture.md +126 -0
  582. package/skills/notion-knowledge-capture/examples/how-to-guide.md +118 -0
  583. package/skills/notion-knowledge-capture/reference/database-best-practices.md +112 -0
  584. package/skills/notion-knowledge-capture/reference/decision-log-database.md +58 -0
  585. package/skills/notion-knowledge-capture/reference/documentation-database.md +93 -0
  586. package/skills/notion-knowledge-capture/reference/faq-database.md +57 -0
  587. package/skills/notion-knowledge-capture/reference/how-to-guide-database.md +38 -0
  588. package/skills/notion-knowledge-capture/reference/learning-database.md +35 -0
  589. package/skills/notion-knowledge-capture/reference/team-wiki-database.md +27 -0
  590. package/skills/notion-meeting-intelligence/LICENSE.txt +7 -0
  591. package/skills/notion-meeting-intelligence/SKILL.md +60 -0
  592. package/skills/notion-meeting-intelligence/agents/openai.yaml +14 -0
  593. package/skills/notion-meeting-intelligence/assets/notion-small.svg +11 -0
  594. package/skills/notion-meeting-intelligence/assets/notion.png +0 -0
  595. package/skills/notion-meeting-intelligence/evaluations/README.md +101 -0
  596. package/skills/notion-meeting-intelligence/evaluations/decision-meeting-prep.json +35 -0
  597. package/skills/notion-meeting-intelligence/evaluations/status-meeting-prep.json +35 -0
  598. package/skills/notion-meeting-intelligence/examples/customer-meeting.md +125 -0
  599. package/skills/notion-meeting-intelligence/examples/executive-review.md +78 -0
  600. package/skills/notion-meeting-intelligence/examples/project-decision.md +431 -0
  601. package/skills/notion-meeting-intelligence/examples/sprint-planning.md +80 -0
  602. package/skills/notion-meeting-intelligence/reference/brainstorming-template.md +81 -0
  603. package/skills/notion-meeting-intelligence/reference/decision-meeting-template.md +94 -0
  604. package/skills/notion-meeting-intelligence/reference/one-on-one-template.md +58 -0
  605. package/skills/notion-meeting-intelligence/reference/retrospective-template.md +58 -0
  606. package/skills/notion-meeting-intelligence/reference/sprint-planning-template.md +68 -0
  607. package/skills/notion-meeting-intelligence/reference/status-update-template.md +74 -0
  608. package/skills/notion-meeting-intelligence/reference/template-selection-guide.md +56 -0
  609. package/skills/notion-research-documentation/LICENSE.txt +7 -0
  610. package/skills/notion-research-documentation/SKILL.md +59 -0
  611. package/skills/notion-research-documentation/agents/openai.yaml +14 -0
  612. package/skills/notion-research-documentation/assets/notion-small.svg +11 -0
  613. package/skills/notion-research-documentation/assets/notion.png +0 -0
  614. package/skills/notion-research-documentation/evaluations/README.md +109 -0
  615. package/skills/notion-research-documentation/evaluations/basic-research.json +28 -0
  616. package/skills/notion-research-documentation/evaluations/research-to-database.json +29 -0
  617. package/skills/notion-research-documentation/examples/competitor-analysis.md +283 -0
  618. package/skills/notion-research-documentation/examples/market-research.md +62 -0
  619. package/skills/notion-research-documentation/examples/technical-investigation.md +233 -0
  620. package/skills/notion-research-documentation/examples/trip-planning.md +128 -0
  621. package/skills/notion-research-documentation/reference/advanced-search.md +212 -0
  622. package/skills/notion-research-documentation/reference/citations.md +190 -0
  623. package/skills/notion-research-documentation/reference/comparison-format.md +37 -0
  624. package/skills/notion-research-documentation/reference/comparison-template.md +44 -0
  625. package/skills/notion-research-documentation/reference/comprehensive-report-format.md +41 -0
  626. package/skills/notion-research-documentation/reference/comprehensive-report-template.md +64 -0
  627. package/skills/notion-research-documentation/reference/format-selection-guide.md +95 -0
  628. package/skills/notion-research-documentation/reference/quick-brief-format.md +37 -0
  629. package/skills/notion-research-documentation/reference/quick-brief-template.md +25 -0
  630. package/skills/notion-research-documentation/reference/research-summary-format.md +33 -0
  631. package/skills/notion-research-documentation/reference/research-summary-template.md +49 -0
  632. package/skills/notion-spec-to-implementation/LICENSE.txt +7 -0
  633. package/skills/notion-spec-to-implementation/SKILL.md +58 -0
  634. package/skills/notion-spec-to-implementation/agents/openai.yaml +14 -0
  635. package/skills/notion-spec-to-implementation/assets/notion-small.svg +11 -0
  636. package/skills/notion-spec-to-implementation/assets/notion.png +0 -0
  637. package/skills/notion-spec-to-implementation/evaluations/README.md +120 -0
  638. package/skills/notion-spec-to-implementation/evaluations/basic-spec-implementation.json +32 -0
  639. package/skills/notion-spec-to-implementation/evaluations/spec-to-tasks.json +35 -0
  640. package/skills/notion-spec-to-implementation/examples/api-feature.md +461 -0
  641. package/skills/notion-spec-to-implementation/examples/database-migration.md +81 -0
  642. package/skills/notion-spec-to-implementation/examples/ui-component.md +68 -0
  643. package/skills/notion-spec-to-implementation/reference/milestone-summary-template.md +27 -0
  644. package/skills/notion-spec-to-implementation/reference/progress-tracking.md +458 -0
  645. package/skills/notion-spec-to-implementation/reference/progress-update-template.md +25 -0
  646. package/skills/notion-spec-to-implementation/reference/quick-implementation-plan.md +26 -0
  647. package/skills/notion-spec-to-implementation/reference/spec-parsing.md +383 -0
  648. package/skills/notion-spec-to-implementation/reference/standard-implementation-plan.md +146 -0
  649. package/skills/notion-spec-to-implementation/reference/task-creation-template.md +34 -0
  650. package/skills/notion-spec-to-implementation/reference/task-creation.md +441 -0
  651. package/skills/openai-docs/LICENSE.txt +201 -0
  652. package/skills/openai-docs/SKILL.md +68 -0
  653. package/skills/openai-docs/agents/openai.yaml +14 -0
  654. package/skills/openai-docs/assets/openai-small.svg +3 -0
  655. package/skills/openai-docs/assets/openai.png +0 -0
  656. package/skills/openai-docs/references/gpt-5p4-prompting-guide.md +433 -0
  657. package/skills/openai-docs/references/latest-model.md +35 -0
  658. package/skills/openai-docs/references/upgrading-to-gpt-5p4.md +164 -0
  659. package/skills/pdf/LICENSE.txt +201 -0
  660. package/skills/pdf/SKILL.md +67 -0
  661. package/skills/pdf/agents/openai.yaml +5 -0
  662. package/skills/pdf/assets/pdf.png +0 -0
  663. package/skills/phase-prompting/SKILL.md +162 -0
  664. package/skills/playwright/LICENSE.txt +201 -0
  665. package/skills/playwright/NOTICE.txt +14 -0
  666. package/skills/playwright/SKILL.md +147 -0
  667. package/skills/playwright/agents/openai.yaml +6 -0
  668. package/skills/playwright/assets/playwright-small.svg +3 -0
  669. package/skills/playwright/assets/playwright.png +0 -0
  670. package/skills/playwright/references/cli.md +116 -0
  671. package/skills/playwright/references/workflows.md +95 -0
  672. package/skills/playwright/scripts/playwright_cli.sh +25 -0
  673. package/skills/playwright-interactive/LICENSE.txt +201 -0
  674. package/skills/playwright-interactive/NOTICE.txt +13 -0
  675. package/skills/playwright-interactive/SKILL.md +693 -0
  676. package/skills/playwright-interactive/agents/openai.yaml +6 -0
  677. package/skills/playwright-interactive/assets/playwright-small.svg +3 -0
  678. package/skills/playwright-interactive/assets/playwright.png +0 -0
  679. package/skills/problem-solving/ABOUT.md +40 -0
  680. package/skills/problem-solving/collision-zone-thinking/SKILL.md +62 -0
  681. package/skills/problem-solving/inversion-exercise/SKILL.md +58 -0
  682. package/skills/problem-solving/meta-pattern-recognition/SKILL.md +54 -0
  683. package/skills/problem-solving/scale-game/SKILL.md +63 -0
  684. package/skills/problem-solving/simplification-cascades/SKILL.md +76 -0
  685. package/skills/problem-solving/when-stuck/SKILL.md +88 -0
  686. package/skills/render-deploy/LICENSE.txt +201 -0
  687. package/skills/render-deploy/SKILL.md +479 -0
  688. package/skills/render-deploy/agents/openai.yaml +14 -0
  689. package/skills/render-deploy/assets/docker.yaml +62 -0
  690. package/skills/render-deploy/assets/go-api.yaml +35 -0
  691. package/skills/render-deploy/assets/nextjs-postgres.yaml +35 -0
  692. package/skills/render-deploy/assets/node-express.yaml +25 -0
  693. package/skills/render-deploy/assets/python-django.yaml +89 -0
  694. package/skills/render-deploy/assets/render-small.svg +3 -0
  695. package/skills/render-deploy/assets/render.png +0 -0
  696. package/skills/render-deploy/assets/static-site.yaml +54 -0
  697. package/skills/render-deploy/references/blueprint-spec.md +718 -0
  698. package/skills/render-deploy/references/codebase-analysis.md +49 -0
  699. package/skills/render-deploy/references/configuration-guide.md +603 -0
  700. package/skills/render-deploy/references/deployment-details.md +224 -0
  701. package/skills/render-deploy/references/direct-creation.md +113 -0
  702. package/skills/render-deploy/references/error-patterns.md +13 -0
  703. package/skills/render-deploy/references/post-deploy-checks.md +36 -0
  704. package/skills/render-deploy/references/runtimes.md +473 -0
  705. package/skills/render-deploy/references/service-types.md +450 -0
  706. package/skills/render-deploy/references/troubleshooting-basics.md +36 -0
  707. package/skills/research/ABOUT.md +20 -0
  708. package/skills/research/tracing-knowledge-lineages/SKILL.md +203 -0
  709. package/skills/round-prompting/SKILL.md +148 -0
  710. package/skills/screenshot/LICENSE.txt +201 -0
  711. package/skills/screenshot/SKILL.md +267 -0
  712. package/skills/screenshot/agents/openai.yaml +6 -0
  713. package/skills/screenshot/assets/screenshot-small.svg +5 -0
  714. package/skills/screenshot/assets/screenshot.png +0 -0
  715. package/skills/screenshot/scripts/ensure_macos_permissions.sh +54 -0
  716. package/skills/screenshot/scripts/macos_display_info.swift +22 -0
  717. package/skills/screenshot/scripts/macos_permissions.swift +40 -0
  718. package/skills/screenshot/scripts/macos_window_info.swift +126 -0
  719. package/skills/screenshot/scripts/take_screenshot.ps1 +163 -0
  720. package/skills/screenshot/scripts/take_screenshot.py +585 -0
  721. package/skills/security-best-practices/LICENSE.txt +201 -0
  722. package/skills/security-best-practices/SKILL.md +86 -0
  723. package/skills/security-best-practices/agents/openai.yaml +4 -0
  724. package/skills/security-best-practices/references/golang-general-backend-security.md +826 -0
  725. package/skills/security-best-practices/references/javascript-express-web-server-security.md +1158 -0
  726. package/skills/security-best-practices/references/javascript-general-web-frontend-security.md +747 -0
  727. package/skills/security-best-practices/references/javascript-jquery-web-frontend-security.md +678 -0
  728. package/skills/security-best-practices/references/javascript-typescript-nextjs-web-server-security.md +1144 -0
  729. package/skills/security-best-practices/references/javascript-typescript-react-web-frontend-security.md +990 -0
  730. package/skills/security-best-practices/references/javascript-typescript-vue-web-frontend-security.md +791 -0
  731. package/skills/security-best-practices/references/python-django-web-server-security.md +882 -0
  732. package/skills/security-best-practices/references/python-fastapi-web-server-security.md +1036 -0
  733. package/skills/security-best-practices/references/python-flask-web-server-security.md +705 -0
  734. package/skills/security-ownership-map/LICENSE.txt +201 -0
  735. package/skills/security-ownership-map/SKILL.md +206 -0
  736. package/skills/security-ownership-map/agents/openai.yaml +4 -0
  737. package/skills/security-ownership-map/references/neo4j-import.md +60 -0
  738. package/skills/security-ownership-map/scripts/build_ownership_map.py +956 -0
  739. package/skills/security-ownership-map/scripts/community_maintainers.py +544 -0
  740. package/skills/security-ownership-map/scripts/query_ownership.py +483 -0
  741. package/skills/security-ownership-map/scripts/run_ownership_map.py +200 -0
  742. package/skills/security-threat-model/LICENSE.txt +201 -0
  743. package/skills/security-threat-model/SKILL.md +81 -0
  744. package/skills/security-threat-model/agents/openai.yaml +4 -0
  745. package/skills/security-threat-model/references/prompt-template.md +255 -0
  746. package/skills/security-threat-model/references/security-controls-and-assets.md +32 -0
  747. package/skills/sentry/LICENSE.txt +201 -0
  748. package/skills/sentry/SKILL.md +123 -0
  749. package/skills/sentry/agents/openai.yaml +6 -0
  750. package/skills/sentry/assets/sentry-small.svg +3 -0
  751. package/skills/sentry/assets/sentry.png +0 -0
  752. package/skills/sentry/scripts/sentry_api.py +238 -0
  753. package/skills/slides/LICENSE.txt +201 -0
  754. package/skills/slides/SKILL.md +71 -0
  755. package/skills/slides/agents/openai.yaml +6 -0
  756. package/skills/slides/assets/pptxgenjs_helpers/code.js +104 -0
  757. package/skills/slides/assets/pptxgenjs_helpers/image.js +333 -0
  758. package/skills/slides/assets/pptxgenjs_helpers/index.js +33 -0
  759. package/skills/slides/assets/pptxgenjs_helpers/latex.js +51 -0
  760. package/skills/slides/assets/pptxgenjs_helpers/layout.js +643 -0
  761. package/skills/slides/assets/pptxgenjs_helpers/layout_builders.js +358 -0
  762. package/skills/slides/assets/pptxgenjs_helpers/svg.js +36 -0
  763. package/skills/slides/assets/pptxgenjs_helpers/text.js +789 -0
  764. package/skills/slides/assets/pptxgenjs_helpers/util.js +24 -0
  765. package/skills/slides/assets/slides-small.svg +3 -0
  766. package/skills/slides/assets/slides.png +0 -0
  767. package/skills/slides/references/pptxgenjs-helpers.md +61 -0
  768. package/skills/slides/scripts/create_montage.py +300 -0
  769. package/skills/slides/scripts/detect_font.py +873 -0
  770. package/skills/slides/scripts/ensure_raster_image.py +202 -0
  771. package/skills/slides/scripts/render_slides.py +273 -0
  772. package/skills/slides/scripts/slides_test.py +201 -0
  773. package/skills/sora/LICENSE.txt +201 -0
  774. package/skills/sora/SKILL.md +153 -0
  775. package/skills/sora/agents/openai.yaml +6 -0
  776. package/skills/sora/assets/sora-small.svg +4 -0
  777. package/skills/sora/assets/sora.png +0 -0
  778. package/skills/sora/references/cinematic-shots.md +53 -0
  779. package/skills/sora/references/cli.md +248 -0
  780. package/skills/sora/references/codex-network.md +28 -0
  781. package/skills/sora/references/prompting.md +137 -0
  782. package/skills/sora/references/sample-prompts.md +95 -0
  783. package/skills/sora/references/social-ads.md +42 -0
  784. package/skills/sora/references/troubleshooting.md +58 -0
  785. package/skills/sora/references/video-api.md +45 -0
  786. package/skills/sora/scripts/sora.py +970 -0
  787. package/skills/speech/LICENSE.txt +201 -0
  788. package/skills/speech/SKILL.md +144 -0
  789. package/skills/speech/agents/openai.yaml +6 -0
  790. package/skills/speech/assets/speech-small.svg +3 -0
  791. package/skills/speech/assets/speech.png +0 -0
  792. package/skills/speech/references/accessibility.md +32 -0
  793. package/skills/speech/references/audio-api.md +31 -0
  794. package/skills/speech/references/cli.md +99 -0
  795. package/skills/speech/references/codex-network.md +28 -0
  796. package/skills/speech/references/ivr.md +32 -0
  797. package/skills/speech/references/narration.md +31 -0
  798. package/skills/speech/references/prompting.md +38 -0
  799. package/skills/speech/references/sample-prompts.md +44 -0
  800. package/skills/speech/references/voice-directions.md +80 -0
  801. package/skills/speech/references/voiceover.md +31 -0
  802. package/skills/speech/scripts/text_to_speech.py +528 -0
  803. package/skills/spreadsheet/LICENSE.txt +201 -0
  804. package/skills/spreadsheet/SKILL.md +145 -0
  805. package/skills/spreadsheet/agents/openai.yaml +6 -0
  806. package/skills/spreadsheet/assets/spreadsheet-small.svg +3 -0
  807. package/skills/spreadsheet/assets/spreadsheet.png +0 -0
  808. package/skills/spreadsheet/references/examples/openpyxl/create_basic_spreadsheet.py +51 -0
  809. package/skills/spreadsheet/references/examples/openpyxl/create_spreadsheet_with_styling.py +96 -0
  810. package/skills/spreadsheet/references/examples/openpyxl/read_existing_spreadsheet.py +59 -0
  811. package/skills/spreadsheet/references/examples/openpyxl/styling_spreadsheet.py +79 -0
  812. package/skills/testing/condition-based-waiting/SKILL.md +123 -0
  813. package/skills/testing/condition-based-waiting/example.ts +158 -0
  814. package/skills/testing/test-driven-development/SKILL.md +367 -0
  815. package/skills/testing/testing-anti-patterns/SKILL.md +304 -0
  816. package/skills/transcribe/LICENSE.txt +201 -0
  817. package/skills/transcribe/SKILL.md +81 -0
  818. package/skills/transcribe/agents/openai.yaml +6 -0
  819. package/skills/transcribe/assets/transcribe-small.svg +3 -0
  820. package/skills/transcribe/assets/transcribe.png +0 -0
  821. package/skills/transcribe/references/api.md +8 -0
  822. package/skills/transcribe/scripts/transcribe_diarize.py +276 -0
  823. package/skills/using-skills/SKILL.md +102 -0
  824. package/skills/using-skills/find-skills +107 -0
  825. package/skills/using-skills/skill-run +44 -0
  826. package/skills/vercel-deploy/LICENSE.txt +21 -0
  827. package/skills/vercel-deploy/SKILL.md +77 -0
  828. package/skills/vercel-deploy/agents/openai.yaml +6 -0
  829. package/skills/vercel-deploy/assets/vercel-small.svg +5 -0
  830. package/skills/vercel-deploy/assets/vercel.png +0 -0
  831. package/skills/vercel-deploy/scripts/deploy.sh +301 -0
  832. package/skills/winui-app/LICENSE.txt +202 -0
  833. package/skills/winui-app/SKILL.md +94 -0
  834. package/skills/winui-app/agents/openai.yaml +5 -0
  835. package/skills/winui-app/assets/winui.png +0 -0
  836. package/skills/winui-app/config.yaml +50 -0
  837. package/skills/winui-app/references/_sections.md +96 -0
  838. package/skills/winui-app/references/accessibility-input-and-localization.md +51 -0
  839. package/skills/winui-app/references/build-run-and-launch-verification.md +72 -0
  840. package/skills/winui-app/references/community-toolkit-controls-and-helpers.md +57 -0
  841. package/skills/winui-app/references/controls-layout-and-adaptive-ui.md +84 -0
  842. package/skills/winui-app/references/foundation-environment-audit-and-remediation.md +82 -0
  843. package/skills/winui-app/references/foundation-setup-and-project-selection.md +67 -0
  844. package/skills/winui-app/references/foundation-template-first-recovery.md +62 -0
  845. package/skills/winui-app/references/foundation-winui-app-structure.md +62 -0
  846. package/skills/winui-app/references/motion-animations-and-polish.md +45 -0
  847. package/skills/winui-app/references/performance-diagnostics-and-responsiveness.md +46 -0
  848. package/skills/winui-app/references/sample-source-map.md +37 -0
  849. package/skills/winui-app/references/shell-navigation-and-windowing.md +67 -0
  850. package/skills/winui-app/references/styling-theming-materials-and-icons.md +71 -0
  851. package/skills/winui-app/references/testing-debugging-and-review-checklists.md +77 -0
  852. package/skills/winui-app/references/windows-app-sdk-lifecycle-notifications-and-deployment.md +52 -0
  853. package/skills/yeet/LICENSE.txt +201 -0
  854. package/skills/yeet/SKILL.md +28 -0
  855. package/skills/yeet/agents/openai.yaml +6 -0
  856. package/skills/yeet/assets/yeet-small.svg +3 -0
  857. package/skills/yeet/assets/yeet.png +0 -0
  858. package/dist/src/agents/definitions.d.ts +0 -16
  859. package/dist/src/agents/definitions.d.ts.map +0 -1
  860. package/dist/src/agents/definitions.js +0 -148
  861. package/dist/src/agents/definitions.js.map +0 -1
@@ -0,0 +1,956 @@
1
+ #!/usr/bin/env python3
2
+ """Build a security ownership map from git history."""
3
+
4
+ from __future__ import annotations
5
+
6
+ import argparse
7
+ import csv
8
+ import datetime as dt
9
+ import fnmatch
10
+ import json
11
+ import math
12
+ import os
13
+ import re
14
+ import subprocess
15
+ import sys
16
+ from collections import defaultdict
17
+ from pathlib import Path
18
+ from typing import Iterable
19
+
20
+ DEFAULT_SENSITIVE_RULES: list[tuple[str, str, float]] = [
21
+ ("**/auth/**", "auth", 1.0),
22
+ ("**/oauth/**", "auth", 1.0),
23
+ ("**/rbac/**", "auth", 1.0),
24
+ ("**/session/**", "auth", 1.0),
25
+ ("**/token/**", "auth", 1.0),
26
+ ("**/crypto/**", "crypto", 1.0),
27
+ ("**/tls/**", "crypto", 1.0),
28
+ ("**/ssl/**", "crypto", 1.0),
29
+ ("**/secrets/**", "secrets", 1.0),
30
+ ("**/keys/**", "secrets", 1.0),
31
+ ("**/*.pem", "secrets", 1.0),
32
+ ("**/*.key", "secrets", 1.0),
33
+ ("**/*.p12", "secrets", 1.0),
34
+ ("**/*.pfx", "secrets", 1.0),
35
+ ("**/iam/**", "auth", 1.0),
36
+ ("**/sso/**", "auth", 1.0),
37
+ ]
38
+
39
+ DEFAULT_AUTHOR_EXCLUDE_REGEXES = [
40
+ "dependabot",
41
+ ]
42
+
43
+ DEFAULT_COCHANGE_EXCLUDES = [
44
+ "**/Cargo.lock",
45
+ "**/Cargo.toml",
46
+ "**/package-lock.json",
47
+ "**/yarn.lock",
48
+ "**/pnpm-lock.yaml",
49
+ "**/go.sum",
50
+ "**/go.mod",
51
+ "**/Gemfile.lock",
52
+ "**/Pipfile.lock",
53
+ "**/poetry.lock",
54
+ "**/composer.lock",
55
+ "**/.github/**",
56
+ "**/.gitignore",
57
+ "**/.gitattributes",
58
+ "**/.gitmodules",
59
+ "**/.editorconfig",
60
+ "**/.vscode/**",
61
+ "**/.idea/**",
62
+ ]
63
+
64
+
65
+ def parse_args() -> argparse.Namespace:
66
+ parser = argparse.ArgumentParser(
67
+ description="Build ownership graphs and security ownership summaries from git history."
68
+ )
69
+ parser.add_argument("--repo", default=".", help="Path to the git repo (default: .)")
70
+ parser.add_argument(
71
+ "--out",
72
+ default="ownership-map-out",
73
+ help="Output directory for graph artifacts",
74
+ )
75
+ parser.add_argument("--since", default=None, help="Limit git log to commits since date")
76
+ parser.add_argument("--until", default=None, help="Limit git log to commits until date")
77
+ parser.add_argument(
78
+ "--identity",
79
+ choices=("author", "committer"),
80
+ default="author",
81
+ help="Identity to attribute touches to",
82
+ )
83
+ parser.add_argument(
84
+ "--date-field",
85
+ choices=("author", "committer"),
86
+ default="author",
87
+ help="Date field to use for recency and bucketing",
88
+ )
89
+ parser.add_argument(
90
+ "--include-merges",
91
+ action="store_true",
92
+ help="Include merge commits (excluded by default)",
93
+ )
94
+ parser.add_argument(
95
+ "--half-life-days",
96
+ type=float,
97
+ default=180.0,
98
+ help="Half life for recency weighting",
99
+ )
100
+ parser.add_argument(
101
+ "--sensitive-config",
102
+ default=None,
103
+ help="CSV file with pattern,tag,weight for sensitive paths",
104
+ )
105
+ parser.add_argument(
106
+ "--owner-threshold",
107
+ type=float,
108
+ default=0.5,
109
+ help="Share threshold for hidden owner detection",
110
+ )
111
+ parser.add_argument(
112
+ "--bus-factor-threshold",
113
+ type=int,
114
+ default=1,
115
+ help="Bus factor threshold for hotspots",
116
+ )
117
+ parser.add_argument(
118
+ "--stale-days",
119
+ type=int,
120
+ default=365,
121
+ help="Days since last touch to consider stale",
122
+ )
123
+ parser.add_argument(
124
+ "--min-touches",
125
+ type=int,
126
+ default=1,
127
+ help="Minimum touches to keep an edge",
128
+ )
129
+ parser.add_argument(
130
+ "--emit-commits",
131
+ action="store_true",
132
+ help="Write commit list to commits.jsonl",
133
+ )
134
+ parser.add_argument(
135
+ "--author-exclude-regex",
136
+ action="append",
137
+ default=[],
138
+ help="Regex for author name/email to exclude (repeatable)",
139
+ )
140
+ parser.add_argument(
141
+ "--no-default-author-excludes",
142
+ action="store_true",
143
+ help="Disable default author excludes (dependabot)",
144
+ )
145
+ parser.add_argument(
146
+ "--no-cochange",
147
+ action="store_true",
148
+ help="Disable co-change graph output",
149
+ )
150
+ parser.add_argument(
151
+ "--cochange-max-files",
152
+ type=int,
153
+ default=50,
154
+ help="Ignore commits touching more than this many files for co-change graph",
155
+ )
156
+ parser.add_argument(
157
+ "--cochange-min-count",
158
+ type=int,
159
+ default=2,
160
+ help="Minimum co-change count to keep file-file edge",
161
+ )
162
+ parser.add_argument(
163
+ "--cochange-min-jaccard",
164
+ type=float,
165
+ default=0.05,
166
+ help="Minimum Jaccard similarity to keep file-file edge",
167
+ )
168
+ parser.add_argument(
169
+ "--cochange-exclude",
170
+ action="append",
171
+ default=[],
172
+ help="Glob to exclude from co-change graph (repeatable)",
173
+ )
174
+ parser.add_argument(
175
+ "--no-default-cochange-excludes",
176
+ action="store_true",
177
+ help="Disable default co-change excludes (lockfiles, .github, editor config)",
178
+ )
179
+ parser.add_argument(
180
+ "--no-communities",
181
+ dest="communities",
182
+ action="store_false",
183
+ help="Disable community detection (enabled by default, requires networkx)",
184
+ )
185
+ parser.add_argument(
186
+ "--graphml",
187
+ action="store_true",
188
+ help="Emit ownership.graphml (requires networkx)",
189
+ )
190
+ parser.add_argument(
191
+ "--max-community-files",
192
+ type=int,
193
+ default=50,
194
+ help="Max files listed per community",
195
+ )
196
+ parser.add_argument(
197
+ "--community-top-owners",
198
+ type=int,
199
+ default=5,
200
+ help="Top maintainers saved per community",
201
+ )
202
+ parser.set_defaults(communities=True)
203
+ return parser.parse_args()
204
+
205
+
206
+ def load_sensitive_rules(path: str | None) -> list[tuple[str, str, float]]:
207
+ if not path:
208
+ return list(DEFAULT_SENSITIVE_RULES)
209
+ rules: list[tuple[str, str, float]] = []
210
+ with open(path, "r", encoding="utf-8") as handle:
211
+ for raw in handle:
212
+ line = raw.strip()
213
+ if not line or line.startswith("#"):
214
+ continue
215
+ parts = [part.strip() for part in line.split(",")]
216
+ if not parts:
217
+ continue
218
+ pattern = parts[0]
219
+ tag = parts[1] if len(parts) > 1 and parts[1] else "sensitive"
220
+ weight = float(parts[2]) if len(parts) > 2 and parts[2] else 1.0
221
+ rules.append((pattern, tag, weight))
222
+ return rules
223
+
224
+
225
+ def parse_date(value: str) -> dt.datetime:
226
+ parsed = dt.datetime.fromisoformat(value)
227
+ if parsed.tzinfo is None:
228
+ parsed = parsed.replace(tzinfo=dt.timezone.utc)
229
+ return parsed
230
+
231
+
232
+ def offset_minutes(timestamp: dt.datetime) -> int | None:
233
+ offset = timestamp.utcoffset()
234
+ if offset is None:
235
+ return None
236
+ return int(offset.total_seconds() / 60)
237
+
238
+
239
+ def format_offset(minutes: int) -> str:
240
+ sign = "+" if minutes >= 0 else "-"
241
+ minutes = abs(minutes)
242
+ return f"{sign}{minutes // 60:02d}:{minutes % 60:02d}"
243
+
244
+
245
+ def recency_weighted(now: dt.datetime, when: dt.datetime, half_life_days: float) -> float:
246
+ if half_life_days <= 0:
247
+ return 1.0
248
+ age_days = max(0.0, (now - when).total_seconds() / 86400.0)
249
+ return math.exp(-math.log(2) * age_days / half_life_days)
250
+
251
+
252
+ def match_sensitive(path: str, rules: Iterable[tuple[str, str, float]]) -> dict[str, float]:
253
+ tags: dict[str, float] = defaultdict(float)
254
+ posix = path.replace("\\", "/")
255
+ for pattern, tag, weight in rules:
256
+ patterns = [pattern]
257
+ if pattern.startswith("**/"):
258
+ patterns.append(pattern[3:])
259
+ for candidate in patterns:
260
+ if fnmatch.fnmatchcase(posix, candidate):
261
+ tags[tag] += weight
262
+ break
263
+ return tags
264
+
265
+
266
+ def matches_glob(path: str, pattern: str) -> bool:
267
+ posix = path.replace("\\", "/")
268
+ patterns = [pattern]
269
+ if pattern.startswith("**/"):
270
+ patterns.append(pattern[3:])
271
+ return any(fnmatch.fnmatchcase(posix, candidate) for candidate in patterns)
272
+
273
+
274
+ def is_excluded(path: str, patterns: Iterable[str]) -> bool:
275
+ return any(matches_glob(path, pattern) for pattern in patterns)
276
+
277
+
278
+ def author_excluded(name: str, email: str, patterns: Iterable[re.Pattern[str]]) -> bool:
279
+ if not patterns:
280
+ return False
281
+ haystack = f"{name} {email}".strip()
282
+ return any(pattern.search(haystack) for pattern in patterns)
283
+
284
+
285
+ def compute_community_owners(
286
+ community_files: Iterable[str],
287
+ people: dict[str, dict[str, object]],
288
+ file_people_touches: dict[str, dict[str, int]],
289
+ file_people_recency: dict[str, dict[str, float]],
290
+ file_people_sensitive: dict[str, dict[str, float]],
291
+ top_n: int,
292
+ ) -> dict[str, object]:
293
+ touches_by_person: dict[str, int] = defaultdict(int)
294
+ recency_by_person: dict[str, float] = defaultdict(float)
295
+ sensitive_by_person: dict[str, float] = defaultdict(float)
296
+
297
+ for path in community_files:
298
+ for person, touches in file_people_touches.get(path, {}).items():
299
+ touches_by_person[person] += touches
300
+ for person, recency in file_people_recency.get(path, {}).items():
301
+ recency_by_person[person] += recency
302
+ for person, weight in file_people_sensitive.get(path, {}).items():
303
+ sensitive_by_person[person] += weight
304
+
305
+ total_touches = sum(touches_by_person.values())
306
+ total_recency = sum(recency_by_person.values())
307
+ total_sensitive = sum(sensitive_by_person.values())
308
+
309
+ ranked = sorted(touches_by_person.items(), key=lambda item: item[1], reverse=True)
310
+ owners = []
311
+ for person_id, touches in ranked[:top_n]:
312
+ recency = recency_by_person.get(person_id, 0.0)
313
+ sensitive = sensitive_by_person.get(person_id, 0.0)
314
+ owners.append(
315
+ {
316
+ "person_id": person_id,
317
+ "name": people.get(person_id, {}).get("name", person_id),
318
+ "touches": touches,
319
+ "touch_share": round(touches / total_touches, 4) if total_touches else 0.0,
320
+ "recency_share": round(recency / total_recency, 4) if total_recency else 0.0,
321
+ "sensitive_share": round(sensitive / total_sensitive, 4)
322
+ if total_sensitive
323
+ else 0.0,
324
+ "primary_tz_offset": people.get(person_id, {}).get("primary_tz_offset", ""),
325
+ }
326
+ )
327
+
328
+ return {
329
+ "bus_factor": len(touches_by_person),
330
+ "owner_count": len(touches_by_person),
331
+ "totals": {
332
+ "touches": total_touches,
333
+ "recency_weight": round(total_recency, 6),
334
+ "sensitive_weight": round(total_sensitive, 2),
335
+ },
336
+ "top_maintainers": owners,
337
+ }
338
+
339
+
340
+ def run_git_log(
341
+ repo: str, since: str | None, until: str | None, include_merges: bool
342
+ ) -> Iterable[list[str]]:
343
+ cmd = [
344
+ "git",
345
+ "-C",
346
+ repo,
347
+ "log",
348
+ "--name-only",
349
+ "--no-renames",
350
+ "--date=iso-strict",
351
+ "--format=---%n%H%n%P%n%an%n%ae%n%ad%n%cn%n%ce%n%cd",
352
+ ]
353
+ if not include_merges:
354
+ cmd.append("--no-merges")
355
+ if since:
356
+ cmd.extend(["--since", since])
357
+ if until:
358
+ cmd.extend(["--until", until])
359
+
360
+ proc = subprocess.Popen(
361
+ cmd,
362
+ stdout=subprocess.PIPE,
363
+ stderr=subprocess.PIPE,
364
+ text=True,
365
+ )
366
+ assert proc.stdout is not None
367
+
368
+ batch: list[str] = []
369
+ for line in proc.stdout:
370
+ batch.append(line.rstrip("\n"))
371
+ if line.rstrip("\n") == "---" and len(batch) > 1:
372
+ yield batch[:-1]
373
+ batch = ["---"]
374
+
375
+ if batch:
376
+ yield batch
377
+
378
+ stderr = proc.stderr.read() if proc.stderr else ""
379
+ exit_code = proc.wait()
380
+ if exit_code != 0:
381
+ raise RuntimeError(stderr.strip() or "git log failed")
382
+
383
+
384
+ def iter_commits(lines: Iterable[list[str]]) -> Iterable[tuple[dict[str, object], list[str]]]:
385
+ for chunk in lines:
386
+ if not chunk or chunk[0] != "---":
387
+ continue
388
+ header = chunk[1:9]
389
+ if len(header) < 8:
390
+ continue
391
+ parents = [entry for entry in header[1].split(" ") if entry]
392
+ commit = {
393
+ "hash": header[0],
394
+ "parents": parents,
395
+ "is_merge": len(parents) > 1,
396
+ "author_name": header[2],
397
+ "author_email": header[3],
398
+ "author_date": header[4],
399
+ "committer_name": header[5],
400
+ "committer_email": header[6],
401
+ "committer_date": header[7],
402
+ }
403
+ files = [line for line in chunk[9:] if line.strip()]
404
+ yield commit, files
405
+
406
+
407
+ def ensure_out_dir(path: str) -> Path:
408
+ out_dir = Path(path)
409
+ out_dir.mkdir(parents=True, exist_ok=True)
410
+ return out_dir
411
+
412
+
413
+ def write_csv(path: Path, header: list[str], rows: Iterable[list[str]]) -> None:
414
+ with path.open("w", encoding="utf-8", newline="") as handle:
415
+ writer = csv.writer(handle)
416
+ writer.writerow(header)
417
+ for row in rows:
418
+ writer.writerow(row)
419
+
420
+
421
+ def build_ownership_map(args: argparse.Namespace) -> Path:
422
+ now = dt.datetime.now(dt.timezone.utc)
423
+ rules = load_sensitive_rules(args.sensitive_config)
424
+ out_dir = ensure_out_dir(args.out)
425
+
426
+ people: dict[str, dict[str, object]] = {}
427
+ files: dict[str, dict[str, object]] = {}
428
+ edges: dict[tuple[str, str], dict[str, object]] = {}
429
+ file_people_touches: dict[str, dict[str, int]] = defaultdict(lambda: defaultdict(int))
430
+ file_people_recency: dict[str, dict[str, float]] = defaultdict(lambda: defaultdict(float))
431
+ file_people_sensitive: dict[str, dict[str, float]] = defaultdict(lambda: defaultdict(float))
432
+ tag_totals: dict[str, float] = defaultdict(float)
433
+ tag_person_totals: dict[str, dict[str, float]] = defaultdict(lambda: defaultdict(float))
434
+ person_timezone_counts: dict[str, dict[int, int]] = defaultdict(lambda: defaultdict(int))
435
+ cochange_counts: dict[tuple[str, str], int] = defaultdict(int)
436
+ cochange_file_commits: dict[str, int] = defaultdict(int)
437
+ cochange_commits_used = 0
438
+ cochange_commits_skipped = 0
439
+ cochange_commits_filtered = 0
440
+ cochange_files_excluded = 0
441
+
442
+ commits_path = out_dir / "commits.jsonl"
443
+ commit_handle = None
444
+ if args.emit_commits:
445
+ commit_handle = commits_path.open("w", encoding="utf-8")
446
+
447
+ total_commits_seen = 0
448
+ total_commits_included = 0
449
+ commits_excluded_identities = 0
450
+ commits_excluded_merges = 0
451
+ total_edges = 0
452
+
453
+ author_exclude_regexes = []
454
+ if not args.no_default_author_excludes:
455
+ author_exclude_regexes.extend(DEFAULT_AUTHOR_EXCLUDE_REGEXES)
456
+ author_exclude_regexes.extend(args.author_exclude_regex)
457
+ author_exclude_patterns = [
458
+ re.compile(pattern, re.IGNORECASE) for pattern in author_exclude_regexes
459
+ ]
460
+
461
+ cochange_excludes = []
462
+ if not args.no_default_cochange_excludes:
463
+ cochange_excludes.extend(DEFAULT_COCHANGE_EXCLUDES)
464
+ cochange_excludes.extend(args.cochange_exclude)
465
+
466
+ log_lines = run_git_log(args.repo, args.since, args.until, args.include_merges)
467
+ for commit, touched_files in iter_commits(log_lines):
468
+ total_commits_seen += 1
469
+
470
+ if commit.get("is_merge") and not args.include_merges:
471
+ commits_excluded_merges += 1
472
+ continue
473
+
474
+ identity_name = commit.get(f"{args.identity}_name", "")
475
+ identity_email = commit.get(f"{args.identity}_email", "")
476
+ if author_excluded(
477
+ identity_name,
478
+ identity_email,
479
+ author_exclude_patterns,
480
+ ):
481
+ commits_excluded_identities += 1
482
+ continue
483
+
484
+ if not touched_files:
485
+ continue
486
+
487
+ total_commits_included += 1
488
+ if commit_handle:
489
+ commit_handle.write(json.dumps({**commit, "files": touched_files}) + "\n")
490
+
491
+ identity_name = commit.get(f"{args.identity}_name", "")
492
+ identity_email = commit.get(f"{args.identity}_email", "") or identity_name
493
+ commit_date = parse_date(commit.get(f"{args.date_field}_date", ""))
494
+ recency = recency_weighted(now, commit_date, args.half_life_days)
495
+ tz_minutes = offset_minutes(commit_date)
496
+ if tz_minutes is not None:
497
+ person_timezone_counts[identity_email][tz_minutes] += 1
498
+ unique_files = sorted(set(touched_files))
499
+ if not args.no_cochange and len(unique_files) > 1:
500
+ if len(unique_files) > args.cochange_max_files:
501
+ cochange_commits_skipped += 1
502
+ else:
503
+ filtered_files = [
504
+ path for path in unique_files if not is_excluded(path, cochange_excludes)
505
+ ]
506
+ excluded = len(unique_files) - len(filtered_files)
507
+ if excluded:
508
+ cochange_files_excluded += excluded
509
+ if len(filtered_files) < 2:
510
+ cochange_commits_filtered += 1
511
+ if filtered_files:
512
+ for path in filtered_files:
513
+ cochange_file_commits[path] += 1
514
+ if len(filtered_files) >= 2:
515
+ cochange_commits_used += 1
516
+ for idx, path in enumerate(filtered_files):
517
+ for other in filtered_files[idx + 1 :]:
518
+ cochange_counts[(path, other)] += 1
519
+
520
+ person = people.setdefault(
521
+ identity_email,
522
+ {
523
+ "name": identity_name,
524
+ "email": identity_email,
525
+ "first_seen": commit_date,
526
+ "last_seen": commit_date,
527
+ "commit_count": 0,
528
+ "touches": 0,
529
+ "sensitive_touches": 0.0,
530
+ },
531
+ )
532
+ person["commit_count"] = int(person["commit_count"]) + 1
533
+ person["first_seen"] = min(person["first_seen"], commit_date)
534
+ person["last_seen"] = max(person["last_seen"], commit_date)
535
+
536
+ for path in touched_files:
537
+ file_entry = files.setdefault(
538
+ path,
539
+ {
540
+ "path": path,
541
+ "first_seen": commit_date,
542
+ "last_seen": commit_date,
543
+ "commit_count": 0,
544
+ "touches": 0,
545
+ "authors": set(),
546
+ "sensitive_tags": {},
547
+ },
548
+ )
549
+ file_entry["commit_count"] = int(file_entry["commit_count"]) + 1
550
+ file_entry["first_seen"] = min(file_entry["first_seen"], commit_date)
551
+ file_entry["last_seen"] = max(file_entry["last_seen"], commit_date)
552
+ file_entry["touches"] = int(file_entry["touches"]) + 1
553
+ file_entry["authors"].add(identity_email)
554
+
555
+ edge = edges.setdefault(
556
+ (identity_email, path),
557
+ {
558
+ "touches": 0,
559
+ "first_seen": commit_date,
560
+ "last_seen": commit_date,
561
+ "recency_weight": 0.0,
562
+ "sensitive_weight": 0.0,
563
+ },
564
+ )
565
+ edge["touches"] = int(edge["touches"]) + 1
566
+ edge["first_seen"] = min(edge["first_seen"], commit_date)
567
+ edge["last_seen"] = max(edge["last_seen"], commit_date)
568
+ edge["recency_weight"] = float(edge["recency_weight"]) + recency
569
+
570
+ tags = match_sensitive(path, rules)
571
+ if tags:
572
+ file_entry["sensitive_tags"] = tags
573
+ sensitive_weight = sum(tags.values())
574
+ edge["sensitive_weight"] = float(edge["sensitive_weight"]) + sensitive_weight
575
+ person["sensitive_touches"] = float(person["sensitive_touches"]) + sensitive_weight
576
+ file_people_sensitive[path][identity_email] += sensitive_weight
577
+ for tag, weight in tags.items():
578
+ tag_totals[tag] += weight
579
+ tag_person_totals[tag][identity_email] += weight
580
+
581
+ person["touches"] = int(person["touches"]) + 1
582
+ file_people_touches[path][identity_email] += 1
583
+ file_people_recency[path][identity_email] += recency
584
+ total_edges += 1
585
+
586
+ if commit_handle:
587
+ commit_handle.close()
588
+
589
+ people_rows = []
590
+ for email, person in sorted(people.items()):
591
+ tz_counts = person_timezone_counts.get(email, {})
592
+ primary_tz_offset = ""
593
+ primary_tz_minutes = ""
594
+ timezone_offsets = ""
595
+ if tz_counts:
596
+ primary_tz_minutes_value = max(tz_counts.items(), key=lambda item: (item[1], item[0]))[
597
+ 0
598
+ ]
599
+ primary_tz_offset = format_offset(primary_tz_minutes_value)
600
+ primary_tz_minutes = str(primary_tz_minutes_value)
601
+ timezone_offsets = ";".join(
602
+ f"{format_offset(minutes)}:{count}"
603
+ for minutes, count in sorted(tz_counts.items(), key=lambda item: item[0])
604
+ )
605
+ person["primary_tz_offset"] = primary_tz_offset
606
+ people_rows.append(
607
+ [
608
+ email,
609
+ str(person["name"]),
610
+ email,
611
+ person["first_seen"].isoformat(),
612
+ person["last_seen"].isoformat(),
613
+ str(person["commit_count"]),
614
+ str(person["touches"]),
615
+ f"{person['sensitive_touches']:.2f}",
616
+ primary_tz_offset,
617
+ primary_tz_minutes,
618
+ timezone_offsets,
619
+ ]
620
+ )
621
+
622
+ file_rows = []
623
+ for path, file_entry in sorted(files.items()):
624
+ authors = file_entry["authors"]
625
+ bus_factor = len(authors)
626
+ tags = file_entry["sensitive_tags"]
627
+ tag_list = ";".join(sorted(tags.keys()))
628
+ sensitivity_score = sum(tags.values()) if tags else 0.0
629
+ file_rows.append(
630
+ [
631
+ path,
632
+ path,
633
+ file_entry["first_seen"].isoformat(),
634
+ file_entry["last_seen"].isoformat(),
635
+ str(file_entry["commit_count"]),
636
+ str(file_entry["touches"]),
637
+ str(bus_factor),
638
+ f"{sensitivity_score:.2f}",
639
+ tag_list,
640
+ ]
641
+ )
642
+
643
+ edge_rows = []
644
+ for (email, path), edge in edges.items():
645
+ if int(edge["touches"]) < args.min_touches:
646
+ continue
647
+ edge_rows.append(
648
+ [
649
+ email,
650
+ path,
651
+ str(edge["touches"]),
652
+ f"{edge['recency_weight']:.6f}",
653
+ edge["first_seen"].isoformat(),
654
+ edge["last_seen"].isoformat(),
655
+ f"{edge['sensitive_weight']:.2f}",
656
+ ]
657
+ )
658
+
659
+ cochange_rows: list[list[str]] = []
660
+ if not args.no_cochange:
661
+ for (file_a, file_b), count in cochange_counts.items():
662
+ if count < args.cochange_min_count:
663
+ continue
664
+ commits_a = cochange_file_commits.get(file_a, 0)
665
+ commits_b = cochange_file_commits.get(file_b, 0)
666
+ denom = commits_a + commits_b - count
667
+ if denom <= 0:
668
+ continue
669
+ jaccard = count / denom
670
+ if jaccard < args.cochange_min_jaccard:
671
+ continue
672
+ cochange_rows.append([file_a, file_b, str(count), f"{jaccard:.6f}"])
673
+
674
+ write_csv(
675
+ out_dir / "people.csv",
676
+ [
677
+ "person_id",
678
+ "name",
679
+ "email",
680
+ "first_seen",
681
+ "last_seen",
682
+ "commit_count",
683
+ "touches",
684
+ "sensitive_touches",
685
+ "primary_tz_offset",
686
+ "primary_tz_minutes",
687
+ "timezone_offsets",
688
+ ],
689
+ people_rows,
690
+ )
691
+ write_csv(
692
+ out_dir / "files.csv",
693
+ [
694
+ "file_id",
695
+ "path",
696
+ "first_seen",
697
+ "last_seen",
698
+ "commit_count",
699
+ "touches",
700
+ "bus_factor",
701
+ "sensitivity_score",
702
+ "sensitivity_tags",
703
+ ],
704
+ file_rows,
705
+ )
706
+ write_csv(
707
+ out_dir / "edges.csv",
708
+ [
709
+ "person_id",
710
+ "file_id",
711
+ "touches",
712
+ "recency_weight",
713
+ "first_seen",
714
+ "last_seen",
715
+ "sensitive_weight",
716
+ ],
717
+ edge_rows,
718
+ )
719
+ if not args.no_cochange:
720
+ write_csv(
721
+ out_dir / "cochange_edges.csv",
722
+ [
723
+ "file_a",
724
+ "file_b",
725
+ "cochange_count",
726
+ "jaccard",
727
+ ],
728
+ cochange_rows,
729
+ )
730
+
731
+ orphaned_sensitive_code = []
732
+ bus_factor_hotspots = []
733
+ for path, file_entry in files.items():
734
+ tags = file_entry["sensitive_tags"]
735
+ if not tags:
736
+ continue
737
+ bus_factor = len(file_entry["authors"])
738
+ last_seen = file_entry["last_seen"]
739
+ age_days = (now - last_seen).days
740
+ top_owner = None
741
+ if path in file_people_touches:
742
+ top_owner = max(file_people_touches[path].items(), key=lambda item: item[1])[0]
743
+ hotspot = {
744
+ "path": path,
745
+ "bus_factor": bus_factor,
746
+ "last_touch": last_seen.isoformat(),
747
+ "sensitivity_tags": sorted(tags.keys()),
748
+ "top_owner": top_owner,
749
+ }
750
+ if bus_factor <= args.bus_factor_threshold:
751
+ bus_factor_hotspots.append(hotspot)
752
+ if age_days >= args.stale_days:
753
+ orphaned_sensitive_code.append(
754
+ {
755
+ **hotspot,
756
+ "last_security_touch": last_seen.isoformat(),
757
+ }
758
+ )
759
+
760
+ hidden_owners = []
761
+ for tag, total in tag_totals.items():
762
+ if total <= 0:
763
+ continue
764
+ person_totals = tag_person_totals[tag]
765
+ if not person_totals:
766
+ continue
767
+ top_email, top_value = max(person_totals.items(), key=lambda item: item[1])
768
+ share = top_value / total
769
+ if share >= args.owner_threshold:
770
+ person_name = people.get(top_email, {}).get("name", top_email)
771
+ hidden_owners.append(
772
+ {
773
+ "person": top_email,
774
+ "name": person_name,
775
+ "controls": f"{share * 100:.0f}% of {tag} code",
776
+ "category": tag,
777
+ "share": round(share, 4),
778
+ }
779
+ )
780
+
781
+ summary = {
782
+ "generated_at": now.isoformat(),
783
+ "repo": os.path.abspath(args.repo),
784
+ "parameters": {
785
+ "since": args.since,
786
+ "until": args.until,
787
+ "half_life_days": args.half_life_days,
788
+ "bus_factor_threshold": args.bus_factor_threshold,
789
+ "stale_days": args.stale_days,
790
+ "owner_threshold": args.owner_threshold,
791
+ "sensitive_config": args.sensitive_config,
792
+ "identity": args.identity,
793
+ "date_field": args.date_field,
794
+ "include_merges": args.include_merges,
795
+ "cochange_enabled": not args.no_cochange,
796
+ "cochange_max_files": args.cochange_max_files,
797
+ "cochange_min_count": args.cochange_min_count,
798
+ "cochange_min_jaccard": args.cochange_min_jaccard,
799
+ "cochange_default_excludes": not args.no_default_cochange_excludes,
800
+ "cochange_excludes": cochange_excludes,
801
+ "author_default_excludes": not args.no_default_author_excludes,
802
+ "author_exclude_regexes": author_exclude_regexes,
803
+ "community_top_owners": args.community_top_owners,
804
+ },
805
+ "orphaned_sensitive_code": orphaned_sensitive_code,
806
+ "hidden_owners": hidden_owners,
807
+ "bus_factor_hotspots": bus_factor_hotspots,
808
+ "stats": {
809
+ "commits": total_commits_included,
810
+ "commits_seen": total_commits_seen,
811
+ "commits_excluded_identities": commits_excluded_identities,
812
+ "commits_excluded_merges": commits_excluded_merges,
813
+ "edges": total_edges,
814
+ "people": len(people),
815
+ "files": len(files),
816
+ "cochange_pairs_total": len(cochange_counts) if not args.no_cochange else 0,
817
+ "cochange_edges": len(cochange_rows) if not args.no_cochange else 0,
818
+ "cochange_commits_used": cochange_commits_used if not args.no_cochange else 0,
819
+ "cochange_commits_skipped": cochange_commits_skipped if not args.no_cochange else 0,
820
+ "cochange_commits_filtered": cochange_commits_filtered if not args.no_cochange else 0,
821
+ "cochange_files_excluded": cochange_files_excluded if not args.no_cochange else 0,
822
+ },
823
+ }
824
+
825
+ with (out_dir / "summary.json").open("w", encoding="utf-8") as handle:
826
+ json.dump(summary, handle, indent=2)
827
+
828
+ if args.communities or args.graphml:
829
+ try:
830
+ import networkx as nx
831
+ from networkx.algorithms import bipartite
832
+ except ImportError:
833
+ raise RuntimeError(
834
+ "networkx is required for communities/graphml output. Install with: pip install networkx"
835
+ )
836
+ else:
837
+ graph_bipartite = None
838
+ graph_cochange = None
839
+ person_nodes = set()
840
+ file_nodes = set()
841
+ community_index: dict[str, int] = {}
842
+ community_metadata: list[dict[str, object]] = []
843
+
844
+ if args.graphml or (args.communities and (args.no_cochange or not cochange_rows)):
845
+ graph_bipartite = nx.Graph()
846
+ for (email, path), edge in edges.items():
847
+ if int(edge["touches"]) < args.min_touches:
848
+ continue
849
+ graph_bipartite.add_node(email, node_type="person")
850
+ graph_bipartite.add_node(path, node_type="file")
851
+ graph_bipartite.add_edge(email, path, weight=float(edge["touches"]))
852
+ person_nodes.add(email)
853
+ file_nodes.add(path)
854
+
855
+ if not args.no_cochange and cochange_rows:
856
+ graph_cochange = nx.Graph()
857
+ for file_a, file_b, count, jaccard in cochange_rows:
858
+ graph_cochange.add_edge(
859
+ file_a,
860
+ file_b,
861
+ weight=float(jaccard),
862
+ count=int(count),
863
+ )
864
+
865
+ if args.communities:
866
+ communities_result = None
867
+ if graph_cochange is not None:
868
+ communities_result = list(
869
+ nx.algorithms.community.greedy_modularity_communities(
870
+ graph_cochange, weight="weight"
871
+ )
872
+ )
873
+ elif graph_bipartite is not None and file_nodes:
874
+ projected = bipartite.weighted_projected_graph(graph_bipartite, file_nodes)
875
+ communities_result = list(
876
+ nx.algorithms.community.greedy_modularity_communities(projected)
877
+ )
878
+
879
+ if communities_result is not None:
880
+ serialized = []
881
+ for idx, community in enumerate(communities_result, start=1):
882
+ files_list = sorted(community)
883
+ owners = compute_community_owners(
884
+ files_list,
885
+ people,
886
+ file_people_touches,
887
+ file_people_recency,
888
+ file_people_sensitive,
889
+ args.community_top_owners,
890
+ )
891
+ for path in files_list:
892
+ community_index[path] = idx
893
+ entry = {
894
+ "id": idx,
895
+ "size": len(files_list),
896
+ "files": files_list[: args.max_community_files],
897
+ "maintainers": owners["top_maintainers"],
898
+ "bus_factor": owners["bus_factor"],
899
+ "owner_count": owners["owner_count"],
900
+ "totals": owners["totals"],
901
+ }
902
+ serialized.append(entry)
903
+ metadata = dict(entry)
904
+ metadata.pop("files", None)
905
+ community_metadata.append(metadata)
906
+ with (out_dir / "communities.json").open("w", encoding="utf-8") as handle:
907
+ json.dump(serialized, handle, indent=2)
908
+
909
+ if args.communities:
910
+ for node, community_id in community_index.items():
911
+ if graph_cochange is not None and node in graph_cochange:
912
+ graph_cochange.nodes[node]["community_id"] = community_id
913
+ if graph_bipartite is not None and node in graph_bipartite:
914
+ graph_bipartite.nodes[node]["community_id"] = community_id
915
+
916
+ graph_for_json = graph_cochange or graph_bipartite
917
+ if graph_for_json is not None:
918
+ try:
919
+ from networkx.readwrite import json_graph
920
+ except ImportError:
921
+ pass
922
+ else:
923
+ data = json_graph.node_link_data(graph_for_json, edges="edges")
924
+ data.setdefault("graph", {})
925
+ data["graph"]["community_maintainers"] = community_metadata
926
+ json_name = (
927
+ "cochange.graph.json"
928
+ if graph_for_json is graph_cochange
929
+ else "ownership.graph.json"
930
+ )
931
+ with (out_dir / json_name).open("w", encoding="utf-8") as handle:
932
+ json.dump(data, handle, indent=2)
933
+
934
+ if args.graphml:
935
+ if graph_bipartite is not None:
936
+ nx.write_graphml(graph_bipartite, out_dir / "ownership.graphml")
937
+ if graph_cochange is not None:
938
+ nx.write_graphml(graph_cochange, out_dir / "cochange.graphml")
939
+
940
+ return out_dir
941
+
942
+
943
+ def main() -> int:
944
+ args = parse_args()
945
+ try:
946
+ out_dir = build_ownership_map(args)
947
+ except RuntimeError as exc:
948
+ print(str(exc), file=sys.stderr)
949
+ return 1
950
+
951
+ print(f"Ownership map written to {out_dir}")
952
+ return 0
953
+
954
+
955
+ if __name__ == "__main__":
956
+ raise SystemExit(main())