2016-06-21 19:37:36 +02:00
|
|
|
|
|
|
|
import os
|
|
|
|
import subprocess
|
2017-06-13 17:04:54 +02:00
|
|
|
from scripts.lib.zulip_tools import run, ENDC, WARNING
|
2016-08-11 13:32:17 +02:00
|
|
|
from scripts.lib.hash_reqs import expand_reqs
|
2016-06-21 19:37:36 +02:00
|
|
|
|
2017-09-22 08:15:01 +02:00
|
|
|
ZULIP_PATH = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
|
2016-06-21 19:37:36 +02:00
|
|
|
VENV_CACHE_PATH = "/srv/zulip-venv-cache"
|
|
|
|
|
2016-06-28 06:35:19 +02:00
|
|
|
if 'TRAVIS' in os.environ:
|
2016-06-21 19:37:36 +02:00
|
|
|
# In Travis CI, we don't have root access
|
2016-06-28 06:35:19 +02:00
|
|
|
VENV_CACHE_PATH = "/home/travis/zulip-venv-cache"
|
2016-06-21 19:37:36 +02:00
|
|
|
|
|
|
|
if False:
|
|
|
|
# Don't add a runtime dependency on typing
|
2016-08-11 13:32:17 +02:00
|
|
|
from typing import List, Optional, Tuple, Set
|
2016-06-21 19:37:36 +02:00
|
|
|
|
2016-06-22 18:17:46 +02:00
|
|
|
VENV_DEPENDENCIES = [
|
2016-06-24 18:09:36 +02:00
|
|
|
"build-essential",
|
2016-06-22 18:17:46 +02:00
|
|
|
"libffi-dev",
|
2016-07-20 23:42:49 +02:00
|
|
|
"libfreetype6-dev", # Needed for image types with Pillow
|
|
|
|
"libz-dev", # Needed to handle compressed PNGs with Pillow
|
|
|
|
"libjpeg-dev", # Needed to handle JPEGs with Pillow
|
2016-06-22 18:17:46 +02:00
|
|
|
"libldap2-dev",
|
|
|
|
"libmemcached-dev",
|
|
|
|
"python3-dev", # Needed to install typed-ast dependency of mypy
|
|
|
|
"python-dev",
|
2017-08-04 01:43:32 +02:00
|
|
|
"python3-pip",
|
2016-06-24 18:09:36 +02:00
|
|
|
"python-pip",
|
2017-08-04 01:43:32 +02:00
|
|
|
"python-virtualenv", # Trusty lacks `python3-virtualenv`.
|
|
|
|
# Fortunately we don't need the library,
|
|
|
|
# only the command, and this suffices.
|
|
|
|
"python3-six",
|
|
|
|
"python-six",
|
2016-07-12 14:55:02 +02:00
|
|
|
"libxml2-dev", # Used for installing talon
|
|
|
|
"libxslt1-dev", # Used for installing talon
|
2016-07-20 18:45:06 +02:00
|
|
|
"libpq-dev", # Needed by psycopg2
|
2016-06-22 18:17:46 +02:00
|
|
|
]
|
|
|
|
|
2017-06-13 17:04:54 +02:00
|
|
|
def install_venv_deps(requirements_file):
|
|
|
|
# type: (str) -> None
|
2017-06-19 11:27:21 +02:00
|
|
|
pip_requirements = os.path.join(ZULIP_PATH, "requirements", "pip.txt")
|
|
|
|
run(["pip", "install", "-U", "--requirement", pip_requirements])
|
2017-06-13 17:04:54 +02:00
|
|
|
run(["pip", "install", "--no-deps", "--requirement", requirements_file])
|
|
|
|
|
2016-08-11 13:32:17 +02:00
|
|
|
def get_index_filename(venv_path):
|
|
|
|
# type: (str) -> str
|
|
|
|
return os.path.join(venv_path, 'package_index')
|
|
|
|
|
|
|
|
def get_package_names(requirements_file):
|
|
|
|
# type: (str) -> List[str]
|
|
|
|
packages = expand_reqs(requirements_file)
|
|
|
|
cleaned = []
|
|
|
|
operators = ['~=', '==', '!=', '<', '>']
|
|
|
|
for package in packages:
|
2016-10-16 08:58:09 +02:00
|
|
|
if package.startswith("git+https://") and '#egg=' in package:
|
|
|
|
split_package = package.split("#egg=")
|
|
|
|
if len(split_package) != 2:
|
|
|
|
raise Exception("Unexpected duplicate #egg in package %s" % (package,))
|
|
|
|
# Extract the package name from Git requirements entries
|
|
|
|
package = split_package[1]
|
|
|
|
|
2016-08-11 13:32:17 +02:00
|
|
|
for operator in operators:
|
|
|
|
if operator in package:
|
|
|
|
package = package.split(operator)[0]
|
|
|
|
|
|
|
|
package = package.strip()
|
|
|
|
if package:
|
|
|
|
cleaned.append(package.lower())
|
|
|
|
|
|
|
|
return sorted(cleaned)
|
|
|
|
|
|
|
|
def create_requirements_index_file(venv_path, requirements_file):
|
|
|
|
# type: (str, str) -> str
|
|
|
|
"""
|
|
|
|
Creates a file, called package_index, in the virtual environment
|
|
|
|
directory that contains all the PIP packages installed in the
|
|
|
|
virtual environment. This file is used to determine the packages
|
|
|
|
that can be copied to a new virtual environment.
|
|
|
|
"""
|
|
|
|
index_filename = get_index_filename(venv_path)
|
|
|
|
packages = get_package_names(requirements_file)
|
|
|
|
with open(index_filename, 'w') as writer:
|
|
|
|
writer.write('\n'.join(packages))
|
|
|
|
writer.write('\n')
|
|
|
|
|
|
|
|
return index_filename
|
|
|
|
|
|
|
|
def get_venv_packages(venv_path):
|
|
|
|
# type: (str) -> Set[str]
|
|
|
|
"""
|
|
|
|
Returns the packages installed in the virtual environment using the
|
|
|
|
package index file.
|
|
|
|
"""
|
|
|
|
with open(get_index_filename(venv_path)) as reader:
|
|
|
|
return set(p.strip() for p in reader.read().split('\n') if p.strip())
|
|
|
|
|
|
|
|
def try_to_copy_venv(venv_path, new_packages):
|
|
|
|
# type: (str, Set[str]) -> bool
|
|
|
|
"""
|
|
|
|
Tries to copy packages from an old virtual environment in the cache
|
|
|
|
to the new virtual environment. The algorithm works as follows:
|
|
|
|
1. Find a virtual environment, v, from the cache that has the
|
|
|
|
highest overlap with the new requirements such that:
|
|
|
|
a. The new requirements only add to the packages of v.
|
|
|
|
b. The new requirements only upgrade packages of v.
|
|
|
|
2. Copy the contents of v to the new virtual environment using
|
|
|
|
virtualenv-clone.
|
|
|
|
3. Delete all .pyc files in the new virtual environment.
|
|
|
|
"""
|
2016-09-20 15:55:08 +02:00
|
|
|
if not os.path.exists(VENV_CACHE_PATH):
|
|
|
|
return False
|
|
|
|
|
2016-08-11 13:32:17 +02:00
|
|
|
venv_name = os.path.basename(venv_path)
|
|
|
|
|
|
|
|
overlaps = [] # type: List[Tuple[int, str, Set[str]]]
|
|
|
|
old_packages = set() # type: Set[str]
|
|
|
|
for sha1sum in os.listdir(VENV_CACHE_PATH):
|
|
|
|
curr_venv_path = os.path.join(VENV_CACHE_PATH, sha1sum, venv_name)
|
|
|
|
if (curr_venv_path == venv_path or
|
|
|
|
not os.path.exists(get_index_filename(curr_venv_path))):
|
|
|
|
continue
|
|
|
|
|
|
|
|
old_packages = get_venv_packages(curr_venv_path)
|
|
|
|
# We only consider using using old virtualenvs that only
|
|
|
|
# contain packages that we want in our new virtualenv.
|
|
|
|
if not (old_packages - new_packages):
|
|
|
|
overlap = new_packages & old_packages
|
|
|
|
overlaps.append((len(overlap), curr_venv_path, overlap))
|
|
|
|
|
|
|
|
target_log = get_logfile_name(venv_path)
|
|
|
|
source_venv_path = None
|
|
|
|
if overlaps:
|
|
|
|
# Here, we select the old virtualenv with the largest overlap
|
|
|
|
overlaps = sorted(overlaps)
|
|
|
|
_, source_venv_path, copied_packages = overlaps[-1]
|
|
|
|
print('Copying packages from {}'.format(source_venv_path))
|
|
|
|
clone_ve = "{}/bin/virtualenv-clone".format(source_venv_path)
|
|
|
|
cmd = "sudo {exe} {source} {target}".format(exe=clone_ve,
|
|
|
|
source=source_venv_path,
|
|
|
|
target=venv_path).split()
|
|
|
|
try:
|
|
|
|
run(cmd)
|
|
|
|
except Exception:
|
|
|
|
# Virtualenv-clone is not installed. Install it and try running
|
|
|
|
# the command again.
|
2017-01-09 20:14:16 +01:00
|
|
|
try:
|
|
|
|
run("{}/bin/pip install --no-deps virtualenv-clone".format(
|
|
|
|
source_venv_path).split())
|
|
|
|
run(cmd)
|
|
|
|
except Exception:
|
|
|
|
# virtualenv-clone isn't working, so just make a new venv
|
|
|
|
return False
|
2016-08-11 13:32:17 +02:00
|
|
|
|
|
|
|
run(["sudo", "chown", "-R",
|
|
|
|
"{}:{}".format(os.getuid(), os.getgid()), venv_path])
|
|
|
|
source_log = get_logfile_name(source_venv_path)
|
|
|
|
copy_parent_log(source_log, target_log)
|
|
|
|
create_log_entry(target_log, source_venv_path, copied_packages,
|
|
|
|
new_packages - copied_packages)
|
|
|
|
return True
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
def get_logfile_name(venv_path):
|
|
|
|
# type: (str) -> str
|
|
|
|
return "{}/setup-venv.log".format(venv_path)
|
|
|
|
|
|
|
|
def create_log_entry(target_log, parent, copied_packages, new_packages):
|
|
|
|
# type: (str, str, Set[str], Set[str]) -> None
|
|
|
|
|
2017-09-22 08:15:01 +02:00
|
|
|
venv_path = os.path.dirname(target_log)
|
2016-08-11 13:32:17 +02:00
|
|
|
with open(target_log, 'a') as writer:
|
|
|
|
writer.write("{}\n".format(venv_path))
|
|
|
|
if copied_packages:
|
|
|
|
writer.write(
|
|
|
|
"Copied from {}:\n".format(parent))
|
|
|
|
writer.write("\n".join('- {}'.format(p) for p in sorted(copied_packages)))
|
|
|
|
writer.write("\n")
|
|
|
|
|
|
|
|
writer.write("New packages:\n")
|
|
|
|
writer.write("\n".join('- {}'.format(p) for p in sorted(new_packages)))
|
|
|
|
writer.write("\n\n")
|
|
|
|
|
|
|
|
def copy_parent_log(source_log, target_log):
|
|
|
|
# type: (str, str) -> None
|
|
|
|
if os.path.exists(source_log):
|
|
|
|
run('cp {} {}'.format(source_log, target_log).split())
|
|
|
|
|
2016-07-20 21:42:33 +02:00
|
|
|
def do_patch_activate_script(venv_path):
|
|
|
|
# type: (str) -> None
|
|
|
|
"""
|
|
|
|
Patches the bin/activate script so that the value of the environment variable VIRTUAL_ENV
|
|
|
|
is set to venv_path during the script's execution whenever it is sourced.
|
|
|
|
"""
|
2016-07-20 15:34:31 +02:00
|
|
|
# venv_path should be what we want to have in VIRTUAL_ENV after patching
|
|
|
|
script_path = os.path.join(venv_path, "bin", "activate")
|
|
|
|
|
|
|
|
file_obj = open(script_path)
|
|
|
|
lines = file_obj.readlines()
|
|
|
|
for i, line in enumerate(lines):
|
|
|
|
if line.startswith('VIRTUAL_ENV='):
|
|
|
|
lines[i] = 'VIRTUAL_ENV="%s"\n' % (venv_path,)
|
|
|
|
file_obj.close()
|
|
|
|
|
|
|
|
file_obj = open(script_path, 'w')
|
|
|
|
file_obj.write("".join(lines))
|
|
|
|
file_obj.close()
|
|
|
|
|
2016-07-20 21:42:33 +02:00
|
|
|
def setup_virtualenv(target_venv_path, requirements_file, virtualenv_args=None, patch_activate_script=False):
|
|
|
|
# type: (Optional[str], str, Optional[List[str]], bool) -> str
|
2016-06-21 19:37:36 +02:00
|
|
|
|
|
|
|
# Check if a cached version already exists
|
2016-06-22 14:42:08 +02:00
|
|
|
path = os.path.join(ZULIP_PATH, 'scripts', 'lib', 'hash_reqs.py')
|
2016-07-02 21:41:13 +02:00
|
|
|
output = subprocess.check_output([path, requirements_file], universal_newlines=True)
|
2016-06-21 19:37:36 +02:00
|
|
|
sha1sum = output.split()[0]
|
2016-06-22 20:04:14 +02:00
|
|
|
if target_venv_path is None:
|
|
|
|
cached_venv_path = os.path.join(VENV_CACHE_PATH, sha1sum, 'venv')
|
|
|
|
else:
|
|
|
|
cached_venv_path = os.path.join(VENV_CACHE_PATH, sha1sum, os.path.basename(target_venv_path))
|
2016-06-21 19:37:36 +02:00
|
|
|
success_stamp = os.path.join(cached_venv_path, "success-stamp")
|
|
|
|
if not os.path.exists(success_stamp):
|
|
|
|
do_setup_virtualenv(cached_venv_path, requirements_file, virtualenv_args or [])
|
|
|
|
run(["touch", success_stamp])
|
|
|
|
|
|
|
|
print("Using cached Python venv from %s" % (cached_venv_path,))
|
2016-06-22 20:04:14 +02:00
|
|
|
if target_venv_path is not None:
|
|
|
|
run(["sudo", "ln", "-nsf", cached_venv_path, target_venv_path])
|
2016-07-20 21:42:33 +02:00
|
|
|
if patch_activate_script:
|
|
|
|
do_patch_activate_script(target_venv_path)
|
2016-06-22 20:04:14 +02:00
|
|
|
activate_this = os.path.join(cached_venv_path, "bin", "activate_this.py")
|
2016-10-17 08:22:00 +02:00
|
|
|
exec(open(activate_this).read(), {}, dict(__file__=activate_this))
|
2016-06-22 20:04:14 +02:00
|
|
|
return cached_venv_path
|
2016-06-21 19:37:36 +02:00
|
|
|
|
|
|
|
def do_setup_virtualenv(venv_path, requirements_file, virtualenv_args):
|
|
|
|
# type: (str, str, List[str]) -> None
|
|
|
|
|
|
|
|
# Setup Python virtualenv
|
2016-08-11 13:32:17 +02:00
|
|
|
new_packages = set(get_package_names(requirements_file))
|
|
|
|
|
2016-06-21 19:37:36 +02:00
|
|
|
run(["sudo", "rm", "-rf", venv_path])
|
2016-08-11 13:32:17 +02:00
|
|
|
if not try_to_copy_venv(venv_path, new_packages):
|
|
|
|
# Create new virtualenv.
|
|
|
|
run(["sudo", "mkdir", "-p", venv_path])
|
|
|
|
run(["sudo", "virtualenv"] + virtualenv_args + [venv_path])
|
|
|
|
run(["sudo", "chown", "-R",
|
|
|
|
"{}:{}".format(os.getuid(), os.getgid()), venv_path])
|
|
|
|
create_log_entry(get_logfile_name(venv_path), "", set(), new_packages)
|
2016-06-21 19:37:36 +02:00
|
|
|
|
2016-08-11 13:32:17 +02:00
|
|
|
create_requirements_index_file(venv_path, requirements_file)
|
2016-06-21 19:37:36 +02:00
|
|
|
# Switch current Python context to the virtualenv.
|
|
|
|
activate_this = os.path.join(venv_path, "bin", "activate_this.py")
|
2016-10-17 08:22:00 +02:00
|
|
|
exec(open(activate_this).read(), {}, dict(__file__=activate_this))
|
2016-06-21 19:37:36 +02:00
|
|
|
|
2017-06-13 17:04:54 +02:00
|
|
|
try:
|
|
|
|
install_venv_deps(requirements_file)
|
|
|
|
except subprocess.CalledProcessError:
|
|
|
|
# Might be a failure due to network connection issues. Retrying...
|
|
|
|
print(WARNING + "`pip install` failed; retrying..." + ENDC)
|
|
|
|
install_venv_deps(requirements_file)
|
2016-06-28 21:02:53 +02:00
|
|
|
run(["sudo", "chmod", "-R", "a+rX", venv_path])
|