main.py 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. from fastapi import FastAPI, HTTPException
  2. from fastapi.middleware.cors import CORSMiddleware
  3. from pydantic import BaseModel
  4. import pandas as pd
  5. import plotly.express as px
  6. import json
  7. from fastapi.responses import StreamingResponse
  8. from typing import AsyncGenerator, List, Dict, Any
  9. import uvicorn
  10. import traceback
  11. import asyncio
  12. from xuanzhi_query import router as xz_router
  13. from sql_generator import SQLGenerator
  14. app = FastAPI(title="Land Analysis API")
  15. app.include_router(xz_router)
  16. # 配置CORS
  17. app.add_middleware(
  18. CORSMiddleware,
  19. allow_origins=["*"],
  20. allow_credentials=True,
  21. allow_methods=["*"],
  22. allow_headers=["*"],
  23. )
  24. class QueryRequest(BaseModel):
  25. description: str
  26. class AnalysisResult(BaseModel):
  27. sql: str
  28. data: list
  29. visualization: dict = None
  30. similar_examples: List[Dict[str, Any]] = None
  31. sql_generator = SQLGenerator()
  32. @app.post("/land_analysis/stream")
  33. async def stream_land_analysis(request: QueryRequest):
  34. """
  35. 流式返回土地分析结果
  36. """
  37. async def generate_stream() -> AsyncGenerator[str, None]:
  38. try:
  39. similar_examples = None
  40. # 流式生成SQL
  41. async for chunk in sql_generator.generate_sql_stream(request.description):
  42. data = json.loads(chunk)
  43. if data["type"] == "similar_examples":
  44. similar_examples = data["content"]
  45. yield chunk
  46. elif data["type"] == "sql_generation":
  47. yield chunk
  48. elif data["type"] == "sql_result":
  49. # 获取到完整的SQL后执行
  50. sql = data["content"]
  51. result = await sql_generator.execute_sql(sql)
  52. if result["status"] == "error":
  53. yield json.dumps({
  54. "type": "error",
  55. "content": result["content"]
  56. }, ensure_ascii=False) + "\n"
  57. return
  58. # 返回最终结果
  59. yield json.dumps({
  60. "type": "result",
  61. "data": {
  62. "sql": sql,
  63. "exec_result": result["data"]
  64. }
  65. }, ensure_ascii=False) + "\n"
  66. else:
  67. yield chunk
  68. except Exception as e:
  69. traceback.print_exc()
  70. yield json.dumps({
  71. "type": "error",
  72. "content": str(e)
  73. }, ensure_ascii=False) + "\n"
  74. return StreamingResponse(
  75. generate_stream(),
  76. media_type="text/event-stream"
  77. )
  78. @app.post("/land_analysis", response_model=AnalysisResult)
  79. async def generate_and_execute_sql(request: QueryRequest):
  80. try:
  81. # 获取相似示例
  82. similar_examples = sql_generator._get_similar_examples(request.description)
  83. # 构建增强提示词
  84. enhanced_prompt = f"""
  85. 基于以下相似示例:
  86. {json.dumps(similar_examples, ensure_ascii=False, indent=2)}
  87. 请根据以下描述生成SQL查询:
  88. {request.description}
  89. """
  90. # 生成SQL
  91. sql = await sql_generator.chain.arun(enhanced_prompt)
  92. # 执行SQL
  93. result = await sql_generator.execute_sql(sql)
  94. if result["status"] == "error":
  95. raise HTTPException(status_code=400, detail=result["message"])
  96. return AnalysisResult(
  97. sql=sql,
  98. exec_result=result["data"]
  99. )
  100. except Exception as e:
  101. traceback.print_exc()
  102. raise HTTPException(status_code=500, detail=str(e))
  103. @app.on_event("shutdown")
  104. async def shutdown_event():
  105. """
  106. 应用关闭时清理资源
  107. """
  108. await sql_generator.close()
  109. if __name__ == "__main__":
  110. uvicorn.run(app, host="0.0.0.0", port=8521)