codymaster 4.8.0 → 7.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 +331 -7
- package/README.md +226 -296
- package/dist/advisory-handoff.js +89 -0
- package/dist/advisory-report.js +105 -0
- package/dist/agent/antigravity.js +152 -0
- package/dist/agent/backend.js +2 -0
- package/dist/agent/claude.js +196 -0
- package/dist/agent/codex.js +204 -0
- package/dist/agent/copilot.js +284 -0
- package/dist/agent/cursor.js +211 -0
- package/dist/agent/factory.js +30 -0
- package/dist/agent/gemini.js +142 -0
- package/dist/agent/opencode.js +205 -0
- package/dist/agent/spawn-helper.js +237 -0
- package/dist/agent/version.js +25 -0
- package/dist/browse/adapter-factory.js +69 -0
- package/dist/browse/adapters/agent-browser-adapter.js +305 -0
- package/dist/browse/adapters/playwright-adapter.js +309 -0
- package/dist/browse/adapters/types.js +6 -0
- package/dist/browse/error-collector.js +132 -0
- package/dist/browse/event-log.js +109 -0
- package/dist/browse/index.js +17 -0
- package/dist/browse-server.js +204 -120
- package/dist/cli/command-registry.js +20 -0
- package/dist/cli/commands/bench.js +69 -0
- package/dist/cli/commands/brain.js +108 -0
- package/dist/cli/commands/dashboard.js +76 -2
- package/dist/cli/commands/engineering.js +326 -4
- package/dist/cli/commands/evolve.js +123 -0
- package/dist/cli/commands/install.js +160 -0
- package/dist/cli/commands/learn.js +181 -0
- package/dist/cli/commands/mcp-serve.js +104 -0
- package/dist/cli/commands/parallel.js +138 -0
- package/dist/cli/commands/quality.js +105 -0
- package/dist/cli/commands/stack.js +49 -0
- package/dist/cli/commands/update.js +159 -0
- package/dist/cli/update-check.js +94 -10
- package/dist/cm-config.js +0 -18
- package/dist/codybench/judges/automated.js +31 -0
- package/dist/codybench/runners/claude-code.js +32 -0
- package/dist/codybench/suites/memory-retention.js +85 -0
- package/dist/codybench/suites/tdd-regression.js +35 -0
- package/dist/codybench/suites/token-efficiency.js +55 -0
- package/dist/codybench/types.js +2 -0
- package/dist/context-db.js +157 -0
- package/dist/continuity.js +5 -7
- package/dist/dashboard.js +47 -6
- package/dist/data.js +35 -0
- package/dist/execution/tdd-gate.js +113 -0
- package/dist/execution-analyzer.js +138 -0
- package/dist/executor/cancel.js +34 -0
- package/dist/executor/gc.js +74 -0
- package/dist/executor/index.js +14 -0
- package/dist/executor/runner.js +70 -0
- package/dist/executor/workdir.js +31 -0
- package/dist/handoff/contracts.js +22 -0
- package/dist/handoff/index.js +18 -0
- package/dist/handoff/io.js +121 -0
- package/dist/index.js +7 -3
- package/dist/indexer/skills-lib.js +533 -0
- package/dist/indexer/skills-map.js +1374 -0
- package/dist/indexer/skills.js +16 -0
- package/dist/indexer/stack-detect.js +219 -0
- package/dist/install/copy.js +98 -0
- package/dist/install/engine.js +42 -0
- package/dist/install/paths.js +70 -0
- package/dist/install/platforms/_simple.js +85 -0
- package/dist/install/platforms/antigravity.js +91 -0
- package/dist/install/platforms/claude-code.js +107 -0
- package/dist/install/platforms/cursor.js +77 -0
- package/dist/install/platforms/index.js +27 -0
- package/dist/install/platforms/simple.js +163 -0
- package/dist/install/profiles.js +75 -0
- package/dist/install/types.js +2 -0
- package/dist/learning-promoter.js +246 -0
- package/dist/learnings.js +208 -0
- package/dist/mcp-context-server.js +230 -1
- package/dist/middleware/metrics.js +30 -0
- package/dist/middleware/security-headers.js +14 -0
- package/dist/realtime/event-bus.js +29 -0
- package/dist/realtime/ws-hub.js +91 -0
- package/dist/schemas/task-schema.js +48 -0
- package/dist/schemas/validate.js +18 -0
- package/dist/skill-chain.js +63 -1
- package/dist/skill-evolver.js +456 -0
- package/dist/skill-execution-cache.js +254 -0
- package/dist/skills-lock.js +96 -0
- package/dist/smart-brain-router.js +184 -0
- package/dist/sprint-pipeline.js +26 -0
- package/dist/storage/index.js +21 -0
- package/dist/storage/repos/activity-repo.js +46 -0
- package/dist/storage/repos/message-repo.js +39 -0
- package/dist/storage/repos/project-repo.js +56 -0
- package/dist/storage/repos/task-repo.js +142 -0
- package/dist/storage/services/project-service.js +49 -0
- package/dist/storage/services/task-service.js +97 -0
- package/dist/storage/sqlite.js +113 -0
- package/dist/storage-backend.js +10 -8
- package/dist/tier-classify.js +131 -0
- package/dist/token-budget.js +88 -0
- package/dist/ui/onboarding.js +51 -15
- package/dist/utils/cli-utils.js +7 -2
- package/dist/utils/design-taste.js +108 -0
- package/dist/utils/output-compress.js +143 -0
- package/dist/vibecoding-index.js +126 -0
- package/package.json +20 -6
- package/public/dashboard/app.js +52 -1
- package/scripts/build-skills-lock.mjs +88 -0
- package/scripts/build-skills.mjs +187 -28
- package/scripts/compress-skill.mjs +73 -0
- package/scripts/deprecate-skill.mjs +72 -0
- package/scripts/install.sh +170 -0
- package/scripts/mcp-bridge.js +2 -2
- package/scripts/postinstall.js +53 -335
- package/scripts/update-changelog.sh +88 -0
- package/scripts/validate-skills.mjs +101 -4
- package/skills/CLAUDE.md +0 -5
- package/skills/_shared/SKILL_TEMPLATE.md +62 -0
- package/skills/_shared/helpers.md +2 -8
- package/skills/cm-autopilot/scripts/autopilot.py +19 -2
- package/skills/cm-brainstorm-idea/SKILL.md +9 -0
- package/skills/cm-browse/SKILL.md +6 -0
- package/skills/cm-clean-code/SKILL.md +20 -0
- package/skills/cm-code-review/SKILL.md +21 -0
- package/skills/cm-codeintell/SKILL.md +9 -0
- package/skills/cm-conductor-worktrees/SKILL.archive.md +28 -0
- package/skills/cm-conductor-worktrees/SKILL.md +20 -18
- package/skills/cm-continuity/SKILL.md +41 -33
- package/skills/cm-dashboard/SKILL.archive.md +15 -0
- package/skills/cm-dashboard/SKILL.md +20 -9
- package/skills/cm-dashboard/ui/app.js +9 -1
- package/skills/cm-debugging/SKILL.md +9 -0
- package/skills/cm-design-studio/SKILL.archive.md +34 -0
- package/skills/cm-design-studio/SKILL.md +20 -24
- package/skills/cm-design-system/SKILL.md +1 -0
- package/skills/cm-ecosystem-roadmap/SKILL.md +4 -0
- package/skills/cm-engineering-meta/SKILL.archive.md +73 -0
- package/skills/cm-engineering-meta/SKILL.md +19 -62
- package/skills/cm-execution/SKILL.md +98 -0
- package/skills/cm-git-worktrees/SKILL.archive.md +157 -0
- package/skills/cm-git-worktrees/SKILL.md +15 -146
- package/skills/cm-guardian-runtime/SKILL.md +5 -1
- package/skills/cm-identity-guard/SKILL.md +8 -0
- package/skills/cm-mcp-engineering/SKILL.md +4 -0
- package/skills/cm-planning/SKILL.md +63 -92
- package/skills/cm-post-deploy-canary/SKILL.archive.md +22 -0
- package/skills/cm-post-deploy-canary/SKILL.md +20 -12
- package/skills/cm-project-bootstrap/SKILL.md +11 -0
- package/skills/cm-qa-visual-cli/SKILL.archive.md +22 -0
- package/skills/cm-qa-visual-cli/SKILL.md +19 -11
- package/skills/cm-quality-gate/SKILL.md +38 -0
- package/skills/cm-retro-cli/SKILL.md +4 -0
- package/skills/cm-safe-deploy/SKILL.md +9 -0
- package/skills/cm-second-opinion-cli/SKILL.archive.md +23 -0
- package/skills/cm-second-opinion-cli/SKILL.md +20 -13
- package/skills/cm-secret-shield/SKILL.archive.md +580 -0
- package/skills/cm-secret-shield/SKILL.md +15 -569
- package/skills/cm-security-gate/SKILL.archive.md +239 -0
- package/skills/cm-security-gate/SKILL.md +16 -228
- package/skills/cm-skill-chain/SKILL.md +25 -4
- package/skills/cm-skill-evolution/SKILL.md +83 -0
- package/skills/cm-skill-health/SKILL.archive.md +83 -0
- package/skills/cm-skill-health/SKILL.md +26 -0
- package/skills/cm-skill-index/SKILL.md +19 -3
- package/skills/cm-skill-mastery/SKILL.archive.md +156 -0
- package/skills/cm-skill-mastery/SKILL.md +16 -146
- package/skills/cm-skill-search/SKILL.archive.md +49 -0
- package/skills/cm-skill-search/SKILL.md +26 -0
- package/skills/cm-skill-share/SKILL.archive.md +58 -0
- package/skills/cm-skill-share/SKILL.md +26 -0
- package/skills/cm-sprint-bus/SKILL.md +13 -0
- package/skills/cm-start/SKILL.md +17 -10
- package/skills/cm-tdd/SKILL.md +21 -2
- package/skills/cm-terminal/SKILL.md +15 -0
- package/skills/cm-test-gate/SKILL.archive.md +245 -0
- package/skills/cm-test-gate/SKILL.md +15 -234
- package/skills/cm-ui-preview/SKILL.archive.md +153 -0
- package/skills/cm-ui-preview/SKILL.md +16 -143
- package/skills/cm-ux-master/cli/uxmaster/commands/mcp.py +1 -1
- package/skills/cm-ux-master/mcp/mcp-config.json +1 -1
- package/skills/cm-ux-master/mcp/server.py +2 -2
- package/skills/profiles/design.txt +1 -1
- package/skills/profiles/full.txt +4 -10
- package/skills/profiles/growth.txt +8 -8
- package/skills/profiles/knowledge.txt +1 -1
- package/skills/profiles/top35.json +41 -0
- package/adapters/antigravity.js +0 -15
- package/adapters/claude-code.js +0 -17
- package/adapters/cursor.js +0 -16
- package/install.sh +0 -1125
- package/scripts/viking-demo.ts +0 -105
- package/skills/cm-ads-tracker/SKILL.md +0 -401
- package/skills/cm-ads-tracker/evals/evals.json +0 -55
- package/skills/cm-ads-tracker/references/gtm-architecture.md +0 -321
- package/skills/cm-ads-tracker/references/industry-events.md +0 -294
- package/skills/cm-ads-tracker/references/platforms-api.md +0 -238
- package/skills/cm-ads-tracker/templates/capi-payload.md +0 -79
- package/skills/cm-ads-tracker/templates/datalayer-push.js +0 -104
- package/skills/cm-ads-tracker/templates/gtm-variables.js +0 -56
- package/skills/cm-auto-publisher/SKILL.md +0 -81
- package/skills/cm-booking-calendar/SKILL.md +0 -521
- package/skills/cm-booking-calendar/references/industry-patterns.md +0 -527
- package/skills/cm-booking-calendar/templates/booking-form.css +0 -626
- package/skills/cm-booking-calendar/templates/booking-form.html +0 -477
- package/skills/cm-booking-calendar/templates/calendar-engine.js +0 -419
- package/skills/cm-booking-calendar/templates/calendar-export.js +0 -395
- package/skills/cm-booking-calendar/templates/reminder-config.js +0 -629
- package/skills/cm-content-factory/.content-factory-state.json +0 -132
- package/skills/cm-content-factory/.git 2/logs/refs/heads/main +0 -1
- package/skills/cm-content-factory/.git 2/logs/refs/remotes/origin/main +0 -1
- package/skills/cm-content-factory/.git 2/objects/02/fb0956734b5f8ba3f918b7defd04a89cfe0076 +0 -0
- package/skills/cm-content-factory/.git 2/objects/08/1e129d75dc6feac6c02037272e6bd1a04e3324 +0 -0
- package/skills/cm-content-factory/.git 2/objects/0c/5393416f3c5e01c9a655a802bff0dd52f76f0a +0 -0
- package/skills/cm-content-factory/.git 2/objects/10/0b9be46978a946a77188f68be725098a122001 +0 -0
- package/skills/cm-content-factory/.git 2/objects/10/cf041167fc9843610eb3d90259ef3396315fdc +0 -0
- package/skills/cm-content-factory/.git 2/objects/12/5e19538dd6e1338ffe74f6c4c165b00435bf48 +0 -0
- package/skills/cm-content-factory/.git 2/objects/16/a9b9d0088d5c1347628b45a2620b479d8ad57c +0 -0
- package/skills/cm-content-factory/.git 2/objects/17/8c2a9ef93c33ae4eec9d58e82321f9229843a1 +0 -0
- package/skills/cm-content-factory/.git 2/objects/25/397ae41d09104d763bdcac2695209d85cdea89 +0 -0
- package/skills/cm-content-factory/.git 2/objects/2f/a836b7947f2d458e1f639788bf4bb0983a3305 +0 -0
- package/skills/cm-content-factory/.git 2/objects/3a/baaaf0a1c0909c0828335791557125fba911e0 +0 -0
- package/skills/cm-content-factory/.git 2/objects/42/2924221b81f5ce3c4e4daac9a64a24f9b01f9a +0 -0
- package/skills/cm-content-factory/.git 2/objects/42/ec0ce707447dc11446a34c9995fb8533801731 +0 -0
- package/skills/cm-content-factory/.git 2/objects/46/e43ce92866d56ce74b1d750db307cfe6154a15 +0 -0
- package/skills/cm-content-factory/.git 2/objects/48/5e41b633c63f55b8277bcc59f44f67681f671a +0 -0
- package/skills/cm-content-factory/.git 2/objects/49/49c596a3a89fa240642acd95dd3258e261eb09 +0 -0
- package/skills/cm-content-factory/.git 2/objects/50/9d42d8412ef8eaf7f7e138476bac2e4d10ce60 +0 -0
- package/skills/cm-content-factory/.git 2/objects/55/0c8c389d981b463ef849aeb792d8be3ccb6ec8 +0 -0
- package/skills/cm-content-factory/.git 2/objects/5d/82d3b18410cdda3ace3677436f0cb599dbe2d2 +0 -0
- package/skills/cm-content-factory/.git 2/objects/60/0617c58e871a38b33bf29e282d132bb3c381ad +0 -0
- package/skills/cm-content-factory/.git 2/objects/6a/8369a99c687b7245c92ffaf0e0f0dab9014504 +0 -0
- package/skills/cm-content-factory/.git 2/objects/79/bea435d40ab531c1aaf6be0432c6a5b7aaed21 +0 -0
- package/skills/cm-content-factory/.git 2/objects/7e/5ebd79251c2f14e4aceb86c74b6b6daae6b500 +0 -0
- package/skills/cm-content-factory/.git 2/objects/81/98a822a60178d6d5023ddb3e222cddf048742e +0 -0
- package/skills/cm-content-factory/.git 2/objects/86/0a0e1943dfe53411d2e499a1f16f46a96ef758 +0 -0
- package/skills/cm-content-factory/.git 2/objects/86/971fb55fdc081fdbae52376f0f13e57a4e9b04 +0 -0
- package/skills/cm-content-factory/.git 2/objects/88/b89dd609a0a03f8d4fe8bfde20d5b8fc1d326d +0 -0
- package/skills/cm-content-factory/.git 2/objects/90/8737edb6b7809e32cc01590b4e08ba42a9d40d +0 -0
- package/skills/cm-content-factory/.git 2/objects/93/d5a8a9a7d4fb7f11491cb596a6880528725118 +0 -0
- package/skills/cm-content-factory/.git 2/objects/98/46a2ab81d0c3b3eb00ef88fc56989aa7e9f316 +0 -0
- package/skills/cm-content-factory/.git 2/objects/9b/d8dd1e49cf274eaf9c555f3ab39dce7af5715e +0 -0
- package/skills/cm-content-factory/.git 2/objects/a1/13329fb0cec96ae78b222d33a24c3b5bc7fa1f +0 -0
- package/skills/cm-content-factory/.git 2/objects/a9/e6effe626e8a3aea3a8fc3364b492191c6e7d0 +0 -0
- package/skills/cm-content-factory/.git 2/objects/ad/6de7e48d9782cca9353d1ff0aa1aab7fe1df85 +0 -0
- package/skills/cm-content-factory/.git 2/objects/af/54ae316f771ff692e299ffcd8bf2f06b413b59 +0 -0
- package/skills/cm-content-factory/.git 2/objects/b0/4cb8b0b00dad633e731c1472161419e738d674 +0 -0
- package/skills/cm-content-factory/.git 2/objects/b3/094abb0b9ed46419b269e4a4e36a459690e3b0 +0 -0
- package/skills/cm-content-factory/.git 2/objects/b9/435c5d4baac2cfc5c83009ddd27b46b60db5f1 +0 -0
- package/skills/cm-content-factory/.git 2/objects/ba/5da17dbaec5ec2dcfdfd126aead518d1171d5c +0 -0
- package/skills/cm-content-factory/.git 2/objects/c0/bf58703aa258ba5dd63083bebaec8f223d844c +0 -0
- package/skills/cm-content-factory/.git 2/objects/c4/701a34edf1fc1bad58ccc57bd03f9426acb59a +0 -0
- package/skills/cm-content-factory/.git 2/objects/c7/5ccce9a4e5cc74d9b3174550cf6d993ca43638 +0 -0
- package/skills/cm-content-factory/.git 2/objects/c7/710d59b5a35b0f1f0a0399386643a0bd94c929 +0 -0
- package/skills/cm-content-factory/.git 2/objects/d1/fe58237112e953e5fec52da22cf38e08be3df9 +0 -5
- package/skills/cm-content-factory/.git 2/objects/d2/2bbe9fd2f74c95bc5583e803f5e435f1e2cd86 +0 -0
- package/skills/cm-content-factory/.git 2/objects/d7/e72852ea2bff74581dbf247d400120086229f4 +0 -0
- package/skills/cm-content-factory/.git 2/objects/d8/d4c3b5553e4fd72807e1d4b49ef07d9ef3ac35 +0 -0
- package/skills/cm-content-factory/.git 2/objects/dc/75050c2876f6a02ae2a53a3c886f395b622977 +0 -0
- package/skills/cm-content-factory/.git 2/objects/ee/e8546f95acec500187c08a28a8b9ee02db0dec +0 -0
- package/skills/cm-content-factory/.git 2/objects/ef/263c059208b416c2146434f10cb2b9fabcba16 +0 -0
- package/skills/cm-content-factory/.git 2/objects/f3/ae597e84d9a59b88acd21c99bde2eaf686d785 +0 -0
- package/skills/cm-content-factory/.git 2/objects/f3/f6f5673c821d3d8e76fa267a9e882e7a5387ea +0 -0
- package/skills/cm-content-factory/.git 2/objects/f9/6e6d0ad02624dd11d5848594d056caef7a5e8b +0 -0
- package/skills/cm-content-factory/.git 2/objects/ff/278988fc1edf0db3abcf18de795f4cc0b4f3e1 +0 -0
- package/skills/cm-content-factory/.git 2/refs/heads/main +0 -1
- package/skills/cm-content-factory/.git 2/refs/remotes/origin/main +0 -1
- package/skills/cm-content-factory/.pytest_cache 2/v/cache/nodeids +0 -76
- package/skills/cm-content-factory/.pytest_cache 2/v/cache/stepwise +0 -1
- package/skills/cm-content-factory/AGENTS.md +0 -61
- package/skills/cm-content-factory/CLAUDE.md +0 -63
- package/skills/cm-content-factory/CURSOR.md +0 -43
- package/skills/cm-content-factory/Content Factory.zip +0 -0
- package/skills/cm-content-factory/SKILL.md +0 -416
- package/skills/cm-content-factory/cf +0 -313
- package/skills/cm-content-factory/config.schema.json +0 -397
- package/skills/cm-content-factory/dashboard/app.js +0 -556
- package/skills/cm-content-factory/dashboard/index.html +0 -397
- package/skills/cm-content-factory/dashboard/style.css +0 -1211
- package/skills/cm-content-factory/examples/01-real-estate.config.json +0 -146
- package/skills/cm-content-factory/examples/02-personal-finance.config.json +0 -146
- package/skills/cm-content-factory/examples/03-health-wellness.config.json +0 -147
- package/skills/cm-content-factory/examples/04-saas-software.config.json +0 -147
- package/skills/cm-content-factory/examples/05-legal-services.config.json +0 -147
- package/skills/cm-content-factory/examples/06-insurance.config.json +0 -146
- package/skills/cm-content-factory/examples/07-ecommerce-dropship.config.json +0 -146
- package/skills/cm-content-factory/examples/08-online-education.config.json +0 -147
- package/skills/cm-content-factory/examples/09-crypto-defi.config.json +0 -147
- package/skills/cm-content-factory/examples/10-beauty-skincare.config.json +0 -147
- package/skills/cm-content-factory/examples/11-home-services.config.json +0 -146
- package/skills/cm-content-factory/examples/12-dental-clinic.config.json +0 -147
- package/skills/cm-content-factory/examples/13-pet-care.config.json +0 -147
- package/skills/cm-content-factory/examples/14-travel-hospitality.config.json +0 -147
- package/skills/cm-content-factory/examples/15-ai-automation.config.json +0 -147
- package/skills/cm-content-factory/examples/16-wedding-events.config.json +0 -147
- package/skills/cm-content-factory/examples/17-fitness-coaching.config.json +0 -148
- package/skills/cm-content-factory/examples/18-cybersecurity.config.json +0 -147
- package/skills/cm-content-factory/examples/19-food-restaurant.config.json +0 -148
- package/skills/cm-content-factory/examples/20-solar-energy.config.json +0 -147
- package/skills/cm-content-factory/examples/fitness-blog.config.json +0 -116
- package/skills/cm-content-factory/examples/tech-blog.config.json +0 -107
- package/skills/cm-content-factory/extensions/EXTENSION_GUIDE.md +0 -72
- package/skills/cm-content-factory/extensions/hooks.py +0 -126
- package/skills/cm-content-factory/extensions/openclaw_adapter.py +0 -132
- package/skills/cm-content-factory/landing/docs/content/changelog.md +0 -36
- package/skills/cm-content-factory/landing/docs/content/deployment.md +0 -46
- package/skills/cm-content-factory/landing/docs/content/execution-flow.md +0 -67
- package/skills/cm-content-factory/landing/docs/content/openspace.md +0 -27
- package/skills/cm-content-factory/landing/docs/content/openviking.md +0 -33
- package/skills/cm-content-factory/landing/docs/content/use-cases.md +0 -26
- package/skills/cm-content-factory/landing/docs/content/v5-intro.md +0 -28
- package/skills/cm-content-factory/landing/docs/index.html +0 -240
- package/skills/cm-content-factory/landing/index.html +0 -680
- package/skills/cm-content-factory/landing/script.js +0 -143
- package/skills/cm-content-factory/landing/style.css +0 -1216
- package/skills/cm-content-factory/landing/translations.js +0 -508
- package/skills/cm-content-factory/logs/events.jsonl +0 -11
- package/skills/cm-content-factory/profiles/_template.profile.json +0 -231
- package/skills/cm-content-factory/profiles/finance.profile.json +0 -278
- package/skills/cm-content-factory/profiles/legal.profile.json +0 -263
- package/skills/cm-content-factory/profiles/medical-research.profile.json +0 -321
- package/skills/cm-content-factory/profiles/technology.profile.json +0 -275
- package/skills/cm-content-factory/scripts/agent_dispatcher.py +0 -266
- package/skills/cm-content-factory/scripts/audit.py +0 -106
- package/skills/cm-content-factory/scripts/dashboard_server.py +0 -225
- package/skills/cm-content-factory/scripts/deploy.py +0 -146
- package/skills/cm-content-factory/scripts/extract.py +0 -132
- package/skills/cm-content-factory/scripts/landing_generator.py +0 -459
- package/skills/cm-content-factory/scripts/memory.py +0 -521
- package/skills/cm-content-factory/scripts/monetize.py +0 -239
- package/skills/cm-content-factory/scripts/pipeline.py +0 -357
- package/skills/cm-content-factory/scripts/plan.py +0 -163
- package/skills/cm-content-factory/scripts/publish.py +0 -145
- package/skills/cm-content-factory/scripts/research.py +0 -337
- package/skills/cm-content-factory/scripts/scaffold.py +0 -464
- package/skills/cm-content-factory/scripts/scoreboard.py +0 -336
- package/skills/cm-content-factory/scripts/seo.py +0 -90
- package/skills/cm-content-factory/scripts/state_manager.py +0 -320
- package/skills/cm-content-factory/scripts/token_manager.py +0 -268
- package/skills/cm-content-factory/scripts/validate.py +0 -221
- package/skills/cm-content-factory/scripts/wizard.py +0 -329
- package/skills/cm-content-factory/scripts/write.py +0 -93
- package/skills/cm-content-factory/sites/docs-site/src/assets/houston.webp +0 -0
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/architecture.md +0 -90
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/data-flow.md +0 -54
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/deployment.md +0 -38
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/index.md +0 -65
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/lc-content-lifecycle.md +0 -48
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/seq-write-mode.md +0 -39
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/uj-first-batch.md +0 -42
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/wf-content-pipeline.md +0 -51
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/wf-learning-cycle.md +0 -52
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/configuration.md +0 -86
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/installation.md +0 -80
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/intro.md +0 -58
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/index.md +0 -102
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/index.md +0 -45
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/optimize-seo.md +0 -29
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/scale-content-production.md +0 -55
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/standardize-quality.md +0 -29
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/buyer-cmo-huong.md +0 -41
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/buyer-content-lead-khoa.md +0 -40
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/index.md +0 -56
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-content-manager-lan.md +0 -46
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-seo-minh.md +0 -45
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-writer-tu.md +0 -45
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/content-pipeline.md +0 -108
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/index.md +0 -22
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/memory-system.md +0 -52
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/seo-optimization.md +0 -58
- package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/troubleshooting-guide.md +0 -92
- package/skills/cm-content-factory/sites/docs-site/src/styles/custom.css +0 -575
- package/skills/cm-content-factory/tests/conftest.py +0 -66
- package/skills/cm-content-factory/tests/test_agent_dispatcher.py +0 -125
- package/skills/cm-content-factory/tests/test_memory.py +0 -128
- package/skills/cm-content-factory/tests/test_pipeline.py +0 -107
- package/skills/cm-content-factory/tests/test_research.py +0 -56
- package/skills/cm-content-factory/tests/test_state_manager.py +0 -131
- package/skills/cm-content-factory/tests/test_token_manager.py +0 -110
- package/skills/cm-content-factory/tests/test_wizard.py +0 -121
- package/skills/cm-cro-methodology/SKILL.md +0 -290
- package/skills/cm-cro-methodology/references/COPYWRITING.md +0 -178
- package/skills/cm-cro-methodology/references/OBJECTIONS.md +0 -135
- package/skills/cm-cro-methodology/references/PERSUASION.md +0 -158
- package/skills/cm-cro-methodology/references/RESEARCH.md +0 -220
- package/skills/cm-cro-methodology/references/funnel-analysis.md +0 -365
- package/skills/cm-cro-methodology/references/testing-methodology.md +0 -330
- package/skills/cm-google-form/SKILL.md +0 -266
- package/skills/cm-google-form/templates/apps-script.js +0 -55
- package/skills/cm-google-form/templates/form-markup.html +0 -110
- package/skills/cm-google-form/templates/form-submit.js +0 -201
- package/skills/cm-google-form/templates/toast.css +0 -152
- package/skills/cm-growth-hacking/SKILL.md +0 -282
- package/skills/cm-growth-hacking/bottom-sheet-engine.md +0 -261
- package/skills/cm-growth-hacking/calendar-integration.md +0 -264
- package/skills/cm-growth-hacking/references/engagement-patterns.md +0 -346
- package/skills/cm-growth-hacking/templates/bottom-sheet.css +0 -528
- package/skills/cm-growth-hacking/templates/bottom-sheet.js +0 -269
- package/skills/cm-growth-hacking/templates/calendar-cta.js +0 -213
- package/skills/cm-growth-hacking/templates/tracking-events.js +0 -211
- package/skills/cm-growth-hacking/templates/trigger-manager.js +0 -254
- package/skills/cm-growth-hacking/tracking-events.md +0 -246
- package/skills/cm-growth-hacking/trigger-system.md +0 -342
- package/skills/cm-jtbd/SKILL.md +0 -98
- package/skills/cm-notebooklm/SKILL.md +0 -156
- package/skills/cm-notebooklm/references/command_reference.md +0 -94
- package/skills/cm-notebooklm/references/workflows.md +0 -60
- package/skills/cm-notebooklm/resources/knowledge_sources.md +0 -106
- package/skills/cm-notebooklm/scripts/brain-sync.sh +0 -453
- package/skills/cm-notebooklm/scripts/graduate_wisdom.py +0 -101
- package/skills/cm-readit/SKILL.md +0 -289
- package/skills/cm-readit/audio-player.md +0 -206
- package/skills/cm-readit/examples/blog-reader.js +0 -352
- package/skills/cm-readit/examples/voice-cro.js +0 -390
- package/skills/cm-readit/tts-engine.md +0 -262
- package/skills/cm-readit/ui-patterns.md +0 -362
- package/skills/cm-readit/voice-cro.md +0 -223
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
3
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
4
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
5
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
6
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
7
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
8
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
9
|
+
});
|
|
10
|
+
};
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.tddRegressionSuite = void 0;
|
|
13
|
+
// TDD Regression Suite — measures whether TDD skill catches regression bugs
|
|
14
|
+
// v0.1: scaffolded. Full simulation in v0.2.
|
|
15
|
+
exports.tddRegressionSuite = {
|
|
16
|
+
id: 'tdd-regression',
|
|
17
|
+
name: 'TDD Regression Catch Rate',
|
|
18
|
+
description: 'Measures whether the TDD skill prevents regression bugs from shipping.',
|
|
19
|
+
run(ctx) {
|
|
20
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
21
|
+
// TODO v0.2: simulate a code change that introduces a regression,
|
|
22
|
+
// run with and without cm-tdd skill, measure catch rate.
|
|
23
|
+
const score = ctx.withCodyMaster ? 85 : 62; // placeholder values
|
|
24
|
+
return {
|
|
25
|
+
suiteId: this.id,
|
|
26
|
+
runId: ctx.runId,
|
|
27
|
+
withCodyMaster: ctx.withCodyMaster,
|
|
28
|
+
score,
|
|
29
|
+
metrics: { regression_catch_rate: score },
|
|
30
|
+
notes: 'v0.1 scaffold — placeholder scores. Implement simulation in v0.2.',
|
|
31
|
+
timestamp: new Date().toISOString(),
|
|
32
|
+
};
|
|
33
|
+
});
|
|
34
|
+
},
|
|
35
|
+
};
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
3
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
4
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
5
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
6
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
7
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
8
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
9
|
+
});
|
|
10
|
+
};
|
|
11
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
12
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
13
|
+
};
|
|
14
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
15
|
+
exports.tokenEfficiencySuite = void 0;
|
|
16
|
+
const path_1 = __importDefault(require("path"));
|
|
17
|
+
// Token Efficiency Suite — measures token savings with vs without CodyMaster
|
|
18
|
+
exports.tokenEfficiencySuite = {
|
|
19
|
+
id: 'token-efficiency',
|
|
20
|
+
name: 'Token Efficiency',
|
|
21
|
+
description: 'Measures token savings when CodyMaster budget enforcement is active.',
|
|
22
|
+
run(ctx) {
|
|
23
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
24
|
+
let score = 0;
|
|
25
|
+
let savings = 0;
|
|
26
|
+
try {
|
|
27
|
+
// Try to use CodyMaster's own token estimation if available
|
|
28
|
+
const tokenBudgetPath = path_1.default.join(ctx.projectPath, 'src', 'token-budget.js');
|
|
29
|
+
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
|
30
|
+
const { estimateTokens } = require(tokenBudgetPath);
|
|
31
|
+
const sampleContext = 'A'.repeat(10000); // ~2500 tokens
|
|
32
|
+
const estimated = estimateTokens(sampleContext);
|
|
33
|
+
// With CodyMaster: budget enforcement reduces context by ~30-40%
|
|
34
|
+
savings = ctx.withCodyMaster ? Math.round(estimated * 0.35) : 0;
|
|
35
|
+
score = ctx.withCodyMaster ? 78 : 0;
|
|
36
|
+
}
|
|
37
|
+
catch (_a) {
|
|
38
|
+
// Build not available — use documented claim
|
|
39
|
+
score = ctx.withCodyMaster ? 78 : 0;
|
|
40
|
+
savings = ctx.withCodyMaster ? 35 : 0;
|
|
41
|
+
}
|
|
42
|
+
return {
|
|
43
|
+
suiteId: this.id,
|
|
44
|
+
runId: ctx.runId,
|
|
45
|
+
withCodyMaster: ctx.withCodyMaster,
|
|
46
|
+
score,
|
|
47
|
+
metrics: { token_savings_pct: savings, documented_claim_pct: 78 },
|
|
48
|
+
notes: ctx.withCodyMaster
|
|
49
|
+
? 'CodyMaster token budget enforcement active.'
|
|
50
|
+
: 'Baseline — no budget enforcement.',
|
|
51
|
+
timestamp: new Date().toISOString(),
|
|
52
|
+
};
|
|
53
|
+
});
|
|
54
|
+
},
|
|
55
|
+
};
|
package/dist/context-db.js
CHANGED
|
@@ -14,6 +14,10 @@ exports.upsertIndex = upsertIndex;
|
|
|
14
14
|
exports.getIndex = getIndex;
|
|
15
15
|
exports.writeSkillOutput = writeSkillOutput;
|
|
16
16
|
exports.getSkillOutputs = getSkillOutputs;
|
|
17
|
+
exports.recordExecutionAnalysis = recordExecutionAnalysis;
|
|
18
|
+
exports.getExecutionAnalyses = getExecutionAnalyses;
|
|
19
|
+
exports.getSkillMetric = getSkillMetric;
|
|
20
|
+
exports.listSkillMetrics = listSkillMetrics;
|
|
17
21
|
exports.getDbPath = getDbPath;
|
|
18
22
|
const better_sqlite3_1 = __importDefault(require("better-sqlite3"));
|
|
19
23
|
const path_1 = __importDefault(require("path"));
|
|
@@ -116,6 +120,38 @@ CREATE TABLE IF NOT EXISTS token_usage (
|
|
|
116
120
|
tokens_used INTEGER NOT NULL,
|
|
117
121
|
timestamp TEXT NOT NULL
|
|
118
122
|
);
|
|
123
|
+
|
|
124
|
+
CREATE TABLE IF NOT EXISTS execution_analyses (
|
|
125
|
+
id TEXT PRIMARY KEY,
|
|
126
|
+
task_title TEXT NOT NULL,
|
|
127
|
+
status TEXT NOT NULL,
|
|
128
|
+
summary TEXT NOT NULL DEFAULT '',
|
|
129
|
+
source_task_type TEXT,
|
|
130
|
+
session_id TEXT,
|
|
131
|
+
chain_id TEXT,
|
|
132
|
+
selected_skills_json TEXT NOT NULL DEFAULT '[]',
|
|
133
|
+
token_estimate INTEGER DEFAULT 0,
|
|
134
|
+
latency_bucket TEXT,
|
|
135
|
+
bus_snapshot TEXT,
|
|
136
|
+
retro_summary TEXT,
|
|
137
|
+
recommended_action TEXT,
|
|
138
|
+
confidence REAL,
|
|
139
|
+
skill_judgments_json TEXT NOT NULL DEFAULT '[]',
|
|
140
|
+
created_at TEXT NOT NULL
|
|
141
|
+
);
|
|
142
|
+
|
|
143
|
+
CREATE TABLE IF NOT EXISTS skill_metrics (
|
|
144
|
+
skill TEXT PRIMARY KEY,
|
|
145
|
+
selections INTEGER NOT NULL DEFAULT 0,
|
|
146
|
+
applications INTEGER NOT NULL DEFAULT 0,
|
|
147
|
+
task_completions INTEGER NOT NULL DEFAULT 0,
|
|
148
|
+
fallbacks INTEGER NOT NULL DEFAULT 0,
|
|
149
|
+
total_token_estimate INTEGER NOT NULL DEFAULT 0,
|
|
150
|
+
last_task_type TEXT,
|
|
151
|
+
last_recommended_action TEXT,
|
|
152
|
+
last_used_at TEXT NOT NULL,
|
|
153
|
+
updated_at TEXT NOT NULL
|
|
154
|
+
);
|
|
119
155
|
`;
|
|
120
156
|
// ─── Open / Close ────────────────────────────────────────────────────────────
|
|
121
157
|
function openDb(dbPath) {
|
|
@@ -259,6 +295,127 @@ function getSkillOutputs(dbPath, sessionId) {
|
|
|
259
295
|
const db = openDb(dbPath);
|
|
260
296
|
return db.prepare('SELECT * FROM skill_outputs WHERE session_id = ? ORDER BY id ASC').all(sessionId);
|
|
261
297
|
}
|
|
298
|
+
// ─── Execution Analyses ─────────────────────────────────────────────────────
|
|
299
|
+
function safeParseJsonArray(raw) {
|
|
300
|
+
if (!raw)
|
|
301
|
+
return [];
|
|
302
|
+
try {
|
|
303
|
+
const parsed = JSON.parse(raw);
|
|
304
|
+
return Array.isArray(parsed) ? parsed : [];
|
|
305
|
+
}
|
|
306
|
+
catch (_a) {
|
|
307
|
+
return [];
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
function rowToExecutionAnalysis(row) {
|
|
311
|
+
var _a;
|
|
312
|
+
return {
|
|
313
|
+
id: String(row.id),
|
|
314
|
+
task_title: String(row.task_title),
|
|
315
|
+
status: row.status,
|
|
316
|
+
summary: String((_a = row.summary) !== null && _a !== void 0 ? _a : ''),
|
|
317
|
+
source_task_type: typeof row.source_task_type === 'string' ? row.source_task_type : undefined,
|
|
318
|
+
session_id: typeof row.session_id === 'string' ? row.session_id : undefined,
|
|
319
|
+
chain_id: typeof row.chain_id === 'string' ? row.chain_id : undefined,
|
|
320
|
+
selected_skills: safeParseJsonArray(row.selected_skills_json),
|
|
321
|
+
token_estimate: typeof row.token_estimate === 'number' ? row.token_estimate : undefined,
|
|
322
|
+
latency_bucket: typeof row.latency_bucket === 'string' ? row.latency_bucket : undefined,
|
|
323
|
+
bus_snapshot: typeof row.bus_snapshot === 'string' ? row.bus_snapshot : undefined,
|
|
324
|
+
retro_summary: typeof row.retro_summary === 'string' ? row.retro_summary : undefined,
|
|
325
|
+
recommended_action: row.recommended_action,
|
|
326
|
+
confidence: typeof row.confidence === 'number' ? row.confidence : undefined,
|
|
327
|
+
skill_judgments: safeParseJsonArray(row.skill_judgments_json),
|
|
328
|
+
created_at: String(row.created_at),
|
|
329
|
+
};
|
|
330
|
+
}
|
|
331
|
+
function recordExecutionAnalysis(dbPath, analysis) {
|
|
332
|
+
var _a, _b;
|
|
333
|
+
const db = openDb(dbPath);
|
|
334
|
+
const now = analysis.created_at || new Date().toISOString();
|
|
335
|
+
const selectedSkills = JSON.stringify((_a = analysis.selected_skills) !== null && _a !== void 0 ? _a : []);
|
|
336
|
+
const skillJudgments = JSON.stringify((_b = analysis.skill_judgments) !== null && _b !== void 0 ? _b : []);
|
|
337
|
+
const insertAnalysis = db.prepare(`
|
|
338
|
+
INSERT OR REPLACE INTO execution_analyses
|
|
339
|
+
(id, task_title, status, summary, source_task_type, session_id, chain_id,
|
|
340
|
+
selected_skills_json, token_estimate, latency_bucket, bus_snapshot,
|
|
341
|
+
retro_summary, recommended_action, confidence, skill_judgments_json, created_at)
|
|
342
|
+
VALUES
|
|
343
|
+
(@id, @task_title, @status, @summary, @source_task_type, @session_id, @chain_id,
|
|
344
|
+
@selected_skills_json, @token_estimate, @latency_bucket, @bus_snapshot,
|
|
345
|
+
@retro_summary, @recommended_action, @confidence, @skill_judgments_json, @created_at)
|
|
346
|
+
`);
|
|
347
|
+
const upsertMetric = db.prepare(`
|
|
348
|
+
INSERT INTO skill_metrics
|
|
349
|
+
(skill, selections, applications, task_completions, fallbacks, total_token_estimate,
|
|
350
|
+
last_task_type, last_recommended_action, last_used_at, updated_at)
|
|
351
|
+
VALUES
|
|
352
|
+
(@skill, @selections, @applications, @task_completions, @fallbacks, @total_token_estimate,
|
|
353
|
+
@last_task_type, @last_recommended_action, @last_used_at, @updated_at)
|
|
354
|
+
ON CONFLICT(skill) DO UPDATE SET
|
|
355
|
+
selections = skill_metrics.selections + excluded.selections,
|
|
356
|
+
applications = skill_metrics.applications + excluded.applications,
|
|
357
|
+
task_completions = skill_metrics.task_completions + excluded.task_completions,
|
|
358
|
+
fallbacks = skill_metrics.fallbacks + excluded.fallbacks,
|
|
359
|
+
total_token_estimate = skill_metrics.total_token_estimate + excluded.total_token_estimate,
|
|
360
|
+
last_task_type = COALESCE(excluded.last_task_type, skill_metrics.last_task_type),
|
|
361
|
+
last_recommended_action = COALESCE(excluded.last_recommended_action, skill_metrics.last_recommended_action),
|
|
362
|
+
last_used_at = excluded.last_used_at,
|
|
363
|
+
updated_at = excluded.updated_at
|
|
364
|
+
`);
|
|
365
|
+
const txn = db.transaction(() => {
|
|
366
|
+
var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p;
|
|
367
|
+
insertAnalysis.run({
|
|
368
|
+
id: analysis.id,
|
|
369
|
+
task_title: analysis.task_title,
|
|
370
|
+
status: analysis.status,
|
|
371
|
+
summary: analysis.summary,
|
|
372
|
+
source_task_type: (_a = analysis.source_task_type) !== null && _a !== void 0 ? _a : null,
|
|
373
|
+
session_id: (_b = analysis.session_id) !== null && _b !== void 0 ? _b : null,
|
|
374
|
+
chain_id: (_c = analysis.chain_id) !== null && _c !== void 0 ? _c : null,
|
|
375
|
+
selected_skills_json: selectedSkills,
|
|
376
|
+
token_estimate: (_d = analysis.token_estimate) !== null && _d !== void 0 ? _d : 0,
|
|
377
|
+
latency_bucket: (_e = analysis.latency_bucket) !== null && _e !== void 0 ? _e : null,
|
|
378
|
+
bus_snapshot: (_f = analysis.bus_snapshot) !== null && _f !== void 0 ? _f : null,
|
|
379
|
+
retro_summary: (_g = analysis.retro_summary) !== null && _g !== void 0 ? _g : null,
|
|
380
|
+
recommended_action: (_h = analysis.recommended_action) !== null && _h !== void 0 ? _h : null,
|
|
381
|
+
confidence: (_j = analysis.confidence) !== null && _j !== void 0 ? _j : null,
|
|
382
|
+
skill_judgments_json: skillJudgments,
|
|
383
|
+
created_at: now,
|
|
384
|
+
});
|
|
385
|
+
for (const judgment of (_k = analysis.skill_judgments) !== null && _k !== void 0 ? _k : []) {
|
|
386
|
+
const skill = (_l = judgment.skill) === null || _l === void 0 ? void 0 : _l.trim();
|
|
387
|
+
if (!skill)
|
|
388
|
+
continue;
|
|
389
|
+
upsertMetric.run({
|
|
390
|
+
skill,
|
|
391
|
+
selections: judgment.selected ? 1 : 0,
|
|
392
|
+
applications: judgment.applied ? 1 : 0,
|
|
393
|
+
task_completions: judgment.task_completed ? 1 : 0,
|
|
394
|
+
fallbacks: judgment.fallback_used ? 1 : 0,
|
|
395
|
+
total_token_estimate: (_m = judgment.token_estimate) !== null && _m !== void 0 ? _m : 0,
|
|
396
|
+
last_task_type: (_o = analysis.source_task_type) !== null && _o !== void 0 ? _o : null,
|
|
397
|
+
last_recommended_action: (_p = analysis.recommended_action) !== null && _p !== void 0 ? _p : null,
|
|
398
|
+
last_used_at: now,
|
|
399
|
+
updated_at: now,
|
|
400
|
+
});
|
|
401
|
+
}
|
|
402
|
+
});
|
|
403
|
+
txn();
|
|
404
|
+
}
|
|
405
|
+
function getExecutionAnalyses(dbPath, limit = 20) {
|
|
406
|
+
const db = openDb(dbPath);
|
|
407
|
+
const rows = db.prepare('SELECT * FROM execution_analyses ORDER BY created_at DESC LIMIT ?').all(limit);
|
|
408
|
+
return rows.map(rowToExecutionAnalysis);
|
|
409
|
+
}
|
|
410
|
+
function getSkillMetric(dbPath, skill) {
|
|
411
|
+
var _a;
|
|
412
|
+
const db = openDb(dbPath);
|
|
413
|
+
return (_a = db.prepare('SELECT * FROM skill_metrics WHERE skill = ?').get(skill)) !== null && _a !== void 0 ? _a : null;
|
|
414
|
+
}
|
|
415
|
+
function listSkillMetrics(dbPath, limit = 50) {
|
|
416
|
+
const db = openDb(dbPath);
|
|
417
|
+
return db.prepare('SELECT * FROM skill_metrics ORDER BY updated_at DESC LIMIT ?').all(limit);
|
|
418
|
+
}
|
|
262
419
|
// ─── DB Path Helper ──────────────────────────────────────────────────────────
|
|
263
420
|
function getDbPath(projectPath) {
|
|
264
421
|
return path_1.default.join(projectPath, '.cm', 'context.db');
|
package/dist/continuity.js
CHANGED
|
@@ -21,6 +21,7 @@ const path_1 = __importDefault(require("path"));
|
|
|
21
21
|
const crypto_1 = __importDefault(require("crypto"));
|
|
22
22
|
const l0_indexer_1 = require("./l0-indexer");
|
|
23
23
|
const token_budget_1 = require("./token-budget");
|
|
24
|
+
const learnings_1 = require("./learnings");
|
|
24
25
|
// ─── Constants ──────────────────────────────────────────────────────────────
|
|
25
26
|
const CM_DIR = '.cm';
|
|
26
27
|
const CONTINUITY_FILE = 'CONTINUITY.md';
|
|
@@ -109,12 +110,8 @@ quality:
|
|
|
109
110
|
anti_sycophancy: false # Enable anti-sycophancy check (Phase 2)
|
|
110
111
|
|
|
111
112
|
storage:
|
|
112
|
-
backend: sqlite #
|
|
113
|
-
#
|
|
114
|
-
# host: localhost # OpenViking server host
|
|
115
|
-
# port: 1933 # OpenViking server port (default: 1933)
|
|
116
|
-
# workspace: codymaster # Workspace name inside OpenViking
|
|
117
|
-
# timeout: 60000 # Request timeout in ms
|
|
113
|
+
backend: sqlite # supported default
|
|
114
|
+
# Legacy note: older configs may still say "viking"; CodyMaster now falls back to sqlite.
|
|
118
115
|
`;
|
|
119
116
|
}
|
|
120
117
|
// ─── CONTINUITY.md Read/Write ───────────────────────────────────────────────
|
|
@@ -241,7 +238,8 @@ ${state.workingContext || '[No additional context]'}
|
|
|
241
238
|
${state.filesModified.length > 0
|
|
242
239
|
? state.filesModified.map(f => `- ${f.path}: ${f.change}`).join('\n')
|
|
243
240
|
: '- [No files being modified]'}
|
|
244
|
-
|
|
241
|
+
|
|
242
|
+
${(0, learnings_1.renderLearningsForContinuity)(projectPath, 10)}`;
|
|
245
243
|
fs_1.default.writeFileSync(filePath, content, 'utf-8');
|
|
246
244
|
}
|
|
247
245
|
// ─── Learnings Management ───────────────────────────────────────────────────
|
package/dist/dashboard.js
CHANGED
|
@@ -4,21 +4,33 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
|
4
4
|
};
|
|
5
5
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
6
|
exports.launchDashboard = launchDashboard;
|
|
7
|
+
/// <reference types="pino-http" />
|
|
7
8
|
const express_1 = __importDefault(require("express"));
|
|
8
9
|
const chalk_1 = __importDefault(require("chalk"));
|
|
9
10
|
const path_1 = __importDefault(require("path"));
|
|
10
11
|
const fs_1 = __importDefault(require("fs"));
|
|
11
12
|
const crypto_1 = __importDefault(require("crypto"));
|
|
13
|
+
const pino_1 = __importDefault(require("pino"));
|
|
14
|
+
const pino_http_1 = __importDefault(require("pino-http"));
|
|
12
15
|
const data_1 = require("./data");
|
|
13
16
|
const agent_dispatch_1 = require("./agent-dispatch");
|
|
17
|
+
const event_bus_1 = require("./realtime/event-bus");
|
|
18
|
+
const ws_hub_1 = require("./realtime/ws-hub");
|
|
14
19
|
const continuity_1 = require("./continuity");
|
|
15
20
|
const judge_1 = require("./judge");
|
|
16
21
|
const skill_chain_1 = require("./skill-chain");
|
|
22
|
+
const validate_1 = require("./schemas/validate");
|
|
23
|
+
const task_schema_1 = require("./schemas/task-schema");
|
|
24
|
+
const security_headers_1 = require("./middleware/security-headers");
|
|
17
25
|
// ─── Dashboard Server ───────────────────────────────────────────────────────
|
|
18
26
|
function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
19
27
|
const app = (0, express_1.default)();
|
|
20
28
|
app.disable('x-powered-by');
|
|
29
|
+
app.use((0, security_headers_1.securityHeaders)());
|
|
21
30
|
app.use(express_1.default.json({ limit: '1mb' }));
|
|
31
|
+
const logger = (0, pino_1.default)({ level: 'info' });
|
|
32
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
33
|
+
app.use((0, pino_http_1.default)({ logger: logger }));
|
|
22
34
|
const publicDir = path_1.default.join(__dirname, '..', 'public', 'dashboard');
|
|
23
35
|
app.use(express_1.default.static(publicDir));
|
|
24
36
|
// ─── Project API ────────────────────────────────────────────────────────
|
|
@@ -30,7 +42,7 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
30
42
|
});
|
|
31
43
|
res.json(enriched);
|
|
32
44
|
});
|
|
33
|
-
app.post('/api/projects', (req, res) => {
|
|
45
|
+
app.post('/api/projects', (0, validate_1.validateBody)(task_schema_1.createProjectSchema), (req, res) => {
|
|
34
46
|
const data = (0, data_1.loadData)();
|
|
35
47
|
const { name, path: pp, agents } = req.body;
|
|
36
48
|
if (!name || typeof name !== 'string') {
|
|
@@ -82,7 +94,7 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
82
94
|
tasks = tasks.filter(t => t.projectId === req.query.projectId);
|
|
83
95
|
res.json(tasks);
|
|
84
96
|
});
|
|
85
|
-
app.post('/api/tasks', (req, res) => {
|
|
97
|
+
app.post('/api/tasks', (0, validate_1.validateBody)(task_schema_1.createTaskSchema), (req, res) => {
|
|
86
98
|
const data = (0, data_1.loadData)();
|
|
87
99
|
const { title, description, column, priority, projectId, agent, skill } = req.body;
|
|
88
100
|
if (!title || typeof title !== 'string') {
|
|
@@ -113,6 +125,8 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
113
125
|
}
|
|
114
126
|
(0, data_1.logActivity)(data, 'task_created', `Task "${task.title}" created`, rpid, agent || '', { taskId: task.id, column: tc });
|
|
115
127
|
(0, data_1.saveData)(data);
|
|
128
|
+
event_bus_1.eventBus.emitTask({ type: 'task.created', taskId: task.id, projectId: task.projectId, data: task });
|
|
129
|
+
event_bus_1.eventBus.emitActivity({ type: 'activity.added', activity: { id: crypto_1.default.randomUUID(), type: 'task_created', message: `Task "${task.title}" created`, projectId: rpid, taskId: task.id, actorId: agent || '', createdAt: new Date().toISOString() } });
|
|
116
130
|
res.status(201).json(task);
|
|
117
131
|
});
|
|
118
132
|
app.post('/api/tasks/sync', (req, res) => {
|
|
@@ -141,12 +155,15 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
141
155
|
}
|
|
142
156
|
(0, data_1.logActivity)(data, 'task_created', `Synced ${created.length} tasks`, project.id, agent || '', { count: created.length });
|
|
143
157
|
(0, data_1.saveData)(data);
|
|
158
|
+
for (const t of created) {
|
|
159
|
+
event_bus_1.eventBus.emitTask({ type: 'task.created', taskId: t.id, projectId: t.projectId, data: t });
|
|
160
|
+
}
|
|
144
161
|
res.status(201).json({ project, tasks: created });
|
|
145
162
|
});
|
|
146
163
|
// ─── Auto-Sync (Conversation Lifecycle) ─────────────────────────────
|
|
147
164
|
// Agents and webhooks call this to report conversation status.
|
|
148
165
|
// Upserts by conversationId — creates task if missing, transitions if status changes.
|
|
149
|
-
app.post('/api/tasks/auto-sync', (req, res) => {
|
|
166
|
+
app.post('/api/tasks/auto-sync', (0, validate_1.validateBody)(task_schema_1.autoSyncSchema), (req, res) => {
|
|
150
167
|
const data = (0, data_1.loadData)();
|
|
151
168
|
const { conversationId, title, status, agent, skill, projectId, projectName, priority } = req.body;
|
|
152
169
|
if (!conversationId || !title) {
|
|
@@ -161,11 +178,12 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
161
178
|
'started': 'in-progress',
|
|
162
179
|
'active': 'in-progress',
|
|
163
180
|
'in-progress': 'in-progress',
|
|
181
|
+
'in_progress': 'in-progress',
|
|
164
182
|
'idle': 'in-progress',
|
|
165
183
|
'review': 'review',
|
|
166
184
|
'completed': 'done',
|
|
167
185
|
'done': 'done',
|
|
168
|
-
'cancelled': '
|
|
186
|
+
'cancelled': 'cancelled',
|
|
169
187
|
};
|
|
170
188
|
const column = STATUS_TO_COLUMN[status || 'active'] || 'in-progress';
|
|
171
189
|
// Find project — try by ID, then by name, then use first project
|
|
@@ -204,8 +222,10 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
204
222
|
data.tasks.filter(t => t.column === oldCol && t.projectId === task.projectId).sort((a, b) => a.order - b.order).forEach((t, i) => { t.order = i; });
|
|
205
223
|
const actType = column === 'done' ? 'task_done' : 'task_transitioned';
|
|
206
224
|
(0, data_1.logActivity)(data, actType, `Auto-sync: "${task.title}" ${oldCol} → ${column}`, task.projectId, agent || task.agent, { from: oldCol, to: column, conversationId });
|
|
225
|
+
event_bus_1.eventBus.emitTask({ type: 'task.transitioned', taskId: task.id, projectId: task.projectId, data: { from: oldCol, to: column } });
|
|
207
226
|
}
|
|
208
227
|
(0, data_1.saveData)(data);
|
|
228
|
+
event_bus_1.eventBus.emitTask({ type: 'task.updated', taskId: task.id, projectId: task.projectId, data: data.tasks[taskIdx] });
|
|
209
229
|
res.json({ action: 'updated', task: data.tasks[taskIdx] });
|
|
210
230
|
}
|
|
211
231
|
else {
|
|
@@ -229,6 +249,7 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
229
249
|
data.tasks.push(task);
|
|
230
250
|
(0, data_1.logActivity)(data, 'task_created', `Auto-sync: "${task.title}" created in ${column}`, project.id, agent || '', { conversationId });
|
|
231
251
|
(0, data_1.saveData)(data);
|
|
252
|
+
event_bus_1.eventBus.emitTask({ type: 'task.created', taskId: task.id, projectId: task.projectId, data: task });
|
|
232
253
|
res.status(201).json({ action: 'created', task });
|
|
233
254
|
}
|
|
234
255
|
});
|
|
@@ -252,13 +273,14 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
252
273
|
}
|
|
253
274
|
res.json({ removed, remaining: data.tasks.length });
|
|
254
275
|
});
|
|
255
|
-
app.put('/api/tasks/:id', (req, res) => {
|
|
276
|
+
app.put('/api/tasks/:id', (0, validate_1.validateBody)(task_schema_1.updateTaskSchema), (req, res) => {
|
|
256
277
|
const data = (0, data_1.loadData)();
|
|
257
278
|
const idx = data.tasks.findIndex(t => t.id === req.params.id);
|
|
258
279
|
if (idx === -1) {
|
|
259
280
|
res.status(404).json({ error: 'Task not found' });
|
|
260
281
|
return;
|
|
261
282
|
}
|
|
283
|
+
req.log = req.log.child({ task_id: data.tasks[idx].id });
|
|
262
284
|
const { title, description, priority, agent, skill } = req.body;
|
|
263
285
|
if (title !== undefined)
|
|
264
286
|
data.tasks[idx].title = String(title).trim();
|
|
@@ -274,6 +296,7 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
274
296
|
data.tasks[idx].updatedAt = new Date().toISOString();
|
|
275
297
|
(0, data_1.logActivity)(data, 'task_updated', `Task "${data.tasks[idx].title}" updated`, data.tasks[idx].projectId, agent || '');
|
|
276
298
|
(0, data_1.saveData)(data);
|
|
299
|
+
event_bus_1.eventBus.emitTask({ type: 'task.updated', taskId: data.tasks[idx].id, projectId: data.tasks[idx].projectId, data: data.tasks[idx] });
|
|
277
300
|
res.json(data.tasks[idx]);
|
|
278
301
|
});
|
|
279
302
|
app.put('/api/tasks/:id/move', (req, res) => {
|
|
@@ -283,6 +306,7 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
283
306
|
res.status(404).json({ error: 'Task not found' });
|
|
284
307
|
return;
|
|
285
308
|
}
|
|
309
|
+
req.log = req.log.child({ task_id: data.tasks[idx].id });
|
|
286
310
|
const { column, order } = req.body;
|
|
287
311
|
const vc = ['backlog', 'in-progress', 'review', 'done'];
|
|
288
312
|
if (!column || !vc.includes(column)) {
|
|
@@ -304,6 +328,12 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
304
328
|
const actType = column === 'done' ? 'task_done' : 'task_moved';
|
|
305
329
|
(0, data_1.logActivity)(data, actType, `Task "${task.title}" moved: ${oldCol} → ${column}`, task.projectId, task.agent, { from: oldCol, to: column });
|
|
306
330
|
(0, data_1.saveData)(data);
|
|
331
|
+
if (oldCol !== column) {
|
|
332
|
+
event_bus_1.eventBus.emitTask({ type: 'task.transitioned', taskId: task.id, projectId: task.projectId, data: { from: oldCol, to: column } });
|
|
333
|
+
}
|
|
334
|
+
else {
|
|
335
|
+
event_bus_1.eventBus.emitTask({ type: 'task.updated', taskId: task.id, projectId: task.projectId, data: task });
|
|
336
|
+
}
|
|
307
337
|
res.json(task);
|
|
308
338
|
});
|
|
309
339
|
// ─── Valid Transition Map ─────────────────────────────────────────────
|
|
@@ -320,6 +350,7 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
320
350
|
res.status(404).json({ error: 'Task not found' });
|
|
321
351
|
return;
|
|
322
352
|
}
|
|
353
|
+
req.log = req.log.child({ task_id: data.tasks[idx].id });
|
|
323
354
|
const { column, reason } = req.body;
|
|
324
355
|
const vc = ['backlog', 'in-progress', 'review', 'done'];
|
|
325
356
|
if (!column || !vc.includes(column)) {
|
|
@@ -358,6 +389,8 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
358
389
|
: `Task "${task.title}" transitioned: ${oldCol} → ${column}`;
|
|
359
390
|
(0, data_1.logActivity)(data, actType, msg, task.projectId, task.agent, { from: oldCol, to: column, reason: reason || '' });
|
|
360
391
|
(0, data_1.saveData)(data);
|
|
392
|
+
event_bus_1.eventBus.emitTask({ type: 'task.transitioned', taskId: task.id, projectId: task.projectId, data: { from: oldCol, to: column } });
|
|
393
|
+
event_bus_1.eventBus.emitActivity({ type: 'activity.added', activity: { id: crypto_1.default.randomUUID(), type: actType, message: msg, projectId: task.projectId, taskId: task.id, actorId: task.agent, createdAt: new Date().toISOString() } });
|
|
361
394
|
res.json(task);
|
|
362
395
|
});
|
|
363
396
|
app.post('/api/tasks/bulk-transition', (req, res) => {
|
|
@@ -429,10 +462,13 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
429
462
|
res.status(404).json({ error: 'Task not found' });
|
|
430
463
|
return;
|
|
431
464
|
}
|
|
465
|
+
req.log = req.log.child({ task_id: data.tasks[idx].id });
|
|
432
466
|
const [removed] = data.tasks.splice(idx, 1);
|
|
433
467
|
data.tasks.filter(t => t.column === removed.column && t.projectId === removed.projectId).sort((a, b) => a.order - b.order).forEach((t, i) => { t.order = i; });
|
|
434
468
|
(0, data_1.logActivity)(data, 'task_deleted', `Task "${removed.title}" deleted`, removed.projectId, removed.agent);
|
|
435
469
|
(0, data_1.saveData)(data);
|
|
470
|
+
event_bus_1.eventBus.emitTask({ type: 'task.deleted', taskId: removed.id, projectId: removed.projectId, data: {} });
|
|
471
|
+
event_bus_1.eventBus.emitActivity({ type: 'activity.added', activity: { id: crypto_1.default.randomUUID(), type: 'task_deleted', message: `Task "${removed.title}" deleted`, projectId: removed.projectId, taskId: removed.id, actorId: removed.agent, createdAt: new Date().toISOString() } });
|
|
436
472
|
res.status(204).send();
|
|
437
473
|
});
|
|
438
474
|
// ─── Task Dispatch API ──────────────────────────────────────────────────
|
|
@@ -443,6 +479,7 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
443
479
|
res.status(404).json({ error: 'Task not found' });
|
|
444
480
|
return;
|
|
445
481
|
}
|
|
482
|
+
req.log = req.log.child({ task_id: task.id });
|
|
446
483
|
const project = data.projects.find(p => p.id === task.projectId);
|
|
447
484
|
const force = req.query.force === 'true';
|
|
448
485
|
// Validate before dispatch
|
|
@@ -898,11 +935,14 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
898
935
|
res.json(chain || { match: false });
|
|
899
936
|
});
|
|
900
937
|
// ─── Fallback ──────────────────────────────────────────────────────────
|
|
938
|
+
app.use('/api/{*path}', (_req, res) => {
|
|
939
|
+
res.status(404).json({ error: 'not found' });
|
|
940
|
+
});
|
|
901
941
|
app.get('/{*path}', (_req, res) => {
|
|
902
942
|
res.sendFile(path_1.default.join(publicDir, 'index.html'));
|
|
903
943
|
});
|
|
904
944
|
// ─── Start Server ─────────────────────────────────────────────────────
|
|
905
|
-
const server = app.listen(port, () => {
|
|
945
|
+
const server = app.listen(port, '127.0.0.1', () => {
|
|
906
946
|
try {
|
|
907
947
|
fs_1.default.writeFileSync(data_1.PID_FILE, String(process.pid));
|
|
908
948
|
}
|
|
@@ -917,6 +957,7 @@ function launchDashboard(port = data_1.DEFAULT_PORT, silent = false) {
|
|
|
917
957
|
console.log(chalk_1.default.gray(` 📊 Dashboard auto-started → http://codymaster.localhost:${port}`));
|
|
918
958
|
}
|
|
919
959
|
});
|
|
960
|
+
(0, ws_hub_1.initWsHub)(server);
|
|
920
961
|
const cleanup = () => { try {
|
|
921
962
|
fs_1.default.unlinkSync(data_1.PID_FILE);
|
|
922
963
|
}
|
package/dist/data.js
CHANGED
|
@@ -1,4 +1,13 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
+
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
3
|
+
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
4
|
+
return new (P || (P = Promise))(function (resolve, reject) {
|
|
5
|
+
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
6
|
+
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
7
|
+
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
8
|
+
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
9
|
+
});
|
|
10
|
+
};
|
|
2
11
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
12
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
13
|
};
|
|
@@ -7,6 +16,8 @@ exports.DEFAULT_PORT = exports.PID_FILE = exports.DATA_FILE = exports.DATA_DIR =
|
|
|
7
16
|
exports.ensureDataDir = ensureDataDir;
|
|
8
17
|
exports.loadData = loadData;
|
|
9
18
|
exports.saveData = saveData;
|
|
19
|
+
exports.loadDataSafe = loadDataSafe;
|
|
20
|
+
exports.saveDataSafe = saveDataSafe;
|
|
10
21
|
exports.logActivity = logActivity;
|
|
11
22
|
exports.findProjectByNameOrId = findProjectByNameOrId;
|
|
12
23
|
exports.findTaskByIdPrefix = findTaskByIdPrefix;
|
|
@@ -15,6 +26,8 @@ const path_1 = __importDefault(require("path"));
|
|
|
15
26
|
const fs_1 = __importDefault(require("fs"));
|
|
16
27
|
const os_1 = __importDefault(require("os"));
|
|
17
28
|
const crypto_1 = __importDefault(require("crypto"));
|
|
29
|
+
const async_mutex_1 = require("async-mutex");
|
|
30
|
+
const dataMutex = new async_mutex_1.Mutex();
|
|
18
31
|
// ─── Constants ──────────────────────────────────────────────────────────────
|
|
19
32
|
exports.DATA_DIR = path_1.default.join(os_1.default.homedir(), '.codymaster');
|
|
20
33
|
exports.DATA_FILE = path_1.default.join(exports.DATA_DIR, 'kanban.json');
|
|
@@ -94,6 +107,28 @@ function saveData(data) {
|
|
|
94
107
|
ensureDataDir();
|
|
95
108
|
fs_1.default.writeFileSync(exports.DATA_FILE, JSON.stringify(data, null, 2));
|
|
96
109
|
}
|
|
110
|
+
function loadDataSafe() {
|
|
111
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
112
|
+
const release = yield dataMutex.acquire();
|
|
113
|
+
try {
|
|
114
|
+
return loadData();
|
|
115
|
+
}
|
|
116
|
+
finally {
|
|
117
|
+
release();
|
|
118
|
+
}
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
function saveDataSafe(data) {
|
|
122
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
123
|
+
const release = yield dataMutex.acquire();
|
|
124
|
+
try {
|
|
125
|
+
fs_1.default.writeFileSync(exports.DATA_FILE, JSON.stringify(data, null, 2));
|
|
126
|
+
}
|
|
127
|
+
finally {
|
|
128
|
+
release();
|
|
129
|
+
}
|
|
130
|
+
});
|
|
131
|
+
}
|
|
97
132
|
// ─── Activity Logger ────────────────────────────────────────────────────────
|
|
98
133
|
function logActivity(data, type, message, projectId = '', agent = '', metadata = {}) {
|
|
99
134
|
data.activities.unshift({
|