Skip to content

Commit ddcae6e

Browse files
authored
Merge pull request #252 from xuhuanzy/diagnostic
diagnostic
2 parents fefd3e5 + 251019b commit ddcae6e

File tree

13 files changed

+185
-78
lines changed

13 files changed

+185
-78
lines changed

crates/emmylua_code_analysis/locales/lint.yml

+9-4
Original file line numberDiff line numberDiff line change
@@ -101,10 +101,10 @@ tuple member %{idx} not match, expect %{typ}, but got %{got}:
101101
zh_CN: '元组成员 %{idx} 不匹配,期望 %{typ},但得到 %{got}'
102102
zh_HK: '元組成員 %{idx} 不匹配,期望 %{typ},但得到 %{got}'
103103

104-
Annotations specify that return value %{index} has a type of `%{source}`, returning value of type `%{found}` here instead.:
105-
en: Annotations specify that return value %{index} has a type of `%{source}`, returning value of type `%{found}` here instead.
106-
zh_CN: '第 %{index} 个返回值的类型为 `%{source}`,但实际返回类型为 `%{found}`。'
107-
zh_HK: '第 %{index} 個回傳值的類型為 `%{source}` ,但實際回傳的是 `%{found}`。'
104+
Annotations specify that return value %{index} has a type of `%{source}`, returning value of type `%{found}` here instead. %{reason}.:
105+
en: Annotations specify that return value %{index} has a type of `%{source}`, returning value of type `%{found}` here instead. %{reason}.
106+
zh_CN: '第 %{index} 个返回值的类型为 `%{source}`,但实际返回类型为 `%{found}`。 %{reason}。'
107+
zh_HK: '第 %{index} 個回傳值的類型為 `%{source}` ,但實際回傳的是 `%{found}`。 %{reason}。'
108108
Annotations specify that at most %{max} return value(s) are required, found %{rmax} returned here instead.:
109109
en: 'Annotations specify that at most %{max} return value(s) are required, found %{rmax} returned here instead.'
110110
zh_CN: '最多只有 %{max} 个返回值,但此处返回了 %{rmax} 个'
@@ -217,3 +217,8 @@ Cannot use `...` outside a vararg function.:
217217
en: "type `%{name}` not found."
218218
zh_CN: "类型 `%{name}` 未找到。"
219219
zh_HK: "類型 `%{name}` 未找到。"
220+
"Duplicate class constructor '%{name}'. constructor must have only one.":
221+
en: "Duplicate class constructor '%{name}'. constructor must have only one."
222+
zh_CN: "类有重复的 (constructor) 定义 '%{name}'。(constructor) 必须只有一个。"
223+
zh_HK: "類有重複的 (constructor) 定義 '%{name}'。(constructor) 必須只有一個。"
224+

crates/emmylua_code_analysis/resources/std/coroutine.lua

+3-3
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ coroutine = {}
1919
---
2020
--- Creates a new coroutine, with body `f`. `f` must be a Lua function. Returns
2121
--- this new coroutine, an object with type `"thread"`.
22-
---@param f async fun(...):...
22+
---@param f async fun(...):any...
2323
---@return thread
2424
---@nodiscard
2525
function coroutine.create(f) end
@@ -93,8 +93,8 @@ function coroutine.status(co) end
9393
--- passed to the function behave as the extra arguments to `resume`. Returns
9494
--- the same values returned by `resume`, except the first
9595
--- boolean. In case of error, propagates the error.
96-
---@param f async fun(...):...
97-
---@return fun(...):...
96+
---@param f async fun(...):any...
97+
---@return fun(...):any...
9898
---@nodiscard
9999
function coroutine.wrap(f) end
100100

crates/emmylua_code_analysis/resources/std/debug.lua

+1-1
Original file line numberDiff line numberDiff line change
@@ -93,7 +93,7 @@ function debug.gethook(thread) end
9393
--- with a name for the current function, if a reasonable name can be found,
9494
--- and the expression `debug.getinfo(print)` returns a table with all available
9595
--- information about the `print` function.
96-
---@overload fun(f: int|function, what?: InfoWhat):DebugInfo
96+
---@overload fun(f: int|function, what?: Infowhat):DebugInfo
9797
---@param thread thread
9898
---@param f function
9999
---@param what? Infowhat

crates/emmylua_code_analysis/resources/std/os.lua

+1-1
Original file line numberDiff line numberDiff line change
@@ -193,7 +193,7 @@ function os.setlocale(locale, category) end
193193
--- documented in the `os.date` function, so that they represent the same time
194194
--- as before the call but with values inside their valid ranges.
195195
---@param date? std.osdateparam
196-
---@return number
196+
---@return integer
197197
function os.time(date) end
198198

199199
---

crates/emmylua_code_analysis/resources/std/table.lua

+1-2
Original file line numberDiff line numberDiff line change
@@ -98,10 +98,9 @@ function table.remove(list, pos) end
9898
---
9999
--- The sort algorithm is not stable: elements considered equal by the given
100100
--- order may have their relative positions changed by the sort.
101-
---@overload fun(list:table):integer
102101
---@generic V
103102
---@param list V[]
104-
---@param comp fun(a:V, b:V):boolean
103+
---@param comp? fun(a:V, b:V):boolean
105104
---@return integer
106105
function table.sort(list, comp) end
107106

crates/emmylua_code_analysis/src/compilation/analyzer/decl/docs.rs

+3
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,9 @@ fn get_attrib_value(
4747
"exact" => {
4848
attr |= LuaTypeAttribute::Exact;
4949
}
50+
"constructor" => {
51+
attr |= LuaTypeAttribute::Constructor;
52+
}
5053
_ => {}
5154
}
5255
}

crates/emmylua_code_analysis/src/db_index/module/mod.rs

+4
Original file line numberDiff line numberDiff line change
@@ -390,6 +390,10 @@ impl LuaModuleIndex {
390390
file_ids
391391
}
392392

393+
pub fn is_std(&self, file_id: &FileId) -> bool {
394+
self.get_std_file_ids().contains(file_id)
395+
}
396+
393397
pub fn get_main_workspace_file_ids(&self) -> Vec<FileId> {
394398
let mut file_ids = Vec::new();
395399
for module_info in self.file_module_map.values() {

crates/emmylua_code_analysis/src/db_index/type/type_decl.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,8 @@ flags! {
2121
Key,
2222
Partial,
2323
Exact,
24-
Meta
24+
Meta,
25+
Constructor,
2526
}
2627
}
2728

crates/emmylua_code_analysis/src/diagnostic/checker/duplicate_type.rs

+15
Original file line numberDiff line numberDiff line change
@@ -43,13 +43,16 @@ fn check_duplicate_class(context: &mut DiagnosticContext, class_tag: LuaDocTagCl
4343
if locations.len() > 1 {
4444
let mut type_times = 0;
4545
let mut partial_times = 0;
46+
let mut constructor_times = 0;
4647
for location in locations {
4748
let attrib = location.attrib;
4849
if attrib.contains(LuaTypeAttribute::Meta) {
4950
continue;
5051
}
5152
if attrib.contains(LuaTypeAttribute::Partial) {
5253
partial_times += 1;
54+
} else if attrib.contains(LuaTypeAttribute::Constructor) {
55+
constructor_times += 1;
5356
} else {
5457
type_times += 1;
5558
}
@@ -70,6 +73,18 @@ fn check_duplicate_class(context: &mut DiagnosticContext, class_tag: LuaDocTagCl
7073
None,
7174
);
7275
}
76+
if constructor_times > 1 {
77+
context.add_diagnostic(
78+
DiagnosticCode::DuplicateType,
79+
range,
80+
t!(
81+
"Duplicate class constructor '%{name}'. constructor must have only one.",
82+
name = name
83+
)
84+
.to_string(),
85+
None,
86+
);
87+
}
7388
}
7489

7590
Some(())

crates/emmylua_code_analysis/src/diagnostic/checker/return_type_mismatch.rs

+55-10
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,9 @@
1-
use emmylua_parser::{LuaAstNode, LuaClosureExpr, LuaReturnStat};
2-
use rowan::TextRange;
1+
use emmylua_parser::{LuaAstNode, LuaClosureExpr, LuaExpr, LuaReturnStat};
2+
use rowan::{NodeOrToken, TextRange};
33

44
use crate::{
5-
humanize_type, DiagnosticCode, LuaSignatureId, LuaType, RenderLevel, SemanticModel,
6-
SignatureReturnStatus, TypeCheckFailReason, TypeCheckResult,
5+
humanize_type, DiagnosticCode, LuaSemanticDeclId, LuaSignatureId, LuaType, RenderLevel,
6+
SemanticModel, SignatureReturnStatus, TypeCheckFailReason, TypeCheckResult,
77
};
88

99
use super::{get_own_return_stats, Checker, DiagnosticContext};
@@ -49,7 +49,12 @@ fn check_return_stat(
4949
&return_stat.get_expr_list().collect::<Vec<_>>(),
5050
None,
5151
)?;
52-
let return_expr_types = infos.iter().map(|(typ, _)| typ.clone()).collect::<Vec<_>>();
52+
let mut return_expr_types = infos.iter().map(|(typ, _)| typ.clone()).collect::<Vec<_>>();
53+
// 解决 setmetatable 的返回值类型问题
54+
let setmetatable_index = has_setmetatable(semantic_model, return_stat);
55+
if let Some(setmetatable_index) = setmetatable_index {
56+
return_expr_types[setmetatable_index] = LuaType::Any;
57+
}
5358
let return_expr_ranges = infos
5459
.iter()
5560
.map(|(_, range)| range.clone())
@@ -62,7 +67,6 @@ fn check_return_stat(
6267
if return_expr_types.len() < index {
6368
break;
6469
}
65-
6670
check_variadic_return_type_match(
6771
context,
6872
semantic_model,
@@ -136,25 +140,38 @@ fn add_type_check_diagnostic(
136140
Ok(_) => return,
137141
Err(reason) => match reason {
138142
TypeCheckFailReason::TypeNotMatchWithReason(reason) => {
139-
context.add_diagnostic(DiagnosticCode::ParamTypeNotMatch, range, reason, None);
143+
context.add_diagnostic(
144+
DiagnosticCode::ReturnTypeMismatch,
145+
range,
146+
t!(
147+
"Annotations specify that return value %{index} has a type of `%{source}`, returning value of type `%{found}` here instead. %{reason}.",
148+
index = index + 1,
149+
source = humanize_type(db, &param_type, RenderLevel::Simple),
150+
found = humanize_type(db, &expr_type, RenderLevel::Simple),
151+
reason = reason
152+
)
153+
.to_string(),
154+
None,
155+
);
140156
}
141157
TypeCheckFailReason::TypeNotMatch => {
142158
context.add_diagnostic(
143159
DiagnosticCode::ReturnTypeMismatch,
144160
range,
145161
t!(
146-
"Annotations specify that return value %{index} has a type of `%{source}`, returning value of type `%{found}` here instead.",
162+
"Annotations specify that return value %{index} has a type of `%{source}`, returning value of type `%{found}` here instead. %{reason}.",
147163
index = index + 1,
148164
source = humanize_type(db, &param_type, RenderLevel::Simple),
149-
found = humanize_type(db, &expr_type, RenderLevel::Simple)
165+
found = humanize_type(db, &expr_type, RenderLevel::Simple),
166+
reason = ""
150167
)
151168
.to_string(),
152169
None,
153170
);
154171
}
155172
TypeCheckFailReason::TypeRecursion => {
156173
context.add_diagnostic(
157-
DiagnosticCode::ParamTypeNotMatch,
174+
DiagnosticCode::ReturnTypeMismatch,
158175
range,
159176
"type recursion".into(),
160177
None,
@@ -163,3 +180,31 @@ fn add_type_check_diagnostic(
163180
},
164181
}
165182
}
183+
184+
fn has_setmetatable(semantic_model: &SemanticModel, return_stat: &LuaReturnStat) -> Option<usize> {
185+
for (index, expr) in return_stat.get_expr_list().enumerate() {
186+
if let LuaExpr::CallExpr(call_expr) = expr {
187+
if let Some(prefix_expr) = call_expr.get_prefix_expr() {
188+
let semantic_info = semantic_model
189+
.get_semantic_info(NodeOrToken::Node(prefix_expr.syntax().clone().into()))?;
190+
191+
if let Some(LuaSemanticDeclId::LuaDecl(decl_id)) = semantic_info.semantic_decl {
192+
let decl = semantic_model.get_db().get_decl_index().get_decl(&decl_id);
193+
194+
if let Some(decl) = decl {
195+
if decl.is_global()
196+
&& semantic_model
197+
.get_db()
198+
.get_module_index()
199+
.is_std(&decl.get_file_id())
200+
&& decl.get_name() == "setmetatable"
201+
{
202+
return Some(index);
203+
}
204+
}
205+
}
206+
}
207+
}
208+
}
209+
None
210+
}

crates/emmylua_code_analysis/src/diagnostic/test/missing_parameter_test.rs

+20
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,26 @@
22
mod test {
33
use crate::{DiagnosticCode, VirtualWorkspace};
44

5+
#[test]
6+
fn test_issue_249() {
7+
let mut ws = VirtualWorkspace::new();
8+
9+
assert!(ws.check_code_for(
10+
DiagnosticCode::MissingParameter,
11+
r#"
12+
---@param path string
13+
---@return string? realpath
14+
---@overload fun(path:string, callback:function):userdata
15+
function realpath(path)
16+
end
17+
18+
local path = realpath('/', function(err, path)
19+
end)
20+
21+
"#
22+
));
23+
}
24+
525
#[test]
626
fn test_1() {
727
let mut ws = VirtualWorkspace::new();

crates/emmylua_code_analysis/src/diagnostic/test/return_type_mismatch_test.rs

+37
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,43 @@
22
mod tests {
33
use crate::{DiagnosticCode, VirtualWorkspace};
44

5+
#[test]
6+
fn test_issue_242() {
7+
let mut ws = VirtualWorkspace::new_with_init_std_lib();
8+
9+
assert!(ws.check_code_for_namespace(
10+
DiagnosticCode::ReturnTypeMismatch,
11+
r#"
12+
local setmetatable = setmetatable
13+
--- @class A
14+
local A = {}
15+
16+
function A:method() end
17+
18+
--- @return A
19+
function new()
20+
return setmetatable({}, { __index = A })
21+
end
22+
"#
23+
));
24+
25+
assert!(ws.check_code_for_namespace(
26+
DiagnosticCode::ReturnTypeMismatch,
27+
r#"
28+
--- @class A
29+
local A = {}
30+
A.__index = A
31+
32+
function A:method() end
33+
34+
--- @return A
35+
function new()
36+
return setmetatable({}, A)
37+
end
38+
"#
39+
));
40+
}
41+
542
#[test]
643
fn test_issue_220() {
744
let mut ws = VirtualWorkspace::new();

0 commit comments

Comments
 (0)