nexo-brain 2.3.0 → 2.3.2

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.
Files changed (299) hide show
  1. package/README.md +1 -1
  2. package/bin/nexo-brain.js +92 -9
  3. package/bin/postinstall.js +22 -15
  4. package/package.json +7 -4
  5. package/src/auto_update.py +194 -5
  6. package/src/crons/sync.py +6 -2
  7. package/src/db/_core.py +1 -0
  8. package/src/db/_entities.py +1 -0
  9. package/src/db/_episodic.py +1 -0
  10. package/src/db/_learnings.py +1 -0
  11. package/src/db/_reminders.py +1 -0
  12. package/src/db/_schema.py +11 -1
  13. package/src/db/_sessions.py +1 -0
  14. package/src/db/_skills.py +1 -0
  15. package/src/hooks/capture-tool-logs.sh +23 -6
  16. package/src/hooks/session-start.sh +4 -3
  17. package/src/plugin_loader.py +1 -0
  18. package/src/plugins/update.py +377 -26
  19. package/src/scripts/deep-sleep/apply_findings.py +1 -0
  20. package/src/scripts/deep-sleep/collect.py +1 -0
  21. package/src/scripts/deep-sleep/extract.py +1 -0
  22. package/src/scripts/deep-sleep/synthesize.py +1 -0
  23. package/src/scripts/nexo-catchup.py +29 -4
  24. package/src/scripts/nexo-daily-self-audit.py +21 -1
  25. package/src/scripts/nexo-evolution-run.py +21 -1
  26. package/src/scripts/nexo-learning-housekeep.py +1 -0
  27. package/src/scripts/nexo-postmortem-consolidator.py +34 -9
  28. package/src/scripts/nexo-sleep.py +32 -10
  29. package/src/scripts/nexo-synthesis.py +29 -9
  30. package/src/scripts/nexo-update.sh +109 -7
  31. package/src/scripts/nexo-watchdog.sh +122 -58
  32. package/src/server.py +66 -1
  33. package/src/tools_coordination.py +1 -0
  34. package/src/tools_sessions.py +1 -0
  35. package/scripts/migrate-to-unified 2.sh +0 -813
  36. package/scripts/migrate-to-unified.sh +0 -813
  37. package/scripts/migrate-v1.5-to-v1.6 2.py +0 -778
  38. package/scripts/migrate-v1.5-to-v1.6.py +0 -778
  39. package/scripts/migrate-v1.7-to-v1.8 2.py +0 -214
  40. package/scripts/migrate-v1.7-to-v1.8.py +0 -214
  41. package/scripts/nexo-preflight.sh +0 -236
  42. package/scripts/pre-commit-check 2.sh +0 -55
  43. package/scripts/pre-commit-check.sh +0 -55
  44. package/src/__pycache__/auto_close_sessions.cpython-314.pyc +0 -0
  45. package/src/__pycache__/auto_update.cpython-310.pyc +0 -0
  46. package/src/__pycache__/hnsw_index.cpython-310.pyc +0 -0
  47. package/src/__pycache__/hnsw_index.cpython-314.pyc +0 -0
  48. package/src/__pycache__/kg_populate.cpython-310.pyc +0 -0
  49. package/src/__pycache__/knowledge_graph.cpython-310.pyc +0 -0
  50. package/src/__pycache__/plugin_loader.cpython-310.pyc +0 -0
  51. package/src/__pycache__/plugin_loader.cpython-314.pyc +0 -0
  52. package/src/__pycache__/tools_coordination.cpython-310.pyc +0 -0
  53. package/src/__pycache__/tools_credentials.cpython-310.pyc +0 -0
  54. package/src/__pycache__/tools_learnings.cpython-310.pyc +0 -0
  55. package/src/__pycache__/tools_menu.cpython-310.pyc +0 -0
  56. package/src/__pycache__/tools_reminders.cpython-310.pyc +0 -0
  57. package/src/__pycache__/tools_reminders_crud.cpython-310.pyc +0 -0
  58. package/src/__pycache__/tools_sessions.cpython-310.pyc +0 -0
  59. package/src/__pycache__/tools_task_history.cpython-310.pyc +0 -0
  60. package/src/auto_close_sessions 2.py +0 -159
  61. package/src/auto_update 2.py +0 -634
  62. package/src/claim_graph 2.py +0 -323
  63. package/src/cognitive/__init__ 2.py +0 -62
  64. package/src/cognitive/__pycache__/__init__.cpython-310.pyc +0 -0
  65. package/src/cognitive/__pycache__/__init__.cpython-312.pyc +0 -0
  66. package/src/cognitive/__pycache__/__init__.cpython-314.pyc +0 -0
  67. package/src/cognitive/__pycache__/_core.cpython-310.pyc +0 -0
  68. package/src/cognitive/__pycache__/_core.cpython-312.pyc +0 -0
  69. package/src/cognitive/__pycache__/_core.cpython-314.pyc +0 -0
  70. package/src/cognitive/__pycache__/_decay.cpython-310.pyc +0 -0
  71. package/src/cognitive/__pycache__/_decay.cpython-312.pyc +0 -0
  72. package/src/cognitive/__pycache__/_decay.cpython-314.pyc +0 -0
  73. package/src/cognitive/__pycache__/_ingest.cpython-310.pyc +0 -0
  74. package/src/cognitive/__pycache__/_ingest.cpython-312.pyc +0 -0
  75. package/src/cognitive/__pycache__/_ingest.cpython-314.pyc +0 -0
  76. package/src/cognitive/__pycache__/_memory.cpython-310.pyc +0 -0
  77. package/src/cognitive/__pycache__/_memory.cpython-312.pyc +0 -0
  78. package/src/cognitive/__pycache__/_memory.cpython-314.pyc +0 -0
  79. package/src/cognitive/__pycache__/_search.cpython-310.pyc +0 -0
  80. package/src/cognitive/__pycache__/_search.cpython-312.pyc +0 -0
  81. package/src/cognitive/__pycache__/_search.cpython-314.pyc +0 -0
  82. package/src/cognitive/__pycache__/_trust.cpython-310.pyc +0 -0
  83. package/src/cognitive/__pycache__/_trust.cpython-312.pyc +0 -0
  84. package/src/cognitive/__pycache__/_trust.cpython-314.pyc +0 -0
  85. package/src/cognitive/_core 2.py +0 -567
  86. package/src/cognitive/_decay 2.py +0 -382
  87. package/src/cognitive/_ingest 2.py +0 -892
  88. package/src/cognitive/_memory 2.py +0 -912
  89. package/src/cognitive/_search 2.py +0 -949
  90. package/src/cognitive/_trust 2.py +0 -464
  91. package/src/crons/__pycache__/sync.cpython-314.pyc +0 -0
  92. package/src/crons/manifest 2.json +0 -106
  93. package/src/crons/sync 2.py +0 -217
  94. package/src/dashboard/__init__ 2.py +0 -0
  95. package/src/dashboard/__pycache__/__init__.cpython-310.pyc +0 -0
  96. package/src/dashboard/__pycache__/app.cpython-310.pyc +0 -0
  97. package/src/dashboard/app 2.py +0 -789
  98. package/src/db/__init__ 2.py +0 -89
  99. package/src/db/__pycache__/__init__.cpython-310.pyc +0 -0
  100. package/src/db/__pycache__/__init__.cpython-312.pyc +0 -0
  101. package/src/db/__pycache__/__init__.cpython-314.pyc +0 -0
  102. package/src/db/__pycache__/_core.cpython-310.pyc +0 -0
  103. package/src/db/__pycache__/_core.cpython-312.pyc +0 -0
  104. package/src/db/__pycache__/_core.cpython-314.pyc +0 -0
  105. package/src/db/__pycache__/_credentials.cpython-310.pyc +0 -0
  106. package/src/db/__pycache__/_credentials.cpython-312.pyc +0 -0
  107. package/src/db/__pycache__/_credentials.cpython-314.pyc +0 -0
  108. package/src/db/__pycache__/_cron_runs.cpython-310.pyc +0 -0
  109. package/src/db/__pycache__/_cron_runs.cpython-314.pyc +0 -0
  110. package/src/db/__pycache__/_entities.cpython-310.pyc +0 -0
  111. package/src/db/__pycache__/_entities.cpython-312.pyc +0 -0
  112. package/src/db/__pycache__/_entities.cpython-314.pyc +0 -0
  113. package/src/db/__pycache__/_episodic.cpython-310.pyc +0 -0
  114. package/src/db/__pycache__/_episodic.cpython-312.pyc +0 -0
  115. package/src/db/__pycache__/_episodic.cpython-314.pyc +0 -0
  116. package/src/db/__pycache__/_evolution.cpython-310.pyc +0 -0
  117. package/src/db/__pycache__/_evolution.cpython-312.pyc +0 -0
  118. package/src/db/__pycache__/_evolution.cpython-314.pyc +0 -0
  119. package/src/db/__pycache__/_fts.cpython-310.pyc +0 -0
  120. package/src/db/__pycache__/_fts.cpython-312.pyc +0 -0
  121. package/src/db/__pycache__/_fts.cpython-314.pyc +0 -0
  122. package/src/db/__pycache__/_learnings.cpython-310.pyc +0 -0
  123. package/src/db/__pycache__/_learnings.cpython-312.pyc +0 -0
  124. package/src/db/__pycache__/_learnings.cpython-314.pyc +0 -0
  125. package/src/db/__pycache__/_reminders.cpython-310.pyc +0 -0
  126. package/src/db/__pycache__/_reminders.cpython-312.pyc +0 -0
  127. package/src/db/__pycache__/_reminders.cpython-314.pyc +0 -0
  128. package/src/db/__pycache__/_schema.cpython-310.pyc +0 -0
  129. package/src/db/__pycache__/_schema.cpython-312.pyc +0 -0
  130. package/src/db/__pycache__/_schema.cpython-314.pyc +0 -0
  131. package/src/db/__pycache__/_sessions.cpython-310.pyc +0 -0
  132. package/src/db/__pycache__/_sessions.cpython-312.pyc +0 -0
  133. package/src/db/__pycache__/_sessions.cpython-314.pyc +0 -0
  134. package/src/db/__pycache__/_skills.cpython-310.pyc +0 -0
  135. package/src/db/__pycache__/_skills.cpython-312.pyc +0 -0
  136. package/src/db/__pycache__/_skills.cpython-314.pyc +0 -0
  137. package/src/db/__pycache__/_tasks.cpython-310.pyc +0 -0
  138. package/src/db/__pycache__/_tasks.cpython-312.pyc +0 -0
  139. package/src/db/__pycache__/_tasks.cpython-314.pyc +0 -0
  140. package/src/db/_core 2.py +0 -417
  141. package/src/db/_credentials 2.py +0 -124
  142. package/src/db/_entities 2.py +0 -178
  143. package/src/db/_episodic 2.py +0 -738
  144. package/src/db/_evolution 2.py +0 -54
  145. package/src/db/_fts 2.py +0 -406
  146. package/src/db/_learnings 2.py +0 -168
  147. package/src/db/_reminders 2.py +0 -338
  148. package/src/db/_schema 2.py +0 -364
  149. package/src/db/_sessions 2.py +0 -300
  150. package/src/db/_tasks 2.py +0 -91
  151. package/src/evolution_cycle 2.py +0 -266
  152. package/src/hnsw_index 2.py +0 -254
  153. package/src/hooks/auto_capture 2.py +0 -208
  154. package/src/hooks/caffeinate-guard 2.sh +0 -8
  155. package/src/hooks/capture-session 2.sh +0 -21
  156. package/src/hooks/capture-tool-logs 2.sh +0 -127
  157. package/src/hooks/daily-briefing-check 2.sh +0 -33
  158. package/src/hooks/inbox-hook 2.sh +0 -76
  159. package/src/hooks/post-compact 2.sh +0 -148
  160. package/src/hooks/pre-compact 2.sh +0 -151
  161. package/src/hooks/session-start 2.sh +0 -268
  162. package/src/hooks/session-stop 2.sh +0 -140
  163. package/src/kg_populate 2.py +0 -290
  164. package/src/knowledge_graph 2.py +0 -257
  165. package/src/maintenance 2.py +0 -59
  166. package/src/migrate_embeddings 2.py +0 -122
  167. package/src/plugin_loader 2.py +0 -202
  168. package/src/plugins/__init__ 2.py +0 -0
  169. package/src/plugins/__pycache__/__init__ 2.cpython-310.pyc +0 -0
  170. package/src/plugins/__pycache__/__init__.cpython-310.pyc +0 -0
  171. package/src/plugins/__pycache__/__init__.cpython-314.pyc +0 -0
  172. package/src/plugins/__pycache__/adaptive_mode 2.cpython-310.pyc +0 -0
  173. package/src/plugins/__pycache__/adaptive_mode.cpython-310.pyc +0 -0
  174. package/src/plugins/__pycache__/adaptive_mode.cpython-314.pyc +0 -0
  175. package/src/plugins/__pycache__/agents 2.cpython-310.pyc +0 -0
  176. package/src/plugins/__pycache__/agents.cpython-310.pyc +0 -0
  177. package/src/plugins/__pycache__/artifact_registry 2.cpython-310.pyc +0 -0
  178. package/src/plugins/__pycache__/artifact_registry.cpython-310.pyc +0 -0
  179. package/src/plugins/__pycache__/backup 2.cpython-310.pyc +0 -0
  180. package/src/plugins/__pycache__/backup.cpython-310.pyc +0 -0
  181. package/src/plugins/__pycache__/cognitive_memory 2.cpython-310.pyc +0 -0
  182. package/src/plugins/__pycache__/cognitive_memory.cpython-310.pyc +0 -0
  183. package/src/plugins/__pycache__/core_rules 2.cpython-310.pyc +0 -0
  184. package/src/plugins/__pycache__/core_rules.cpython-310.pyc +0 -0
  185. package/src/plugins/__pycache__/cortex 2.cpython-310.pyc +0 -0
  186. package/src/plugins/__pycache__/cortex.cpython-310.pyc +0 -0
  187. package/src/plugins/__pycache__/entities 2.cpython-310.pyc +0 -0
  188. package/src/plugins/__pycache__/entities.cpython-310.pyc +0 -0
  189. package/src/plugins/__pycache__/episodic_memory 2.cpython-310.pyc +0 -0
  190. package/src/plugins/__pycache__/episodic_memory.cpython-310.pyc +0 -0
  191. package/src/plugins/__pycache__/evolution 2.cpython-310.pyc +0 -0
  192. package/src/plugins/__pycache__/evolution.cpython-310.pyc +0 -0
  193. package/src/plugins/__pycache__/guard 2.cpython-310.pyc +0 -0
  194. package/src/plugins/__pycache__/guard.cpython-310.pyc +0 -0
  195. package/src/plugins/__pycache__/knowledge_graph_tools 2.cpython-310.pyc +0 -0
  196. package/src/plugins/__pycache__/knowledge_graph_tools.cpython-310.pyc +0 -0
  197. package/src/plugins/__pycache__/preferences 2.cpython-310.pyc +0 -0
  198. package/src/plugins/__pycache__/preferences.cpython-310.pyc +0 -0
  199. package/src/plugins/__pycache__/schedule.cpython-310.pyc +0 -0
  200. package/src/plugins/__pycache__/schedule.cpython-314.pyc +0 -0
  201. package/src/plugins/__pycache__/skills.cpython-310.pyc +0 -0
  202. package/src/plugins/__pycache__/skills.cpython-314.pyc +0 -0
  203. package/src/plugins/__pycache__/update 2.cpython-310.pyc +0 -0
  204. package/src/plugins/__pycache__/update.cpython-310.pyc +0 -0
  205. package/src/plugins/adaptive_mode 2.py +0 -805
  206. package/src/plugins/agents 2.py +0 -52
  207. package/src/plugins/artifact_registry 2.py +0 -450
  208. package/src/plugins/backup 2.py +0 -104
  209. package/src/plugins/cognitive_memory 2.py +0 -564
  210. package/src/plugins/core_rules 2.py +0 -252
  211. package/src/plugins/cortex 2.py +0 -299
  212. package/src/plugins/entities 2.py +0 -67
  213. package/src/plugins/episodic_memory 2.py +0 -533
  214. package/src/plugins/evolution 2.py +0 -115
  215. package/src/plugins/guard 2.py +0 -746
  216. package/src/plugins/knowledge_graph_tools 2.py +0 -105
  217. package/src/plugins/preferences 2.py +0 -47
  218. package/src/plugins/update 2.py +0 -256
  219. package/src/requirements 2.txt +0 -12
  220. package/src/rules/__init__ 2.py +0 -0
  221. package/src/rules/core-rules 2.json +0 -331
  222. package/src/rules/migrate 2.py +0 -207
  223. package/src/scripts/__pycache__/nexo-auto-update.cpython-314.pyc +0 -0
  224. package/src/scripts/__pycache__/nexo-catchup.cpython-314.pyc +0 -0
  225. package/src/scripts/__pycache__/nexo-cognitive-decay.cpython-314.pyc +0 -0
  226. package/src/scripts/__pycache__/nexo-daily-self-audit.cpython-314.pyc +0 -0
  227. package/src/scripts/__pycache__/nexo-evolution-run.cpython-314.pyc +0 -0
  228. package/src/scripts/__pycache__/nexo-followup-hygiene.cpython-314.pyc +0 -0
  229. package/src/scripts/__pycache__/nexo-immune.cpython-314.pyc +0 -0
  230. package/src/scripts/__pycache__/nexo-install.cpython-314.pyc +0 -0
  231. package/src/scripts/__pycache__/nexo-learning-housekeep.cpython-314.pyc +0 -0
  232. package/src/scripts/__pycache__/nexo-learning-validator.cpython-314.pyc +0 -0
  233. package/src/scripts/__pycache__/nexo-migrate.cpython-314.pyc +0 -0
  234. package/src/scripts/__pycache__/nexo-postmortem-consolidator.cpython-314.pyc +0 -0
  235. package/src/scripts/__pycache__/nexo-pre-commit.cpython-314.pyc +0 -0
  236. package/src/scripts/__pycache__/nexo-proactive-dashboard.cpython-314.pyc +0 -0
  237. package/src/scripts/__pycache__/nexo-reflection.cpython-314.pyc +0 -0
  238. package/src/scripts/__pycache__/nexo-runtime-preflight.cpython-314.pyc +0 -0
  239. package/src/scripts/__pycache__/nexo-send-email.cpython-314.pyc +0 -0
  240. package/src/scripts/__pycache__/nexo-send-reply.cpython-314.pyc +0 -0
  241. package/src/scripts/__pycache__/nexo-sleep.cpython-314.pyc +0 -0
  242. package/src/scripts/__pycache__/nexo-synthesis.cpython-314.pyc +0 -0
  243. package/src/scripts/__pycache__/nexo-watchdog-smoke.cpython-314.pyc +0 -0
  244. package/src/scripts/check-context 2.py +0 -264
  245. package/src/scripts/nexo-auto-update 2.py +0 -6
  246. package/src/scripts/nexo-backup 2.sh +0 -25
  247. package/src/scripts/nexo-brain-activation 2.sh +0 -140
  248. package/src/scripts/nexo-catchup 2.py +0 -242
  249. package/src/scripts/nexo-cognitive-decay 2.py +0 -182
  250. package/src/scripts/nexo-daily-self-audit 2.py +0 -552
  251. package/src/scripts/nexo-deep-sleep 2.sh +0 -97
  252. package/src/scripts/nexo-evolution-run 2.py +0 -597
  253. package/src/scripts/nexo-followup-hygiene 2.py +0 -112
  254. package/src/scripts/nexo-github-monitor 2.py +0 -256
  255. package/src/scripts/nexo-immune 2.py +0 -927
  256. package/src/scripts/nexo-inbox-hook 2.sh +0 -74
  257. package/src/scripts/nexo-install 2.py +0 -6
  258. package/src/scripts/nexo-learning-housekeep 2.py +0 -245
  259. package/src/scripts/nexo-learning-validator 2.py +0 -207
  260. package/src/scripts/nexo-migrate 2.py +0 -232
  261. package/src/scripts/nexo-postmortem-consolidator 2.py +0 -421
  262. package/src/scripts/nexo-pre-commit 2.py +0 -120
  263. package/src/scripts/nexo-prevent-sleep 2.sh +0 -29
  264. package/src/scripts/nexo-proactive-dashboard 2.py +0 -345
  265. package/src/scripts/nexo-reflection 2.py +0 -253
  266. package/src/scripts/nexo-runtime-preflight 2.py +0 -274
  267. package/src/scripts/nexo-send-email 2.py +0 -25
  268. package/src/scripts/nexo-send-email.py +0 -25
  269. package/src/scripts/nexo-send-reply 2.py +0 -178
  270. package/src/scripts/nexo-send-reply.py +0 -178
  271. package/src/scripts/nexo-sleep 2.py +0 -592
  272. package/src/scripts/nexo-snapshot-restore 2.sh +0 -35
  273. package/src/scripts/nexo-synthesis 2.py +0 -253
  274. package/src/scripts/nexo-tcc-approve 2.sh +0 -79
  275. package/src/scripts/nexo-update 2.sh +0 -161
  276. package/src/scripts/nexo-watchdog 2.sh +0 -878
  277. package/src/scripts/nexo-watchdog-smoke 2.py +0 -119
  278. package/src/server 2.py +0 -733
  279. package/src/storage_router 2.py +0 -32
  280. package/src/tools_coordination 2.py +0 -102
  281. package/src/tools_credentials 2.py +0 -68
  282. package/src/tools_learnings 2.py +0 -220
  283. package/src/tools_menu 2.py +0 -227
  284. package/src/tools_reminders 2.py +0 -86
  285. package/src/tools_reminders_crud 2.py +0 -159
  286. package/src/tools_sessions 2.py +0 -476
  287. package/src/tools_task_history 2.py +0 -57
  288. package/templates/CLAUDE.md 2.template +0 -63
  289. package/templates/openclaw 2.json +0 -13
  290. package/tests/__init__ 2.py +0 -0
  291. package/tests/__init__.py +0 -0
  292. package/tests/conftest 2.py +0 -71
  293. package/tests/conftest.py +0 -71
  294. package/tests/test_cognitive 2.py +0 -205
  295. package/tests/test_cognitive.py +0 -205
  296. package/tests/test_knowledge_graph 2.py +0 -140
  297. package/tests/test_knowledge_graph.py +0 -140
  298. package/tests/test_migrations 2.py +0 -137
  299. package/tests/test_migrations.py +0 -137
@@ -1,264 +0,0 @@
1
- #!/usr/bin/env python3
2
- """Context checker for NEXO operations - prevents duplicate actions.
3
-
4
- Mechanical checks (email sent, file exists, action done) run in Python.
5
- When the 'smart' command is used, passes context to Claude CLI for
6
- intelligent duplicate/conflict detection that goes beyond file checks.
7
- """
8
-
9
- import os
10
- import sys
11
- import json
12
- import hashlib
13
- import subprocess
14
- from datetime import datetime
15
- from pathlib import Path
16
-
17
- NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
18
-
19
- CLAUDE_CLI = Path.home() / ".local" / "bin" / "claude"
20
-
21
- class ContextChecker:
22
- def __init__(self):
23
- self.state_dir = NEXO_HOME / 'state'
24
- self.state_dir.mkdir(exist_ok=True)
25
-
26
- def check_email_sent(self, to_addr, subject, since_hours=72):
27
- """Check if email was already sent to address with subject."""
28
- sent_path = Path.home() / 'mail' / '.nexo-sent' / '.Sent' # Configure for your mail setup
29
- if not sent_path.exists():
30
- return False
31
-
32
- subject_lower = subject.lower()
33
- to_lower = to_addr.lower()
34
- cutoff = datetime.now().timestamp() - (since_hours * 3600)
35
- cur_dir = sent_path / 'cur'
36
- if not cur_dir.exists():
37
- return False
38
-
39
- for msg_file in cur_dir.iterdir():
40
- try:
41
- if msg_file.stat().st_mtime < cutoff:
42
- continue
43
- content = msg_file.read_text(errors='ignore')
44
- except (OSError, UnicodeDecodeError):
45
- continue
46
-
47
- content_lower = content.lower()
48
- if f"to:{to_lower}" in content_lower or f"to: {to_lower}" in content_lower:
49
- if subject_lower in content_lower:
50
- return True
51
- return False
52
-
53
- def check_file_exists(self, pattern, search_dirs=None):
54
- """Check if file matching pattern exists in common locations."""
55
- if search_dirs is None:
56
- search_dirs = [
57
- '/var/www/vhosts',
58
- str(NEXO_HOME),
59
- '/opt'
60
- ]
61
-
62
- for base_dir in search_dirs:
63
- if not os.path.exists(base_dir):
64
- continue
65
- matches = []
66
- try:
67
- for root, _, files in os.walk(base_dir):
68
- for filename in files:
69
- if pattern in filename:
70
- matches.append(str(Path(root) / filename))
71
- if len(matches) >= 5:
72
- return matches
73
- except OSError:
74
- continue
75
- return []
76
-
77
- def check_action_done(self, action_type, identifier, ttl_days=7):
78
- """Check if action was already performed recently."""
79
- action_file = self.state_dir / 'actions.json'
80
-
81
- # Load existing actions
82
- actions = {}
83
- if action_file.exists():
84
- with open(action_file) as f:
85
- actions = json.load(f)
86
-
87
- # Create action key
88
- key = hashlib.md5(f"{action_type}:{identifier}".encode()).hexdigest()
89
-
90
- # Check if exists and not expired
91
- if key in actions:
92
- action_time = datetime.fromisoformat(actions[key]['timestamp'])
93
- age_days = (datetime.now() - action_time).days
94
- if age_days < ttl_days:
95
- return True, actions[key]
96
-
97
- return False, None
98
-
99
- def mark_action_done(self, action_type, identifier, metadata=None):
100
- """Mark action as completed."""
101
- action_file = self.state_dir / 'actions.json'
102
-
103
- # Load existing actions
104
- actions = {}
105
- if action_file.exists():
106
- with open(action_file) as f:
107
- actions = json.load(f)
108
-
109
- # Add new action
110
- key = hashlib.md5(f"{action_type}:{identifier}".encode()).hexdigest()
111
- actions[key] = {
112
- 'type': action_type,
113
- 'identifier': identifier,
114
- 'timestamp': datetime.now().isoformat(),
115
- 'metadata': metadata or {}
116
- }
117
-
118
- # Save
119
- with open(action_file, 'w') as f:
120
- json.dump(actions, f, indent=2)
121
-
122
- return key
123
-
124
- def smart_check(action_description: str, context: str = "") -> dict:
125
- """Use Claude CLI to intelligently check if an action would be redundant.
126
-
127
- Goes beyond simple file/hash checks — understands intent and context
128
- to detect semantic duplicates (e.g., "send welcome email" vs
129
- "email onboarding message" to same person).
130
- """
131
- checker = ContextChecker()
132
-
133
- # Gather mechanical context first
134
- state_file = checker.state_dir / 'actions.json'
135
- recent_actions = {}
136
- if state_file.exists():
137
- try:
138
- all_actions = json.loads(state_file.read_text())
139
- cutoff = datetime.now().timestamp() - (7 * 86400)
140
- for k, v in all_actions.items():
141
- try:
142
- ts = datetime.fromisoformat(v['timestamp']).timestamp()
143
- if ts > cutoff:
144
- recent_actions[k] = v
145
- except (ValueError, KeyError):
146
- pass
147
- except Exception:
148
- pass
149
-
150
- if not CLAUDE_CLI.exists():
151
- return {"redundant": False, "reason": "CLI unavailable, cannot smart-check"}
152
-
153
- prompt = f"""You are a context deduplication engine for NEXO operations.
154
-
155
- PROPOSED ACTION:
156
- {action_description}
157
-
158
- ADDITIONAL CONTEXT:
159
- {context or "None"}
160
-
161
- RECENT ACTIONS (last 7 days):
162
- {json.dumps(list(recent_actions.values()), indent=1, default=str)}
163
-
164
- Respond with ONLY valid JSON (no markdown):
165
- {{
166
- "redundant": true/false,
167
- "confidence": 0.0-1.0,
168
- "reason": "<one line explanation>",
169
- "matching_action": "<identifier of matching action if redundant, else null>"
170
- }}
171
-
172
- Rules:
173
- - Same recipient + same intent within 72h = redundant
174
- - Same file modification with same content = redundant
175
- - Similar but different scope (e.g., different recipients) = NOT redundant
176
- - When in doubt, say not redundant (false negatives are cheaper than false positives)"""
177
- )
178
- if auth_check.returncode != 0:
179
- # CLI not authenticated, skip gracefully
180
- return {"redundant": False, "reason": "CLI not authenticated — skipped analysis", "suggestion": "N/A"}
181
-
182
- env = os.environ.copy()
183
- env["NEXO_HEADLESS"] = "1" # Skip stop hook post-mortem
184
- env.pop("CLAUDECODE", None)
185
- env.pop("CLAUDE_CODE", None)
186
-
187
- try:
188
- result = subprocess.run(
189
- [str(CLAUDE_CLI), "-p", prompt, "--model", "opus", "--output-format", "text",
190
- "--allowedTools", "Read,Write,Edit,Glob,Grep,Bash,mcp__nexo__*"],
191
- capture_output=True, text=True, timeout=21600, env=env
192
- )
193
- if result.returncode == 0:
194
- text = result.stdout.strip()
195
- if "```json" in text:
196
- text = text.split("```json")[1].split("```")[0]
197
- elif "```" in text:
198
- text = text.split("```")[1].split("```")[0]
199
- return json.loads(text.strip())
200
- except Exception:
201
- pass
202
-
203
- return {"redundant": False, "reason": "CLI check failed, defaulting to not redundant"}
204
-
205
-
206
- def main():
207
- """CLI interface for context checking."""
208
- if len(sys.argv) < 3:
209
- print("Usage: check-context.py <command> <args>")
210
- print("Commands:")
211
- print(" email <to> <subject> - Check if email was sent")
212
- print(" file <pattern> - Check if file exists")
213
- print(" action <type> <id> - Check if action was done")
214
- print(" smart <description> [ctx] - Intelligent duplicate check via CLI")
215
- sys.exit(1)
216
-
217
- checker = ContextChecker()
218
- command = sys.argv[1]
219
-
220
- if command == 'email':
221
- if len(sys.argv) < 4:
222
- print("Usage: check-context.py email <to> <subject>")
223
- sys.exit(1)
224
- exists = checker.check_email_sent(sys.argv[2], sys.argv[3])
225
- print("EXISTS" if exists else "NOT_FOUND")
226
- sys.exit(0 if not exists else 1)
227
-
228
- elif command == 'file':
229
- files = checker.check_file_exists(sys.argv[2])
230
- if files:
231
- print("\n".join(files))
232
- sys.exit(1)
233
- else:
234
- print("NOT_FOUND")
235
- sys.exit(0)
236
-
237
- elif command == 'action':
238
- if len(sys.argv) < 4:
239
- print("Usage: check-context.py action <type> <id>")
240
- sys.exit(1)
241
- done, data = checker.check_action_done(sys.argv[2], sys.argv[3])
242
- if done:
243
- print(f"DONE: {data}")
244
- sys.exit(1)
245
- else:
246
- print("NOT_DONE")
247
- sys.exit(0)
248
-
249
- elif command == 'smart':
250
- if len(sys.argv) < 3:
251
- print("Usage: check-context.py smart <description> [context]")
252
- sys.exit(1)
253
- description = sys.argv[2]
254
- context = sys.argv[3] if len(sys.argv) > 3 else ""
255
- result = smart_check(description, context)
256
- print(json.dumps(result, indent=2))
257
- sys.exit(1 if result.get("redundant") else 0)
258
-
259
- else:
260
- print(f"Unknown command: {command}")
261
- sys.exit(1)
262
-
263
- if __name__ == '__main__':
264
- main()
@@ -1,6 +0,0 @@
1
- #!/usr/bin/env python3
2
- """DEPRECATED: Updates are handled automatically by NEXO on startup."""
3
- import sys
4
- print("This script is deprecated. NEXO auto-updates on startup.")
5
- print("To update manually, use the nexo_update MCP tool.")
6
- sys.exit(0)
@@ -1,25 +0,0 @@
1
- #!/bin/bash
2
- # NEXO DB hourly backup — crontab: 0 * * * * $NEXO_HOME/scripts/nexo-backup.sh
3
- NEXO_HOME="${NEXO_HOME:-$HOME/.nexo}"
4
- NEXO_DIR="$NEXO_HOME"
5
- BACKUP_DIR="$NEXO_HOME/backups"
6
- WEEKLY_DIR="$BACKUP_DIR/weekly"
7
- DB="$NEXO_HOME/data/nexo.db"
8
- RETENTION_HOURS=48
9
-
10
- mkdir -p "$BACKUP_DIR" "$WEEKLY_DIR"
11
-
12
- # Hourly backup
13
- TIMESTAMP=$(date +%Y-%m-%d-%H%M)
14
- sqlite3 "$DB" ".backup '$BACKUP_DIR/nexo-$TIMESTAMP.db'"
15
-
16
- # Weekly backup — save one per week (Sundays)
17
- WEEK=$(date +%Y-W%V)
18
- WEEKLY_FILE="$WEEKLY_DIR/weekly-$WEEK.db"
19
- if [ ! -f "$WEEKLY_FILE" ] && [ "$(date +%u)" = "7" ]; then
20
- cp "$BACKUP_DIR/nexo-$TIMESTAMP.db" "$WEEKLY_FILE"
21
- fi
22
-
23
- # Cleanup: hourly >48h, weekly >90 days
24
- find "$BACKUP_DIR" -maxdepth 1 -name "nexo-*.db" -mmin +$((RETENTION_HOURS * 60)) -delete
25
- find "$WEEKLY_DIR" -name "weekly-*.db" -mtime +90 -delete
@@ -1,140 +0,0 @@
1
- #!/usr/bin/env bash
2
- # nexo-brain-activation.sh — NF24: Spontaneous Activation
3
- # Reads user_model.json, detects significant shifts, and generates insights
4
- # for NEXO startup. If nothing relevant: exit 0 with no output.
5
-
6
- set -euo pipefail
7
-
8
- BRAIN_DIR="~/.nexo/brain"
9
- MODEL_FILE="$BRAIN_DIR/user_model.json"
10
- SUMMARIES_DIR="$BRAIN_DIR/daily_summaries"
11
-
12
- # --- Guard: required file ---
13
- if [[ ! -f "$MODEL_FILE" ]]; then
14
- exit 0
15
- fi
16
-
17
- # --- Inline Python to parse JSON and analyze ---
18
- python3 - <<'PYEOF'
19
- import json
20
- import sys
21
- import os
22
- import glob
23
- from datetime import datetime, timedelta
24
-
25
- BRAIN_DIR = os.path.expanduser("~/.nexo/brain")
26
- MODEL_FILE = os.path.join(BRAIN_DIR, "user_model.json")
27
- SUMMARIES_DIR = os.path.join(BRAIN_DIR, "daily_summaries")
28
-
29
- # ── Load model ──────────────────────────────────────────────────────────
30
- try:
31
- with open(MODEL_FILE) as f:
32
- model = json.load(f)
33
- except Exception:
34
- sys.exit(0)
35
-
36
- insights = []
37
-
38
- # ── 1. Analyze evolution_log — last 3-5 entries ─────────────────────
39
- evolution = model.get("evolution_log", [])
40
- recent = evolution[-5:] if len(evolution) >= 5 else evolution
41
-
42
- # Detect entries from the last 3 days
43
- today = datetime.now().date()
44
- cutoff = today - timedelta(days=3)
45
-
46
- recent_obs = []
47
- for entry in recent:
48
- try:
49
- entry_date = datetime.strptime(entry["date"], "%Y-%m-%d").date()
50
- if entry_date >= cutoff:
51
- recent_obs.append(entry)
52
- except Exception:
53
- pass
54
-
55
- if recent_obs:
56
- for obs in recent_obs[-2:]: # max 2 most recent
57
- insights.append(obs["observation"].strip())
58
-
59
- # ── 2. Detect trait changes >0.1 between entries (if history exists) ──
60
- # The current model only has a snapshot; if in the future there is history
61
- # in evolution_log with trait deltas, this can be expanded here.
62
- # Detect the most extreme trait as the dominant identity signal.
63
- traits = model.get("traits", {})
64
- if traits:
65
- dominant = max(traits, key=lambda k: traits[k])
66
- dominant_val = traits[dominant]
67
- weakest = min(traits, key=lambda k: traits[k])
68
- weakest_val = traits[weakest]
69
- # Only report if extreme (>0.9 or <0.2) to avoid noise
70
- if dominant_val >= 0.9:
71
- insights.append(f"Dominant trait: {dominant}={dominant_val} (highest recorded)")
72
- if weakest_val <= 0.2:
73
- insights.append(f"Lowest trait: {weakest}={weakest_val} (low tolerance active)")
74
-
75
- # ── 3. Recent contradictions (last 3 days) ─────────────────────────
76
- contradictions = model.get("contradictions", [])
77
- recent_contradictions = []
78
- for c in contradictions:
79
- try:
80
- c_date = datetime.strptime(c["date"], "%Y-%m-%d").date()
81
- if c_date >= cutoff:
82
- recent_contradictions.append(c)
83
- except Exception:
84
- pass
85
-
86
- for c in recent_contradictions:
87
- insights.append(f"Contradiction detected: {c['description'].strip()}")
88
-
89
- # ── 4. Current active focus ─────────────────────────────────────────────────
90
- current_focus = model.get("current_focus", [])
91
- # Only include if there are focus items (and it's not initial startup)
92
- if len(current_focus) > 0 and len(insights) == 0:
93
- # If no other insights, report focus as minimum context
94
- focus_str = ", ".join(current_focus)
95
- insights.append(f"Current focus: {focus_str}")
96
-
97
- # ── 5. Goals: active/dormant changes ─────────────────────────────────────
98
- goals_active = model.get("goals_active", [])
99
- goals_dormant = model.get("goals_dormant", [])
100
- # If there are dormant goals, mention them (may need reactivation)
101
- if goals_dormant:
102
- insights.append(f"Goal dormant: {goals_dormant[0]}")
103
-
104
- # ── Read last daily summary ─────────────────────────────────────────────
105
- summary_text = ""
106
- try:
107
- files = sorted(glob.glob(os.path.join(SUMMARIES_DIR, "*.md")))
108
- if files:
109
- with open(files[-1]) as f:
110
- lines = [l.rstrip() for l in f.readlines() if l.strip()]
111
- # Take up to 3 lines of content (exclude header)
112
- content_lines = [l for l in lines if not l.startswith("# Resumen") and not l.startswith("# Summary")]
113
- summary_text = " | ".join(content_lines[:3])
114
- except Exception:
115
- pass
116
-
117
- # ── Output ────────────────────────────────────────────────────────────────
118
- # If no real insights, exit without output
119
- if not insights and not summary_text:
120
- sys.exit(0)
121
-
122
- # Deduplicate and limit
123
- seen = set()
124
- unique_insights = []
125
- for ins in insights:
126
- key = ins[:60]
127
- if key not in seen:
128
- seen.add(key)
129
- unique_insights.append(ins)
130
-
131
- if unique_insights:
132
- print("BRAIN_INSIGHTS:")
133
- for ins in unique_insights[:5]:
134
- print(f"- {ins}")
135
-
136
- if summary_text:
137
- print("RECENT_SUMMARY:")
138
- print(summary_text[:400])
139
-
140
- PYEOF
@@ -1,242 +0,0 @@
1
- #!/usr/bin/env python3
2
- """
3
- NEXO Catch-Up — Runs at Mac boot to execute any missed scheduled tasks.
4
-
5
- When the Mac was asleep/off during scheduled times, launchd does NOT retry
6
- missed StartCalendarInterval jobs. This script detects what was missed and
7
- runs them in the correct order.
8
-
9
- Scheduled tasks (ordered by intended run time):
10
- 03:00 — cognitive-decay (Ebbinghaus decay + STM→LTM promotion)
11
- 03:00 — evolution (weekly, Sundays only)
12
- 04:00 — sleep (session cleanup)
13
- 07:00 — self-audit (health checks + weekly cognitive GC on Sundays)
14
- 23:30 — postmortem (consolidation + sensory register)
15
-
16
- Logic: For each task, check if its last successful run was before the
17
- most recent scheduled time. If so, run it now.
18
- """
19
-
20
- import json
21
- import os
22
- import subprocess
23
- import sys
24
- from datetime import datetime, timedelta
25
- from pathlib import Path
26
-
27
- CLAUDE_CLI = Path.home() / ".local" / "bin" / "claude"
28
-
29
- HOME = Path.home()
30
- NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
31
- LOG_DIR = NEXO_HOME / "logs"
32
- LOG_DIR.mkdir(parents=True, exist_ok=True)
33
- LOG_FILE = LOG_DIR / "catchup.log"
34
- STATE_FILE = NEXO_HOME / "operations" / ".catchup-state.json"
35
-
36
- SCRIPTS = NEXO_HOME / "scripts"
37
-
38
- # Resolve Python: prefer NEXO's venv, then the same Python running this script
39
- def _resolve_python() -> str:
40
- """Find the best Python to use for subprocess calls."""
41
- # Check for NEXO_CODE env var pointing to the repo's src/
42
- nexo_code = os.environ.get("NEXO_CODE", "")
43
- if nexo_code:
44
- venv_python = Path(nexo_code).parent / ".venv" / "bin" / "python"
45
- if venv_python.exists():
46
- return str(venv_python)
47
- # Check for venv relative to NEXO_HOME
48
- venv_python = NEXO_HOME / ".venv" / "bin" / "python"
49
- if venv_python.exists():
50
- return str(venv_python)
51
- # Fall back to the same Python running this script
52
- return sys.executable
53
-
54
- NEXO_PYTHON = _resolve_python()
55
-
56
-
57
- def log(msg: str):
58
- ts = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
59
- line = f"[{ts}] {msg}"
60
- print(line, flush=True)
61
- with open(LOG_FILE, "a") as f:
62
- f.write(line + "\n")
63
-
64
-
65
- def load_state() -> dict:
66
- if STATE_FILE.exists():
67
- try:
68
- return json.loads(STATE_FILE.read_text())
69
- except Exception:
70
- pass
71
- return {}
72
-
73
-
74
- def save_state(state: dict):
75
- STATE_FILE.parent.mkdir(parents=True, exist_ok=True)
76
- STATE_FILE.write_text(json.dumps(state, indent=2))
77
-
78
-
79
- def last_scheduled_time(hour: int, minute: int, weekday: int = None) -> datetime:
80
- """Calculate the most recent time this task should have run."""
81
- now = datetime.now()
82
- today_at = now.replace(hour=hour, minute=minute, second=0, microsecond=0)
83
-
84
- if weekday is not None:
85
- # Weekly task — find the most recent matching weekday
86
- days_since = (now.weekday() - weekday) % 7
87
- target = now - timedelta(days=days_since)
88
- target = target.replace(hour=hour, minute=minute, second=0, microsecond=0)
89
- if target > now:
90
- target -= timedelta(weeks=1)
91
- return target
92
-
93
- # Daily task
94
- if today_at <= now:
95
- return today_at
96
- else:
97
- return today_at - timedelta(days=1)
98
-
99
-
100
- def should_run(task_name: str, hour: int, minute: int, state: dict, weekday: int = None) -> bool:
101
- """Check if task needs catch-up: last run was before last scheduled time."""
102
- last_run_str = state.get(task_name)
103
- last_scheduled = last_scheduled_time(hour, minute, weekday)
104
-
105
- if not last_run_str:
106
- # Never ran — should run
107
- return True
108
-
109
- try:
110
- last_run = datetime.fromisoformat(last_run_str)
111
- except ValueError:
112
- return True
113
-
114
- return last_run < last_scheduled
115
-
116
-
117
- def run_task(name: str, python: str, script: str, state: dict) -> bool:
118
- """Execute a task and update state."""
119
- script_path = str(SCRIPTS / script)
120
- if not Path(script_path).exists():
121
- log(f" SKIP {name}: script not found ({script_path})")
122
- return False
123
-
124
- log(f" RUNNING {name}: {script}")
125
- try:
126
- result = subprocess.run(
127
- [python, script_path],
128
- capture_output=True, text=True, timeout=21600,
129
- env={**os.environ, "HOME": str(HOME), "NEXO_CATCHUP": "1"}
130
- )
131
- if result.returncode == 0:
132
- log(f" OK {name} (exit 0)")
133
- else:
134
- log(f" WARN {name} (exit {result.returncode})")
135
- if result.stderr:
136
- log(f" stderr: {result.stderr[:300]}")
137
- state[name] = datetime.now().isoformat()
138
- save_state(state)
139
- return True
140
- except subprocess.TimeoutExpired:
141
- log(f" TIMEOUT {name} (300s)")
142
- return False
143
- except Exception as e:
144
- log(f" ERROR {name}: {e}")
145
- return False
146
-
147
-
148
- def main():
149
- log("=== NEXO Catch-Up starting (boot/wake) ===")
150
- state = load_state()
151
-
152
- # Define tasks in execution order (matching their intended schedule order)
153
- # Note: auto-update is handled by the MCP server on startup, not by catchup.
154
- tasks = [
155
- # (name, hour, minute, python, script, weekday)
156
- ("cognitive-decay", 3, 0, NEXO_PYTHON, "nexo-cognitive-decay.py", None),
157
- ("evolution", 3, 0, NEXO_PYTHON, "nexo-evolution-run.py", 6), # Sunday = 6
158
- ("sleep", 4, 0, NEXO_PYTHON, "nexo-sleep.py", None),
159
- ("self-audit", 7, 0, NEXO_PYTHON, "nexo-daily-self-audit.py", None),
160
- ("github-monitor", 8, 0, NEXO_PYTHON, "nexo-github-monitor.py", None),
161
- ("postmortem", 23, 30, NEXO_PYTHON, "nexo-postmortem-consolidator.py", None),
162
- ]
163
-
164
- ran = 0
165
- skipped = 0
166
- for name, hour, minute, python, script, weekday in tasks:
167
- if should_run(name, hour, minute, state, weekday):
168
- log(f" {name} — missed scheduled run, catching up...")
169
- if run_task(name, python, script, state):
170
- ran += 1
171
- else:
172
- skipped += 1
173
-
174
- if ran == 0:
175
- log("All tasks up to date, nothing to catch up.")
176
- elif ran >= 3:
177
- # Many tasks caught up — ask CLI to assess system state
178
- _cli_post_catchup_assessment(ran, skipped, state)
179
- else:
180
- log(f"Caught up {ran} tasks, {skipped} already current.")
181
-
182
- log("=== Catch-Up complete ===")
183
-
184
-
185
- def _cli_post_catchup_assessment(ran: int, skipped: int, state: dict):
186
- """When 3+ tasks were missed, use CLI to assess if there are concerns."""
187
- if not CLAUDE_CLI.exists():
188
- log(f"Caught up {ran} tasks, {skipped} already current. (CLI unavailable for assessment)")
189
- return
190
- )
191
- if auth_check.returncode != 0:
192
- # CLI not authenticated, skip gracefully
193
- print(f"[{datetime.now().strftime('%H:%M:%S')}] Claude CLI not authenticated. Skipping CLI analysis.")
194
- return
195
-
196
- assessment_file = LOG_DIR / "catchup-assessment.md"
197
- state_summary = json.dumps(state, indent=2, default=str)
198
-
199
- prompt = f"""You are the NEXO Catch-Up system. The Mac was off/asleep and {ran} scheduled tasks just ran as catch-up ({skipped} were already current).
200
-
201
- Task run state (timestamps of last successful runs):
202
- {state_summary}
203
-
204
- Assess:
205
- 1. How long was the system likely offline? (compare timestamps to now)
206
- 2. Are there any tasks that depend on each other where order matters?
207
- 3. Any tasks that may have produced stale results because they ran late?
208
- 4. Should any task be re-run at its normal time today?
209
-
210
- Write a brief assessment (max 20 lines) to: {assessment_file}
211
-
212
- Format:
213
- ## Catch-Up Assessment — {datetime.now().strftime('%Y-%m-%d %H:%M')}
214
- - Offline duration: ~Xh
215
- - Tasks caught up: {ran}
216
- - Concerns: ...
217
- - Recommendation: ..."""
218
-
219
- log(f"Caught up {ran} tasks — running CLI assessment...")
220
- env = os.environ.copy()
221
- env["NEXO_HEADLESS"] = "1" # Skip stop hook post-mortem
222
- env.pop("CLAUDECODE", None)
223
- env.pop("CLAUDE_CODE", None)
224
-
225
- try:
226
- result = subprocess.run(
227
- [str(CLAUDE_CLI), "-p", prompt, "--model", "opus", "--output-format", "text",
228
- "--allowedTools", "Read,Write,Edit,Glob,Grep,Bash,mcp__nexo__*"],
229
- capture_output=True, text=True, timeout=21600, env=env
230
- )
231
- if result.returncode == 0:
232
- log(f"Assessment written to {assessment_file}")
233
- else:
234
- log(f"CLI assessment exited {result.returncode}")
235
- except subprocess.TimeoutExpired:
236
- log("CLI assessment timed out (90s)")
237
- except Exception as e:
238
- log(f"CLI assessment error: {e}")
239
-
240
-
241
- if __name__ == "__main__":
242
- main()