annotate vendor/nikic/php-parser/test/PhpParser/NodeTraverserTest.php @ 13:5fb285c0d0e3

Update Drupal core to 8.4.7 via Composer. Security update; I *think* we've been lucky to get away with this so far, as we don't support self-registration which seems to be used by the so-called "drupalgeddon 2" attack that 8.4.5 was vulnerable to.
author Chris Cannam
date Mon, 23 Apr 2018 09:33:26 +0100
parents 4c8ae668cc8c
children 129ea1e6d783
rev   line source
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 }