Skip to content

Commit 3a983c3

Browse files
committed
Merge lintcheck popular-crates bin as a subcommand
1 parent a2c9782 commit 3a983c3

File tree

6 files changed

+77
-88
lines changed

6 files changed

+77
-88
lines changed

lintcheck/Cargo.toml

+1-12
Original file line numberDiff line numberDiff line change
@@ -11,30 +11,19 @@ publish = false
1111
default-run = "lintcheck"
1212

1313
[dependencies]
14-
anyhow = "1.0.69"
1514
cargo_metadata = "0.15.3"
1615
clap = { version = "4.4", features = ["derive", "env"] }
17-
crates_io_api = "0.8.1"
1816
crossbeam-channel = "0.5.6"
1917
diff = "0.1.13"
2018
flate2 = "1.0"
21-
indicatif = "0.17.3"
2219
rayon = "1.5.1"
2320
serde = { version = "1.0", features = ["derive"] }
2421
serde_json = "1.0.85"
2522
strip-ansi-escapes = "0.1.1"
2623
tar = "0.4"
2724
toml = "0.7.3"
28-
ureq = "2.2"
25+
ureq = { version = "2.2", features = ["json"] }
2926
walkdir = "2.3"
3027

3128
[features]
3229
deny-warnings = []
33-
34-
[[bin]]
35-
name = "lintcheck"
36-
path = "src/main.rs"
37-
38-
[[bin]]
39-
name = "popular-crates"
40-
path = "src/popular-crates.rs"

lintcheck/README.md

+3-3
Original file line numberDiff line numberDiff line change
@@ -26,11 +26,11 @@ the repo root.
2626
The results will then be saved to `lintcheck-logs/custom_logs.toml`.
2727

2828
The `custom.toml` file may be built using <https://crates.io> recently most
29-
downloaded crates by using the `popular-crates` binary from the `lintcheck`
30-
directory. For example, to retrieve the 100 recently most downloaded crates:
29+
downloaded crates by using `cargo lintcheck popular`. For example, to retrieve
30+
the 200 recently most downloaded crates:
3131

3232
```
33-
cargo run --release --bin popular-crates -- -n 100 custom.toml
33+
cargo lintcheck popular -n 200 custom.toml
3434
```
3535

3636

lintcheck/src/config.rs

+9
Original file line numberDiff line numberDiff line change
@@ -48,7 +48,16 @@ pub(crate) struct LintcheckConfig {
4848

4949
#[derive(Subcommand, Clone, Debug)]
5050
pub(crate) enum Commands {
51+
/// Display a markdown diff between two lintcheck log files in JSON format
5152
Diff { old: PathBuf, new: PathBuf },
53+
/// Create a lintcheck crates TOML file containing the top N popular crates
54+
Popular {
55+
/// Output TOML file name
56+
output: PathBuf,
57+
/// Number of crate names to download
58+
#[clap(short, long, default_value_t = 100)]
59+
number: usize,
60+
},
5261
}
5362

5463
#[derive(ValueEnum, Debug, Clone, Copy, PartialEq, Eq)]

lintcheck/src/main.rs

+12-8
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
mod config;
1818
mod driver;
1919
mod json;
20+
mod popular_crates;
2021
mod recursive;
2122

2223
use crate::config::{Commands, LintcheckConfig, OutputFormat};
@@ -43,21 +44,21 @@ const LINTCHECK_DOWNLOADS: &str = "target/lintcheck/downloads";
4344
const LINTCHECK_SOURCES: &str = "target/lintcheck/sources";
4445

4546
/// List of sources to check, loaded from a .toml file
46-
#[derive(Debug, Serialize, Deserialize)]
47+
#[derive(Debug, Deserialize)]
4748
struct SourceList {
4849
crates: HashMap<String, TomlCrate>,
4950
#[serde(default)]
5051
recursive: RecursiveOptions,
5152
}
5253

53-
#[derive(Debug, Serialize, Deserialize, Default)]
54+
#[derive(Debug, Deserialize, Default)]
5455
struct RecursiveOptions {
5556
ignore: HashSet<String>,
5657
}
5758

5859
/// A crate source stored inside the .toml
5960
/// will be translated into on one of the `CrateSource` variants
60-
#[derive(Debug, Serialize, Deserialize)]
61+
#[derive(Debug, Deserialize)]
6162
struct TomlCrate {
6263
name: String,
6364
versions: Option<Vec<String>>,
@@ -69,7 +70,7 @@ struct TomlCrate {
6970

7071
/// Represents an archive we download from crates.io, or a git repo, or a local repo/folder
7172
/// Once processed (downloaded/extracted/cloned/copied...), this will be translated into a `Crate`
72-
#[derive(Debug, Serialize, Deserialize, Eq, Hash, PartialEq, Ord, PartialOrd)]
73+
#[derive(Debug, Deserialize, Eq, Hash, PartialEq, Ord, PartialOrd)]
7374
enum CrateSource {
7475
CratesIo {
7576
name: String,
@@ -609,7 +610,6 @@ fn gather_stats(warnings: &[ClippyWarning]) -> (String, HashMap<&String, usize>)
609610
(stats_string, counter)
610611
}
611612

612-
#[allow(clippy::too_many_lines)]
613613
fn main() {
614614
// We're being executed as a `RUSTC_WRAPPER` as part of `--recursive`
615615
if let Ok(addr) = env::var("LINTCHECK_SERVER") {
@@ -624,11 +624,15 @@ fn main() {
624624

625625
let config = LintcheckConfig::new();
626626

627-
if let Some(Commands::Diff { old, new }) = config.subcommand {
628-
json::diff(&old, &new);
629-
return;
627+
match config.subcommand {
628+
Some(Commands::Diff { old, new }) => json::diff(&old, &new),
629+
Some(Commands::Popular { output, number }) => popular_crates::fetch(output, number).unwrap(),
630+
None => lintcheck(config),
630631
}
632+
}
631633

634+
#[allow(clippy::too_many_lines)]
635+
fn lintcheck(config: LintcheckConfig) {
632636
println!("Compiling clippy...");
633637
build_clippy();
634638
println!("Done compiling");

lintcheck/src/popular-crates.rs

-65
This file was deleted.

lintcheck/src/popular_crates.rs

+52
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
use serde::Deserialize;
2+
use std::error::Error;
3+
use std::fmt::Write;
4+
use std::fs;
5+
use std::path::PathBuf;
6+
7+
#[derive(Deserialize, Debug)]
8+
struct Page {
9+
crates: Vec<Crate>,
10+
meta: Meta,
11+
}
12+
13+
#[derive(Deserialize, Debug)]
14+
struct Crate {
15+
name: String,
16+
max_version: String,
17+
}
18+
19+
#[derive(Deserialize, Debug)]
20+
struct Meta {
21+
next_page: String,
22+
}
23+
24+
pub(crate) fn fetch(output: PathBuf, number: usize) -> Result<(), Box<dyn Error>> {
25+
let agent = ureq::builder()
26+
.user_agent("clippy/lintcheck (github.com/rust-lang/rust-clippy/)")
27+
.build();
28+
29+
let mut crates = Vec::with_capacity(number);
30+
let mut query = "?sort=recent-downloads&per_page=100".to_string();
31+
while crates.len() < number {
32+
let page: Page = agent
33+
.get(&format!("https://crates.io/api/v1/crates{query}"))
34+
.call()?
35+
.into_json()?;
36+
37+
query = page.meta.next_page;
38+
crates.extend(page.crates);
39+
crates.truncate(number);
40+
41+
let width = number.ilog10() as usize + 1;
42+
println!("Fetched {:>width$}/{number} crates", crates.len());
43+
}
44+
45+
let mut out = "[crates]\n".to_string();
46+
for Crate { name, max_version } in crates {
47+
writeln!(out, "{name} = {{ name = '{name}', versions = ['{max_version}'] }}").unwrap();
48+
}
49+
fs::write(output, out)?;
50+
51+
Ok(())
52+
}

0 commit comments

Comments
 (0)