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