2017-08-23 10:41:27 -04:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import argparse
|
|
|
|
import subprocess
|
|
|
|
|
|
|
|
from importlib import import_module
|
|
|
|
|
2017-09-20 00:12:40 -04:00
|
|
|
CURRENT_DIR = os.path.dirname(os.path.abspath(__file__))
|
|
|
|
ZULIP_BOTS_DIR = os.path.join(CURRENT_DIR, '..', 'zulip_bots')
|
|
|
|
sys.path.append(ZULIP_BOTS_DIR)
|
|
|
|
|
|
|
|
import generate_manifest
|
|
|
|
|
2017-11-30 04:53:45 -05:00
|
|
|
red = '\033[91m'
|
2017-11-23 11:46:03 -05:00
|
|
|
green = '\033[92m'
|
|
|
|
end_format = '\033[0m'
|
|
|
|
bold = '\033[1m'
|
|
|
|
|
2017-08-23 10:41:27 -04:00
|
|
|
def main():
|
2017-08-24 15:36:04 -04:00
|
|
|
usage = """./tools/provision
|
|
|
|
|
|
|
|
Creates a Python virtualenv. Its Python version is equal to
|
|
|
|
the Python version this command is executed with."""
|
|
|
|
parser = argparse.ArgumentParser(usage=usage)
|
2017-11-10 16:11:54 -05:00
|
|
|
parser.add_argument('--python-interpreter', '-p',
|
|
|
|
metavar='PATH_TO_PYTHON_INTERPRETER',
|
|
|
|
default=os.path.abspath(sys.executable),
|
|
|
|
help='Path to the Python interpreter to use when provisioning.')
|
2017-11-30 04:53:45 -05:00
|
|
|
parser.add_argument('--force', '-f', action='store_true',
|
|
|
|
help='create venv even with outdated Python version.')
|
2017-11-10 16:11:54 -05:00
|
|
|
options = parser.parse_args()
|
2017-08-23 10:41:27 -04:00
|
|
|
|
|
|
|
base_dir = os.path.abspath(os.path.join(__file__, '..', '..'))
|
2017-11-23 11:39:54 -05:00
|
|
|
py_version_output = subprocess.check_output([options.python_interpreter, '--version'],
|
|
|
|
stderr=subprocess.STDOUT, universal_newlines=True)
|
|
|
|
# The output has the format "Python 1.2.3"
|
|
|
|
py_version_list = py_version_output.split()[1].split('.')
|
|
|
|
py_version = tuple(int(num) for num in py_version_list)
|
|
|
|
venv_name = 'zulip-api-py{}-venv'.format(py_version[0])
|
2017-08-23 10:41:27 -04:00
|
|
|
|
|
|
|
venv_dir = os.path.join(base_dir, venv_name)
|
|
|
|
if not os.path.isdir(venv_dir):
|
2017-09-26 20:35:21 -04:00
|
|
|
try:
|
2017-11-10 16:11:54 -05:00
|
|
|
return_code = subprocess.call(['virtualenv', '-p', options.python_interpreter, venv_dir])
|
2017-09-26 20:35:21 -04:00
|
|
|
except OSError:
|
|
|
|
if subprocess.call(['which', 'virtualenv']):
|
|
|
|
print("{red}Please install the virtualenv package and try again.{end_format}"
|
|
|
|
.format(red='\033[91m', end_format='\033[0m'))
|
|
|
|
sys.exit(1)
|
|
|
|
raise
|
2017-08-23 10:41:27 -04:00
|
|
|
else:
|
2017-09-26 20:35:21 -04:00
|
|
|
if return_code:
|
|
|
|
raise OSError("The command `virtualenv -p {} {}` failed. Virtualenv not created!"
|
2017-11-10 16:11:54 -05:00
|
|
|
.format(options.python_interpreter, venv_dir))
|
2017-08-23 10:41:27 -04:00
|
|
|
print("New virtualenv created.")
|
|
|
|
else:
|
|
|
|
print("Virtualenv already exists.")
|
|
|
|
|
|
|
|
if os.path.isdir(os.path.join(venv_dir, 'Scripts')):
|
|
|
|
# POSIX compatibility layer and Linux environment emulation for Windows
|
|
|
|
# Virtual uses /Scripts instead of /bin on Windows.
|
|
|
|
# Read https://virtualenv.pypa.io/en/stable/userguide/
|
|
|
|
venv_exec_dir = 'Scripts'
|
|
|
|
else:
|
|
|
|
venv_exec_dir = 'bin'
|
|
|
|
|
|
|
|
# In order to install all required packages for the venv, we need to activate it. Since
|
|
|
|
# the activation script sets environmental variables, it needs to be executed inline with
|
|
|
|
# `import_module`.
|
|
|
|
activate_module_dir = os.path.abspath(os.path.join(venv_dir, venv_exec_dir))
|
|
|
|
sys.path.append(activate_module_dir)
|
|
|
|
import_module('activate_this')
|
|
|
|
|
|
|
|
# In order to install all required packages for the venv, `pip` needs to be executed by
|
|
|
|
# the venv's Python interpreter. `--prefix venv_dir` ensures that all modules are installed
|
|
|
|
# in the right place.
|
2017-09-15 07:24:35 -04:00
|
|
|
def install_dependencies(requirements_filename):
|
2017-09-28 15:08:42 -04:00
|
|
|
pip_path = os.path.join(venv_dir, venv_exec_dir, 'pip')
|
|
|
|
# We first install a modern version of pip that supports --prefix
|
|
|
|
subprocess.call([pip_path, 'install', 'pip>=9.0'])
|
|
|
|
if subprocess.call([pip_path, 'install', '--prefix', venv_dir, '-r',
|
|
|
|
os.path.join(base_dir, requirements_filename)]):
|
2017-09-15 07:24:35 -04:00
|
|
|
raise OSError("The command `pip install -r {}` failed. Dependencies not installed!"
|
|
|
|
.format(os.path.join(base_dir, requirements_filename)))
|
|
|
|
|
|
|
|
install_dependencies('requirements.txt')
|
2017-11-23 11:39:54 -05:00
|
|
|
if py_version > (3, 1):
|
2017-09-15 07:24:35 -04:00
|
|
|
install_dependencies('py3_requirements.txt')
|
2017-11-30 04:53:45 -05:00
|
|
|
elif not options.force:
|
|
|
|
print(red + "Provision failed: Cannot create venv with outdated Python version ({}).\n"
|
|
|
|
"Run this script with --force if you want to proceed anyway."
|
|
|
|
.format(py_version_output.strip()) + end_format)
|
|
|
|
sys.exit(1)
|
2017-11-10 16:03:40 -05:00
|
|
|
|
|
|
|
print(green + 'Success!' + end_format)
|
|
|
|
|
|
|
|
activate_command = os.path.join(base_dir,
|
|
|
|
venv_dir,
|
|
|
|
venv_exec_dir,
|
|
|
|
'activate')
|
|
|
|
print('\nNow, run this!:')
|
|
|
|
print(bold + 'source ' + activate_command + end_format)
|
|
|
|
|
2017-08-23 10:41:27 -04:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|