File size: 6,514 Bytes
4cbbddf
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
python3 << 'PYEOF'
code = '''
import asyncio, json, os, sys, time
from datetime import datetime
from typing import AsyncGenerator
from fastapi import FastAPI, HTTPException, Request
from fastapi.middleware.cors import CORSMiddleware
from fastapi.responses import HTMLResponse, StreamingResponse
from pydantic import BaseModel, Field

sys.path.insert(0, "/opt/aegis")
from agents.swarm import (signal_agent, intelligence_agent, forecast_agent,
    simulation_agent, decision_agent, alert_agent, execution_agent,
    run_aegis_pipeline, scrape_marine_traffic)
from models.forecaster import get_forecaster

app = FastAPI(title="Aegis", version="2.0.0")
app.add_middleware(CORSMiddleware, allow_origins=["*"], allow_methods=["*"], allow_headers=["*"])

@app.middleware("http")
async def timing(request: Request, call_next):
    start = time.time()
    response = await call_next(request)
    response.headers["X-Response-Time"] = f"{round((time.time()-start)*1000,1)}ms"
    response.headers["X-Powered-By"] = "AMD Instinct MI300X"
    return response

class CrisisEvent(BaseModel):
    oil_price_change_pct: float = Field(default=18.0)
    shipping_disruption:  str   = Field(default="Strait of Hormuz disruption")
    news_headline:        str   = Field(default="Regional conflict near Persian Gulf")
    severity:             str   = Field(default="HIGH")
    disruption_factor:    float = Field(default=0.7, ge=0, le=1)
    horizon_days:         int   = Field(default=14, ge=1, le=90)

class ForecastRequest(BaseModel):
    oil_shock_pct:     float = Field(default=0.0)
    disruption_factor: float = Field(default=0.0, ge=0, le=1)
    horizon_days:      int   = Field(default=14, ge=1, le=90)

@app.on_event("startup")
async def startup():
    print("Aegis starting on AMD Developer Cloud...")
    try:
        get_forecaster()
        print("Forecaster ready")
    except Exception as e:
        print(f"Forecaster warning: {e}")

@app.get("/health")
async def health():
    return {
        "status": "online", "system": "Aegis", "version": "2.0.0",
        "timestamp": datetime.utcnow().isoformat(),
        "agents": 7, "platform": "AMD Developer Cloud",
        "model": "llama-3.3-70b-versatile",
        "forecast": "ARIMA(2,1,2) + XGBoost hybrid",
    }

@app.get("/api/marine")
async def marine_feed():
    """Live MarineTraffic shipping news feed."""
    try:
        data = scrape_marine_traffic()
        return {"success": True, "data": data}
    except Exception as e:
        raise HTTPException(status_code=500, detail=str(e))

@app.post("/api/forecast")
async def forecast_only(req: ForecastRequest):
    try:
        fc = get_forecaster()
        result = fc.forecast(req.horizon_days, req.oil_shock_pct, req.disruption_factor)
        return {"success": True, "data": result}
    except Exception as e:
        raise HTTPException(status_code=500, detail=str(e))

@app.post("/api/crisis")
async def run_crisis(event: CrisisEvent):
    try:
        fc = get_forecaster()
        forecast_data = fc.forecast(event.horizon_days, event.oil_price_change_pct, event.disruption_factor)
        results = await run_aegis_pipeline(event.model_dump(), forecast_data)
        return {"success": True, "timestamp": datetime.utcnow().isoformat(), "data": results}
    except Exception as e:
        import traceback
        raise HTTPException(status_code=500, detail=f"{e}\\n{traceback.format_exc()}")

@app.get("/api/stream")
async def stream_crisis(oil_change: float=18.0, disruption: float=0.7, severity: str="HIGH"):
    async def gen() -> AsyncGenerator[str, None]:
        def sse(d): return f"data: {json.dumps(d)}\\n\\n"
        yield sse({"type":"start","message":"Aegis pipeline initiated"})
        try:
            fc = get_forecaster()
            fd = fc.forecast(14, oil_change, disruption)
            yield sse({"type":"forecast_ready","data":fd["summary"]})
            ev = {"oil_price_change_pct":oil_change,"shipping_disruption":"Strait of Hormuz",
                  "news_headline":"Geopolitical escalation","severity":severity,"disruption_factor":disruption}
            yield sse({"type":"agent_start","agent":"signal","index":1})
            sig = await signal_agent(ev)
            yield sse({"type":"agent_done","agent":"signal","data":sig})
            yield sse({"type":"agent_start","agent":"intelligence","index":2})
            intel = await intelligence_agent(sig, ev)
            yield sse({"type":"agent_done","agent":"intelligence","data":intel})
            yield sse({"type":"agent_start","agent":"forecast","index":3})
            fore = await forecast_agent(intel, fd)
            yield sse({"type":"agent_done","agent":"forecast","data":fore})
            yield sse({"type":"agent_start","agent":"simulation","index":4})
            sim = await simulation_agent(fore, ev)
            yield sse({"type":"agent_done","agent":"simulation","data":sim})
            yield sse({"type":"agent_start","agent":"decision","index":5})
            dec = await decision_agent(sim, fore, intel)
            yield sse({"type":"agent_done","agent":"decision","data":dec})
            yield sse({"type":"agent_start","agent":"alert","index":6})
            alert = await alert_agent(dec, fore)
            yield sse({"type":"agent_done","agent":"alert","data":alert})
            yield sse({"type":"agent_start","agent":"execution","index":7})
            exe = await execution_agent(dec)
            yield sse({"type":"agent_done","agent":"execution","data":exe})
            yield sse({"type":"complete","message":"All 7 agents finished",
                       "summary":dec.get("executive_summary",""),"threat":dec.get("threat_level","")})
        except Exception as e:
            yield sse({"type":"error","message":str(e)})
    return StreamingResponse(gen(), media_type="text/event-stream",
                             headers={"Cache-Control":"no-cache","X-Accel-Buffering":"no"})

@app.get("/api/status")
async def status():
    import random
    return {"agents_online":7,"platform":"AMD Developer Cloud",
            "oil_price":round(82+random.uniform(-2,2),2),
            "risk_level":"LOW","groq_model":"llama-3.3-70b-versatile",
            "forecast_model":"ARIMA(2,1,2)+XGBoost"}

@app.get("/", response_class=HTMLResponse)
async def index():
    with open("/opt/aegis/frontend.html") as f:
        return f.read()
'''
with open("/opt/aegis/api/server.py","w") as f:
    f.write(code)
print("server.py written OK")
PYEOF

✅ Should print: server.py written OK