bps-kit 1.0.1 → 1.0.2

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 (368) hide show
  1. package/package.json +1 -1
  2. package/templates/.agents/agents/backend-specialist.md +263 -0
  3. package/templates/.agents/agents/code-archaeologist.md +106 -0
  4. package/templates/.agents/agents/database-architect.md +226 -0
  5. package/templates/.agents/agents/debugger.md +225 -0
  6. package/templates/.agents/agents/devops-engineer.md +242 -0
  7. package/templates/.agents/agents/documentation-writer.md +104 -0
  8. package/templates/.agents/agents/explorer-agent.md +73 -0
  9. package/templates/.agents/agents/frontend-specialist.md +593 -0
  10. package/templates/.agents/agents/game-developer.md +162 -0
  11. package/templates/.agents/agents/mobile-developer.md +377 -0
  12. package/templates/.agents/agents/orchestrator.md +416 -0
  13. package/templates/.agents/agents/penetration-tester.md +188 -0
  14. package/templates/.agents/agents/performance-optimizer.md +187 -0
  15. package/templates/.agents/agents/product-manager.md +112 -0
  16. package/templates/.agents/agents/product-owner.md +95 -0
  17. package/templates/.agents/agents/project-planner.md +406 -0
  18. package/templates/.agents/agents/qa-automation-engineer.md +103 -0
  19. package/templates/.agents/agents/security-auditor.md +170 -0
  20. package/templates/.agents/agents/seo-specialist.md +111 -0
  21. package/templates/.agents/agents/test-engineer.md +158 -0
  22. package/templates/.agents/rules/GEMINI.md +219 -0
  23. package/templates/.agents/scripts/auto_preview.py +148 -0
  24. package/templates/.agents/scripts/checklist.py +217 -0
  25. package/templates/.agents/scripts/session_manager.py +120 -0
  26. package/templates/.agents/scripts/verify_all.py +327 -0
  27. package/templates/.agents/workflows/brainstorm.md +113 -0
  28. package/templates/.agents/workflows/create.md +59 -0
  29. package/templates/.agents/workflows/debug.md +103 -0
  30. package/templates/.agents/workflows/deploy.md +176 -0
  31. package/templates/.agents/workflows/enhance.md +63 -0
  32. package/templates/.agents/workflows/orchestrate.md +237 -0
  33. package/templates/.agents/workflows/plan.md +89 -0
  34. package/templates/.agents/workflows/preview.md +81 -0
  35. package/templates/.agents/workflows/setup-brain.md +39 -0
  36. package/templates/.agents/workflows/status.md +86 -0
  37. package/templates/.agents/workflows/test.md +144 -0
  38. package/templates/.agents/workflows/ui-ux-pro-max.md +296 -0
  39. package/templates/skills_normal/api-patterns/scripts/api_validator.py +211 -0
  40. package/templates/skills_normal/database-design/scripts/schema_validator.py +172 -0
  41. package/templates/skills_normal/frontend-design/scripts/accessibility_checker.py +183 -0
  42. package/templates/skills_normal/frontend-design/scripts/ux_audit.py +722 -0
  43. package/templates/skills_normal/git-pushing/scripts/smart_commit.sh +19 -0
  44. package/templates/skills_normal/lint-and-validate/scripts/lint_runner.py +184 -0
  45. package/templates/skills_normal/lint-and-validate/scripts/type_coverage.py +173 -0
  46. package/templates/skills_normal/performance-profiling/scripts/lighthouse_audit.py +76 -0
  47. package/templates/skills_normal/senior-fullstack/scripts/code_quality_analyzer.py +114 -0
  48. package/templates/skills_normal/senior-fullstack/scripts/fullstack_scaffolder.py +114 -0
  49. package/templates/skills_normal/senior-fullstack/scripts/project_scaffolder.py +114 -0
  50. package/templates/skills_normal/seo-fundamentals/scripts/seo_checker.py +219 -0
  51. package/templates/skills_normal/testing-patterns/scripts/test_runner.py +219 -0
  52. package/templates/skills_normal/vulnerability-scanner/scripts/security_scan.py +458 -0
  53. package/templates/vault/007/scripts/config.py +472 -0
  54. package/templates/vault/007/scripts/full_audit.py +1306 -0
  55. package/templates/vault/007/scripts/quick_scan.py +481 -0
  56. package/templates/vault/007/scripts/requirements.txt +26 -0
  57. package/templates/vault/007/scripts/scanners/__init__.py +0 -0
  58. package/templates/vault/007/scripts/scanners/dependency_scanner.py +1305 -0
  59. package/templates/vault/007/scripts/scanners/injection_scanner.py +1104 -0
  60. package/templates/vault/007/scripts/scanners/secrets_scanner.py +1008 -0
  61. package/templates/vault/007/scripts/score_calculator.py +693 -0
  62. package/templates/vault/agent-orchestrator/scripts/match_skills.py +329 -0
  63. package/templates/vault/agent-orchestrator/scripts/orchestrate.py +304 -0
  64. package/templates/vault/agent-orchestrator/scripts/requirements.txt +1 -0
  65. package/templates/vault/agent-orchestrator/scripts/scan_registry.py +508 -0
  66. package/templates/vault/ai-studio-image/scripts/config.py +613 -0
  67. package/templates/vault/ai-studio-image/scripts/generate.py +630 -0
  68. package/templates/vault/ai-studio-image/scripts/prompt_engine.py +424 -0
  69. package/templates/vault/ai-studio-image/scripts/requirements.txt +4 -0
  70. package/templates/vault/ai-studio-image/scripts/templates.py +349 -0
  71. package/templates/vault/android_ui_verification/scripts/verify_ui.sh +32 -0
  72. package/templates/vault/apify-audience-analysis/reference/scripts/run_actor.js +363 -0
  73. package/templates/vault/apify-brand-reputation-monitoring/reference/scripts/run_actor.js +363 -0
  74. package/templates/vault/apify-competitor-intelligence/reference/scripts/run_actor.js +363 -0
  75. package/templates/vault/apify-content-analytics/reference/scripts/run_actor.js +363 -0
  76. package/templates/vault/apify-ecommerce/reference/scripts/package.json +3 -0
  77. package/templates/vault/apify-ecommerce/reference/scripts/run_actor.js +369 -0
  78. package/templates/vault/apify-influencer-discovery/reference/scripts/run_actor.js +363 -0
  79. package/templates/vault/apify-lead-generation/reference/scripts/run_actor.js +363 -0
  80. package/templates/vault/apify-market-research/reference/scripts/run_actor.js +363 -0
  81. package/templates/vault/apify-trend-analysis/reference/scripts/run_actor.js +363 -0
  82. package/templates/vault/apify-ultimate-scraper/reference/scripts/run_actor.js +363 -0
  83. package/templates/vault/audio-transcriber/scripts/install-requirements.sh +190 -0
  84. package/templates/vault/audio-transcriber/scripts/transcribe.py +486 -0
  85. package/templates/vault/claude-monitor/scripts/api_bench.py +240 -0
  86. package/templates/vault/claude-monitor/scripts/config.py +69 -0
  87. package/templates/vault/claude-monitor/scripts/health_check.py +362 -0
  88. package/templates/vault/claude-monitor/scripts/monitor.py +296 -0
  89. package/templates/vault/content-creator/scripts/brand_voice_analyzer.py +185 -0
  90. package/templates/vault/content-creator/scripts/seo_optimizer.py +419 -0
  91. package/templates/vault/context-agent/scripts/active_context.py +227 -0
  92. package/templates/vault/context-agent/scripts/compressor.py +149 -0
  93. package/templates/vault/context-agent/scripts/config.py +69 -0
  94. package/templates/vault/context-agent/scripts/context_loader.py +155 -0
  95. package/templates/vault/context-agent/scripts/context_manager.py +302 -0
  96. package/templates/vault/context-agent/scripts/models.py +103 -0
  97. package/templates/vault/context-agent/scripts/project_registry.py +132 -0
  98. package/templates/vault/context-agent/scripts/requirements.txt +6 -0
  99. package/templates/vault/context-agent/scripts/search.py +115 -0
  100. package/templates/vault/context-agent/scripts/session_parser.py +206 -0
  101. package/templates/vault/context-agent/scripts/session_summary.py +319 -0
  102. package/templates/vault/context-guardian/scripts/context_snapshot.py +229 -0
  103. package/templates/vault/docx/ooxml/scripts/pack.py +159 -0
  104. package/templates/vault/docx/ooxml/scripts/unpack.py +29 -0
  105. package/templates/vault/docx/ooxml/scripts/validate.py +69 -0
  106. package/templates/vault/docx/ooxml/scripts/validation/__init__.py +15 -0
  107. package/templates/vault/docx/ooxml/scripts/validation/base.py +951 -0
  108. package/templates/vault/docx/ooxml/scripts/validation/docx.py +274 -0
  109. package/templates/vault/docx/ooxml/scripts/validation/pptx.py +315 -0
  110. package/templates/vault/docx/ooxml/scripts/validation/redlining.py +279 -0
  111. package/templates/vault/docx/scripts/__init__.py +1 -0
  112. package/templates/vault/docx/scripts/document.py +1276 -0
  113. package/templates/vault/docx/scripts/templates/comments.xml +3 -0
  114. package/templates/vault/docx/scripts/templates/commentsExtended.xml +3 -0
  115. package/templates/vault/docx/scripts/templates/commentsExtensible.xml +3 -0
  116. package/templates/vault/docx/scripts/templates/commentsIds.xml +3 -0
  117. package/templates/vault/docx/scripts/templates/people.xml +3 -0
  118. package/templates/vault/docx/scripts/utilities.py +374 -0
  119. package/templates/vault/docx-official/ooxml/scripts/pack.py +159 -0
  120. package/templates/vault/docx-official/ooxml/scripts/unpack.py +29 -0
  121. package/templates/vault/docx-official/ooxml/scripts/validate.py +69 -0
  122. package/templates/vault/docx-official/ooxml/scripts/validation/__init__.py +15 -0
  123. package/templates/vault/docx-official/ooxml/scripts/validation/base.py +951 -0
  124. package/templates/vault/docx-official/ooxml/scripts/validation/docx.py +274 -0
  125. package/templates/vault/docx-official/ooxml/scripts/validation/pptx.py +315 -0
  126. package/templates/vault/docx-official/ooxml/scripts/validation/redlining.py +279 -0
  127. package/templates/vault/docx-official/scripts/__init__.py +1 -0
  128. package/templates/vault/docx-official/scripts/document.py +1276 -0
  129. package/templates/vault/docx-official/scripts/templates/comments.xml +3 -0
  130. package/templates/vault/docx-official/scripts/templates/commentsExtended.xml +3 -0
  131. package/templates/vault/docx-official/scripts/templates/commentsExtensible.xml +3 -0
  132. package/templates/vault/docx-official/scripts/templates/commentsIds.xml +3 -0
  133. package/templates/vault/docx-official/scripts/templates/people.xml +3 -0
  134. package/templates/vault/docx-official/scripts/utilities.py +374 -0
  135. package/templates/vault/geo-fundamentals/scripts/geo_checker.py +289 -0
  136. package/templates/vault/helm-chart-scaffolding/scripts/validate-chart.sh +244 -0
  137. package/templates/vault/i18n-localization/scripts/i18n_checker.py +241 -0
  138. package/templates/vault/instagram/scripts/account_setup.py +233 -0
  139. package/templates/vault/instagram/scripts/analyze.py +221 -0
  140. package/templates/vault/instagram/scripts/api_client.py +444 -0
  141. package/templates/vault/instagram/scripts/auth.py +411 -0
  142. package/templates/vault/instagram/scripts/comments.py +160 -0
  143. package/templates/vault/instagram/scripts/config.py +111 -0
  144. package/templates/vault/instagram/scripts/db.py +467 -0
  145. package/templates/vault/instagram/scripts/export.py +138 -0
  146. package/templates/vault/instagram/scripts/governance.py +233 -0
  147. package/templates/vault/instagram/scripts/hashtags.py +114 -0
  148. package/templates/vault/instagram/scripts/insights.py +170 -0
  149. package/templates/vault/instagram/scripts/media.py +65 -0
  150. package/templates/vault/instagram/scripts/messages.py +103 -0
  151. package/templates/vault/instagram/scripts/profile.py +58 -0
  152. package/templates/vault/instagram/scripts/publish.py +449 -0
  153. package/templates/vault/instagram/scripts/requirements.txt +5 -0
  154. package/templates/vault/instagram/scripts/run_all.py +189 -0
  155. package/templates/vault/instagram/scripts/schedule.py +189 -0
  156. package/templates/vault/instagram/scripts/serve_api.py +234 -0
  157. package/templates/vault/instagram/scripts/templates.py +155 -0
  158. package/templates/vault/junta-leiloeiros/scripts/db.py +216 -0
  159. package/templates/vault/junta-leiloeiros/scripts/export.py +137 -0
  160. package/templates/vault/junta-leiloeiros/scripts/requirements.txt +15 -0
  161. package/templates/vault/junta-leiloeiros/scripts/run_all.py +190 -0
  162. package/templates/vault/junta-leiloeiros/scripts/scraper/__init__.py +4 -0
  163. package/templates/vault/junta-leiloeiros/scripts/scraper/base_scraper.py +209 -0
  164. package/templates/vault/junta-leiloeiros/scripts/scraper/generic_scraper.py +110 -0
  165. package/templates/vault/junta-leiloeiros/scripts/scraper/jucap.py +110 -0
  166. package/templates/vault/junta-leiloeiros/scripts/scraper/juceac.py +72 -0
  167. package/templates/vault/junta-leiloeiros/scripts/scraper/juceal.py +72 -0
  168. package/templates/vault/junta-leiloeiros/scripts/scraper/juceb.py +68 -0
  169. package/templates/vault/junta-leiloeiros/scripts/scraper/jucec.py +63 -0
  170. package/templates/vault/junta-leiloeiros/scripts/scraper/jucema.py +211 -0
  171. package/templates/vault/junta-leiloeiros/scripts/scraper/jucemg.py +218 -0
  172. package/templates/vault/junta-leiloeiros/scripts/scraper/jucep.py +70 -0
  173. package/templates/vault/junta-leiloeiros/scripts/scraper/jucepa.py +74 -0
  174. package/templates/vault/junta-leiloeiros/scripts/scraper/jucepar.py +80 -0
  175. package/templates/vault/junta-leiloeiros/scripts/scraper/jucepe.py +78 -0
  176. package/templates/vault/junta-leiloeiros/scripts/scraper/jucepi.py +69 -0
  177. package/templates/vault/junta-leiloeiros/scripts/scraper/jucer.py +256 -0
  178. package/templates/vault/junta-leiloeiros/scripts/scraper/jucerja.py +170 -0
  179. package/templates/vault/junta-leiloeiros/scripts/scraper/jucern.py +71 -0
  180. package/templates/vault/junta-leiloeiros/scripts/scraper/jucesc.py +89 -0
  181. package/templates/vault/junta-leiloeiros/scripts/scraper/jucesp.py +233 -0
  182. package/templates/vault/junta-leiloeiros/scripts/scraper/jucetins.py +134 -0
  183. package/templates/vault/junta-leiloeiros/scripts/scraper/jucis_df.py +63 -0
  184. package/templates/vault/junta-leiloeiros/scripts/scraper/jucisrs.py +299 -0
  185. package/templates/vault/junta-leiloeiros/scripts/scraper/states.py +99 -0
  186. package/templates/vault/junta-leiloeiros/scripts/serve_api.py +164 -0
  187. package/templates/vault/junta-leiloeiros/scripts/web_scraper_fallback.py +233 -0
  188. package/templates/vault/last30days/scripts/last30days.py +521 -0
  189. package/templates/vault/last30days/scripts/lib/__init__.py +1 -0
  190. package/templates/vault/last30days/scripts/lib/cache.py +152 -0
  191. package/templates/vault/last30days/scripts/lib/dates.py +124 -0
  192. package/templates/vault/last30days/scripts/lib/dedupe.py +120 -0
  193. package/templates/vault/last30days/scripts/lib/env.py +149 -0
  194. package/templates/vault/last30days/scripts/lib/http.py +152 -0
  195. package/templates/vault/last30days/scripts/lib/models.py +175 -0
  196. package/templates/vault/last30days/scripts/lib/normalize.py +160 -0
  197. package/templates/vault/last30days/scripts/lib/openai_reddit.py +230 -0
  198. package/templates/vault/last30days/scripts/lib/reddit_enrich.py +232 -0
  199. package/templates/vault/last30days/scripts/lib/render.py +383 -0
  200. package/templates/vault/last30days/scripts/lib/schema.py +336 -0
  201. package/templates/vault/last30days/scripts/lib/score.py +311 -0
  202. package/templates/vault/last30days/scripts/lib/ui.py +324 -0
  203. package/templates/vault/last30days/scripts/lib/websearch.py +401 -0
  204. package/templates/vault/last30days/scripts/lib/xai_x.py +217 -0
  205. package/templates/vault/leiloeiro-avaliacao/scripts/governance.py +106 -0
  206. package/templates/vault/leiloeiro-avaliacao/scripts/requirements.txt +1 -0
  207. package/templates/vault/leiloeiro-edital/scripts/governance.py +106 -0
  208. package/templates/vault/leiloeiro-edital/scripts/requirements.txt +1 -0
  209. package/templates/vault/leiloeiro-ia/scripts/governance.py +106 -0
  210. package/templates/vault/leiloeiro-ia/scripts/requirements.txt +1 -0
  211. package/templates/vault/leiloeiro-juridico/scripts/governance.py +106 -0
  212. package/templates/vault/leiloeiro-juridico/scripts/requirements.txt +1 -0
  213. package/templates/vault/leiloeiro-mercado/scripts/governance.py +106 -0
  214. package/templates/vault/leiloeiro-mercado/scripts/requirements.txt +1 -0
  215. package/templates/vault/leiloeiro-risco/scripts/governance.py +106 -0
  216. package/templates/vault/leiloeiro-risco/scripts/requirements.txt +1 -0
  217. package/templates/vault/loki-mode/examples/todo-app-generated/backend/src/db/database.ts +24 -0
  218. package/templates/vault/loki-mode/examples/todo-app-generated/backend/src/db/db.ts +35 -0
  219. package/templates/vault/loki-mode/examples/todo-app-generated/backend/src/db/index.ts +2 -0
  220. package/templates/vault/loki-mode/examples/todo-app-generated/backend/src/db/migrations.ts +31 -0
  221. package/templates/vault/loki-mode/examples/todo-app-generated/backend/src/db/schema.sql +8 -0
  222. package/templates/vault/loki-mode/examples/todo-app-generated/backend/src/index.ts +44 -0
  223. package/templates/vault/loki-mode/examples/todo-app-generated/backend/src/routes/todos.ts +155 -0
  224. package/templates/vault/loki-mode/examples/todo-app-generated/backend/src/types/index.ts +35 -0
  225. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/App.css +384 -0
  226. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/App.tsx +81 -0
  227. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/api/todos.ts +57 -0
  228. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/components/ConfirmDialog.tsx +26 -0
  229. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/components/EmptyState.tsx +8 -0
  230. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/components/TodoForm.tsx +43 -0
  231. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/components/TodoItem.tsx +36 -0
  232. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/components/TodoList.tsx +27 -0
  233. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/hooks/useTodos.ts +81 -0
  234. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/index.css +48 -0
  235. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/main.tsx +10 -0
  236. package/templates/vault/loki-mode/examples/todo-app-generated/frontend/src/vite-env.d.ts +1 -0
  237. package/templates/vault/loki-mode/scripts/export-to-vibe-kanban.sh +178 -0
  238. package/templates/vault/loki-mode/scripts/loki-wrapper.sh +281 -0
  239. package/templates/vault/loki-mode/scripts/take-screenshots.js +55 -0
  240. package/templates/vault/matematico-tao/scripts/complexity_analyzer.py +544 -0
  241. package/templates/vault/matematico-tao/scripts/dependency_graph.py +538 -0
  242. package/templates/vault/mcp-builder/scripts/connections.py +151 -0
  243. package/templates/vault/mcp-builder/scripts/evaluation.py +373 -0
  244. package/templates/vault/mcp-builder/scripts/example_evaluation.xml +22 -0
  245. package/templates/vault/mcp-builder/scripts/requirements.txt +2 -0
  246. package/templates/vault/mobile-design/scripts/mobile_audit.py +670 -0
  247. package/templates/vault/notebooklm/scripts/__init__.py +81 -0
  248. package/templates/vault/notebooklm/scripts/ask_question.py +256 -0
  249. package/templates/vault/notebooklm/scripts/auth_manager.py +358 -0
  250. package/templates/vault/notebooklm/scripts/browser_session.py +255 -0
  251. package/templates/vault/notebooklm/scripts/browser_utils.py +107 -0
  252. package/templates/vault/notebooklm/scripts/cleanup_manager.py +302 -0
  253. package/templates/vault/notebooklm/scripts/config.py +44 -0
  254. package/templates/vault/notebooklm/scripts/notebook_manager.py +410 -0
  255. package/templates/vault/notebooklm/scripts/run.py +102 -0
  256. package/templates/vault/notebooklm/scripts/setup_environment.py +204 -0
  257. package/templates/vault/pdf/scripts/check_bounding_boxes.py +70 -0
  258. package/templates/vault/pdf/scripts/check_bounding_boxes_test.py +226 -0
  259. package/templates/vault/pdf/scripts/check_fillable_fields.py +12 -0
  260. package/templates/vault/pdf/scripts/convert_pdf_to_images.py +35 -0
  261. package/templates/vault/pdf/scripts/create_validation_image.py +41 -0
  262. package/templates/vault/pdf/scripts/extract_form_field_info.py +152 -0
  263. package/templates/vault/pdf/scripts/fill_fillable_fields.py +114 -0
  264. package/templates/vault/pdf/scripts/fill_pdf_form_with_annotations.py +108 -0
  265. package/templates/vault/pdf-official/scripts/check_bounding_boxes.py +70 -0
  266. package/templates/vault/pdf-official/scripts/check_bounding_boxes_test.py +226 -0
  267. package/templates/vault/pdf-official/scripts/check_fillable_fields.py +12 -0
  268. package/templates/vault/pdf-official/scripts/convert_pdf_to_images.py +35 -0
  269. package/templates/vault/pdf-official/scripts/create_validation_image.py +41 -0
  270. package/templates/vault/pdf-official/scripts/extract_form_field_info.py +152 -0
  271. package/templates/vault/pdf-official/scripts/fill_fillable_fields.py +114 -0
  272. package/templates/vault/pdf-official/scripts/fill_pdf_form_with_annotations.py +108 -0
  273. package/templates/vault/planning-with-files/scripts/check-complete.sh +44 -0
  274. package/templates/vault/planning-with-files/scripts/init-session.sh +120 -0
  275. package/templates/vault/pptx/ooxml/scripts/pack.py +159 -0
  276. package/templates/vault/pptx/ooxml/scripts/unpack.py +29 -0
  277. package/templates/vault/pptx/ooxml/scripts/validate.py +69 -0
  278. package/templates/vault/pptx/ooxml/scripts/validation/__init__.py +15 -0
  279. package/templates/vault/pptx/ooxml/scripts/validation/base.py +951 -0
  280. package/templates/vault/pptx/ooxml/scripts/validation/docx.py +274 -0
  281. package/templates/vault/pptx/ooxml/scripts/validation/pptx.py +315 -0
  282. package/templates/vault/pptx/ooxml/scripts/validation/redlining.py +279 -0
  283. package/templates/vault/pptx/scripts/html2pptx.js +979 -0
  284. package/templates/vault/pptx/scripts/inventory.py +1020 -0
  285. package/templates/vault/pptx/scripts/rearrange.py +231 -0
  286. package/templates/vault/pptx/scripts/replace.py +385 -0
  287. package/templates/vault/pptx/scripts/thumbnail.py +450 -0
  288. package/templates/vault/pptx-official/ooxml/scripts/pack.py +159 -0
  289. package/templates/vault/pptx-official/ooxml/scripts/unpack.py +29 -0
  290. package/templates/vault/pptx-official/ooxml/scripts/validate.py +69 -0
  291. package/templates/vault/pptx-official/ooxml/scripts/validation/__init__.py +15 -0
  292. package/templates/vault/pptx-official/ooxml/scripts/validation/base.py +951 -0
  293. package/templates/vault/pptx-official/ooxml/scripts/validation/docx.py +274 -0
  294. package/templates/vault/pptx-official/ooxml/scripts/validation/pptx.py +315 -0
  295. package/templates/vault/pptx-official/ooxml/scripts/validation/redlining.py +279 -0
  296. package/templates/vault/pptx-official/scripts/html2pptx.js +979 -0
  297. package/templates/vault/pptx-official/scripts/inventory.py +1020 -0
  298. package/templates/vault/pptx-official/scripts/rearrange.py +231 -0
  299. package/templates/vault/pptx-official/scripts/replace.py +385 -0
  300. package/templates/vault/pptx-official/scripts/thumbnail.py +450 -0
  301. package/templates/vault/product-manager-toolkit/scripts/customer_interview_analyzer.py +441 -0
  302. package/templates/vault/product-manager-toolkit/scripts/rice_prioritizer.py +296 -0
  303. package/templates/vault/prompt-engineering-patterns/scripts/optimize-prompt.py +279 -0
  304. package/templates/vault/scripts/.skill_cache.json +7538 -0
  305. package/templates/vault/scripts/skill_search.py +228 -0
  306. package/templates/vault/senior-architect/scripts/architecture_diagram_generator.py +114 -0
  307. package/templates/vault/senior-architect/scripts/dependency_analyzer.py +114 -0
  308. package/templates/vault/senior-architect/scripts/project_architect.py +114 -0
  309. package/templates/vault/shopify-development/scripts/requirements.txt +19 -0
  310. package/templates/vault/shopify-development/scripts/shopify_graphql.py +428 -0
  311. package/templates/vault/shopify-development/scripts/shopify_init.py +441 -0
  312. package/templates/vault/shopify-development/scripts/tests/test_shopify_init.py +379 -0
  313. package/templates/vault/skill-creator/scripts/init_skill.py +303 -0
  314. package/templates/vault/skill-creator/scripts/package_skill.py +110 -0
  315. package/templates/vault/skill-creator/scripts/quick_validate.py +95 -0
  316. package/templates/vault/skill-installer/scripts/detect_skills.py +318 -0
  317. package/templates/vault/skill-installer/scripts/install_skill.py +1708 -0
  318. package/templates/vault/skill-installer/scripts/package_skill.py +417 -0
  319. package/templates/vault/skill-installer/scripts/requirements.txt +1 -0
  320. package/templates/vault/skill-installer/scripts/validate_skill.py +430 -0
  321. package/templates/vault/skill-sentinel/scripts/analyzers/__init__.py +13 -0
  322. package/templates/vault/skill-sentinel/scripts/analyzers/code_quality.py +247 -0
  323. package/templates/vault/skill-sentinel/scripts/analyzers/cross_skill.py +134 -0
  324. package/templates/vault/skill-sentinel/scripts/analyzers/dependencies.py +121 -0
  325. package/templates/vault/skill-sentinel/scripts/analyzers/documentation.py +189 -0
  326. package/templates/vault/skill-sentinel/scripts/analyzers/governance_audit.py +153 -0
  327. package/templates/vault/skill-sentinel/scripts/analyzers/performance.py +164 -0
  328. package/templates/vault/skill-sentinel/scripts/analyzers/security.py +189 -0
  329. package/templates/vault/skill-sentinel/scripts/config.py +158 -0
  330. package/templates/vault/skill-sentinel/scripts/cost_optimizer.py +146 -0
  331. package/templates/vault/skill-sentinel/scripts/db.py +354 -0
  332. package/templates/vault/skill-sentinel/scripts/governance.py +58 -0
  333. package/templates/vault/skill-sentinel/scripts/recommender.py +228 -0
  334. package/templates/vault/skill-sentinel/scripts/report_generator.py +224 -0
  335. package/templates/vault/skill-sentinel/scripts/requirements.txt +1 -0
  336. package/templates/vault/skill-sentinel/scripts/run_audit.py +290 -0
  337. package/templates/vault/skill-sentinel/scripts/scanner.py +271 -0
  338. package/templates/vault/stability-ai/scripts/config.py +266 -0
  339. package/templates/vault/stability-ai/scripts/generate.py +687 -0
  340. package/templates/vault/stability-ai/scripts/requirements.txt +4 -0
  341. package/templates/vault/stability-ai/scripts/styles.py +174 -0
  342. package/templates/vault/telegram/assets/boilerplate/nodejs/src/bot-client.ts +86 -0
  343. package/templates/vault/telegram/assets/boilerplate/nodejs/src/handlers.ts +79 -0
  344. package/templates/vault/telegram/assets/boilerplate/nodejs/src/index.ts +32 -0
  345. package/templates/vault/telegram/scripts/send_message.py +143 -0
  346. package/templates/vault/telegram/scripts/setup_project.py +103 -0
  347. package/templates/vault/telegram/scripts/test_bot.py +144 -0
  348. package/templates/vault/typescript-expert/scripts/ts_diagnostic.py +203 -0
  349. package/templates/vault/ui-ux-pro-max/scripts/__pycache__/core.cpython-314.pyc +0 -0
  350. package/templates/vault/ui-ux-pro-max/scripts/__pycache__/design_system.cpython-314.pyc +0 -0
  351. package/templates/vault/ui-ux-pro-max/scripts/core.py +257 -0
  352. package/templates/vault/ui-ux-pro-max/scripts/design_system.py +487 -0
  353. package/templates/vault/ui-ux-pro-max/scripts/search.py +76 -0
  354. package/templates/vault/videodb/scripts/ws_listener.py +204 -0
  355. package/templates/vault/web-artifacts-builder/scripts/bundle-artifact.sh +54 -0
  356. package/templates/vault/web-artifacts-builder/scripts/init-artifact.sh +322 -0
  357. package/templates/vault/web-artifacts-builder/scripts/shadcn-components.tar.gz +0 -0
  358. package/templates/vault/webapp-testing/scripts/with_server.py +106 -0
  359. package/templates/vault/whatsapp-cloud-api/assets/boilerplate/nodejs/src/index.ts +125 -0
  360. package/templates/vault/whatsapp-cloud-api/assets/boilerplate/nodejs/src/template-manager.ts +67 -0
  361. package/templates/vault/whatsapp-cloud-api/assets/boilerplate/nodejs/src/types.ts +216 -0
  362. package/templates/vault/whatsapp-cloud-api/assets/boilerplate/nodejs/src/webhook-handler.ts +173 -0
  363. package/templates/vault/whatsapp-cloud-api/assets/boilerplate/nodejs/src/whatsapp-client.ts +193 -0
  364. package/templates/vault/whatsapp-cloud-api/scripts/send_test_message.py +137 -0
  365. package/templates/vault/whatsapp-cloud-api/scripts/setup_project.py +118 -0
  366. package/templates/vault/whatsapp-cloud-api/scripts/validate_config.py +190 -0
  367. package/templates/vault/youtube-summarizer/scripts/extract-transcript.py +65 -0
  368. package/templates/vault/youtube-summarizer/scripts/install-dependencies.sh +28 -0
@@ -0,0 +1,450 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Create thumbnail grids from PowerPoint presentation slides.
4
+
5
+ Creates a grid layout of slide thumbnails with configurable columns (max 6).
6
+ Each grid contains up to cols×(cols+1) images. For presentations with more
7
+ slides, multiple numbered grid files are created automatically.
8
+
9
+ The program outputs the names of all files created.
10
+
11
+ Output:
12
+ - Single grid: {prefix}.jpg (if slides fit in one grid)
13
+ - Multiple grids: {prefix}-1.jpg, {prefix}-2.jpg, etc.
14
+
15
+ Grid limits by column count:
16
+ - 3 cols: max 12 slides per grid (3×4)
17
+ - 4 cols: max 20 slides per grid (4×5)
18
+ - 5 cols: max 30 slides per grid (5×6) [default]
19
+ - 6 cols: max 42 slides per grid (6×7)
20
+
21
+ Usage:
22
+ python thumbnail.py input.pptx [output_prefix] [--cols N] [--outline-placeholders]
23
+
24
+ Examples:
25
+ python thumbnail.py presentation.pptx
26
+ # Creates: thumbnails.jpg (using default prefix)
27
+ # Outputs:
28
+ # Created 1 grid(s):
29
+ # - thumbnails.jpg
30
+
31
+ python thumbnail.py large-deck.pptx grid --cols 4
32
+ # Creates: grid-1.jpg, grid-2.jpg, grid-3.jpg
33
+ # Outputs:
34
+ # Created 3 grid(s):
35
+ # - grid-1.jpg
36
+ # - grid-2.jpg
37
+ # - grid-3.jpg
38
+
39
+ python thumbnail.py template.pptx analysis --outline-placeholders
40
+ # Creates thumbnail grids with red outlines around text placeholders
41
+ """
42
+
43
+ import argparse
44
+ import subprocess
45
+ import sys
46
+ import tempfile
47
+ from pathlib import Path
48
+
49
+ from inventory import extract_text_inventory
50
+ from PIL import Image, ImageDraw, ImageFont
51
+ from pptx import Presentation
52
+
53
+ # Constants
54
+ THUMBNAIL_WIDTH = 300 # Fixed thumbnail width in pixels
55
+ CONVERSION_DPI = 100 # DPI for PDF to image conversion
56
+ MAX_COLS = 6 # Maximum number of columns
57
+ DEFAULT_COLS = 5 # Default number of columns
58
+ JPEG_QUALITY = 95 # JPEG compression quality
59
+
60
+ # Grid layout constants
61
+ GRID_PADDING = 20 # Padding between thumbnails
62
+ BORDER_WIDTH = 2 # Border width around thumbnails
63
+ FONT_SIZE_RATIO = 0.12 # Font size as fraction of thumbnail width
64
+ LABEL_PADDING_RATIO = 0.4 # Label padding as fraction of font size
65
+
66
+
67
+ def main():
68
+ parser = argparse.ArgumentParser(
69
+ description="Create thumbnail grids from PowerPoint slides."
70
+ )
71
+ parser.add_argument("input", help="Input PowerPoint file (.pptx)")
72
+ parser.add_argument(
73
+ "output_prefix",
74
+ nargs="?",
75
+ default="thumbnails",
76
+ help="Output prefix for image files (default: thumbnails, will create prefix.jpg or prefix-N.jpg)",
77
+ )
78
+ parser.add_argument(
79
+ "--cols",
80
+ type=int,
81
+ default=DEFAULT_COLS,
82
+ help=f"Number of columns (default: {DEFAULT_COLS}, max: {MAX_COLS})",
83
+ )
84
+ parser.add_argument(
85
+ "--outline-placeholders",
86
+ action="store_true",
87
+ help="Outline text placeholders with a colored border",
88
+ )
89
+
90
+ args = parser.parse_args()
91
+
92
+ # Validate columns
93
+ cols = min(args.cols, MAX_COLS)
94
+ if args.cols > MAX_COLS:
95
+ print(f"Warning: Columns limited to {MAX_COLS} (requested {args.cols})")
96
+
97
+ # Validate input
98
+ input_path = Path(args.input)
99
+ if not input_path.exists() or input_path.suffix.lower() != ".pptx":
100
+ print(f"Error: Invalid PowerPoint file: {args.input}")
101
+ sys.exit(1)
102
+
103
+ # Construct output path (always JPG)
104
+ output_path = Path(f"{args.output_prefix}.jpg")
105
+
106
+ print(f"Processing: {args.input}")
107
+
108
+ try:
109
+ with tempfile.TemporaryDirectory() as temp_dir:
110
+ # Get placeholder regions if outlining is enabled
111
+ placeholder_regions = None
112
+ slide_dimensions = None
113
+ if args.outline_placeholders:
114
+ print("Extracting placeholder regions...")
115
+ placeholder_regions, slide_dimensions = get_placeholder_regions(
116
+ input_path
117
+ )
118
+ if placeholder_regions:
119
+ print(f"Found placeholders on {len(placeholder_regions)} slides")
120
+
121
+ # Convert slides to images
122
+ slide_images = convert_to_images(input_path, Path(temp_dir), CONVERSION_DPI)
123
+ if not slide_images:
124
+ print("Error: No slides found")
125
+ sys.exit(1)
126
+
127
+ print(f"Found {len(slide_images)} slides")
128
+
129
+ # Create grids (max cols×(cols+1) images per grid)
130
+ grid_files = create_grids(
131
+ slide_images,
132
+ cols,
133
+ THUMBNAIL_WIDTH,
134
+ output_path,
135
+ placeholder_regions,
136
+ slide_dimensions,
137
+ )
138
+
139
+ # Print saved files
140
+ print(f"Created {len(grid_files)} grid(s):")
141
+ for grid_file in grid_files:
142
+ print(f" - {grid_file}")
143
+
144
+ except Exception as e:
145
+ print(f"Error: {e}")
146
+ sys.exit(1)
147
+
148
+
149
+ def create_hidden_slide_placeholder(size):
150
+ """Create placeholder image for hidden slides."""
151
+ img = Image.new("RGB", size, color="#F0F0F0")
152
+ draw = ImageDraw.Draw(img)
153
+ line_width = max(5, min(size) // 100)
154
+ draw.line([(0, 0), size], fill="#CCCCCC", width=line_width)
155
+ draw.line([(size[0], 0), (0, size[1])], fill="#CCCCCC", width=line_width)
156
+ return img
157
+
158
+
159
+ def get_placeholder_regions(pptx_path):
160
+ """Extract ALL text regions from the presentation.
161
+
162
+ Returns a tuple of (placeholder_regions, slide_dimensions).
163
+ text_regions is a dict mapping slide indices to lists of text regions.
164
+ Each region is a dict with 'left', 'top', 'width', 'height' in inches.
165
+ slide_dimensions is a tuple of (width_inches, height_inches).
166
+ """
167
+ prs = Presentation(str(pptx_path))
168
+ inventory = extract_text_inventory(pptx_path, prs)
169
+ placeholder_regions = {}
170
+
171
+ # Get actual slide dimensions in inches (EMU to inches conversion)
172
+ slide_width_inches = (prs.slide_width or 9144000) / 914400.0
173
+ slide_height_inches = (prs.slide_height or 5143500) / 914400.0
174
+
175
+ for slide_key, shapes in inventory.items():
176
+ # Extract slide index from "slide-N" format
177
+ slide_idx = int(slide_key.split("-")[1])
178
+ regions = []
179
+
180
+ for shape_key, shape_data in shapes.items():
181
+ # The inventory only contains shapes with text, so all shapes should be highlighted
182
+ regions.append(
183
+ {
184
+ "left": shape_data.left,
185
+ "top": shape_data.top,
186
+ "width": shape_data.width,
187
+ "height": shape_data.height,
188
+ }
189
+ )
190
+
191
+ if regions:
192
+ placeholder_regions[slide_idx] = regions
193
+
194
+ return placeholder_regions, (slide_width_inches, slide_height_inches)
195
+
196
+
197
+ def convert_to_images(pptx_path, temp_dir, dpi):
198
+ """Convert PowerPoint to images via PDF, handling hidden slides."""
199
+ # Detect hidden slides
200
+ print("Analyzing presentation...")
201
+ prs = Presentation(str(pptx_path))
202
+ total_slides = len(prs.slides)
203
+
204
+ # Find hidden slides (1-based indexing for display)
205
+ hidden_slides = {
206
+ idx + 1
207
+ for idx, slide in enumerate(prs.slides)
208
+ if slide.element.get("show") == "0"
209
+ }
210
+
211
+ print(f"Total slides: {total_slides}")
212
+ if hidden_slides:
213
+ print(f"Hidden slides: {sorted(hidden_slides)}")
214
+
215
+ pdf_path = temp_dir / f"{pptx_path.stem}.pdf"
216
+
217
+ # Convert to PDF
218
+ print("Converting to PDF...")
219
+ result = subprocess.run(
220
+ [
221
+ "soffice",
222
+ "--headless",
223
+ "--convert-to",
224
+ "pdf",
225
+ "--outdir",
226
+ str(temp_dir),
227
+ str(pptx_path),
228
+ ],
229
+ capture_output=True,
230
+ text=True,
231
+ )
232
+ if result.returncode != 0 or not pdf_path.exists():
233
+ raise RuntimeError("PDF conversion failed")
234
+
235
+ # Convert PDF to images
236
+ print(f"Converting to images at {dpi} DPI...")
237
+ result = subprocess.run(
238
+ ["pdftoppm", "-jpeg", "-r", str(dpi), str(pdf_path), str(temp_dir / "slide")],
239
+ capture_output=True,
240
+ text=True,
241
+ )
242
+ if result.returncode != 0:
243
+ raise RuntimeError("Image conversion failed")
244
+
245
+ visible_images = sorted(temp_dir.glob("slide-*.jpg"))
246
+
247
+ # Create full list with placeholders for hidden slides
248
+ all_images = []
249
+ visible_idx = 0
250
+
251
+ # Get placeholder dimensions from first visible slide
252
+ if visible_images:
253
+ with Image.open(visible_images[0]) as img:
254
+ placeholder_size = img.size
255
+ else:
256
+ placeholder_size = (1920, 1080)
257
+
258
+ for slide_num in range(1, total_slides + 1):
259
+ if slide_num in hidden_slides:
260
+ # Create placeholder image for hidden slide
261
+ placeholder_path = temp_dir / f"hidden-{slide_num:03d}.jpg"
262
+ placeholder_img = create_hidden_slide_placeholder(placeholder_size)
263
+ placeholder_img.save(placeholder_path, "JPEG")
264
+ all_images.append(placeholder_path)
265
+ else:
266
+ # Use the actual visible slide image
267
+ if visible_idx < len(visible_images):
268
+ all_images.append(visible_images[visible_idx])
269
+ visible_idx += 1
270
+
271
+ return all_images
272
+
273
+
274
+ def create_grids(
275
+ image_paths,
276
+ cols,
277
+ width,
278
+ output_path,
279
+ placeholder_regions=None,
280
+ slide_dimensions=None,
281
+ ):
282
+ """Create multiple thumbnail grids from slide images, max cols×(cols+1) images per grid."""
283
+ # Maximum images per grid is cols × (cols + 1) for better proportions
284
+ max_images_per_grid = cols * (cols + 1)
285
+ grid_files = []
286
+
287
+ print(
288
+ f"Creating grids with {cols} columns (max {max_images_per_grid} images per grid)"
289
+ )
290
+
291
+ # Split images into chunks
292
+ for chunk_idx, start_idx in enumerate(
293
+ range(0, len(image_paths), max_images_per_grid)
294
+ ):
295
+ end_idx = min(start_idx + max_images_per_grid, len(image_paths))
296
+ chunk_images = image_paths[start_idx:end_idx]
297
+
298
+ # Create grid for this chunk
299
+ grid = create_grid(
300
+ chunk_images, cols, width, start_idx, placeholder_regions, slide_dimensions
301
+ )
302
+
303
+ # Generate output filename
304
+ if len(image_paths) <= max_images_per_grid:
305
+ # Single grid - use base filename without suffix
306
+ grid_filename = output_path
307
+ else:
308
+ # Multiple grids - insert index before extension with dash
309
+ stem = output_path.stem
310
+ suffix = output_path.suffix
311
+ grid_filename = output_path.parent / f"{stem}-{chunk_idx + 1}{suffix}"
312
+
313
+ # Save grid
314
+ grid_filename.parent.mkdir(parents=True, exist_ok=True)
315
+ grid.save(str(grid_filename), quality=JPEG_QUALITY)
316
+ grid_files.append(str(grid_filename))
317
+
318
+ return grid_files
319
+
320
+
321
+ def create_grid(
322
+ image_paths,
323
+ cols,
324
+ width,
325
+ start_slide_num=0,
326
+ placeholder_regions=None,
327
+ slide_dimensions=None,
328
+ ):
329
+ """Create thumbnail grid from slide images with optional placeholder outlining."""
330
+ font_size = int(width * FONT_SIZE_RATIO)
331
+ label_padding = int(font_size * LABEL_PADDING_RATIO)
332
+
333
+ # Get dimensions
334
+ with Image.open(image_paths[0]) as img:
335
+ aspect = img.height / img.width
336
+ height = int(width * aspect)
337
+
338
+ # Calculate grid size
339
+ rows = (len(image_paths) + cols - 1) // cols
340
+ grid_w = cols * width + (cols + 1) * GRID_PADDING
341
+ grid_h = rows * (height + font_size + label_padding * 2) + (rows + 1) * GRID_PADDING
342
+
343
+ # Create grid
344
+ grid = Image.new("RGB", (grid_w, grid_h), "white")
345
+ draw = ImageDraw.Draw(grid)
346
+
347
+ # Load font with size based on thumbnail width
348
+ try:
349
+ # Use Pillow's default font with size
350
+ font = ImageFont.load_default(size=font_size)
351
+ except Exception:
352
+ # Fall back to basic default font if size parameter not supported
353
+ font = ImageFont.load_default()
354
+
355
+ # Place thumbnails
356
+ for i, img_path in enumerate(image_paths):
357
+ row, col = i // cols, i % cols
358
+ x = col * width + (col + 1) * GRID_PADDING
359
+ y_base = (
360
+ row * (height + font_size + label_padding * 2) + (row + 1) * GRID_PADDING
361
+ )
362
+
363
+ # Add label with actual slide number
364
+ label = f"{start_slide_num + i}"
365
+ bbox = draw.textbbox((0, 0), label, font=font)
366
+ text_w = bbox[2] - bbox[0]
367
+ draw.text(
368
+ (x + (width - text_w) // 2, y_base + label_padding),
369
+ label,
370
+ fill="black",
371
+ font=font,
372
+ )
373
+
374
+ # Add thumbnail below label with proportional spacing
375
+ y_thumbnail = y_base + label_padding + font_size + label_padding
376
+
377
+ with Image.open(img_path) as img:
378
+ # Get original dimensions before thumbnail
379
+ orig_w, orig_h = img.size
380
+
381
+ # Apply placeholder outlines if enabled
382
+ if placeholder_regions and (start_slide_num + i) in placeholder_regions:
383
+ # Convert to RGBA for transparency support
384
+ if img.mode != "RGBA":
385
+ img = img.convert("RGBA")
386
+
387
+ # Get the regions for this slide
388
+ regions = placeholder_regions[start_slide_num + i]
389
+
390
+ # Calculate scale factors using actual slide dimensions
391
+ if slide_dimensions:
392
+ slide_width_inches, slide_height_inches = slide_dimensions
393
+ else:
394
+ # Fallback: estimate from image size at CONVERSION_DPI
395
+ slide_width_inches = orig_w / CONVERSION_DPI
396
+ slide_height_inches = orig_h / CONVERSION_DPI
397
+
398
+ x_scale = orig_w / slide_width_inches
399
+ y_scale = orig_h / slide_height_inches
400
+
401
+ # Create a highlight overlay
402
+ overlay = Image.new("RGBA", img.size, (255, 255, 255, 0))
403
+ overlay_draw = ImageDraw.Draw(overlay)
404
+
405
+ # Highlight each placeholder region
406
+ for region in regions:
407
+ # Convert from inches to pixels in the original image
408
+ px_left = int(region["left"] * x_scale)
409
+ px_top = int(region["top"] * y_scale)
410
+ px_width = int(region["width"] * x_scale)
411
+ px_height = int(region["height"] * y_scale)
412
+
413
+ # Draw highlight outline with red color and thick stroke
414
+ # Using a bright red outline instead of fill
415
+ stroke_width = max(
416
+ 5, min(orig_w, orig_h) // 150
417
+ ) # Thicker proportional stroke width
418
+ overlay_draw.rectangle(
419
+ [(px_left, px_top), (px_left + px_width, px_top + px_height)],
420
+ outline=(255, 0, 0, 255), # Bright red, fully opaque
421
+ width=stroke_width,
422
+ )
423
+
424
+ # Composite the overlay onto the image using alpha blending
425
+ img = Image.alpha_composite(img, overlay)
426
+ # Convert back to RGB for JPEG saving
427
+ img = img.convert("RGB")
428
+
429
+ img.thumbnail((width, height), Image.Resampling.LANCZOS)
430
+ w, h = img.size
431
+ tx = x + (width - w) // 2
432
+ ty = y_thumbnail + (height - h) // 2
433
+ grid.paste(img, (tx, ty))
434
+
435
+ # Add border
436
+ if BORDER_WIDTH > 0:
437
+ draw.rectangle(
438
+ [
439
+ (tx - BORDER_WIDTH, ty - BORDER_WIDTH),
440
+ (tx + w + BORDER_WIDTH - 1, ty + h + BORDER_WIDTH - 1),
441
+ ],
442
+ outline="gray",
443
+ width=BORDER_WIDTH,
444
+ )
445
+
446
+ return grid
447
+
448
+
449
+ if __name__ == "__main__":
450
+ main()
@@ -0,0 +1,159 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Tool to pack a directory into a .docx, .pptx, or .xlsx file with XML formatting undone.
4
+
5
+ Example usage:
6
+ python pack.py <input_directory> <office_file> [--force]
7
+ """
8
+
9
+ import argparse
10
+ import shutil
11
+ import subprocess
12
+ import sys
13
+ import tempfile
14
+ import defusedxml.minidom
15
+ import zipfile
16
+ from pathlib import Path
17
+
18
+
19
+ def main():
20
+ parser = argparse.ArgumentParser(description="Pack a directory into an Office file")
21
+ parser.add_argument("input_directory", help="Unpacked Office document directory")
22
+ parser.add_argument("output_file", help="Output Office file (.docx/.pptx/.xlsx)")
23
+ parser.add_argument("--force", action="store_true", help="Skip validation")
24
+ args = parser.parse_args()
25
+
26
+ try:
27
+ success = pack_document(
28
+ args.input_directory, args.output_file, validate=not args.force
29
+ )
30
+
31
+ # Show warning if validation was skipped
32
+ if args.force:
33
+ print("Warning: Skipped validation, file may be corrupt", file=sys.stderr)
34
+ # Exit with error if validation failed
35
+ elif not success:
36
+ print("Contents would produce a corrupt file.", file=sys.stderr)
37
+ print("Please validate XML before repacking.", file=sys.stderr)
38
+ print("Use --force to skip validation and pack anyway.", file=sys.stderr)
39
+ sys.exit(1)
40
+
41
+ except ValueError as e:
42
+ sys.exit(f"Error: {e}")
43
+
44
+
45
+ def pack_document(input_dir, output_file, validate=False):
46
+ """Pack a directory into an Office file (.docx/.pptx/.xlsx).
47
+
48
+ Args:
49
+ input_dir: Path to unpacked Office document directory
50
+ output_file: Path to output Office file
51
+ validate: If True, validates with soffice (default: False)
52
+
53
+ Returns:
54
+ bool: True if successful, False if validation failed
55
+ """
56
+ input_dir = Path(input_dir)
57
+ output_file = Path(output_file)
58
+
59
+ if not input_dir.is_dir():
60
+ raise ValueError(f"{input_dir} is not a directory")
61
+ if output_file.suffix.lower() not in {".docx", ".pptx", ".xlsx"}:
62
+ raise ValueError(f"{output_file} must be a .docx, .pptx, or .xlsx file")
63
+
64
+ # Work in temporary directory to avoid modifying original
65
+ with tempfile.TemporaryDirectory() as temp_dir:
66
+ temp_content_dir = Path(temp_dir) / "content"
67
+ shutil.copytree(input_dir, temp_content_dir)
68
+
69
+ # Process XML files to remove pretty-printing whitespace
70
+ for pattern in ["*.xml", "*.rels"]:
71
+ for xml_file in temp_content_dir.rglob(pattern):
72
+ condense_xml(xml_file)
73
+
74
+ # Create final Office file as zip archive
75
+ output_file.parent.mkdir(parents=True, exist_ok=True)
76
+ with zipfile.ZipFile(output_file, "w", zipfile.ZIP_DEFLATED) as zf:
77
+ for f in temp_content_dir.rglob("*"):
78
+ if f.is_file():
79
+ zf.write(f, f.relative_to(temp_content_dir))
80
+
81
+ # Validate if requested
82
+ if validate:
83
+ if not validate_document(output_file):
84
+ output_file.unlink() # Delete the corrupt file
85
+ return False
86
+
87
+ return True
88
+
89
+
90
+ def validate_document(doc_path):
91
+ """Validate document by converting to HTML with soffice."""
92
+ # Determine the correct filter based on file extension
93
+ match doc_path.suffix.lower():
94
+ case ".docx":
95
+ filter_name = "html:HTML"
96
+ case ".pptx":
97
+ filter_name = "html:impress_html_Export"
98
+ case ".xlsx":
99
+ filter_name = "html:HTML (StarCalc)"
100
+
101
+ with tempfile.TemporaryDirectory() as temp_dir:
102
+ try:
103
+ result = subprocess.run(
104
+ [
105
+ "soffice",
106
+ "--headless",
107
+ "--convert-to",
108
+ filter_name,
109
+ "--outdir",
110
+ temp_dir,
111
+ str(doc_path),
112
+ ],
113
+ capture_output=True,
114
+ timeout=10,
115
+ text=True,
116
+ )
117
+ if not (Path(temp_dir) / f"{doc_path.stem}.html").exists():
118
+ error_msg = result.stderr.strip() or "Document validation failed"
119
+ print(f"Validation error: {error_msg}", file=sys.stderr)
120
+ return False
121
+ return True
122
+ except FileNotFoundError:
123
+ print("Warning: soffice not found. Skipping validation.", file=sys.stderr)
124
+ return True
125
+ except subprocess.TimeoutExpired:
126
+ print("Validation error: Timeout during conversion", file=sys.stderr)
127
+ return False
128
+ except Exception as e:
129
+ print(f"Validation error: {e}", file=sys.stderr)
130
+ return False
131
+
132
+
133
+ def condense_xml(xml_file):
134
+ """Strip unnecessary whitespace and remove comments."""
135
+ with open(xml_file, "r", encoding="utf-8") as f:
136
+ dom = defusedxml.minidom.parse(f)
137
+
138
+ # Process each element to remove whitespace and comments
139
+ for element in dom.getElementsByTagName("*"):
140
+ # Skip w:t elements and their processing
141
+ if element.tagName.endswith(":t"):
142
+ continue
143
+
144
+ # Remove whitespace-only text nodes and comment nodes
145
+ for child in list(element.childNodes):
146
+ if (
147
+ child.nodeType == child.TEXT_NODE
148
+ and child.nodeValue
149
+ and child.nodeValue.strip() == ""
150
+ ) or child.nodeType == child.COMMENT_NODE:
151
+ element.removeChild(child)
152
+
153
+ # Write back the condensed XML
154
+ with open(xml_file, "wb") as f:
155
+ f.write(dom.toxml(encoding="UTF-8"))
156
+
157
+
158
+ if __name__ == "__main__":
159
+ main()
@@ -0,0 +1,29 @@
1
+ #!/usr/bin/env python3
2
+ """Unpack and format XML contents of Office files (.docx, .pptx, .xlsx)"""
3
+
4
+ import random
5
+ import sys
6
+ import defusedxml.minidom
7
+ import zipfile
8
+ from pathlib import Path
9
+
10
+ # Get command line arguments
11
+ assert len(sys.argv) == 3, "Usage: python unpack.py <office_file> <output_dir>"
12
+ input_file, output_dir = sys.argv[1], sys.argv[2]
13
+
14
+ # Extract and format
15
+ output_path = Path(output_dir)
16
+ output_path.mkdir(parents=True, exist_ok=True)
17
+ zipfile.ZipFile(input_file).extractall(output_path)
18
+
19
+ # Pretty print all XML files
20
+ xml_files = list(output_path.rglob("*.xml")) + list(output_path.rglob("*.rels"))
21
+ for xml_file in xml_files:
22
+ content = xml_file.read_text(encoding="utf-8")
23
+ dom = defusedxml.minidom.parseString(content)
24
+ xml_file.write_bytes(dom.toprettyxml(indent=" ", encoding="ascii"))
25
+
26
+ # For .docx files, suggest an RSID for tracked changes
27
+ if input_file.endswith(".docx"):
28
+ suggested_rsid = "".join(random.choices("0123456789ABCDEF", k=8))
29
+ print(f"Suggested RSID for edit session: {suggested_rsid}")