Skip to main content

php_ast/owned/
fold.rs

1//! Owned AST transformation via the [`FoldOwned`] trait.
2//!
3//! [`FoldOwned`] is the transformation counterpart of [`super::visitor::OwnedVisitor`].
4//! Where `OwnedVisitor` reads nodes in place, `FoldOwned` rebuilds them — reading
5//! from an input node (borrowed) and returning a new owned value. Override only the
6//! node types you want to change; all others are rebuilt identically by the default
7//! implementations (equivalent to `Clone`).
8//!
9//! # Example
10//!
11//! ```
12//! use php_ast::owned::fold::{FoldOwned, fold_owned_expr};
13//! use php_ast::owned::{Expr, ExprKind};
14//!
15//! struct NegateInts;
16//!
17//! impl FoldOwned for NegateInts {
18//!     fn fold_expr(&mut self, expr: &Expr) -> Expr {
19//!         if let ExprKind::Int(n) = &expr.kind {
20//!             return Expr { kind: ExprKind::Int(-n), span: expr.span };
21//!         }
22//!         fold_owned_expr(self, expr)
23//!     }
24//! }
25//! ```
26
27use super::*;
28
29// =============================================================================
30// FoldOwned trait
31// =============================================================================
32
33/// Trait for transforming owned PHP AST nodes.
34///
35/// All methods have identity-fold default implementations that call the
36/// corresponding free `fold_owned_*` function. Override only the node types
37/// you want to change; the rest recurse automatically.
38pub trait FoldOwned {
39    fn fold_program(&mut self, program: &Program) -> Program {
40        fold_owned_program(self, program)
41    }
42
43    fn fold_stmt(&mut self, stmt: &Stmt) -> Stmt {
44        fold_owned_stmt(self, stmt)
45    }
46
47    fn fold_block(&mut self, block: &Block) -> Block {
48        fold_owned_block(self, block)
49    }
50
51    fn fold_expr(&mut self, expr: &Expr) -> Expr {
52        fold_owned_expr(self, expr)
53    }
54
55    fn fold_param(&mut self, param: &Param) -> Param {
56        fold_owned_param(self, param)
57    }
58
59    fn fold_arg(&mut self, arg: &Arg) -> Arg {
60        fold_owned_arg(self, arg)
61    }
62
63    fn fold_class_member(&mut self, member: &ClassMember) -> ClassMember {
64        fold_owned_class_member(self, member)
65    }
66
67    fn fold_enum_member(&mut self, member: &EnumMember) -> EnumMember {
68        fold_owned_enum_member(self, member)
69    }
70
71    fn fold_property_hook(&mut self, hook: &PropertyHook) -> PropertyHook {
72        fold_owned_property_hook(self, hook)
73    }
74
75    fn fold_type_hint(&mut self, type_hint: &TypeHint) -> TypeHint {
76        fold_owned_type_hint(self, type_hint)
77    }
78
79    fn fold_attribute(&mut self, attribute: &Attribute) -> Attribute {
80        fold_owned_attribute(self, attribute)
81    }
82
83    fn fold_catch_clause(&mut self, catch: &CatchClause) -> CatchClause {
84        fold_owned_catch_clause(self, catch)
85    }
86
87    fn fold_match_arm(&mut self, arm: &MatchArm) -> MatchArm {
88        fold_owned_match_arm(self, arm)
89    }
90
91    fn fold_closure_use_var(&mut self, var: &ClosureUseVar) -> ClosureUseVar {
92        fold_owned_closure_use_var(self, var)
93    }
94
95    fn fold_name(&mut self, name: &Name) -> Name {
96        fold_owned_name(self, name)
97    }
98}
99
100// =============================================================================
101// Free fold functions
102// =============================================================================
103
104pub fn fold_owned_program<F: FoldOwned + ?Sized>(folder: &mut F, program: &Program) -> Program {
105    Program {
106        stmts: program.stmts.iter().map(|s| folder.fold_stmt(s)).collect(),
107        span: program.span,
108    }
109}
110
111pub fn fold_owned_stmt<F: FoldOwned + ?Sized>(folder: &mut F, stmt: &Stmt) -> Stmt {
112    Stmt {
113        kind: fold_owned_stmt_kind(folder, &stmt.kind),
114        span: stmt.span,
115        doc_comment: stmt.doc_comment.clone(),
116    }
117}
118
119fn fold_owned_stmts<F: FoldOwned + ?Sized>(folder: &mut F, stmts: &[Stmt]) -> Box<[Stmt]> {
120    stmts.iter().map(|s| folder.fold_stmt(s)).collect()
121}
122
123pub fn fold_owned_block<F: FoldOwned + ?Sized>(folder: &mut F, block: &Block) -> Block {
124    Block {
125        stmts: fold_owned_stmts(folder, &block.stmts),
126        span: block.span,
127    }
128}
129
130fn fold_owned_exprs<F: FoldOwned + ?Sized>(folder: &mut F, exprs: &[Expr]) -> Box<[Expr]> {
131    exprs.iter().map(|e| folder.fold_expr(e)).collect()
132}
133
134fn fold_owned_args<F: FoldOwned + ?Sized>(folder: &mut F, args: &[Arg]) -> Box<[Arg]> {
135    args.iter().map(|a| folder.fold_arg(a)).collect()
136}
137
138fn fold_owned_attrs<F: FoldOwned + ?Sized>(
139    folder: &mut F,
140    attrs: &[Attribute],
141) -> Box<[Attribute]> {
142    attrs.iter().map(|a| folder.fold_attribute(a)).collect()
143}
144
145fn fold_owned_params<F: FoldOwned + ?Sized>(folder: &mut F, params: &[Param]) -> Box<[Param]> {
146    params.iter().map(|p| folder.fold_param(p)).collect()
147}
148
149fn fold_owned_hooks<F: FoldOwned + ?Sized>(
150    folder: &mut F,
151    hooks: &[PropertyHook],
152) -> Box<[PropertyHook]> {
153    hooks.iter().map(|h| folder.fold_property_hook(h)).collect()
154}
155
156fn fold_owned_members<F: FoldOwned + ?Sized>(
157    folder: &mut F,
158    members: &[ClassMember],
159) -> Box<[ClassMember]> {
160    members
161        .iter()
162        .map(|m| folder.fold_class_member(m))
163        .collect()
164}
165
166fn fold_owned_string_parts<F: FoldOwned + ?Sized>(
167    folder: &mut F,
168    parts: &[StringPart],
169) -> Box<[StringPart]> {
170    parts
171        .iter()
172        .map(|p| match p {
173            StringPart::Literal(s) => StringPart::Literal(s.clone()),
174            StringPart::Expr(e) => StringPart::Expr(folder.fold_expr(e)),
175        })
176        .collect()
177}
178
179fn fold_owned_stmt_kind<F: FoldOwned + ?Sized>(folder: &mut F, k: &StmtKind) -> StmtKind {
180    match k {
181        StmtKind::Expression(e) => StmtKind::Expression(Box::new(folder.fold_expr(e))),
182        StmtKind::Echo(exprs) => StmtKind::Echo(fold_owned_exprs(folder, exprs)),
183        StmtKind::Return(e) => StmtKind::Return(e.as_ref().map(|e| Box::new(folder.fold_expr(e)))),
184        StmtKind::Block(block) => StmtKind::Block(Box::new(folder.fold_block(block))),
185        StmtKind::If(s) => StmtKind::If(Box::new(IfStmt {
186            condition: folder.fold_expr(&s.condition),
187            then_branch: Box::new(folder.fold_stmt(&s.then_branch)),
188            elseif_branches: s
189                .elseif_branches
190                .iter()
191                .map(|b| ElseIfBranch {
192                    condition: folder.fold_expr(&b.condition),
193                    body: folder.fold_stmt(&b.body),
194                    span: b.span,
195                })
196                .collect(),
197            else_branch: s
198                .else_branch
199                .as_ref()
200                .map(|b| Box::new(folder.fold_stmt(b))),
201            else_kw_start: s.else_kw_start,
202            uses_alternative: s.uses_alternative,
203        })),
204        StmtKind::While(s) => StmtKind::While(Box::new(WhileStmt {
205            condition: folder.fold_expr(&s.condition),
206            body: Box::new(folder.fold_stmt(&s.body)),
207            uses_alternative: s.uses_alternative,
208        })),
209        StmtKind::For(s) => StmtKind::For(Box::new(ForStmt {
210            init: fold_owned_exprs(folder, &s.init),
211            condition: fold_owned_exprs(folder, &s.condition),
212            update: fold_owned_exprs(folder, &s.update),
213            body: Box::new(folder.fold_stmt(&s.body)),
214            uses_alternative: s.uses_alternative,
215        })),
216        StmtKind::Foreach(s) => StmtKind::Foreach(Box::new(ForeachStmt {
217            expr: folder.fold_expr(&s.expr),
218            key: s.key.as_ref().map(|e| folder.fold_expr(e)),
219            value: folder.fold_expr(&s.value),
220            body: Box::new(folder.fold_stmt(&s.body)),
221            uses_alternative: s.uses_alternative,
222        })),
223        StmtKind::DoWhile(s) => StmtKind::DoWhile(Box::new(DoWhileStmt {
224            body: Box::new(folder.fold_stmt(&s.body)),
225            condition: folder.fold_expr(&s.condition),
226        })),
227        StmtKind::Function(f) => StmtKind::Function(Box::new(FunctionDecl {
228            name: f.name.clone(),
229            params: fold_owned_params(folder, &f.params),
230            body: Box::new(folder.fold_block(&f.body)),
231            return_type: f.return_type.as_ref().map(|t| folder.fold_type_hint(t)),
232            by_ref: f.by_ref,
233            attributes: fold_owned_attrs(folder, &f.attributes),
234            doc_comment: f.doc_comment.clone(),
235        })),
236        StmtKind::Break(e) => StmtKind::Break(e.as_ref().map(|e| Box::new(folder.fold_expr(e)))),
237        StmtKind::Continue(e) => {
238            StmtKind::Continue(e.as_ref().map(|e| Box::new(folder.fold_expr(e))))
239        }
240        StmtKind::Switch(s) => StmtKind::Switch(Box::new(SwitchStmt {
241            expr: folder.fold_expr(&s.expr),
242            body: SwitchBody {
243                cases: s
244                    .body
245                    .cases
246                    .iter()
247                    .map(|c| SwitchCase {
248                        value: c.value.as_ref().map(|v| folder.fold_expr(v)),
249                        body: fold_owned_stmts(folder, &c.body),
250                        span: c.span,
251                    })
252                    .collect(),
253                span: s.body.span,
254            },
255            uses_alternative: s.uses_alternative,
256        })),
257        StmtKind::Goto(ident) => StmtKind::Goto(ident.clone()),
258        StmtKind::Label(s) => StmtKind::Label(s.clone()),
259        StmtKind::Declare(d) => StmtKind::Declare(Box::new(DeclareStmt {
260            directives: d
261                .directives
262                .iter()
263                .map(|(k, v)| (k.clone(), folder.fold_expr(v)))
264                .collect(),
265            body: d.body.as_ref().map(|b| Box::new(folder.fold_stmt(b))),
266            uses_alternative: d.uses_alternative,
267        })),
268        StmtKind::Unset(exprs) => StmtKind::Unset(fold_owned_exprs(folder, exprs)),
269        StmtKind::Throw(e) => StmtKind::Throw(Box::new(folder.fold_expr(e))),
270        StmtKind::TryCatch(t) => StmtKind::TryCatch(Box::new(TryCatchStmt {
271            body: Box::new(folder.fold_block(&t.body)),
272            catches: t
273                .catches
274                .iter()
275                .map(|c| folder.fold_catch_clause(c))
276                .collect(),
277            finally: t.finally.as_deref().map(|f| Box::new(folder.fold_block(f))),
278            finally_kw_start: t.finally_kw_start,
279        })),
280        StmtKind::Global(exprs) => StmtKind::Global(fold_owned_exprs(folder, exprs)),
281        StmtKind::Class(cls) => StmtKind::Class(Box::new(fold_owned_class_decl(folder, cls))),
282        StmtKind::Interface(iface) => {
283            StmtKind::Interface(Box::new(fold_owned_interface_decl(folder, iface)))
284        }
285        StmtKind::Trait(tr) => StmtKind::Trait(Box::new(fold_owned_trait_decl(folder, tr))),
286        StmtKind::Enum(en) => StmtKind::Enum(Box::new(fold_owned_enum_decl(folder, en))),
287        StmtKind::Namespace(ns) => StmtKind::Namespace(Box::new(NamespaceDecl {
288            name: ns.name.as_ref().map(|n| folder.fold_name(n)),
289            body: match &ns.body {
290                NamespaceBody::Braced(block) => {
291                    NamespaceBody::Braced(Box::new(folder.fold_block(block)))
292                }
293                NamespaceBody::Simple => NamespaceBody::Simple,
294            },
295        })),
296        StmtKind::Use(u) => StmtKind::Use(Box::new(UseDecl {
297            kind: u.kind,
298            uses: u
299                .uses
300                .iter()
301                .map(|item| UseItem {
302                    name: folder.fold_name(&item.name),
303                    alias: item.alias.clone(),
304                    kind: item.kind,
305                    span: item.span,
306                })
307                .collect(),
308        })),
309        StmtKind::Const(items) => StmtKind::Const(
310            items
311                .iter()
312                .map(|item| ConstItem {
313                    name: item.name.clone(),
314                    value: folder.fold_expr(&item.value),
315                    attributes: fold_owned_attrs(folder, &item.attributes),
316                    span: item.span,
317                    doc_comment: item.doc_comment.clone(),
318                })
319                .collect(),
320        ),
321        StmtKind::StaticVar(vars) => StmtKind::StaticVar(
322            vars.iter()
323                .map(|v| StaticVar {
324                    name: v.name.clone(),
325                    default: v.default.as_ref().map(|e| folder.fold_expr(e)),
326                    span: v.span,
327                })
328                .collect(),
329        ),
330        StmtKind::HaltCompiler(s) => StmtKind::HaltCompiler(s.clone()),
331        StmtKind::Nop => StmtKind::Nop,
332        StmtKind::InlineHtml(s) => StmtKind::InlineHtml(s.clone()),
333        StmtKind::Error => StmtKind::Error,
334    }
335}
336
337pub fn fold_owned_expr<F: FoldOwned + ?Sized>(folder: &mut F, expr: &Expr) -> Expr {
338    Expr {
339        kind: fold_owned_expr_kind(folder, &expr.kind),
340        span: expr.span,
341    }
342}
343
344fn fold_owned_expr_kind<F: FoldOwned + ?Sized>(folder: &mut F, k: &ExprKind) -> ExprKind {
345    match k {
346        ExprKind::Int(v) => ExprKind::Int(*v),
347        ExprKind::Float(v) => ExprKind::Float(*v),
348        ExprKind::String(s) => ExprKind::String(s.clone()),
349        ExprKind::InterpolatedString(parts) => {
350            ExprKind::InterpolatedString(fold_owned_string_parts(folder, parts))
351        }
352        ExprKind::Heredoc { label, parts } => ExprKind::Heredoc {
353            label: label.clone(),
354            parts: fold_owned_string_parts(folder, parts),
355        },
356        ExprKind::Nowdoc { label, value } => ExprKind::Nowdoc {
357            label: label.clone(),
358            value: value.clone(),
359        },
360        ExprKind::ShellExec(parts) => ExprKind::ShellExec(fold_owned_string_parts(folder, parts)),
361        ExprKind::Bool(v) => ExprKind::Bool(*v),
362        ExprKind::Null => ExprKind::Null,
363        ExprKind::Variable(s) => ExprKind::Variable(s.clone()),
364        ExprKind::VariableVariable(inner) => {
365            ExprKind::VariableVariable(Box::new(folder.fold_expr(inner)))
366        }
367        ExprKind::Identifier(s) => ExprKind::Identifier(s.clone()),
368        ExprKind::Assign(a) => ExprKind::Assign(AssignExpr {
369            target: Box::new(folder.fold_expr(&a.target)),
370            op: a.op,
371            value: Box::new(folder.fold_expr(&a.value)),
372            by_ref: a.by_ref,
373        }),
374        ExprKind::Binary(b) => ExprKind::Binary(BinaryExpr {
375            left: Box::new(folder.fold_expr(&b.left)),
376            op: b.op,
377            right: Box::new(folder.fold_expr(&b.right)),
378        }),
379        ExprKind::UnaryPrefix(u) => ExprKind::UnaryPrefix(UnaryPrefixExpr {
380            op: u.op,
381            operand: Box::new(folder.fold_expr(&u.operand)),
382        }),
383        ExprKind::UnaryPostfix(u) => ExprKind::UnaryPostfix(UnaryPostfixExpr {
384            operand: Box::new(folder.fold_expr(&u.operand)),
385            op: u.op,
386        }),
387        ExprKind::Ternary(t) => ExprKind::Ternary(TernaryExpr {
388            condition: Box::new(folder.fold_expr(&t.condition)),
389            then_expr: t.then_expr.as_ref().map(|e| Box::new(folder.fold_expr(e))),
390            else_expr: Box::new(folder.fold_expr(&t.else_expr)),
391        }),
392        ExprKind::NullCoalesce(n) => ExprKind::NullCoalesce(NullCoalesceExpr {
393            left: Box::new(folder.fold_expr(&n.left)),
394            right: Box::new(folder.fold_expr(&n.right)),
395        }),
396        ExprKind::FunctionCall(f) => ExprKind::FunctionCall(FunctionCallExpr {
397            name: Box::new(folder.fold_expr(&f.name)),
398            args: fold_owned_args(folder, &f.args),
399        }),
400        ExprKind::Array(elems) => ExprKind::Array(
401            elems
402                .iter()
403                .map(|e| ArrayElement {
404                    key: e.key.as_ref().map(|k| folder.fold_expr(k)),
405                    value: folder.fold_expr(&e.value),
406                    unpack: e.unpack,
407                    by_ref: e.by_ref,
408                    span: e.span,
409                })
410                .collect(),
411        ),
412        ExprKind::ArrayAccess(a) => ExprKind::ArrayAccess(ArrayAccessExpr {
413            array: Box::new(folder.fold_expr(&a.array)),
414            index: a.index.as_ref().map(|e| Box::new(folder.fold_expr(e))),
415        }),
416        ExprKind::Print(e) => ExprKind::Print(Box::new(folder.fold_expr(e))),
417        ExprKind::Parenthesized(e) => ExprKind::Parenthesized(Box::new(folder.fold_expr(e))),
418        ExprKind::Cast(kind, e) => ExprKind::Cast(*kind, Box::new(folder.fold_expr(e))),
419        ExprKind::ErrorSuppress(e) => ExprKind::ErrorSuppress(Box::new(folder.fold_expr(e))),
420        ExprKind::Isset(exprs) => ExprKind::Isset(fold_owned_exprs(folder, exprs)),
421        ExprKind::Empty(e) => ExprKind::Empty(Box::new(folder.fold_expr(e))),
422        ExprKind::Include(kind, e) => ExprKind::Include(*kind, Box::new(folder.fold_expr(e))),
423        ExprKind::Eval(e) => ExprKind::Eval(Box::new(folder.fold_expr(e))),
424        ExprKind::Exit(e) => ExprKind::Exit(e.as_ref().map(|e| Box::new(folder.fold_expr(e)))),
425        ExprKind::MagicConst(m) => ExprKind::MagicConst(*m),
426        ExprKind::Clone(e) => ExprKind::Clone(Box::new(folder.fold_expr(e))),
427        ExprKind::CloneWith(obj, props) => ExprKind::CloneWith(
428            Box::new(folder.fold_expr(obj)),
429            Box::new(folder.fold_expr(props)),
430        ),
431        ExprKind::New(n) => ExprKind::New(NewExpr {
432            class: Box::new(folder.fold_expr(&n.class)),
433            args: fold_owned_args(folder, &n.args),
434        }),
435        ExprKind::PropertyAccess(p) => ExprKind::PropertyAccess(PropertyAccessExpr {
436            object: Box::new(folder.fold_expr(&p.object)),
437            property: Box::new(folder.fold_expr(&p.property)),
438        }),
439        ExprKind::NullsafePropertyAccess(p) => {
440            ExprKind::NullsafePropertyAccess(PropertyAccessExpr {
441                object: Box::new(folder.fold_expr(&p.object)),
442                property: Box::new(folder.fold_expr(&p.property)),
443            })
444        }
445        ExprKind::MethodCall(m) => ExprKind::MethodCall(Box::new(MethodCallExpr {
446            object: Box::new(folder.fold_expr(&m.object)),
447            method: Box::new(folder.fold_expr(&m.method)),
448            args: fold_owned_args(folder, &m.args),
449        })),
450        ExprKind::NullsafeMethodCall(m) => ExprKind::NullsafeMethodCall(Box::new(MethodCallExpr {
451            object: Box::new(folder.fold_expr(&m.object)),
452            method: Box::new(folder.fold_expr(&m.method)),
453            args: fold_owned_args(folder, &m.args),
454        })),
455        ExprKind::StaticPropertyAccess(s) => ExprKind::StaticPropertyAccess(StaticAccessExpr {
456            class: Box::new(folder.fold_expr(&s.class)),
457            member: Box::new(folder.fold_expr(&s.member)),
458        }),
459        ExprKind::StaticMethodCall(s) => {
460            ExprKind::StaticMethodCall(Box::new(StaticMethodCallExpr {
461                class: Box::new(folder.fold_expr(&s.class)),
462                method: Box::new(folder.fold_expr(&s.method)),
463                args: fold_owned_args(folder, &s.args),
464            }))
465        }
466        ExprKind::StaticDynMethodCall(s) => {
467            ExprKind::StaticDynMethodCall(Box::new(StaticDynMethodCallExpr {
468                class: Box::new(folder.fold_expr(&s.class)),
469                method: Box::new(folder.fold_expr(&s.method)),
470                args: fold_owned_args(folder, &s.args),
471            }))
472        }
473        ExprKind::ClassConstAccess(s) => ExprKind::ClassConstAccess(StaticAccessExpr {
474            class: Box::new(folder.fold_expr(&s.class)),
475            member: Box::new(folder.fold_expr(&s.member)),
476        }),
477        ExprKind::ClassConstAccessDynamic { class, member } => ExprKind::ClassConstAccessDynamic {
478            class: Box::new(folder.fold_expr(class)),
479            member: Box::new(folder.fold_expr(member)),
480        },
481        ExprKind::StaticPropertyAccessDynamic { class, member } => {
482            ExprKind::StaticPropertyAccessDynamic {
483                class: Box::new(folder.fold_expr(class)),
484                member: Box::new(folder.fold_expr(member)),
485            }
486        }
487        ExprKind::Closure(c) => ExprKind::Closure(Box::new(ClosureExpr {
488            is_static: c.is_static,
489            by_ref: c.by_ref,
490            params: fold_owned_params(folder, &c.params),
491            use_vars: c
492                .use_vars
493                .iter()
494                .map(|v| folder.fold_closure_use_var(v))
495                .collect(),
496            return_type: c.return_type.as_ref().map(|t| folder.fold_type_hint(t)),
497            body: Box::new(folder.fold_block(&c.body)),
498            attributes: fold_owned_attrs(folder, &c.attributes),
499        })),
500        ExprKind::ArrowFunction(f) => ExprKind::ArrowFunction(Box::new(ArrowFunctionExpr {
501            is_static: f.is_static,
502            by_ref: f.by_ref,
503            params: fold_owned_params(folder, &f.params),
504            return_type: f.return_type.as_ref().map(|t| folder.fold_type_hint(t)),
505            body: Box::new(folder.fold_expr(&f.body)),
506            attributes: fold_owned_attrs(folder, &f.attributes),
507        })),
508        ExprKind::Match(m) => ExprKind::Match(MatchExpr {
509            subject: Box::new(folder.fold_expr(&m.subject)),
510            arms: m
511                .arms
512                .iter()
513                .map(|arm| folder.fold_match_arm(arm))
514                .collect(),
515            brace_start: m.brace_start,
516        }),
517        ExprKind::ThrowExpr(e) => ExprKind::ThrowExpr(Box::new(folder.fold_expr(e))),
518        ExprKind::Yield(y) => ExprKind::Yield(YieldExpr {
519            key: y.key.as_ref().map(|e| Box::new(folder.fold_expr(e))),
520            value: y.value.as_ref().map(|e| Box::new(folder.fold_expr(e))),
521            is_from: y.is_from,
522        }),
523        ExprKind::AnonymousClass(cls) => {
524            ExprKind::AnonymousClass(Box::new(fold_owned_class_decl(folder, cls)))
525        }
526        ExprKind::CallableCreate(c) => ExprKind::CallableCreate(CallableCreateExpr {
527            kind: match &c.kind {
528                CallableCreateKind::Function(e) => {
529                    CallableCreateKind::Function(Box::new(folder.fold_expr(e)))
530                }
531                CallableCreateKind::Method { object, method } => CallableCreateKind::Method {
532                    object: Box::new(folder.fold_expr(object)),
533                    method: Box::new(folder.fold_expr(method)),
534                },
535                CallableCreateKind::NullsafeMethod { object, method } => {
536                    CallableCreateKind::NullsafeMethod {
537                        object: Box::new(folder.fold_expr(object)),
538                        method: Box::new(folder.fold_expr(method)),
539                    }
540                }
541                CallableCreateKind::StaticMethod { class, method } => {
542                    CallableCreateKind::StaticMethod {
543                        class: Box::new(folder.fold_expr(class)),
544                        method: Box::new(folder.fold_expr(method)),
545                    }
546                }
547            },
548        }),
549        ExprKind::Omit => ExprKind::Omit,
550        ExprKind::Error => ExprKind::Error,
551    }
552}
553
554pub fn fold_owned_param<F: FoldOwned + ?Sized>(folder: &mut F, p: &Param) -> Param {
555    Param {
556        name: p.name.clone(),
557        type_hint: p.type_hint.as_ref().map(|t| folder.fold_type_hint(t)),
558        default: p.default.as_ref().map(|e| folder.fold_expr(e)),
559        by_ref: p.by_ref,
560        variadic: p.variadic,
561        is_readonly: p.is_readonly,
562        is_final: p.is_final,
563        visibility: p.visibility,
564        set_visibility: p.set_visibility,
565        attributes: fold_owned_attrs(folder, &p.attributes),
566        hooks: fold_owned_hooks(folder, &p.hooks),
567        span: p.span,
568    }
569}
570
571pub fn fold_owned_arg<F: FoldOwned + ?Sized>(folder: &mut F, arg: &Arg) -> Arg {
572    Arg {
573        name: arg.name.as_ref().map(|n| folder.fold_name(n)),
574        value: folder.fold_expr(&arg.value),
575        unpack: arg.unpack,
576        by_ref: arg.by_ref,
577        span: arg.span,
578    }
579}
580
581pub fn fold_owned_closure_use_var<F: FoldOwned + ?Sized>(
582    _folder: &mut F,
583    var: &ClosureUseVar,
584) -> ClosureUseVar {
585    var.clone()
586}
587
588pub fn fold_owned_name<F: FoldOwned + ?Sized>(_folder: &mut F, name: &Name) -> Name {
589    name.clone()
590}
591
592pub fn fold_owned_class_member<F: FoldOwned + ?Sized>(
593    folder: &mut F,
594    member: &ClassMember,
595) -> ClassMember {
596    ClassMember {
597        kind: match &member.kind {
598            ClassMemberKind::Property(p) => ClassMemberKind::Property(PropertyDecl {
599                name: p.name.clone(),
600                visibility: p.visibility,
601                set_visibility: p.set_visibility,
602                is_static: p.is_static,
603                is_readonly: p.is_readonly,
604                type_hint: p.type_hint.as_ref().map(|t| folder.fold_type_hint(t)),
605                default: p.default.as_ref().map(|e| folder.fold_expr(e)),
606                attributes: fold_owned_attrs(folder, &p.attributes),
607                hooks: fold_owned_hooks(folder, &p.hooks),
608                doc_comment: p.doc_comment.clone(),
609            }),
610            ClassMemberKind::Method(m) => ClassMemberKind::Method(MethodDecl {
611                name: m.name.clone(),
612                visibility: m.visibility,
613                is_static: m.is_static,
614                is_abstract: m.is_abstract,
615                is_final: m.is_final,
616                by_ref: m.by_ref,
617                params: fold_owned_params(folder, &m.params),
618                return_type: m.return_type.as_ref().map(|t| folder.fold_type_hint(t)),
619                body: m.body.as_ref().map(|b| Box::new(folder.fold_block(b))),
620                attributes: fold_owned_attrs(folder, &m.attributes),
621                doc_comment: m.doc_comment.clone(),
622            }),
623            ClassMemberKind::ClassConst(c) => {
624                ClassMemberKind::ClassConst(fold_owned_class_const(folder, c))
625            }
626            ClassMemberKind::TraitUse(t) => {
627                ClassMemberKind::TraitUse(fold_owned_trait_use(folder, t))
628            }
629        },
630        span: member.span,
631    }
632}
633
634pub fn fold_owned_enum_member<F: FoldOwned + ?Sized>(
635    folder: &mut F,
636    member: &EnumMember,
637) -> EnumMember {
638    EnumMember {
639        kind: match &member.kind {
640            EnumMemberKind::Case(c) => EnumMemberKind::Case(EnumCase {
641                name: c.name.clone(),
642                value: c.value.as_ref().map(|e| folder.fold_expr(e)),
643                attributes: fold_owned_attrs(folder, &c.attributes),
644                doc_comment: c.doc_comment.clone(),
645            }),
646            EnumMemberKind::Method(m) => EnumMemberKind::Method(MethodDecl {
647                name: m.name.clone(),
648                visibility: m.visibility,
649                is_static: m.is_static,
650                is_abstract: m.is_abstract,
651                is_final: m.is_final,
652                by_ref: m.by_ref,
653                params: fold_owned_params(folder, &m.params),
654                return_type: m.return_type.as_ref().map(|t| folder.fold_type_hint(t)),
655                body: m.body.as_ref().map(|b| Box::new(folder.fold_block(b))),
656                attributes: fold_owned_attrs(folder, &m.attributes),
657                doc_comment: m.doc_comment.clone(),
658            }),
659            EnumMemberKind::ClassConst(c) => {
660                EnumMemberKind::ClassConst(fold_owned_class_const(folder, c))
661            }
662            EnumMemberKind::TraitUse(t) => {
663                EnumMemberKind::TraitUse(fold_owned_trait_use(folder, t))
664            }
665        },
666        span: member.span,
667    }
668}
669
670pub fn fold_owned_property_hook<F: FoldOwned + ?Sized>(
671    folder: &mut F,
672    hook: &PropertyHook,
673) -> PropertyHook {
674    PropertyHook {
675        kind: hook.kind,
676        body: match &hook.body {
677            PropertyHookBody::Block(block) => {
678                PropertyHookBody::Block(Box::new(folder.fold_block(block)))
679            }
680            PropertyHookBody::Expression(e) => PropertyHookBody::Expression(folder.fold_expr(e)),
681            PropertyHookBody::Abstract => PropertyHookBody::Abstract,
682        },
683        is_final: hook.is_final,
684        by_ref: hook.by_ref,
685        params: fold_owned_params(folder, &hook.params),
686        attributes: fold_owned_attrs(folder, &hook.attributes),
687        span: hook.span,
688    }
689}
690
691pub fn fold_owned_type_hint<F: FoldOwned + ?Sized>(
692    folder: &mut F,
693    type_hint: &TypeHint,
694) -> TypeHint {
695    TypeHint {
696        kind: match &type_hint.kind {
697            TypeHintKind::Named(n) => TypeHintKind::Named(folder.fold_name(n)),
698            TypeHintKind::Keyword(b, span) => TypeHintKind::Keyword(*b, *span),
699            TypeHintKind::Nullable(inner) => {
700                TypeHintKind::Nullable(Box::new(folder.fold_type_hint(inner)))
701            }
702            TypeHintKind::Union(types) => {
703                TypeHintKind::Union(types.iter().map(|t| folder.fold_type_hint(t)).collect())
704            }
705            TypeHintKind::Intersection(types) => {
706                TypeHintKind::Intersection(types.iter().map(|t| folder.fold_type_hint(t)).collect())
707            }
708        },
709        span: type_hint.span,
710    }
711}
712
713pub fn fold_owned_attribute<F: FoldOwned + ?Sized>(
714    folder: &mut F,
715    attribute: &Attribute,
716) -> Attribute {
717    Attribute {
718        name: folder.fold_name(&attribute.name),
719        args: fold_owned_args(folder, &attribute.args),
720        span: attribute.span,
721    }
722}
723
724pub fn fold_owned_catch_clause<F: FoldOwned + ?Sized>(
725    folder: &mut F,
726    catch: &CatchClause,
727) -> CatchClause {
728    CatchClause {
729        types: catch.types.iter().map(|n| folder.fold_name(n)).collect(),
730        var: catch.var.clone(),
731        body: Box::new(folder.fold_block(&catch.body)),
732        span: catch.span,
733    }
734}
735
736pub fn fold_owned_match_arm<F: FoldOwned + ?Sized>(folder: &mut F, arm: &MatchArm) -> MatchArm {
737    MatchArm {
738        conditions: arm
739            .conditions
740            .as_ref()
741            .map(|conds| fold_owned_exprs(folder, conds)),
742        body: folder.fold_expr(&arm.body),
743        span: arm.span,
744    }
745}
746
747fn fold_owned_class_const<F: FoldOwned + ?Sized>(
748    folder: &mut F,
749    c: &ClassConstDecl,
750) -> ClassConstDecl {
751    ClassConstDecl {
752        name: c.name.clone(),
753        visibility: c.visibility,
754        is_final: c.is_final,
755        type_hint: c
756            .type_hint
757            .as_ref()
758            .map(|th| Box::new(folder.fold_type_hint(th))),
759        value: folder.fold_expr(&c.value),
760        attributes: fold_owned_attrs(folder, &c.attributes),
761        doc_comment: c.doc_comment.clone(),
762    }
763}
764
765fn fold_owned_trait_use<F: FoldOwned + ?Sized>(folder: &mut F, t: &TraitUseDecl) -> TraitUseDecl {
766    TraitUseDecl {
767        traits: t.traits.iter().map(|n| folder.fold_name(n)).collect(),
768        adaptations: t
769            .adaptations
770            .iter()
771            .map(|a| TraitAdaptation {
772                kind: match &a.kind {
773                    TraitAdaptationKind::Precedence {
774                        trait_name,
775                        method,
776                        insteadof,
777                    } => TraitAdaptationKind::Precedence {
778                        trait_name: folder.fold_name(trait_name),
779                        method: folder.fold_name(method),
780                        insteadof: insteadof.iter().map(|n| folder.fold_name(n)).collect(),
781                    },
782                    TraitAdaptationKind::Alias {
783                        trait_name,
784                        method,
785                        new_modifier,
786                        new_name,
787                    } => TraitAdaptationKind::Alias {
788                        trait_name: trait_name.as_ref().map(|n| folder.fold_name(n)),
789                        method: folder.fold_name(method),
790                        new_modifier: *new_modifier,
791                        new_name: new_name.as_ref().map(|n| folder.fold_name(n)),
792                    },
793                },
794                span: a.span,
795            })
796            .collect(),
797        adaptations_brace_start: t.adaptations_brace_start,
798    }
799}
800
801fn fold_owned_class_decl<F: FoldOwned + ?Sized>(folder: &mut F, cls: &ClassDecl) -> ClassDecl {
802    ClassDecl {
803        name: cls.name.clone(),
804        modifiers: cls.modifiers.clone(),
805        extends: cls.extends.as_ref().map(|n| folder.fold_name(n)),
806        implements: cls.implements.iter().map(|n| folder.fold_name(n)).collect(),
807        body: ClassBody {
808            members: fold_owned_members(folder, &cls.body.members),
809            span: cls.body.span,
810        },
811        attributes: fold_owned_attrs(folder, &cls.attributes),
812        doc_comment: cls.doc_comment.clone(),
813    }
814}
815
816fn fold_owned_interface_decl<F: FoldOwned + ?Sized>(
817    folder: &mut F,
818    iface: &InterfaceDecl,
819) -> InterfaceDecl {
820    InterfaceDecl {
821        name: iface.name.clone(),
822        extends: iface.extends.iter().map(|n| folder.fold_name(n)).collect(),
823        body: ClassBody {
824            members: fold_owned_members(folder, &iface.body.members),
825            span: iface.body.span,
826        },
827        attributes: fold_owned_attrs(folder, &iface.attributes),
828        doc_comment: iface.doc_comment.clone(),
829    }
830}
831
832fn fold_owned_trait_decl<F: FoldOwned + ?Sized>(folder: &mut F, tr: &TraitDecl) -> TraitDecl {
833    TraitDecl {
834        name: tr.name.clone(),
835        body: ClassBody {
836            members: fold_owned_members(folder, &tr.body.members),
837            span: tr.body.span,
838        },
839        attributes: fold_owned_attrs(folder, &tr.attributes),
840        doc_comment: tr.doc_comment.clone(),
841    }
842}
843
844fn fold_owned_enum_decl<F: FoldOwned + ?Sized>(folder: &mut F, en: &EnumDecl) -> EnumDecl {
845    EnumDecl {
846        name: en.name.clone(),
847        scalar_type: en.scalar_type.as_ref().map(|n| folder.fold_name(n)),
848        implements: en.implements.iter().map(|n| folder.fold_name(n)).collect(),
849        body: EnumBody {
850            members: en
851                .body
852                .members
853                .iter()
854                .map(|m| folder.fold_enum_member(m))
855                .collect(),
856            span: en.body.span,
857        },
858        attributes: fold_owned_attrs(folder, &en.attributes),
859        doc_comment: en.doc_comment.clone(),
860    }
861}
862
863// =============================================================================
864// Tests
865// =============================================================================
866
867#[cfg(test)]
868mod tests {
869    use super::*;
870    use crate::ast::AssignOp;
871    use crate::Span;
872
873    fn dummy_var(name: &str) -> Expr {
874        Expr {
875            kind: ExprKind::Variable(Box::from(name)),
876            span: Span::DUMMY,
877        }
878    }
879
880    fn dummy_int(n: i64) -> Expr {
881        Expr {
882            kind: ExprKind::Int(n),
883            span: Span::DUMMY,
884        }
885    }
886
887    fn assign(target: Expr, value: Expr) -> Expr {
888        Expr {
889            kind: ExprKind::Assign(AssignExpr {
890                target: Box::new(target),
891                op: AssignOp::Assign,
892                value: Box::new(value),
893                by_ref: false,
894            }),
895            span: Span::DUMMY,
896        }
897    }
898
899    fn expr_stmt(e: Expr) -> Stmt {
900        Stmt {
901            kind: StmtKind::Expression(Box::new(e)),
902            span: Span::DUMMY,
903            doc_comment: None,
904        }
905    }
906
907    fn empty_block() -> Block {
908        Block {
909            stmts: Box::from([]),
910            span: Span::DUMMY,
911        }
912    }
913
914    fn program(stmts: impl IntoIterator<Item = Stmt>) -> Program {
915        Program {
916            stmts: stmts.into_iter().collect(),
917            span: Span::DUMMY,
918        }
919    }
920
921    /// Identity fold on `$x = $y;` must produce structurally equal output.
922    #[test]
923    fn identity_fold_roundtrip() {
924        let p = program([expr_stmt(assign(dummy_var("x"), dummy_var("y")))]);
925        struct Identity;
926        impl FoldOwned for Identity {}
927        let folded = Identity.fold_program(&p);
928        assert_eq!(
929            serde_json::to_string(&folded).unwrap(),
930            serde_json::to_string(&p).unwrap(),
931        );
932    }
933
934    /// NegateInts fold on `$x = 42;` must produce `$x = -42;`.
935    #[test]
936    fn negate_ints() {
937        let p = program([expr_stmt(assign(dummy_var("x"), dummy_int(42)))]);
938        struct NegateInts;
939        impl FoldOwned for NegateInts {
940            fn fold_expr(&mut self, expr: &Expr) -> Expr {
941                if let ExprKind::Int(n) = &expr.kind {
942                    return Expr {
943                        kind: ExprKind::Int(-n),
944                        span: expr.span,
945                    };
946                }
947                fold_owned_expr(self, expr)
948            }
949        }
950        let folded = NegateInts.fold_program(&p);
951        let stmt = &folded.stmts[0];
952        if let StmtKind::Expression(expr) = &stmt.kind {
953            if let ExprKind::Assign(a) = &expr.kind {
954                if let ExprKind::Int(n) = &a.value.kind {
955                    assert_eq!(*n, -42);
956                    return;
957                }
958            }
959        }
960        panic!("expected negated int assignment");
961    }
962
963    /// Fold that rewrites variable names: all occurrences of `$x` become `$renamed`.
964    #[test]
965    fn rename_variable() {
966        // $x = $x;
967        let p = program([expr_stmt(assign(dummy_var("x"), dummy_var("x")))]);
968        struct Rename;
969        impl FoldOwned for Rename {
970            fn fold_expr(&mut self, expr: &Expr) -> Expr {
971                if let ExprKind::Variable(name) = &expr.kind {
972                    if name.as_ref() == "x" {
973                        return Expr {
974                            kind: ExprKind::Variable(Box::from("renamed")),
975                            span: expr.span,
976                        };
977                    }
978                }
979                fold_owned_expr(self, expr)
980            }
981        }
982        let folded = Rename.fold_program(&p);
983        let stmt = &folded.stmts[0];
984        if let StmtKind::Expression(expr) = &stmt.kind {
985            if let ExprKind::Assign(a) = &expr.kind {
986                assert!(matches!(&a.target.kind, ExprKind::Variable(n) if n.as_ref() == "renamed"));
987                assert!(matches!(&a.value.kind, ExprKind::Variable(n) if n.as_ref() == "renamed"));
988                return;
989            }
990        }
991        panic!("expected renamed variable assignment");
992    }
993
994    /// Rename fold must reach into closure `use` capture lists.
995    #[test]
996    fn rename_variable_in_closure_use() {
997        // function() use ($x) {}
998        let closure = Expr {
999            kind: ExprKind::Closure(Box::new(ClosureExpr {
1000                is_static: false,
1001                by_ref: false,
1002                params: Box::from([]),
1003                use_vars: Box::from([ClosureUseVar {
1004                    name: Box::from("x"),
1005                    by_ref: false,
1006                    span: Span::DUMMY,
1007                }]),
1008                return_type: None,
1009                body: Box::new(empty_block()),
1010                attributes: Box::from([]),
1011            })),
1012            span: Span::DUMMY,
1013        };
1014        let p = program([expr_stmt(closure)]);
1015
1016        struct Rename;
1017        impl FoldOwned for Rename {
1018            fn fold_closure_use_var(&mut self, var: &ClosureUseVar) -> ClosureUseVar {
1019                ClosureUseVar {
1020                    name: if var.name.as_ref() == "x" {
1021                        Box::from("renamed")
1022                    } else {
1023                        var.name.clone()
1024                    },
1025                    by_ref: var.by_ref,
1026                    span: var.span,
1027                }
1028            }
1029        }
1030
1031        let folded = Rename.fold_program(&p);
1032        if let StmtKind::Expression(expr) = &folded.stmts[0].kind {
1033            if let ExprKind::Closure(c) = &expr.kind {
1034                assert_eq!(c.use_vars[0].name.as_ref(), "renamed");
1035                return;
1036            }
1037        }
1038        panic!("expected closure with renamed use-var");
1039    }
1040}