1use std::collections::HashMap;
5
6use php_ast::{
7 ClassMemberKind, EnumMemberKind, ExprKind, NamespaceBody, Stmt, StmtKind, TypeHint,
8 TypeHintKind,
9};
10use tower_lsp::lsp_types::Position;
11
12use crate::document::ast::{ParsedDoc, SourceView};
13use crate::lang::docblock::{docblock_before, parse_docblock};
14use crate::lang::phpstorm_meta::PhpStormMeta;
15use crate::text::fqn_short_name;
16
17#[derive(Debug, Default, Clone)]
19pub struct TypeMap(HashMap<String, String>);
20
21impl TypeMap {
22 #[cfg(test)]
23 pub fn from_doc(doc: &ParsedDoc) -> Self {
24 Self::from_doc_with_meta(doc, None)
25 }
26
27 pub fn from_doc_with_meta(doc: &ParsedDoc, meta: Option<&PhpStormMeta>) -> Self {
30 let mut map = HashMap::new();
31 collect_types_stmts(
32 doc.source(),
33 &doc.program().stmts,
34 &mut map,
35 meta,
36 None,
37 doc,
38 );
39 let aliases = collect_file_type_aliases(doc.source(), &doc.program().stmts);
40 expand_type_aliases(&mut map, &aliases);
41 TypeMap(map)
42 }
43
44 pub fn from_doc_at_position(
50 doc: &ParsedDoc,
51 meta: Option<&PhpStormMeta>,
52 position: Position,
53 ) -> Self {
54 let cursor_byte = {
55 let line_starts = doc.line_starts();
56 let line = position.line as usize;
57 if line < line_starts.len() {
58 let line_start = line_starts[line] as usize;
59 let col_byte = crate::text::utf16_offset_to_byte(
60 &doc.source()[line_start..],
61 position.character as usize,
62 );
63 Some((line_start + col_byte) as u32)
64 } else {
65 None
66 }
67 };
68 let mut map = HashMap::new();
69 collect_types_stmts(
70 doc.source(),
71 &doc.program().stmts,
72 &mut map,
73 meta,
74 cursor_byte,
75 doc,
76 );
77 let aliases = collect_file_type_aliases(doc.source(), &doc.program().stmts);
78 expand_type_aliases(&mut map, &aliases);
79 TypeMap(map)
80 }
81
82 pub fn get<'a>(&'a self, var: &str) -> Option<&'a str> {
84 self.0.get(var).map(|s| s.as_str())
85 }
86}
87
88fn type_hint_to_class_string(
96 hint: &TypeHint<'_, '_>,
97 enclosing_class: Option<&str>,
98 doc: Option<&ParsedDoc>,
99) -> Option<String> {
100 use mir_types::Atomic;
101 let union = mir_analyzer::parser::type_from_hint(hint, enclosing_class);
102 let classes: Vec<String> = union
103 .types
104 .iter()
105 .filter_map(|a| match a {
106 Atomic::TNamedObject { fqcn, .. }
107 | Atomic::TSelf { fqcn }
108 | Atomic::TStaticObject { fqcn } => {
109 let short = fqn_short_name(fqcn);
110 Some(short.to_string())
111 }
112 Atomic::TParent { fqcn } => {
113 if let (Some(doc), Some(enc_class)) = (doc, enclosing_class) {
114 if let Some(parent) = parent_class_name(doc, enc_class) {
115 let short = fqn_short_name(&parent);
116 Some(short.to_string())
117 } else {
118 let short = fqn_short_name(fqcn);
119 Some(short.to_string())
120 }
121 } else {
122 let short = fqn_short_name(fqcn);
123 Some(short.to_string())
124 }
125 }
126 Atomic::TIntersection { parts } => {
127 let intersection_classes: Vec<String> = parts
128 .iter()
129 .flat_map(|part| {
130 part.types.iter().filter_map(|a| match a {
131 Atomic::TNamedObject { fqcn, .. }
132 | Atomic::TSelf { fqcn }
133 | Atomic::TStaticObject { fqcn } => {
134 let short = fqn_short_name(fqcn);
135 Some(short.to_string())
136 }
137 Atomic::TParent { fqcn } => {
138 if let (Some(doc), Some(enc_class)) = (doc, enclosing_class) {
139 if let Some(parent) = parent_class_name(doc, enc_class) {
140 let short = fqn_short_name(&parent);
141 Some(short.to_string())
142 } else {
143 let short =
144 fqcn.rsplit('\\').next().unwrap_or(fqcn.as_ref());
145 Some(short.to_string())
146 }
147 } else {
148 let short = fqn_short_name(fqcn);
149 Some(short.to_string())
150 }
151 }
152 _ => None,
153 })
154 })
155 .collect();
156 if intersection_classes.is_empty() {
157 None
158 } else {
159 Some(intersection_classes.join("|"))
160 }
161 }
162 _ => None,
163 })
164 .collect();
165 if classes.is_empty() {
166 None
167 } else {
168 Some(classes.join("|"))
169 }
170}
171
172fn docblock_class_parts(type_hint: &str) -> Vec<String> {
186 type_hint
187 .split('|')
188 .flat_map(|p| {
189 let p = p.trim().trim_start_matches('\\').trim_start_matches('?');
190 if let Some(elem) = php_iterable_element_type(p) {
192 return docblock_class_parts(elem);
193 }
194 if p.chars().next().map(|c| c.is_uppercase()).unwrap_or(false) {
196 let base = p.split_once('<').map(|(b, _)| b).unwrap_or(p);
197 if let Some(short) = base.rsplit('\\').next() {
198 return vec![short.to_string()];
199 }
200 }
201 vec![]
202 })
203 .collect()
204}
205
206fn php_iterable_element_type(ty: &str) -> Option<&str> {
210 let base = ty.split_once('<').map(|(b, _)| b).unwrap_or(ty);
211 if !matches!(
212 base,
213 "list" | "array" | "iterable" | "non-empty-list" | "non-empty-array"
214 ) {
215 return None;
216 }
217 let inner = ty.split_once('<')?.1.strip_suffix('>')?;
218 Some(if let Some((_, last)) = inner.rsplit_once(',') {
220 last.trim()
221 } else {
222 inner.trim()
223 })
224}
225
226fn collect_file_type_aliases(source: &str, stmts: &[Stmt<'_, '_>]) -> HashMap<String, String> {
230 let mut aliases = HashMap::new();
231 collect_aliases_in_stmts(source, stmts, &mut aliases);
232 aliases
233}
234
235fn collect_aliases_in_stmts(
236 source: &str,
237 stmts: &[Stmt<'_, '_>],
238 aliases: &mut HashMap<String, String>,
239) {
240 for stmt in stmts {
241 if let Some(raw) = docblock_before(source, stmt.span.start) {
242 let db = parse_docblock(&raw);
243 for ta in &db.type_aliases {
244 aliases.insert(ta.name.clone(), ta.type_expr.clone());
245 }
246 }
247 if let StmtKind::Namespace(ns) = &stmt.kind
248 && let NamespaceBody::Braced(inner) = &ns.body
249 {
250 collect_aliases_in_stmts(source, &inner.stmts, aliases);
251 }
252 }
253}
254
255fn expand_type_aliases(map: &mut HashMap<String, String>, aliases: &HashMap<String, String>) {
261 if aliases.is_empty() {
262 return;
263 }
264 for value in map.values_mut() {
265 if let Some(expanded_expr) = aliases.get(value.as_str()) {
266 let expanded_classes = docblock_class_parts(expanded_expr);
267 if !expanded_classes.is_empty() {
268 *value = expanded_classes.join("|");
269 }
270 }
271 }
272}
273
274fn collect_param_docblock_types(source: &str, span_start: u32, map: &mut HashMap<String, String>) {
278 let Some(raw) = docblock_before(source, span_start) else {
279 return;
280 };
281 let db = parse_docblock(&raw);
282 for param in &db.params {
283 let classes = docblock_class_parts(¶m.type_hint);
284 if classes.is_empty() {
285 continue;
286 }
287 let key = if param.name.starts_with('$') {
288 param.name.clone()
289 } else {
290 format!("${}", param.name)
291 };
292 map.entry(key).or_insert_with(|| classes.join("|"));
293 }
294}
295
296#[allow(clippy::too_many_arguments)]
297fn collect_types_stmts(
298 source: &str,
299 stmts: &[Stmt<'_, '_>],
300 map: &mut HashMap<String, String>,
301 meta: Option<&PhpStormMeta>,
302 cursor_byte: Option<u32>,
303 doc: &ParsedDoc,
304) {
305 for stmt in stmts {
306 if let Some(raw) = docblock_before(source, stmt.span.start) {
307 let db = parse_docblock(&raw);
308 if let Some(type_str) = db.var_type {
309 let class_name = docblock_class_parts(&type_str).into_iter().next();
310 if let Some(class_name) = class_name {
311 if let Some(vname) = db.var_name {
312 map.insert(format!("${}", vname.as_str()), class_name);
313 } else if let StmtKind::Expression(e) = &stmt.kind
314 && let ExprKind::Assign(a) = &e.kind
315 && let ExprKind::Variable(vn) = &a.target.kind
316 {
317 map.insert(format!("${}", vn.as_str()), class_name);
318 }
319 }
320 }
321 }
322
323 match &stmt.kind {
324 StmtKind::Expression(e) => collect_types_expr(source, e, map, meta, cursor_byte, doc),
325 StmtKind::Function(f) => {
326 let in_scope =
327 cursor_byte.is_none_or(|c| stmt.span.start <= c && c <= stmt.span.end);
328 if !in_scope {
329 continue;
330 }
331 collect_param_docblock_types(source, stmt.span.start, map);
332 for p in f.params.iter() {
333 if let Some(hint) = &p.type_hint
334 && let Some(class_str) = type_hint_to_class_string(hint, None, Some(doc))
335 {
336 map.insert(format!("${}", p.name), class_str);
337 }
338 }
339 collect_types_stmts(source, &f.body.stmts, map, meta, cursor_byte, doc);
340 }
341 StmtKind::Class(c) => {
342 let class_name = c.name.map(|n| n.to_string());
343 for member in c.body.members.iter() {
344 if let ClassMemberKind::Method(m) = &member.kind {
345 let in_scope = cursor_byte
346 .is_none_or(|cb| member.span.start <= cb && cb <= member.span.end);
347 if !in_scope {
348 continue;
349 }
350 collect_param_docblock_types(source, member.span.start, map);
351 for p in m.params.iter() {
352 if let Some(hint) = &p.type_hint
353 && let Some(class_str) = type_hint_to_class_string(
354 hint,
355 class_name.as_deref(),
356 Some(doc),
357 )
358 {
359 map.insert(format!("${}", p.name), class_str);
360 }
361 }
362 if !m.is_static
363 && let Some(ref cname) = class_name
364 {
365 map.insert("$this".to_string(), cname.clone());
366 }
367 if let Some(body) = &m.body {
368 collect_types_stmts(source, &body.stmts, map, meta, cursor_byte, doc);
369 }
370 }
371 }
372 }
373 StmtKind::Trait(t) => {
374 for member in t.body.members.iter() {
375 if let ClassMemberKind::Method(m) = &member.kind {
376 let in_scope = cursor_byte
377 .is_none_or(|cb| member.span.start <= cb && cb <= member.span.end);
378 if !in_scope {
379 continue;
380 }
381 for p in m.params.iter() {
382 if let Some(hint) = &p.type_hint
383 && let Some(class_str) =
384 type_hint_to_class_string(hint, None, Some(doc))
385 {
386 map.insert(format!("${}", p.name), class_str);
387 }
388 }
389 if let Some(body) = &m.body {
390 collect_types_stmts(source, &body.stmts, map, meta, cursor_byte, doc);
391 }
392 }
393 }
394 }
395 StmtKind::Enum(e) => {
396 for member in e.body.members.iter() {
397 if let EnumMemberKind::Method(m) = &member.kind {
398 let in_scope = cursor_byte
399 .is_none_or(|cb| member.span.start <= cb && cb <= member.span.end);
400 if !in_scope {
401 continue;
402 }
403 for p in m.params.iter() {
404 if let Some(hint) = &p.type_hint
405 && let Some(class_str) =
406 type_hint_to_class_string(hint, None, Some(doc))
407 {
408 map.insert(format!("${}", p.name), class_str);
409 }
410 }
411 if let Some(body) = &m.body {
412 collect_types_stmts(source, &body.stmts, map, meta, cursor_byte, doc);
413 }
414 }
415 }
416 }
417 StmtKind::Namespace(ns) => {
418 if let NamespaceBody::Braced(inner) = &ns.body {
419 collect_types_stmts(source, &inner.stmts, map, meta, cursor_byte, doc);
420 }
421 }
422 StmtKind::If(if_stmt) => {
424 collect_types_stmts(
425 source,
426 std::slice::from_ref(if_stmt.then_branch),
427 map,
428 meta,
429 cursor_byte,
430 doc,
431 );
432 for elseif in if_stmt.elseif_branches.iter() {
433 collect_types_stmts(
434 source,
435 std::slice::from_ref(&elseif.body),
436 map,
437 meta,
438 cursor_byte,
439 doc,
440 );
441 }
442 if let Some(else_branch) = if_stmt.else_branch {
443 collect_types_stmts(
444 source,
445 std::slice::from_ref(else_branch),
446 map,
447 meta,
448 cursor_byte,
449 doc,
450 );
451 }
452 }
453
454 StmtKind::Foreach(f) => {
455 if let ExprKind::Variable(arr_name) = &f.expr.kind
460 && let ExprKind::Variable(val_name) = &f.value.kind
461 {
462 let arr_key = format!("${}", arr_name.as_str());
463 if let Some(elem_type) = map.get(&arr_key).cloned() {
464 map.entry(format!("${}", val_name.as_str()))
465 .or_insert(elem_type);
466 }
467 }
468 collect_types_stmts(
469 source,
470 std::slice::from_ref(f.body),
471 map,
472 meta,
473 cursor_byte,
474 doc,
475 );
476 }
477 StmtKind::TryCatch(t) => {
478 collect_types_stmts(source, &t.body.stmts, map, meta, cursor_byte, doc);
479 for catch in t.catches.iter() {
480 collect_types_stmts(source, &catch.body.stmts, map, meta, cursor_byte, doc);
481 }
482 if let Some(finally) = &t.finally {
483 collect_types_stmts(source, &finally.stmts, map, meta, cursor_byte, doc);
484 }
485 }
486
487 StmtKind::StaticVar(vars) => {
488 for var in vars.iter() {
489 let var_key = format!("${}", &var.name.to_string());
490 if let Some(default) = &var.default {
491 if let ExprKind::New(new_expr) = &default.kind
492 && let Some(class_name) = extract_class_name(new_expr.class)
493 {
494 map.insert(var_key.clone(), class_name);
495 }
496 if let ExprKind::Array(_) = &default.kind {
497 map.insert(var_key, "array".to_string());
498 }
499 }
500 }
501 }
502
503 _ => {}
504 }
505 }
506}
507
508fn collect_types_expr(
509 source: &str,
510 expr: &php_ast::Expr<'_, '_>,
511 map: &mut HashMap<String, String>,
512 meta: Option<&PhpStormMeta>,
513 cursor_byte: Option<u32>,
514 doc: &ParsedDoc,
515) {
516 match &expr.kind {
517 ExprKind::Assign(assign) => {
518 if let ExprKind::Variable(var_name) = &assign.target.kind {
519 if let ExprKind::New(new_expr) = &assign.value.kind
520 && let Some(class_name) = extract_class_name(new_expr.class)
521 {
522 map.insert(format!("${}", var_name.as_str()), class_name);
523 }
524 if let Some(meta) = meta
528 && let Some(inferred) = infer_from_meta_method_call(assign.value, map, meta)
529 {
530 map.insert(format!("${}", var_name.as_str()), inferred);
531 }
532 if let ExprKind::CallableCreate(_) = &assign.value.kind {
536 map.entry(format!("${}", var_name.as_str()))
537 .or_insert_with(|| "Closure".to_string());
538 }
539 }
540 collect_types_expr(source, assign.value, map, meta, cursor_byte, doc);
541 }
542
543 ExprKind::Closure(c) => {
544 for p in c.params.iter() {
545 if let Some(hint) = &p.type_hint
546 && let TypeHintKind::Named(name) = &hint.kind
547 {
548 map.insert(format!("${}", p.name), name.to_string_repr().to_string());
549 }
550 }
551 let use_var_snapshot: Vec<(String, String)> = c
555 .use_vars
556 .iter()
557 .filter_map(|uv| {
558 let key = format!("${}", &uv.name.to_string());
559 map.get(&key).map(|ty| (key, ty.clone()))
560 })
561 .collect();
562 collect_types_stmts(source, &c.body.stmts, map, meta, cursor_byte, doc);
563 for (key, ty) in use_var_snapshot {
566 map.insert(key, ty);
567 }
568 }
569
570 ExprKind::ArrowFunction(af) => {
571 for p in af.params.iter() {
572 if let Some(hint) = &p.type_hint
573 && let TypeHintKind::Named(name) = &hint.kind
574 {
575 map.insert(format!("${}", p.name), name.to_string_repr().to_string());
576 }
577 }
578 collect_types_expr(source, af.body, map, meta, cursor_byte, doc);
579 }
580
581 _ => {}
582 }
583}
584
585fn extract_class_name(expr: &php_ast::Expr<'_, '_>) -> Option<String> {
586 match &expr.kind {
587 ExprKind::Identifier(name) => Some(name.as_str().to_string()),
588 _ => None,
589 }
590}
591
592fn infer_from_meta_method_call(
595 expr: &php_ast::Expr<'_, '_>,
596 var_map: &HashMap<String, String>,
597 meta: &PhpStormMeta,
598) -> Option<String> {
599 let ExprKind::MethodCall(m) = &expr.kind else {
600 return None;
601 };
602 let receiver_class = match &m.object.kind {
604 ExprKind::Variable(v) => {
605 let key = format!("${}", v.as_str());
606 var_map.get(&key)?.clone()
607 }
608 _ => return None,
609 };
610 let method_name = match &m.method.kind {
612 ExprKind::Identifier(n) => n.to_string(),
613 _ => return None,
614 };
615 let arg = m.args.first()?;
617 let arg_str = match &arg.value.kind {
618 ExprKind::String(s) => s.trim_start_matches('\\').to_string(),
619 ExprKind::ClassConstAccess(c) if c.member.name_str() == Some("class") => {
620 match &c.class.kind {
621 ExprKind::Identifier(n) => n
622 .trim_start_matches('\\')
623 .rsplit('\\')
624 .next()
625 .unwrap_or(n)
626 .to_string(),
627 _ => return None,
628 }
629 }
630 _ => return None,
631 };
632 meta.resolve_return_type(&receiver_class, &method_name, &arg_str)
633 .map(|s| s.to_string())
634}
635
636pub fn parent_class_name(doc: &ParsedDoc, class_name: &str) -> Option<String> {
638 parent_in_stmts(&doc.program().stmts, class_name)
639}
640
641fn parent_in_stmts(stmts: &[Stmt<'_, '_>], class_name: &str) -> Option<String> {
642 for stmt in stmts {
643 match &stmt.kind {
644 StmtKind::Class(c)
645 if c.name.as_ref().map(|n| n.to_string()) == Some(class_name.to_string()) =>
646 {
647 return c.extends.as_ref().map(|n| n.to_string_repr().to_string());
648 }
649 StmtKind::Namespace(ns) => {
650 if let NamespaceBody::Braced(inner) = &ns.body
651 && let found @ Some(_) = parent_in_stmts(&inner.stmts, class_name)
652 {
653 return found;
654 }
655 }
656 _ => {}
657 }
658 }
659 None
660}
661
662#[derive(Debug, Default)]
664pub struct ClassMembers {
665 pub methods: Vec<(String, bool)>,
667 pub properties: Vec<(String, bool)>,
669 pub readonly_properties: Vec<String>,
671 pub constants: Vec<String>,
672 pub parent: Option<String>,
674 pub trait_uses: Vec<String>,
676 pub found: bool,
680}
681
682pub fn members_of_class(doc: &ParsedDoc, class_name: &str) -> ClassMembers {
685 let mut out = ClassMembers::default();
686 out.parent = collect_members_stmts(doc.source(), &doc.program().stmts, class_name, &mut out);
687 out
688}
689
690fn collect_members_stmts(
691 source: &str,
692 stmts: &[Stmt<'_, '_>],
693 class_name: &str,
694 out: &mut ClassMembers,
695) -> Option<String> {
696 for stmt in stmts {
697 match &stmt.kind {
698 StmtKind::Class(c)
699 if c.name.as_ref().map(|n| n.to_string()) == Some(class_name.to_string()) =>
700 {
701 out.found = true;
702 if let Some(raw) = docblock_before(source, stmt.span.start) {
704 let db = parse_docblock(&raw);
705 for prop in &db.properties {
706 out.properties.push((prop.name.clone(), false));
707 }
708 for method in &db.methods {
709 out.methods.push((method.name.clone(), method.is_static));
710 }
711 }
712 for member in c.body.members.iter() {
713 match &member.kind {
714 ClassMemberKind::Method(m) => {
715 out.methods.push((m.name.to_string(), m.is_static));
716 if m.name == "__construct" {
717 for p in m.params.iter() {
718 if p.visibility.is_some() {
719 out.properties.push((p.name.to_string(), false));
720 let param_src =
722 &source[p.span.start as usize..p.span.end as usize];
723 if param_src.contains("readonly") {
724 out.readonly_properties.push(p.name.to_string());
725 }
726 }
727 }
728 }
729 }
730 ClassMemberKind::Property(p) => {
731 out.properties.push((p.name.to_string(), p.is_static));
732 if p.is_readonly {
733 out.readonly_properties.push(p.name.to_string());
734 }
735 }
736 ClassMemberKind::ClassConst(c) => {
737 out.constants.push(c.name.to_string());
738 }
739 ClassMemberKind::TraitUse(t) => {
740 for name in t.traits.iter() {
741 out.trait_uses.push(name.to_string_repr().to_string());
742 }
743 }
744 }
745 }
746 return c.extends.as_ref().map(|n| n.to_string_repr().to_string());
747 }
748 StmtKind::Enum(e) if e.name == class_name => {
749 out.found = true;
750 let is_backed = e.scalar_type.is_some();
751 out.properties.push(("name".to_string(), false));
752 if is_backed {
753 out.properties.push(("value".to_string(), false));
754 }
755 out.methods.push(("cases".to_string(), true));
756 if is_backed {
757 out.methods.push(("from".to_string(), true));
758 out.methods.push(("tryFrom".to_string(), true));
759 }
760 for member in e.body.members.iter() {
761 match &member.kind {
762 EnumMemberKind::Case(c) => {
763 out.constants.push(c.name.to_string());
764 }
765 EnumMemberKind::Method(m) => {
766 out.methods.push((m.name.to_string(), m.is_static));
767 }
768 EnumMemberKind::ClassConst(c) => {
769 out.constants.push(c.name.to_string());
770 }
771 _ => {}
772 }
773 }
774 return None; }
776 StmtKind::Trait(t) if t.name == class_name => {
777 out.found = true;
778 for member in t.body.members.iter() {
779 match &member.kind {
780 ClassMemberKind::Method(m) => {
781 out.methods.push((m.name.to_string(), m.is_static));
782 }
783 ClassMemberKind::Property(p) => {
784 out.properties.push((p.name.to_string(), p.is_static));
785 }
786 ClassMemberKind::ClassConst(c) => {
787 out.constants.push(c.name.to_string());
788 }
789 ClassMemberKind::TraitUse(t) => {
790 for name in t.traits.iter() {
791 out.trait_uses.push(name.to_string_repr().to_string());
792 }
793 }
794 }
795 }
796 return None; }
798 StmtKind::Namespace(ns) => {
799 if let NamespaceBody::Braced(inner) = &ns.body {
800 let result = collect_members_stmts(source, &inner.stmts, class_name, out);
801 if result.is_some() {
802 return result;
803 }
804 }
805 }
806 _ => {}
807 }
808 }
809 None
810}
811
812pub fn mixin_classes_of(doc: &ParsedDoc, class_name: &str) -> Vec<String> {
814 let source = doc.source();
815 mixin_classes_in_stmts(source, &doc.program().stmts, class_name)
816}
817
818fn mixin_classes_in_stmts(source: &str, stmts: &[Stmt<'_, '_>], class_name: &str) -> Vec<String> {
819 for stmt in stmts {
820 match &stmt.kind {
821 StmtKind::Class(c)
822 if c.name.as_ref().map(|n| n.to_string()) == Some(class_name.to_string()) =>
823 {
824 if let Some(raw) = docblock_before(source, stmt.span.start) {
825 return parse_docblock(&raw).mixins;
826 }
827 return vec![];
828 }
829 StmtKind::Namespace(ns) => {
830 if let NamespaceBody::Braced(inner) = &ns.body {
831 let found = mixin_classes_in_stmts(source, &inner.stmts, class_name);
832 if !found.is_empty() {
833 return found;
834 }
835 }
836 }
837 _ => {}
838 }
839 }
840 vec![]
841}
842
843pub fn enclosing_class_at(_source: &str, doc: &ParsedDoc, position: Position) -> Option<String> {
845 let sv = doc.view();
846 enclosing_class_in_stmts(sv, &doc.program().stmts, position)
847}
848
849pub fn enclosing_class_fqn_at(
855 _source: &str,
856 doc: &ParsedDoc,
857 position: Position,
858) -> Option<String> {
859 let sv = doc.view();
860 enclosing_class_fqn_in_stmts(sv, &doc.program().stmts, position, "")
861}
862
863fn enclosing_class_fqn_in_stmts(
864 sv: SourceView<'_>,
865 stmts: &[Stmt<'_, '_>],
866 pos: Position,
867 ns_prefix: &str,
868) -> Option<String> {
869 let make_fqn = |ns: &str, short: &str| -> String {
870 if ns.is_empty() {
871 short.to_owned()
872 } else {
873 format!("{}\\{}", ns, short)
874 }
875 };
876 let mut current_ns = ns_prefix.to_owned();
877 for stmt in stmts {
878 match &stmt.kind {
879 StmtKind::Class(c) => {
880 let start = sv.position_of(stmt.span.start).line;
881 let end = sv.position_of(stmt.span.end).line;
882 if pos.line >= start && pos.line <= end {
883 return c.name.map(|n| make_fqn(¤t_ns, &n.to_string()));
884 }
885 }
886 StmtKind::Interface(i) => {
887 let start = sv.position_of(stmt.span.start).line;
888 let end = sv.position_of(stmt.span.end).line;
889 if pos.line >= start && pos.line <= end {
890 return Some(make_fqn(¤t_ns, &i.name.to_string()));
891 }
892 }
893 StmtKind::Trait(t) => {
894 let start = sv.position_of(stmt.span.start).line;
895 let end = sv.position_of(stmt.span.end).line;
896 if pos.line >= start && pos.line <= end {
897 return Some(make_fqn(¤t_ns, &t.name.to_string()));
898 }
899 }
900 StmtKind::Enum(e) => {
901 let start = sv.position_of(stmt.span.start).line;
902 let end = sv.position_of(stmt.span.end).line;
903 if pos.line >= start && pos.line <= end {
904 return Some(make_fqn(¤t_ns, &e.name.to_string()));
905 }
906 }
907 StmtKind::Namespace(ns) => {
908 let ns_name = ns
909 .name
910 .as_ref()
911 .map(|n| n.to_string_repr().to_string())
912 .unwrap_or_default();
913 match &ns.body {
914 NamespaceBody::Braced(inner) => {
915 if let Some(found) =
916 enclosing_class_fqn_in_stmts(sv, &inner.stmts, pos, &ns_name)
917 {
918 return Some(found);
919 }
920 }
921 NamespaceBody::Simple => {
922 current_ns = ns_name;
923 }
924 }
925 }
926 _ => {}
927 }
928 }
929 None
930}
931
932pub fn enclosing_class_range_at(
937 doc: &ParsedDoc,
938 position: Position,
939) -> Option<tower_lsp::lsp_types::Range> {
940 let sv = doc.view();
941 enclosing_class_range_in_stmts(sv, &doc.program().stmts, position)
942}
943
944pub fn collect_all_class_ranges(doc: &ParsedDoc) -> Vec<tower_lsp::lsp_types::Range> {
948 let sv = doc.view();
949 let mut out = Vec::new();
950 collect_class_ranges_in_stmts(sv, &doc.program().stmts, &mut out);
951 out
952}
953
954fn collect_class_ranges_in_stmts(
955 sv: SourceView<'_>,
956 stmts: &[Stmt<'_, '_>],
957 out: &mut Vec<tower_lsp::lsp_types::Range>,
958) {
959 for stmt in stmts {
960 match &stmt.kind {
961 StmtKind::Class(_)
962 | StmtKind::Interface(_)
963 | StmtKind::Trait(_)
964 | StmtKind::Enum(_) => {
965 out.push(sv.range_of(stmt.span));
966 }
967 StmtKind::Namespace(ns) => {
968 if let NamespaceBody::Braced(inner) = &ns.body {
969 collect_class_ranges_in_stmts(sv, &inner.stmts, out);
970 }
971 }
972 _ => {}
973 }
974 }
975}
976
977fn enclosing_class_range_in_stmts(
978 sv: SourceView<'_>,
979 stmts: &[Stmt<'_, '_>],
980 pos: Position,
981) -> Option<tower_lsp::lsp_types::Range> {
982 for stmt in stmts {
983 match &stmt.kind {
984 StmtKind::Class(_)
985 | StmtKind::Interface(_)
986 | StmtKind::Trait(_)
987 | StmtKind::Enum(_) => {
988 let r = sv.range_of(stmt.span);
989 if pos.line >= r.start.line && pos.line <= r.end.line {
990 return Some(r);
991 }
992 }
993 StmtKind::Namespace(ns) => {
994 if let NamespaceBody::Braced(inner) = &ns.body
995 && let Some(r) = enclosing_class_range_in_stmts(sv, &inner.stmts, pos)
996 {
997 return Some(r);
998 }
999 }
1000 _ => {}
1001 }
1002 }
1003 None
1004}
1005
1006fn enclosing_class_in_stmts(
1007 sv: SourceView<'_>,
1008 stmts: &[Stmt<'_, '_>],
1009 pos: Position,
1010) -> Option<String> {
1011 for stmt in stmts {
1012 match &stmt.kind {
1013 StmtKind::Class(c) => {
1014 let start = sv.position_of(stmt.span.start).line;
1015 let end = sv.position_of(stmt.span.end).line;
1016 if pos.line >= start && pos.line <= end {
1017 return c.name.map(|n| n.to_string());
1018 }
1019 }
1020 StmtKind::Interface(i) => {
1021 let start = sv.position_of(stmt.span.start).line;
1022 let end = sv.position_of(stmt.span.end).line;
1023 if pos.line >= start && pos.line <= end {
1024 return Some(i.name.to_string());
1025 }
1026 }
1027 StmtKind::Trait(t) => {
1028 let start = sv.position_of(stmt.span.start).line;
1029 let end = sv.position_of(stmt.span.end).line;
1030 if pos.line >= start && pos.line <= end {
1031 return Some(t.name.to_string());
1032 }
1033 }
1034 StmtKind::Enum(e) => {
1035 let start = sv.position_of(stmt.span.start).line;
1036 let end = sv.position_of(stmt.span.end).line;
1037 if pos.line >= start && pos.line <= end {
1038 return Some(e.name.to_string());
1039 }
1040 }
1041 StmtKind::Namespace(ns) => {
1042 if let NamespaceBody::Braced(inner) = &ns.body
1043 && let Some(found) = enclosing_class_in_stmts(sv, &inner.stmts, pos)
1044 {
1045 return Some(found);
1046 }
1047 }
1048 _ => {}
1049 }
1050 }
1051 None
1052}
1053
1054pub fn params_of_function(doc: &ParsedDoc, func_name: &str) -> Vec<String> {
1056 let mut out = Vec::new();
1057 collect_params_stmts(&doc.program().stmts, func_name, &mut out);
1058 out
1059}
1060
1061pub fn params_of_method(doc: &ParsedDoc, class_name: &str, method_name: &str) -> Vec<String> {
1064 let mut out = Vec::new();
1065 collect_method_params_stmts(&doc.program().stmts, class_name, method_name, &mut out);
1066 out
1067}
1068
1069fn collect_method_params_stmts(
1070 stmts: &[php_ast::Stmt<'_, '_>],
1071 class_name: &str,
1072 method_name: &str,
1073 out: &mut Vec<String>,
1074) {
1075 for stmt in stmts {
1076 match &stmt.kind {
1077 StmtKind::Class(c)
1078 if c.name.as_ref().map(|n| n.to_string()) == Some(class_name.to_string()) =>
1079 {
1080 for member in c.body.members.iter() {
1081 if let ClassMemberKind::Method(m) = &member.kind
1082 && m.name == method_name
1083 {
1084 for p in m.params.iter() {
1085 out.push(p.name.to_string());
1086 }
1087 return;
1088 }
1089 }
1090 }
1091 StmtKind::Namespace(ns) => {
1092 if let NamespaceBody::Braced(inner) = &ns.body {
1093 collect_method_params_stmts(&inner.stmts, class_name, method_name, out);
1094 }
1095 }
1096 _ => {}
1097 }
1098 }
1099}
1100
1101pub fn is_enum(doc: &ParsedDoc, class_name: &str) -> bool {
1103 is_enum_in_stmts(&doc.program().stmts, class_name)
1104}
1105
1106fn is_enum_in_stmts(stmts: &[Stmt<'_, '_>], name: &str) -> bool {
1107 for stmt in stmts {
1108 match &stmt.kind {
1109 StmtKind::Enum(e) if e.name == name => return true,
1110 StmtKind::Namespace(ns) => {
1111 if let NamespaceBody::Braced(inner) = &ns.body
1112 && is_enum_in_stmts(&inner.stmts, name)
1113 {
1114 return true;
1115 }
1116 }
1117 _ => {}
1118 }
1119 }
1120 false
1121}
1122
1123pub fn is_backed_enum(doc: &ParsedDoc, class_name: &str) -> bool {
1126 is_backed_enum_in_stmts(&doc.program().stmts, class_name)
1127}
1128
1129fn is_backed_enum_in_stmts(stmts: &[Stmt<'_, '_>], name: &str) -> bool {
1130 for stmt in stmts {
1131 match &stmt.kind {
1132 StmtKind::Enum(e) if e.name == name => return e.scalar_type.is_some(),
1133 StmtKind::Namespace(ns) => {
1134 if let NamespaceBody::Braced(inner) = &ns.body
1135 && is_backed_enum_in_stmts(&inner.stmts, name)
1136 {
1137 return true;
1138 }
1139 }
1140 _ => {}
1141 }
1142 }
1143 false
1144}
1145
1146fn collect_params_stmts(stmts: &[Stmt<'_, '_>], func_name: &str, out: &mut Vec<String>) {
1147 for stmt in stmts {
1148 match &stmt.kind {
1149 StmtKind::Function(f) if f.name == func_name => {
1150 for p in f.params.iter() {
1151 out.push(p.name.to_string());
1152 }
1153 return;
1154 }
1155 StmtKind::Class(c) => {
1156 for member in c.body.members.iter() {
1157 if let ClassMemberKind::Method(m) = &member.kind
1158 && m.name == func_name
1159 {
1160 for p in m.params.iter() {
1161 out.push(p.name.to_string());
1162 }
1163 return;
1164 }
1165 }
1166 }
1167 StmtKind::Namespace(ns) => {
1168 if let NamespaceBody::Braced(inner) = &ns.body {
1169 collect_params_stmts(&inner.stmts, func_name, out);
1170 }
1171 }
1172 _ => {}
1173 }
1174 }
1175}
1176
1177#[cfg(test)]
1178mod tests {
1179 use super::*;
1180
1181 #[test]
1182 fn infers_type_from_new_expression() {
1183 let src = "<?php\n$obj = new Foo();";
1184 let doc = ParsedDoc::parse(src.to_string());
1185 let tm = TypeMap::from_doc(&doc);
1186 assert_eq!(tm.get("$obj"), Some("Foo"));
1187 }
1188
1189 #[test]
1190 fn unknown_variable_returns_none() {
1191 let src = "<?php\n$obj = new Foo();";
1192 let doc = ParsedDoc::parse(src.to_string());
1193 let tm = TypeMap::from_doc(&doc);
1194 assert!(tm.get("$other").is_none());
1195 }
1196
1197 #[test]
1198 fn multiple_assignments() {
1199 let src = "<?php\n$a = new Foo();\n$b = new Bar();";
1200 let doc = ParsedDoc::parse(src.to_string());
1201 let tm = TypeMap::from_doc(&doc);
1202 assert_eq!(tm.get("$a"), Some("Foo"));
1203 assert_eq!(tm.get("$b"), Some("Bar"));
1204 }
1205
1206 #[test]
1207 fn later_assignment_overwrites() {
1208 let src = "<?php\n$a = new Foo();\n$a = new Bar();";
1209 let doc = ParsedDoc::parse(src.to_string());
1210 let tm = TypeMap::from_doc(&doc);
1211 assert_eq!(tm.get("$a"), Some("Bar"));
1212 }
1213
1214 #[test]
1215 fn infers_type_from_typed_param() {
1216 let src = "<?php\nfunction process(Mailer $mailer): void { $mailer-> }";
1217 let doc = ParsedDoc::parse(src.to_string());
1218 let tm = TypeMap::from_doc(&doc);
1219 assert_eq!(tm.get("$mailer"), Some("Mailer"));
1220 }
1221
1222 #[test]
1223 fn parent_class_name_finds_parent() {
1224 let src = "<?php\nclass Base {}\nclass Child extends Base {}";
1225 let doc = ParsedDoc::parse(src.to_string());
1226 assert_eq!(parent_class_name(&doc, "Child"), Some("Base".to_string()));
1227 }
1228
1229 #[test]
1230 fn parent_class_name_returns_none_for_top_level() {
1231 let src = "<?php\nclass Base {}";
1232 let doc = ParsedDoc::parse(src.to_string());
1233 assert!(parent_class_name(&doc, "Base").is_none());
1234 }
1235
1236 #[test]
1237 fn members_of_class_includes_parent_field() {
1238 let src = "<?php\nclass Base {}\nclass Child extends Base {}";
1239 let doc = ParsedDoc::parse(src.to_string());
1240 let m = members_of_class(&doc, "Child");
1241 assert_eq!(m.parent.as_deref(), Some("Base"));
1242 }
1243
1244 #[test]
1245 fn members_of_class_finds_methods() {
1246 let src = "<?php\nclass Calc { public function add() {} public function sub() {} }";
1247 let doc = ParsedDoc::parse(src.to_string());
1248 let members = members_of_class(&doc, "Calc");
1249 let names: Vec<&str> = members.methods.iter().map(|(n, _)| n.as_str()).collect();
1250 assert!(names.contains(&"add"), "missing 'add'");
1251 assert!(names.contains(&"sub"), "missing 'sub'");
1252 }
1253
1254 #[test]
1255 fn members_of_unknown_class_is_empty() {
1256 let src = "<?php\nclass Calc { public function add() {} }";
1257 let doc = ParsedDoc::parse(src.to_string());
1258 let members = members_of_class(&doc, "Unknown");
1259 assert!(members.methods.is_empty());
1260 }
1261
1262 #[test]
1263 fn constructor_promoted_params_appear_as_properties() {
1264 let src = "<?php\nclass Point {\n public function __construct(\n public float $x,\n public float $y,\n ) {}\n}";
1265 let doc = ParsedDoc::parse(src.to_string());
1266 let members = members_of_class(&doc, "Point");
1267 let prop_names: Vec<&str> = members.properties.iter().map(|(n, _)| n.as_str()).collect();
1268 assert!(
1269 prop_names.contains(&"x"),
1270 "promoted param x should be a property"
1271 );
1272 assert!(
1273 prop_names.contains(&"y"),
1274 "promoted param y should be a property"
1275 );
1276 }
1277
1278 #[test]
1279 fn promoted_readonly_params_appear_in_readonly_properties() {
1280 let src = "<?php\nclass User {\n public function __construct(\n public readonly string $name,\n public int $age,\n ) {}\n}";
1281 let doc = ParsedDoc::parse(src.to_string());
1282 let members = members_of_class(&doc, "User");
1283 let prop_names: Vec<&str> = members.properties.iter().map(|(n, _)| n.as_str()).collect();
1284 assert!(
1285 prop_names.contains(&"name"),
1286 "promoted param name should be a property"
1287 );
1288 assert!(
1289 prop_names.contains(&"age"),
1290 "promoted param age should be a property"
1291 );
1292 assert!(
1293 members.readonly_properties.contains(&"name".to_string()),
1294 "readonly promoted param name should be in readonly_properties"
1295 );
1296 assert!(
1297 !members.readonly_properties.contains(&"age".to_string()),
1298 "non-readonly promoted param age should not be in readonly_properties"
1299 );
1300 }
1301
1302 #[test]
1303 fn enum_instance_members_include_name() {
1304 let src = "<?php\nenum Status { case Active; case Inactive; }";
1305 let doc = ParsedDoc::parse(src.to_string());
1306 let members = members_of_class(&doc, "Status");
1307 let prop_names: Vec<&str> = members.properties.iter().map(|(n, _)| n.as_str()).collect();
1308 assert!(
1309 prop_names.contains(&"name"),
1310 "pure enum should expose ->name"
1311 );
1312 assert!(
1313 !prop_names.contains(&"value"),
1314 "pure enum should not expose ->value"
1315 );
1316 }
1317
1318 #[test]
1319 fn backed_enum_exposes_value_and_factory_methods() {
1320 let src = "<?php\nenum Color: string { case Red = 'red'; }";
1321 let doc = ParsedDoc::parse(src.to_string());
1322 let members = members_of_class(&doc, "Color");
1323 let prop_names: Vec<&str> = members.properties.iter().map(|(n, _)| n.as_str()).collect();
1324 let method_names: Vec<&str> = members.methods.iter().map(|(n, _)| n.as_str()).collect();
1325 assert!(
1326 prop_names.contains(&"value"),
1327 "backed enum should expose ->value"
1328 );
1329 assert!(
1330 method_names.contains(&"from"),
1331 "backed enum should have ::from()"
1332 );
1333 assert!(
1334 method_names.contains(&"tryFrom"),
1335 "backed enum should have ::tryFrom()"
1336 );
1337 assert!(
1338 method_names.contains(&"cases"),
1339 "enum should have ::cases()"
1340 );
1341 }
1342
1343 #[test]
1344 fn enum_cases_appear_as_constants() {
1345 let src = "<?php\nenum Status { case Active; case Inactive; }";
1346 let doc = ParsedDoc::parse(src.to_string());
1347 let members = members_of_class(&doc, "Status");
1348 assert!(members.constants.contains(&"Active".to_string()));
1349 assert!(members.constants.contains(&"Inactive".to_string()));
1350 }
1351
1352 #[test]
1353 fn trait_members_are_collected() {
1354 let src = "<?php\ntrait Logging { public function log() {} public string $logFile; }";
1355 let doc = ParsedDoc::parse(src.to_string());
1356 let members = members_of_class(&doc, "Logging");
1357 let method_names: Vec<&str> = members.methods.iter().map(|(n, _)| n.as_str()).collect();
1358 let prop_names: Vec<&str> = members.properties.iter().map(|(n, _)| n.as_str()).collect();
1359 assert!(
1360 method_names.contains(&"log"),
1361 "trait method log should be collected"
1362 );
1363 assert!(
1364 prop_names.contains(&"logFile"),
1365 "trait property logFile should be collected"
1366 );
1367 }
1368
1369 #[test]
1370 fn class_with_trait_use_lists_trait() {
1371 let src = "<?php\ntrait Logging { public function log() {} }\nclass App { use Logging; }";
1372 let doc = ParsedDoc::parse(src.to_string());
1373 let members = members_of_class(&doc, "App");
1374 assert!(
1375 members.trait_uses.contains(&"Logging".to_string()),
1376 "should list used trait"
1377 );
1378 }
1379
1380 #[test]
1381 fn var_docblock_with_explicit_varname_infers_type() {
1382 let src = "<?php\n/** @var Mailer $mailer */\n$mailer = $container->get('mailer');";
1383 let doc = ParsedDoc::parse(src.to_string());
1384 let tm = TypeMap::from_doc(&doc);
1385 assert_eq!(
1386 tm.get("$mailer"),
1387 Some("Mailer"),
1388 "@var with explicit name should map the variable"
1389 );
1390 }
1391
1392 #[test]
1393 fn var_docblock_without_varname_infers_from_assignment() {
1394 let src = "<?php\n/** @var Repository */\n$repo = $this->getRepository();";
1395 let doc = ParsedDoc::parse(src.to_string());
1396 let tm = TypeMap::from_doc(&doc);
1397 assert_eq!(
1398 tm.get("$repo"),
1399 Some("Repository"),
1400 "@var without name should use assignment LHS"
1401 );
1402 }
1403
1404 #[test]
1405 fn var_docblock_does_not_map_primitive_types() {
1406 let src = "<?php\n/** @var string */\n$name = 'hello';";
1407 let doc = ParsedDoc::parse(src.to_string());
1408 let tm = TypeMap::from_doc(&doc);
1409 assert!(
1411 tm.get("$name").is_none(),
1412 "primitive @var should not produce a class mapping"
1413 );
1414 }
1415
1416 #[test]
1417 fn var_nullable_docblock_maps_to_class() {
1418 let src = "<?php\n/** @var ?Mailer $mailer */\n$mailer = null;";
1421 let doc = ParsedDoc::parse(src.to_string());
1422 let tm = TypeMap::from_doc(&doc);
1423 assert_eq!(
1424 tm.get("$mailer"),
1425 Some("Mailer"),
1426 "@var ?Foo should map to 'Foo', not 'Foo|null'"
1427 );
1428 }
1429
1430 #[test]
1431 fn var_union_docblock_maps_first_class() {
1432 let src = "<?php\n/** @var Repository|null $repo */\n$repo = null;";
1434 let doc = ParsedDoc::parse(src.to_string());
1435 let tm = TypeMap::from_doc(&doc);
1436 assert_eq!(
1437 tm.get("$repo"),
1438 Some("Repository"),
1439 "@var Foo|null should map to 'Foo', not 'Foo|null'"
1440 );
1441 }
1442
1443 #[test]
1444 fn is_enum_pure() {
1445 let src = "<?php\nenum Suit { case Hearts; case Clubs; }";
1446 let doc = ParsedDoc::parse(src.to_string());
1447 assert!(is_enum(&doc, "Suit"));
1448 assert!(!is_backed_enum(&doc, "Suit"));
1449 }
1450
1451 #[test]
1452 fn is_backed_enum_string() {
1453 let src = "<?php\nenum Status: string { case Active = 'active'; }";
1454 let doc = ParsedDoc::parse(src.to_string());
1455 assert!(is_enum(&doc, "Status"));
1456 assert!(is_backed_enum(&doc, "Status"));
1457 }
1458
1459 #[test]
1460 fn is_enum_false_for_class() {
1461 let src = "<?php\nclass Foo {}";
1462 let doc = ParsedDoc::parse(src.to_string());
1463 assert!(!is_enum(&doc, "Foo"));
1464 assert!(!is_backed_enum(&doc, "Foo"));
1465 }
1466
1467 #[test]
1468 fn closure_use_var_type_is_available_inside_body() {
1469 let src = "<?php\n$svc = new PaymentService();\n$fn = function() use ($svc) { $svc->process(); };";
1470 let doc = ParsedDoc::parse(src.to_string());
1471 let tm = TypeMap::from_doc(&doc);
1472 assert_eq!(
1473 tm.get("$svc"),
1474 Some("PaymentService"),
1475 "captured use variable should retain its outer type inside closure body"
1476 );
1477 }
1478
1479 #[test]
1480 fn closure_use_var_inner_assignment_does_not_override_outer_type() {
1481 let src = "<?php\n$svc = new PaymentService();\n$fn = function() use ($svc) { $svc = new OtherService(); };";
1484 let doc = ParsedDoc::parse(src.to_string());
1485 let tm = TypeMap::from_doc(&doc);
1486 assert_eq!(
1488 tm.get("$svc"),
1489 Some("PaymentService"),
1490 "outer type should not be overwritten by inner assignment in closure"
1491 );
1492 }
1493
1494 #[test]
1495 fn param_docblock_type_inferred() {
1496 let src = "<?php\n/**\n * @param Mailer $mailer\n */\nfunction send($mailer) { $mailer-> }";
1497 let doc = ParsedDoc::parse(src.to_string());
1498 let tm = TypeMap::from_doc(&doc);
1499 assert_eq!(tm.get("$mailer"), Some("Mailer"));
1500 }
1501
1502 #[test]
1503 fn param_docblock_does_not_override_ast_hint() {
1504 let src = "<?php\n/**\n * @param OtherClass $x\n */\nfunction foo(Foo $x) {}";
1505 let doc = ParsedDoc::parse(src.to_string());
1506 let tm = TypeMap::from_doc(&doc);
1507 assert_eq!(tm.get("$x"), Some("Foo"));
1509 }
1510
1511 #[test]
1512 fn not_null_check_preserves_existing_type() {
1513 let src = "<?php\n$x = new Foo();\nif ($x !== null) { $x-> }";
1514 let doc = ParsedDoc::parse(src.to_string());
1515 let tm = TypeMap::from_doc(&doc);
1516 assert_eq!(tm.get("$x"), Some("Foo"));
1517 }
1518
1519 #[test]
1520 fn docblock_property_appears_in_members() {
1521 let src =
1522 "<?php\n/**\n * @property string $email\n * @property-read int $id\n */\nclass User {}";
1523 let doc = ParsedDoc::parse(src.to_string());
1524 let members = members_of_class(&doc, "User");
1525 let props: Vec<&str> = members.properties.iter().map(|(n, _)| n.as_str()).collect();
1526 assert!(props.contains(&"email"));
1527 assert!(props.contains(&"id"));
1528 }
1529
1530 #[test]
1531 fn docblock_method_appears_in_members() {
1532 let src = "<?php\n/**\n * @method User find(int $id)\n * @method static Builder where(string $col, mixed $val)\n */\nclass Model {}";
1533 let doc = ParsedDoc::parse(src.to_string());
1534 let members = members_of_class(&doc, "Model");
1535 let method_names: Vec<&str> = members.methods.iter().map(|(n, _)| n.as_str()).collect();
1536 assert!(method_names.contains(&"find"));
1537 assert!(method_names.contains(&"where"));
1538 let where_static = members
1539 .methods
1540 .iter()
1541 .find(|(n, _)| n == "where")
1542 .map(|(_, s)| *s);
1543 assert_eq!(where_static, Some(true));
1544 }
1545
1546 #[test]
1547 fn union_type_param_maps_both_classes() {
1548 let src = "<?php\nfunction f(Foo|Bar $x) {}";
1550 let doc = ParsedDoc::parse(src.to_string());
1551 let tm = TypeMap::from_doc(&doc);
1552 let val = tm.get("$x").expect("$x should be in the type map");
1553 assert!(
1554 val.contains("Foo"),
1555 "union type should contain 'Foo', got: {}",
1556 val
1557 );
1558 assert!(
1559 val.contains("Bar"),
1560 "union type should contain 'Bar', got: {}",
1561 val
1562 );
1563 }
1564
1565 #[test]
1566 fn nullable_param_resolves_to_class() {
1567 let src = "<?php\nfunction f(?Foo $x) {}";
1569 let doc = ParsedDoc::parse(src.to_string());
1570 let tm = TypeMap::from_doc(&doc);
1571 assert_eq!(
1572 tm.get("$x"),
1573 Some("Foo"),
1574 "nullable type hint ?Foo should map $x to Foo"
1575 );
1576 }
1577
1578 #[test]
1579 fn null_assignment_does_not_overwrite_class() {
1580 let src = "<?php\n$x = new Foo();\n$x = null;\n";
1583 let doc = ParsedDoc::parse(src.to_string());
1584 let tm = TypeMap::from_doc(&doc);
1585 assert_eq!(
1588 tm.get("$x"),
1589 Some("Foo"),
1590 "$x should retain its Foo type after being assigned null"
1591 );
1592 }
1593
1594 #[test]
1595 fn infers_type_from_assignment_inside_trait_method() {
1596 let src = "<?php\ntrait Builder { public function make(): void { $obj = new Widget(); } }";
1597 let doc = ParsedDoc::parse(src.to_string());
1598 let tm = TypeMap::from_doc(&doc);
1599 assert_eq!(
1600 tm.get("$obj"),
1601 Some("Widget"),
1602 "type map should walk into trait method bodies"
1603 );
1604 }
1605
1606 #[test]
1607 fn infers_type_from_assignment_inside_enum_method() {
1608 let src = "<?php\nenum Color { case Red; public function make(): void { $obj = new Palette(); } }";
1609 let doc = ParsedDoc::parse(src.to_string());
1610 let tm = TypeMap::from_doc(&doc);
1611 assert_eq!(
1612 tm.get("$obj"),
1613 Some("Palette"),
1614 "type map should walk into enum method bodies"
1615 );
1616 }
1617
1618 #[test]
1621 fn generic_type_annotation_stripped_to_base_class() {
1622 let src = "<?php\n/** @var Collection<User> $coll */\n$coll = get();";
1625 let doc = ParsedDoc::parse(src.to_string());
1626 let tm = TypeMap::from_doc(&doc);
1627 assert_eq!(
1628 tm.get("$coll"),
1629 Some("Collection"),
1630 "generic param should be stripped; expected Collection, not Collection<User>"
1631 );
1632 }
1633
1634 #[test]
1637 fn list_var_annotation_maps_to_element_type() {
1638 let src = "<?php\n/** @var list<Widget> $items */\n$items = get();";
1641 let doc = ParsedDoc::parse(src.to_string());
1642 let tm = TypeMap::from_doc(&doc);
1643 assert_eq!(
1644 tm.get("$items"),
1645 Some("Widget"),
1646 "element type Widget should be extracted from list<Widget>"
1647 );
1648 }
1649
1650 #[test]
1651 fn array_element_annotation_maps_to_element_type() {
1652 let src = "<?php\n/** @var array<Widget> $map */\n$map = get();";
1657 let doc = ParsedDoc::parse(src.to_string());
1658 let tm = TypeMap::from_doc(&doc);
1659 assert_eq!(
1660 tm.get("$map"),
1661 Some("Widget"),
1662 "element type Widget should be extracted from array<Widget>"
1663 );
1664 }
1665
1666 #[test]
1667 fn foreach_value_var_inherits_element_type() {
1668 let src = "<?php\n/** @var list<Widget> $items */\n$items = get();\nforeach ($items as $w) { $w; }";
1671 let doc = ParsedDoc::parse(src.to_string());
1672 let tm = TypeMap::from_doc(&doc);
1673 assert_eq!(
1674 tm.get("$w"),
1675 Some("Widget"),
1676 "$w should inherit Widget element type inside foreach"
1677 );
1678 }
1679
1680 #[test]
1683 fn psalm_type_alias_expanded_for_param() {
1684 let src = r#"<?php
1687/** @psalm-type Result = Success|Failure */
1688class Processor {
1689 /** @param Result $r */
1690 public function handle($r): void {}
1691}"#;
1692 let doc = ParsedDoc::parse(src.to_string());
1693 let tm = TypeMap::from_doc(&doc);
1694 let val = tm.get("$r").expect("$r should be in the type map");
1695 assert!(
1696 val.contains("Success"),
1697 "alias should expand to include Success; got: {val}"
1698 );
1699 assert!(
1700 val.contains("Failure"),
1701 "alias should expand to include Failure; got: {val}"
1702 );
1703 }
1704
1705 #[test]
1708 fn first_class_callable_maps_to_closure() {
1709 let src = "<?php\n$fn = strlen(...);";
1710 let doc = ParsedDoc::parse(src.to_string());
1711 let tm = TypeMap::from_doc(&doc);
1712 assert_eq!(
1713 tm.get("$fn"),
1714 Some("Closure"),
1715 "first-class callable should be mapped to Closure"
1716 );
1717 }
1718
1719 #[test]
1720 fn first_class_method_callable_maps_to_closure() {
1721 let src = "<?php\n$obj = new Foo();\n$m = $obj->bar(...);";
1722 let doc = ParsedDoc::parse(src.to_string());
1723 let tm = TypeMap::from_doc(&doc);
1724 assert_eq!(
1725 tm.get("$m"),
1726 Some("Closure"),
1727 "method first-class callable should be mapped to Closure"
1728 );
1729 }
1730}