python-ox/ox/movie.py

496 lines
20 KiB
Python
Raw Permalink Normal View History

2011-10-13 10:21:16 +00:00
# -*- coding: utf-8 -*-
# vi:si:et:sw=4:sts=4:ts=4
# GPL 2012
2011-10-13 10:21:16 +00:00
from __future__ import division
2011-10-13 13:08:40 +00:00
import hashlib
import os
import re
2012-09-11 12:36:51 +00:00
import unicodedata
2011-10-13 10:21:16 +00:00
2014-09-30 19:04:46 +00:00
from .normalize import normalize_name
from .text import get_sort_name, find_re
from .file import EXTENSIONS
2011-10-13 10:21:16 +00:00
2011-10-13 13:08:40 +00:00
__all__ = ['parse_movie_path', 'create_movie_path', 'get_oxid']
2011-10-13 10:21:16 +00:00
LANGUAGES = ['en', 'fr', 'de', 'es', 'it']
'''
Naming scheme:
2012-08-16 23:59:09 +00:00
X/[Group, The; Lastname, Firstname/]The Title[ (YEAR[-[YEAR]])]/
The Title[ ([SXX][EYY[+ZZ|-ZZ]])[ Episode Title]][.Version][.Part XY[.Part Title][.en][.fr].xyz
'''
def format_path(data, directory_key='director'):
def format_underscores(string):
2024-08-30 11:30:47 +00:00
return re.sub(r'^\.|\.$|:|/|\?|<|>', '_', string)
2012-09-11 12:36:51 +00:00
director = data['directorSort'] or ['Unknown Director']
2012-09-11 16:30:42 +00:00
title = data['seriesTitle' if data['isEpisode'] else 'title'] or 'Untitled'
year = data['seriesYear' if data['isEpisode'] else 'year'] or None
2017-02-18 12:02:21 +00:00
parts = list(map(format_underscores, filter(lambda x: x is not None, [
u'; '.join(director[:10]),
u'%s%s' % (title, u' (%s)' % year if year else ''),
2012-09-11 12:36:51 +00:00
u'%s%s%s%s%s%s' % (
data['title'] or 'Untitled',
2012-09-11 12:36:51 +00:00
u'.%s' % data['version'] if data['version'] else '',
u'.Part %s' % data['part'] if data['part'] else '',
u'.%s' % data['partTitle'] if data['partTitle'] else '',
u'.%s' % data['language'] if data['language'] else '',
u'.%s' % data['extension'] if data['extension'] else ''
)
])))
2012-09-11 12:36:51 +00:00
if data.get('subdirectory'):
parts.insert(-1, data['subdirectory'])
2012-09-11 12:36:51 +00:00
return unicodedata.normalize('NFD', u'/'.join(parts))
2012-08-20 16:19:17 +00:00
def parse_item_files(files):
# parses a list of file objects associated with one item (file objects
2012-08-22 10:27:32 +00:00
# as returned by parse_path, but extended with 'path' and 'time')
# and returns a list of version objects (in case of english-only subtitles,
# version[i]['files'][j]['normalizedPath'] will be modified)
def get_file_key(file):
return '\n'.join([
file['version'] or '',
file['part'] or '',
2012-08-22 15:17:22 +00:00
file['language'] or '',
file['extension'] or ''
])
def get_version_key(file, extension=True):
return '%s/%s-part/%s' % (
2012-08-22 10:50:13 +00:00
file['version'] or '',
2017-02-18 12:02:21 +00:00
'single' if file['part'] is None else 'multi',
file['extension'] if extension else ''
)
# filter out duplicate files (keep shortest path, sorted alphabetically)
# since same version+part+language+extension can still differ in part title,
# ''/'en' or 'mpg'/'mpeg', or have an unparsed section in their path
unique_files = []
duplicate_files = []
for key in [get_file_key(file) for file in files]:
key_files = sorted(
2017-02-18 12:01:33 +00:00
[file for file in files if get_file_key(file) == key],
2012-08-22 10:27:32 +00:00
key=lambda x: len(x['path'])
)
2012-08-22 10:24:29 +00:00
unique_files.append(key_files[0])
duplicate_files += key_files[1:]
# determine versions ('version.single|multi-part.videoextension')
version_files = {}
time = {}
video_files = [file for file in unique_files if file['type'] == 'video']
versions = set([file['version'] for file in video_files])
for version in versions:
for file in [file for file in video_files if file['version'] == version]:
version_key = get_version_key(file)
version_files[version_key] = (version_files[version_key] if version_key in version_files else []) + [file]
time[version_key] = sorted([time[version_key], file['time']])[-1] if version_key in time else file['time']
# determine preferred video extension (newest)
extension = {}
2012-08-22 10:54:56 +00:00
for key in set(['/'.join(version_key.split('/')[:-1]) + '/' for version_key in version_files]):
extensions = set([version_key.split('/')[-1] for version_key in version_files if version_key.startswith(key)])
extension[key] = sorted(extensions, key=lambda x: time[key + x])[-1]
# associate other (non-video) files
other_files = [file for file in unique_files if file['type'] != 'video']
versions = set([file['version'] for file in other_files])
for version in versions:
for file in [file for file in other_files if file['version'] == version]:
key = get_version_key(file, extension=False)
if key in extension:
version_files[key + extension[key]].append(file)
else:
version_files[key] = (version_files[key] if key in version_files else []) + [file]
extension[key] = ''
2012-08-22 15:10:54 +00:00
# determine main files (video + srt)
full = {}
language = {}
main_files = {}
for version_key in version_files:
parts = sorted(list(set([file['part'] for file in version_files[version_key]])))
2012-08-22 15:10:54 +00:00
# determine if all parts have one video file
video_files = [file for file in version_files[version_key] if file['type'] == 'video']
full[version_key] = len(video_files) == len(parts)
main_files[version_key] = video_files if full[version_key] else []
# determine preferred subtitle language
language[version_key] = None
2012-08-22 10:31:16 +00:00
subtitle_files = [file for file in version_files[version_key] if file['extension'] == 'srt']
2017-02-18 12:01:33 +00:00
subtitle_languages = list(set([file['language'] for file in subtitle_files]))
for subtitle_language in sorted(subtitle_languages, key=subtitle_sort):
language_files = [file for file in subtitle_files if file['language'] == subtitle_language]
if len(subtitle_files) == len(parts):
language[version_key] = subtitle_language
main_files[version_key] += language_files
break
2012-08-22 10:36:17 +00:00
# determine main version (best srt language, then video time)
main_version = None
full_version_keys = sorted(
[version_key for version_key in version_files if full[version_key]],
2012-08-22 10:36:17 +00:00
key=lambda x: time[x],
reverse=True
)
if full_version_keys:
language_version_keys = sorted(
[version_key for version_key in full_version_keys if language[version_key]],
key=lambda x: LANGUAGES.index(language[x]) if language[x] in LANGUAGES else language[x]
)
main_version = language_version_keys[0] if language_version_keys else full_version_keys[0]
# add duplicate files
for file in duplicate_files:
2012-08-22 12:13:17 +00:00
key = get_version_key(file, extension=False)
version_key = '%s%s' % (key, extension[key] if key in extension else '')
2012-08-22 12:15:32 +00:00
version_files[version_key] = (version_files[version_key] if version_key in version_files else []) + [file]
# remove unneeded '.en'
for version_key in version_files:
for extension in EXTENSIONS['subtitle']:
subtitle_files = [file for file in version_files[version_key] if file['extension'] == extension]
2013-05-31 14:01:55 +00:00
subtitle_languages = list(set([file['language'] for file in subtitle_files]))
if len(subtitle_languages) == 1 and subtitle_languages[0] == LANGUAGES[0]:
for subtitle_file in subtitle_files:
subtitle_file['normalizedPath'] = format_path(dict(subtitle_file, **{'language': None}))
# return data
data = []
for version_key in version_files:
data.append({
'files': sorted(
[dict(file, isMainFile=file in main_files[version_key]) for file in version_files[version_key]],
key=lambda x: x['path']
),
'isFullVersion': full[version_key],
'isMainVersion': version_key == main_version,
'subtitleLanguage': language[version_key] if version_key in language else None,
'version': version_key
})
return data
def parse_path(path, directory_key='director'):
'''
# all keys
>>> parse_path('Frost, Mark; Lynch, David/Twin Peaks (1991)/Twin Peaks (S01E01) Pilot.European Version.Part 1.Welcome to Twin Peaks.en.fr.MPEG')['normalizedPath']
'Frost, Mark; Lynch, David/Twin Peaks (1991)/Twin Peaks (S01E00) Pilot.European Version.Part 1.Welcome to Twin Peaks.en.fr.mpg'
2012-09-09 17:28:11 +00:00
# pop directory title off file name
>>> parse_path("Unknown Director/www.xxx.com.._/www.xxx.com....Director's Cut.avi")['version']
2012-09-09 17:28:11 +00:00
"Director's Cut"
# handle dots
>>> parse_path("Unknown Director/Unknown Title (2000)/... Mr. .com....Director's Cut.srt")['version']
2012-09-09 17:28:11 +00:00
"Director's Cut"
# multiple years, season zero, multiple episodes, dots in episode title and part title
>>> parse_path('Groening, Matt/The Simpsons (1989-2012)/The Simpsons (S00E01-02) D.I.Y..Uncensored Version.Part 1.D.I.Y..de.avi')['normalizedPath']
'Groening, Matt/The Simpsons (1989-2012)/The Simpsons (S01E01+02) D.I.Y..Uncensored Version.Part 1.D.I.Y..de.avi'
2012-09-09 17:28:11 +00:00
# handle underscores
>>> parse_path('Unknown Director/_com_ 1_0 _ NaN.._/_com_ 1_0 _ NaN....avi')['title']
'.com: 1/0 / NaN...'
2012-09-09 17:28:11 +00:00
2012-08-16 23:59:09 +00:00
# TODO: '.com.avi'
'''
2016-09-23 10:48:02 +00:00
def parse_type(string):
for type in EXTENSIONS:
if string in EXTENSIONS[type]:
return type
return None
def parse_underscores(string):
string = unicodedata.normalize('NFC', string)
2012-09-11 17:58:30 +00:00
# '^_' or '_$' is '.'
string = re.sub('^_', '.', string)
string = re.sub('_$', '.', string)
2012-09-11 17:58:30 +00:00
# '_.foo$' or '_ (' is '?'
2024-08-30 11:30:47 +00:00
string = re.sub(re.compile(r'_(?=(\.\w+$| \())', re.U), '?', string)
2012-09-11 17:58:30 +00:00
# ' _..._ ' is '<...>'
2024-09-11 21:52:01 +00:00
string = re.sub(r'(?<= )_(.+)_(?= )', r'<\g<1>>', string)
2012-09-11 17:58:30 +00:00
# 'foo_bar' or 'foo _ bar' is '/'
2024-08-30 11:30:47 +00:00
string = re.sub(re.compile(r'(?<=\w)_(?=\w)', re.U), '/', string)
string = re.sub(' _ ', ' / ', string)
2012-09-11 17:58:30 +00:00
# 'foo_ ' is ':'
2024-08-30 11:30:47 +00:00
string = re.sub(re.compile(r'(?<=[\w\)\]])_ ', re.U), ': ', string)
string = unicodedata.normalize('NFD', string)
return string
data = {}
parts = list(map(lambda x: parse_underscores(x.strip()), unicodedata.normalize('NFD', path).split('/')))
# subdirectory
if len(parts) > 4:
data['subdirectory'] = '/'.join(parts[3:-1])
parts = parts[:3] + parts[-1:]
else:
data['subdirectory'] = None
length = len(parts)
director, title, file = [
parts[-3] if length > 2 else None,
parts[-2] if length > 1 else None,
parts[-1]
]
# directorSort, director
data['directorSort'] = data['director'] = []
if director:
data['directorSort'] = list(filter(
lambda x: x != 'Unknown Director',
director.split('; ')
))
data['director'] = list(map(
lambda x: ' '.join(reversed(x.split(', '))),
data['directorSort']
))
# title, year
data['title'] = data['year'] = None
if title:
2024-08-30 11:30:47 +00:00
match = re.search(r' \(\d{4}(-(\d{4})?)?\)$', title)
2012-08-16 16:18:44 +00:00
data['title'] = title[:-len(match.group(0))] if match else title
data['year'] = match.group(0)[2:-1] if match else None
2012-08-18 19:35:58 +00:00
file_title = re.sub('[/:]', '_', data['title'])
# (remove title from beginning of filename if the rest contains a dot)
2024-09-11 21:52:01 +00:00
file = re.sub(r'^' + re.escape(file_title) + r'(?=.*\.)', '', file)
2012-08-20 13:32:06 +00:00
# (split by nospace+dot+word, but remove spaces preceding extension)
2024-08-30 11:30:47 +00:00
parts = re.split(r'(?<!\s)\.(?=\w)', re.sub(r'\s+(?=.\w+$)', '', file))
title, parts, extension = [
parts[0],
parts[1:-1],
parts[-1] if len(parts) > 1 else None
]
if not data['title'] and title:
data['title'] = title
# season, episode, episodes, episodeTitle
data['season'] = data['episode'] = data['episodeTitle'] = None
data['episodes'] = []
2024-08-30 11:30:47 +00:00
match = re.search(r' \((S\d{2})?(E\d{2}([+-]\d{2})?)?\)(.+)?', title)
if match:
if match.group(1):
data['season'] = int(match.group(1)[1:])
if match.group(2):
if len(match.group(2)) == 3:
data['episode'] = int(match.group(2)[1:])
else:
data['episodes'] = range(int(match.group(2)[1:3]), int(match.group(2)[-2:]) + 1)
if match.group(4):
data['episodeTitle'] = match.group(4)[1:]
2024-08-30 11:30:47 +00:00
while data['episodeTitle'] and len(parts) and re.search(r'^\w+\.*$', parts[0]) and not re.search(r'^[a-z]{2}$', parts[0]):
data['episodeTitle'] += '.%s' % parts.pop(0)
# isEpisode, seriesTitle, seriesYear
data['isEpisode'] = False
data['seriesTitle'] = data['seriesYear'] = None
2017-02-18 12:02:21 +00:00
if data['season'] is not None or data['episode'] is not None or data['episodes']:
2012-08-15 22:34:12 +00:00
data['isEpisode'] = True
data['seriesTitle'] = data['title']
2017-02-18 12:02:21 +00:00
season = 'S%02d' % data['season'] if data['season'] is not None else ''
episode = ''
2017-02-18 12:02:21 +00:00
if data['episode'] is not None:
episode = 'E%02d' % data['episode']
elif data['episodes']:
episode = 'E%02d%s%02d' % (
data['episodes'][0], '+' if len(data['episodes']) == 2 else '-', data['episodes'][-1]
)
episodeTitle = ' %s' % data['episodeTitle'] if data['episodeTitle'] else ''
data['title'] += ' (%s%s)%s' % (season, episode, episodeTitle)
2012-08-15 22:34:12 +00:00
data['seriesYear'] = data['year']
data['year'] = None
# version
data['version'] = parts.pop(0) if len(parts) and re.search('^[A-Z0-9]', parts[0]) and not re.search('^Part .', parts[0]) else None
# part
data['part'] = parts.pop(0)[5:] if len(parts) and re.search('^Part .', parts[0]) else None
2012-08-15 22:34:12 +00:00
# partTitle
data['partTitle'] = parts.pop(0) if len(parts) and re.search('^[A-Z0-9]', parts[0]) and data['part'] else None
while data['partTitle'] and len(parts) and not re.search('^[a-z]{2}$', parts[0]):
data['partTitle'] += '.%s' % parts.pop(0)
# language
2013-02-17 14:39:48 +00:00
data['language'] = parts.pop(0) if len(parts) and re.search('^[a-z]{2}$', parts[0]) else None
# extension
data['extension'] = re.sub('^mpeg$', 'mpg', extension.lower()) if extension else None
# type
data['type'] = parse_type(data['extension'])
# normalizedPath
data['normalizedPath'] = format_path(data)
return data
2011-10-13 10:21:16 +00:00
def parse_movie_path(path):
"""
"A/Abrams, J.J.; Lieber, Jeffrey; Lindelof, Damon/Lost (2004)/Lost.Season 3.Episode 21.Greatest Hits.avi"
"B/Balada, Ivan/Metrum (1967)/Metrum.Part 1.en.srt"
"N/Nakata, Hideo/L - Change the World (2008)/L - Change the World.Part 2.srt"
"R/Reitz, Edgar/Heimat (1984-2006)/Heimat.Season 2.Episode 8.The Wedding.Part 2.avi"
"F/Feuillade, Louis/Les vampires (1915)/Les vampires.Episode 10.Part 2.avi"
title: 'Les vampires', year: '1915', episode: 10, part: 2
"G/Godard, Jean-Luc/Histoire(s) du cinema_ Toutes les histoires (1988)/Histoire(s) du cinema_ Toutes les histoires.avi"
"G/Godard, Jean-Luc/Six fois deux (1976)/Six fois deux.Part 1A.Y a personne.avi"
"G/Godard, Jean-Luc; Miéville, Anne-Marie/France_tour_detour_deux_enfants (1977)/France_tour_detour_deux_enfants.Part 5.Impression_Dictée.avi"
"L/Labarthe, André S_/Cinéastes de notre temps (1964-)/Cinéastes de notre temps.Episode.Jean Renoir le patron, première partie_ La Recherche du relatif.avi"
"S/Scott, Ridley/Blade Runner (1982)/Blade Runner.Directors's Cut.avi"
2011-10-13 17:28:18 +00:00
or
T/Title (Year)/Title.avi
2011-10-13 10:21:16 +00:00
"""
2011-10-14 19:50:50 +00:00
episodeTitle = episodeYear = seriesTitle = None
2011-10-13 10:21:16 +00:00
episodeDirector = []
parts = unicodedata.normalize('NFD', path).split('/')
2011-10-13 10:21:16 +00:00
#title/year
2011-10-13 17:28:18 +00:00
if len(parts) == 4:
title = parts[2]
2012-01-31 00:28:58 +00:00
elif len(parts) > 1:
2011-10-13 17:28:18 +00:00
title = parts[1]
2012-01-31 00:28:58 +00:00
else:
title = parts[0]
2011-11-22 21:35:06 +00:00
title = title.replace('_ ', ': ')
2012-03-18 14:38:51 +00:00
if title.endswith('_'):
title = title[:-1] + '.'
2012-08-15 15:15:40 +00:00
if title.startswith('_'):
title = '.' + title[1:]
2011-11-22 21:35:06 +00:00
2024-08-30 11:30:47 +00:00
year = find_re(title, r'(\(\d{4}\))')
2011-10-13 10:21:16 +00:00
if not year:
2024-08-30 11:30:47 +00:00
year = find_re(title, r'(\(\d{4}-\d*\))')
2011-10-13 10:21:16 +00:00
if year and title.endswith(year):
title = title[:-len(year)].strip()
year = year[1:-1]
if '-' in year:
2024-08-30 11:30:47 +00:00
year = find_re(year, r'\d{4}')
2011-10-13 17:28:18 +00:00
2011-10-13 10:21:16 +00:00
#director
2011-10-13 17:28:18 +00:00
if len(parts) == 4:
director = parts[1]
if director.endswith('_'):
director = "%s." % director[:-1]
director = director.split('; ')
director = [normalize_name(d).strip() for d in director]
2016-09-23 10:48:02 +00:00
director = list(filter(lambda d: d not in ('Unknown Director', 'Various Directors'), director))
2011-10-13 17:28:18 +00:00
else:
director = []
2011-10-13 10:21:16 +00:00
#extension/language
2011-10-17 11:56:49 +00:00
fileparts = [x.replace('||', '. ') for x in parts[-1].replace('. ', '||').split('.')]
2012-01-31 00:28:58 +00:00
extension = len(fileparts) > 1 and fileparts[-1] or ''
2011-10-13 10:21:16 +00:00
2012-01-31 00:28:58 +00:00
if len(fileparts) > 1 and len(fileparts[-2]) == 2:
2011-10-13 10:21:16 +00:00
language = fileparts[-2]
else:
language = ''
#season/episode/episodeTitle
2024-08-30 11:30:47 +00:00
match = re.compile(r'(.+?) \((S(\d+))?(E(\d+))?\)( (.+?))?\.').match(parts[-1])
2012-09-01 03:27:59 +00:00
if match:
seriesTitle = match.group(1)
season = match.group(3)
episode = match.group(5)
episodeTitle = (match.group(6) or '').strip()
2017-02-18 12:02:21 +00:00
if episode is not None:
2012-09-01 03:27:59 +00:00
episode = int(episode)
2017-02-18 12:02:21 +00:00
if season is not None:
2012-09-01 03:27:59 +00:00
season = int(season)
if episode and not season:
season = 1
2011-10-13 10:21:16 +00:00
else:
2024-08-30 11:30:47 +00:00
season = find_re(parts[-1], r'\.Season (\d+)\.')
2012-09-01 03:27:59 +00:00
if season:
season = int(season)
else:
season = None
2011-10-13 10:21:16 +00:00
2024-08-30 11:30:47 +00:00
episode = find_re(parts[-1], r'\.Episode[s]* ([\d+]+)\.')
2012-09-01 03:27:59 +00:00
if episode:
episode = episode.split('+')[0]
episode = int(episode)
else:
episode = None
2011-10-13 10:21:16 +00:00
2017-02-18 12:02:21 +00:00
if episode and 'Episode %d' % episode in fileparts:
2012-09-01 03:27:59 +00:00
episodeTitle = fileparts.index('Episode %d' % episode) + 1
episodeTitle = fileparts[episodeTitle]
if episodeTitle == extension or episodeTitle.startswith('Part'):
episodeTitle = None
2011-11-22 21:21:29 +00:00
2012-09-01 03:27:59 +00:00
if not season and 'Episode' in fileparts:
episodeTitle = fileparts.index('Episode') + 1
episodeTitle = fileparts[episodeTitle]
if episodeTitle == extension or episodeTitle.startswith('Part'):
episodeTitle = None
else:
season = 1
2011-11-22 21:21:29 +00:00
2011-10-14 19:50:50 +00:00
if season:
seriesTitle = title
title = u'%s (S%02d)' % (seriesTitle, season)
2011-11-22 21:30:29 +00:00
if isinstance(episode, int):
2011-10-14 19:50:50 +00:00
title = u'%s (S%02dE%02d)' % (seriesTitle, season, episode)
if episodeTitle:
title = u'%s %s' % (title, episodeTitle)
2011-10-13 10:21:16 +00:00
#part
2024-08-30 11:30:47 +00:00
part = find_re(parts[-1], r'\.Part (\d+)\.')
2011-10-13 10:21:16 +00:00
if part:
part = int(part)
else:
part = 0
return {
'director': director,
'episodeDirector': episodeDirector,
'episode': episode,
'episodeTitle': episodeTitle,
'episodeYear': episodeYear,
'extension': extension,
'language': language,
'part': part,
'season': season,
2011-10-14 19:50:50 +00:00
'seriesTitle': seriesTitle,
2011-10-13 10:21:16 +00:00
'title': title,
'year': year,
}
def create_movie_path(title, director, year,
season, episode, episodeTitle, episodeDirector, episodeYear,
part, language, extension):
'''
{
title: '', director: [''], year: '',
season: int, episode: int, episodeTitle: '', episodeDirector: [''], episodeYear: '',
part: int, language: '', extension: '', extra: bool
})
'''
partTitle = None
director = '; '.join(map(get_sort_name, director))
episodeDirector = '; '.join(map(get_sort_name, episodeDirector))
filename = [title]
if season:
filename += ['Season %d' % season]
if episode:
filename += ['Episode %d' % episode]
if episodeTitle:
filename += [episodeTitle]
if part:
filename += ['Part %s' % part]
if partTitle:
filename += [partTitle]
2012-01-31 00:28:58 +00:00
if extension:
filename += [extension]
2011-10-13 10:21:16 +00:00
filename = '.'.join(filename)
path = os.path.join(director[0], director, '%s (%s)' % (title, year), filename)
return path
2011-10-13 13:08:40 +00:00
def get_oxid(title, director=[], year='',
season='', episode='', episode_title='', episode_director=[], episode_year=''):
def get_hash(string):
return hashlib.sha1(string.encode('utf-8')).hexdigest().upper()
director = ', '.join(director)
episode_director = ', '.join(episode_director)
if not season and not episode and not episode_title:
2011-10-13 13:08:40 +00:00
oxid = get_hash(director)[:8] + get_hash('\n'.join([title, str(year)]))[:8]
else:
oxid = get_hash('\n'.join([director, title, str(year), str(season)]))[:8] + \
get_hash('\n'.join([str(episode), episode_director, episode_title, str(episode_year)]))[:8]
return u'0x' + oxid
2017-02-18 12:01:33 +00:00
def subtitle_sort(language):
if language in LANGUAGES:
return str(LANGUAGES.index(language))
elif language is None:
return str(len(LANGUAGES))
else:
return language