nexo-brain 2.1.0 → 2.3.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 +7 -7
- package/bin/nexo-brain.js +53 -26
- package/package.json +1 -1
- 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/migrate-v1.7-to-v1.8.py +2 -2
- package/scripts/nexo-preflight.sh +236 -0
- package/scripts/pre-commit-check 2.sh +55 -0
- package/src/__pycache__/auto_close_sessions.cpython-314.pyc +0 -0
- package/src/__pycache__/auto_update.cpython-310.pyc +0 -0
- package/src/__pycache__/hnsw_index.cpython-310.pyc +0 -0
- package/src/__pycache__/hnsw_index.cpython-314.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__/plugin_loader.cpython-314.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/auto_update.py +25 -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__/__init__.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_decay.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_decay.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_decay.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_ingest.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_ingest.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_ingest.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_memory.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_memory.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_memory.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_search.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_search.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_search.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_trust.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_trust.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_trust.cpython-314.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/__pycache__/sync.cpython-314.pyc +0 -0
- package/src/crons/manifest 2.json +106 -0
- package/src/crons/manifest.json +6 -13
- package/src/crons/sync 2.py +217 -0
- package/src/crons/sync.py +151 -6
- 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/db/__init__ 2.py +89 -0
- package/src/db/__init__.py +13 -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__/_cron_runs.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_cron_runs.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__/_skills.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_skills.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_skills.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/_cron_runs.py +74 -0
- package/src/db/_entities 2.py +178 -0
- package/src/db/_episodic 2.py +738 -0
- package/src/db/_episodic.py +40 -6
- 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/_schema 2.py +364 -0
- package/src/db/_schema.py +64 -0
- package/src/db/_sessions 2.py +300 -0
- package/src/db/_skills.py +514 -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 +14 -102
- 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__/__init__.cpython-314.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__/adaptive_mode.cpython-314.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__/schedule.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/schedule.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/skills.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/skills.cpython-314.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/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/episodic_memory.py +5 -3
- 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/schedule.py +212 -0
- package/src/plugins/skills.py +264 -0
- package/src/plugins/update 2.py +256 -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/__pycache__/nexo-auto-update.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-catchup.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-cognitive-decay.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-daily-self-audit.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-evolution-run.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-followup-hygiene.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-immune.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-install.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-learning-housekeep.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-learning-validator.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-migrate.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-postmortem-consolidator.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-pre-commit.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-proactive-dashboard.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-reflection.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-runtime-preflight.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-send-email.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-send-reply.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-sleep.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-synthesis.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-watchdog-smoke.cpython-314.pyc +0 -0
- package/src/scripts/check-context 2.py +264 -0
- package/src/scripts/deep-sleep/apply_findings.py +168 -8
- package/src/scripts/deep-sleep/collect.py +33 -11
- package/src/scripts/deep-sleep/extract-prompt.md +38 -0
- package/src/scripts/deep-sleep/extract.py +80 -8
- package/src/scripts/deep-sleep/synthesize-prompt.md +59 -2
- package/src/scripts/deep-sleep/synthesize.py +3 -1
- 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 +65 -29
- package/src/scripts/nexo-cognitive-decay 2.py +182 -0
- package/src/scripts/nexo-cron-wrapper.sh +53 -0
- package/src/scripts/nexo-daily-self-audit 2.py +552 -0
- package/src/scripts/nexo-daily-self-audit.py +4 -2
- package/src/scripts/nexo-deep-sleep 2.sh +97 -0
- package/src/scripts/nexo-deep-sleep.sh +66 -77
- package/src/scripts/nexo-evolution-run 2.py +597 -0
- package/src/scripts/nexo-evolution-run.py +13 -0
- package/src/scripts/nexo-followup-hygiene 2.py +112 -0
- package/src/scripts/nexo-immune 2.py +927 -0
- 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-housekeep.py +156 -1
- package/src/scripts/nexo-learning-validator 2.py +207 -0
- package/src/scripts/nexo-learning-validator.py +19 -0
- 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 +3 -2
- 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-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 +16 -11
- 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 +46 -3
- 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/scripts/nexo-watchdog.sh +72 -19
- package/src/server 2.py +733 -0
- package/src/server.py +11 -2
- 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_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_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/scripts/deep-sleep/__pycache__/extract.cpython-314.pyc +0 -0
- /package/src/scripts/{nexo-github-monitor.py → nexo-github-monitor 2.py} +0 -0
|
@@ -0,0 +1,274 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
NEXO Runtime Preflight
|
|
4
|
+
|
|
5
|
+
Runs safe end-to-end smoke tests for Cortex and Evolution using a temporary
|
|
6
|
+
workspace and a copied SQLite database. No external API calls are performed.
|
|
7
|
+
Results are written to NEXO_HOME/logs/runtime-preflight-summary.json.
|
|
8
|
+
"""
|
|
9
|
+
|
|
10
|
+
from __future__ import annotations
|
|
11
|
+
|
|
12
|
+
import importlib.util
|
|
13
|
+
import json
|
|
14
|
+
import os
|
|
15
|
+
import shutil
|
|
16
|
+
import sqlite3
|
|
17
|
+
import sys
|
|
18
|
+
import tempfile
|
|
19
|
+
import traceback
|
|
20
|
+
from datetime import datetime
|
|
21
|
+
from pathlib import Path
|
|
22
|
+
|
|
23
|
+
HOME = Path.home()
|
|
24
|
+
NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(HOME / ".nexo")))
|
|
25
|
+
# Auto-detect: if running from repo (src/scripts/), use src/ as NEXO_CODE
|
|
26
|
+
_script_dir = Path(__file__).resolve().parent
|
|
27
|
+
_repo_src = _script_dir.parent # src/scripts/ -> src/
|
|
28
|
+
NEXO_CODE = Path(os.environ.get("NEXO_CODE", str(_repo_src) if (_repo_src / "server.py").exists() else str(NEXO_HOME)))
|
|
29
|
+
|
|
30
|
+
LOG_DIR = NEXO_HOME / "logs"
|
|
31
|
+
LOG_DIR.mkdir(parents=True, exist_ok=True)
|
|
32
|
+
SUMMARY_FILE = LOG_DIR / "runtime-preflight-summary.json"
|
|
33
|
+
DB_FILE = NEXO_HOME / "data" / "nexo.db"
|
|
34
|
+
# Evolution config: NEXO_HOME/brain/ (canonical), NEXO_HOME/cortex/ (legacy fallback), NEXO_CODE (dev fallback)
|
|
35
|
+
def _find_evolution_file(name: str) -> Path:
|
|
36
|
+
for candidate in [NEXO_HOME / "brain" / name, NEXO_HOME / "cortex" / name, NEXO_CODE / name]:
|
|
37
|
+
if candidate.exists():
|
|
38
|
+
return candidate
|
|
39
|
+
return NEXO_HOME / "brain" / name # default canonical path
|
|
40
|
+
|
|
41
|
+
CORTEX_OBJECTIVE = _find_evolution_file("evolution-objective.json")
|
|
42
|
+
CORTEX_PROMPT = _find_evolution_file("evolution-prompt.md")
|
|
43
|
+
|
|
44
|
+
|
|
45
|
+
def _load_module(name: str, path: Path):
|
|
46
|
+
spec = importlib.util.spec_from_file_location(name, str(path))
|
|
47
|
+
module = importlib.util.module_from_spec(spec)
|
|
48
|
+
sys.modules[name] = module
|
|
49
|
+
assert spec.loader is not None
|
|
50
|
+
spec.loader.exec_module(module)
|
|
51
|
+
return module
|
|
52
|
+
|
|
53
|
+
|
|
54
|
+
def _write_summary(summary: dict):
|
|
55
|
+
SUMMARY_FILE.write_text(json.dumps(summary, indent=2, ensure_ascii=False))
|
|
56
|
+
|
|
57
|
+
|
|
58
|
+
def _fake_cortex_response(model: str) -> str:
|
|
59
|
+
if "opus" in model:
|
|
60
|
+
payload = {
|
|
61
|
+
"analysis": "Smoke evolution run over temp workspace",
|
|
62
|
+
"proposals": [
|
|
63
|
+
{
|
|
64
|
+
"dimension": "self_improvement",
|
|
65
|
+
"classification": "propose",
|
|
66
|
+
"action": "Add a regression smoke before live cycles",
|
|
67
|
+
"reasoning": "Smoke run should stay side-effect free",
|
|
68
|
+
}
|
|
69
|
+
],
|
|
70
|
+
"dimension_scores": {
|
|
71
|
+
"episodic_memory": 86,
|
|
72
|
+
"autonomy": 53,
|
|
73
|
+
"proactivity": 34,
|
|
74
|
+
"self_improvement": 30,
|
|
75
|
+
"agi": 6,
|
|
76
|
+
},
|
|
77
|
+
"score_evidence": {
|
|
78
|
+
"self_improvement": "Temp evolution smoke completed successfully",
|
|
79
|
+
},
|
|
80
|
+
}
|
|
81
|
+
else:
|
|
82
|
+
payload = {
|
|
83
|
+
"actions_taken": ["preflight perception completed"],
|
|
84
|
+
"signals_detected": 1,
|
|
85
|
+
"pending_questions": [],
|
|
86
|
+
"execute": [],
|
|
87
|
+
"next_interval_suggestion": 600,
|
|
88
|
+
"reflection_done": False,
|
|
89
|
+
"dmn_done": False,
|
|
90
|
+
"briefing_update": {
|
|
91
|
+
"actions_taken": ["perception smoke ok"],
|
|
92
|
+
"signals_active": [{"summary": "Smoke signal", "urgency": "INFO", "score": 10}],
|
|
93
|
+
"recommendations": ["keep runtime checks green"],
|
|
94
|
+
"pending_questions_unanswered": [],
|
|
95
|
+
"dmn_summary": "",
|
|
96
|
+
},
|
|
97
|
+
"working_memory": {
|
|
98
|
+
"current_threads": ["runtime preflight"],
|
|
99
|
+
"attention_focus": "runtime integrity",
|
|
100
|
+
"last_reasoning": "API mocked successfully",
|
|
101
|
+
"watching": ["state writes", "briefing writes"],
|
|
102
|
+
"momentum": "stable",
|
|
103
|
+
},
|
|
104
|
+
"error": None,
|
|
105
|
+
}
|
|
106
|
+
return json.dumps(payload, ensure_ascii=False)
|
|
107
|
+
|
|
108
|
+
|
|
109
|
+
def _fake_runner_response() -> tuple[str, dict]:
|
|
110
|
+
payload = {
|
|
111
|
+
"analysis": "Standalone evolution smoke over temp DB",
|
|
112
|
+
"patterns": [
|
|
113
|
+
{"type": "smoke", "description": "runner executed over temp workspace", "frequency": "once"}
|
|
114
|
+
],
|
|
115
|
+
"proposals": [
|
|
116
|
+
{
|
|
117
|
+
"dimension": "self_improvement",
|
|
118
|
+
"classification": "propose",
|
|
119
|
+
"action": "Keep standalone evolution covered by preflight",
|
|
120
|
+
"reasoning": "Regression prevention",
|
|
121
|
+
}
|
|
122
|
+
],
|
|
123
|
+
"dimension_scores": {
|
|
124
|
+
"episodic_memory": 87,
|
|
125
|
+
"autonomy": 54,
|
|
126
|
+
"proactivity": 35,
|
|
127
|
+
"self_improvement": 31,
|
|
128
|
+
"agi": 6,
|
|
129
|
+
},
|
|
130
|
+
"score_evidence": {
|
|
131
|
+
"self_improvement": "standalone runner smoke ok",
|
|
132
|
+
},
|
|
133
|
+
}
|
|
134
|
+
usage = {"input_tokens": 1234, "output_tokens": 432}
|
|
135
|
+
return json.dumps(payload, ensure_ascii=False), usage
|
|
136
|
+
|
|
137
|
+
|
|
138
|
+
def main() -> int:
|
|
139
|
+
started = datetime.now().isoformat()
|
|
140
|
+
summary = {
|
|
141
|
+
"timestamp": started,
|
|
142
|
+
"ok": False,
|
|
143
|
+
"checks": {},
|
|
144
|
+
"errors": [],
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
if not DB_FILE.exists():
|
|
148
|
+
summary["errors"].append("nexo.db missing")
|
|
149
|
+
_write_summary(summary)
|
|
150
|
+
return 1
|
|
151
|
+
|
|
152
|
+
preflight_root = HOME / ".codex" / "memories"
|
|
153
|
+
preflight_root.mkdir(parents=True, exist_ok=True)
|
|
154
|
+
temp_root = Path(tempfile.mkdtemp(prefix="nexo-runtime-preflight-", dir=str(preflight_root)))
|
|
155
|
+
try:
|
|
156
|
+
temp_db = temp_root / "nexo.db"
|
|
157
|
+
shutil.copy2(DB_FILE, temp_db)
|
|
158
|
+
|
|
159
|
+
temp_cortex_dir = temp_root / "brain"
|
|
160
|
+
temp_logs_dir = temp_root / "logs"
|
|
161
|
+
temp_coord_dir = temp_root / "coordination"
|
|
162
|
+
temp_daily_dir = temp_root / "daily_summaries"
|
|
163
|
+
temp_dmn_dir = temp_root / "dmn_insights"
|
|
164
|
+
temp_snapshots_dir = temp_root / "snapshots"
|
|
165
|
+
temp_sandbox_dir = temp_root / "sandbox" / "workspace"
|
|
166
|
+
temp_scripts_dir = temp_root / "scripts"
|
|
167
|
+
for directory in [
|
|
168
|
+
temp_cortex_dir, temp_logs_dir, temp_coord_dir, temp_daily_dir,
|
|
169
|
+
temp_dmn_dir, temp_snapshots_dir, temp_sandbox_dir, temp_scripts_dir,
|
|
170
|
+
]:
|
|
171
|
+
directory.mkdir(parents=True, exist_ok=True)
|
|
172
|
+
|
|
173
|
+
temp_objective = temp_cortex_dir / "evolution-objective.json"
|
|
174
|
+
temp_prompt = temp_cortex_dir / "evolution-prompt.md"
|
|
175
|
+
if CORTEX_OBJECTIVE.exists():
|
|
176
|
+
shutil.copy2(CORTEX_OBJECTIVE, temp_objective)
|
|
177
|
+
else:
|
|
178
|
+
# Create a minimal objective so evolution smoke tests can proceed
|
|
179
|
+
temp_objective.write_text(json.dumps({"evolution_enabled": True, "dimensions": {}}, indent=2))
|
|
180
|
+
if CORTEX_PROMPT.exists():
|
|
181
|
+
shutil.copy2(CORTEX_PROMPT, temp_prompt)
|
|
182
|
+
snapshot_restore = NEXO_CODE / "scripts" / "nexo-snapshot-restore.sh"
|
|
183
|
+
if snapshot_restore.exists():
|
|
184
|
+
shutil.copy2(snapshot_restore, temp_scripts_dir / "nexo-snapshot-restore.sh")
|
|
185
|
+
|
|
186
|
+
temp_api_key = temp_root / "anthropic-api-key.txt"
|
|
187
|
+
temp_api_key.write_text("smoke-test-key")
|
|
188
|
+
|
|
189
|
+
evolution_cycle = _load_module("evolution_cycle", NEXO_CODE / "evolution_cycle.py")
|
|
190
|
+
evolution_cycle.NEXO_DB = temp_db
|
|
191
|
+
evolution_cycle.NEXO_HOME = temp_root
|
|
192
|
+
evolution_cycle.SANDBOX_DIR = temp_sandbox_dir
|
|
193
|
+
evolution_cycle.SNAPSHOTS_DIR = temp_snapshots_dir
|
|
194
|
+
evolution_cycle.OBJECTIVE_FILE = temp_objective
|
|
195
|
+
evolution_cycle.PROMPT_FILE = temp_prompt
|
|
196
|
+
evolution_cycle.RESTORE_LOG = temp_logs_dir / "snapshot-restores.log"
|
|
197
|
+
|
|
198
|
+
week_data = evolution_cycle.get_week_data(str(temp_db))
|
|
199
|
+
prompt = evolution_cycle.build_evolution_prompt(week_data, evolution_cycle.load_objective())
|
|
200
|
+
restore_ok = evolution_cycle.dry_run_restore_test()
|
|
201
|
+
if not restore_ok:
|
|
202
|
+
raise RuntimeError("dry_run_restore_test failed")
|
|
203
|
+
summary["checks"]["evolution_cycle"] = {
|
|
204
|
+
"learnings": len(week_data.get("learnings", [])),
|
|
205
|
+
"decisions": len(week_data.get("decisions", [])),
|
|
206
|
+
"prompt_chars": len(prompt),
|
|
207
|
+
"restore_ok": restore_ok,
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
cortex = _load_module("cortex_plugin", NEXO_CODE / "plugins" / "cortex.py")
|
|
211
|
+
cortex.NEXO_DB = temp_db
|
|
212
|
+
cortex.DRY_RUN = True
|
|
213
|
+
cortex.BRIEFING_FILE = temp_cortex_dir / "briefing.json"
|
|
214
|
+
cortex.HEALTH_FILE = temp_cortex_dir / "health.json"
|
|
215
|
+
cortex.STATE_FILE = temp_cortex_dir / "state.json"
|
|
216
|
+
cortex.STATE_TMP = temp_cortex_dir / "state.tmp"
|
|
217
|
+
cortex.LOG_DIR = temp_logs_dir
|
|
218
|
+
cortex.COORD_DIR = temp_coord_dir
|
|
219
|
+
cortex.SIGNALS_FILE = temp_coord_dir / "pending-signals.json"
|
|
220
|
+
cortex.DAILY_SUMMARIES_DIR = temp_daily_dir
|
|
221
|
+
cortex.DMN_INSIGHTS_DIR = temp_dmn_dir
|
|
222
|
+
cortex.FAILURE_COUNT_FILE = temp_cortex_dir / ".failure-count"
|
|
223
|
+
cortex.PID_FILE = temp_cortex_dir / "cortex.pid"
|
|
224
|
+
cortex.API_KEY_FILE = temp_api_key
|
|
225
|
+
cortex.wa_notify = lambda *args, **kwargs: None
|
|
226
|
+
cortex.poll_wa_inbox = lambda state: None
|
|
227
|
+
cortex._check_health_endpoints = lambda: []
|
|
228
|
+
cortex._call_anthropic = lambda prompt, model=None, max_tokens=4096: _fake_cortex_response(model or "")
|
|
229
|
+
|
|
230
|
+
# Smoke test: verify cortex plugin loads and has expected tools
|
|
231
|
+
cortex_path = NEXO_CODE / "plugins" / "cortex.py"
|
|
232
|
+
if cortex_path.exists():
|
|
233
|
+
import importlib.util
|
|
234
|
+
spec = importlib.util.spec_from_file_location("cortex_plugin", str(cortex_path))
|
|
235
|
+
cortex_mod = importlib.util.module_from_spec(spec)
|
|
236
|
+
spec.loader.exec_module(cortex_mod)
|
|
237
|
+
assert hasattr(cortex_mod, 'TOOLS'), "cortex plugin missing TOOLS"
|
|
238
|
+
tool_names = [t[1] for t in cortex_mod.TOOLS]
|
|
239
|
+
assert "nexo_cortex_check" in tool_names, "cortex plugin missing nexo_cortex_check tool"
|
|
240
|
+
else:
|
|
241
|
+
tool_names = ["cortex.py not found"]
|
|
242
|
+
summary["checks"]["cortex_plugin"] = {
|
|
243
|
+
"status": "pass",
|
|
244
|
+
"tools_found": tool_names,
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
# Evolution runner: component verification (not full execution — that needs CLI)
|
|
248
|
+
runner_path = NEXO_CODE / "scripts" / "nexo-evolution-run.py"
|
|
249
|
+
if runner_path.exists():
|
|
250
|
+
runner = _load_module("nexo_evolution_run", runner_path)
|
|
251
|
+
checks = {
|
|
252
|
+
"module_loads": True,
|
|
253
|
+
"has_run": hasattr(runner, 'run'),
|
|
254
|
+
"has_load_objective": hasattr(runner, 'load_objective'),
|
|
255
|
+
"has_get_week_data": hasattr(runner, 'get_week_data'),
|
|
256
|
+
}
|
|
257
|
+
summary["checks"]["standalone_runner"] = checks
|
|
258
|
+
else:
|
|
259
|
+
summary["checks"]["standalone_runner"] = {"status": "skip", "reason": "runner not found"}
|
|
260
|
+
|
|
261
|
+
summary["ok"] = True
|
|
262
|
+
_write_summary(summary)
|
|
263
|
+
return 0
|
|
264
|
+
except Exception as exc:
|
|
265
|
+
summary["errors"].append(f"{type(exc).__name__}: {exc}")
|
|
266
|
+
summary["errors"].append(traceback.format_exc())
|
|
267
|
+
_write_summary(summary)
|
|
268
|
+
return 1
|
|
269
|
+
finally:
|
|
270
|
+
shutil.rmtree(temp_root, ignore_errors=True)
|
|
271
|
+
|
|
272
|
+
|
|
273
|
+
if __name__ == "__main__":
|
|
274
|
+
sys.exit(main())
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""Quick email sender for NEXO progress updates."""
|
|
3
|
+
import smtplib, sys
|
|
4
|
+
from email.mime.text import MIMEText
|
|
5
|
+
from email.utils import formataddr
|
|
6
|
+
|
|
7
|
+
def send(subject, body, to="user@example.com", cc="user@example.com"):
|
|
8
|
+
msg = MIMEText(body, 'plain', 'utf-8')
|
|
9
|
+
msg['From'] = formataddr(('NEXO', 'nexo@example.com'))
|
|
10
|
+
msg['To'] = to
|
|
11
|
+
msg['Cc'] = cc
|
|
12
|
+
msg['Subject'] = subject
|
|
13
|
+
smtp = smtplib.SMTP_SSL(os.environ.get('NEXO_SMTP_HOST', 'smtp.example.com'), int(os.environ.get('NEXO_SMTP_PORT', '465')))
|
|
14
|
+
smtp.login(FROM_EMAIL, os.environ.get('NEXO_SMTP_PASSWORD', ''))
|
|
15
|
+
recipients = [to]
|
|
16
|
+
if cc:
|
|
17
|
+
recipients.append(cc)
|
|
18
|
+
smtp.send_message(msg)
|
|
19
|
+
smtp.quit()
|
|
20
|
+
print(f"OK — sent to {to}")
|
|
21
|
+
|
|
22
|
+
if __name__ == "__main__":
|
|
23
|
+
subject = sys.argv[1] if len(sys.argv) > 1 else "NEXO Update"
|
|
24
|
+
body = sys.argv[2] if len(sys.argv) > 2 else "No body"
|
|
25
|
+
send(subject, body)
|
|
@@ -0,0 +1,178 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
NEXO Email Reply Helper — Sends email replies with correct threading headers.
|
|
4
|
+
NEXO calls this instead of building SMTP manually.
|
|
5
|
+
|
|
6
|
+
Usage:
|
|
7
|
+
nexo-send-reply.py --to addr --subject "Re: ..." --in-reply-to "<msg-id>" --body "text"
|
|
8
|
+
nexo-send-reply.py --to addr --subject "Re: ..." --in-reply-to "<msg-id>" --body-file /tmp/reply.html --html
|
|
9
|
+
nexo-send-reply.py --to addr --subject "New subject" --body "text" (new email, no threading)
|
|
10
|
+
|
|
11
|
+
Options:
|
|
12
|
+
--to Recipient (required)
|
|
13
|
+
--cc CC recipients (comma-separated, default: info@example.com)
|
|
14
|
+
--subject Subject line (required)
|
|
15
|
+
--in-reply-to Message-ID of the email being replied to (for threading)
|
|
16
|
+
--references Full References chain (optional, defaults to in-reply-to value)
|
|
17
|
+
--body Plain text body (inline)
|
|
18
|
+
--body-file Read body from file instead
|
|
19
|
+
--html Treat body as HTML
|
|
20
|
+
--quote Original message text to include as quoted reply
|
|
21
|
+
--quote-file Read original message from file to quote
|
|
22
|
+
--quote-from Sender of the original message (for "On date, X wrote:")
|
|
23
|
+
--quote-date Date of the original message
|
|
24
|
+
--attachment Path to file attachment (can be repeated)
|
|
25
|
+
"""
|
|
26
|
+
|
|
27
|
+
import argparse
|
|
28
|
+
import imaplib
|
|
29
|
+
import json
|
|
30
|
+
import os
|
|
31
|
+
import smtplib
|
|
32
|
+
import sys
|
|
33
|
+
import time
|
|
34
|
+
from email.message import EmailMessage
|
|
35
|
+
from email.utils import make_msgid, formatdate
|
|
36
|
+
from pathlib import Path
|
|
37
|
+
import mimetypes
|
|
38
|
+
|
|
39
|
+
NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
|
|
40
|
+
CONFIG_PATH = NEXO_HOME / "nexo-email" / "config.json"
|
|
41
|
+
|
|
42
|
+
|
|
43
|
+
def load_config():
|
|
44
|
+
with open(CONFIG_PATH) as f:
|
|
45
|
+
return json.load(f)
|
|
46
|
+
|
|
47
|
+
|
|
48
|
+
def build_message(args, config):
|
|
49
|
+
msg = EmailMessage()
|
|
50
|
+
|
|
51
|
+
# From / To / CC
|
|
52
|
+
msg["From"] = f"NEXO <{config['email']}>"
|
|
53
|
+
msg["To"] = args.to
|
|
54
|
+
if args.cc:
|
|
55
|
+
msg["Cc"] = args.cc
|
|
56
|
+
|
|
57
|
+
# Subject
|
|
58
|
+
msg["Subject"] = args.subject
|
|
59
|
+
|
|
60
|
+
# Threading headers — this is the whole point of this script
|
|
61
|
+
if args.in_reply_to:
|
|
62
|
+
msg["In-Reply-To"] = args.in_reply_to
|
|
63
|
+
msg["References"] = args.references or args.in_reply_to
|
|
64
|
+
|
|
65
|
+
# Standard headers
|
|
66
|
+
msg["Message-ID"] = make_msgid(domain="example.com")
|
|
67
|
+
msg["Date"] = formatdate(localtime=True)
|
|
68
|
+
|
|
69
|
+
# Body
|
|
70
|
+
if args.body_file:
|
|
71
|
+
body = Path(args.body_file).read_text(encoding="utf-8")
|
|
72
|
+
else:
|
|
73
|
+
body = args.body or ""
|
|
74
|
+
|
|
75
|
+
# Quoted original message
|
|
76
|
+
quote_text = ""
|
|
77
|
+
if args.quote_file:
|
|
78
|
+
quote_text = Path(args.quote_file).read_text(encoding="utf-8")
|
|
79
|
+
elif args.quote:
|
|
80
|
+
quote_text = args.quote
|
|
81
|
+
|
|
82
|
+
if quote_text:
|
|
83
|
+
quote_from = args.quote_from or args.to
|
|
84
|
+
quote_date = args.quote_date or ""
|
|
85
|
+
attribution = f"On {quote_date}, {quote_from} wrote:" if quote_date else f"{quote_from} wrote:"
|
|
86
|
+
|
|
87
|
+
if args.html:
|
|
88
|
+
quoted_html = quote_text.replace("\n", "<br>") if "<" not in quote_text else quote_text
|
|
89
|
+
body = (
|
|
90
|
+
f"{body}<br><br>"
|
|
91
|
+
f"<div style=\"color:#666;\">{attribution}</div>"
|
|
92
|
+
f"<blockquote style=\"margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex;color:#666;\">"
|
|
93
|
+
f"{quoted_html}"
|
|
94
|
+
f"</blockquote>"
|
|
95
|
+
)
|
|
96
|
+
else:
|
|
97
|
+
quoted_lines = "\n".join(f"> {line}" for line in quote_text.splitlines())
|
|
98
|
+
body = f"{body}\n\n{attribution}\n{quoted_lines}"
|
|
99
|
+
|
|
100
|
+
if args.html:
|
|
101
|
+
msg.set_content(body, subtype="html")
|
|
102
|
+
else:
|
|
103
|
+
msg.set_content(body)
|
|
104
|
+
|
|
105
|
+
# Attachments
|
|
106
|
+
if args.attachment:
|
|
107
|
+
for filepath in args.attachment:
|
|
108
|
+
p = Path(filepath)
|
|
109
|
+
if not p.exists():
|
|
110
|
+
print(f"WARNING: attachment not found: {filepath}", file=sys.stderr)
|
|
111
|
+
continue
|
|
112
|
+
mime_type, _ = mimetypes.guess_type(str(p))
|
|
113
|
+
if mime_type is None:
|
|
114
|
+
mime_type = "application/octet-stream"
|
|
115
|
+
maintype, subtype = mime_type.split("/", 1)
|
|
116
|
+
with open(p, "rb") as f:
|
|
117
|
+
msg.add_attachment(
|
|
118
|
+
f.read(),
|
|
119
|
+
maintype=maintype,
|
|
120
|
+
subtype=subtype,
|
|
121
|
+
filename=p.name
|
|
122
|
+
)
|
|
123
|
+
|
|
124
|
+
return msg
|
|
125
|
+
|
|
126
|
+
|
|
127
|
+
def send(msg, config):
|
|
128
|
+
with smtplib.SMTP_SSL(config["smtp_host"], config["smtp_port"]) as server:
|
|
129
|
+
server.login(config["email"], config["password"])
|
|
130
|
+
server.send_message(msg)
|
|
131
|
+
|
|
132
|
+
|
|
133
|
+
def save_to_sent(msg, config):
|
|
134
|
+
"""Append the sent message to the IMAP Sent folder."""
|
|
135
|
+
try:
|
|
136
|
+
with imaplib.IMAP4_SSL(config["imap_host"], config["imap_port"]) as imap:
|
|
137
|
+
imap.login(config["email"], config["password"])
|
|
138
|
+
imap.append(
|
|
139
|
+
"INBOX.Sent",
|
|
140
|
+
r"\Seen",
|
|
141
|
+
imaplib.Time2Internaldate(time.time()),
|
|
142
|
+
msg.as_bytes(),
|
|
143
|
+
)
|
|
144
|
+
except Exception as e:
|
|
145
|
+
print(f"WARNING: could not save to Sent folder: {e}", file=sys.stderr)
|
|
146
|
+
|
|
147
|
+
|
|
148
|
+
def main():
|
|
149
|
+
parser = argparse.ArgumentParser(description="NEXO Email Reply Helper")
|
|
150
|
+
parser.add_argument("--to", required=True, help="Recipient email")
|
|
151
|
+
parser.add_argument("--cc", default=None, help="CC recipients (comma-separated, no default)")
|
|
152
|
+
parser.add_argument("--subject", required=True, help="Subject line")
|
|
153
|
+
parser.add_argument("--in-reply-to", default=None, help="Message-ID for threading")
|
|
154
|
+
parser.add_argument("--references", default=None, help="References chain for threading")
|
|
155
|
+
parser.add_argument("--body", default=None, help="Body text inline")
|
|
156
|
+
parser.add_argument("--body-file", default=None, help="Read body from file")
|
|
157
|
+
parser.add_argument("--html", action="store_true", help="Body is HTML")
|
|
158
|
+
parser.add_argument("--quote", default=None, help="Original message text to quote inline")
|
|
159
|
+
parser.add_argument("--quote-file", default=None, help="Read original message from file to quote")
|
|
160
|
+
parser.add_argument("--quote-from", default=None, help="Sender of original (for attribution line)")
|
|
161
|
+
parser.add_argument("--quote-date", default=None, help="Date of original message")
|
|
162
|
+
parser.add_argument("--attachment", action="append", help="File to attach (repeatable)")
|
|
163
|
+
args = parser.parse_args()
|
|
164
|
+
|
|
165
|
+
if not args.body and not args.body_file:
|
|
166
|
+
print("ERROR: --body or --body-file required", file=sys.stderr)
|
|
167
|
+
sys.exit(1)
|
|
168
|
+
|
|
169
|
+
config = load_config()
|
|
170
|
+
msg = build_message(args, config)
|
|
171
|
+
send(msg, config)
|
|
172
|
+
save_to_sent(msg, config)
|
|
173
|
+
|
|
174
|
+
print(f"OK: sent to {args.to} | subject: {args.subject} | threaded: {bool(args.in_reply_to)}")
|
|
175
|
+
|
|
176
|
+
|
|
177
|
+
if __name__ == "__main__":
|
|
178
|
+
main()
|