Builder plugins¶
See the documentation for build configuration.
Built-in¶
Wheel¶
A wheel is a binary distribution of a Python package that can be installed directly into an environment.
Configuration¶
The builder plugin name is wheel
.
[tool.hatch.build.targets.wheel]
[build.targets.wheel]
Options¶
Option | Default | Description |
---|---|---|
core-metadata-version | "2.1" | The version of core metadata to use |
shared-data | A mapping similar to the explicit selection option corresponding to data that will be installed globally in a given Python environment, usually under sys.prefix | |
extra-metadata | A mapping similar to the explicit selection option corresponding to extra metadata that will be shipped in a directory named extra_metadata |
Versions¶
Version | Description |
---|---|
standard (default) | The latest standardized format |
editable | A wheel that only ships .pth files or import hooks for real-time development |
Default file selection¶
When the user has not set any file selection options, the project name will be used to determine the package to ship in the following heuristic order:
<PACKAGE>/__init__.py
src/<PACKAGE>/__init__.py
<NAMESPACE>/<PACKAGE>/__init__.py
- Otherwise, every Python package and file that does not start with the word
test
will be included
Reproducibility¶
Reproducible builds are supported.
Build data¶
This is data that can be modified by build hooks.
Data | Default | Description |
---|---|---|
tag | The full tag part of the filename (e.g. py3-none-any ), defaulting to a cross-platform wheel with the supported major versions of Python based on project metadata | |
infer_tag | False | When tag is not set, this may be enabled to use the one most specific to the platform, Python interpreter, and ABI |
pure_python | True | Whether or not to write metadata indicating that the package does not contain any platform-specific files |
dependencies | Extra project dependencies | |
force_include_editable | Similar to the force_include option but specifically for the editable version |
Source distribution¶
A source distribution, or sdist
, is an archive of Python "source code". Although largely unspecified, by convention it should include everything that is required to build a wheel without making network requests.
Configuration¶
The builder plugin name is sdist
.
[tool.hatch.build.targets.sdist]
[build.targets.sdist]
Options¶
Option | Default | Description |
---|---|---|
core-metadata-version | "2.1" | The version of core metadata to use |
support-legacy | false | Whether or not to include a setup.py file to support legacy installation mechanisms |
Versions¶
Version | Description |
---|---|
standard (default) | The latest conventional format |
Default file selection¶
When the user has not set any file selection options, all files that are not ignored by your VCS will be included.
Reproducibility¶
Reproducible builds are supported.
Build data¶
This is data that can be modified by build hooks.
Data | Default | Description |
---|---|---|
dependencies | Extra project dependencies |
Custom¶
This is a custom class in a given Python file that inherits from the BuilderInterface.
Configuration¶
The builder plugin name is custom
.
[tool.hatch.build.targets.custom]
[build.targets.custom]
An option path
is used to specify the path of the Python file, defaulting to hatch_build.py
.
Example¶
from hatchling.builders.plugin.interface import BuilderInterface
class CustomBuilder(BuilderInterface):
...
If multiple subclasses are found, you must define a function named get_builder
that returns the desired builder.
Note
Any defined PLUGIN_NAME is ignored and will always be custom
.
BuilderInterface (ABC)
¶
Example usage:
from hatchling.builders.plugin.interface import BuilderInterface
class SpecialBuilder(BuilderInterface):
PLUGIN_NAME = 'special'
...
from hatchling.plugin import hookimpl
from .plugin import SpecialBuilder
@hookimpl
def hatch_register_builder():
return SpecialBuilder
Source code in hatchling/builders/plugin/interface.py
class BuilderInterface(ABC):
"""
Example usage:
=== ":octicons-file-code-16: plugin.py"
```python
from hatchling.builders.plugin.interface import BuilderInterface
class SpecialBuilder(BuilderInterface):
PLUGIN_NAME = 'special'
...
```
=== ":octicons-file-code-16: hooks.py"
```python
from hatchling.plugin import hookimpl
from .plugin import SpecialBuilder
@hookimpl
def hatch_register_builder():
return SpecialBuilder
```
"""
PLUGIN_NAME = ''
"""The name used for selection."""
def __init__(self, root, plugin_manager=None, config=None, metadata=None, app=None):
self.__root = root
self.__plugin_manager = plugin_manager
self.__raw_config = config
self.__metadata = metadata
self.__app = app
self.__config = None
self.__project_config = None
self.__hatch_config = None
self.__build_config = None
self.__build_targets = None
self.__target_config = None
# Metadata
self.__project_id = None
def build(
self,
directory=None,
versions=None,
hooks_only=None,
clean=None,
clean_hooks_after=None,
clean_only=False,
) -> Generator[str, None, None]:
# Fail early for invalid project metadata
self.metadata.validate_fields()
if directory is None:
if BuildEnvVars.LOCATION in os.environ:
directory = self.config.normalize_build_directory(os.environ[BuildEnvVars.LOCATION])
else:
directory = self.config.directory
if not os.path.isdir(directory):
os.makedirs(directory)
version_api = self.get_version_api()
if not versions:
versions = self.config.versions
else:
unknown_versions = set(versions) - set(version_api)
if unknown_versions:
raise ValueError(
f'Unknown versions for target `{self.PLUGIN_NAME}`: {", ".join(map(str, sorted(unknown_versions)))}'
)
if hooks_only is None:
hooks_only = env_var_enabled(BuildEnvVars.HOOKS_ONLY)
configured_build_hooks = self.get_build_hooks(directory)
build_hooks = list(configured_build_hooks.values())
if clean_only:
clean = True
elif clean is None:
clean = env_var_enabled(BuildEnvVars.CLEAN)
if clean:
if not hooks_only:
self.clean(directory, versions)
for build_hook in build_hooks:
build_hook.clean(versions)
if clean_only:
return
if clean_hooks_after is None:
clean_hooks_after = env_var_enabled(BuildEnvVars.CLEAN_HOOKS_AFTER)
for version in versions:
self.app.display_debug(f'Building `{self.PLUGIN_NAME}` version `{version}`')
build_data = self.get_default_build_data()
# Make sure reserved fields are set
build_data.setdefault('artifacts', [])
build_data.setdefault('force_include', {})
# Allow inspection of configured build hooks and the order in which they run
build_data['build_hooks'] = tuple(configured_build_hooks)
# Execute all `initialize` build hooks
for build_hook in build_hooks:
build_hook.initialize(version, build_data)
if hooks_only:
self.app.display_debug(f'Only ran build hooks for `{self.PLUGIN_NAME}` version `{version}`')
continue
# Build the artifact
with self.config.set_build_data(build_data):
artifact = version_api[version](directory, **build_data)
# Execute all `finalize` build hooks
for build_hook in build_hooks:
build_hook.finalize(version, build_data, artifact)
if clean_hooks_after:
for build_hook in build_hooks:
build_hook.clean([version])
yield artifact
def recurse_included_files(self) -> Generator[IncludedFile, None, None]:
"""
Returns a consistently generated series of file objects for every file that should be distributed. Each file
object has three `str` attributes:
- `path` - the absolute path
- `relative_path` - the path relative to the project root; will be an empty string for external files
- `distribution_path` - the path to be distributed as
"""
for project_file in self.recurse_project_files():
yield project_file
for explicit_file in self.recurse_explicit_files():
yield explicit_file
def recurse_project_files(self) -> Generator[IncludedFile, None, None]:
for root, dirs, files in safe_walk(self.root):
relative_path = os.path.relpath(root, self.root)
# First iteration
if relative_path == '.':
relative_path = ''
if self.config.skip_excluded_dirs:
dirs[:] = sorted(
d
for d in dirs
# The trailing slash is necessary so e.g. `bar/` matches `foo/bar`
if not self.config.path_is_excluded(f'{os.path.join(relative_path, d)}/')
)
else:
dirs.sort()
files.sort()
is_package = '__init__.py' in files
for f in files:
relative_file_path = os.path.join(relative_path, f)
if self.config.include_path(relative_file_path, is_package=is_package):
yield IncludedFile(
os.path.join(root, f), relative_file_path, self.config.get_distribution_path(relative_file_path)
)
def recurse_explicit_files(self, inclusion_map=None) -> Generator[IncludedFile, None, None]:
if inclusion_map is None:
inclusion_map = self.config.get_force_include()
for source, target_path in inclusion_map.items():
external = not source.startswith(self.root)
if os.path.isfile(source):
yield IncludedFile(source, '' if external else os.path.relpath(source, self.root), target_path)
elif os.path.isdir(source):
for root, dirs, files in safe_walk(source):
relative_path = os.path.relpath(root, source)
# First iteration
if relative_path == '.':
relative_path = ''
dirs.sort()
files.sort()
for f in files:
relative_file_path = os.path.join(relative_path, f)
yield IncludedFile(
os.path.join(root, f),
'' if external else os.path.relpath(relative_file_path, self.root),
os.path.join(target_path, relative_file_path),
)
@property
def root(self):
"""
The root of the project tree.
"""
return self.__root
@property
def plugin_manager(self):
if self.__plugin_manager is None:
from ...plugin.manager import PluginManager
self.__plugin_manager = PluginManager()
return self.__plugin_manager
@property
def metadata(self):
if self.__metadata is None:
from ...metadata.core import ProjectMetadata
self.__metadata = ProjectMetadata(self.root, self.plugin_manager, self.__raw_config)
return self.__metadata
@property
def app(self):
"""
An instance of [Application](utilities.md#hatchling.bridge.app.Application).
"""
if self.__app is None:
from ...bridge.app import Application
self.__app = Application().get_safe_application()
return self.__app
@property
def raw_config(self):
if self.__raw_config is None:
self.__raw_config = self.metadata.config
return self.__raw_config
@property
def project_config(self):
if self.__project_config is None:
self.__project_config = self.metadata.core.config
return self.__project_config
@property
def hatch_config(self):
if self.__hatch_config is None:
self.__hatch_config = self.metadata.hatch.config
return self.__hatch_config
@property
def config(self):
"""
An instance of [BuilderConfig](utilities.md#hatchling.builders.config.BuilderConfig).
"""
if self.__config is None:
self.__config = self.get_config_class()(
self, self.root, self.PLUGIN_NAME, self.build_config, self.target_config
)
return self.__config
@property
def build_config(self):
"""
=== ":octicons-file-code-16: pyproject.toml"
```toml
[tool.hatch.build]
```
=== ":octicons-file-code-16: hatch.toml"
```toml
[build]
```
"""
if self.__build_config is None:
self.__build_config = self.metadata.hatch.build_config
return self.__build_config
@property
def target_config(self):
"""
=== ":octicons-file-code-16: pyproject.toml"
```toml
[tool.hatch.build.targets.<PLUGIN_NAME>]
```
=== ":octicons-file-code-16: hatch.toml"
```toml
[build.targets.<PLUGIN_NAME>]
```
"""
if self.__target_config is None:
target_config = self.metadata.hatch.build_targets.get(self.PLUGIN_NAME, {})
if not isinstance(target_config, dict):
raise TypeError(f'Field `tool.hatch.build.targets.{self.PLUGIN_NAME}` must be a table')
self.__target_config = target_config
return self.__target_config
@property
def project_id(self):
if self.__project_id is None:
# https://discuss.python.org/t/clarify-naming-of-dist-info-directories/5565
self.__project_id = f'{self.normalize_file_name_component(self.metadata.core.name)}-{self.metadata.version}'
return self.__project_id
def get_build_hooks(self, directory):
configured_build_hooks = OrderedDict()
for hook_name, config in self.config.hook_config.items():
build_hook = self.plugin_manager.build_hook.get(hook_name)
if build_hook is None:
from ...plugin.exceptions import UnknownPluginError
raise UnknownPluginError(f'Unknown build hook: {hook_name}')
configured_build_hooks[hook_name] = build_hook(
self.root, config, self.config, self.metadata, directory, self.PLUGIN_NAME, self.app
)
return configured_build_hooks
@abstractmethod
def get_version_api(self) -> dict[str, Callable]:
"""
A mapping of `str` versions to a callable that is used for building.
Each callable must have the following signature:
```python
def ...(build_dir: str, build_data: dict) -> str:
```
The return value must be the absolute path to the built artifact.
"""
def get_default_versions(self):
"""
A list of versions to build when users do not specify any, defaulting to all versions.
"""
return list(self.get_version_api())
def get_default_build_data(self):
"""
A mapping that can be modified by [build hooks](build-hook.md) to influence the behavior of builds.
"""
return {}
def clean(self, directory, versions):
"""
Called before builds if the `-c`/`--clean` flag was passed to the
[`build`](../cli/reference.md#hatch-build) command.
"""
@classmethod
def get_config_class(cls):
"""
Must return a subclass of [BuilderConfig](utilities.md#hatchling.builders.config.BuilderConfig).
"""
return BuilderConfig
@staticmethod
def normalize_file_name_component(file_name):
"""
https://peps.python.org/pep-0427/#escaping-and-unicode
"""
return re.sub(r'[^\w\d.]+', '_', file_name, re.UNICODE)
PLUGIN_NAME
¶
The name used for selection.
app
property
readonly
¶
An instance of Application.
build_config
property
readonly
¶
[tool.hatch.build]
[build]
config
property
readonly
¶
An instance of BuilderConfig.
root
property
readonly
¶
The root of the project tree.
target_config
property
readonly
¶
[tool.hatch.build.targets.<PLUGIN_NAME>]
[build.targets.<PLUGIN_NAME>]
clean(self, directory, versions)
¶
Called before builds if the -c
/--clean
flag was passed to the build
command.
Source code in hatchling/builders/plugin/interface.py
def clean(self, directory, versions):
"""
Called before builds if the `-c`/`--clean` flag was passed to the
[`build`](../cli/reference.md#hatch-build) command.
"""
get_config_class()
classmethod
¶
Must return a subclass of BuilderConfig.
Source code in hatchling/builders/plugin/interface.py
@classmethod
def get_config_class(cls):
"""
Must return a subclass of [BuilderConfig](utilities.md#hatchling.builders.config.BuilderConfig).
"""
return BuilderConfig
get_default_build_data(self)
¶
A mapping that can be modified by build hooks to influence the behavior of builds.
Source code in hatchling/builders/plugin/interface.py
def get_default_build_data(self):
"""
A mapping that can be modified by [build hooks](build-hook.md) to influence the behavior of builds.
"""
return {}
get_default_versions(self)
¶
A list of versions to build when users do not specify any, defaulting to all versions.
Source code in hatchling/builders/plugin/interface.py
def get_default_versions(self):
"""
A list of versions to build when users do not specify any, defaulting to all versions.
"""
return list(self.get_version_api())
get_version_api(self) -> dict[str, Callable]
¶
A mapping of str
versions to a callable that is used for building. Each callable must have the following signature:
def ...(build_dir: str, build_data: dict) -> str:
The return value must be the absolute path to the built artifact.
Source code in hatchling/builders/plugin/interface.py
@abstractmethod
def get_version_api(self) -> dict[str, Callable]:
"""
A mapping of `str` versions to a callable that is used for building.
Each callable must have the following signature:
```python
def ...(build_dir: str, build_data: dict) -> str:
```
The return value must be the absolute path to the built artifact.
"""
recurse_included_files(self) -> Generator[IncludedFile, None, None]
¶
Returns a consistently generated series of file objects for every file that should be distributed. Each file object has three str
attributes:
path
- the absolute pathrelative_path
- the path relative to the project root; will be an empty string for external filesdistribution_path
- the path to be distributed as
Source code in hatchling/builders/plugin/interface.py
def recurse_included_files(self) -> Generator[IncludedFile, None, None]:
"""
Returns a consistently generated series of file objects for every file that should be distributed. Each file
object has three `str` attributes:
- `path` - the absolute path
- `relative_path` - the path relative to the project root; will be an empty string for external files
- `distribution_path` - the path to be distributed as
"""
for project_file in self.recurse_project_files():
yield project_file
for explicit_file in self.recurse_explicit_files():
yield explicit_file