mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-27 20:35:09 +00:00
Refactor xtasks
This commit is contained in:
parent
b02576d562
commit
91f9bc2b86
5 changed files with 288 additions and 279 deletions
|
@ -1,33 +0,0 @@
|
||||||
//! FIXME: write short doc here
|
|
||||||
|
|
||||||
pub const GLOBAL_HELP: &str = "tasks
|
|
||||||
|
|
||||||
USAGE:
|
|
||||||
ra_tools <SUBCOMMAND>
|
|
||||||
|
|
||||||
FLAGS:
|
|
||||||
-h, --help Prints help information
|
|
||||||
|
|
||||||
SUBCOMMANDS:
|
|
||||||
format
|
|
||||||
install-pre-commit-hook
|
|
||||||
fuzz-tests
|
|
||||||
codegen
|
|
||||||
install
|
|
||||||
lint";
|
|
||||||
|
|
||||||
pub const INSTALL_HELP: &str = "ra_tools-install
|
|
||||||
|
|
||||||
USAGE:
|
|
||||||
ra_tools.exe install [FLAGS]
|
|
||||||
|
|
||||||
FLAGS:
|
|
||||||
--client-code
|
|
||||||
-h, --help Prints help information
|
|
||||||
--jemalloc
|
|
||||||
--server";
|
|
||||||
|
|
||||||
pub const INSTALL_RA_CONFLICT: &str =
|
|
||||||
"error: The argument `--server` cannot be used with `--client-code`
|
|
||||||
|
|
||||||
For more information try --help";
|
|
178
xtask/src/install.rs
Normal file
178
xtask/src/install.rs
Normal file
|
@ -0,0 +1,178 @@
|
||||||
|
//! Installs rust-analyzer langauge server and/or editor plugin.
|
||||||
|
|
||||||
|
use std::{env, path::PathBuf, str};
|
||||||
|
|
||||||
|
use anyhow::{Context, Result};
|
||||||
|
|
||||||
|
use crate::{run, run_with_output, Cmd};
|
||||||
|
|
||||||
|
// Latest stable, feel free to send a PR if this lags behind.
|
||||||
|
const REQUIRED_RUST_VERSION: u32 = 40;
|
||||||
|
|
||||||
|
pub struct InstallCmd {
|
||||||
|
pub client: Option<ClientOpt>,
|
||||||
|
pub server: Option<ServerOpt>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub enum ClientOpt {
|
||||||
|
VsCode,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ServerOpt {
|
||||||
|
pub jemalloc: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl InstallCmd {
|
||||||
|
pub fn run(self) -> Result<()> {
|
||||||
|
if cfg!(target_os = "macos") {
|
||||||
|
fix_path_for_mac().context("Fix path for mac")?
|
||||||
|
}
|
||||||
|
if let Some(server) = self.server {
|
||||||
|
install_server(server).context("install server")?;
|
||||||
|
}
|
||||||
|
if let Some(client) = self.client {
|
||||||
|
install_client(client).context("install client")?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn fix_path_for_mac() -> Result<()> {
|
||||||
|
let mut vscode_path: Vec<PathBuf> = {
|
||||||
|
const COMMON_APP_PATH: &str =
|
||||||
|
r"/Applications/Visual Studio Code.app/Contents/Resources/app/bin";
|
||||||
|
const ROOT_DIR: &str = "";
|
||||||
|
let home_dir = match env::var("HOME") {
|
||||||
|
Ok(home) => home,
|
||||||
|
Err(e) => anyhow::bail!("Failed getting HOME from environment with error: {}.", e),
|
||||||
|
};
|
||||||
|
|
||||||
|
[ROOT_DIR, &home_dir]
|
||||||
|
.iter()
|
||||||
|
.map(|dir| String::from(*dir) + COMMON_APP_PATH)
|
||||||
|
.map(PathBuf::from)
|
||||||
|
.filter(|path| path.exists())
|
||||||
|
.collect()
|
||||||
|
};
|
||||||
|
|
||||||
|
if !vscode_path.is_empty() {
|
||||||
|
let vars = match env::var_os("PATH") {
|
||||||
|
Some(path) => path,
|
||||||
|
None => anyhow::bail!("Could not get PATH variable from env."),
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut paths = env::split_paths(&vars).collect::<Vec<_>>();
|
||||||
|
paths.append(&mut vscode_path);
|
||||||
|
let new_paths = env::join_paths(paths).context("build env PATH")?;
|
||||||
|
env::set_var("PATH", &new_paths);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn install_client(ClientOpt::VsCode: ClientOpt) -> Result<()> {
|
||||||
|
let npm_version = Cmd {
|
||||||
|
unix: r"npm --version",
|
||||||
|
windows: r"cmd.exe /c npm --version",
|
||||||
|
work_dir: "./editors/code",
|
||||||
|
}
|
||||||
|
.run();
|
||||||
|
|
||||||
|
if npm_version.is_err() {
|
||||||
|
eprintln!("\nERROR: `npm --version` failed, `npm` is required to build the VS Code plugin")
|
||||||
|
}
|
||||||
|
|
||||||
|
Cmd { unix: r"npm install", windows: r"cmd.exe /c npm install", work_dir: "./editors/code" }
|
||||||
|
.run()?;
|
||||||
|
Cmd {
|
||||||
|
unix: r"npm run package --scripts-prepend-node-path",
|
||||||
|
windows: r"cmd.exe /c npm run package",
|
||||||
|
work_dir: "./editors/code",
|
||||||
|
}
|
||||||
|
.run()?;
|
||||||
|
|
||||||
|
let code_binary = ["code", "code-insiders", "codium"].iter().find(|bin| {
|
||||||
|
Cmd {
|
||||||
|
unix: &format!("{} --version", bin),
|
||||||
|
windows: &format!("cmd.exe /c {}.cmd --version", bin),
|
||||||
|
work_dir: "./editors/code",
|
||||||
|
}
|
||||||
|
.run()
|
||||||
|
.is_ok()
|
||||||
|
});
|
||||||
|
|
||||||
|
let code_binary = match code_binary {
|
||||||
|
Some(it) => it,
|
||||||
|
None => anyhow::bail!("Can't execute `code --version`. Perhaps it is not in $PATH?"),
|
||||||
|
};
|
||||||
|
|
||||||
|
Cmd {
|
||||||
|
unix: &format!(r"{} --install-extension ./ra-lsp-0.0.1.vsix --force", code_binary),
|
||||||
|
windows: &format!(
|
||||||
|
r"cmd.exe /c {}.cmd --install-extension ./ra-lsp-0.0.1.vsix --force",
|
||||||
|
code_binary
|
||||||
|
),
|
||||||
|
work_dir: "./editors/code",
|
||||||
|
}
|
||||||
|
.run()?;
|
||||||
|
|
||||||
|
let output = Cmd {
|
||||||
|
unix: &format!(r"{} --list-extensions", code_binary),
|
||||||
|
windows: &format!(r"cmd.exe /c {}.cmd --list-extensions", code_binary),
|
||||||
|
work_dir: ".",
|
||||||
|
}
|
||||||
|
.run_with_output()?;
|
||||||
|
|
||||||
|
if !str::from_utf8(&output.stdout)?.contains("ra-lsp") {
|
||||||
|
anyhow::bail!(
|
||||||
|
"Could not install the Visual Studio Code extension. \
|
||||||
|
Please make sure you have at least NodeJS 10.x together with the latest version of VS Code installed and try again."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn install_server(opts: ServerOpt) -> Result<()> {
|
||||||
|
let mut old_rust = false;
|
||||||
|
if let Ok(output) = run_with_output("cargo --version", ".") {
|
||||||
|
if let Ok(stdout) = String::from_utf8(output.stdout) {
|
||||||
|
println!("{}", stdout);
|
||||||
|
if !check_version(&stdout, REQUIRED_RUST_VERSION) {
|
||||||
|
old_rust = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if old_rust {
|
||||||
|
eprintln!(
|
||||||
|
"\nWARNING: at least rust 1.{}.0 is required to compile rust-analyzer\n",
|
||||||
|
REQUIRED_RUST_VERSION,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
let res = if opts.jemalloc {
|
||||||
|
run("cargo install --path crates/ra_lsp_server --locked --force --features jemalloc", ".")
|
||||||
|
} else {
|
||||||
|
run("cargo install --path crates/ra_lsp_server --locked --force", ".")
|
||||||
|
};
|
||||||
|
|
||||||
|
if res.is_err() && old_rust {
|
||||||
|
eprintln!(
|
||||||
|
"\nWARNING: at least rust 1.{}.0 is required to compile rust-analyzer\n",
|
||||||
|
REQUIRED_RUST_VERSION,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
res
|
||||||
|
}
|
||||||
|
|
||||||
|
fn check_version(version_output: &str, min_minor_version: u32) -> bool {
|
||||||
|
// Parse second the number out of
|
||||||
|
// cargo 1.39.0-beta (1c6ec66d5 2019-09-30)
|
||||||
|
let minor: Option<u32> = version_output.split('.').nth(1).and_then(|it| it.parse().ok());
|
||||||
|
match minor {
|
||||||
|
None => true,
|
||||||
|
Some(minor) => minor >= min_minor_version,
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,13 +1,14 @@
|
||||||
//! FIXME: write short doc here
|
//! FIXME: write short doc here
|
||||||
|
|
||||||
pub mod codegen;
|
pub mod codegen;
|
||||||
|
pub mod install;
|
||||||
|
pub mod pre_commit;
|
||||||
mod ast_src;
|
mod ast_src;
|
||||||
|
|
||||||
use anyhow::Context;
|
use anyhow::Context;
|
||||||
pub use anyhow::Result;
|
pub use anyhow::Result;
|
||||||
use std::{
|
use std::{
|
||||||
env, fs,
|
env,
|
||||||
io::{Error as IoError, ErrorKind},
|
|
||||||
path::{Path, PathBuf},
|
path::{Path, PathBuf},
|
||||||
process::{Command, Output, Stdio},
|
process::{Command, Output, Stdio},
|
||||||
};
|
};
|
||||||
|
@ -79,23 +80,11 @@ pub fn run_rustfmt(mode: Mode) -> Result<()> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn install_rustfmt() -> Result<()> {
|
fn install_rustfmt() -> Result<()> {
|
||||||
run(&format!("rustup toolchain install {}", TOOLCHAIN), ".")?;
|
run(&format!("rustup toolchain install {}", TOOLCHAIN), ".")?;
|
||||||
run(&format!("rustup component add rustfmt --toolchain {}", TOOLCHAIN), ".")
|
run(&format!("rustup component add rustfmt --toolchain {}", TOOLCHAIN), ".")
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn install_pre_commit_hook() -> Result<()> {
|
|
||||||
let result_path =
|
|
||||||
PathBuf::from(format!("./.git/hooks/pre-commit{}", std::env::consts::EXE_SUFFIX));
|
|
||||||
if !result_path.exists() {
|
|
||||||
let me = std::env::current_exe()?;
|
|
||||||
fs::copy(me, result_path)?;
|
|
||||||
} else {
|
|
||||||
Err(IoError::new(ErrorKind::AlreadyExists, "Git hook already created"))?;
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn run_clippy() -> Result<()> {
|
pub fn run_clippy() -> Result<()> {
|
||||||
match Command::new("rustup")
|
match Command::new("rustup")
|
||||||
.args(&["run", TOOLCHAIN, "--", "cargo", "clippy", "--version"])
|
.args(&["run", TOOLCHAIN, "--", "cargo", "clippy", "--version"])
|
||||||
|
@ -144,28 +133,6 @@ pub fn run_fuzzer() -> Result<()> {
|
||||||
run("rustup run nightly -- cargo fuzz run parser", "./crates/ra_syntax")
|
run("rustup run nightly -- cargo fuzz run parser", "./crates/ra_syntax")
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn reformat_staged_files() -> Result<()> {
|
|
||||||
run_rustfmt(Mode::Overwrite)?;
|
|
||||||
let root = project_root();
|
|
||||||
let output = Command::new("git")
|
|
||||||
.arg("diff")
|
|
||||||
.arg("--diff-filter=MAR")
|
|
||||||
.arg("--name-only")
|
|
||||||
.arg("--cached")
|
|
||||||
.current_dir(&root)
|
|
||||||
.output()?;
|
|
||||||
if !output.status.success() {
|
|
||||||
anyhow::bail!(
|
|
||||||
"`git diff --diff-filter=MAR --name-only --cached` exited with {}",
|
|
||||||
output.status
|
|
||||||
);
|
|
||||||
}
|
|
||||||
for line in String::from_utf8(output.stdout)?.lines() {
|
|
||||||
run(&format!("git update-index --add {}", root.join(line).to_string_lossy()), ".")?;
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn do_run<F>(cmdline: &str, dir: &str, mut f: F) -> Result<Output>
|
fn do_run<F>(cmdline: &str, dir: &str, mut f: F) -> Result<Output>
|
||||||
where
|
where
|
||||||
F: FnMut(&mut Command),
|
F: FnMut(&mut Command),
|
||||||
|
|
|
@ -7,244 +7,105 @@
|
||||||
//!
|
//!
|
||||||
//! This binary is integrated into the `cargo` command line by using an alias in
|
//! This binary is integrated into the `cargo` command line by using an alias in
|
||||||
//! `.cargo/config`.
|
//! `.cargo/config`.
|
||||||
mod help;
|
|
||||||
|
|
||||||
use std::{env, fmt::Write, path::PathBuf, str};
|
use std::env;
|
||||||
|
|
||||||
use anyhow::Context;
|
|
||||||
use pico_args::Arguments;
|
use pico_args::Arguments;
|
||||||
use xtask::{
|
use xtask::{
|
||||||
codegen::{self, Mode},
|
codegen::{self, Mode},
|
||||||
install_pre_commit_hook, reformat_staged_files, run, run_clippy, run_fuzzer, run_rustfmt,
|
install::{ClientOpt, InstallCmd, ServerOpt},
|
||||||
run_with_output, Cmd, Result,
|
pre_commit, run_clippy, run_fuzzer, run_rustfmt, Result,
|
||||||
};
|
};
|
||||||
|
|
||||||
// Latest stable, feel free to send a PR if this lags behind.
|
|
||||||
const REQUIRED_RUST_VERSION: u32 = 40;
|
|
||||||
|
|
||||||
struct InstallOpt {
|
|
||||||
client: Option<ClientOpt>,
|
|
||||||
server: Option<ServerOpt>,
|
|
||||||
}
|
|
||||||
|
|
||||||
enum ClientOpt {
|
|
||||||
VsCode,
|
|
||||||
}
|
|
||||||
|
|
||||||
struct ServerOpt {
|
|
||||||
jemalloc: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
fn main() -> Result<()> {
|
fn main() -> Result<()> {
|
||||||
if env::args().next().map(|it| it.contains("pre-commit")) == Some(true) {
|
if env::args().next().map(|it| it.contains("pre-commit")) == Some(true) {
|
||||||
return reformat_staged_files();
|
return pre_commit::run_hook();
|
||||||
}
|
}
|
||||||
|
|
||||||
let subcommand = match std::env::args_os().nth(1) {
|
let subcommand = std::env::args().nth(1).unwrap_or_default();
|
||||||
None => {
|
let mut args = Arguments::from_vec(std::env::args_os().skip(2).collect());
|
||||||
eprintln!("{}", help::GLOBAL_HELP);
|
|
||||||
return Ok(());
|
match subcommand.as_str() {
|
||||||
}
|
|
||||||
Some(s) => s,
|
|
||||||
};
|
|
||||||
let mut matches = Arguments::from_vec(std::env::args_os().skip(2).collect());
|
|
||||||
let subcommand = &*subcommand.to_string_lossy();
|
|
||||||
match subcommand {
|
|
||||||
"install" => {
|
"install" => {
|
||||||
if matches.contains(["-h", "--help"]) {
|
if args.contains(["-h", "--help"]) {
|
||||||
eprintln!("{}", help::INSTALL_HELP);
|
eprintln!(
|
||||||
|
"\
|
||||||
|
cargo xtask install
|
||||||
|
Install rust-analyzer server or editor plugin.
|
||||||
|
|
||||||
|
USAGE:
|
||||||
|
cargo xtask install [FLAGS]
|
||||||
|
|
||||||
|
FLAGS:
|
||||||
|
--client-code Install only VS Code plugin
|
||||||
|
--server Install only the language server
|
||||||
|
--jemalloc Use jemalloc for server
|
||||||
|
-h, --help Prints help information
|
||||||
|
"
|
||||||
|
);
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
let server = matches.contains("--server");
|
let server = args.contains("--server");
|
||||||
let client_code = matches.contains("--client-code");
|
let client_code = args.contains("--client-code");
|
||||||
if server && client_code {
|
if server && client_code {
|
||||||
eprintln!("{}", help::INSTALL_RA_CONFLICT);
|
eprintln!(
|
||||||
|
"error: The argument `--server` cannot be used with `--client-code`\n\n\
|
||||||
|
For more information try --help"
|
||||||
|
);
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
let jemalloc = matches.contains("--jemalloc");
|
|
||||||
matches.finish().or_else(handle_extra_flags)?;
|
let jemalloc = args.contains("--jemalloc");
|
||||||
let opts = InstallOpt {
|
|
||||||
|
args.finish()?;
|
||||||
|
|
||||||
|
InstallCmd {
|
||||||
client: if server { None } else { Some(ClientOpt::VsCode) },
|
client: if server { None } else { Some(ClientOpt::VsCode) },
|
||||||
server: if client_code { None } else { Some(ServerOpt { jemalloc }) },
|
server: if client_code { None } else { Some(ServerOpt { jemalloc }) },
|
||||||
};
|
}
|
||||||
install(opts)?
|
.run()
|
||||||
}
|
}
|
||||||
"codegen" => {
|
"codegen" => {
|
||||||
|
args.finish()?;
|
||||||
codegen::generate_syntax(Mode::Overwrite)?;
|
codegen::generate_syntax(Mode::Overwrite)?;
|
||||||
codegen::generate_parser_tests(Mode::Overwrite)?;
|
codegen::generate_parser_tests(Mode::Overwrite)?;
|
||||||
codegen::generate_assists_docs(Mode::Overwrite)?;
|
codegen::generate_assists_docs(Mode::Overwrite)?;
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
"format" => run_rustfmt(Mode::Overwrite)?,
|
"format" => {
|
||||||
"install-pre-commit-hook" => install_pre_commit_hook()?,
|
args.finish()?;
|
||||||
"lint" => run_clippy()?,
|
run_rustfmt(Mode::Overwrite)
|
||||||
"fuzz-tests" => run_fuzzer()?,
|
|
||||||
_ => eprintln!("{}", help::GLOBAL_HELP),
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn handle_extra_flags(e: pico_args::Error) -> Result<()> {
|
|
||||||
if let pico_args::Error::UnusedArgsLeft(flags) = e {
|
|
||||||
let mut invalid_flags = String::new();
|
|
||||||
for flag in flags {
|
|
||||||
write!(&mut invalid_flags, "{}, ", flag)?;
|
|
||||||
}
|
}
|
||||||
let (invalid_flags, _) = invalid_flags.split_at(invalid_flags.len() - 2);
|
"install-pre-commit-hook" => {
|
||||||
anyhow::bail!("Invalid flags: {}", invalid_flags)
|
args.finish()?;
|
||||||
} else {
|
pre_commit::install_hook()
|
||||||
anyhow::bail!(e.to_string())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn install(opts: InstallOpt) -> Result<()> {
|
|
||||||
if cfg!(target_os = "macos") {
|
|
||||||
fix_path_for_mac().context("Fix path for mac")?
|
|
||||||
}
|
|
||||||
if let Some(server) = opts.server {
|
|
||||||
install_server(server).context("install server")?;
|
|
||||||
}
|
|
||||||
if let Some(client) = opts.client {
|
|
||||||
install_client(client).context("install client")?;
|
|
||||||
}
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn fix_path_for_mac() -> Result<()> {
|
|
||||||
let mut vscode_path: Vec<PathBuf> = {
|
|
||||||
const COMMON_APP_PATH: &str =
|
|
||||||
r"/Applications/Visual Studio Code.app/Contents/Resources/app/bin";
|
|
||||||
const ROOT_DIR: &str = "";
|
|
||||||
let home_dir = match env::var("HOME") {
|
|
||||||
Ok(home) => home,
|
|
||||||
Err(e) => anyhow::bail!("Failed getting HOME from environment with error: {}.", e),
|
|
||||||
};
|
|
||||||
|
|
||||||
[ROOT_DIR, &home_dir]
|
|
||||||
.iter()
|
|
||||||
.map(|dir| String::from(*dir) + COMMON_APP_PATH)
|
|
||||||
.map(PathBuf::from)
|
|
||||||
.filter(|path| path.exists())
|
|
||||||
.collect()
|
|
||||||
};
|
|
||||||
|
|
||||||
if !vscode_path.is_empty() {
|
|
||||||
let vars = match env::var_os("PATH") {
|
|
||||||
Some(path) => path,
|
|
||||||
None => anyhow::bail!("Could not get PATH variable from env."),
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut paths = env::split_paths(&vars).collect::<Vec<_>>();
|
|
||||||
paths.append(&mut vscode_path);
|
|
||||||
let new_paths = env::join_paths(paths).context("build env PATH")?;
|
|
||||||
env::set_var("PATH", &new_paths);
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn install_client(ClientOpt::VsCode: ClientOpt) -> Result<()> {
|
|
||||||
let npm_version = Cmd {
|
|
||||||
unix: r"npm --version",
|
|
||||||
windows: r"cmd.exe /c npm --version",
|
|
||||||
work_dir: "./editors/code",
|
|
||||||
}
|
|
||||||
.run();
|
|
||||||
|
|
||||||
if npm_version.is_err() {
|
|
||||||
eprintln!("\nERROR: `npm --version` failed, `npm` is required to build the VS Code plugin")
|
|
||||||
}
|
|
||||||
|
|
||||||
Cmd { unix: r"npm install", windows: r"cmd.exe /c npm install", work_dir: "./editors/code" }
|
|
||||||
.run()?;
|
|
||||||
Cmd {
|
|
||||||
unix: r"npm run package --scripts-prepend-node-path",
|
|
||||||
windows: r"cmd.exe /c npm run package",
|
|
||||||
work_dir: "./editors/code",
|
|
||||||
}
|
|
||||||
.run()?;
|
|
||||||
|
|
||||||
let code_binary = ["code", "code-insiders", "codium"].iter().find(|bin| {
|
|
||||||
Cmd {
|
|
||||||
unix: &format!("{} --version", bin),
|
|
||||||
windows: &format!("cmd.exe /c {}.cmd --version", bin),
|
|
||||||
work_dir: "./editors/code",
|
|
||||||
}
|
}
|
||||||
.run()
|
"lint" => {
|
||||||
.is_ok()
|
args.finish()?;
|
||||||
});
|
run_clippy()
|
||||||
|
}
|
||||||
|
"fuzz-tests" => {
|
||||||
|
args.finish()?;
|
||||||
|
run_fuzzer()
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
eprintln!(
|
||||||
|
"\
|
||||||
|
cargo xtask
|
||||||
|
Run custom build command.
|
||||||
|
|
||||||
let code_binary = match code_binary {
|
USAGE:
|
||||||
Some(it) => it,
|
cargo xtask <SUBCOMMAND>
|
||||||
None => anyhow::bail!("Can't execute `code --version`. Perhaps it is not in $PATH?"),
|
|
||||||
};
|
|
||||||
|
|
||||||
Cmd {
|
SUBCOMMANDS:
|
||||||
unix: &format!(r"{} --install-extension ./ra-lsp-0.0.1.vsix --force", code_binary),
|
format
|
||||||
windows: &format!(
|
install-pre-commit-hook
|
||||||
r"cmd.exe /c {}.cmd --install-extension ./ra-lsp-0.0.1.vsix --force",
|
fuzz-tests
|
||||||
code_binary
|
codegen
|
||||||
),
|
install
|
||||||
work_dir: "./editors/code",
|
lint"
|
||||||
}
|
);
|
||||||
.run()?;
|
Ok(())
|
||||||
|
|
||||||
let output = Cmd {
|
|
||||||
unix: &format!(r"{} --list-extensions", code_binary),
|
|
||||||
windows: &format!(r"cmd.exe /c {}.cmd --list-extensions", code_binary),
|
|
||||||
work_dir: ".",
|
|
||||||
}
|
|
||||||
.run_with_output()?;
|
|
||||||
|
|
||||||
if !str::from_utf8(&output.stdout)?.contains("ra-lsp") {
|
|
||||||
anyhow::bail!(
|
|
||||||
"Could not install the Visual Studio Code extension. \
|
|
||||||
Please make sure you have at least NodeJS 10.x together with the latest version of VS Code installed and try again."
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn install_server(opts: ServerOpt) -> Result<()> {
|
|
||||||
let mut old_rust = false;
|
|
||||||
if let Ok(output) = run_with_output("cargo --version", ".") {
|
|
||||||
if let Ok(stdout) = String::from_utf8(output.stdout) {
|
|
||||||
println!("{}", stdout);
|
|
||||||
if !check_version(&stdout, REQUIRED_RUST_VERSION) {
|
|
||||||
old_rust = true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if old_rust {
|
|
||||||
eprintln!(
|
|
||||||
"\nWARNING: at least rust 1.{}.0 is required to compile rust-analyzer\n",
|
|
||||||
REQUIRED_RUST_VERSION,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
let res = if opts.jemalloc {
|
|
||||||
run("cargo install --path crates/ra_lsp_server --locked --force --features jemalloc", ".")
|
|
||||||
} else {
|
|
||||||
run("cargo install --path crates/ra_lsp_server --locked --force", ".")
|
|
||||||
};
|
|
||||||
|
|
||||||
if res.is_err() && old_rust {
|
|
||||||
eprintln!(
|
|
||||||
"\nWARNING: at least rust 1.{}.0 is required to compile rust-analyzer\n",
|
|
||||||
REQUIRED_RUST_VERSION,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
res
|
|
||||||
}
|
|
||||||
|
|
||||||
fn check_version(version_output: &str, min_minor_version: u32) -> bool {
|
|
||||||
// Parse second the number out of
|
|
||||||
// cargo 1.39.0-beta (1c6ec66d5 2019-09-30)
|
|
||||||
let minor: Option<u32> = version_output.split('.').nth(1).and_then(|it| it.parse().ok());
|
|
||||||
match minor {
|
|
||||||
None => true,
|
|
||||||
Some(minor) => minor >= min_minor_version,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
36
xtask/src/pre_commit.rs
Normal file
36
xtask/src/pre_commit.rs
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
//! pre-commit hook for code formatting.
|
||||||
|
|
||||||
|
use std::{fs, path::PathBuf};
|
||||||
|
|
||||||
|
use anyhow::{bail, Result};
|
||||||
|
|
||||||
|
use crate::{project_root, run, run_rustfmt, run_with_output, Mode};
|
||||||
|
|
||||||
|
// FIXME: if there are changed `.ts` files, also reformat TypeScript (by
|
||||||
|
// shelling out to `npm fmt`).
|
||||||
|
pub fn run_hook() -> Result<()> {
|
||||||
|
run_rustfmt(Mode::Overwrite)?;
|
||||||
|
|
||||||
|
let diff = run_with_output("git diff --diff-filter=MAR --name-only --cached", ".")?;
|
||||||
|
|
||||||
|
let root = project_root();
|
||||||
|
for line in String::from_utf8(diff.stdout)?.lines() {
|
||||||
|
run(&format!("git update-index --add {}", root.join(line).to_string_lossy()), ".")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn install_hook() -> Result<()> {
|
||||||
|
let hook_path: PathBuf =
|
||||||
|
format!("./.git/hooks/pre-commit{}", std::env::consts::EXE_SUFFIX).into();
|
||||||
|
|
||||||
|
if hook_path.exists() {
|
||||||
|
bail!("Git hook already created");
|
||||||
|
}
|
||||||
|
|
||||||
|
let me = std::env::current_exe()?;
|
||||||
|
fs::copy(me, hook_path)?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
Loading…
Reference in a new issue