mirror of
https://github.com/meta-llama/llama-stack.git
synced 2025-10-03 19:57:35 +00:00
Rather than have a single `LLAMA_STACK_VERSION`, we need to have a `_V1`, `_V1ALPHA`, and `_V1BETA` constant. This also necessitated addition of `level` to the `WebMethod` so that routing can be handeled properly. For backwards compat, the `v1` routes are being kept around and marked as `deprecated`. When used, the server will log a deprecation warning. move: post_training to v1alpha as it is under heavy development and not near its final state eval: job scheduling is not implemented. Relies heavily on the datasetio API which is under development missing implementations of specific routes indicating the structure of those routes might change. Additionally eval depends on the inference API which is going to be deprecated, eval will likely need a major API surface change to conform to using completions properly Signed-off-by: Charlie Doern <cdoern@redhat.com>
187 lines
6.6 KiB
Python
187 lines
6.6 KiB
Python
# Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
# All rights reserved.
|
|
#
|
|
# This source code is licensed under the terms described in the LICENSE file in
|
|
# the root directory of this source tree.
|
|
|
|
from enum import StrEnum
|
|
from typing import Annotated, ClassVar, Literal, Protocol, runtime_checkable
|
|
|
|
from fastapi import File, Form, Response, UploadFile
|
|
from pydantic import BaseModel, Field
|
|
|
|
from llama_stack.apis.common.responses import Order
|
|
from llama_stack.apis.version import LLAMA_STACK_API_V1
|
|
from llama_stack.providers.utils.telemetry.trace_protocol import trace_protocol
|
|
from llama_stack.schema_utils import json_schema_type, webmethod
|
|
|
|
|
|
# OpenAI Files API Models
|
|
class OpenAIFilePurpose(StrEnum):
|
|
"""
|
|
Valid purpose values for OpenAI Files API.
|
|
"""
|
|
|
|
ASSISTANTS = "assistants"
|
|
BATCH = "batch"
|
|
# TODO: Add other purposes as needed
|
|
|
|
|
|
@json_schema_type
|
|
class OpenAIFileObject(BaseModel):
|
|
"""
|
|
OpenAI File object as defined in the OpenAI Files API.
|
|
|
|
:param object: The object type, which is always "file"
|
|
:param id: The file identifier, which can be referenced in the API endpoints
|
|
:param bytes: The size of the file, in bytes
|
|
:param created_at: The Unix timestamp (in seconds) for when the file was created
|
|
:param expires_at: The Unix timestamp (in seconds) for when the file expires
|
|
:param filename: The name of the file
|
|
:param purpose: The intended purpose of the file
|
|
"""
|
|
|
|
object: Literal["file"] = "file"
|
|
id: str
|
|
bytes: int
|
|
created_at: int
|
|
expires_at: int
|
|
filename: str
|
|
purpose: OpenAIFilePurpose
|
|
|
|
|
|
@json_schema_type
|
|
class ExpiresAfter(BaseModel):
|
|
"""
|
|
Control expiration of uploaded files.
|
|
|
|
Params:
|
|
- anchor, must be "created_at"
|
|
- seconds, must be int between 3600 and 2592000 (1 hour to 30 days)
|
|
"""
|
|
|
|
MIN: ClassVar[int] = 3600 # 1 hour
|
|
MAX: ClassVar[int] = 2592000 # 30 days
|
|
|
|
anchor: Literal["created_at"]
|
|
seconds: int = Field(..., ge=3600, le=2592000)
|
|
|
|
|
|
@json_schema_type
|
|
class ListOpenAIFileResponse(BaseModel):
|
|
"""
|
|
Response for listing files in OpenAI Files API.
|
|
|
|
:param data: List of file objects
|
|
:param has_more: Whether there are more files available beyond this page
|
|
:param first_id: ID of the first file in the list for pagination
|
|
:param last_id: ID of the last file in the list for pagination
|
|
:param object: The object type, which is always "list"
|
|
"""
|
|
|
|
data: list[OpenAIFileObject]
|
|
has_more: bool
|
|
first_id: str
|
|
last_id: str
|
|
object: Literal["list"] = "list"
|
|
|
|
|
|
@json_schema_type
|
|
class OpenAIFileDeleteResponse(BaseModel):
|
|
"""
|
|
Response for deleting a file in OpenAI Files API.
|
|
|
|
:param id: The file identifier that was deleted
|
|
:param object: The object type, which is always "file"
|
|
:param deleted: Whether the file was successfully deleted
|
|
"""
|
|
|
|
id: str
|
|
object: Literal["file"] = "file"
|
|
deleted: bool
|
|
|
|
|
|
@runtime_checkable
|
|
@trace_protocol
|
|
class Files(Protocol):
|
|
# OpenAI Files API Endpoints
|
|
@webmethod(route="/openai/v1/files", method="POST", level=LLAMA_STACK_API_V1)
|
|
async def openai_upload_file(
|
|
self,
|
|
file: Annotated[UploadFile, File()],
|
|
purpose: Annotated[OpenAIFilePurpose, Form()],
|
|
expires_after_anchor: Annotated[str | None, Form(alias="expires_after[anchor]")] = None,
|
|
expires_after_seconds: Annotated[int | None, Form(alias="expires_after[seconds]")] = None,
|
|
# TODO: expires_after is producing strange openapi spec, params are showing up as a required w/ oneOf being null
|
|
) -> OpenAIFileObject:
|
|
"""
|
|
Upload a file that can be used across various endpoints.
|
|
|
|
The file upload should be a multipart form request with:
|
|
- file: The File object (not file name) to be uploaded.
|
|
- purpose: The intended purpose of the uploaded file.
|
|
- expires_after: Optional form values describing expiration for the file. Expected expires_after[anchor] = "created_at", expires_after[seconds] = {integer}. Seconds must be between 3600 and 2592000 (1 hour to 30 days).
|
|
|
|
:param file: The uploaded file object containing content and metadata (filename, content_type, etc.).
|
|
:param purpose: The intended purpose of the uploaded file (e.g., "assistants", "fine-tune").
|
|
:returns: An OpenAIFileObject representing the uploaded file.
|
|
"""
|
|
...
|
|
|
|
@webmethod(route="/openai/v1/files", method="GET", level=LLAMA_STACK_API_V1)
|
|
async def openai_list_files(
|
|
self,
|
|
after: str | None = None,
|
|
limit: int | None = 10000,
|
|
order: Order | None = Order.desc,
|
|
purpose: OpenAIFilePurpose | None = None,
|
|
) -> ListOpenAIFileResponse:
|
|
"""
|
|
Returns a list of files that belong to the user's organization.
|
|
|
|
:param after: A cursor for use in pagination. `after` is an object ID that defines your place in the list. For instance, if you make a list request and receive 100 objects, ending with obj_foo, your subsequent call can include after=obj_foo in order to fetch the next page of the list.
|
|
:param limit: A limit on the number of objects to be returned. Limit can range between 1 and 10,000, and the default is 10,000.
|
|
:param order: Sort order by the `created_at` timestamp of the objects. `asc` for ascending order and `desc` for descending order.
|
|
:param purpose: Only return files with the given purpose.
|
|
:returns: An ListOpenAIFileResponse containing the list of files.
|
|
"""
|
|
...
|
|
|
|
@webmethod(route="/openai/v1/files/{file_id}", method="GET", level=LLAMA_STACK_API_V1)
|
|
async def openai_retrieve_file(
|
|
self,
|
|
file_id: str,
|
|
) -> OpenAIFileObject:
|
|
"""
|
|
Returns information about a specific file.
|
|
|
|
:param file_id: The ID of the file to use for this request.
|
|
:returns: An OpenAIFileObject containing file information.
|
|
"""
|
|
...
|
|
|
|
@webmethod(route="/openai/v1/files/{file_id}", method="DELETE", level=LLAMA_STACK_API_V1)
|
|
async def openai_delete_file(
|
|
self,
|
|
file_id: str,
|
|
) -> OpenAIFileDeleteResponse:
|
|
"""
|
|
Delete a file.
|
|
|
|
:param file_id: The ID of the file to use for this request.
|
|
:returns: An OpenAIFileDeleteResponse indicating successful deletion.
|
|
"""
|
|
...
|
|
|
|
@webmethod(route="/openai/v1/files/{file_id}/content", method="GET", level=LLAMA_STACK_API_V1)
|
|
async def openai_retrieve_file_content(
|
|
self,
|
|
file_id: str,
|
|
) -> Response:
|
|
"""
|
|
Returns the contents of the specified file.
|
|
|
|
:param file_id: The ID of the file to use for this request.
|
|
:returns: The raw file content as a binary response.
|
|
"""
|
|
...
|