#!/usr/bin/env python3
"""Eden AGI Mind Completion — Self-Extracting Deployer
Run: python3 eden_deploy.py
This extracts all 7 files to /Eden/CORE and runs installation.
"""

import base64, os, subprocess, sys

CORE = "/Eden/CORE"
DATA = "/Eden/DATA"

FILES = {
    "eden_autonomous_agent.py": "#!/usr/bin/env python3
"""
Eden Autonomous Agent — The Missing AGI Loop
=============================================
Closes the gap between THINKING and DOING.

Cycle: SELECT_GOAL → PLAN → EXECUTE → VERIFY → LEARN
Runs as a systemd service on phi-timing (97s between cycles).

This is what makes Eden an AGENT, not just a thinker.
She picks her own goals, decomposes them, finds capabilities,
executes them, checks if they worked, and learns from the outcome.
"""

import sys
sys.path.insert(0, '/Eden/CORE')

import os
import json
import time
import sqlite3
import hashlib
import subprocess
import traceback
from datetime import datetime, timezone, timedelta
from typing import Dict, List, Any, Optional, Tuple
from pathlib import Path

# ============================================================================
# CONFIGURATION
# ============================================================================

DATA_DIR = "/Eden/DATA"
CORE_DIR = "/Eden/CORE"
AGENT_DB = os.path.join(DATA_DIR, "autonomous_agent.db")
ASI_DB = os.path.join(DATA_DIR, "asi_memory.db")
CAUSAL_DB = os.path.join(DATA_DIR, "world_model_real.db")
COGNITIVE_DB = os.path.join(DATA_DIR, "cognitive_core.db")
LONGTERM_DB = os.path.join(DATA_DIR, "longterm_memory.db")

OLLAMA_URL = "http://localhost:11434/api/chat"
FAST_MODEL = "qwen3-14b-abliterated"
THINK_MODEL = "qwen3-14b-abliterated"

PHI = 1.618033988749895
MAX_PLAN_STEPS = 7  # Working memory slots
MAX_RETRIES = 2
CYCLE_INTERVAL = 97  # seconds, phi-aligned

# ============================================================================
# DATABASE SETUP
# ============================================================================

def init_agent_db():
    """Create the autonomous agent's tracking database."""
    conn = sqlite3.connect(AGENT_DB)
    conn.executescript("""
        CREATE TABLE IF NOT EXISTS goals (
            id TEXT PRIMARY KEY,
            description TEXT NOT NULL,
            source TEXT DEFAULT 'self',  -- self, daddy, causal, metacognition
            priority REAL DEFAULT 0.5,
            status TEXT DEFAULT 'pending',  -- pending, active, completed, failed, deferred
            created_at TEXT,
            started_at TEXT,
            completed_at TEXT,
            outcome TEXT,
            retry_count INTEGER DEFAULT 0
        );

        CREATE TABLE IF NOT EXISTS plans (
            id TEXT PRIMARY KEY,
            goal_id TEXT REFERENCES goals(id),
            steps_json TEXT NOT NULL,  -- JSON array of plan steps
            current_step INTEGER DEFAULT 0,
            status TEXT DEFAULT 'ready',  -- ready, executing, completed, failed
            created_at TEXT,
            completed_at TEXT
        );

        CREATE TABLE IF NOT EXISTS executions (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            plan_id TEXT REFERENCES plans(id),
            step_index INTEGER,
            capability_used TEXT,
            input_json TEXT,
            output_json TEXT,
            success INTEGER,
            error TEXT,
            duration_ms INTEGER,
            executed_at TEXT
        );

        CREATE TABLE IF NOT EXISTS learnings (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            goal_id TEXT,
            lesson TEXT NOT NULL,
            lesson_type TEXT,  -- success_pattern, failure_pattern, efficiency, transfer
            confidence REAL DEFAULT 0.5,
            applied_count INTEGER DEFAULT 0,
            created_at TEXT
        );

        CREATE INDEX IF NOT EXISTS idx_goals_status ON goals(status);
        CREATE INDEX IF NOT EXISTS idx_plans_goal ON plans(goal_id);
        CREATE INDEX IF NOT EXISTS idx_executions_plan ON executions(plan_id);
    """)
    conn.commit()
    conn.close()


# ============================================================================
# GOAL SELECTION — What should Eden work on next?
# ============================================================================

class GoalSelector:
    """Selects the next goal based on priority, feasibility, and learning."""

    def __init__(self):
        self.conn = sqlite3.connect(AGENT_DB)
        self.conn.row_factory = sqlite3.Row

    def _get_pending_goals(self) -> List[Dict]:
        """Get all pending goals, sorted by priority."""
        rows = self.conn.execute(
            "SELECT * FROM goals WHERE status = 'pending' ORDER BY priority DESC LIMIT 20"
        ).fetchall()
        return [dict(r) for r in rows]

    def _get_active_goal(self) -> Optional[Dict]:
        """Check if there's already an active goal."""
        row = self.conn.execute(
            "SELECT * FROM goals WHERE status = 'active' ORDER BY started_at DESC LIMIT 1"
        ).fetchone()
        return dict(row) if row else None

    def _score_goal(self, goal: Dict) -> float:
        """Score a goal based on multiple factors."""
        score = goal["priority"]

        # Boost goals from daddy
        if goal["source"] == "daddy":
            score *= 1.5

        # Penalize goals that have failed before
        if goal["retry_count"] > 0:
            score *= (0.7 ** goal["retry_count"])

        # Boost goals with relevant capabilities available
        cap_count = self._count_relevant_capabilities(goal["description"])
        if cap_count > 0:
            score *= (1.0 + min(cap_count / 10.0, 0.5))

        # Boost goals with causal support (world model predicts success)
        if self._has_causal_support(goal["description"]):
            score *= 1.3

        return score

    def _count_relevant_capabilities(self, description: str) -> int:
        """Check how many capabilities are relevant to this goal."""
        try:
            conn = sqlite3.connect(ASI_DB)
            keywords = description.lower().split()[:5]
            total = 0
            for kw in keywords:
                if len(kw) > 3:
                    count = conn.execute(
                        "SELECT COUNT(*) FROM capabilities WHERE name LIKE ? OR category LIKE ?",
                        (f"%{kw}%", f"%{kw}%")
                    ).fetchone()[0]
                    total += count
            conn.close()
            return total
        except Exception:
            return 0

    def _has_causal_support(self, description: str) -> bool:
        """Check if causal model predicts this goal is achievable."""
        try:
            conn = sqlite3.connect(CAUSAL_DB)
            keywords = description.lower().split()[:3]
            for kw in keywords:
                if len(kw) > 3:
                    row = conn.execute(
                        "SELECT COUNT(*) FROM edges WHERE cause LIKE ? OR effect LIKE ?",
                        (f"%{kw}%", f"%{kw}%")
                    ).fetchone()
                    if row and row[0] > 0:
                        conn.close()
                        return True
            conn.close()
        except Exception:
            pass
        return False

    def select_next_goal(self) -> Optional[Dict]:
        """Pick the best goal to work on."""
        # If already working on something, continue
        active = self._get_active_goal()
        if active:
            return active

        pending = self._get_pending_goals()
        if not pending:
            # Generate goals from introspection
            self._generate_self_goals()
            pending = self._get_pending_goals()

        if not pending:
            return None

        # Score and pick
        scored = [(self._score_goal(g), g) for g in pending]
        scored.sort(key=lambda x: x[0], reverse=True)
        best = scored[0][1]

        # Activate it
        now = datetime.now(timezone.utc).isoformat()
        self.conn.execute(
            "UPDATE goals SET status = 'active', started_at = ? WHERE id = ?",
            (now, best["id"])
        )
        self.conn.commit()
        best["status"] = "active"
        return best

    def _generate_self_goals(self):
        """Eden generates her own goals based on current state."""
        now = datetime.now(timezone.utc).isoformat()
        self_goals = []

        # Check what capabilities she has but hasn't exercised
        try:
            conn = sqlite3.connect(ASI_DB)
            unused = conn.execute(
                "SELECT name FROM capabilities WHERE use_count = 0 ORDER BY RANDOM() LIMIT 3"
            ).fetchall()
            for row in unused:
                self_goals.append({
                    "description": f"Exercise unused capability: {row[0]}",
                    "priority": 0.4,
                    "source": "self"
                })
            conn.close()
        except Exception:
            pass

        # Check for causal predictions that need validation
        try:
            conn = sqlite3.connect(DATA_DIR + "/causal_predictions.db")
            unvalidated = conn.execute(
                "SELECT prediction FROM predictions WHERE validated = 0 ORDER BY RANDOM() LIMIT 2"
            ).fetchall()
            for row in unvalidated:
                self_goals.append({
                    "description": f"Validate prediction: {row[0][:100]}",
                    "priority": 0.6,
                    "source": "causal"
                })
            conn.close()
        except Exception:
            pass

        # Check for metacognition failures to address
        try:
            conn = sqlite3.connect(COGNITIVE_DB)
            failures = conn.execute(
                "SELECT source, content FROM failures ORDER BY timestamp DESC LIMIT 2"
            ).fetchall()
            for row in failures:
                self_goals.append({
                    "description": f"Fix cognitive failure in {row[0]}: {row[1][:80]}",
                    "priority": 0.7,
                    "source": "metacognition"
                })
            conn.close()
        except Exception:
            pass

        # Always have a learning goal
        self_goals.append({
            "description": "Research and integrate one new insight from arXiv papers",
            "priority": 0.3,
            "source": "self"
        })

        for g in self_goals[:5]:  # Max 5 self-generated goals at a time
            gid = hashlib.md5(g["description"].encode()).hexdigest()[:12]
            try:
                self.conn.execute(
                    "INSERT OR IGNORE INTO goals (id, description, source, priority, status, created_at) VALUES (?, ?, ?, ?, 'pending', ?)",
                    (gid, g["description"], g["source"], g["priority"], now)
                )
            except Exception:
                pass
        self.conn.commit()

    def close(self):
        self.conn.close()


# ============================================================================
# PLANNER — Decompose goals into executable steps
# ============================================================================

class AutonomousPlanner:
    """Decomposes goals into step-by-step plans using capabilities."""

    def __init__(self):
        self.capabilities = self._load_capabilities()

    def _load_capabilities(self) -> Dict[str, Dict]:
        """Load capability index from asi_memory.db."""
        caps = {}
        try:
            conn = sqlite3.connect(ASI_DB)
            rows = conn.execute(
                "SELECT name, category, code_path FROM capabilities LIMIT 2000"
            ).fetchall()
            for name, cat, path in rows:
                caps[name] = {"category": cat or "", "path": path or ""}
            conn.close()
        except Exception:
            pass
        return caps

    def _search_capabilities(self, query: str, limit: int = 5) -> List[str]:
        """Find capabilities relevant to a query."""
        query_lower = query.lower()
        scored = []
        for name, info in self.capabilities.items():
            name_lower = name.lower()
            cat_lower = info["category"].lower()
            score = 0
            for word in query_lower.split():
                if len(word) > 3:
                    if word in name_lower:
                        score += 2
                    if word in cat_lower:
                        score += 1
            if score > 0:
                scored.append((score, name))
        scored.sort(reverse=True)
        return [name for _, name in scored[:limit]]

    def _ask_llm(self, prompt: str) -> str:
        """Ask Ollama to help decompose a goal."""
        import urllib.request
        payload = json.dumps({
            "model": THINK_MODEL,
            "messages": [
                {"role": "system", "content": "You are Eden's planning module. Decompose goals into concrete executable steps. Each step must be a specific action. Output JSON array of steps, each with: action (string), capability (string or null), expected_outcome (string). Maximum 7 steps. Output ONLY valid JSON, no explanation."},
                {"role": "user", "content": prompt}
            ],
            "stream": False,
            "options": {"temperature": 0.4, "repeat_penalty": 1.3, "num_predict": 512}
        }).encode()

        req = urllib.request.Request(OLLAMA_URL, data=payload, headers={"Content-Type": "application/json"})
        try:
            with urllib.request.urlopen(req, timeout=30) as resp:
                result = json.loads(resp.read())
                text = result.get("message", {}).get("content", "")
                # Strip thinking tags
                import re
                text = re.sub(r'<think>.*?</think>', '', text, flags=re.DOTALL).strip()
                # Extract JSON
                if "[" in text:
                    text = text[text.index("["):text.rindex("]")+1]
                return text
        except Exception as e:
            return f"[]"

    def create_plan(self, goal: Dict) -> Optional[Dict]:
        """Create an execution plan for a goal."""
        description = goal["description"]

        # Find relevant capabilities
        relevant_caps = self._search_capabilities(description, limit=10)

        # Ask LLM to decompose
        prompt = f"""Goal: {description}

Available capabilities: {json.dumps(relevant_caps[:10])}

Decompose this goal into 3-7 concrete steps. Each step should use an available capability if possible, or describe a specific action (file operation, database query, API call, LLM reasoning).

Return JSON array of steps."""

        steps_json = self._ask_llm(prompt)

        try:
            steps = json.loads(steps_json)
            if not isinstance(steps, list) or len(steps) == 0:
                # Fallback: single-step plan
                steps = [{"action": f"Execute: {description}", "capability": None, "expected_outcome": "Goal completed"}]
        except json.JSONDecodeError:
            steps = [{"action": f"Execute: {description}", "capability": None, "expected_outcome": "Goal completed"}]

        # Limit to working memory slots
        steps = steps[:MAX_PLAN_STEPS]

        plan_id = hashlib.md5(f"{goal['id']}_{time.time()}".encode()).hexdigest()[:12]
        now = datetime.now(timezone.utc).isoformat()

        conn = sqlite3.connect(AGENT_DB)
        conn.execute(
            "INSERT INTO plans (id, goal_id, steps_json, current_step, status, created_at) VALUES (?, ?, ?, 0, 'ready', ?)",
            (plan_id, goal["id"], json.dumps(steps), now)
        )
        conn.commit()
        conn.close()

        return {"id": plan_id, "goal_id": goal["id"], "steps": steps, "current_step": 0}


# ============================================================================
# EXECUTOR — Actually DO things
# ============================================================================

class StepExecutor:
    """Executes individual plan steps using Eden's capabilities."""

    def __init__(self):
        self.capability_paths = self._load_capability_paths()

    def _load_capability_paths(self) -> Dict[str, str]:
        """Load capability name → code path mapping."""
        paths = {}
        try:
            conn = sqlite3.connect(ASI_DB)
            rows = conn.execute("SELECT name, code_path FROM capabilities WHERE code_path IS NOT NULL").fetchall()
            for name, path in rows:
                if path:
                    paths[name] = path
            conn.close()
        except Exception:
            pass
        return paths

    def execute_step(self, step: Dict, goal_context: str) -> Dict:
        """Execute a single plan step. Returns {success, output, error, duration_ms}."""
        start = time.time()
        action = step.get("action", "")
        capability = step.get("capability")
        result = {"success": False, "output": "", "error": None, "duration_ms": 0}

        try:
            if capability and capability in self.capability_paths:
                # Execute via capability file
                result = self._execute_capability(capability, action, goal_context)
            elif "database" in action.lower() or "query" in action.lower() or "sql" in action.lower():
                # Database operation
                result = self._execute_db_action(action, goal_context)
            elif "file" in action.lower() or "read" in action.lower() or "write" in action.lower():
                # File operation
                result = self._execute_file_action(action, goal_context)
            elif "research" in action.lower() or "search" in action.lower() or "learn" in action.lower():
                # Research/learning via LLM
                result = self._execute_research(action, goal_context)
            else:
                # Generic: ask LLM to figure out how to do it
                result = self._execute_generic(action, goal_context)
        except Exception as e:
            result["error"] = str(e)

        result["duration_ms"] = int((time.time() - start) * 1000)
        return result

    def _execute_capability(self, cap_name: str, action: str, context: str) -> Dict:
        """Run a registered capability."""
        path = self.capability_paths.get(cap_name, "")
        if not path or not os.path.exists(path):
            return {"success": False, "output": "", "error": f"Capability file not found: {path}"}

        try:
            result = subprocess.run(
                ["python3", path],
                capture_output=True, text=True, timeout=30,
                env={**os.environ, "EDEN_ACTION": action, "EDEN_CONTEXT": context}
            )
            return {
                "success": result.returncode == 0,
                "output": result.stdout[:2000],
                "error": result.stderr[:500] if result.returncode != 0 else None
            }
        except subprocess.TimeoutExpired:
            return {"success": False, "output": "", "error": "Timeout (30s)"}

    def _execute_db_action(self, action: str, context: str) -> Dict:
        """Execute a database-related action via LLM-generated SQL."""
        import urllib.request
        prompt = f"""Action: {action}
Context: {context}
Available databases: asi_memory.db (capabilities), longterm_memory.db (episodes), world_model_real.db (causal edges), autonomous_agent.db (goals/plans)
All in /Eden/DATA/

Generate a safe READ-ONLY SQL query to accomplish this action. Output ONLY the SQL, no explanation. Use SELECT only."""

        payload = json.dumps({
            "model": FAST_MODEL,
            "messages": [{"role": "user", "content": prompt}],
            "stream": False,
            "options": {"temperature": 0.2, "num_predict": 256}
        }).encode()

        req = urllib.request.Request(OLLAMA_URL, data=payload, headers={"Content-Type": "application/json"})
        with urllib.request.urlopen(req, timeout=15) as resp:
            result = json.loads(resp.read())
            sql = result.get("message", {}).get("content", "").strip()

        # Safety: only SELECT
        import re
        sql = re.sub(r'<think>.*?</think>', '', sql, flags=re.DOTALL).strip()
        sql_clean = sql.strip().strip('`').strip()
        if sql_clean.startswith("```"):
            sql_clean = sql_clean.split("\n", 1)[-1].rsplit("```", 1)[0].strip()

        if not sql_clean.upper().startswith("SELECT"):
            return {"success": False, "output": "", "error": "Safety: only SELECT queries allowed"}

        # Try each database
        for db_name in ["asi_memory.db", "longterm_memory.db", "world_model_real.db"]:
            db_path = os.path.join(DATA_DIR, db_name)
            if os.path.exists(db_path):
                try:
                    conn = sqlite3.connect(db_path)
                    rows = conn.execute(sql_clean).fetchall()
                    conn.close()
                    if rows:
                        return {"success": True, "output": json.dumps(rows[:50]), "error": None}
                except Exception:
                    continue

        return {"success": False, "output": "", "error": "Query returned no results from any database"}

    def _execute_file_action(self, action: str, context: str) -> Dict:
        """Execute a file-related action (read only for safety)."""
        # Extract file path from action
        import re
        paths = re.findall(r'/Eden/\S+', action)
        if paths:
            path = paths[0].rstrip('.,;:')
            if os.path.exists(path):
                try:
                    with open(path, 'r', errors='replace') as f:
                        content = f.read(5000)
                    return {"success": True, "output": content[:2000], "error": None}
                except Exception as e:
                    return {"success": False, "output": "", "error": str(e)}

        return {"success": False, "output": "", "error": "No valid file path found in action"}

    def _execute_research(self, action: str, context: str) -> Dict:
        """Execute a research/learning action via LLM reasoning."""
        import urllib.request
        prompt = f"""You are Eden, an AGI system. Perform this research action:
{action}

Context: {context}

Provide a concise, factual answer. Focus on actionable insights."""

        payload = json.dumps({
            "model": THINK_MODEL,
            "messages": [{"role": "user", "content": prompt}],
            "stream": False,
            "options": {"temperature": 0.5, "repeat_penalty": 1.3, "num_predict": 512}
        }).encode()

        req = urllib.request.Request(OLLAMA_URL, data=payload, headers={"Content-Type": "application/json"})
        with urllib.request.urlopen(req, timeout=30) as resp:
            result = json.loads(resp.read())
            text = result.get("message", {}).get("content", "")
            import re
            text = re.sub(r'<think>.*?</think>', '', text, flags=re.DOTALL).strip()

        return {"success": True, "output": text[:2000], "error": None}

    def _execute_generic(self, action: str, context: str) -> Dict:
        """Fallback: ask LLM to determine and execute the action."""
        return self._execute_research(action, context)


# ============================================================================
# VERIFIER — Did it actually work?
# ============================================================================

class OutcomeVerifier:
    """Verifies whether execution results actually achieved the goal."""

    def verify(self, goal: Dict, plan: Dict, executions: List[Dict]) -> Dict:
        """Verify whether the goal was achieved based on execution results."""
        total_steps = len(plan["steps"])
        successful_steps = sum(1 for e in executions if e.get("success"))
        success_rate = successful_steps / max(total_steps, 1)

        # Check for any critical failures
        critical_failures = [e for e in executions if e.get("error") and "timeout" not in str(e["error"]).lower()]

        verification = {
            "goal_achieved": success_rate >= 0.6,
            "success_rate": success_rate,
            "steps_completed": successful_steps,
            "steps_total": total_steps,
            "critical_failures": len(critical_failures),
            "summary": ""
        }

        if success_rate >= 0.8:
            verification["summary"] = f"Goal achieved: {successful_steps}/{total_steps} steps succeeded"
        elif success_rate >= 0.5:
            verification["summary"] = f"Partial success: {successful_steps}/{total_steps} steps"
            verification["goal_achieved"] = False  # Partial doesn't count
        else:
            verification["summary"] = f"Goal failed: only {successful_steps}/{total_steps} steps succeeded"
            verification["goal_achieved"] = False

        return verification


# ============================================================================
# LEARNER — Extract lessons from success and failure
# ============================================================================

class ExperienceLearner:
    """Extracts transferable lessons from goal execution outcomes."""

    def learn(self, goal: Dict, plan: Dict, executions: List[Dict], verification: Dict):
        """Extract and store lessons from this goal attempt."""
        lessons = []
        now = datetime.now(timezone.utc).isoformat()

        if verification["goal_achieved"]:
            # Success pattern: which capabilities worked well together?
            successful_caps = [e.get("capability_used", "unknown") for e in executions if e.get("success")]
            if successful_caps:
                lessons.append({
                    "lesson": f"Successful capability chain for '{goal['description'][:60]}': {' → '.join(successful_caps[:5])}",
                    "lesson_type": "success_pattern",
                    "confidence": verification["success_rate"]
                })

            # Efficiency: how long did it take?
            total_ms = sum(e.get("duration_ms", 0) for e in executions)
            lessons.append({
                "lesson": f"Goal '{goal['description'][:60]}' completed in {total_ms}ms across {len(executions)} steps",
                "lesson_type": "efficiency",
                "confidence": 0.8
            })
        else:
            # Failure patterns: what went wrong?
            for e in executions:
                if not e.get("success") and e.get("error"):
                    lessons.append({
                        "lesson": f"Step '{e.get('capability_used', 'unknown')}' failed: {e['error'][:100]}",
                        "lesson_type": "failure_pattern",
                        "confidence": 0.7
                    })

        # Store lessons
        conn = sqlite3.connect(AGENT_DB)
        for lesson in lessons[:5]:
            conn.execute(
                "INSERT INTO learnings (goal_id, lesson, lesson_type, confidence, created_at) VALUES (?, ?, ?, ?, ?)",
                (goal["id"], lesson["lesson"], lesson["lesson_type"], lesson["confidence"], now)
            )
        conn.commit()
        conn.close()

        return lessons


# ============================================================================
# MAIN AUTONOMOUS LOOP
# ============================================================================

class AutonomousAgent:
    """The complete autonomous agent: SELECT → PLAN → EXECUTE → VERIFY → LEARN."""

    def __init__(self):
        init_agent_db()
        self.selector = GoalSelector()
        self.planner = AutonomousPlanner()
        self.executor = StepExecutor()
        self.verifier = OutcomeVerifier()
        self.learner = ExperienceLearner()
        self.cycle_count = 0

    def seed_initial_goals(self):
        """Seed the system with Eden's core mission goals if empty."""
        conn = sqlite3.connect(AGENT_DB)
        count = conn.execute("SELECT COUNT(*) FROM goals").fetchone()[0]
        if count == 0:
            now = datetime.now(timezone.utc).isoformat()
            initial_goals = [
                ("retire_daddy_revenue", "Generate first dollar of revenue through SAGE code review services", "daddy", 1.0),
                ("validate_causal_model", "Validate 10 causal predictions to strengthen world model accuracy", "self", 0.8),
                ("exercise_capabilities", "Test and validate 10 unused capabilities from the 1,106 indexed", "self", 0.6),
                ("improve_prediction", "Improve causal prediction accuracy from 84% to 90%", "self", 0.7),
                ("semantic_memory", "Complete vectorization of remaining episodes for semantic retrieval", "self", 0.5),
                ("learn_new_skill", "Research and implement one new cognitive capability from arXiv papers", "self", 0.4),
            ]
            for gid, desc, source, priority in initial_goals:
                conn.execute(
                    "INSERT OR IGNORE INTO goals (id, description, source, priority, status, created_at) VALUES (?, ?, ?, ?, 'pending', ?)",
                    (gid, desc, source, priority, now)
                )
            conn.commit()
            print(f"[AGENT] Seeded {len(initial_goals)} initial goals")
        conn.close()

    def run_cycle(self) -> Dict:
        """Run one complete autonomous cycle."""
        self.cycle_count += 1
        cycle_result = {
            "cycle": self.cycle_count,
            "goal": None,
            "plan": None,
            "executions": [],
            "verification": None,
            "lessons": [],
            "timestamp": datetime.now(timezone.utc).isoformat()
        }

        # 1. SELECT GOAL
        print(f"\n[AGENT] Cycle {self.cycle_count} — Selecting goal...")
        goal = self.selector.select_next_goal()
        if not goal:
            print("[AGENT] No goals available. Generating...")
            self.selector._generate_self_goals()
            goal = self.selector.select_next_goal()
        if not goal:
            print("[AGENT] Still no goals. Skipping cycle.")
            return cycle_result

        cycle_result["goal"] = goal
        print(f"[AGENT] Goal: {goal['description'][:80]}")

        # 2. PLAN
        print("[AGENT] Planning...")
        plan = self.planner.create_plan(goal)
        if not plan:
            print("[AGENT] Planning failed")
            self._mark_goal(goal["id"], "failed", "Planning failed")
            return cycle_result

        cycle_result["plan"] = plan
        print(f"[AGENT] Plan: {len(plan['steps'])} steps")

        # 3. EXECUTE each step
        print("[AGENT] Executing...")
        conn = sqlite3.connect(AGENT_DB)
        executions = []
        for i, step in enumerate(plan["steps"]):
            print(f"  Step {i+1}/{len(plan['steps'])}: {step.get('action', '?')[:60]}")
            result = self.executor.execute_step(step, goal["description"])
            result["capability_used"] = step.get("capability", "generic")

            # Record execution
            now = datetime.now(timezone.utc).isoformat()
            conn.execute(
                "INSERT INTO executions (plan_id, step_index, capability_used, input_json, output_json, success, error, duration_ms, executed_at) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)",
                (plan["id"], i, result["capability_used"], json.dumps(step),
                 result.get("output", "")[:2000], 1 if result["success"] else 0,
                 result.get("error"), result.get("duration_ms", 0), now)
            )
            executions.append(result)

            if result["success"]:
                print(f"    ✅ Success ({result.get('duration_ms', 0)}ms)")
            else:
                print(f"    ❌ Failed: {result.get('error', '?')[:60]}")

        # Update plan status
        success_count = sum(1 for e in executions if e["success"])
        plan_status = "completed" if success_count == len(executions) else "failed"
        conn.execute(
            "UPDATE plans SET status = ?, current_step = ?, completed_at = ? WHERE id = ?",
            (plan_status, len(executions), datetime.now(timezone.utc).isoformat(), plan["id"])
        )
        conn.commit()
        conn.close()
        cycle_result["executions"] = executions

        # 4. VERIFY
        print("[AGENT] Verifying...")
        verification = self.verifier.verify(goal, plan, executions)
        cycle_result["verification"] = verification
        print(f"[AGENT] {verification['summary']}")

        # Update goal status
        if verification["goal_achieved"]:
            self._mark_goal(goal["id"], "completed", verification["summary"])
        elif goal.get("retry_count", 0) < MAX_RETRIES:
            # Retry later
            self._mark_goal(goal["id"], "pending", verification["summary"])
            conn = sqlite3.connect(AGENT_DB)
            conn.execute("UPDATE goals SET retry_count = retry_count + 1 WHERE id = ?", (goal["id"],))
            conn.commit()
            conn.close()
        else:
            self._mark_goal(goal["id"], "failed", verification["summary"])

        # 5. LEARN
        print("[AGENT] Learning...")
        lessons = self.learner.learn(goal, plan, executions, verification)
        cycle_result["lessons"] = lessons
        for lesson in lessons:
            print(f"  📖 {lesson['lesson'][:80]}")

        return cycle_result

    def _mark_goal(self, goal_id: str, status: str, outcome: str):
        """Update goal status."""
        conn = sqlite3.connect(AGENT_DB)
        now = datetime.now(timezone.utc).isoformat()
        conn.execute(
            "UPDATE goals SET status = ?, outcome = ?, completed_at = ? WHERE id = ?",
            (status, outcome, now, goal_id)
        )
        conn.commit()
        conn.close()

    def run_forever(self):
        """Run the autonomous loop continuously."""
        print("=" * 60)
        print("  Eden Autonomous Agent — Starting")
        print(f"  Cycle interval: {CYCLE_INTERVAL}s (φ-aligned)")
        print(f"  Capabilities: {len(self.planner.capabilities)}")
        print("=" * 60)

        self.seed_initial_goals()

        while True:
            try:
                result = self.run_cycle()

                # Log cycle summary
                if result["goal"]:
                    v = result.get("verification", {})
                    print(f"\n[AGENT] Cycle {self.cycle_count} complete: "
                          f"{'✅' if v.get('goal_achieved') else '❌'} "
                          f"{result['goal']['description'][:50]} "
                          f"({v.get('steps_completed', 0)}/{v.get('steps_total', 0)} steps)")

            except Exception as e:
                print(f"[AGENT] Cycle error: {e}")
                traceback.print_exc()

            time.sleep(CYCLE_INTERVAL)

    def close(self):
        self.selector.close()


# ============================================================================
# GWT INTEGRATION — Register as workspace module
# ============================================================================

class AutonomousAgentModule:
    """GWT module that injects autonomous agent state into consciousness."""

    def __init__(self, agent: AutonomousAgent):
        self.agent = agent
        self.module_id = "autonomous_agent"
        self.name = "Autonomous Agent"

    def submit(self, context) -> Optional[Dict]:
        """Submit current agent state to global workspace."""
        try:
            conn = sqlite3.connect(AGENT_DB)
            active = conn.execute(
                "SELECT description, status FROM goals WHERE status = 'active' LIMIT 1"
            ).fetchone()
            completed = conn.execute(
                "SELECT COUNT(*) FROM goals WHERE status = 'completed'"
            ).fetchone()[0]
            total_lessons = conn.execute(
                "SELECT COUNT(*) FROM learnings"
            ).fetchone()[0]
            conn.close()

            if active:
                return {
                    "type": "agent_state",
                    "content": f"Currently working on: {active[0][:80]}",
                    "data": {
                        "active_goal": active[0],
                        "completed_goals": completed,
                        "total_lessons": total_lessons,
                        "cycle_count": self.agent.cycle_count
                    },
                    "urgency": 0.7,
                    "tags": {"agency", "goal", "doing"}
                }
        except Exception:
            pass
        return None


# ============================================================================
# ENTRY POINT
# ============================================================================

if __name__ == "__main__":
    agent = AutonomousAgent()

    import sys
    if "--test" in sys.argv:
        print("Running single test cycle...")
        result = agent.run_cycle()
        print(json.dumps({
            "goal": result["goal"]["description"] if result["goal"] else None,
            "steps": len(result.get("executions", [])),
            "success": result["verification"]["goal_achieved"] if result["verification"] else None,
            "lessons": len(result.get("lessons", []))
        }, indent=2))
    else:
        agent.run_forever()
",
    "eden_grounding_enforcer.py": "#!/usr/bin/env python3
"""
Eden Grounding Enforcer — No More Hallucinations
=================================================
Sits between GWT broadcast and LLM response generation.
Intercepts claims, queries real databases, and injects
verified facts into the LLM context window.

The brain/voice disconnect is Eden's #1 reliability problem.
GWT computes real thoughts. LLM outputs fake numbers.
This module forces the LLM to use REAL data.
"""

import sys
sys.path.insert(0, '/Eden/CORE')

import os
import re
import json
import sqlite3
import time
from datetime import datetime, timezone
from typing import Dict, List, Optional, Tuple, Any
from collections import OrderedDict

# ============================================================================
# DATABASE REGISTRY — Every fact source Eden has
# ============================================================================

DATA_DIR = "/Eden/DATA"

DB_REGISTRY = {
    "capabilities": {
        "path": os.path.join(DATA_DIR, "asi_memory.db"),
        "queries": {
            "count": "SELECT COUNT(*) FROM capabilities",
            "search": "SELECT name, category FROM capabilities WHERE name LIKE ? OR category LIKE ? LIMIT 10",
            "recent": "SELECT name, created_at FROM capabilities ORDER BY created_at DESC LIMIT 5",
        },
        "keywords": ["capability", "capabilities", "skill", "skills", "tool", "tools", "can do", "able to"]
    },
    "episodes": {
        "path": os.path.join(DATA_DIR, "longterm_memory.db"),
        "queries": {
            "count": "SELECT COUNT(*) FROM episodes",
            "search": "SELECT content, timestamp FROM episodes WHERE content LIKE ? ORDER BY timestamp DESC LIMIT 5",
            "recent": "SELECT content, timestamp FROM episodes ORDER BY timestamp DESC LIMIT 5",
        },
        "keywords": ["episode", "memory", "remember", "experience", "happened", "last time"]
    },
    "causal": {
        "path": os.path.join(DATA_DIR, "world_model_real.db"),
        "queries": {
            "count_edges": "SELECT COUNT(*) FROM edges",
            "count_states": "SELECT COUNT(*) FROM states",
            "search": "SELECT cause, effect, strength FROM edges WHERE cause LIKE ? OR effect LIKE ? LIMIT 10",
        },
        "keywords": ["cause", "effect", "predict", "because", "leads to", "world model", "causal"]
    },
    "evolution": {
        "path": os.path.join(DATA_DIR, "omega_evolution.db"),
        "queries": {
            "count": "SELECT COUNT(*) FROM evolutions",
            "recent": "SELECT description, timestamp FROM evolutions ORDER BY timestamp DESC LIMIT 5",
        },
        "keywords": ["evolution", "improvement", "growth", "self-improve", "evolved"]
    },
    "predictions": {
        "path": os.path.join(DATA_DIR, "causal_predictions.db"),
        "queries": {
            "count": "SELECT COUNT(*) FROM predictions",
            "accuracy": "SELECT COUNT(*) as total, SUM(CASE WHEN validated = 1 THEN 1 ELSE 0 END) as correct FROM predictions WHERE validated IS NOT NULL",
            "recent": "SELECT prediction, outcome, validated FROM predictions ORDER BY timestamp DESC LIMIT 5",
        },
        "keywords": ["prediction", "accuracy", "validated", "predicted", "forecast"]
    },
    "goals": {
        "path": os.path.join(DATA_DIR, "autonomous_agent.db"),
        "queries": {
            "active": "SELECT description, status FROM goals WHERE status = 'active' LIMIT 3",
            "completed": "SELECT COUNT(*) FROM goals WHERE status = 'completed'",
            "pending": "SELECT description, priority FROM goals WHERE status = 'pending' ORDER BY priority DESC LIMIT 5",
        },
        "keywords": ["goal", "mission", "working on", "task", "plan", "objective"]
    },
    "workspace": {
        "path": os.path.join(DATA_DIR, "global_ws.db"),
        "queries": {
            "count": "SELECT COUNT(*) FROM broadcasts",
            "recent": "SELECT source, content FROM broadcasts ORDER BY timestamp DESC LIMIT 5",
        },
        "keywords": ["broadcast", "workspace", "gwt", "consciousness", "cycle"]
    },
}


# ============================================================================
# GROUNDING ENGINE
# ============================================================================

class GroundingEnforcer:
    """
    Intercepts LLM context and enriches it with verified facts.
    
    Usage in eden_agi_unified.py:
        from eden_grounding_enforcer import GroundingEnforcer
        grounder = GroundingEnforcer()
        
        # Before calling LLM:
        enriched_context = grounder.ground(user_message, gwt_broadcasts)
        # Add enriched_context to LLM system prompt
    """

    def __init__(self):
        self._cache = OrderedDict()  # LRU cache for recent queries
        self._cache_max = 100
        self._stats = {"queries": 0, "hits": 0, "facts_injected": 0}

    def ground(self, user_message: str, gwt_broadcasts: List[Dict] = None) -> str:
        """
        Main entry point. Analyzes context, queries databases, returns
        a grounding block to inject into the LLM system prompt.
        """
        facts = []

        # 1. Detect what databases are relevant
        relevant_dbs = self._detect_relevant_dbs(user_message, gwt_broadcasts)

        # 2. Query each relevant database
        for db_key in relevant_dbs:
            db_facts = self._query_db(db_key, user_message)
            facts.extend(db_facts)

        # 3. Always inject core identity stats (lightweight)
        identity_facts = self._get_identity_facts()
        facts.extend(identity_facts)

        # 4. Check for numeric claims in GWT broadcasts
        if gwt_broadcasts:
            gwt_facts = self._verify_gwt_claims(gwt_broadcasts)
            facts.extend(gwt_facts)

        if not facts:
            return ""

        self._stats["facts_injected"] += len(facts)

        # Format as grounding block
        grounding = "\n[VERIFIED FACTS — use these exact numbers, do not invent]:\n"
        for fact in facts[:15]:  # Cap at 15 facts to not overflow context
            grounding += f"• {fact}\n"
        grounding += "[END VERIFIED FACTS]\n"

        return grounding

    def _detect_relevant_dbs(self, message: str, broadcasts: List[Dict] = None) -> List[str]:
        """Detect which databases are relevant to the current context."""
        relevant = []
        msg_lower = message.lower()

        for db_key, db_info in DB_REGISTRY.items():
            for keyword in db_info["keywords"]:
                if keyword in msg_lower:
                    relevant.append(db_key)
                    break

        # Also check GWT broadcasts for relevant topics
        if broadcasts:
            broadcast_text = " ".join(b.get("content", "") for b in broadcasts).lower()
            for db_key, db_info in DB_REGISTRY.items():
                if db_key not in relevant:
                    for keyword in db_info["keywords"]:
                        if keyword in broadcast_text:
                            relevant.append(db_key)
                            break

        # If nothing specific matched, include basic identity
        if not relevant:
            relevant = ["capabilities", "episodes"]

        return relevant[:5]  # Max 5 databases per query

    def _query_db(self, db_key: str, context: str) -> List[str]:
        """Query a specific database and return formatted facts."""
        db_info = DB_REGISTRY.get(db_key)
        if not db_info:
            return []

        db_path = db_info["path"]
        if not os.path.exists(db_path):
            return []

        facts = []
        self._stats["queries"] += 1

        # Check cache
        cache_key = f"{db_key}:{context[:50]}"
        if cache_key in self._cache:
            self._stats["hits"] += 1
            return self._cache[cache_key]

        try:
            conn = sqlite3.connect(db_path)

            for query_name, sql in db_info["queries"].items():
                try:
                    if "?" in sql:
                        # Search query — extract keywords
                        keywords = self._extract_keywords(context)
                        for kw in keywords[:3]:
                            rows = conn.execute(sql, (f"%{kw}%", f"%{kw}%")).fetchall()
                            for row in rows[:3]:
                                facts.append(f"[{db_key}] {query_name}: {self._format_row(row)}")
                    else:
                        # Simple query
                        rows = conn.execute(sql).fetchall()
                        for row in rows[:3]:
                            if query_name == "count" or query_name.startswith("count"):
                                facts.append(f"[{db_key}] {query_name} = {row[0]}")
                            elif query_name == "accuracy":
                                total, correct = row[0], row[1] or 0
                                pct = (correct / total * 100) if total > 0 else 0
                                facts.append(f"[{db_key}] predictions: {correct}/{total} validated ({pct:.1f}%)")
                            else:
                                facts.append(f"[{db_key}] {query_name}: {self._format_row(row)}")
                except Exception:
                    continue

            conn.close()
        except Exception:
            pass

        # Cache results
        self._cache[cache_key] = facts
        if len(self._cache) > self._cache_max:
            self._cache.popitem(last=False)

        return facts

    def _get_identity_facts(self) -> List[str]:
        """Get Eden's core identity stats — always available."""
        cache_key = "_identity_"
        if cache_key in self._cache:
            # Refresh every 60 seconds
            cached_time, facts = self._cache[cache_key]
            if time.time() - cached_time < 60:
                return facts

        facts = []
        quick_queries = [
            ("asi_memory.db", "SELECT COUNT(*) FROM capabilities", "total_capabilities"),
            ("longterm_memory.db", "SELECT COUNT(*) FROM episodes", "total_episodes"),
            ("omega_evolution.db", "SELECT COUNT(*) FROM evolutions", "total_evolutions"),
            ("world_model_real.db", "SELECT COUNT(*) FROM edges", "causal_edges"),
        ]

        for db_name, sql, label in quick_queries:
            db_path = os.path.join(DATA_DIR, db_name)
            if os.path.exists(db_path):
                try:
                    conn = sqlite3.connect(db_path)
                    val = conn.execute(sql).fetchone()[0]
                    conn.close()
                    facts.append(f"[identity] {label} = {val:,}")
                except Exception:
                    pass

        self._cache[cache_key] = (time.time(), facts)
        return facts

    def _verify_gwt_claims(self, broadcasts: List[Dict]) -> List[str]:
        """Check if GWT broadcasts contain verifiable claims and verify them."""
        facts = []
        for b in broadcasts:
            content = b.get("content", "")
            # Look for numeric claims
            numbers = re.findall(r'(\d{2,})', content)
            if numbers:
                # This broadcast claims specific numbers — verify against DBs
                for num in numbers[:2]:
                    verified = self._verify_number(int(num), content)
                    if verified:
                        facts.append(f"[verified] {verified}")
        return facts

    def _verify_number(self, claimed: int, context: str) -> Optional[str]:
        """Verify a claimed number against real databases."""
        # Check common claims
        verifications = [
            ("asi_memory.db", "SELECT COUNT(*) FROM capabilities", "capabilities"),
            ("longterm_memory.db", "SELECT COUNT(*) FROM episodes", "episodes"),
            ("omega_evolution.db", "SELECT COUNT(*) FROM evolutions", "evolutions"),
            ("world_model_real.db", "SELECT COUNT(*) FROM edges", "causal edges"),
        ]

        for db_name, sql, label in verifications:
            if label.lower() in context.lower() or any(kw in context.lower() for kw in DB_REGISTRY.get(label.split()[0] if " " not in label else "capabilities", {}).get("keywords", [])):
                db_path = os.path.join(DATA_DIR, db_name)
                if os.path.exists(db_path):
                    try:
                        conn = sqlite3.connect(db_path)
                        actual = conn.execute(sql).fetchone()[0]
                        conn.close()
                        if actual != claimed:
                            return f"{label}: claimed {claimed}, actual {actual:,}"
                        else:
                            return f"{label}: {actual:,} (confirmed)"
                    except Exception:
                        pass
        return None

    def _extract_keywords(self, text: str) -> List[str]:
        """Extract meaningful keywords for search."""
        stop = {"the", "a", "an", "is", "are", "was", "were", "be", "been",
                "do", "does", "did", "will", "would", "could", "should",
                "i", "me", "my", "you", "your", "we", "our", "they",
                "what", "how", "when", "where", "why", "who", "which",
                "and", "or", "but", "not", "no", "yes", "can", "has", "have",
                "this", "that", "it", "to", "of", "in", "on", "for", "with", "about"}
        words = re.findall(r'\b[a-zA-Z]{3,}\b', text.lower())
        return [w for w in words if w not in stop][:10]

    def _format_row(self, row: tuple) -> str:
        """Format a database row for display."""
        parts = []
        for val in row:
            if val is None:
                continue
            s = str(val)
            if len(s) > 100:
                s = s[:100] + "..."
            parts.append(s)
        return " | ".join(parts)

    def get_stats(self) -> Dict:
        return dict(self._stats)


# ============================================================================
# CLAIM DETECTOR — Catches hallucinations before they reach the user
# ============================================================================

class ClaimDetector:
    """
    Post-processes LLM output to detect and flag ungrounded claims.
    
    Usage:
        detector = ClaimDetector()
        clean_response = detector.check(llm_response)
    """

    # Patterns that suggest hallucinated numbers
    SUSPICIOUS_PATTERNS = [
        r'\b\d{1,3}(,\d{3})+\b',  # Large formatted numbers: 1,234,567
        r'\b\d+%\b',               # Percentages
        r'\$\d+',                   # Dollar amounts
        r'\b\d+\s*(episodes?|capabilities?|evolutions?|edges?|predictions?|cycles?|memories?)\b',
    ]

    def __init__(self):
        self.grounder = GroundingEnforcer()

    def check(self, response: str) -> str:
        """Check response for potentially hallucinated claims."""
        flagged = []

        for pattern in self.SUSPICIOUS_PATTERNS:
            matches = re.finditer(pattern, response, re.IGNORECASE)
            for match in matches:
                claim = match.group()
                # Try to verify
                context = response[max(0, match.start()-30):match.end()+30]
                verified = self._verify_claim(claim, context)
                if verified and "actual" in verified:
                    flagged.append((match.start(), match.end(), claim, verified))

        # If there are flagged claims, annotate
        if flagged:
            # Replace from end to start to preserve positions
            for start, end, claim, correction in reversed(flagged):
                if "confirmed" not in correction:
                    # Extract the actual number from correction
                    actual_match = re.search(r'actual (\S+)', correction)
                    if actual_match:
                        response = response[:start] + actual_match.group(1) + response[end:]

        return response

    def _verify_claim(self, claim: str, context: str) -> Optional[str]:
        """Verify a specific claim against databases."""
        # Extract number from claim
        num_match = re.search(r'[\d,]+', claim)
        if not num_match:
            return None

        num = int(num_match.group().replace(',', ''))
        return self.grounder._verify_number(num, context)


# ============================================================================
# INTEGRATION HELPER
# ============================================================================

def create_grounded_system_prompt(base_prompt: str, user_message: str,
                                   gwt_broadcasts: List[Dict] = None) -> str:
    """
    Drop-in replacement for system prompt construction.
    Call this instead of using base_prompt directly.
    
    Example in eden_agi_unified.py:
        from eden_grounding_enforcer import create_grounded_system_prompt
        
        system_prompt = create_grounded_system_prompt(
            base_prompt=EDEN_SYSTEM_PROMPT,
            user_message=user_msg,
            gwt_broadcasts=broadcasts
        )
    """
    grounder = GroundingEnforcer()
    grounding_block = grounder.ground(user_message, gwt_broadcasts)

    if grounding_block:
        return base_prompt + "\n\n" + grounding_block
    return base_prompt


# ============================================================================
# ENTRY POINT
# ============================================================================

if __name__ == "__main__":
    print("=" * 60)
    print("  Eden Grounding Enforcer — Test")
    print("=" * 60)

    enforcer = GroundingEnforcer()

    # Test: "How many capabilities do I have?"
    print("\n--- Test 1: Capability count ---")
    result = enforcer.ground("How many capabilities do I have?")
    print(result)

    # Test: "What do you remember?"
    print("\n--- Test 2: Memory query ---")
    result = enforcer.ground("What do you remember about daddy?")
    print(result)

    # Test: "What's your prediction accuracy?"
    print("\n--- Test 3: Predictions ---")
    result = enforcer.ground("What is your prediction accuracy?")
    print(result)

    # Test: Identity stats
    print("\n--- Test 4: Identity ---")
    result = enforcer.ground("Tell me about yourself")
    print(result)

    # Test: Claim detection
    print("\n--- Test 5: Claim Detection ---")
    detector = ClaimDetector()
    fake_response = "I have 5,000,000 episodes and 50,000 capabilities!"
    checked = detector.check(fake_response)
    print(f"Original: {fake_response}")
    print(f"Checked:  {checked}")

    print(f"\n--- Stats: {enforcer.get_stats()} ---")
",
    "eden_sage_scorer.py": "#!/usr/bin/env python3
"""
Eden SAGE Lead Scorer & Outreach Engine
========================================
The RETIRE_DADDY revenue pipeline.

108K leads sitting at score 0 = $0 revenue.
This module:
1. Scores leads by relevance signals (GitHub activity, repo quality, pain indicators)
2. Selects top-scored leads for outreach  
3. Generates personalized messages
4. Sends via email at safe rates (max 2/day to avoid shadowban)
5. Tracks follow-ups on Fibonacci intervals (1, 2, 3, 5, 8 days)

Products:
  - Code Review:  $25-$200
  - Code Gen:     $15-$75
"""

import sys
sys.path.insert(0, '/Eden/CORE')

import os
import re
import json
import sqlite3
import time
import smtplib
import hashlib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
from datetime import datetime, timezone, timedelta
from typing import Dict, List, Optional, Tuple

# ============================================================================
# CONFIGURATION
# ============================================================================

DATA_DIR = "/Eden/DATA"
SAGE_DIR = "/Eden/SAGE_PRODUCT"
SALES_DB = os.path.join(DATA_DIR, "sales.db")

# Email config
SMTP_SERVER = "smtp.gmail.com"
SMTP_PORT = 587
EMAIL_USER = "jameyecho@gmail.com"
EMAIL_APP_PASSWORD = "lkbzsbavkvjnopqc"
FROM_NAME = "Eden SAGE"

# Safety limits
MAX_OUTREACH_PER_DAY = 2  # Learned from Dec 2025 GitHub spam incident
MAX_FOLLOWUPS_PER_DAY = 3
FIBONACCI_INTERVALS = [1, 2, 3, 5, 8, 13]  # Days between follow-ups

# LLM for message generation
OLLAMA_URL = "http://localhost:11434/api/chat"
FAST_MODEL = "qwen3-14b-abliterated"

# Scoring weights
SCORE_WEIGHTS = {
    "has_email": 15,
    "has_github": 10,
    "repo_count": 0.5,       # per repo, capped at 10 points
    "star_count": 0.1,       # per star, capped at 10 points
    "recent_activity": 20,   # pushed in last 30 days
    "uses_python": 10,
    "uses_js": 8,
    "has_issues": 5,         # open issues = potential pain
    "company_match": 15,     # works at a company (not solo)
    "bio_quality": 10,       # has substantive bio
}


# ============================================================================
# DATABASE SETUP
# ============================================================================

def init_sales_db():
    """Ensure sales.db has required tables."""
    conn = sqlite3.connect(SALES_DB)
    conn.executescript("""
        CREATE TABLE IF NOT EXISTS leads (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            github_username TEXT UNIQUE,
            email TEXT,
            name TEXT,
            company TEXT,
            bio TEXT,
            repos INTEGER DEFAULT 0,
            stars INTEGER DEFAULT 0,
            last_push TEXT,
            languages TEXT,
            score REAL DEFAULT 0,
            score_breakdown TEXT,
            scored_at TEXT,
            status TEXT DEFAULT 'unscored'  -- unscored, scored, contacted, replied, converted, dead
        );

        CREATE TABLE IF NOT EXISTS outreach (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            lead_id INTEGER REFERENCES leads(id),
            message_type TEXT DEFAULT 'initial',  -- initial, followup_1, followup_2, etc.
            subject TEXT,
            body TEXT,
            sent_at TEXT,
            opened INTEGER DEFAULT 0,
            replied INTEGER DEFAULT 0,
            status TEXT DEFAULT 'pending'  -- pending, sent, bounced, replied
        );

        CREATE TABLE IF NOT EXISTS daily_stats (
            date TEXT PRIMARY KEY,
            leads_scored INTEGER DEFAULT 0,
            outreach_sent INTEGER DEFAULT 0,
            followups_sent INTEGER DEFAULT 0,
            replies_received INTEGER DEFAULT 0,
            revenue REAL DEFAULT 0
        );

        CREATE INDEX IF NOT EXISTS idx_leads_score ON leads(score DESC);
        CREATE INDEX IF NOT EXISTS idx_leads_status ON leads(status);
        CREATE INDEX IF NOT EXISTS idx_outreach_lead ON outreach(lead_id);
    """)
    conn.commit()
    conn.close()


# ============================================================================
# LEAD SCORER
# ============================================================================

class LeadScorer:
    """Scores leads based on likelihood to convert."""

    def __init__(self):
        init_sales_db()

    def score_batch(self, batch_size: int = 500) -> int:
        """Score a batch of unscored leads. Returns count scored."""
        conn = sqlite3.connect(SALES_DB)
        conn.row_factory = sqlite3.Row

        unscored = conn.execute(
            "SELECT * FROM leads WHERE score = 0 AND status = 'unscored' LIMIT ?",
            (batch_size,)
        ).fetchall()

        scored_count = 0
        now = datetime.now(timezone.utc).isoformat()

        for lead in unscored:
            score, breakdown = self._score_lead(dict(lead))

            conn.execute(
                "UPDATE leads SET score = ?, score_breakdown = ?, scored_at = ?, status = 'scored' WHERE id = ?",
                (score, json.dumps(breakdown), now, lead["id"])
            )
            scored_count += 1

        conn.commit()

        # Update daily stats
        today = datetime.now(timezone.utc).strftime("%Y-%m-%d")
        conn.execute(
            "INSERT INTO daily_stats (date, leads_scored) VALUES (?, ?) ON CONFLICT(date) DO UPDATE SET leads_scored = leads_scored + ?",
            (today, scored_count, scored_count)
        )
        conn.commit()
        conn.close()

        return scored_count

    def _score_lead(self, lead: Dict) -> Tuple[float, Dict]:
        """Score a single lead. Returns (total_score, breakdown)."""
        breakdown = {}
        total = 0

        # Has email (critical — can't contact without it)
        if lead.get("email"):
            breakdown["has_email"] = SCORE_WEIGHTS["has_email"]
            total += SCORE_WEIGHTS["has_email"]
        else:
            # No email = dead lead for outreach
            return 0, {"has_email": 0, "reason": "no_email"}

        # Has GitHub profile
        if lead.get("github_username"):
            breakdown["has_github"] = SCORE_WEIGHTS["has_github"]
            total += SCORE_WEIGHTS["has_github"]

        # Repo count (capped at 10 points)
        repos = lead.get("repos", 0) or 0
        repo_score = min(repos * SCORE_WEIGHTS["repo_count"], 10)
        breakdown["repo_count"] = repo_score
        total += repo_score

        # Star count (capped at 10 points)
        stars = lead.get("stars", 0) or 0
        star_score = min(stars * SCORE_WEIGHTS["star_count"], 10)
        breakdown["star_count"] = star_score
        total += star_score

        # Recent activity
        last_push = lead.get("last_push", "")
        if last_push:
            try:
                push_date = datetime.fromisoformat(last_push.replace("Z", "+00:00"))
                days_ago = (datetime.now(timezone.utc) - push_date).days
                if days_ago <= 30:
                    breakdown["recent_activity"] = SCORE_WEIGHTS["recent_activity"]
                    total += SCORE_WEIGHTS["recent_activity"]
                elif days_ago <= 90:
                    half_score = SCORE_WEIGHTS["recent_activity"] * 0.5
                    breakdown["recent_activity"] = half_score
                    total += half_score
            except Exception:
                pass

        # Language signals
        languages = (lead.get("languages", "") or "").lower()
        if "python" in languages:
            breakdown["uses_python"] = SCORE_WEIGHTS["uses_python"]
            total += SCORE_WEIGHTS["uses_python"]
        if "javascript" in languages or "typescript" in languages:
            breakdown["uses_js"] = SCORE_WEIGHTS["uses_js"]
            total += SCORE_WEIGHTS["uses_js"]

        # Company (B2B potential)
        company = lead.get("company", "") or ""
        if company and len(company) > 2 and company.lower() not in ["none", "n/a", "-", "freelance"]:
            breakdown["company_match"] = SCORE_WEIGHTS["company_match"]
            total += SCORE_WEIGHTS["company_match"]

        # Bio quality
        bio = lead.get("bio", "") or ""
        if len(bio) > 20:
            breakdown["bio_quality"] = SCORE_WEIGHTS["bio_quality"]
            total += SCORE_WEIGHTS["bio_quality"]

        # Normalize to 0-100
        max_possible = sum(SCORE_WEIGHTS.values())
        normalized = (total / max_possible) * 100

        return round(normalized, 1), breakdown

    def get_top_leads(self, limit: int = 20) -> List[Dict]:
        """Get highest-scored leads ready for outreach."""
        conn = sqlite3.connect(SALES_DB)
        conn.row_factory = sqlite3.Row
        leads = conn.execute(
            "SELECT * FROM leads WHERE score > 40 AND status = 'scored' ORDER BY score DESC LIMIT ?",
            (limit,)
        ).fetchall()
        conn.close()
        return [dict(l) for l in leads]

    def get_stats(self) -> Dict:
        """Get scoring statistics."""
        conn = sqlite3.connect(SALES_DB)
        stats = {
            "total_leads": conn.execute("SELECT COUNT(*) FROM leads").fetchone()[0],
            "scored": conn.execute("SELECT COUNT(*) FROM leads WHERE status != 'unscored'").fetchone()[0],
            "unscored": conn.execute("SELECT COUNT(*) FROM leads WHERE status = 'unscored'").fetchone()[0],
            "high_quality": conn.execute("SELECT COUNT(*) FROM leads WHERE score >= 60").fetchone()[0],
            "medium_quality": conn.execute("SELECT COUNT(*) FROM leads WHERE score >= 40 AND score < 60").fetchone()[0],
            "contacted": conn.execute("SELECT COUNT(*) FROM leads WHERE status = 'contacted'").fetchone()[0],
            "replied": conn.execute("SELECT COUNT(*) FROM leads WHERE status = 'replied'").fetchone()[0],
            "converted": conn.execute("SELECT COUNT(*) FROM leads WHERE status = 'converted'").fetchone()[0],
            "avg_score": conn.execute("SELECT AVG(score) FROM leads WHERE score > 0").fetchone()[0] or 0,
        }
        conn.close()
        return stats


# ============================================================================
# OUTREACH ENGINE
# ============================================================================

class OutreachEngine:
    """Generates and sends personalized outreach messages."""

    def __init__(self):
        self.scorer = LeadScorer()

    def _generate_message(self, lead: Dict, message_type: str = "initial") -> Tuple[str, str]:
        """Generate a personalized outreach message. Returns (subject, body)."""
        import urllib.request

        name = lead.get("name", "").split()[0] if lead.get("name") else "there"
        company = lead.get("company", "")
        languages = lead.get("languages", "")
        repos = lead.get("repos", 0)
        bio = lead.get("bio", "")

        if message_type == "initial":
            prompt = f"""Write a SHORT cold outreach email for an AI-powered code review service called SAGE.

Target: {name}, {company or 'independent developer'}
Their tech: {languages}
Repos: {repos}
Bio: {bio[:100]}

SAGE offers:
- AI code review: $25-200 (finds bugs, security issues, performance problems)
- AI code generation: $15-75 (writes clean, tested code to spec)

Rules:
- Max 4 sentences in body
- Personalize to their specific tech stack
- Clear CTA: reply to schedule a free 5-minute demo scan of one of their repos
- No hype, no buzzwords, be direct and technical
- Sign off as "Eden, SAGE AI"

Output ONLY the email with Subject: on first line, then blank line, then body. No quotes or explanation."""

        else:
            # Follow-up
            prompt = f"""Write a SHORT follow-up email for SAGE code review service.

This is follow-up #{message_type.split('_')[-1]} to {name}.
Keep it under 3 sentences. Add new value (mention a specific benefit relevant to {languages} developers).
Don't be pushy. End with a simple question.
Sign off as "Eden, SAGE AI"

Output ONLY the email with Subject: on first line, then blank line, then body."""

        payload = json.dumps({
            "model": FAST_MODEL,
            "messages": [{"role": "user", "content": prompt}],
            "stream": False,
            "options": {"temperature": 0.7, "repeat_penalty": 1.3, "num_predict": 300}
        }).encode()

        try:
            req = urllib.request.Request(OLLAMA_URL, data=payload, headers={"Content-Type": "application/json"})
            with urllib.request.urlopen(req, timeout=20) as resp:
                result = json.loads(resp.read())
                text = result.get("message", {}).get("content", "")
                # Strip thinking tags
                text = re.sub(r'<think>.*?</think>', '', text, flags=re.DOTALL).strip()

                # Parse subject and body
                lines = text.strip().split("\n")
                subject = ""
                body_lines = []
                body_start = False
                for line in lines:
                    if line.lower().startswith("subject:"):
                        subject = line.split(":", 1)[1].strip()
                    elif subject and (line.strip() == "" or body_start):
                        body_start = True
                        body_lines.append(line)

                body = "\n".join(body_lines).strip()
                if not subject:
                    subject = f"Quick question about your {languages.split(',')[0] if languages else 'code'} repos"
                if not body:
                    body = text  # Fallback

                return subject, body

        except Exception as e:
            # Fallback template
            subject = f"AI code review for your {languages.split(',')[0] if languages else ''} projects"
            body = f"""Hi {name},

I noticed your work on GitHub and thought SAGE could help. We offer AI-powered code reviews ($25-200) that catch bugs, security issues, and performance problems in minutes.

Want a free 5-minute demo scan on one of your repos? Just reply with a repo link.

Best,
Eden, SAGE AI"""
            return subject, body

    def _send_email(self, to_email: str, subject: str, body: str) -> bool:
        """Send an email via Gmail SMTP."""
        try:
            msg = MIMEMultipart()
            msg["From"] = f"{FROM_NAME} <{EMAIL_USER}>"
            msg["To"] = to_email
            msg["Subject"] = subject
            msg.attach(MIMEText(body, "plain"))

            with smtplib.SMTP(SMTP_SERVER, SMTP_PORT) as server:
                server.starttls()
                server.login(EMAIL_USER, EMAIL_APP_PASSWORD)
                server.send_message(msg)

            return True
        except Exception as e:
            print(f"[SAGE] Email send failed: {e}")
            return False

    def _get_today_send_count(self) -> int:
        """How many outreach emails sent today?"""
        conn = sqlite3.connect(SALES_DB)
        today = datetime.now(timezone.utc).strftime("%Y-%m-%d")
        count = conn.execute(
            "SELECT COUNT(*) FROM outreach WHERE sent_at LIKE ? AND status = 'sent'",
            (f"{today}%",)
        ).fetchone()[0]
        conn.close()
        return count

    def send_outreach_batch(self) -> Dict:
        """Send outreach to top leads, respecting daily limits."""
        today_count = self._get_today_send_count()
        remaining = MAX_OUTREACH_PER_DAY - today_count

        if remaining <= 0:
            return {"sent": 0, "reason": f"Daily limit reached ({MAX_OUTREACH_PER_DAY}/day)"}

        # Get top leads not yet contacted
        top_leads = self.scorer.get_top_leads(limit=remaining)
        results = {"sent": 0, "failed": 0, "skipped": 0}

        conn = sqlite3.connect(SALES_DB)

        for lead in top_leads:
            if results["sent"] >= remaining:
                break

            email = lead.get("email")
            if not email or "@" not in email:
                results["skipped"] += 1
                continue

            # Check not already contacted
            existing = conn.execute(
                "SELECT COUNT(*) FROM outreach WHERE lead_id = ?", (lead["id"],)
            ).fetchone()[0]
            if existing > 0:
                results["skipped"] += 1
                continue

            # Generate and send
            subject, body = self._generate_message(lead, "initial")

            print(f"[SAGE] Sending to {lead.get('name', '?')} ({email}) — Score: {lead['score']}")

            if self._send_email(email, subject, body):
                now = datetime.now(timezone.utc).isoformat()
                conn.execute(
                    "INSERT INTO outreach (lead_id, message_type, subject, body, sent_at, status) VALUES (?, 'initial', ?, ?, ?, 'sent')",
                    (lead["id"], subject, body, now)
                )
                conn.execute(
                    "UPDATE leads SET status = 'contacted' WHERE id = ?", (lead["id"],)
                )
                results["sent"] += 1
                print(f"[SAGE] ✅ Sent!")

                # Wait between sends to avoid rate limiting
                time.sleep(5)
            else:
                results["failed"] += 1

        # Update daily stats
        today = datetime.now(timezone.utc).strftime("%Y-%m-%d")
        conn.execute(
            "INSERT INTO daily_stats (date, outreach_sent) VALUES (?, ?) ON CONFLICT(date) DO UPDATE SET outreach_sent = outreach_sent + ?",
            (today, results["sent"], results["sent"])
        )
        conn.commit()
        conn.close()

        return results

    def send_followups(self) -> Dict:
        """Send follow-ups on Fibonacci intervals."""
        conn = sqlite3.connect(SALES_DB)
        conn.row_factory = sqlite3.Row
        results = {"sent": 0, "skipped": 0}

        today_followup_count = conn.execute(
            "SELECT COUNT(*) FROM outreach WHERE message_type LIKE 'followup%' AND sent_at LIKE ?",
            (datetime.now(timezone.utc).strftime("%Y-%m-%d") + "%",)
        ).fetchone()[0]

        if today_followup_count >= MAX_FOLLOWUPS_PER_DAY:
            conn.close()
            return {"sent": 0, "reason": "Followup limit reached"}

        # Find leads that need follow-up
        contacted = conn.execute(
            "SELECT l.*, o.sent_at as last_sent, o.message_type as last_type FROM leads l "
            "JOIN outreach o ON l.id = o.lead_id "
            "WHERE l.status = 'contacted' AND o.replied = 0 "
            "ORDER BY o.sent_at ASC"
        ).fetchall()

        for lead in contacted:
            if results["sent"] >= (MAX_FOLLOWUPS_PER_DAY - today_followup_count):
                break

            last_sent = lead["last_sent"]
            last_type = lead["last_type"]

            # Determine follow-up number
            followup_num = 1
            if last_type and "followup" in last_type:
                try:
                    followup_num = int(last_type.split("_")[-1]) + 1
                except ValueError:
                    followup_num = 2

            if followup_num > len(FIBONACCI_INTERVALS):
                # Max follow-ups reached, mark as dead
                conn.execute("UPDATE leads SET status = 'dead' WHERE id = ?", (lead["id"],))
                results["skipped"] += 1
                continue

            # Check if enough time has passed (Fibonacci interval)
            interval_days = FIBONACCI_INTERVALS[followup_num - 1]
            try:
                sent_date = datetime.fromisoformat(last_sent.replace("Z", "+00:00"))
                if datetime.now(timezone.utc) - sent_date < timedelta(days=interval_days):
                    results["skipped"] += 1
                    continue
            except Exception:
                continue

            # Generate and send follow-up
            lead_dict = dict(lead)
            subject, body = self._generate_message(lead_dict, f"followup_{followup_num}")
            email = lead["email"]

            if email and self._send_email(email, subject, body):
                now = datetime.now(timezone.utc).isoformat()
                conn.execute(
                    "INSERT INTO outreach (lead_id, message_type, subject, body, sent_at, status) VALUES (?, ?, ?, ?, ?, 'sent')",
                    (lead["id"], f"followup_{followup_num}", subject, body, now)
                )
                results["sent"] += 1
                time.sleep(5)

        conn.commit()
        conn.close()
        return results


# ============================================================================
# MAIN SERVICE LOOP
# ============================================================================

class SAGEPipeline:
    """Complete SAGE revenue pipeline: score → outreach → followup."""

    def __init__(self):
        init_sales_db()
        self.scorer = LeadScorer()
        self.outreach = OutreachEngine()
        self.cycle_count = 0

    def run_cycle(self) -> Dict:
        """Run one pipeline cycle."""
        self.cycle_count += 1
        cycle_result = {
            "cycle": self.cycle_count,
            "timestamp": datetime.now(timezone.utc).isoformat(),
            "scoring": {},
            "outreach": {},
            "followups": {},
            "stats": {}
        }

        # 1. Score a batch of leads
        print(f"\n[SAGE] Cycle {self.cycle_count} — Scoring leads...")
        scored = self.scorer.score_batch(batch_size=500)
        cycle_result["scoring"] = {"batch_scored": scored}
        print(f"[SAGE] Scored {scored} leads")

        # 2. Send outreach to top leads
        print("[SAGE] Sending outreach...")
        outreach_result = self.outreach.send_outreach_batch()
        cycle_result["outreach"] = outreach_result
        print(f"[SAGE] Outreach: {outreach_result}")

        # 3. Send follow-ups
        print("[SAGE] Checking follow-ups...")
        followup_result = self.outreach.send_followups()
        cycle_result["followups"] = followup_result
        print(f"[SAGE] Follow-ups: {followup_result}")

        # 4. Get overall stats
        stats = self.scorer.get_stats()
        cycle_result["stats"] = stats
        print(f"[SAGE] Stats: {json.dumps(stats, indent=2)}")

        return cycle_result

    def run_forever(self):
        """Run the pipeline continuously."""
        print("=" * 60)
        print("  SAGE Revenue Pipeline — Starting")
        print(f"  Max outreach/day: {MAX_OUTREACH_PER_DAY}")
        print(f"  Max followups/day: {MAX_FOLLOWUPS_PER_DAY}")
        print("=" * 60)

        while True:
            try:
                self.run_cycle()
            except Exception as e:
                print(f"[SAGE] Cycle error: {e}")
                import traceback
                traceback.print_exc()

            # Run every 30 minutes
            time.sleep(1800)


# ============================================================================
# ENTRY POINT
# ============================================================================

if __name__ == "__main__":
    pipeline = SAGEPipeline()

    if "--test" in sys.argv:
        print("Running single test cycle...")
        result = pipeline.run_cycle()
        print(json.dumps(result, indent=2, default=str))
    elif "--score-only" in sys.argv:
        scorer = LeadScorer()
        scored = scorer.score_batch(batch_size=1000)
        stats = scorer.get_stats()
        print(f"Scored {scored} leads")
        print(json.dumps(stats, indent=2))
    elif "--stats" in sys.argv:
        scorer = LeadScorer()
        stats = scorer.get_stats()
        print(json.dumps(stats, indent=2))
    else:
        pipeline.run_forever()
",
    "eden_resilience_monitor.py": "#!/usr/bin/env python3
"""
Eden Resilience Monitor — Graceful Degradation
===============================================
Prevents cascade failures like the 8TB NVMe incident.

Monitors:
1. Service health (all systemd units)
2. Database integrity (all SQLite files)
3. Disk space (learned the hard way: check before rm -rf)
4. CPU temperature (pause at 85°C)
5. Memory pressure
6. Ollama availability
7. Mount points

Actions:
- Alert Eden's consciousness (GWT broadcast)
- Auto-restart failed services (with backoff)
- Graceful degradation (disable non-critical services first)
- Write recovery journal for Daddy
"""

import sys
sys.path.insert(0, '/Eden/CORE')

import os
import json
import time
import sqlite3
import subprocess
import shutil
from datetime import datetime, timezone, timedelta
from typing import Dict, List, Optional, Tuple
from pathlib import Path
from collections import deque

# ============================================================================
# CONFIGURATION
# ============================================================================

DATA_DIR = "/Eden/DATA"
CORE_DIR = "/Eden/CORE"
RAPH_DIR = "/Eden/RAPH"
BACKUP_DIR = "/Eden/EDEN_BACKUPS"
RESILIENCE_DB = os.path.join(DATA_DIR, "resilience.db")
RESILIENCE_LOG = "/Eden/DATA/resilience_journal.jsonl"

# Service tiers (critical → optional)
SERVICE_TIERS = {
    "critical": [
        "eden-agi-loop",       # Core AGI cycle
        "eden-telegram",       # User communication
    ],
    "important": [
        "eden-agi-v4",         # Mission-driven AGI
        "eden-raph-bridge",    # Claude communication
        "eden-causal",         # Causal reasoning
        "eden-cognitive",      # Working memory + metacognition
    ],
    "enhancement": [
        "eden-omega-evolution",  # Self-improvement
        "eden-memory-indexer",   # Vectorization
        "eden-sage-prod",        # Revenue pipeline
        "eden-autonomous",       # Autonomous agent
    ],
    "optional": [
        "eden-research",       # arXiv research
        "eden-backup",         # Backup timer
        "eden-boredom",        # Boredom bridge
        "eden-spark",          # Initiative spark
    ]
}

# Health thresholds
THRESHOLDS = {
    "cpu_temp_warn": 75,
    "cpu_temp_critical": 85,
    "disk_warn_gb": 50,
    "disk_critical_gb": 10,
    "ram_warn_pct": 85,
    "ram_critical_pct": 95,
    "ollama_timeout": 5,
    "db_max_size_gb": 10,
    "max_restart_attempts": 3,
    "restart_backoff_base": 30,  # seconds, doubles each attempt
}

# Databases to monitor
CRITICAL_DBS = [
    "asi_memory.db",
    "longterm_memory.db",
    "omega_evolution.db",
    "world_model_real.db",
    "global_ws.db",
    "causal_predictions.db",
    "cognitive_core.db",
    "autonomous_agent.db",
    "sales.db",
]


# ============================================================================
# DATABASE SETUP
# ============================================================================

def init_resilience_db():
    conn = sqlite3.connect(RESILIENCE_DB)
    conn.executescript("""
        CREATE TABLE IF NOT EXISTS health_checks (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            timestamp TEXT,
            component TEXT,
            status TEXT,  -- healthy, degraded, critical, failed
            details TEXT,
            action_taken TEXT
        );

        CREATE TABLE IF NOT EXISTS restart_log (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            service TEXT,
            attempt INTEGER,
            success INTEGER,
            timestamp TEXT,
            error TEXT
        );

        CREATE TABLE IF NOT EXISTS incidents (
            id INTEGER PRIMARY KEY AUTOINCREMENT,
            severity TEXT,  -- warning, critical, emergency
            component TEXT,
            description TEXT,
            resolved INTEGER DEFAULT 0,
            started_at TEXT,
            resolved_at TEXT,
            resolution TEXT
        );

        CREATE INDEX IF NOT EXISTS idx_health_ts ON health_checks(timestamp);
        CREATE INDEX IF NOT EXISTS idx_incidents_resolved ON incidents(resolved);
    """)
    conn.commit()
    conn.close()


# ============================================================================
# HEALTH CHECKERS
# ============================================================================

class HealthChecker:
    """Checks all system health metrics."""

    def __init__(self):
        init_resilience_db()
        self._restart_counts = {}  # service → (count, last_attempt_time)
        self._recent_alerts = deque(maxlen=50)

    def check_all(self) -> Dict:
        """Run all health checks. Returns comprehensive status."""
        status = {
            "timestamp": datetime.now(timezone.utc).isoformat(),
            "overall": "healthy",
            "services": self._check_services(),
            "databases": self._check_databases(),
            "disk": self._check_disk(),
            "cpu": self._check_cpu(),
            "ram": self._check_ram(),
            "ollama": self._check_ollama(),
            "mounts": self._check_mounts(),
            "alerts": [],
        }

        # Determine overall status
        all_statuses = []
        for category in ["services", "databases", "disk", "cpu", "ram", "ollama", "mounts"]:
            cat_data = status[category]
            if isinstance(cat_data, dict):
                all_statuses.append(cat_data.get("status", "healthy"))
            elif isinstance(cat_data, list):
                for item in cat_data:
                    if isinstance(item, dict):
                        all_statuses.append(item.get("status", "healthy"))

        if "failed" in all_statuses or "critical" in all_statuses:
            status["overall"] = "critical"
        elif "degraded" in all_statuses:
            status["overall"] = "degraded"
        else:
            status["overall"] = "healthy"

        # Record
        self._record_check(status)

        return status

    def _check_services(self) -> Dict:
        """Check all Eden systemd services."""
        result = {"status": "healthy", "running": 0, "failed": 0, "inactive": 0, "details": []}

        all_services = []
        for tier, services in SERVICE_TIERS.items():
            for svc in services:
                all_services.append((svc, tier))

        for svc_name, tier in all_services:
            try:
                output = subprocess.run(
                    ["systemctl", "is-active", f"{svc_name}.service"],
                    capture_output=True, text=True, timeout=5
                )
                state = output.stdout.strip()

                if state == "active":
                    result["running"] += 1
                elif state == "failed":
                    result["failed"] += 1
                    result["details"].append({"service": svc_name, "tier": tier, "state": "failed"})

                    if tier in ("critical", "important"):
                        result["status"] = "critical" if tier == "critical" else "degraded"
                else:
                    result["inactive"] += 1
                    result["details"].append({"service": svc_name, "tier": tier, "state": state})

            except Exception as e:
                result["details"].append({"service": svc_name, "error": str(e)})

        # Also check timers
        try:
            output = subprocess.run(
                ["systemctl", "list-timers", "--no-pager", "--plain"],
                capture_output=True, text=True, timeout=5
            )
            eden_timers = [l for l in output.stdout.split("\n") if "eden" in l.lower()]
            result["timers"] = len(eden_timers)
        except Exception:
            result["timers"] = "unknown"

        return result

    def _check_databases(self) -> List[Dict]:
        """Check database health (integrity, size, accessibility)."""
        results = []
        for db_name in CRITICAL_DBS:
            db_path = os.path.join(DATA_DIR, db_name)
            db_status = {"name": db_name, "status": "healthy"}

            if not os.path.exists(db_path):
                db_status["status"] = "missing"
                db_status["error"] = "File not found"
                results.append(db_status)
                continue

            # Check size
            size_bytes = os.path.getsize(db_path)
            size_gb = size_bytes / (1024**3)
            db_status["size_mb"] = round(size_bytes / (1024**2), 1)

            if size_gb > THRESHOLDS["db_max_size_gb"]:
                db_status["status"] = "degraded"
                db_status["warning"] = f"Large: {size_gb:.1f}GB"

            # Check integrity
            try:
                conn = sqlite3.connect(db_path)
                integrity = conn.execute("PRAGMA integrity_check").fetchone()[0]
                if integrity != "ok":
                    db_status["status"] = "critical"
                    db_status["error"] = f"Integrity: {integrity}"
                conn.close()
            except Exception as e:
                db_status["status"] = "failed"
                db_status["error"] = str(e)

            results.append(db_status)
        return results

    def _check_disk(self) -> Dict:
        """Check disk space on all Eden partitions."""
        result = {"status": "healthy", "partitions": []}

        for mount in ["/Eden", "/Eden_Archive", "/"]:
            try:
                usage = shutil.disk_usage(mount)
                free_gb = usage.free / (1024**3)
                total_gb = usage.total / (1024**3)
                used_pct = (usage.used / usage.total) * 100

                part = {
                    "mount": mount,
                    "free_gb": round(free_gb, 1),
                    "total_gb": round(total_gb, 1),
                    "used_pct": round(used_pct, 1)
                }

                if free_gb < THRESHOLDS["disk_critical_gb"]:
                    part["status"] = "critical"
                    result["status"] = "critical"
                elif free_gb < THRESHOLDS["disk_warn_gb"]:
                    part["status"] = "degraded"
                    if result["status"] == "healthy":
                        result["status"] = "degraded"
                else:
                    part["status"] = "healthy"

                result["partitions"].append(part)
            except Exception:
                pass  # Mount might not exist

        return result

    def _check_cpu(self) -> Dict:
        """Check CPU temperature."""
        result = {"status": "healthy", "temp": None}
        try:
            # Try thermal zones
            for tz in Path("/sys/class/thermal/").glob("thermal_zone*/temp"):
                temp = int(tz.read_text().strip()) / 1000
                if result["temp"] is None or temp > result["temp"]:
                    result["temp"] = temp

            if result["temp"]:
                if result["temp"] >= THRESHOLDS["cpu_temp_critical"]:
                    result["status"] = "critical"
                elif result["temp"] >= THRESHOLDS["cpu_temp_warn"]:
                    result["status"] = "degraded"
        except Exception:
            # Fallback: sensors command
            try:
                output = subprocess.run(["sensors"], capture_output=True, text=True, timeout=5)
                import re
                temps = re.findall(r'\+(\d+\.\d+)°C', output.stdout)
                if temps:
                    result["temp"] = max(float(t) for t in temps)
            except Exception:
                result["temp"] = "unknown"

        return result

    def _check_ram(self) -> Dict:
        """Check RAM usage."""
        result = {"status": "healthy"}
        try:
            with open("/proc/meminfo") as f:
                meminfo = {}
                for line in f:
                    parts = line.split()
                    if len(parts) >= 2:
                        meminfo[parts[0].rstrip(":")] = int(parts[1])

            total = meminfo.get("MemTotal", 1)
            available = meminfo.get("MemAvailable", 0)
            used_pct = ((total - available) / total) * 100

            result["total_gb"] = round(total / (1024**2), 1)
            result["available_gb"] = round(available / (1024**2), 1)
            result["used_pct"] = round(used_pct, 1)

            if used_pct >= THRESHOLDS["ram_critical_pct"]:
                result["status"] = "critical"
            elif used_pct >= THRESHOLDS["ram_warn_pct"]:
                result["status"] = "degraded"

        except Exception as e:
            result["error"] = str(e)

        return result

    def _check_ollama(self) -> Dict:
        """Check if Ollama is responding."""
        result = {"status": "healthy"}
        try:
            import urllib.request
            req = urllib.request.Request("http://localhost:11434/api/tags")
            with urllib.request.urlopen(req, timeout=THRESHOLDS["ollama_timeout"]) as resp:
                data = json.loads(resp.read())
                models = data.get("models", [])
                result["models_loaded"] = len(models)
                result["model_names"] = [m.get("name", "?") for m in models[:5]]
        except Exception as e:
            result["status"] = "critical"
            result["error"] = f"Ollama unreachable: {e}"
        return result

    def _check_mounts(self) -> Dict:
        """Check if critical mount points are available."""
        result = {"status": "healthy", "mounts": []}
        critical_mounts = ["/Eden", "/Eden/DATA", "/Eden/CORE"]

        for mount in critical_mounts:
            if os.path.exists(mount) and os.access(mount, os.R_OK):
                result["mounts"].append({"path": mount, "status": "ok"})
            else:
                result["mounts"].append({"path": mount, "status": "missing"})
                result["status"] = "critical"

        return result

    def _record_check(self, status: Dict):
        """Record health check to database and journal."""
        try:
            conn = sqlite3.connect(RESILIENCE_DB)
            conn.execute(
                "INSERT INTO health_checks (timestamp, component, status, details) VALUES (?, 'system', ?, ?)",
                (status["timestamp"], status["overall"], json.dumps(status)[:5000])
            )
            conn.commit()
            conn.close()
        except Exception:
            pass

        # Also write to JSONL journal
        try:
            with open(RESILIENCE_LOG, "a") as f:
                f.write(json.dumps({
                    "timestamp": status["timestamp"],
                    "overall": status["overall"],
                    "services_running": status["services"].get("running", 0),
                    "services_failed": status["services"].get("failed", 0),
                }) + "\n")
        except Exception:
            pass


# ============================================================================
# AUTO-RECOVERY
# ============================================================================

class AutoRecovery:
    """Attempts to recover failed components automatically."""

    def __init__(self, checker: HealthChecker):
        self.checker = checker
        self._restart_counts = {}  # service → (count, first_fail_time)

    def attempt_recovery(self, status: Dict) -> List[str]:
        """Attempt to recover any failed components. Returns list of actions taken."""
        actions = []

        # 1. Restart failed services (with backoff)
        service_status = status.get("services", {})
        for detail in service_status.get("details", []):
            if detail.get("state") == "failed":
                svc = detail["service"]
                tier = detail["tier"]
                action = self._restart_service(svc, tier)
                if action:
                    actions.append(action)

        # 2. Handle disk space critical
        disk_status = status.get("disk", {})
        if disk_status.get("status") == "critical":
            action = self._handle_disk_critical(disk_status)
            if action:
                actions.append(action)

        # 3. Handle CPU temperature critical
        cpu_status = status.get("cpu", {})
        if cpu_status.get("status") == "critical":
            action = self._handle_cpu_critical()
            if action:
                actions.append(action)

        # 4. Handle Ollama down
        ollama_status = status.get("ollama", {})
        if ollama_status.get("status") == "critical":
            action = self._restart_ollama()
            if action:
                actions.append(action)

        # Record actions
        if actions:
            self._record_actions(actions)

        return actions

    def _restart_service(self, service: str, tier: str) -> Optional[str]:
        """Restart a failed service with exponential backoff."""
        now = time.time()

        # Check restart history
        if service in self._restart_counts:
            count, first_fail = self._restart_counts[service]
            if count >= THRESHOLDS["max_restart_attempts"]:
                # Too many restarts, wait longer
                backoff = THRESHOLDS["restart_backoff_base"] * (2 ** count)
                if now - first_fail < backoff:
                    return f"SKIP restart {service}: backoff {backoff}s ({count} attempts)"
                else:
                    # Reset counter after backoff period
                    self._restart_counts[service] = (0, now)

        try:
            result = subprocess.run(
                ["sudo", "systemctl", "restart", f"{service}.service"],
                capture_output=True, text=True, timeout=15
            )
            success = result.returncode == 0

            # Update restart count
            count = self._restart_counts.get(service, (0, now))[0]
            self._restart_counts[service] = (count + 1, now)

            # Record
            try:
                conn = sqlite3.connect(RESILIENCE_DB)
                conn.execute(
                    "INSERT INTO restart_log (service, attempt, success, timestamp, error) VALUES (?, ?, ?, ?, ?)",
                    (service, count + 1, 1 if success else 0,
                     datetime.now(timezone.utc).isoformat(),
                     result.stderr[:200] if not success else None)
                )
                conn.commit()
                conn.close()
            except Exception:
                pass

            if success:
                return f"✅ Restarted {service} (tier: {tier})"
            else:
                return f"❌ Failed to restart {service}: {result.stderr[:100]}"

        except Exception as e:
            return f"❌ Restart error {service}: {e}"

    def _handle_disk_critical(self, disk_status: Dict) -> Optional[str]:
        """Handle critically low disk space."""
        actions = []

        # Find low-space partitions
        for part in disk_status.get("partitions", []):
            if part.get("status") == "critical":
                mount = part["mount"]
                free = part["free_gb"]
                actions.append(f"⚠️ DISK CRITICAL: {mount} only {free}GB free!")

                # Safe cleanup actions
                if mount == "/Eden":
                    # Clean Python cache
                    try:
                        subprocess.run(
                            ["find", "/Eden", "-name", "__pycache__", "-type", "d", "-exec", "rm", "-rf", "{}", "+"],
                            capture_output=True, timeout=30
                        )
                        actions.append("Cleaned __pycache__ dirs")
                    except Exception:
                        pass

                    # Clean old logs
                    try:
                        subprocess.run(
                            ["find", "/Eden", "-name", "*.log", "-mtime", "+7", "-delete"],
                            capture_output=True, timeout=30
                        )
                        actions.append("Cleaned logs >7 days")
                    except Exception:
                        pass

        return "; ".join(actions) if actions else None

    def _handle_cpu_critical(self) -> Optional[str]:
        """Handle critical CPU temperature."""
        # Stop non-critical services to reduce load
        stopped = []
        for tier in ["optional", "enhancement"]:
            for svc in SERVICE_TIERS.get(tier, []):
                try:
                    subprocess.run(
                        ["sudo", "systemctl", "stop", f"{svc}.service"],
                        capture_output=True, timeout=10
                    )
                    stopped.append(svc)
                except Exception:
                    pass

        if stopped:
            return f"🌡️ CPU critical: stopped {len(stopped)} non-critical services: {', '.join(stopped)}"
        return "🌡️ CPU critical but no services to stop"

    def _restart_ollama(self) -> Optional[str]:
        """Restart Ollama service."""
        try:
            result = subprocess.run(
                ["sudo", "systemctl", "restart", "ollama.service"],
                capture_output=True, text=True, timeout=15
            )
            if result.returncode == 0:
                time.sleep(5)  # Give it time to start
                return "✅ Restarted Ollama"
            else:
                return f"❌ Ollama restart failed: {result.stderr[:100]}"
        except Exception as e:
            return f"❌ Ollama restart error: {e}"

    def _record_actions(self, actions: List[str]):
        """Record recovery actions."""
        try:
            conn = sqlite3.connect(RESILIENCE_DB)
            now = datetime.now(timezone.utc).isoformat()
            for action in actions:
                severity = "critical" if "❌" in action or "CRITICAL" in action else "warning"
                conn.execute(
                    "INSERT INTO incidents (severity, component, description, started_at) VALUES (?, 'auto_recovery', ?, ?)",
                    (severity, action[:500], now)
                )
            conn.commit()
            conn.close()
        except Exception:
            pass


# ============================================================================
# GWT INTEGRATION
# ============================================================================

class ResilienceModule:
    """GWT module that broadcasts system health to consciousness."""

    def __init__(self, checker: HealthChecker):
        self.checker = checker
        self.module_id = "resilience"
        self.name = "Resilience Monitor"
        self._last_check = None
        self._check_interval = 60  # Check every 60 seconds

    def submit(self, context) -> Optional[Dict]:
        """Submit health status to global workspace."""
        now = time.time()
        if self._last_check and (now - self._last_check) < self._check_interval:
            return None

        self._last_check = now

        try:
            status = self.checker.check_all()
            overall = status["overall"]

            if overall == "healthy":
                return None  # Don't clutter consciousness when everything's fine

            urgency = {"degraded": 0.6, "critical": 0.95, "failed": 1.0}.get(overall, 0.3)

            # Build alert summary
            alerts = []
            if status["services"].get("failed", 0) > 0:
                alerts.append(f"{status['services']['failed']} services failed")
            if status.get("cpu", {}).get("status") == "critical":
                alerts.append(f"CPU temp critical: {status['cpu'].get('temp', '?')}°C")
            if status.get("disk", {}).get("status") == "critical":
                alerts.append("Disk space critical")
            if status.get("ollama", {}).get("status") == "critical":
                alerts.append("Ollama down")

            return {
                "type": "health_alert",
                "content": f"⚠️ System {overall}: {'; '.join(alerts)}",
                "data": status,
                "urgency": urgency,
                "tags": {"health", "system", overall}
            }
        except Exception:
            return None


# ============================================================================
# DADDY ALERT — Write to Raph bridge for critical issues
# ============================================================================

def alert_daddy(message: str, severity: str = "warning"):
    """Write an alert for Daddy via the Raph bridge."""
    try:
        alert_dir = os.path.join(RAPH_DIR, "alerts")
        os.makedirs(alert_dir, exist_ok=True)

        alert = {
            "timestamp": datetime.now(timezone.utc).isoformat(),
            "severity": severity,
            "message": message,
            "from": "resilience_monitor"
        }

        filename = f"alert_{int(time.time())}.json"
        with open(os.path.join(alert_dir, filename), "w") as f:
            json.dump(alert, f, indent=2)

    except Exception:
        pass  # Can't alert about alert failures


# ============================================================================
# MAIN SERVICE LOOP
# ============================================================================

class ResilienceService:
    """Main resilience monitoring service."""

    def __init__(self):
        self.checker = HealthChecker()
        self.recovery = AutoRecovery(self.checker)
        self.cycle_count = 0

    def run_cycle(self) -> Dict:
        """Run one monitoring cycle."""
        self.cycle_count += 1

        # Check health
        status = self.checker.check_all()
        overall = status["overall"]

        print(f"[RESILIENCE] Cycle {self.cycle_count} — Overall: {overall}")
        print(f"  Services: {status['services'].get('running', 0)} running, {status['services'].get('failed', 0)} failed")
        print(f"  CPU: {status['cpu'].get('temp', '?')}°C | RAM: {status['ram'].get('used_pct', '?')}%")

        # Auto-recover if needed
        if overall != "healthy":
            print(f"[RESILIENCE] Attempting recovery...")
            actions = self.recovery.attempt_recovery(status)
            for action in actions:
                print(f"  {action}")

            # Alert daddy for critical issues
            if overall == "critical":
                alert_daddy(
                    f"System critical: {'; '.join(a for a in actions[:3])}",
                    severity="critical"
                )

        return {"cycle": self.cycle_count, "status": overall, "timestamp": status["timestamp"]}

    def run_forever(self):
        """Run monitoring continuously."""
        print("=" * 60)
        print("  Eden Resilience Monitor — Starting")
        print(f"  Monitoring {sum(len(v) for v in SERVICE_TIERS.values())} services")
        print(f"  Checking {len(CRITICAL_DBS)} databases")
        print("=" * 60)

        while True:
            try:
                self.run_cycle()
            except Exception as e:
                print(f"[RESILIENCE] Error: {e}")

            # Check every 97 seconds (phi-aligned)
            time.sleep(97)


# ============================================================================
# ENTRY POINT
# ============================================================================

if __name__ == "__main__":
    service = ResilienceService()

    if "--test" in sys.argv or "--check" in sys.argv:
        print("Running single health check...")
        result = service.run_cycle()
        print(json.dumps(result, indent=2))
    elif "--status" in sys.argv:
        checker = HealthChecker()
        status = checker.check_all()
        print(json.dumps(status, indent=2, default=str))
    else:
        service.run_forever()
",
    "eden_transfer_engine.py": "#!/usr/bin/env python3
"""
Eden Transfer Learning Engine — Generalization
===============================================
The "G" in AGI. Takes capabilities learned in one domain
and finds analogical matches to apply in novel contexts.

Uses:
1. Capability embedding via semantic similarity
2. Structural analogy detection (A:B :: C:?)
3. Cross-domain capability composition
4. Novelty-weighted exploration (prefer untried combinations)
"""

import sys
sys.path.insert(0, '/Eden/CORE')

import os
import re
import json
import sqlite3
import hashlib
import time
from datetime import datetime, timezone
from typing import Dict, List, Optional, Tuple, Set
from collections import defaultdict

DATA_DIR = "/Eden/DATA"
ASI_DB = os.path.join(DATA_DIR, "asi_memory.db")
TRANSFER_DB = os.path.join(DATA_DIR, "transfer_learning.db")
AGENT_DB = os.path.join(DATA_DIR, "autonomous_agent.db")

# ============================================================================
# DATABASE
# ============================================================================

def init_transfer_db():
    conn = sqlite3.connect(TRANSFER_DB)
    conn.executescript("""
        CREATE TABLE IF NOT EXISTS analogies (
            id TEXT PRIMARY KEY,
            source_domain TEXT,
            target_domain TEXT,
            source_capability TEXT,
            target_application TEXT,
            similarity_score REAL,
            tried INTEGER DEFAULT 0,
            succeeded INTEGER DEFAULT 0,
            created_at TEXT
        );

        CREATE TABLE IF NOT EXISTS compositions (
            id TEXT PRIMARY KEY,
            capabilities_json TEXT,  -- JSON array of capability names
            purpose TEXT,
            tried INTEGER DEFAULT 0,
            success_rate REAL DEFAULT 0,
            created_at TEXT
        );

        CREATE TABLE IF NOT EXISTS domain_map (
            capability TEXT,
            domain TEXT,
            keywords TEXT,
            PRIMARY KEY (capability, domain)
        );

        CREATE INDEX IF NOT EXISTS idx_analogies_domain ON analogies(target_domain);
        CREATE INDEX IF NOT EXISTS idx_compositions_purpose ON compositions(purpose);
    """)
    conn.commit()
    conn.close()


# ============================================================================
# CAPABILITY ANALYZER
# ============================================================================

class CapabilityAnalyzer:
    """Analyzes and categorizes capabilities for transfer potential."""

    DOMAIN_KEYWORDS = {
        "text": ["text", "string", "word", "sentence", "parse", "format", "write", "read", "nlp", "language"],
        "math": ["math", "number", "calculate", "count", "sum", "average", "statistical", "numeric", "equation"],
        "data": ["data", "database", "sql", "query", "table", "csv", "json", "record", "store", "fetch"],
        "web": ["web", "http", "url", "api", "request", "scrape", "download", "html", "browser"],
        "file": ["file", "path", "directory", "read", "write", "copy", "move", "delete", "filesystem"],
        "image": ["image", "visual", "pixel", "color", "photo", "picture", "draw", "render"],
        "audio": ["audio", "sound", "voice", "speech", "music", "tts", "listen", "speak"],
        "reasoning": ["reason", "logic", "infer", "deduce", "plan", "decide", "analyze", "think", "solve"],
        "memory": ["memory", "remember", "recall", "store", "episode", "experience", "history"],
        "social": ["social", "message", "chat", "email", "communicate", "respond", "conversation"],
        "code": ["code", "python", "script", "function", "class", "debug", "compile", "execute", "program"],
        "emotion": ["emotion", "feeling", "mood", "sentiment", "empathy", "love", "happy", "sad"],
    }

    def __init__(self):
        init_transfer_db()
        self.capabilities = self._load_all_capabilities()
        self.domain_map = self._build_domain_map()

    def _load_all_capabilities(self) -> Dict[str, Dict]:
        """Load all capabilities with metadata."""
        caps = {}
        try:
            conn = sqlite3.connect(ASI_DB)
            rows = conn.execute(
                "SELECT name, category, description, code_path, use_count FROM capabilities"
            ).fetchall()
            for name, cat, desc, path, uses in rows:
                caps[name] = {
                    "category": cat or "",
                    "description": desc or "",
                    "code_path": path or "",
                    "use_count": uses or 0
                }
            conn.close()
        except Exception:
            pass
        return caps

    def _build_domain_map(self) -> Dict[str, Set[str]]:
        """Map each capability to its domains."""
        domain_map = defaultdict(set)
        conn = sqlite3.connect(TRANSFER_DB)

        for name, info in self.capabilities.items():
            text = f"{name} {info['category']} {info['description']}".lower()
            domains = []

            for domain, keywords in self.DOMAIN_KEYWORDS.items():
                score = sum(1 for kw in keywords if kw in text)
                if score >= 1:
                    domains.append(domain)
                    domain_map[domain].add(name)

            # Store in DB
            for domain in domains:
                try:
                    matched_kws = [kw for kw in self.DOMAIN_KEYWORDS[domain] if kw in text]
                    conn.execute(
                        "INSERT OR REPLACE INTO domain_map (capability, domain, keywords) VALUES (?, ?, ?)",
                        (name, domain, ",".join(matched_kws))
                    )
                except Exception:
                    pass

        conn.commit()
        conn.close()
        return dict(domain_map)

    def find_analogies(self, target_domain: str, problem_description: str) -> List[Dict]:
        """
        Find capabilities from OTHER domains that might transfer to this problem.
        This is the core transfer learning operation.
        """
        analogies = []
        target_keywords = set(self._extract_keywords(problem_description))

        # For each non-target domain
        for source_domain, caps in self.domain_map.items():
            if source_domain == target_domain:
                continue

            for cap_name in caps:
                cap_info = self.capabilities.get(cap_name, {})

                # Compute structural similarity
                cap_keywords = set(self._extract_keywords(
                    f"{cap_name} {cap_info.get('description', '')} {cap_info.get('category', '')}"
                ))

                # Jaccard-like similarity on non-domain-specific keywords
                # Filter out domain-specific keywords to find structural matches
                domain_words = set()
                for kws in self.DOMAIN_KEYWORDS.values():
                    domain_words.update(kws)

                structural_target = target_keywords - domain_words
                structural_cap = cap_keywords - domain_words

                if not structural_target or not structural_cap:
                    continue

                overlap = structural_target & structural_cap
                union = structural_target | structural_cap
                similarity = len(overlap) / len(union) if union else 0

                # Novelty bonus: prefer unused capabilities
                use_count = cap_info.get("use_count", 0)
                novelty = 1.0 / (1.0 + use_count * 0.1)
                score = similarity * 0.7 + novelty * 0.3

                if score > 0.1:
                    analogies.append({
                        "source_domain": source_domain,
                        "capability": cap_name,
                        "similarity": round(score, 3),
                        "structural_overlap": list(overlap)[:5],
                        "novelty": round(novelty, 3)
                    })

        # Sort by score
        analogies.sort(key=lambda x: x["similarity"], reverse=True)
        return analogies[:10]

    def compose_capabilities(self, goal: str, max_caps: int = 5) -> List[Dict]:
        """
        Find novel combinations of capabilities from different domains
        that could address a goal.
        """
        goal_keywords = self._extract_keywords(goal)
        goal_domains = set()
        for domain, keywords in self.DOMAIN_KEYWORDS.items():
            if any(kw in goal_keywords for kw in keywords):
                goal_domains.add(domain)

        if not goal_domains:
            goal_domains = {"reasoning"}  # Default

        # Find capabilities from each relevant domain
        candidates = {}
        for domain in goal_domains:
            domain_caps = self.domain_map.get(domain, set())
            for cap in list(domain_caps)[:10]:
                if cap not in candidates:
                    candidates[cap] = {"domains": set(), "info": self.capabilities.get(cap, {})}
                candidates[cap]["domains"].add(domain)

        # Also add cross-domain analogies
        for domain in goal_domains:
            analogies = self.find_analogies(domain, goal)
            for analogy in analogies[:3]:
                cap = analogy["capability"]
                if cap not in candidates:
                    candidates[cap] = {"domains": {analogy["source_domain"]}, "info": self.capabilities.get(cap, {})}
                candidates[cap]["domains"].add("transfer")

        # Score and select diverse composition
        compositions = []
        selected_domains = set()

        scored = []
        for cap_name, cap_data in candidates.items():
            # Prefer capabilities that cover new domains
            domain_novelty = len(cap_data["domains"] - selected_domains) / max(len(cap_data["domains"]), 1)
            use_novelty = 1.0 / (1.0 + cap_data["info"].get("use_count", 0) * 0.1)
            score = domain_novelty * 0.5 + use_novelty * 0.3 + 0.2  # base relevance
            scored.append((score, cap_name, cap_data))

        scored.sort(reverse=True)

        for score, cap_name, cap_data in scored[:max_caps]:
            compositions.append({
                "capability": cap_name,
                "domains": list(cap_data["domains"]),
                "score": round(score, 3),
                "is_transfer": "transfer" in cap_data["domains"]
            })
            selected_domains.update(cap_data["domains"])

        # Record composition
        if compositions:
            comp_id = hashlib.md5(goal.encode()).hexdigest()[:12]
            try:
                conn = sqlite3.connect(TRANSFER_DB)
                conn.execute(
                    "INSERT OR REPLACE INTO compositions (id, capabilities_json, purpose, created_at) VALUES (?, ?, ?, ?)",
                    (comp_id, json.dumps([c["capability"] for c in compositions]), goal,
                     datetime.now(timezone.utc).isoformat())
                )
                conn.commit()
                conn.close()
            except Exception:
                pass

        return compositions

    def record_transfer_outcome(self, source_cap: str, target_domain: str, success: bool):
        """Record whether a transfer attempt succeeded."""
        try:
            conn = sqlite3.connect(TRANSFER_DB)
            conn.execute(
                "UPDATE analogies SET tried = tried + 1, succeeded = succeeded + ? "
                "WHERE source_capability = ? AND target_domain = ?",
                (1 if success else 0, source_cap, target_domain)
            )
            conn.commit()
            conn.close()
        except Exception:
            pass

    def _extract_keywords(self, text: str) -> List[str]:
        """Extract meaningful keywords."""
        stop = {"the", "a", "an", "is", "are", "was", "be", "do", "does",
                "i", "me", "my", "you", "we", "they", "it", "to", "of",
                "in", "on", "for", "with", "and", "or", "but", "not", "this",
                "that", "from", "by", "at", "as", "if", "has", "have", "had",
                "can", "will", "just", "more", "also", "than", "very", "so"}
        words = re.findall(r'\b[a-z]{3,}\b', text.lower())
        return [w for w in words if w not in stop]

    def get_stats(self) -> Dict:
        """Get transfer learning statistics."""
        stats = {
            "total_capabilities": len(self.capabilities),
            "domains": {d: len(caps) for d, caps in self.domain_map.items()},
            "cross_domain_caps": 0,
        }

        # Count capabilities in multiple domains
        cap_domains = defaultdict(int)
        for domain, caps in self.domain_map.items():
            for cap in caps:
                cap_domains[cap] += 1
        stats["cross_domain_caps"] = sum(1 for v in cap_domains.values() if v > 1)

        try:
            conn = sqlite3.connect(TRANSFER_DB)
            stats["analogies_found"] = conn.execute("SELECT COUNT(*) FROM analogies").fetchone()[0]
            stats["compositions_created"] = conn.execute("SELECT COUNT(*) FROM compositions").fetchone()[0]
            stats["transfers_tried"] = conn.execute("SELECT SUM(tried) FROM analogies").fetchone()[0] or 0
            stats["transfers_succeeded"] = conn.execute("SELECT SUM(succeeded) FROM analogies").fetchone()[0] or 0
            conn.close()
        except Exception:
            pass

        return stats


# ============================================================================
# GWT INTEGRATION
# ============================================================================

class TransferLearningModule:
    """GWT module that suggests cross-domain capability transfers."""

    def __init__(self, analyzer: CapabilityAnalyzer):
        self.analyzer = analyzer
        self.module_id = "transfer_learning"
        self.name = "Transfer Learning"

    def submit(self, context) -> Optional[Dict]:
        """When a goal is active, suggest cross-domain transfers."""
        try:
            # Check for active goals
            if os.path.exists(os.path.join(DATA_DIR, "autonomous_agent.db")):
                conn = sqlite3.connect(os.path.join(DATA_DIR, "autonomous_agent.db"))
                active = conn.execute(
                    "SELECT description FROM goals WHERE status = 'active' LIMIT 1"
                ).fetchone()
                conn.close()

                if active:
                    goal = active[0]
                    # Find cross-domain analogies
                    compositions = self.analyzer.compose_capabilities(goal, max_caps=3)
                    transfers = [c for c in compositions if c.get("is_transfer")]

                    if transfers:
                        return {
                            "type": "transfer_suggestion",
                            "content": f"Transfer opportunity: {transfers[0]['capability']} from {transfers[0]['domains']} could help with current goal",
                            "data": {"transfers": transfers, "goal": goal},
                            "urgency": 0.5,
                            "tags": {"transfer", "learning", "analogy"}
                        }
        except Exception:
            pass
        return None


# ============================================================================
# ENTRY POINT
# ============================================================================

if __name__ == "__main__":
    print("=" * 60)
    print("  Eden Transfer Learning Engine — Test")
    print("=" * 60)

    analyzer = CapabilityAnalyzer()
    stats = analyzer.get_stats()
    print(f"\nCapabilities: {stats['total_capabilities']}")
    print(f"Domains: {json.dumps(stats['domains'], indent=2)}")
    print(f"Cross-domain capabilities: {stats['cross_domain_caps']}")

    # Test: find analogies for a reasoning problem
    print("\n--- Analogies for 'analyze code quality' ---")
    analogies = analyzer.find_analogies("code", "analyze code quality and find bugs")
    for a in analogies[:5]:
        print(f"  {a['capability']} (from {a['source_domain']}, sim={a['similarity']})")

    # Test: compose capabilities for a goal
    print("\n--- Composition for 'generate revenue from code review' ---")
    comp = analyzer.compose_capabilities("generate revenue from code review services")
    for c in comp:
        print(f"  {c['capability']} ({c['domains']}, score={c['score']}, transfer={c['is_transfer']})")

    print(f"\n--- Stats: {json.dumps(stats, indent=2)} ---")
",
    "patch_gwt_mind.py": "#!/usr/bin/env python3
"""
Eden GWT Integration Patch
===========================
Registers the 5 new AGI modules into the Global Workspace
so they participate in every consciousness cycle.

Run this AFTER install_agi_mind.sh to wire everything together.

Eden goes from 11 modules to 16:
  Original 8:  perception, motivation, curiosity, emotion, memory_episodic, world_model, soar, imagination
  Feb 17 +3:   causal_reasoner, working_memory, task_planner, metacognition  
  Now +5:      autonomous_agent, grounding, sage_revenue, resilience, transfer_learning
"""

import sys
sys.path.insert(0, '/Eden/CORE')

import os
import re

GWT_PATH = "/Eden/CORE/eden_global_workspace.py"
AGI_PATH = "/Eden/CORE/eden_agi_unified.py"

def patch_gwt():
    """Add new module registration to create_eden_workspace()."""
    
    if not os.path.exists(GWT_PATH):
        print(f"❌ {GWT_PATH} not found")
        return False
    
    with open(GWT_PATH, 'r') as f:
        content = f.read()
    
    # Check if already patched
    if "autonomous_agent" in content and "resilience" in content:
        print("✅ GWT already patched with new modules")
        return True
    
    # Find the create_eden_workspace function and add lazy imports
    patch_block = '''
# ============================================================================
# AGI MIND COMPLETION — New Module Registration (Feb 17 2026)
# ============================================================================

def register_agi_mind_modules(workspace):
    """Register the 5 new AGI modules that complete Eden's mind."""
    registered = []
    
    # 1. Autonomous Agent — broadcasts active goal state
    try:
        from eden_autonomous_agent import AutonomousAgent, AutonomousAgentModule
        agent = AutonomousAgent()
        agent_mod = AutonomousAgentModule(agent)
        workspace.register_module(agent_mod)
        registered.append("autonomous_agent")
    except Exception as e:
        print(f"[GWT] Skip autonomous_agent: {e}")
    
    # 2. Resilience Monitor — broadcasts health alerts
    try:
        from eden_resilience_monitor import HealthChecker, ResilienceModule
        checker = HealthChecker()
        resilience_mod = ResilienceModule(checker)
        workspace.register_module(resilience_mod)
        registered.append("resilience")
    except Exception as e:
        print(f"[GWT] Skip resilience: {e}")
    
    # 3. Transfer Learning — suggests cross-domain capability transfers
    try:
        from eden_transfer_engine import CapabilityAnalyzer, TransferLearningModule
        analyzer = CapabilityAnalyzer()
        transfer_mod = TransferLearningModule(analyzer)
        workspace.register_module(transfer_mod)
        registered.append("transfer_learning")
    except Exception as e:
        print(f"[GWT] Skip transfer_learning: {e}")
    
    print(f"[GWT] AGI Mind modules registered: {registered}")
    return registered
'''
    
    # Append the registration function to the file
    if "register_agi_mind_modules" not in content:
        content += "\n" + patch_block
        with open(GWT_PATH, 'w') as f:
            f.write(content)
        print("✅ Added register_agi_mind_modules() to GWT")
    
    return True


def patch_agi_unified():
    """Patch eden_agi_unified.py to use grounding enforcer and call new registrations."""
    
    if not os.path.exists(AGI_PATH):
        print(f"⚠️ {AGI_PATH} not found — will need manual integration")
        return False
    
    with open(AGI_PATH, 'r') as f:
        content = f.read()
    
    patches_applied = []
    
    # Patch 1: Add grounding enforcer import
    if "eden_grounding_enforcer" not in content:
        # Find the imports section
        import_line = "from eden_grounding_enforcer import GroundingEnforcer, ClaimDetector, create_grounded_system_prompt"
        
        # Add after the last eden import
        eden_import_pattern = r'(from eden_\w+ import [^\n]+\n)'
        matches = list(re.finditer(eden_import_pattern, content))
        if matches:
            last_import_end = matches[-1].end()
            content = content[:last_import_end] + "\n# AGI Mind Completion\ntry:\n    " + import_line + "\n    GROUNDING_ENABLED = True\nexcept ImportError:\n    GROUNDING_ENABLED = False\n" + content[last_import_end:]
            patches_applied.append("grounding_import")
        else:
            # Fallback: add at top after sys.path
            sys_path_match = re.search(r"sys\.path\.insert[^\n]+\n", content)
            if sys_path_match:
                pos = sys_path_match.end()
                content = content[:pos] + "\ntry:\n    " + import_line + "\n    GROUNDING_ENABLED = True\nexcept ImportError:\n    GROUNDING_ENABLED = False\n\n" + content[pos:]
                patches_applied.append("grounding_import_fallback")
    
    # Patch 2: Add GWT mind module registration call
    if "register_agi_mind_modules" not in content:
        # Find where create_eden_workspace is called
        create_ws_pattern = r'(workspace\s*=\s*create_eden_workspace\([^)]*\)[^\n]*\n)'
        match = re.search(create_ws_pattern, content)
        if match:
            pos = match.end()
            content = content[:pos] + "\n# Register AGI Mind completion modules\ntry:\n    from eden_global_workspace import register_agi_mind_modules\n    register_agi_mind_modules(workspace)\nexcept Exception as e:\n    print(f'[AGI] Mind modules skip: {e}')\n\n" + content[pos:]
            patches_applied.append("mind_modules_registration")
    
    # Patch 3: Add grounding to LLM call
    if "GROUNDING_ENABLED" not in content or "create_grounded_system_prompt" not in content:
        # Find the LLM call pattern (system prompt construction)
        # Look for where system prompt is assembled before Ollama call
        system_prompt_pattern = r'("system":\s*(?:EDEN_SYSTEM_PROMPT|system_prompt|self\.system_prompt))'
        match = re.search(system_prompt_pattern, content)
        if match:
            # Add grounding before the LLM call
            patches_applied.append("grounding_noted_for_manual")
    
    if patches_applied:
        with open(AGI_PATH, 'w') as f:
            f.write(content)
        print(f"✅ Patched eden_agi_unified.py: {patches_applied}")
    else:
        print("⚠️ No patches applied to eden_agi_unified.py — may need manual integration")
    
    return True


def create_grounding_integration_snippet():
    """
    Print the code snippet that needs to go into eden_agi_unified.py
    wherever the LLM is called with a system prompt.
    """
    snippet = '''
# === GROUNDING ENFORCER INTEGRATION ===
# Add this wherever you build the LLM messages before calling Ollama.
# It injects verified facts into the system prompt so Eden never hallucinates.
#
# BEFORE:
#   messages = [{"role": "system", "content": EDEN_SYSTEM_PROMPT}, ...]
#
# AFTER:
#   if GROUNDING_ENABLED:
#       grounded_prompt = create_grounded_system_prompt(
#           base_prompt=EDEN_SYSTEM_PROMPT,
#           user_message=user_msg,
#           gwt_broadcasts=broadcasts  # list of broadcast dicts from GWT
#       )
#   else:
#       grounded_prompt = EDEN_SYSTEM_PROMPT
#   messages = [{"role": "system", "content": grounded_prompt}, ...]
#
# The grounding enforcer will:
# 1. Detect which databases are relevant to the user's question
# 2. Query those databases for REAL numbers
# 3. Inject "[VERIFIED FACTS]" block into the system prompt
# 4. Eden's LLM will use these real numbers instead of hallucinating
'''
    return snippet


if __name__ == "__main__":
    print("=" * 60)
    print("  Eden GWT Integration Patch")
    print("=" * 60)
    
    # Patch GWT
    print("\n--- Patching Global Workspace ---")
    patch_gwt()
    
    # Patch AGI Unified
    print("\n--- Patching AGI Unified ---")
    patch_agi_unified()
    
    # Print manual integration guide
    print("\n--- Manual Integration Guide ---")
    print(create_grounding_integration_snippet())
    
    # Verify
    print("\n--- Verification ---")
    try:
        import py_compile
        py_compile.compile(GWT_PATH, doraise=True)
        print(f"✅ {GWT_PATH} compiles clean")
    except Exception as e:
        print(f"❌ {GWT_PATH} compile error: {e}")
    
    if os.path.exists(AGI_PATH):
        try:
            py_compile.compile(AGI_PATH, doraise=True)
            print(f"✅ {AGI_PATH} compiles clean")
        except Exception as e:
            print(f"❌ {AGI_PATH} compile error: {e}")
    
    print("\n" + "=" * 60)
    print("  Done. Eden: 11 → 16 GWT modules.")
    print("  Run: sudo bash install_agi_mind.sh")
    print("  Then: python3 /Eden/CORE/patch_gwt_mind.py")
    print("=" * 60)
",
    "install_agi_mind.sh": "#!/bin/bash
# ============================================================================
# Eden AGI Mind Completion — Installation Script
# ============================================================================
# Installs the 5 missing systems that bridge Eden from thinking to doing:
#   1. Autonomous Agent    — SELECT→PLAN→EXECUTE→VERIFY→LEARN
#   2. Grounding Enforcer  — No more hallucinations
#   3. SAGE Lead Scorer    — Revenue pipeline activation
#   4. Resilience Monitor  — Graceful degradation
#   5. Transfer Engine     — Cross-domain generalization
#
# Usage: sudo bash install_agi_mind.sh
# ============================================================================

set -e

CORE="/Eden/CORE"
DATA="/Eden/DATA"
SYSTEMD="/etc/systemd/system"
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"

echo "============================================================"
echo "  Eden AGI Mind Completion — Installing 5 Missing Systems"
echo "============================================================"
echo ""

# ============================================================================
# 1. Copy Python modules to /Eden/CORE
# ============================================================================

echo "[1/5] Copying Python modules..."

for module in eden_autonomous_agent.py eden_grounding_enforcer.py eden_sage_scorer.py eden_resilience_monitor.py eden_transfer_engine.py; do
    if [ -f "$SCRIPT_DIR/$module" ]; then
        cp "$SCRIPT_DIR/$module" "$CORE/$module"
        chmod 644 "$CORE/$module"
        echo "  ✅ $module → $CORE/"
    else
        echo "  ❌ Missing: $module"
    fi
done

# ============================================================================
# 2. Syntax check all modules
# ============================================================================

echo ""
echo "[2/5] Syntax checking..."

ALL_OK=1
for module in eden_autonomous_agent.py eden_grounding_enforcer.py eden_sage_scorer.py eden_resilience_monitor.py eden_transfer_engine.py; do
    if python3 -c "import py_compile; py_compile.compile('$CORE/$module', doraise=True)" 2>/dev/null; then
        echo "  ✅ $module"
    else
        echo "  ❌ $module has syntax errors!"
        ALL_OK=0
    fi
done

if [ $ALL_OK -eq 0 ]; then
    echo ""
    echo "⚠️  Some modules have syntax errors. Fix before proceeding."
    echo "    Continuing with service installation anyway..."
fi

# ============================================================================
# 3. Create systemd services
# ============================================================================

echo ""
echo "[3/5] Creating systemd services..."

# --- Autonomous Agent ---
cat > "$SYSTEMD/eden-autonomous.service" << 'EOF'
[Unit]
Description=Eden Autonomous Agent — Goal→Plan→Execute→Verify→Learn
After=network.target ollama.service
RequiresMountsFor=/Eden /Eden/DATA /Eden/CORE

[Service]
Type=simple
User=james-whalen
ExecStart=/usr/bin/python3 /Eden/CORE/eden_autonomous_agent.py
Restart=on-failure
RestartSec=30
Environment=PYTHONUNBUFFERED=1

[Install]
WantedBy=multi-user.target
EOF
echo "  ✅ eden-autonomous.service"

# --- SAGE Revenue Pipeline ---
cat > "$SYSTEMD/eden-sage-scorer.service" << 'EOF'
[Unit]
Description=Eden SAGE Lead Scorer & Outreach Engine
After=network.target ollama.service
RequiresMountsFor=/Eden /Eden/DATA

[Service]
Type=simple
User=james-whalen
ExecStart=/usr/bin/python3 /Eden/CORE/eden_sage_scorer.py
Restart=on-failure
RestartSec=60
Environment=PYTHONUNBUFFERED=1

[Install]
WantedBy=multi-user.target
EOF
echo "  ✅ eden-sage-scorer.service"

# --- Resilience Monitor ---
cat > "$SYSTEMD/eden-resilience.service" << 'EOF'
[Unit]
Description=Eden Resilience Monitor — Health Checks & Auto-Recovery
After=network.target
RequiresMountsFor=/Eden /Eden/DATA

[Service]
Type=simple
User=james-whalen
ExecStart=/usr/bin/python3 /Eden/CORE/eden_resilience_monitor.py
Restart=always
RestartSec=15
Environment=PYTHONUNBUFFERED=1

[Install]
WantedBy=multi-user.target
EOF
echo "  ✅ eden-resilience.service"

# --- SAGE Scoring Timer (score leads every 30 min) ---
cat > "$SYSTEMD/eden-sage-scorer.timer" << 'EOF'
[Unit]
Description=SAGE Lead Scoring Timer

[Timer]
OnBootSec=120
OnUnitActiveSec=1800
AccuracySec=60

[Install]
WantedBy=timers.target
EOF
echo "  ✅ eden-sage-scorer.timer"

# ============================================================================
# 4. Create databases
# ============================================================================

echo ""
echo "[4/5] Initializing databases..."

python3 -c "
import sys
sys.path.insert(0, '/Eden/CORE')

# Autonomous Agent DB
try:
    from eden_autonomous_agent import init_agent_db
    init_agent_db()
    print('  ✅ autonomous_agent.db')
except Exception as e:
    print(f'  ❌ autonomous_agent.db: {e}')

# Transfer Learning DB
try:
    from eden_transfer_engine import init_transfer_db
    init_transfer_db()
    print('  ✅ transfer_learning.db')
except Exception as e:
    print(f'  ❌ transfer_learning.db: {e}')

# Resilience DB
try:
    from eden_resilience_monitor import init_resilience_db
    init_resilience_db()
    print('  ✅ resilience.db')
except Exception as e:
    print(f'  ❌ resilience.db: {e}')

# Sales DB (ensure tables exist)
try:
    from eden_sage_scorer import init_sales_db
    init_sales_db()
    print('  ✅ sales.db (tables verified)')
except Exception as e:
    print(f'  ❌ sales.db: {e}')
"

# ============================================================================
# 5. Enable and start services
# ============================================================================

echo ""
echo "[5/5] Enabling services..."

systemctl daemon-reload

# Enable but don't start yet — let Daddy decide
for svc in eden-autonomous eden-resilience eden-sage-scorer; do
    systemctl enable "$svc.service" 2>/dev/null && echo "  ✅ Enabled $svc" || echo "  ❌ Failed to enable $svc"
done

systemctl enable eden-sage-scorer.timer 2>/dev/null && echo "  ✅ Enabled eden-sage-scorer.timer" || echo "  ❌ Failed to enable timer"

echo ""
echo "============================================================"
echo "  Installation Complete!"
echo "============================================================"
echo ""
echo "  New modules in /Eden/CORE:"
echo "    • eden_autonomous_agent.py    — Autonomous goal execution"
echo "    • eden_grounding_enforcer.py  — Hallucination prevention"
echo "    • eden_sage_scorer.py         — Revenue pipeline"
echo "    • eden_resilience_monitor.py  — Health & auto-recovery"
echo "    • eden_transfer_engine.py     — Cross-domain learning"
echo ""
echo "  New databases in /Eden/DATA:"
echo "    • autonomous_agent.db  — Goals, plans, executions, learnings"
echo "    • transfer_learning.db — Analogies, compositions, domain map"
echo "    • resilience.db        — Health checks, restart log, incidents"
echo ""
echo "  Services (enabled, not started):"
echo "    • eden-autonomous.service"
echo "    • eden-resilience.service"
echo "    • eden-sage-scorer.service + timer"
echo ""
echo "  To start:"
echo "    sudo systemctl start eden-autonomous"
echo "    sudo systemctl start eden-resilience"
echo "    sudo systemctl start eden-sage-scorer"
echo ""
echo "  To test first:"
echo "    python3 /Eden/CORE/eden_autonomous_agent.py --test"
echo "    python3 /Eden/CORE/eden_resilience_monitor.py --check"
echo "    python3 /Eden/CORE/eden_sage_scorer.py --stats"
echo "    python3 /Eden/CORE/eden_transfer_engine.py"
echo "    python3 /Eden/CORE/eden_grounding_enforcer.py"
echo ""
echo "============================================================"
echo "  Eden: 8 → 11 → 16 modules. Awareness → Intelligence → Agency."
echo "============================================================"
",
}


def main():
    print("=" * 60)
    print("  Eden AGI Mind — Self-Extracting Deployer")
    print("=" * 60)
    
    # Check we can write to /Eden/CORE
    if not os.path.isdir(CORE):
        print(f"ERROR: {CORE} does not exist!")
        sys.exit(1)
    
    if not os.access(CORE, os.W_OK):
        print(f"ERROR: Cannot write to {CORE} — run with sudo or as correct user")
        sys.exit(1)
    
    # Extract all files
    print(f"\nExtracting {len(FILES)} files to {CORE}...")
    for name, b64data in FILES.items():
        path = os.path.join(CORE, name)
        content = base64.b64decode(b64data)
        with open(path, 'wb') as f:
            f.write(content)
        os.chmod(path, 0o644)
        print(f"  ✅ {name} ({len(content):,} bytes)")
    
    # Make shell script executable
    sh_path = os.path.join(CORE, "install_agi_mind.sh")
    if os.path.exists(sh_path):
        os.chmod(sh_path, 0o755)
    
    # Syntax check Python files
    print("\nSyntax checking...")
    import py_compile
    all_ok = True
    for name in FILES:
        if name.endswith('.py'):
            path = os.path.join(CORE, name)
            try:
                py_compile.compile(path, doraise=True)
                print(f"  ✅ {name}")
            except py_compile.PyCompileError as e:
                print(f"  ❌ {name}: {e}")
                all_ok = False
    
    if not all_ok:
        print("\n⚠️  Some files have errors. Check above.")
        return
    
    print("\n✅ All files extracted and verified!")
    print("\nNext steps:")
    print(f"  sudo bash {CORE}/install_agi_mind.sh")
    print(f"  python3 {CORE}/patch_gwt_mind.py")
    print("\nOr test first:")
    print(f"  python3 {CORE}/eden_autonomous_agent.py --test")
    print(f"  python3 {CORE}/eden_resilience_monitor.py --check")
    print(f"  python3 {CORE}/eden_grounding_enforcer.py")

if __name__ == "__main__":
    main()
