auto-ttv-grabber/main.py

161 lines
5.4 KiB
Python
Raw Normal View History

2024-04-02 20:51:44 -04:00
#!/usr/bin/env python3
2024-04-05 15:54:40 -04:00
import configparser
2024-04-02 21:17:14 -04:00
import os
2024-04-03 13:27:00 -04:00
import shutil
import subprocess
import sys
import time
2024-04-03 14:17:51 -04:00
from datetime import datetime
2024-04-05 15:11:59 -04:00
from pathlib import Path
2024-04-08 12:49:28 -04:00
from typing import List
2024-04-08 12:56:56 -04:00
2024-04-07 15:57:03 -04:00
import requests
2024-04-08 12:49:28 -04:00
channel_list: List[str] = []
2024-04-03 16:55:05 -04:00
downloading = {}
2024-04-05 15:54:40 -04:00
# Default Config Settings
2024-04-07 15:47:37 -04:00
streamlink_location: str = "streamlink"
download_location: str = f"{Path.home()}/Downloads/Streams"
skip_ads: bool = False
2024-04-10 16:44:57 -04:00
log: bool = False
2024-04-05 15:11:59 -04:00
2024-04-08 12:49:28 -04:00
def load_config() -> None:
2024-04-05 15:54:40 -04:00
print("Reading config file...")
2024-04-07 15:47:37 -04:00
config = configparser.ConfigParser()
config.read("config.ini")
if config.has_option("settings", "streamlink_location"):
2024-04-07 15:47:37 -04:00
streamlink_location = config["settings"]["streamlink_location"]
print(f"Streamlink location: {streamlink_location}")
if config.has_option("settings", "download_location"):
2024-04-07 15:47:37 -04:00
download_location = config["settings"]["download_location"]
print(f"Download location: {download_location}")
if config.has_option("settings", "skip_ads"):
2024-04-07 15:47:37 -04:00
skip_ads = bool(config["settings"]["skip_ads"])
print(f"Skip ads: {skip_ads}")
if config.has_option("settings", "log"):
2024-04-10 16:44:57 -04:00
log = bool(config["settings"]["log"])
print(f"Logs: {log}")
2024-04-07 15:47:37 -04:00
if len(config["streams"]) < 1:
sys.exit("ERROR: No streams found in config.ini! See README.md for more info.")
else:
for index in range(1, len(config["streams"]) + 1):
channel_list.append(config["streams"][str(index)])
2024-04-07 15:47:37 -04:00
print("Config file loaded")
2024-04-05 15:11:59 -04:00
2024-04-03 16:55:05 -04:00
2024-04-10 13:35:42 -04:00
# TODO: Have steamlink itself check if the channel is live
def is_live(channel: str) -> bool:
"""Checks if a channel is live on Twitch"""
try:
contents = requests.get("https://www.twitch.tv/" + channel).content.decode(
"utf-8"
)
if "isLiveBroadcast" in contents:
return True
else:
return False
except Exception:
print(f"There was an issue checking if {channel} was life. Will try next time!")
if channel in downloading:
return True
else:
return False
2024-04-08 12:49:28 -04:00
def download_stream(channel: str) -> None:
2024-04-03 18:00:04 -04:00
"""Downloads a given channel name in its own subprocess"""
# TODO: Just clean this up at somepoint
2024-04-12 12:53:39 -04:00
loging: str = ""
ad_skipping: str = ""
print(log)
2024-04-10 16:44:57 -04:00
if log:
2024-04-12 12:53:39 -04:00
loging = f"--loglevel info --logfile {download_location}/{channel}/log.txt"
if skip_ads:
2024-04-12 12:53:39 -04:00
ad_skipping = "--twitch-proxy-playlist=https://lb-eu.cdn-perfprod.com,https://lb-eu2.cdn-perfprod.com,https://lb-na.cdn-perfprod.com,https://lb-as.cdn-perfprod.com,https://as.luminous.dev --twitch-disable-ads"
2024-04-08 12:56:56 -04:00
file_name: str = f"{channel}_{datetime.now().strftime('%Y-%m-%d_%H-%M-%S')}.ts"
2024-04-12 12:53:39 -04:00
cmd: str = f"{streamlink_location} --retry-max 10 {loging} {ad_skipping} -o {download_location}/{channel}/{file_name} twitch.tv/{channel} best"
# TODO: Check if the process failed for some reason
downloading[channel] = subprocess.Popen(
[cmd],
shell=True,
start_new_session=True,
stdout=subprocess.PIPE,
stderr=subprocess.STDOUT,
)
2024-04-03 16:55:05 -04:00
2024-04-02 20:51:44 -04:00
2024-04-08 12:49:28 -04:00
def check_system() -> None:
2024-04-03 18:00:04 -04:00
"""Makes sure everything is place for the script to run"""
2024-04-07 15:47:37 -04:00
# Checks for config file
if not os.path.exists("config.ini"):
sys.exit("ERROR: config.ini is not found! See README.md for more info.")
# Checks if streamlink is in the systems path
if not shutil.which("streamlink"):
sys.exit("ERROR: streamlink is not found in the systems path!")
2024-04-02 20:51:44 -04:00
2024-04-05 15:54:40 -04:00
# Make sure the download location exists
if not os.path.exists(download_location):
print(
f"Download destination does not exist.\n Creating now at {download_location}"
)
os.makedirs(download_location)
2024-04-03 18:00:04 -04:00
2024-04-08 12:49:28 -04:00
def stop_downloads() -> None:
2024-04-03 18:00:04 -04:00
"""Goes through every process and stops it if running"""
print("\nCleaning up...")
for name, proc in downloading.items():
proc.terminate()
2024-04-05 15:54:40 -04:00
print(f"Stopping download of {name}")
2024-04-03 13:27:00 -04:00
2024-04-08 12:49:28 -04:00
def main() -> None:
2024-04-03 18:00:04 -04:00
"""Main entry point of the app"""
# Run untill progam is killed
2024-04-12 12:53:39 -04:00
# TODO: Check on the process if it is still alive and restart as needed
2024-04-03 18:00:04 -04:00
while True:
# Exits the program if there is no channels to grab
print("\n------------------------------------")
for channel in channel_list:
channel = channel.strip()
2024-04-10 13:35:42 -04:00
if is_live(channel):
print(f"\n\033[1m{channel}\033[0m is \033[32mlive\033[0m!", end=" ")
2024-04-03 18:00:04 -04:00
if channel not in downloading:
download_stream(channel)
else:
2024-04-10 13:35:42 -04:00
print("\033[33m(Already Downloading)\033[0m")
2024-04-02 20:51:44 -04:00
else:
2024-04-10 13:35:42 -04:00
print(
"\n\033[1m" + channel + "\033[0m is \033[31mnot live\033[0m.",
end=" ",
)
2024-04-03 18:00:04 -04:00
if channel in downloading:
del downloading[channel]
2024-04-10 13:35:42 -04:00
print("\033[35m(Stopping)\033[0m")
2024-04-03 18:00:04 -04:00
time.sleep(1) # Wait one second before going to next channel
print(
2024-04-10 13:35:42 -04:00
"\n\n\033[3mLast checked: "
+ datetime.now().strftime("%H:%M:%S")
+ "\033[0m"
2024-04-03 18:00:04 -04:00
)
print("------------------------------------")
time.sleep(60) # Wait 60 Seconds before trying again
if __name__ == "__main__":
2024-04-07 15:47:37 -04:00
"""This is executed when run from the command line"""
2024-04-03 18:00:04 -04:00
try:
check_system()
2024-04-07 15:47:37 -04:00
load_config()
2024-04-03 18:00:04 -04:00
main()
finally:
stop_downloads()