rustlings/src/project.rs

99 lines
3.2 KiB
Rust
Raw Normal View History

use anyhow::{bail, Context, Result};
2022-06-16 06:53:41 +03:00
use glob::glob;
use serde::{Deserialize, Serialize};
use std::env;
2022-06-16 06:53:41 +03:00
use std::error::Error;
use std::path::PathBuf;
2024-03-25 19:14:41 +03:00
use std::process::{Command, Stdio};
2022-06-16 06:53:41 +03:00
/// Contains the structure of resulting rust-project.json file
/// and functions to build the data required to create the file
#[derive(Serialize, Deserialize)]
pub struct RustAnalyzerProject {
sysroot_src: String,
pub crates: Vec<Crate>,
}
#[derive(Serialize, Deserialize)]
pub struct Crate {
root_module: String,
edition: String,
deps: Vec<String>,
cfg: Vec<String>,
}
impl RustAnalyzerProject {
pub fn build() -> Result<Self> {
// check if RUST_SRC_PATH is set
if let Ok(sysroot_src) = env::var("RUST_SRC_PATH") {
return Ok(Self {
sysroot_src,
crates: Vec::new(),
});
2022-06-16 06:53:41 +03:00
}
let toolchain = Command::new("rustc")
.arg("--print")
.arg("sysroot")
2024-03-25 19:14:41 +03:00
.stderr(Stdio::inherit())
.output()
.context("Failed to get the sysroot from `rustc`. Do you have `rustc` installed?")?
.stdout;
let toolchain =
String::from_utf8(toolchain).context("The toolchain path is invalid UTF8")?;
let toolchain = toolchain.trim_end();
println!("Determined toolchain: {toolchain}\n");
let mut sysroot_src = PathBuf::with_capacity(256);
sysroot_src.extend([toolchain, "lib", "rustlib", "src", "rust", "library"]);
let Ok(sysroot_src) = sysroot_src.into_os_string().into_string() else {
bail!("The sysroot path is invalid UTF8");
};
Ok(Self {
sysroot_src,
crates: Vec::new(),
})
2022-06-16 06:53:41 +03:00
}
/// Write rust-project.json to disk
pub fn write_to_disk(&self) -> Result<(), std::io::Error> {
// Using the capacity 2^14 = 16384 since the file length in bytes is higher than 2^13.
// The final length is not known exactly because it depends on the user's sysroot path,
// the current number of exercises etc.
let mut buf = Vec::with_capacity(16384);
serde_json::to_writer(&mut buf, &self).expect("Failed to serialize to JSON");
std::fs::write("rust-project.json", buf)?;
2022-06-16 06:53:41 +03:00
Ok(())
}
/// If path contains .rs extension, add a crate to `rust-project.json`
fn path_to_json(&mut self, path: PathBuf) -> Result<(), Box<dyn Error>> {
if let Some(ext) = path.extension() {
2022-06-16 06:53:41 +03:00
if ext == "rs" {
self.crates.push(Crate {
root_module: path.display().to_string(),
2022-06-16 06:53:41 +03:00
edition: "2021".to_string(),
deps: Vec::new(),
// This allows rust_analyzer to work inside #[test] blocks
cfg: vec!["test".to_string()],
})
}
}
Ok(())
2022-06-16 06:53:41 +03:00
}
/// Parse the exercises folder for .rs files, any matches will create
/// a new `crate` in rust-project.json which allows rust-analyzer to
/// treat it like a normal binary
2023-01-03 16:14:07 +03:00
pub fn exercises_to_json(&mut self) -> Result<(), Box<dyn Error>> {
for path in glob("./exercises/**/*")? {
self.path_to_json(path?)?;
2022-06-16 06:53:41 +03:00
}
Ok(())
}
}