mirror of
https://github.com/csunny/DB-GPT.git
synced 2025-07-27 13:57:46 +00:00
Co-authored-by: 夏姜 <wenfengjiang.jwf@digital-engine.com> Co-authored-by: aries_ckt <916701291@qq.com> Co-authored-by: wb-lh513319 <wb-lh513319@alibaba-inc.com> Co-authored-by: csunny <cfqsunny@163.com>
380 lines
9.8 KiB
Python
380 lines
9.8 KiB
Python
from functools import cache
|
|
from typing import List, Optional, Union
|
|
|
|
from fastapi import (
|
|
APIRouter,
|
|
Depends,
|
|
File,
|
|
Form,
|
|
Header,
|
|
HTTPException,
|
|
Query,
|
|
UploadFile,
|
|
)
|
|
from fastapi.security.http import HTTPAuthorizationCredentials, HTTPBearer
|
|
|
|
from dbgpt.component import SystemApp
|
|
from dbgpt.rag.chunk_manager import ChunkParameters
|
|
from dbgpt.serve.core import Result
|
|
from dbgpt.serve.rag.api.schemas import (
|
|
DocumentServeRequest,
|
|
DocumentServeResponse,
|
|
KnowledgeRetrieveRequest,
|
|
KnowledgeSyncRequest,
|
|
SpaceServeRequest,
|
|
SpaceServeResponse,
|
|
)
|
|
from dbgpt.serve.rag.config import SERVE_SERVICE_COMPONENT_NAME
|
|
from dbgpt.serve.rag.service.service import Service
|
|
from dbgpt.util import PaginationResult
|
|
|
|
router = APIRouter()
|
|
|
|
# Add your API endpoints here
|
|
|
|
global_system_app: Optional[SystemApp] = None
|
|
|
|
|
|
def get_service() -> Service:
|
|
"""Get the service instance"""
|
|
return global_system_app.get_component(SERVE_SERVICE_COMPONENT_NAME, Service)
|
|
|
|
|
|
get_bearer_token = HTTPBearer(auto_error=False)
|
|
|
|
|
|
@cache
|
|
def _parse_api_keys(api_keys: str) -> List[str]:
|
|
"""Parse the string api keys to a list
|
|
|
|
Args:
|
|
api_keys (str): The string api keys
|
|
|
|
Returns:
|
|
List[str]: The list of api keys
|
|
"""
|
|
if not api_keys:
|
|
return []
|
|
return [key.strip() for key in api_keys.split(",")]
|
|
|
|
|
|
async def check_api_key(
|
|
auth: Optional[HTTPAuthorizationCredentials] = Depends(get_bearer_token),
|
|
service: Service = Depends(get_service),
|
|
) -> Optional[str]:
|
|
"""Check the api key
|
|
|
|
If the api key is not set, allow all.
|
|
|
|
Your can pass the token in you request header like this:
|
|
|
|
.. code-block:: python
|
|
|
|
import requests
|
|
|
|
client_api_key = "your_api_key"
|
|
headers = {"Authorization": "Bearer " + client_api_key}
|
|
res = requests.get("http://test/hello", headers=headers)
|
|
assert res.status_code == 200
|
|
|
|
"""
|
|
if service.config.api_keys:
|
|
api_keys = _parse_api_keys(service.config.api_keys)
|
|
if auth is None or (token := auth.credentials) not in api_keys:
|
|
raise HTTPException(
|
|
status_code=401,
|
|
detail={
|
|
"error": {
|
|
"message": "",
|
|
"type": "invalid_request_error",
|
|
"param": None,
|
|
"code": "invalid_api_key",
|
|
}
|
|
},
|
|
)
|
|
return token
|
|
else:
|
|
# api_keys not set; allow all
|
|
return None
|
|
|
|
|
|
@router.get("/health", dependencies=[Depends(check_api_key)])
|
|
async def health():
|
|
"""Health check endpoint"""
|
|
return {"status": "ok"}
|
|
|
|
|
|
@router.get("/test_auth", dependencies=[Depends(check_api_key)])
|
|
async def test_auth():
|
|
"""Test auth endpoint"""
|
|
return {"status": "ok"}
|
|
|
|
|
|
@router.post("/spaces")
|
|
async def create(
|
|
request: SpaceServeRequest,
|
|
service: Service = Depends(get_service),
|
|
) -> Result:
|
|
"""Create a new Space entity
|
|
|
|
Args:
|
|
request (SpaceServeRequest): The request
|
|
service (Service): The service
|
|
Returns:
|
|
ServerResponse: The response
|
|
"""
|
|
return Result.succ(service.create_space(request))
|
|
|
|
|
|
@router.put("/spaces", dependencies=[Depends(check_api_key)])
|
|
async def update(
|
|
request: SpaceServeRequest, service: Service = Depends(get_service)
|
|
) -> Result:
|
|
"""Update a Space entity
|
|
|
|
Args:
|
|
request (SpaceServeRequest): The request
|
|
service (Service): The service
|
|
Returns:
|
|
ServerResponse: The response
|
|
"""
|
|
return Result.succ(service.update_space(request))
|
|
|
|
|
|
@router.delete(
|
|
"/spaces/{space_id}",
|
|
response_model=Result[None],
|
|
dependencies=[Depends(check_api_key)],
|
|
)
|
|
async def delete(
|
|
space_id: str, service: Service = Depends(get_service)
|
|
) -> Result[None]:
|
|
"""Delete a Space entity
|
|
|
|
Args:
|
|
request (SpaceServeRequest): The request
|
|
service (Service): The service
|
|
Returns:
|
|
ServerResponse: The response
|
|
"""
|
|
return Result.succ(service.delete(space_id))
|
|
|
|
|
|
@router.get(
|
|
"/spaces/{space_id}",
|
|
response_model=Result[List],
|
|
)
|
|
async def query(
|
|
space_id: str,
|
|
service: Service = Depends(get_service),
|
|
) -> Result[List[SpaceServeResponse]]:
|
|
"""Query Space entities
|
|
|
|
Args:
|
|
request (SpaceServeRequest): The request
|
|
service (Service): The service
|
|
Returns:
|
|
List[ServeResponse]: The response
|
|
"""
|
|
request = {"id": space_id}
|
|
return Result.succ(service.get(request))
|
|
|
|
|
|
@router.get(
|
|
"/spaces",
|
|
response_model=Result[PaginationResult[SpaceServeResponse]],
|
|
)
|
|
async def query_page(
|
|
page: int = Query(default=1, description="current page"),
|
|
page_size: int = Query(default=20, description="page size"),
|
|
service: Service = Depends(get_service),
|
|
) -> Result[PaginationResult[SpaceServeResponse]]:
|
|
"""Query Space entities
|
|
|
|
Args:
|
|
page (int): The page number
|
|
page_size (int): The page size
|
|
service (Service): The service
|
|
Returns:
|
|
ServerResponse: The response
|
|
"""
|
|
return Result.succ(service.get_list_by_page({}, page, page_size))
|
|
|
|
|
|
@router.post("/spaces/{space_id}/retrieve")
|
|
async def space_retrieve(
|
|
space_id: int,
|
|
request: KnowledgeRetrieveRequest,
|
|
service: Service = Depends(get_service),
|
|
) -> Result:
|
|
"""Create a new Document entity
|
|
|
|
Args:
|
|
space_id (int): The space id
|
|
request (SpaceServeRequest): The request
|
|
service (Service): The service
|
|
Returns:
|
|
ServerResponse: The response
|
|
"""
|
|
request.space_id = space_id
|
|
space_request = service.build_space_request(token, space_id)
|
|
spaces = service.get_list(space_request)
|
|
if not spaces:
|
|
raise HTTPException(status_code=404, detail="Space not found")
|
|
return Result.succ(await service.retrieve(request))
|
|
|
|
|
|
@router.post("/documents")
|
|
async def create_document(
|
|
doc_name: str = Form(...),
|
|
doc_type: str = Form(...),
|
|
space_id: str = Form(...),
|
|
content: Optional[str] = Form(None),
|
|
doc_file: Union[UploadFile, str] = Form(None),
|
|
service: Service = Depends(get_service),
|
|
) -> Result:
|
|
"""Create a new Document entity
|
|
|
|
Args:
|
|
request (SpaceServeRequest): The request
|
|
service (Service): The service
|
|
Returns:
|
|
ServerResponse: The response
|
|
"""
|
|
request = DocumentServeRequest(
|
|
doc_name=doc_name,
|
|
doc_type=doc_type,
|
|
content=content,
|
|
doc_file=doc_file,
|
|
space_id=space_id,
|
|
)
|
|
return Result.succ(await service.create_document(request))
|
|
|
|
|
|
@router.get(
|
|
"/documents/{document_id}",
|
|
response_model=Result[List],
|
|
)
|
|
async def query(
|
|
document_id: int,
|
|
service: Service = Depends(get_service),
|
|
) -> Result[List[SpaceServeResponse]]:
|
|
"""Query Space entities
|
|
|
|
Args:
|
|
request (SpaceServeRequest): The request
|
|
service (Service): The service
|
|
Returns:
|
|
List[ServeResponse]: The response
|
|
"""
|
|
request = {"id": document_id}
|
|
return Result.succ(service.get_document(request))
|
|
|
|
|
|
@router.get(
|
|
"/documents",
|
|
response_model=Result[PaginationResult[SpaceServeResponse]],
|
|
)
|
|
async def query_page(
|
|
page: int = Query(default=1, description="current page"),
|
|
page_size: int = Query(default=20, description="page size"),
|
|
service: Service = Depends(get_service),
|
|
) -> Result[PaginationResult[DocumentServeResponse]]:
|
|
"""Query Space entities
|
|
|
|
Args:
|
|
page (int): The page number
|
|
page_size (int): The page size
|
|
service (Service): The service
|
|
Returns:
|
|
ServerResponse: The response
|
|
"""
|
|
return Result.succ(service.get_document_list({}, page, page_size))
|
|
|
|
|
|
@router.post("/documents/chunks/add")
|
|
async def add_documents_chunks(
|
|
doc_name: str = Form(...),
|
|
space_id: int = Form(...),
|
|
content: List[str] = Form(None),
|
|
service: Service = Depends(get_service),
|
|
) -> Result:
|
|
""" """
|
|
|
|
|
|
@router.post("/documents/sync", dependencies=[Depends(check_api_key)])
|
|
async def sync_documents(
|
|
requests: List[KnowledgeSyncRequest], service: Service = Depends(get_service)
|
|
) -> Result:
|
|
"""Create a new Document entity
|
|
|
|
Args:
|
|
request (SpaceServeRequest): The request
|
|
service (Service): The service
|
|
Returns:
|
|
ServerResponse: The response
|
|
"""
|
|
return Result.succ(service.sync_document(requests))
|
|
|
|
|
|
@router.post("/documents/batch_sync")
|
|
async def sync_documents(
|
|
requests: List[KnowledgeSyncRequest],
|
|
service: Service = Depends(get_service),
|
|
) -> Result:
|
|
"""Create a new Document entity
|
|
|
|
Args:
|
|
request (SpaceServeRequest): The request
|
|
service (Service): The service
|
|
Returns:
|
|
ServerResponse: The response
|
|
"""
|
|
return Result.succ(service.sync_document(requests))
|
|
|
|
|
|
@router.post("/documents/{document_id}/sync")
|
|
async def sync_document(
|
|
document_id: int,
|
|
request: KnowledgeSyncRequest,
|
|
service: Service = Depends(get_service),
|
|
) -> Result:
|
|
"""Create a new Document entity
|
|
|
|
Args:
|
|
request (SpaceServeRequest): The request
|
|
service (Service): The service
|
|
Returns:
|
|
ServerResponse: The response
|
|
"""
|
|
request.doc_id = document_id
|
|
if request.chunk_parameters is None:
|
|
request.chunk_parameters = ChunkParameters(chunk_strategy="Automatic")
|
|
return Result.succ(service.sync_document([request]))
|
|
|
|
|
|
@router.delete(
|
|
"/documents/{document_id}",
|
|
dependencies=[Depends(check_api_key)],
|
|
response_model=Result[None],
|
|
)
|
|
async def delete_document(
|
|
document_id: str, service: Service = Depends(get_service)
|
|
) -> Result[None]:
|
|
"""Delete a Space entity
|
|
|
|
Args:
|
|
request (SpaceServeRequest): The request
|
|
service (Service): The service
|
|
Returns:
|
|
ServerResponse: The response
|
|
"""
|
|
return Result.succ(service.delete_document(document_id))
|
|
|
|
|
|
def init_endpoints(system_app: SystemApp) -> None:
|
|
"""Initialize the endpoints"""
|
|
global global_system_app
|
|
system_app.register(Service)
|
|
global_system_app = system_app
|