2024-03-26 00:41:14 +03:00
|
|
|
use anyhow::{Context, Result};
|
2024-03-26 00:30:16 +03:00
|
|
|
use serde::Serialize;
|
2023-05-17 22:04:32 +03:00
|
|
|
use std::env;
|
2022-06-16 06:53:41 +03:00
|
|
|
use std::error::Error;
|
2024-03-25 05:59:21 +03:00
|
|
|
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
|
|
|
|
2024-03-26 00:20:00 +03:00
|
|
|
use crate::exercise::Exercise;
|
|
|
|
|
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
|
2024-03-26 00:30:16 +03:00
|
|
|
#[derive(Serialize)]
|
2022-06-16 06:53:41 +03:00
|
|
|
pub struct RustAnalyzerProject {
|
2024-03-26 00:41:14 +03:00
|
|
|
sysroot_src: PathBuf,
|
2024-03-26 00:29:33 +03:00
|
|
|
crates: Vec<Crate>,
|
2022-06-16 06:53:41 +03:00
|
|
|
}
|
|
|
|
|
2024-03-26 00:30:16 +03:00
|
|
|
#[derive(Serialize)]
|
2024-03-26 00:41:14 +03:00
|
|
|
struct Crate {
|
|
|
|
root_module: PathBuf,
|
|
|
|
edition: &'static str,
|
|
|
|
// Not used, but required in the JSON file.
|
|
|
|
deps: Vec<()>,
|
|
|
|
cfg: [&'static str; 1],
|
2022-06-16 06:53:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl RustAnalyzerProject {
|
2024-03-25 05:49:10 +03:00
|
|
|
pub fn build() -> Result<Self> {
|
|
|
|
// check if RUST_SRC_PATH is set
|
2024-03-26 00:41:14 +03:00
|
|
|
if let Some(path) = env::var_os("RUST_SRC_PATH") {
|
2024-03-25 05:49:10 +03:00
|
|
|
return Ok(Self {
|
2024-03-26 00:41:14 +03:00
|
|
|
sysroot_src: PathBuf::from(path),
|
2024-03-25 05:49:10 +03:00
|
|
|
crates: Vec::new(),
|
|
|
|
});
|
2022-06-16 06:53:41 +03:00
|
|
|
}
|
2024-03-25 05:49:10 +03:00
|
|
|
|
|
|
|
let toolchain = Command::new("rustc")
|
|
|
|
.arg("--print")
|
|
|
|
.arg("sysroot")
|
2024-03-25 19:14:41 +03:00
|
|
|
.stderr(Stdio::inherit())
|
2024-03-25 05:49:10 +03:00
|
|
|
.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");
|
|
|
|
|
2024-03-25 05:59:21 +03:00
|
|
|
let mut sysroot_src = PathBuf::with_capacity(256);
|
|
|
|
sysroot_src.extend([toolchain, "lib", "rustlib", "src", "rust", "library"]);
|
2024-03-25 05:49:10 +03:00
|
|
|
|
|
|
|
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> {
|
2024-03-25 05:33:14 +03:00
|
|
|
// 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(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 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
|
2024-03-26 00:20:00 +03:00
|
|
|
pub fn exercises_to_json(&mut self, exercises: Vec<Exercise>) -> Result<(), Box<dyn Error>> {
|
|
|
|
self.crates = exercises
|
|
|
|
.into_iter()
|
|
|
|
.map(|exercise| Crate {
|
2024-03-26 00:41:14 +03:00
|
|
|
root_module: exercise.path,
|
|
|
|
edition: "2021",
|
2024-03-26 00:20:00 +03:00
|
|
|
deps: Vec::new(),
|
|
|
|
// This allows rust_analyzer to work inside #[test] blocks
|
2024-03-26 00:41:14 +03:00
|
|
|
cfg: ["test"],
|
2024-03-26 00:20:00 +03:00
|
|
|
})
|
|
|
|
.collect();
|
2022-06-16 06:53:41 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|