Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

RFC: Give users control over feature unification #3692

Merged
merged 37 commits into from
Nov 2, 2024
Merged
Changes from 3 commits
Commits
Show all changes
37 commits
Select commit Hold shift + click to select a range
97d49c2
Skeleton
epage Sep 11, 2024
c3b17e8
Initial
epage Sep 11, 2024
a921e09
Set RFC number
epage Sep 11, 2024
7d6810d
Fix typo
epage Sep 12, 2024
5b13f93
Fix typo
epage Sep 12, 2024
750b5f1
Fix typo
epage Sep 12, 2024
4ebf0ff
Remove boilerplate
epage Sep 12, 2024
703a636
Remove copy/pasted header
epage Sep 12, 2024
e7a4f27
Move mutually exclusive features to a Drawback
epage Sep 12, 2024
f2bb64a
Cover no_std with std features being unified
epage Sep 12, 2024
c5bbeee
Fix typo
epage Sep 12, 2024
65e6988
Note cargo-hakari's build/host unification features
epage Sep 12, 2024
e752830
Add build/host unification qualification
epage Sep 12, 2024
0cebf5f
Fix typo
epage Sep 12, 2024
363be82
Add citation
epage Sep 12, 2024
6860e51
Note ecosystem importance of build/host unification
epage Sep 12, 2024
8b114b3
Fix typos
epage Sep 12, 2024
4bbdc51
fix: Clarify some points
epage Sep 17, 2024
8f848a4
Expand on future possibilities
epage Sep 17, 2024
c2a4ac1
Expand on dev/normal unification
epage Sep 17, 2024
d35c626
fix: Typo
epage Sep 19, 2024
760b618
fix: Clarify statement
epage Sep 19, 2024
cf07a12
fix: Whatever
epage Sep 19, 2024
51839e5
fix: Clarify statements
epage Sep 19, 2024
6318f47
fix: Clarify statement
epage Sep 19, 2024
cd2af9a
fix: Typi
epage Sep 19, 2024
1b2a3d2
fix: Clarify statement
epage Sep 19, 2024
eb8ffb2
fix: Typo
epage Sep 19, 2024
ff5e82a
fix: Link out to another related Issue
epage Sep 20, 2024
0aac9ab
fix: Remove issue link
epage Sep 30, 2024
c0158e7
fix: Disambiguate each use of 'target'
epage Oct 1, 2024
969cb03
Make cargo-install interaction explicit
epage Oct 2, 2024
a071fe4
alt: Build X as if its a dependency of Y
epage Oct 3, 2024
0e1fbc4
fix: Clarify summary
epage Oct 3, 2024
e3fa072
fix: Clarify this RFC offers no new concepts
epage Oct 3, 2024
f2075e1
fix(drawbacks): Put more emphasis on relationship with --workspace, m…
epage Oct 28, 2024
203e6fc
Add tracking issue
ehuss Nov 2, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
130 changes: 130 additions & 0 deletions text/3692-feature-unification.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,130 @@
- Feature Name: `feature-unification`
- Start Date: 2024-09-11
- RFC PR: [rust-lang/rfcs#3692](https://github.com/rust-lang/rfcs/pull/3692)
- Rust Issue: [rust-lang/rust#3692](https://github.com/rust-lang/rust/issues/3692)
epage marked this conversation as resolved.
Show resolved Hide resolved

# Summary
[summary]: #summary

Allow users to control feature unifcation.
epage marked this conversation as resolved.
Show resolved Hide resolved

Related issues:
- [#4463: Feature selection in workspace depends on the set of packages compiled](https://github.com/rust-lang/cargo/issues/4463)
- [#8157: --bin B resolves features differently than -p B in a workspace](https://github.com/rust-lang/cargo/issues/8157)
- [#13844: The cargo build --bins re-builds binaries again after cargo build --all-targets](https://github.com/rust-lang/cargo/issues/13844)

# Motivation
[motivation]: #motivation

Today, when Cargo is building, features in dependencies are enabled baed on the set of packages selected to build.
This is an attempt to balance
- Build speed: we should reuse builds between packages within the same invocation
- Ability to verify features for a given package

This isn't always ideal.

If a user is building an application, they may be jumping around the application's components which are packages within the workspace.
The final artifact is the same but Cargo will select different features depending on which package they are currently building,
causing build churn for the same set of dependencies that, in the end, will only be used with the same set of features.
The "cargo-workspace-hack" is a pattern that has existed for years
(e.g. [`rustc-workspace-hack`](https://crates.io/crates/rustc-workspace-hack))
where users have all workspace members that depend on a generated package that depends on direct-dependemncies in the workspace along with their features.
epage marked this conversation as resolved.
Show resolved Hide resolved
Tools like [`cargo-hakari`](https://crates.io/crates/cargo-hakari) automate this process.
To allow others to pull in a package depending on a workspace-hack package as a git dependency, you then need to publish the workspace-hack as an empty package with no dependencies
and then locally patch in the real instance of it.

This also makes testing of features more difficult because a user can't just run `cargo check --workspace` to verify that the correct set of features are enabled.
This has led to the rise of tools like [cargo-hack](https://crates.io/crates/cargo-hack) which de-unify packages.

# Guide-level explanation
[guide-level-explanation]: #guide-level-explanation

Explain the proposal as if it was already included in the language and you were teaching it to another Rust programmer. That generally means:

- Introducing new named concepts.
- Explaining the feature largely in terms of examples.
- Explaining how Rust programmers should *think* about the feature, and how it should impact the way they use Rust. It should explain the impact as concretely as possible.
- If applicable, provide sample error messages, deprecation warnings, or migration guidance.
- If applicable, describe the differences between teaching this to existing Rust programmers and new Rust programmers.
- Discuss how this impacts the ability to read, understand, and maintain Rust code. Code is read and modified far more often than written; will the proposed feature make code easier to maintain?

For implementation-oriented RFCs (e.g. for compiler internals), this section should focus on how compiler contributors should think about the change, and give examples of its concrete impact. For policy RFCs, this section should provide an example-driven introduction to the policy, and explain its impact in concrete terms.
epage marked this conversation as resolved.
Show resolved Hide resolved

# Reference-level explanation
[reference-level-explanation]: #reference-level-explanation

### Rust Version
epage marked this conversation as resolved.
Show resolved Hide resolved

We'll add two new modes to feature unifcation:
epage marked this conversation as resolved.
Show resolved Hide resolved

**Unify features across the workspace, independent of the selected packages**

This would be built-in support for "cargo-workspace-hack".

This would require effectively changing from
1. Resolve dependencies
2. Filter dependencies down for current target and selected packages
3. Resolve features

To
1. Resolve dependencies
2. Filter dependencies down for current target
3. Resolve features
4. Filter for selected packages

**Features will be evaluated for each package in isolation**
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The main situation I encounter this with is a workspace that has a single package, but the different cargo targets have different dependencies and that affects feature resolution in a way that cargo build vs cargo test leads to rebuilds. Will this RFC help there? If yes, then the text should be clarified, because it seems to talk entirely about packages but not about targets within a package. If not -- what would it take to help in that situation?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for calling this out! I had considered this but forgot it and didn't want to hold it up until I could remember.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@sunshowers says cargo-hakari will unify normal and dev-dependencies. Platforms are not unified.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@sunshowers says that is also unifies the features that workspace member features enable

Copy link
Contributor Author

@epage epage Sep 12, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Some cases with unifying dev-dependencies include

  • fail being used for dev-dependencies but not production
  • Tests enabling std but not wanting them for normal build
  • Some use case related to private keys where the impl Clone is only provided on debug builds and not release builds

There are likely other cases of this same sort of problem

Copy link
Member

@RalfJung RalfJung Sep 12, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In Miri's case it's dev-dependencies enabling different features for libc or some other fundamental crate, which means that even after cargo test, doing cargo build does an almost complete re-build of Miri -- that's waiting time I'd rather avoid. :)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've summarized this under "Future possibilities"


This will require building duplicate copies of build units when there is disjoint sets of features.
epage marked this conversation as resolved.
Show resolved Hide resolved

For example purposes., this could be implemented as either
epage marked this conversation as resolved.
Show resolved Hide resolved
- Loop over the packages, resolving, and then run a build plan for that package
- Resolve for each package and generate everything into the same build plan

This is not prescriptive of the implementation but to illustrate what the feature does.

**Note:** these features do not need to be stabilized together.

##### `resolver.feature-unification`

*(update to [Configuration](https://doc.rust-lang.org/cargo/reference/config.html))*

* Type: string
* Default: "selected"
* Environment: `CARGO_RESOLVER_FEATURE_UNIFICATION`

Specify which packages participate in [feature unification](https://doc.rust-lang.org/cargo/reference/features.html#feature-unification).

* `selected`: merge dependency features from all package specified for the current build
* `workspace`: merge dependency features across all workspace members, regardless of which packages are specified for the current build
* `package`: dependency features are only enabled for each package, preferring duplicate builds of dependencies to when different feature sets are selected
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
* `package`: dependency features are only enabled for each package, preferring duplicate builds of dependencies to when different feature sets are selected
* `package`: dependency features are only enabled for each package, preferring duplicate builds of dependencies when different feature sets are selected by their dependents


# Drawbacks
[drawbacks]: #drawbacks

This increases entropy within Cargo and the universe at large.

# Rationale and alternatives
[rationale-and-alternatives]: #rationale-and-alternatives

This is done in the config instead of the manifest:
- As this can change from run-to-run, this covers more use cases
- As this fits easily into the `resolver` table. there is less design work
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@dlight

Could this be added into the workspace's Cargo.toml rather than .cargo/config.toml? Or is this intended to be applied to all workspaces inside a directory?

(moved here for easier to follow thread)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We do mention manifest support in future possibilities.

Also a config doesn't require it to apply to all workspaces. It depends on where you put your config, where your workspaces are, and what directory you are in.

epage marked this conversation as resolved.
Show resolved Hide resolved

This will not support exceptions for mutually exclusive features because those are officially unsupported.
Instead, effort should be put towards [official mutually exclusive globals](https://internals.rust-lang.org/t/pre-rfc-mutually-excusive-global-features/19618).

# Prior art
[prior-art]: #prior-art

[`cargo-hakari`](https://crates.io/crates/cargo-hakari) is a "cargo-workspace-hack" generator that builds a graph off of `cargo metadata` and re-implements feature unification.

[cargo-hack](https://crates.io/crates/cargo-hack) can run each selected package in a separate `cargo` invocation to prevent unification.

# Unresolved questions
[unresolved-questions]: #unresolved-questions


# Future possibilities
[future-possibilities]: #future-possibilities

Add a related field to manifests that the config can override.