main.py 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. from fastapi import FastAPI, HTTPException
  2. from fastapi.middleware.cors import CORSMiddleware
  3. from pydantic import BaseModel
  4. import json
  5. from fastapi.responses import StreamingResponse
  6. from typing import AsyncGenerator, List, Dict, Any
  7. import uvicorn
  8. import traceback
  9. from xuanzhi_query import router as xz_router
  10. from sql_generator import SQLGenerator
  11. from config import DEFAULT_MODEL_TYPE
  12. import re
  13. app = FastAPI(title="Land Analysis API")
  14. app.include_router(xz_router)
  15. # 配置CORS
  16. app.add_middleware(
  17. CORSMiddleware,
  18. allow_origins=["*"],
  19. allow_credentials=True,
  20. allow_methods=["*"],
  21. allow_headers=["*"],
  22. )
  23. class QueryRequest(BaseModel):
  24. description: str
  25. class AnalysisResult(BaseModel):
  26. sql: str
  27. data: list
  28. visualization: dict = None
  29. similar_examples: List[Dict[str, Any]] = None
  30. @app.post("/land_analysis/stream")
  31. async def stream_land_analysis(request: QueryRequest):
  32. """
  33. 流式返回土地分析结果
  34. """
  35. def remove_think_tag(text: str) -> str:
  36. # 移除<think>标签及其内容
  37. return re.sub(r"<think>[\s\S]*?</think>\\n*", "", text)
  38. sql_generator = SQLGenerator(model_type=DEFAULT_MODEL_TYPE)
  39. async def generate_stream() -> AsyncGenerator[str, None]:
  40. try:
  41. similar_examples = None
  42. # 流式生成SQL
  43. async for chunk in sql_generator.generate_sql_stream(request.description):
  44. data = json.loads(chunk)
  45. if data["type"] == "similar_examples":
  46. similar_examples = data["content"]
  47. yield chunk
  48. elif data["type"] == "sql_generation":
  49. # zjstai模型时,移除<think>标签内容
  50. # if DEFAULT_MODEL_TYPE == "zjstai":
  51. # data["content"] = remove_think_tag(data["content"])
  52. # yield json.dumps(data, ensure_ascii=False) + "\n"
  53. # else:
  54. yield chunk
  55. elif data["type"] == "sql_result":
  56. sql = data["content"]
  57. result = await sql_generator.execute_sql(sql)
  58. if result["status"] == "error":
  59. yield json.dumps({
  60. "type": "error",
  61. "content": result["content"]
  62. }, ensure_ascii=False) + "\n"
  63. return
  64. yield json.dumps({
  65. "type": "result",
  66. "data": {
  67. "sql": sql,
  68. "exec_result": result["data"]
  69. }
  70. }, ensure_ascii=False) + "\n"
  71. else:
  72. yield chunk
  73. except Exception as e:
  74. traceback.print_exc()
  75. yield json.dumps({
  76. "type": "error",
  77. "content": str(e)
  78. }, ensure_ascii=False) + "\n"
  79. return StreamingResponse(
  80. generate_stream(),
  81. media_type="text/event-stream"
  82. )
  83. @app.on_event("shutdown")
  84. async def shutdown_event():
  85. """
  86. 应用关闭时清理资源
  87. """
  88. pass
  89. if __name__ == "__main__":
  90. uvicorn.run(app, host="0.0.0.0", port=8521)