nexo-brain 2.0.0 → 2.2.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/README.md +143 -44
- package/bin/nexo-brain.js +53 -26
- package/package.json +15 -3
- package/scripts/migrate-to-unified 2.sh +813 -0
- package/scripts/migrate-v1.5-to-v1.6 2.py +778 -0
- package/scripts/migrate-v1.7-to-v1.8 2.py +214 -0
- package/scripts/pre-commit-check 2.sh +55 -0
- package/src/__pycache__/auto_update.cpython-310.pyc +0 -0
- package/src/__pycache__/hnsw_index.cpython-310.pyc +0 -0
- package/src/__pycache__/kg_populate.cpython-310.pyc +0 -0
- package/src/__pycache__/knowledge_graph.cpython-310.pyc +0 -0
- package/src/__pycache__/plugin_loader.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_coordination.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_credentials.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_learnings.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_menu.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_reminders.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_reminders_crud.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_sessions.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_task_history.cpython-310.pyc +0 -0
- package/src/auto_close_sessions 2.py +159 -0
- package/src/auto_update 2.py +634 -0
- package/src/claim_graph 2.py +323 -0
- package/src/cognitive/__init__ 2.py +62 -0
- package/src/cognitive/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_decay.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_ingest.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_memory.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_search.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_trust.cpython-310.pyc +0 -0
- package/src/cognitive/_core 2.py +567 -0
- package/src/cognitive/_decay 2.py +382 -0
- package/src/cognitive/_ingest 2.py +892 -0
- package/src/cognitive/_memory 2.py +912 -0
- package/src/cognitive/_search 2.py +949 -0
- package/src/cognitive/_trust 2.py +464 -0
- package/src/cognitive/_trust.py +10 -36
- package/src/crons/manifest 2.json +106 -0
- package/src/crons/manifest.json +106 -0
- package/src/crons/sync 2.py +217 -0
- package/src/crons/sync.py +217 -0
- package/src/dashboard/__init__ 2.py +0 -0
- package/src/dashboard/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/dashboard/__pycache__/app.cpython-310.pyc +0 -0
- package/src/dashboard/app 2.py +789 -0
- package/src/dashboard/app.py +16 -2
- package/src/dashboard/templates/dashboard.html +3 -2
- package/src/db/__init__ 2.py +89 -0
- package/src/db/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/db/__pycache__/__init__.cpython-312.pyc +0 -0
- package/src/db/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_core.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_core.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_core.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_credentials.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_credentials.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_credentials.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_entities.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_entities.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_entities.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_episodic.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_episodic.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_episodic.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_evolution.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_evolution.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_evolution.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_fts.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_fts.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_fts.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_learnings.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_learnings.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_learnings.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_reminders.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_reminders.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_reminders.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_schema.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_schema.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_schema.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_sessions.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_sessions.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_sessions.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_tasks.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_tasks.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_tasks.cpython-314.pyc +0 -0
- package/src/db/_core 2.py +417 -0
- package/src/db/_credentials 2.py +124 -0
- package/src/db/_entities 2.py +178 -0
- package/src/db/_episodic 2.py +738 -0
- package/src/db/_episodic.py +1 -1
- package/src/db/_evolution 2.py +54 -0
- package/src/db/_fts 2.py +406 -0
- package/src/db/_learnings 2.py +168 -0
- package/src/db/_reminders 2.py +338 -0
- package/src/db/_reminders.py +9 -5
- package/src/db/_schema 2.py +364 -0
- package/src/db/_sessions 2.py +300 -0
- package/src/db/_tasks 2.py +91 -0
- package/src/evolution_cycle 2.py +266 -0
- package/src/hnsw_index 2.py +254 -0
- package/src/hooks/auto_capture 2.py +208 -0
- package/src/hooks/caffeinate-guard 2.sh +8 -0
- package/src/hooks/capture-session 2.sh +21 -0
- package/src/hooks/capture-session.sh +2 -0
- package/src/hooks/capture-tool-logs 2.sh +127 -0
- package/src/hooks/capture-tool-logs.sh +3 -2
- package/src/hooks/daily-briefing-check 2.sh +33 -0
- package/src/hooks/inbox-hook 2.sh +76 -0
- package/src/hooks/inbox-hook.sh +3 -2
- package/src/hooks/post-compact 2.sh +148 -0
- package/src/hooks/post-compact.sh +1 -1
- package/src/hooks/pre-compact 2.sh +151 -0
- package/src/hooks/pre-compact.sh +1 -1
- package/src/hooks/session-start 2.sh +268 -0
- package/src/hooks/session-start.sh +6 -3
- package/src/hooks/session-stop 2.sh +140 -0
- package/src/hooks/session-stop.sh +3 -2
- package/src/kg_populate 2.py +290 -0
- package/src/knowledge_graph 2.py +257 -0
- package/src/maintenance 2.py +59 -0
- package/src/migrate_embeddings 2.py +122 -0
- package/src/plugin_loader 2.py +202 -0
- package/src/plugins/__init__ 2.py +0 -0
- package/src/plugins/__pycache__/__init__ 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/adaptive_mode 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/adaptive_mode.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/agents 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/agents.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/artifact_registry 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/artifact_registry.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/backup 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/backup.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cognitive_memory 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cognitive_memory.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/core_rules 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/core_rules.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cortex 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cortex.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/entities 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/entities.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/episodic_memory 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/episodic_memory.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/evolution 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/evolution.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/guard 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/guard.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/knowledge_graph_tools 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/knowledge_graph_tools.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/preferences 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/preferences.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/update 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/update.cpython-310.pyc +0 -0
- package/src/plugins/adaptive_mode 2.py +805 -0
- package/src/plugins/agents 2.py +52 -0
- package/src/plugins/artifact_registry 2.py +450 -0
- package/src/plugins/backup 2.py +104 -0
- package/src/plugins/cognitive_memory 2.py +564 -0
- package/src/plugins/core_rules 2.py +252 -0
- package/src/plugins/core_rules.py +34 -17
- package/src/plugins/cortex 2.py +299 -0
- package/src/plugins/entities 2.py +67 -0
- package/src/plugins/episodic_memory 2.py +533 -0
- package/src/plugins/evolution 2.py +115 -0
- package/src/plugins/guard 2.py +746 -0
- package/src/plugins/knowledge_graph_tools 2.py +105 -0
- package/src/plugins/preferences 2.py +47 -0
- package/src/plugins/update 2.py +256 -0
- package/src/plugins/update.py +18 -0
- package/src/requirements 2.txt +12 -0
- package/src/rules/__init__ 2.py +0 -0
- package/src/rules/core-rules 2.json +331 -0
- package/src/rules/migrate 2.py +207 -0
- package/src/scripts/check-context 2.py +264 -0
- package/src/scripts/check-context.py +4 -7
- package/src/scripts/deep-sleep/apply_findings.py +570 -167
- package/src/scripts/deep-sleep/collect.py +480 -0
- package/src/scripts/deep-sleep/extract-prompt.md +233 -0
- package/src/scripts/deep-sleep/extract.py +249 -0
- package/src/scripts/deep-sleep/synthesize-prompt.md +197 -0
- package/src/scripts/deep-sleep/synthesize.py +191 -0
- package/src/scripts/nexo-auto-update 2.py +6 -0
- package/src/scripts/nexo-backup 2.sh +25 -0
- package/src/scripts/nexo-brain-activation 2.sh +140 -0
- package/src/scripts/nexo-catchup 2.py +242 -0
- package/src/scripts/nexo-catchup.py +5 -8
- package/src/scripts/nexo-cognitive-decay 2.py +182 -0
- package/src/scripts/nexo-daily-self-audit 2.py +552 -0
- package/src/scripts/nexo-daily-self-audit.py +28 -19
- package/src/scripts/nexo-deep-sleep 2.sh +97 -0
- package/src/scripts/nexo-deep-sleep.sh +31 -16
- package/src/scripts/nexo-evolution-run 2.py +597 -0
- package/src/scripts/nexo-evolution-run.py +5 -20
- package/src/scripts/nexo-followup-hygiene 2.py +112 -0
- package/src/scripts/nexo-followup-hygiene.py +4 -2
- package/src/scripts/nexo-github-monitor 2.py +256 -0
- package/src/scripts/nexo-github-monitor.py +6 -9
- package/src/scripts/nexo-immune 2.py +927 -0
- package/src/scripts/nexo-immune.py +4 -17
- package/src/scripts/nexo-inbox-hook 2.sh +74 -0
- package/src/scripts/nexo-install 2.py +6 -0
- package/src/scripts/nexo-learning-housekeep 2.py +245 -0
- package/src/scripts/nexo-learning-validator 2.py +207 -0
- package/src/scripts/nexo-learning-validator.py +0 -29
- package/src/scripts/nexo-migrate 2.py +232 -0
- package/src/scripts/nexo-postmortem-consolidator 2.py +421 -0
- package/src/scripts/nexo-postmortem-consolidator.py +9 -20
- package/src/scripts/nexo-pre-commit 2.py +120 -0
- package/src/scripts/nexo-prevent-sleep 2.sh +29 -0
- package/src/scripts/nexo-proactive-dashboard 2.py +345 -0
- package/src/scripts/nexo-proactive-dashboard.py +1 -0
- package/src/scripts/nexo-reflection 2.py +253 -0
- package/src/scripts/nexo-runtime-preflight 2.py +274 -0
- package/src/scripts/nexo-send-email 2.py +25 -0
- package/src/scripts/nexo-send-reply 2.py +178 -0
- package/src/scripts/nexo-sleep 2.py +592 -0
- package/src/scripts/nexo-sleep.py +8 -18
- package/src/scripts/nexo-snapshot-restore 2.sh +35 -0
- package/src/scripts/nexo-synthesis 2.py +253 -0
- package/src/scripts/nexo-synthesis.py +8 -19
- package/src/scripts/nexo-tcc-approve 2.sh +79 -0
- package/src/scripts/nexo-update 2.sh +161 -0
- package/src/scripts/nexo-watchdog 2.sh +878 -0
- package/src/scripts/nexo-watchdog-smoke 2.py +119 -0
- package/src/server 2.py +733 -0
- package/src/server.py +6 -1
- package/src/storage_router 2.py +32 -0
- package/src/tools_coordination 2.py +102 -0
- package/src/tools_credentials 2.py +68 -0
- package/src/tools_learnings 2.py +220 -0
- package/src/tools_menu 2.py +227 -0
- package/src/tools_menu.py +1 -1
- package/src/tools_reminders 2.py +86 -0
- package/src/tools_reminders_crud 2.py +159 -0
- package/src/tools_reminders_crud.py +7 -0
- package/src/tools_sessions 2.py +476 -0
- package/src/tools_sessions.py +67 -0
- package/src/tools_task_history 2.py +57 -0
- package/templates/CLAUDE.md 2.template +63 -0
- package/templates/openclaw 2.json +13 -0
- package/tests/__init__ 2.py +0 -0
- package/tests/conftest 2.py +71 -0
- package/tests/test_cognitive 2.py +205 -0
- package/tests/test_knowledge_graph 2.py +140 -0
- package/tests/test_migrations 2.py +137 -0
- package/src/__pycache__/auto_close_sessions.cpython-310.pyc +0 -0
- package/src/__pycache__/auto_close_sessions.cpython-314.pyc +0 -0
- package/src/__pycache__/auto_update.cpython-314.pyc +0 -0
- package/src/__pycache__/claim_graph.cpython-310.pyc +0 -0
- package/src/__pycache__/claim_graph.cpython-314.pyc +0 -0
- package/src/__pycache__/evolution_cycle.cpython-310.pyc +0 -0
- package/src/__pycache__/evolution_cycle.cpython-314.pyc +0 -0
- package/src/__pycache__/hnsw_index.cpython-314.pyc +0 -0
- package/src/__pycache__/kg_populate.cpython-314.pyc +0 -0
- package/src/__pycache__/knowledge_graph.cpython-314.pyc +0 -0
- package/src/__pycache__/maintenance.cpython-310.pyc +0 -0
- package/src/__pycache__/maintenance.cpython-314.pyc +0 -0
- package/src/__pycache__/migrate_embeddings.cpython-310.pyc +0 -0
- package/src/__pycache__/migrate_embeddings.cpython-314.pyc +0 -0
- package/src/__pycache__/plugin_loader.cpython-314.pyc +0 -0
- package/src/__pycache__/server.cpython-310.pyc +0 -0
- package/src/__pycache__/server.cpython-314.pyc +0 -0
- package/src/__pycache__/storage_router.cpython-310.pyc +0 -0
- package/src/__pycache__/storage_router.cpython-314.pyc +0 -0
- package/src/__pycache__/tools_coordination.cpython-314.pyc +0 -0
- package/src/__pycache__/tools_credentials.cpython-314.pyc +0 -0
- package/src/__pycache__/tools_learnings.cpython-314.pyc +0 -0
- package/src/__pycache__/tools_menu.cpython-314.pyc +0 -0
- package/src/__pycache__/tools_reminders.cpython-314.pyc +0 -0
- package/src/__pycache__/tools_reminders_crud.cpython-314.pyc +0 -0
- package/src/__pycache__/tools_sessions.cpython-314.pyc +0 -0
- package/src/__pycache__/tools_task_history.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_decay.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_ingest.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_memory.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_search.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_trust.cpython-314.pyc +0 -0
- package/src/dashboard/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/dashboard/__pycache__/app.cpython-314.pyc +0 -0
- package/src/hooks/__pycache__/auto_capture.cpython-310.pyc +0 -0
- package/src/hooks/__pycache__/auto_capture.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/adaptive_mode.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/agents.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/artifact_registry.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/backup.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/cognitive_memory.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/core_rules.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/cortex.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/entities.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/episodic_memory.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/evolution.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/guard.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/knowledge_graph_tools.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/preferences.cpython-314.pyc +0 -0
- package/src/rules/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/rules/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/rules/__pycache__/migrate.cpython-310.pyc +0 -0
- package/src/rules/__pycache__/migrate.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/check-context.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/check-context.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-auto-update.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-auto-update.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-catchup.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-catchup.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-cognitive-decay.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-cognitive-decay.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-daily-self-audit.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-daily-self-audit.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-evolution-run.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-evolution-run.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-followup-hygiene.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-followup-hygiene.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-github-monitor.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-github-monitor.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-immune.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-immune.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-install.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-install.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-learning-housekeep.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-learning-housekeep.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-learning-validator.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-learning-validator.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-migrate.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-migrate.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-postmortem-consolidator.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-postmortem-consolidator.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-pre-commit.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-pre-commit.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-proactive-dashboard.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-proactive-dashboard.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-reflection.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-reflection.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-runtime-preflight.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-runtime-preflight.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-send-email.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-send-email.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-send-reply.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-send-reply.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-sleep.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-sleep.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-synthesis.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-synthesis.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-watchdog-smoke.cpython-310.pyc +0 -0
- package/src/scripts/__pycache__/nexo-watchdog-smoke.cpython-314.pyc +0 -0
- package/src/scripts/deep-sleep/__pycache__/analyze_session.cpython-310.pyc +0 -0
- package/src/scripts/deep-sleep/__pycache__/analyze_session.cpython-314.pyc +0 -0
- package/src/scripts/deep-sleep/__pycache__/apply_findings.cpython-310.pyc +0 -0
- package/src/scripts/deep-sleep/__pycache__/apply_findings.cpython-314.pyc +0 -0
- package/src/scripts/deep-sleep/__pycache__/collect_transcripts.cpython-310.pyc +0 -0
- package/src/scripts/deep-sleep/__pycache__/collect_transcripts.cpython-314.pyc +0 -0
- package/src/scripts/deep-sleep/analyze_session.py +0 -217
- package/src/scripts/deep-sleep/collect_transcripts.py +0 -145
- package/src/scripts/deep-sleep/prompt.md +0 -109
- package/tests/__pycache__/__init__.cpython-310.pyc +0 -0
- package/tests/__pycache__/__init__.cpython-314.pyc +0 -0
- package/tests/__pycache__/conftest.cpython-310-pytest-9.0.2.pyc +0 -0
- package/tests/__pycache__/conftest.cpython-310.pyc +0 -0
- package/tests/__pycache__/conftest.cpython-314-pytest-9.0.2.pyc +0 -0
- package/tests/__pycache__/test_cognitive.cpython-310-pytest-9.0.2.pyc +0 -0
- package/tests/__pycache__/test_cognitive.cpython-310.pyc +0 -0
- package/tests/__pycache__/test_cognitive.cpython-314-pytest-9.0.2.pyc +0 -0
- package/tests/__pycache__/test_knowledge_graph.cpython-310-pytest-9.0.2.pyc +0 -0
- package/tests/__pycache__/test_knowledge_graph.cpython-310.pyc +0 -0
- package/tests/__pycache__/test_knowledge_graph.cpython-314-pytest-9.0.2.pyc +0 -0
- package/tests/__pycache__/test_migrations.cpython-310-pytest-9.0.2.pyc +0 -0
- package/tests/__pycache__/test_migrations.cpython-310.pyc +0 -0
- package/tests/__pycache__/test_migrations.cpython-314-pytest-9.0.2.pyc +0 -0
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
{
|
|
2
|
+
"$schema": "NEXO cron manifest — synced by nexo_update to LaunchAgents (macOS) or systemd timers (Linux)",
|
|
3
|
+
"version": 2,
|
|
4
|
+
"crons": [
|
|
5
|
+
{
|
|
6
|
+
"id": "deep-sleep",
|
|
7
|
+
"script": "scripts/nexo-deep-sleep.sh",
|
|
8
|
+
"type": "shell",
|
|
9
|
+
"schedule": {"hour": 4, "minute": 30},
|
|
10
|
+
"description": "Overnight session analysis — 4 phases: collect, extract, synthesize, apply",
|
|
11
|
+
"core": true
|
|
12
|
+
},
|
|
13
|
+
{
|
|
14
|
+
"id": "sleep",
|
|
15
|
+
"script": "scripts/nexo-sleep.py",
|
|
16
|
+
"schedule": {"hour": 4, "minute": 0},
|
|
17
|
+
"description": "Nightly memory consolidation and dream cycle",
|
|
18
|
+
"core": true
|
|
19
|
+
},
|
|
20
|
+
{
|
|
21
|
+
"id": "cognitive-decay",
|
|
22
|
+
"script": "scripts/nexo-cognitive-decay.py",
|
|
23
|
+
"schedule": {"hour": 3, "minute": 0},
|
|
24
|
+
"description": "Memory decay — reduce strength of unaccessed memories",
|
|
25
|
+
"core": true
|
|
26
|
+
},
|
|
27
|
+
{
|
|
28
|
+
"id": "learning-housekeep",
|
|
29
|
+
"script": "scripts/nexo-learning-housekeep.py",
|
|
30
|
+
"schedule": {"hour": 3, "minute": 15},
|
|
31
|
+
"description": "Archive stale learnings, deduplicate, validate",
|
|
32
|
+
"core": true
|
|
33
|
+
},
|
|
34
|
+
{
|
|
35
|
+
"id": "immune",
|
|
36
|
+
"script": "scripts/nexo-immune.py",
|
|
37
|
+
"interval_seconds": 1800,
|
|
38
|
+
"description": "Health monitor — checks MCP, DB, services, auto-repairs",
|
|
39
|
+
"core": true
|
|
40
|
+
},
|
|
41
|
+
{
|
|
42
|
+
"id": "watchdog",
|
|
43
|
+
"script": "scripts/nexo-watchdog.sh",
|
|
44
|
+
"type": "shell",
|
|
45
|
+
"interval_seconds": 1800,
|
|
46
|
+
"description": "System health checks — snapshots, logs, alerts",
|
|
47
|
+
"core": true
|
|
48
|
+
},
|
|
49
|
+
{
|
|
50
|
+
"id": "self-audit",
|
|
51
|
+
"script": "scripts/nexo-daily-self-audit.py",
|
|
52
|
+
"schedule": {"hour": 7, "minute": 0},
|
|
53
|
+
"description": "Daily self-audit — validates learnings, protocols, drift",
|
|
54
|
+
"core": true
|
|
55
|
+
},
|
|
56
|
+
{
|
|
57
|
+
"id": "postmortem",
|
|
58
|
+
"script": "scripts/nexo-postmortem-consolidator.py",
|
|
59
|
+
"schedule": {"hour": 23, "minute": 30},
|
|
60
|
+
"description": "Consolidate session post-mortems into patterns",
|
|
61
|
+
"core": true
|
|
62
|
+
},
|
|
63
|
+
{
|
|
64
|
+
"id": "evolution",
|
|
65
|
+
"script": "scripts/nexo-evolution-run.py",
|
|
66
|
+
"schedule": {"hour": 5, "minute": 0, "weekday": 0},
|
|
67
|
+
"description": "Weekly self-improvement cycle — propose and evaluate changes",
|
|
68
|
+
"core": true
|
|
69
|
+
},
|
|
70
|
+
{
|
|
71
|
+
"id": "followup-hygiene",
|
|
72
|
+
"script": "scripts/nexo-followup-hygiene.py",
|
|
73
|
+
"schedule": {"hour": 5, "minute": 0},
|
|
74
|
+
"description": "Clean stale followups, archive completed, validate dates",
|
|
75
|
+
"core": true
|
|
76
|
+
},
|
|
77
|
+
{
|
|
78
|
+
"id": "synthesis",
|
|
79
|
+
"script": "scripts/nexo-synthesis.py",
|
|
80
|
+
"interval_seconds": 7200,
|
|
81
|
+
"description": "Periodic synthesis — cross-reference learnings, decisions, changes",
|
|
82
|
+
"core": true
|
|
83
|
+
},
|
|
84
|
+
{
|
|
85
|
+
"id": "auto-close-sessions",
|
|
86
|
+
"script": "scripts/nexo-auto-close-sessions.py",
|
|
87
|
+
"interval_seconds": 300,
|
|
88
|
+
"description": "Close stale sessions that lost their parent process",
|
|
89
|
+
"core": true
|
|
90
|
+
},
|
|
91
|
+
{
|
|
92
|
+
"id": "github-monitor",
|
|
93
|
+
"script": "scripts/nexo-github-monitor.py",
|
|
94
|
+
"schedule": {"hour": 8, "minute": 0},
|
|
95
|
+
"description": "Monitor GitHub repo — issues, PRs, stars, auto-respond",
|
|
96
|
+
"core": true
|
|
97
|
+
},
|
|
98
|
+
{
|
|
99
|
+
"id": "catchup",
|
|
100
|
+
"script": "scripts/nexo-catchup.py",
|
|
101
|
+
"schedule": {"hour": 8, "minute": 30},
|
|
102
|
+
"description": "Morning catchup briefing for the user",
|
|
103
|
+
"core": true
|
|
104
|
+
}
|
|
105
|
+
]
|
|
106
|
+
}
|
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
NEXO Cron Sync — Synchronize crons/manifest.json with system LaunchAgents (macOS).
|
|
4
|
+
|
|
5
|
+
Called by nexo_update after pulling new code. Ensures:
|
|
6
|
+
- New crons in manifest → installed
|
|
7
|
+
- Removed crons from manifest → unloaded + deleted
|
|
8
|
+
- Changed schedule/interval → plist updated + reloaded
|
|
9
|
+
- Personal (non-core) crons → left untouched
|
|
10
|
+
|
|
11
|
+
Usage:
|
|
12
|
+
python3 crons/sync.py [--dry-run]
|
|
13
|
+
|
|
14
|
+
Environment:
|
|
15
|
+
NEXO_HOME — root of NEXO installation
|
|
16
|
+
NEXO_CODE — path to NEXO source (defaults to script parent's parent)
|
|
17
|
+
"""
|
|
18
|
+
|
|
19
|
+
import json
|
|
20
|
+
import os
|
|
21
|
+
import platform
|
|
22
|
+
import plistlib
|
|
23
|
+
import subprocess
|
|
24
|
+
import sys
|
|
25
|
+
from pathlib import Path
|
|
26
|
+
|
|
27
|
+
NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
|
|
28
|
+
NEXO_CODE = Path(os.environ.get("NEXO_CODE", str(Path(__file__).resolve().parent.parent)))
|
|
29
|
+
MANIFEST = Path(__file__).resolve().parent / "manifest.json"
|
|
30
|
+
LAUNCH_AGENTS_DIR = Path.home() / "Library" / "LaunchAgents"
|
|
31
|
+
LABEL_PREFIX = "com.nexo."
|
|
32
|
+
LOG_DIR = NEXO_HOME / "logs"
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
def log(msg: str):
|
|
36
|
+
print(f"[cron-sync] {msg}", flush=True)
|
|
37
|
+
|
|
38
|
+
|
|
39
|
+
def load_manifest() -> list[dict]:
|
|
40
|
+
with open(MANIFEST) as f:
|
|
41
|
+
data = json.load(f)
|
|
42
|
+
return data.get("crons", [])
|
|
43
|
+
|
|
44
|
+
|
|
45
|
+
def build_plist(cron: dict) -> dict:
|
|
46
|
+
"""Build a macOS LaunchAgent plist dict from a manifest entry."""
|
|
47
|
+
cron_id = cron["id"]
|
|
48
|
+
label = f"{LABEL_PREFIX}{cron_id}"
|
|
49
|
+
script_path = str(NEXO_CODE / cron["script"])
|
|
50
|
+
script_type = cron.get("type", "python")
|
|
51
|
+
|
|
52
|
+
if script_type == "shell":
|
|
53
|
+
program_args = ["/bin/bash", script_path]
|
|
54
|
+
else:
|
|
55
|
+
# Find python3
|
|
56
|
+
python_candidates = [
|
|
57
|
+
"/opt/homebrew/bin/python3",
|
|
58
|
+
"/usr/local/bin/python3",
|
|
59
|
+
"/Library/Frameworks/Python.framework/Versions/3.12/bin/python3",
|
|
60
|
+
"/usr/bin/python3",
|
|
61
|
+
]
|
|
62
|
+
python_bin = "python3"
|
|
63
|
+
for p in python_candidates:
|
|
64
|
+
if Path(p).exists():
|
|
65
|
+
python_bin = p
|
|
66
|
+
break
|
|
67
|
+
program_args = [python_bin, script_path]
|
|
68
|
+
|
|
69
|
+
plist = {
|
|
70
|
+
"Label": label,
|
|
71
|
+
"ProgramArguments": program_args,
|
|
72
|
+
"StandardOutPath": str(LOG_DIR / f"{cron_id}-stdout.log"),
|
|
73
|
+
"StandardErrorPath": str(LOG_DIR / f"{cron_id}-stderr.log"),
|
|
74
|
+
"EnvironmentVariables": {
|
|
75
|
+
"PATH": "/usr/local/bin:/usr/bin:/bin:/opt/homebrew/bin:"
|
|
76
|
+
+ str(Path.home() / ".local" / "bin") + ":"
|
|
77
|
+
+ str(Path.home() / ".nvm/versions/node/v22.14.0/bin") + ":"
|
|
78
|
+
+ "/Library/Frameworks/Python.framework/Versions/3.12/bin",
|
|
79
|
+
"HOME": str(Path.home()),
|
|
80
|
+
"NEXO_HOME": str(NEXO_HOME),
|
|
81
|
+
"NEXO_CODE": str(NEXO_CODE),
|
|
82
|
+
"PYTHONUNBUFFERED": "1",
|
|
83
|
+
},
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
# Schedule
|
|
87
|
+
if "interval_seconds" in cron:
|
|
88
|
+
plist["StartInterval"] = cron["interval_seconds"]
|
|
89
|
+
elif "schedule" in cron:
|
|
90
|
+
cal = {}
|
|
91
|
+
s = cron["schedule"]
|
|
92
|
+
if "hour" in s:
|
|
93
|
+
cal["Hour"] = s["hour"]
|
|
94
|
+
if "minute" in s:
|
|
95
|
+
cal["Minute"] = s["minute"]
|
|
96
|
+
if "weekday" in s:
|
|
97
|
+
cal["Weekday"] = s["weekday"]
|
|
98
|
+
plist["StartCalendarInterval"] = cal
|
|
99
|
+
|
|
100
|
+
return plist
|
|
101
|
+
|
|
102
|
+
|
|
103
|
+
def get_installed_nexo_crons() -> dict[str, Path]:
|
|
104
|
+
"""Return dict of cron_id → plist_path for installed NEXO crons."""
|
|
105
|
+
installed = {}
|
|
106
|
+
if not LAUNCH_AGENTS_DIR.exists():
|
|
107
|
+
return installed
|
|
108
|
+
for f in LAUNCH_AGENTS_DIR.glob(f"{LABEL_PREFIX}*.plist"):
|
|
109
|
+
cron_id = f.stem.replace(LABEL_PREFIX, "")
|
|
110
|
+
installed[cron_id] = f
|
|
111
|
+
return installed
|
|
112
|
+
|
|
113
|
+
|
|
114
|
+
def plist_needs_update(existing_path: Path, new_plist: dict) -> bool:
|
|
115
|
+
"""Check if the installed plist differs from what we'd generate."""
|
|
116
|
+
try:
|
|
117
|
+
with open(existing_path, "rb") as f:
|
|
118
|
+
existing = plistlib.load(f)
|
|
119
|
+
except Exception:
|
|
120
|
+
return True
|
|
121
|
+
|
|
122
|
+
# Compare key fields
|
|
123
|
+
if existing.get("ProgramArguments") != new_plist.get("ProgramArguments"):
|
|
124
|
+
return True
|
|
125
|
+
if existing.get("StartInterval") != new_plist.get("StartInterval"):
|
|
126
|
+
return True
|
|
127
|
+
if existing.get("StartCalendarInterval") != new_plist.get("StartCalendarInterval"):
|
|
128
|
+
return True
|
|
129
|
+
return False
|
|
130
|
+
|
|
131
|
+
|
|
132
|
+
def install_plist(label: str, plist: dict, plist_path: Path, dry_run: bool):
|
|
133
|
+
"""Write plist and load it."""
|
|
134
|
+
if dry_run:
|
|
135
|
+
log(f" DRY-RUN: would install {plist_path.name}")
|
|
136
|
+
return
|
|
137
|
+
|
|
138
|
+
# Unload if already loaded
|
|
139
|
+
subprocess.run(["launchctl", "unload", str(plist_path)], capture_output=True)
|
|
140
|
+
|
|
141
|
+
with open(plist_path, "wb") as f:
|
|
142
|
+
plistlib.dump(plist, f)
|
|
143
|
+
|
|
144
|
+
subprocess.run(["launchctl", "load", str(plist_path)], capture_output=True)
|
|
145
|
+
log(f" Installed + loaded: {plist_path.name}")
|
|
146
|
+
|
|
147
|
+
|
|
148
|
+
def unload_plist(plist_path: Path, dry_run: bool):
|
|
149
|
+
"""Unload and remove a plist."""
|
|
150
|
+
if dry_run:
|
|
151
|
+
log(f" DRY-RUN: would remove {plist_path.name}")
|
|
152
|
+
return
|
|
153
|
+
|
|
154
|
+
subprocess.run(["launchctl", "unload", str(plist_path)], capture_output=True)
|
|
155
|
+
plist_path.unlink(missing_ok=True)
|
|
156
|
+
log(f" Removed: {plist_path.name}")
|
|
157
|
+
|
|
158
|
+
|
|
159
|
+
def sync(dry_run: bool = False):
|
|
160
|
+
if platform.system() != "Darwin":
|
|
161
|
+
log("Not macOS — cron sync only supports LaunchAgents. Skipping.")
|
|
162
|
+
return
|
|
163
|
+
|
|
164
|
+
LOG_DIR.mkdir(parents=True, exist_ok=True)
|
|
165
|
+
LAUNCH_AGENTS_DIR.mkdir(parents=True, exist_ok=True)
|
|
166
|
+
|
|
167
|
+
manifest_crons = load_manifest()
|
|
168
|
+
manifest_ids = {c["id"] for c in manifest_crons}
|
|
169
|
+
installed = get_installed_nexo_crons()
|
|
170
|
+
|
|
171
|
+
log(f"Manifest: {len(manifest_crons)} core crons")
|
|
172
|
+
log(f"Installed: {len(installed)} NEXO crons")
|
|
173
|
+
|
|
174
|
+
# 1. Install or update crons from manifest
|
|
175
|
+
for cron in manifest_crons:
|
|
176
|
+
cron_id = cron["id"]
|
|
177
|
+
label = f"{LABEL_PREFIX}{cron_id}"
|
|
178
|
+
plist_path = LAUNCH_AGENTS_DIR / f"{label}.plist"
|
|
179
|
+
new_plist = build_plist(cron)
|
|
180
|
+
|
|
181
|
+
if cron_id not in installed:
|
|
182
|
+
log(f" NEW: {cron_id}")
|
|
183
|
+
install_plist(label, new_plist, plist_path, dry_run)
|
|
184
|
+
elif plist_needs_update(installed[cron_id], new_plist):
|
|
185
|
+
log(f" UPDATE: {cron_id}")
|
|
186
|
+
install_plist(label, new_plist, plist_path, dry_run)
|
|
187
|
+
else:
|
|
188
|
+
log(f" OK: {cron_id}")
|
|
189
|
+
|
|
190
|
+
# 2. Remove crons that are in installed but NOT in manifest and ARE core
|
|
191
|
+
# (personal crons like shopify-backup, email-monitor are left alone)
|
|
192
|
+
for cron_id, plist_path in installed.items():
|
|
193
|
+
if cron_id not in manifest_ids:
|
|
194
|
+
# Check if this was previously a core cron by reading the plist
|
|
195
|
+
# If it points to NEXO_CODE scripts → it's core, safe to remove
|
|
196
|
+
try:
|
|
197
|
+
with open(plist_path, "rb") as f:
|
|
198
|
+
existing = plistlib.load(f)
|
|
199
|
+
args = existing.get("ProgramArguments", [])
|
|
200
|
+
is_core = any(str(NEXO_CODE) in str(a) for a in args)
|
|
201
|
+
except Exception:
|
|
202
|
+
is_core = False
|
|
203
|
+
|
|
204
|
+
if is_core:
|
|
205
|
+
log(f" REMOVE (no longer in manifest): {cron_id}")
|
|
206
|
+
unload_plist(plist_path, dry_run)
|
|
207
|
+
else:
|
|
208
|
+
log(f" SKIP (personal): {cron_id}")
|
|
209
|
+
|
|
210
|
+
log("Sync complete.")
|
|
211
|
+
|
|
212
|
+
|
|
213
|
+
if __name__ == "__main__":
|
|
214
|
+
dry_run = "--dry-run" in sys.argv
|
|
215
|
+
if dry_run:
|
|
216
|
+
log("DRY RUN MODE — no changes will be made")
|
|
217
|
+
sync(dry_run=dry_run)
|
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
NEXO Cron Sync — Synchronize crons/manifest.json with system LaunchAgents (macOS).
|
|
4
|
+
|
|
5
|
+
Called by nexo_update after pulling new code. Ensures:
|
|
6
|
+
- New crons in manifest → installed
|
|
7
|
+
- Removed crons from manifest → unloaded + deleted
|
|
8
|
+
- Changed schedule/interval → plist updated + reloaded
|
|
9
|
+
- Personal (non-core) crons → left untouched
|
|
10
|
+
|
|
11
|
+
Usage:
|
|
12
|
+
python3 crons/sync.py [--dry-run]
|
|
13
|
+
|
|
14
|
+
Environment:
|
|
15
|
+
NEXO_HOME — root of NEXO installation
|
|
16
|
+
NEXO_CODE — path to NEXO source (defaults to script parent's parent)
|
|
17
|
+
"""
|
|
18
|
+
|
|
19
|
+
import json
|
|
20
|
+
import os
|
|
21
|
+
import platform
|
|
22
|
+
import plistlib
|
|
23
|
+
import subprocess
|
|
24
|
+
import sys
|
|
25
|
+
from pathlib import Path
|
|
26
|
+
|
|
27
|
+
NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
|
|
28
|
+
NEXO_CODE = Path(os.environ.get("NEXO_CODE", str(Path(__file__).resolve().parent.parent)))
|
|
29
|
+
MANIFEST = Path(__file__).resolve().parent / "manifest.json"
|
|
30
|
+
LAUNCH_AGENTS_DIR = Path.home() / "Library" / "LaunchAgents"
|
|
31
|
+
LABEL_PREFIX = "com.nexo."
|
|
32
|
+
LOG_DIR = NEXO_HOME / "logs"
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
def log(msg: str):
|
|
36
|
+
print(f"[cron-sync] {msg}", flush=True)
|
|
37
|
+
|
|
38
|
+
|
|
39
|
+
def load_manifest() -> list[dict]:
|
|
40
|
+
with open(MANIFEST) as f:
|
|
41
|
+
data = json.load(f)
|
|
42
|
+
return data.get("crons", [])
|
|
43
|
+
|
|
44
|
+
|
|
45
|
+
def build_plist(cron: dict) -> dict:
|
|
46
|
+
"""Build a macOS LaunchAgent plist dict from a manifest entry."""
|
|
47
|
+
cron_id = cron["id"]
|
|
48
|
+
label = f"{LABEL_PREFIX}{cron_id}"
|
|
49
|
+
script_path = str(NEXO_CODE / cron["script"])
|
|
50
|
+
script_type = cron.get("type", "python")
|
|
51
|
+
|
|
52
|
+
if script_type == "shell":
|
|
53
|
+
program_args = ["/bin/bash", script_path]
|
|
54
|
+
else:
|
|
55
|
+
# Find python3
|
|
56
|
+
python_candidates = [
|
|
57
|
+
"/opt/homebrew/bin/python3",
|
|
58
|
+
"/usr/local/bin/python3",
|
|
59
|
+
"/Library/Frameworks/Python.framework/Versions/3.12/bin/python3",
|
|
60
|
+
"/usr/bin/python3",
|
|
61
|
+
]
|
|
62
|
+
python_bin = "python3"
|
|
63
|
+
for p in python_candidates:
|
|
64
|
+
if Path(p).exists():
|
|
65
|
+
python_bin = p
|
|
66
|
+
break
|
|
67
|
+
program_args = [python_bin, script_path]
|
|
68
|
+
|
|
69
|
+
plist = {
|
|
70
|
+
"Label": label,
|
|
71
|
+
"ProgramArguments": program_args,
|
|
72
|
+
"StandardOutPath": str(LOG_DIR / f"{cron_id}-stdout.log"),
|
|
73
|
+
"StandardErrorPath": str(LOG_DIR / f"{cron_id}-stderr.log"),
|
|
74
|
+
"EnvironmentVariables": {
|
|
75
|
+
"PATH": "/usr/local/bin:/usr/bin:/bin:/opt/homebrew/bin:"
|
|
76
|
+
+ str(Path.home() / ".local" / "bin") + ":"
|
|
77
|
+
+ str(Path.home() / ".nvm/versions/node/v22.14.0/bin") + ":"
|
|
78
|
+
+ "/Library/Frameworks/Python.framework/Versions/3.12/bin",
|
|
79
|
+
"HOME": str(Path.home()),
|
|
80
|
+
"NEXO_HOME": str(NEXO_HOME),
|
|
81
|
+
"NEXO_CODE": str(NEXO_CODE),
|
|
82
|
+
"PYTHONUNBUFFERED": "1",
|
|
83
|
+
},
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
# Schedule
|
|
87
|
+
if "interval_seconds" in cron:
|
|
88
|
+
plist["StartInterval"] = cron["interval_seconds"]
|
|
89
|
+
elif "schedule" in cron:
|
|
90
|
+
cal = {}
|
|
91
|
+
s = cron["schedule"]
|
|
92
|
+
if "hour" in s:
|
|
93
|
+
cal["Hour"] = s["hour"]
|
|
94
|
+
if "minute" in s:
|
|
95
|
+
cal["Minute"] = s["minute"]
|
|
96
|
+
if "weekday" in s:
|
|
97
|
+
cal["Weekday"] = s["weekday"]
|
|
98
|
+
plist["StartCalendarInterval"] = cal
|
|
99
|
+
|
|
100
|
+
return plist
|
|
101
|
+
|
|
102
|
+
|
|
103
|
+
def get_installed_nexo_crons() -> dict[str, Path]:
|
|
104
|
+
"""Return dict of cron_id → plist_path for installed NEXO crons."""
|
|
105
|
+
installed = {}
|
|
106
|
+
if not LAUNCH_AGENTS_DIR.exists():
|
|
107
|
+
return installed
|
|
108
|
+
for f in LAUNCH_AGENTS_DIR.glob(f"{LABEL_PREFIX}*.plist"):
|
|
109
|
+
cron_id = f.stem.replace(LABEL_PREFIX, "")
|
|
110
|
+
installed[cron_id] = f
|
|
111
|
+
return installed
|
|
112
|
+
|
|
113
|
+
|
|
114
|
+
def plist_needs_update(existing_path: Path, new_plist: dict) -> bool:
|
|
115
|
+
"""Check if the installed plist differs from what we'd generate."""
|
|
116
|
+
try:
|
|
117
|
+
with open(existing_path, "rb") as f:
|
|
118
|
+
existing = plistlib.load(f)
|
|
119
|
+
except Exception:
|
|
120
|
+
return True
|
|
121
|
+
|
|
122
|
+
# Compare key fields
|
|
123
|
+
if existing.get("ProgramArguments") != new_plist.get("ProgramArguments"):
|
|
124
|
+
return True
|
|
125
|
+
if existing.get("StartInterval") != new_plist.get("StartInterval"):
|
|
126
|
+
return True
|
|
127
|
+
if existing.get("StartCalendarInterval") != new_plist.get("StartCalendarInterval"):
|
|
128
|
+
return True
|
|
129
|
+
return False
|
|
130
|
+
|
|
131
|
+
|
|
132
|
+
def install_plist(label: str, plist: dict, plist_path: Path, dry_run: bool):
|
|
133
|
+
"""Write plist and load it."""
|
|
134
|
+
if dry_run:
|
|
135
|
+
log(f" DRY-RUN: would install {plist_path.name}")
|
|
136
|
+
return
|
|
137
|
+
|
|
138
|
+
# Unload if already loaded
|
|
139
|
+
subprocess.run(["launchctl", "unload", str(plist_path)], capture_output=True)
|
|
140
|
+
|
|
141
|
+
with open(plist_path, "wb") as f:
|
|
142
|
+
plistlib.dump(plist, f)
|
|
143
|
+
|
|
144
|
+
subprocess.run(["launchctl", "load", str(plist_path)], capture_output=True)
|
|
145
|
+
log(f" Installed + loaded: {plist_path.name}")
|
|
146
|
+
|
|
147
|
+
|
|
148
|
+
def unload_plist(plist_path: Path, dry_run: bool):
|
|
149
|
+
"""Unload and remove a plist."""
|
|
150
|
+
if dry_run:
|
|
151
|
+
log(f" DRY-RUN: would remove {plist_path.name}")
|
|
152
|
+
return
|
|
153
|
+
|
|
154
|
+
subprocess.run(["launchctl", "unload", str(plist_path)], capture_output=True)
|
|
155
|
+
plist_path.unlink(missing_ok=True)
|
|
156
|
+
log(f" Removed: {plist_path.name}")
|
|
157
|
+
|
|
158
|
+
|
|
159
|
+
def sync(dry_run: bool = False):
|
|
160
|
+
if platform.system() != "Darwin":
|
|
161
|
+
log("Not macOS — cron sync only supports LaunchAgents. Skipping.")
|
|
162
|
+
return
|
|
163
|
+
|
|
164
|
+
LOG_DIR.mkdir(parents=True, exist_ok=True)
|
|
165
|
+
LAUNCH_AGENTS_DIR.mkdir(parents=True, exist_ok=True)
|
|
166
|
+
|
|
167
|
+
manifest_crons = load_manifest()
|
|
168
|
+
manifest_ids = {c["id"] for c in manifest_crons}
|
|
169
|
+
installed = get_installed_nexo_crons()
|
|
170
|
+
|
|
171
|
+
log(f"Manifest: {len(manifest_crons)} core crons")
|
|
172
|
+
log(f"Installed: {len(installed)} NEXO crons")
|
|
173
|
+
|
|
174
|
+
# 1. Install or update crons from manifest
|
|
175
|
+
for cron in manifest_crons:
|
|
176
|
+
cron_id = cron["id"]
|
|
177
|
+
label = f"{LABEL_PREFIX}{cron_id}"
|
|
178
|
+
plist_path = LAUNCH_AGENTS_DIR / f"{label}.plist"
|
|
179
|
+
new_plist = build_plist(cron)
|
|
180
|
+
|
|
181
|
+
if cron_id not in installed:
|
|
182
|
+
log(f" NEW: {cron_id}")
|
|
183
|
+
install_plist(label, new_plist, plist_path, dry_run)
|
|
184
|
+
elif plist_needs_update(installed[cron_id], new_plist):
|
|
185
|
+
log(f" UPDATE: {cron_id}")
|
|
186
|
+
install_plist(label, new_plist, plist_path, dry_run)
|
|
187
|
+
else:
|
|
188
|
+
log(f" OK: {cron_id}")
|
|
189
|
+
|
|
190
|
+
# 2. Remove crons that are in installed but NOT in manifest and ARE core
|
|
191
|
+
# (personal crons like shopify-backup, email-monitor are left alone)
|
|
192
|
+
for cron_id, plist_path in installed.items():
|
|
193
|
+
if cron_id not in manifest_ids:
|
|
194
|
+
# Check if this was previously a core cron by reading the plist
|
|
195
|
+
# If it points to NEXO_CODE scripts → it's core, safe to remove
|
|
196
|
+
try:
|
|
197
|
+
with open(plist_path, "rb") as f:
|
|
198
|
+
existing = plistlib.load(f)
|
|
199
|
+
args = existing.get("ProgramArguments", [])
|
|
200
|
+
is_core = any(str(NEXO_CODE) in str(a) for a in args)
|
|
201
|
+
except Exception:
|
|
202
|
+
is_core = False
|
|
203
|
+
|
|
204
|
+
if is_core:
|
|
205
|
+
log(f" REMOVE (no longer in manifest): {cron_id}")
|
|
206
|
+
unload_plist(plist_path, dry_run)
|
|
207
|
+
else:
|
|
208
|
+
log(f" SKIP (personal): {cron_id}")
|
|
209
|
+
|
|
210
|
+
log("Sync complete.")
|
|
211
|
+
|
|
212
|
+
|
|
213
|
+
if __name__ == "__main__":
|
|
214
|
+
dry_run = "--dry-run" in sys.argv
|
|
215
|
+
if dry_run:
|
|
216
|
+
log("DRY RUN MODE — no changes will be made")
|
|
217
|
+
sync(dry_run=dry_run)
|
|
File without changes
|
|
Binary file
|
|
Binary file
|