Files
pipenv/pipenv.py
T
2017-01-20 13:12:17 -05:00

296 lines
8.0 KiB
Python

import codecs
import json
import os
import sys
import toml
import delegator
import click
import crayons
import _pipfile as pipfile
__version__ = '0.0.0'
def ensure_latest_pip():
# Ensure that pip is installed.
c = delegator.run('pip install pip')
# Check if version is out of date.
if 'however' in c.err:
# If version is out of date, update.
click.echo(crayons.yellow('Pip is out of date... updating to latest.'))
c = delegator.run('pip install pip --upgrade', block=False)
click.echo(crayons.blue(c.out))
def ensure_virtualenv():
# c = delegator.run('pip install virtualenv')
pass
def format_toml(data):
"""Pretty-formats a given toml string."""
data = data.split('\n')
for i, line in enumerate(data):
if i > 0:
if line.startswith('['):
data[i] = '\n{}'.format(line)
return '\n'.join(data)
def add_package_to_pipfile(package_name, dev=False):
pipfile_path = pipfile.Pipfile.find()
# Read and append Pipfile.
with open(pipfile_path, 'r') as f:
p = toml.loads(f.read())
key = 'develop' if dev else 'packages'
# Set empty group if it doesn't exist yet.
if key not in p:
p[key] = {}
package = convert_deps_from_pip(package_name)
# Add the package to the group.
if package_name not in p[key]:
# TODO: Support >1.0.1
p[key][package_name] = package
# Write Pipfile.
data = format_toml(toml.dumps(p))
with open(pipfile_path, 'w') as f:
f.write(data)
def multi_split(s, split):
for r in split:
s = s.replace(r, '|')
return [i for i in s.split('|') if len(i) > 0]
def remove_package_from_pipfile(package_name, dev=False):
pipfile_path = pipfile.Pipfile.find()
# Read and append Pipfile.
with open(pipfile_path, 'r') as f:
p = toml.loads(f.read())
key = 'develop' if dev else 'packages'
if package_name in p[key]:
del p[key][package_name]
# Write Pipfile.
data = format_toml(toml.dumps(p))
with open(pipfile_path, 'w') as f:
f.write(data)
def convert_deps_from_pip(dep):
dependency = {}
# Comparison operators: e.g. Django>1.10
if '=' in dep or '<' in dep or '>' in dep:
r = multi_split(dep, '=<>')
dependency[r[0]] = dep[len(r[0]):]
# Extras: e.g. requests[socks]
elif '[' in dep:
r = multi_split(dep, '[]')
dependency[r[0]] = {'extras': r[1].split(',')}
# TODO: Editable installs.
# Bare dependencies: e.g. requests
else:
dependency[dep] = '*'
return dependency
def convert_deps_to_pip(deps):
dependencies = []
for dep in deps.keys():
# Default (e.g. '>1.10').
extra = deps[dep]
# Get rid of '*'.
if deps[dep] == '*' or str(extra) == '{}':
extra = ''
# Support for extras (e.g. requests[socks])
if 'extras' in deps[dep]:
extra = '[{}]'.format(deps[dep]['extras'][0])
# Support for git.
if 'git' in deps[dep]:
extra = 'git+{}'.format(deps[dep]['git'])
# Support for @refs.
if 'ref' in deps[dep]:
extra += '@{}'.format(deps[dep]['ref'])
# Support for editable.
if 'editable' in deps[dep]:
# Support for --egg.
extra += ' --egg={}'.format(dep)
dep = '-e '
dependencies.append('{}{}'.format(dep, extra))
return dependencies
def virtualenv_location():
return os.sep.join(pipfile.Pipfile.find().split(os.sep)[:-1] + ['.venv'])
def pipfile_location():
return pipfile.Pipfile.find()
def lockfile_location():
return '{}.freeze'.format(pipfile_location())
def lockfile_exists():
return os.path.isfile(lockfile_location())
def do_where(virtualenv=False, bare=True):
if not virtualenv:
location = pipfile_location()
if not bare:
click.echo('Pipfile found at {}. Considering this to be the project home.'.format(crayons.green(location)))
else:
click.echo(location)
else:
location = virtualenv_location()
if not bare:
click.echo('Virtualenv location: {}'.format(crayons.green(location)))
else:
click.echo(location)
@click.group()
# @click.option('--version', is_flag=True, callback=display_version, help='Display version information')
@click.version_option(prog_name=crayons.yellow('pip2'), version=__version__)
def cli(*args, **kwargs):
# Ensure that pip is installed and up-to-date.
ensure_latest_pip()
# Ensure that virtualenv is installed.
ensure_virtualenv()
def which_pip():
return os.sep.join([virtualenv_location()] + ['bin/pip'])
@click.command()
@click.option('--dev', is_flag=True, default=False)
def prepare(dev=False):
do_where(bare=False)
click.echo(crayons.yellow('Creating a virtualenv for this project...'))
# Actually create the virtualenv.
c = delegator.run('virtualenv {}'.format(virtualenv_location()), block=False)
# c.block()
click.echo(crayons.blue(c.out))
# Say where the virtualenv is.
do_where(virtualenv=True, bare=False)
# Write out the lockfile if it doesn't exist.
if lockfile_exists():
click.echo(crayons.yellow('Installing dependencies from Pipfile.freeze...'))
with codecs.open(lockfile_location(), 'r') as f:
lockfile = json.load(f)
# TODO: Update the lockfile if it is out-of-date.
p = pipfile.load(pipfile_location())
if not lockfile['_meta']['Pipfile-sha256'] == p.hash:
click.echo(crayons.red('Pipfile.freeze out of date, updating...'))
# Update the lockfile.
# TODO: Add sub-dependencies.
with open(lockfile_location(), 'w') as f:
f.write(p.freeze())
else:
# Load the pipfile.
click.echo(crayons.yellow('Installing dependencies from Pipfile...'))
p = pipfile.load(pipfile_location())
lockfile = json.loads(p.freeze())
# Install default dependencies, always.
deps = lockfile['default']
# Add development deps if --dev was passed.
if dev:
deps.update(lockfile['develop'])
# Convert the deps to pip-compatbile arguments.
deps = convert_deps_to_pip(deps)
# Actually install each dependency into the virtualenv.
for package_name in deps:
click.echo('Installing {}...'.format(crayons.green(package_name)))
c = delegator.run('{} install {}'.format(which_pip(), package_name),)
click.echo(crayons.blue(c.out))
# Write out the lockfile if it doesn't exist.
if not lockfile_exists():
click.echo(crayons.yellow('Pipfile.freeze not found, creating...'))
with codecs.open(lockfile_location(), 'w', 'utf-8') as f:
f.write(p.freeze())
@click.command()
@click.option('--virtualenv', is_flag=True, default=False)
@click.option('--bare', is_flag=True, default=False)
def where(virtualenv=False, bare=False):
do_where(virtualenv, bare)
@click.command()
@click.argument('package_name')
@click.option('--dev', is_flag=True, default=False)
def install(package_name, dev=False):
click.echo('Installing {}...'.format(crayons.green(package_name)))
c = delegator.run('{} install {}'.format(which_pip(), package_name))
click.echo(crayons.blue(c.out))
click.echo('Adding {} to Pipfile...'.format(crayons.green(package_name)))
add_package_to_pipfile(package_name, dev)
@click.command()
@click.argument('package_name')
def uninstall(package_name):
click.echo('Un-installing {}...'.format(crayons.green(package_name)))
c = delegator.run('{} uninstall {} -y'.format(which_pip(), package_name))
click.echo(crayons.blue(c.out))
click.echo('Removing {} from Pipfile...'.format(crayons.green(package_name)))
remove_package_from_pipfile(package_name)
# Install click commands.
cli.add_command(prepare)
cli.add_command(where)
cli.add_command(install)
cli.add_command(uninstall)
if __name__ == '__main__':
cli()