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.2.tar.gz (385.8 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.2-py3-none-any.whl (107.0 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: homesec-1.0.2.tar.gz
  • Upload date:
  • Size: 385.8 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.2.tar.gz
Algorithm Hash digest
SHA256 3d1d76f74f39949cf1ebf703f2f158fa24f74748792537095c456763d247d65f
MD5 d2f9e991a57b2e9281fb8f87a389dd4c
BLAKE2b-256 83b62ef99bc25e0d001dd83e771dad8a660ead3bf33e0a108cc0a42c94c111cf

See more details on using hashes here.

Provenance

The following attestation bundles were made for homesec-1.0.2.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.2-py3-none-any.whl.

File metadata

  • Download URL: homesec-1.0.2-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.2-py3-none-any.whl
Algorithm Hash digest
SHA256 d40a4a28ec69cccf472c9d636035c9a73f967a664174a5ffa14b3431909d05a5
MD5 d5192e4c1bd1fbe0eda0517b5372d6cf
BLAKE2b-256 a96ec9313f15ba2c7810f43a953caa8ed075b6c2eb49f42a822cfb0871776981

See more details on using hashes here.

Provenance

The following attestation bundles were made for homesec-1.0.2-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