* use posix spawn on alpine * Avoid subprocess memory copy when c library supports posix_spawn By default python 3.10 will use the fork() which has to copy all the memory of the parent process (in our case this can be huge since Home Assistant core can use hundreds of megabytes of RAM). By using posix_spawn this is avoided. In python 3.11 vfork will also be available https://github.com/python/cpython/issues/80004#issuecomment-1093810689 https://github.com/python/cpython/pull/11671 but we won't always be able to use it and posix_spawn is considered safer https://bugzilla.kernel.org/show_bug.cgi?id=215813#c14 The subprocess library doesn't know about musl though even though it supports posix_spawn https://git.musl-libc.org/cgit/musl/log/src/process/posix_spawn.c so we have to teach it since it only has checks for glibc1b736838e6/Lib/subprocess.py (L745)
The constant is documented as being able to be flipped here: https://docs.python.org/3/library/subprocess.html#disabling-use-of-vfork-or-posix-spawn * Avoid subprocess memory copy when c library supports posix_spawn By default python 3.10 will use the fork() which has to copy memory of the parent process (in our case this can be huge since Home Assistant core can use hundreds of megabytes of RAM). By using posix_spawn this is avoided and subprocess creation does not get discernibly slow the larger the Home Assistant python process grows. In python 3.11 vfork will also be available https://github.com/python/cpython/issues/80004#issuecomment-1093810689 https://github.com/python/cpython/pull/11671 but we won't always be able to use it and posix_spawn is considered safer https://bugzilla.kernel.org/show_bug.cgi?id=215813#c14 The subprocess library doesn't know about musl though even though it supports posix_spawn https://git.musl-libc.org/cgit/musl/log/src/process/posix_spawn.c so we have to teach it since it only has checks for glibc1b736838e6/Lib/subprocess.py (L745)
The constant is documented as being able to be flipped here: https://docs.python.org/3/library/subprocess.html#disabling-use-of-vfork-or-posix-spawn * missed some * adjust more tests * coverage
159 lines
4.8 KiB
Python
159 lines
4.8 KiB
Python
"""Camera platform that has a Raspberry Pi camera."""
|
|
from __future__ import annotations
|
|
|
|
import logging
|
|
import os
|
|
import shutil
|
|
import subprocess
|
|
from tempfile import NamedTemporaryFile
|
|
|
|
from homeassistant.components.camera import Camera
|
|
from homeassistant.const import CONF_FILE_PATH, CONF_NAME, EVENT_HOMEASSISTANT_STOP
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
|
|
|
from .const import (
|
|
CONF_HORIZONTAL_FLIP,
|
|
CONF_IMAGE_HEIGHT,
|
|
CONF_IMAGE_QUALITY,
|
|
CONF_IMAGE_ROTATION,
|
|
CONF_IMAGE_WIDTH,
|
|
CONF_OVERLAY_METADATA,
|
|
CONF_OVERLAY_TIMESTAMP,
|
|
CONF_TIMELAPSE,
|
|
CONF_VERTICAL_FLIP,
|
|
DOMAIN,
|
|
)
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
def kill_raspistill(*args):
|
|
"""Kill any previously running raspistill process.."""
|
|
with subprocess.Popen(
|
|
["killall", "raspistill"],
|
|
stdout=subprocess.DEVNULL,
|
|
stderr=subprocess.STDOUT,
|
|
close_fds=False, # required for posix_spawn
|
|
):
|
|
pass
|
|
|
|
|
|
def setup_platform(
|
|
hass: HomeAssistant,
|
|
config: ConfigType,
|
|
add_entities: AddEntitiesCallback,
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
) -> None:
|
|
"""Set up the Raspberry Camera."""
|
|
# We only want this platform to be set up via discovery.
|
|
# prevent initializing by erroneous platform config section in yaml conf
|
|
if discovery_info is None:
|
|
return
|
|
|
|
if shutil.which("raspistill") is None:
|
|
_LOGGER.error("'raspistill' was not found")
|
|
return
|
|
|
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, kill_raspistill)
|
|
|
|
setup_config = hass.data[DOMAIN]
|
|
file_path = setup_config[CONF_FILE_PATH]
|
|
|
|
def delete_temp_file(*args):
|
|
"""Delete the temporary file to prevent saving multiple temp images.
|
|
|
|
Only used when no path is defined
|
|
"""
|
|
os.remove(file_path)
|
|
|
|
# If no file path is defined, use a temporary file
|
|
if file_path is None:
|
|
with NamedTemporaryFile(suffix=".jpg", delete=False) as temp_file:
|
|
file_path = temp_file.name
|
|
setup_config[CONF_FILE_PATH] = file_path
|
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_STOP, delete_temp_file)
|
|
|
|
# Check whether the file path has been whitelisted
|
|
elif not hass.config.is_allowed_path(file_path):
|
|
_LOGGER.error("'%s' is not a whitelisted directory", file_path)
|
|
return
|
|
|
|
add_entities([RaspberryCamera(setup_config)])
|
|
|
|
|
|
class RaspberryCamera(Camera):
|
|
"""Representation of a Raspberry Pi camera."""
|
|
|
|
def __init__(self, device_info):
|
|
"""Initialize Raspberry Pi camera component."""
|
|
super().__init__()
|
|
|
|
self._name = device_info[CONF_NAME]
|
|
self._config = device_info
|
|
|
|
# Kill if there's raspistill instance
|
|
kill_raspistill()
|
|
|
|
cmd_args = [
|
|
"raspistill",
|
|
"--nopreview",
|
|
"-o",
|
|
device_info[CONF_FILE_PATH],
|
|
"-t",
|
|
"0",
|
|
"-w",
|
|
str(device_info[CONF_IMAGE_WIDTH]),
|
|
"-h",
|
|
str(device_info[CONF_IMAGE_HEIGHT]),
|
|
"-tl",
|
|
str(device_info[CONF_TIMELAPSE]),
|
|
"-q",
|
|
str(device_info[CONF_IMAGE_QUALITY]),
|
|
"-rot",
|
|
str(device_info[CONF_IMAGE_ROTATION]),
|
|
]
|
|
if device_info[CONF_HORIZONTAL_FLIP]:
|
|
cmd_args.append("-hf")
|
|
|
|
if device_info[CONF_VERTICAL_FLIP]:
|
|
cmd_args.append("-vf")
|
|
|
|
if device_info[CONF_OVERLAY_METADATA]:
|
|
cmd_args.append("-a")
|
|
cmd_args.append(str(device_info[CONF_OVERLAY_METADATA]))
|
|
|
|
if device_info[CONF_OVERLAY_TIMESTAMP]:
|
|
cmd_args.append("-a")
|
|
cmd_args.append("4")
|
|
cmd_args.append("-a")
|
|
cmd_args.append(str(device_info[CONF_OVERLAY_TIMESTAMP]))
|
|
|
|
# The raspistill process started below must run "forever" in
|
|
# the background until killed when Home Assistant is stopped.
|
|
# Therefore it must not be wrapped with "with", since that
|
|
# waits for the subprocess to exit before continuing.
|
|
subprocess.Popen( # pylint: disable=consider-using-with
|
|
cmd_args,
|
|
stdout=subprocess.DEVNULL,
|
|
stderr=subprocess.STDOUT,
|
|
close_fds=False, # required for posix_spawn
|
|
)
|
|
|
|
def camera_image(
|
|
self, width: int | None = None, height: int | None = None
|
|
) -> bytes | None:
|
|
"""Return raspistill image response."""
|
|
with open(self._config[CONF_FILE_PATH], "rb") as file:
|
|
return file.read()
|
|
|
|
@property
|
|
def name(self):
|
|
"""Return the name of this camera."""
|
|
return self._name
|
|
|
|
@property
|
|
def frame_interval(self):
|
|
"""Return the interval between frames of the stream."""
|
|
return self._config[CONF_TIMELAPSE] / 1000
|