Chris@13
|
1 <?php declare(strict_types=1);
|
Chris@0
|
2
|
Chris@0
|
3 namespace PhpParser;
|
Chris@0
|
4
|
Chris@0
|
5 use PhpParser\Node\Expr;
|
Chris@0
|
6 use PhpParser\Node\Scalar\String_;
|
Chris@13
|
7 use PhpParser\NodeVisitor;
|
Chris@13
|
8 use PHPUnit\Framework\TestCase;
|
Chris@0
|
9
|
Chris@13
|
10 class NodeTraverserTest extends TestCase
|
Chris@0
|
11 {
|
Chris@0
|
12 public function testNonModifying() {
|
Chris@0
|
13 $str1Node = new String_('Foo');
|
Chris@0
|
14 $str2Node = new String_('Bar');
|
Chris@13
|
15 $echoNode = new Node\Stmt\Echo_([$str1Node, $str2Node]);
|
Chris@13
|
16 $stmts = [$echoNode];
|
Chris@0
|
17
|
Chris@13
|
18 $visitor = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@0
|
19
|
Chris@0
|
20 $visitor->expects($this->at(0))->method('beforeTraverse')->with($stmts);
|
Chris@0
|
21 $visitor->expects($this->at(1))->method('enterNode')->with($echoNode);
|
Chris@0
|
22 $visitor->expects($this->at(2))->method('enterNode')->with($str1Node);
|
Chris@0
|
23 $visitor->expects($this->at(3))->method('leaveNode')->with($str1Node);
|
Chris@0
|
24 $visitor->expects($this->at(4))->method('enterNode')->with($str2Node);
|
Chris@0
|
25 $visitor->expects($this->at(5))->method('leaveNode')->with($str2Node);
|
Chris@0
|
26 $visitor->expects($this->at(6))->method('leaveNode')->with($echoNode);
|
Chris@0
|
27 $visitor->expects($this->at(7))->method('afterTraverse')->with($stmts);
|
Chris@0
|
28
|
Chris@0
|
29 $traverser = new NodeTraverser;
|
Chris@0
|
30 $traverser->addVisitor($visitor);
|
Chris@0
|
31
|
Chris@0
|
32 $this->assertEquals($stmts, $traverser->traverse($stmts));
|
Chris@0
|
33 }
|
Chris@0
|
34
|
Chris@0
|
35 public function testModifying() {
|
Chris@0
|
36 $str1Node = new String_('Foo');
|
Chris@0
|
37 $str2Node = new String_('Bar');
|
Chris@0
|
38 $printNode = new Expr\Print_($str1Node);
|
Chris@0
|
39
|
Chris@0
|
40 // first visitor changes the node, second verifies the change
|
Chris@13
|
41 $visitor1 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
42 $visitor2 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@0
|
43
|
Chris@0
|
44 // replace empty statements with string1 node
|
Chris@13
|
45 $visitor1->expects($this->at(0))->method('beforeTraverse')->with([])
|
Chris@13
|
46 ->will($this->returnValue([$str1Node]));
|
Chris@13
|
47 $visitor2->expects($this->at(0))->method('beforeTraverse')->with([$str1Node]);
|
Chris@0
|
48
|
Chris@0
|
49 // replace string1 node with print node
|
Chris@0
|
50 $visitor1->expects($this->at(1))->method('enterNode')->with($str1Node)
|
Chris@0
|
51 ->will($this->returnValue($printNode));
|
Chris@0
|
52 $visitor2->expects($this->at(1))->method('enterNode')->with($printNode);
|
Chris@0
|
53
|
Chris@0
|
54 // replace string1 node with string2 node
|
Chris@0
|
55 $visitor1->expects($this->at(2))->method('enterNode')->with($str1Node)
|
Chris@0
|
56 ->will($this->returnValue($str2Node));
|
Chris@0
|
57 $visitor2->expects($this->at(2))->method('enterNode')->with($str2Node);
|
Chris@0
|
58
|
Chris@0
|
59 // replace string2 node with string1 node again
|
Chris@0
|
60 $visitor1->expects($this->at(3))->method('leaveNode')->with($str2Node)
|
Chris@0
|
61 ->will($this->returnValue($str1Node));
|
Chris@0
|
62 $visitor2->expects($this->at(3))->method('leaveNode')->with($str1Node);
|
Chris@0
|
63
|
Chris@0
|
64 // replace print node with string1 node again
|
Chris@0
|
65 $visitor1->expects($this->at(4))->method('leaveNode')->with($printNode)
|
Chris@0
|
66 ->will($this->returnValue($str1Node));
|
Chris@0
|
67 $visitor2->expects($this->at(4))->method('leaveNode')->with($str1Node);
|
Chris@0
|
68
|
Chris@0
|
69 // replace string1 node with empty statements again
|
Chris@13
|
70 $visitor1->expects($this->at(5))->method('afterTraverse')->with([$str1Node])
|
Chris@13
|
71 ->will($this->returnValue([]));
|
Chris@13
|
72 $visitor2->expects($this->at(5))->method('afterTraverse')->with([]);
|
Chris@0
|
73
|
Chris@0
|
74 $traverser = new NodeTraverser;
|
Chris@0
|
75 $traverser->addVisitor($visitor1);
|
Chris@0
|
76 $traverser->addVisitor($visitor2);
|
Chris@0
|
77
|
Chris@0
|
78 // as all operations are reversed we end where we start
|
Chris@13
|
79 $this->assertEquals([], $traverser->traverse([]));
|
Chris@0
|
80 }
|
Chris@0
|
81
|
Chris@0
|
82 public function testRemove() {
|
Chris@0
|
83 $str1Node = new String_('Foo');
|
Chris@0
|
84 $str2Node = new String_('Bar');
|
Chris@0
|
85
|
Chris@13
|
86 $visitor = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@0
|
87
|
Chris@0
|
88 // remove the string1 node, leave the string2 node
|
Chris@0
|
89 $visitor->expects($this->at(2))->method('leaveNode')->with($str1Node)
|
Chris@13
|
90 ->will($this->returnValue(NodeTraverser::REMOVE_NODE));
|
Chris@0
|
91
|
Chris@0
|
92 $traverser = new NodeTraverser;
|
Chris@0
|
93 $traverser->addVisitor($visitor);
|
Chris@0
|
94
|
Chris@13
|
95 $this->assertEquals([$str2Node], $traverser->traverse([$str1Node, $str2Node]));
|
Chris@0
|
96 }
|
Chris@0
|
97
|
Chris@0
|
98 public function testMerge() {
|
Chris@0
|
99 $strStart = new String_('Start');
|
Chris@0
|
100 $strMiddle = new String_('End');
|
Chris@0
|
101 $strEnd = new String_('Middle');
|
Chris@0
|
102 $strR1 = new String_('Replacement 1');
|
Chris@0
|
103 $strR2 = new String_('Replacement 2');
|
Chris@0
|
104
|
Chris@13
|
105 $visitor = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@0
|
106
|
Chris@0
|
107 // replace strMiddle with strR1 and strR2 by merge
|
Chris@0
|
108 $visitor->expects($this->at(4))->method('leaveNode')->with($strMiddle)
|
Chris@13
|
109 ->will($this->returnValue([$strR1, $strR2]));
|
Chris@0
|
110
|
Chris@0
|
111 $traverser = new NodeTraverser;
|
Chris@0
|
112 $traverser->addVisitor($visitor);
|
Chris@0
|
113
|
Chris@0
|
114 $this->assertEquals(
|
Chris@13
|
115 [$strStart, $strR1, $strR2, $strEnd],
|
Chris@13
|
116 $traverser->traverse([$strStart, $strMiddle, $strEnd])
|
Chris@0
|
117 );
|
Chris@0
|
118 }
|
Chris@0
|
119
|
Chris@13
|
120 /**
|
Chris@13
|
121 * @expectedException \LogicException
|
Chris@13
|
122 * @expectedExceptionMessage Invalid node structure: Contains nested arrays
|
Chris@13
|
123 */
|
Chris@13
|
124 public function testInvalidDeepArray() {
|
Chris@0
|
125 $strNode = new String_('Foo');
|
Chris@13
|
126 $stmts = [[[$strNode]]];
|
Chris@0
|
127
|
Chris@0
|
128 $traverser = new NodeTraverser;
|
Chris@0
|
129 $this->assertEquals($stmts, $traverser->traverse($stmts));
|
Chris@0
|
130 }
|
Chris@0
|
131
|
Chris@0
|
132 public function testDontTraverseChildren() {
|
Chris@0
|
133 $strNode = new String_('str');
|
Chris@0
|
134 $printNode = new Expr\Print_($strNode);
|
Chris@0
|
135 $varNode = new Expr\Variable('foo');
|
Chris@0
|
136 $mulNode = new Expr\BinaryOp\Mul($varNode, $varNode);
|
Chris@0
|
137 $negNode = new Expr\UnaryMinus($mulNode);
|
Chris@13
|
138 $stmts = [$printNode, $negNode];
|
Chris@0
|
139
|
Chris@13
|
140 $visitor1 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
141 $visitor2 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@0
|
142
|
Chris@0
|
143 $visitor1->expects($this->at(1))->method('enterNode')->with($printNode)
|
Chris@0
|
144 ->will($this->returnValue(NodeTraverser::DONT_TRAVERSE_CHILDREN));
|
Chris@0
|
145 $visitor2->expects($this->at(1))->method('enterNode')->with($printNode);
|
Chris@0
|
146
|
Chris@0
|
147 $visitor1->expects($this->at(2))->method('leaveNode')->with($printNode);
|
Chris@0
|
148 $visitor2->expects($this->at(2))->method('leaveNode')->with($printNode);
|
Chris@0
|
149
|
Chris@0
|
150 $visitor1->expects($this->at(3))->method('enterNode')->with($negNode);
|
Chris@0
|
151 $visitor2->expects($this->at(3))->method('enterNode')->with($negNode);
|
Chris@0
|
152
|
Chris@0
|
153 $visitor1->expects($this->at(4))->method('enterNode')->with($mulNode);
|
Chris@0
|
154 $visitor2->expects($this->at(4))->method('enterNode')->with($mulNode)
|
Chris@0
|
155 ->will($this->returnValue(NodeTraverser::DONT_TRAVERSE_CHILDREN));
|
Chris@0
|
156
|
Chris@0
|
157 $visitor1->expects($this->at(5))->method('leaveNode')->with($mulNode);
|
Chris@0
|
158 $visitor2->expects($this->at(5))->method('leaveNode')->with($mulNode);
|
Chris@0
|
159
|
Chris@0
|
160 $visitor1->expects($this->at(6))->method('leaveNode')->with($negNode);
|
Chris@0
|
161 $visitor2->expects($this->at(6))->method('leaveNode')->with($negNode);
|
Chris@0
|
162
|
Chris@0
|
163 $traverser = new NodeTraverser;
|
Chris@0
|
164 $traverser->addVisitor($visitor1);
|
Chris@0
|
165 $traverser->addVisitor($visitor2);
|
Chris@0
|
166
|
Chris@0
|
167 $this->assertEquals($stmts, $traverser->traverse($stmts));
|
Chris@0
|
168 }
|
Chris@0
|
169
|
Chris@0
|
170 public function testStopTraversal() {
|
Chris@0
|
171 $varNode1 = new Expr\Variable('a');
|
Chris@0
|
172 $varNode2 = new Expr\Variable('b');
|
Chris@0
|
173 $varNode3 = new Expr\Variable('c');
|
Chris@0
|
174 $mulNode = new Expr\BinaryOp\Mul($varNode1, $varNode2);
|
Chris@0
|
175 $printNode = new Expr\Print_($varNode3);
|
Chris@0
|
176 $stmts = [$mulNode, $printNode];
|
Chris@0
|
177
|
Chris@0
|
178 // From enterNode() with array parent
|
Chris@13
|
179 $visitor = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@0
|
180 $visitor->expects($this->at(1))->method('enterNode')->with($mulNode)
|
Chris@0
|
181 ->will($this->returnValue(NodeTraverser::STOP_TRAVERSAL));
|
Chris@0
|
182 $visitor->expects($this->at(2))->method('afterTraverse');
|
Chris@0
|
183 $traverser = new NodeTraverser;
|
Chris@0
|
184 $traverser->addVisitor($visitor);
|
Chris@0
|
185 $this->assertEquals($stmts, $traverser->traverse($stmts));
|
Chris@0
|
186
|
Chris@0
|
187 // From enterNode with Node parent
|
Chris@13
|
188 $visitor = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@0
|
189 $visitor->expects($this->at(2))->method('enterNode')->with($varNode1)
|
Chris@0
|
190 ->will($this->returnValue(NodeTraverser::STOP_TRAVERSAL));
|
Chris@0
|
191 $visitor->expects($this->at(3))->method('afterTraverse');
|
Chris@0
|
192 $traverser = new NodeTraverser;
|
Chris@0
|
193 $traverser->addVisitor($visitor);
|
Chris@0
|
194 $this->assertEquals($stmts, $traverser->traverse($stmts));
|
Chris@0
|
195
|
Chris@0
|
196 // From leaveNode with Node parent
|
Chris@13
|
197 $visitor = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@0
|
198 $visitor->expects($this->at(3))->method('leaveNode')->with($varNode1)
|
Chris@0
|
199 ->will($this->returnValue(NodeTraverser::STOP_TRAVERSAL));
|
Chris@0
|
200 $visitor->expects($this->at(4))->method('afterTraverse');
|
Chris@0
|
201 $traverser = new NodeTraverser;
|
Chris@0
|
202 $traverser->addVisitor($visitor);
|
Chris@0
|
203 $this->assertEquals($stmts, $traverser->traverse($stmts));
|
Chris@0
|
204
|
Chris@0
|
205 // From leaveNode with array parent
|
Chris@13
|
206 $visitor = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@0
|
207 $visitor->expects($this->at(6))->method('leaveNode')->with($mulNode)
|
Chris@0
|
208 ->will($this->returnValue(NodeTraverser::STOP_TRAVERSAL));
|
Chris@0
|
209 $visitor->expects($this->at(7))->method('afterTraverse');
|
Chris@0
|
210 $traverser = new NodeTraverser;
|
Chris@0
|
211 $traverser->addVisitor($visitor);
|
Chris@0
|
212 $this->assertEquals($stmts, $traverser->traverse($stmts));
|
Chris@0
|
213
|
Chris@0
|
214 // Check that pending array modifications are still carried out
|
Chris@13
|
215 $visitor = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@0
|
216 $visitor->expects($this->at(6))->method('leaveNode')->with($mulNode)
|
Chris@0
|
217 ->will($this->returnValue(NodeTraverser::REMOVE_NODE));
|
Chris@0
|
218 $visitor->expects($this->at(7))->method('enterNode')->with($printNode)
|
Chris@0
|
219 ->will($this->returnValue(NodeTraverser::STOP_TRAVERSAL));
|
Chris@0
|
220 $visitor->expects($this->at(8))->method('afterTraverse');
|
Chris@0
|
221 $traverser = new NodeTraverser;
|
Chris@0
|
222 $traverser->addVisitor($visitor);
|
Chris@0
|
223 $this->assertEquals([$printNode], $traverser->traverse($stmts));
|
Chris@0
|
224
|
Chris@0
|
225 }
|
Chris@0
|
226
|
Chris@0
|
227 public function testRemovingVisitor() {
|
Chris@13
|
228 $visitor1 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
229 $visitor2 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
230 $visitor3 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@0
|
231
|
Chris@0
|
232 $traverser = new NodeTraverser;
|
Chris@0
|
233 $traverser->addVisitor($visitor1);
|
Chris@0
|
234 $traverser->addVisitor($visitor2);
|
Chris@0
|
235 $traverser->addVisitor($visitor3);
|
Chris@0
|
236
|
Chris@13
|
237 $preExpected = [$visitor1, $visitor2, $visitor3];
|
Chris@0
|
238 $this->assertAttributeSame($preExpected, 'visitors', $traverser, 'The appropriate visitors have not been added');
|
Chris@0
|
239
|
Chris@0
|
240 $traverser->removeVisitor($visitor2);
|
Chris@0
|
241
|
Chris@13
|
242 $postExpected = [0 => $visitor1, 2 => $visitor3];
|
Chris@0
|
243 $this->assertAttributeSame($postExpected, 'visitors', $traverser, 'The appropriate visitors are not present after removal');
|
Chris@0
|
244 }
|
Chris@0
|
245
|
Chris@0
|
246 public function testNoCloneNodes() {
|
Chris@13
|
247 $stmts = [new Node\Stmt\Echo_([new String_('Foo'), new String_('Bar')])];
|
Chris@0
|
248
|
Chris@0
|
249 $traverser = new NodeTraverser;
|
Chris@0
|
250
|
Chris@0
|
251 $this->assertSame($stmts, $traverser->traverse($stmts));
|
Chris@0
|
252 }
|
Chris@0
|
253
|
Chris@0
|
254 /**
|
Chris@13
|
255 * @dataProvider provideTestInvalidReturn
|
Chris@0
|
256 */
|
Chris@13
|
257 public function testInvalidReturn($visitor, $message) {
|
Chris@13
|
258 $this->expectException(\LogicException::class);
|
Chris@13
|
259 $this->expectExceptionMessage($message);
|
Chris@0
|
260
|
Chris@13
|
261 $stmts = [new Node\Stmt\Expression(new Node\Scalar\LNumber(42))];
|
Chris@0
|
262
|
Chris@0
|
263 $traverser = new NodeTraverser();
|
Chris@0
|
264 $traverser->addVisitor($visitor);
|
Chris@0
|
265 $traverser->traverse($stmts);
|
Chris@0
|
266 }
|
Chris@13
|
267
|
Chris@13
|
268 public function provideTestInvalidReturn() {
|
Chris@13
|
269 $visitor1 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
270 $visitor1->expects($this->at(1))->method('enterNode')
|
Chris@13
|
271 ->willReturn('foobar');
|
Chris@13
|
272
|
Chris@13
|
273 $visitor2 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
274 $visitor2->expects($this->at(2))->method('enterNode')
|
Chris@13
|
275 ->willReturn('foobar');
|
Chris@13
|
276
|
Chris@13
|
277 $visitor3 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
278 $visitor3->expects($this->at(3))->method('leaveNode')
|
Chris@13
|
279 ->willReturn('foobar');
|
Chris@13
|
280
|
Chris@13
|
281 $visitor4 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
282 $visitor4->expects($this->at(4))->method('leaveNode')
|
Chris@13
|
283 ->willReturn('foobar');
|
Chris@13
|
284
|
Chris@13
|
285 $visitor5 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
286 $visitor5->expects($this->at(3))->method('leaveNode')
|
Chris@13
|
287 ->willReturn([new Node\Scalar\DNumber(42.0)]);
|
Chris@13
|
288
|
Chris@13
|
289 $visitor6 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
290 $visitor6->expects($this->at(4))->method('leaveNode')
|
Chris@13
|
291 ->willReturn(false);
|
Chris@13
|
292
|
Chris@13
|
293 $visitor7 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
294 $visitor7->expects($this->at(1))->method('enterNode')
|
Chris@13
|
295 ->willReturn(new Node\Scalar\LNumber(42));
|
Chris@13
|
296
|
Chris@13
|
297 $visitor8 = $this->getMockBuilder(NodeVisitor::class)->getMock();
|
Chris@13
|
298 $visitor8->expects($this->at(2))->method('enterNode')
|
Chris@13
|
299 ->willReturn(new Node\Stmt\Return_());
|
Chris@13
|
300
|
Chris@13
|
301 return [
|
Chris@13
|
302 [$visitor1, 'enterNode() returned invalid value of type string'],
|
Chris@13
|
303 [$visitor2, 'enterNode() returned invalid value of type string'],
|
Chris@13
|
304 [$visitor3, 'leaveNode() returned invalid value of type string'],
|
Chris@13
|
305 [$visitor4, 'leaveNode() returned invalid value of type string'],
|
Chris@13
|
306 [$visitor5, 'leaveNode() may only return an array if the parent structure is an array'],
|
Chris@13
|
307 [$visitor6, 'bool(false) return from leaveNode() no longer supported. Return NodeTraverser::REMOVE_NODE instead'],
|
Chris@13
|
308 [$visitor7, 'Trying to replace statement (Stmt_Expression) with expression (Scalar_LNumber). Are you missing a Stmt_Expression wrapper?'],
|
Chris@13
|
309 [$visitor8, 'Trying to replace expression (Scalar_LNumber) with statement (Stmt_Return)'],
|
Chris@13
|
310 ];
|
Chris@13
|
311 }
|
Chris@0
|
312 }
|