bazarr/bazarr.py

182 lines
5.5 KiB
Python
Raw Normal View History

# coding=utf-8
2019-09-28 16:38:26 +00:00
import bazarr.libs
import os
import signal
2020-01-31 22:04:01 +00:00
import subprocess
import sys
import time
from bazarr.get_args import args
2019-09-05 15:30:14 +00:00
def check_python_version():
2020-01-30 15:14:51 +00:00
python_version = (sys.version_info.major, sys.version_info.minor, sys.version_info.micro)
python_version_str = '.'.join(map(str, python_version))
minimum_python3_version = (3, 6, 0)
minimum_python3_version_str = '.'.join(map(str, minimum_python3_version))
if python_version >= minimum_python3_version:
pass
else:
print("Python " + minimum_python3_version_str + " or greater required. Current version is " +
python_version_str + ". Please upgrade Python.")
2019-09-05 15:30:14 +00:00
os._exit(0)
check_python_version()
2018-10-11 01:23:30 +00:00
dir_name = os.path.dirname(__file__)
class ProcessRegistry:
def register(self, process):
pass
def unregister(self, process):
pass
class DaemonStatus(ProcessRegistry):
def __init__(self):
self.__should_stop = False
self.__processes = set()
def register(self, process):
self.__processes.add(process)
def unregister(self, process):
self.__processes.remove(process)
@staticmethod
def __wait_for_processes(processes, timeout):
2020-01-31 22:04:01 +00:00
"""
Waits all the provided processes for the specified amount of time in seconds.
"""
reference_ts = time.time()
elapsed = 0
remaining_processes = list(processes)
while elapsed < timeout and len(remaining_processes) > 0:
remaining_time = timeout - elapsed
for ep in list(remaining_processes):
if ep.poll() is not None:
remaining_processes.remove(ep)
else:
if remaining_time > 0:
2020-02-01 03:24:47 +00:00
try:
ep.wait(remaining_time)
remaining_processes.remove(ep)
except subprocess.TimeoutExpired:
pass
elapsed = time.time() - reference_ts
remaining_time = timeout - elapsed
return remaining_processes
@staticmethod
def __send_signal(processes, signal_no, live_processes=None):
2020-01-31 22:04:01 +00:00
"""
Sends to every single of the specified processes the given signal and (if live_processes is not None) append to
it processes which are still alive.
"""
for ep in processes:
if ep.poll() is None:
if live_processes is not None:
live_processes.append(ep)
try:
ep.send_signal(signal_no)
except Exception as e:
2020-01-31 22:04:01 +00:00
print('Failed sending signal %s to process %s because of an unexpected error: %s' % (
signal_no, ep.pid, e))
return live_processes
def stop(self):
2020-01-31 22:04:01 +00:00
"""
Flags this instance as should stop and terminates as smoothly as possible children processes.
"""
self.__should_stop = True
live_processes = DaemonStatus.__send_signal(self.__processes, signal.SIGINT, list())
live_processes = DaemonStatus.__wait_for_processes(live_processes, 120)
DaemonStatus.__send_signal(live_processes, signal.SIGTERM)
def should_stop(self):
return self.__should_stop
def start_bazarr(process_registry=ProcessRegistry()):
script = [sys.executable, "-u", os.path.normcase(os.path.join(dir_name, 'bazarr', 'main.py'))] + sys.argv[1:]
2020-01-31 22:04:01 +00:00
ep = subprocess.Popen(script, stdout=subprocess.PIPE, stderr=subprocess.STDOUT, stdin=subprocess.PIPE)
process_registry.register(ep)
2019-09-13 19:12:26 +00:00
print("Bazarr starting...")
try:
2019-12-07 03:40:40 +00:00
while True:
line = ep.stdout.readline()
if line == '' or not line:
# Process ended so let's unregister it
process_registry.unregister(ep)
2019-12-07 03:40:40 +00:00
break
2020-01-30 15:14:51 +00:00
sys.stdout.buffer.write(line)
sys.stdout.flush()
except KeyboardInterrupt:
pass
if __name__ == '__main__':
restartfile = os.path.normcase(os.path.join(args.config_dir, 'bazarr.restart'))
stopfile = os.path.normcase(os.path.join(args.config_dir, 'bazarr.stop'))
2020-01-31 22:04:01 +00:00
try:
os.remove(restartfile)
2020-01-31 22:04:01 +00:00
except Exception:
2018-10-11 01:23:30 +00:00
pass
2020-01-31 22:04:01 +00:00
2018-10-11 01:23:30 +00:00
try:
os.remove(stopfile)
2020-01-31 22:04:01 +00:00
except Exception:
pass
2020-01-31 22:04:01 +00:00
def daemon(bazarr_runner=lambda: start_bazarr()):
if os.path.exists(stopfile):
try:
os.remove(stopfile)
2020-01-31 22:04:01 +00:00
except Exception:
2019-09-13 19:12:26 +00:00
print('Unable to delete stop file.')
else:
2019-09-13 19:12:26 +00:00
print('Bazarr exited.')
2020-01-31 22:04:01 +00:00
sys.exit(0)
if os.path.exists(restartfile):
try:
os.remove(restartfile)
2020-01-31 22:04:01 +00:00
except Exception:
2019-09-13 19:12:26 +00:00
print('Unable to delete restart file.')
else:
bazarr_runner()
2020-01-31 22:04:01 +00:00
bazarr_runner = lambda: start_bazarr()
2020-01-31 22:04:01 +00:00
should_stop = lambda: False
2020-01-30 15:36:33 +00:00
daemonStatus = DaemonStatus()
2020-01-30 15:36:33 +00:00
def shutdown():
# indicates that everything should stop
daemonStatus.stop()
# emulate a Ctrl C command on itself (bypasses the signal thing but, then, emulates the "Ctrl+C break")
os.kill(os.getpid(), signal.SIGINT)
2020-01-30 15:36:33 +00:00
signal.signal(signal.SIGTERM, lambda signal_no, frame: shutdown())
2020-01-30 15:36:33 +00:00
should_stop = lambda: daemonStatus.should_stop()
bazarr_runner = lambda: start_bazarr(daemonStatus)
bazarr_runner()
# Keep the script running forever until stop is requested through term or keyboard interrupt
while not should_stop():
daemon(bazarr_runner)
2019-05-31 23:00:32 +00:00
time.sleep(1)