pypa-hatch/tests/env/plugin/test_interface.py

1895 lines
62 KiB
Python

import pytest
from hatch.config.constants import AppEnvVars
from hatch.env.plugin.interface import EnvironmentInterface
from hatch.project.core import Project
from hatch.utils.structures import EnvVars
from hatchling.utils.constants import DEFAULT_BUILD_SCRIPT
class MockEnvironment(EnvironmentInterface): # no cov
PLUGIN_NAME = 'mock'
def find(self):
pass
def create(self):
pass
def remove(self):
pass
def exists(self):
pass
def install_project(self):
pass
def install_project_dev_mode(self):
pass
def dependencies_in_sync(self):
pass
def sync_dependencies(self):
pass
class TestEnvVars:
def test_default(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.env_vars == environment.env_vars == {AppEnvVars.ENV_ACTIVE: 'default'}
def test_not_table(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'env-vars': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.env-vars` must be a mapping'):
_ = environment.env_vars
def test_value_not_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'env-vars': {'foo': 9000}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
TypeError, match='Environment variable `foo` of field `tool.hatch.envs.default.env-vars` must be a string'
):
_ = environment.env_vars
def test_correct(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'env-vars': {'foo': 'bar'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.env_vars == {AppEnvVars.ENV_ACTIVE: 'default', 'foo': 'bar'}
def test_context_formatting(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'env-vars': {'foo': '{env:FOOBAZ}-{matrix:bar}'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{'bar': '42'},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with EnvVars({'FOOBAZ': 'baz'}):
assert environment.env_vars == {AppEnvVars.ENV_ACTIVE: 'default', 'foo': 'baz-42'}
class TestEnvInclude:
def test_default(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.env_include == environment.env_include == []
def test_not_array(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'env-include': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.env-include` must be an array'):
_ = environment.env_include
def test_pattern_not_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'env-include': [9000]}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
TypeError, match='Pattern #1 of field `tool.hatch.envs.default.env-include` must be a string'
):
_ = environment.env_include
def test_correct(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'env-include': ['FOO*']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.env_include == ['HATCH_BUILD_*', 'FOO*']
class TestEnvExclude:
def test_default(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.env_exclude == environment.env_exclude == []
def test_not_array(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'env-exclude': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.env-exclude` must be an array'):
_ = environment.env_exclude
def test_pattern_not_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'env-exclude': [9000]}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
TypeError, match='Pattern #1 of field `tool.hatch.envs.default.env-exclude` must be a string'
):
_ = environment.env_exclude
def test_correct(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'env-exclude': ['FOO*']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.env_exclude == ['FOO*']
class TestPlatforms:
def test_default(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.platforms == environment.platforms == []
def test_not_array(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'platforms': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.platforms` must be an array'):
_ = environment.platforms
def test_entry_not_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'platforms': [9000]}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
TypeError, match='Platform #1 of field `tool.hatch.envs.default.platforms` must be a string'
):
_ = environment.platforms
def test_correct(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'platforms': ['macOS']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.platforms == ['macos']
class TestSkipInstall:
def test_default_project(self, temp_dir, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(temp_dir, config=config)
environment = MockEnvironment(
temp_dir,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
(temp_dir / 'pyproject.toml').touch()
with temp_dir.as_cwd():
assert environment.skip_install is environment.skip_install is False
def test_default_no_project(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.skip_install is environment.skip_install is True
def test_not_boolean(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'skip-install': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.skip-install` must be a boolean'):
_ = environment.skip_install
def test_enable(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'skip-install': True}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.skip_install is True
class TestDevMode:
def test_default(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.dev_mode is environment.dev_mode is True
def test_not_boolean(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'dev-mode': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.dev-mode` must be a boolean'):
_ = environment.dev_mode
def test_disable(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'dev-mode': False}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.dev_mode is False
class TestFeatures:
def test_default(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.features == environment.features == []
def test_invalid_type(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'features': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.features` must be an array of strings'):
_ = environment.features
def test_correct(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'optional-dependencies': {'foo-bar': [], 'baz': []}},
'tool': {'hatch': {'envs': {'default': {'features': ['Foo...Bar', 'Baz', 'baZ']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.features == ['baz', 'foo-bar']
def test_feature_not_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'optional-dependencies': {'foo': [], 'bar': []}},
'tool': {'hatch': {'envs': {'default': {'features': [9000]}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Feature #1 of field `tool.hatch.envs.default.features` must be a string'):
_ = environment.features
def test_feature_empty_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'optional-dependencies': {'foo': [], 'bar': []}},
'tool': {'hatch': {'envs': {'default': {'features': ['']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
ValueError, match='Feature #1 of field `tool.hatch.envs.default.features` cannot be an empty string'
):
_ = environment.features
def test_feature_undefined(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'optional-dependencies': {'foo': []}},
'tool': {'hatch': {'envs': {'default': {'features': ['foo', 'bar', '']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
ValueError,
match=(
'Feature `bar` of field `tool.hatch.envs.default.features` is not defined in '
'field `project.optional-dependencies`'
),
):
_ = environment.features
class TestDescription:
def test_default(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.description == environment.description == ''
def test_not_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'description': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.description` must be a string'):
_ = environment.description
def test_correct(self, isolation, isolated_data_dir, platform):
description = 'foo'
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'description': description}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.description is description
class TestDependencies:
def test_default(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dependencies': ['dep1']},
'tool': {'hatch': {'envs': {'default': {'skip-install': False}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.dependencies == environment.dependencies == ['dep1']
assert len(environment.dependencies) == len(environment.dependencies_complex)
def test_not_array(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dependencies': ['dep1']},
'tool': {'hatch': {'envs': {'default': {'dependencies': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.dependencies` must be an array'):
_ = environment.dependencies
def test_entry_not_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dependencies': ['dep1']},
'tool': {'hatch': {'envs': {'default': {'dependencies': [9000]}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
TypeError, match='Dependency #1 of field `tool.hatch.envs.default.dependencies` must be a string'
):
_ = environment.dependencies
def test_invalid(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dependencies': ['dep1']},
'tool': {'hatch': {'envs': {'default': {'dependencies': ['foo^1']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
ValueError, match='Dependency #1 of field `tool.hatch.envs.default.dependencies` is invalid: .+'
):
_ = environment.dependencies
def test_extra_not_array(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dependencies': ['dep1']},
'tool': {'hatch': {'envs': {'default': {'extra-dependencies': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.extra-dependencies` must be an array'):
_ = environment.dependencies
def test_extra_entry_not_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dependencies': ['dep1']},
'tool': {'hatch': {'envs': {'default': {'extra-dependencies': [9000]}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
TypeError, match='Dependency #1 of field `tool.hatch.envs.default.extra-dependencies` must be a string'
):
_ = environment.dependencies
def test_extra_invalid(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dependencies': ['dep1']},
'tool': {'hatch': {'envs': {'default': {'extra-dependencies': ['foo^1']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
ValueError, match='Dependency #1 of field `tool.hatch.envs.default.extra-dependencies` is invalid: .+'
):
_ = environment.dependencies
def test_full(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dependencies': ['dep1']},
'tool': {
'hatch': {
'envs': {
'default': {'skip-install': False, 'dependencies': ['dep2'], 'extra-dependencies': ['dep3']}
}
}
},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.dependencies == ['dep2', 'dep3', 'dep1']
def test_context_formatting(self, isolation, isolated_data_dir, platform, uri_slash_prefix):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dependencies': ['dep1']},
'tool': {
'hatch': {
'envs': {
'default': {
'skip-install': False,
'dependencies': ['dep2'],
'extra-dependencies': ['proj @ {root:uri}'],
}
}
}
},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
normalized_path = str(isolation).replace('\\', '/')
assert environment.dependencies == ['dep2', f'proj@ file:{uri_slash_prefix}{normalized_path}', 'dep1']
def test_full_skip_install(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dependencies': ['dep1']},
'tool': {
'hatch': {
'envs': {
'default': {'dependencies': ['dep2'], 'extra-dependencies': ['dep3'], 'skip-install': True}
}
}
},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.dependencies == ['dep2', 'dep3']
def test_full_skip_install_and_features(self, isolation, isolated_data_dir, platform):
config = {
'project': {
'name': 'my_app',
'version': '0.0.1',
'dependencies': ['dep1'],
'optional-dependencies': {'feat': ['dep4']},
},
'tool': {
'hatch': {
'envs': {
'default': {
'dependencies': ['dep2'],
'extra-dependencies': ['dep3'],
'skip-install': True,
'features': ['feat'],
}
}
}
},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.dependencies == ['dep2', 'dep3', 'dep4']
def test_full_dev_mode(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dependencies': ['dep1']},
'tool': {
'hatch': {
'envs': {'default': {'dependencies': ['dep2'], 'extra-dependencies': ['dep3'], 'dev-mode': False}}
}
},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.dependencies == ['dep2', 'dep3']
def test_unknown_dynamic_feature(self, helpers, temp_dir, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1', 'dynamic': ['optional-dependencies']},
'tool': {
'hatch': {
'metadata': {'hooks': {'custom': {}}},
'envs': {'default': {'skip-install': False, 'features': ['foo']}},
},
},
}
project = Project(temp_dir, config=config)
environment = MockEnvironment(
temp_dir,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
build_script = temp_dir / DEFAULT_BUILD_SCRIPT
build_script.write_text(
helpers.dedent(
"""
from hatchling.metadata.plugin.interface import MetadataHookInterface
class CustomHook(MetadataHookInterface):
def update(self, metadata):
metadata['optional-dependencies'] = {'bar': ['binary']}
"""
)
)
with pytest.raises(
ValueError,
match=(
'Feature `foo` of field `tool.hatch.envs.default.features` is not defined in the dynamic '
'field `project.optional-dependencies`'
),
):
_ = environment.dependencies
class TestScripts:
def test_not_table(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.scripts` must be a table'):
_ = environment.scripts
def test_name_contains_spaces(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo bar': []}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
ValueError, match='Script name `foo bar` in field `tool.hatch.envs.default.scripts` must not contain spaces'
):
_ = environment.scripts
def test_default(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.scripts == environment.scripts == {}
def test_single_commands(self, isolation, isolated_data_dir, platform):
script_config = {'foo': 'command1', 'bar': 'command2'}
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': script_config}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.scripts == {'foo': ['command1'], 'bar': ['command2']}
def test_multiple_commands(self, isolation, isolated_data_dir, platform):
script_config = {'foo': 'command1', 'bar': ['command3', 'command2']}
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': script_config}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.scripts == {'foo': ['command1'], 'bar': ['command3', 'command2']}
def test_multiple_commands_not_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': [9000]}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
TypeError, match='Command #1 in field `tool.hatch.envs.default.scripts.foo` must be a string'
):
_ = environment.scripts
def test_config_invalid_type(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': 9000}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
TypeError, match='Field `tool.hatch.envs.default.scripts.foo` must be a string or an array of strings'
):
_ = environment.scripts
def test_command_expansion_basic(self, isolation, isolated_data_dir, platform):
script_config = {'foo': 'command1', 'bar': ['command3', 'foo']}
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': script_config}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.scripts == {'foo': ['command1'], 'bar': ['command3', 'command1']}
def test_command_expansion_multiple_nested(self, isolation, isolated_data_dir, platform):
script_config = {
'foo': 'command3',
'baz': ['command5', 'bar', 'foo', 'command1'],
'bar': ['command4', 'foo', 'command2'],
}
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': script_config}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.scripts == {
'foo': ['command3'],
'baz': ['command5', 'command4', 'command3', 'command2', 'command3', 'command1'],
'bar': ['command4', 'command3', 'command2'],
}
def test_command_expansion_multiple_nested_ignore_exit_code(self, isolation, isolated_data_dir, platform):
script_config = {
'foo': 'command3',
'baz': ['command5', '- bar', 'foo', 'command1'],
'bar': ['command4', '- foo', 'command2'],
}
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': script_config}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.scripts == {
'foo': ['command3'],
'baz': ['command5', '- command4', '- command3', '- command2', 'command3', 'command1'],
'bar': ['command4', '- command3', 'command2'],
}
def test_command_expansion_modification(self, isolation, isolated_data_dir, platform):
script_config = {
'foo': 'command3',
'baz': ['command5', 'bar world', 'foo', 'command1'],
'bar': ['command4', 'foo hello', 'command2'],
}
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': script_config}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.scripts == {
'foo': ['command3'],
'baz': ['command5', 'command4 world', 'command3 hello world', 'command2 world', 'command3', 'command1'],
'bar': ['command4', 'command3 hello', 'command2'],
}
def test_command_expansion_circular_inheritance(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': 'bar', 'bar': 'foo'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
ValueError,
match='Circular expansion detected for field `tool.hatch.envs.default.scripts`: foo -> bar -> foo',
):
_ = environment.scripts
class TestPreInstallCommands:
def test_default(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.pre_install_commands == environment.pre_install_commands == []
def test_not_array(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'pre-install-commands': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.pre-install-commands` must be an array'):
_ = environment.pre_install_commands
def test_entry_not_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'pre-install-commands': [9000]}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
TypeError, match='Command #1 of field `tool.hatch.envs.default.pre-install-commands` must be a string'
):
_ = environment.pre_install_commands
def test_correct(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'pre-install-commands': ['baz test']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.pre_install_commands == ['baz test']
class TestPostInstallCommands:
def test_default(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.post_install_commands == environment.post_install_commands == []
def test_not_array(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'post-install-commands': 9000}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Field `tool.hatch.envs.default.post-install-commands` must be an array'):
_ = environment.post_install_commands
def test_entry_not_string(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'post-install-commands': [9000]}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(
TypeError, match='Command #1 of field `tool.hatch.envs.default.post-install-commands` must be a string'
):
_ = environment.post_install_commands
def test_correct(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'post-install-commands': ['baz test']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.post_install_commands == ['baz test']
class TestEnvVarOption:
def test_unset(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.get_env_var_option('foo') == ''
def test_set(self, isolation, isolated_data_dir, platform):
config = {'project': {'name': 'my_app', 'version': '0.0.1'}}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with EnvVars({'HATCH_ENV_TYPE_MOCK_FOO': 'bar'}):
assert environment.get_env_var_option('foo') == 'bar'
class TestContextFormatting:
def test_env_name(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': 'command {env_name}'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert list(environment.expand_command('foo')) == ['command default']
def test_env_type(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': 'command {env_type}'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert list(environment.expand_command('foo')) == ['command mock']
def test_verbosity_default(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': 'command -v={verbosity}'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
9000,
)
assert list(environment.expand_command('foo')) == ['command -v=9000']
def test_verbosity_unknown_modifier(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': 'command {verbosity:bar}'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(ValueError, match='Unknown verbosity modifier: bar'):
next(environment.expand_command('foo'))
def test_verbosity_flag_adjustment_not_integer(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': 'command {verbosity:flag:-1.0}'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(TypeError, match='Verbosity flag adjustment must be an integer: -1.0'):
next(environment.expand_command('foo'))
@pytest.mark.parametrize(
('verbosity', 'command'),
[
(-9000, 'command -qqq'),
(-3, 'command -qqq'),
(-2, 'command -qq'),
(-1, 'command -q'),
(0, 'command'),
(1, 'command -v'),
(2, 'command -vv'),
(3, 'command -vvv'),
(9000, 'command -vvv'),
],
)
def test_verbosity_flag_default(self, isolation, isolated_data_dir, platform, verbosity, command):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': 'command {verbosity:flag}'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
verbosity,
)
assert list(environment.expand_command('foo')) == [command]
@pytest.mark.parametrize(
('adjustment', 'command'),
[
(-9000, 'command -qqq'),
(-3, 'command -qqq'),
(-2, 'command -qq'),
(-1, 'command -q'),
(0, 'command'),
(1, 'command -v'),
(2, 'command -vv'),
(3, 'command -vvv'),
(9000, 'command -vvv'),
],
)
def test_verbosity_flag_adjustment(self, isolation, isolated_data_dir, platform, adjustment, command):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': f'command {{verbosity:flag:{adjustment}}}'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert list(environment.expand_command('foo')) == [command]
def test_args_undefined(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': 'command {args}'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert list(environment.expand_command('foo')) == ['command']
def test_args_default(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': 'command {args: -bar > /dev/null}'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert list(environment.expand_command('foo')) == ['command -bar > /dev/null']
def test_args_default_override(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'scripts': {'foo': 'command {args: -bar > /dev/null}'}}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert list(environment.expand_command('foo baz')) == ['command baz']
def test_matrix_no_selection(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'dependencies': ['pkg=={matrix}']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(ValueError, match='The `matrix` context formatting field requires a modifier'):
_ = environment.dependencies
def test_matrix_no_default(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'dependencies': ['pkg=={matrix:bar}']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
with pytest.raises(ValueError, match='Nonexistent matrix variable must set a default: bar'):
_ = environment.dependencies
def test_matrix_default(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'dependencies': ['pkg=={matrix:bar:9000}']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.dependencies == ['pkg==9000']
def test_matrix_default_override(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {'hatch': {'envs': {'default': {'dependencies': ['pkg=={matrix:bar:baz}']}}}},
}
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{'bar': '42'},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.dependencies == ['pkg==42']
def test_env_vars_override(self, isolation, isolated_data_dir, platform):
config = {
'project': {'name': 'my_app', 'version': '0.0.1'},
'tool': {
'hatch': {
'envs': {
'default': {
'dependencies': ['pkg{env:DEP_PIN}'],
'env-vars': {'DEP_PIN': '==0.0.1'},
'overrides': {'env': {'DEP_ANY': {'env-vars': 'DEP_PIN='}}},
},
},
},
},
}
with EnvVars({'DEP_ANY': 'true'}):
project = Project(isolation, config=config)
environment = MockEnvironment(
isolation,
project.metadata,
'default',
project.config.envs['default'],
{},
isolated_data_dir,
isolated_data_dir,
platform,
0,
)
assert environment.dependencies == ['pkg']