pypa-hatch/tests/commands/test_new.py

257 lines
9.0 KiB
Python

import os
import sys
import toml
from click.testing import CliRunner
from hatch.cli import hatch
from hatch.config import get_venv_dir
from hatch.env import get_editable_packages, get_python_version
from hatch.settings import (
SETTINGS_FILE, copy_default_settings, restore_settings, save_settings
)
from hatch.utils import create_file, remove_path, temp_chdir, temp_move_path
from hatch.venv import create_venv, get_new_venv_name, is_venv, venv
from ..utils import matching_file, read_file, wait_until, requires_internet
def test_config_not_exist():
with temp_chdir() as d:
runner = CliRunner()
with temp_move_path(SETTINGS_FILE, d):
result = runner.invoke(hatch, ['new', 'ok', '--basic', '-ne'])
assert result.exit_code == 0
assert (
'Unable to locate config file; try `hatch config --restore`. '
'The default project structure will be used.'
) in result.output
assert 'Created project `ok`' in result.output
def test_missing_name_invokes_interactive_mode():
with temp_chdir() as d:
runner = CliRunner()
result = runner.invoke(hatch, ['new', '--basic', '-ne'], input='invalid-name')
assert result.exit_code == 0
assert os.path.exists(os.path.join(d, 'invalid-name', 'invalid_name', '__init__.py'))
def test_interactive_mode():
with temp_chdir() as d:
runner = CliRunner()
with temp_move_path(SETTINGS_FILE, d):
result = runner.invoke(
hatch, ['new', '-i', '--basic', '-ne'],
input='ok\n0.1.0\nTest Description\nPicard\npicard@startrek.com\nmpl\n'
)
assert result.exit_code == 0
assert os.path.exists(os.path.join(d, 'ok', 'ok', '__init__.py'))
assert "__version__ = '0.1.0'\n" == read_file(os.path.join(d, 'ok', 'ok', '__init__.py'))
assert os.path.exists(os.path.join(d, 'ok', 'LICENSE-MPL'))
assert os.path.exists(os.path.join(d, 'ok', 'pyproject.toml'))
pyproject = toml.load(os.path.join(d, 'ok', 'pyproject.toml'))
metadata = pyproject['metadata']
assert metadata['name'] == 'ok'
assert metadata['version'] == '0.1.0'
assert metadata['description'] == 'Test Description'
assert metadata['author'] == 'Picard'
assert metadata['author_email'] == 'picard@startrek.com'
assert metadata['license'] == 'MPL-2.0'
assert metadata['url'] == 'https://github.com/_/ok'
def test_invalid_name():
with temp_chdir() as d:
runner = CliRunner()
runner.invoke(hatch, ['new', 'invalid-name', '--basic', '-ne'])
assert os.path.exists(os.path.join(d, 'invalid-name', 'invalid_name', '__init__.py'))
def test_output():
with temp_chdir():
runner = CliRunner()
result = runner.invoke(hatch, ['new', 'new-project', '--basic', '-ne'])
assert result.exit_code == 0
assert 'Created project `new-project`' in result.output
def test_already_exists():
with temp_chdir() as d:
d = os.path.join(d, 'ok')
os.makedirs(d)
runner = CliRunner()
result = runner.invoke(hatch, ['new', 'ok', '--basic', '-ne'])
assert result.exit_code == 1
assert 'Directory `{}` already exists.'.format(d) in result.output
@requires_internet
def test_env():
with temp_chdir() as d:
runner = CliRunner()
result = runner.invoke(hatch, ['new', 'new-project', '--basic'])
venv_dir = os.path.join(d, 'new-project', 'venv')
wait_until(is_venv, venv_dir)
with venv(venv_dir):
assert 'new-project' in get_editable_packages()
assert result.exit_code == 0
assert 'Creating its own virtual env... complete!' in result.output
assert 'Installing locally in the virtual env... complete!' in result.output
def test_pyname():
with temp_chdir() as d:
runner = CliRunner()
env_name = get_new_venv_name()
venv_dir = os.path.join(get_venv_dir(), env_name)
try:
with temp_move_path(SETTINGS_FILE, d):
settings = copy_default_settings()
settings['pypaths']['python'] = sys.executable
save_settings(settings)
result = runner.invoke(hatch, ['new', 'ok', '-py', 'python'])
venv_dir = os.path.join(d, 'ok', 'venv')
global_version = get_python_version()
wait_until(is_venv, venv_dir)
with venv(venv_dir):
assert get_python_version() == global_version
finally:
remove_path(venv_dir)
assert result.exit_code == 0
def test_pyname_config_not_exist():
with temp_chdir() as d:
runner = CliRunner()
with temp_move_path(SETTINGS_FILE, d):
result = runner.invoke(hatch, ['new', 'ok', '-py', 'python'])
assert result.exit_code == 1
assert 'Unable to locate config file. Try `hatch config --restore`.' in result.output
def test_pyname_key_not_exist():
with temp_chdir() as d:
runner = CliRunner()
with temp_move_path(SETTINGS_FILE, d):
restore_settings()
result = runner.invoke(hatch, ['new', 'ok', '-py', 'pyname'])
assert result.exit_code == 1
assert 'Unable to find a Python path named `pyname`.' in result.output
def test_basic():
with temp_chdir() as d:
runner = CliRunner()
runner.invoke(hatch, ['new', 'ok', '--basic', '-ne'])
d = os.path.join(d, 'ok')
assert os.path.exists(os.path.join(d, 'ok', '__init__.py'))
assert os.path.exists(os.path.join(d, 'tests', '__init__.py'))
assert os.path.exists(os.path.join(d, 'setup.py'))
assert os.path.exists(os.path.join(d, 'MANIFEST.in'))
assert os.path.exists(os.path.join(d, 'requirements.txt'))
assert os.path.exists(os.path.join(d, '.coveragerc'))
assert os.path.exists(os.path.join(d, 'tox.ini'))
assert matching_file(r'^LICENSE.*', os.listdir(d))
assert matching_file(r'^README.*', os.listdir(d))
def test_cli():
with temp_chdir() as d:
runner = CliRunner()
runner.invoke(hatch, ['new', 'ok', '--cli', '-ne'])
d = os.path.join(d, 'ok')
assert os.path.exists(os.path.join(d, 'ok', 'cli.py'))
assert os.path.exists(os.path.join(d, 'ok', '__main__.py'))
def test_license_single():
with temp_chdir() as d:
runner = CliRunner()
runner.invoke(hatch, ['new', 'ok', '-l', 'cc0', '-ne'])
assert os.path.exists(os.path.join(d, 'ok', 'LICENSE-CC0'))
def test_license_multiple():
with temp_chdir() as d:
runner = CliRunner()
runner.invoke(hatch, ['new', 'ok', '-l', 'cc0,mpl', '-ne'])
d = os.path.join(d, 'ok')
assert os.path.exists(os.path.join(d, 'LICENSE-CC0'))
assert os.path.exists(os.path.join(d, 'LICENSE-MPL'))
def test_extras():
with temp_chdir() as d:
runner = CliRunner()
test_dir = os.path.join(d, 'a', 'b')
test_file1 = os.path.join(test_dir, 'file1.txt')
test_file2 = os.path.join(d, 'x', 'y', 'file2.txt')
test_glob = '{}{}*'.format(os.path.join(d, 'x'), os.path.sep)
fake_file = os.path.join(test_dir, 'file.py')
create_file(test_file1)
create_file(test_file2)
with temp_move_path(SETTINGS_FILE, d):
new_settings = copy_default_settings()
new_settings['extras'] = [test_dir, test_file1, test_glob, fake_file]
save_settings(new_settings)
runner.invoke(hatch, ['new', 'ok', '--basic', '-ne'])
d = os.path.join(d, 'ok')
assert os.path.exists(os.path.join(d, 'b', 'file1.txt'))
assert os.path.exists(os.path.join(d, 'file1.txt'))
assert os.path.exists(os.path.join(d, 'y', 'file2.txt'))
assert not os.path.exists(os.path.join(d, 'file2.txt'))
assert not os.path.exists(os.path.join(d, 'file.py'))
@requires_internet
def test_envs():
with temp_chdir():
runner = CliRunner()
env_name1, env_name2 = get_new_venv_name(count=2)
venv_dir1 = os.path.join(get_venv_dir(), env_name1)
venv_dir2 = os.path.join(get_venv_dir(), env_name2)
create_venv(venv_dir1)
try:
result = runner.invoke(hatch, [
'new', '-ne', '--basic', 'ok', '-e', '{}/{}'.format(env_name1, env_name2)
])
wait_until(is_venv, venv_dir2)
with venv(venv_dir1):
assert 'ok' in get_editable_packages()
with venv(venv_dir2):
assert 'ok' in get_editable_packages()
finally:
remove_path(venv_dir1)
remove_path(venv_dir2)
assert result.exit_code == 0
assert 'Creating virtual env `{}`... complete!'.format(env_name1) not in result.output
assert 'Creating virtual env `{}`... complete!'.format(env_name2) in result.output
assert 'Installing locally in virtual env `{}`... complete!'.format(env_name1) in result.output
assert 'Installing locally in virtual env `{}`... complete!'.format(env_name2) in result.output