2020-03-17 10:26:29 +01:00
|
|
|
use std::{
|
|
|
|
collections::HashMap,
|
|
|
|
path::{Path, PathBuf},
|
|
|
|
};
|
|
|
|
|
2020-10-16 19:46:03 +02:00
|
|
|
use xshell::{cmd, read_file};
|
2020-05-06 10:25:25 +02:00
|
|
|
use xtask::{
|
2021-02-03 22:01:09 +08:00
|
|
|
cargo_files,
|
2020-05-06 10:25:25 +02:00
|
|
|
codegen::{self, Mode},
|
|
|
|
project_root, run_rustfmt, rust_files,
|
|
|
|
};
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn generated_grammar_is_fresh() {
|
|
|
|
if let Err(error) = codegen::generate_syntax(Mode::Verify) {
|
|
|
|
panic!("{}. Please update it by running `cargo xtask codegen`", error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn generated_tests_are_fresh() {
|
|
|
|
if let Err(error) = codegen::generate_parser_tests(Mode::Verify) {
|
|
|
|
panic!("{}. Please update tests by running `cargo xtask codegen`", error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn generated_assists_are_fresh() {
|
2020-06-03 18:22:05 +02:00
|
|
|
if let Err(error) = codegen::generate_assists_tests(Mode::Verify) {
|
2020-05-06 10:25:25 +02:00
|
|
|
panic!("{}. Please update assists by running `cargo xtask codegen`", error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn check_code_formatting() {
|
|
|
|
if let Err(error) = run_rustfmt(Mode::Verify) {
|
|
|
|
panic!("{}. Please format the code by running `cargo format`", error);
|
|
|
|
}
|
|
|
|
}
|
2020-03-17 10:26:29 +01:00
|
|
|
|
2020-10-06 15:17:16 +02:00
|
|
|
#[test]
|
|
|
|
fn smoke_test_docs_generation() {
|
|
|
|
// We don't commit docs to the repo, so we can just overwrite in tests.
|
|
|
|
codegen::generate_assists_docs(Mode::Overwrite).unwrap();
|
|
|
|
codegen::generate_feature_docs(Mode::Overwrite).unwrap();
|
2020-10-19 21:07:40 +03:00
|
|
|
codegen::generate_diagnostic_docs(Mode::Overwrite).unwrap();
|
2020-10-06 15:17:16 +02:00
|
|
|
}
|
|
|
|
|
2020-10-14 13:30:06 +02:00
|
|
|
#[test]
|
|
|
|
fn check_lsp_extensions_docs() {
|
|
|
|
let expected_hash = {
|
|
|
|
let lsp_ext_rs =
|
2020-10-16 19:46:03 +02:00
|
|
|
read_file(project_root().join("crates/rust-analyzer/src/lsp_ext.rs")).unwrap();
|
2020-10-14 13:30:06 +02:00
|
|
|
stable_hash(lsp_ext_rs.as_str())
|
|
|
|
};
|
|
|
|
|
|
|
|
let actual_hash = {
|
|
|
|
let lsp_extensions_md =
|
2020-10-16 19:46:03 +02:00
|
|
|
read_file(project_root().join("docs/dev/lsp-extensions.md")).unwrap();
|
2020-10-14 13:30:06 +02:00
|
|
|
let text = lsp_extensions_md
|
|
|
|
.lines()
|
|
|
|
.find_map(|line| line.strip_prefix("lsp_ext.rs hash:"))
|
|
|
|
.unwrap()
|
|
|
|
.trim();
|
|
|
|
u64::from_str_radix(text, 16).unwrap()
|
|
|
|
};
|
|
|
|
|
|
|
|
if actual_hash != expected_hash {
|
|
|
|
panic!(
|
|
|
|
"
|
|
|
|
lsp_ext.rs was changed without touching lsp-extensions.md.
|
|
|
|
|
|
|
|
Expected hash: {:x}
|
|
|
|
Actual hash: {:x}
|
|
|
|
|
|
|
|
Please adjust docs/dev/lsp-extensions.md.
|
|
|
|
",
|
|
|
|
expected_hash, actual_hash
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-17 10:26:29 +01:00
|
|
|
#[test]
|
|
|
|
fn rust_files_are_tidy() {
|
|
|
|
let mut tidy_docs = TidyDocs::default();
|
2021-01-21 16:37:08 +03:00
|
|
|
for path in rust_files() {
|
2020-10-16 19:46:03 +02:00
|
|
|
let text = read_file(&path).unwrap();
|
2020-03-17 10:26:29 +01:00
|
|
|
check_todo(&path, &text);
|
2020-12-28 11:27:54 +08:00
|
|
|
check_dbg(&path, &text);
|
2020-03-17 10:46:46 +01:00
|
|
|
check_trailing_ws(&path, &text);
|
2020-08-12 12:45:38 +02:00
|
|
|
deny_clippy(&path, &text);
|
2020-03-17 10:26:29 +01:00
|
|
|
tidy_docs.visit(&path, &text);
|
|
|
|
}
|
|
|
|
tidy_docs.finish();
|
|
|
|
}
|
|
|
|
|
2021-02-03 22:01:09 +08:00
|
|
|
#[test]
|
|
|
|
fn cargo_files_are_tidy() {
|
|
|
|
for cargo in cargo_files() {
|
|
|
|
let mut section = None;
|
|
|
|
for (line_no, text) in read_file(&cargo).unwrap().lines().enumerate() {
|
|
|
|
let text = text.trim();
|
|
|
|
if text.starts_with("[") {
|
|
|
|
section = Some(text);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if !section.map(|it| it.starts_with("[dependencies")).unwrap_or(false) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
let text: String = text.split_whitespace().collect();
|
|
|
|
if text.contains("path=") && !text.contains("version") {
|
|
|
|
panic!(
|
|
|
|
"\ncargo internal dependencies should have version.\n\
|
|
|
|
{}:{}\n",
|
|
|
|
cargo.display(),
|
|
|
|
line_no + 1
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-24 15:17:00 -06:00
|
|
|
#[test]
|
|
|
|
fn check_merge_commits() {
|
2020-12-28 01:21:41 +08:00
|
|
|
let stdout = cmd!("git rev-list --merges --invert-grep --author 'bors\\[bot\\]' HEAD~19..")
|
|
|
|
.read()
|
|
|
|
.unwrap();
|
2020-09-01 09:38:17 +02:00
|
|
|
if !stdout.is_empty() {
|
|
|
|
panic!(
|
|
|
|
"
|
|
|
|
Merge commits are not allowed in the history.
|
|
|
|
|
|
|
|
When updating a pull-request, please rebase your feature branch
|
|
|
|
on top of master by running `git rebase master`. If rebase fails,
|
|
|
|
you can re-apply your changes like this:
|
|
|
|
|
2021-01-07 12:00:07 +03:00
|
|
|
# Just look around to see the current state.
|
|
|
|
$ git status
|
|
|
|
$ git log
|
|
|
|
|
|
|
|
# Abort in-progress rebase and merges, if any.
|
2020-09-01 09:38:17 +02:00
|
|
|
$ git rebase --abort
|
2021-01-07 12:00:07 +03:00
|
|
|
$ git merge --abort
|
2020-09-01 09:38:17 +02:00
|
|
|
|
|
|
|
# Make the branch point to the latest commit from master,
|
|
|
|
# while maintaining your local changes uncommited.
|
|
|
|
$ git reset --soft origin/master
|
|
|
|
|
|
|
|
# Commit all changes in a single batch.
|
|
|
|
$ git commit -am'My changes'
|
|
|
|
|
2021-01-07 12:00:07 +03:00
|
|
|
# Verify that everything looks alright.
|
|
|
|
$ git status
|
|
|
|
$ git log
|
|
|
|
|
2020-09-01 09:38:17 +02:00
|
|
|
# Push the changes. We did a rebase, so we need `--force` option.
|
|
|
|
# `--force-with-lease` is a more safe (Rusty) version of `--force`.
|
|
|
|
$ git push --force-with-lease
|
|
|
|
|
2021-01-07 12:00:07 +03:00
|
|
|
# Verify that both local and remote branch point to the same commit.
|
|
|
|
$ git log
|
|
|
|
|
2020-09-01 09:38:17 +02:00
|
|
|
And don't fear to mess something up during a rebase -- you can
|
|
|
|
always restore the previous state using `git ref-log`:
|
|
|
|
|
|
|
|
https://github.blog/2015-06-08-how-to-undo-almost-anything-with-git/#redo-after-undo-local
|
|
|
|
"
|
|
|
|
);
|
2020-08-24 15:17:00 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-12 12:45:38 +02:00
|
|
|
fn deny_clippy(path: &PathBuf, text: &String) {
|
2020-10-20 21:29:31 +02:00
|
|
|
let ignore = &[
|
|
|
|
// The documentation in string literals may contain anything for its own purposes
|
2021-02-17 17:53:31 +03:00
|
|
|
"ide_completion/src/generated_lint_completions.rs",
|
2020-10-20 21:29:31 +02:00
|
|
|
];
|
|
|
|
if ignore.iter().any(|p| path.ends_with(p)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-01-01 01:58:04 +03:00
|
|
|
if text.contains("\u{61}llow(clippy") {
|
2020-08-12 12:45:38 +02:00
|
|
|
panic!(
|
|
|
|
"\n\nallowing lints is forbidden: {}.
|
|
|
|
rust-analyzer intentionally doesn't check clippy on CI.
|
|
|
|
You can allow lint globally via `xtask clippy`.
|
2020-08-12 13:03:43 +02:00
|
|
|
See https://github.com/rust-lang/rust-clippy/issues/5537 for discussion.
|
2020-08-12 12:45:38 +02:00
|
|
|
|
|
|
|
",
|
|
|
|
path.display()
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-14 17:14:00 +02:00
|
|
|
#[test]
|
|
|
|
fn check_licenses() {
|
|
|
|
let expected = "
|
|
|
|
0BSD OR MIT OR Apache-2.0
|
2020-08-01 12:55:04 +12:00
|
|
|
Apache-2.0
|
2020-07-14 17:14:00 +02:00
|
|
|
Apache-2.0 OR BSL-1.0
|
|
|
|
Apache-2.0 OR MIT
|
|
|
|
Apache-2.0/MIT
|
|
|
|
BSD-3-Clause
|
|
|
|
CC0-1.0
|
|
|
|
ISC
|
|
|
|
MIT
|
|
|
|
MIT / Apache-2.0
|
|
|
|
MIT OR Apache-2.0
|
2020-11-17 09:39:25 -05:00
|
|
|
MIT OR Apache-2.0 OR Zlib
|
2020-10-08 09:43:00 -04:00
|
|
|
MIT OR Zlib OR Apache-2.0
|
2020-07-14 17:14:00 +02:00
|
|
|
MIT/Apache-2.0
|
|
|
|
Unlicense OR MIT
|
|
|
|
Unlicense/MIT
|
2020-08-18 10:49:18 +02:00
|
|
|
Zlib OR Apache-2.0 OR MIT
|
2020-07-14 17:14:00 +02:00
|
|
|
"
|
|
|
|
.lines()
|
|
|
|
.filter(|it| !it.is_empty())
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
2020-10-16 19:46:03 +02:00
|
|
|
let meta = cmd!("cargo metadata --format-version 1").read().unwrap();
|
2020-07-14 17:14:00 +02:00
|
|
|
let mut licenses = meta
|
|
|
|
.split(|c| c == ',' || c == '{' || c == '}')
|
|
|
|
.filter(|it| it.contains(r#""license""#))
|
|
|
|
.map(|it| it.trim())
|
|
|
|
.map(|it| it[r#""license":"#.len()..].trim_matches('"'))
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
licenses.sort();
|
|
|
|
licenses.dedup();
|
2020-10-16 19:46:03 +02:00
|
|
|
if licenses != expected {
|
|
|
|
let mut diff = String::new();
|
|
|
|
|
|
|
|
diff += &format!("New Licenses:\n");
|
|
|
|
for &l in licenses.iter() {
|
|
|
|
if !expected.contains(&l) {
|
|
|
|
diff += &format!(" {}\n", l)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
diff += &format!("\nMissing Licenses:\n");
|
|
|
|
for &l in expected.iter() {
|
|
|
|
if !licenses.contains(&l) {
|
|
|
|
diff += &format!(" {}\n", l)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
panic!("different set of licenses!\n{}", diff);
|
|
|
|
}
|
2020-08-01 11:52:24 +12:00
|
|
|
assert_eq!(licenses, expected);
|
2020-07-14 17:14:00 +02:00
|
|
|
}
|
|
|
|
|
2020-03-17 10:26:29 +01:00
|
|
|
fn check_todo(path: &Path, text: &str) {
|
2020-07-08 22:47:50 +02:00
|
|
|
let need_todo = &[
|
|
|
|
// This file itself obviously needs to use todo (<- like this!).
|
2020-12-28 11:27:54 +08:00
|
|
|
"tests/tidy.rs",
|
2020-07-08 22:47:50 +02:00
|
|
|
// Some of our assists generate `todo!()`.
|
2020-05-20 00:07:00 +02:00
|
|
|
"handlers/add_turbo_fish.rs",
|
2020-07-03 18:15:03 +02:00
|
|
|
"handlers/generate_function.rs",
|
2020-07-08 22:47:50 +02:00
|
|
|
// To support generating `todo!()` in assists, we have `expr_todo()` in
|
|
|
|
// `ast::make`.
|
2020-04-10 13:41:11 -07:00
|
|
|
"ast/make.rs",
|
2020-08-10 00:35:42 +07:00
|
|
|
// The documentation in string literals may contain anything for its own purposes
|
2021-02-17 17:53:31 +03:00
|
|
|
"ide_completion/src/generated_lint_completions.rs",
|
2020-04-10 13:41:11 -07:00
|
|
|
];
|
2020-07-08 22:47:50 +02:00
|
|
|
if need_todo.iter().any(|p| path.ends_with(p)) {
|
2020-03-17 10:26:29 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if text.contains("TODO") || text.contains("TOOD") || text.contains("todo!") {
|
2020-11-09 13:07:18 +01:00
|
|
|
// Generated by an assist
|
|
|
|
if text.contains("${0:todo!()}") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-03-17 10:26:29 +01:00
|
|
|
panic!(
|
2020-04-17 10:32:12 +02:00
|
|
|
"\nTODO markers or todo! macros should not be committed to the master branch,\n\
|
2020-03-17 10:26:29 +01:00
|
|
|
use FIXME instead\n\
|
|
|
|
{}\n",
|
|
|
|
path.display(),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-28 11:27:54 +08:00
|
|
|
fn check_dbg(path: &Path, text: &str) {
|
|
|
|
let need_dbg = &[
|
|
|
|
// This file itself obviously needs to use dbg.
|
|
|
|
"tests/tidy.rs",
|
|
|
|
// Assists to remove `dbg!()`
|
|
|
|
"handlers/remove_dbg.rs",
|
|
|
|
// We have .dbg postfix
|
2021-02-17 17:53:31 +03:00
|
|
|
"ide_completion/src/completions/postfix.rs",
|
2020-12-28 11:27:54 +08:00
|
|
|
// The documentation in string literals may contain anything for its own purposes
|
2021-02-17 17:53:31 +03:00
|
|
|
"ide_completion/src/lib.rs",
|
|
|
|
"ide_completion/src/generated_lint_completions.rs",
|
2020-12-28 11:27:54 +08:00
|
|
|
// test for doc test for remove_dbg
|
|
|
|
"src/tests/generated.rs",
|
|
|
|
];
|
|
|
|
if need_dbg.iter().any(|p| path.ends_with(p)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if text.contains("dbg!") {
|
|
|
|
panic!(
|
|
|
|
"\ndbg! macros should not be committed to the master branch,\n\
|
|
|
|
{}\n",
|
|
|
|
path.display(),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-17 10:46:46 +01:00
|
|
|
fn check_trailing_ws(path: &Path, text: &str) {
|
|
|
|
if is_exclude_dir(path, &["test_data"]) {
|
|
|
|
return;
|
|
|
|
}
|
2020-04-17 10:32:12 +02:00
|
|
|
for (line_number, line) in text.lines().enumerate() {
|
2020-03-17 10:46:46 +01:00
|
|
|
if line.chars().last().map(char::is_whitespace) == Some(true) {
|
2020-04-17 10:32:12 +02:00
|
|
|
panic!("Trailing whitespace in {} at line {}", path.display(), line_number)
|
2020-03-17 10:46:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-17 10:26:29 +01:00
|
|
|
#[derive(Default)]
|
|
|
|
struct TidyDocs {
|
|
|
|
missing_docs: Vec<String>,
|
|
|
|
contains_fixme: Vec<PathBuf>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TidyDocs {
|
|
|
|
fn visit(&mut self, path: &Path, text: &str) {
|
2020-03-17 10:46:46 +01:00
|
|
|
// Test hopefully don't really need comments, and for assists we already
|
|
|
|
// have special comments which are source of doc tests and user docs.
|
2020-05-31 09:59:38 +02:00
|
|
|
if is_exclude_dir(path, &["tests", "test_data"]) {
|
2020-03-17 10:46:46 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if is_exclude_file(path) {
|
2020-03-17 10:26:29 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let first_line = match text.lines().next() {
|
|
|
|
Some(it) => it,
|
|
|
|
None => return,
|
|
|
|
};
|
|
|
|
|
|
|
|
if first_line.starts_with("//!") {
|
|
|
|
if first_line.contains("FIXME") {
|
2020-05-31 09:59:38 +02:00
|
|
|
self.contains_fixme.push(path.to_path_buf());
|
2020-03-17 10:26:29 +01:00
|
|
|
}
|
|
|
|
} else {
|
2020-05-31 09:59:38 +02:00
|
|
|
if text.contains("// Feature:") || text.contains("// Assist:") {
|
|
|
|
return;
|
|
|
|
}
|
2020-03-17 10:26:29 +01:00
|
|
|
self.missing_docs.push(path.display().to_string());
|
|
|
|
}
|
|
|
|
|
|
|
|
fn is_exclude_file(d: &Path) -> bool {
|
2021-01-12 20:19:13 +01:00
|
|
|
let file_names = ["tests.rs", "famous_defs_fixture.rs"];
|
2020-03-17 10:26:29 +01:00
|
|
|
|
|
|
|
d.file_name()
|
|
|
|
.unwrap_or_default()
|
|
|
|
.to_str()
|
|
|
|
.map(|f_n| file_names.iter().any(|name| *name == f_n))
|
|
|
|
.unwrap_or(false)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn finish(self) {
|
|
|
|
if !self.missing_docs.is_empty() {
|
|
|
|
panic!(
|
|
|
|
"\nMissing docs strings\n\n\
|
|
|
|
modules:\n{}\n\n",
|
|
|
|
self.missing_docs.join("\n")
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2020-07-08 22:47:50 +02:00
|
|
|
let poorly_documented = [
|
2020-08-13 16:36:55 +02:00
|
|
|
"hir",
|
2020-08-13 16:26:29 +02:00
|
|
|
"hir_expand",
|
2020-08-13 17:42:52 +02:00
|
|
|
"ide",
|
2020-08-13 10:08:11 +02:00
|
|
|
"mbe",
|
2020-08-12 17:06:49 +02:00
|
|
|
"parser",
|
2020-08-12 16:32:36 +02:00
|
|
|
"profile",
|
2020-08-13 12:05:30 +02:00
|
|
|
"project_model",
|
2020-08-12 18:26:51 +02:00
|
|
|
"syntax",
|
2020-08-12 16:46:20 +02:00
|
|
|
"tt",
|
2020-08-13 16:35:29 +02:00
|
|
|
"hir_ty",
|
2020-03-17 10:26:29 +01:00
|
|
|
];
|
|
|
|
|
|
|
|
let mut has_fixmes =
|
2020-07-08 22:47:50 +02:00
|
|
|
poorly_documented.iter().map(|it| (*it, false)).collect::<HashMap<&str, bool>>();
|
2020-03-17 10:26:29 +01:00
|
|
|
'outer: for path in self.contains_fixme {
|
2020-07-08 22:47:50 +02:00
|
|
|
for krate in poorly_documented.iter() {
|
2020-03-17 10:26:29 +01:00
|
|
|
if path.components().any(|it| it.as_os_str() == *krate) {
|
|
|
|
has_fixmes.insert(krate, true);
|
|
|
|
continue 'outer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
panic!("FIXME doc in a fully-documented crate: {}", path.display())
|
|
|
|
}
|
|
|
|
|
|
|
|
for (krate, has_fixme) in has_fixmes.iter() {
|
|
|
|
if !has_fixme {
|
2020-07-08 22:47:50 +02:00
|
|
|
panic!("crate {} is fully documented :tada:, remove it from the list of poorly documented crates", krate)
|
2020-03-17 10:26:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-17 10:46:46 +01:00
|
|
|
fn is_exclude_dir(p: &Path, dirs_to_exclude: &[&str]) -> bool {
|
2020-06-02 23:15:23 +03:00
|
|
|
p.strip_prefix(project_root())
|
|
|
|
.unwrap()
|
|
|
|
.components()
|
|
|
|
.rev()
|
|
|
|
.skip(1)
|
|
|
|
.filter_map(|it| it.as_os_str().to_str())
|
|
|
|
.any(|it| dirs_to_exclude.contains(&it))
|
2020-03-17 10:46:46 +01:00
|
|
|
}
|
2020-10-14 13:30:06 +02:00
|
|
|
|
|
|
|
#[allow(deprecated)]
|
|
|
|
fn stable_hash(text: &str) -> u64 {
|
|
|
|
use std::hash::{Hash, Hasher, SipHasher};
|
|
|
|
|
|
|
|
let text = text.replace('\r', "");
|
|
|
|
let mut hasher = SipHasher::default();
|
|
|
|
text.hash(&mut hasher);
|
|
|
|
hasher.finish()
|
|
|
|
}
|