import json
import sys
import asyncio
from pathlib import Path
from urllib.parse import quote
from urllib.request import Request, urlopen

from flask import Flask, Response, jsonify, request, render_template_string
from rdflib import Graph, URIRef, BNode, RDFS, Literal
from mcp import ClientSession
from mcp.client.streamable_http import streamable_http_client

ROOT = Path(__file__).resolve().parent
if str(ROOT) not in sys.path:
    sys.path.insert(0, str(ROOT))

from signifier import bind_common_prefixes, HMAS

DEFAULT_SEM_BASE_URL = "http://localhost:5000"
DEFAULT_SEM_MCP_URL = "http://localhost:8200/mcp"
AGENT_ID = "human_agent"
AGENT_PORT = 9000

app = Flask(__name__)
agent_profile_context = "I am a human agent interacting with web services."
agent_minimum_relevance_value = 0.5
agent_maximum_signifiers = 10
agent_uri = URIRef(f"http://localhost:{AGENT_PORT}/profile")

HTML_TEMPLATE = """
<!DOCTYPE html>
<html>
<head>
    <title>Human Agent - SEM Interaction</title>
    <style>
        * { box-sizing: border-box; }
        body {
            font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
            max-width: 1200px;
            margin: 0 auto;
            padding: 20px;
            background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
            min-height: 100vh;
        }
        .header {
            background: white;
            padding: 20px;
            border-radius: 8px;
            margin-bottom: 20px;
            box-shadow: 0 2px 8px rgba(0,0,0,0.1);
        }
        .header h1 { margin: 0 0 10px 0; color: #333; }
        .header p { margin: 0; color: #666; font-size: 14px; }
        .container {
            display: grid;
            grid-template-columns: 1fr 1fr;
            gap: 20px;
        }
        .panel {
            background: white;
            border-radius: 8px;
            padding: 20px;
            box-shadow: 0 2px 8px rgba(0,0,0,0.1);
        }
        .panel h2 {
            margin: 0 0 15px 0;
            color: #333;
            font-size: 18px;
            border-bottom: 2px solid #667eea;
            padding-bottom: 10px;
        }
        label {
            display: block;
            margin: 15px 0 5px 0;
            color: #555;
            font-weight: 500;
            font-size: 13px;
        }
        input, textarea, select {
            width: 100%;
            padding: 10px;
            margin-bottom: 10px;
            border: 1px solid #ddd;
            border-radius: 4px;
            font-size: 13px;
            font-family: inherit;
        }
        textarea { resize: vertical; font-family: 'Monaco', monospace; }
        button {
            background: #667eea;
            color: white;
            padding: 10px 16px;
            border: none;
            border-radius: 4px;
            cursor: pointer;
            font-size: 13px;
            font-weight: 500;
            transition: background 0.2s;
            margin-right: 8px;
            margin-bottom: 10px;
        }
        button:hover { background: #5568d3; }
        button:active { background: #4957b8; }
        a.button-link {
            display: inline-block;
            background: #2f855a;
            color: white;
            padding: 10px 16px;
            border-radius: 4px;
            text-decoration: none;
            font-size: 13px;
            font-weight: 500;
            margin-right: 8px;
            margin-bottom: 10px;
        }
        a.button-link:hover { background: #276749; }
        .status {
            padding: 12px;
            margin: 10px 0;
            border-radius: 4px;
            font-size: 13px;
            display: none;
        }
        .status.show { display: block; }
        .status.success {
            background: #d4edda;
            color: #155724;
            border: 1px solid #c3e6cb;
        }
        .status.error {
            background: #f8d7da;
            color: #721c24;
            border: 1px solid #f5c6cb;
        }
        .output {
            background: #1e1e1e;
            color: #d4d4d4;
            border: 1px solid #444;
            border-radius: 4px;
            padding: 12px;
            margin: 10px 0;
            max-height: 350px;
            overflow-y: auto;
            font-family: 'Monaco', monospace;
            font-size: 12px;
            white-space: pre-wrap;
            word-wrap: break-word;
            line-height: 1.5;
        }
        .output.empty::before {
            content: 'Ready...';
            color: #666;
        }
        @media (max-width: 900px) {
            .container { grid-template-columns: 1fr; }
        }
    </style>
</head>
<body>
    <div class="header">
        <h1>🤖 Human Agent</h1>
        <p>Interact with SEM signifiers and manage your profile context</p>
    </div>

    <div class="container">
        <div class="panel">
            <h2>Profile Management</h2>

            <label>Profile Context (Natural Language):</label>
            <textarea id="profileContext" rows="4" placeholder="Describe your agent's capabilities and goals..."></textarea>

            <label>Minimum Relevance Value:</label>
            <input type="number" id="minimumRelevanceValue" min="0" max="1" step="0.01" value="0.5">

            <label>Maximum Signifiers:</label>
            <input type="number" id="maximumSignifiers" min="1" step="1" value="10">

            <button onclick="updateProfile()">💾 Update Profile</button>
            <button onclick="viewProfile()">👁️ View Profile (RDF)</button>
            <a id="profileLink" class="button-link" href="/profile" target="_blank" rel="noopener">Open /profile</a>

            <div id="profileStatus" class="status"></div>

            <h2 style="margin-top: 25px;">Signifier Discovery</h2>
            <button onclick="readSignifiers()" style="width: 100%;">🔍 Fetch Signifiers for My Profile</button>
            <button onclick="fetchSignifiersSem()" style="width: 100%; margin-top: 8px;">📄 Fetch Signifiers from SEM API (Turtle)</button>

            <label style="margin-top: 15px;">Profile URL:</label>
            <input type="text" id="profileUrl" readonly value="">

            <div id="discoveryStatus" class="status"></div>

            <h2 style="margin-top: 25px;">MCP Tools</h2>
            <button onclick="refreshMcpTools()">Refresh Tools</button>
            <label>Available Tool:</label>
            <select id="mcpToolSelect"></select>
            <label>Tool Arguments (JSON):</label>
            <textarea id="mcpToolArgs" rows="5" placeholder='{"example": "value"}'>{}</textarea>
            <button onclick="callMcpTool()" style="width: 100%;">Call Selected MCP Tool</button>
            <div id="mcpStatus" class="status"></div>
        </div>

        <div class="panel">
            <h2>Output</h2>
            <div id="output" class="output empty"></div>
            <button onclick="clearOutput()" style="width: 100%; margin-top: 10px;">Clear</button>
        </div>
    </div>

    <script>
        const BASE_URL = 'http://localhost:' + window.location.port;

        window.onload = function() {
            loadProfileContext();
            updateProfileUrl();
        };

        function updateProfileUrl() {
            const profileUrl = BASE_URL + '/profile';
            document.getElementById('profileUrl').value = profileUrl;
            document.getElementById('profileLink').href = profileUrl;
        }

        function log(msg) {
            const output = document.getElementById('output');
            output.classList.remove('empty');
            const timestamp = new Date().toLocaleTimeString();
            output.textContent += `[${timestamp}] ${msg}\n`;
            output.scrollTop = output.scrollHeight;
        }

        function clearOutput() {
            document.getElementById('output').textContent = '';
            document.getElementById('output').classList.add('empty');
        }

        function showStatus(elementId, msg, isSuccess) {
            const el = document.getElementById(elementId);
            el.className = 'status show ' + (isSuccess ? 'success' : 'error');
            el.textContent = msg;
            setTimeout(() => el.classList.remove('show'), 5000);
        }

        function loadProfileContext() {
            fetch('/api/profile-context')
                .then(r => r.json())
                .then(data => {
                    document.getElementById('profileContext').value = data.context;
                    document.getElementById('minimumRelevanceValue').value = data.minimum_relevance_value;
                    document.getElementById('maximumSignifiers').value = data.maximum_signifiers;
                })
                .catch(e => log('Error loading profile: ' + e.message));
        }

        function updateProfile() {
            const context = document.getElementById('profileContext').value;
            const minimumRelevanceValue = parseFloat(document.getElementById('minimumRelevanceValue').value);
            const maximumSignifiers = parseInt(document.getElementById('maximumSignifiers').value, 10);
            if (!context.trim()) {
                showStatus('profileStatus', 'Profile context cannot be empty', false);
                return;
            }
            if (Number.isNaN(minimumRelevanceValue) || minimumRelevanceValue < 0 || minimumRelevanceValue > 1) {
                showStatus('profileStatus', 'Minimum relevance value must be between 0 and 1', false);
                return;
            }
            if (Number.isNaN(maximumSignifiers) || maximumSignifiers < 1) {
                showStatus('profileStatus', 'Maximum signifiers must be at least 1', false);
                return;
            }

            log('Updating profile context...');
            fetch('/api/profile-context', {
                method: 'POST',
                headers: {'Content-Type': 'application/json'},
                body: JSON.stringify({
                    context: context,
                    minimum_relevance_value: minimumRelevanceValue,
                    maximum_signifiers: maximumSignifiers
                })
            })
            .then(r => r.json())
            .then(data => {
                if (data.success) {
                    showStatus('profileStatus', '✓ Profile updated and registered with SEM', true);
                    log('Profile successfully updated');
                } else {
                    showStatus('profileStatus', '✗ Failed: ' + (data.error || 'Unknown error'), false);
                    log('Error: ' + (data.error || 'Unknown error'));
                }
            })
            .catch(e => {
                showStatus('profileStatus', '✗ Network error: ' + e.message, false);
                log('Network error: ' + e.message);
            });
        }

        function viewProfile() {
            log('Fetching profile in RDF format...');
            fetch('/profile?format=json-ld')
                .then(r => r.text())
                .then(data => {
                    try {
                        const json = JSON.parse(data);
                        log(JSON.stringify(json, null, 2));
                    } catch {
                        log(data);
                    }
                })
                .catch(e => log('Error fetching profile: ' + e.message));
        }

        function readSignifiers() {
            const profileUrl = document.getElementById('profileUrl').value;
            if (!profileUrl) {
                showStatus('discoveryStatus', 'Profile URL is required', false);
                return;
            }

            log('Fetching signifiers through SEM MCP for profile: ' + profileUrl);
            fetch('/api/fetch-signifiers', {
                method: 'POST',
                headers: {'Content-Type': 'application/json'},
                body: JSON.stringify({profile_url: profileUrl})
            })
            .then(r => r.json())
            .then(data => {
                if (data.success) {
                    showStatus('discoveryStatus', '✓ ' + (data.message || 'Signifiers fetched'), true);
                    log('SEM MCP read_signifiers result: ' + JSON.stringify(data.result, null, 2));
                    log('Available MCP tools: ' + (data.tools || []).map(t => t.name).join(', '));
                    populateMcpTools(data.tools || []);
                    if (data.details) {
                        log('Details: ' + JSON.stringify(data.details, null, 2));
                    }
                } else {
                    showStatus('discoveryStatus', '✗ ' + (data.error || 'Failed to fetch signifiers'), false);
                    log('Error: ' + (data.error || 'Unknown error'));
                }
            })
            .catch(e => {
                showStatus('discoveryStatus', '✗ Network error', false);
                log('Error: ' + e.message);
            });
        }

        function fetchSignifiersSem() {
            const profileUrl = BASE_URL + '/profile';
            log('Fetching signifiers from SEM API for profile: ' + profileUrl);
            fetch('/api/fetch-signifiers-sem')
                .then(r => r.json())
                .then(data => {
                    if (data.success) {
                        showStatus('discoveryStatus', '✓ Signifiers retrieved in Turtle format', true);
                        log('Signifiers (Turtle/RDF format):');
                        log(data.signifiers);
                    } else {
                        showStatus('discoveryStatus', '✗ ' + (data.error || 'Failed to fetch signifiers'), false);
                        log('Error: ' + (data.error || 'Unknown error'));
                    }
                })
                .catch(e => {
                    showStatus('discoveryStatus', '✗ Network error', false);
                    log('Error: ' + e.message);
                });
        }

        function populateMcpTools(tools) {
            const select = document.getElementById('mcpToolSelect');
            select.innerHTML = '';
            tools.forEach(tool => {
                const option = document.createElement('option');
                option.value = tool.name;
                option.textContent = tool.name;
                option.title = tool.description || '';
                select.appendChild(option);
            });
        }

        function refreshMcpTools() {
            log('Refreshing MCP tools...');
            fetch('/api/mcp/tools')
                .then(r => r.json())
                .then(data => {
                    if (data.success) {
                        populateMcpTools(data.tools || []);
                        showStatus('mcpStatus', '✓ MCP tools refreshed', true);
                        log('MCP tools: ' + (data.tools || []).map(t => t.name).join(', '));
                    } else {
                        showStatus('mcpStatus', '✗ ' + (data.error || 'Failed to refresh tools'), false);
                        log('MCP error: ' + (data.error || 'Unknown error'));
                    }
                })
                .catch(e => {
                    showStatus('mcpStatus', '✗ Network error', false);
                    log('Error: ' + e.message);
                });
        }

        function callMcpTool() {
            const toolName = document.getElementById('mcpToolSelect').value;
            if (!toolName) {
                showStatus('mcpStatus', 'Select a tool first', false);
                return;
            }

            let args;
            try {
                args = JSON.parse(document.getElementById('mcpToolArgs').value || '{}');
            } catch (e) {
                showStatus('mcpStatus', 'Tool arguments must be valid JSON', false);
                return;
            }

            log('Calling MCP tool: ' + toolName);
            fetch('/api/mcp/call', {
                method: 'POST',
                headers: {'Content-Type': 'application/json'},
                body: JSON.stringify({tool_name: toolName, arguments: args})
            })
            .then(r => r.json())
            .then(data => {
                if (data.success) {
                    showStatus('mcpStatus', '✓ Tool call completed', true);
                    log('Tool result: ' + JSON.stringify(data.result, null, 2));
                } else {
                    showStatus('mcpStatus', '✗ ' + (data.error || 'Tool call failed'), false);
                    log('MCP error: ' + (data.error || 'Unknown error'));
                }
            })
            .catch(e => {
                showStatus('mcpStatus', '✗ Network error', false);
                log('Error: ' + e.message);
            });
        }
    </script>
</body>
</html>
"""


async def _mcp_call_tool(tool_name: str, arguments: dict):
    async with streamable_http_client(DEFAULT_SEM_MCP_URL) as (read_stream, write_stream, _sid):
        async with ClientSession(read_stream, write_stream) as session:
            await session.initialize()
            result = await session.call_tool(tool_name, arguments=arguments)
            content = []
            for item in result.content or []:
                text = getattr(item, "text", None)
                if getattr(item, "type", None) == "text" and isinstance(text, str):
                    try:
                        content.append(json.loads(text))
                    except json.JSONDecodeError:
                        content.append(text)
                else:
                    content.append(str(item))
            return {"content": content, "is_error": bool(getattr(result, "isError", False))}


async def _mcp_list_tools():
    async with streamable_http_client(DEFAULT_SEM_MCP_URL) as (read_stream, write_stream, _sid):
        async with ClientSession(read_stream, write_stream) as session:
            await session.initialize()
            response = await session.list_tools()
            tools = []
            for tool in response.tools:
                input_schema = getattr(tool, "input_schema", None) or getattr(tool, "inputSchema", None)
                tools.append(
                    {
                        "name": tool.name,
                        "description": tool.description,
                        "input_schema": input_schema,
                    }
                )
            return tools


def _mcp_result_has_error(result: dict) -> bool:
    if result.get("is_error"):
        return True
    for item in result.get("content", []):
        if isinstance(item, dict) and item.get("error"):
            return True
    return False


def register_with_sem():
    """Register this agent's profile with the SEM."""
    try:
        encoded_id = quote(AGENT_ID, safe="/")
        profile_graph = get_profile_graph()
        serialized = profile_graph.serialize(format="turtle")

        target = f"{DEFAULT_SEM_BASE_URL}/profile/{encoded_id}"
        req = Request(
            target,
            data=serialized.encode("utf-8"),
            headers={"Content-Type": "text/turtle"},
            method="PUT",
        )
        with urlopen(req, timeout=10) as resp:
            return resp.status == 204
    except Exception as exc:
        print(f"Failed to register with SEM: {exc}")
        return False


def get_profile_graph():
    """Generate the current profile graph."""
    profile_graph = Graph()
    context_node = BNode()
    profile_graph.add((agent_uri, HMAS["hasContext"], context_node))
    profile_graph.add((context_node, RDFS["comment"], Literal(agent_profile_context)))
    profile_graph.add(
        (agent_uri, HMAS["minimumRelevanceValue"], Literal(agent_minimum_relevance_value))
    )
    profile_graph.add((agent_uri, HMAS["maximumSignifiers"], Literal(agent_maximum_signifiers)))
    bind_common_prefixes(profile_graph)
    return profile_graph


@app.route("/")
def index():
    return render_template_string(HTML_TEMPLATE)


@app.route("/profile")
def get_profile():
    """Expose the agent's profile in RDF format."""
    profile_graph = get_profile_graph()
    fmt = request.args.get("format", "turtle")

    if fmt == "json-ld":
        return Response(profile_graph.serialize(format="json-ld"), mimetype="application/ld+json")
    else:
        return Response(profile_graph.serialize(format="turtle"), mimetype="text/turtle")


@app.route("/api/profile-context", methods=["GET"])
def get_profile_context():
    """Get the current profile context."""
    return jsonify(
        {
            "context": agent_profile_context,
            "minimum_relevance_value": agent_minimum_relevance_value,
            "maximum_signifiers": agent_maximum_signifiers,
        }
    )


@app.route("/api/profile-context", methods=["POST"])
def set_profile_context():
    """Update the profile context and register with SEM."""
    global agent_profile_context, agent_minimum_relevance_value, agent_maximum_signifiers
    try:
        data = request.json
        context = data.get("context", "").strip()
        if not context:
            return jsonify({"success": False, "error": "Context cannot be empty"})

        minimum_relevance_value = float(data.get("minimum_relevance_value", 0.5))
        maximum_signifiers = int(data.get("maximum_signifiers", 10))
        if minimum_relevance_value < 0 or minimum_relevance_value > 1:
            return jsonify(
                {"success": False, "error": "Minimum relevance value must be between 0 and 1"}
            )
        if maximum_signifiers < 1:
            return jsonify({"success": False, "error": "Maximum signifiers must be at least 1"})

        agent_profile_context = context
        agent_minimum_relevance_value = minimum_relevance_value
        agent_maximum_signifiers = maximum_signifiers
        success = register_with_sem()
        return jsonify(
            {
                "success": success,
                "context": agent_profile_context,
                "minimum_relevance_value": agent_minimum_relevance_value,
                "maximum_signifiers": agent_maximum_signifiers,
            }
        )
    except Exception as e:
        return jsonify({"success": False, "error": str(e)})


@app.route("/api/fetch-signifiers", methods=["POST"])
def fetch_signifiers_route():
    """Fetch signifiers through the SEM MCP server based on the provided profile URL."""
    try:
        data = request.json
        profile_url = data.get("profile_url", f"http://localhost:{AGENT_PORT}/profile")

        if not profile_url.strip():
            return jsonify({"success": False, "error": "Profile URL is required"})

        result = asyncio.run(_mcp_call_tool("read_signifiers", {"profile_url": profile_url}))
        tools = asyncio.run(_mcp_list_tools())
        has_error = _mcp_result_has_error(result)

        return jsonify(
            {
                "success": not has_error,
                "message": "Retrieved signifiers through SEM MCP",
                "result": result,
                "tools": tools,
                "mcp_url": DEFAULT_SEM_MCP_URL,
            }
        )
    except Exception as e:
        return jsonify({"success": False, "error": str(e)})


@app.route("/api/fetch-signifiers-sem", methods=["GET"])
def fetch_signifiers_sem_route():
    """Fetch signifiers directly from the SEM API using this agent's profile."""
    try:
        profile_url = f"http://localhost:{AGENT_PORT}/profile"
        sem_signifiers_url = f"{DEFAULT_SEM_BASE_URL}/signifiers?profile={quote(profile_url, safe='')}"

        req = Request(
            sem_signifiers_url,
            headers={"Accept": "text/turtle"}
        )
        with urlopen(req, timeout=10) as resp:
            signifiers_rdf = resp.read().decode("utf-8")
            return jsonify({
                "success": True,
                "signifiers": signifiers_rdf,
                "content_type": resp.headers.get("Content-Type", "text/turtle")
            })
    except Exception as e:
        return jsonify({"success": False, "error": str(e)})


@app.route("/api/mcp/tools", methods=["GET"])
def list_mcp_tools_route():
    """List tools currently exposed by the SEM MCP server."""
    try:
        return jsonify({"success": True, "tools": asyncio.run(_mcp_list_tools())})
    except Exception as e:
        return jsonify({"success": False, "error": str(e)})


@app.route("/api/mcp/call", methods=["POST"])
def call_mcp_tool_route():
    """Call a tool exposed by the SEM MCP server."""
    try:
        data = request.json
        tool_name = data.get("tool_name", "").strip()
        arguments = data.get("arguments", {})
        if not tool_name:
            return jsonify({"success": False, "error": "Tool name is required"})
        if not isinstance(arguments, dict):
            return jsonify({"success": False, "error": "Tool arguments must be a JSON object"})

        result = asyncio.run(_mcp_call_tool(tool_name, arguments))
        return jsonify({"success": not _mcp_result_has_error(result), "result": result})
    except Exception as e:
        return jsonify({"success": False, "error": str(e)})


if __name__ == "__main__":
    print(f"\n{'='*60}")
    print("  Human Agent")
    print(f"{'='*60}")
    print(f"  GUI:     http://localhost:{AGENT_PORT}")
    print(f"  Profile: http://localhost:{AGENT_PORT}/profile")
    print(f"  SEM:     {DEFAULT_SEM_BASE_URL}")
    print(f"  SEM MCP: {DEFAULT_SEM_MCP_URL}")
    print(f"{'='*60}\n")

    print("Registering with SEM...")
    if register_with_sem():
        print("✓ Successfully registered with SEM\n")
    else:
        print("⚠ Failed to register with SEM (is it running?)\n")

    app.run(host="0.0.0.0", port=AGENT_PORT, debug=False, threaded=True)
