twitch-autosave/daemon.py

233 lines
8.3 KiB
Python
Raw Normal View History

2021-07-18 03:06:08 +03:00
#!/usr/bin/python3
2021-09-25 14:26:34 +03:00
# TODO: Перезапускать скрипт при обнаружении новой версии
2021-07-18 03:06:08 +03:00
import os
import sys
2021-07-18 03:06:08 +03:00
from threading import Thread
import configparser
import schedule
from twitchAPI.twitch import Twitch
import subprocess
import time
import logging
from logging.handlers import TimedRotatingFileHandler
2021-09-23 06:27:54 +03:00
log_format = logging.Formatter('%(asctime)s %(levelname)s - %(message)s')
log_file = 'output.log'
cfg_file = 'config.ini'
def set_config():
"""
Эта функция либо читает существующий конфиг, либо создает новый.
Возвращает объект конфига (configparser.ConfigParser())
"""
config = configparser.ConfigParser()
if not config.read('cfg_file.ini'):
config["app"] = {
"path": "",
"check_period": 5,
"max_files": 3
}
config["twitch"] = {
"app_id": "",
"app_secret": "",
"streamers": "asdf,qqqqq"
}
with open('cfg_file.ini', 'w') as cfg_file:
config.write(cfg_file)
return config
2021-07-18 03:06:08 +03:00
def which(command) -> bool:
2021-07-22 09:13:13 +03:00
# Пиздец, почему нет нормального аналога which из bash???
2022-11-13 11:31:48 +03:00
"""
Мой аналог which из bash'а, который отдает true или false
при наличии или отсутствии утилиты
2022-11-13 11:31:48 +03:00
"""
2021-07-22 09:13:13 +03:00
for dirs in os.get_exec_path():
if command in os.listdir(dirs):
# Если что-нибудь нашли, то True
return True
2021-07-23 02:20:32 +03:00
# Если ничего не нашли во всех дирах, то выходим с False
2021-07-22 09:13:13 +03:00
return False
def check_installed_tools() -> bool:
2022-11-13 11:31:48 +03:00
"""
Проверяет, установлены ли необходимые утилиты
2022-11-13 11:31:48 +03:00
"""
tools = ('youtube-dl', 'ffmpeg')
for tool in tools:
if not which(tool):
log.critical(tool + " не установлен")
return False
return True
def start_recording(streamer):
2022-11-13 11:31:48 +03:00
"""
2021-07-18 03:06:08 +03:00
Функция, которая запускает в отдельном потоке запись стрима - recorder(i)
2022-11-13 11:31:48 +03:00
"""
th = Thread(target=recorder, args=(streamer,))
2021-07-18 03:06:08 +03:00
th.start()
def recorder(streamer):
2022-11-13 11:31:48 +03:00
"""
2021-07-18 03:06:08 +03:00
Функция, которая запускает youtube-dl, фактически записывает стрим
2022-11-13 11:31:48 +03:00
"""
path = config['app']['path'] + "/" + streamer
log.info("Записываем стрим %s\n" % streamer)
# cmdline для запуска youtube-dl
cmdline = ["youtube-dl", "-q", "-o",
path + "/%(upload_date)s_%(title)s__%(timestamp)s_%(id)s.%(ext)s",
"https://twitch.tv/" + streamer]
subprocess.call(cmdline)
log.info("Запись стрима %s закончена\n" % streamer)
2022-11-13 11:31:48 +03:00
if os.path.exists(path + "/pid"):
os.remove(path + "/pid")
log.info("lock файл удален")
2021-07-18 03:06:08 +03:00
def check_stream():
# FIXME: Распилить на более мелкие функции
2022-11-13 11:31:48 +03:00
"""
2021-07-18 03:06:08 +03:00
1. Проверка на наличие стрима
1.1 Если нет - удалить lock файл, если он есть
1.2 Если есть - создать lock файл, запустить записывалку
2022-11-13 11:31:48 +03:00
"""
for streamer in config['twitch']['streamers'].split(','):
# Путь до диры со стримами
path = config['app']['path'] + "/" + streamer
# Получаем инфо о стримере, если не получается, выходим с ошибкой
# resolved_id = client.users.translate_usernames_to_ids(i)
resolved_id = twitch_client.get_users(logins=[streamer])
if not resolved_id['data']:
log.error(
"Аккаунт " + streamer + " не найден"
)
continue
# Создаем путь до диры со стримером, если папка не существует
if not (os.path.exists(path)):
os.makedirs(path)
log.info("Создана директория " + streamer)
# Достаем ID стримера из инфо
user_id = resolved_id['data'][0]['id']
user_stream = twitch_client.get_streams(user_id=user_id)
# Если стрим идет, то идем дальше
if user_stream['data']:
# Если стрим идет и лок файла нет, то записываем и ставим лок
if (user_stream['data'][0]['type'] == 'live') and not (
os.path.exists(config['app']['path'] + "/" + streamer + "/pid")):
log.info(streamer + " стримит")
start_recording(streamer)
open(config['app']['path'] + "/pid", 'w').close
else:
2021-09-23 06:27:54 +03:00
log.info(
"Идет запись " + streamer
2021-09-23 06:27:54 +03:00
)
2021-07-18 03:06:08 +03:00
else:
# Если стрим не идет, то пишем об этом и убираем его из залоченных
log.info(streamer + " Не стримит")
2021-07-18 22:16:03 +03:00
# Если есть лок, то удаляем
2022-11-13 11:31:48 +03:00
if os.path.exists(path + "/pid"):
os.remove(path + "/pid")
2021-07-18 03:06:08 +03:00
def remove_old_streams():
# https://clck.ru/WHh32
records_path = config['app']['path']
# По каждой папке со стримерами
for streamer in config['twitch']['streamers']:
try:
os.chdir(records_path + "/" + streamer)
# Если файлов в папке со стримами больше чем указано в конфиге
if len(os.listdir(records_path + "/" + streamer)) > int(config['app']['max_files']):
# Получаем список файлов
# и смотрим, превышает ли кол-во mp4 файлов заданное в конфиге
# Если превышает - удаляем старейший
oldest = min(os.listdir(records_path + "/" + streamer),
key=os.path.getctime)
os.unlink(oldest)
log.warning("Удален файл: " + oldest)
except Exception as e:
log.error(e)
def get_console_handler():
console_handler = logging.StreamHandler(sys.stdout)
console_handler.setFormatter(log_format)
return console_handler
def get_file_handler():
file_handler = TimedRotatingFileHandler(log_file, when='midnight')
file_handler.setFormatter(log_format)
return file_handler
class CustomFormatter(logging.Formatter):
grey = "\x1b[38;20m"
yellow = "\x1b[33;20m"
red = "\x1b[31;20m"
bold_red = "\x1b[31;1m"
reset = "\x1b[0m"
format = "%(asctime)s %(levelname)s - %(message)s"
FORMATS = {
logging.DEBUG: grey + format + reset,
logging.INFO: grey + format + reset,
logging.WARNING: yellow + format + reset,
logging.ERROR: red + format + reset,
logging.CRITICAL: bold_red + format + reset
}
def format(self, record):
log_fmt = self.FORMATS.get(record.levelno)
formatter = logging.Formatter(log_fmt)
return formatter.format(record)
def get_logger(logger_name):
"""
Инициализация лога
"""
logger = logging.getLogger(logger_name)
logger.setLevel(logging.DEBUG)
# Console logging
console = get_console_handler()
console.setFormatter(CustomFormatter())
logger.addHandler(console)
logger.addHandler(get_file_handler())
logger.propagate = False
return logger
2021-07-18 03:06:08 +03:00
if __name__ == "__main__":
# Проверить, установлены ли нужные утилиты
if not check_installed_tools():
exit()
# Set config
config = set_config()
# Log config
log = get_logger("main")
log.info("Запущен")
# Проверять стримы раз в check_period
schedule.every(int(config['app']['check_period'])).seconds.do(check_stream)
# Каждый час удалять старые стримы
schedule.every(1).hours.do(remove_old_streams)
twitch_client = Twitch(config['twitch']['app_id'], config['twitch']['app_secret'])
while True:
schedule.run_pending()
time.sleep(1)