diff --git a/docs/ext_lock_import.md b/docs/ext_lock_import.md index c3cf4f50..0264558c 100644 --- a/docs/ext_lock_import.md +++ b/docs/ext_lock_import.md @@ -12,8 +12,9 @@ lock_import.import_pdm( default_alias_single_version, default_build_dependencies, development_groups, disallow_builds, local_wheels, lock_file, optional_groups, project_file, repo, require_static_urls, target_environments) -lock_import.import_poetry(default_alias_single_version, default_build_dependencies, disallow_builds, - local_wheels, lock_file, project_file, repo, target_environments) +lock_import.import_poetry(all_optional_groups, default, default_alias_single_version, + default_build_dependencies, disallow_builds, local_wheels, lock_file, + optional_groups, project_file, repo, target_environments) lock_import.import_uv(all_development_groups, all_optional_groups, default, default_alias_single_version, default_build_dependencies, development_groups, disallow_builds, local_wheels, lock_file, optional_groups, project_file, repo, @@ -60,11 +61,14 @@ Import a Poetry lock file. | Name | Description | Type | Mandatory | Default | | :------------- | :------------- | :------------- | :------------- | :------------- | +| all_optional_groups | Install all optional dependencies. | Boolean | optional | `False` | +| default | Whether to install dependencies from the default group. | Boolean | optional | `True` | | default_alias_single_version | Generate aliases for all packages that have a single version in the lock file. | Boolean | optional | `False` | | default_build_dependencies | A list of package keys (name or name@version) that will be used as default build dependencies. | List of strings | optional | `[]` | | disallow_builds | If True, only pre-built wheels are allowed. | Boolean | optional | `False` | | local_wheels | A list of local .whl files to consider when processing lock files. | List of labels | optional | `[]` | | lock_file | The poetry.lock file. | Label | required | | +| optional_groups | List of optional dependency groups to install. | List of strings | optional | `[]` | | project_file | The pyproject.toml file. | Label | required | | | repo | The repository name | String | required | | | target_environments | A list of target environment descriptors. | List of labels | optional | `["@pycross_environments//:environments"]` | diff --git a/docs/rules.md b/docs/rules.md index 5092c834..b4057c50 100644 --- a/docs/rules.md +++ b/docs/rules.md @@ -70,7 +70,8 @@ pycross_pdm_lock_model(name, -pycross_poetry_lock_model(name, lock_file, project_file) +pycross_poetry_lock_model(name, all_optional_groups, default, lock_file, optional_groups, + project_file) @@ -81,7 +82,10 @@ pycross_poetry_lock_model(name, name | A unique name for this target. | Name | required | | +| all_optional_groups | Install all optional dependencies. | Boolean | optional | `False` | +| default | Whether to install dependencies from the default group. | Boolean | optional | `True` | | lock_file | The poetry.lock file. | Label | required | | +| optional_groups | List of optional dependency groups to install. | List of strings | optional | `[]` | | project_file | The pyproject.toml file. | Label | required | | diff --git a/docs/workspace_rules.md b/docs/workspace_rules.md index 393f57c5..c06991a0 100644 --- a/docs/workspace_rules.md +++ b/docs/workspace_rules.md @@ -33,7 +33,7 @@ lock_repo_model_pdm(project_file ## lock_repo_model_poetry
-lock_repo_model_poetry(project_file, lock_file) +lock_repo_model_poetry(project_file, lock_file, default, optional_groups, all_optional_groups)@@ -45,6 +45,9 @@ lock_repo_model_poetry(project_fi | :------------- | :------------- | :------------- | | project_file |
-
| none | | lock_file |-
| none | +| default |-
| `True` | +| optional_groups |-
| `[]` | +| all_optional_groups |-
| `False` | diff --git a/examples/poetry/BUILD.bazel b/examples/poetry/BUILD.bazel index f54608e8..91e24cb2 100644 --- a/examples/poetry/BUILD.bazel +++ b/examples/poetry/BUILD.bazel @@ -73,6 +73,7 @@ pycross_target_environment( pycross_poetry_lock_model( name = "example_lock_model", lock_file = "poetry.lock", + optional_groups = ["dev"], project_file = "pyproject.toml", ) diff --git a/examples/poetry/example_lock.bzl b/examples/poetry/example_lock.bzl index f8a4cc31..aec7e575 100755 --- a/examples/poetry/example_lock.bzl +++ b/examples/poetry/example_lock.bzl @@ -22,6 +22,7 @@ PINS = { "charset-normalizer": "charset-normalizer@3.3.2", "click": "click@8.1.7", "cognitojwt": "cognitojwt@1.4.1", + "cowsay": "cowsay@6.1", "cryptography": "cryptography@41.0.5", "cython": "cython@0.29.36", "decorator": "decorator@5.1.1", @@ -367,6 +368,16 @@ def targets(): wheel = ":_wheel_cognitojwt@1.4.1", ) + native.alias( + name = "_wheel_cowsay@6.1", + actual = "@example_lock_wheel_cowsay_6.1_py3_none_any//file", + ) + + pycross_wheel_library( + name = "cowsay@6.1", + wheel = ":_wheel_cowsay@6.1", + ) + _cryptography_41_0_5_deps = [ ":cffi@1.16.0", ] @@ -1874,6 +1885,16 @@ def repositories(): index = "https://pypi.org", ) + maybe( + pypi_file, + name = "example_lock_wheel_cowsay_6.1_py3_none_any", + package_name = "cowsay", + package_version = "6.1", + filename = "cowsay-6.1-py3-none-any.whl", + sha256 = "274b1e6fc1b966d53976333eb90ac94cb07a450a700b455af9fbdf882244b30a", + index = "https://pypi.org", + ) + maybe( pypi_file, name = "example_lock_wheel_cryptography_41.0.5_cp37_abi3_macosx_10_12_universal2", diff --git a/examples/poetry/poetry.lock b/examples/poetry/poetry.lock index ac2774da..df1faade 100644 --- a/examples/poetry/poetry.lock +++ b/examples/poetry/poetry.lock @@ -2632,3 +2632,13 @@ files = [ lock-version = "2.0" python-versions = ">=3.11,<3.13" content-hash = "058183c94e35a7f1f3105fac99063ccd83b2c240b6e0552711b551b34bd7c72d" + +[[package]] +name = "cowsay" +version = "6.1" +description = "The famous cowsay for GNU/Linux is now available for python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "cowsay-6.1-py3-none-any.whl", hash = "sha256:274b1e6fc1b966d53976333eb90ac94cb07a450a700b455af9fbdf882244b30a"}, +] diff --git a/examples/poetry/pyproject.toml b/examples/poetry/pyproject.toml index d93a2afb..5d7a2cc7 100644 --- a/examples/poetry/pyproject.toml +++ b/examples/poetry/pyproject.toml @@ -21,3 +21,6 @@ tree-sitter = "=0.20.2" future = "=0.18.2" opencv-python = "=4.6.0.66" keyring = "=23.9.1" + +[tool.poetry.group.dev.dependencies] +cowsay = "^6.1" diff --git a/examples/poetry/tools/BUILD.bazel b/examples/poetry/tools/BUILD.bazel index 1ccd6559..07789c3f 100644 --- a/examples/poetry/tools/BUILD.bazel +++ b/examples/poetry/tools/BUILD.bazel @@ -18,3 +18,10 @@ py_binary( main = "ipython.py", deps = [requirement("ipython")], ) + +py_binary( + name = "app", + srcs = ["app.py"], + main = "app.py", + deps = [requirement("cowsay")], +) diff --git a/examples/poetry/tools/app.py b/examples/poetry/tools/app.py new file mode 100644 index 00000000..1cb1ab92 --- /dev/null +++ b/examples/poetry/tools/app.py @@ -0,0 +1,3 @@ +import cowsay + +cowsay.cow("Hello rules_pycross") diff --git a/pycross/private/lock_attrs.bzl b/pycross/private/lock_attrs.bzl index 89f85b55..8eb09a35 100644 --- a/pycross/private/lock_attrs.bzl +++ b/pycross/private/lock_attrs.bzl @@ -139,6 +139,16 @@ POETRY_IMPORT_ATTRS = dict( allow_single_file = True, mandatory = True, ), + default = attr.bool( + doc = "Whether to install dependencies from the default group.", + default = True, + ), + optional_groups = attr.string_list( + doc = "List of optional dependency groups to install.", + ), + all_optional_groups = attr.bool( + doc = "Install all optional dependencies.", + ), ) def handle_resolve_attrs(attrs, environment_files_and_labels, local_wheel_names_and_labels): diff --git a/pycross/private/poetry_lock_model.bzl b/pycross/private/poetry_lock_model.bzl index ffda441e..5ba89115 100644 --- a/pycross/private/poetry_lock_model.bzl +++ b/pycross/private/poetry_lock_model.bzl @@ -5,13 +5,32 @@ load(":lock_attrs.bzl", "POETRY_IMPORT_ATTRS") TRANSLATOR_TOOL = Label("//pycross/private/tools:poetry_translator.py") +def _handle_args(attrs, project_file, lock_file, output): + args = [] + args.extend(["--project-file", project_file]) + args.extend(["--lock-file", lock_file]) + args.extend(["--output", output]) + + if attrs.default: + args.append("--default") + + for group in attrs.optional_groups: + args.extend(["--optional-group", group]) + + return args + def _pycross_poetry_lock_model_impl(ctx): out = ctx.actions.declare_file(ctx.attr.name + ".json") args = ctx.actions.args().use_param_file("--flagfile=%s") - args.add("--project-file", ctx.file.project_file) - args.add("--lock-file", ctx.file.lock_file) - args.add("--output", out) + args.add_all( + _handle_args( + ctx.attr, + ctx.file.project_file.path, + ctx.file.lock_file.path, + out.path, + ), + ) ctx.actions.run( inputs = ( @@ -40,11 +59,14 @@ pycross_poetry_lock_model = rule( } | POETRY_IMPORT_ATTRS, ) -def lock_repo_model_poetry(*, project_file, lock_file): +def lock_repo_model_poetry(*, project_file, lock_file, default = True, optional_groups = [], all_optional_groups = False): return json.encode(dict( model_type = "poetry", project_file = str(project_file), lock_file = str(lock_file), + default = default, + optional_groups = optional_groups, + all_optional_groups = all_optional_groups, )) def repo_create_poetry_model(rctx, params, output): @@ -59,14 +81,12 @@ def repo_create_poetry_model(rctx, params, output): attrs = struct(**params) else: attrs = params - args = [ - "--project-file", + args = _handle_args( + attrs, str(rctx.path(Label(attrs.project_file))), - "--lock-file", str(rctx.path(Label(attrs.lock_file))), - "--output", output, - ] + ) exec_internal_tool( rctx, diff --git a/pycross/private/tools/poetry_translator.py b/pycross/private/tools/poetry_translator.py index c80b6edc..e767adc1 100644 --- a/pycross/private/tools/poetry_translator.py +++ b/pycross/private/tools/poetry_translator.py @@ -99,7 +99,13 @@ def get_files_for_package( return result -def translate(project_file: Path, lock_file: Path) -> RawLockSet: +def translate( + project_file: Path, + lock_file: Path, + default_group: bool, + optional_groups: List[str], + all_optional_groups: bool, +) -> RawLockSet: try: with open(project_file, "rb") as f: project_dict = tomli.load(f) @@ -113,7 +119,19 @@ def translate(project_file: Path, lock_file: Path) -> RawLockSet: raise Exception(f"Could not load lock file: {lock_file}: {e}") pinned_package_specs = {} - for pin, pin_info in (project_dict.get("tool", {}).get("poetry", {}).get("dependencies", {})).items(): + + dependency_items = [] + + if default_group: + dependency_items.extend((project_dict.get("tool", {}).get("poetry", {}).get("dependencies", {})).items()) + + groups = project_dict.get("tool", {}).get("poetry", {}).get("group", {}) + + for group_name, group in groups.items(): + if all_optional_groups or group_name in optional_groups: + dependency_items.extend(group.get("dependencies", {}).items()) + + for pin, pin_info in dependency_items: pin = package_canonical_name(pin) if pin == "python": # Skip the special line indicating python version. @@ -239,7 +257,13 @@ def parse_file_info(file_info) -> PackageFile: def main(args: Any) -> None: output = args.output - lock_set = translate(args.project_file, args.lock_file) + lock_set = translate( + project_file=args.project_file, + lock_file=args.lock_file, + default_group=args.default, + optional_groups=args.optional_group, + all_optional_groups=args.all_optional_groups, + ) with open(output, "w") as f: f.write(lock_set.to_json(indent=2)) @@ -262,6 +286,25 @@ def parse_flags() -> Any: help="The path to pdm.lock.", ) + parser.add_argument( + "--default", + action="store_true", + help="Whether to install dependencies from the default group.", + ) + + parser.add_argument( + "--optional-group", + action="append", + default=[], + help="Optional dependency groups to install.", + ) + + parser.add_argument( + "--all-optional-groups", + action="store_true", + help="Install all optional dependency groups.", + ) + parser.add_argument( "--output", type=Path,