mirror of
https://github.com/invoke-ai/InvokeAI
synced 2024-08-30 20:32:17 +00:00
402cf9b0ee
Refactor services folder/module structure. **Motivation** While working on our services I've repeatedly encountered circular imports and a general lack of clarity regarding where to put things. The structure introduced goes a long way towards resolving those issues, setting us up for a clean structure going forward. **Services** Services are now in their own folder with a few files: - `services/{service_name}/__init__.py`: init as needed, mostly empty now - `services/{service_name}/{service_name}_base.py`: the base class for the service - `services/{service_name}/{service_name}_{impl_type}.py`: the default concrete implementation of the service - typically one of `sqlite`, `default`, or `memory` - `services/{service_name}/{service_name}_common.py`: any common items - models, exceptions, utilities, etc Though it's a bit verbose to have the service name both as the folder name and the prefix for files, I found it is _extremely_ confusing to have all of the base classes just be named `base.py`. So, at the cost of some verbosity when importing things, I've included the service name in the filename. There are some minor logic changes. For example, in `InvocationProcessor`, instead of assigning the model manager service to a variable to be used later in the file, the service is used directly via the `Invoker`. **Shared** Things that are used across disparate services are in `services/shared/`: - `default_graphs.py`: previously in `services/` - `graphs.py`: previously in `services/` - `paginatation`: generic pagination models used in a few services - `sqlite`: the `SqliteDatabase` class, other sqlite-specific things
151 lines
5.2 KiB
Python
151 lines
5.2 KiB
Python
from typing import Optional, Union
|
|
|
|
from fastapi import Body, HTTPException, Path, Query
|
|
from fastapi.routing import APIRouter
|
|
from pydantic import BaseModel, Field
|
|
|
|
from invokeai.app.services.board_records.board_records_common import BoardChanges
|
|
from invokeai.app.services.boards.boards_common import BoardDTO
|
|
from invokeai.app.services.shared.pagination import OffsetPaginatedResults
|
|
|
|
from ..dependencies import ApiDependencies
|
|
|
|
boards_router = APIRouter(prefix="/v1/boards", tags=["boards"])
|
|
|
|
|
|
class DeleteBoardResult(BaseModel):
|
|
board_id: str = Field(description="The id of the board that was deleted.")
|
|
deleted_board_images: list[str] = Field(
|
|
description="The image names of the board-images relationships that were deleted."
|
|
)
|
|
deleted_images: list[str] = Field(description="The names of the images that were deleted.")
|
|
|
|
|
|
@boards_router.post(
|
|
"/",
|
|
operation_id="create_board",
|
|
responses={
|
|
201: {"description": "The board was created successfully"},
|
|
},
|
|
status_code=201,
|
|
response_model=BoardDTO,
|
|
)
|
|
async def create_board(
|
|
board_name: str = Query(description="The name of the board to create"),
|
|
) -> BoardDTO:
|
|
"""Creates a board"""
|
|
try:
|
|
result = ApiDependencies.invoker.services.boards.create(board_name=board_name)
|
|
return result
|
|
except Exception:
|
|
raise HTTPException(status_code=500, detail="Failed to create board")
|
|
|
|
|
|
@boards_router.get("/{board_id}", operation_id="get_board", response_model=BoardDTO)
|
|
async def get_board(
|
|
board_id: str = Path(description="The id of board to get"),
|
|
) -> BoardDTO:
|
|
"""Gets a board"""
|
|
|
|
try:
|
|
result = ApiDependencies.invoker.services.boards.get_dto(board_id=board_id)
|
|
return result
|
|
except Exception:
|
|
raise HTTPException(status_code=404, detail="Board not found")
|
|
|
|
|
|
@boards_router.patch(
|
|
"/{board_id}",
|
|
operation_id="update_board",
|
|
responses={
|
|
201: {
|
|
"description": "The board was updated successfully",
|
|
},
|
|
},
|
|
status_code=201,
|
|
response_model=BoardDTO,
|
|
)
|
|
async def update_board(
|
|
board_id: str = Path(description="The id of board to update"),
|
|
changes: BoardChanges = Body(description="The changes to apply to the board"),
|
|
) -> BoardDTO:
|
|
"""Updates a board"""
|
|
try:
|
|
result = ApiDependencies.invoker.services.boards.update(board_id=board_id, changes=changes)
|
|
return result
|
|
except Exception:
|
|
raise HTTPException(status_code=500, detail="Failed to update board")
|
|
|
|
|
|
@boards_router.delete("/{board_id}", operation_id="delete_board", response_model=DeleteBoardResult)
|
|
async def delete_board(
|
|
board_id: str = Path(description="The id of board to delete"),
|
|
include_images: Optional[bool] = Query(description="Permanently delete all images on the board", default=False),
|
|
) -> DeleteBoardResult:
|
|
"""Deletes a board"""
|
|
try:
|
|
if include_images is True:
|
|
deleted_images = ApiDependencies.invoker.services.board_images.get_all_board_image_names_for_board(
|
|
board_id=board_id
|
|
)
|
|
ApiDependencies.invoker.services.images.delete_images_on_board(board_id=board_id)
|
|
ApiDependencies.invoker.services.boards.delete(board_id=board_id)
|
|
return DeleteBoardResult(
|
|
board_id=board_id,
|
|
deleted_board_images=[],
|
|
deleted_images=deleted_images,
|
|
)
|
|
else:
|
|
deleted_board_images = ApiDependencies.invoker.services.board_images.get_all_board_image_names_for_board(
|
|
board_id=board_id
|
|
)
|
|
ApiDependencies.invoker.services.boards.delete(board_id=board_id)
|
|
return DeleteBoardResult(
|
|
board_id=board_id,
|
|
deleted_board_images=deleted_board_images,
|
|
deleted_images=[],
|
|
)
|
|
except Exception:
|
|
raise HTTPException(status_code=500, detail="Failed to delete board")
|
|
|
|
|
|
@boards_router.get(
|
|
"/",
|
|
operation_id="list_boards",
|
|
response_model=Union[OffsetPaginatedResults[BoardDTO], list[BoardDTO]],
|
|
)
|
|
async def list_boards(
|
|
all: Optional[bool] = Query(default=None, description="Whether to list all boards"),
|
|
offset: Optional[int] = Query(default=None, description="The page offset"),
|
|
limit: Optional[int] = Query(default=None, description="The number of boards per page"),
|
|
) -> Union[OffsetPaginatedResults[BoardDTO], list[BoardDTO]]:
|
|
"""Gets a list of boards"""
|
|
if all:
|
|
return ApiDependencies.invoker.services.boards.get_all()
|
|
elif offset is not None and limit is not None:
|
|
return ApiDependencies.invoker.services.boards.get_many(
|
|
offset,
|
|
limit,
|
|
)
|
|
else:
|
|
raise HTTPException(
|
|
status_code=400,
|
|
detail="Invalid request: Must provide either 'all' or both 'offset' and 'limit'",
|
|
)
|
|
|
|
|
|
@boards_router.get(
|
|
"/{board_id}/image_names",
|
|
operation_id="list_all_board_image_names",
|
|
response_model=list[str],
|
|
)
|
|
async def list_all_board_image_names(
|
|
board_id: str = Path(description="The id of the board"),
|
|
) -> list[str]:
|
|
"""Gets a list of images for a board"""
|
|
|
|
image_names = ApiDependencies.invoker.services.board_images.get_all_board_image_names_for_board(
|
|
board_id,
|
|
)
|
|
return image_names
|