virtualenv/web/virtualenv.py
changeset 243 1f2840354865
child 272 1c774f7a0341
equal deleted inserted replaced
242:5c3f8477efcc 243:1f2840354865
       
     1 #!/usr/bin/env python
       
     2 """Create a "virtual" Python installation
       
     3 """
       
     4 
       
     5 # If you change the version here, change it in setup.py
       
     6 # and docs/conf.py as well.
       
     7 virtualenv_version = "1.7.1.2"
       
     8 
       
     9 import base64
       
    10 import sys
       
    11 import os
       
    12 import optparse
       
    13 import re
       
    14 import shutil
       
    15 import logging
       
    16 import tempfile
       
    17 import zlib
       
    18 import errno
       
    19 import distutils.sysconfig
       
    20 from distutils.util import strtobool
       
    21 
       
    22 try:
       
    23     import subprocess
       
    24 except ImportError:
       
    25     if sys.version_info <= (2, 3):
       
    26         print('ERROR: %s' % sys.exc_info()[1])
       
    27         print('ERROR: this script requires Python 2.4 or greater; or at least the subprocess module.')
       
    28         print('If you copy subprocess.py from a newer version of Python this script will probably work')
       
    29         sys.exit(101)
       
    30     else:
       
    31         raise
       
    32 try:
       
    33     set
       
    34 except NameError:
       
    35     from sets import Set as set
       
    36 try:
       
    37     basestring
       
    38 except NameError:
       
    39     basestring = str
       
    40 
       
    41 try:
       
    42     import ConfigParser
       
    43 except ImportError:
       
    44     import configparser as ConfigParser
       
    45 
       
    46 join = os.path.join
       
    47 py_version = 'python%s.%s' % (sys.version_info[0], sys.version_info[1])
       
    48 
       
    49 is_jython = sys.platform.startswith('java')
       
    50 is_pypy = hasattr(sys, 'pypy_version_info')
       
    51 is_win  = (sys.platform == 'win32')
       
    52 abiflags = getattr(sys, 'abiflags', '')
       
    53 
       
    54 user_dir = os.path.expanduser('~')
       
    55 if sys.platform == 'win32':
       
    56     user_dir = os.environ.get('APPDATA', user_dir)  # Use %APPDATA% for roaming
       
    57     default_storage_dir = os.path.join(user_dir, 'virtualenv')
       
    58 else:
       
    59     default_storage_dir = os.path.join(user_dir, '.virtualenv')
       
    60 default_config_file = os.path.join(default_storage_dir, 'virtualenv.ini')
       
    61 
       
    62 if is_pypy:
       
    63     expected_exe = 'pypy'
       
    64 elif is_jython:
       
    65     expected_exe = 'jython'
       
    66 else:
       
    67     expected_exe = 'python'
       
    68 
       
    69 
       
    70 REQUIRED_MODULES = ['os', 'posix', 'posixpath', 'nt', 'ntpath', 'genericpath',
       
    71                     'fnmatch', 'locale', 'encodings', 'codecs',
       
    72                     'stat', 'UserDict', 'readline', 'copy_reg', 'types',
       
    73                     're', 'sre', 'sre_parse', 'sre_constants', 'sre_compile',
       
    74                     'zlib']
       
    75 
       
    76 REQUIRED_FILES = ['lib-dynload', 'config']
       
    77 
       
    78 majver, minver = sys.version_info[:2]
       
    79 if majver == 2:
       
    80     if minver >= 6:
       
    81         REQUIRED_MODULES.extend(['warnings', 'linecache', '_abcoll', 'abc'])
       
    82     if minver >= 7:
       
    83         REQUIRED_MODULES.extend(['_weakrefset'])
       
    84     if minver <= 3:
       
    85         REQUIRED_MODULES.extend(['sets', '__future__'])
       
    86 elif majver == 3:
       
    87     # Some extra modules are needed for Python 3, but different ones
       
    88     # for different versions.
       
    89     REQUIRED_MODULES.extend(['_abcoll', 'warnings', 'linecache', 'abc', 'io',
       
    90                              '_weakrefset', 'copyreg', 'tempfile', 'random',
       
    91                              '__future__', 'collections', 'keyword', 'tarfile',
       
    92                              'shutil', 'struct', 'copy'])
       
    93     if minver >= 2:
       
    94         REQUIRED_FILES[-1] = 'config-%s' % majver
       
    95     if minver == 3:
       
    96         # The whole list of 3.3 modules is reproduced below - the current
       
    97         # uncommented ones are required for 3.3 as of now, but more may be
       
    98         # added as 3.3 development continues.
       
    99         REQUIRED_MODULES.extend([
       
   100             #"aifc",
       
   101             #"antigravity",
       
   102             #"argparse",
       
   103             #"ast",
       
   104             #"asynchat",
       
   105             #"asyncore",
       
   106             "base64",
       
   107             #"bdb",
       
   108             #"binhex",
       
   109             "bisect",
       
   110             #"calendar",
       
   111             #"cgi",
       
   112             #"cgitb",
       
   113             #"chunk",
       
   114             #"cmd",
       
   115             #"codeop",
       
   116             #"code",
       
   117             #"colorsys",
       
   118             #"_compat_pickle",
       
   119             #"compileall",
       
   120             #"concurrent",
       
   121             #"configparser",
       
   122             #"contextlib",
       
   123             #"cProfile",
       
   124             #"crypt",
       
   125             #"csv",
       
   126             #"ctypes",
       
   127             #"curses",
       
   128             #"datetime",
       
   129             #"dbm",
       
   130             #"decimal",
       
   131             #"difflib",
       
   132             #"dis",
       
   133             #"doctest",
       
   134             #"dummy_threading",
       
   135             "_dummy_thread",
       
   136             #"email",
       
   137             #"filecmp",
       
   138             #"fileinput",
       
   139             #"formatter",
       
   140             #"fractions",
       
   141             #"ftplib",
       
   142             #"functools",
       
   143             #"getopt",
       
   144             #"getpass",
       
   145             #"gettext",
       
   146             #"glob",
       
   147             #"gzip",
       
   148             "hashlib",
       
   149             "heapq",
       
   150             "hmac",
       
   151             #"html",
       
   152             #"http",
       
   153             #"idlelib",
       
   154             #"imaplib",
       
   155             #"imghdr",
       
   156             #"importlib",
       
   157             #"inspect",
       
   158             #"json",
       
   159             #"lib2to3",
       
   160             #"logging",
       
   161             #"macpath",
       
   162             #"macurl2path",
       
   163             #"mailbox",
       
   164             #"mailcap",
       
   165             #"_markupbase",
       
   166             #"mimetypes",
       
   167             #"modulefinder",
       
   168             #"multiprocessing",
       
   169             #"netrc",
       
   170             #"nntplib",
       
   171             #"nturl2path",
       
   172             #"numbers",
       
   173             #"opcode",
       
   174             #"optparse",
       
   175             #"os2emxpath",
       
   176             #"pdb",
       
   177             #"pickle",
       
   178             #"pickletools",
       
   179             #"pipes",
       
   180             #"pkgutil",
       
   181             #"platform",
       
   182             #"plat-linux2",
       
   183             #"plistlib",
       
   184             #"poplib",
       
   185             #"pprint",
       
   186             #"profile",
       
   187             #"pstats",
       
   188             #"pty",
       
   189             #"pyclbr",
       
   190             #"py_compile",
       
   191             #"pydoc_data",
       
   192             #"pydoc",
       
   193             #"_pyio",
       
   194             #"queue",
       
   195             #"quopri",
       
   196             "reprlib",
       
   197             "rlcompleter",
       
   198             #"runpy",
       
   199             #"sched",
       
   200             #"shelve",
       
   201             #"shlex",
       
   202             #"smtpd",
       
   203             #"smtplib",
       
   204             #"sndhdr",
       
   205             #"socket",
       
   206             #"socketserver",
       
   207             #"sqlite3",
       
   208             #"ssl",
       
   209             #"stringprep",
       
   210             #"string",
       
   211             #"_strptime",
       
   212             #"subprocess",
       
   213             #"sunau",
       
   214             #"symbol",
       
   215             #"symtable",
       
   216             #"sysconfig",
       
   217             #"tabnanny",
       
   218             #"telnetlib",
       
   219             #"test",
       
   220             #"textwrap",
       
   221             #"this",
       
   222             #"_threading_local",
       
   223             #"threading",
       
   224             #"timeit",
       
   225             #"tkinter",
       
   226             #"tokenize",
       
   227             #"token",
       
   228             #"traceback",
       
   229             #"trace",
       
   230             #"tty",
       
   231             #"turtledemo",
       
   232             #"turtle",
       
   233             #"unittest",
       
   234             #"urllib",
       
   235             #"uuid",
       
   236             #"uu",
       
   237             #"wave",
       
   238             "weakref",
       
   239             #"webbrowser",
       
   240             #"wsgiref",
       
   241             #"xdrlib",
       
   242             #"xml",
       
   243             #"xmlrpc",
       
   244             #"zipfile",
       
   245         ])
       
   246 
       
   247 if is_pypy:
       
   248     # these are needed to correctly display the exceptions that may happen
       
   249     # during the bootstrap
       
   250     REQUIRED_MODULES.extend(['traceback', 'linecache'])
       
   251 
       
   252 class Logger(object):
       
   253 
       
   254     """
       
   255     Logging object for use in command-line script.  Allows ranges of
       
   256     levels, to avoid some redundancy of displayed information.
       
   257     """
       
   258 
       
   259     DEBUG = logging.DEBUG
       
   260     INFO = logging.INFO
       
   261     NOTIFY = (logging.INFO+logging.WARN)/2
       
   262     WARN = WARNING = logging.WARN
       
   263     ERROR = logging.ERROR
       
   264     FATAL = logging.FATAL
       
   265 
       
   266     LEVELS = [DEBUG, INFO, NOTIFY, WARN, ERROR, FATAL]
       
   267 
       
   268     def __init__(self, consumers):
       
   269         self.consumers = consumers
       
   270         self.indent = 0
       
   271         self.in_progress = None
       
   272         self.in_progress_hanging = False
       
   273 
       
   274     def debug(self, msg, *args, **kw):
       
   275         self.log(self.DEBUG, msg, *args, **kw)
       
   276     def info(self, msg, *args, **kw):
       
   277         self.log(self.INFO, msg, *args, **kw)
       
   278     def notify(self, msg, *args, **kw):
       
   279         self.log(self.NOTIFY, msg, *args, **kw)
       
   280     def warn(self, msg, *args, **kw):
       
   281         self.log(self.WARN, msg, *args, **kw)
       
   282     def error(self, msg, *args, **kw):
       
   283         self.log(self.WARN, msg, *args, **kw)
       
   284     def fatal(self, msg, *args, **kw):
       
   285         self.log(self.FATAL, msg, *args, **kw)
       
   286     def log(self, level, msg, *args, **kw):
       
   287         if args:
       
   288             if kw:
       
   289                 raise TypeError(
       
   290                     "You may give positional or keyword arguments, not both")
       
   291         args = args or kw
       
   292         rendered = None
       
   293         for consumer_level, consumer in self.consumers:
       
   294             if self.level_matches(level, consumer_level):
       
   295                 if (self.in_progress_hanging
       
   296                     and consumer in (sys.stdout, sys.stderr)):
       
   297                     self.in_progress_hanging = False
       
   298                     sys.stdout.write('\n')
       
   299                     sys.stdout.flush()
       
   300                 if rendered is None:
       
   301                     if args:
       
   302                         rendered = msg % args
       
   303                     else:
       
   304                         rendered = msg
       
   305                     rendered = ' '*self.indent + rendered
       
   306                 if hasattr(consumer, 'write'):
       
   307                     consumer.write(rendered+'\n')
       
   308                 else:
       
   309                     consumer(rendered)
       
   310 
       
   311     def start_progress(self, msg):
       
   312         assert not self.in_progress, (
       
   313             "Tried to start_progress(%r) while in_progress %r"
       
   314             % (msg, self.in_progress))
       
   315         if self.level_matches(self.NOTIFY, self._stdout_level()):
       
   316             sys.stdout.write(msg)
       
   317             sys.stdout.flush()
       
   318             self.in_progress_hanging = True
       
   319         else:
       
   320             self.in_progress_hanging = False
       
   321         self.in_progress = msg
       
   322 
       
   323     def end_progress(self, msg='done.'):
       
   324         assert self.in_progress, (
       
   325             "Tried to end_progress without start_progress")
       
   326         if self.stdout_level_matches(self.NOTIFY):
       
   327             if not self.in_progress_hanging:
       
   328                 # Some message has been printed out since start_progress
       
   329                 sys.stdout.write('...' + self.in_progress + msg + '\n')
       
   330                 sys.stdout.flush()
       
   331             else:
       
   332                 sys.stdout.write(msg + '\n')
       
   333                 sys.stdout.flush()
       
   334         self.in_progress = None
       
   335         self.in_progress_hanging = False
       
   336 
       
   337     def show_progress(self):
       
   338         """If we are in a progress scope, and no log messages have been
       
   339         shown, write out another '.'"""
       
   340         if self.in_progress_hanging:
       
   341             sys.stdout.write('.')
       
   342             sys.stdout.flush()
       
   343 
       
   344     def stdout_level_matches(self, level):
       
   345         """Returns true if a message at this level will go to stdout"""
       
   346         return self.level_matches(level, self._stdout_level())
       
   347 
       
   348     def _stdout_level(self):
       
   349         """Returns the level that stdout runs at"""
       
   350         for level, consumer in self.consumers:
       
   351             if consumer is sys.stdout:
       
   352                 return level
       
   353         return self.FATAL
       
   354 
       
   355     def level_matches(self, level, consumer_level):
       
   356         """
       
   357         >>> l = Logger([])
       
   358         >>> l.level_matches(3, 4)
       
   359         False
       
   360         >>> l.level_matches(3, 2)
       
   361         True
       
   362         >>> l.level_matches(slice(None, 3), 3)
       
   363         False
       
   364         >>> l.level_matches(slice(None, 3), 2)
       
   365         True
       
   366         >>> l.level_matches(slice(1, 3), 1)
       
   367         True
       
   368         >>> l.level_matches(slice(2, 3), 1)
       
   369         False
       
   370         """
       
   371         if isinstance(level, slice):
       
   372             start, stop = level.start, level.stop
       
   373             if start is not None and start > consumer_level:
       
   374                 return False
       
   375             if stop is not None and stop <= consumer_level:
       
   376                 return False
       
   377             return True
       
   378         else:
       
   379             return level >= consumer_level
       
   380 
       
   381     #@classmethod
       
   382     def level_for_integer(cls, level):
       
   383         levels = cls.LEVELS
       
   384         if level < 0:
       
   385             return levels[0]
       
   386         if level >= len(levels):
       
   387             return levels[-1]
       
   388         return levels[level]
       
   389 
       
   390     level_for_integer = classmethod(level_for_integer)
       
   391 
       
   392 # create a silent logger just to prevent this from being undefined
       
   393 # will be overridden with requested verbosity main() is called.
       
   394 logger = Logger([(Logger.LEVELS[-1], sys.stdout)])
       
   395 
       
   396 def mkdir(path):
       
   397     if not os.path.exists(path):
       
   398         logger.info('Creating %s', path)
       
   399         os.makedirs(path)
       
   400     else:
       
   401         logger.info('Directory %s already exists', path)
       
   402 
       
   403 def copyfileordir(src, dest):
       
   404     if os.path.isdir(src):
       
   405         shutil.copytree(src, dest, True)
       
   406     else:
       
   407         shutil.copy2(src, dest)
       
   408 
       
   409 def copyfile(src, dest, symlink=True):
       
   410     if not os.path.exists(src):
       
   411         # Some bad symlink in the src
       
   412         logger.warn('Cannot find file %s (bad symlink)', src)
       
   413         return
       
   414     if os.path.exists(dest):
       
   415         logger.debug('File %s already exists', dest)
       
   416         return
       
   417     if not os.path.exists(os.path.dirname(dest)):
       
   418         logger.info('Creating parent directories for %s' % os.path.dirname(dest))
       
   419         os.makedirs(os.path.dirname(dest))
       
   420     if not os.path.islink(src):
       
   421         srcpath = os.path.abspath(src)
       
   422     else:
       
   423         srcpath = os.readlink(src)
       
   424     if symlink and hasattr(os, 'symlink') and not is_win:
       
   425         logger.info('Symlinking %s', dest)
       
   426         try:
       
   427             os.symlink(srcpath, dest)
       
   428         except (OSError, NotImplementedError):
       
   429             logger.info('Symlinking failed, copying to %s', dest)
       
   430             copyfileordir(src, dest)
       
   431     else:
       
   432         logger.info('Copying to %s', dest)
       
   433         copyfileordir(src, dest)
       
   434 
       
   435 def writefile(dest, content, overwrite=True):
       
   436     if not os.path.exists(dest):
       
   437         logger.info('Writing %s', dest)
       
   438         f = open(dest, 'wb')
       
   439         f.write(content.encode('utf-8'))
       
   440         f.close()
       
   441         return
       
   442     else:
       
   443         f = open(dest, 'rb')
       
   444         c = f.read()
       
   445         f.close()
       
   446         if c != content:
       
   447             if not overwrite:
       
   448                 logger.notify('File %s exists with different content; not overwriting', dest)
       
   449                 return
       
   450             logger.notify('Overwriting %s with new content', dest)
       
   451             f = open(dest, 'wb')
       
   452             f.write(content.encode('utf-8'))
       
   453             f.close()
       
   454         else:
       
   455             logger.info('Content %s already in place', dest)
       
   456 
       
   457 def rmtree(dir):
       
   458     if os.path.exists(dir):
       
   459         logger.notify('Deleting tree %s', dir)
       
   460         shutil.rmtree(dir)
       
   461     else:
       
   462         logger.info('Do not need to delete %s; already gone', dir)
       
   463 
       
   464 def make_exe(fn):
       
   465     if hasattr(os, 'chmod'):
       
   466         oldmode = os.stat(fn).st_mode & 0xFFF # 0o7777
       
   467         newmode = (oldmode | 0x16D) & 0xFFF # 0o555, 0o7777
       
   468         os.chmod(fn, newmode)
       
   469         logger.info('Changed mode of %s to %s', fn, oct(newmode))
       
   470 
       
   471 def _find_file(filename, dirs):
       
   472     for dir in reversed(dirs):
       
   473         if os.path.exists(join(dir, filename)):
       
   474             return join(dir, filename)
       
   475     return filename
       
   476 
       
   477 def _install_req(py_executable, unzip=False, distribute=False,
       
   478                  search_dirs=None, never_download=False):
       
   479 
       
   480     if search_dirs is None:
       
   481         search_dirs = file_search_dirs()
       
   482 
       
   483     if not distribute:
       
   484         setup_fn = 'setuptools-0.6c11-py%s.egg' % sys.version[:3]
       
   485         project_name = 'setuptools'
       
   486         bootstrap_script = EZ_SETUP_PY
       
   487         source = None
       
   488     else:
       
   489         setup_fn = None
       
   490         source = 'distribute-0.6.24.tar.gz'
       
   491         project_name = 'distribute'
       
   492         bootstrap_script = DISTRIBUTE_SETUP_PY
       
   493 
       
   494     if setup_fn is not None:
       
   495         setup_fn = _find_file(setup_fn, search_dirs)
       
   496 
       
   497     if source is not None:
       
   498         source = _find_file(source, search_dirs)
       
   499 
       
   500     if is_jython and os._name == 'nt':
       
   501         # Jython's .bat sys.executable can't handle a command line
       
   502         # argument with newlines
       
   503         fd, ez_setup = tempfile.mkstemp('.py')
       
   504         os.write(fd, bootstrap_script)
       
   505         os.close(fd)
       
   506         cmd = [py_executable, ez_setup]
       
   507     else:
       
   508         cmd = [py_executable, '-c', bootstrap_script]
       
   509     if unzip:
       
   510         cmd.append('--always-unzip')
       
   511     env = {}
       
   512     remove_from_env = []
       
   513     if logger.stdout_level_matches(logger.DEBUG):
       
   514         cmd.append('-v')
       
   515 
       
   516     old_chdir = os.getcwd()
       
   517     if setup_fn is not None and os.path.exists(setup_fn):
       
   518         logger.info('Using existing %s egg: %s' % (project_name, setup_fn))
       
   519         cmd.append(setup_fn)
       
   520         if os.environ.get('PYTHONPATH'):
       
   521             env['PYTHONPATH'] = setup_fn + os.path.pathsep + os.environ['PYTHONPATH']
       
   522         else:
       
   523             env['PYTHONPATH'] = setup_fn
       
   524     else:
       
   525         # the source is found, let's chdir
       
   526         if source is not None and os.path.exists(source):
       
   527             logger.info('Using existing %s egg: %s' % (project_name, source))
       
   528             os.chdir(os.path.dirname(source))
       
   529             # in this case, we want to be sure that PYTHONPATH is unset (not
       
   530             # just empty, really unset), else CPython tries to import the
       
   531             # site.py that it's in virtualenv_support
       
   532             remove_from_env.append('PYTHONPATH')
       
   533         else:
       
   534             if never_download:
       
   535                 logger.fatal("Can't find any local distributions of %s to install "
       
   536                              "and --never-download is set.  Either re-run virtualenv "
       
   537                              "without the --never-download option, or place a %s "
       
   538                              "distribution (%s) in one of these "
       
   539                              "locations: %r" % (project_name, project_name,
       
   540                                                 setup_fn or source,
       
   541                                                 search_dirs))
       
   542                 sys.exit(1)
       
   543 
       
   544             logger.info('No %s egg found; downloading' % project_name)
       
   545         cmd.extend(['--always-copy', '-U', project_name])
       
   546     logger.start_progress('Installing %s...' % project_name)
       
   547     logger.indent += 2
       
   548     cwd = None
       
   549     if project_name == 'distribute':
       
   550         env['DONT_PATCH_SETUPTOOLS'] = 'true'
       
   551 
       
   552     def _filter_ez_setup(line):
       
   553         return filter_ez_setup(line, project_name)
       
   554 
       
   555     if not os.access(os.getcwd(), os.W_OK):
       
   556         cwd = tempfile.mkdtemp()
       
   557         if source is not None and os.path.exists(source):
       
   558             # the current working dir is hostile, let's copy the
       
   559             # tarball to a temp dir
       
   560             target = os.path.join(cwd, os.path.split(source)[-1])
       
   561             shutil.copy(source, target)
       
   562     try:
       
   563         call_subprocess(cmd, show_stdout=False,
       
   564                         filter_stdout=_filter_ez_setup,
       
   565                         extra_env=env,
       
   566                         remove_from_env=remove_from_env,
       
   567                         cwd=cwd)
       
   568     finally:
       
   569         logger.indent -= 2
       
   570         logger.end_progress()
       
   571         if os.getcwd() != old_chdir:
       
   572             os.chdir(old_chdir)
       
   573         if is_jython and os._name == 'nt':
       
   574             os.remove(ez_setup)
       
   575 
       
   576 def file_search_dirs():
       
   577     here = os.path.dirname(os.path.abspath(__file__))
       
   578     dirs = ['.', here,
       
   579             join(here, 'virtualenv_support')]
       
   580     if os.path.splitext(os.path.dirname(__file__))[0] != 'virtualenv':
       
   581         # Probably some boot script; just in case virtualenv is installed...
       
   582         try:
       
   583             import virtualenv
       
   584         except ImportError:
       
   585             pass
       
   586         else:
       
   587             dirs.append(os.path.join(os.path.dirname(virtualenv.__file__), 'virtualenv_support'))
       
   588     return [d for d in dirs if os.path.isdir(d)]
       
   589 
       
   590 def install_setuptools(py_executable, unzip=False,
       
   591                        search_dirs=None, never_download=False):
       
   592     _install_req(py_executable, unzip,
       
   593                  search_dirs=search_dirs, never_download=never_download)
       
   594 
       
   595 def install_distribute(py_executable, unzip=False,
       
   596                        search_dirs=None, never_download=False):
       
   597     _install_req(py_executable, unzip, distribute=True,
       
   598                  search_dirs=search_dirs, never_download=never_download)
       
   599 
       
   600 _pip_re = re.compile(r'^pip-.*(zip|tar.gz|tar.bz2|tgz|tbz)$', re.I)
       
   601 def install_pip(py_executable, search_dirs=None, never_download=False):
       
   602     if search_dirs is None:
       
   603         search_dirs = file_search_dirs()
       
   604 
       
   605     filenames = []
       
   606     for dir in search_dirs:
       
   607         filenames.extend([join(dir, fn) for fn in os.listdir(dir)
       
   608                           if _pip_re.search(fn)])
       
   609     filenames = [(os.path.basename(filename).lower(), i, filename) for i, filename in enumerate(filenames)]
       
   610     filenames.sort()
       
   611     filenames = [filename for basename, i, filename in filenames]
       
   612     if not filenames:
       
   613         filename = 'pip'
       
   614     else:
       
   615         filename = filenames[-1]
       
   616     easy_install_script = 'easy_install'
       
   617     if sys.platform == 'win32':
       
   618         easy_install_script = 'easy_install-script.py'
       
   619     cmd = [join(os.path.dirname(py_executable), easy_install_script), filename]
       
   620     if sys.platform == 'win32':
       
   621         cmd.insert(0, py_executable)
       
   622     if filename == 'pip':
       
   623         if never_download:
       
   624             logger.fatal("Can't find any local distributions of pip to install "
       
   625                          "and --never-download is set.  Either re-run virtualenv "
       
   626                          "without the --never-download option, or place a pip "
       
   627                          "source distribution (zip/tar.gz/tar.bz2) in one of these "
       
   628                          "locations: %r" % search_dirs)
       
   629             sys.exit(1)
       
   630         logger.info('Installing pip from network...')
       
   631     else:
       
   632         logger.info('Installing existing %s distribution: %s' % (
       
   633                 os.path.basename(filename), filename))
       
   634     logger.start_progress('Installing pip...')
       
   635     logger.indent += 2
       
   636     def _filter_setup(line):
       
   637         return filter_ez_setup(line, 'pip')
       
   638     try:
       
   639         call_subprocess(cmd, show_stdout=False,
       
   640                         filter_stdout=_filter_setup)
       
   641     finally:
       
   642         logger.indent -= 2
       
   643         logger.end_progress()
       
   644 
       
   645 def filter_ez_setup(line, project_name='setuptools'):
       
   646     if not line.strip():
       
   647         return Logger.DEBUG
       
   648     if project_name == 'distribute':
       
   649         for prefix in ('Extracting', 'Now working', 'Installing', 'Before',
       
   650                        'Scanning', 'Setuptools', 'Egg', 'Already',
       
   651                        'running', 'writing', 'reading', 'installing',
       
   652                        'creating', 'copying', 'byte-compiling', 'removing',
       
   653                        'Processing'):
       
   654             if line.startswith(prefix):
       
   655                 return Logger.DEBUG
       
   656         return Logger.DEBUG
       
   657     for prefix in ['Reading ', 'Best match', 'Processing setuptools',
       
   658                    'Copying setuptools', 'Adding setuptools',
       
   659                    'Installing ', 'Installed ']:
       
   660         if line.startswith(prefix):
       
   661             return Logger.DEBUG
       
   662     return Logger.INFO
       
   663 
       
   664 
       
   665 class UpdatingDefaultsHelpFormatter(optparse.IndentedHelpFormatter):
       
   666     """
       
   667     Custom help formatter for use in ConfigOptionParser that updates
       
   668     the defaults before expanding them, allowing them to show up correctly
       
   669     in the help listing
       
   670     """
       
   671     def expand_default(self, option):
       
   672         if self.parser is not None:
       
   673             self.parser.update_defaults(self.parser.defaults)
       
   674         return optparse.IndentedHelpFormatter.expand_default(self, option)
       
   675 
       
   676 
       
   677 class ConfigOptionParser(optparse.OptionParser):
       
   678     """
       
   679     Custom option parser which updates its defaults by by checking the
       
   680     configuration files and environmental variables
       
   681     """
       
   682     def __init__(self, *args, **kwargs):
       
   683         self.config = ConfigParser.RawConfigParser()
       
   684         self.files = self.get_config_files()
       
   685         self.config.read(self.files)
       
   686         optparse.OptionParser.__init__(self, *args, **kwargs)
       
   687 
       
   688     def get_config_files(self):
       
   689         config_file = os.environ.get('VIRTUALENV_CONFIG_FILE', False)
       
   690         if config_file and os.path.exists(config_file):
       
   691             return [config_file]
       
   692         return [default_config_file]
       
   693 
       
   694     def update_defaults(self, defaults):
       
   695         """
       
   696         Updates the given defaults with values from the config files and
       
   697         the environ. Does a little special handling for certain types of
       
   698         options (lists).
       
   699         """
       
   700         # Then go and look for the other sources of configuration:
       
   701         config = {}
       
   702         # 1. config files
       
   703         config.update(dict(self.get_config_section('virtualenv')))
       
   704         # 2. environmental variables
       
   705         config.update(dict(self.get_environ_vars()))
       
   706         # Then set the options with those values
       
   707         for key, val in config.items():
       
   708             key = key.replace('_', '-')
       
   709             if not key.startswith('--'):
       
   710                 key = '--%s' % key  # only prefer long opts
       
   711             option = self.get_option(key)
       
   712             if option is not None:
       
   713                 # ignore empty values
       
   714                 if not val:
       
   715                     continue
       
   716                 # handle multiline configs
       
   717                 if option.action == 'append':
       
   718                     val = val.split()
       
   719                 else:
       
   720                     option.nargs = 1
       
   721                 if option.action in ('store_true', 'store_false', 'count'):
       
   722                     val = strtobool(val)
       
   723                 try:
       
   724                     val = option.convert_value(key, val)
       
   725                 except optparse.OptionValueError:
       
   726                     e = sys.exc_info()[1]
       
   727                     print("An error occured during configuration: %s" % e)
       
   728                     sys.exit(3)
       
   729                 defaults[option.dest] = val
       
   730         return defaults
       
   731 
       
   732     def get_config_section(self, name):
       
   733         """
       
   734         Get a section of a configuration
       
   735         """
       
   736         if self.config.has_section(name):
       
   737             return self.config.items(name)
       
   738         return []
       
   739 
       
   740     def get_environ_vars(self, prefix='VIRTUALENV_'):
       
   741         """
       
   742         Returns a generator with all environmental vars with prefix VIRTUALENV
       
   743         """
       
   744         for key, val in os.environ.items():
       
   745             if key.startswith(prefix):
       
   746                 yield (key.replace(prefix, '').lower(), val)
       
   747 
       
   748     def get_default_values(self):
       
   749         """
       
   750         Overridding to make updating the defaults after instantiation of
       
   751         the option parser possible, update_defaults() does the dirty work.
       
   752         """
       
   753         if not self.process_default_values:
       
   754             # Old, pre-Optik 1.5 behaviour.
       
   755             return optparse.Values(self.defaults)
       
   756 
       
   757         defaults = self.update_defaults(self.defaults.copy()) # ours
       
   758         for option in self._get_all_options():
       
   759             default = defaults.get(option.dest)
       
   760             if isinstance(default, basestring):
       
   761                 opt_str = option.get_opt_string()
       
   762                 defaults[option.dest] = option.check_value(opt_str, default)
       
   763         return optparse.Values(defaults)
       
   764 
       
   765 
       
   766 def main():
       
   767     parser = ConfigOptionParser(
       
   768         version=virtualenv_version,
       
   769         usage="%prog [OPTIONS] DEST_DIR",
       
   770         formatter=UpdatingDefaultsHelpFormatter())
       
   771 
       
   772     parser.add_option(
       
   773         '-v', '--verbose',
       
   774         action='count',
       
   775         dest='verbose',
       
   776         default=0,
       
   777         help="Increase verbosity")
       
   778 
       
   779     parser.add_option(
       
   780         '-q', '--quiet',
       
   781         action='count',
       
   782         dest='quiet',
       
   783         default=0,
       
   784         help='Decrease verbosity')
       
   785 
       
   786     parser.add_option(
       
   787         '-p', '--python',
       
   788         dest='python',
       
   789         metavar='PYTHON_EXE',
       
   790         help='The Python interpreter to use, e.g., --python=python2.5 will use the python2.5 '
       
   791         'interpreter to create the new environment.  The default is the interpreter that '
       
   792         'virtualenv was installed with (%s)' % sys.executable)
       
   793 
       
   794     parser.add_option(
       
   795         '--clear',
       
   796         dest='clear',
       
   797         action='store_true',
       
   798         help="Clear out the non-root install and start from scratch")
       
   799 
       
   800     parser.add_option(
       
   801         '--no-site-packages',
       
   802         dest='no_site_packages',
       
   803         action='store_true',
       
   804         help="Don't give access to the global site-packages dir to the "
       
   805              "virtual environment")
       
   806 
       
   807     parser.add_option(
       
   808         '--system-site-packages',
       
   809         dest='system_site_packages',
       
   810         action='store_true',
       
   811         help="Give access to the global site-packages dir to the "
       
   812              "virtual environment")
       
   813 
       
   814     parser.add_option(
       
   815         '--unzip-setuptools',
       
   816         dest='unzip_setuptools',
       
   817         action='store_true',
       
   818         help="Unzip Setuptools or Distribute when installing it")
       
   819 
       
   820     parser.add_option(
       
   821         '--relocatable',
       
   822         dest='relocatable',
       
   823         action='store_true',
       
   824         help='Make an EXISTING virtualenv environment relocatable.  '
       
   825         'This fixes up scripts and makes all .pth files relative')
       
   826 
       
   827     parser.add_option(
       
   828         '--distribute',
       
   829         dest='use_distribute',
       
   830         action='store_true',
       
   831         help='Use Distribute instead of Setuptools. Set environ variable '
       
   832         'VIRTUALENV_DISTRIBUTE to make it the default ')
       
   833 
       
   834     default_search_dirs = file_search_dirs()
       
   835     parser.add_option(
       
   836         '--extra-search-dir',
       
   837         dest="search_dirs",
       
   838         action="append",
       
   839         default=default_search_dirs,
       
   840         help="Directory to look for setuptools/distribute/pip distributions in. "
       
   841         "You can add any number of additional --extra-search-dir paths.")
       
   842 
       
   843     parser.add_option(
       
   844         '--never-download',
       
   845         dest="never_download",
       
   846         action="store_true",
       
   847         help="Never download anything from the network.  Instead, virtualenv will fail "
       
   848         "if local distributions of setuptools/distribute/pip are not present.")
       
   849 
       
   850     parser.add_option(
       
   851         '--prompt=',
       
   852         dest='prompt',
       
   853         help='Provides an alternative prompt prefix for this environment')
       
   854 
       
   855     if 'extend_parser' in globals():
       
   856         extend_parser(parser)
       
   857 
       
   858     options, args = parser.parse_args()
       
   859 
       
   860     global logger
       
   861 
       
   862     if 'adjust_options' in globals():
       
   863         adjust_options(options, args)
       
   864 
       
   865     verbosity = options.verbose - options.quiet
       
   866     logger = Logger([(Logger.level_for_integer(2-verbosity), sys.stdout)])
       
   867 
       
   868     if options.python and not os.environ.get('VIRTUALENV_INTERPRETER_RUNNING'):
       
   869         env = os.environ.copy()
       
   870         interpreter = resolve_interpreter(options.python)
       
   871         if interpreter == sys.executable:
       
   872             logger.warn('Already using interpreter %s' % interpreter)
       
   873         else:
       
   874             logger.notify('Running virtualenv with interpreter %s' % interpreter)
       
   875             env['VIRTUALENV_INTERPRETER_RUNNING'] = 'true'
       
   876             file = __file__
       
   877             if file.endswith('.pyc'):
       
   878                 file = file[:-1]
       
   879             popen = subprocess.Popen([interpreter, file] + sys.argv[1:], env=env)
       
   880             raise SystemExit(popen.wait())
       
   881 
       
   882     # Force --distribute on Python 3, since setuptools is not available.
       
   883     if majver > 2:
       
   884         options.use_distribute = True
       
   885 
       
   886     if os.environ.get('PYTHONDONTWRITEBYTECODE') and not options.use_distribute:
       
   887         print(
       
   888             "The PYTHONDONTWRITEBYTECODE environment variable is "
       
   889             "not compatible with setuptools. Either use --distribute "
       
   890             "or unset PYTHONDONTWRITEBYTECODE.")
       
   891         sys.exit(2)
       
   892     if not args:
       
   893         print('You must provide a DEST_DIR')
       
   894         parser.print_help()
       
   895         sys.exit(2)
       
   896     if len(args) > 1:
       
   897         print('There must be only one argument: DEST_DIR (you gave %s)' % (
       
   898             ' '.join(args)))
       
   899         parser.print_help()
       
   900         sys.exit(2)
       
   901 
       
   902     home_dir = args[0]
       
   903 
       
   904     if os.environ.get('WORKING_ENV'):
       
   905         logger.fatal('ERROR: you cannot run virtualenv while in a workingenv')
       
   906         logger.fatal('Please deactivate your workingenv, then re-run this script')
       
   907         sys.exit(3)
       
   908 
       
   909     if 'PYTHONHOME' in os.environ:
       
   910         logger.warn('PYTHONHOME is set.  You *must* activate the virtualenv before using it')
       
   911         del os.environ['PYTHONHOME']
       
   912 
       
   913     if options.relocatable:
       
   914         make_environment_relocatable(home_dir)
       
   915         return
       
   916 
       
   917     if options.no_site_packages:
       
   918         logger.warn('The --no-site-packages flag is deprecated; it is now '
       
   919                     'the default behavior.')
       
   920 
       
   921     create_environment(home_dir,
       
   922                        site_packages=options.system_site_packages,
       
   923                        clear=options.clear,
       
   924                        unzip_setuptools=options.unzip_setuptools,
       
   925                        use_distribute=options.use_distribute,
       
   926                        prompt=options.prompt,
       
   927                        search_dirs=options.search_dirs,
       
   928                        never_download=options.never_download)
       
   929     if 'after_install' in globals():
       
   930         after_install(options, home_dir)
       
   931 
       
   932 def call_subprocess(cmd, show_stdout=True,
       
   933                     filter_stdout=None, cwd=None,
       
   934                     raise_on_returncode=True, extra_env=None,
       
   935                     remove_from_env=None):
       
   936     cmd_parts = []
       
   937     for part in cmd:
       
   938         if len(part) > 45:
       
   939             part = part[:20]+"..."+part[-20:]
       
   940         if ' ' in part or '\n' in part or '"' in part or "'" in part:
       
   941             part = '"%s"' % part.replace('"', '\\"')
       
   942         if hasattr(part, 'decode'):
       
   943             try:
       
   944                 part = part.decode(sys.getdefaultencoding())
       
   945             except UnicodeDecodeError:
       
   946                 part = part.decode(sys.getfilesystemencoding())
       
   947         cmd_parts.append(part)
       
   948     cmd_desc = ' '.join(cmd_parts)
       
   949     if show_stdout:
       
   950         stdout = None
       
   951     else:
       
   952         stdout = subprocess.PIPE
       
   953     logger.debug("Running command %s" % cmd_desc)
       
   954     if extra_env or remove_from_env:
       
   955         env = os.environ.copy()
       
   956         if extra_env:
       
   957             env.update(extra_env)
       
   958         if remove_from_env:
       
   959             for varname in remove_from_env:
       
   960                 env.pop(varname, None)
       
   961     else:
       
   962         env = None
       
   963     try:
       
   964         proc = subprocess.Popen(
       
   965             cmd, stderr=subprocess.STDOUT, stdin=None, stdout=stdout,
       
   966             cwd=cwd, env=env)
       
   967     except Exception:
       
   968         e = sys.exc_info()[1]
       
   969         logger.fatal(
       
   970             "Error %s while executing command %s" % (e, cmd_desc))
       
   971         raise
       
   972     all_output = []
       
   973     if stdout is not None:
       
   974         stdout = proc.stdout
       
   975         encoding = sys.getdefaultencoding()
       
   976         fs_encoding = sys.getfilesystemencoding()
       
   977         while 1:
       
   978             line = stdout.readline()
       
   979             try:
       
   980                 line = line.decode(encoding)
       
   981             except UnicodeDecodeError:
       
   982                 line = line.decode(fs_encoding)
       
   983             if not line:
       
   984                 break
       
   985             line = line.rstrip()
       
   986             all_output.append(line)
       
   987             if filter_stdout:
       
   988                 level = filter_stdout(line)
       
   989                 if isinstance(level, tuple):
       
   990                     level, line = level
       
   991                 logger.log(level, line)
       
   992                 if not logger.stdout_level_matches(level):
       
   993                     logger.show_progress()
       
   994             else:
       
   995                 logger.info(line)
       
   996     else:
       
   997         proc.communicate()
       
   998     proc.wait()
       
   999     if proc.returncode:
       
  1000         if raise_on_returncode:
       
  1001             if all_output:
       
  1002                 logger.notify('Complete output from command %s:' % cmd_desc)
       
  1003                 logger.notify('\n'.join(all_output) + '\n----------------------------------------')
       
  1004             raise OSError(
       
  1005                 "Command %s failed with error code %s"
       
  1006                 % (cmd_desc, proc.returncode))
       
  1007         else:
       
  1008             logger.warn(
       
  1009                 "Command %s had error code %s"
       
  1010                 % (cmd_desc, proc.returncode))
       
  1011 
       
  1012 
       
  1013 def create_environment(home_dir, site_packages=False, clear=False,
       
  1014                        unzip_setuptools=False, use_distribute=False,
       
  1015                        prompt=None, search_dirs=None, never_download=False):
       
  1016     """
       
  1017     Creates a new environment in ``home_dir``.
       
  1018 
       
  1019     If ``site_packages`` is true, then the global ``site-packages/``
       
  1020     directory will be on the path.
       
  1021 
       
  1022     If ``clear`` is true (default False) then the environment will
       
  1023     first be cleared.
       
  1024     """
       
  1025     home_dir, lib_dir, inc_dir, bin_dir = path_locations(home_dir)
       
  1026 
       
  1027     py_executable = os.path.abspath(install_python(
       
  1028         home_dir, lib_dir, inc_dir, bin_dir,
       
  1029         site_packages=site_packages, clear=clear))
       
  1030 
       
  1031     install_distutils(home_dir)
       
  1032 
       
  1033     # use_distribute also is True if VIRTUALENV_DISTRIBUTE env var is set
       
  1034     # we also check VIRTUALENV_USE_DISTRIBUTE for backwards compatibility
       
  1035     if use_distribute or os.environ.get('VIRTUALENV_USE_DISTRIBUTE'):
       
  1036         install_distribute(py_executable, unzip=unzip_setuptools,
       
  1037                            search_dirs=search_dirs, never_download=never_download)
       
  1038     else:
       
  1039         install_setuptools(py_executable, unzip=unzip_setuptools,
       
  1040                            search_dirs=search_dirs, never_download=never_download)
       
  1041 
       
  1042     install_pip(py_executable, search_dirs=search_dirs, never_download=never_download)
       
  1043 
       
  1044     install_activate(home_dir, bin_dir, prompt)
       
  1045 
       
  1046 def path_locations(home_dir):
       
  1047     """Return the path locations for the environment (where libraries are,
       
  1048     where scripts go, etc)"""
       
  1049     # XXX: We'd use distutils.sysconfig.get_python_inc/lib but its
       
  1050     # prefix arg is broken: http://bugs.python.org/issue3386
       
  1051     if sys.platform == 'win32':
       
  1052         # Windows has lots of problems with executables with spaces in
       
  1053         # the name; this function will remove them (using the ~1
       
  1054         # format):
       
  1055         mkdir(home_dir)
       
  1056         if ' ' in home_dir:
       
  1057             try:
       
  1058                 import win32api
       
  1059             except ImportError:
       
  1060                 print('Error: the path "%s" has a space in it' % home_dir)
       
  1061                 print('To handle these kinds of paths, the win32api module must be installed:')
       
  1062                 print('  http://sourceforge.net/projects/pywin32/')
       
  1063                 sys.exit(3)
       
  1064             home_dir = win32api.GetShortPathName(home_dir)
       
  1065         lib_dir = join(home_dir, 'Lib')
       
  1066         inc_dir = join(home_dir, 'Include')
       
  1067         bin_dir = join(home_dir, 'Scripts')
       
  1068     elif is_jython:
       
  1069         lib_dir = join(home_dir, 'Lib')
       
  1070         inc_dir = join(home_dir, 'Include')
       
  1071         bin_dir = join(home_dir, 'bin')
       
  1072     elif is_pypy:
       
  1073         lib_dir = home_dir
       
  1074         inc_dir = join(home_dir, 'include')
       
  1075         bin_dir = join(home_dir, 'bin')
       
  1076     else:
       
  1077         lib_dir = join(home_dir, 'lib', py_version)
       
  1078         inc_dir = join(home_dir, 'include', py_version + abiflags)
       
  1079         bin_dir = join(home_dir, 'bin')
       
  1080     return home_dir, lib_dir, inc_dir, bin_dir
       
  1081 
       
  1082 
       
  1083 def change_prefix(filename, dst_prefix):
       
  1084     prefixes = [sys.prefix]
       
  1085 
       
  1086     if sys.platform == "darwin":
       
  1087         prefixes.extend((
       
  1088             os.path.join("/Library/Python", sys.version[:3], "site-packages"),
       
  1089             os.path.join(sys.prefix, "Extras", "lib", "python"),
       
  1090             os.path.join("~", "Library", "Python", sys.version[:3], "site-packages")))
       
  1091 
       
  1092     if hasattr(sys, 'real_prefix'):
       
  1093         prefixes.append(sys.real_prefix)
       
  1094     prefixes = list(map(os.path.abspath, prefixes))
       
  1095     filename = os.path.abspath(filename)
       
  1096     for src_prefix in prefixes:
       
  1097         if filename.startswith(src_prefix):
       
  1098             _, relpath = filename.split(src_prefix, 1)
       
  1099             assert relpath[0] == os.sep
       
  1100             relpath = relpath[1:]
       
  1101             return join(dst_prefix, relpath)
       
  1102     assert False, "Filename %s does not start with any of these prefixes: %s" % \
       
  1103         (filename, prefixes)
       
  1104 
       
  1105 def copy_required_modules(dst_prefix):
       
  1106     import imp
       
  1107     # If we are running under -p, we need to remove the current
       
  1108     # directory from sys.path temporarily here, so that we
       
  1109     # definitely get the modules from the site directory of
       
  1110     # the interpreter we are running under, not the one
       
  1111     # virtualenv.py is installed under (which might lead to py2/py3
       
  1112     # incompatibility issues)
       
  1113     _prev_sys_path = sys.path
       
  1114     if os.environ.get('VIRTUALENV_INTERPRETER_RUNNING'):
       
  1115         sys.path = sys.path[1:]
       
  1116     try:
       
  1117         for modname in REQUIRED_MODULES:
       
  1118             if modname in sys.builtin_module_names:
       
  1119                 logger.info("Ignoring built-in bootstrap module: %s" % modname)
       
  1120                 continue
       
  1121             try:
       
  1122                 f, filename, _ = imp.find_module(modname)
       
  1123             except ImportError:
       
  1124                 logger.info("Cannot import bootstrap module: %s" % modname)
       
  1125             else:
       
  1126                 if f is not None:
       
  1127                     f.close()
       
  1128                 dst_filename = change_prefix(filename, dst_prefix)
       
  1129                 copyfile(filename, dst_filename)
       
  1130                 if filename.endswith('.pyc'):
       
  1131                     pyfile = filename[:-1]
       
  1132                     if os.path.exists(pyfile):
       
  1133                         copyfile(pyfile, dst_filename[:-1])
       
  1134     finally:
       
  1135         sys.path = _prev_sys_path
       
  1136 
       
  1137 def install_python(home_dir, lib_dir, inc_dir, bin_dir, site_packages, clear):
       
  1138     """Install just the base environment, no distutils patches etc"""
       
  1139     if sys.executable.startswith(bin_dir):
       
  1140         print('Please use the *system* python to run this script')
       
  1141         return
       
  1142 
       
  1143     if clear:
       
  1144         rmtree(lib_dir)
       
  1145         ## FIXME: why not delete it?
       
  1146         ## Maybe it should delete everything with #!/path/to/venv/python in it
       
  1147         logger.notify('Not deleting %s', bin_dir)
       
  1148 
       
  1149     if hasattr(sys, 'real_prefix'):
       
  1150         logger.notify('Using real prefix %r' % sys.real_prefix)
       
  1151         prefix = sys.real_prefix
       
  1152     else:
       
  1153         prefix = sys.prefix
       
  1154     mkdir(lib_dir)
       
  1155     fix_lib64(lib_dir)
       
  1156     stdlib_dirs = [os.path.dirname(os.__file__)]
       
  1157     if sys.platform == 'win32':
       
  1158         stdlib_dirs.append(join(os.path.dirname(stdlib_dirs[0]), 'DLLs'))
       
  1159     elif sys.platform == 'darwin':
       
  1160         stdlib_dirs.append(join(stdlib_dirs[0], 'site-packages'))
       
  1161     if hasattr(os, 'symlink'):
       
  1162         logger.info('Symlinking Python bootstrap modules')
       
  1163     else:
       
  1164         logger.info('Copying Python bootstrap modules')
       
  1165     logger.indent += 2
       
  1166     try:
       
  1167         # copy required files...
       
  1168         for stdlib_dir in stdlib_dirs:
       
  1169             if not os.path.isdir(stdlib_dir):
       
  1170                 continue
       
  1171             for fn in os.listdir(stdlib_dir):
       
  1172                 bn = os.path.splitext(fn)[0]
       
  1173                 if fn != 'site-packages' and bn in REQUIRED_FILES:
       
  1174                     copyfile(join(stdlib_dir, fn), join(lib_dir, fn))
       
  1175         # ...and modules
       
  1176         copy_required_modules(home_dir)
       
  1177     finally:
       
  1178         logger.indent -= 2
       
  1179     mkdir(join(lib_dir, 'site-packages'))
       
  1180     import site
       
  1181     site_filename = site.__file__
       
  1182     if site_filename.endswith('.pyc'):
       
  1183         site_filename = site_filename[:-1]
       
  1184     elif site_filename.endswith('$py.class'):
       
  1185         site_filename = site_filename.replace('$py.class', '.py')
       
  1186     site_filename_dst = change_prefix(site_filename, home_dir)
       
  1187     site_dir = os.path.dirname(site_filename_dst)
       
  1188     writefile(site_filename_dst, SITE_PY)
       
  1189     writefile(join(site_dir, 'orig-prefix.txt'), prefix)
       
  1190     site_packages_filename = join(site_dir, 'no-global-site-packages.txt')
       
  1191     if not site_packages:
       
  1192         writefile(site_packages_filename, '')
       
  1193     else:
       
  1194         if os.path.exists(site_packages_filename):
       
  1195             logger.info('Deleting %s' % site_packages_filename)
       
  1196             os.unlink(site_packages_filename)
       
  1197 
       
  1198     if is_pypy or is_win:
       
  1199         stdinc_dir = join(prefix, 'include')
       
  1200     else:
       
  1201         stdinc_dir = join(prefix, 'include', py_version + abiflags)
       
  1202     if os.path.exists(stdinc_dir):
       
  1203         copyfile(stdinc_dir, inc_dir)
       
  1204     else:
       
  1205         logger.debug('No include dir %s' % stdinc_dir)
       
  1206 
       
  1207     # pypy never uses exec_prefix, just ignore it
       
  1208     if sys.exec_prefix != prefix and not is_pypy:
       
  1209         if sys.platform == 'win32':
       
  1210             exec_dir = join(sys.exec_prefix, 'lib')
       
  1211         elif is_jython:
       
  1212             exec_dir = join(sys.exec_prefix, 'Lib')
       
  1213         else:
       
  1214             exec_dir = join(sys.exec_prefix, 'lib', py_version)
       
  1215         for fn in os.listdir(exec_dir):
       
  1216             copyfile(join(exec_dir, fn), join(lib_dir, fn))
       
  1217 
       
  1218     if is_jython:
       
  1219         # Jython has either jython-dev.jar and javalib/ dir, or just
       
  1220         # jython.jar
       
  1221         for name in 'jython-dev.jar', 'javalib', 'jython.jar':
       
  1222             src = join(prefix, name)
       
  1223             if os.path.exists(src):
       
  1224                 copyfile(src, join(home_dir, name))
       
  1225         # XXX: registry should always exist after Jython 2.5rc1
       
  1226         src = join(prefix, 'registry')
       
  1227         if os.path.exists(src):
       
  1228             copyfile(src, join(home_dir, 'registry'), symlink=False)
       
  1229         copyfile(join(prefix, 'cachedir'), join(home_dir, 'cachedir'),
       
  1230                  symlink=False)
       
  1231 
       
  1232     mkdir(bin_dir)
       
  1233     py_executable = join(bin_dir, os.path.basename(sys.executable))
       
  1234     if 'Python.framework' in prefix:
       
  1235         if re.search(r'/Python(?:-32|-64)*$', py_executable):
       
  1236             # The name of the python executable is not quite what
       
  1237             # we want, rename it.
       
  1238             py_executable = os.path.join(
       
  1239                     os.path.dirname(py_executable), 'python')
       
  1240 
       
  1241     logger.notify('New %s executable in %s', expected_exe, py_executable)
       
  1242     pcbuild_dir = os.path.dirname(sys.executable)
       
  1243     pyd_pth = os.path.join(lib_dir, 'site-packages', 'virtualenv_builddir_pyd.pth')
       
  1244     if is_win and os.path.exists(os.path.join(pcbuild_dir, 'build.bat')):
       
  1245         logger.notify('Detected python running from build directory %s', pcbuild_dir)
       
  1246         logger.notify('Writing .pth file linking to build directory for *.pyd files')
       
  1247         writefile(pyd_pth, pcbuild_dir)
       
  1248     else:
       
  1249         pcbuild_dir = None
       
  1250         if os.path.exists(pyd_pth):
       
  1251             logger.info('Deleting %s (not Windows env or not build directory python)' % pyd_pth)
       
  1252             os.unlink(pyd_pth)
       
  1253         
       
  1254     if sys.executable != py_executable:
       
  1255         ## FIXME: could I just hard link?
       
  1256         executable = sys.executable
       
  1257         if sys.platform == 'cygwin' and os.path.exists(executable + '.exe'):
       
  1258             # Cygwin misreports sys.executable sometimes
       
  1259             executable += '.exe'
       
  1260             py_executable += '.exe'
       
  1261             logger.info('Executable actually exists in %s' % executable)
       
  1262         shutil.copyfile(executable, py_executable)
       
  1263         make_exe(py_executable)
       
  1264         if sys.platform == 'win32' or sys.platform == 'cygwin':
       
  1265             pythonw = os.path.join(os.path.dirname(sys.executable), 'pythonw.exe')
       
  1266             if os.path.exists(pythonw):
       
  1267                 logger.info('Also created pythonw.exe')
       
  1268                 shutil.copyfile(pythonw, os.path.join(os.path.dirname(py_executable), 'pythonw.exe'))
       
  1269             python_d = os.path.join(os.path.dirname(sys.executable), 'python_d.exe')
       
  1270             python_d_dest = os.path.join(os.path.dirname(py_executable), 'python_d.exe')
       
  1271             if os.path.exists(python_d):
       
  1272                 logger.info('Also created python_d.exe')
       
  1273                 shutil.copyfile(python_d, python_d_dest)
       
  1274             elif os.path.exists(python_d_dest):
       
  1275                 logger.info('Removed python_d.exe as it is no longer at the source')
       
  1276                 os.unlink(python_d_dest)
       
  1277             # we need to copy the DLL to enforce that windows will load the correct one.
       
  1278             # may not exist if we are cygwin.
       
  1279             py_executable_dll = 'python%s%s.dll' % (
       
  1280                 sys.version_info[0], sys.version_info[1])
       
  1281             py_executable_dll_d = 'python%s%s_d.dll' % (
       
  1282                 sys.version_info[0], sys.version_info[1])
       
  1283             pythondll = os.path.join(os.path.dirname(sys.executable), py_executable_dll)
       
  1284             pythondll_d = os.path.join(os.path.dirname(sys.executable), py_executable_dll_d)
       
  1285             pythondll_d_dest = os.path.join(os.path.dirname(py_executable), py_executable_dll_d)
       
  1286             if os.path.exists(pythondll):
       
  1287                 logger.info('Also created %s' % py_executable_dll)
       
  1288                 shutil.copyfile(pythondll, os.path.join(os.path.dirname(py_executable), py_executable_dll))
       
  1289             if os.path.exists(pythondll_d):
       
  1290                 logger.info('Also created %s' % py_executable_dll_d)
       
  1291                 shutil.copyfile(pythondll_d, pythondll_d_dest)
       
  1292             elif os.path.exists(pythondll_d_dest):
       
  1293                 logger.info('Removed %s as the source does not exist' % pythondll_d_dest)
       
  1294                 os.unlink(pythondll_d_dest)
       
  1295         if is_pypy:
       
  1296             # make a symlink python --> pypy-c
       
  1297             python_executable = os.path.join(os.path.dirname(py_executable), 'python')
       
  1298             logger.info('Also created executable %s' % python_executable)
       
  1299             copyfile(py_executable, python_executable)
       
  1300 
       
  1301     if os.path.splitext(os.path.basename(py_executable))[0] != expected_exe:
       
  1302         secondary_exe = os.path.join(os.path.dirname(py_executable),
       
  1303                                      expected_exe)
       
  1304         py_executable_ext = os.path.splitext(py_executable)[1]
       
  1305         if py_executable_ext == '.exe':
       
  1306             # python2.4 gives an extension of '.4' :P
       
  1307             secondary_exe += py_executable_ext
       
  1308         if os.path.exists(secondary_exe):
       
  1309             logger.warn('Not overwriting existing %s script %s (you must use %s)'
       
  1310                         % (expected_exe, secondary_exe, py_executable))
       
  1311         else:
       
  1312             logger.notify('Also creating executable in %s' % secondary_exe)
       
  1313             shutil.copyfile(sys.executable, secondary_exe)
       
  1314             make_exe(secondary_exe)
       
  1315 
       
  1316     if '.framework' in prefix:
       
  1317         if 'Python.framework' in prefix:
       
  1318             logger.debug('MacOSX Python framework detected')
       
  1319             # Make sure we use the the embedded interpreter inside
       
  1320             # the framework, even if sys.executable points to
       
  1321             # the stub executable in ${sys.prefix}/bin
       
  1322             # See http://groups.google.com/group/python-virtualenv/
       
  1323             #                              browse_thread/thread/17cab2f85da75951
       
  1324             original_python = os.path.join(
       
  1325                 prefix, 'Resources/Python.app/Contents/MacOS/Python')
       
  1326         if 'EPD' in prefix:
       
  1327             logger.debug('EPD framework detected')
       
  1328             original_python = os.path.join(prefix, 'bin/python')
       
  1329         shutil.copy(original_python, py_executable)
       
  1330 
       
  1331         # Copy the framework's dylib into the virtual
       
  1332         # environment
       
  1333         virtual_lib = os.path.join(home_dir, '.Python')
       
  1334 
       
  1335         if os.path.exists(virtual_lib):
       
  1336             os.unlink(virtual_lib)
       
  1337         copyfile(
       
  1338             os.path.join(prefix, 'Python'),
       
  1339             virtual_lib)
       
  1340 
       
  1341         # And then change the install_name of the copied python executable
       
  1342         try:
       
  1343             call_subprocess(
       
  1344                 ["install_name_tool", "-change",
       
  1345                  os.path.join(prefix, 'Python'),
       
  1346                  '@executable_path/../.Python',
       
  1347                  py_executable])
       
  1348         except:
       
  1349             logger.fatal(
       
  1350                 "Could not call install_name_tool -- you must have Apple's development tools installed")
       
  1351             raise
       
  1352 
       
  1353         # Some tools depend on pythonX.Y being present
       
  1354         py_executable_version = '%s.%s' % (
       
  1355             sys.version_info[0], sys.version_info[1])
       
  1356         if not py_executable.endswith(py_executable_version):
       
  1357             # symlinking pythonX.Y > python
       
  1358             pth = py_executable + '%s.%s' % (
       
  1359                     sys.version_info[0], sys.version_info[1])
       
  1360             if os.path.exists(pth):
       
  1361                 os.unlink(pth)
       
  1362             os.symlink('python', pth)
       
  1363         else:
       
  1364             # reverse symlinking python -> pythonX.Y (with --python)
       
  1365             pth = join(bin_dir, 'python')
       
  1366             if os.path.exists(pth):
       
  1367                 os.unlink(pth)
       
  1368             os.symlink(os.path.basename(py_executable), pth)
       
  1369 
       
  1370     if sys.platform == 'win32' and ' ' in py_executable:
       
  1371         # There's a bug with subprocess on Windows when using a first
       
  1372         # argument that has a space in it.  Instead we have to quote
       
  1373         # the value:
       
  1374         py_executable = '"%s"' % py_executable
       
  1375     cmd = [py_executable, '-c', """
       
  1376 import sys
       
  1377 prefix = sys.prefix
       
  1378 if sys.version_info[0] == 3:
       
  1379     prefix = prefix.encode('utf8')
       
  1380 if hasattr(sys.stdout, 'detach'):
       
  1381     sys.stdout = sys.stdout.detach()
       
  1382 elif hasattr(sys.stdout, 'buffer'):
       
  1383     sys.stdout = sys.stdout.buffer
       
  1384 sys.stdout.write(prefix)
       
  1385 """]
       
  1386     logger.info('Testing executable with %s %s "%s"' % tuple(cmd))
       
  1387     try:
       
  1388         proc = subprocess.Popen(cmd,
       
  1389                             stdout=subprocess.PIPE)
       
  1390         proc_stdout, proc_stderr = proc.communicate()
       
  1391     except OSError:
       
  1392         e = sys.exc_info()[1]
       
  1393         if e.errno == errno.EACCES:
       
  1394             logger.fatal('ERROR: The executable %s could not be run: %s' % (py_executable, e))
       
  1395             sys.exit(100)
       
  1396         else:
       
  1397           raise e
       
  1398 
       
  1399     proc_stdout = proc_stdout.strip().decode("utf-8")
       
  1400     proc_stdout = os.path.normcase(os.path.abspath(proc_stdout))
       
  1401     norm_home_dir = os.path.normcase(os.path.abspath(home_dir))
       
  1402     if hasattr(norm_home_dir, 'decode'):
       
  1403         norm_home_dir = norm_home_dir.decode(sys.getfilesystemencoding())
       
  1404     if proc_stdout != norm_home_dir:
       
  1405         logger.fatal(
       
  1406             'ERROR: The executable %s is not functioning' % py_executable)
       
  1407         logger.fatal(
       
  1408             'ERROR: It thinks sys.prefix is %r (should be %r)'
       
  1409             % (proc_stdout, norm_home_dir))
       
  1410         logger.fatal(
       
  1411             'ERROR: virtualenv is not compatible with this system or executable')
       
  1412         if sys.platform == 'win32':
       
  1413             logger.fatal(
       
  1414                 'Note: some Windows users have reported this error when they '
       
  1415                 'installed Python for "Only this user" or have multiple '
       
  1416                 'versions of Python installed. Copying the appropriate '
       
  1417                 'PythonXX.dll to the virtualenv Scripts/ directory may fix '
       
  1418                 'this problem.')
       
  1419         sys.exit(100)
       
  1420     else:
       
  1421         logger.info('Got sys.prefix result: %r' % proc_stdout)
       
  1422 
       
  1423     pydistutils = os.path.expanduser('~/.pydistutils.cfg')
       
  1424     if os.path.exists(pydistutils):
       
  1425         logger.notify('Please make sure you remove any previous custom paths from '
       
  1426                       'your %s file.' % pydistutils)
       
  1427     ## FIXME: really this should be calculated earlier
       
  1428 
       
  1429     fix_local_scheme(home_dir)
       
  1430 
       
  1431     return py_executable
       
  1432 
       
  1433 def install_activate(home_dir, bin_dir, prompt=None):
       
  1434     home_dir = os.path.abspath(home_dir)
       
  1435     if sys.platform == 'win32' or is_jython and os._name == 'nt':
       
  1436         files = {
       
  1437             'activate.bat': ACTIVATE_BAT,
       
  1438             'deactivate.bat': DEACTIVATE_BAT,
       
  1439             'activate.ps1': ACTIVATE_PS,
       
  1440         }
       
  1441 
       
  1442         # MSYS needs paths of the form /c/path/to/file
       
  1443         drive, tail = os.path.splitdrive(home_dir.replace(os.sep, '/'))
       
  1444         home_dir_msys = (drive and "/%s%s" or "%s%s") % (drive[:1], tail)
       
  1445 
       
  1446         # Run-time conditional enables (basic) Cygwin compatibility
       
  1447         home_dir_sh = ("""$(if [ "$OSTYPE" "==" "cygwin" ]; then cygpath -u '%s'; else echo '%s'; fi;)""" % 
       
  1448                        (home_dir, home_dir_msys))
       
  1449         files['activate'] = ACTIVATE_SH.replace('__VIRTUAL_ENV__', home_dir_sh)
       
  1450 
       
  1451     else:
       
  1452         files = {'activate': ACTIVATE_SH}
       
  1453 
       
  1454         # suppling activate.fish in addition to, not instead of, the
       
  1455         # bash script support.
       
  1456         files['activate.fish'] = ACTIVATE_FISH
       
  1457 
       
  1458         # same for csh/tcsh support...
       
  1459         files['activate.csh'] = ACTIVATE_CSH
       
  1460 
       
  1461     files['activate_this.py'] = ACTIVATE_THIS
       
  1462     if hasattr(home_dir, 'decode'):
       
  1463         home_dir = home_dir.decode(sys.getfilesystemencoding())
       
  1464     vname = os.path.basename(home_dir)
       
  1465     for name, content in files.items():
       
  1466         content = content.replace('__VIRTUAL_PROMPT__', prompt or '')
       
  1467         content = content.replace('__VIRTUAL_WINPROMPT__', prompt or '(%s)' % vname)
       
  1468         content = content.replace('__VIRTUAL_ENV__', home_dir)
       
  1469         content = content.replace('__VIRTUAL_NAME__', vname)
       
  1470         content = content.replace('__BIN_NAME__', os.path.basename(bin_dir))
       
  1471         writefile(os.path.join(bin_dir, name), content)
       
  1472 
       
  1473 def install_distutils(home_dir):
       
  1474     distutils_path = change_prefix(distutils.__path__[0], home_dir)
       
  1475     mkdir(distutils_path)
       
  1476     ## FIXME: maybe this prefix setting should only be put in place if
       
  1477     ## there's a local distutils.cfg with a prefix setting?
       
  1478     home_dir = os.path.abspath(home_dir)
       
  1479     ## FIXME: this is breaking things, removing for now:
       
  1480     #distutils_cfg = DISTUTILS_CFG + "\n[install]\nprefix=%s\n" % home_dir
       
  1481     writefile(os.path.join(distutils_path, '__init__.py'), DISTUTILS_INIT)
       
  1482     writefile(os.path.join(distutils_path, 'distutils.cfg'), DISTUTILS_CFG, overwrite=False)
       
  1483 
       
  1484 def fix_local_scheme(home_dir):
       
  1485     """
       
  1486     Platforms that use the "posix_local" install scheme (like Ubuntu with
       
  1487     Python 2.7) need to be given an additional "local" location, sigh.
       
  1488     """
       
  1489     try:
       
  1490         import sysconfig
       
  1491     except ImportError:
       
  1492         pass
       
  1493     else:
       
  1494         if sysconfig._get_default_scheme() == 'posix_local':
       
  1495             local_path = os.path.join(home_dir, 'local')
       
  1496             if not os.path.exists(local_path):
       
  1497                 os.mkdir(local_path)
       
  1498                 for subdir_name in os.listdir(home_dir):
       
  1499                     if subdir_name == 'local':
       
  1500                         continue
       
  1501                     os.symlink(os.path.abspath(os.path.join(home_dir, subdir_name)), \
       
  1502                                                             os.path.join(local_path, subdir_name))
       
  1503 
       
  1504 def fix_lib64(lib_dir):
       
  1505     """
       
  1506     Some platforms (particularly Gentoo on x64) put things in lib64/pythonX.Y
       
  1507     instead of lib/pythonX.Y.  If this is such a platform we'll just create a
       
  1508     symlink so lib64 points to lib
       
  1509     """
       
  1510     if [p for p in distutils.sysconfig.get_config_vars().values()
       
  1511         if isinstance(p, basestring) and 'lib64' in p]:
       
  1512         logger.debug('This system uses lib64; symlinking lib64 to lib')
       
  1513         assert os.path.basename(lib_dir) == 'python%s' % sys.version[:3], (
       
  1514             "Unexpected python lib dir: %r" % lib_dir)
       
  1515         lib_parent = os.path.dirname(lib_dir)
       
  1516         assert os.path.basename(lib_parent) == 'lib', (
       
  1517             "Unexpected parent dir: %r" % lib_parent)
       
  1518         copyfile(lib_parent, os.path.join(os.path.dirname(lib_parent), 'lib64'))
       
  1519 
       
  1520 def resolve_interpreter(exe):
       
  1521     """
       
  1522     If the executable given isn't an absolute path, search $PATH for the interpreter
       
  1523     """
       
  1524     if os.path.abspath(exe) != exe:
       
  1525         paths = os.environ.get('PATH', '').split(os.pathsep)
       
  1526         for path in paths:
       
  1527             if os.path.exists(os.path.join(path, exe)):
       
  1528                 exe = os.path.join(path, exe)
       
  1529                 break
       
  1530     if not os.path.exists(exe):
       
  1531         logger.fatal('The executable %s (from --python=%s) does not exist' % (exe, exe))
       
  1532         raise SystemExit(3)
       
  1533     if not is_executable(exe):
       
  1534         logger.fatal('The executable %s (from --python=%s) is not executable' % (exe, exe))
       
  1535         raise SystemExit(3)
       
  1536     return exe
       
  1537 
       
  1538 def is_executable(exe):
       
  1539     """Checks a file is executable"""
       
  1540     return os.access(exe, os.X_OK)
       
  1541 
       
  1542 ############################################################
       
  1543 ## Relocating the environment:
       
  1544 
       
  1545 def make_environment_relocatable(home_dir):
       
  1546     """
       
  1547     Makes the already-existing environment use relative paths, and takes out
       
  1548     the #!-based environment selection in scripts.
       
  1549     """
       
  1550     home_dir, lib_dir, inc_dir, bin_dir = path_locations(home_dir)
       
  1551     activate_this = os.path.join(bin_dir, 'activate_this.py')
       
  1552     if not os.path.exists(activate_this):
       
  1553         logger.fatal(
       
  1554             'The environment doesn\'t have a file %s -- please re-run virtualenv '
       
  1555             'on this environment to update it' % activate_this)
       
  1556     fixup_scripts(home_dir)
       
  1557     fixup_pth_and_egg_link(home_dir)
       
  1558     ## FIXME: need to fix up distutils.cfg
       
  1559 
       
  1560 OK_ABS_SCRIPTS = ['python', 'python%s' % sys.version[:3],
       
  1561                   'activate', 'activate.bat', 'activate_this.py']
       
  1562 
       
  1563 def fixup_scripts(home_dir):
       
  1564     # This is what we expect at the top of scripts:
       
  1565     shebang = '#!%s/bin/python' % os.path.normcase(os.path.abspath(home_dir))
       
  1566     # This is what we'll put:
       
  1567     new_shebang = '#!/usr/bin/env python%s' % sys.version[:3]
       
  1568     activate = "import os; activate_this=os.path.join(os.path.dirname(os.path.realpath(__file__)), 'activate_this.py'); execfile(activate_this, dict(__file__=activate_this)); del os, activate_this"
       
  1569     if sys.platform == 'win32':
       
  1570         bin_suffix = 'Scripts'
       
  1571     else:
       
  1572         bin_suffix = 'bin'
       
  1573     bin_dir = os.path.join(home_dir, bin_suffix)
       
  1574     home_dir, lib_dir, inc_dir, bin_dir = path_locations(home_dir)
       
  1575     for filename in os.listdir(bin_dir):
       
  1576         filename = os.path.join(bin_dir, filename)
       
  1577         if not os.path.isfile(filename):
       
  1578             # ignore subdirs, e.g. .svn ones.
       
  1579             continue
       
  1580         f = open(filename, 'rb')
       
  1581         try:
       
  1582             try:
       
  1583                 lines = f.read().decode('utf-8').splitlines()
       
  1584             except UnicodeDecodeError:
       
  1585                 # This is probably a binary program instead
       
  1586                 # of a script, so just ignore it.
       
  1587                 continue
       
  1588         finally:
       
  1589             f.close()
       
  1590         if not lines:
       
  1591             logger.warn('Script %s is an empty file' % filename)
       
  1592             continue
       
  1593         if not lines[0].strip().startswith(shebang):
       
  1594             if os.path.basename(filename) in OK_ABS_SCRIPTS:
       
  1595                 logger.debug('Cannot make script %s relative' % filename)
       
  1596             elif lines[0].strip() == new_shebang:
       
  1597                 logger.info('Script %s has already been made relative' % filename)
       
  1598             else:
       
  1599                 logger.warn('Script %s cannot be made relative (it\'s not a normal script that starts with %s)'
       
  1600                             % (filename, shebang))
       
  1601             continue
       
  1602         logger.notify('Making script %s relative' % filename)
       
  1603         lines = [new_shebang+'\n', activate+'\n'] + lines[1:]
       
  1604         f = open(filename, 'wb')
       
  1605         f.write('\n'.join(lines).encode('utf-8'))
       
  1606         f.close()
       
  1607 
       
  1608 def fixup_pth_and_egg_link(home_dir, sys_path=None):
       
  1609     """Makes .pth and .egg-link files use relative paths"""
       
  1610     home_dir = os.path.normcase(os.path.abspath(home_dir))
       
  1611     if sys_path is None:
       
  1612         sys_path = sys.path
       
  1613     for path in sys_path:
       
  1614         if not path:
       
  1615             path = '.'
       
  1616         if not os.path.isdir(path):
       
  1617             continue
       
  1618         path = os.path.normcase(os.path.abspath(path))
       
  1619         if not path.startswith(home_dir):
       
  1620             logger.debug('Skipping system (non-environment) directory %s' % path)
       
  1621             continue
       
  1622         for filename in os.listdir(path):
       
  1623             filename = os.path.join(path, filename)
       
  1624             if filename.endswith('.pth'):
       
  1625                 if not os.access(filename, os.W_OK):
       
  1626                     logger.warn('Cannot write .pth file %s, skipping' % filename)
       
  1627                 else:
       
  1628                     fixup_pth_file(filename)
       
  1629             if filename.endswith('.egg-link'):
       
  1630                 if not os.access(filename, os.W_OK):
       
  1631                     logger.warn('Cannot write .egg-link file %s, skipping' % filename)
       
  1632                 else:
       
  1633                     fixup_egg_link(filename)
       
  1634 
       
  1635 def fixup_pth_file(filename):
       
  1636     lines = []
       
  1637     prev_lines = []
       
  1638     f = open(filename)
       
  1639     prev_lines = f.readlines()
       
  1640     f.close()
       
  1641     for line in prev_lines:
       
  1642         line = line.strip()
       
  1643         if (not line or line.startswith('#') or line.startswith('import ')
       
  1644             or os.path.abspath(line) != line):
       
  1645             lines.append(line)
       
  1646         else:
       
  1647             new_value = make_relative_path(filename, line)
       
  1648             if line != new_value:
       
  1649                 logger.debug('Rewriting path %s as %s (in %s)' % (line, new_value, filename))
       
  1650             lines.append(new_value)
       
  1651     if lines == prev_lines:
       
  1652         logger.info('No changes to .pth file %s' % filename)
       
  1653         return
       
  1654     logger.notify('Making paths in .pth file %s relative' % filename)
       
  1655     f = open(filename, 'w')
       
  1656     f.write('\n'.join(lines) + '\n')
       
  1657     f.close()
       
  1658 
       
  1659 def fixup_egg_link(filename):
       
  1660     f = open(filename)
       
  1661     link = f.read().strip()
       
  1662     f.close()
       
  1663     if os.path.abspath(link) != link:
       
  1664         logger.debug('Link in %s already relative' % filename)
       
  1665         return
       
  1666     new_link = make_relative_path(filename, link)
       
  1667     logger.notify('Rewriting link %s in %s as %s' % (link, filename, new_link))
       
  1668     f = open(filename, 'w')
       
  1669     f.write(new_link)
       
  1670     f.close()
       
  1671 
       
  1672 def make_relative_path(source, dest, dest_is_directory=True):
       
  1673     """
       
  1674     Make a filename relative, where the filename is dest, and it is
       
  1675     being referred to from the filename source.
       
  1676 
       
  1677         >>> make_relative_path('/usr/share/something/a-file.pth',
       
  1678         ...                    '/usr/share/another-place/src/Directory')
       
  1679         '../another-place/src/Directory'
       
  1680         >>> make_relative_path('/usr/share/something/a-file.pth',
       
  1681         ...                    '/home/user/src/Directory')
       
  1682         '../../../home/user/src/Directory'
       
  1683         >>> make_relative_path('/usr/share/a-file.pth', '/usr/share/')
       
  1684         './'
       
  1685     """
       
  1686     source = os.path.dirname(source)
       
  1687     if not dest_is_directory:
       
  1688         dest_filename = os.path.basename(dest)
       
  1689         dest = os.path.dirname(dest)
       
  1690     dest = os.path.normpath(os.path.abspath(dest))
       
  1691     source = os.path.normpath(os.path.abspath(source))
       
  1692     dest_parts = dest.strip(os.path.sep).split(os.path.sep)
       
  1693     source_parts = source.strip(os.path.sep).split(os.path.sep)
       
  1694     while dest_parts and source_parts and dest_parts[0] == source_parts[0]:
       
  1695         dest_parts.pop(0)
       
  1696         source_parts.pop(0)
       
  1697     full_parts = ['..']*len(source_parts) + dest_parts
       
  1698     if not dest_is_directory:
       
  1699         full_parts.append(dest_filename)
       
  1700     if not full_parts:
       
  1701         # Special case for the current directory (otherwise it'd be '')
       
  1702         return './'
       
  1703     return os.path.sep.join(full_parts)
       
  1704 
       
  1705 
       
  1706 
       
  1707 ############################################################
       
  1708 ## Bootstrap script creation:
       
  1709 
       
  1710 def create_bootstrap_script(extra_text, python_version=''):
       
  1711     """
       
  1712     Creates a bootstrap script, which is like this script but with
       
  1713     extend_parser, adjust_options, and after_install hooks.
       
  1714 
       
  1715     This returns a string that (written to disk of course) can be used
       
  1716     as a bootstrap script with your own customizations.  The script
       
  1717     will be the standard virtualenv.py script, with your extra text
       
  1718     added (your extra text should be Python code).
       
  1719 
       
  1720     If you include these functions, they will be called:
       
  1721 
       
  1722     ``extend_parser(optparse_parser)``:
       
  1723         You can add or remove options from the parser here.
       
  1724 
       
  1725     ``adjust_options(options, args)``:
       
  1726         You can change options here, or change the args (if you accept
       
  1727         different kinds of arguments, be sure you modify ``args`` so it is
       
  1728         only ``[DEST_DIR]``).
       
  1729 
       
  1730     ``after_install(options, home_dir)``:
       
  1731 
       
  1732         After everything is installed, this function is called.  This
       
  1733         is probably the function you are most likely to use.  An
       
  1734         example would be::
       
  1735 
       
  1736             def after_install(options, home_dir):
       
  1737                 subprocess.call([join(home_dir, 'bin', 'easy_install'),
       
  1738                                  'MyPackage'])
       
  1739                 subprocess.call([join(home_dir, 'bin', 'my-package-script'),
       
  1740                                  'setup', home_dir])
       
  1741 
       
  1742         This example immediately installs a package, and runs a setup
       
  1743         script from that package.
       
  1744 
       
  1745     If you provide something like ``python_version='2.4'`` then the
       
  1746     script will start with ``#!/usr/bin/env python2.4`` instead of
       
  1747     ``#!/usr/bin/env python``.  You can use this when the script must
       
  1748     be run with a particular Python version.
       
  1749     """
       
  1750     filename = __file__
       
  1751     if filename.endswith('.pyc'):
       
  1752         filename = filename[:-1]
       
  1753     f = open(filename, 'rb')
       
  1754     content = f.read()
       
  1755     f.close()
       
  1756     py_exe = 'python%s' % python_version
       
  1757     content = (('#!/usr/bin/env %s\n' % py_exe)
       
  1758                + '## WARNING: This file is generated\n'
       
  1759                + content)
       
  1760     return content.replace('##EXT' 'END##', extra_text)
       
  1761 
       
  1762 ##EXTEND##
       
  1763 
       
  1764 def convert(s):
       
  1765     b = base64.b64decode(s.encode('ascii'))
       
  1766     return zlib.decompress(b).decode('utf-8')
       
  1767 
       
  1768 ##file site.py
       
  1769 SITE_PY = convert("""
       
  1770 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
       
  1905 Q7wQmAIMcman2gOwRiH4P2ts6wE=
       
  1906 """)
       
  1907 
       
  1908 ##file ez_setup.py
       
  1909 EZ_SETUP_PY = convert("""
       
  1910 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
       
  1966 +G+mirJU+fmCq5Ta1xyCRccC897nZW+WyGsxiMawF7e329Zb2621wQDo2I7tLv7jrv9/AfAaXNUU
       
  1967 TOsyF6jViUG46+NBJqZXv+rRK7Evv2i81ZEw33DQ8y6YowH05r+BuxfN92SX3RbVP8bNymDOGnY7
       
  1968 16PfvzG+4ecrzfzkjPZya/H/ScnXyqwX/JtSrrL5pbrryu1hPKFrZzsrJD6sUuyPwDGdKerJyxmq
       
  1969 dvmdHNCrrzU/+2W0pQ6gSvPl/Mertmi+7hBlDhB80kRUqcNeJCGapHNCz1cvCFwsf0A/Ne++jGMf
       
  1970 TuOJcm6+ZnP9TRR7tWjHreOhZ6huiKnPAP2zfmqpIqHHLG/emnNhyHxSs+JJYfIwj6t2AlLdVneO
       
  1971 3Is9u0R33ef+Wv2pVizPfbUW0rGhps1FRRfnZ/2xsnr3oT2Slh2tvngsLXu6M0OgIen7ufrjprrD
       
  1972 vzXQAgNE22ualqzbyAb97uvl6qF/2a5hcU+eBzVWzOdmVjA0PXQMQoAhsulmBv39oU13134SjSlb
       
  1973 dX85nKW3umfYbtu8713Sylhb2i3v2qaoc8C7S2P3pME8uIGedi1IxXbL+adi+P2fT8Xy/m+/PrxZ
       
  1974 /TrXDcpqOMjotwdo9AJmg8r1N7BySygc+Gp+XaYdJhpV8f/7Oy3Y1s330l09YBDTjnyjn5qHGF7x
       
  1975 6O7hZfMXz21OyLZB6lUfOGAGMzo/bjaL7VaV7Ha76D/1yJVEqKmr+L2nCbH7+959wDtv38JZplQG
       
  1976 BDaonX65d/fwEjNqlDjLVIvM9X+XVxF7
       
  1977 """)
       
  1978 
       
  1979 ##file distribute_setup.py
       
  1980 DISTRIBUTE_SETUP_PY = convert("""
       
  1981 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
       
  2060 """)
       
  2061 
       
  2062 ##file activate.sh
       
  2063 ACTIVATE_SH = convert("""
       
  2064 eJytVVFvokAQfudXTLEP2pw1fW3jg01NNGm1KV4vd22zrDDIJrhrYJHay/33m0VEKGpyufIg7s63
       
  2065 M9/OfDO0YBaKBAIRISzTRMMcIU3Qh0zoEOxEpbGHMBeyxz0t1lyjDRdBrJYw50l4YbVgo1LwuJRK
       
  2066 Q5xKEBp8EaOno41l+bg7Be0O/LaAnhbEmKAGFfmAci1iJZcoNax5LPg8wiRHiQBeoCvBPmfT+zv2
       
  2067 PH6afR/cs8fBbGTDG9yADlHmSPOY7f4haInA95WKdQ4s91JpeDQO5fZAnKTxczaaTkbTh+EhMqWx
       
  2068 QWl/rEGsNJ2kV0cRySKleRGTUKWUVB81pT+vD3Dpw0cSfoMsFF4IIV8jcHqRyVPLpTHrkOu89IUr
       
  2069 EoDHo4gkoBUsiAFVlP4FKjaLFSeNFEeTS4AfJBOV6sKshVwUbmpAkyA4N8kFL+RygQlkpDfum58N
       
  2070 GO1QWNLFipij/yn1twOHit5V29UvZ8Seh0/OeDo5kPz8at24lp5jRXSuDlXPuWqUjYCNejlXJwtV
       
  2071 mHcUtpCddTh53hM7I15EpA+2VNLHRMep6Rn8xK0FDkYB7ABnn6J3jWnXbLvQfyzqz61dxDFGVP1a
       
  2072 o1Xasx7bsipU+zZjlSVjtlUkoXofq9FHlMZtDxaLCrrH2O14wiaDhyFj1wWs2qIl773iTbZohyza
       
  2073 iD0TUQQBF5HZr6ISgzKKNZrD5UpvgO5FwoT2tgkIMec+tcYm45sO+fPytqGpBy75aufpTG/gmhRb
       
  2074 +u3AjQtC5l1l7QV1dBAcadt+7UhFGpXONprZRviAWtbY3dgZ3N4P2ePT9OFxdjJiruJSuLk7+31f
       
  2075 x60HKiWc9eH9SBc04XuPGCVYce1SXlDyJcJrjfKr7ebSNpEaQVpg+l3wiAYOJZ9GCAxoR9JMWAiv
       
  2076 +IyoWBSfhOIIIoRar657vSzLLj9Q0xRZX9Kk6SUq0BmPsceNl179Mi8Vii65Pkj21XXf4MAlSy/t
       
  2077 Exft7A8WX4/iVRkZprZfNK2/YFL/55T+9wm9m86Uhr8A0Hwt
       
  2078 """)
       
  2079 
       
  2080 ##file activate.fish
       
  2081 ACTIVATE_FISH = convert("""
       
  2082 eJydVm1v4jgQ/s6vmA1wBxUE7X2stJVYlVWR2lK13d6d9laRk0yIr8HmbIe0++tvnIQQB9pbXT5A
       
  2083 Ys/LM55nZtyHx5RrSHiGsMm1gRAh1xhDwU0Kng8hFzMWGb5jBv2E69SDs0TJDdj3MxilxmzPZzP7
       
  2084 pVPMMl+q9bjXh1eZQ8SEkAZULoAbiLnCyGSvvV6SC7IoBcS4Nw0wjcFbvJDcjiuTswzFDpiIQaHJ
       
  2085 lQAjQUi1YRmUboC2uZJig8J4PaCnT5IaDcgsbm/CjinOwgx1KcUTMEhhTgV4g2B1fRk8Le8fv86v
       
  2086 g7v545UHpZB9rKnp+gXsMhxLunIIpwVQxP/l9c/Hq9Xt1epm4R27bva6AJqN92G4YhbMG2i+LB+u
       
  2087 grv71c3dY7B6WtzfLy9bePbp0taDTXSwJQJszUnnp0y57mvpPcrF7ZODyhswtd59+/jdgw+fwBNS
       
  2088 xLSscksUPIDqwwNmCez3PpxGeyBYg6HE0YdcWBxcKczYzuVJi5Wu915vn5oWePCCoPUZBN5B7IgV
       
  2089 MCi54ZDLG7TUZ0HweXkb3M5vFmSpFm/gthhBx0UrveoPpv9AJ9unIbQYdUoe21bKg2q48sPFGVwu
       
  2090 H+afrxd1qvclaNlRFyh1EQ2sSccEuNAGWQwysfVpz1tPajUqbqJUnEcIJkWo6OXDaodK8ZiLdbmM
       
  2091 L1wb+9H0D+pcyPSrX5u5kgWSygRYXCnJUi/KKcuU4cqsAyTKZBiissLc7NFwizvjxtieKBVCIdWz
       
  2092 fzilzPaYyljZN0cGN1v7NnaIPNCGmVy3GKuJaQ6iVjE1Qfm+36hglErwmnAD8hu0dDy4uICBA8ZV
       
  2093 pQr/q/+O0KFW2kjelu9Dgb9SDBsWV4F4x5CswgS0zBVlk5tDMP5bVtUGpslbm81Lu2sdKq7uNMGh
       
  2094 MVQ4fy9xhogC1lS5guhISa0DlBWv0O8odT6/LP+4WZzDV6FzIkEqC0uolGZSZoMnlpxplmD2euaT
       
  2095 O4hkTpPnbztDccey0bhjDaBIqaWQa0uwEtQEwtyU56i4fq54F9IE3ORR6mKriODM4XOYZwaVYLYz
       
  2096 7SPbKkz4i7VkB6/Ot1upDE3znNqYKpM8raa0Bx8vfvntJ32UENsM4aI6gJL+jJwhxhh3jVIDOcpi
       
  2097 m0r2hmEtS8XXXNBk71QCDXTBNhhPiHX2LtHkrVIlhoEshH/EZgdq53Eirqs5iFKMnkOmqZTtr3Xq
       
  2098 djvPTWZT4S3NT5aVLgurMPUWI07BRVYqkQrmtCKohNY8qu9EdACoT6ki0a66XxVF4f9AQ3W38yO5
       
  2099 mWmZmIIpnDFrbXakvKWeZhLwhvrbUH8fahhqD0YUcBDJjEBMQwiznE4y5QbHrbhHBOnUAYzb2tVN
       
  2100 jJa65e+eE2Ya30E2GurxUP8ssA6e/wOnvo3V78d3vTcvMB3n7l3iX1JXWqk=
       
  2101 """)
       
  2102 
       
  2103 ##file activate.csh
       
  2104 ACTIVATE_CSH = convert("""
       
  2105 eJx9U11vmzAUffevOCVRu+UB9pws29Kl0iq1aVWllaZlcgxciiViItsQdb9+xiQp+dh4QOB7Pu49
       
  2106 XHqY59IgkwVhVRmLmFAZSrGRNkdgykonhFiqSCRW1sJSmJg8wCDT5QrucRCyHn6WFRKhVGmhKwVp
       
  2107 kUpNiS3emup3TY6XIn7DVNQyJUwlrgthJD6n/iCNv72uhCzCpFx9CRkThRQGKe08cWXJ9db/yh/u
       
  2108 pvzl9mn+PLnjj5P5D1yM8QmXlzBkSdXwZ0H/BBc0mEo5FE5qI2jKhclHOOvy9HD/OO/6YO1mX9vx
       
  2109 sY0H/tPIV0dtqel0V7iZvWyNg8XFcBA0ToEqVeqOdNUEQFvN41SumAv32VtJrakQNSmLWmgp4oJM
       
  2110 yDoBHgoydtoEAs47r5wHHnUal5vbJ8oOI+9wI86vb2d8Nrm/4Xy4RZ8R85E4uTZPB5EZPnTaaAGu
       
  2111 E59J8BE2J8XgrkbLeXMlVoQxznEYFYY8uFFdxsKQRx90Giwx9vSueHP1YNaUSFG4vTaErNSYuBOF
       
  2112 lXiVyXa9Sy3JdClEyK1dD6Nos9mEf8iKlOpmqSNTZnYjNEWiUYn2pKNB3ttcLJ3HmYYXy6Un76f7
       
  2113 r8rRsC1TpTJj7f19m5sUf/V3Ir+x/yjtLu8KjLX/CmN/AcVGUUo=
       
  2114 """)
       
  2115 
       
  2116 ##file activate.bat
       
  2117 ACTIVATE_BAT = convert("""
       
  2118 eJyFUkEKgzAQvAfyhz0YaL9QEWpRqlSjWGspFPZQTevFHOr/adQaU1GaUzI7Mzu7ZF89XhKkEJS8
       
  2119 qxaKMMsvboQ+LxxE44VICSW1gEa2UFaibqoS0iyJ0xw2lIA6nX5AHCu1jpRsv5KRjknkac9VLVug
       
  2120 sX9mtzxIeJDE/mg4OGp47qoLo3NHX2jsMB3AiDht5hryAUOEifoTdCXbSh7V0My2NMq/Xbh5MEjU
       
  2121 ZT63gpgNT9lKOJ/CtHsvT99re3pX303kydn4HeyOeAg5cjf2EW1D6HOPkg9NGKhu
       
  2122 """)
       
  2123 
       
  2124 ##file deactivate.bat
       
  2125 DEACTIVATE_BAT = convert("""
       
  2126 eJxzSE3OyFfIT0vj4spMU0hJTcvMS01RiPf3cYkP8wwKCXX0iQ8I8vcNCFHQ4FIAguLUEgWIgK0q
       
  2127 FlWqXJpcICVYpGzx2BAZ4uHv5+Hv6wq1BWINXBTdKriEKkI1DhW2QAfhttcxxANiFZCBbglQSJUL
       
  2128 i2dASrm4rFz9XLgAwJNbyQ==
       
  2129 """)
       
  2130 
       
  2131 ##file activate.ps1
       
  2132 ACTIVATE_PS = convert("""
       
  2133 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
       
  2219 QastYw==
       
  2220 """)
       
  2221 
       
  2222 ##file distutils-init.py
       
  2223 DISTUTILS_INIT = convert("""
       
  2224 eJytV92L4zYQf/dfMU0ottuse7RvC6FQrg8Lxz2Ugz4si9HacqKuIxlJ2ST313dG8odkO9d7aGBB
       
  2225 luZLv/nNjFacOqUtKJMIvzK3cXlhWgp5MDBsqK5SNYftsBAGpLLA4F1oe2Ytl+9wUvW55TswCi4c
       
  2226 KibhbFDSglXQCFmDPXIwtm7FawLRbwtPzg2T9gf4gupKv4GS0N262w7V0NvpbCy8cvTo3eAus6C5
       
  2227 ETU3ICQZX1hFTw/dzR6V/AW1RCN4/XAtbsVXqIXmlVX6liS4lOzEYY9QFB2zx6LfoSNjz1a0pqT9
       
  2228 QOIfJWQ2E888NEVZNqLlZZnvIB0NpHkimlFdKn2iRRY7yGG/CCJb6Iz280d34SFXBS2yEYPNF0Q7
       
  2229 yM7oCjpWvbEDQmnhRwOs6zjThpKE8HogwRAgraqYFZgGZvzmzVh+mgz9vskT3hruwyjdFcqyENJw
       
  2230 bbMPO5jdzonxK68QKT7B57CMRRG5shRSWDTX3dI8LzRndZbnSWL1zfvriUmK4TcGWSnZiEPCrxXv
       
  2231 bM+sP7VW2is2WgWXCO3sAu3Rzysz3FiNCA8WPyM4gb1JAAmCiyTZbhFjWx3h9SzauuRXC9MFoVbc
       
  2232 yNTCm1QXOOIfIn/g1kGMhDUBN72hI5XCBQtIXQw8UEEdma6Jaz4vJIJ51Orc15hzzmu6TdFp3ogr
       
  2233 Aof0c98tsw1SiaiWotHffk3XYCkqdToxWRfTFXqgpg2khcLluOHMVC0zZhLKIomesfSreUNNgbXi
       
  2234 Ky9VRzwzkBneNoGQyyvGjbsFQqOZvpWIjqH281lJ/jireFgR3cPzSyTGWzQpDNIU+03Fs4XKLkhp
       
  2235 /n0uFnuF6VphB44b3uWRneSbBoMSioqE8oeF0JY+qTvYfEK+bPLYdoR4McfYQ7wMZj39q0kfP8q+
       
  2236 FfsymO0GzNlPh644Jje06ulqHpOEQqdJUfoidI2O4CWx4qOglLye6RrFQirpCRXvhoRqXH3sYdVJ
       
  2237 AItvc+VUsLO2v2hVAWrNIfVGtkG351cUMNncbh/WdowtSPtCdkzYFv6mwYc9o2Jt68ud6wectBr8
       
  2238 hYAulPSlgzH44YbV3ikjrulEaNJxt+/H3wZ7bXSXje/YY4tfVVrVmUstaDwwOBLMg6iduDB0lMVC
       
  2239 UyzYx7Ab4kjCqdViEJmDcdk/SKbgsjYXgfMznUWcrtS4z4fmJ/XOM1LPk/iIpqass5XwNbdnLb1Y
       
  2240 8h3ERXSWZI6rZJxKs1LBqVH65w0Oy4ra0CBYxEeuOMbDmV5GI6E0Ha/wgVTtkX0+OXvqsD02CKLf
       
  2241 XHbeft85D7tTCMYy2Njp4DJP7gWJr6paVWXZ1+/6YXLv/iE0M90FktiI7yFJD9e7SOLhEkkaMTUO
       
  2242 azq9i2woBNR0/0eoF1HFMf0H8ChxH/jgcB34GZIz3Qn4/vid+VEamQrOVqAPTrOfmD4MPdVh09tb
       
  2243 8dLLjvh/61lEP4yW5vJaH4vHcevG8agXvzPGoOhhXNncpTr99PTHx6e/UvffFLaxUSjuSeP286Dw
       
  2244 gtEMcW1xKr/he4/6IQ6FUXP+0gkioHY5iwC9Eyx3HKO7af0zPPe+XyLn7fAY78k4aiR387bCr5XT
       
  2245 5C4rFgwLGfMvJuAMew==
       
  2246 """)
       
  2247 
       
  2248 ##file distutils.cfg
       
  2249 DISTUTILS_CFG = convert("""
       
  2250 eJxNj00KwkAMhfc9xYNuxe4Ft57AjYiUtDO1wXSmNJnK3N5pdSEEAu8nH6lxHVlRhtDHMPATA4uH
       
  2251 xJ4EFmGbvfJiicSHFRzUSISMY6hq3GLCRLnIvSTnEefN0FIjw5tF0Hkk9Q5dRunBsVoyFi24aaLg
       
  2252 9FDOlL0FPGluf4QjcInLlxd6f6rqkgPu/5nHLg0cXCscXoozRrP51DRT3j9QNl99AP53T2Q=
       
  2253 """)
       
  2254 
       
  2255 ##file activate_this.py
       
  2256 ACTIVATE_THIS = convert("""
       
  2257 eJyNU01v2zAMvetXEB4K21jmDOstQA4dMGCHbeihlyEIDMWmG62yJEiKE//7kXKdpN2KzYBt8euR
       
  2258 fKSyLPs8wiEo8wh4wqZTGou4V6Hm0wJa1cSiTkJdr8+GsoTRHuCotBayiWqQEYGtMCgfD1KjGYBe
       
  2259 5a3p0cRKiAe2NtLADikftnDco0ko/SFEVgEZ8aRC5GLux7i3BpSJ6J1H+i7A2CjiHq9z7JRZuuQq
       
  2260 siwTIvpxJYCeuWaBpwZdhB+yxy/eWz+ZvVSU8C4E9FFZkyxFsvCT/ZzL8gcz9aXVE14Yyp2M+2W0
       
  2261 y7n5mp0qN+avKXvbsyyzUqjeWR8hjGE+2iCE1W1tQ82hsCZN9UzlJr+/e/iab8WfqsmPI6pWeUPd
       
  2262 FrMsd4H/55poeO9n54COhUs+sZNEzNtg/wanpjpuqHJaxs76HtZryI/K3H7KJ/KDIhqcbJ7kI4ar
       
  2263 XL+sMgXnX0D+Te2Iy5xdP8yueSlQB/x/ED2BTAtyE3K4SYUN6AMNfbO63f4lBW3bUJPbTL+mjSxS
       
  2264 PyRfJkZRgj+VbFv+EzHFi5pKwUEepa4JslMnwkowSRCXI+m5XvEOvtuBrxHdhLalG0JofYBok6qj
       
  2265 YdN2dEngUlbC4PG60M1WEN0piu7Nq7on0mgyyUw3iV1etLo6r/81biWdQ9MWHFaePWZYaq+nmp+t
       
  2266 s3az+sj7eA0jfgPfeoN1
       
  2267 """)
       
  2268 
       
  2269 if __name__ == '__main__':
       
  2270     main()
       
  2271 
       
  2272 ## TODO:
       
  2273 ## Copy python.exe.manifest
       
  2274 ## Monkeypatch distutils.sysconfig