@aslomon/effectum 0.3.3 → 0.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bin/install.js +68 -4
- package/bin/lib/cli-tools.js +288 -0
- package/bin/lib/specializations.js +11 -1
- package/bin/lib/template.js +14 -0
- package/bin/lib/ui.js +109 -0
- package/package.json +1 -1
- package/system/agents/data-engineer.md +268 -0
- package/system/agents/mobile-developer.md +257 -0
- package/system/hooks/observe.sh +164 -0
- package/system/hooks/quick-diff.sh +87 -0
- package/system/hooks/save-results.sh +56 -0
- package/system/hooks/scan.sh +170 -0
- package/system/skills/README.md +86 -0
- package/system/skills/SKILL.md +117 -0
- package/system/skills/agents/observer.md +137 -0
- package/system/skills/agents/start-observer.sh +143 -0
- package/system/skills/algorithmic-art/LICENSE.txt +202 -0
- package/system/skills/algorithmic-art/SKILL.md +405 -0
- package/system/skills/algorithmic-art/templates/generator_template.js +223 -0
- package/system/skills/algorithmic-art/templates/viewer.html +599 -0
- package/system/skills/api-endpoint/SKILL.md +106 -0
- package/system/skills/canvas-design/LICENSE.txt +202 -0
- package/system/skills/canvas-design/SKILL.md +130 -0
- package/system/skills/canvas-design/canvas-fonts/ArsenalSC-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/ArsenalSC-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/BigShoulders-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/BigShoulders-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/BigShoulders-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Boldonse-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/Boldonse-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/BricolageGrotesque-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/BricolageGrotesque-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/BricolageGrotesque-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/CrimsonPro-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/CrimsonPro-Italic.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/CrimsonPro-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/CrimsonPro-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/DMMono-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/DMMono-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/EricaOne-OFL.txt +94 -0
- package/system/skills/canvas-design/canvas-fonts/EricaOne-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/GeistMono-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/GeistMono-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/GeistMono-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Gloock-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/Gloock-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/IBMPlexMono-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/IBMPlexMono-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/IBMPlexMono-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/IBMPlexSerif-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/IBMPlexSerif-BoldItalic.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/IBMPlexSerif-Italic.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/IBMPlexSerif-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/InstrumentSans-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/InstrumentSans-BoldItalic.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/InstrumentSans-Italic.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/InstrumentSans-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/InstrumentSans-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/InstrumentSerif-Italic.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/InstrumentSerif-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Italiana-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/Italiana-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/JetBrainsMono-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/JetBrainsMono-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/JetBrainsMono-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Jura-Light.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Jura-Medium.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Jura-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/LibreBaskerville-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/LibreBaskerville-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Lora-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Lora-BoldItalic.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Lora-Italic.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Lora-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/Lora-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/NationalPark-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/NationalPark-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/NationalPark-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/NothingYouCouldDo-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/NothingYouCouldDo-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Outfit-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Outfit-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/Outfit-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/PixelifySans-Medium.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/PixelifySans-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/PoiretOne-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/PoiretOne-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/RedHatMono-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/RedHatMono-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/RedHatMono-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Silkscreen-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/Silkscreen-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/SmoochSans-Medium.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/SmoochSans-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/Tektur-Medium.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/Tektur-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/Tektur-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/WorkSans-Bold.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/WorkSans-BoldItalic.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/WorkSans-Italic.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/WorkSans-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/WorkSans-Regular.ttf +0 -0
- package/system/skills/canvas-design/canvas-fonts/YoungSerif-OFL.txt +93 -0
- package/system/skills/canvas-design/canvas-fonts/YoungSerif-Regular.ttf +0 -0
- package/system/skills/component/SKILL.md +103 -0
- package/system/skills/config.json +18 -0
- package/system/skills/doc-coauthoring/SKILL.md +375 -0
- package/system/skills/docx/LICENSE.txt +30 -0
- package/system/skills/docx/SKILL.md +590 -0
- package/system/skills/docx/scripts/__init__.py +1 -0
- package/system/skills/docx/scripts/accept_changes.py +135 -0
- package/system/skills/docx/scripts/comment.py +318 -0
- package/system/skills/docx/scripts/office/helpers/__init__.py +0 -0
- package/system/skills/docx/scripts/office/helpers/merge_runs.py +199 -0
- package/system/skills/docx/scripts/office/helpers/simplify_redlines.py +197 -0
- package/system/skills/docx/scripts/office/pack.py +159 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
- package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
- package/system/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
- package/system/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
- package/system/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
- package/system/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
- package/system/skills/docx/scripts/office/schemas/mce/mc.xsd +75 -0
- package/system/skills/docx/scripts/office/schemas/microsoft/wml-2010.xsd +560 -0
- package/system/skills/docx/scripts/office/schemas/microsoft/wml-2012.xsd +67 -0
- package/system/skills/docx/scripts/office/schemas/microsoft/wml-2018.xsd +14 -0
- package/system/skills/docx/scripts/office/schemas/microsoft/wml-cex-2018.xsd +20 -0
- package/system/skills/docx/scripts/office/schemas/microsoft/wml-cid-2016.xsd +13 -0
- package/system/skills/docx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
- package/system/skills/docx/scripts/office/schemas/microsoft/wml-symex-2015.xsd +8 -0
- package/system/skills/docx/scripts/office/soffice.py +183 -0
- package/system/skills/docx/scripts/office/unpack.py +132 -0
- package/system/skills/docx/scripts/office/validate.py +111 -0
- package/system/skills/docx/scripts/office/validators/__init__.py +15 -0
- package/system/skills/docx/scripts/office/validators/base.py +847 -0
- package/system/skills/docx/scripts/office/validators/docx.py +446 -0
- package/system/skills/docx/scripts/office/validators/pptx.py +275 -0
- package/system/skills/docx/scripts/office/validators/redlining.py +247 -0
- package/system/skills/docx/scripts/templates/comments.xml +3 -0
- package/system/skills/docx/scripts/templates/commentsExtended.xml +3 -0
- package/system/skills/docx/scripts/templates/commentsExtensible.xml +3 -0
- package/system/skills/docx/scripts/templates/commentsIds.xml +3 -0
- package/system/skills/docx/scripts/templates/people.xml +3 -0
- package/system/skills/evaluate-session.sh +69 -0
- package/system/skills/feature/SKILL.md +73 -0
- package/system/skills/frontend-design/LICENSE.txt +177 -0
- package/system/skills/frontend-design/SKILL.md +52 -0
- package/system/skills/hooks/observe.sh +164 -0
- package/system/skills/mcp-builder/LICENSE.txt +202 -0
- package/system/skills/mcp-builder/SKILL.md +236 -0
- package/system/skills/mcp-builder/reference/evaluation.md +602 -0
- package/system/skills/mcp-builder/reference/mcp_best_practices.md +249 -0
- package/system/skills/mcp-builder/reference/node_mcp_server.md +970 -0
- package/system/skills/mcp-builder/reference/python_mcp_server.md +719 -0
- package/system/skills/mcp-builder/scripts/connections.py +151 -0
- package/system/skills/mcp-builder/scripts/evaluation.py +373 -0
- package/system/skills/mcp-builder/scripts/example_evaluation.xml +22 -0
- package/system/skills/mcp-builder/scripts/requirements.txt +2 -0
- package/system/skills/pdf/LICENSE.txt +30 -0
- package/system/skills/pdf/SKILL.md +314 -0
- package/system/skills/pdf/forms.md +294 -0
- package/system/skills/pdf/reference.md +612 -0
- package/system/skills/pdf/scripts/check_bounding_boxes.py +65 -0
- package/system/skills/pdf/scripts/check_fillable_fields.py +11 -0
- package/system/skills/pdf/scripts/convert_pdf_to_images.py +33 -0
- package/system/skills/pdf/scripts/create_validation_image.py +37 -0
- package/system/skills/pdf/scripts/extract_form_field_info.py +122 -0
- package/system/skills/pdf/scripts/extract_form_structure.py +115 -0
- package/system/skills/pdf/scripts/fill_fillable_fields.py +98 -0
- package/system/skills/pdf/scripts/fill_pdf_form_with_annotations.py +107 -0
- package/system/skills/pptx/LICENSE.txt +30 -0
- package/system/skills/pptx/SKILL.md +232 -0
- package/system/skills/pptx/editing.md +205 -0
- package/system/skills/pptx/pptxgenjs.md +420 -0
- package/system/skills/pptx/scripts/__init__.py +0 -0
- package/system/skills/pptx/scripts/add_slide.py +195 -0
- package/system/skills/pptx/scripts/clean.py +286 -0
- package/system/skills/pptx/scripts/office/helpers/__init__.py +0 -0
- package/system/skills/pptx/scripts/office/helpers/merge_runs.py +199 -0
- package/system/skills/pptx/scripts/office/helpers/simplify_redlines.py +197 -0
- package/system/skills/pptx/scripts/office/pack.py +159 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
- package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
- package/system/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
- package/system/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
- package/system/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
- package/system/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
- package/system/skills/pptx/scripts/office/schemas/mce/mc.xsd +75 -0
- package/system/skills/pptx/scripts/office/schemas/microsoft/wml-2010.xsd +560 -0
- package/system/skills/pptx/scripts/office/schemas/microsoft/wml-2012.xsd +67 -0
- package/system/skills/pptx/scripts/office/schemas/microsoft/wml-2018.xsd +14 -0
- package/system/skills/pptx/scripts/office/schemas/microsoft/wml-cex-2018.xsd +20 -0
- package/system/skills/pptx/scripts/office/schemas/microsoft/wml-cid-2016.xsd +13 -0
- package/system/skills/pptx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
- package/system/skills/pptx/scripts/office/schemas/microsoft/wml-symex-2015.xsd +8 -0
- package/system/skills/pptx/scripts/office/soffice.py +183 -0
- package/system/skills/pptx/scripts/office/unpack.py +132 -0
- package/system/skills/pptx/scripts/office/validate.py +111 -0
- package/system/skills/pptx/scripts/office/validators/__init__.py +15 -0
- package/system/skills/pptx/scripts/office/validators/base.py +847 -0
- package/system/skills/pptx/scripts/office/validators/docx.py +446 -0
- package/system/skills/pptx/scripts/office/validators/pptx.py +275 -0
- package/system/skills/pptx/scripts/office/validators/redlining.py +247 -0
- package/system/skills/pptx/scripts/thumbnail.py +289 -0
- package/system/skills/scripts/quick-diff.sh +87 -0
- package/system/skills/scripts/save-results.sh +56 -0
- package/system/skills/scripts/scan.sh +170 -0
- package/system/skills/security-check/SKILL.md +70 -0
- package/system/skills/skill-creator/LICENSE.txt +202 -0
- package/system/skills/skill-creator/SKILL.md +479 -0
- package/system/skills/skill-creator/agents/analyzer.md +274 -0
- package/system/skills/skill-creator/agents/comparator.md +202 -0
- package/system/skills/skill-creator/agents/grader.md +223 -0
- package/system/skills/skill-creator/assets/eval_review.html +146 -0
- package/system/skills/skill-creator/eval-viewer/generate_review.py +471 -0
- package/system/skills/skill-creator/eval-viewer/viewer.html +1325 -0
- package/system/skills/skill-creator/references/schemas.md +430 -0
- package/system/skills/skill-creator/scripts/__init__.py +0 -0
- package/system/skills/skill-creator/scripts/aggregate_benchmark.py +401 -0
- package/system/skills/skill-creator/scripts/generate_report.py +326 -0
- package/system/skills/skill-creator/scripts/improve_description.py +248 -0
- package/system/skills/skill-creator/scripts/package_skill.py +136 -0
- package/system/skills/skill-creator/scripts/quick_validate.py +103 -0
- package/system/skills/skill-creator/scripts/run_eval.py +310 -0
- package/system/skills/skill-creator/scripts/run_loop.py +332 -0
- package/system/skills/skill-creator/scripts/utils.py +47 -0
- package/system/skills/suggest-compact.sh +54 -0
- package/system/skills/supabase-migration/SKILL.md +59 -0
- package/system/skills/theme-factory/LICENSE.txt +202 -0
- package/system/skills/theme-factory/SKILL.md +59 -0
- package/system/skills/theme-factory/theme-showcase.pdf +0 -0
- package/system/skills/theme-factory/themes/arctic-frost.md +19 -0
- package/system/skills/theme-factory/themes/botanical-garden.md +19 -0
- package/system/skills/theme-factory/themes/desert-rose.md +19 -0
- package/system/skills/theme-factory/themes/forest-canopy.md +19 -0
- package/system/skills/theme-factory/themes/golden-hour.md +19 -0
- package/system/skills/theme-factory/themes/midnight-galaxy.md +19 -0
- package/system/skills/theme-factory/themes/modern-minimalist.md +19 -0
- package/system/skills/theme-factory/themes/ocean-depths.md +19 -0
- package/system/skills/theme-factory/themes/sunset-boulevard.md +19 -0
- package/system/skills/theme-factory/themes/tech-innovation.md +19 -0
- package/system/skills/web-artifacts-builder/LICENSE.txt +202 -0
- package/system/skills/web-artifacts-builder/SKILL.md +74 -0
- package/system/skills/web-artifacts-builder/scripts/bundle-artifact.sh +54 -0
- package/system/skills/web-artifacts-builder/scripts/init-artifact.sh +322 -0
- package/system/skills/web-artifacts-builder/scripts/shadcn-components.tar.gz +0 -0
- package/system/skills/webapp-testing/LICENSE.txt +202 -0
- package/system/skills/webapp-testing/SKILL.md +96 -0
- package/system/skills/webapp-testing/examples/console_logging.py +35 -0
- package/system/skills/webapp-testing/examples/element_discovery.py +40 -0
- package/system/skills/webapp-testing/examples/static_html_automation.py +33 -0
- package/system/skills/webapp-testing/scripts/with_server.py +106 -0
- package/system/skills/xlsx/LICENSE.txt +30 -0
- package/system/skills/xlsx/SKILL.md +292 -0
- package/system/skills/xlsx/scripts/office/helpers/__init__.py +0 -0
- package/system/skills/xlsx/scripts/office/helpers/merge_runs.py +199 -0
- package/system/skills/xlsx/scripts/office/helpers/simplify_redlines.py +197 -0
- package/system/skills/xlsx/scripts/office/pack.py +159 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
- package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
- package/system/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
- package/system/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
- package/system/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
- package/system/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
- package/system/skills/xlsx/scripts/office/schemas/mce/mc.xsd +75 -0
- package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-2010.xsd +560 -0
- package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-2012.xsd +67 -0
- package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-2018.xsd +14 -0
- package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-cex-2018.xsd +20 -0
- package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-cid-2016.xsd +13 -0
- package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
- package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-symex-2015.xsd +8 -0
- package/system/skills/xlsx/scripts/office/soffice.py +183 -0
- package/system/skills/xlsx/scripts/office/unpack.py +132 -0
- package/system/skills/xlsx/scripts/office/validate.py +111 -0
- package/system/skills/xlsx/scripts/office/validators/__init__.py +15 -0
- package/system/skills/xlsx/scripts/office/validators/base.py +847 -0
- package/system/skills/xlsx/scripts/office/validators/docx.py +446 -0
- package/system/skills/xlsx/scripts/office/validators/pptx.py +275 -0
- package/system/skills/xlsx/scripts/office/validators/redlining.py +247 -0
- package/system/skills/xlsx/scripts/recalc.py +184 -0
- package/system/teams/profiles.md +76 -0
- package/system/templates/CLAUDE.md.tmpl +6 -0
- package/system/templates/settings.json.tmpl +11 -0
|
@@ -0,0 +1,289 @@
|
|
|
1
|
+
"""Create thumbnail grids from PowerPoint presentation slides.
|
|
2
|
+
|
|
3
|
+
Creates a grid layout of slide thumbnails for quick visual analysis.
|
|
4
|
+
Labels each thumbnail with its XML filename (e.g., slide1.xml).
|
|
5
|
+
Hidden slides are shown with a placeholder pattern.
|
|
6
|
+
|
|
7
|
+
Usage:
|
|
8
|
+
python thumbnail.py input.pptx [output_prefix] [--cols N]
|
|
9
|
+
|
|
10
|
+
Examples:
|
|
11
|
+
python thumbnail.py presentation.pptx
|
|
12
|
+
# Creates: thumbnails.jpg
|
|
13
|
+
|
|
14
|
+
python thumbnail.py template.pptx grid --cols 4
|
|
15
|
+
# Creates: grid.jpg (or grid-1.jpg, grid-2.jpg for large decks)
|
|
16
|
+
"""
|
|
17
|
+
|
|
18
|
+
import argparse
|
|
19
|
+
import subprocess
|
|
20
|
+
import sys
|
|
21
|
+
import tempfile
|
|
22
|
+
import zipfile
|
|
23
|
+
from pathlib import Path
|
|
24
|
+
|
|
25
|
+
import defusedxml.minidom
|
|
26
|
+
from office.soffice import get_soffice_env
|
|
27
|
+
from PIL import Image, ImageDraw, ImageFont
|
|
28
|
+
|
|
29
|
+
THUMBNAIL_WIDTH = 300
|
|
30
|
+
CONVERSION_DPI = 100
|
|
31
|
+
MAX_COLS = 6
|
|
32
|
+
DEFAULT_COLS = 3
|
|
33
|
+
JPEG_QUALITY = 95
|
|
34
|
+
GRID_PADDING = 20
|
|
35
|
+
BORDER_WIDTH = 2
|
|
36
|
+
FONT_SIZE_RATIO = 0.10
|
|
37
|
+
LABEL_PADDING_RATIO = 0.4
|
|
38
|
+
|
|
39
|
+
|
|
40
|
+
def main():
|
|
41
|
+
parser = argparse.ArgumentParser(
|
|
42
|
+
description="Create thumbnail grids from PowerPoint slides."
|
|
43
|
+
)
|
|
44
|
+
parser.add_argument("input", help="Input PowerPoint file (.pptx)")
|
|
45
|
+
parser.add_argument(
|
|
46
|
+
"output_prefix",
|
|
47
|
+
nargs="?",
|
|
48
|
+
default="thumbnails",
|
|
49
|
+
help="Output prefix for image files (default: thumbnails)",
|
|
50
|
+
)
|
|
51
|
+
parser.add_argument(
|
|
52
|
+
"--cols",
|
|
53
|
+
type=int,
|
|
54
|
+
default=DEFAULT_COLS,
|
|
55
|
+
help=f"Number of columns (default: {DEFAULT_COLS}, max: {MAX_COLS})",
|
|
56
|
+
)
|
|
57
|
+
|
|
58
|
+
args = parser.parse_args()
|
|
59
|
+
|
|
60
|
+
cols = min(args.cols, MAX_COLS)
|
|
61
|
+
if args.cols > MAX_COLS:
|
|
62
|
+
print(f"Warning: Columns limited to {MAX_COLS}")
|
|
63
|
+
|
|
64
|
+
input_path = Path(args.input)
|
|
65
|
+
if not input_path.exists() or input_path.suffix.lower() != ".pptx":
|
|
66
|
+
print(f"Error: Invalid PowerPoint file: {args.input}", file=sys.stderr)
|
|
67
|
+
sys.exit(1)
|
|
68
|
+
|
|
69
|
+
output_path = Path(f"{args.output_prefix}.jpg")
|
|
70
|
+
|
|
71
|
+
try:
|
|
72
|
+
slide_info = get_slide_info(input_path)
|
|
73
|
+
|
|
74
|
+
with tempfile.TemporaryDirectory() as temp_dir:
|
|
75
|
+
temp_path = Path(temp_dir)
|
|
76
|
+
visible_images = convert_to_images(input_path, temp_path)
|
|
77
|
+
|
|
78
|
+
if not visible_images and not any(s["hidden"] for s in slide_info):
|
|
79
|
+
print("Error: No slides found", file=sys.stderr)
|
|
80
|
+
sys.exit(1)
|
|
81
|
+
|
|
82
|
+
slides = build_slide_list(slide_info, visible_images, temp_path)
|
|
83
|
+
|
|
84
|
+
grid_files = create_grids(slides, cols, THUMBNAIL_WIDTH, output_path)
|
|
85
|
+
|
|
86
|
+
print(f"Created {len(grid_files)} grid(s):")
|
|
87
|
+
for grid_file in grid_files:
|
|
88
|
+
print(f" {grid_file}")
|
|
89
|
+
|
|
90
|
+
except Exception as e:
|
|
91
|
+
print(f"Error: {e}", file=sys.stderr)
|
|
92
|
+
sys.exit(1)
|
|
93
|
+
|
|
94
|
+
|
|
95
|
+
def get_slide_info(pptx_path: Path) -> list[dict]:
|
|
96
|
+
with zipfile.ZipFile(pptx_path, "r") as zf:
|
|
97
|
+
rels_content = zf.read("ppt/_rels/presentation.xml.rels").decode("utf-8")
|
|
98
|
+
rels_dom = defusedxml.minidom.parseString(rels_content)
|
|
99
|
+
|
|
100
|
+
rid_to_slide = {}
|
|
101
|
+
for rel in rels_dom.getElementsByTagName("Relationship"):
|
|
102
|
+
rid = rel.getAttribute("Id")
|
|
103
|
+
target = rel.getAttribute("Target")
|
|
104
|
+
rel_type = rel.getAttribute("Type")
|
|
105
|
+
if "slide" in rel_type and target.startswith("slides/"):
|
|
106
|
+
rid_to_slide[rid] = target.replace("slides/", "")
|
|
107
|
+
|
|
108
|
+
pres_content = zf.read("ppt/presentation.xml").decode("utf-8")
|
|
109
|
+
pres_dom = defusedxml.minidom.parseString(pres_content)
|
|
110
|
+
|
|
111
|
+
slides = []
|
|
112
|
+
for sld_id in pres_dom.getElementsByTagName("p:sldId"):
|
|
113
|
+
rid = sld_id.getAttribute("r:id")
|
|
114
|
+
if rid in rid_to_slide:
|
|
115
|
+
hidden = sld_id.getAttribute("show") == "0"
|
|
116
|
+
slides.append({"name": rid_to_slide[rid], "hidden": hidden})
|
|
117
|
+
|
|
118
|
+
return slides
|
|
119
|
+
|
|
120
|
+
|
|
121
|
+
def build_slide_list(
|
|
122
|
+
slide_info: list[dict],
|
|
123
|
+
visible_images: list[Path],
|
|
124
|
+
temp_dir: Path,
|
|
125
|
+
) -> list[tuple[Path, str]]:
|
|
126
|
+
if visible_images:
|
|
127
|
+
with Image.open(visible_images[0]) as img:
|
|
128
|
+
placeholder_size = img.size
|
|
129
|
+
else:
|
|
130
|
+
placeholder_size = (1920, 1080)
|
|
131
|
+
|
|
132
|
+
slides = []
|
|
133
|
+
visible_idx = 0
|
|
134
|
+
|
|
135
|
+
for info in slide_info:
|
|
136
|
+
if info["hidden"]:
|
|
137
|
+
placeholder_path = temp_dir / f"hidden-{info['name']}.jpg"
|
|
138
|
+
placeholder_img = create_hidden_placeholder(placeholder_size)
|
|
139
|
+
placeholder_img.save(placeholder_path, "JPEG")
|
|
140
|
+
slides.append((placeholder_path, f"{info['name']} (hidden)"))
|
|
141
|
+
else:
|
|
142
|
+
if visible_idx < len(visible_images):
|
|
143
|
+
slides.append((visible_images[visible_idx], info["name"]))
|
|
144
|
+
visible_idx += 1
|
|
145
|
+
|
|
146
|
+
return slides
|
|
147
|
+
|
|
148
|
+
|
|
149
|
+
def create_hidden_placeholder(size: tuple[int, int]) -> Image.Image:
|
|
150
|
+
img = Image.new("RGB", size, color="#F0F0F0")
|
|
151
|
+
draw = ImageDraw.Draw(img)
|
|
152
|
+
line_width = max(5, min(size) // 100)
|
|
153
|
+
draw.line([(0, 0), size], fill="#CCCCCC", width=line_width)
|
|
154
|
+
draw.line([(size[0], 0), (0, size[1])], fill="#CCCCCC", width=line_width)
|
|
155
|
+
return img
|
|
156
|
+
|
|
157
|
+
|
|
158
|
+
def convert_to_images(pptx_path: Path, temp_dir: Path) -> list[Path]:
|
|
159
|
+
pdf_path = temp_dir / f"{pptx_path.stem}.pdf"
|
|
160
|
+
|
|
161
|
+
result = subprocess.run(
|
|
162
|
+
[
|
|
163
|
+
"soffice",
|
|
164
|
+
"--headless",
|
|
165
|
+
"--convert-to",
|
|
166
|
+
"pdf",
|
|
167
|
+
"--outdir",
|
|
168
|
+
str(temp_dir),
|
|
169
|
+
str(pptx_path),
|
|
170
|
+
],
|
|
171
|
+
capture_output=True,
|
|
172
|
+
text=True,
|
|
173
|
+
env=get_soffice_env(),
|
|
174
|
+
)
|
|
175
|
+
if result.returncode != 0 or not pdf_path.exists():
|
|
176
|
+
raise RuntimeError("PDF conversion failed")
|
|
177
|
+
|
|
178
|
+
result = subprocess.run(
|
|
179
|
+
[
|
|
180
|
+
"pdftoppm",
|
|
181
|
+
"-jpeg",
|
|
182
|
+
"-r",
|
|
183
|
+
str(CONVERSION_DPI),
|
|
184
|
+
str(pdf_path),
|
|
185
|
+
str(temp_dir / "slide"),
|
|
186
|
+
],
|
|
187
|
+
capture_output=True,
|
|
188
|
+
text=True,
|
|
189
|
+
)
|
|
190
|
+
if result.returncode != 0:
|
|
191
|
+
raise RuntimeError("Image conversion failed")
|
|
192
|
+
|
|
193
|
+
return sorted(temp_dir.glob("slide-*.jpg"))
|
|
194
|
+
|
|
195
|
+
|
|
196
|
+
def create_grids(
|
|
197
|
+
slides: list[tuple[Path, str]],
|
|
198
|
+
cols: int,
|
|
199
|
+
width: int,
|
|
200
|
+
output_path: Path,
|
|
201
|
+
) -> list[str]:
|
|
202
|
+
max_per_grid = cols * (cols + 1)
|
|
203
|
+
grid_files = []
|
|
204
|
+
|
|
205
|
+
for chunk_idx, start_idx in enumerate(range(0, len(slides), max_per_grid)):
|
|
206
|
+
end_idx = min(start_idx + max_per_grid, len(slides))
|
|
207
|
+
chunk_slides = slides[start_idx:end_idx]
|
|
208
|
+
|
|
209
|
+
grid = create_grid(chunk_slides, cols, width)
|
|
210
|
+
|
|
211
|
+
if len(slides) <= max_per_grid:
|
|
212
|
+
grid_filename = output_path
|
|
213
|
+
else:
|
|
214
|
+
stem = output_path.stem
|
|
215
|
+
suffix = output_path.suffix
|
|
216
|
+
grid_filename = output_path.parent / f"{stem}-{chunk_idx + 1}{suffix}"
|
|
217
|
+
|
|
218
|
+
grid_filename.parent.mkdir(parents=True, exist_ok=True)
|
|
219
|
+
grid.save(str(grid_filename), quality=JPEG_QUALITY)
|
|
220
|
+
grid_files.append(str(grid_filename))
|
|
221
|
+
|
|
222
|
+
return grid_files
|
|
223
|
+
|
|
224
|
+
|
|
225
|
+
def create_grid(
|
|
226
|
+
slides: list[tuple[Path, str]],
|
|
227
|
+
cols: int,
|
|
228
|
+
width: int,
|
|
229
|
+
) -> Image.Image:
|
|
230
|
+
font_size = int(width * FONT_SIZE_RATIO)
|
|
231
|
+
label_padding = int(font_size * LABEL_PADDING_RATIO)
|
|
232
|
+
|
|
233
|
+
with Image.open(slides[0][0]) as img:
|
|
234
|
+
aspect = img.height / img.width
|
|
235
|
+
height = int(width * aspect)
|
|
236
|
+
|
|
237
|
+
rows = (len(slides) + cols - 1) // cols
|
|
238
|
+
grid_w = cols * width + (cols + 1) * GRID_PADDING
|
|
239
|
+
grid_h = rows * (height + font_size + label_padding * 2) + (rows + 1) * GRID_PADDING
|
|
240
|
+
|
|
241
|
+
grid = Image.new("RGB", (grid_w, grid_h), "white")
|
|
242
|
+
draw = ImageDraw.Draw(grid)
|
|
243
|
+
|
|
244
|
+
try:
|
|
245
|
+
font = ImageFont.load_default(size=font_size)
|
|
246
|
+
except Exception:
|
|
247
|
+
font = ImageFont.load_default()
|
|
248
|
+
|
|
249
|
+
for i, (img_path, slide_name) in enumerate(slides):
|
|
250
|
+
row, col = i // cols, i % cols
|
|
251
|
+
x = col * width + (col + 1) * GRID_PADDING
|
|
252
|
+
y_base = (
|
|
253
|
+
row * (height + font_size + label_padding * 2) + (row + 1) * GRID_PADDING
|
|
254
|
+
)
|
|
255
|
+
|
|
256
|
+
label = slide_name
|
|
257
|
+
bbox = draw.textbbox((0, 0), label, font=font)
|
|
258
|
+
text_w = bbox[2] - bbox[0]
|
|
259
|
+
draw.text(
|
|
260
|
+
(x + (width - text_w) // 2, y_base + label_padding),
|
|
261
|
+
label,
|
|
262
|
+
fill="black",
|
|
263
|
+
font=font,
|
|
264
|
+
)
|
|
265
|
+
|
|
266
|
+
y_thumbnail = y_base + label_padding + font_size + label_padding
|
|
267
|
+
|
|
268
|
+
with Image.open(img_path) as img:
|
|
269
|
+
img.thumbnail((width, height), Image.Resampling.LANCZOS)
|
|
270
|
+
w, h = img.size
|
|
271
|
+
tx = x + (width - w) // 2
|
|
272
|
+
ty = y_thumbnail + (height - h) // 2
|
|
273
|
+
grid.paste(img, (tx, ty))
|
|
274
|
+
|
|
275
|
+
if BORDER_WIDTH > 0:
|
|
276
|
+
draw.rectangle(
|
|
277
|
+
[
|
|
278
|
+
(tx - BORDER_WIDTH, ty - BORDER_WIDTH),
|
|
279
|
+
(tx + w + BORDER_WIDTH - 1, ty + h + BORDER_WIDTH - 1),
|
|
280
|
+
],
|
|
281
|
+
outline="gray",
|
|
282
|
+
width=BORDER_WIDTH,
|
|
283
|
+
)
|
|
284
|
+
|
|
285
|
+
return grid
|
|
286
|
+
|
|
287
|
+
|
|
288
|
+
if __name__ == "__main__":
|
|
289
|
+
main()
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
# quick-diff.sh — compare skill file mtimes against results.json evaluated_at
|
|
3
|
+
# Usage: quick-diff.sh RESULTS_JSON [CWD_SKILLS_DIR]
|
|
4
|
+
# Output: JSON array of changed/new files to stdout (empty [] if no changes)
|
|
5
|
+
#
|
|
6
|
+
# When CWD_SKILLS_DIR is omitted, defaults to $PWD/.claude/skills so the
|
|
7
|
+
# script always picks up project-level skills without relying on the caller.
|
|
8
|
+
#
|
|
9
|
+
# Environment:
|
|
10
|
+
# SKILL_STOCKTAKE_GLOBAL_DIR Override ~/.claude/skills (for testing only;
|
|
11
|
+
# do not set in production — intended for bats tests)
|
|
12
|
+
# SKILL_STOCKTAKE_PROJECT_DIR Override project dir detection (for testing only)
|
|
13
|
+
|
|
14
|
+
set -euo pipefail
|
|
15
|
+
|
|
16
|
+
RESULTS_JSON="${1:-}"
|
|
17
|
+
CWD_SKILLS_DIR="${SKILL_STOCKTAKE_PROJECT_DIR:-${2:-$PWD/.claude/skills}}"
|
|
18
|
+
GLOBAL_DIR="${SKILL_STOCKTAKE_GLOBAL_DIR:-$HOME/.claude/skills}"
|
|
19
|
+
|
|
20
|
+
if [[ -z "$RESULTS_JSON" || ! -f "$RESULTS_JSON" ]]; then
|
|
21
|
+
echo "Error: RESULTS_JSON not found: ${RESULTS_JSON:-<empty>}" >&2
|
|
22
|
+
exit 1
|
|
23
|
+
fi
|
|
24
|
+
|
|
25
|
+
# Validate CWD_SKILLS_DIR looks like a .claude/skills path (defense-in-depth).
|
|
26
|
+
# Only warn when the path exists — a nonexistent path poses no traversal risk.
|
|
27
|
+
if [[ -n "$CWD_SKILLS_DIR" && -d "$CWD_SKILLS_DIR" && "$CWD_SKILLS_DIR" != */.claude/skills* ]]; then
|
|
28
|
+
echo "Warning: CWD_SKILLS_DIR does not look like a .claude/skills path: $CWD_SKILLS_DIR" >&2
|
|
29
|
+
fi
|
|
30
|
+
|
|
31
|
+
evaluated_at=$(jq -r '.evaluated_at' "$RESULTS_JSON")
|
|
32
|
+
|
|
33
|
+
# Fail fast on a missing or malformed evaluated_at rather than producing
|
|
34
|
+
# unpredictable results from ISO 8601 string comparison against "null".
|
|
35
|
+
if [[ ! "$evaluated_at" =~ ^[0-9]{4}-[0-9]{2}-[0-9]{2}T[0-9]{2}:[0-9]{2}:[0-9]{2}Z$ ]]; then
|
|
36
|
+
echo "Error: invalid or missing evaluated_at in $RESULTS_JSON: $evaluated_at" >&2
|
|
37
|
+
exit 1
|
|
38
|
+
fi
|
|
39
|
+
|
|
40
|
+
# Pre-extract known paths from results.json once (O(1) lookup per file instead of O(n*m))
|
|
41
|
+
known_paths=$(jq -r '.skills[].path' "$RESULTS_JSON" 2>/dev/null)
|
|
42
|
+
|
|
43
|
+
tmpdir=$(mktemp -d)
|
|
44
|
+
# Use a function to avoid embedding $tmpdir in a quoted string (prevents injection
|
|
45
|
+
# if TMPDIR were crafted to contain shell metacharacters).
|
|
46
|
+
_cleanup() { rm -rf "$tmpdir"; }
|
|
47
|
+
trap _cleanup EXIT
|
|
48
|
+
|
|
49
|
+
# Shared counter across process_dir calls — intentionally NOT local
|
|
50
|
+
i=0
|
|
51
|
+
|
|
52
|
+
process_dir() {
|
|
53
|
+
local dir="$1"
|
|
54
|
+
while IFS= read -r file; do
|
|
55
|
+
local mtime dp is_new
|
|
56
|
+
mtime=$(date -u -r "$file" +%Y-%m-%dT%H:%M:%SZ)
|
|
57
|
+
dp="${file/#$HOME/~}"
|
|
58
|
+
|
|
59
|
+
# Check if this file is known to results.json (exact whole-line match to
|
|
60
|
+
# avoid substring false-positives, e.g. "python-patterns" matching "python-patterns-v2").
|
|
61
|
+
if echo "$known_paths" | grep -qxF "$dp"; then
|
|
62
|
+
is_new="false"
|
|
63
|
+
# Known file: only emit if mtime changed (ISO 8601 string comparison is safe)
|
|
64
|
+
[[ "$mtime" > "$evaluated_at" ]] || continue
|
|
65
|
+
else
|
|
66
|
+
is_new="true"
|
|
67
|
+
# New file: always emit regardless of mtime
|
|
68
|
+
fi
|
|
69
|
+
|
|
70
|
+
jq -n \
|
|
71
|
+
--arg path "$dp" \
|
|
72
|
+
--arg mtime "$mtime" \
|
|
73
|
+
--argjson is_new "$is_new" \
|
|
74
|
+
'{path:$path,mtime:$mtime,is_new:$is_new}' \
|
|
75
|
+
> "$tmpdir/$i.json"
|
|
76
|
+
i=$((i+1))
|
|
77
|
+
done < <(find "$dir" -name "*.md" -type f 2>/dev/null | sort)
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
[[ -d "$GLOBAL_DIR" ]] && process_dir "$GLOBAL_DIR"
|
|
81
|
+
[[ -n "$CWD_SKILLS_DIR" && -d "$CWD_SKILLS_DIR" ]] && process_dir "$CWD_SKILLS_DIR"
|
|
82
|
+
|
|
83
|
+
if [[ $i -eq 0 ]]; then
|
|
84
|
+
echo "[]"
|
|
85
|
+
else
|
|
86
|
+
jq -s '.' "$tmpdir"/*.json
|
|
87
|
+
fi
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
# save-results.sh — merge evaluated skills into results.json with correct UTC timestamp
|
|
3
|
+
# Usage: save-results.sh RESULTS_JSON <<< "$EVAL_JSON"
|
|
4
|
+
#
|
|
5
|
+
# stdin format:
|
|
6
|
+
# { "skills": {...}, "mode"?: "full"|"quick", "batch_progress"?: {...} }
|
|
7
|
+
#
|
|
8
|
+
# Always sets evaluated_at to current UTC time via `date -u`.
|
|
9
|
+
# Merges stdin .skills into existing results.json (new entries override old).
|
|
10
|
+
# Optionally updates .mode and .batch_progress if present in stdin.
|
|
11
|
+
|
|
12
|
+
set -euo pipefail
|
|
13
|
+
|
|
14
|
+
RESULTS_JSON="${1:-}"
|
|
15
|
+
|
|
16
|
+
if [[ -z "$RESULTS_JSON" ]]; then
|
|
17
|
+
echo "Error: RESULTS_JSON argument required" >&2
|
|
18
|
+
echo "Usage: save-results.sh RESULTS_JSON <<< \"\$EVAL_JSON\"" >&2
|
|
19
|
+
exit 1
|
|
20
|
+
fi
|
|
21
|
+
|
|
22
|
+
EVALUATED_AT=$(date -u +%Y-%m-%dT%H:%M:%SZ)
|
|
23
|
+
|
|
24
|
+
# Read eval results from stdin and validate JSON before touching the results file
|
|
25
|
+
input_json=$(cat)
|
|
26
|
+
if ! echo "$input_json" | jq empty 2>/dev/null; then
|
|
27
|
+
echo "Error: stdin is not valid JSON" >&2
|
|
28
|
+
exit 1
|
|
29
|
+
fi
|
|
30
|
+
|
|
31
|
+
if [[ ! -f "$RESULTS_JSON" ]]; then
|
|
32
|
+
# Bootstrap: create new results.json from stdin JSON + current UTC timestamp
|
|
33
|
+
echo "$input_json" | jq --arg ea "$EVALUATED_AT" \
|
|
34
|
+
'. + { evaluated_at: $ea }' > "$RESULTS_JSON"
|
|
35
|
+
exit 0
|
|
36
|
+
fi
|
|
37
|
+
|
|
38
|
+
# Merge: new .skills override existing ones; old skills not in input_json are kept.
|
|
39
|
+
# Optionally update .mode and .batch_progress if provided.
|
|
40
|
+
#
|
|
41
|
+
# Use mktemp for a collision-safe temp file (concurrent runs on the same RESULTS_JSON
|
|
42
|
+
# would race on a predictable ".tmp" suffix; random suffix prevents silent overwrites).
|
|
43
|
+
tmp=$(mktemp "${RESULTS_JSON}.XXXXXX")
|
|
44
|
+
trap 'rm -f "$tmp"' EXIT
|
|
45
|
+
|
|
46
|
+
jq -s \
|
|
47
|
+
--arg ea "$EVALUATED_AT" \
|
|
48
|
+
'.[0] as $existing | .[1] as $new |
|
|
49
|
+
$existing |
|
|
50
|
+
.evaluated_at = $ea |
|
|
51
|
+
.skills = ($existing.skills + ($new.skills // {})) |
|
|
52
|
+
if ($new | has("mode")) then .mode = $new.mode else . end |
|
|
53
|
+
if ($new | has("batch_progress")) then .batch_progress = $new.batch_progress else . end' \
|
|
54
|
+
"$RESULTS_JSON" <(echo "$input_json") > "$tmp"
|
|
55
|
+
|
|
56
|
+
mv "$tmp" "$RESULTS_JSON"
|
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
# scan.sh — enumerate skill files, extract frontmatter and UTC mtime
|
|
3
|
+
# Usage: scan.sh [CWD_SKILLS_DIR]
|
|
4
|
+
# Output: JSON to stdout
|
|
5
|
+
#
|
|
6
|
+
# When CWD_SKILLS_DIR is omitted, defaults to $PWD/.claude/skills so the
|
|
7
|
+
# script always picks up project-level skills without relying on the caller.
|
|
8
|
+
#
|
|
9
|
+
# Environment:
|
|
10
|
+
# SKILL_STOCKTAKE_GLOBAL_DIR Override ~/.claude/skills (for testing only;
|
|
11
|
+
# do not set in production — intended for bats tests)
|
|
12
|
+
# SKILL_STOCKTAKE_PROJECT_DIR Override project dir detection (for testing only)
|
|
13
|
+
|
|
14
|
+
set -euo pipefail
|
|
15
|
+
|
|
16
|
+
GLOBAL_DIR="${SKILL_STOCKTAKE_GLOBAL_DIR:-$HOME/.claude/skills}"
|
|
17
|
+
CWD_SKILLS_DIR="${SKILL_STOCKTAKE_PROJECT_DIR:-${1:-$PWD/.claude/skills}}"
|
|
18
|
+
# Path to JSONL file containing tool-use observations (optional; used for usage frequency counts).
|
|
19
|
+
# Override via SKILL_STOCKTAKE_OBSERVATIONS env var if your setup uses a different path.
|
|
20
|
+
OBSERVATIONS="${SKILL_STOCKTAKE_OBSERVATIONS:-$HOME/.claude/observations.jsonl}"
|
|
21
|
+
|
|
22
|
+
# Validate CWD_SKILLS_DIR looks like a .claude/skills path (defense-in-depth).
|
|
23
|
+
# Only warn when the path exists — a nonexistent path poses no traversal risk.
|
|
24
|
+
if [[ -n "$CWD_SKILLS_DIR" && -d "$CWD_SKILLS_DIR" && "$CWD_SKILLS_DIR" != */.claude/skills* ]]; then
|
|
25
|
+
echo "Warning: CWD_SKILLS_DIR does not look like a .claude/skills path: $CWD_SKILLS_DIR" >&2
|
|
26
|
+
fi
|
|
27
|
+
|
|
28
|
+
# Extract a frontmatter field (handles both quoted and unquoted single-line values).
|
|
29
|
+
# Does NOT support multi-line YAML blocks (| or >) or nested YAML keys.
|
|
30
|
+
extract_field() {
|
|
31
|
+
local file="$1" field="$2"
|
|
32
|
+
awk -v f="$field" '
|
|
33
|
+
BEGIN { fm=0 }
|
|
34
|
+
/^---$/ { fm++; next }
|
|
35
|
+
fm==1 {
|
|
36
|
+
n = length(f) + 2
|
|
37
|
+
if (substr($0, 1, n) == f ": ") {
|
|
38
|
+
val = substr($0, n+1)
|
|
39
|
+
gsub(/^"/, "", val)
|
|
40
|
+
gsub(/"$/, "", val)
|
|
41
|
+
print val
|
|
42
|
+
exit
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
fm>=2 { exit }
|
|
46
|
+
' "$file"
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
# Get UTC timestamp N days ago (supports both macOS and GNU date)
|
|
50
|
+
date_ago() {
|
|
51
|
+
local n="$1"
|
|
52
|
+
date -u -v-"${n}d" +%Y-%m-%dT%H:%M:%SZ 2>/dev/null ||
|
|
53
|
+
date -u -d "${n} days ago" +%Y-%m-%dT%H:%M:%SZ
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
# Count observations matching a file path since a cutoff timestamp
|
|
57
|
+
count_obs() {
|
|
58
|
+
local file="$1" cutoff="$2"
|
|
59
|
+
if [[ ! -f "$OBSERVATIONS" ]]; then
|
|
60
|
+
echo 0
|
|
61
|
+
return
|
|
62
|
+
fi
|
|
63
|
+
jq -r --arg p "$file" --arg c "$cutoff" \
|
|
64
|
+
'select(.tool=="Read" and .path==$p and .timestamp>=$c) | 1' \
|
|
65
|
+
"$OBSERVATIONS" 2>/dev/null | wc -l | tr -d ' '
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
# Scan a directory and produce a JSON array of skill objects
|
|
69
|
+
scan_dir_to_json() {
|
|
70
|
+
local dir="$1"
|
|
71
|
+
local c7 c30
|
|
72
|
+
c7=$(date_ago 7)
|
|
73
|
+
c30=$(date_ago 30)
|
|
74
|
+
|
|
75
|
+
local tmpdir
|
|
76
|
+
tmpdir=$(mktemp -d)
|
|
77
|
+
# Use a function to avoid embedding $tmpdir in a quoted string (prevents injection
|
|
78
|
+
# if TMPDIR were crafted to contain shell metacharacters).
|
|
79
|
+
local _scan_tmpdir="$tmpdir"
|
|
80
|
+
_scan_cleanup() { rm -rf "$_scan_tmpdir"; }
|
|
81
|
+
trap _scan_cleanup RETURN
|
|
82
|
+
|
|
83
|
+
# Pre-aggregate observation counts in two passes (one per window) instead of
|
|
84
|
+
# calling jq per-file — reduces from O(n*m) to O(n+m) jq invocations.
|
|
85
|
+
local obs_7d_counts obs_30d_counts
|
|
86
|
+
obs_7d_counts=""
|
|
87
|
+
obs_30d_counts=""
|
|
88
|
+
if [[ -f "$OBSERVATIONS" ]]; then
|
|
89
|
+
obs_7d_counts=$(jq -r --arg c "$c7" \
|
|
90
|
+
'select(.tool=="Read" and .timestamp>=$c) | .path' \
|
|
91
|
+
"$OBSERVATIONS" 2>/dev/null | sort | uniq -c)
|
|
92
|
+
obs_30d_counts=$(jq -r --arg c "$c30" \
|
|
93
|
+
'select(.tool=="Read" and .timestamp>=$c) | .path' \
|
|
94
|
+
"$OBSERVATIONS" 2>/dev/null | sort | uniq -c)
|
|
95
|
+
fi
|
|
96
|
+
|
|
97
|
+
local i=0
|
|
98
|
+
while IFS= read -r file; do
|
|
99
|
+
local name desc mtime u7 u30 dp
|
|
100
|
+
name=$(extract_field "$file" "name")
|
|
101
|
+
desc=$(extract_field "$file" "description")
|
|
102
|
+
mtime=$(date -u -r "$file" +%Y-%m-%dT%H:%M:%SZ)
|
|
103
|
+
# Use awk exact field match to avoid substring false-positives from grep -F.
|
|
104
|
+
# uniq -c output format: " N /path/to/file" — path is always field 2.
|
|
105
|
+
u7=$(echo "$obs_7d_counts" | awk -v f="$file" '$2 == f {print $1}' | head -1)
|
|
106
|
+
u7="${u7:-0}"
|
|
107
|
+
u30=$(echo "$obs_30d_counts" | awk -v f="$file" '$2 == f {print $1}' | head -1)
|
|
108
|
+
u30="${u30:-0}"
|
|
109
|
+
dp="${file/#$HOME/~}"
|
|
110
|
+
|
|
111
|
+
jq -n \
|
|
112
|
+
--arg path "$dp" \
|
|
113
|
+
--arg name "$name" \
|
|
114
|
+
--arg description "$desc" \
|
|
115
|
+
--arg mtime "$mtime" \
|
|
116
|
+
--argjson use_7d "$u7" \
|
|
117
|
+
--argjson use_30d "$u30" \
|
|
118
|
+
'{path:$path,name:$name,description:$description,use_7d:$use_7d,use_30d:$use_30d,mtime:$mtime}' \
|
|
119
|
+
> "$tmpdir/$i.json"
|
|
120
|
+
i=$((i+1))
|
|
121
|
+
done < <(find "$dir" -name "*.md" -type f 2>/dev/null | sort)
|
|
122
|
+
|
|
123
|
+
if [[ $i -eq 0 ]]; then
|
|
124
|
+
echo "[]"
|
|
125
|
+
else
|
|
126
|
+
jq -s '.' "$tmpdir"/*.json
|
|
127
|
+
fi
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
# --- Main ---
|
|
131
|
+
|
|
132
|
+
global_found="false"
|
|
133
|
+
global_count=0
|
|
134
|
+
global_skills="[]"
|
|
135
|
+
|
|
136
|
+
if [[ -d "$GLOBAL_DIR" ]]; then
|
|
137
|
+
global_found="true"
|
|
138
|
+
global_skills=$(scan_dir_to_json "$GLOBAL_DIR")
|
|
139
|
+
global_count=$(echo "$global_skills" | jq 'length')
|
|
140
|
+
fi
|
|
141
|
+
|
|
142
|
+
project_found="false"
|
|
143
|
+
project_path=""
|
|
144
|
+
project_count=0
|
|
145
|
+
project_skills="[]"
|
|
146
|
+
|
|
147
|
+
if [[ -n "$CWD_SKILLS_DIR" && -d "$CWD_SKILLS_DIR" ]]; then
|
|
148
|
+
project_found="true"
|
|
149
|
+
project_path="$CWD_SKILLS_DIR"
|
|
150
|
+
project_skills=$(scan_dir_to_json "$CWD_SKILLS_DIR")
|
|
151
|
+
project_count=$(echo "$project_skills" | jq 'length')
|
|
152
|
+
fi
|
|
153
|
+
|
|
154
|
+
# Merge global + project skills into one array
|
|
155
|
+
all_skills=$(jq -s 'add' <(echo "$global_skills") <(echo "$project_skills"))
|
|
156
|
+
|
|
157
|
+
jq -n \
|
|
158
|
+
--arg global_found "$global_found" \
|
|
159
|
+
--argjson global_count "$global_count" \
|
|
160
|
+
--arg project_found "$project_found" \
|
|
161
|
+
--arg project_path "$project_path" \
|
|
162
|
+
--argjson project_count "$project_count" \
|
|
163
|
+
--argjson skills "$all_skills" \
|
|
164
|
+
'{
|
|
165
|
+
scan_summary: {
|
|
166
|
+
global: { found: ($global_found == "true"), count: $global_count },
|
|
167
|
+
project: { found: ($project_found == "true"), path: $project_path, count: $project_count }
|
|
168
|
+
},
|
|
169
|
+
skills: $skills
|
|
170
|
+
}'
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: security-check
|
|
3
|
+
description: Audit Supabase security — RLS policies, auth configuration, exposed secrets, and API surface. Use after schema changes or periodically.
|
|
4
|
+
context: fork
|
|
5
|
+
agent: Explore
|
|
6
|
+
allowed-tools: Read, Grep, Glob, Bash(grep *), Bash(git diff *)
|
|
7
|
+
---
|
|
8
|
+
|
|
9
|
+
Run a security audit on the current project and its Supabase configuration.
|
|
10
|
+
|
|
11
|
+
## Checks to Perform
|
|
12
|
+
|
|
13
|
+
### 1. Supabase Security Advisors
|
|
14
|
+
- Call `get_advisors(type: "security")` on the project
|
|
15
|
+
- Report ALL findings with severity and remediation links
|
|
16
|
+
|
|
17
|
+
### 2. RLS Policy Coverage
|
|
18
|
+
- Call `list_tables(schemas: ["public"])` to get all tables
|
|
19
|
+
- For each table, run:
|
|
20
|
+
```sql
|
|
21
|
+
SELECT tablename, rowsecurity FROM pg_tables WHERE schemaname = 'public';
|
|
22
|
+
```
|
|
23
|
+
- Flag any table where `rowsecurity = false`
|
|
24
|
+
- For tables WITH RLS, check policy completeness:
|
|
25
|
+
```sql
|
|
26
|
+
SELECT schemaname, tablename, policyname, permissive, roles, cmd, qual
|
|
27
|
+
FROM pg_policies WHERE schemaname = 'public';
|
|
28
|
+
```
|
|
29
|
+
- Flag tables missing SELECT/INSERT/UPDATE/DELETE policies
|
|
30
|
+
|
|
31
|
+
### 3. Auth Configuration
|
|
32
|
+
- Check if middleware auth is implemented (search for `createServerClient` in middleware)
|
|
33
|
+
- Search for `createBrowserClient` — verify it's only used in Client Components
|
|
34
|
+
- Search for hardcoded API keys or tokens: grep for patterns like `sk_`, `key_`, `secret`, `password`, `token` in source files (exclude node_modules, .git)
|
|
35
|
+
|
|
36
|
+
### 4. Environment Variables
|
|
37
|
+
- Check `.env*` files are in `.gitignore`
|
|
38
|
+
- Search for `process.env` usage — verify all env vars are validated (Zod or runtime check)
|
|
39
|
+
- Flag any env var used directly without validation
|
|
40
|
+
|
|
41
|
+
### 5. API Surface
|
|
42
|
+
- List all Route Handlers in `src/app/api/`
|
|
43
|
+
- Check each has authentication (search for `getUser` or `getSession`)
|
|
44
|
+
- Flag any public endpoints (no auth check) — these need explicit justification
|
|
45
|
+
|
|
46
|
+
### 6. Client-Side Data Exposure
|
|
47
|
+
- Search for `"use client"` components that directly query Supabase
|
|
48
|
+
- Verify they only use `createBrowserClient` (never server client)
|
|
49
|
+
- Check that sensitive columns aren't selected in client-side queries
|
|
50
|
+
|
|
51
|
+
## Output Format
|
|
52
|
+
|
|
53
|
+
```
|
|
54
|
+
## Security Audit Report
|
|
55
|
+
|
|
56
|
+
### Critical (fix immediately)
|
|
57
|
+
- [ ] ...
|
|
58
|
+
|
|
59
|
+
### Warning (fix soon)
|
|
60
|
+
- [ ] ...
|
|
61
|
+
|
|
62
|
+
### Info (review)
|
|
63
|
+
- [ ] ...
|
|
64
|
+
|
|
65
|
+
### Passed
|
|
66
|
+
- [x] ...
|
|
67
|
+
|
|
68
|
+
### Summary
|
|
69
|
+
X critical, Y warnings, Z info items found.
|
|
70
|
+
```
|