deprec / app.py
exploring-solver's picture
updated logic to use qwen with intended tasks
5510ae2
"""
FastAPI server for DevOpsEnv - Linux DevOps/SRE troubleshooting environment.
Endpoints:
---------
POST /reset Create a new episode
POST /step Advance the episode
GET /state Current episode state
GET /tasks List tasks and action schema
POST /grader Grade a finished episode
GET /health Liveness check
GET / Info / spec link
"""
from __future__ import annotations
import os
import uuid
import json
from typing import Any, Dict, List, Optional
from datetime import datetime
from fastapi import FastAPI, HTTPException, Query
from fastapi.middleware.cors import CORSMiddleware
from pydantic import BaseModel
import environment as env
from data import TASK_META
from models import (
Action,
Observation,
State,
StepResult,
TaskInfo,
Reward,
GraderResponse,
)
app = FastAPI(
title="DevOpsEnv",
description="An OpenEnv-compliant Linux DevOps/SRE troubleshooting environment.",
version="1.0.0",
docs_url="/docs",
redoc_url="/redoc",
)
app.add_middleware(
CORSMiddleware,
allow_origins=["*"],
allow_methods=["*"],
allow_headers=["*"],
)
class ResetRequest(BaseModel):
task_id: str
class StepRequest(BaseModel):
episode_id: str
action: Action
class GraderRequest(BaseModel):
episode_id: str
# ---------------------------------------------------------------------------
# Endpoints
# ---------------------------------------------------------------------------
@app.get("/", tags=["meta"])
def root():
return {
"name": "DevOpsEnv",
"version": "1.0.0",
"description": "OpenEnv DevOps/SRE troubleshooting environment",
"openenv_spec": "https://github.com/meta-pytorch/OpenEnv",
"tasks": list(TASK_META.keys()),
"endpoints": {
"reset": "POST /reset",
"step": "POST /step",
"state": "GET /state?episode_id=...",
"tasks": "GET /tasks",
"grader": "POST /grader",
"health": "GET /health",
"docs": "GET /docs",
},
}
@app.get("/health", tags=["meta"])
def health():
return {"status": "healthy", "timestamp": datetime.now().isoformat()}
@app.get("/tasks", tags=["tasks"])
def tasks():
result = []
for task_id, meta in TASK_META.items():
result.append(
TaskInfo(
task_id=task_id,
name=meta["name"],
description=meta["description"],
difficulty=meta["difficulty"],
max_steps=meta["max_steps"],
)
)
return result
@app.post("/reset", tags=["control"])
def reset(req: ResetRequest):
try:
obs = env.reset(req.task_id)
return obs.model_dump()
except ValueError as e:
raise HTTPException(status_code=400, detail=str(e))
@app.post("/step", tags=["control"])
def step(req: StepRequest):
try:
result = env.step(req.episode_id, req.action)
return result.model_dump()
except (KeyError, ValueError) as e:
raise HTTPException(status_code=400, detail=str(e))
@app.get("/state", tags=["observation"])
def state(episode_id: str = Query(...)):
try:
st = env.get_state(episode_id)
return st.model_dump()
except KeyError as e:
raise HTTPException(status_code=404, detail=str(e))
@app.post("/grader", tags=["evaluation"])
def grader(req: GraderRequest):
try:
score, breakdown, feedback = env.grade(req.episode_id)
return GraderResponse(
episode_id=req.episode_id,
task_id=env._EPISODES[req.episode_id]["task_id"],
score=score,
breakdown=breakdown,
feedback=feedback,
).model_dump()
except KeyError as e:
raise HTTPException(status_code=404, detail=str(e))
if __name__ == "__main__":
import uvicorn
port = int(os.environ.get("PORT", 7860))
uvicorn.run(app, host="0.0.0.0", port=port, workers=1)
def run_server() -> None:
"""Console-script entry point used by packaging validators."""
import uvicorn
port = int(os.environ.get("PORT", 7860))
uvicorn.run(app, host="0.0.0.0", port=port, workers=1)