forked from ccjensen/PlexMediaTagger
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathVideoItemProcessor.py
executable file
·429 lines (382 loc) · 20.6 KB
/
VideoItemProcessor.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
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
#!/usr/bin/env python
#encoding:utf-8
#author:ccjensen/Chris
#project:PlexMediaTagger
#repository:http://github.com/ccjensen/plexmediatagger
#license:Creative Commons GNU GPL v2
# (http://creativecommons.org/licenses/GPL/2.0/)
import logging
import sys
import os
import glob
import subprocess
from Summary import *
from LibraryStatistics import *
from DataTokens import *
from unicodedata import normalize
class VideoItemProcessor:
"""docstring for VideoItemProcessor"""
def __init__(self, opts, video_item):
self.opts = opts
self.video_item = video_item
self.media_items = self.video_item.media_items
#end def __init__
def canTag(self, part_item):
if part_item.file_type == '.m4v' or part_item.file_type == '.mp4':
return True
else:
logging.info("PlexMediaTagger cannot tag '%s' files" % part_item.file_type)
return False
#end if
#end def canTag
def shouldTag(self, part_item):
if self.opts.force:
return True
shouldTag = True
comment_tag_contents = self.getFileCommentTagContents(part_item)
if DataTokens.itunes_tag_data_token in comment_tag_contents:
logging.info("File previously tagged")
if self.opts.tag_update:
itunes_tag_data = comment_tag_contents.replace(DataTokens.itunes_tag_data_token, '')
for data_entry in itunes_tag_data.split(DataTokens.tag_data_delimiter):
if len(data_entry):
data = data_entry.split(DataTokens.token_delimiter)
if len(data) == 2:
token = data[0]
if token == DataTokens.updated_at_token:
tag_data_updated_at = data[1]
metadata_updated_at = self.video_item.updated_at
logging.debug("Date tagged: file:%s vs. pms:%s" % (tag_data_updated_at, metadata_updated_at))
if (metadata_updated_at > tag_data_updated_at):
logging.info("Metadata has changed since last time")
shouldTag = True
else:
logging.info("No change to the metadata since last time")
shouldTag = False
#end if metadata_updated_at
elif token == DataTokens.itunes_playcount_token:
#store the playcount in case we write out the tags
logging.debug("Playcount tagged: file:%s vs. pms:%s" % (data[1], self.video_item.view_count))
try:
if int(data[1]) > int(self.video_item.view_count):
logging.info("Using playcount from embedded metadata")
self.video_item.view_count = data[1]
except:
continue
elif token == DataTokens.itunes_rating_token:
#do nothing
shouldTag = shouldTag
#end if token == x
#end if len(data)
#end if len(data_entry)
#end for data_entry
else:
#file previously tagged, no need to tag it again
shouldTag = False
#end if tag_update
#end if DataTokens.itunes_tag_data_token
return shouldTag
#end def shouldTag
def getFileCommentTagContents(self, part_item):
"""docstring for getFileCommentTagContents"""
# use latest subler as it can read metadata
SublerCLI = os.path.join(sys.path[0], "SublerCLI")
#Create the command line string
get_tags_cmd = ['%s' % SublerCLI]
get_tags_cmd.append("-source")
get_tags_cmd.append('%s' % part_item.modified_file_path())
get_tags_cmd.append('-listmetadata')
#check if file has already been tagged
result = subprocess.Popen(get_tags_cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE).communicate()[0]
#error checking
if "Error" in result:
logging.critical("Failed to determine file tagged status")
return ""
#end if "Error" in result:
for line in result.split("\n"):
if DataTokens.subler_comment_token in line:
return line.replace(DataTokens.subler_comment_token, '')
logging.info("File untagged")
return ""
#end if tagString in result
#end getFileCommentTagContents
def preexec(self): # Don't forward signals.
os.setpgrp()
def execute_command(self, actionable_file_path, command, action_description):
logging.debug("'%s' arguments: %s" % (action_description, command))
if self.opts.dryrun:
result = "dryrun"
else:
#check if file exists
if os.path.isfile(actionable_file_path):
#run command
result = subprocess.Popen(command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, preexec_fn = self.preexec).communicate()[0]
else:
result = "Error: Unable to find file."
#end if isfile
#end if dryrun
if "Error" in result:
logging.critical("Failed %s for '%s': %s" % (action_description, actionable_file_path, result.strip()) )
return False
else:
#success
logging.warning("'%s': '%s'" % (action_description, actionable_file_path))
return True
#end if "Error"
#end def execute_command
def remove_tags(self, part_item):
SublerCLI = os.path.join(sys.path[0], "SublerCLI-v010")
filepath = part_item.modified_file_path()
#removal of artwork doesn't seem to work
all_tags = ["{Artwork:}", "{HD Video:}", "{Gapless:}", "{Content Rating:}", "{Media Kind:}", "{Name:}", "{Artist:}", "{Album Artist:}", "{Album:}", "{Grouping:}", "{Composer:}", "{Comments:}", "{Genre:}", "{Release Date:}", "{Track #:}", "{Disk #:}", "{TV Show:}", "{TV Episode #:}", "{TV Network:}", "{TV Episode ID:}", "{TV Season:}", "{Description:}", "{Long Description:}", "{Rating:}", "{Rating Annotation:}", "{Studio:}", "{Cast:}", "{Director:}", "{Codirector:}", "{Producers:}", "{Screenwriters:}", "{Lyrics:}", "{Copyright:}", "{Encoding Tool:}", "{Encoded By:}", "{contentID:}"]#these are currently not supported in subler cli tool, "{XID:}", "{iTunes Account:}", "{Sort Name:}", "{Sort Artist:}", "{Sort Album Artist:}", "{Sort Album:}", "{Sort Composer:}", "{Sort TV Show:}"]
logging.warning("removing tags...")
#Create the command line command
tag_removal_cmd = ['%s' % SublerCLI]
if self.opts.optimize:
action_description = "Tags removed and optimized"
tag_removal_cmd.append("-O")
else:
action_description = "Tags removed"
#end if optimize
tag_removal_cmd.append("-t")
tag_removal_cmd.append("".join(all_tags))
tag_removal_cmd.append("-i")
tag_removal_cmd.append(filepath)
success = self.execute_command(filepath, tag_removal_cmd, action_description)
if success:
Summary().metadata_removal_succeeded()
else:
Summary().metadata_removal_failed()
#end success
#end remove_tags
def tag(self, part_item):
SublerCLI = os.path.join(sys.path[0], "SublerCLI-v010")
filepath = part_item.modified_file_path()
directory = os.path.dirname(filepath)
filename = os.path.basename(filepath)
filename_without_extension = os.path.splitext(filename)[0]
logging.warning("tagging...")
#Create the command line command
tag_cmd = ['%s' % SublerCLI]
if self.opts.optimize:
action_description = "Tags added and optimized"
tag_cmd.append("-O")
else:
action_description = "Tags added"
#end if optimize
if self.opts.embed_subtitles:
action_description += ", and any subtitles embedded"
compatible_sidecar_subtitles = self.get_all_sidecar_subtitles(directory, filename_without_extension, codec='srt')
if len(compatible_sidecar_subtitles) == 0:
logging.warning("Found no 'srt' subtitle files to embed...")
else:
for sub in compatible_sidecar_subtitles:
tag_cmd.append("-s")
tag_cmd.append(os.path.join(directory, sub))
if self.opts.chapter_previews:
action_description += ", and chapter previews generated"
tag_cmd.append("-p")
tag_cmd.append("-t")
tag_cmd.append(part_item.tag_string()) #also downloads the artwork
tag_cmd.append("-i")
tag_cmd.append(filepath)
success = self.execute_command(filepath, tag_cmd, action_description)
if success:
Summary().metadata_embedded_succeeded()
else:
Summary().metadata_embedded_failed()
#end success
#end tag
def optimize(self, part_item):
SublerCLI = os.path.join(sys.path[0], "SublerCLI-v010")
filepath = part_item.modified_file_path()
logging.warning("optimizing file...")
action_description = "Tags optimized"
#Create the command line command
optimize_cmd = ['%s' % SublerCLI]
optimize_cmd.append("-O")
optimize_cmd.append("-i")
optimize_cmd.append(filepath)
success = self.execute_command(filepath, optimize_cmd, action_description)
if success:
Summary().metadata_optimized_succeeded()
else:
Summary().metadata_optimized_failed()
#end success
#end optimize
def export_resources(self, part_item):
part_item_file_path = part_item.modified_file_path()
directory = os.path.dirname(part_item_file_path)
filename = os.path.basename(part_item_file_path)
filename_without_extension = os.path.splitext(filename)[0]
#=== subtitles ===
#build up language_code dict
if self.opts.export_subtitles:
logging.warning("attempting to export subtitles...")
subtitle_stream_type = "3"
all_non_embedded_subtitles = []
for stream_item in part_item.stream_items:
if stream_item.stream_type == subtitle_stream_type and stream_item.key != "":
all_non_embedded_subtitles.append(stream_item)
#end if
#end for
number_of_non_embedded_subtitles = len(all_non_embedded_subtitles)
if number_of_non_embedded_subtitles == 0:
logging.warning("no subtitles found")
else:
logging.warning("found %d subtitle(s)" % number_of_non_embedded_subtitles)
categorized_subtitles = {}
for subtitle in all_non_embedded_subtitles:
key = (subtitle.language_code, subtitle.codec)
if categorized_subtitles.has_key(key):
categorized_subtitles[key].append(subtitle)
else:
categorized_subtitles[key] = [subtitle]
#end if has_key
#end for all_subtitles
exported_subtitles = 0
for key, subtitles in categorized_subtitles.iteritems():
#key = (eng, srt), (eng, sub), (fre, srt), etc.
language_code = key[0]
codec = key[1]
#get all existing sub files. example filename: Sopranos - S01E01 - The Pilot*.eng.srt
compatible_sidecar_subtitles = self.get_all_sidecar_subtitles(directory, filename_without_extension, language_code, codec)
if len(compatible_sidecar_subtitles) > 0:
logging.warning("Subtitle file(s) with language code '%s' of type '%s' already exist. Skipping all matching..." % (language_code, codec))
continue
#end if
#export subs
i = 0
for subtitle in subtitles:
if i == 0:
subtitle_filename = "%s.%s.%s" % (filename_without_extension, language_code, codec)
else:
subtitle_filename = "%s.%02d.%s.%s" % (filename_without_extension, i, language_code, codec)
#end if
subtitle_full_path = os.path.join(directory, subtitle_filename)
success = subtitle.export_to_path(subtitle_full_path)
if success:
i += 1
exported_subtitles += 1
Summary().subtitle_export_succeeded()
else:
Summary().subtitle_export_failed()
#end success
#end for subtitles
#end for categorized_subtitles
logging.warning( "exported %d subtitle(s), skipped %d" % ( exported_subtitles, (number_of_non_embedded_subtitles-exported_subtitles) ) )
#end if len(all_non_embedded_subtitles) == 0:
#end if subtitles
if self.opts.export_artwork:
logging.error("artwork export not yet implemented...")
#logging.warning("attempting to export artwork...")
#end def export_resources
def get_all_sidecar_subtitles(self, directory, filename_without_extension, language_code=None, codec=None):
#get all existing sub files. example filename: Sopranos - S01E01 - The Pilot*.eng.srt
current_directory = os.getcwd()
try:
os.chdir(directory)
except OSError as (errorstr):
logging.critical("Failed 'resource export': %s" % (errorstr) )
logging.critical('Do you have any "yellow exclamation marks" in the Plex Media Manager?')
return False
#end try
glob_str = "%s*" % (filename_without_extension)
if language_code:
glob_str += ".%s" % (language_code)
if codec:
glob_str += ".%s" % (codec)
subs = glob.glob(glob_str)
os.chdir(current_directory)
return subs
def add_to_itunes(self, part_item):
if self.video_item.__class__.__name__ == "MovieItem":
itunes_playlist = "Movies"
else:
itunes_playlist = "TV Shows"
item_title = self.video_item.title
actionable_file_path = part_item.modified_file_path()
if not self.opts.force:
logging.warning("Finding '%s' in iTunes..." % actionable_file_path)
search_string = 'set currentItems to search playlist "%s" for "%s" only displayed' % (itunes_playlist, item_title)
delimiter = "###"
result_creation_string = 'set output to output & "%s"' % delimiter
does_item_exist_command = ["osascript", '-e', 'tell application "iTunes"', '-e', 'try' ,'-e', search_string, '-e', 'set output to ""', '-e', 'repeat with currentItem in currentItems', '-e', 'set loc to (location of currentItem)', '-e', 'if output is not equal to "" then', '-e', result_creation_string, '-e', 'end if', '-e', 'set output to output & POSIX path of loc', '-e', 'end repeat', '-e', 'output', '-e', 'end try', '-e', 'end tell']
logging.debug("'find in iTunes' script: %s" % (does_item_exist_command))
result_string = subprocess.Popen(does_item_exist_command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, preexec_fn = self.preexec).communicate()[0].rstrip()
result_file_paths = result_string.split(delimiter)
logging.debug("Find results in iTunes: %s" % result_file_paths)
for file_path in result_file_paths:
unicode_filepath = normalize('NFC', file_path.decode('utf-8'))
if unicode_filepath == actionable_file_path:
logging.warning(" Already added to iTunes")
return
#end if result == actionable_file_path:
#end for file_path in result_file_paths
#end if not self.opts.force
logging.warning(" Adding to iTunes...")
file_str = 'set p to (POSIX file "%s")' % actionable_file_path
add_to_playlist_str = 'add p to playlist "%s"' % itunes_playlist
add_to_itunes_command = ['osascript', '-e', 'try', '-e', file_str, '-e', 'tell application "iTunes"', '-e', add_to_playlist_str, '-e', 'end tell', '-e', 'end try']
logging.debug("'add to iTunes' script: %s" % (add_to_itunes_command))
result = subprocess.Popen(add_to_itunes_command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, preexec_fn = self.preexec).communicate()[0].rstrip()
if result.startswith("file track id"):
Summary().add_to_itunes_succeeded()
logging.warning(" Adding additional metadata to item in iTunes...")
current_item_str = 'tell %s' % result
set_rating_str = 'set rating to %i as integer' % self.video_item.itunes_rating()
set_play_count_str = 'set played count to %s as integer' % self.video_item.view_count
add_metadata_to_itunes_command = ['osascript', '-e', 'try', '-e', 'tell application "iTunes"', '-e', current_item_str, '-e', set_play_count_str, '-e', set_rating_str, '-e', 'end tell', '-e', 'end tell', '-e', 'end try']
logging.debug("'add additional metadata iTunes' script: %s" % (add_metadata_to_itunes_command))
subprocess.call(add_metadata_to_itunes_command)
else:
Summary().add_to_itunes_failed()
logging.critical("Failed 'add to iTunes' for '%s'. Incorrect path or not a compatible file type?" % (actionable_file_path) )
#end if result.startswith
#end def add_to_itunes
def process(self):
skipped_all = True
if self.opts.gather_statistics:
skipped_all = False
LibraryStatistics().add_item(self.video_item)
#end if gather_statistics
for index, media_item in enumerate(self.media_items):
logging.warning( "processing %d/%d media_items" % ( index+1, len(self.media_items)) )
part_items = media_item.part_items
for index2, part_item in enumerate(part_items):
no_action = True
logging.warning( " processing %d/%d part_items" % ( index2+1, len(part_items)) )
Summary().increment_parts_processed()
if self.opts.removetags and self.canTag(part_item):
skipped_all = no_action = False
self.remove_tags(part_item)
#end if removetags
if self.opts.tag and self.canTag(part_item) and self.shouldTag(part_item):
skipped_all = no_action = False
self.tag(part_item)
#end if tag
if self.opts.optimize and not self.opts.tag and not self.opts.removetags and self.canTag(part_item):
#optimize is done together with tagging or removing, so only needs to be done here if it's the exclusive action
skipped_all = no_action = False
self.optimize(part_item)
#end if optimize
if self.opts.export_resources:
skipped_all = no_action = False
self.export_resources(part_item)
#end if export_resouces
if self.opts.add_to_itunes:
skipped_all = no_action = False
self.add_to_itunes(part_item)
#end if add_to_itunes
if self.opts.open_file_location and not no_action:
logging.warning("opening '%s'..." % part_item.modified_file_path())
command = ['open', "-R", part_item.modified_file_path()]
subprocess.call(command)
#end if self.opts.open_file_location
#end for part_items
#end for media_items
if skipped_all:
logging.warning("skipping: found no suitable files for specified tasks")
#end if skipped_all
#end def process_item
#end MediaPartProcessor