-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdoctor-mike.py
executable file
·235 lines (193 loc) · 9.77 KB
/
doctor-mike.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
#!/opt/projects/mytube/venv/bin/python3
# Import Modules
import os
import re
import shutil
import socket
import subprocess
import sys
import pytubefix
import pytubefix.helpers
from libs.functions import ProofOfLife, CheckHistory, CheckProcess, FileName, LogIt, NotifyMe, WriteHistory, PlexLibraryUpdate
####[ REQUIRED VARIABLES ]####
LOGGER = str('doctormike')
OUTPUT_PATH = str(pytubefix.helpers.target_directory('/opt/media/tv.docs/Doctor Mike (2017) {tvdb-359268}'))
SERIES_PREFIX = str('Doctor Mike (2017) - ')
YOUTUBE_URL = str('https://www.youtube.com/@DoctorMike/videos')
SECTION_ID = str('6')
SERIES_URL = str('http://plex.int.snyderfamily.co:32400/web/index.html#!/server/50d6b668401e93d23054d59158dfff33bc988de4/details?key=%2Flibrary%2Fmetadata%2F48985&context=source%3Acontent.library~0~0')
PLAYLIST = False
CHANNEL = True
INITIALIZE = False
####[ REQUIRED VARIABLES ]####
if not ProofOfLife:
LogIt(LOGGER, "Plex Server is offline, so don't add new media to its libraries.", "error")
sys.exit(1)
# Get the hostname, for later
THISBOX = socket.gethostname()
# Define a lockfile, so we can increase
# the run scheduled without running over ourselves
pidfile = f"/tmp/{LOGGER}.lock"
# If the PID File exists, check to see if the contained PID is actually running
# If its not, delete the file.
if os.path.exists(pidfile):
if not CheckProcess(pidfile):
try:
subprocess.run(['rm', '-f', pidfile], check=True)
except subprocess.CalledProcessError:
pass
else:
sys.exit(1) # Reaching this step means the PID exists and it is currently running. Exit.
# Create the lock file
with open(pidfile, "w") as f:
f.write(str(os.getpid()))
def main():
# Validate the conditions before proceeding
if PLAYLIST == CHANNEL: # This checks both if they are both True or both False
raise ValueError("Either PLAYLIST or CHANNEL must be True, but not both.")
if PLAYLIST:
# Create a playlist object
x = pytubefix.Playlist(YOUTUBE_URL)
elif CHANNEL:
# Create a channel object
x = pytubefix.Channel(YOUTUBE_URL)
# Iterate through the playlist
for index, VID in enumerate(x.videos, start=1):
if not INITIALIZE:
# Limit to the first 10 items in the playlist
if index == 11:
# Report that we've reached the limit (minus 1, because we're halting before processing the 11th.)
LogIt(LOGGER, f"Reached the index limit ({index - 1} playlist items).")
break
else:
LogIt(LOGGER, f"Working on video {index} of {len(x.video_urls)}")
VIDEO = VID.watch_url
# Build the YouTube Object
yt = pytubefix.YouTube(str(VIDEO), use_oauth=True, allow_oauth_cache=True)
# Set the temporary directory
TEMP_DIR = str(pytubefix.helpers.target_directory(f'/opt/projects/mytube/downloads/{LOGGER}'))
# Set the video ID, title, and publish date
ID = str(yt.video_id)
TITLE = str(yt.title).strip()
PUBLISH_DATE = (yt.publish_date).strftime("%Y-%m-%d")
HISTORY_PATH = str(pytubefix.helpers.target_directory('/opt/projects/mytube/history'))
OUTPUT_FILENAME = FileName(f"{SERIES_PREFIX}", f"{PUBLISH_DATE}", f"{TITLE}")
HISTORY_LOG = str(f"{HISTORY_PATH}/{LOGGER}_history.txt")
LENGTH = int(yt.length // 60)
THUMBNAIL_URL = yt.thumbnail_url
DESCRIPTION = f"{yt.description}\n\n\n\nVideo URL: {VIDEO}\nThumbnail URL: {THUMBNAIL_URL}"
# Check if the history file exists, and if not, create it
if not os.path.exists(HISTORY_LOG):
with open(HISTORY_LOG, "w") as f:
f.write(f"### {LOGGER} history log ###\n")
TEMP_OUTPUT = f"{TEMP_DIR}/{OUTPUT_FILENAME}"
FINAL_OUTPUT = f"{OUTPUT_PATH}/{OUTPUT_FILENAME}"
if os.path.exists(FINAL_OUTPUT):
LogIt(LOGGER, f"\"{FINAL_OUTPUT}\" already exists!", "warn")
if (not(CheckHistory(HISTORY_LOG, VIDEO))):
WriteHistory(HISTORY_LOG, VIDEO)
PlexLibraryUpdate(SECTION_ID, SERIES_URL, FINAL_OUTPUT, THUMBNAIL_URL, LOGGER, DESCRIPTION)
continue
else:
PlexLibraryUpdate(SECTION_ID, SERIES_URL, FINAL_OUTPUT, THUMBNAIL_URL, LOGGER, DESCRIPTION)
continue
# # Only capture videos from a specific date range
# if not(yt.publish_date.year >= 2024):
# LogIt(LOGGER, f"{index} of {len(x.video_urls)}: '{yt.title}' ({yt.video_id}) was published before 2024, and will be disgarded.")
# WriteHistory(HISTORY_LOG, VIDEO)
# continue
pattern = r'(Reacts To|Compilation)'
if re.search(pattern, TITLE, re.IGNORECASE):
LogIt(LOGGER, f"Episode \"{TITLE}\" ({ID}) is not a desired episode!")
if not CheckHistory(HISTORY_LOG, VIDEO):
WriteHistory(HISTORY_LOG, VIDEO)
continue
else:
continue
# Only interested in long-form interviews
if LENGTH < 50:
LogIt(LOGGER, f"Episode \"{TITLE}\" ({ID}) is too short ({LENGTH} minutes!)")
if (not(CheckHistory(HISTORY_LOG, VIDEO))):
WriteHistory(HISTORY_LOG, VIDEO)
continue
else:
continue
# Video is NOT in the history file
if (not(CheckHistory(HISTORY_LOG, VIDEO))):
LogIt(LOGGER, f"{index} of {len(x.video_urls)}: \"{TITLE}\" ({ID}) was NOT in history, and will be downloaded.")
# Download the audio stream, try 160kbps, if that fails, try 128kbps. If that fails, skip it.
try:
input_audio = yt.streams.filter(adaptive=True, mime_type="audio/webm", abr="160kbps").first().download(f"{TEMP_DIR}",f"{PUBLISH_DATE}.audio.webm")
except AttributeError:
try:
input_audio = yt.streams.filter(adaptive=True, mime_type="audio/webm", abr="128kbps").first().download(f"{TEMP_DIR}",f"{PUBLISH_DATE}.audio.webm")
except Exception:
LogIt(LOGGER, f"There was an error downloading the audio stream for \"{TITLE}\" ({ID})", "error")
WriteHistory(HISTORY_LOG, VIDEO)
continue
# Download the video stream, try 1080p, if that fails, try 720p. If that fails, skip it.
try:
input_video = yt.streams.filter(adaptive=True, mime_type="video/webm",res="1080p").first().download(f"{TEMP_DIR}", f"{PUBLISH_DATE}.video.webm")
except AttributeError:
try:
input_video = yt.streams.filter(adaptive=True, mime_type="video/webm",res="720p").first().download(f"{TEMP_DIR}", f"{PUBLISH_DATE}.video.webm")
except Exception:
LogIt(LOGGER, f"There was an error downloading the video stream for \"{TITLE}\" ({ID})", "error")
if os.path.exists(input_audio):
os.remove(input_audio)
WriteHistory(HISTORY_LOG, VIDEO)
continue
# Check to make sure the audio and video files exist
if not (os.path.exists(input_audio) or (os.path.exists(input_video))):
LogIt(LOGGER, f"Required media does not exist.", "warn")
continue
# Command to mux video and audio will differ, depending on the hostname
command = [
"/usr/bin/ffmpeg",
'-hwaccel', 'cuda',
"-i", input_audio,
"-i", input_video,
'-c:v', 'hevc_nvenc',
'-preset', 'medium',
'-c:a', 'aac',
'-strict', 'experimental',
TEMP_OUTPUT
]
# If an entry for the video ID does not yet exist in the history file, then download it.
OUTPUT = subprocess.run(command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, check=True, text=True)
# If the FFMPEG command was successful, log the success and send a notification
if OUTPUT.returncode == 0:
# Log the success
LogIt(LOGGER, f"Downloaded \"{TITLE}\" for {LOGGER}")
# Update the history file
WriteHistory(HISTORY_LOG, VIDEO)
# Clean up our mess
os.remove(input_audio)
os.remove(input_video)
shutil.move(TEMP_OUTPUT, FINAL_OUTPUT)
if os.path.exists(f"{TEMP_OUTPUT}"):
os.remove(f"{TEMP_OUTPUT}")
os.chown(f"{FINAL_OUTPUT}", 3000, 3000)
os.chmod(f"{FINAL_OUTPUT}", 0o2775)
# Update the Plex Library
PlexLibraryUpdate(SECTION_ID, SERIES_URL, FINAL_OUTPUT, THUMBNAIL_URL, LOGGER, DESCRIPTION)
# Send an NTFY notification
NotifyMe('New Episode!','2','dolphin',f"Downloaded {TITLE} for {LOGGER}")
else:
# Log the error output of the FFMPEG command
LogIt(LOGGER, OUTPUT.stderr, "error")
# Send an NTFY notification
NotifyMe('Error!','5','face_with_spiral_eyes','There was an error in the FFMPEG')
# Clean up our mess
os.remove(input_audio)
os.remove(input_video)
os.remove(TEMP_OUTPUT)
# Remove the lock file when the script finishes
sys.exit(1)
else: # Video IS in the history file
LogIt(LOGGER, f"{index} of {len(x.video_urls)}: \"{TITLE}\" ({ID}) WAS in history, and will be disgarded.")
continue
# Remove the lock file when the script finishes
os.remove(pidfile)
if __name__ == '__main__':
main()