1
- use emmylua_parser:: { LuaAstNode , LuaClosureExpr , LuaReturnStat } ;
2
- use rowan:: TextRange ;
1
+ use emmylua_parser:: { LuaAstNode , LuaClosureExpr , LuaExpr , LuaReturnStat } ;
2
+ use rowan:: { NodeOrToken , TextRange } ;
3
3
4
4
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 ,
7
7
} ;
8
8
9
9
use super :: { get_own_return_stats, Checker , DiagnosticContext } ;
@@ -49,7 +49,12 @@ fn check_return_stat(
49
49
& return_stat. get_expr_list ( ) . collect :: < Vec < _ > > ( ) ,
50
50
None ,
51
51
) ?;
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
+ }
53
58
let return_expr_ranges = infos
54
59
. iter ( )
55
60
. map ( |( _, range) | range. clone ( ) )
@@ -62,7 +67,6 @@ fn check_return_stat(
62
67
if return_expr_types. len ( ) < index {
63
68
break ;
64
69
}
65
-
66
70
check_variadic_return_type_match (
67
71
context,
68
72
semantic_model,
@@ -136,25 +140,38 @@ fn add_type_check_diagnostic(
136
140
Ok ( _) => return ,
137
141
Err ( reason) => match reason {
138
142
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
+ ) ;
140
156
}
141
157
TypeCheckFailReason :: TypeNotMatch => {
142
158
context. add_diagnostic (
143
159
DiagnosticCode :: ReturnTypeMismatch ,
144
160
range,
145
161
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}. " ,
147
163
index = index + 1 ,
148
164
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 = ""
150
167
)
151
168
. to_string ( ) ,
152
169
None ,
153
170
) ;
154
171
}
155
172
TypeCheckFailReason :: TypeRecursion => {
156
173
context. add_diagnostic (
157
- DiagnosticCode :: ParamTypeNotMatch ,
174
+ DiagnosticCode :: ReturnTypeMismatch ,
158
175
range,
159
176
"type recursion" . into ( ) ,
160
177
None ,
@@ -163,3 +180,31 @@ fn add_type_check_diagnostic(
163
180
} ,
164
181
}
165
182
}
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
+ }
0 commit comments