1use super::*;
28
29pub 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
100pub 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#[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 #[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 #[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 #[test]
965 fn rename_variable() {
966 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 #[test]
996 fn rename_variable_in_closure_use() {
997 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}