core/homeassistant/__main__.py

204 lines
6.1 KiB
Python
Raw Normal View History

"""Start Home Assistant."""
2021-03-17 16:34:55 +00:00
from __future__ import annotations
2016-02-19 05:27:50 +00:00
import argparse
2021-09-28 07:05:06 +00:00
import faulthandler
2016-02-19 05:27:50 +00:00
import os
import sys
import threading
from .const import REQUIRED_PYTHON_VER, RESTART_EXIT_CODE, __version__
2021-09-28 07:05:06 +00:00
FAULT_LOG_FILENAME = "home-assistant.log.fault"
2015-05-01 05:44:24 +00:00
def validate_os() -> None:
"""Validate that Home Assistant is running in a supported operating system."""
if not sys.platform.startswith(("darwin", "linux")):
print("Home Assistant only supports Linux, OSX and Windows using WSL")
sys.exit(1)
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 "
f"{REQUIRED_PYTHON_VER[0]}.{REQUIRED_PYTHON_VER[1]}.{REQUIRED_PYTHON_VER[2]}"
2019-07-31 19:25:30 +00:00
)
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."""
# pylint: disable=import-outside-toplevel
from . import 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(
f"Fatal Error: Specified configuration directory {config_dir} "
"does not exist"
2019-07-31 19:25:30 +00:00
)
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 "
f"directory {config_dir}"
2019-07-31 19:25:30 +00:00
)
sys.exit(1)
# Test if library directory exists
if not os.path.isdir(lib_dir):
try:
os.mkdir(lib_dir)
except OSError:
print(f"Fatal Error: Unable to create library directory {lib_dir}")
sys.exit(1)
2015-08-30 06:02:07 +00:00
def get_arguments() -> argparse.Namespace:
2016-03-07 23:06:04 +00:00
"""Get parsed passed in arguments."""
# pylint: disable=import-outside-toplevel
from . import config as config_util
2019-07-31 19:25:30 +00:00
2015-08-30 06:02:07 +00:00
parser = argparse.ArgumentParser(
description="Home Assistant: Observe, Control, Automate.",
epilog=f"If restart is requested, exits with code {RESTART_EXIT_CODE}",
2019-07-31 19:25:30 +00:00
)
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(
"--safe-mode", action="store_true", help="Start Home Assistant in safe mode"
2019-07-31 19:25:30 +00:00
)
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
"--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,
help="Log file to write to. If not set, CONFIG/home-assistant.log is used",
2019-07-31 19:25:30 +00:00
)
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
"--script", nargs=argparse.REMAINDER, help="Run one of the embedded scripts"
)
parser.add_argument(
"--ignore-os-check",
action="store_true",
help="Skips validation of operating system",
)
arguments = parser.parse_args()
return arguments
2021-03-17 16:34:55 +00:00
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, "-m", "homeassistant"] + list(sys.argv[1:])
return sys.argv
def check_threads() -> None:
"""Check if there are any lingering threads."""
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")
def main() -> int:
2016-03-07 23:06:04 +00:00
"""Start Home Assistant."""
validate_python()
args = get_arguments()
if not args.ignore_os_check:
validate_os()
if args.script is not None:
# pylint: disable=import-outside-toplevel
from . import scripts
2019-07-31 19:25:30 +00:00
return scripts.run(args.script)
config_dir = os.path.abspath(os.path.join(os.getcwd(), args.config))
2015-08-30 06:02:07 +00:00
ensure_config_path(config_dir)
# pylint: disable=import-outside-toplevel
from . import runner
runtime_conf = runner.RuntimeConfig(
config_dir=config_dir,
verbose=args.verbose,
log_rotate_days=args.log_rotate_days,
log_file=args.log_file,
log_no_color=args.log_no_color,
skip_pip=args.skip_pip,
safe_mode=args.safe_mode,
debug=args.debug,
open_ui=args.open_ui,
)
2021-09-28 07:05:06 +00:00
fault_file_name = os.path.join(config_dir, FAULT_LOG_FILENAME)
with open(fault_file_name, mode="a", encoding="utf8") as fault_file:
faulthandler.enable(fault_file)
exit_code = runner.run(runtime_conf)
faulthandler.disable()
if os.path.getsize(fault_file_name) == 0:
os.remove(fault_file_name)
check_threads()
2019-12-16 06:29:19 +00:00
return exit_code
if __name__ == "__main__":
sys.exit(main())