xtrm-tools 2.0.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/CHANGELOG.md +496 -0
- package/README.md +762 -0
- package/cli/dist/index.cjs +55245 -0
- package/cli/dist/index.cjs.map +1 -0
- package/cli/dist/index.d.cts +2 -0
- package/cli/package.json +43 -0
- package/config/.env.example +40 -0
- package/config/hooks.json +36 -0
- package/config/mcp_servers.json +46 -0
- package/config/mcp_servers_optional.json +53 -0
- package/config/settings.json +70 -0
- package/hooks/README.md +156 -0
- package/hooks/__pycache__/agent_context.cpython-314.pyc +0 -0
- package/hooks/agent_context.py +105 -0
- package/hooks/gitnexus/gitnexus-hook.cjs +133 -0
- package/hooks/serena-workflow-reminder.py +74 -0
- package/hooks/skill-discovery.py +90 -0
- package/hooks/skill-suggestion.py +112 -0
- package/hooks/test_agent_context.py +112 -0
- package/hooks/type-safety-enforcement.py +107 -0
- package/package.json +48 -0
- package/project-skills/main-guard/.claude/hooks/main-guard.cjs +188 -0
- package/project-skills/main-guard/.claude/settings.json +16 -0
- package/project-skills/main-guard/.claude/skills/using-main-guard/SKILL.md +135 -0
- package/project-skills/main-guard/README.md +163 -0
- package/project-skills/py-quality-gate/.claude/hooks/quality-check.py +311 -0
- package/project-skills/py-quality-gate/.claude/settings.json +16 -0
- package/project-skills/py-quality-gate/.claude/skills/using-py-quality-gate/SKILL.md +112 -0
- package/project-skills/py-quality-gate/README.md +147 -0
- package/project-skills/service-skills-set/.claude/git-hooks/__pycache__/doc_reminder.cpython-314.pyc +0 -0
- package/project-skills/service-skills-set/.claude/git-hooks/__pycache__/skill_staleness.cpython-314.pyc +0 -0
- package/project-skills/service-skills-set/.claude/git-hooks/doc_reminder.py +67 -0
- package/project-skills/service-skills-set/.claude/git-hooks/skill_staleness.py +194 -0
- package/project-skills/service-skills-set/.claude/service-registry.json +4 -0
- package/project-skills/service-skills-set/.claude/settings.json +37 -0
- package/project-skills/service-skills-set/.claude/skills/creating-service-skills/SKILL.md +433 -0
- package/project-skills/service-skills-set/.claude/skills/creating-service-skills/references/script_quality_standards.md +412 -0
- package/project-skills/service-skills-set/.claude/skills/creating-service-skills/references/service_skill_system_guide.md +264 -0
- package/project-skills/service-skills-set/.claude/skills/creating-service-skills/scripts/bootstrap.py +308 -0
- package/project-skills/service-skills-set/.claude/skills/creating-service-skills/scripts/deep_dive.py +304 -0
- package/project-skills/service-skills-set/.claude/skills/creating-service-skills/scripts/scaffolder.py +482 -0
- package/project-skills/service-skills-set/.claude/skills/scoping-service-skills/SKILL.md +231 -0
- package/project-skills/service-skills-set/.claude/skills/scoping-service-skills/scripts/scope.py +74 -0
- package/project-skills/service-skills-set/.claude/skills/updating-service-skills/SKILL.md +136 -0
- package/project-skills/service-skills-set/.claude/skills/updating-service-skills/scripts/__pycache__/drift_detector.cpython-314.pyc +0 -0
- package/project-skills/service-skills-set/.claude/skills/updating-service-skills/scripts/drift_detector.py +222 -0
- package/project-skills/service-skills-set/.claude/skills/using-service-skills/SKILL.md +108 -0
- package/project-skills/service-skills-set/.claude/skills/using-service-skills/scripts/__pycache__/cataloger.cpython-314.pyc +0 -0
- package/project-skills/service-skills-set/.claude/skills/using-service-skills/scripts/__pycache__/skill_activator.cpython-314.pyc +0 -0
- package/project-skills/service-skills-set/.claude/skills/using-service-skills/scripts/cataloger.py +74 -0
- package/project-skills/service-skills-set/.claude/skills/using-service-skills/scripts/skill_activator.py +152 -0
- package/project-skills/service-skills-set/README.md +93 -0
- package/project-skills/service-skills-set/__pycache__/install-service-skills.cpython-314.pyc +0 -0
- package/project-skills/service-skills-set/install-service-skills.py +163 -0
- package/project-skills/service-skills-set/service-skills-readme.md +236 -0
- package/project-skills/tdd-guard/.claude/settings.json +38 -0
- package/project-skills/tdd-guard/.claude/skills/using-tdd-guard/SKILL.md +74 -0
- package/project-skills/tdd-guard/CLAUDE.md +98 -0
- package/project-skills/tdd-guard/CONTRIBUTING.md +38 -0
- package/project-skills/tdd-guard/DEVELOPMENT.md +127 -0
- package/project-skills/tdd-guard/LICENSE +21 -0
- package/project-skills/tdd-guard/README.md +396 -0
- package/project-skills/tdd-guard/docs/adr/001-claude-session-subdirectory.md +52 -0
- package/project-skills/tdd-guard/docs/adr/002-secure-claude-binary-path.md +56 -0
- package/project-skills/tdd-guard/docs/adr/003-remove-configurable-data-directory.md +56 -0
- package/project-skills/tdd-guard/docs/adr/004-monorepo-architecture.md +64 -0
- package/project-skills/tdd-guard/docs/adr/005-claude-project-dir-support.md +55 -0
- package/project-skills/tdd-guard/docs/adr/006-phpunit-separate-repository.md +93 -0
- package/project-skills/tdd-guard/docs/adr/007-golangci-lint-path-support.md +83 -0
- package/project-skills/tdd-guard/docs/adr/008-storybook-reporter-design.md +182 -0
- package/project-skills/tdd-guard/docs/assets/tdd-guard-demo-screenshot.gif +0 -0
- package/project-skills/tdd-guard/docs/config-migration.md +143 -0
- package/project-skills/tdd-guard/docs/configuration.md +137 -0
- package/project-skills/tdd-guard/docs/custom-instructions.md +43 -0
- package/project-skills/tdd-guard/docs/enforcement.md +46 -0
- package/project-skills/tdd-guard/docs/ignore-patterns.md +81 -0
- package/project-skills/tdd-guard/docs/linting.md +109 -0
- package/project-skills/tdd-guard/docs/quick-commands.md +52 -0
- package/project-skills/tdd-guard/docs/session-management.md +75 -0
- package/project-skills/tdd-guard/docs/storybook-vitest-addon.md +120 -0
- package/project-skills/tdd-guard/docs/validation-model.md +63 -0
- package/project-skills/tdd-guard/eslint.config.mjs +140 -0
- package/project-skills/tdd-guard/package-lock.json +16937 -0
- package/project-skills/tdd-guard/package.json +102 -0
- package/project-skills/tdd-guard/reporters/go/README.md +67 -0
- package/project-skills/tdd-guard/reporters/go/cmd/tdd-guard-go/main.go +127 -0
- package/project-skills/tdd-guard/reporters/go/cmd/tdd-guard-go/main_test.go +280 -0
- package/project-skills/tdd-guard/reporters/go/go.mod +3 -0
- package/project-skills/tdd-guard/reporters/go/go.sum +0 -0
- package/project-skills/tdd-guard/reporters/go/internal/formatter/formatter.go +126 -0
- package/project-skills/tdd-guard/reporters/go/internal/formatter/formatter_test.go +264 -0
- package/project-skills/tdd-guard/reporters/go/internal/io/tee_reader.go +26 -0
- package/project-skills/tdd-guard/reporters/go/internal/io/tee_reader_test.go +37 -0
- package/project-skills/tdd-guard/reporters/go/internal/parser/mixed_reader.go +94 -0
- package/project-skills/tdd-guard/reporters/go/internal/parser/mixed_reader_test.go +198 -0
- package/project-skills/tdd-guard/reporters/go/internal/parser/parser.go +245 -0
- package/project-skills/tdd-guard/reporters/go/internal/parser/parser_test.go +547 -0
- package/project-skills/tdd-guard/reporters/go/internal/storage/storage.go +35 -0
- package/project-skills/tdd-guard/reporters/go/internal/storage/storage_test.go +113 -0
- package/project-skills/tdd-guard/reporters/go/internal/transformer/transformer.go +103 -0
- package/project-skills/tdd-guard/reporters/go/internal/transformer/transformer_test.go +303 -0
- package/project-skills/tdd-guard/reporters/jest/README.md +102 -0
- package/project-skills/tdd-guard/reporters/jest/package.json +38 -0
- package/project-skills/tdd-guard/reporters/phpunit/.php-cs-fixer.php +28 -0
- package/project-skills/tdd-guard/reporters/phpunit/README.md +97 -0
- package/project-skills/tdd-guard/reporters/phpunit/SYNC_README.md +29 -0
- package/project-skills/tdd-guard/reporters/phpunit/composer.json +55 -0
- package/project-skills/tdd-guard/reporters/phpunit/phpunit.xml.dist +19 -0
- package/project-skills/tdd-guard/reporters/phpunit/psalm.xml +44 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/Event/ErroredTestSubscriber.php +28 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/Event/FailedTestSubscriber.php +28 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/Event/IncompleteTestSubscriber.php +28 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/Event/PassedTestSubscriber.php +27 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/Event/SkippedTestSubscriber.php +28 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/Event/TestRunnerFinishedSubscriber.php +24 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/PathValidator.php +88 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/Storage.php +26 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/TddGuardExtension.php +33 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/TddGuardListener.php +158 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/TddGuardSubscriber.php +35 -0
- package/project-skills/tdd-guard/reporters/phpunit/src/TestResultCollector.php +105 -0
- package/project-skills/tdd-guard/reporters/phpunit/tests/PathValidatorTest.php +74 -0
- package/project-skills/tdd-guard/reporters/phpunit/tests/TddGuardExtensionFailedTest.php +241 -0
- package/project-skills/tdd-guard/reporters/phpunit/tests/TddGuardExtensionTest.php +84 -0
- package/project-skills/tdd-guard/reporters/phpunit/tests/TddGuardStorageLocationTest.php +71 -0
- package/project-skills/tdd-guard/reporters/pytest/README.md +77 -0
- package/project-skills/tdd-guard/reporters/pytest/pyproject.toml +43 -0
- package/project-skills/tdd-guard/reporters/pytest/pytest.ini.example +7 -0
- package/project-skills/tdd-guard/reporters/pytest/tdd_guard_pytest/__init__.py +1 -0
- package/project-skills/tdd-guard/reporters/pytest/tdd_guard_pytest/pytest_reporter.py +134 -0
- package/project-skills/tdd-guard/reporters/pytest/tests/__init__.py +1 -0
- package/project-skills/tdd-guard/reporters/pytest/tests/conftest.py +3 -0
- package/project-skills/tdd-guard/reporters/pytest/tests/helpers.py +293 -0
- package/project-skills/tdd-guard/reporters/pytest/tests/test_config_option.py +38 -0
- package/project-skills/tdd-guard/reporters/pytest/tests/test_path_validation.py +59 -0
- package/project-skills/tdd-guard/reporters/pytest/tests/test_plugin_config.py +32 -0
- package/project-skills/tdd-guard/reporters/pytest/tests/test_project_root.py +296 -0
- package/project-skills/tdd-guard/reporters/pytest/tests/test_pytest_reporter.py +137 -0
- package/project-skills/tdd-guard/reporters/rspec/Gemfile +3 -0
- package/project-skills/tdd-guard/reporters/rust/Cargo.lock +458 -0
- package/project-skills/tdd-guard/reporters/rust/Cargo.toml +33 -0
- package/project-skills/tdd-guard/reporters/rust/Makefile.example +95 -0
- package/project-skills/tdd-guard/reporters/rust/README.md +88 -0
- package/project-skills/tdd-guard/reporters/rust/src/error_parser.rs +309 -0
- package/project-skills/tdd-guard/reporters/rust/src/main.rs +464 -0
- package/project-skills/tdd-guard/reporters/rust/src/parser.rs +225 -0
- package/project-skills/tdd-guard/reporters/rust/src/transformer.rs +409 -0
- package/project-skills/tdd-guard/reporters/storybook/README.md +108 -0
- package/project-skills/tdd-guard/reporters/storybook/package-lock.json +9482 -0
- package/project-skills/tdd-guard/reporters/storybook/package.json +43 -0
- package/project-skills/tdd-guard/reporters/storybook/src/StorybookReporter.test-data.ts +22 -0
- package/project-skills/tdd-guard/reporters/storybook/src/StorybookReporter.test.ts +190 -0
- package/project-skills/tdd-guard/reporters/storybook/src/StorybookReporter.ts +88 -0
- package/project-skills/tdd-guard/reporters/storybook/src/index.ts +12 -0
- package/project-skills/tdd-guard/reporters/storybook/src/types.ts +37 -0
- package/project-skills/tdd-guard/reporters/storybook/tsconfig.json +11 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/go/failing/go.mod +3 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/go/failing/single_failing_test.go +13 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/go/import/go.mod +3 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/go/import/single_import_error_test.go +17 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/go/passing/go.mod +3 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/go/passing/single_passing_test.go +13 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/jest/single-failing.test.js +5 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/jest/single-import-error.test.js +8 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/jest/single-passing.test.js +5 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/phpunit/SingleFailingTest.php +11 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/phpunit/SingleImportErrorTest.php +14 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/phpunit/SinglePassingTest.php +11 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/pytest/test_single_failing.py +3 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/pytest/test_single_import_error.py +6 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/pytest/test_single_passing.py +3 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/rust/failing/Cargo.lock +7 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/rust/failing/Cargo.toml +4 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/rust/failing/src/lib.rs +14 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/rust/import/Cargo.lock +7 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/rust/import/Cargo.toml +4 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/rust/import/src/lib.rs +13 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/rust/passing/Cargo.lock +7 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/rust/passing/Cargo.toml +4 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/rust/passing/src/lib.rs +14 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/storybook/Calculator.js +4 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/storybook/single-failing.stories.js +15 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/storybook/single-import-error.stories.js +14 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/storybook/single-passing.stories.js +15 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/vitest/single-failing.test.js +7 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/vitest/single-import-error.test.js +9 -0
- package/project-skills/tdd-guard/reporters/test/artifacts/vitest/single-passing.test.js +7 -0
- package/project-skills/tdd-guard/reporters/test/factories/go.ts +59 -0
- package/project-skills/tdd-guard/reporters/test/factories/helpers.ts +48 -0
- package/project-skills/tdd-guard/reporters/test/factories/index.ts +7 -0
- package/project-skills/tdd-guard/reporters/test/factories/jest.ts +51 -0
- package/project-skills/tdd-guard/reporters/test/factories/phpunit.ts +63 -0
- package/project-skills/tdd-guard/reporters/test/factories/pytest.ts +41 -0
- package/project-skills/tdd-guard/reporters/test/factories/rust.ts +158 -0
- package/project-skills/tdd-guard/reporters/test/factories/storybook.ts +198 -0
- package/project-skills/tdd-guard/reporters/test/factories/vitest.ts +51 -0
- package/project-skills/tdd-guard/reporters/test/reporters.integration.test.ts +735 -0
- package/project-skills/tdd-guard/reporters/test/types.ts +28 -0
- package/project-skills/tdd-guard/reporters/vitest/README.md +64 -0
- package/project-skills/tdd-guard/reporters/vitest/package.json +35 -0
- package/project-skills/tdd-guard/src/cli/buildContext.test.ts +200 -0
- package/project-skills/tdd-guard/src/cli/buildContext.ts +48 -0
- package/project-skills/tdd-guard/src/cli/tdd-guard.test.ts +159 -0
- package/project-skills/tdd-guard/src/cli/tdd-guard.ts +48 -0
- package/project-skills/tdd-guard/src/config/Config.test.ts +538 -0
- package/project-skills/tdd-guard/src/config/Config.ts +172 -0
- package/project-skills/tdd-guard/src/contracts/schemas/guardSchemas.test.ts +58 -0
- package/project-skills/tdd-guard/src/contracts/schemas/guardSchemas.ts +8 -0
- package/project-skills/tdd-guard/src/contracts/schemas/lintSchemas.test.ts +347 -0
- package/project-skills/tdd-guard/src/contracts/schemas/lintSchemas.ts +61 -0
- package/project-skills/tdd-guard/src/contracts/schemas/pytestSchemas.test.ts +24 -0
- package/project-skills/tdd-guard/src/contracts/schemas/pytestSchemas.ts +7 -0
- package/project-skills/tdd-guard/src/contracts/schemas/reporterSchemas.test.ts +377 -0
- package/project-skills/tdd-guard/src/contracts/schemas/reporterSchemas.ts +75 -0
- package/project-skills/tdd-guard/src/contracts/schemas/toolSchemas.test.ts +563 -0
- package/project-skills/tdd-guard/src/contracts/schemas/toolSchemas.ts +140 -0
- package/project-skills/tdd-guard/src/contracts/types/ClientType.ts +1 -0
- package/project-skills/tdd-guard/src/contracts/types/ConfigOptions.ts +12 -0
- package/project-skills/tdd-guard/src/contracts/types/Context.ts +16 -0
- package/project-skills/tdd-guard/src/contracts/types/ModelClient.ts +3 -0
- package/project-skills/tdd-guard/src/contracts/types/ValidationResult.ts +6 -0
- package/project-skills/tdd-guard/src/guard/GuardManager.test.ts +336 -0
- package/project-skills/tdd-guard/src/guard/GuardManager.ts +83 -0
- package/project-skills/tdd-guard/src/hooks/HookEvents.test.ts +107 -0
- package/project-skills/tdd-guard/src/hooks/HookEvents.ts +39 -0
- package/project-skills/tdd-guard/src/hooks/fileTypeDetection.ts +16 -0
- package/project-skills/tdd-guard/src/hooks/postToolLint.test.ts +327 -0
- package/project-skills/tdd-guard/src/hooks/postToolLint.ts +165 -0
- package/project-skills/tdd-guard/src/hooks/processHookData.test.ts +465 -0
- package/project-skills/tdd-guard/src/hooks/processHookData.ts +203 -0
- package/project-skills/tdd-guard/src/hooks/sessionHandler.test.ts +136 -0
- package/project-skills/tdd-guard/src/hooks/sessionHandler.ts +31 -0
- package/project-skills/tdd-guard/src/hooks/userPromptHandler.test.ts +131 -0
- package/project-skills/tdd-guard/src/hooks/userPromptHandler.ts +55 -0
- package/project-skills/tdd-guard/src/index.ts +19 -0
- package/project-skills/tdd-guard/src/linters/Linter.ts +5 -0
- package/project-skills/tdd-guard/src/linters/eslint/ESLint.test.ts +183 -0
- package/project-skills/tdd-guard/src/linters/eslint/ESLint.ts +82 -0
- package/project-skills/tdd-guard/src/linters/golangci/GolangciLint.test.ts +170 -0
- package/project-skills/tdd-guard/src/linters/golangci/GolangciLint.ts +148 -0
- package/project-skills/tdd-guard/src/processors/index.ts +1 -0
- package/project-skills/tdd-guard/src/processors/lintProcessor.ts +77 -0
- package/project-skills/tdd-guard/src/processors/testResults/TestResultsProcessor.test.ts +303 -0
- package/project-skills/tdd-guard/src/processors/testResults/TestResultsProcessor.ts +255 -0
- package/project-skills/tdd-guard/src/providers/LinterProvider.test.ts +43 -0
- package/project-skills/tdd-guard/src/providers/LinterProvider.ts +20 -0
- package/project-skills/tdd-guard/src/providers/ModelClientProvider.test.ts +68 -0
- package/project-skills/tdd-guard/src/providers/ModelClientProvider.ts +22 -0
- package/project-skills/tdd-guard/src/storage/FileStorage.test.ts +76 -0
- package/project-skills/tdd-guard/src/storage/FileStorage.ts +108 -0
- package/project-skills/tdd-guard/src/storage/MemoryStorage.ts +57 -0
- package/project-skills/tdd-guard/src/storage/Storage.test.ts +227 -0
- package/project-skills/tdd-guard/src/storage/Storage.ts +17 -0
- package/project-skills/tdd-guard/src/validation/context/context.test.ts +364 -0
- package/project-skills/tdd-guard/src/validation/context/context.ts +155 -0
- package/project-skills/tdd-guard/src/validation/models/AnthropicApi.test.ts +171 -0
- package/project-skills/tdd-guard/src/validation/models/AnthropicApi.ts +49 -0
- package/project-skills/tdd-guard/src/validation/models/ClaudeAgentSdk.test.ts +167 -0
- package/project-skills/tdd-guard/src/validation/models/ClaudeAgentSdk.ts +54 -0
- package/project-skills/tdd-guard/src/validation/models/ClaudeCli.test.ts +239 -0
- package/project-skills/tdd-guard/src/validation/models/ClaudeCli.ts +57 -0
- package/project-skills/tdd-guard/src/validation/prompts/file-types.ts +52 -0
- package/project-skills/tdd-guard/src/validation/prompts/operations/edit.ts +58 -0
- package/project-skills/tdd-guard/src/validation/prompts/operations/multi-edit.ts +54 -0
- package/project-skills/tdd-guard/src/validation/prompts/operations/write.ts +54 -0
- package/project-skills/tdd-guard/src/validation/prompts/response.ts +40 -0
- package/project-skills/tdd-guard/src/validation/prompts/rules.ts +51 -0
- package/project-skills/tdd-guard/src/validation/prompts/system-prompt.ts +10 -0
- package/project-skills/tdd-guard/src/validation/prompts/tools/lint-results.ts +15 -0
- package/project-skills/tdd-guard/src/validation/prompts/tools/test-output.ts +14 -0
- package/project-skills/tdd-guard/src/validation/prompts/tools/todos.ts +9 -0
- package/project-skills/tdd-guard/src/validation/validator.test.ts +268 -0
- package/project-skills/tdd-guard/src/validation/validator.ts +159 -0
- package/project-skills/tdd-guard/test/artifacts/go/.golangci.yml +6 -0
- package/project-skills/tdd-guard/test/artifacts/go/with-issues/file-with-issues.go +12 -0
- package/project-skills/tdd-guard/test/artifacts/go/with-issues/go.mod +3 -0
- package/project-skills/tdd-guard/test/artifacts/go/without-issues/file-without-issues.go +7 -0
- package/project-skills/tdd-guard/test/artifacts/go/without-issues/go.mod +3 -0
- package/project-skills/tdd-guard/test/artifacts/javascript/eslint.config.js +20 -0
- package/project-skills/tdd-guard/test/artifacts/javascript/file-with-issues.js +12 -0
- package/project-skills/tdd-guard/test/artifacts/javascript/file-without-issues.js +10 -0
- package/project-skills/tdd-guard/test/hooks/fileTypeDetection.test.ts +26 -0
- package/project-skills/tdd-guard/test/hooks/processHookData.fileType.test.ts +46 -0
- package/project-skills/tdd-guard/test/hooks/processHookData.python.test.ts +68 -0
- package/project-skills/tdd-guard/test/integration/test-context.test.ts +66 -0
- package/project-skills/tdd-guard/test/integration/validator.core.test.ts +96 -0
- package/project-skills/tdd-guard/test/integration/validator.scenarios.test.ts +497 -0
- package/project-skills/tdd-guard/test/utils/assertions.ts +29 -0
- package/project-skills/tdd-guard/test/utils/factories/contextFactory.ts +30 -0
- package/project-skills/tdd-guard/test/utils/factories/editFactory.ts +82 -0
- package/project-skills/tdd-guard/test/utils/factories/helpers.test.ts +46 -0
- package/project-skills/tdd-guard/test/utils/factories/helpers.ts +46 -0
- package/project-skills/tdd-guard/test/utils/factories/lintFactory.ts +352 -0
- package/project-skills/tdd-guard/test/utils/factories/modelClientProviderFactory.ts +21 -0
- package/project-skills/tdd-guard/test/utils/factories/multiEditFactory.ts +79 -0
- package/project-skills/tdd-guard/test/utils/factories/operations.ts +57 -0
- package/project-skills/tdd-guard/test/utils/factories/reporterFactory.ts +55 -0
- package/project-skills/tdd-guard/test/utils/factories/scenarios/index.ts +22 -0
- package/project-skills/tdd-guard/test/utils/factories/scenarios/languages/python.ts +745 -0
- package/project-skills/tdd-guard/test/utils/factories/scenarios/languages/typescript.ts +767 -0
- package/project-skills/tdd-guard/test/utils/factories/scenarios/types.ts +77 -0
- package/project-skills/tdd-guard/test/utils/factories/scenarios/utils.ts +15 -0
- package/project-skills/tdd-guard/test/utils/factories/sessionStartFactory.ts +36 -0
- package/project-skills/tdd-guard/test/utils/factories/testDefaults.ts +90 -0
- package/project-skills/tdd-guard/test/utils/factories/testResultsFactory.ts +234 -0
- package/project-skills/tdd-guard/test/utils/factories/todoFactory.ts +99 -0
- package/project-skills/tdd-guard/test/utils/factories/userPromptSubmitFactory.ts +39 -0
- package/project-skills/tdd-guard/test/utils/factories/writeFactory.ts +70 -0
- package/project-skills/tdd-guard/test/utils/index.ts +131 -0
- package/project-skills/tdd-guard/tsconfig.build.json +16 -0
- package/project-skills/tdd-guard/tsconfig.eslint.json +17 -0
- package/project-skills/tdd-guard/tsconfig.json +32 -0
- package/project-skills/tdd-guard/tsconfig.node.json +10 -0
- package/project-skills/tdd-guard/vitest.config.ts +85 -0
- package/project-skills/ts-quality-gate/.claude/hooks/hook-config.json +66 -0
- package/project-skills/ts-quality-gate/.claude/hooks/quality-check.cjs +1251 -0
- package/project-skills/ts-quality-gate/.claude/settings.json +16 -0
- package/project-skills/ts-quality-gate/.claude/skills/using-ts-quality-gate/SKILL.md +81 -0
- package/project-skills/ts-quality-gate/README.md +115 -0
- package/skills/README.txt +31 -0
- package/skills/clean-code/SKILL.md +201 -0
- package/skills/delegating/SKILL.md +196 -0
- package/skills/delegating/config.yaml +210 -0
- package/skills/delegating/references/orchestration-protocols.md +41 -0
- package/skills/docker-expert/SKILL.md +409 -0
- package/skills/documenting/CHANGELOG.md +23 -0
- package/skills/documenting/README.md +148 -0
- package/skills/documenting/SKILL.md +113 -0
- package/skills/documenting/examples/example_pattern.md +70 -0
- package/skills/documenting/examples/example_reference.md +70 -0
- package/skills/documenting/examples/example_ssot_analytics.md +64 -0
- package/skills/documenting/examples/example_workflow.md +141 -0
- package/skills/documenting/references/changelog-format.md +97 -0
- package/skills/documenting/references/metadata-schema.md +136 -0
- package/skills/documenting/references/taxonomy.md +81 -0
- package/skills/documenting/references/versioning-rules.md +78 -0
- package/skills/documenting/scripts/__pycache__/drift_detector.cpython-314.pyc +0 -0
- package/skills/documenting/scripts/__pycache__/orchestrator.cpython-314.pyc +0 -0
- package/skills/documenting/scripts/__pycache__/validate_metadata.cpython-314.pyc +0 -0
- package/skills/documenting/scripts/bump_version.sh +60 -0
- package/skills/documenting/scripts/changelog/__init__.py +0 -0
- package/skills/documenting/scripts/changelog/__pycache__/__init__.cpython-314.pyc +0 -0
- package/skills/documenting/scripts/changelog/__pycache__/add_entry.cpython-314.pyc +0 -0
- package/skills/documenting/scripts/changelog/__pycache__/bump_release.cpython-314.pyc +0 -0
- package/skills/documenting/scripts/changelog/__pycache__/validate_changelog.cpython-314.pyc +0 -0
- package/skills/documenting/scripts/changelog/add_entry.py +216 -0
- package/skills/documenting/scripts/changelog/bump_release.py +117 -0
- package/skills/documenting/scripts/changelog/init_changelog.py +54 -0
- package/skills/documenting/scripts/changelog/validate_changelog.py +128 -0
- package/skills/documenting/scripts/drift_detector.py +266 -0
- package/skills/documenting/scripts/generate_template.py +311 -0
- package/skills/documenting/scripts/list_by_category.sh +84 -0
- package/skills/documenting/scripts/orchestrator.py +255 -0
- package/skills/documenting/scripts/validate_metadata.py +242 -0
- package/skills/documenting/templates/CHANGELOG.md.template +13 -0
- package/skills/documenting/tests/__pycache__/test_changelog.cpython-314-pytest-9.0.2.pyc +0 -0
- package/skills/documenting/tests/__pycache__/test_drift_detector.cpython-314-pytest-9.0.2.pyc +0 -0
- package/skills/documenting/tests/__pycache__/test_orchestrator.cpython-314-pytest-9.0.2.pyc +0 -0
- package/skills/documenting/tests/__pycache__/test_validate_metadata.cpython-314-pytest-9.0.2.pyc +0 -0
- package/skills/documenting/tests/integration_test.sh +70 -0
- package/skills/documenting/tests/test_changelog.py +201 -0
- package/skills/documenting/tests/test_drift_detector.py +80 -0
- package/skills/documenting/tests/test_orchestrator.py +52 -0
- package/skills/documenting/tests/test_validate_metadata.py +64 -0
- package/skills/find-skills/SKILL.md +133 -0
- package/skills/gitnexus-debugging/SKILL.md +85 -0
- package/skills/gitnexus-exploring/SKILL.md +75 -0
- package/skills/gitnexus-impact-analysis/SKILL.md +94 -0
- package/skills/gitnexus-refactoring/SKILL.md +113 -0
- package/skills/hook-development/SKILL.md +797 -0
- package/skills/hook-development/examples/load-context.sh +55 -0
- package/skills/hook-development/examples/quality-check.js +1168 -0
- package/skills/hook-development/examples/validate-bash.sh +43 -0
- package/skills/hook-development/examples/validate-write.sh +38 -0
- package/skills/hook-development/references/advanced.md +527 -0
- package/skills/hook-development/references/migration.md +369 -0
- package/skills/hook-development/references/patterns.md +412 -0
- package/skills/hook-development/scripts/README.md +164 -0
- package/skills/hook-development/scripts/hook-linter.sh +153 -0
- package/skills/hook-development/scripts/test-hook.sh +252 -0
- package/skills/hook-development/scripts/validate-hook-schema.sh +159 -0
- package/skills/obsidian-cli/SKILL.md +106 -0
- package/skills/orchestrating-agents/SKILL.md +135 -0
- package/skills/orchestrating-agents/config.yaml +45 -0
- package/skills/orchestrating-agents/references/agent-context-integration.md +37 -0
- package/skills/orchestrating-agents/references/examples.md +45 -0
- package/skills/orchestrating-agents/references/handover-protocol.md +31 -0
- package/skills/orchestrating-agents/references/workflows.md +42 -0
- package/skills/orchestrating-agents/scripts/detect_neighbors.py +23 -0
- package/skills/prompt-improving/README.md +162 -0
- package/skills/prompt-improving/SKILL.md +74 -0
- package/skills/prompt-improving/references/analysis_commands.md +24 -0
- package/skills/prompt-improving/references/chain_of_thought.md +24 -0
- package/skills/prompt-improving/references/mcp_definitions.md +20 -0
- package/skills/prompt-improving/references/multishot.md +23 -0
- package/skills/prompt-improving/references/xml_core.md +60 -0
- package/skills/python-testing/SKILL.md +815 -0
- package/skills/senior-backend/SKILL.md +209 -0
- package/skills/senior-backend/references/api_design_patterns.md +103 -0
- package/skills/senior-backend/references/backend_security_practices.md +103 -0
- package/skills/senior-backend/references/database_optimization_guide.md +103 -0
- package/skills/senior-backend/scripts/api_load_tester.py +114 -0
- package/skills/senior-backend/scripts/api_scaffolder.py +114 -0
- package/skills/senior-backend/scripts/database_migration_tool.py +114 -0
- package/skills/senior-data-scientist/SKILL.md +226 -0
- package/skills/senior-data-scientist/references/experiment_design_frameworks.md +80 -0
- package/skills/senior-data-scientist/references/feature_engineering_patterns.md +80 -0
- package/skills/senior-data-scientist/references/statistical_methods_advanced.md +80 -0
- package/skills/senior-data-scientist/scripts/experiment_designer.py +100 -0
- package/skills/senior-data-scientist/scripts/feature_engineering_pipeline.py +100 -0
- package/skills/senior-data-scientist/scripts/model_evaluation_suite.py +100 -0
- package/skills/senior-devops/SKILL.md +209 -0
- package/skills/senior-devops/references/cicd_pipeline_guide.md +103 -0
- package/skills/senior-devops/references/deployment_strategies.md +103 -0
- package/skills/senior-devops/references/infrastructure_as_code.md +103 -0
- package/skills/senior-devops/scripts/deployment_manager.py +114 -0
- package/skills/senior-devops/scripts/pipeline_generator.py +114 -0
- package/skills/senior-devops/scripts/terraform_scaffolder.py +114 -0
- package/skills/senior-security/SKILL.md +209 -0
- package/skills/senior-security/references/cryptography_implementation.md +103 -0
- package/skills/senior-security/references/penetration_testing_guide.md +103 -0
- package/skills/senior-security/references/security_architecture_patterns.md +103 -0
- package/skills/senior-security/scripts/pentest_automator.py +114 -0
- package/skills/senior-security/scripts/security_auditor.py +114 -0
- package/skills/senior-security/scripts/threat_modeler.py +114 -0
- package/skills/skill-creator/LICENSE.txt +202 -0
- package/skills/skill-creator/SKILL.md +479 -0
- package/skills/skill-creator/agents/analyzer.md +274 -0
- package/skills/skill-creator/agents/comparator.md +202 -0
- package/skills/skill-creator/agents/grader.md +223 -0
- package/skills/skill-creator/assets/eval_review.html +146 -0
- package/skills/skill-creator/eval-viewer/generate_review.py +471 -0
- package/skills/skill-creator/eval-viewer/viewer.html +1325 -0
- package/skills/skill-creator/references/schemas.md +430 -0
- package/skills/skill-creator/scripts/__init__.py +0 -0
- package/skills/skill-creator/scripts/aggregate_benchmark.py +401 -0
- package/skills/skill-creator/scripts/generate_report.py +326 -0
- package/skills/skill-creator/scripts/improve_description.py +248 -0
- package/skills/skill-creator/scripts/package_skill.py +136 -0
- package/skills/skill-creator/scripts/quick_validate.py +103 -0
- package/skills/skill-creator/scripts/run_eval.py +310 -0
- package/skills/skill-creator/scripts/run_loop.py +332 -0
- package/skills/skill-creator/scripts/utils.py +47 -0
- package/skills/using-TDD/SKILL.md +410 -0
- package/skills/using-serena-lsp/README.md +8 -0
- package/skills/using-serena-lsp/REFERENCE.md +194 -0
- package/skills/using-serena-lsp/SKILL.md +82 -0
|
@@ -0,0 +1,242 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
Validate SSOT memory metadata compliance.
|
|
4
|
+
|
|
5
|
+
Checks if a memory file has all required frontmatter fields and follows
|
|
6
|
+
the naming conventions defined in the SSOT guidelines.
|
|
7
|
+
"""
|
|
8
|
+
|
|
9
|
+
import sys
|
|
10
|
+
import re
|
|
11
|
+
from pathlib import Path
|
|
12
|
+
import yaml
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
def extract_headings(content: str) -> list:
|
|
16
|
+
"""Extract (heading, first_sentence) for every ## section (not ###)."""
|
|
17
|
+
results = []
|
|
18
|
+
lines = content.splitlines()
|
|
19
|
+
i = 0
|
|
20
|
+
while i < len(lines):
|
|
21
|
+
line = lines[i]
|
|
22
|
+
if line.startswith("## ") and not line.startswith("### "):
|
|
23
|
+
heading = line[3:].strip()
|
|
24
|
+
summary = ""
|
|
25
|
+
j = i + 1
|
|
26
|
+
in_code = False
|
|
27
|
+
while j < len(lines):
|
|
28
|
+
l = lines[j].strip()
|
|
29
|
+
if l.startswith("```"):
|
|
30
|
+
in_code = not in_code
|
|
31
|
+
j += 1
|
|
32
|
+
continue
|
|
33
|
+
if not in_code and l and not l.startswith("#") and not l.startswith("|") and not l.startswith("-"):
|
|
34
|
+
# Take up to first period, strip trailing period
|
|
35
|
+
sentence = l.split(".")[0].strip()
|
|
36
|
+
summary = sentence[:120]
|
|
37
|
+
break
|
|
38
|
+
j += 1
|
|
39
|
+
results.append((heading, summary))
|
|
40
|
+
i += 1
|
|
41
|
+
return results
|
|
42
|
+
|
|
43
|
+
|
|
44
|
+
def generate_index_table(headings: list) -> str:
|
|
45
|
+
"""Generate a full markdown table string from headings list."""
|
|
46
|
+
rows = ["| Section | Summary |", "|---|---|"]
|
|
47
|
+
for heading, summary in headings:
|
|
48
|
+
anchor = heading.lower()
|
|
49
|
+
for ch in "/()":
|
|
50
|
+
anchor = anchor.replace(ch, "")
|
|
51
|
+
anchor = re.sub(r"[\s_]+", "-", anchor)
|
|
52
|
+
anchor = re.sub(r"-+", "-", anchor)
|
|
53
|
+
anchor = anchor.strip("-")
|
|
54
|
+
rows.append(f"| [{heading}](#{anchor}) | {summary or '_no summary_'} |")
|
|
55
|
+
return "\n".join(rows) + "\n"
|
|
56
|
+
|
|
57
|
+
|
|
58
|
+
def inject_index(content: str, table: str) -> str:
|
|
59
|
+
"""Replace existing INDEX block or insert one after frontmatter closing ---."""
|
|
60
|
+
header = "<!-- INDEX: auto-generated by validate_metadata.py — do not edit manually -->\n"
|
|
61
|
+
footer = "<!-- END INDEX -->"
|
|
62
|
+
block = f"{header}{table}{footer}"
|
|
63
|
+
|
|
64
|
+
# Replace existing block
|
|
65
|
+
existing = re.search(r"<!-- INDEX:.*?-->.*?<!-- END INDEX -->", content, re.DOTALL)
|
|
66
|
+
if existing:
|
|
67
|
+
return content[:existing.start()] + block + content[existing.end():]
|
|
68
|
+
|
|
69
|
+
# Insert after second --- (end of frontmatter)
|
|
70
|
+
fm_match = re.match(r"^(---\n.*?\n---\n)(.*)", content, re.DOTALL)
|
|
71
|
+
if fm_match:
|
|
72
|
+
return fm_match.group(1) + "\n" + block + "\n" + fm_match.group(2)
|
|
73
|
+
|
|
74
|
+
return block + "\n" + content
|
|
75
|
+
|
|
76
|
+
|
|
77
|
+
REQUIRED_FIELDS = {
|
|
78
|
+
"ssot": ["title", "version", "updated", "scope", "category", "subcategory", "domain"],
|
|
79
|
+
"pattern": ["title", "version", "updated", "scope", "category", "domain"],
|
|
80
|
+
"plan": ["title", "version", "updated", "scope", "category"],
|
|
81
|
+
"reference": ["title", "scope", "category"],
|
|
82
|
+
"archive": ["title", "version", "updated", "scope", "category", "archived_date"],
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
CATEGORY_SUFFIXES = ["ssot", "pattern", "plan", "reference", "archive", "troubleshoot", "commit_log"]
|
|
86
|
+
|
|
87
|
+
|
|
88
|
+
def extract_frontmatter(content):
|
|
89
|
+
"""Extract YAML frontmatter from markdown content."""
|
|
90
|
+
match = re.match(r'^---\n(.*?)\n---\n', content, re.DOTALL)
|
|
91
|
+
if not match:
|
|
92
|
+
return None
|
|
93
|
+
try:
|
|
94
|
+
return yaml.safe_load(match.group(1))
|
|
95
|
+
except yaml.YAMLError as e:
|
|
96
|
+
print(f"ERROR: Invalid YAML frontmatter: {e}")
|
|
97
|
+
return None
|
|
98
|
+
|
|
99
|
+
|
|
100
|
+
def validate_naming(filename):
|
|
101
|
+
"""Validate filename follows naming conventions."""
|
|
102
|
+
errors = []
|
|
103
|
+
|
|
104
|
+
# Check if ends with .md
|
|
105
|
+
if not filename.endswith('.md'):
|
|
106
|
+
errors.append("Filename should end with .md")
|
|
107
|
+
return errors
|
|
108
|
+
|
|
109
|
+
stem = filename[:-3] # Remove .md
|
|
110
|
+
|
|
111
|
+
# Check if ends with known category suffix
|
|
112
|
+
has_suffix = any(stem.endswith(f"_{suffix}") for suffix in CATEGORY_SUFFIXES)
|
|
113
|
+
if not has_suffix:
|
|
114
|
+
errors.append(f"Filename should end with one of: {', '.join(['_' + s + '.md' for s in CATEGORY_SUFFIXES])}")
|
|
115
|
+
|
|
116
|
+
return errors
|
|
117
|
+
|
|
118
|
+
|
|
119
|
+
def validate_version(version):
|
|
120
|
+
"""Validate semantic version format (x.y.z)."""
|
|
121
|
+
if not version:
|
|
122
|
+
return False
|
|
123
|
+
pattern = r'^\d+\.\d+\.\d+$'
|
|
124
|
+
return bool(re.match(pattern, str(version)))
|
|
125
|
+
|
|
126
|
+
|
|
127
|
+
def validate_metadata(filepath):
|
|
128
|
+
"""Validate memory metadata."""
|
|
129
|
+
path = Path(filepath)
|
|
130
|
+
|
|
131
|
+
if not path.exists():
|
|
132
|
+
print(f"ERROR: File not found: {filepath}")
|
|
133
|
+
return False
|
|
134
|
+
|
|
135
|
+
print(f"Validating: {path.name}")
|
|
136
|
+
print("=" * 60)
|
|
137
|
+
|
|
138
|
+
errors = []
|
|
139
|
+
warnings = []
|
|
140
|
+
|
|
141
|
+
# Check naming convention
|
|
142
|
+
naming_errors = validate_naming(path.name)
|
|
143
|
+
errors.extend(naming_errors)
|
|
144
|
+
|
|
145
|
+
# Read content
|
|
146
|
+
content = path.read_text(encoding='utf-8')
|
|
147
|
+
|
|
148
|
+
# Extract frontmatter
|
|
149
|
+
metadata = extract_frontmatter(content)
|
|
150
|
+
if metadata is None:
|
|
151
|
+
errors.append("Missing or invalid YAML frontmatter (should be between --- markers)")
|
|
152
|
+
print_results(errors, warnings)
|
|
153
|
+
return False
|
|
154
|
+
|
|
155
|
+
# Determine category from filename suffix
|
|
156
|
+
category = None
|
|
157
|
+
stem = path.name[:-3]
|
|
158
|
+
for suffix in CATEGORY_SUFFIXES:
|
|
159
|
+
if stem.endswith(f"_{suffix}"):
|
|
160
|
+
category = suffix
|
|
161
|
+
break
|
|
162
|
+
|
|
163
|
+
# Check required fields based on category
|
|
164
|
+
# If category is troubleshoot or commit_log, they might not have strict requirements yet
|
|
165
|
+
if category in REQUIRED_FIELDS:
|
|
166
|
+
required = REQUIRED_FIELDS[category]
|
|
167
|
+
missing_fields = [field for field in required if field not in metadata]
|
|
168
|
+
if missing_fields:
|
|
169
|
+
errors.append(f"Missing required fields: {', '.join(missing_fields)}")
|
|
170
|
+
else:
|
|
171
|
+
# Default to ssot requirements if unknown but has suffix
|
|
172
|
+
required = REQUIRED_FIELDS["ssot"]
|
|
173
|
+
missing_fields = [field for field in required if field not in metadata]
|
|
174
|
+
if missing_fields:
|
|
175
|
+
errors.append(f"Missing required fields: {', '.join(missing_fields)}")
|
|
176
|
+
|
|
177
|
+
# Validate version format (if present)
|
|
178
|
+
if "version" in metadata:
|
|
179
|
+
if not validate_version(metadata["version"]):
|
|
180
|
+
errors.append(f"Invalid version format: {metadata['version']} (should be x.y.z)")
|
|
181
|
+
|
|
182
|
+
# Validate updated timestamp (if present)
|
|
183
|
+
if "updated" in metadata:
|
|
184
|
+
updated = str(metadata["updated"])
|
|
185
|
+
# Check for ISO8601-ish format (basic validation)
|
|
186
|
+
if not re.match(r'^\d{4}-\d{2}-\d{2}', updated):
|
|
187
|
+
warnings.append(f"Updated timestamp should be ISO8601 format: {updated}")
|
|
188
|
+
|
|
189
|
+
# Check for domain field (should be array)
|
|
190
|
+
if "domain" in metadata:
|
|
191
|
+
if not isinstance(metadata["domain"], list):
|
|
192
|
+
errors.append("Domain field should be an array/list")
|
|
193
|
+
|
|
194
|
+
# Check for changelog in SSOT files
|
|
195
|
+
if category == "ssot" and "changelog" not in metadata:
|
|
196
|
+
warnings.append("SSOT files should include a changelog section in frontmatter")
|
|
197
|
+
|
|
198
|
+
print_results(errors, warnings)
|
|
199
|
+
|
|
200
|
+
# Regenerate INDEX block unconditionally (navigation aid, independent of schema validity)
|
|
201
|
+
headings = extract_headings(content)
|
|
202
|
+
if headings:
|
|
203
|
+
table = generate_index_table(headings)
|
|
204
|
+
new_content = inject_index(content, table)
|
|
205
|
+
if new_content != content:
|
|
206
|
+
path.write_text(new_content, encoding="utf-8")
|
|
207
|
+
print(" ✏️ INDEX regenerated.")
|
|
208
|
+
|
|
209
|
+
return len(errors) == 0
|
|
210
|
+
|
|
211
|
+
|
|
212
|
+
def print_results(errors, warnings):
|
|
213
|
+
"""Print validation results."""
|
|
214
|
+
if errors:
|
|
215
|
+
print("\n❌ ERRORS:")
|
|
216
|
+
for error in errors:
|
|
217
|
+
print(f" - {error}")
|
|
218
|
+
|
|
219
|
+
if warnings:
|
|
220
|
+
print("\n⚠️ WARNINGS:")
|
|
221
|
+
for warning in warnings:
|
|
222
|
+
print(f" - {warning}")
|
|
223
|
+
|
|
224
|
+
if not errors and not warnings:
|
|
225
|
+
print("\n✅ VALID: All checks passed!")
|
|
226
|
+
|
|
227
|
+
print("=" * 60)
|
|
228
|
+
|
|
229
|
+
|
|
230
|
+
def main():
|
|
231
|
+
if len(sys.argv) < 2:
|
|
232
|
+
print("Usage: validate_metadata.py <memory-file.md>")
|
|
233
|
+
print("Example: validate_metadata.py analytics_volatility_ssot.md")
|
|
234
|
+
sys.exit(1)
|
|
235
|
+
|
|
236
|
+
filepath = sys.argv[1]
|
|
237
|
+
success = validate_metadata(filepath)
|
|
238
|
+
sys.exit(0 if success else 1)
|
|
239
|
+
|
|
240
|
+
|
|
241
|
+
if __name__ == "__main__":
|
|
242
|
+
main()
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
# Changelog
|
|
2
|
+
|
|
3
|
+
All notable changes to this project will be documented in this file.
|
|
4
|
+
|
|
5
|
+
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
|
6
|
+
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
|
7
|
+
|
|
8
|
+
## [Unreleased]
|
|
9
|
+
|
|
10
|
+
## [{initial_version}] - {release_date}
|
|
11
|
+
|
|
12
|
+
### Added
|
|
13
|
+
- Initial release
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
package/skills/documenting/tests/__pycache__/test_validate_metadata.cpython-314-pytest-9.0.2.pyc
ADDED
|
Binary file
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
# Integration test for documenting skill workflows
|
|
3
|
+
|
|
4
|
+
set -e # Exit on error
|
|
5
|
+
|
|
6
|
+
TEST_DIR=$(mktemp -d)
|
|
7
|
+
echo "Test directory: $TEST_DIR"
|
|
8
|
+
|
|
9
|
+
# Ensure we are in the skill root (skills/documenting)
|
|
10
|
+
cd "$(dirname "$0")/.."
|
|
11
|
+
SKILL_ROOT=$(pwd)
|
|
12
|
+
echo "Skill root: $SKILL_ROOT"
|
|
13
|
+
|
|
14
|
+
# Test 1: Initialize CHANGELOG
|
|
15
|
+
echo "Test 1: Initialize CHANGELOG"
|
|
16
|
+
python3 -m scripts.changelog.init_changelog "$TEST_DIR/CHANGELOG.md"
|
|
17
|
+
python3 -m scripts.changelog.validate_changelog "$TEST_DIR/CHANGELOG.md"
|
|
18
|
+
echo "✅ CHANGELOG initialization passed"
|
|
19
|
+
|
|
20
|
+
# Test 2: Add entries
|
|
21
|
+
echo ""
|
|
22
|
+
echo "Test 2: Add entries to CHANGELOG"
|
|
23
|
+
python3 -m scripts.changelog.add_entry "$TEST_DIR/CHANGELOG.md" Added "Feature A"
|
|
24
|
+
python3 -m scripts.changelog.add_entry "$TEST_DIR/CHANGELOG.md" Fixed "Bug B"
|
|
25
|
+
python3 -m scripts.changelog.add_entry "$TEST_DIR/CHANGELOG.md" Changed "Refactor C"
|
|
26
|
+
python3 -m scripts.changelog.validate_changelog "$TEST_DIR/CHANGELOG.md"
|
|
27
|
+
echo "✅ Entry addition passed"
|
|
28
|
+
|
|
29
|
+
# Test 3: Bump release
|
|
30
|
+
echo ""
|
|
31
|
+
echo "Test 3: Bump release version"
|
|
32
|
+
python3 -m scripts.changelog.bump_release "$TEST_DIR/CHANGELOG.md" "1.0.0"
|
|
33
|
+
python3 -m scripts.changelog.validate_changelog "$TEST_DIR/CHANGELOG.md"
|
|
34
|
+
|
|
35
|
+
# Verify [Unreleased] is empty and [1.0.0] has entries
|
|
36
|
+
# Check 3 lines after Unreleased (header, empty line, next header)
|
|
37
|
+
if grep -A 2 "\[Unreleased\]" "$TEST_DIR/CHANGELOG.md" | grep -q "^- "; then
|
|
38
|
+
echo "❌ [Unreleased] should be empty after bump"
|
|
39
|
+
cat "$TEST_DIR/CHANGELOG.md"
|
|
40
|
+
exit 1
|
|
41
|
+
fi
|
|
42
|
+
|
|
43
|
+
if ! grep -A 10 "\[1.0.0\]" "$TEST_DIR/CHANGELOG.md" | grep -q "Feature A"; then
|
|
44
|
+
echo "❌ [1.0.0] should contain Feature A"
|
|
45
|
+
exit 1
|
|
46
|
+
fi
|
|
47
|
+
echo "✅ Release bump passed"
|
|
48
|
+
|
|
49
|
+
# Test 4: Orchestrator
|
|
50
|
+
echo ""
|
|
51
|
+
echo "Test 4: Orchestrator workflow"
|
|
52
|
+
mkdir -p "$TEST_DIR/.serena/memories"
|
|
53
|
+
cp "$TEST_DIR/CHANGELOG.md" "$TEST_DIR/CHANGELOG.md.backup"
|
|
54
|
+
|
|
55
|
+
python3 -m scripts.orchestrator "$TEST_DIR" feature "Orchestrator test feature" --scope=test --category=testing
|
|
56
|
+
|
|
57
|
+
# Verify CHANGELOG updated
|
|
58
|
+
if ! grep -q "Orchestrator test feature" "$TEST_DIR/CHANGELOG.md"; then
|
|
59
|
+
echo "❌ Orchestrator should have updated CHANGELOG"
|
|
60
|
+
exit 1
|
|
61
|
+
fi
|
|
62
|
+
echo "✅ Orchestrator passed"
|
|
63
|
+
|
|
64
|
+
# Cleanup
|
|
65
|
+
rm -rf "$TEST_DIR"
|
|
66
|
+
|
|
67
|
+
echo ""
|
|
68
|
+
echo "========================================="
|
|
69
|
+
echo "✅ All integration tests passed!"
|
|
70
|
+
echo "========================================="
|
|
@@ -0,0 +1,201 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""Tests for CHANGELOG management scripts."""
|
|
3
|
+
|
|
4
|
+
import pytest
|
|
5
|
+
from pathlib import Path
|
|
6
|
+
from scripts.changelog.validate_changelog import validate_changelog
|
|
7
|
+
from scripts.changelog.add_entry import add_entry, ChangeCategory
|
|
8
|
+
from scripts.changelog.bump_release import bump_release
|
|
9
|
+
from datetime import date
|
|
10
|
+
import tempfile
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
def test_valid_changelog_passes():
|
|
14
|
+
"""Valid Keep a Changelog format should pass validation."""
|
|
15
|
+
valid_content = """# Changelog
|
|
16
|
+
|
|
17
|
+
All notable changes to this project will be documented in this file.
|
|
18
|
+
|
|
19
|
+
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
|
20
|
+
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
|
21
|
+
|
|
22
|
+
## [Unreleased]
|
|
23
|
+
|
|
24
|
+
### Added
|
|
25
|
+
- New feature description
|
|
26
|
+
|
|
27
|
+
## [1.0.0] - 2026-02-01
|
|
28
|
+
|
|
29
|
+
### Added
|
|
30
|
+
- Initial release
|
|
31
|
+
"""
|
|
32
|
+
result = validate_changelog(valid_content)
|
|
33
|
+
assert result["valid"] is True
|
|
34
|
+
assert len(result["errors"]) == 0
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
def test_missing_unreleased_section_fails():
|
|
38
|
+
"""CHANGELOG without [Unreleased] section should fail."""
|
|
39
|
+
invalid_content = """# Changelog
|
|
40
|
+
|
|
41
|
+
## [1.0.0] - 2026-02-01
|
|
42
|
+
|
|
43
|
+
### Added
|
|
44
|
+
- Initial release
|
|
45
|
+
"""
|
|
46
|
+
result = validate_changelog(invalid_content)
|
|
47
|
+
assert result["valid"] is False
|
|
48
|
+
assert any("Unreleased" in err for err in result["errors"])
|
|
49
|
+
|
|
50
|
+
|
|
51
|
+
def test_invalid_semver_fails():
|
|
52
|
+
"""CHANGELOG with invalid semver should fail."""
|
|
53
|
+
invalid_content = """# Changelog
|
|
54
|
+
|
|
55
|
+
## [Unreleased]
|
|
56
|
+
|
|
57
|
+
## [1.0] - 2026-02-01
|
|
58
|
+
|
|
59
|
+
### Added
|
|
60
|
+
- Initial release
|
|
61
|
+
"""
|
|
62
|
+
result = validate_changelog(invalid_content)
|
|
63
|
+
assert result["valid"] is False
|
|
64
|
+
assert any("semantic version" in err.lower() for err in result["errors"])
|
|
65
|
+
|
|
66
|
+
|
|
67
|
+
def test_invalid_category_fails():
|
|
68
|
+
"""CHANGELOG with invalid category should fail."""
|
|
69
|
+
invalid_content = """# Changelog
|
|
70
|
+
|
|
71
|
+
## [Unreleased]
|
|
72
|
+
|
|
73
|
+
### InvalidCategory
|
|
74
|
+
- Some change
|
|
75
|
+
|
|
76
|
+
## [1.0.0] - 2026-02-01
|
|
77
|
+
|
|
78
|
+
### Added
|
|
79
|
+
- Initial release
|
|
80
|
+
"""
|
|
81
|
+
result = validate_changelog(invalid_content)
|
|
82
|
+
assert result["valid"] is False
|
|
83
|
+
assert any("category" in err.lower() for err in result["errors"])
|
|
84
|
+
|
|
85
|
+
|
|
86
|
+
def test_add_entry_to_unreleased():
|
|
87
|
+
"""Adding entry should place it under [Unreleased] in correct category."""
|
|
88
|
+
changelog_content = """# Changelog
|
|
89
|
+
|
|
90
|
+
## [Unreleased]
|
|
91
|
+
|
|
92
|
+
## [1.0.0] - 2026-02-01
|
|
93
|
+
|
|
94
|
+
### Added
|
|
95
|
+
- Initial release
|
|
96
|
+
"""
|
|
97
|
+
|
|
98
|
+
result = add_entry(
|
|
99
|
+
changelog_content,
|
|
100
|
+
category=ChangeCategory.ADDED,
|
|
101
|
+
description="New feature X"
|
|
102
|
+
)
|
|
103
|
+
|
|
104
|
+
assert "### Added" in result
|
|
105
|
+
assert "- New feature X" in result
|
|
106
|
+
# Should be under [Unreleased], not under [1.0.0]
|
|
107
|
+
unreleased_section = result.split("## [1.0.0]")[0]
|
|
108
|
+
assert "- New feature X" in unreleased_section
|
|
109
|
+
|
|
110
|
+
|
|
111
|
+
def test_add_entry_creates_category_if_missing():
|
|
112
|
+
"""Adding entry should create category section if it doesn't exist."""
|
|
113
|
+
changelog_content = """# Changelog
|
|
114
|
+
|
|
115
|
+
## [Unreleased]
|
|
116
|
+
|
|
117
|
+
## [1.0.0] - 2026-02-01
|
|
118
|
+
|
|
119
|
+
### Added
|
|
120
|
+
- Initial release
|
|
121
|
+
"""
|
|
122
|
+
|
|
123
|
+
result = add_entry(
|
|
124
|
+
changelog_content,
|
|
125
|
+
category=ChangeCategory.FIXED,
|
|
126
|
+
description="Bug fix Y"
|
|
127
|
+
)
|
|
128
|
+
|
|
129
|
+
assert "### Fixed" in result
|
|
130
|
+
assert "- Bug fix Y" in result
|
|
131
|
+
|
|
132
|
+
|
|
133
|
+
def test_add_entry_maintains_category_order():
|
|
134
|
+
"""Categories should be ordered: Added, Changed, Deprecated, Removed, Fixed, Security."""
|
|
135
|
+
changelog_content = """# Changelog
|
|
136
|
+
|
|
137
|
+
## [Unreleased]
|
|
138
|
+
|
|
139
|
+
### Added
|
|
140
|
+
- Feature A
|
|
141
|
+
|
|
142
|
+
### Fixed
|
|
143
|
+
- Bug B
|
|
144
|
+
|
|
145
|
+
## [1.0.0] - 2026-02-01
|
|
146
|
+
"""
|
|
147
|
+
|
|
148
|
+
result = add_entry(
|
|
149
|
+
changelog_content,
|
|
150
|
+
category=ChangeCategory.CHANGED,
|
|
151
|
+
description="Change C"
|
|
152
|
+
)
|
|
153
|
+
|
|
154
|
+
# Changed should appear between Added and Fixed
|
|
155
|
+
lines = result.split('\n')
|
|
156
|
+
added_idx = next(i for i, line in enumerate(lines) if "### Added" in line)
|
|
157
|
+
changed_idx = next(i for i, line in enumerate(lines) if "### Changed" in line)
|
|
158
|
+
fixed_idx = next(i for i, line in enumerate(lines) if "### Fixed" in line)
|
|
159
|
+
|
|
160
|
+
assert added_idx < changed_idx < fixed_idx
|
|
161
|
+
|
|
162
|
+
|
|
163
|
+
def test_bump_release_moves_unreleased():
|
|
164
|
+
"""Bumping release should move [Unreleased] to [X.Y.Z] - DATE."""
|
|
165
|
+
changelog_content = """# Changelog
|
|
166
|
+
|
|
167
|
+
## [Unreleased]
|
|
168
|
+
|
|
169
|
+
### Added
|
|
170
|
+
- New feature
|
|
171
|
+
|
|
172
|
+
## [1.0.0] - 2026-01-01
|
|
173
|
+
|
|
174
|
+
### Added
|
|
175
|
+
- Initial release
|
|
176
|
+
"""
|
|
177
|
+
|
|
178
|
+
result = bump_release(changelog_content, "1.1.0")
|
|
179
|
+
|
|
180
|
+
# Should have new version section
|
|
181
|
+
assert "## [1.1.0]" in result
|
|
182
|
+
assert f"## [1.1.0] - {date.today().strftime('%Y-%m-%d')}" in result
|
|
183
|
+
|
|
184
|
+
# New feature should be under 1.1.0 now
|
|
185
|
+
version_section = result.split("## [1.0.0]")[0]
|
|
186
|
+
assert "- New feature" in version_section
|
|
187
|
+
|
|
188
|
+
# [Unreleased] should be empty
|
|
189
|
+
unreleased_section = result.split("## [1.1.0]")[0]
|
|
190
|
+
assert "## [Unreleased]" in unreleased_section
|
|
191
|
+
# No ### categories under [Unreleased]
|
|
192
|
+
unreleased_lines = unreleased_section.split("## [Unreleased]")[1].split('\\n')
|
|
193
|
+
assert not any(line.startswith("###") for line in unreleased_lines)
|
|
194
|
+
|
|
195
|
+
|
|
196
|
+
def test_bump_release_validates_semver():
|
|
197
|
+
"""Bumping with invalid semver should raise error."""
|
|
198
|
+
changelog_content = "## [Unreleased]"
|
|
199
|
+
|
|
200
|
+
with pytest.raises(ValueError, match="semantic version"):
|
|
201
|
+
bump_release(changelog_content, "1.0")
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
import pytest
|
|
2
|
+
from pathlib import Path
|
|
3
|
+
import sys
|
|
4
|
+
sys.path.insert(0, str(Path(__file__).parent.parent / "scripts"))
|
|
5
|
+
from drift_detector import extract_tracks, match_files_to_tracks, format_scan_report
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
MEMORY_WITH_TRACKS = """---
|
|
9
|
+
title: Test SSOT
|
|
10
|
+
version: 1.0.0
|
|
11
|
+
updated: 2026-02-01
|
|
12
|
+
tracks:
|
|
13
|
+
- "cli/src/**/*.ts"
|
|
14
|
+
- "hooks/**/*.py"
|
|
15
|
+
---
|
|
16
|
+
|
|
17
|
+
## Purpose
|
|
18
|
+
Test memory.
|
|
19
|
+
"""
|
|
20
|
+
|
|
21
|
+
MEMORY_NO_TRACKS = """---
|
|
22
|
+
title: No Tracks
|
|
23
|
+
version: 1.0.0
|
|
24
|
+
updated: 2026-02-01
|
|
25
|
+
---
|
|
26
|
+
|
|
27
|
+
## Purpose
|
|
28
|
+
No tracking.
|
|
29
|
+
"""
|
|
30
|
+
|
|
31
|
+
|
|
32
|
+
def test_extract_tracks_finds_globs():
|
|
33
|
+
tracks = extract_tracks(MEMORY_WITH_TRACKS)
|
|
34
|
+
assert tracks == ["cli/src/**/*.ts", "hooks/**/*.py"]
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
def test_extract_tracks_empty_when_missing():
|
|
38
|
+
tracks = extract_tracks(MEMORY_NO_TRACKS)
|
|
39
|
+
assert tracks == []
|
|
40
|
+
|
|
41
|
+
|
|
42
|
+
def test_match_files_to_tracks_hit():
|
|
43
|
+
files = ["cli/src/core/diff.ts", "hooks/skill-suggestion.py"]
|
|
44
|
+
tracks = ["cli/src/**/*.ts", "hooks/**/*.py"]
|
|
45
|
+
matched = match_files_to_tracks(files, tracks)
|
|
46
|
+
assert "cli/src/core/diff.ts" in matched
|
|
47
|
+
assert "hooks/skill-suggestion.py" in matched
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
def test_match_files_to_tracks_miss():
|
|
51
|
+
files = ["docs/README.md"]
|
|
52
|
+
tracks = ["cli/src/**/*.ts"]
|
|
53
|
+
assert match_files_to_tracks(files, tracks) == []
|
|
54
|
+
|
|
55
|
+
|
|
56
|
+
def test_format_scan_report_stale():
|
|
57
|
+
stale = {"my_memory": {"files": ["cli/src/core/diff.ts"], "updated": "2026-02-01"}}
|
|
58
|
+
report = format_scan_report(stale)
|
|
59
|
+
assert "my_memory" in report
|
|
60
|
+
assert "cli/src/core/diff.ts" in report
|
|
61
|
+
|
|
62
|
+
|
|
63
|
+
def test_format_scan_report_clean():
|
|
64
|
+
report = format_scan_report({})
|
|
65
|
+
assert any(word in report.lower() for word in ["clean", "no stale", "up to date"])
|
|
66
|
+
|
|
67
|
+
|
|
68
|
+
def test_match_files_to_tracks_no_false_positive():
|
|
69
|
+
"""A .py file in an unrelated dir must NOT match hooks/**/*.py"""
|
|
70
|
+
files = ["cli/src/transform.py", "docs/something.py"]
|
|
71
|
+
tracks = ["hooks/**/*.py"]
|
|
72
|
+
assert match_files_to_tracks(files, tracks) == []
|
|
73
|
+
|
|
74
|
+
|
|
75
|
+
def test_match_files_to_tracks_direct_child():
|
|
76
|
+
"""hooks/**/*.py must match files directly under hooks/ (no subdir)"""
|
|
77
|
+
files = ["hooks/skill-suggestion.py"]
|
|
78
|
+
tracks = ["hooks/**/*.py"]
|
|
79
|
+
matched = match_files_to_tracks(files, tracks)
|
|
80
|
+
assert "hooks/skill-suggestion.py" in matched
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""Tests for documentation orchestrator."""
|
|
3
|
+
|
|
4
|
+
import pytest
|
|
5
|
+
from pathlib import Path
|
|
6
|
+
from scripts.orchestrator import DocumentingOrchestrator, ChangeType
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
def test_orchestrator_routes_to_correct_docs(tmp_path):
|
|
10
|
+
"""Orchestrator should update relevant docs based on change type."""
|
|
11
|
+
# Setup test project
|
|
12
|
+
project_root = tmp_path / "test_project"
|
|
13
|
+
project_root.mkdir()
|
|
14
|
+
|
|
15
|
+
changelog = project_root / "CHANGELOG.md"
|
|
16
|
+
changelog.write_text("""# Changelog
|
|
17
|
+
|
|
18
|
+
## [Unreleased]
|
|
19
|
+
|
|
20
|
+
## [0.1.0] - 2026-02-01
|
|
21
|
+
|
|
22
|
+
### Added
|
|
23
|
+
- Initial release
|
|
24
|
+
""", encoding="utf-8")
|
|
25
|
+
|
|
26
|
+
serena_memories = project_root / ".serena" / "memories"
|
|
27
|
+
serena_memories.mkdir(parents=True)
|
|
28
|
+
|
|
29
|
+
orchestrator = DocumentingOrchestrator(project_root)
|
|
30
|
+
|
|
31
|
+
result = orchestrator.document_change(
|
|
32
|
+
change_type=ChangeType.FEATURE,
|
|
33
|
+
description="Add new API endpoint",
|
|
34
|
+
details={
|
|
35
|
+
"scope": "api-endpoints",
|
|
36
|
+
"category": "backend",
|
|
37
|
+
"files_changed": ["server.py", "routes/api.py"]
|
|
38
|
+
}
|
|
39
|
+
)
|
|
40
|
+
|
|
41
|
+
# Should update CHANGELOG
|
|
42
|
+
assert result["changelog_updated"] is True
|
|
43
|
+
# Should create/update SSOT
|
|
44
|
+
assert result["ssot_updated"] is False # Placeholder impl in plan returns False
|
|
45
|
+
# Should suggest README update
|
|
46
|
+
assert "readme_suggestions" in result
|
|
47
|
+
|
|
48
|
+
|
|
49
|
+
def test_orchestrator_validates_all_docs():
|
|
50
|
+
"""Orchestrator should validate all documentation after updates."""
|
|
51
|
+
# Test validation integration
|
|
52
|
+
pass
|