Skip to content
This repository has been archived by the owner on Jul 9, 2023. It is now read-only.

Tool to enforce that Cargo.toml dependencies are written in sorted order #29

Closed
dtolnay opened this issue Apr 8, 2019 · 20 comments · Fixed by #37
Closed

Tool to enforce that Cargo.toml dependencies are written in sorted order #29

dtolnay opened this issue Apr 8, 2019 · 20 comments · Fixed by #37

Comments

@dtolnay
Copy link
Owner

dtolnay commented Apr 8, 2019

$ cargo dependencies-sorted? path/to/Cargo.toml; echo $?
0

Large projects tend to like for things to stay sorted to cut down the occurrence of merge conflicts between two developers appending to the end of the same unsorted list in parallel changes. Similar idea behind the remain crate.

The command should parse a Cargo.toml and validate that particular pieces are sorted, exiting 0 if all sorted and nonzero otherwise.

We would at least want to check:

  • [dependencies]
  • [dev-dependencies]
  • [build-dependencies]
  • [workspace.members]
  • [workspace.exclude]

Take a look at the documentation for Cargo.toml to see whether any other parts make sense to enforce as sorted.

@jpoles1
Copy link

jpoles1 commented Apr 10, 2019

I'm somewhat new to Rust, but I'd like to try to take a stab at this one as a way to learn some more!

@dtolnay
Copy link
Owner Author

dtolnay commented Apr 10, 2019

Yay! Let me know how it goes or if you get stuck on any part.

@jpoles1
Copy link

jpoles1 commented Apr 10, 2019

Thanks @dtolnay! Will keep you updated as the project progresses.

@jpoles1
Copy link

jpoles1 commented Apr 12, 2019

Still working on some of the basic functionality, and I've gotta write tests for everything, but I've got a prototype up and running at this repo: https://github.com/jpoles1/cargo-dep-sort

As a note, I had originally wanted to use the TOML crate, but as far as I can tell, parsing with this library does not respect the order of the TOML tables, and as such is not suitable for this application. It looks like it's possible to enable preservation of ordering, but this is an experimental feature, and thus requires nightly rustc builds to work, which I wanted to avoid for the time being.

@dtolnay
Copy link
Owner Author

dtolnay commented Apr 12, 2019

You will need to use the toml crate. Regex are not a reliable way to interpret a TOML file. The preserve_order feature is neither experimental nor requires nightly. You can turn it on as:

[dependencies]
toml = { version = "0.5", features = ["preserve_order"] }

@jpoles1
Copy link

jpoles1 commented Apr 13, 2019

Ok, will give that try, thanks for the guidance!

@jpoles1
Copy link

jpoles1 commented Apr 17, 2019

Nothing perfect, but I've got the basic functionality up and running using the toml crate. My big concern right now is writing tests.

One other thing I'm having some trouble with is a feature to write sorted Cargo.toml files out of the software. The big issue is that the serialization features of the toml crate are limited, such that you can't use inline tables (like the toml dep line you suggested to me above). Still trying to work out a way around that.

@pickfire
Copy link

Should this be in clippy instead?

@polybuildr
Copy link

In case it helps, in rust-lang/rust#57443 (comment) I came across the toml_edit crate (a format preserving TOML parser that cargo-edit uses). It has some issues but works well overall.

@gsquire
Copy link

gsquire commented Jun 1, 2019

I decided to give this a go and published an initial version here. I will add some unit tests soon and ensure that it works as expected. Please file issues if you find any!

Edit: Another idea could be an actual cargo sub-command that uses cargo_metadata.

@dtolnay
Copy link
Owner Author

dtolnay commented Jun 21, 2019

Thanks @gsquire! I wasn't able to get it working (gsquire/toml-sorted#2) but once that works I'll close out this issue.

@DevinR528
Copy link

DevinR528 commented Jun 23, 2019

I added some to @jpoles1, implementation wondering if I'm moving in the right direction https://github.com/DevinR528/cargo-dep-sort. Rewriting the file is the goal right? Having it work like cargo itself, by assuming everything from cwd, is probably the most ergonomic?

@dtolnay
Copy link
Owner Author

dtolnay commented Jun 23, 2019

In my case I wouldn't want it rewriting the file, just a reliable 0 or 1 exit code and consideration for common patterns like out-of-line dependency sections (syn in cargo-expand/Cargo.toml).

@DevinR528
Copy link

So fail if:

[dependencies.b]
version = "0.15"

[dependencies.a]
version = "0.15"

@dtolnay
Copy link
Owner Author

dtolnay commented Jun 23, 2019

Yes I would want that to exit nonzero. The out-of-line dependencies should need to be sorted among themselves.

@Zykino
Copy link

Zykino commented Jun 23, 2019

Maybe it is not your use case but having an option to override the file could be nice. Or at least write the correct order somewhere: console/file.

@DevinR528
Copy link

DevinR528 commented Jun 24, 2019

Here's my crack at sort checking cargo.toml you can install it cargo install cargo-sort-ck. I'm glad I found this repo any excuse to write some rust is a good one thanks @dtolnay. @Zykino I also started adding that feature to https://github.com/DevinR528/cargo-dep-sort cloned from @jpoles1.

@DevinR528
Copy link

cargo-sort-ck now has a toml tokenizer, as such can handle comments, nested headers, out of group nested headers and each "token" impls display so printing to a file would be as easy as .to_string().

@DevinR528
Copy link

cargo-sort-ck is finished, other than bug fixes. Any feed back would be much appreciated, working alone can create a real echo chamber.

Thanks

@dtolnay
Copy link
Owner Author

dtolnay commented Jul 6, 2019

Thanks @DevinR528! This is terrific. I filed some issues for you but I am really impressed by your crate.

Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

Successfully merging a pull request may close this issue.

7 participants