Backend#
The ViPPET backend lives in vippet/.
It is a Python 3.12 / FastAPI service that exposes a REST API on port 7860,
orchestrates GStreamer + DLStreamer pipelines as subprocesses, runs
benchmarks, and proxies model installation to the model-download
microservice.
This page covers what a contributor needs in order to navigate the code, run it locally, follow the conventions, and validate changes before opening a pull request. For an end-to-end architectural overview see ViPPET Backend.
Source layout#
vippet/
├── api/ # FastAPI app
│ ├── main.py # App entrypoint, router registration, middleware
│ ├── api_schemas.py # Pydantic v2 request/response models
│ ├── middleware.py # Custom middleware (e.g. upload size limits)
│ ├── routes/ # Route handlers, one module per resource
│ │ ├── pipelines.py
│ │ ├── pipeline_templates.py
│ │ ├── models.py
│ │ ├── jobs.py
│ │ ├── cameras.py
│ │ ├── videos.py
│ │ ├── images.py
│ │ ├── devices.py
│ │ ├── convert.py
│ │ ├── tests.py
│ │ └── health.py
│ └── static/ # Static assets served by the API
├── managers/ # Business logic; thread-safe singletons
│ ├── pipeline_manager.py
│ ├── pipeline_template_manager.py
│ ├── camera_manager.py
│ ├── model_manager.py
│ ├── metadata_manager.py
│ ├── optimization_manager.py
│ ├── validation_manager.py
│ ├── app_state_manager.py
│ └── tests_manager.py
├── pipelines/ # Built-in GStreamer pipeline YAMLs + loader
├── graph.py # In-memory pipeline graph (parse / serialize / simple view)
├── pipeline_runner.py # Subprocess-based pipeline executor
├── gst_runner.py # Low-level GStreamer runner (invoked as subprocess)
├── benchmark.py # Density benchmarking
├── video_encoder.py # Encoding / live-streaming sub-pipelines
├── video_decoder.py # Decoder selection / caps building
├── device.py # OpenVINO device detection (CPU/GPU/NPU)
├── camera.py # Camera enumeration helpers
├── videos.py # Input/output video management
├── images.py # Image set management
├── models.py # Supported models catalog
├── resources.py # Shared resource managers (labels, scripts, model-procs)
├── utils.py # Generic helpers (ids, timestamps, slugify, ...)
├── internal_types.py # Internal dataclasses used between managers
├── requirements.txt # Production dependencies
├── requirements-dev.txt # Dev / test dependencies
├── tests/ # unittest + pytest suites
└── Dockerfile # Multi-stage image (prod / test targets)
A few invariants worth knowing:
Managers (
PipelineManager,CameraManager, etc.) are thread-safe singletons. Create them withManager(), do not pass instances around.The
vippet/package uses relative imports and is meant to be run from the container context (PYTHONPATH=/app). Tests run inside the same image.GStreamer pipelines are not executed in-process.
pipeline_runner.pybuilds a command line and startsgst_runner.pyas a subprocess.OpenVINO device detection (
device.py) happens at startup and drives which hardware profile (cpu,gpu,npu) is selected at compose time.
Tech stack#
Layer |
Technology |
|---|---|
Language |
Python 3.12 |
Web framework |
FastAPI + uvicorn |
Validation |
Pydantic v2 |
AI inference |
OpenVINO™ 2025.x, DLStreamer 2026.x, GStreamer 1.0 |
Metrics |
Telegraf, qmassa (GPU), InfluxDB line protocol |
Containerization |
Docker Compose (profiles: |
Lint / type |
ruff, pyright (strict) |
Tests |
unittest + coverage (unit), pytest (functional) |
Local development loop#
The recommended workflow uses Docker Compose with the dev override, which mounts the source tree into the running container and disables the health check so you can iterate without rebuilding the image:
./setup_env.sh # auto-detect CPU/GPU/NPU, write .env
make env-setup # create shared/ subdirectories
make build-dev # build images with target=dev
make run-dev # start all services with compose.dev.yml overlay
make shell-vippet # exec into the vippet container
Useful targets while iterating:
Target |
Description |
|---|---|
|
Start the stack with live code reload. |
|
Stop all compose services. |
|
Stop containers and remove generated volumes. |
|
Open a shell in the |
|
Shell in |
|
Shell in |
|
Regenerate the OpenAPI schema after changing routes/models. |
The API is exposed at http://localhost:7860/api/v1/ and the
auto-generated Swagger UI at http://localhost:7860/docs.
Coding standards#
Strictly follow the rules below, they are enforced by ruff, pyright
and code review.
Python typing (Python 3.12+)#
Use built-in generics:
list,dict,set,tuple.Use
|for unions andT | Nonefor optional values.Do not use
List,Dict,Union,Optionalfromtyping.Import from
typingonly when actually necessary (Literal,Protocol,TypeVar, …).
def process(data: list[dict[str, int]] | None) -> bool:
return data is not None
FastAPI routes#
Place every route in
vippet/api/routes/, one module per resource.Use
async deffor handlers.Use Pydantic v2 models from
api_schemas.pyfor both request bodies and responses; call.model_dump()(never the deprecated.dict()).Document each endpoint with a markdown docstring, Swagger renders it. See the example in
AGENTS.md.Use
logging(module-level logger), neverprint().
Managers and helpers#
Regular (non-API) docstrings follow Google / NumPy style, no markdown.
Class names:
PascalCase. Functions/variables:snake_case. Constants:UPPER_SNAKE_CASE. Private methods:_leading_underscore.Keep changes small and focused; do not bundle unrelated refactors.
Logging#
Acquire a logger with
logging.getLogger(__name__)(or a stable manager-specific name such as"PipelineManager").Levels are configured by environment variables:
APP_LOG_LEVEL(application),RUNNER_LOG_LEVEL(gst_runner.pysubprocess),WEB_SERVER_LOG_LEVEL(uvicorn),GST_DEBUG(GStreamer native, integer0-9).
Tests#
Unit tests live in vippet/tests/unit/ and use Python’s unittest
framework with coverage. They are device-agnostic and run inside the
cpu profile image:
make test
This builds an image with TARGET=test, mounts the source tree, runs
unittest discover against tests/unit/, and prints a coverage report
plus an HTML report under /tmp/.vippet-coverage-html inside the
container.
Functional / smoke tests use pytest:
make test-smoke # marker: smoke
make test-full # full functional suite
When you add a feature:
Put unit tests next to existing ones in
vippet/tests/unit/<area>_tests/(for examplemanagers_tests/,api_tests/).Cover both the happy path and the error paths (invalid input, missing resource, conflicting state).
For new API endpoints, add tests that exercise the route through FastAPI’s
TestClient.
Linting and formatting#
make lint # markdownlint + ruff check + ruff format --check + pyright
make fix-linter # ruff check --fix
make format # ruff format
Rules:
ruffruns with the project configuration.pyrightruns in strict mode (seepyrightconfig.json). Avoid# type: ignore, when truly required add a one-line justification.Markdown files are linted by
markdownlint.
API schema and clients#
The OpenAPI schema is the contract between the backend and the UI. Whenever you add, remove, or change a route, a request body, or a response model you must regenerate both the schema and the TypeScript client used by the UI.
1. Regenerate the OpenAPI schema#
From the repository root:
make generate_openapi
This runs generate_openapi.py against the FastAPI app and writes the
resulting schema to docs/user-guide/_assets/vippet.json. Commit this file
together with your backend changes.
2. Regenerate the UI client#
The UI consumes the schema through @rtk-query/codegen-openapi, configured
in ui/src/api/api.config.json. After the schema has been regenerated,
rebuild the typed client from the ui/ directory:
cd ui
npm run generate:api
This produces ui/src/api/api.generated.ts (RTK Query hooks and types).
Commit the regenerated file together with the schema. If the build or the
UI starts reporting unknown endpoints or type mismatches after a backend
change, the most common cause is that one of these two steps was skipped.
Adding a new route, quick checklist#
Add or extend a Pydantic model in
vippet/api/api_schemas.py.Create the handler in
vippet/api/routes/<resource>.py(or a new module, registered invippet/api/main.py).Use
async def, type hints, and a markdown docstring with response codes and examples.Delegate business logic to the appropriate manager, keep route handlers thin.
Add unit tests under
vippet/tests/unit/api_tests/.Run
make lint test, thenmake generate_openapi, thencd ui && npm run generate:api.
Important constraints#
Do not commit
.envfiles or model artifacts.All new Dockerfiles must follow the existing multi-stage
prod/testpattern.New environment variables must be documented in the README and in
AGENTS.md(the Key Environment Variables table).