Skip to main content

Pluggable async home security camera pipeline with detection, VLM analysis, and alerts.

Project description

HomeSec

License: Apache 2.0 Python: 3.10+ Typing: Typed

HomeSec is a self-hosted, extensible network video recorder that puts you in control. Store clips wherever you want, analyze them with AI, and get smart notifications—all while keeping your footage private and off third-party clouds.

Under the hood, it's a pluggable async pipeline for home security cameras. It records short clips, runs object detection, optionally calls a vision-language model (VLM) for a structured summary, and sends alerts via MQTT or email. The design leans toward reliability: clips land on disk first, state/event writes are best-effort, and non-critical stages can fail without losing the alert.

Highlights

  • Bring your own input: RTSP motion detection, FTP uploads, or a watched folder
  • Parallel upload + filter (YOLOv8) with frame sampling and early exit
  • OpenAI-compatible VLM analysis with structured output
  • Policy-driven alerts with per-camera overrides
  • Fan-out notifiers (MQTT for Home Assistant, SendGrid email)
  • Postgres-backed state + events with graceful degradation
  • Built around small, stable interfaces so new plugins drop in cleanly
  • Health endpoint plus optional Postgres telemetry logging

Pipeline at a glance

ClipSource -> (Upload + Filter) -> VLM (optional) -> Alert Policy -> Notifier(s)
  • Upload and filter run in parallel; VLM runs only when trigger classes are detected.
  • Upload failures do not block alerts; filter failures stop processing.
  • State is stored in Postgres (clip_states + clip_events) when available.

Quickstart

Requirements

  • Docker and Docker Compose
  • Optional: MQTT broker, Dropbox credentials, OpenAI-compatible API key

Setup

  1. Create a config file:
    cp config/example.yaml config/config.yaml
    # Edit config/config.yaml with your settings
    
  2. Set environment variables:
    cp .env.example .env
    # Edit .env with your credentials
    
  3. Start HomeSec + Postgres:
    make up
    
  4. Stop:
    make down
    

Running without Docker

If you prefer to run locally:

  1. Install Python 3.10+ and ffmpeg
  2. uv sync
  3. make db (starts Postgres)
  4. make run

Configuration

Configs are YAML and validated with Pydantic. See config/example.yaml for all options.

Minimal example (RTSP + Dropbox + MQTT):

version: 1

cameras:
  - name: front_door
    source:
      type: rtsp
      config:
        rtsp_url_env: FRONT_DOOR_RTSP_URL
        output_dir: "./recordings"

storage:
  backend: dropbox
  dropbox:
    root: "/homecam"
    token_env: DROPBOX_TOKEN
    app_key_env: DROPBOX_APP_KEY
    app_secret_env: DROPBOX_APP_SECRET
    refresh_token_env: DROPBOX_REFRESH_TOKEN

state_store:
  dsn_env: DB_DSN

notifiers:
  - backend: mqtt
    config:
      host: "localhost"
      port: 1883

filter:
  plugin: yolo
  config:
    classes: ["person"]
    min_confidence: 0.5

vlm:
  backend: openai
  llm:
    api_key_env: OPENAI_API_KEY
    model: gpt-4o

alert_policy:
  backend: default
  enabled: true
  config:
    min_risk_level: medium

per_camera_alert:
  front_door:
    min_risk_level: low
    notify_on_activity_types: ["person_at_door", "delivery"]

A few things worth knowing:

  • Secrets never go in YAML. Use env var names (*_env) and set values in your shell or .env.
  • At least one notifier must be enabled (mqtt or sendgrid_email).
  • Built-in YOLO classes: person, car, truck, motorcycle, bicycle, dog, cat, bird, backpack, handbag, suitcase.
  • Local storage for development:
storage:
  backend: local
  local:
    root: "./storage"
  • Set alert_policy.enabled: false to disable notifications (a noop policy is used).
  • For a quick local run, pair local_folder with local storage and drop a clip into recordings/.

Extensible by design

HomeSec is intentionally modular. Each major capability is an interface (ClipSource, StorageBackend, ObjectFilter, VLMAnalyzer, AlertPolicy, Notifier) defined in src/homesec/interfaces.py, and plugins are discovered at runtime via entry points. This keeps the core pipeline small while making it easy to add new backends without editing core code.

What this means in practice:

  • Swap storage or notifications by changing config, not code.
  • Add a new plugin type as a separate package and register it.
  • Keep config validation strict by pairing each plugin with a Pydantic model.

Extension points (all pluggable):

  • Sources: RTSP motion detection, FTP uploads, local folders
  • Storage backends: Dropbox, local disk, or your own
  • Filters: object detection (YOLO or custom models)
  • VLM analyzers: OpenAI-compatible APIs or local models
  • Alert policies: per-camera rules and thresholds
  • Notifiers: MQTT, email, or anything else you can send from Python

CLI

  • Run the pipeline: uv run python -m homesec.cli run --config config/config.yaml --log_level INFO
  • Validate config: uv run python -m homesec.cli validate --config config/config.yaml
  • Cleanup (reanalyze and optionally delete empty clips): uv run python -m homesec.cli cleanup --config config/config.yaml --older_than_days 7 --dry_run True

Built-in plugins

  • Filters: yolo
  • VLM analyzers: openai (OpenAI-compatible API)
  • Storage: dropbox, local
  • Notifiers: mqtt, sendgrid_email
  • Alert policies: default, noop

Writing a plugin

HomeSec discovers plugins via entry points in the homesec.plugins group. A plugin module just needs to import and register itself.

Each plugin provides:

  • A unique name (used in config)
  • A Pydantic config model for validation
  • A factory that builds the concrete implementation
# my_package/filters/custom.py
from pydantic import BaseModel
from homesec.interfaces import ObjectFilter
from homesec.plugins.filters import FilterPlugin, filter_plugin

class CustomConfig(BaseModel):
    threshold: float = 0.5

class CustomFilter(ObjectFilter):
    ...

@filter_plugin(name="custom")
def register() -> FilterPlugin:
    return FilterPlugin(
        name="custom",
        config_model=CustomConfig,
        factory=lambda cfg: CustomFilter(cfg),
    )
# pyproject.toml
[project.entry-points."homesec.plugins"]
my_filters = "my_package.filters.custom"

Observability

  • Health endpoint: GET /health (configurable in health.host/health.port)
  • Optional telemetry logs to Postgres when DB_DSN is set:
    • Start local DB: make db
    • Run migrations: make db-migrate

Development

  • Run tests: make test
  • Run type checking (strict): make typecheck
  • Run both: make check
  • Tests must include Given/When/Then comments.
  • Architecture notes: DESIGN.md

License

Apache 2.0. See LICENSE.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

homesec-1.0.1.tar.gz (385.7 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

homesec-1.0.1-py3-none-any.whl (107.0 kB view details)

Uploaded Python 3

File details

Details for the file homesec-1.0.1.tar.gz.

File metadata

  • Download URL: homesec-1.0.1.tar.gz
  • Upload date:
  • Size: 385.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for homesec-1.0.1.tar.gz
Algorithm Hash digest
SHA256 5f19cd44af2db7d1dba7529fc9eb3fb360af82275ba9ae82cb83ce90858f7839
MD5 fa1abf11c66c3a1fe3a342af9433a79d
BLAKE2b-256 102a97d4114e3d980e9891cac41774e6237801e949c1ab3683aaa9b7d6e558d8

See more details on using hashes here.

Provenance

The following attestation bundles were made for homesec-1.0.1.tar.gz:

Publisher: release.yaml on lan17/homesec

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file homesec-1.0.1-py3-none-any.whl.

File metadata

  • Download URL: homesec-1.0.1-py3-none-any.whl
  • Upload date:
  • Size: 107.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for homesec-1.0.1-py3-none-any.whl
Algorithm Hash digest
SHA256 6fa63b82da8236cebf277ba06dfaaa903b8931188edbf2f660d95527b7ac2562
MD5 4bd6320fdd5a64bfc40838fa770ae7ef
BLAKE2b-256 d6c21c8345110882ad9ed0141ab341a201ea6a85ebd3eb56481ec7cb67401cfd

See more details on using hashes here.

Provenance

The following attestation bundles were made for homesec-1.0.1-py3-none-any.whl:

Publisher: release.yaml on lan17/homesec

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page