#!/usr/bin/env python2.7 # -*- coding: utf-8 -*- import sys import struct import urllib import os import hashlib import mutagen import binascii import subprocess import collections import errno import argparse import shutil import re import tempfile import signal audio_ext = (".mp3", ".m4a", ".m4b", ".m4p", ".aa", ".wav") list_ext = (".pls", ".m3u") def make_dir_if_absent(path): try: os.makedirs(path) except OSError as exc: if exc.errno != errno.EEXIST: raise def raises_unicode_error(str): try: str.decode('utf-8').encode('latin-1') return False except (UnicodeEncodeError, UnicodeDecodeError): return True def hash_error_unicode(item): return "".join(["{0:02X}".format(ord(x)) for x in reversed(hashlib.md5(item).digest()[:8])]) pass def validate_unicode(path): path_list = path.split('/') last_raise = False for i in xrange(len(path_list)): if raises_unicode_error(path_list[i]): path_list[i] = hash_error_unicode(path_list[i]) last_raise = True else: last_raise = False extension = os.path.splitext(path)[1].lower() return "/".join(path_list) + (extension if last_raise and extension in audio_ext else '') def exec_exists_in_path(command): with open(os.devnull, 'w') as FNULL: try: with open(os.devnull, 'r') as RFNULL: subprocess.call([command], stdout=FNULL, stderr=subprocess.STDOUT, stdin=RFNULL) return True except OSError as e: return False def splitpath(path): return path.split(os.sep) def get_relpath(path, basepath): commonprefix = os.sep.join(os.path.commonprefix(map(splitpath, [path, basepath]))) return os.path.relpath(path, commonprefix) def is_path_prefix(prefix, path): return prefix == os.sep.join(os.path.commonprefix(map(splitpath, [prefix, path]))) class Text2Speech(object): valid_tts = {'pico2wave': True, 'RHVoice': True, 'espeak': True} @staticmethod def check_support(): voiceoverAvailable = False # Check for pico2wave voiceover if not exec_exists_in_path("pico2wave"): Text2Speech.valid_tts['pico2wave'] = False print "Warning: pico2wave not found, voicever won't be generated using it." else: voiceoverAvailable = True # Check for espeak voiceover if not exec_exists_in_path("espeak"): Text2Speech.valid_tts['espeak'] = False print "Warning: espeak not found, voicever won't be generated using it." else: voiceoverAvailable = True # Check for Russian RHVoice voiceover if not exec_exists_in_path("RHVoice"): Text2Speech.valid_tts['RHVoice'] = False print "Warning: RHVoice not found, Russian voicever won't be generated." else: voiceoverAvailable = True # Return if we at least found one voiceover program. # Otherwise this will result in silent voiceover for tracks and "Playlist N" for playlists. return voiceoverAvailable @staticmethod def text2speech(out_wav_path, text): # Skip voiceover generation if a track with the same name is used. # This might happen with "Track001" or "01. Intro" names for example. if os.path.isfile(out_wav_path): print "Using existing", out_wav_path return True # ensure we deal with unicode later if not isinstance(text, unicode): text = unicode(text, 'utf-8') lang = Text2Speech.guess_lang(text) if lang == "ru-RU": return Text2Speech.rhvoice(out_wav_path, text) else: if Text2Speech.pico2wave(out_wav_path, text): return True elif Text2Speech.espeak(out_wav_path, text): return True else: return False # guess-language seems like an overkill for now @staticmethod def guess_lang(unicodetext): lang = 'en-GB' if re.search(u"[А-Яа-я]", unicodetext) is not None: lang = 'ru-RU' return lang @staticmethod def pico2wave(out_wav_path, unicodetext): if not Text2Speech.valid_tts['pico2wave']: return False subprocess.call(["pico2wave", "-l", "en-GB", "-w", out_wav_path, unicodetext]) return True @staticmethod def espeak(out_wav_path, unicodetext): if not Text2Speech.valid_tts['espeak']: return False subprocess.call(["espeak", "-v", "english_rp", "-s", "150", "-w", out_wav_path, unicodetext]) return True @staticmethod def rhvoice(out_wav_path, unicodetext): if not Text2Speech.valid_tts['RHVoice']: return False tmp_file = tempfile.NamedTemporaryFile(suffix=".wav", delete=False) tmp_file.close() proc = subprocess.Popen(["RHVoice", "--voice=Elena", "--variant=Russian", "--volume=100", "-o", tmp_file.name], stdin=subprocess.PIPE) proc.communicate(input=unicodetext.encode('utf-8')) # make a little bit louder to be comparable with pico2wave subprocess.call(["sox", tmp_file.name, out_wav_path, "norm"]) os.remove(tmp_file.name) return True class Record(object): def __init__(self, parent): self.parent = parent self._struct = collections.OrderedDict([]) self._fields = {} self.voiceover = parent.voiceover self.playlist_voiceover = parent.playlist_voiceover self.rename = parent.rename self.trackgain = parent.trackgain def __getitem__(self, item): if item not in self._struct.keys(): raise KeyError return self._fields.get(item, self._struct[item][1]) def __setitem__(self, item, value): self._fields[item] = value def construct(self): output = "" for i in self._struct.keys(): (fmt, default) = self._struct[i] if fmt == "4s": fmt, default = "I", int(binascii.hexlify(default), 16) output += struct.pack("<" + fmt, self._fields.get(i, default)) return output def text_to_speech(self, text, dbid, playlist = False): if self.voiceover and not playlist or self.playlist_voiceover and playlist: # Create the voiceover wav file fn = "".join(["{0:02X}".format(ord(x)) for x in reversed(dbid)]) path = os.path.join(self.base, "iPod_Control", "Speakable", "Tracks" if not playlist else "Playlists", fn + ".wav") return Text2Speech.text2speech(path, text) return False def path_to_ipod(self, filename): if os.path.commonprefix([os.path.abspath(filename), self.base]) != self.base: raise IOError("Cannot get Ipod filename, since file is outside the IPOD path") baselen = len(self.base) if self.base.endswith(os.path.sep): baselen -= 1 ipodname = "/".join(os.path.abspath(filename)[baselen:].split(os.path.sep)) return ipodname def ipod_to_path(self, ipodname): return os.path.abspath(os.path.join(self.base, os.path.sep.join(ipodname.split("/")))) @property def shuffledb(self): parent = self.parent while parent.__class__ != Shuffler: parent = parent.parent return parent @property def base(self): return self.shuffledb.path @property def tracks(self): return self.shuffledb.tracks @property def albums(self): return self.shuffledb.albums @property def artists(self): return self.shuffledb.artists @property def lists(self): return self.shuffledb.lists class TunesSD(Record): def __init__(self, parent): Record.__init__(self, parent) self.track_header = TrackHeader(self) self.play_header = PlaylistHeader(self) self._struct = collections.OrderedDict([ ("header_id", ("4s", "shdb")), ("unknown1", ("I", 0x02000003)), ("total_length", ("I", 64)), ("total_number_of_tracks", ("I", 0)), ("total_number_of_playlists", ("I", 0)), ("unknown2", ("Q", 0)), ("max_volume", ("B", 0)), ("voiceover_enabled", ("B", int(self.voiceover))), ("unknown3", ("H", 0)), ("total_tracks_without_podcasts", ("I", 0)), ("track_header_offset", ("I", 64)), ("playlist_header_offset", ("I", 0)), ("unknown4", ("20s", "\x00" * 20)), ]) def construct(self): # The header is a fixed length, so no need to calculate it self.track_header.base_offset = 64 track_header = self.track_header.construct() # The playlist offset will depend on the number of tracks self.play_header.base_offset = self.track_header.base_offset + len(track_header) play_header = self.play_header.construct(self.track_header.tracks) self["playlist_header_offset"] = self.play_header.base_offset self["total_number_of_tracks"] = self.track_header["number_of_tracks"] self["total_tracks_without_podcasts"] = self.track_header["number_of_tracks"] self["total_number_of_playlists"] = self.play_header["number_of_playlists"] output = Record.construct(self) return output + track_header + play_header class TrackHeader(Record): def __init__(self, parent): self.base_offset = 0 Record.__init__(self, parent) self._struct = collections.OrderedDict([ ("header_id", ("4s", "shth")), ("total_length", ("I", 0)), ("number_of_tracks", ("I", 0)), ("unknown1", ("Q", 0)), ]) def construct(self): self["number_of_tracks"] = len(self.tracks) self["total_length"] = 20 + (len(self.tracks) * 4) output = Record.construct(self) # Construct the underlying tracks track_chunk = "" for i in self.tracks: track = Track(self) print "[*] Adding track", i track.populate(i) output += struct.pack("I", self.base_offset + self["total_length"] + len(track_chunk)) track_chunk += track.construct() return output + track_chunk class Track(Record): def __init__(self, parent): Record.__init__(self, parent) self._struct = collections.OrderedDict([ ("header_id", ("4s", "shtr")), ("header_length", ("I", 0x174)), ("start_at_pos_ms", ("I", 0)), ("stop_at_pos_ms", ("I", 0)), ("volume_gain", ("I", int(self.trackgain))), ("filetype", ("I", 1)), ("filename", ("256s", "\x00" * 256)), ("bookmark", ("I", 0)), ("dontskip", ("B", 1)), ("remember", ("B", 0)), ("unintalbum", ("B", 0)), ("unknown", ("B", 0)), ("pregap", ("I", 0x200)), ("postgap", ("I", 0x200)), ("numsamples", ("I", 0)), ("unknown2", ("I", 0)), ("gapless", ("I", 0)), ("unknown3", ("I", 0)), ("albumid", ("I", 0)), ("track", ("H", 1)), ("disc", ("H", 0)), ("unknown4", ("Q", 0)), ("dbid", ("8s", 0)), ("artistid", ("I", 0)), ("unknown5", ("32s", "\x00" * 32)), ]) def populate(self, filename): self["filename"] = self.path_to_ipod(filename) if os.path.splitext(filename)[1].lower() in (".m4a", ".m4b", ".m4p", ".aa"): self["filetype"] = 2 text = os.path.splitext(os.path.basename(filename))[0] audio = None try: audio = mutagen.File(filename, easy = True) except: print "Error calling mutagen. Possible invalid filename/ID3Tags (hyphen in filename?)" if audio: # Note: Rythmbox IPod plugin sets this value always 0. self["stop_at_pos_ms"] = int(audio.info.length * 1000) artist = audio.get("artist", [u"Unknown"])[0] if artist in self.artists: self["artistid"] = self.artists.index(artist) else: self["artistid"] = len(self.artists) self.artists.append(artist) album = audio.get("album", [u"Unknown"])[0] if album in self.albums: self["albumid"] = self.albums.index(album) else: self["albumid"] = len(self.albums) self.albums.append(album) if audio.get("title", "") and audio.get("artist", ""): text = u" - ".join(audio.get("title", u"") + audio.get("artist", u"")) # Handle the VoiceOverData if isinstance(text, unicode): text = text.encode('utf-8', 'ignore') self["dbid"] = hashlib.md5(text).digest()[:8] #pylint: disable-msg=E1101 self.text_to_speech(text, self["dbid"]) class PlaylistHeader(Record): def __init__(self, parent): self.base_offset = 0 Record.__init__(self, parent) self._struct = collections.OrderedDict([ ("header_id", ("4s", "shph")), ("total_length", ("I", 0)), ("number_of_playlists", ("I", 0)), ("number_of_non_podcast_lists", ("2s", "\xFF\xFF")), ("number_of_master_lists", ("2s", "\x01\x00")), ("number_of_non_audiobook_lists", ("2s", "\xFF\xFF")), ("unknown2", ("2s", "\x00" * 2)), ]) def construct(self, tracks): #pylint: disable-msg=W0221 # Build the master list masterlist = Playlist(self) print "[+] Adding master playlist" masterlist.set_master(tracks) chunks = [masterlist.construct(tracks)] # Build all the remaining playlists playlistcount = 1 for i in self.lists: playlist = Playlist(self) print "[+] Adding playlist", i playlist.populate(i) construction = playlist.construct(tracks) if playlist["number_of_songs"] > 0: playlistcount += 1 chunks += [construction] else: print "Error: Playlist does not contain a single track. Skipping playlist." self["number_of_playlists"] = playlistcount self["total_length"] = 0x14 + (self["number_of_playlists"] * 4) # Start the header output = Record.construct(self) offset = self.base_offset + self["total_length"] for i in range(len(chunks)): output += struct.pack("I", offset) offset += len(chunks[i]) return output + "".join(chunks) class Playlist(Record): def __init__(self, parent): self.listtracks = [] Record.__init__(self, parent) self._struct = collections.OrderedDict([ ("header_id", ("4s", "shpl")), ("total_length", ("I", 0)), ("number_of_songs", ("I", 0)), ("number_of_nonaudio", ("I", 0)), ("dbid", ("8s", "\x00" * 8)), ("listtype", ("I", 2)), ("unknown1", ("16s", "\x00" * 16)) ]) def set_master(self, tracks): # By default use "All Songs" builtin voiceover (dbid all zero) # Else generate alternative "All Songs" to fit the speaker voice of other playlists if self.playlist_voiceover and (Text2Speech.valid_tts['pico2wave'] or Text2Speech.valid_tts['espeak']): self["dbid"] = hashlib.md5("masterlist").digest()[:8] #pylint: disable-msg=E1101 self.text_to_speech("All songs", self["dbid"], True) self["listtype"] = 1 self.listtracks = tracks def populate_m3u(self, data): listtracks = [] for i in data: if not i.startswith("#"): path = i.strip() if self.rename: path = validate_unicode(path) listtracks.append(path) return listtracks def populate_pls(self, data): sorttracks = [] for i in data: dataarr = i.strip().split("=", 1) if dataarr[0].lower().startswith("file"): num = int(dataarr[0][4:]) filename = urllib.unquote(dataarr[1]).strip() if filename.lower().startswith('file://'): filename = filename[7:] if self.rename: filename = validate_unicode(filename) sorttracks.append((num, filename)) listtracks = [ x for (_, x) in sorted(sorttracks) ] return listtracks def populate_directory(self, playlistpath, recursive = True): # Add all tracks inside the folder and its subfolders recursively. # Folders containing no music and only a single Album # would generate duplicated playlists. That is intended and "wont fix". # Empty folders (inside the music path) will generate an error -> "wont fix". listtracks = [] for (dirpath, dirnames, filenames) in os.walk(playlistpath): dirnames.sort() # Ignore any hidden directories if "/." not in dirpath: for filename in sorted(filenames, key = lambda x: x.lower()): # Only add valid music files to playlist if os.path.splitext(filename)[1].lower() in (".mp3", ".m4a", ".m4b", ".m4p", ".aa", ".wav"): fullPath = os.path.abspath(os.path.join(dirpath, filename)) listtracks.append(fullPath) if not recursive: break return listtracks def remove_relatives(self, relative, filename): base = os.path.dirname(os.path.abspath(filename)) if not os.path.exists(relative): relative = os.path.join(base, relative) fullPath = relative return fullPath def populate(self, filename): # Create a playlist of the folder and all subfolders if os.path.isdir(filename): self.listtracks = self.populate_directory(filename) # Read the playlist file else: with open(filename, 'rb') as f: data = f.readlines() extension = os.path.splitext(filename)[1].lower() if extension == '.pls': self.listtracks = self.populate_pls(data) elif extension == '.m3u': self.listtracks = self.populate_m3u(data) else: raise # Ensure all paths are not relative to the playlist file for i in range(len(self.listtracks)): self.listtracks[i] = self.remove_relatives(self.listtracks[i], filename) # Handle the VoiceOverData text = os.path.splitext(os.path.basename(filename))[0] self["dbid"] = hashlib.md5(text).digest()[:8] #pylint: disable-msg=E1101 self.text_to_speech(text, self["dbid"], True) def construct(self, tracks): #pylint: disable-msg=W0221 self["total_length"] = 44 + (4 * len(self.listtracks)) self["number_of_songs"] = 0 chunks = "" for i in self.listtracks: path = self.ipod_to_path(i) position = -1 try: position = tracks.index(path) except: # Print an error if no track was found. # Empty playlists are handeled in the PlaylistHeader class. print "Error: Could not find track \"" + path + "\"." print "Maybe its an invalid FAT filesystem name. Please fix your playlist. Skipping track." if position > -1: chunks += struct.pack("I", position) self["number_of_songs"] += 1 self["number_of_nonaudio"] = self["number_of_songs"] output = Record.construct(self) return output + chunks class Shuffler(object): def __init__(self, path, voiceover=False, playlist_voiceover=False, rename=False, trackgain=0, auto_playlists=None): self.path = os.path.abspath(path) self.tracks = [] self.albums = [] self.artists = [] self.lists = [] self.tunessd = None self.voiceover = voiceover self.playlist_voiceover = playlist_voiceover self.rename = rename self.trackgain = trackgain self.auto_playlists = auto_playlists def initialize(self): # remove existing voiceover files (they are either useless or will be overwritten anyway) for dirname in ('iPod_Control/Speakable/Playlists', 'iPod_Control/Speakable/Tracks'): shutil.rmtree(os.path.join(self.path, dirname), ignore_errors=True) for dirname in ('iPod_Control/iTunes', 'iPod_Control/Music', 'iPod_Control/Speakable/Playlists', 'iPod_Control/Speakable/Tracks'): make_dir_if_absent(os.path.join(self.path, dirname)) def dump_state(self): print "Shuffle DB state" print "Tracks", self.tracks print "Albums", self.albums print "Artists", self.artists print "Playlists", self.lists def populate(self): self.tunessd = TunesSD(self) for (dirpath, dirnames, filenames) in os.walk(self.path): dirnames.sort() relpath = get_relpath(dirpath, self.path) # Ignore the speakable directory and any hidden directories if not is_path_prefix("iPod_Control/Speakable", relpath) and "/." not in dirpath: for filename in sorted(filenames, key = lambda x: x.lower()): fullPath = os.path.abspath(os.path.join(dirpath, filename)) if os.path.splitext(filename)[1].lower() in (".mp3", ".m4a", ".m4b", ".m4p", ".aa", ".wav"): self.tracks.append(fullPath) if os.path.splitext(filename)[1].lower() in (".pls", ".m3u"): self.lists.append(fullPath) # Create automatic playlists in music directory. # Ignore the (music) root and any hidden directories. if self.auto_playlists and "iPod_Control/Music/" in dirpath and "/." not in dirpath: # Only go to a specific depth. -1 is unlimted, 0 is ignored as there is already a master playlist. depth = dirpath[len(self.path) + len(os.path.sep):].count(os.path.sep) - 1 if self.auto_playlists < 0 or depth <= self.auto_playlists: self.lists.append(os.path.abspath(dirpath)) def write_database(self): with open(os.path.join(self.path, "iPod_Control", "iTunes", "iTunesSD"), "wb") as f: f.write(self.tunessd.construct()) # # Read all files from the directory # Construct the appropriate iTunesDB file # Construct the appropriate iTunesSD file # http://shuffle3db.wikispaces.com/iTunesSD3gen # Use SVOX pico2wave and RHVoice to produce voiceover data # def check_unicode(path): ret_flag = False # True if there is a recognizable file within this level for item in os.listdir(path): if os.path.isfile(os.path.join(path, item)): if os.path.splitext(item)[1].lower() in audio_ext+list_ext: ret_flag = True if raises_unicode_error(item): src = os.path.join(path, item) dest = os.path.join(path, hash_error_unicode(item)) + os.path.splitext(item)[1].lower() print 'Renaming %s -> %s' % (src, dest) os.rename(src, dest) else: ret_flag = (check_unicode(os.path.join(path, item)) or ret_flag) if ret_flag and raises_unicode_error(item): src = os.path.join(path, item) new_name = hash_error_unicode(item) dest = os.path.join(path, new_name) print 'Renaming %s -> %s' % (src, dest) os.rename(src, dest) return ret_flag def nonnegative_int(string): try: intval = int(string) except ValueError: raise argparse.ArgumentTypeError("'%s' must be an integer" % string) if intval < 0 or intval > 99: raise argparse.ArgumentTypeError("Track gain value should be in range 0-99") return intval def checkPathValidity(path): if not os.path.isdir(result.path): print "Error finding IPod directory. Maybe it is not connected or mounted?" sys.exit(1) if not os.access(result.path, os.W_OK): print 'Unable to get write permissions in the IPod directory' sys.exit(1) def handle_interrupt(signal, frame): print "Interrupt detected, exiting..." sys.exit(1) if __name__ == '__main__': signal.signal(signal.SIGINT, handle_interrupt) parser = argparse.ArgumentParser(description= 'Python script for building the Track and Playlist database ' 'for the newer gen IPod Shuffle. Version 1.3') parser.add_argument('--voiceover', action='store_true', help='Enable track voiceover feature') parser.add_argument('--playlist-voiceover', action='store_true', help='Enable playlist voiceover feature') parser.add_argument('--rename-unicode', action='store_true', help='Rename files causing unicode errors, will do minimal required renaming') parser.add_argument('--track-gain', type=nonnegative_int, default='0', help='Specify volume gain (0-99) for all tracks; ' '0 (default) means no gain and is usually fine; ' 'e.g. 60 is very loud even on minimal player volume') parser.add_argument('--auto-playlists', type=int, default=None, const=-1, nargs='?', help='Generate automatic playlists for each folder recursively inside ' '"IPod_Control/Music/". You can optionally limit the depth: ' '0=root, 1=artist, 2=album, n=subfoldername, default=-1 (No Limit).') parser.add_argument('path', help='Path to the IPod\'s root directory') result = parser.parse_args() checkPathValidity(result.path) if result.rename_unicode: check_unicode(result.path) if result.voiceover and not Text2Speech.check_support(): print "Error: Did not find any voiceover program. Voiceover disabled." result.voiceover = False shuffle = Shuffler(result.path, voiceover=result.voiceover, playlist_voiceover=result.playlist_voiceover, rename=result.rename_unicode, trackgain=result.track_gain, auto_playlists=result.auto_playlists) shuffle.initialize() shuffle.populate() shuffle.write_database()