mirror of https://github.com/sgoudham/Enso-Bot.git
319 lines
10 KiB
Python
319 lines
10 KiB
Python
5 years ago
|
#!/usr/bin/env python
|
||
|
from __future__ import with_statement
|
||
|
|
||
|
import logging
|
||
|
import optparse
|
||
|
import os
|
||
|
import os.path
|
||
|
import re
|
||
|
import shutil
|
||
|
import subprocess
|
||
|
import sys
|
||
|
import itertools
|
||
|
|
||
|
__version__ = '0.5.4'
|
||
|
|
||
|
|
||
|
logger = logging.getLogger()
|
||
|
|
||
|
|
||
|
env_bin_dir = 'bin'
|
||
|
if sys.platform == 'win32':
|
||
|
env_bin_dir = 'Scripts'
|
||
|
|
||
|
|
||
|
class UserError(Exception):
|
||
|
pass
|
||
|
|
||
|
|
||
|
def _dirmatch(path, matchwith):
|
||
|
"""Check if path is within matchwith's tree.
|
||
|
|
||
|
>>> _dirmatch('/home/foo/bar', '/home/foo/bar')
|
||
|
True
|
||
|
>>> _dirmatch('/home/foo/bar/', '/home/foo/bar')
|
||
|
True
|
||
|
>>> _dirmatch('/home/foo/bar/etc', '/home/foo/bar')
|
||
|
True
|
||
|
>>> _dirmatch('/home/foo/bar2', '/home/foo/bar')
|
||
|
False
|
||
|
>>> _dirmatch('/home/foo/bar2/etc', '/home/foo/bar')
|
||
|
False
|
||
|
"""
|
||
|
matchlen = len(matchwith)
|
||
|
if (path.startswith(matchwith)
|
||
|
and path[matchlen:matchlen + 1] in [os.sep, '']):
|
||
|
return True
|
||
|
return False
|
||
|
|
||
|
|
||
|
def _virtualenv_sys(venv_path):
|
||
|
"obtain version and path info from a virtualenv."
|
||
|
executable = os.path.join(venv_path, env_bin_dir, 'python')
|
||
|
# Must use "executable" as the first argument rather than as the
|
||
|
# keyword argument "executable" to get correct value from sys.path
|
||
|
p = subprocess.Popen([executable,
|
||
|
'-c', 'import sys;'
|
||
|
'print (sys.version[:3]);'
|
||
|
'print ("\\n".join(sys.path));'],
|
||
|
env={},
|
||
|
stdout=subprocess.PIPE)
|
||
|
stdout, err = p.communicate()
|
||
|
assert not p.returncode and stdout
|
||
|
lines = stdout.decode('utf-8').splitlines()
|
||
|
return lines[0], list(filter(bool, lines[1:]))
|
||
|
|
||
|
|
||
|
def clone_virtualenv(src_dir, dst_dir):
|
||
|
if not os.path.exists(src_dir):
|
||
|
raise UserError('src dir %r does not exist' % src_dir)
|
||
|
if os.path.exists(dst_dir):
|
||
|
raise UserError('dest dir %r exists' % dst_dir)
|
||
|
#sys_path = _virtualenv_syspath(src_dir)
|
||
|
logger.info('cloning virtualenv \'%s\' => \'%s\'...' %
|
||
|
(src_dir, dst_dir))
|
||
|
shutil.copytree(src_dir, dst_dir, symlinks=True,
|
||
|
ignore=shutil.ignore_patterns('*.pyc'))
|
||
|
version, sys_path = _virtualenv_sys(dst_dir)
|
||
|
logger.info('fixing scripts in bin...')
|
||
|
fixup_scripts(src_dir, dst_dir, version)
|
||
|
|
||
|
has_old = lambda s: any(i for i in s if _dirmatch(i, src_dir))
|
||
|
|
||
|
if has_old(sys_path):
|
||
|
# only need to fix stuff in sys.path if we have old
|
||
|
# paths in the sys.path of new python env. right?
|
||
|
logger.info('fixing paths in sys.path...')
|
||
|
fixup_syspath_items(sys_path, src_dir, dst_dir)
|
||
|
v_sys = _virtualenv_sys(dst_dir)
|
||
|
remaining = has_old(v_sys[1])
|
||
|
assert not remaining, v_sys
|
||
|
fix_symlink_if_necessary(src_dir, dst_dir)
|
||
|
|
||
|
def fix_symlink_if_necessary(src_dir, dst_dir):
|
||
|
#sometimes the source virtual environment has symlinks that point to itself
|
||
|
#one example is $OLD_VIRTUAL_ENV/local/lib points to $OLD_VIRTUAL_ENV/lib
|
||
|
#this function makes sure
|
||
|
#$NEW_VIRTUAL_ENV/local/lib will point to $NEW_VIRTUAL_ENV/lib
|
||
|
#usually this goes unnoticed unless one tries to upgrade a package though pip, so this bug is hard to find.
|
||
|
logger.info("scanning for internal symlinks that point to the original virtual env")
|
||
|
for dirpath, dirnames, filenames in os.walk(dst_dir):
|
||
|
for a_file in itertools.chain(filenames, dirnames):
|
||
|
full_file_path = os.path.join(dirpath, a_file)
|
||
|
if os.path.islink(full_file_path):
|
||
|
target = os.path.realpath(full_file_path)
|
||
|
if target.startswith(src_dir):
|
||
|
new_target = target.replace(src_dir, dst_dir)
|
||
|
logger.debug('fixing symlink in %s' % (full_file_path,))
|
||
|
os.remove(full_file_path)
|
||
|
os.symlink(new_target, full_file_path)
|
||
|
|
||
|
|
||
|
def fixup_scripts(old_dir, new_dir, version, rewrite_env_python=False):
|
||
|
bin_dir = os.path.join(new_dir, env_bin_dir)
|
||
|
root, dirs, files = next(os.walk(bin_dir))
|
||
|
pybinre = re.compile(r'pythonw?([0-9]+(\.[0-9]+(\.[0-9]+)?)?)?$')
|
||
|
for file_ in files:
|
||
|
filename = os.path.join(root, file_)
|
||
|
if file_ in ['python', 'python%s' % version, 'activate_this.py']:
|
||
|
continue
|
||
|
elif file_.startswith('python') and pybinre.match(file_):
|
||
|
# ignore other possible python binaries
|
||
|
continue
|
||
|
elif file_.endswith('.pyc'):
|
||
|
# ignore compiled files
|
||
|
continue
|
||
|
elif file_ == 'activate' or file_.startswith('activate.'):
|
||
|
fixup_activate(os.path.join(root, file_), old_dir, new_dir)
|
||
|
elif os.path.islink(filename):
|
||
|
fixup_link(filename, old_dir, new_dir)
|
||
|
elif os.path.isfile(filename):
|
||
|
fixup_script_(root, file_, old_dir, new_dir, version,
|
||
|
rewrite_env_python=rewrite_env_python)
|
||
|
|
||
|
|
||
|
def fixup_script_(root, file_, old_dir, new_dir, version,
|
||
|
rewrite_env_python=False):
|
||
|
old_shebang = '#!%s/bin/python' % os.path.normcase(os.path.abspath(old_dir))
|
||
|
new_shebang = '#!%s/bin/python' % os.path.normcase(os.path.abspath(new_dir))
|
||
|
env_shebang = '#!/usr/bin/env python'
|
||
|
|
||
|
filename = os.path.join(root, file_)
|
||
|
with open(filename, 'rb') as f:
|
||
|
if f.read(2) != b'#!':
|
||
|
# no shebang
|
||
|
return
|
||
|
f.seek(0)
|
||
|
lines = f.readlines()
|
||
|
|
||
|
if not lines:
|
||
|
# warn: empty script
|
||
|
return
|
||
|
|
||
|
def rewrite_shebang(version=None):
|
||
|
logger.debug('fixing %s' % filename)
|
||
|
shebang = new_shebang
|
||
|
if version:
|
||
|
shebang = shebang + version
|
||
|
shebang = (shebang + '\n').encode('utf-8')
|
||
|
with open(filename, 'wb') as f:
|
||
|
f.write(shebang)
|
||
|
f.writelines(lines[1:])
|
||
|
|
||
|
try:
|
||
|
bang = lines[0].decode('utf-8').strip()
|
||
|
except UnicodeDecodeError:
|
||
|
# binary file
|
||
|
return
|
||
|
|
||
|
# This takes care of the scheme in which shebang is of type
|
||
|
# '#!/venv/bin/python3' while the version of system python
|
||
|
# is of type 3.x e.g. 3.5.
|
||
|
short_version = bang[len(old_shebang):]
|
||
|
|
||
|
if not bang.startswith('#!'):
|
||
|
return
|
||
|
elif bang == old_shebang:
|
||
|
rewrite_shebang()
|
||
|
elif (bang.startswith(old_shebang)
|
||
|
and bang[len(old_shebang):] == version):
|
||
|
rewrite_shebang(version)
|
||
|
elif (bang.startswith(old_shebang)
|
||
|
and short_version
|
||
|
and bang[len(old_shebang):] == short_version):
|
||
|
rewrite_shebang(short_version)
|
||
|
elif rewrite_env_python and bang.startswith(env_shebang):
|
||
|
if bang == env_shebang:
|
||
|
rewrite_shebang()
|
||
|
elif bang[len(env_shebang):] == version:
|
||
|
rewrite_shebang(version)
|
||
|
else:
|
||
|
# can't do anything
|
||
|
return
|
||
|
|
||
|
|
||
|
def fixup_activate(filename, old_dir, new_dir):
|
||
|
logger.debug('fixing %s' % filename)
|
||
|
with open(filename, 'rb') as f:
|
||
|
data = f.read().decode('utf-8')
|
||
|
|
||
|
data = data.replace(old_dir, new_dir)
|
||
|
with open(filename, 'wb') as f:
|
||
|
f.write(data.encode('utf-8'))
|
||
|
|
||
|
|
||
|
def fixup_link(filename, old_dir, new_dir, target=None):
|
||
|
logger.debug('fixing %s' % filename)
|
||
|
if target is None:
|
||
|
target = os.readlink(filename)
|
||
|
|
||
|
origdir = os.path.dirname(os.path.abspath(filename)).replace(
|
||
|
new_dir, old_dir)
|
||
|
if not os.path.isabs(target):
|
||
|
target = os.path.abspath(os.path.join(origdir, target))
|
||
|
rellink = True
|
||
|
else:
|
||
|
rellink = False
|
||
|
|
||
|
if _dirmatch(target, old_dir):
|
||
|
if rellink:
|
||
|
# keep relative links, but don't keep original in case it
|
||
|
# traversed up out of, then back into the venv.
|
||
|
# so, recreate a relative link from absolute.
|
||
|
target = target[len(origdir):].lstrip(os.sep)
|
||
|
else:
|
||
|
target = target.replace(old_dir, new_dir, 1)
|
||
|
|
||
|
# else: links outside the venv, replaced with absolute path to target.
|
||
|
_replace_symlink(filename, target)
|
||
|
|
||
|
|
||
|
def _replace_symlink(filename, newtarget):
|
||
|
tmpfn = "%s.new" % filename
|
||
|
os.symlink(newtarget, tmpfn)
|
||
|
os.rename(tmpfn, filename)
|
||
|
|
||
|
|
||
|
def fixup_syspath_items(syspath, old_dir, new_dir):
|
||
|
for path in syspath:
|
||
|
if not os.path.isdir(path):
|
||
|
continue
|
||
|
path = os.path.normcase(os.path.abspath(path))
|
||
|
if _dirmatch(path, old_dir):
|
||
|
path = path.replace(old_dir, new_dir, 1)
|
||
|
if not os.path.exists(path):
|
||
|
continue
|
||
|
elif not _dirmatch(path, new_dir):
|
||
|
continue
|
||
|
root, dirs, files = next(os.walk(path))
|
||
|
for file_ in files:
|
||
|
filename = os.path.join(root, file_)
|
||
|
if filename.endswith('.pth'):
|
||
|
fixup_pth_file(filename, old_dir, new_dir)
|
||
|
elif filename.endswith('.egg-link'):
|
||
|
fixup_egglink_file(filename, old_dir, new_dir)
|
||
|
|
||
|
|
||
|
def fixup_pth_file(filename, old_dir, new_dir):
|
||
|
logger.debug('fixup_pth_file %s' % filename)
|
||
|
|
||
|
with open(filename, 'r') as f:
|
||
|
lines = f.readlines()
|
||
|
|
||
|
has_change = False
|
||
|
|
||
|
for num, line in enumerate(lines):
|
||
|
line = (line.decode('utf-8') if hasattr(line, 'decode') else line).strip()
|
||
|
|
||
|
if not line or line.startswith('#') or line.startswith('import '):
|
||
|
continue
|
||
|
elif _dirmatch(line, old_dir):
|
||
|
lines[num] = line.replace(old_dir, new_dir, 1)
|
||
|
has_change = True
|
||
|
|
||
|
if has_change:
|
||
|
with open(filename, 'w') as f:
|
||
|
payload = os.linesep.join([l.strip() for l in lines]) + os.linesep
|
||
|
f.write(payload)
|
||
|
|
||
|
|
||
|
def fixup_egglink_file(filename, old_dir, new_dir):
|
||
|
logger.debug('fixing %s' % filename)
|
||
|
with open(filename, 'rb') as f:
|
||
|
link = f.read().decode('utf-8').strip()
|
||
|
if _dirmatch(link, old_dir):
|
||
|
link = link.replace(old_dir, new_dir, 1)
|
||
|
with open(filename, 'wb') as f:
|
||
|
link = (link + '\n').encode('utf-8')
|
||
|
f.write(link)
|
||
|
|
||
|
|
||
|
def main():
|
||
|
parser = optparse.OptionParser("usage: %prog [options]"
|
||
|
" /path/to/existing/venv /path/to/cloned/venv")
|
||
|
parser.add_option('-v',
|
||
|
action="count",
|
||
|
dest='verbose',
|
||
|
default=False,
|
||
|
help='verbosity')
|
||
|
options, args = parser.parse_args()
|
||
|
try:
|
||
|
old_dir, new_dir = args
|
||
|
except ValueError:
|
||
|
print("virtualenv-clone %s" % (__version__,))
|
||
|
parser.error("not enough arguments given.")
|
||
|
old_dir = os.path.realpath(old_dir)
|
||
|
new_dir = os.path.realpath(new_dir)
|
||
|
loglevel = (logging.WARNING, logging.INFO, logging.DEBUG)[min(2,
|
||
|
options.verbose)]
|
||
|
logging.basicConfig(level=loglevel, format='%(message)s')
|
||
|
try:
|
||
|
clone_virtualenv(old_dir, new_dir)
|
||
|
except UserError:
|
||
|
e = sys.exc_info()[1]
|
||
|
parser.error(str(e))
|
||
|
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
main()
|