#!/usr/bin/python # Strawberry Music Player # This file was part of Clementine. # # Strawberry is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # Strawberry is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with Strawberry. If not, see . from distutils import spawn import logging import os import re import subprocess import sys import traceback LOGGER = logging.getLogger('macdeploy') LIBRARY_SEARCH_PATH = ['/usr/local/lib', '/usr/local/opt/icu4c/lib'] FRAMEWORK_SEARCH_PATH = [ '/Library/Frameworks', os.path.join(os.environ['HOME'], 'Library/Frameworks'), '/Library/Frameworks/Sparkle.framework/Versions' ] QT_PLUGINS = [ 'platforms/libqcocoa.dylib', 'platforminputcontexts/libqtvirtualkeyboardplugin.dylib', 'styles/libqmacstyle.dylib', 'sqldrivers/libqsqlite.dylib', 'bearer/libqgenericbearer.dylib', 'iconengines/libqsvgicon.dylib', 'imageformats/libqgif.dylib', 'imageformats/libqicns.dylib', 'imageformats/libqico.dylib', 'imageformats/libqjpeg.dylib', 'imageformats/libqsvg.dylib', 'imageformats/libqtiff.dylib', 'printsupport/libcocoaprintersupport.dylib', 'virtualkeyboard/libqtvirtualkeyboard_hangul.dylib', 'virtualkeyboard/libqtvirtualkeyboard_openwnn.dylib', 'virtualkeyboard/libqtvirtualkeyboard_pinyin.dylib', 'virtualkeyboard/libqtvirtualkeyboard_tcime.dylib', 'virtualkeyboard/libqtvirtualkeyboard_thai.dylib', ] QT_PLUGINS_SEARCH_PATH = [ '/usr/local/opt/qt/plugins', ] GSTREAMER_SEARCH_PATH = [ '/usr/local/lib/gstreamer-1.0', '/usr/local/Cellar/gstreamer', ] GSTREAMER_PLUGINS = [ 'libgstapetag.dylib', 'libgstapp.dylib', 'libgstaudioconvert.dylib', 'libgstaudiofx.dylib', 'libgstaudiomixer.dylib', 'libgstaudioparsers.dylib', 'libgstaudiorate.dylib', 'libgstaudioresample.dylib', 'libgstaudiotestsrc.dylib', 'libgstaudiovisualizers.dylib', 'libgstauparse.dylib', 'libgstautoconvert.dylib', 'libgstautodetect.dylib', 'libgstcoreelements.dylib', 'libgstequalizer.dylib', 'libgstgio.dylib', 'libgsticydemux.dylib', 'libgstid3demux.dylib', 'libgstlevel.dylib', 'libgstosxaudio.dylib', 'libgstplayback.dylib', 'libgstrawparse.dylib', 'libgstrealmedia.dylib', 'libgstreplaygain.dylib', 'libgstsoup.dylib', 'libgstspectrum.dylib', 'libgsttypefindfunctions.dylib', 'libgstvolume.dylib', 'libgstxingmux.dylib', 'libgsttcp.dylib', 'libgstudp.dylib', 'libgstpbtypes.dylib', 'libgstrtp.dylib', 'libgstrtsp.dylib', 'libgstflac.dylib', 'libgstwavparse.dylib', 'libgstfaac.dylib', 'libgstfaad.dylib', 'libgstogg.dylib', 'libgstopus.dylib', 'libgstopusparse.dylib', 'libgstasf.dylib', 'libgstspeex.dylib', 'libgsttaglib.dylib', 'libgstvorbis.dylib', 'libgstisomp4.dylib', 'libgstlibav.dylib', 'libgstaiff.dylib', 'libgstlame.dylib', 'libgstmusepack.dylib', ] GIO_MODULES_SEARCH_PATH = ['/usr/local/lib/gio/modules',] INSTALL_NAME_TOOL_APPLE = 'install_name_tool' INSTALL_NAME_TOOL_CROSS = 'x86_64-apple-darwin-%s' % INSTALL_NAME_TOOL_APPLE INSTALL_NAME_TOOL = INSTALL_NAME_TOOL_CROSS if spawn.find_executable( INSTALL_NAME_TOOL_CROSS) else INSTALL_NAME_TOOL_APPLE OTOOL_APPLE = 'otool' OTOOL_CROSS = 'x86_64-apple-darwin-%s' % OTOOL_APPLE OTOOL = OTOOL_CROSS if spawn.find_executable(OTOOL_CROSS) else OTOOL_APPLE class Error(Exception): pass class CouldNotFindFrameworkError(Error): pass class CouldNotFindGioModuleError(Error): pass class CouldNotFindQtPluginError(Error): pass class CouldNotParseFrameworkNameError(Error): pass class InstallNameToolError(Error): pass class CouldNotFindGstreamerPluginError(Error): pass if len(sys.argv) < 2: print('Usage: %s ' % sys.argv[0]) bundle_dir = sys.argv[1] bundle_name = os.path.basename(bundle_dir).split('.')[0] commands = [] frameworks_dir = os.path.join(bundle_dir, 'Contents', 'Frameworks') commands.append(['mkdir', '-p', frameworks_dir]) resources_dir = os.path.join(bundle_dir, 'Contents', 'Resources') commands.append(['mkdir', '-p', resources_dir]) plugins_dir = os.path.join(bundle_dir, 'Contents', 'PlugIns') binary = os.path.join(bundle_dir, 'Contents', 'MacOS', bundle_name) fixed_libraries = set() fixed_frameworks = set() def GetBrokenLibraries(binary): #print("Checking libs for binary: %s" % binary) output = subprocess.Popen([OTOOL, '-L', binary], stdout=subprocess.PIPE).communicate()[0].decode('utf-8') broken_libs = {'frameworks': [], 'libs': []} for line in [x.split(' ')[0].lstrip() for x in output.split('\n')[1:]]: #print("Checking line: %s" % line) if not line: # skip empty lines continue if os.path.basename(binary) == os.path.basename(line): #print("mnope %s-%s" % (os.path.basename(binary), os.path.basename(line))) continue if re.match(r'^\s*/System/', line): #print("system framework: %s" % line) continue # System framework elif re.match(r'^\s*/usr/lib/', line): #print("unix style system lib: %s" % line) continue # unix style system library elif re.match(r'^\s*@executable_path', line) or re.match(r'^\s*@rpath', line) or re.match(r'^\s*@loader_path', line): # Potentially already fixed library if line.count('/') == 1: relative_path = os.path.join(*line.split('/')[1:]) if not os.path.exists(os.path.join(frameworks_dir, relative_path)): broken_libs['libs'].append(relative_path) elif line.count('/') == 2: relative_path = os.path.join(*line.split('/')[2:]) if not os.path.exists(os.path.join(frameworks_dir, relative_path)): broken_libs['libs'].append(relative_path) elif line.count('/') >= 3: relative_path = os.path.join(*line.split('/')[3:]) if not os.path.exists(os.path.join(frameworks_dir, relative_path)): broken_libs['frameworks'].append(relative_path) else: print("GetBrokenLibraries Error: %s" % line) elif re.search(r'\w+\.framework', line): #print("framework: %s" % line) broken_libs['frameworks'].append(line) else: broken_libs['libs'].append(line) return broken_libs def FindFramework(path): for search_path in FRAMEWORK_SEARCH_PATH: abs_path = os.path.join(search_path, path) if os.path.exists(abs_path): LOGGER.debug("Found framework '%s' in '%s'", path, search_path) return abs_path raise CouldNotFindFrameworkError(path) def FindLibrary(path): if os.path.exists(path): return path for search_path in LIBRARY_SEARCH_PATH: abs_path = os.path.join(search_path, path) if os.path.exists(abs_path): LOGGER.debug("Found library '%s' in '%s'", path, search_path) return abs_path else: # try harder---look for lib name in library folders newpath = os.path.join(search_path,os.path.basename(path)) if os.path.exists(newpath): return newpath raise CouldNotFindFrameworkError(path) def FixAllLibraries(broken_libs): for framework in broken_libs['frameworks']: FixFramework(framework) for lib in broken_libs['libs']: FixLibrary(lib) def FixFramework(path): if path in fixed_frameworks: return else: fixed_frameworks.add(path) abs_path = FindFramework(path) broken_libs = GetBrokenLibraries(abs_path) FixAllLibraries(broken_libs) new_path = CopyFramework(abs_path) id = os.sep.join(new_path.split(os.sep)[3:]) FixFrameworkId(new_path, id) for framework in broken_libs['frameworks']: FixFrameworkInstallPath(framework, new_path) for library in broken_libs['libs']: FixLibraryInstallPath(library, new_path) def FixLibrary(path): if path in fixed_libraries: return # Always bundle libraries provided by homebrew (/usr/local). if not re.match(r'^\s*/usr/local', path) and FindSystemLibrary(os.path.basename(path)) is not None: return fixed_libraries.add(path) abs_path = FindLibrary(path) if abs_path == "": print("Could not resolve %s, not fixing!" % path) return broken_libs = GetBrokenLibraries(abs_path) FixAllLibraries(broken_libs) new_path = CopyLibrary(abs_path) FixLibraryId(new_path) for framework in broken_libs['frameworks']: FixFrameworkInstallPath(framework, new_path) for library in broken_libs['libs']: FixLibraryInstallPath(library, new_path) def FixPlugin(abs_path, subdir): broken_libs = GetBrokenLibraries(abs_path) FixAllLibraries(broken_libs) new_path = CopyPlugin(abs_path, subdir) for framework in broken_libs['frameworks']: FixFrameworkInstallPath(framework, new_path) for library in broken_libs['libs']: FixLibraryInstallPath(library, new_path) def FixBinary(path): broken_libs = GetBrokenLibraries(path) FixAllLibraries(broken_libs) for framework in broken_libs['frameworks']: FixFrameworkInstallPath(framework, path) for library in broken_libs['libs']: FixLibraryInstallPath(library, path) def CopyLibrary(path): new_path = os.path.join(frameworks_dir, os.path.basename(path)) #args = ['cp', path, new_path] args = ['ditto', '--arch=i386', '--arch=x86_64', path, new_path] commands.append(args) commands.append(['chmod', '+w', new_path]) LOGGER.info("Copying library '%s'", path) return new_path def CopyPlugin(path, subdir): new_path = os.path.join(plugins_dir, subdir, os.path.basename(path)) args = ['mkdir', '-p', os.path.dirname(new_path)] commands.append(args) #args = ['cp', path, new_path] args = ['ditto', '--arch=i386', '--arch=x86_64', path, new_path] commands.append(args) commands.append(['chmod', '+w', new_path]) LOGGER.info("Copying plugin '%s'", path) return new_path def CopyFramework(path): parts = path.split(os.sep) for i, part in enumerate(parts): if re.match(r'\w+\.framework', part): full_path = os.path.join(frameworks_dir, *parts[i:-1]) framework_name = part.split(".framework")[0] break def CopyFramework(src_binary): while os.path.islink(src_binary): src_binary = os.path.realpath(src_binary) m = re.match(r'(.*/([^/]+)\.framework)/Versions/([^/]+)/.*', src_binary) if not m: raise CouldNotParseFrameworkNameError(src_binary) src_base = m.group(1) name = m.group(2) version = m.group(3) LOGGER.info('Copying framework %s version %s', name, version) dest_base = os.path.join(frameworks_dir, '%s.framework' % name) dest_dir = os.path.join(dest_base, 'Versions', version) dest_binary = os.path.join(dest_dir, name) commands.append(['mkdir', '-p', dest_dir]) commands.append(['cp', src_binary, dest_binary]) commands.append(['chmod', '+w', dest_binary]) # Copy special files from various places: # QtCore has Resources/qt_menu.nib (copy to app's Resources) # Sparkle has Resources/* # Qt* have Resources/Info.plist resources_src = os.path.join(src_base, 'Resources') menu_nib = os.path.join(resources_src, 'qt_menu.nib') if os.path.exists(menu_nib): LOGGER.info("Copying qt_menu.nib '%s'", menu_nib) commands.append(['cp', '-r', menu_nib, resources_dir]) elif os.path.exists(resources_src): LOGGER.info("Copying resources dir '%s'", resources_src) commands.append(['cp', '-r', resources_src, dest_dir]) info_plist = os.path.join(src_base, 'Contents', 'Info.plist') if os.path.exists(info_plist): LOGGER.info("Copying special file '%s'", info_plist) resources_dest = os.path.join(dest_dir, 'Resources') commands.append(['mkdir', resources_dest]) commands.append(['cp', '-r', info_plist, resources_dest]) # Create symlinks in the Framework to make it look like # https://developer.apple.com/library/mac/documentation/MacOSX/Conceptual/BPFrameworks/Concepts/FrameworkAnatomy.html commands.append([ 'ln', '-sf', 'Versions/Current/%s' % name, os.path.join(dest_base, name) ]) commands.append([ 'ln', '-sf', 'Versions/Current/Resources', os.path.join(dest_base, 'Resources') ]) commands.append( ['ln', '-sf', version, os.path.join(dest_base, 'Versions/Current')]) return dest_binary def FixId(path, library_name): id = '@executable_path/../Frameworks/%s' % library_name args = [INSTALL_NAME_TOOL, '-id', id, path] commands.append(args) def FixLibraryId(path): library_name = os.path.basename(path) FixId(path, library_name) def FixFrameworkId(path, id): FixId(path, id) def FixInstallPath(library_path, library, new_path): args = [INSTALL_NAME_TOOL, '-change', library_path, new_path, library] commands.append(args) def FindSystemLibrary(library_name): for path in ['/lib', '/usr/lib']: full_path = os.path.join(path, library_name) if os.path.exists(full_path): return full_path return None def FixLibraryInstallPath(library_path, library): system_library = FindSystemLibrary(os.path.basename(library_path)) if system_library is None or re.match(r'^\s*/usr/local', library_path): new_path = '@executable_path/../Frameworks/%s' % os.path.basename(library_path) FixInstallPath(library_path, library, new_path) else: FixInstallPath(library_path, library, system_library) def FixFrameworkInstallPath(library_path, library): parts = library_path.split(os.sep) full_path = "" for i, part in enumerate(parts): if re.match(r'\w+\.framework', part): full_path = os.path.join(*parts[i:]) break if full_path: new_path = '@executable_path/../Frameworks/%s' % full_path FixInstallPath(library_path, library, new_path) def FindQtPlugin(name): for path in QT_PLUGINS_SEARCH_PATH: if os.path.exists(path): if os.path.exists(os.path.join(path, name)): return os.path.join(path, name) raise CouldNotFindQtPluginError(name) def FindGstreamerPlugin(name): for path in GSTREAMER_SEARCH_PATH: if os.path.exists(path): for dir, dirs, files in os.walk(path): if name in files: return os.path.join(dir, name) raise CouldNotFindGstreamerPluginError(name) def FindGioModule(name): for path in GIO_MODULES_SEARCH_PATH: if os.path.exists(path): for dir, dirs, files in os.walk(path): if name in files: return os.path.join(dir, name) raise CouldNotFindGioModuleError(name) def main(): logging.basicConfig(filename='macdeploy.log', level=logging.DEBUG, format='%(asctime)s %(levelname)-8s %(message)s') FixBinary(binary) try: FixPlugin('strawberry-tagreader', '.') except: print('Failed to find blob: %s' % traceback.format_exc()) for plugin in GSTREAMER_PLUGINS: FixPlugin(FindGstreamerPlugin(plugin), 'gstreamer') FixPlugin(FindGstreamerPlugin('gst-plugin-scanner'), '.') FixPlugin(FindGioModule('libgiognutls.so'), 'gio-modules') #FixPlugin(FindGioModule('libgiognomeproxy.so'), 'gio-modules') #for plugin in QT_PLUGINS: #FixPlugin(FindQtPlugin(plugin), os.path.dirname(plugin)) if len(sys.argv) <= 2: print('Would run %d commands:' % len(commands)) for command in commands: print(' '.join(command)) #print('OK?') #raw_input() for command in commands: p = subprocess.Popen(command) os.waitpid(p.pid, 0) if __name__ == "__main__": main()