core/homeassistant/__main__.py

407 lines
12 KiB
Python
Raw Normal View History

"""Start Home Assistant."""
2016-02-19 05:27:50 +00:00
import argparse
import os
import platform
import subprocess
import sys
import threading
from typing import List, Dict, Any, TYPE_CHECKING
from homeassistant import monkey_patch
from homeassistant.const import __version__, REQUIRED_PYTHON_VER, RESTART_EXIT_CODE
if TYPE_CHECKING:
from homeassistant import core
2015-05-01 05:44:24 +00:00
def set_loop() -> None:
"""Attempt to use uvloop."""
import asyncio
from asyncio.events import BaseDefaultEventLoopPolicy
policy = None
2019-07-31 19:25:30 +00:00
if sys.platform == "win32":
if hasattr(asyncio, "WindowsProactorEventLoopPolicy"):
2018-10-02 08:35:00 +00:00
# pylint: disable=no-member
policy = asyncio.WindowsProactorEventLoopPolicy()
else:
2019-07-31 19:25:30 +00:00
class ProactorPolicy(BaseDefaultEventLoopPolicy):
"""Event loop policy to create proactor loops."""
_loop_factory = asyncio.ProactorEventLoop
policy = ProactorPolicy()
else:
try:
import uvloop
except ImportError:
pass
else:
policy = uvloop.EventLoopPolicy()
if policy is not None:
asyncio.set_event_loop_policy(policy)
def validate_python() -> None:
"""Validate that the right Python version is running."""
if sys.version_info[:3] < REQUIRED_PYTHON_VER:
2019-07-31 19:25:30 +00:00
print(
"Home Assistant requires at least Python {}.{}.{}".format(
*REQUIRED_PYTHON_VER
)
)
2015-08-31 15:53:59 +00:00
sys.exit(1)
def ensure_config_path(config_dir: str) -> None:
2016-03-07 23:06:04 +00:00
"""Validate the configuration directory."""
2016-03-27 15:44:15 +00:00
import homeassistant.config as config_util
2019-07-31 19:25:30 +00:00
lib_dir = os.path.join(config_dir, "deps")
# Test if configuration directory exists
2014-11-23 20:57:29 +00:00
if not os.path.isdir(config_dir):
if config_dir != config_util.get_default_config_dir():
2019-07-31 19:25:30 +00:00
print(
(
"Fatal Error: Specified configuration directory does "
"not exist {} "
).format(config_dir)
)
sys.exit(1)
try:
os.mkdir(config_dir)
except OSError:
2019-07-31 19:25:30 +00:00
print(
(
"Fatal Error: Unable to create default configuration "
"directory {} "
).format(config_dir)
)
sys.exit(1)
# Test if library directory exists
if not os.path.isdir(lib_dir):
try:
os.mkdir(lib_dir)
except OSError:
2019-07-31 19:25:30 +00:00
print(
("Fatal Error: Unable to create library " "directory {} ").format(
lib_dir
)
)
sys.exit(1)
2015-08-30 06:02:07 +00:00
2019-07-31 19:25:30 +00:00
async def ensure_config_file(hass: "core.HomeAssistant", config_dir: str) -> str:
2016-03-07 23:06:04 +00:00
"""Ensure configuration file exists."""
2016-03-27 15:44:15 +00:00
import homeassistant.config as config_util
2019-07-31 19:25:30 +00:00
config_path = await config_util.async_ensure_config_exists(hass, config_dir)
if config_path is None:
2019-07-31 19:25:30 +00:00
print("Error getting configuration path")
sys.exit(1)
2014-11-08 19:01:47 +00:00
return config_path
def get_arguments() -> argparse.Namespace:
2016-03-07 23:06:04 +00:00
"""Get parsed passed in arguments."""
2016-03-27 15:44:15 +00:00
import homeassistant.config as config_util
2019-07-31 19:25:30 +00:00
2015-08-30 06:02:07 +00:00
parser = argparse.ArgumentParser(
2019-07-31 19:25:30 +00:00
description="Home Assistant: Observe, Control, Automate."
)
parser.add_argument("--version", action="version", version=__version__)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"-c",
"--config",
metavar="path_to_config_dir",
default=config_util.get_default_config_dir(),
2019-07-31 19:25:30 +00:00
help="Directory that contains the Home Assistant configuration",
)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--demo-mode", action="store_true", help="Start Home Assistant in demo mode"
)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--debug", action="store_true", help="Start Home Assistant in debug mode"
)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--open-ui", action="store_true", help="Open the webinterface in a browser"
)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--skip-pip",
action="store_true",
help="Skips pip install of required packages on startup",
)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"-v", "--verbose", action="store_true", help="Enable verbose logging to file."
)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--pid-file",
metavar="path_to_pid_file",
default=None,
2019-07-31 19:25:30 +00:00
help="Path to PID file useful for running as daemon",
)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--log-rotate-days",
type=int,
default=None,
2019-07-31 19:25:30 +00:00
help="Enables daily log rotation and keeps up to the specified days",
)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--log-file",
type=str,
default=None,
2019-07-31 19:25:30 +00:00
help="Log file to write to. If not set, CONFIG/home-assistant.log " "is used",
)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--log-no-color", action="store_true", help="Disable color logs"
)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--runner",
action="store_true",
help=f"On restart exit with code {RESTART_EXIT_CODE}",
2019-07-31 19:25:30 +00:00
)
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--script", nargs=argparse.REMAINDER, help="Run one of the embedded scripts"
)
if os.name == "posix":
parser.add_argument(
2019-07-31 19:25:30 +00:00
"--daemon", action="store_true", help="Run Home Assistant as daemon"
)
arguments = parser.parse_args()
if os.name != "posix" or arguments.debug or arguments.runner:
2019-07-31 19:25:30 +00:00
setattr(arguments, "daemon", False)
return arguments
def daemonize() -> None:
2016-03-07 23:06:04 +00:00
"""Move current process to daemon process."""
# Create first fork
pid = os.fork()
if pid > 0:
sys.exit(0)
2016-03-07 23:06:04 +00:00
# Decouple fork
os.setsid()
2016-03-07 23:06:04 +00:00
# Create second fork
pid = os.fork()
if pid > 0:
sys.exit(0)
# redirect standard file descriptors to devnull
2019-07-31 19:25:30 +00:00
infd = open(os.devnull, "r")
outfd = open(os.devnull, "a+")
sys.stdout.flush()
sys.stderr.flush()
os.dup2(infd.fileno(), sys.stdin.fileno())
os.dup2(outfd.fileno(), sys.stdout.fileno())
os.dup2(outfd.fileno(), sys.stderr.fileno())
def check_pid(pid_file: str) -> None:
"""Check that Home Assistant is not already running."""
2016-03-07 23:06:04 +00:00
# Check pid file
try:
2019-07-31 19:25:30 +00:00
with open(pid_file, "r") as file:
pid = int(file.readline())
2019-09-04 17:09:24 +00:00
except OSError:
2015-09-01 07:22:43 +00:00
# PID File does not exist
return
# If we just restarted, we just found our own pidfile.
if pid == os.getpid():
return
try:
os.kill(pid, 0)
except OSError:
# PID does not exist
return
2019-07-31 19:25:30 +00:00
print("Fatal Error: HomeAssistant is already running.")
sys.exit(1)
def write_pid(pid_file: str) -> None:
2016-03-07 23:06:04 +00:00
"""Create a PID File."""
pid = os.getpid()
try:
2019-07-31 19:25:30 +00:00
with open(pid_file, "w") as file:
file.write(str(pid))
2019-09-04 17:09:24 +00:00
except OSError:
print(f"Fatal Error: Unable to write pid file {pid_file}")
sys.exit(1)
2015-09-15 06:17:01 +00:00
def closefds_osx(min_fd: int, max_fd: int) -> None:
"""Make sure file descriptors get closed when we restart.
2016-03-07 23:06:04 +00:00
We cannot call close on guarded fds, and we cannot easily test which fds
are guarded. But we can set the close-on-exec flag on everything we want to
get rid of.
"""
from fcntl import fcntl, F_GETFD, F_SETFD, FD_CLOEXEC
for _fd in range(min_fd, max_fd):
try:
val = fcntl(_fd, F_GETFD)
if not val & FD_CLOEXEC:
fcntl(_fd, F_SETFD, val | FD_CLOEXEC)
2019-09-04 17:09:24 +00:00
except OSError:
pass
def cmdline() -> List[str]:
"""Collect path and arguments to re-execute the current hass instance."""
2019-07-31 19:25:30 +00:00
if os.path.basename(sys.argv[0]) == "__main__.py":
modulepath = os.path.dirname(sys.argv[0])
2019-07-31 19:25:30 +00:00
os.environ["PYTHONPATH"] = os.path.dirname(modulepath)
return [sys.executable] + [arg for arg in sys.argv if arg != "--daemon"]
2019-07-31 19:25:30 +00:00
return [arg for arg in sys.argv if arg != "--daemon"]
2019-07-31 19:25:30 +00:00
async def setup_and_run_hass(config_dir: str, args: argparse.Namespace) -> int:
"""Set up HASS and run."""
from homeassistant import bootstrap, core
2016-03-27 15:44:15 +00:00
hass = core.HomeAssistant()
if args.demo_mode:
config: Dict[str, Any] = {"frontend": {}, "demo": {}}
bootstrap.async_from_config_dict(
2019-07-31 19:25:30 +00:00
config,
hass,
config_dir=config_dir,
verbose=args.verbose,
skip_pip=args.skip_pip,
log_rotate_days=args.log_rotate_days,
log_file=args.log_file,
log_no_color=args.log_no_color,
)
else:
config_file = await ensure_config_file(hass, config_dir)
2019-07-31 19:25:30 +00:00
print("Config directory:", config_dir)
await bootstrap.async_from_config_file(
2019-07-31 19:25:30 +00:00
config_file,
hass,
verbose=args.verbose,
skip_pip=args.skip_pip,
log_rotate_days=args.log_rotate_days,
log_file=args.log_file,
log_no_color=args.log_no_color,
)
if args.open_ui and hass.config.api is not None:
import webbrowser
hass.add_job(webbrowser.open, hass.config.api.base_url)
return await hass.async_run()
def try_to_restart() -> None:
"""Attempt to clean up state and start a new Home Assistant instance."""
# Things should be mostly shut down already at this point, now just try
# to clean up things that may have been left behind.
2019-07-31 19:25:30 +00:00
sys.stderr.write("Home Assistant attempting to restart.\n")
# Count remaining threads, ideally there should only be one non-daemonized
# thread left (which is us). Nothing we really do with it, but it might be
# useful when debugging shutdown/restart issues.
try:
2019-07-31 19:25:30 +00:00
nthreads = sum(
thread.is_alive() and not thread.daemon for thread in threading.enumerate()
)
if nthreads > 1:
sys.stderr.write(f"Found {nthreads} non-daemonic threads.\n")
# Somehow we sometimes seem to trigger an assertion in the python threading
# module. It seems we find threads that have no associated OS level thread
# which are not marked as stopped at the python level.
except AssertionError:
sys.stderr.write("Failed to count non-daemonic threads.\n")
# Try to not leave behind open filedescriptors with the emphasis on try.
try:
max_fd = os.sysconf("SC_OPEN_MAX")
except ValueError:
max_fd = 256
2019-07-31 19:25:30 +00:00
if platform.system() == "Darwin":
closefds_osx(3, max_fd)
else:
os.closerange(3, max_fd)
# Now launch into a new instance of Home Assistant. If this fails we
# fall through and exit with error 100 (RESTART_EXIT_CODE) in which case
# systemd will restart us when RestartForceExitStatus=100 is set in the
# systemd.service file.
sys.stderr.write("Restarting Home Assistant\n")
args = cmdline()
os.execv(args[0], args)
def main() -> int:
2016-03-07 23:06:04 +00:00
"""Start Home Assistant."""
validate_python()
monkey_patch_needed = sys.version_info[:3] < (3, 6, 3)
2019-07-31 19:25:30 +00:00
if monkey_patch_needed and os.environ.get("HASS_NO_MONKEY") != "1":
monkey_patch.disable_c_asyncio()
monkey_patch.patch_weakref_tasks()
set_loop()
# Run a simple daemon runner process on Windows to handle restarts
2019-07-31 19:25:30 +00:00
if os.name == "nt" and "--runner" not in sys.argv:
nt_args = cmdline() + ["--runner"]
while True:
try:
subprocess.check_call(nt_args)
sys.exit(0)
except KeyboardInterrupt:
sys.exit(0)
except subprocess.CalledProcessError as exc:
if exc.returncode != RESTART_EXIT_CODE:
sys.exit(exc.returncode)
args = get_arguments()
if args.script is not None:
from homeassistant import scripts
2019-07-31 19:25:30 +00:00
return scripts.run(args.script)
config_dir = os.path.join(os.getcwd(), args.config)
2015-08-30 06:02:07 +00:00
ensure_config_path(config_dir)
2016-03-07 23:06:04 +00:00
# Daemon functions
if args.pid_file:
check_pid(args.pid_file)
if args.daemon:
daemonize()
if args.pid_file:
write_pid(args.pid_file)
from homeassistant.util.async_ import asyncio_run
2019-07-31 19:25:30 +00:00
exit_code = asyncio_run(setup_and_run_hass(config_dir, args))
if exit_code == RESTART_EXIT_CODE and not args.runner:
try_to_restart()
return exit_code # type: ignore
if __name__ == "__main__":
sys.exit(main())