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

Explicitly add schemars to ruff_python_ast Cargo.toml #12275

Merged
merged 2 commits into from
Jul 11, 2024

Conversation

GaetanLepage
Copy link
Contributor

@GaetanLepage GaetanLepage commented Jul 10, 2024

Summary

Context: Updating ruff from 0.5.0 to 0.5.1 on nixpkgs.

We noticed that since 5109b50, ruff was not building successfully in our pipeline:

thread 'main' panicked at cargo-auditable/src/collect_audit_data.rs:77:9:
error: Package `ruff_python_ast v0.0.0 (/build/source/crates/ruff_python_ast)` does not have feature `schemars`. It has an optional dependency with that name, but that dependency us>

note: run with `RUST_BACKTRACE=1` environment variable to display a backtrace
error: could not compile `ruff_python_formatter` (bin "ruff_python_formatter")

This patch by @eljamm seems to fix it on our end.

Test Plan

We were able to build ruff 0.5.1 with this patch and its test suite is passing.

@MichaReiser
Copy link
Member

Thanks for the PR.

I'm hesitant merging the PR because it's unclear to me where this feature is used (I can't find any usage in our code). Ideally we would change the usage rather than the features exposed by the crate.

@MichaReiser
Copy link
Member

Do you have a link to the CI log where the build on nix fails? I wonder what command it is using

@GaetanLepage
Copy link
Contributor Author

GaetanLepage commented Jul 10, 2024

Do you have a link to the CI log where the build on nix fails? I wonder what command it is using

Here is the entire log. I am afraid that it won't help you much more than the error message I shared.
https://pastebin.com/RCDPtNmj (the build command is at line 50).

It is weird, because manually running the exact same cargo build command on the exact same source files works fine.
Our sandbox might be doing something weird here, but I have no idea what.

@eljamm
Copy link

eljamm commented Jul 10, 2024

After a bit more of digging, this is my best guess of what's going on. According to Features - The Cargo Book:

By default, this optional dependency implicitly defines a feature that looks like this:

[features]
gif = ["dep:gif"]

But, it also says that:

If you specify the optional dependency with the dep: prefix anywhere in the [features] table, that disables the implicit feature

Which is exactly what's happening here:

[features]
serde = ["dep:serde", "ruff_text_size/serde", "dep:ruff_cache", "compact_str/serde", "dep:ruff_macros", "dep:schemars"]

This is what the error is complaining about :

error: Package ruff_python_ast v0.0.0 (/build/source/crates/ruff_python_ast) does not have feature schemars. It has an optional dependency with that name, but that dependency uses the "dep:" syntax in the features table, so it does not have an implicit feature with that name.

Adding schemars = ["dep:schemars"] to the features satisfies this condition and makes the program compile correctly.

But why is this happening? I have no clue 🤷

schemars is apparently needed in ruff_python_ast/src/name.rs#L183

#[cfg(feature = "serde")]
impl schemars::JsonSchema for Name {
    fn is_referenceable() -> bool {
        String::is_referenceable()
    }
...

Because when I remove "dep:schemars" from the features:

 [features]
-serde = ["dep:serde", "ruff_text_size/serde", "dep:ruff_cache", "compact_str/serde", "dep:ruff_macros", "dep:schemars"]
+serde = ["dep:serde", "ruff_text_size/serde", "dep:ruff_cache", "compact_str/serde", "dep:ruff_macros"]

That's where it complains about it being missing in ruff.log

But even though schemars is included as a dep of serde, it's not being used. In other crates like ruff_python_formatter, we can see that schemars is defined independently from serde, so I don't know why it's different for ruff_python_ast.

Note: Another solution is to make schemars non-optional:

non-optional-schemars.patch
---
 crates/ruff_python_ast/Cargo.toml | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/crates/ruff_python_ast/Cargo.toml b/crates/ruff_python_ast/Cargo.toml
index bd41c71b6..e4e8e4304 100644
--- a/crates/ruff_python_ast/Cargo.toml
+++ b/crates/ruff_python_ast/Cargo.toml
@@ -25,12 +25,12 @@ is-macro = { workspace = true }
 itertools = { workspace = true }
 once_cell = { workspace = true }
 rustc-hash = { workspace = true }
-schemars = { workspace = true, optional = true }
+schemars = { workspace = true }
 serde = { workspace = true, optional = true }
 compact_str = { workspace = true }
 
 [features]
-serde = ["dep:serde", "ruff_text_size/serde", "dep:ruff_cache", "compact_str/serde", "dep:ruff_macros", "dep:schemars"]
+serde = ["dep:serde", "ruff_text_size/serde", "dep:ruff_cache", "compact_str/serde", "dep:ruff_macros"]
 
 [lints]
 workspace = true
-- 
2.45.1

@MichaReiser
Copy link
Member

Your analysis is correct. The part that I don't understand is that the error can only occur if some code adds a dependency to ruff_python_ast with features=["schemars"] or has a feature that enables ruff_python_asts schemars feature, which doesn't exist.

I haven't found any code that enables that feature or requires that feature from ruff_python_ast. The only reference I've seen reference the serde feature, which does exist.

@eljamm
Copy link

eljamm commented Jul 10, 2024

I finally found an explanation of why this is happening, from parcel-bundler/lightningcss#713 (comment):

cargo-auditable is a tool to store the exact versions of dependencies in rust binaries and makes it easier/possible to audit the dependency tree of rust binaries.

It is a drop-in replacement of cargo. Mainly this lets easier identification of vulnerabilities (CVEs). Nixpkgs by default uses cargo auditable to build rust binaries.

Stricter checks like these are implemented by cargo-auditable. And hence this PR. cargo auditable is compatible with cargo itself, so this change won't affect builds with cargo.

As proposed by the PR above, to fix this issues for cargo-auditable, it's enough to just remove the dep: prefix:

 [features]
-serde = ["dep:serde", "ruff_text_size/serde", "dep:ruff_cache", "compact_str/serde", "dep:ruff_macros", "dep:schemars"]
+serde = ["dep:serde", "ruff_text_size/serde", "dep:ruff_cache", "compact_str/serde", "dep:ruff_macros", "schemars"]

With this, ruff compiled successfully both manually (with cargo) and in nixpkgs (with cargo-auditable).

I tracked the main issue behind this down to rust-secure-code/cargo-auditable#124 which appears to be caused by cargo in which it passes the wrong features to cargo-auditable.

It would be nice to fix this in ruff, but I'm also worried that it would be confusing as schemars is marked optional but is not called with the dep: prefix. If this is wrongly assumed to be a mistake in the future, it can be re-added and thus break the compilation again in nixpkgs.

What do you think @MichaReiser @GaetanLepage ?

@MichaReiser
Copy link
Member

Wow nice find! I think what I'll do is to split the feature into three:

  • schemars
  • Serde
  • Cache

Which is also more consistent with other crates and should reduce the risk of breaking in the future because someone readds the dep prefix

@eljamm eljamm mentioned this pull request Jul 10, 2024
13 tasks
@GaetanLepage
Copy link
Contributor Author

Thanks so much @eljamm for this amazing deep dive !
What should we do now ? Should I apply your patch in this PR ?

@eljamm
Copy link

eljamm commented Jul 10, 2024

I'll leave that call to @MichaReiser if he wants to add this change now or wait until he implements his suggestion.

@MichaReiser
Copy link
Member

The exact command to reproduce the error is cargo auditable build --target x86_64-unknown-linux-gnu Building without a target succeeds

@MichaReiser MichaReiser added the internal An internal refactor or improvement label Jul 11, 2024
@MichaReiser MichaReiser enabled auto-merge (squash) July 11, 2024 06:43
@MichaReiser MichaReiser merged commit d0298dc into astral-sh:main Jul 11, 2024
17 checks passed
@GaetanLepage GaetanLepage deleted the cargo branch July 11, 2024 06:46
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
internal An internal refactor or improvement
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants