OLD | NEW |
(Empty) | |
| 1 # Copyright 2015 The Chromium Authors. All rights reserved. |
| 2 # Use of this source code is governed by a BSD-style license that can be |
| 3 # found in the LICENSE file. |
| 4 |
| 5 """Attempts to remove references to Conda installation directory to make the |
| 6 environment more relocatable in exchange for its modifiability. |
| 7 |
| 8 No new package can be installed into a butchered environment. |
| 9 |
| 10 Works great on Mac and Win, but doesn't really makes the environment prefix |
| 11 independent on Linux, since Linux binaries like to have prefix hardcoded. |
| 12 |
| 13 Everything seems to work even on Linux though, but watch out for weirdness. |
| 14 |
| 15 Helpful links: |
| 16 https://github.com/conda/conda-recipes |
| 17 http://conda.pydata.org/docs/building/meta-yaml.html#relocatable |
| 18 """ |
| 19 |
| 20 import ast |
| 21 import glob |
| 22 import os |
| 23 import shutil |
| 24 import sys |
| 25 |
| 26 |
| 27 IS_WIN = sys.platform in ('win32', 'cygwin') |
| 28 |
| 29 |
| 30 def main(conda_dir): |
| 31 butcher_bin(conda_dir) |
| 32 butcher_conda_meta(conda_dir) |
| 33 butcher_include(conda_dir) |
| 34 butcher_lib(conda_dir) |
| 35 butcher_pkgs(conda_dir) |
| 36 butcher_pyqt(conda_dir) |
| 37 |
| 38 |
| 39 def butcher_bin(conda_dir): |
| 40 if IS_WIN: |
| 41 bin_dir = os.path.join(conda_dir, 'Library', 'bin') |
| 42 else: |
| 43 bin_dir = os.path.join(conda_dir, 'bin') |
| 44 |
| 45 whitelist = [ |
| 46 'python', # relative symlink to python2.7, keep it, it's cute |
| 47 ] |
| 48 blacklist = [ |
| 49 'c_rehash', # perl script with references to local prefix |
| 50 'cairo-trace', # not needed |
| 51 'freetype-config', # we won't be linking to freetype |
| 52 'libpng16-config', # same |
| 53 'xml2-config', # same |
| 54 'pyuic4', # just use PyQt4/uic/pyuic.py directly |
| 55 'pyuic4.bat', # same |
| 56 'qt.conf', # Qt manages to work without it |
| 57 ] |
| 58 |
| 59 def is_naughty_shell_script(path): |
| 60 """True if script's shebang points to local prefix.""" |
| 61 if IS_WIN: |
| 62 return False |
| 63 with open(path, 'r') as f: |
| 64 return f.read(1024).startswith('#!' + bin_dir) |
| 65 |
| 66 # Remove meaningless symlinks and shell scripts that'll broke when relocated. |
| 67 # 'python' stays, it is all that matters. |
| 68 for p in os.listdir(bin_dir): |
| 69 full = os.path.join(bin_dir, p) |
| 70 kill_it = not (p in whitelist or os.path.isdir(full)) and ( |
| 71 p in blacklist or |
| 72 os.path.islink(full) or |
| 73 is_naughty_shell_script(full)) |
| 74 if kill_it: |
| 75 kill_file(full) |
| 76 |
| 77 |
| 78 def butcher_conda_meta(conda_dir): |
| 79 # 'conda-meta' contains history of local commands with full paths, as well |
| 80 # as ton of *.json files referencing local paths. We aren't going to install |
| 81 # any more conda packages, no need for meta files. |
| 82 kill_dir(os.path.join(conda_dir, 'conda-meta')) |
| 83 |
| 84 |
| 85 def butcher_include(conda_dir): |
| 86 # Header files? Where we're going we don't need header files. |
| 87 # But in case we do, a special care must be given to openssl/opensslconf.h, |
| 88 # it references local prefix. |
| 89 kill_dir(os.path.join(conda_dir, 'include')) |
| 90 |
| 91 |
| 92 def butcher_lib(conda_dir): |
| 93 if IS_WIN: |
| 94 lib = os.path.join(conda_dir, 'Library', 'lib') |
| 95 else: |
| 96 lib = os.path.join(conda_dir, 'lib') |
| 97 |
| 98 # We aren't going to build Tk\Tcl extensions, it's not 80s. |
| 99 kill_file(os.path.join(lib, 'tclConfig.sh')) |
| 100 kill_file(os.path.join(lib, 'tkConfig.sh')) |
| 101 |
| 102 # That's all for Win. |
| 103 if IS_WIN: |
| 104 return |
| 105 |
| 106 # We won't be using cmake. |
| 107 kill_dir(os.path.join(lib, 'cmake')) |
| 108 |
| 109 # We aren't going to build C stuff, kill libtool and pkg-config files. |
| 110 kill_glob(os.path.join(lib, '*.la')) |
| 111 kill_glob(os.path.join(lib, 'cairo', '*.la')) |
| 112 kill_dir(os.path.join(lib, 'pkgconfig')) |
| 113 |
| 114 # We aren't be building python extensions at all, in fact. |
| 115 kill_file(os.path.join(lib, 'python2.7', 'config', 'Makefile')) |
| 116 |
| 117 # This file looks important, let's patch it instead of removing. |
| 118 sysconf = os.path.join(lib, 'python2.7', '_sysconfigdata.py') |
| 119 if patch_file(sysconf, conda_dir): |
| 120 kill_file(os.path.join(lib, 'python2.7', '_sysconfigdata.pyc')) |
| 121 kill_file(os.path.join(lib, 'python2.7', '_sysconfigdata.pyo')) |
| 122 |
| 123 |
| 124 def butcher_pkgs(conda_dir): |
| 125 # pkgs contains unpacked Conda packages that act as source of hardlinks that |
| 126 # gets installed into actual prefix. Since its hard links, it's OK to remove |
| 127 # originals (thus "converting" hardlinks into regular files). |
| 128 kill_dir(os.path.join(conda_dir, 'pkgs')) # TODO: Windows? |
| 129 |
| 130 |
| 131 def butcher_pyqt(conda_dir): |
| 132 if IS_WIN: |
| 133 prefix = os.path.join(conda_dir, 'Library') |
| 134 else: |
| 135 prefix = conda_dir |
| 136 |
| 137 # We won't be using qmake. |
| 138 kill_glob(os.path.join(prefix, 'lib', '*.prl')) |
| 139 kill_dir(os.path.join(prefix, 'mkspecs')) |
| 140 |
| 141 # We don't care about Qt4 tests. |
| 142 kill_dir(os.path.join(prefix, 'tests', 'qt4')) |
| 143 |
| 144 if not IS_WIN: |
| 145 # We won't by using PyQt build system. |
| 146 kill_file( |
| 147 os.path.join( |
| 148 conda_dir, 'lib', 'python2.7', 'site-packages', 'sipconfig.py')) |
| 149 |
| 150 # This file looks important, let's patch it instead. |
| 151 patch_file( |
| 152 os.path.join( |
| 153 conda_dir, 'lib', 'python2.7', 'site-packages', |
| 154 'PyQt4', 'pyqtconfig.py'), |
| 155 conda_dir) |
| 156 else: |
| 157 kill_file(os.path.join(conda_dir, 'qt.conf')) |
| 158 |
| 159 |
| 160 ### |
| 161 |
| 162 |
| 163 def patch_file(path, prefix): |
| 164 """Replaces references to 'prefix' with '/opt/fake-python-prefix'.""" |
| 165 with open(path, 'rb') as f: |
| 166 blob = f.read() |
| 167 if prefix.endswith('/') or prefix.endswith('\\'): |
| 168 prefix = prefix[:-1] |
| 169 if IS_WIN: |
| 170 fake_prefix = 'C:\\fake-python-prefix' |
| 171 else: |
| 172 fake_prefix = '/opt/fake-python-prefix' |
| 173 modified = blob.replace(prefix, fake_prefix) |
| 174 if modified != blob: |
| 175 print 'Patching %s' % os.path.basename(path) |
| 176 with open(path, 'wb') as f: |
| 177 f.write(modified) |
| 178 return True |
| 179 return False |
| 180 |
| 181 |
| 182 def kill_file(path): |
| 183 if os.path.exists(path) or os.path.lexists(path): |
| 184 print 'Removing %s' % os.path.basename(path) |
| 185 os.remove(path) |
| 186 |
| 187 |
| 188 def kill_dir(path): |
| 189 if os.path.exists(path): |
| 190 print 'Removing %s directory' % os.path.basename(path) |
| 191 shutil.rmtree(path) |
| 192 |
| 193 |
| 194 def kill_glob(path): |
| 195 for p in glob.glob(path): |
| 196 kill_file(p) |
| 197 |
| 198 |
| 199 if __name__ == '__main__': |
| 200 sys.exit(main(os.path.abspath(sys.argv[1]))) |
OLD | NEW |