api_server.py 3.33 KB
Newer Older
Zhuohan Li's avatar
Zhuohan Li committed
1
2
3
4
import argparse
import json
from typing import AsyncGenerator

5
from fastapi import FastAPI, Request
6
from fastapi.responses import JSONResponse, Response, StreamingResponse
Zhuohan Li's avatar
Zhuohan Li committed
7
8
import uvicorn

Woosuk Kwon's avatar
Woosuk Kwon committed
9
10
11
12
from vllm.engine.arg_utils import AsyncEngineArgs
from vllm.engine.async_llm_engine import AsyncLLMEngine
from vllm.sampling_params import SamplingParams
from vllm.utils import random_uuid
Zhuohan Li's avatar
Zhuohan Li committed
13

14
TIMEOUT_KEEP_ALIVE = 5  # seconds.
Zhuohan Li's avatar
Zhuohan Li committed
15
app = FastAPI()
16
engine = None
Zhuohan Li's avatar
Zhuohan Li committed
17
18


19
20
21
22
23
24
@app.get("/health")
async def health() -> Response:
    """Health check."""
    return Response(status_code=200)


Zhuohan Li's avatar
Zhuohan Li committed
25
@app.post("/generate")
26
async def generate(request: Request) -> Response:
27
    """Generate completion for the request.
28
29
30

    The request should be a JSON object with the following fields:
    - prompt: the prompt to use for the generation.
31
    - stream: whether to stream the results or not.
32
33
    - other fields: the sampling parameters (See `SamplingParams` for details).
    """
Zhuohan Li's avatar
Zhuohan Li committed
34
35
    request_dict = await request.json()
    prompt = request_dict.pop("prompt")
36
    prefix_pos = request_dict.pop("prefix_pos", None)
37
    stream = request_dict.pop("stream", False)
Zhuohan Li's avatar
Zhuohan Li committed
38
    sampling_params = SamplingParams(**request_dict)
39
    request_id = random_uuid()
40

41
42
43
44
    results_generator = engine.generate(prompt,
                                        sampling_params,
                                        request_id,
                                        prefix_pos=prefix_pos)
Zhuohan Li's avatar
Zhuohan Li committed
45

46
    # Streaming case
Zhuohan Li's avatar
Zhuohan Li committed
47
48
49
50
    async def stream_results() -> AsyncGenerator[bytes, None]:
        async for request_output in results_generator:
            prompt = request_output.prompt
            text_outputs = [
51
                prompt + output.text for output in request_output.outputs
Zhuohan Li's avatar
Zhuohan Li committed
52
            ]
53
            ret = {"text": text_outputs}
Zhuohan Li's avatar
Zhuohan Li committed
54
55
            yield (json.dumps(ret) + "\0").encode("utf-8")

56
    if stream:
57
        return StreamingResponse(stream_results())
58
59
60
61
62
63

    # Non-streaming case
    final_output = None
    async for request_output in results_generator:
        if await request.is_disconnected():
            # Abort the request if the client disconnects.
Zhuohan Li's avatar
Zhuohan Li committed
64
            await engine.abort(request_id)
65
66
67
68
69
            return Response(status_code=499)
        final_output = request_output

    assert final_output is not None
    prompt = final_output.prompt
70
    text_outputs = [prompt + output.text for output in final_output.outputs]
71
    ret = {"text": text_outputs}
72
    return JSONResponse(ret)
Zhuohan Li's avatar
Zhuohan Li committed
73
74
75
76


if __name__ == "__main__":
    parser = argparse.ArgumentParser()
77
    parser.add_argument("--host", type=str, default=None)
78
    parser.add_argument("--port", type=int, default=8000)
79
80
    parser.add_argument("--ssl-keyfile", type=str, default=None)
    parser.add_argument("--ssl-certfile", type=str, default=None)
81
82
83
84
85
    parser.add_argument(
        "--root-path",
        type=str,
        default=None,
        help="FastAPI root_path when app is behind a path based routing proxy")
Zhuohan Li's avatar
Zhuohan Li committed
86
    parser = AsyncEngineArgs.add_cli_args(parser)
Zhuohan Li's avatar
Zhuohan Li committed
87
88
    args = parser.parse_args()

Zhuohan Li's avatar
Zhuohan Li committed
89
    engine_args = AsyncEngineArgs.from_cli_args(args)
90
    engine = AsyncLLMEngine.from_engine_args(engine_args)
Zhuohan Li's avatar
Zhuohan Li committed
91

92
    app.root_path = args.root_path
93
94
95
96
    uvicorn.run(app,
                host=args.host,
                port=args.port,
                log_level="debug",
97
98
99
                timeout_keep_alive=TIMEOUT_KEEP_ALIVE,
                ssl_keyfile=args.ssl_keyfile,
                ssl_certfile=args.ssl_certfile)