@thanhvn14/csvibe 0.1.4 → 0.1.5

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 (392) hide show
  1. package/.github/agents/schemas/base-output.schema.json +88 -0
  2. package/.github/agents/schemas/brainstorm-output.schema.json +88 -0
  3. package/.github/agents/schemas/scout-output.schema.json +60 -0
  4. package/.github/agents/scripts/fetch-copilot-tools.js +245 -0
  5. package/.github/agents/scripts/lib/parse-agent-file.js +275 -0
  6. package/.github/agents/scripts/package-lock.json +78 -0
  7. package/.github/agents/scripts/package.json +22 -0
  8. package/.github/agents/scripts/schemas/agent-frontmatter.schema.json +83 -0
  9. package/.github/agents/scripts/validate-agent-all.js +157 -0
  10. package/.github/agents/scripts/validate-agent-frontmatter.js +96 -0
  11. package/.github/agents/scripts/validate-agent-handoffs.js +169 -0
  12. package/.github/agents/scripts/validate-agent-output.js +157 -0
  13. package/.github/agents/scripts/validate-agent-tools.js +278 -0
  14. package/.github/skills/.env.example +100 -0
  15. package/.github/skills/.install-state.json +23 -0
  16. package/.github/skills/README.md +149 -0
  17. package/.github/skills/ai-multimodal/.env.example +204 -0
  18. package/.github/skills/ai-multimodal/scripts/.coverage +0 -0
  19. package/.github/skills/ai-multimodal/scripts/check_setup.py +305 -0
  20. package/.github/skills/ai-multimodal/scripts/document_converter.py +395 -0
  21. package/.github/skills/ai-multimodal/scripts/gemini_batch_process.py +1184 -0
  22. package/.github/skills/ai-multimodal/scripts/media_optimizer.py +506 -0
  23. package/.github/skills/ai-multimodal/scripts/requirements.txt +26 -0
  24. package/.github/skills/better-auth/scripts/.coverage +0 -0
  25. package/.github/skills/better-auth/scripts/better_auth_init.py +521 -0
  26. package/.github/skills/better-auth/scripts/requirements.txt +15 -0
  27. package/.github/skills/chrome-devtools/scripts/README.md +272 -0
  28. package/.github/skills/chrome-devtools/scripts/__tests__/selector.test.js +210 -0
  29. package/.github/skills/chrome-devtools/scripts/aria-snapshot.js +362 -0
  30. package/.github/skills/chrome-devtools/scripts/click.js +83 -0
  31. package/.github/skills/chrome-devtools/scripts/console.js +79 -0
  32. package/.github/skills/chrome-devtools/scripts/evaluate.js +53 -0
  33. package/.github/skills/chrome-devtools/scripts/fill.js +76 -0
  34. package/.github/skills/chrome-devtools/scripts/inject-auth.js +229 -0
  35. package/.github/skills/chrome-devtools/scripts/install-deps.sh +181 -0
  36. package/.github/skills/chrome-devtools/scripts/install.sh +83 -0
  37. package/.github/skills/chrome-devtools/scripts/lib/browser.js +318 -0
  38. package/.github/skills/chrome-devtools/scripts/lib/selector.js +178 -0
  39. package/.github/skills/chrome-devtools/scripts/navigate.js +54 -0
  40. package/.github/skills/chrome-devtools/scripts/network.js +106 -0
  41. package/.github/skills/chrome-devtools/scripts/package-lock.json +1589 -0
  42. package/.github/skills/chrome-devtools/scripts/package.json +16 -0
  43. package/.github/skills/chrome-devtools/scripts/performance.js +149 -0
  44. package/.github/skills/chrome-devtools/scripts/screenshot.js +198 -0
  45. package/.github/skills/chrome-devtools/scripts/select-ref.js +131 -0
  46. package/.github/skills/chrome-devtools/scripts/snapshot.js +135 -0
  47. package/.github/skills/common/README.md +120 -0
  48. package/.github/skills/common/api_key_helper.py +411 -0
  49. package/.github/skills/common/api_key_rotator.py +248 -0
  50. package/.github/skills/databases/scripts/.coverage +0 -0
  51. package/.github/skills/databases/scripts/db_backup.py +502 -0
  52. package/.github/skills/databases/scripts/db_migrate.py +425 -0
  53. package/.github/skills/databases/scripts/db_performance_check.py +456 -0
  54. package/.github/skills/databases/scripts/requirements.txt +20 -0
  55. package/.github/skills/debugging/scripts/find-polluter.sh +63 -0
  56. package/.github/skills/devops/.env.example +76 -0
  57. package/.github/skills/devops/scripts/cloudflare_deploy.py +269 -0
  58. package/.github/skills/devops/scripts/docker_optimize.py +331 -0
  59. package/.github/skills/devops/scripts/requirements.txt +20 -0
  60. package/.github/skills/docs-seeker/.env.example +15 -0
  61. package/.github/skills/docs-seeker/package.json +25 -0
  62. package/.github/skills/docs-seeker/scripts/analyze-llms-txt.js +211 -0
  63. package/.github/skills/docs-seeker/scripts/detect-topic.js +172 -0
  64. package/.github/skills/docs-seeker/scripts/fetch-docs.js +213 -0
  65. package/.github/skills/docs-seeker/scripts/utils/env-loader.js +94 -0
  66. package/.github/skills/document-skills/docx/LICENSE.txt +30 -0
  67. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
  68. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
  69. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
  70. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
  71. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
  72. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
  73. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
  74. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
  75. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
  76. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
  77. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
  78. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
  79. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
  80. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
  81. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
  82. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
  83. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
  84. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
  85. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
  86. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
  87. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
  88. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
  89. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
  90. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
  91. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
  92. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
  93. package/.github/skills/document-skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
  94. package/.github/skills/document-skills/docx/ooxml/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
  95. package/.github/skills/document-skills/docx/ooxml/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
  96. package/.github/skills/document-skills/docx/ooxml/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
  97. package/.github/skills/document-skills/docx/ooxml/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
  98. package/.github/skills/document-skills/docx/ooxml/schemas/mce/mc.xsd +75 -0
  99. package/.github/skills/document-skills/docx/ooxml/schemas/microsoft/wml-2010.xsd +560 -0
  100. package/.github/skills/document-skills/docx/ooxml/schemas/microsoft/wml-2012.xsd +67 -0
  101. package/.github/skills/document-skills/docx/ooxml/schemas/microsoft/wml-2018.xsd +14 -0
  102. package/.github/skills/document-skills/docx/ooxml/schemas/microsoft/wml-cex-2018.xsd +20 -0
  103. package/.github/skills/document-skills/docx/ooxml/schemas/microsoft/wml-cid-2016.xsd +13 -0
  104. package/.github/skills/document-skills/docx/ooxml/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
  105. package/.github/skills/document-skills/docx/ooxml/schemas/microsoft/wml-symex-2015.xsd +8 -0
  106. package/.github/skills/document-skills/docx/ooxml/scripts/pack.py +159 -0
  107. package/.github/skills/document-skills/docx/ooxml/scripts/unpack.py +29 -0
  108. package/.github/skills/document-skills/docx/ooxml/scripts/validate.py +69 -0
  109. package/.github/skills/document-skills/docx/ooxml/scripts/validation/__init__.py +15 -0
  110. package/.github/skills/document-skills/docx/ooxml/scripts/validation/base.py +951 -0
  111. package/.github/skills/document-skills/docx/ooxml/scripts/validation/docx.py +274 -0
  112. package/.github/skills/document-skills/docx/ooxml/scripts/validation/pptx.py +315 -0
  113. package/.github/skills/document-skills/docx/ooxml/scripts/validation/redlining.py +279 -0
  114. package/.github/skills/document-skills/docx/scripts/__init__.py +1 -0
  115. package/.github/skills/document-skills/docx/scripts/document.py +1276 -0
  116. package/.github/skills/document-skills/docx/scripts/templates/comments.xml +3 -0
  117. package/.github/skills/document-skills/docx/scripts/templates/commentsExtended.xml +3 -0
  118. package/.github/skills/document-skills/docx/scripts/templates/commentsExtensible.xml +3 -0
  119. package/.github/skills/document-skills/docx/scripts/templates/commentsIds.xml +3 -0
  120. package/.github/skills/document-skills/docx/scripts/templates/people.xml +3 -0
  121. package/.github/skills/document-skills/docx/scripts/utilities.py +374 -0
  122. package/.github/skills/document-skills/pdf/LICENSE.txt +30 -0
  123. package/.github/skills/document-skills/pdf/scripts/check_bounding_boxes.py +70 -0
  124. package/.github/skills/document-skills/pdf/scripts/check_bounding_boxes_test.py +226 -0
  125. package/.github/skills/document-skills/pdf/scripts/check_fillable_fields.py +12 -0
  126. package/.github/skills/document-skills/pdf/scripts/convert_pdf_to_images.py +35 -0
  127. package/.github/skills/document-skills/pdf/scripts/create_validation_image.py +41 -0
  128. package/.github/skills/document-skills/pdf/scripts/extract_form_field_info.py +152 -0
  129. package/.github/skills/document-skills/pdf/scripts/fill_fillable_fields.py +114 -0
  130. package/.github/skills/document-skills/pdf/scripts/fill_pdf_form_with_annotations.py +108 -0
  131. package/.github/skills/document-skills/pptx/LICENSE.txt +30 -0
  132. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
  133. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
  134. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
  135. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
  136. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
  137. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
  138. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
  139. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
  140. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
  141. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
  142. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
  143. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
  144. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
  145. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
  146. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
  147. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
  148. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
  149. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
  150. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
  151. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
  152. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
  153. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
  154. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
  155. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
  156. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
  157. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
  158. package/.github/skills/document-skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
  159. package/.github/skills/document-skills/pptx/ooxml/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
  160. package/.github/skills/document-skills/pptx/ooxml/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
  161. package/.github/skills/document-skills/pptx/ooxml/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
  162. package/.github/skills/document-skills/pptx/ooxml/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
  163. package/.github/skills/document-skills/pptx/ooxml/schemas/mce/mc.xsd +75 -0
  164. package/.github/skills/document-skills/pptx/ooxml/schemas/microsoft/wml-2010.xsd +560 -0
  165. package/.github/skills/document-skills/pptx/ooxml/schemas/microsoft/wml-2012.xsd +67 -0
  166. package/.github/skills/document-skills/pptx/ooxml/schemas/microsoft/wml-2018.xsd +14 -0
  167. package/.github/skills/document-skills/pptx/ooxml/schemas/microsoft/wml-cex-2018.xsd +20 -0
  168. package/.github/skills/document-skills/pptx/ooxml/schemas/microsoft/wml-cid-2016.xsd +13 -0
  169. package/.github/skills/document-skills/pptx/ooxml/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
  170. package/.github/skills/document-skills/pptx/ooxml/schemas/microsoft/wml-symex-2015.xsd +8 -0
  171. package/.github/skills/document-skills/pptx/ooxml/scripts/pack.py +159 -0
  172. package/.github/skills/document-skills/pptx/ooxml/scripts/unpack.py +29 -0
  173. package/.github/skills/document-skills/pptx/ooxml/scripts/validate.py +69 -0
  174. package/.github/skills/document-skills/pptx/ooxml/scripts/validation/__init__.py +15 -0
  175. package/.github/skills/document-skills/pptx/ooxml/scripts/validation/base.py +951 -0
  176. package/.github/skills/document-skills/pptx/ooxml/scripts/validation/docx.py +274 -0
  177. package/.github/skills/document-skills/pptx/ooxml/scripts/validation/pptx.py +315 -0
  178. package/.github/skills/document-skills/pptx/ooxml/scripts/validation/redlining.py +279 -0
  179. package/.github/skills/document-skills/pptx/scripts/html2pptx.js +979 -0
  180. package/.github/skills/document-skills/pptx/scripts/inventory.py +1020 -0
  181. package/.github/skills/document-skills/pptx/scripts/rearrange.py +231 -0
  182. package/.github/skills/document-skills/pptx/scripts/replace.py +385 -0
  183. package/.github/skills/document-skills/pptx/scripts/thumbnail.py +450 -0
  184. package/.github/skills/document-skills/xlsx/LICENSE.txt +30 -0
  185. package/.github/skills/document-skills/xlsx/recalc.py +190 -0
  186. package/.github/skills/install.ps1 +1220 -0
  187. package/.github/skills/install.sh +1032 -0
  188. package/.github/skills/markdown-novel-viewer/assets/directory-browser.css +215 -0
  189. package/.github/skills/markdown-novel-viewer/assets/favicon.png +0 -0
  190. package/.github/skills/markdown-novel-viewer/assets/novel-theme.css +818 -0
  191. package/.github/skills/markdown-novel-viewer/assets/reader.js +262 -0
  192. package/.github/skills/markdown-novel-viewer/assets/template.html +80 -0
  193. package/.github/skills/markdown-novel-viewer/package-lock.json +146 -0
  194. package/.github/skills/markdown-novel-viewer/package.json +15 -0
  195. package/.github/skills/markdown-novel-viewer/scripts/lib/http-server.cjs +434 -0
  196. package/.github/skills/markdown-novel-viewer/scripts/lib/markdown-renderer.cjs +272 -0
  197. package/.github/skills/markdown-novel-viewer/scripts/lib/plan-navigator.cjs +509 -0
  198. package/.github/skills/markdown-novel-viewer/scripts/lib/port-finder.cjs +48 -0
  199. package/.github/skills/markdown-novel-viewer/scripts/lib/process-mgr.cjs +150 -0
  200. package/.github/skills/markdown-novel-viewer/scripts/server.cjs +411 -0
  201. package/.github/skills/mcp-builder/LICENSE.txt +202 -0
  202. package/.github/skills/mcp-builder/scripts/connections.py +151 -0
  203. package/.github/skills/mcp-builder/scripts/evaluation.py +373 -0
  204. package/.github/skills/mcp-builder/scripts/example_evaluation.xml +22 -0
  205. package/.github/skills/mcp-builder/scripts/requirements.txt +2 -0
  206. package/.github/skills/mcp-management/README.md +219 -0
  207. package/.github/skills/mcp-management/assets/tools.json +3146 -0
  208. package/.github/skills/mcp-management/package-lock.json +6 -0
  209. package/.github/skills/mcp-management/scripts/.env.example +10 -0
  210. package/.github/skills/mcp-management/scripts/cli.ts +195 -0
  211. package/.github/skills/mcp-management/scripts/dist/analyze-tools.js +70 -0
  212. package/.github/skills/mcp-management/scripts/dist/cli.js +160 -0
  213. package/.github/skills/mcp-management/scripts/dist/mcp-client.js +183 -0
  214. package/.github/skills/mcp-management/scripts/mcp-client.ts +230 -0
  215. package/.github/skills/mcp-management/scripts/package.json +20 -0
  216. package/.github/skills/media-processing/scripts/README.md +111 -0
  217. package/.github/skills/media-processing/scripts/batch-remove-background.sh +124 -0
  218. package/.github/skills/media-processing/scripts/batch_resize.py +342 -0
  219. package/.github/skills/media-processing/scripts/media_convert.py +311 -0
  220. package/.github/skills/media-processing/scripts/remove-background.sh +96 -0
  221. package/.github/skills/media-processing/scripts/remove-bg-node.js +158 -0
  222. package/.github/skills/media-processing/scripts/requirements.txt +24 -0
  223. package/.github/skills/media-processing/scripts/video_optimize.py +414 -0
  224. package/.github/skills/payment-integration/README.md +185 -0
  225. package/.github/skills/payment-integration/scripts/.env.example +20 -0
  226. package/.github/skills/payment-integration/scripts/checkout-helper.js +244 -0
  227. package/.github/skills/payment-integration/scripts/package.json +17 -0
  228. package/.github/skills/payment-integration/scripts/polar-webhook-verify.js +202 -0
  229. package/.github/skills/payment-integration/scripts/sepay-webhook-verify.js +193 -0
  230. package/.github/skills/payment-integration/scripts/test-scripts.js +237 -0
  231. package/.github/skills/plans-kanban/assets/dashboard-template.html +119 -0
  232. package/.github/skills/plans-kanban/assets/dashboard.css +1594 -0
  233. package/.github/skills/plans-kanban/assets/dashboard.js +596 -0
  234. package/.github/skills/plans-kanban/assets/favicon.png +0 -0
  235. package/.github/skills/plans-kanban/package-lock.json +123 -0
  236. package/.github/skills/plans-kanban/package.json +13 -0
  237. package/.github/skills/plans-kanban/scripts/lib/dashboard-renderer.cjs +884 -0
  238. package/.github/skills/plans-kanban/scripts/lib/http-server.cjs +310 -0
  239. package/.github/skills/plans-kanban/scripts/lib/plan-metadata-extractor.cjs +489 -0
  240. package/.github/skills/plans-kanban/scripts/lib/plan-parser.cjs +175 -0
  241. package/.github/skills/plans-kanban/scripts/lib/plan-scanner.cjs +272 -0
  242. package/.github/skills/plans-kanban/scripts/lib/port-finder.cjs +48 -0
  243. package/.github/skills/plans-kanban/scripts/lib/process-mgr.cjs +128 -0
  244. package/.github/skills/plans-kanban/scripts/server.cjs +260 -0
  245. package/.github/skills/repomix/scripts/.coverage +0 -0
  246. package/.github/skills/repomix/scripts/README.md +179 -0
  247. package/.github/skills/repomix/scripts/repomix_batch.py +455 -0
  248. package/.github/skills/repomix/scripts/repos.example.json +15 -0
  249. package/.github/skills/repomix/scripts/requirements.txt +15 -0
  250. package/.github/skills/scout-validation/scripts/lib/broad-pattern-detector.cjs +124 -0
  251. package/.github/skills/scout-validation/scripts/lib/path-checker.cjs +66 -0
  252. package/.github/skills/scout-validation/scripts/lib/schema-validator.cjs +45 -0
  253. package/.github/skills/scout-validation/scripts/package.json +11 -0
  254. package/.github/skills/scout-validation/scripts/validate-scout-output.cjs +219 -0
  255. package/.github/skills/scout-validation/test/broad-pattern-output.json +18 -0
  256. package/.github/skills/scout-validation/test/invalid-path-output.json +18 -0
  257. package/.github/skills/scout-validation/test/valid-scout-output.json +26 -0
  258. package/.github/skills/sequential-thinking/.env.example +8 -0
  259. package/.github/skills/sequential-thinking/README.md +183 -0
  260. package/.github/skills/sequential-thinking/package.json +31 -0
  261. package/.github/skills/sequential-thinking/scripts/format-thought.js +159 -0
  262. package/.github/skills/sequential-thinking/scripts/process-thought.js +236 -0
  263. package/.github/skills/shopify/README.md +66 -0
  264. package/.github/skills/shopify/scripts/.coverage +0 -0
  265. package/.github/skills/shopify/scripts/requirements.txt +19 -0
  266. package/.github/skills/shopify/scripts/shopify_init.py +423 -0
  267. package/.github/skills/skill-creator/LICENSE.txt +202 -0
  268. package/.github/skills/skill-creator/scripts/init_skill.py +303 -0
  269. package/.github/skills/skill-creator/scripts/package_skill.py +110 -0
  270. package/.github/skills/skill-creator/scripts/quick_validate.py +65 -0
  271. package/.github/skills/ui-styling/LICENSE.txt +202 -0
  272. package/.github/skills/ui-styling/canvas-fonts/ArsenalSC-OFL.txt +93 -0
  273. package/.github/skills/ui-styling/canvas-fonts/ArsenalSC-Regular.ttf +0 -0
  274. package/.github/skills/ui-styling/canvas-fonts/BigShoulders-Bold.ttf +0 -0
  275. package/.github/skills/ui-styling/canvas-fonts/BigShoulders-OFL.txt +93 -0
  276. package/.github/skills/ui-styling/canvas-fonts/BigShoulders-Regular.ttf +0 -0
  277. package/.github/skills/ui-styling/canvas-fonts/Boldonse-OFL.txt +93 -0
  278. package/.github/skills/ui-styling/canvas-fonts/Boldonse-Regular.ttf +0 -0
  279. package/.github/skills/ui-styling/canvas-fonts/BricolageGrotesque-Bold.ttf +0 -0
  280. package/.github/skills/ui-styling/canvas-fonts/BricolageGrotesque-OFL.txt +93 -0
  281. package/.github/skills/ui-styling/canvas-fonts/BricolageGrotesque-Regular.ttf +0 -0
  282. package/.github/skills/ui-styling/canvas-fonts/CrimsonPro-Bold.ttf +0 -0
  283. package/.github/skills/ui-styling/canvas-fonts/CrimsonPro-Italic.ttf +0 -0
  284. package/.github/skills/ui-styling/canvas-fonts/CrimsonPro-OFL.txt +93 -0
  285. package/.github/skills/ui-styling/canvas-fonts/CrimsonPro-Regular.ttf +0 -0
  286. package/.github/skills/ui-styling/canvas-fonts/DMMono-OFL.txt +93 -0
  287. package/.github/skills/ui-styling/canvas-fonts/DMMono-Regular.ttf +0 -0
  288. package/.github/skills/ui-styling/canvas-fonts/EricaOne-OFL.txt +94 -0
  289. package/.github/skills/ui-styling/canvas-fonts/EricaOne-Regular.ttf +0 -0
  290. package/.github/skills/ui-styling/canvas-fonts/GeistMono-Bold.ttf +0 -0
  291. package/.github/skills/ui-styling/canvas-fonts/GeistMono-OFL.txt +93 -0
  292. package/.github/skills/ui-styling/canvas-fonts/GeistMono-Regular.ttf +0 -0
  293. package/.github/skills/ui-styling/canvas-fonts/Gloock-OFL.txt +93 -0
  294. package/.github/skills/ui-styling/canvas-fonts/Gloock-Regular.ttf +0 -0
  295. package/.github/skills/ui-styling/canvas-fonts/IBMPlexMono-Bold.ttf +0 -0
  296. package/.github/skills/ui-styling/canvas-fonts/IBMPlexMono-OFL.txt +93 -0
  297. package/.github/skills/ui-styling/canvas-fonts/IBMPlexMono-Regular.ttf +0 -0
  298. package/.github/skills/ui-styling/canvas-fonts/IBMPlexSerif-Bold.ttf +0 -0
  299. package/.github/skills/ui-styling/canvas-fonts/IBMPlexSerif-BoldItalic.ttf +0 -0
  300. package/.github/skills/ui-styling/canvas-fonts/IBMPlexSerif-Italic.ttf +0 -0
  301. package/.github/skills/ui-styling/canvas-fonts/IBMPlexSerif-Regular.ttf +0 -0
  302. package/.github/skills/ui-styling/canvas-fonts/InstrumentSans-Bold.ttf +0 -0
  303. package/.github/skills/ui-styling/canvas-fonts/InstrumentSans-BoldItalic.ttf +0 -0
  304. package/.github/skills/ui-styling/canvas-fonts/InstrumentSans-Italic.ttf +0 -0
  305. package/.github/skills/ui-styling/canvas-fonts/InstrumentSans-OFL.txt +93 -0
  306. package/.github/skills/ui-styling/canvas-fonts/InstrumentSans-Regular.ttf +0 -0
  307. package/.github/skills/ui-styling/canvas-fonts/InstrumentSerif-Italic.ttf +0 -0
  308. package/.github/skills/ui-styling/canvas-fonts/InstrumentSerif-Regular.ttf +0 -0
  309. package/.github/skills/ui-styling/canvas-fonts/Italiana-OFL.txt +93 -0
  310. package/.github/skills/ui-styling/canvas-fonts/Italiana-Regular.ttf +0 -0
  311. package/.github/skills/ui-styling/canvas-fonts/JetBrainsMono-Bold.ttf +0 -0
  312. package/.github/skills/ui-styling/canvas-fonts/JetBrainsMono-OFL.txt +93 -0
  313. package/.github/skills/ui-styling/canvas-fonts/JetBrainsMono-Regular.ttf +0 -0
  314. package/.github/skills/ui-styling/canvas-fonts/Jura-Light.ttf +0 -0
  315. package/.github/skills/ui-styling/canvas-fonts/Jura-Medium.ttf +0 -0
  316. package/.github/skills/ui-styling/canvas-fonts/Jura-OFL.txt +93 -0
  317. package/.github/skills/ui-styling/canvas-fonts/LibreBaskerville-OFL.txt +93 -0
  318. package/.github/skills/ui-styling/canvas-fonts/LibreBaskerville-Regular.ttf +0 -0
  319. package/.github/skills/ui-styling/canvas-fonts/Lora-Bold.ttf +0 -0
  320. package/.github/skills/ui-styling/canvas-fonts/Lora-BoldItalic.ttf +0 -0
  321. package/.github/skills/ui-styling/canvas-fonts/Lora-Italic.ttf +0 -0
  322. package/.github/skills/ui-styling/canvas-fonts/Lora-OFL.txt +93 -0
  323. package/.github/skills/ui-styling/canvas-fonts/Lora-Regular.ttf +0 -0
  324. package/.github/skills/ui-styling/canvas-fonts/NationalPark-Bold.ttf +0 -0
  325. package/.github/skills/ui-styling/canvas-fonts/NationalPark-OFL.txt +93 -0
  326. package/.github/skills/ui-styling/canvas-fonts/NationalPark-Regular.ttf +0 -0
  327. package/.github/skills/ui-styling/canvas-fonts/NothingYouCouldDo-OFL.txt +93 -0
  328. package/.github/skills/ui-styling/canvas-fonts/NothingYouCouldDo-Regular.ttf +0 -0
  329. package/.github/skills/ui-styling/canvas-fonts/Outfit-Bold.ttf +0 -0
  330. package/.github/skills/ui-styling/canvas-fonts/Outfit-OFL.txt +93 -0
  331. package/.github/skills/ui-styling/canvas-fonts/Outfit-Regular.ttf +0 -0
  332. package/.github/skills/ui-styling/canvas-fonts/PixelifySans-Medium.ttf +0 -0
  333. package/.github/skills/ui-styling/canvas-fonts/PixelifySans-OFL.txt +93 -0
  334. package/.github/skills/ui-styling/canvas-fonts/PoiretOne-OFL.txt +93 -0
  335. package/.github/skills/ui-styling/canvas-fonts/PoiretOne-Regular.ttf +0 -0
  336. package/.github/skills/ui-styling/canvas-fonts/RedHatMono-Bold.ttf +0 -0
  337. package/.github/skills/ui-styling/canvas-fonts/RedHatMono-OFL.txt +93 -0
  338. package/.github/skills/ui-styling/canvas-fonts/RedHatMono-Regular.ttf +0 -0
  339. package/.github/skills/ui-styling/canvas-fonts/Silkscreen-OFL.txt +93 -0
  340. package/.github/skills/ui-styling/canvas-fonts/Silkscreen-Regular.ttf +0 -0
  341. package/.github/skills/ui-styling/canvas-fonts/SmoochSans-Medium.ttf +0 -0
  342. package/.github/skills/ui-styling/canvas-fonts/SmoochSans-OFL.txt +93 -0
  343. package/.github/skills/ui-styling/canvas-fonts/Tektur-Medium.ttf +0 -0
  344. package/.github/skills/ui-styling/canvas-fonts/Tektur-OFL.txt +93 -0
  345. package/.github/skills/ui-styling/canvas-fonts/Tektur-Regular.ttf +0 -0
  346. package/.github/skills/ui-styling/canvas-fonts/WorkSans-Bold.ttf +0 -0
  347. package/.github/skills/ui-styling/canvas-fonts/WorkSans-BoldItalic.ttf +0 -0
  348. package/.github/skills/ui-styling/canvas-fonts/WorkSans-Italic.ttf +0 -0
  349. package/.github/skills/ui-styling/canvas-fonts/WorkSans-OFL.txt +93 -0
  350. package/.github/skills/ui-styling/canvas-fonts/WorkSans-Regular.ttf +0 -0
  351. package/.github/skills/ui-styling/canvas-fonts/YoungSerif-OFL.txt +93 -0
  352. package/.github/skills/ui-styling/canvas-fonts/YoungSerif-Regular.ttf +0 -0
  353. package/.github/skills/ui-styling/scripts/.coverage +0 -0
  354. package/.github/skills/ui-styling/scripts/requirements.txt +17 -0
  355. package/.github/skills/ui-styling/scripts/shadcn_add.py +292 -0
  356. package/.github/skills/ui-styling/scripts/tailwind_config_gen.py +456 -0
  357. package/.github/skills/ui-ux-pro-max/data/charts.csv +26 -0
  358. package/.github/skills/ui-ux-pro-max/data/colors.csv +97 -0
  359. package/.github/skills/ui-ux-pro-max/data/landing.csv +31 -0
  360. package/.github/skills/ui-ux-pro-max/data/products.csv +97 -0
  361. package/.github/skills/ui-ux-pro-max/data/prompts.csv +24 -0
  362. package/.github/skills/ui-ux-pro-max/data/stacks/flutter.csv +53 -0
  363. package/.github/skills/ui-ux-pro-max/data/stacks/html-tailwind.csv +51 -0
  364. package/.github/skills/ui-ux-pro-max/data/stacks/nextjs.csv +53 -0
  365. package/.github/skills/ui-ux-pro-max/data/stacks/react-native.csv +52 -0
  366. package/.github/skills/ui-ux-pro-max/data/stacks/react.csv +54 -0
  367. package/.github/skills/ui-ux-pro-max/data/stacks/svelte.csv +54 -0
  368. package/.github/skills/ui-ux-pro-max/data/stacks/swiftui.csv +51 -0
  369. package/.github/skills/ui-ux-pro-max/data/stacks/vue.csv +50 -0
  370. package/.github/skills/ui-ux-pro-max/data/styles.csv +59 -0
  371. package/.github/skills/ui-ux-pro-max/data/typography.csv +58 -0
  372. package/.github/skills/ui-ux-pro-max/data/ux-guidelines.csv +100 -0
  373. package/.github/skills/ui-ux-pro-max/scripts/core.py +236 -0
  374. package/.github/skills/ui-ux-pro-max/scripts/search.py +76 -0
  375. package/.github/skills/web-frameworks/scripts/.coverage +0 -0
  376. package/.github/skills/web-frameworks/scripts/__init__.py +0 -0
  377. package/.github/skills/web-frameworks/scripts/nextjs_init.py +547 -0
  378. package/.github/skills/web-frameworks/scripts/requirements.txt +16 -0
  379. package/.github/skills/web-frameworks/scripts/turborepo_migrate.py +394 -0
  380. package/dist/config/constants.d.ts +2 -0
  381. package/dist/config/constants.d.ts.map +1 -1
  382. package/dist/config/constants.js +4 -1
  383. package/dist/config/constants.js.map +1 -1
  384. package/dist/domains/github/github-client.d.ts +5 -0
  385. package/dist/domains/github/github-client.d.ts.map +1 -1
  386. package/dist/domains/github/github-client.js +44 -0
  387. package/dist/domains/github/github-client.js.map +1 -1
  388. package/dist/utils/downloader.d.ts +3 -1
  389. package/dist/utils/downloader.d.ts.map +1 -1
  390. package/dist/utils/downloader.js +48 -11
  391. package/dist/utils/downloader.js.map +1 -1
  392. package/package.json +3 -1
@@ -0,0 +1,342 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Batch image resizing with multiple strategies.
4
+
5
+ Supports aspect ratio maintenance, smart cropping, thumbnail generation,
6
+ watermarks, format conversion, and parallel processing.
7
+ """
8
+
9
+ import argparse
10
+ import subprocess
11
+ import sys
12
+ from concurrent.futures import ThreadPoolExecutor, as_completed
13
+ from pathlib import Path
14
+ from typing import List, Optional, Tuple
15
+
16
+
17
+ class ImageResizer:
18
+ """Handle image resizing operations using ImageMagick."""
19
+
20
+ def __init__(self, verbose: bool = False, dry_run: bool = False):
21
+ self.verbose = verbose
22
+ self.dry_run = dry_run
23
+
24
+ def check_imagemagick(self) -> bool:
25
+ """Check if ImageMagick is available."""
26
+ try:
27
+ subprocess.run(
28
+ ['magick', '-version'],
29
+ stdout=subprocess.DEVNULL,
30
+ stderr=subprocess.DEVNULL,
31
+ check=True
32
+ )
33
+ return True
34
+ except (subprocess.CalledProcessError, FileNotFoundError):
35
+ return False
36
+
37
+ def build_resize_command(
38
+ self,
39
+ input_path: Path,
40
+ output_path: Path,
41
+ width: Optional[int],
42
+ height: Optional[int],
43
+ strategy: str,
44
+ quality: int,
45
+ watermark: Optional[Path] = None
46
+ ) -> List[str]:
47
+ """Build ImageMagick resize command based on strategy."""
48
+ cmd = ['magick', str(input_path)]
49
+
50
+ # Apply resize strategy
51
+ if strategy == 'fit':
52
+ # Fit within dimensions, maintain aspect ratio
53
+ geometry = f"{width or ''}x{height or ''}"
54
+ cmd.extend(['-resize', geometry])
55
+
56
+ elif strategy == 'fill':
57
+ # Fill dimensions, crop excess
58
+ if not width or not height:
59
+ raise ValueError("Both width and height required for 'fill' strategy")
60
+ cmd.extend([
61
+ '-resize', f'{width}x{height}^',
62
+ '-gravity', 'center',
63
+ '-extent', f'{width}x{height}'
64
+ ])
65
+
66
+ elif strategy == 'cover':
67
+ # Cover dimensions, may exceed
68
+ if not width or not height:
69
+ raise ValueError("Both width and height required for 'cover' strategy")
70
+ cmd.extend(['-resize', f'{width}x{height}^'])
71
+
72
+ elif strategy == 'exact':
73
+ # Force exact dimensions, ignore aspect ratio
74
+ if not width or not height:
75
+ raise ValueError("Both width and height required for 'exact' strategy")
76
+ cmd.extend(['-resize', f'{width}x{height}!'])
77
+
78
+ elif strategy == 'thumbnail':
79
+ # Create square thumbnail
80
+ size = width or height or 200
81
+ cmd.extend([
82
+ '-resize', f'{size}x{size}^',
83
+ '-gravity', 'center',
84
+ '-extent', f'{size}x{size}'
85
+ ])
86
+
87
+ # Add watermark if specified
88
+ if watermark:
89
+ cmd.extend([
90
+ str(watermark),
91
+ '-gravity', 'southeast',
92
+ '-geometry', '+10+10',
93
+ '-composite'
94
+ ])
95
+
96
+ # Output settings
97
+ cmd.extend([
98
+ '-quality', str(quality),
99
+ '-strip',
100
+ str(output_path)
101
+ ])
102
+
103
+ return cmd
104
+
105
+ def resize_image(
106
+ self,
107
+ input_path: Path,
108
+ output_path: Path,
109
+ width: Optional[int],
110
+ height: Optional[int],
111
+ strategy: str = 'fit',
112
+ quality: int = 85,
113
+ watermark: Optional[Path] = None
114
+ ) -> bool:
115
+ """Resize a single image."""
116
+ try:
117
+ # Ensure output directory exists
118
+ output_path.parent.mkdir(parents=True, exist_ok=True)
119
+
120
+ cmd = self.build_resize_command(
121
+ input_path, output_path, width, height,
122
+ strategy, quality, watermark
123
+ )
124
+
125
+ if self.verbose or self.dry_run:
126
+ print(f"Command: {' '.join(cmd)}")
127
+
128
+ if self.dry_run:
129
+ return True
130
+
131
+ subprocess.run(
132
+ cmd,
133
+ stdout=subprocess.PIPE if not self.verbose else None,
134
+ stderr=subprocess.PIPE if not self.verbose else None,
135
+ check=True
136
+ )
137
+ return True
138
+
139
+ except subprocess.CalledProcessError as e:
140
+ print(f"Error resizing {input_path}: {e}", file=sys.stderr)
141
+ if not self.verbose and e.stderr:
142
+ print(e.stderr.decode(), file=sys.stderr)
143
+ return False
144
+ except Exception as e:
145
+ print(f"Error processing {input_path}: {e}", file=sys.stderr)
146
+ return False
147
+
148
+ def batch_resize(
149
+ self,
150
+ input_paths: List[Path],
151
+ output_dir: Path,
152
+ width: Optional[int],
153
+ height: Optional[int],
154
+ strategy: str = 'fit',
155
+ quality: int = 85,
156
+ format_ext: Optional[str] = None,
157
+ watermark: Optional[Path] = None,
158
+ parallel: int = 1
159
+ ) -> Tuple[int, int]:
160
+ """Resize multiple images."""
161
+ success_count = 0
162
+ fail_count = 0
163
+
164
+ def process_image(input_path: Path) -> Tuple[Path, bool]:
165
+ """Process single image for parallel execution."""
166
+ if not input_path.exists() or not input_path.is_file():
167
+ return input_path, False
168
+
169
+ # Determine output path
170
+ output_name = input_path.stem
171
+ if format_ext:
172
+ output_path = output_dir / f"{output_name}.{format_ext.lstrip('.')}"
173
+ else:
174
+ output_path = output_dir / input_path.name
175
+
176
+ if not self.dry_run:
177
+ print(f"Processing {input_path.name} -> {output_path.name}")
178
+
179
+ success = self.resize_image(
180
+ input_path, output_path, width, height,
181
+ strategy, quality, watermark
182
+ )
183
+
184
+ return input_path, success
185
+
186
+ # Process images
187
+ if parallel > 1:
188
+ with ThreadPoolExecutor(max_workers=parallel) as executor:
189
+ futures = [executor.submit(process_image, path) for path in input_paths]
190
+
191
+ for future in as_completed(futures):
192
+ _, success = future.result()
193
+ if success:
194
+ success_count += 1
195
+ else:
196
+ fail_count += 1
197
+ else:
198
+ for input_path in input_paths:
199
+ _, success = process_image(input_path)
200
+ if success:
201
+ success_count += 1
202
+ else:
203
+ fail_count += 1
204
+
205
+ return success_count, fail_count
206
+
207
+
208
+ def collect_images(paths: List[Path], recursive: bool = False) -> List[Path]:
209
+ """Collect image files from paths."""
210
+ image_exts = {'.jpg', '.jpeg', '.png', '.gif', '.webp', '.bmp', '.tiff', '.tif'}
211
+ images = []
212
+
213
+ for path in paths:
214
+ if path.is_file() and path.suffix.lower() in image_exts:
215
+ images.append(path)
216
+ elif path.is_dir():
217
+ pattern = '**/*' if recursive else '*'
218
+ for img_path in path.glob(pattern):
219
+ if img_path.is_file() and img_path.suffix.lower() in image_exts:
220
+ images.append(img_path)
221
+
222
+ return images
223
+
224
+
225
+ def main():
226
+ """Main entry point."""
227
+ parser = argparse.ArgumentParser(
228
+ description='Batch image resizing with multiple strategies.'
229
+ )
230
+ parser.add_argument(
231
+ 'inputs',
232
+ nargs='+',
233
+ type=Path,
234
+ help='Input image(s) or directory'
235
+ )
236
+ parser.add_argument(
237
+ '-o', '--output',
238
+ type=Path,
239
+ required=True,
240
+ help='Output directory'
241
+ )
242
+ parser.add_argument(
243
+ '-w', '--width',
244
+ type=int,
245
+ help='Target width in pixels'
246
+ )
247
+ parser.add_argument(
248
+ '-h', '--height',
249
+ type=int,
250
+ dest='img_height',
251
+ help='Target height in pixels'
252
+ )
253
+ parser.add_argument(
254
+ '-s', '--strategy',
255
+ choices=['fit', 'fill', 'cover', 'exact', 'thumbnail'],
256
+ default='fit',
257
+ help='Resize strategy (default: fit)'
258
+ )
259
+ parser.add_argument(
260
+ '-q', '--quality',
261
+ type=int,
262
+ default=85,
263
+ help='Output quality 0-100 (default: 85)'
264
+ )
265
+ parser.add_argument(
266
+ '-f', '--format',
267
+ help='Output format (e.g., jpg, png, webp)'
268
+ )
269
+ parser.add_argument(
270
+ '-wm', '--watermark',
271
+ type=Path,
272
+ help='Watermark image to overlay'
273
+ )
274
+ parser.add_argument(
275
+ '-p', '--parallel',
276
+ type=int,
277
+ default=1,
278
+ help='Number of parallel processes (default: 1)'
279
+ )
280
+ parser.add_argument(
281
+ '-r', '--recursive',
282
+ action='store_true',
283
+ help='Process directories recursively'
284
+ )
285
+ parser.add_argument(
286
+ '-n', '--dry-run',
287
+ action='store_true',
288
+ help='Show commands without executing'
289
+ )
290
+ parser.add_argument(
291
+ '-v', '--verbose',
292
+ action='store_true',
293
+ help='Verbose output'
294
+ )
295
+
296
+ args = parser.parse_args()
297
+
298
+ # Validate dimensions
299
+ if not args.width and not args.img_height:
300
+ print("Error: At least one of --width or --height required", file=sys.stderr)
301
+ sys.exit(1)
302
+
303
+ # Initialize resizer
304
+ resizer = ImageResizer(verbose=args.verbose, dry_run=args.dry_run)
305
+
306
+ # Check dependencies
307
+ if not resizer.check_imagemagick():
308
+ print("Error: ImageMagick not found", file=sys.stderr)
309
+ sys.exit(1)
310
+
311
+ # Collect input images
312
+ images = collect_images(args.inputs, args.recursive)
313
+
314
+ if not images:
315
+ print("Error: No images found", file=sys.stderr)
316
+ sys.exit(1)
317
+
318
+ print(f"Found {len(images)} image(s) to process")
319
+
320
+ # Create output directory
321
+ if not args.dry_run:
322
+ args.output.mkdir(parents=True, exist_ok=True)
323
+
324
+ # Process images
325
+ success, fail = resizer.batch_resize(
326
+ images,
327
+ args.output,
328
+ args.width,
329
+ args.img_height,
330
+ args.strategy,
331
+ args.quality,
332
+ args.format,
333
+ args.watermark,
334
+ args.parallel
335
+ )
336
+
337
+ print(f"\nResults: {success} succeeded, {fail} failed")
338
+ sys.exit(0 if fail == 0 else 1)
339
+
340
+
341
+ if __name__ == '__main__':
342
+ main()
@@ -0,0 +1,311 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Unified media conversion tool for video, audio, and images.
4
+
5
+ Auto-detects format and applies appropriate tool (FFmpeg or ImageMagick).
6
+ Supports quality presets, batch processing, and dry-run mode.
7
+ """
8
+
9
+ import argparse
10
+ import subprocess
11
+ import sys
12
+ from pathlib import Path
13
+ from typing import List, Optional, Tuple
14
+
15
+
16
+ # Format mappings
17
+ VIDEO_FORMATS = {'.mp4', '.mkv', '.avi', '.mov', '.webm', '.flv', '.wmv', '.m4v'}
18
+ AUDIO_FORMATS = {'.mp3', '.aac', '.m4a', '.opus', '.flac', '.wav', '.ogg'}
19
+ IMAGE_FORMATS = {'.jpg', '.jpeg', '.png', '.gif', '.webp', '.bmp', '.tiff', '.tif'}
20
+
21
+ # Quality presets
22
+ QUALITY_PRESETS = {
23
+ 'web': {
24
+ 'video_crf': 23,
25
+ 'video_preset': 'medium',
26
+ 'audio_bitrate': '128k',
27
+ 'image_quality': 85
28
+ },
29
+ 'archive': {
30
+ 'video_crf': 18,
31
+ 'video_preset': 'slow',
32
+ 'audio_bitrate': '192k',
33
+ 'image_quality': 95
34
+ },
35
+ 'mobile': {
36
+ 'video_crf': 26,
37
+ 'video_preset': 'fast',
38
+ 'audio_bitrate': '96k',
39
+ 'image_quality': 80
40
+ }
41
+ }
42
+
43
+
44
+ def check_dependencies() -> Tuple[bool, bool]:
45
+ """Check if ffmpeg and imagemagick are available."""
46
+ ffmpeg_available = subprocess.run(
47
+ ['ffmpeg', '-version'],
48
+ stdout=subprocess.DEVNULL,
49
+ stderr=subprocess.DEVNULL
50
+ ).returncode == 0
51
+
52
+ magick_available = subprocess.run(
53
+ ['magick', '-version'],
54
+ stdout=subprocess.DEVNULL,
55
+ stderr=subprocess.DEVNULL
56
+ ).returncode == 0
57
+
58
+ return ffmpeg_available, magick_available
59
+
60
+
61
+ def detect_media_type(file_path: Path) -> str:
62
+ """Detect media type from file extension."""
63
+ ext = file_path.suffix.lower()
64
+
65
+ if ext in VIDEO_FORMATS:
66
+ return 'video'
67
+ elif ext in AUDIO_FORMATS:
68
+ return 'audio'
69
+ elif ext in IMAGE_FORMATS:
70
+ return 'image'
71
+ else:
72
+ return 'unknown'
73
+
74
+
75
+ def build_video_command(
76
+ input_path: Path,
77
+ output_path: Path,
78
+ preset: str = 'web'
79
+ ) -> List[str]:
80
+ """Build FFmpeg command for video conversion."""
81
+ quality = QUALITY_PRESETS[preset]
82
+
83
+ return [
84
+ 'ffmpeg', '-i', str(input_path),
85
+ '-c:v', 'libx264',
86
+ '-preset', quality['video_preset'],
87
+ '-crf', str(quality['video_crf']),
88
+ '-c:a', 'aac',
89
+ '-b:a', quality['audio_bitrate'],
90
+ '-movflags', '+faststart',
91
+ '-y',
92
+ str(output_path)
93
+ ]
94
+
95
+
96
+ def build_audio_command(
97
+ input_path: Path,
98
+ output_path: Path,
99
+ preset: str = 'web'
100
+ ) -> List[str]:
101
+ """Build FFmpeg command for audio conversion."""
102
+ quality = QUALITY_PRESETS[preset]
103
+ output_ext = output_path.suffix.lower()
104
+
105
+ codec_map = {
106
+ '.mp3': 'libmp3lame',
107
+ '.aac': 'aac',
108
+ '.m4a': 'aac',
109
+ '.opus': 'libopus',
110
+ '.flac': 'flac',
111
+ '.wav': 'pcm_s16le',
112
+ '.ogg': 'libvorbis'
113
+ }
114
+
115
+ codec = codec_map.get(output_ext, 'aac')
116
+
117
+ cmd = ['ffmpeg', '-i', str(input_path), '-c:a', codec]
118
+
119
+ # Add bitrate for lossy codecs
120
+ if codec not in ['flac', 'pcm_s16le']:
121
+ cmd.extend(['-b:a', quality['audio_bitrate']])
122
+
123
+ cmd.extend(['-y', str(output_path)])
124
+ return cmd
125
+
126
+
127
+ def build_image_command(
128
+ input_path: Path,
129
+ output_path: Path,
130
+ preset: str = 'web'
131
+ ) -> List[str]:
132
+ """Build ImageMagick command for image conversion."""
133
+ quality = QUALITY_PRESETS[preset]
134
+
135
+ return [
136
+ 'magick', str(input_path),
137
+ '-quality', str(quality['image_quality']),
138
+ '-strip',
139
+ str(output_path)
140
+ ]
141
+
142
+
143
+ def convert_file(
144
+ input_path: Path,
145
+ output_path: Path,
146
+ preset: str = 'web',
147
+ dry_run: bool = False,
148
+ verbose: bool = False
149
+ ) -> bool:
150
+ """Convert a single media file."""
151
+ media_type = detect_media_type(input_path)
152
+
153
+ if media_type == 'unknown':
154
+ print(f"Error: Unsupported format for {input_path}", file=sys.stderr)
155
+ return False
156
+
157
+ # Ensure output directory exists
158
+ output_path.parent.mkdir(parents=True, exist_ok=True)
159
+
160
+ # Build command based on media type
161
+ if media_type == 'video':
162
+ cmd = build_video_command(input_path, output_path, preset)
163
+ elif media_type == 'audio':
164
+ cmd = build_audio_command(input_path, output_path, preset)
165
+ else: # image
166
+ cmd = build_image_command(input_path, output_path, preset)
167
+
168
+ if verbose or dry_run:
169
+ print(f"Command: {' '.join(cmd)}")
170
+
171
+ if dry_run:
172
+ return True
173
+
174
+ try:
175
+ result = subprocess.run(
176
+ cmd,
177
+ stdout=subprocess.PIPE if not verbose else None,
178
+ stderr=subprocess.PIPE if not verbose else None,
179
+ check=True
180
+ )
181
+ return True
182
+ except subprocess.CalledProcessError as e:
183
+ print(f"Error converting {input_path}: {e}", file=sys.stderr)
184
+ if not verbose and e.stderr:
185
+ print(e.stderr.decode(), file=sys.stderr)
186
+ return False
187
+ except Exception as e:
188
+ print(f"Error converting {input_path}: {e}", file=sys.stderr)
189
+ return False
190
+
191
+
192
+ def batch_convert(
193
+ input_paths: List[Path],
194
+ output_dir: Optional[Path] = None,
195
+ output_format: Optional[str] = None,
196
+ preset: str = 'web',
197
+ dry_run: bool = False,
198
+ verbose: bool = False
199
+ ) -> Tuple[int, int]:
200
+ """Convert multiple files."""
201
+ success_count = 0
202
+ fail_count = 0
203
+
204
+ for input_path in input_paths:
205
+ if not input_path.exists():
206
+ print(f"Error: {input_path} not found", file=sys.stderr)
207
+ fail_count += 1
208
+ continue
209
+
210
+ # Determine output path
211
+ if output_dir:
212
+ output_name = input_path.stem
213
+ if output_format:
214
+ output_path = output_dir / f"{output_name}.{output_format.lstrip('.')}"
215
+ else:
216
+ output_path = output_dir / input_path.name
217
+ else:
218
+ if output_format:
219
+ output_path = input_path.with_suffix(f".{output_format.lstrip('.')}")
220
+ else:
221
+ print(f"Error: No output format specified for {input_path}", file=sys.stderr)
222
+ fail_count += 1
223
+ continue
224
+
225
+ print(f"Converting {input_path.name} -> {output_path.name}")
226
+
227
+ if convert_file(input_path, output_path, preset, dry_run, verbose):
228
+ success_count += 1
229
+ else:
230
+ fail_count += 1
231
+
232
+ return success_count, fail_count
233
+
234
+
235
+ def main():
236
+ """Main entry point."""
237
+ parser = argparse.ArgumentParser(
238
+ description='Unified media conversion tool for video, audio, and images.'
239
+ )
240
+ parser.add_argument(
241
+ 'inputs',
242
+ nargs='+',
243
+ type=Path,
244
+ help='Input file(s) to convert'
245
+ )
246
+ parser.add_argument(
247
+ '-o', '--output',
248
+ type=Path,
249
+ help='Output file or directory for batch conversion'
250
+ )
251
+ parser.add_argument(
252
+ '-f', '--format',
253
+ help='Output format (e.g., mp4, jpg, mp3)'
254
+ )
255
+ parser.add_argument(
256
+ '-p', '--preset',
257
+ choices=['web', 'archive', 'mobile'],
258
+ default='web',
259
+ help='Quality preset (default: web)'
260
+ )
261
+ parser.add_argument(
262
+ '-n', '--dry-run',
263
+ action='store_true',
264
+ help='Show commands without executing'
265
+ )
266
+ parser.add_argument(
267
+ '-v', '--verbose',
268
+ action='store_true',
269
+ help='Verbose output'
270
+ )
271
+
272
+ args = parser.parse_args()
273
+
274
+ # Check dependencies
275
+ ffmpeg_ok, magick_ok = check_dependencies()
276
+ if not ffmpeg_ok and not magick_ok:
277
+ print("Error: Neither ffmpeg nor imagemagick found", file=sys.stderr)
278
+ sys.exit(1)
279
+
280
+ # Handle single file vs batch conversion
281
+ if len(args.inputs) == 1 and args.output and not args.output.is_dir():
282
+ # Single file conversion
283
+ success = convert_file(
284
+ args.inputs[0],
285
+ args.output,
286
+ args.preset,
287
+ args.dry_run,
288
+ args.verbose
289
+ )
290
+ sys.exit(0 if success else 1)
291
+ else:
292
+ # Batch conversion
293
+ output_dir = args.output if args.output else Path.cwd()
294
+ if not args.output:
295
+ output_dir = None # Will convert in place with new format
296
+
297
+ success, fail = batch_convert(
298
+ args.inputs,
299
+ output_dir,
300
+ args.format,
301
+ args.preset,
302
+ args.dry_run,
303
+ args.verbose
304
+ )
305
+
306
+ print(f"\nResults: {success} succeeded, {fail} failed")
307
+ sys.exit(0 if fail == 0 else 1)
308
+
309
+
310
+ if __name__ == '__main__':
311
+ main()