271: Implement format hook r=matklad a=DJMcNab

Tentatively: fixes #155.

However, this does add all changes in staged files, which might not be desirable. However, I think we can't solve that without explicit support in rustfmt for it, so it should be fine.

Co-authored-by: DJMcNab <36049421+djmcnab@users.noreply.github.com>
This commit is contained in:
bors[bot] 2018-12-09 12:44:58 +00:00
commit 3725276554
5 changed files with 75 additions and 15 deletions

View File

@ -1,9 +1,10 @@
[alias] [alias]
# Automatically generates the ast and syntax kinds files # Automatically generates the ast and syntax kinds files
gen-syntax = "run --package tools -- gen-syntax" gen-syntax = "run --package tools --bin tools -- gen-syntax"
gen-tests = "run --package tools -- gen-tests" gen-tests = "run --package tools --bin tools -- gen-tests"
install-code = "run --package tools -- install-code" install-code = "run --package tools --bin tools -- install-code"
format = "run --package tools -- format" format = "run --package tools --bin tools -- format"
format-hook = "run --package tools --bin tools -- format-hook"
render-test = "run --package ra_cli -- render-test" render-test = "run --package ra_cli -- render-test"
parse = "run --package ra_cli -- parse" parse = "run --package ra_cli -- parse"

View File

@ -20,7 +20,7 @@ pub fn join_lines(file: &SourceFileNode, range: TextRange) -> LocalEdit {
return LocalEdit { return LocalEdit {
edit: EditBuilder::new().finish(), edit: EditBuilder::new().finish(),
cursor_position: None, cursor_position: None,
} };
} }
Some(pos) => pos, Some(pos) => pos,
}; };

View File

@ -0,0 +1,37 @@
use std::{
process::{Command},
};
use tools::{Result, run_rustfmt, run, project_root};
use failure::bail;
fn main() -> tools::Result<()> {
run_rustfmt(tools::Overwrite)?;
update_staged()
}
fn update_staged() -> Result<()> {
let root = project_root();
let output = Command::new("git")
.arg("diff")
.arg("--name-only")
.arg("--cached")
.current_dir(&root)
.output()?;
if !output.status.success() {
bail!(
"`git diff --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(())
}

View File

@ -1,6 +1,8 @@
use std::{ use std::{
path::{Path, PathBuf}, path::{Path, PathBuf},
process::{Command, Stdio}, process::{Command, Stdio},
fs::copy,
io::{Error, ErrorKind}
}; };
use failure::bail; use failure::bail;
@ -39,7 +41,7 @@ pub fn collect_tests(s: &str) -> Vec<(usize, Test)> {
let (start_line, name) = loop { let (start_line, name) = loop {
match block.next() { match block.next() {
Some((idx, line)) if line.starts_with("test ") => { Some((idx, line)) if line.starts_with("test ") => {
break (idx, line["test ".len()..].to_string()) break (idx, line["test ".len()..].to_string());
} }
Some(_) => (), Some(_) => (),
None => continue 'outer, None => continue 'outer,
@ -65,7 +67,7 @@ pub fn generate(mode: Mode) -> Result<()> {
} }
pub fn project_root() -> PathBuf { pub fn project_root() -> PathBuf {
Path::new(&std::env::var("CARGO_MANIFEST_DIR").unwrap()) Path::new(&env!("CARGO_MANIFEST_DIR"))
.ancestors() .ancestors()
.nth(2) .nth(2)
.unwrap() .unwrap()
@ -116,3 +118,18 @@ fn install_rustfmt() -> Result<()> {
".", ".",
) )
} }
pub fn install_format_hook() -> Result<()> {
let result_path = Path::new("./.git/hooks/pre-commit");
if !result_path.exists() {
run("cargo build --package tools --bin pre-commit", ".")?;
if cfg!(windows) {
copy("./target/debug/pre-commit.exe", result_path)?;
} else {
copy("./target/debug/pre-commit", result_path)?;
}
} else {
return Err(Error::new(ErrorKind::AlreadyExists, "Git hook already created").into());
}
Ok(())
}

View File

@ -7,7 +7,7 @@ use std::{
use clap::{App, Arg, SubCommand}; use clap::{App, Arg, SubCommand};
use failure::bail; use failure::bail;
use tools::{collect_tests, generate, run, run_rustfmt, Mode, Overwrite, Result, Test, Verify}; use tools::{collect_tests, generate, install_format_hook, run, run_rustfmt, Mode, Overwrite, Result, Test, Verify};
const GRAMMAR_DIR: &str = "./crates/ra_syntax/src/grammar"; const GRAMMAR_DIR: &str = "./crates/ra_syntax/src/grammar";
const INLINE_TESTS_DIR: &str = "./crates/ra_syntax/tests/data/parser/inline"; const INLINE_TESTS_DIR: &str = "./crates/ra_syntax/tests/data/parser/inline";
@ -25,17 +25,22 @@ fn main() -> Result<()> {
.subcommand(SubCommand::with_name("gen-tests")) .subcommand(SubCommand::with_name("gen-tests"))
.subcommand(SubCommand::with_name("install-code")) .subcommand(SubCommand::with_name("install-code"))
.subcommand(SubCommand::with_name("format")) .subcommand(SubCommand::with_name("format"))
.subcommand(SubCommand::with_name("format-hook"))
.get_matches(); .get_matches();
let mode = if matches.is_present("verify") { let mode = if matches.is_present("verify") {
Verify Verify
} else { } else {
Overwrite Overwrite
}; };
match matches.subcommand() { match matches
("install-code", _) => install_code_extension()?, .subcommand_name()
("gen-tests", _) => gen_tests(mode)?, .expect("Subcommand must be specified")
("gen-syntax", _) => generate(Overwrite)?, {
("format", _) => run_rustfmt(Overwrite)?, "install-code" => install_code_extension()?,
"gen-tests" => gen_tests(mode)?,
"gen-syntax" => generate(Overwrite)?,
"format" => run_rustfmt(mode)?,
"format-hook" => install_format_hook()?,
_ => unreachable!(), _ => unreachable!(),
} }
Ok(()) Ok(())