nexo-brain 5.3.20 → 5.3.21
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/.claude-plugin/plugin.json +1 -1
- package/package.json +1 -1
- package/src/auto_update.py +11 -8
- package/src/dashboard/static/favicon 2.svg +32 -0
- package/src/dashboard/static/nexo-logo 2.png +0 -0
- package/src/dashboard/static/nexo-logo 2.svg +40 -0
- package/src/dashboard/static/style 2.css +2458 -0
- package/src/dashboard/templates/adaptive 2.html +118 -0
- package/src/dashboard/templates/artifacts 2.html +133 -0
- package/src/dashboard/templates/backups 2.html +136 -0
- package/src/dashboard/templates/base 2.html +417 -0
- package/src/dashboard/templates/calendar 2.html +591 -0
- package/src/dashboard/templates/chat 2.html +356 -0
- package/src/dashboard/templates/claims 2.html +259 -0
- package/src/dashboard/templates/cortex 2.html +321 -0
- package/src/dashboard/templates/credentials 2.html +128 -0
- package/src/dashboard/templates/crons 2.html +370 -0
- package/src/dashboard/templates/dashboard 2.html +494 -0
- package/src/dashboard/templates/dreams 2.html +252 -0
- package/src/dashboard/templates/email 2.html +160 -0
- package/src/dashboard/templates/evolution 2.html +189 -0
- package/src/dashboard/templates/feed 2.html +249 -0
- package/src/dashboard/templates/followup_health 2.html +170 -0
- package/src/dashboard/templates/graph 2.html +201 -0
- package/src/dashboard/templates/guard 2.html +259 -0
- package/src/dashboard/templates/inbox 2.html +251 -0
- package/src/dashboard/templates/memory 2.html +420 -0
- package/src/dashboard/templates/operations 2.html +608 -0
- package/src/dashboard/templates/plugins 2.html +185 -0
- package/src/dashboard/templates/protocol 2.html +199 -0
- package/src/dashboard/templates/rules 2.html +246 -0
- package/src/dashboard/templates/sentiment 2.html +247 -0
- package/src/dashboard/templates/sessions 2.html +218 -0
- package/src/dashboard/templates/skills 2.html +329 -0
- package/src/dashboard/templates/somatic 2.html +73 -0
- package/src/dashboard/templates/triggers 2.html +133 -0
- package/src/dashboard/templates/trust 2.html +360 -0
- package/src/db/__init__ 2.py +259 -0
- package/src/db/_core 2.py +437 -0
- package/src/db/_credentials 2.py +124 -0
- package/src/db/_episodic 2.py +762 -0
- package/src/db/_evolution 2.py +54 -0
- package/src/db/_fts 2.py +406 -0
- package/src/db/_goal_profiles 2.py +376 -0
- package/src/db/_hot_context 2.py +660 -0
- package/src/db/_outcomes 2.py +800 -0
- package/src/db/_personal_scripts 2.py +582 -0
- package/src/db/_sessions 2.py +330 -0
- package/src/db/_tasks 2.py +91 -0
- package/src/db/_watchers 2.py +173 -0
- package/src/doctor/formatters 2.py +52 -0
- package/src/doctor/models 2.py +69 -0
- package/src/doctor/planes 2.py +87 -0
- package/src/doctor/providers/__init__ 2.py +1 -0
- package/src/doctor/providers/deep 2.py +367 -0
- package/src/evolution_cycle 2.py +519 -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-tool-logs 2.sh +158 -0
- package/src/hooks/daily-briefing-check 2.sh +33 -0
- package/src/hooks/heartbeat-enforcement 2.py +90 -0
- package/src/hooks/heartbeat-posttool 2.sh +18 -0
- package/src/hooks/inbox-hook 2.sh +76 -0
- package/src/hooks/post-compact 2.sh +152 -0
- package/src/hooks/pre-compact 2.sh +169 -0
- package/src/hooks/protocol-guardrail 2.sh +10 -0
- package/src/hooks/protocol-pretool-guardrail 2.sh +9 -0
- package/src/hooks/session-stop 2.sh +52 -0
- package/src/kg_populate 2.py +292 -0
- package/src/maintenance 2.py +53 -0
- package/src/memory_backends 2.py +71 -0
- package/src/migrate_embeddings 2.py +124 -0
- package/src/nexo_sdk 2.py +103 -0
- package/src/observability 2.py +199 -0
- package/src/plugin_loader 2.py +217 -0
- package/src/plugins/__init__ 2.py +0 -0
- package/src/plugins/artifact_registry 2.py +450 -0
- package/src/plugins/backup 2.py +127 -0
- package/src/plugins/claims_tools 2.py +119 -0
- package/src/plugins/cognitive_memory 2.py +609 -0
- package/src/plugins/core_rules 2.py +252 -0
- package/src/plugins/cortex 2.py +1155 -0
- package/src/plugins/entities 2.py +67 -0
- package/src/plugins/episodic_memory 2.py +560 -0
- package/src/plugins/evolution 2.py +167 -0
- package/src/plugins/goal_engine 2.py +142 -0
- package/src/plugins/guard 2.py +862 -0
- package/src/plugins/impact 2.py +29 -0
- package/src/plugins/knowledge_graph_tools 2.py +137 -0
- package/src/plugins/media_memory_tools 2.py +98 -0
- package/src/plugins/memory_export 2.py +196 -0
- package/src/plugins/outcomes 2.py +130 -0
- package/src/plugins/personal_scripts 2.py +117 -0
- package/src/plugins/preferences 2.py +47 -0
- package/src/plugins/protocol 2.py +1449 -0
- package/src/plugins/simple_api 2.py +106 -0
- package/src/plugins/skills 2.py +341 -0
- package/src/plugins/state_watchers 2.py +79 -0
- package/src/plugins/update 2.py +986 -0
- package/src/plugins/user_state_tools 2.py +43 -0
- package/src/plugins/workflow 2.py +588 -0
- package/src/protocol_settings 2.py +59 -0
- package/src/public_contribution 2.py +466 -0
- package/src/public_evolution_queue 2.py +241 -0
- package/src/requirements 2.txt +14 -0
- package/src/retroactive_learnings 2.py +373 -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/runtime_power 2.py +874 -0
- package/src/script_registry 2.py +1559 -0
- package/src/scripts/check-context 2.py +272 -0
- package/src/scripts/deep-sleep/apply_findings 2.py +2327 -0
- package/src/scripts/deep-sleep/collect 2.py +928 -0
- package/src/scripts/deep-sleep/extract 2.py +330 -0
- package/src/scripts/deep-sleep/extract-prompt 2.md +285 -0
- package/src/scripts/deep-sleep/synthesize 2.py +312 -0
- package/src/scripts/deep-sleep/synthesize-prompt 2.md +336 -0
- package/src/scripts/nexo-agent-run 2.py +75 -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 +300 -0
- package/src/scripts/nexo-cognitive-decay 2.py +257 -0
- package/src/scripts/nexo-cortex-cycle 2.py +293 -0
- package/src/scripts/nexo-cron-wrapper 2.sh +53 -0
- package/src/scripts/nexo-daily-self-audit 2.py +2161 -0
- package/src/scripts/nexo-dashboard 2.sh +29 -0
- package/src/scripts/nexo-deep-sleep 2.sh +86 -0
- package/src/scripts/nexo-evolution-run 2.py +1664 -0
- package/src/scripts/nexo-followup-hygiene 2.py +139 -0
- package/src/scripts/nexo-hook-record 2.py +42 -0
- package/src/scripts/nexo-immune 2.py +936 -0
- package/src/scripts/nexo-impact-scorer 2.py +117 -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 +401 -0
- package/src/scripts/nexo-learning-validator 2.py +266 -0
- package/src/scripts/nexo-migrate 2.py +260 -0
- package/src/scripts/nexo-outcome-checker 2.py +127 -0
- package/src/scripts/nexo-postmortem-consolidator 2.py +456 -0
- package/src/scripts/nexo-pre-commit 2.py +120 -0
- package/src/scripts/nexo-prevent-sleep 2.sh +35 -0
- package/src/scripts/nexo-proactive-dashboard 2.py +354 -0
- package/src/scripts/nexo-reflection 2.py +256 -0
- package/src/scripts/nexo-runtime-preflight 2.py +274 -0
- package/src/scripts/nexo-sleep 2.py +631 -0
- package/src/scripts/nexo-snapshot-restore 2.sh +35 -0
- package/src/scripts/nexo-sync-clients 2.py +16 -0
- package/src/scripts/nexo-synthesis 2.py +475 -0
- package/src/scripts/nexo-tcc-approve 2.sh +79 -0
- package/src/scripts/nexo-update 2.sh +306 -0
- package/src/scripts/nexo-watchdog 2.sh +1207 -0
- package/src/scripts/nexo-watchdog-smoke 2.py +119 -0
- package/src/scripts/rehydrate_learnings_from_archive 2.py +245 -0
- package/src/server 2.py +1296 -0
- package/src/skills/run-nexo-audit-phase/guide 2.md +43 -0
- package/src/skills/run-nexo-audit-phase/skill 2.json +59 -0
- package/src/skills/run-nexo-core-fix-cycle/guide 2.md +17 -0
- package/src/skills/run-nexo-core-fix-cycle/script 2.py +276 -0
- package/src/skills/run-nexo-core-fix-cycle/skill 2.json +58 -0
- package/src/skills/run-release-final-audit/guide 2.md +16 -0
- package/src/skills/run-release-final-audit/script 2.py +259 -0
- package/src/skills/run-release-final-audit/skill 2.json +77 -0
- package/src/skills/run-runtime-doctor/guide 2.md +12 -0
- package/src/skills/run-runtime-doctor/script 2.py +21 -0
- package/src/skills/run-runtime-doctor/skill 2.json +25 -0
- package/src/skills_runtime 2.py +932 -0
- package/src/state_watchers_runtime 2.py +475 -0
- package/src/storage_router 2.py +32 -0
- package/src/system_catalog 2.py +786 -0
- package/src/tools_coordination 2.py +103 -0
- package/src/tools_credentials 2.py +68 -0
- package/src/tools_drive 2.py +487 -0
- package/src/tools_hot_context 2.py +163 -0
- package/src/tools_learnings 2.py +612 -0
- package/src/tools_menu 2.py +229 -0
- package/src/tools_reminders 2.py +88 -0
- package/src/tools_reminders_crud 2.py +363 -0
- package/src/tools_sessions 2.py +1054 -0
- package/src/tools_system_catalog 2.py +19 -0
- package/src/tools_task_history 2.py +57 -0
- package/src/tools_transcripts 2.py +98 -0
- package/src/transcript_utils 2.py +412 -0
- package/src/user_context 2.py +46 -0
- package/src/user_data_portability 2.py +328 -0
- package/src/user_state_model 2.py +170 -0
- package/templates/CLAUDE.md 2.template +108 -0
- package/templates/CODEX.AGENTS.md 2.template +66 -0
- package/templates/launchagents/README 2.md +132 -0
- package/templates/launchagents/com.nexo.auto-close-sessions 2.plist +39 -0
- package/templates/launchagents/com.nexo.catchup 2.plist +39 -0
- package/templates/launchagents/com.nexo.cognitive-decay 2.plist +40 -0
- package/templates/launchagents/com.nexo.dashboard 2.plist +43 -0
- package/templates/launchagents/com.nexo.deep-sleep 2.plist +43 -0
- package/templates/launchagents/com.nexo.evolution 2.plist +44 -0
- package/templates/launchagents/com.nexo.followup-hygiene 2.plist +45 -0
- package/templates/launchagents/com.nexo.immune 2.plist +41 -0
- package/templates/launchagents/com.nexo.postmortem 2.plist +45 -0
- package/templates/launchagents/com.nexo.self-audit 2.plist +47 -0
- package/templates/launchagents/com.nexo.synthesis 2.plist +45 -0
- package/templates/launchagents/com.nexo.watchdog 2.plist +37 -0
- package/templates/nexo_helper 2.py +301 -0
- package/templates/openclaw 2.json +13 -0
- package/templates/plugin-template 2.py +40 -0
- package/templates/script-template 2.py +59 -0
- package/templates/script-template 2.sh +13 -0
- package/templates/skill-script-template 2.py +48 -0
- package/templates/skill-template 2.md +33 -0
|
@@ -0,0 +1,272 @@
|
|
|
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, NEXO asks the configured automation backend
|
|
6
|
+
for semantic duplicate/conflict detection that goes beyond file checks.
|
|
7
|
+
"""
|
|
8
|
+
|
|
9
|
+
from __future__ import annotations
|
|
10
|
+
|
|
11
|
+
import hashlib
|
|
12
|
+
import json
|
|
13
|
+
import os
|
|
14
|
+
import sys
|
|
15
|
+
from datetime import datetime
|
|
16
|
+
from pathlib import Path
|
|
17
|
+
|
|
18
|
+
NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
|
|
19
|
+
NEXO_CODE = Path(os.environ.get("NEXO_CODE", str(Path(__file__).resolve().parents[1])))
|
|
20
|
+
if str(NEXO_CODE) not in sys.path:
|
|
21
|
+
sys.path.insert(0, str(NEXO_CODE))
|
|
22
|
+
|
|
23
|
+
from agent_runner import AutomationBackendUnavailableError, run_automation_prompt
|
|
24
|
+
try:
|
|
25
|
+
from client_preferences import resolve_user_model as _resolve_user_model
|
|
26
|
+
_USER_MODEL = _resolve_user_model()
|
|
27
|
+
except Exception:
|
|
28
|
+
_USER_MODEL = ""
|
|
29
|
+
|
|
30
|
+
|
|
31
|
+
|
|
32
|
+
class ContextChecker:
|
|
33
|
+
def __init__(self):
|
|
34
|
+
self.state_dir = NEXO_HOME / "state"
|
|
35
|
+
self.state_dir.mkdir(exist_ok=True)
|
|
36
|
+
|
|
37
|
+
def check_email_sent(self, to_addr, subject, since_hours=72):
|
|
38
|
+
"""Check if email was already sent to address with subject."""
|
|
39
|
+
sent_path = Path.home() / "mail" / ".nexo-sent" / ".Sent"
|
|
40
|
+
if not sent_path.exists():
|
|
41
|
+
return False
|
|
42
|
+
|
|
43
|
+
subject_lower = subject.lower()
|
|
44
|
+
to_lower = to_addr.lower()
|
|
45
|
+
cutoff = datetime.now().timestamp() - (since_hours * 3600)
|
|
46
|
+
cur_dir = sent_path / "cur"
|
|
47
|
+
if not cur_dir.exists():
|
|
48
|
+
return False
|
|
49
|
+
|
|
50
|
+
for msg_file in cur_dir.iterdir():
|
|
51
|
+
try:
|
|
52
|
+
if msg_file.stat().st_mtime < cutoff:
|
|
53
|
+
continue
|
|
54
|
+
content = msg_file.read_text(errors="ignore")
|
|
55
|
+
except (OSError, UnicodeDecodeError):
|
|
56
|
+
continue
|
|
57
|
+
|
|
58
|
+
content_lower = content.lower()
|
|
59
|
+
if f"to:{to_lower}" in content_lower or f"to: {to_lower}" in content_lower:
|
|
60
|
+
if subject_lower in content_lower:
|
|
61
|
+
return True
|
|
62
|
+
return False
|
|
63
|
+
|
|
64
|
+
def check_file_exists(self, pattern, search_dirs=None):
|
|
65
|
+
"""Check if file matching pattern exists in common locations."""
|
|
66
|
+
if search_dirs is None:
|
|
67
|
+
search_dirs = [
|
|
68
|
+
"/var/www/vhosts",
|
|
69
|
+
str(NEXO_HOME),
|
|
70
|
+
"/opt",
|
|
71
|
+
]
|
|
72
|
+
|
|
73
|
+
for base_dir in search_dirs:
|
|
74
|
+
if not os.path.exists(base_dir):
|
|
75
|
+
continue
|
|
76
|
+
matches = []
|
|
77
|
+
try:
|
|
78
|
+
for root, _, files in os.walk(base_dir):
|
|
79
|
+
for filename in files:
|
|
80
|
+
if pattern in filename:
|
|
81
|
+
matches.append(str(Path(root) / filename))
|
|
82
|
+
if len(matches) >= 5:
|
|
83
|
+
return matches
|
|
84
|
+
except OSError:
|
|
85
|
+
continue
|
|
86
|
+
return []
|
|
87
|
+
|
|
88
|
+
def check_action_done(self, action_type, identifier, ttl_days=7):
|
|
89
|
+
"""Check if action was already performed recently."""
|
|
90
|
+
action_file = self.state_dir / "actions.json"
|
|
91
|
+
actions = {}
|
|
92
|
+
if action_file.exists():
|
|
93
|
+
with open(action_file) as fh:
|
|
94
|
+
actions = json.load(fh)
|
|
95
|
+
|
|
96
|
+
key = hashlib.md5(f"{action_type}:{identifier}".encode(), usedforsecurity=False).hexdigest()
|
|
97
|
+
if key in actions:
|
|
98
|
+
action_time = datetime.fromisoformat(actions[key]["timestamp"])
|
|
99
|
+
age_days = (datetime.now() - action_time).days
|
|
100
|
+
if age_days < ttl_days:
|
|
101
|
+
return True, actions[key]
|
|
102
|
+
return False, None
|
|
103
|
+
|
|
104
|
+
def mark_action_done(self, action_type, identifier, metadata=None):
|
|
105
|
+
"""Mark action as completed."""
|
|
106
|
+
action_file = self.state_dir / "actions.json"
|
|
107
|
+
actions = {}
|
|
108
|
+
if action_file.exists():
|
|
109
|
+
with open(action_file) as fh:
|
|
110
|
+
actions = json.load(fh)
|
|
111
|
+
|
|
112
|
+
key = hashlib.md5(f"{action_type}:{identifier}".encode(), usedforsecurity=False).hexdigest()
|
|
113
|
+
actions[key] = {
|
|
114
|
+
"type": action_type,
|
|
115
|
+
"identifier": identifier,
|
|
116
|
+
"timestamp": datetime.now().isoformat(),
|
|
117
|
+
"metadata": metadata or {},
|
|
118
|
+
}
|
|
119
|
+
with open(action_file, "w") as fh:
|
|
120
|
+
json.dump(actions, fh, indent=2)
|
|
121
|
+
return key
|
|
122
|
+
|
|
123
|
+
|
|
124
|
+
def _extract_json(text: str) -> dict | None:
|
|
125
|
+
text = (text or "").strip()
|
|
126
|
+
if not text:
|
|
127
|
+
return None
|
|
128
|
+
if text.startswith("```"):
|
|
129
|
+
lines = text.splitlines()
|
|
130
|
+
end = len(lines)
|
|
131
|
+
for idx in range(len(lines) - 1, 0, -1):
|
|
132
|
+
if lines[idx].strip() == "```":
|
|
133
|
+
end = idx
|
|
134
|
+
break
|
|
135
|
+
text = "\n".join(lines[1:end]).strip()
|
|
136
|
+
brace_start = text.find("{")
|
|
137
|
+
if brace_start < 0:
|
|
138
|
+
return None
|
|
139
|
+
depth = 0
|
|
140
|
+
for idx in range(brace_start, len(text)):
|
|
141
|
+
if text[idx] == "{":
|
|
142
|
+
depth += 1
|
|
143
|
+
elif text[idx] == "}":
|
|
144
|
+
depth -= 1
|
|
145
|
+
if depth == 0:
|
|
146
|
+
try:
|
|
147
|
+
return json.loads(text[brace_start:idx + 1])
|
|
148
|
+
except json.JSONDecodeError:
|
|
149
|
+
return None
|
|
150
|
+
return None
|
|
151
|
+
|
|
152
|
+
|
|
153
|
+
def smart_check(action_description: str, context: str = "") -> dict:
|
|
154
|
+
"""Use the automation backend to check if an action would be redundant."""
|
|
155
|
+
checker = ContextChecker()
|
|
156
|
+
|
|
157
|
+
state_file = checker.state_dir / "actions.json"
|
|
158
|
+
recent_actions = {}
|
|
159
|
+
if state_file.exists():
|
|
160
|
+
try:
|
|
161
|
+
all_actions = json.loads(state_file.read_text())
|
|
162
|
+
cutoff = datetime.now().timestamp() - (7 * 86400)
|
|
163
|
+
for key, value in all_actions.items():
|
|
164
|
+
try:
|
|
165
|
+
ts = datetime.fromisoformat(value["timestamp"]).timestamp()
|
|
166
|
+
if ts > cutoff:
|
|
167
|
+
recent_actions[key] = value
|
|
168
|
+
except (ValueError, KeyError):
|
|
169
|
+
pass
|
|
170
|
+
except Exception:
|
|
171
|
+
pass
|
|
172
|
+
|
|
173
|
+
prompt = f"""You are a context deduplication engine for NEXO operations.
|
|
174
|
+
|
|
175
|
+
PROPOSED ACTION:
|
|
176
|
+
{action_description}
|
|
177
|
+
|
|
178
|
+
ADDITIONAL CONTEXT:
|
|
179
|
+
{context or "None"}
|
|
180
|
+
|
|
181
|
+
RECENT ACTIONS (last 7 days):
|
|
182
|
+
{json.dumps(list(recent_actions.values()), indent=1, default=str, ensure_ascii=False)}
|
|
183
|
+
|
|
184
|
+
Respond with ONLY valid JSON:
|
|
185
|
+
{{
|
|
186
|
+
"redundant": true/false,
|
|
187
|
+
"confidence": 0.0-1.0,
|
|
188
|
+
"reason": "<one line explanation>",
|
|
189
|
+
"matching_action": "<identifier of matching action if redundant, else null>"
|
|
190
|
+
}}
|
|
191
|
+
|
|
192
|
+
Rules:
|
|
193
|
+
- Same recipient + same intent within 72h = redundant
|
|
194
|
+
- Same file modification with same content = redundant
|
|
195
|
+
- Similar but different scope (e.g. different recipients) = NOT redundant
|
|
196
|
+
- When in doubt, say not redundant (false negatives are cheaper than false positives)"""
|
|
197
|
+
|
|
198
|
+
try:
|
|
199
|
+
result = run_automation_prompt(
|
|
200
|
+
prompt,
|
|
201
|
+
model=_USER_MODEL or "opus",
|
|
202
|
+
timeout=300,
|
|
203
|
+
output_format="text",
|
|
204
|
+
append_system_prompt="Return exactly one valid JSON object.",
|
|
205
|
+
allowed_tools="Read,Write,Edit,Glob,Grep,Bash,mcp__nexo__*",
|
|
206
|
+
)
|
|
207
|
+
if result.returncode == 0:
|
|
208
|
+
parsed = _extract_json(result.stdout)
|
|
209
|
+
if parsed:
|
|
210
|
+
return parsed
|
|
211
|
+
except AutomationBackendUnavailableError as exc:
|
|
212
|
+
return {"redundant": False, "reason": f"Automation backend unavailable — {exc}"}
|
|
213
|
+
except Exception:
|
|
214
|
+
pass
|
|
215
|
+
|
|
216
|
+
return {"redundant": False, "reason": "Automation check failed, defaulting to not redundant"}
|
|
217
|
+
|
|
218
|
+
|
|
219
|
+
def main():
|
|
220
|
+
"""CLI interface for context checking."""
|
|
221
|
+
if len(sys.argv) < 3:
|
|
222
|
+
print("Usage: check-context.py <command> <args>")
|
|
223
|
+
print("Commands:")
|
|
224
|
+
print(" email <to> <subject> - Check if email was sent")
|
|
225
|
+
print(" file <pattern> - Check if file exists")
|
|
226
|
+
print(" action <type> <id> - Check if action was done")
|
|
227
|
+
print(" smart <description> [ctx] - Intelligent duplicate check via automation backend")
|
|
228
|
+
sys.exit(1)
|
|
229
|
+
|
|
230
|
+
checker = ContextChecker()
|
|
231
|
+
command = sys.argv[1]
|
|
232
|
+
|
|
233
|
+
if command == "email":
|
|
234
|
+
if len(sys.argv) < 4:
|
|
235
|
+
print("Usage: check-context.py email <to> <subject>")
|
|
236
|
+
sys.exit(1)
|
|
237
|
+
exists = checker.check_email_sent(sys.argv[2], sys.argv[3])
|
|
238
|
+
print("EXISTS" if exists else "NOT_FOUND")
|
|
239
|
+
sys.exit(0 if not exists else 1)
|
|
240
|
+
|
|
241
|
+
if command == "file":
|
|
242
|
+
files = checker.check_file_exists(sys.argv[2])
|
|
243
|
+
if files:
|
|
244
|
+
print("\n".join(files))
|
|
245
|
+
sys.exit(1)
|
|
246
|
+
print("NOT_FOUND")
|
|
247
|
+
sys.exit(0)
|
|
248
|
+
|
|
249
|
+
if command == "action":
|
|
250
|
+
if len(sys.argv) < 4:
|
|
251
|
+
print("Usage: check-context.py action <type> <id>")
|
|
252
|
+
sys.exit(1)
|
|
253
|
+
done, data = checker.check_action_done(sys.argv[2], sys.argv[3])
|
|
254
|
+
if done:
|
|
255
|
+
print(f"DONE: {data}")
|
|
256
|
+
sys.exit(1)
|
|
257
|
+
print("NOT_DONE")
|
|
258
|
+
sys.exit(0)
|
|
259
|
+
|
|
260
|
+
if command == "smart":
|
|
261
|
+
description = sys.argv[2]
|
|
262
|
+
context = sys.argv[3] if len(sys.argv) > 3 else ""
|
|
263
|
+
result = smart_check(description, context)
|
|
264
|
+
print(json.dumps(result, indent=2, ensure_ascii=False))
|
|
265
|
+
sys.exit(1 if result.get("redundant") else 0)
|
|
266
|
+
|
|
267
|
+
print(f"Unknown command: {command}")
|
|
268
|
+
sys.exit(1)
|
|
269
|
+
|
|
270
|
+
|
|
271
|
+
if __name__ == "__main__":
|
|
272
|
+
main()
|