2019-11-16 00:56:51 +03:00
|
|
|
mod never_type;
|
|
|
|
mod coercion;
|
2019-12-03 13:38:54 +01:00
|
|
|
mod regression;
|
|
|
|
mod simple;
|
|
|
|
mod patterns;
|
|
|
|
mod traits;
|
|
|
|
mod method_resolution;
|
|
|
|
mod macros;
|
2019-11-16 00:56:51 +03:00
|
|
|
|
2019-07-04 23:05:17 +03:00
|
|
|
use std::sync::Arc;
|
2018-12-20 21:56:28 +01:00
|
|
|
|
2019-11-27 17:46:02 +03:00
|
|
|
use hir_def::{
|
2020-03-06 14:44:44 +01:00
|
|
|
body::{BodySourceMap, SyntheticSyntax},
|
|
|
|
child_by_source::ChildBySource,
|
|
|
|
db::DefDatabase,
|
|
|
|
item_scope::ItemScope,
|
|
|
|
keys,
|
|
|
|
nameres::CrateDefMap,
|
|
|
|
AssocItemId, DefWithBodyId, LocalModuleId, Lookup, ModuleDefId,
|
2019-11-27 17:46:02 +03:00
|
|
|
};
|
2019-11-28 12:50:26 +03:00
|
|
|
use hir_expand::InFile;
|
2019-08-29 16:49:10 +03:00
|
|
|
use insta::assert_snapshot;
|
2019-11-04 13:25:42 +03:00
|
|
|
use ra_db::{fixture::WithFixture, salsa::Database, FilePosition, SourceDatabase};
|
2019-07-04 23:05:17 +03:00
|
|
|
use ra_syntax::{
|
|
|
|
algo,
|
2020-04-09 23:35:05 +02:00
|
|
|
ast::{self, AstNode},
|
2019-07-04 23:05:17 +03:00
|
|
|
};
|
2020-03-28 11:20:34 +01:00
|
|
|
use stdx::format_to;
|
2018-12-20 21:56:28 +01:00
|
|
|
|
2019-11-27 17:46:02 +03:00
|
|
|
use crate::{db::HirDatabase, display::HirDisplay, test_db::TestDB, InferenceResult};
|
2018-12-20 21:56:28 +01:00
|
|
|
|
2018-12-24 15:36:54 +01:00
|
|
|
// These tests compare the inference results for all expressions in a file
|
2019-03-03 12:40:36 +01:00
|
|
|
// against snapshots of the expected results using insta. Use cargo-insta to
|
|
|
|
// update the snapshots.
|
2018-12-24 15:36:54 +01:00
|
|
|
|
2019-11-04 22:21:15 +03:00
|
|
|
fn type_at_pos(db: &TestDB, pos: FilePosition) -> String {
|
2019-05-28 18:07:39 +03:00
|
|
|
let file = db.parse(pos.file_id).ok().unwrap();
|
2019-04-13 01:32:43 +03:00
|
|
|
let expr = algo::find_node_at_offset::<ast::Expr>(file.syntax(), pos.offset).unwrap();
|
2019-12-05 21:17:17 +01:00
|
|
|
let fn_def = expr.syntax().ancestors().find_map(ast::FnDef::cast).unwrap();
|
2019-11-27 17:46:02 +03:00
|
|
|
let module = db.module_for_file(pos.file_id);
|
2019-12-05 23:34:12 +01:00
|
|
|
let func = *module.child_by_source(db)[keys::FUNCTION]
|
|
|
|
.get(&InFile::new(pos.file_id.into(), fn_def))
|
|
|
|
.unwrap();
|
2019-12-05 21:17:17 +01:00
|
|
|
|
|
|
|
let (_body, source_map) = db.body_with_source_map(func.into());
|
|
|
|
if let Some(expr_id) = source_map.node_expr(InFile::new(pos.file_id.into(), &expr)) {
|
|
|
|
let infer = db.infer(func.into());
|
|
|
|
let ty = &infer[expr_id];
|
|
|
|
return ty.display(db).to_string();
|
2019-11-27 17:46:02 +03:00
|
|
|
}
|
|
|
|
panic!("Can't find expression")
|
2019-02-23 12:32:25 +01:00
|
|
|
}
|
|
|
|
|
2019-04-14 16:08:10 +02:00
|
|
|
fn type_at(content: &str) -> String {
|
2019-11-04 22:21:15 +03:00
|
|
|
let (db, file_pos) = TestDB::with_position(content);
|
2019-04-14 16:08:10 +02:00
|
|
|
type_at_pos(&db, file_pos)
|
|
|
|
}
|
|
|
|
|
2020-02-27 16:05:35 +01:00
|
|
|
fn infer(ra_fixture: &str) -> String {
|
|
|
|
infer_with_mismatches(ra_fixture, false)
|
2019-12-05 23:02:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
fn infer_with_mismatches(content: &str, include_mismatches: bool) -> String {
|
2019-11-04 22:21:15 +03:00
|
|
|
let (db, file_id) = TestDB::with_single_file(content);
|
2019-04-18 19:35:47 +01:00
|
|
|
|
2020-03-28 11:20:34 +01:00
|
|
|
let mut buf = String::new();
|
2019-03-30 11:17:31 +00:00
|
|
|
|
|
|
|
let mut infer_def = |inference_result: Arc<InferenceResult>,
|
|
|
|
body_source_map: Arc<BodySourceMap>| {
|
2019-01-06 23:01:33 +01:00
|
|
|
let mut types = Vec::new();
|
2019-12-05 23:02:31 +01:00
|
|
|
let mut mismatches = Vec::new();
|
2019-03-30 11:17:31 +00:00
|
|
|
|
2019-01-06 23:57:39 +01:00
|
|
|
for (pat, ty) in inference_result.type_of_pat.iter() {
|
2019-03-02 15:14:37 +03:00
|
|
|
let syntax_ptr = match body_source_map.pat_syntax(pat) {
|
2020-03-06 14:44:44 +01:00
|
|
|
Ok(sp) => {
|
2019-09-03 11:04:38 +03:00
|
|
|
sp.map(|ast| ast.either(|it| it.syntax_node_ptr(), |it| it.syntax_node_ptr()))
|
|
|
|
}
|
2020-03-06 14:44:44 +01:00
|
|
|
Err(SyntheticSyntax) => continue,
|
2019-01-06 16:47:59 +01:00
|
|
|
};
|
2019-01-06 23:01:33 +01:00
|
|
|
types.push((syntax_ptr, ty));
|
2019-01-06 16:47:59 +01:00
|
|
|
}
|
2019-03-30 11:17:31 +00:00
|
|
|
|
2019-01-06 23:57:39 +01:00
|
|
|
for (expr, ty) in inference_result.type_of_expr.iter() {
|
2019-03-02 15:14:37 +03:00
|
|
|
let syntax_ptr = match body_source_map.expr_syntax(expr) {
|
2020-03-06 14:44:44 +01:00
|
|
|
Ok(sp) => {
|
2019-09-03 11:04:38 +03:00
|
|
|
sp.map(|ast| ast.either(|it| it.syntax_node_ptr(), |it| it.syntax_node_ptr()))
|
|
|
|
}
|
2020-03-06 14:44:44 +01:00
|
|
|
Err(SyntheticSyntax) => continue,
|
2019-01-06 16:47:59 +01:00
|
|
|
};
|
2019-01-06 23:01:33 +01:00
|
|
|
types.push((syntax_ptr, ty));
|
2019-12-05 23:02:31 +01:00
|
|
|
if let Some(mismatch) = inference_result.type_mismatch_for_expr(expr) {
|
|
|
|
mismatches.push((syntax_ptr, mismatch));
|
|
|
|
}
|
2019-01-06 16:47:59 +01:00
|
|
|
}
|
2019-04-18 19:35:47 +01:00
|
|
|
|
2019-01-06 23:01:33 +01:00
|
|
|
// sort ranges for consistency
|
2019-11-20 09:40:36 +03:00
|
|
|
types.sort_by_key(|(src_ptr, _)| {
|
|
|
|
(src_ptr.value.range().start(), src_ptr.value.range().end())
|
|
|
|
});
|
2019-09-03 11:04:38 +03:00
|
|
|
for (src_ptr, ty) in &types {
|
2019-11-20 09:40:36 +03:00
|
|
|
let node = src_ptr.value.to_node(&src_ptr.file_syntax(&db));
|
2019-09-03 11:04:38 +03:00
|
|
|
|
2019-07-19 10:43:01 +03:00
|
|
|
let (range, text) = if let Some(self_param) = ast::SelfParam::cast(node.clone()) {
|
2020-04-09 23:35:05 +02:00
|
|
|
(self_param.self_token().unwrap().text_range(), "self".to_string())
|
2019-03-30 13:25:53 +03:00
|
|
|
} else {
|
2019-11-20 09:40:36 +03:00
|
|
|
(src_ptr.value.range(), node.text().to_string().replace("\n", " "))
|
2019-03-30 13:25:53 +03:00
|
|
|
};
|
2019-09-03 11:04:38 +03:00
|
|
|
let macro_prefix = if src_ptr.file_id != file_id.into() { "!" } else { "" };
|
2020-03-28 11:20:34 +01:00
|
|
|
format_to!(
|
|
|
|
buf,
|
|
|
|
"{}{} '{}': {}\n",
|
2019-09-03 11:04:38 +03:00
|
|
|
macro_prefix,
|
|
|
|
range,
|
|
|
|
ellipsize(text, 15),
|
|
|
|
ty.display(&db)
|
2020-03-28 11:20:34 +01:00
|
|
|
);
|
2018-12-20 21:56:28 +01:00
|
|
|
}
|
2019-12-05 23:02:31 +01:00
|
|
|
if include_mismatches {
|
|
|
|
mismatches.sort_by_key(|(src_ptr, _)| {
|
|
|
|
(src_ptr.value.range().start(), src_ptr.value.range().end())
|
|
|
|
});
|
|
|
|
for (src_ptr, mismatch) in &mismatches {
|
|
|
|
let range = src_ptr.value.range();
|
|
|
|
let macro_prefix = if src_ptr.file_id != file_id.into() { "!" } else { "" };
|
2020-03-28 11:20:34 +01:00
|
|
|
format_to!(
|
|
|
|
buf,
|
|
|
|
"{}{}: expected {}, got {}\n",
|
2019-12-05 23:02:31 +01:00
|
|
|
macro_prefix,
|
|
|
|
range,
|
|
|
|
mismatch.expected.display(&db),
|
|
|
|
mismatch.actual.display(&db),
|
2020-03-28 11:20:34 +01:00
|
|
|
);
|
2019-12-05 23:02:31 +01:00
|
|
|
}
|
|
|
|
}
|
2019-03-30 11:17:31 +00:00
|
|
|
};
|
|
|
|
|
2019-11-27 17:46:02 +03:00
|
|
|
let module = db.module_for_file(file_id);
|
|
|
|
let crate_def_map = db.crate_def_map(module.krate);
|
|
|
|
|
|
|
|
let mut defs: Vec<DefWithBodyId> = Vec::new();
|
2019-11-27 21:31:51 +03:00
|
|
|
visit_module(&db, &crate_def_map, module.local_id, &mut |it| defs.push(it));
|
2019-11-27 17:46:02 +03:00
|
|
|
defs.sort_by_key(|def| match def {
|
|
|
|
DefWithBodyId::FunctionId(it) => {
|
|
|
|
it.lookup(&db).ast_id.to_node(&db).syntax().text_range().start()
|
|
|
|
}
|
|
|
|
DefWithBodyId::ConstId(it) => {
|
|
|
|
it.lookup(&db).ast_id.to_node(&db).syntax().text_range().start()
|
|
|
|
}
|
|
|
|
DefWithBodyId::StaticId(it) => {
|
|
|
|
it.lookup(&db).ast_id.to_node(&db).syntax().text_range().start()
|
2019-04-13 00:56:57 +03:00
|
|
|
}
|
2019-11-27 17:46:02 +03:00
|
|
|
});
|
|
|
|
for def in defs {
|
|
|
|
let (_body, source_map) = db.body_with_source_map(def);
|
|
|
|
let infer = db.infer(def);
|
|
|
|
infer_def(infer, source_map);
|
2019-03-30 11:17:31 +00:00
|
|
|
}
|
|
|
|
|
2020-03-28 11:20:34 +01:00
|
|
|
buf.truncate(buf.trim_end().len());
|
|
|
|
buf
|
2018-12-23 12:05:54 +01:00
|
|
|
}
|
|
|
|
|
2019-11-27 17:46:02 +03:00
|
|
|
fn visit_module(
|
|
|
|
db: &TestDB,
|
|
|
|
crate_def_map: &CrateDefMap,
|
|
|
|
module_id: LocalModuleId,
|
|
|
|
cb: &mut dyn FnMut(DefWithBodyId),
|
|
|
|
) {
|
2019-12-29 14:46:24 +01:00
|
|
|
visit_scope(db, crate_def_map, &crate_def_map[module_id].scope, cb);
|
2019-12-20 15:58:20 +01:00
|
|
|
for impl_id in crate_def_map[module_id].scope.impls() {
|
2019-11-27 17:46:02 +03:00
|
|
|
let impl_data = db.impl_data(impl_id);
|
|
|
|
for &item in impl_data.items.iter() {
|
|
|
|
match item {
|
2019-12-29 14:46:24 +01:00
|
|
|
AssocItemId::FunctionId(it) => {
|
|
|
|
let def = it.into();
|
|
|
|
cb(def);
|
|
|
|
let body = db.body(def);
|
|
|
|
visit_scope(db, crate_def_map, &body.item_scope, cb);
|
|
|
|
}
|
|
|
|
AssocItemId::ConstId(it) => {
|
|
|
|
let def = it.into();
|
|
|
|
cb(def);
|
|
|
|
let body = db.body(def);
|
|
|
|
visit_scope(db, crate_def_map, &body.item_scope, cb);
|
|
|
|
}
|
2019-11-27 17:46:02 +03:00
|
|
|
AssocItemId::TypeAliasId(_) => (),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-12-29 14:46:24 +01:00
|
|
|
|
|
|
|
fn visit_scope(
|
|
|
|
db: &TestDB,
|
|
|
|
crate_def_map: &CrateDefMap,
|
|
|
|
scope: &ItemScope,
|
|
|
|
cb: &mut dyn FnMut(DefWithBodyId),
|
|
|
|
) {
|
|
|
|
for decl in scope.declarations() {
|
|
|
|
match decl {
|
|
|
|
ModuleDefId::FunctionId(it) => {
|
|
|
|
let def = it.into();
|
|
|
|
cb(def);
|
|
|
|
let body = db.body(def);
|
|
|
|
visit_scope(db, crate_def_map, &body.item_scope, cb);
|
|
|
|
}
|
|
|
|
ModuleDefId::ConstId(it) => {
|
|
|
|
let def = it.into();
|
|
|
|
cb(def);
|
|
|
|
let body = db.body(def);
|
|
|
|
visit_scope(db, crate_def_map, &body.item_scope, cb);
|
|
|
|
}
|
|
|
|
ModuleDefId::StaticId(it) => {
|
|
|
|
let def = it.into();
|
|
|
|
cb(def);
|
|
|
|
let body = db.body(def);
|
|
|
|
visit_scope(db, crate_def_map, &body.item_scope, cb);
|
|
|
|
}
|
|
|
|
ModuleDefId::TraitId(it) => {
|
|
|
|
let trait_data = db.trait_data(it);
|
|
|
|
for &(_, item) in trait_data.items.iter() {
|
|
|
|
match item {
|
|
|
|
AssocItemId::FunctionId(it) => cb(it.into()),
|
|
|
|
AssocItemId::ConstId(it) => cb(it.into()),
|
|
|
|
AssocItemId::TypeAliasId(_) => (),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ModuleDefId::ModuleId(it) => visit_module(db, crate_def_map, it.local_id, cb),
|
|
|
|
_ => (),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-11-27 17:46:02 +03:00
|
|
|
}
|
|
|
|
|
2018-12-23 12:05:54 +01:00
|
|
|
fn ellipsize(mut text: String, max_len: usize) -> String {
|
|
|
|
if text.len() <= max_len {
|
|
|
|
return text;
|
|
|
|
}
|
|
|
|
let ellipsis = "...";
|
|
|
|
let e_len = ellipsis.len();
|
|
|
|
let mut prefix_len = (max_len - e_len) / 2;
|
|
|
|
while !text.is_char_boundary(prefix_len) {
|
|
|
|
prefix_len += 1;
|
|
|
|
}
|
|
|
|
let mut suffix_len = max_len - e_len - prefix_len;
|
|
|
|
while !text.is_char_boundary(text.len() - suffix_len) {
|
|
|
|
suffix_len += 1;
|
|
|
|
}
|
|
|
|
text.replace_range(prefix_len..text.len() - suffix_len, ellipsis);
|
|
|
|
text
|
2018-12-20 21:56:28 +01:00
|
|
|
}
|
|
|
|
|
2019-01-05 13:42:47 +01:00
|
|
|
#[test]
|
|
|
|
fn typing_whitespace_inside_a_function_should_not_invalidate_types() {
|
2019-11-04 22:21:15 +03:00
|
|
|
let (mut db, pos) = TestDB::with_position(
|
2019-01-05 13:42:47 +01:00
|
|
|
"
|
|
|
|
//- /lib.rs
|
|
|
|
fn foo() -> i32 {
|
|
|
|
<|>1 + 1
|
|
|
|
}
|
|
|
|
",
|
|
|
|
);
|
|
|
|
{
|
|
|
|
let events = db.log_executed(|| {
|
2019-11-27 17:46:02 +03:00
|
|
|
let module = db.module_for_file(pos.file_id);
|
|
|
|
let crate_def_map = db.crate_def_map(module.krate);
|
2019-11-27 21:31:51 +03:00
|
|
|
visit_module(&db, &crate_def_map, module.local_id, &mut |def| {
|
2019-11-27 17:46:02 +03:00
|
|
|
db.infer(def);
|
|
|
|
});
|
2019-01-05 13:42:47 +01:00
|
|
|
});
|
|
|
|
assert!(format!("{:?}", events).contains("infer"))
|
|
|
|
}
|
|
|
|
|
|
|
|
let new_text = "
|
|
|
|
fn foo() -> i32 {
|
|
|
|
1
|
|
|
|
+
|
|
|
|
1
|
|
|
|
}
|
|
|
|
"
|
|
|
|
.to_string();
|
|
|
|
|
2019-02-08 14:49:43 +03:00
|
|
|
db.query_mut(ra_db::FileTextQuery).set(pos.file_id, Arc::new(new_text));
|
2019-01-05 13:42:47 +01:00
|
|
|
|
|
|
|
{
|
|
|
|
let events = db.log_executed(|| {
|
2019-11-27 17:46:02 +03:00
|
|
|
let module = db.module_for_file(pos.file_id);
|
|
|
|
let crate_def_map = db.crate_def_map(module.krate);
|
2019-11-27 21:31:51 +03:00
|
|
|
visit_module(&db, &crate_def_map, module.local_id, &mut |def| {
|
2019-11-27 17:46:02 +03:00
|
|
|
db.infer(def);
|
|
|
|
});
|
2019-01-05 13:42:47 +01:00
|
|
|
});
|
|
|
|
assert!(!format!("{:?}", events).contains("infer"), "{:#?}", events)
|
|
|
|
}
|
|
|
|
}
|
2019-03-24 11:39:47 +03:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn no_such_field_diagnostics() {
|
2019-11-04 22:21:15 +03:00
|
|
|
let diagnostics = TestDB::with_files(
|
2019-03-24 11:39:47 +03:00
|
|
|
r"
|
|
|
|
//- /lib.rs
|
|
|
|
struct S { foo: i32, bar: () }
|
|
|
|
impl S {
|
|
|
|
fn new() -> S {
|
|
|
|
S {
|
|
|
|
foo: 92,
|
|
|
|
baz: 62,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
",
|
|
|
|
)
|
2020-03-24 04:40:58 -07:00
|
|
|
.diagnostics()
|
|
|
|
.0;
|
2019-03-24 11:39:47 +03:00
|
|
|
|
2019-08-29 16:49:10 +03:00
|
|
|
assert_snapshot!(diagnostics, @r###"
|
|
|
|
"baz: 62": no such field
|
2019-11-24 17:36:30 +01:00
|
|
|
"{\n foo: 92,\n baz: 62,\n }": Missing structure fields:
|
|
|
|
- bar
|
2019-08-29 16:49:10 +03:00
|
|
|
"###
|
2019-03-24 11:39:47 +03:00
|
|
|
);
|
|
|
|
}
|
2020-04-07 17:58:05 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn no_such_field_with_feature_flag_diagnostics() {
|
|
|
|
let diagnostics = TestDB::with_files(
|
|
|
|
r#"
|
2020-04-08 18:12:15 +02:00
|
|
|
//- /lib.rs crate:foo cfg:feature=foo
|
2020-04-07 17:58:05 +02:00
|
|
|
struct MyStruct {
|
|
|
|
my_val: usize,
|
|
|
|
#[cfg(feature = "foo")]
|
|
|
|
bar: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl MyStruct {
|
|
|
|
#[cfg(feature = "foo")]
|
|
|
|
pub(crate) fn new(my_val: usize, bar: bool) -> Self {
|
|
|
|
Self { my_val, bar }
|
|
|
|
}
|
2020-04-09 09:39:17 +02:00
|
|
|
|
2020-04-07 17:58:05 +02:00
|
|
|
#[cfg(not(feature = "foo"))]
|
|
|
|
pub(crate) fn new(my_val: usize, _bar: bool) -> Self {
|
|
|
|
Self { my_val }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
)
|
2020-04-08 18:12:15 +02:00
|
|
|
.diagnostics()
|
|
|
|
.0;
|
2020-04-07 17:58:05 +02:00
|
|
|
|
2020-04-08 18:12:15 +02:00
|
|
|
assert_snapshot!(diagnostics, @r###""###);
|
2020-04-07 17:58:05 +02:00
|
|
|
}
|
2020-04-09 18:32:02 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn no_such_field_with_feature_flag_diagnostics_on_struct_lit() {
|
|
|
|
let diagnostics = TestDB::with_files(
|
|
|
|
r#"
|
|
|
|
//- /lib.rs crate:foo cfg:feature=foo
|
|
|
|
struct S {
|
|
|
|
#[cfg(feature = "foo")]
|
|
|
|
foo: u32,
|
|
|
|
#[cfg(not(feature = "foo"))]
|
|
|
|
bar: u32,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl S {
|
|
|
|
#[cfg(feature = "foo")]
|
|
|
|
fn new(foo: u32) -> Self {
|
|
|
|
Self { foo }
|
|
|
|
}
|
2020-04-09 18:37:34 +02:00
|
|
|
#[cfg(not(feature = "foo"))]
|
2020-04-09 18:32:02 +02:00
|
|
|
fn new(bar: u32) -> Self {
|
|
|
|
Self { bar }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
.diagnostics()
|
|
|
|
.0;
|
|
|
|
|
|
|
|
assert_snapshot!(diagnostics, @r###""###);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn no_such_field_with_feature_flag_diagnostics_on_struct_fields() {
|
|
|
|
let diagnostics = TestDB::with_files(
|
|
|
|
r#"
|
|
|
|
//- /lib.rs crate:foo cfg:feature=foo
|
|
|
|
struct S {
|
|
|
|
#[cfg(feature = "foo")]
|
|
|
|
foo: u32,
|
|
|
|
#[cfg(not(feature = "foo"))]
|
|
|
|
bar: u32,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl S {
|
|
|
|
fn new(val: u32) -> Self {
|
|
|
|
Self {
|
|
|
|
#[cfg(feature = "foo")]
|
|
|
|
foo: val,
|
|
|
|
#[cfg(not(feature = "foo"))]
|
|
|
|
bar: val,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"#,
|
|
|
|
)
|
|
|
|
.diagnostics()
|
|
|
|
.0;
|
|
|
|
|
|
|
|
assert_snapshot!(diagnostics, @r###""###);
|
|
|
|
}
|
2020-04-08 20:23:51 -07:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn missing_record_pat_field_diagnostic() {
|
|
|
|
let diagnostics = TestDB::with_files(
|
|
|
|
r"
|
|
|
|
//- /lib.rs
|
|
|
|
struct S { foo: i32, bar: () }
|
|
|
|
fn baz(s: S) {
|
|
|
|
let S { foo: _ } = s;
|
|
|
|
}
|
|
|
|
",
|
|
|
|
)
|
|
|
|
.diagnostics()
|
|
|
|
.0;
|
|
|
|
|
|
|
|
assert_snapshot!(diagnostics, @r###"
|
|
|
|
"{ foo: _ }": Missing structure fields:
|
|
|
|
- bar
|
|
|
|
"###
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn missing_record_pat_field_no_diagnostic_if_not_exhaustive() {
|
|
|
|
let diagnostics = TestDB::with_files(
|
|
|
|
r"
|
|
|
|
//- /lib.rs
|
|
|
|
struct S { foo: i32, bar: () }
|
|
|
|
fn baz(s: S) -> i32 {
|
|
|
|
match s {
|
|
|
|
S { foo, .. } => foo,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
",
|
|
|
|
)
|
|
|
|
.diagnostics()
|
|
|
|
.0;
|
|
|
|
|
|
|
|
assert_snapshot!(diagnostics, @"");
|
|
|
|
}
|