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.
- package/README.md +1 -1
- package/bin/nexo-brain.js +92 -9
- package/bin/postinstall.js +22 -15
- package/package.json +7 -4
- package/src/auto_update.py +194 -5
- package/src/crons/sync.py +6 -2
- package/src/db/_core.py +1 -0
- package/src/db/_entities.py +1 -0
- package/src/db/_episodic.py +1 -0
- package/src/db/_learnings.py +1 -0
- package/src/db/_reminders.py +1 -0
- package/src/db/_schema.py +11 -1
- package/src/db/_sessions.py +1 -0
- package/src/db/_skills.py +1 -0
- package/src/hooks/capture-tool-logs.sh +23 -6
- package/src/hooks/session-start.sh +4 -3
- package/src/plugin_loader.py +1 -0
- package/src/plugins/update.py +377 -26
- package/src/scripts/deep-sleep/apply_findings.py +1 -0
- package/src/scripts/deep-sleep/collect.py +1 -0
- package/src/scripts/deep-sleep/extract.py +1 -0
- package/src/scripts/deep-sleep/synthesize.py +1 -0
- package/src/scripts/nexo-catchup.py +29 -4
- package/src/scripts/nexo-daily-self-audit.py +21 -1
- package/src/scripts/nexo-evolution-run.py +21 -1
- package/src/scripts/nexo-learning-housekeep.py +1 -0
- package/src/scripts/nexo-postmortem-consolidator.py +34 -9
- package/src/scripts/nexo-sleep.py +32 -10
- package/src/scripts/nexo-synthesis.py +29 -9
- package/src/scripts/nexo-update.sh +109 -7
- package/src/scripts/nexo-watchdog.sh +122 -58
- package/src/server.py +66 -1
- package/src/tools_coordination.py +1 -0
- package/src/tools_sessions.py +1 -0
- package/scripts/migrate-to-unified 2.sh +0 -813
- package/scripts/migrate-to-unified.sh +0 -813
- package/scripts/migrate-v1.5-to-v1.6 2.py +0 -778
- package/scripts/migrate-v1.5-to-v1.6.py +0 -778
- package/scripts/migrate-v1.7-to-v1.8 2.py +0 -214
- package/scripts/migrate-v1.7-to-v1.8.py +0 -214
- package/scripts/nexo-preflight.sh +0 -236
- package/scripts/pre-commit-check 2.sh +0 -55
- package/scripts/pre-commit-check.sh +0 -55
- 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 +0 -159
- package/src/auto_update 2.py +0 -634
- package/src/claim_graph 2.py +0 -323
- package/src/cognitive/__init__ 2.py +0 -62
- 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 +0 -567
- package/src/cognitive/_decay 2.py +0 -382
- package/src/cognitive/_ingest 2.py +0 -892
- package/src/cognitive/_memory 2.py +0 -912
- package/src/cognitive/_search 2.py +0 -949
- package/src/cognitive/_trust 2.py +0 -464
- package/src/crons/__pycache__/sync.cpython-314.pyc +0 -0
- package/src/crons/manifest 2.json +0 -106
- package/src/crons/sync 2.py +0 -217
- 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 +0 -789
- package/src/db/__init__ 2.py +0 -89
- 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 +0 -417
- package/src/db/_credentials 2.py +0 -124
- package/src/db/_entities 2.py +0 -178
- package/src/db/_episodic 2.py +0 -738
- package/src/db/_evolution 2.py +0 -54
- package/src/db/_fts 2.py +0 -406
- package/src/db/_learnings 2.py +0 -168
- package/src/db/_reminders 2.py +0 -338
- package/src/db/_schema 2.py +0 -364
- package/src/db/_sessions 2.py +0 -300
- package/src/db/_tasks 2.py +0 -91
- package/src/evolution_cycle 2.py +0 -266
- package/src/hnsw_index 2.py +0 -254
- package/src/hooks/auto_capture 2.py +0 -208
- package/src/hooks/caffeinate-guard 2.sh +0 -8
- package/src/hooks/capture-session 2.sh +0 -21
- package/src/hooks/capture-tool-logs 2.sh +0 -127
- package/src/hooks/daily-briefing-check 2.sh +0 -33
- package/src/hooks/inbox-hook 2.sh +0 -76
- package/src/hooks/post-compact 2.sh +0 -148
- package/src/hooks/pre-compact 2.sh +0 -151
- package/src/hooks/session-start 2.sh +0 -268
- package/src/hooks/session-stop 2.sh +0 -140
- package/src/kg_populate 2.py +0 -290
- package/src/knowledge_graph 2.py +0 -257
- package/src/maintenance 2.py +0 -59
- package/src/migrate_embeddings 2.py +0 -122
- package/src/plugin_loader 2.py +0 -202
- 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 +0 -805
- package/src/plugins/agents 2.py +0 -52
- package/src/plugins/artifact_registry 2.py +0 -450
- package/src/plugins/backup 2.py +0 -104
- package/src/plugins/cognitive_memory 2.py +0 -564
- package/src/plugins/core_rules 2.py +0 -252
- package/src/plugins/cortex 2.py +0 -299
- package/src/plugins/entities 2.py +0 -67
- package/src/plugins/episodic_memory 2.py +0 -533
- package/src/plugins/evolution 2.py +0 -115
- package/src/plugins/guard 2.py +0 -746
- package/src/plugins/knowledge_graph_tools 2.py +0 -105
- package/src/plugins/preferences 2.py +0 -47
- package/src/plugins/update 2.py +0 -256
- package/src/requirements 2.txt +0 -12
- package/src/rules/__init__ 2.py +0 -0
- package/src/rules/core-rules 2.json +0 -331
- package/src/rules/migrate 2.py +0 -207
- 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 +0 -264
- package/src/scripts/nexo-auto-update 2.py +0 -6
- package/src/scripts/nexo-backup 2.sh +0 -25
- package/src/scripts/nexo-brain-activation 2.sh +0 -140
- package/src/scripts/nexo-catchup 2.py +0 -242
- package/src/scripts/nexo-cognitive-decay 2.py +0 -182
- package/src/scripts/nexo-daily-self-audit 2.py +0 -552
- package/src/scripts/nexo-deep-sleep 2.sh +0 -97
- package/src/scripts/nexo-evolution-run 2.py +0 -597
- package/src/scripts/nexo-followup-hygiene 2.py +0 -112
- package/src/scripts/nexo-github-monitor 2.py +0 -256
- package/src/scripts/nexo-immune 2.py +0 -927
- package/src/scripts/nexo-inbox-hook 2.sh +0 -74
- package/src/scripts/nexo-install 2.py +0 -6
- package/src/scripts/nexo-learning-housekeep 2.py +0 -245
- package/src/scripts/nexo-learning-validator 2.py +0 -207
- package/src/scripts/nexo-migrate 2.py +0 -232
- package/src/scripts/nexo-postmortem-consolidator 2.py +0 -421
- package/src/scripts/nexo-pre-commit 2.py +0 -120
- package/src/scripts/nexo-prevent-sleep 2.sh +0 -29
- package/src/scripts/nexo-proactive-dashboard 2.py +0 -345
- package/src/scripts/nexo-reflection 2.py +0 -253
- package/src/scripts/nexo-runtime-preflight 2.py +0 -274
- package/src/scripts/nexo-send-email 2.py +0 -25
- package/src/scripts/nexo-send-email.py +0 -25
- package/src/scripts/nexo-send-reply 2.py +0 -178
- package/src/scripts/nexo-send-reply.py +0 -178
- package/src/scripts/nexo-sleep 2.py +0 -592
- package/src/scripts/nexo-snapshot-restore 2.sh +0 -35
- package/src/scripts/nexo-synthesis 2.py +0 -253
- package/src/scripts/nexo-tcc-approve 2.sh +0 -79
- package/src/scripts/nexo-update 2.sh +0 -161
- package/src/scripts/nexo-watchdog 2.sh +0 -878
- package/src/scripts/nexo-watchdog-smoke 2.py +0 -119
- package/src/server 2.py +0 -733
- package/src/storage_router 2.py +0 -32
- package/src/tools_coordination 2.py +0 -102
- package/src/tools_credentials 2.py +0 -68
- package/src/tools_learnings 2.py +0 -220
- package/src/tools_menu 2.py +0 -227
- package/src/tools_reminders 2.py +0 -86
- package/src/tools_reminders_crud 2.py +0 -159
- package/src/tools_sessions 2.py +0 -476
- package/src/tools_task_history 2.py +0 -57
- package/templates/CLAUDE.md 2.template +0 -63
- package/templates/openclaw 2.json +0 -13
- package/tests/__init__ 2.py +0 -0
- package/tests/__init__.py +0 -0
- package/tests/conftest 2.py +0 -71
- package/tests/conftest.py +0 -71
- package/tests/test_cognitive 2.py +0 -205
- package/tests/test_cognitive.py +0 -205
- package/tests/test_knowledge_graph 2.py +0 -140
- package/tests/test_knowledge_graph.py +0 -140
- package/tests/test_migrations 2.py +0 -137
- package/tests/test_migrations.py +0 -137
package/tests/conftest.py
DELETED
|
@@ -1,71 +0,0 @@
|
|
|
1
|
-
"""Shared fixtures for NEXO test suite.
|
|
2
|
-
|
|
3
|
-
Uses isolated temp databases so tests never touch production data.
|
|
4
|
-
"""
|
|
5
|
-
|
|
6
|
-
import os
|
|
7
|
-
import sys
|
|
8
|
-
import sqlite3
|
|
9
|
-
|
|
10
|
-
import pytest
|
|
11
|
-
|
|
12
|
-
# Add src/ to path so we can import nexo modules
|
|
13
|
-
sys.path.insert(0, os.path.join(os.path.dirname(__file__), "..", "src"))
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
@pytest.fixture(autouse=True)
|
|
17
|
-
def isolated_db(tmp_path, monkeypatch):
|
|
18
|
-
"""Redirect both nexo.db and cognitive.db to temp files per test."""
|
|
19
|
-
test_db = str(tmp_path / "test_nexo.db")
|
|
20
|
-
test_cog_db = str(tmp_path / "test_cognitive.db")
|
|
21
|
-
|
|
22
|
-
monkeypatch.setenv("NEXO_TEST_DB", test_db)
|
|
23
|
-
monkeypatch.setenv("NEXO_COGNITIVE_DB", test_cog_db)
|
|
24
|
-
monkeypatch.setenv("NEXO_SKIP_FS_INDEX", "1")
|
|
25
|
-
|
|
26
|
-
import db._core as db_core
|
|
27
|
-
import cognitive._core as cog_core
|
|
28
|
-
|
|
29
|
-
# Close existing connections
|
|
30
|
-
db_core.close_db()
|
|
31
|
-
if cog_core._conn is not None:
|
|
32
|
-
try:
|
|
33
|
-
cog_core._conn.close()
|
|
34
|
-
except Exception:
|
|
35
|
-
pass
|
|
36
|
-
cog_core._conn = None
|
|
37
|
-
|
|
38
|
-
# Point to temp paths
|
|
39
|
-
db_core.DB_PATH = test_db
|
|
40
|
-
cog_core.COGNITIVE_DB = test_cog_db
|
|
41
|
-
|
|
42
|
-
# Create a fresh raw connection
|
|
43
|
-
raw = sqlite3.connect(test_db, timeout=30, check_same_thread=False,
|
|
44
|
-
isolation_level=None)
|
|
45
|
-
raw.execute("PRAGMA journal_mode=WAL")
|
|
46
|
-
raw.execute("PRAGMA busy_timeout=30000")
|
|
47
|
-
raw.execute("PRAGMA foreign_keys=ON")
|
|
48
|
-
raw.row_factory = sqlite3.Row
|
|
49
|
-
|
|
50
|
-
wrapped = db_core._SerializedConnection(raw)
|
|
51
|
-
db_core._shared_conn = wrapped
|
|
52
|
-
|
|
53
|
-
# Initialize schemas
|
|
54
|
-
from db._core import init_db
|
|
55
|
-
from db._schema import run_migrations
|
|
56
|
-
init_db()
|
|
57
|
-
run_migrations()
|
|
58
|
-
|
|
59
|
-
yield {
|
|
60
|
-
"nexo_db": test_db,
|
|
61
|
-
"cognitive_db": test_cog_db,
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
# Cleanup
|
|
65
|
-
db_core.close_db()
|
|
66
|
-
if cog_core._conn is not None:
|
|
67
|
-
try:
|
|
68
|
-
cog_core._conn.close()
|
|
69
|
-
except Exception:
|
|
70
|
-
pass
|
|
71
|
-
cog_core._conn = None
|
|
@@ -1,205 +0,0 @@
|
|
|
1
|
-
"""Tests for cognitive engine: embeddings, cosine similarity, search, KG boost, decay."""
|
|
2
|
-
|
|
3
|
-
import math
|
|
4
|
-
import numpy as np
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
def test_cosine_similarity_identical():
|
|
8
|
-
"""Identical vectors should have similarity 1.0."""
|
|
9
|
-
import cognitive
|
|
10
|
-
a = np.array([1.0, 2.0, 3.0], dtype=np.float32)
|
|
11
|
-
assert abs(cognitive.cosine_similarity(a, a) - 1.0) < 1e-6
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
def test_cosine_similarity_orthogonal():
|
|
15
|
-
"""Orthogonal vectors should have similarity 0.0."""
|
|
16
|
-
import cognitive
|
|
17
|
-
a = np.array([1.0, 0.0, 0.0], dtype=np.float32)
|
|
18
|
-
b = np.array([0.0, 1.0, 0.0], dtype=np.float32)
|
|
19
|
-
assert abs(cognitive.cosine_similarity(a, b)) < 1e-6
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
def test_cosine_similarity_zero_norm():
|
|
23
|
-
"""Zero vector should return 0.0 (not NaN)."""
|
|
24
|
-
import cognitive
|
|
25
|
-
a = np.zeros(3, dtype=np.float32)
|
|
26
|
-
b = np.array([1.0, 2.0, 3.0], dtype=np.float32)
|
|
27
|
-
assert cognitive.cosine_similarity(a, b) == 0.0
|
|
28
|
-
assert cognitive.cosine_similarity(b, a) == 0.0
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
def test_blob_roundtrip():
|
|
32
|
-
"""Array → blob → array should be lossless."""
|
|
33
|
-
import cognitive
|
|
34
|
-
arr = np.random.randn(768).astype(np.float32)
|
|
35
|
-
blob = cognitive._array_to_blob(arr)
|
|
36
|
-
recovered = cognitive._blob_to_array(blob)
|
|
37
|
-
np.testing.assert_array_equal(arr, recovered)
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
def test_kg_boost_results_no_kg_data():
|
|
41
|
-
"""KG boost should be a no-op when there are no KG nodes."""
|
|
42
|
-
import cognitive
|
|
43
|
-
results = [
|
|
44
|
-
{"source_type": "learning", "source_id": "L999", "score": 0.7},
|
|
45
|
-
{"source_type": "sensory", "source_id": "buffer#1", "score": 0.6},
|
|
46
|
-
]
|
|
47
|
-
boosted = cognitive._kg_boost_results(results)
|
|
48
|
-
# No KG nodes in test DB → no boost applied
|
|
49
|
-
assert boosted[0]["score"] == 0.7
|
|
50
|
-
assert boosted[1]["score"] == 0.6
|
|
51
|
-
assert "kg_boost" not in boosted[0]
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
def test_kg_boost_results_with_connections():
|
|
55
|
-
"""KG boost should increase scores for connected nodes."""
|
|
56
|
-
import cognitive
|
|
57
|
-
import knowledge_graph as kg
|
|
58
|
-
|
|
59
|
-
db = cognitive._get_db()
|
|
60
|
-
# Initialize KG tables
|
|
61
|
-
db.executescript("""
|
|
62
|
-
CREATE TABLE IF NOT EXISTS kg_nodes (
|
|
63
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
64
|
-
node_type TEXT NOT NULL,
|
|
65
|
-
node_ref TEXT NOT NULL UNIQUE,
|
|
66
|
-
label TEXT NOT NULL DEFAULT '',
|
|
67
|
-
properties TEXT DEFAULT '{}'
|
|
68
|
-
);
|
|
69
|
-
CREATE TABLE IF NOT EXISTS kg_edges (
|
|
70
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
71
|
-
source_id INTEGER NOT NULL,
|
|
72
|
-
target_id INTEGER NOT NULL,
|
|
73
|
-
relation TEXT NOT NULL DEFAULT '',
|
|
74
|
-
weight REAL DEFAULT 1.0,
|
|
75
|
-
confidence REAL DEFAULT 1.0,
|
|
76
|
-
valid_from TEXT,
|
|
77
|
-
valid_until TEXT,
|
|
78
|
-
source_memory_id TEXT DEFAULT '',
|
|
79
|
-
properties TEXT DEFAULT '{}'
|
|
80
|
-
);
|
|
81
|
-
""")
|
|
82
|
-
|
|
83
|
-
# Create a learning node with 8 connections
|
|
84
|
-
node_id = db.execute(
|
|
85
|
-
"INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
|
|
86
|
-
("learning", "learning:42", "Test Learning")
|
|
87
|
-
).lastrowid
|
|
88
|
-
|
|
89
|
-
for i in range(8):
|
|
90
|
-
file_id = db.execute(
|
|
91
|
-
"INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
|
|
92
|
-
("file", f"file:test{i}.py", f"test{i}.py")
|
|
93
|
-
).lastrowid
|
|
94
|
-
db.execute(
|
|
95
|
-
"INSERT INTO kg_edges (source_id, target_id, relation, weight) VALUES (?, ?, ?, ?)",
|
|
96
|
-
(node_id, file_id, "touched", 1.0)
|
|
97
|
-
)
|
|
98
|
-
db.commit()
|
|
99
|
-
|
|
100
|
-
results = [
|
|
101
|
-
{"source_type": "learning", "source_id": "L42", "score": 0.6},
|
|
102
|
-
{"source_type": "sensory", "source_id": "buffer#1", "score": 0.6},
|
|
103
|
-
]
|
|
104
|
-
boosted = cognitive._kg_boost_results(results)
|
|
105
|
-
|
|
106
|
-
# Learning with 8 edges should get boost
|
|
107
|
-
assert boosted[0].get("kg_boost") is not None
|
|
108
|
-
assert boosted[0]["score"] > 0.6
|
|
109
|
-
expected_boost = min(0.08, 0.015 * math.log2(8 + 1))
|
|
110
|
-
assert abs(boosted[0]["kg_boost"] - round(expected_boost, 4)) < 0.001
|
|
111
|
-
|
|
112
|
-
# Sensory with no KG node should NOT get boost
|
|
113
|
-
assert "kg_boost" not in boosted[1]
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
def test_kg_boost_relevance_gate():
|
|
117
|
-
"""KG boost should not apply to low-relevance results (score < 0.45)."""
|
|
118
|
-
import cognitive
|
|
119
|
-
|
|
120
|
-
db = cognitive._get_db()
|
|
121
|
-
db.executescript("""
|
|
122
|
-
CREATE TABLE IF NOT EXISTS kg_nodes (
|
|
123
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
124
|
-
node_type TEXT NOT NULL,
|
|
125
|
-
node_ref TEXT NOT NULL UNIQUE,
|
|
126
|
-
label TEXT NOT NULL DEFAULT '',
|
|
127
|
-
properties TEXT DEFAULT '{}'
|
|
128
|
-
);
|
|
129
|
-
CREATE TABLE IF NOT EXISTS kg_edges (
|
|
130
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
131
|
-
source_id INTEGER NOT NULL,
|
|
132
|
-
target_id INTEGER NOT NULL,
|
|
133
|
-
relation TEXT NOT NULL DEFAULT '',
|
|
134
|
-
weight REAL DEFAULT 1.0,
|
|
135
|
-
confidence REAL DEFAULT 1.0,
|
|
136
|
-
valid_from TEXT,
|
|
137
|
-
valid_until TEXT,
|
|
138
|
-
source_memory_id TEXT DEFAULT '',
|
|
139
|
-
properties TEXT DEFAULT '{}'
|
|
140
|
-
);
|
|
141
|
-
""")
|
|
142
|
-
|
|
143
|
-
node_id = db.execute(
|
|
144
|
-
"INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
|
|
145
|
-
("learning", "learning:99", "Low Score Learning")
|
|
146
|
-
).lastrowid
|
|
147
|
-
for i in range(20):
|
|
148
|
-
fid = db.execute(
|
|
149
|
-
"INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
|
|
150
|
-
("file", f"file:low{i}.py", f"low{i}.py")
|
|
151
|
-
).lastrowid
|
|
152
|
-
db.execute(
|
|
153
|
-
"INSERT INTO kg_edges (source_id, target_id, relation, weight) VALUES (?, ?, ?, ?)",
|
|
154
|
-
(node_id, fid, "touched", 1.0)
|
|
155
|
-
)
|
|
156
|
-
db.commit()
|
|
157
|
-
|
|
158
|
-
results = [
|
|
159
|
-
{"source_type": "learning", "source_id": "L99", "score": 0.3},
|
|
160
|
-
]
|
|
161
|
-
boosted = cognitive._kg_boost_results(results)
|
|
162
|
-
# Score below 0.45 gate → no boost
|
|
163
|
-
assert boosted[0]["score"] == 0.3
|
|
164
|
-
assert "kg_boost" not in boosted[0]
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
def test_decay_formula():
|
|
168
|
-
"""Ebbinghaus decay should reduce strength over time (lambda operates on HOURS)."""
|
|
169
|
-
import cognitive
|
|
170
|
-
# STM decay: lambda=0.004126, after 7 days (168h) strength should be ~0.5
|
|
171
|
-
initial = 1.0
|
|
172
|
-
hours_7d = 7 * 24
|
|
173
|
-
decayed = initial * math.exp(-cognitive.LAMBDA_STM * hours_7d)
|
|
174
|
-
assert 0.45 < decayed < 0.55
|
|
175
|
-
|
|
176
|
-
# LTM decay: lambda=0.000481, after 60 days (1440h) strength should be ~0.5
|
|
177
|
-
hours_60d = 60 * 24
|
|
178
|
-
decayed_ltm = initial * math.exp(-cognitive.LAMBDA_LTM * hours_60d)
|
|
179
|
-
assert 0.45 < decayed_ltm < 0.55
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
def test_apply_temporal_boost_historical():
|
|
183
|
-
"""Historical queries should get no temporal boost."""
|
|
184
|
-
import cognitive
|
|
185
|
-
results = [
|
|
186
|
-
{"source_type": "learning", "source_id": "L1", "score": 0.7,
|
|
187
|
-
"created_at": "2026-03-28T10:00:00"},
|
|
188
|
-
]
|
|
189
|
-
boosted = cognitive._apply_temporal_boost(results, "what happened months ago")
|
|
190
|
-
# Historical cue "months" should disable boost
|
|
191
|
-
assert boosted[0]["score"] == 0.7
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
def test_apply_temporal_boost_operational():
|
|
195
|
-
"""Operational queries should get a higher temporal boost."""
|
|
196
|
-
import cognitive
|
|
197
|
-
from datetime import datetime
|
|
198
|
-
now_str = datetime.now().strftime("%Y-%m-%dT%H:%M:%S")
|
|
199
|
-
results = [
|
|
200
|
-
{"source_type": "learning", "source_id": "L1", "score": 0.7,
|
|
201
|
-
"created_at": now_str},
|
|
202
|
-
]
|
|
203
|
-
boosted = cognitive._apply_temporal_boost(results, "active backend issues today")
|
|
204
|
-
# Very recent + operational → should get noticeable boost
|
|
205
|
-
assert boosted[0]["score"] > 0.7
|
package/tests/test_cognitive.py
DELETED
|
@@ -1,205 +0,0 @@
|
|
|
1
|
-
"""Tests for cognitive engine: embeddings, cosine similarity, search, KG boost, decay."""
|
|
2
|
-
|
|
3
|
-
import math
|
|
4
|
-
import numpy as np
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
def test_cosine_similarity_identical():
|
|
8
|
-
"""Identical vectors should have similarity 1.0."""
|
|
9
|
-
import cognitive
|
|
10
|
-
a = np.array([1.0, 2.0, 3.0], dtype=np.float32)
|
|
11
|
-
assert abs(cognitive.cosine_similarity(a, a) - 1.0) < 1e-6
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
def test_cosine_similarity_orthogonal():
|
|
15
|
-
"""Orthogonal vectors should have similarity 0.0."""
|
|
16
|
-
import cognitive
|
|
17
|
-
a = np.array([1.0, 0.0, 0.0], dtype=np.float32)
|
|
18
|
-
b = np.array([0.0, 1.0, 0.0], dtype=np.float32)
|
|
19
|
-
assert abs(cognitive.cosine_similarity(a, b)) < 1e-6
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
def test_cosine_similarity_zero_norm():
|
|
23
|
-
"""Zero vector should return 0.0 (not NaN)."""
|
|
24
|
-
import cognitive
|
|
25
|
-
a = np.zeros(3, dtype=np.float32)
|
|
26
|
-
b = np.array([1.0, 2.0, 3.0], dtype=np.float32)
|
|
27
|
-
assert cognitive.cosine_similarity(a, b) == 0.0
|
|
28
|
-
assert cognitive.cosine_similarity(b, a) == 0.0
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
def test_blob_roundtrip():
|
|
32
|
-
"""Array → blob → array should be lossless."""
|
|
33
|
-
import cognitive
|
|
34
|
-
arr = np.random.randn(768).astype(np.float32)
|
|
35
|
-
blob = cognitive._array_to_blob(arr)
|
|
36
|
-
recovered = cognitive._blob_to_array(blob)
|
|
37
|
-
np.testing.assert_array_equal(arr, recovered)
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
def test_kg_boost_results_no_kg_data():
|
|
41
|
-
"""KG boost should be a no-op when there are no KG nodes."""
|
|
42
|
-
import cognitive
|
|
43
|
-
results = [
|
|
44
|
-
{"source_type": "learning", "source_id": "L999", "score": 0.7},
|
|
45
|
-
{"source_type": "sensory", "source_id": "buffer#1", "score": 0.6},
|
|
46
|
-
]
|
|
47
|
-
boosted = cognitive._kg_boost_results(results)
|
|
48
|
-
# No KG nodes in test DB → no boost applied
|
|
49
|
-
assert boosted[0]["score"] == 0.7
|
|
50
|
-
assert boosted[1]["score"] == 0.6
|
|
51
|
-
assert "kg_boost" not in boosted[0]
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
def test_kg_boost_results_with_connections():
|
|
55
|
-
"""KG boost should increase scores for connected nodes."""
|
|
56
|
-
import cognitive
|
|
57
|
-
import knowledge_graph as kg
|
|
58
|
-
|
|
59
|
-
db = cognitive._get_db()
|
|
60
|
-
# Initialize KG tables
|
|
61
|
-
db.executescript("""
|
|
62
|
-
CREATE TABLE IF NOT EXISTS kg_nodes (
|
|
63
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
64
|
-
node_type TEXT NOT NULL,
|
|
65
|
-
node_ref TEXT NOT NULL UNIQUE,
|
|
66
|
-
label TEXT NOT NULL DEFAULT '',
|
|
67
|
-
properties TEXT DEFAULT '{}'
|
|
68
|
-
);
|
|
69
|
-
CREATE TABLE IF NOT EXISTS kg_edges (
|
|
70
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
71
|
-
source_id INTEGER NOT NULL,
|
|
72
|
-
target_id INTEGER NOT NULL,
|
|
73
|
-
relation TEXT NOT NULL DEFAULT '',
|
|
74
|
-
weight REAL DEFAULT 1.0,
|
|
75
|
-
confidence REAL DEFAULT 1.0,
|
|
76
|
-
valid_from TEXT,
|
|
77
|
-
valid_until TEXT,
|
|
78
|
-
source_memory_id TEXT DEFAULT '',
|
|
79
|
-
properties TEXT DEFAULT '{}'
|
|
80
|
-
);
|
|
81
|
-
""")
|
|
82
|
-
|
|
83
|
-
# Create a learning node with 8 connections
|
|
84
|
-
node_id = db.execute(
|
|
85
|
-
"INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
|
|
86
|
-
("learning", "learning:42", "Test Learning")
|
|
87
|
-
).lastrowid
|
|
88
|
-
|
|
89
|
-
for i in range(8):
|
|
90
|
-
file_id = db.execute(
|
|
91
|
-
"INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
|
|
92
|
-
("file", f"file:test{i}.py", f"test{i}.py")
|
|
93
|
-
).lastrowid
|
|
94
|
-
db.execute(
|
|
95
|
-
"INSERT INTO kg_edges (source_id, target_id, relation, weight) VALUES (?, ?, ?, ?)",
|
|
96
|
-
(node_id, file_id, "touched", 1.0)
|
|
97
|
-
)
|
|
98
|
-
db.commit()
|
|
99
|
-
|
|
100
|
-
results = [
|
|
101
|
-
{"source_type": "learning", "source_id": "L42", "score": 0.6},
|
|
102
|
-
{"source_type": "sensory", "source_id": "buffer#1", "score": 0.6},
|
|
103
|
-
]
|
|
104
|
-
boosted = cognitive._kg_boost_results(results)
|
|
105
|
-
|
|
106
|
-
# Learning with 8 edges should get boost
|
|
107
|
-
assert boosted[0].get("kg_boost") is not None
|
|
108
|
-
assert boosted[0]["score"] > 0.6
|
|
109
|
-
expected_boost = min(0.08, 0.015 * math.log2(8 + 1))
|
|
110
|
-
assert abs(boosted[0]["kg_boost"] - round(expected_boost, 4)) < 0.001
|
|
111
|
-
|
|
112
|
-
# Sensory with no KG node should NOT get boost
|
|
113
|
-
assert "kg_boost" not in boosted[1]
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
def test_kg_boost_relevance_gate():
|
|
117
|
-
"""KG boost should not apply to low-relevance results (score < 0.45)."""
|
|
118
|
-
import cognitive
|
|
119
|
-
|
|
120
|
-
db = cognitive._get_db()
|
|
121
|
-
db.executescript("""
|
|
122
|
-
CREATE TABLE IF NOT EXISTS kg_nodes (
|
|
123
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
124
|
-
node_type TEXT NOT NULL,
|
|
125
|
-
node_ref TEXT NOT NULL UNIQUE,
|
|
126
|
-
label TEXT NOT NULL DEFAULT '',
|
|
127
|
-
properties TEXT DEFAULT '{}'
|
|
128
|
-
);
|
|
129
|
-
CREATE TABLE IF NOT EXISTS kg_edges (
|
|
130
|
-
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
131
|
-
source_id INTEGER NOT NULL,
|
|
132
|
-
target_id INTEGER NOT NULL,
|
|
133
|
-
relation TEXT NOT NULL DEFAULT '',
|
|
134
|
-
weight REAL DEFAULT 1.0,
|
|
135
|
-
confidence REAL DEFAULT 1.0,
|
|
136
|
-
valid_from TEXT,
|
|
137
|
-
valid_until TEXT,
|
|
138
|
-
source_memory_id TEXT DEFAULT '',
|
|
139
|
-
properties TEXT DEFAULT '{}'
|
|
140
|
-
);
|
|
141
|
-
""")
|
|
142
|
-
|
|
143
|
-
node_id = db.execute(
|
|
144
|
-
"INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
|
|
145
|
-
("learning", "learning:99", "Low Score Learning")
|
|
146
|
-
).lastrowid
|
|
147
|
-
for i in range(20):
|
|
148
|
-
fid = db.execute(
|
|
149
|
-
"INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
|
|
150
|
-
("file", f"file:low{i}.py", f"low{i}.py")
|
|
151
|
-
).lastrowid
|
|
152
|
-
db.execute(
|
|
153
|
-
"INSERT INTO kg_edges (source_id, target_id, relation, weight) VALUES (?, ?, ?, ?)",
|
|
154
|
-
(node_id, fid, "touched", 1.0)
|
|
155
|
-
)
|
|
156
|
-
db.commit()
|
|
157
|
-
|
|
158
|
-
results = [
|
|
159
|
-
{"source_type": "learning", "source_id": "L99", "score": 0.3},
|
|
160
|
-
]
|
|
161
|
-
boosted = cognitive._kg_boost_results(results)
|
|
162
|
-
# Score below 0.45 gate → no boost
|
|
163
|
-
assert boosted[0]["score"] == 0.3
|
|
164
|
-
assert "kg_boost" not in boosted[0]
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
def test_decay_formula():
|
|
168
|
-
"""Ebbinghaus decay should reduce strength over time (lambda operates on HOURS)."""
|
|
169
|
-
import cognitive
|
|
170
|
-
# STM decay: lambda=0.004126, after 7 days (168h) strength should be ~0.5
|
|
171
|
-
initial = 1.0
|
|
172
|
-
hours_7d = 7 * 24
|
|
173
|
-
decayed = initial * math.exp(-cognitive.LAMBDA_STM * hours_7d)
|
|
174
|
-
assert 0.45 < decayed < 0.55
|
|
175
|
-
|
|
176
|
-
# LTM decay: lambda=0.000481, after 60 days (1440h) strength should be ~0.5
|
|
177
|
-
hours_60d = 60 * 24
|
|
178
|
-
decayed_ltm = initial * math.exp(-cognitive.LAMBDA_LTM * hours_60d)
|
|
179
|
-
assert 0.45 < decayed_ltm < 0.55
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
def test_apply_temporal_boost_historical():
|
|
183
|
-
"""Historical queries should get no temporal boost."""
|
|
184
|
-
import cognitive
|
|
185
|
-
results = [
|
|
186
|
-
{"source_type": "learning", "source_id": "L1", "score": 0.7,
|
|
187
|
-
"created_at": "2026-03-28T10:00:00"},
|
|
188
|
-
]
|
|
189
|
-
boosted = cognitive._apply_temporal_boost(results, "what happened months ago")
|
|
190
|
-
# Historical cue "months" should disable boost
|
|
191
|
-
assert boosted[0]["score"] == 0.7
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
def test_apply_temporal_boost_operational():
|
|
195
|
-
"""Operational queries should get a higher temporal boost."""
|
|
196
|
-
import cognitive
|
|
197
|
-
from datetime import datetime
|
|
198
|
-
now_str = datetime.now().strftime("%Y-%m-%dT%H:%M:%S")
|
|
199
|
-
results = [
|
|
200
|
-
{"source_type": "learning", "source_id": "L1", "score": 0.7,
|
|
201
|
-
"created_at": now_str},
|
|
202
|
-
]
|
|
203
|
-
boosted = cognitive._apply_temporal_boost(results, "active backend issues today")
|
|
204
|
-
# Very recent + operational → should get noticeable boost
|
|
205
|
-
assert boosted[0]["score"] > 0.7
|
|
@@ -1,140 +0,0 @@
|
|
|
1
|
-
"""Tests for Knowledge Graph operations."""
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
def test_upsert_and_get_node():
|
|
5
|
-
"""Create a node and retrieve it."""
|
|
6
|
-
import cognitive
|
|
7
|
-
cognitive._get_db() # Ensure KG tables exist
|
|
8
|
-
|
|
9
|
-
import knowledge_graph as kg
|
|
10
|
-
node_id = kg.upsert_node("area", "area:test", "Test Area", {"color": "blue"})
|
|
11
|
-
assert node_id > 0
|
|
12
|
-
|
|
13
|
-
node = kg.get_node("area", "area:test")
|
|
14
|
-
assert node is not None
|
|
15
|
-
assert node["label"] == "Test Area"
|
|
16
|
-
|
|
17
|
-
# Upsert same node should return same id (update)
|
|
18
|
-
node_id2 = kg.upsert_node("area", "area:test", "Test Area Updated")
|
|
19
|
-
assert node_id2 == node_id
|
|
20
|
-
|
|
21
|
-
node2 = kg.get_node("area", "area:test")
|
|
22
|
-
assert node2["label"] == "Test Area Updated"
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
def test_upsert_and_get_edge():
|
|
26
|
-
"""Create nodes and edges, verify traversal."""
|
|
27
|
-
import cognitive
|
|
28
|
-
cognitive._get_db()
|
|
29
|
-
|
|
30
|
-
import knowledge_graph as kg
|
|
31
|
-
kg.upsert_node("learning", "learning:1", "L1")
|
|
32
|
-
kg.upsert_node("file", "file:foo.py", "foo.py")
|
|
33
|
-
|
|
34
|
-
result = kg.upsert_edge(
|
|
35
|
-
"learning", "learning:1", "touched",
|
|
36
|
-
"file", "file:foo.py", weight=1.0,
|
|
37
|
-
)
|
|
38
|
-
assert result["action"] == "ADD"
|
|
39
|
-
|
|
40
|
-
# Same edge again → NOOP
|
|
41
|
-
result2 = kg.upsert_edge(
|
|
42
|
-
"learning", "learning:1", "touched",
|
|
43
|
-
"file", "file:foo.py", weight=1.0,
|
|
44
|
-
)
|
|
45
|
-
assert result2["action"] == "NOOP"
|
|
46
|
-
|
|
47
|
-
# Different weight → UPDATE (closes old, creates new)
|
|
48
|
-
result3 = kg.upsert_edge(
|
|
49
|
-
"learning", "learning:1", "touched",
|
|
50
|
-
"file", "file:foo.py", weight=0.5,
|
|
51
|
-
)
|
|
52
|
-
assert result3["action"] == "UPDATE"
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
def test_neighbors():
|
|
56
|
-
"""Get direct neighbors of a node."""
|
|
57
|
-
import cognitive
|
|
58
|
-
cognitive._get_db()
|
|
59
|
-
|
|
60
|
-
import knowledge_graph as kg
|
|
61
|
-
kg.upsert_node("area", "area:myproject", "MyProject")
|
|
62
|
-
kg.upsert_node("learning", "learning:10", "L10")
|
|
63
|
-
kg.upsert_node("learning", "learning:11", "L11")
|
|
64
|
-
kg.upsert_edge("learning", "learning:10", "belongs_to", "area", "area:myproject")
|
|
65
|
-
kg.upsert_edge("learning", "learning:11", "belongs_to", "area", "area:myproject")
|
|
66
|
-
|
|
67
|
-
area_node = kg.get_node("area", "area:myproject")
|
|
68
|
-
neighbors = kg.get_neighbors(area_node["id"])
|
|
69
|
-
assert len(neighbors) == 2
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
def test_traverse():
|
|
73
|
-
"""BFS traversal from a node."""
|
|
74
|
-
import cognitive
|
|
75
|
-
cognitive._get_db()
|
|
76
|
-
|
|
77
|
-
import knowledge_graph as kg
|
|
78
|
-
# A → B → C chain
|
|
79
|
-
kg.upsert_node("area", "area:a", "A")
|
|
80
|
-
kg.upsert_node("file", "file:b", "B")
|
|
81
|
-
kg.upsert_node("change", "change:c", "C")
|
|
82
|
-
kg.upsert_edge("area", "area:a", "has", "file", "file:b")
|
|
83
|
-
kg.upsert_edge("file", "file:b", "modified_by", "change", "change:c")
|
|
84
|
-
|
|
85
|
-
a_node = kg.get_node("area", "area:a")
|
|
86
|
-
result = kg.traverse(a_node["id"], max_depth=2)
|
|
87
|
-
node_ids = {n["id"] for n in result["nodes"]}
|
|
88
|
-
# Should reach A, B, and C within depth 2
|
|
89
|
-
assert len(node_ids) == 3
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
def test_shortest_path():
|
|
93
|
-
"""Find shortest path between two nodes."""
|
|
94
|
-
import cognitive
|
|
95
|
-
cognitive._get_db()
|
|
96
|
-
|
|
97
|
-
import knowledge_graph as kg
|
|
98
|
-
kg.upsert_node("area", "area:x", "X")
|
|
99
|
-
kg.upsert_node("file", "file:y", "Y")
|
|
100
|
-
kg.upsert_node("change", "change:z", "Z")
|
|
101
|
-
kg.upsert_edge("area", "area:x", "has", "file", "file:y")
|
|
102
|
-
kg.upsert_edge("file", "file:y", "modified_by", "change", "change:z")
|
|
103
|
-
|
|
104
|
-
x = kg.get_node("area", "area:x")
|
|
105
|
-
z = kg.get_node("change", "change:z")
|
|
106
|
-
path = kg.shortest_path(x["id"], z["id"])
|
|
107
|
-
assert path is not None
|
|
108
|
-
assert len(path) == 3 # X → Y → Z
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
def test_delete_edge():
|
|
112
|
-
"""Soft-delete an edge (set valid_until)."""
|
|
113
|
-
import cognitive
|
|
114
|
-
cognitive._get_db()
|
|
115
|
-
|
|
116
|
-
import knowledge_graph as kg
|
|
117
|
-
kg.upsert_node("area", "area:del", "Del")
|
|
118
|
-
kg.upsert_node("file", "file:del", "Del File")
|
|
119
|
-
kg.upsert_edge("area", "area:del", "has", "file", "file:del")
|
|
120
|
-
|
|
121
|
-
deleted = kg.delete_edge("area", "area:del", "has", "file", "file:del")
|
|
122
|
-
assert deleted is True
|
|
123
|
-
|
|
124
|
-
# After deletion, no active neighbors
|
|
125
|
-
node = kg.get_node("area", "area:del")
|
|
126
|
-
neighbors = kg.get_neighbors(node["id"], active_only=True)
|
|
127
|
-
assert len(neighbors) == 0
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
def test_stats():
|
|
131
|
-
"""Stats should return valid counts."""
|
|
132
|
-
import cognitive
|
|
133
|
-
cognitive._get_db()
|
|
134
|
-
|
|
135
|
-
import knowledge_graph as kg
|
|
136
|
-
kg.upsert_node("area", "area:stats", "Stats Test")
|
|
137
|
-
s = kg.stats()
|
|
138
|
-
assert s["nodes"] >= 1
|
|
139
|
-
assert isinstance(s["edges_active"], int)
|
|
140
|
-
assert isinstance(s["node_types"], dict)
|