541 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			541 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| /*
 | |
|  * This file is part of the Symfony package.
 | |
|  *
 | |
|  * (c) Fabien Potencier <fabien@symfony.com>
 | |
|  *
 | |
|  * For the full copyright and license information, please view the LICENSE
 | |
|  * file that was distributed with this source code.
 | |
|  */
 | |
| 
 | |
| namespace Symfony\Component\Yaml\Tests;
 | |
| 
 | |
| use PHPUnit\Framework\TestCase;
 | |
| use Symfony\Component\Yaml\Dumper;
 | |
| use Symfony\Component\Yaml\Parser;
 | |
| use Symfony\Component\Yaml\Tag\TaggedValue;
 | |
| use Symfony\Component\Yaml\Yaml;
 | |
| 
 | |
| class DumperTest extends TestCase
 | |
| {
 | |
|     protected $parser;
 | |
|     protected $dumper;
 | |
|     protected $path;
 | |
| 
 | |
|     protected $array = [
 | |
|         '' => 'bar',
 | |
|         'foo' => '#bar',
 | |
|         'foo\'bar' => [],
 | |
|         'bar' => [1, 'foo'],
 | |
|         'foobar' => [
 | |
|             'foo' => 'bar',
 | |
|             'bar' => [1, 'foo'],
 | |
|             'foobar' => [
 | |
|                 'foo' => 'bar',
 | |
|                 'bar' => [1, 'foo'],
 | |
|             ],
 | |
|         ],
 | |
|     ];
 | |
| 
 | |
|     protected function setUp(): void
 | |
|     {
 | |
|         $this->parser = new Parser();
 | |
|         $this->dumper = new Dumper();
 | |
|         $this->path = __DIR__.'/Fixtures';
 | |
|     }
 | |
| 
 | |
|     protected function tearDown(): void
 | |
|     {
 | |
|         $this->parser = null;
 | |
|         $this->dumper = null;
 | |
|         $this->path = null;
 | |
|         $this->array = null;
 | |
|     }
 | |
| 
 | |
|     public function testIndentationInConstructor()
 | |
|     {
 | |
|         $dumper = new Dumper(7);
 | |
|         $expected = <<<'EOF'
 | |
| '': bar
 | |
| foo: '#bar'
 | |
| 'foo''bar': {  }
 | |
| bar:
 | |
|        - 1
 | |
|        - foo
 | |
| foobar:
 | |
|        foo: bar
 | |
|        bar:
 | |
|               - 1
 | |
|               - foo
 | |
|        foobar:
 | |
|               foo: bar
 | |
|               bar:
 | |
|                      - 1
 | |
|                      - foo
 | |
| 
 | |
| EOF;
 | |
|         $this->assertEquals($expected, $dumper->dump($this->array, 4, 0));
 | |
|     }
 | |
| 
 | |
|     public function testSpecifications()
 | |
|     {
 | |
|         $files = $this->parser->parse(file_get_contents($this->path.'/index.yml'));
 | |
|         foreach ($files as $file) {
 | |
|             $yamls = file_get_contents($this->path.'/'.$file.'.yml');
 | |
| 
 | |
|             // split YAMLs documents
 | |
|             foreach (preg_split('/^---( %YAML\:1\.0)?/m', $yamls) as $yaml) {
 | |
|                 if (!$yaml) {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 $test = $this->parser->parse($yaml);
 | |
|                 if (isset($test['dump_skip']) && $test['dump_skip']) {
 | |
|                     continue;
 | |
|                 } elseif (isset($test['todo']) && $test['todo']) {
 | |
|                     // TODO
 | |
|                 } else {
 | |
|                     eval('$expected = '.trim($test['php']).';');
 | |
|                     $this->assertSame($expected, $this->parser->parse($this->dumper->dump($expected, 10)), $test['test']);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function testInlineLevel()
 | |
|     {
 | |
|         $expected = <<<'EOF'
 | |
| { '': bar, foo: '#bar', 'foo''bar': {  }, bar: [1, foo], foobar: { foo: bar, bar: [1, foo], foobar: { foo: bar, bar: [1, foo] } } }
 | |
| EOF;
 | |
|         $this->assertEquals($expected, $this->dumper->dump($this->array, -10), '->dump() takes an inline level argument');
 | |
|         $this->assertEquals($expected, $this->dumper->dump($this->array, 0), '->dump() takes an inline level argument');
 | |
| 
 | |
|         $expected = <<<'EOF'
 | |
| '': bar
 | |
| foo: '#bar'
 | |
| 'foo''bar': {  }
 | |
| bar: [1, foo]
 | |
| foobar: { foo: bar, bar: [1, foo], foobar: { foo: bar, bar: [1, foo] } }
 | |
| 
 | |
| EOF;
 | |
|         $this->assertEquals($expected, $this->dumper->dump($this->array, 1), '->dump() takes an inline level argument');
 | |
| 
 | |
|         $expected = <<<'EOF'
 | |
| '': bar
 | |
| foo: '#bar'
 | |
| 'foo''bar': {  }
 | |
| bar:
 | |
|     - 1
 | |
|     - foo
 | |
| foobar:
 | |
|     foo: bar
 | |
|     bar: [1, foo]
 | |
|     foobar: { foo: bar, bar: [1, foo] }
 | |
| 
 | |
| EOF;
 | |
|         $this->assertEquals($expected, $this->dumper->dump($this->array, 2), '->dump() takes an inline level argument');
 | |
| 
 | |
|         $expected = <<<'EOF'
 | |
| '': bar
 | |
| foo: '#bar'
 | |
| 'foo''bar': {  }
 | |
| bar:
 | |
|     - 1
 | |
|     - foo
 | |
| foobar:
 | |
|     foo: bar
 | |
|     bar:
 | |
|         - 1
 | |
|         - foo
 | |
|     foobar:
 | |
|         foo: bar
 | |
|         bar: [1, foo]
 | |
| 
 | |
| EOF;
 | |
|         $this->assertEquals($expected, $this->dumper->dump($this->array, 3), '->dump() takes an inline level argument');
 | |
| 
 | |
|         $expected = <<<'EOF'
 | |
| '': bar
 | |
| foo: '#bar'
 | |
| 'foo''bar': {  }
 | |
| bar:
 | |
|     - 1
 | |
|     - foo
 | |
| foobar:
 | |
|     foo: bar
 | |
|     bar:
 | |
|         - 1
 | |
|         - foo
 | |
|     foobar:
 | |
|         foo: bar
 | |
|         bar:
 | |
|             - 1
 | |
|             - foo
 | |
| 
 | |
| EOF;
 | |
|         $this->assertEquals($expected, $this->dumper->dump($this->array, 4), '->dump() takes an inline level argument');
 | |
|         $this->assertEquals($expected, $this->dumper->dump($this->array, 10), '->dump() takes an inline level argument');
 | |
|     }
 | |
| 
 | |
|     public function testObjectSupportEnabled()
 | |
|     {
 | |
|         $dump = $this->dumper->dump(['foo' => new A(), 'bar' => 1], 0, 0, Yaml::DUMP_OBJECT);
 | |
| 
 | |
|         $this->assertEquals('{ foo: !php/object \'O:30:"Symfony\Component\Yaml\Tests\A":1:{s:1:"a";s:3:"foo";}\', bar: 1 }', $dump, '->dump() is able to dump objects');
 | |
|     }
 | |
| 
 | |
|     public function testObjectSupportDisabledButNoExceptions()
 | |
|     {
 | |
|         $dump = $this->dumper->dump(['foo' => new A(), 'bar' => 1]);
 | |
| 
 | |
|         $this->assertEquals('{ foo: null, bar: 1 }', $dump, '->dump() does not dump objects when disabled');
 | |
|     }
 | |
| 
 | |
|     public function testObjectSupportDisabledWithExceptions()
 | |
|     {
 | |
|         $this->expectException('Symfony\Component\Yaml\Exception\DumpException');
 | |
|         $this->dumper->dump(['foo' => new A(), 'bar' => 1], 0, 0, Yaml::DUMP_EXCEPTION_ON_INVALID_TYPE);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider getEscapeSequences
 | |
|      */
 | |
|     public function testEscapedEscapeSequencesInQuotedScalar($input, $expected)
 | |
|     {
 | |
|         $this->assertEquals($expected, $this->dumper->dump($input));
 | |
|     }
 | |
| 
 | |
|     public function getEscapeSequences()
 | |
|     {
 | |
|         return [
 | |
|             'empty string' => ['', "''"],
 | |
|             'null' => ["\x0", '"\\0"'],
 | |
|             'bell' => ["\x7", '"\\a"'],
 | |
|             'backspace' => ["\x8", '"\\b"'],
 | |
|             'horizontal-tab' => ["\t", '"\\t"'],
 | |
|             'line-feed' => ["\n", '"\\n"'],
 | |
|             'vertical-tab' => ["\v", '"\\v"'],
 | |
|             'form-feed' => ["\xC", '"\\f"'],
 | |
|             'carriage-return' => ["\r", '"\\r"'],
 | |
|             'escape' => ["\x1B", '"\\e"'],
 | |
|             'space' => [' ', "' '"],
 | |
|             'double-quote' => ['"', "'\"'"],
 | |
|             'slash' => ['/', '/'],
 | |
|             'backslash' => ['\\', '\\'],
 | |
|             'next-line' => ["\xC2\x85", '"\\N"'],
 | |
|             'non-breaking-space' => ["\xc2\xa0", '"\\_"'],
 | |
|             'line-separator' => ["\xE2\x80\xA8", '"\\L"'],
 | |
|             'paragraph-separator' => ["\xE2\x80\xA9", '"\\P"'],
 | |
|             'colon' => [':', "':'"],
 | |
|         ];
 | |
|     }
 | |
| 
 | |
|     public function testBinaryDataIsDumpedBase64Encoded()
 | |
|     {
 | |
|         $binaryData = file_get_contents(__DIR__.'/Fixtures/arrow.gif');
 | |
|         $expected = '{ data: !!binary '.base64_encode($binaryData).' }';
 | |
| 
 | |
|         $this->assertSame($expected, $this->dumper->dump(['data' => $binaryData]));
 | |
|     }
 | |
| 
 | |
|     public function testNonUtf8DataIsDumpedBase64Encoded()
 | |
|     {
 | |
|         // "für" (ISO-8859-1 encoded)
 | |
|         $this->assertSame('!!binary ZsM/cg==', $this->dumper->dump("f\xc3\x3fr"));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @dataProvider objectAsMapProvider
 | |
|      */
 | |
|     public function testDumpObjectAsMap($object, $expected)
 | |
|     {
 | |
|         $yaml = $this->dumper->dump($object, 0, 0, Yaml::DUMP_OBJECT_AS_MAP);
 | |
| 
 | |
|         $this->assertEquals($expected, Yaml::parse($yaml, Yaml::PARSE_OBJECT_FOR_MAP));
 | |
|     }
 | |
| 
 | |
|     public function objectAsMapProvider()
 | |
|     {
 | |
|         $tests = [];
 | |
| 
 | |
|         $bar = new \stdClass();
 | |
|         $bar->class = 'classBar';
 | |
|         $bar->args = ['bar'];
 | |
|         $zar = new \stdClass();
 | |
|         $foo = new \stdClass();
 | |
|         $foo->bar = $bar;
 | |
|         $foo->zar = $zar;
 | |
|         $object = new \stdClass();
 | |
|         $object->foo = $foo;
 | |
|         $tests['stdClass'] = [$object, $object];
 | |
| 
 | |
|         $arrayObject = new \ArrayObject();
 | |
|         $arrayObject['foo'] = 'bar';
 | |
|         $arrayObject['baz'] = 'foobar';
 | |
|         $parsedArrayObject = new \stdClass();
 | |
|         $parsedArrayObject->foo = 'bar';
 | |
|         $parsedArrayObject->baz = 'foobar';
 | |
|         $tests['ArrayObject'] = [$arrayObject, $parsedArrayObject];
 | |
| 
 | |
|         $a = new A();
 | |
|         $tests['arbitrary-object'] = [$a, null];
 | |
| 
 | |
|         return $tests;
 | |
|     }
 | |
| 
 | |
|     public function testDumpingArrayObjectInstancesRespectsInlineLevel()
 | |
|     {
 | |
|         $deep = new \ArrayObject(['deep1' => 'd', 'deep2' => 'e']);
 | |
|         $inner = new \ArrayObject(['inner1' => 'b', 'inner2' => 'c', 'inner3' => $deep]);
 | |
|         $outer = new \ArrayObject(['outer1' => 'a', 'outer2' => $inner]);
 | |
| 
 | |
|         $yaml = $this->dumper->dump($outer, 2, 0, Yaml::DUMP_OBJECT_AS_MAP);
 | |
| 
 | |
|         $expected = <<<YAML
 | |
| outer1: a
 | |
| outer2:
 | |
|     inner1: b
 | |
|     inner2: c
 | |
|     inner3: { deep1: d, deep2: e }
 | |
| 
 | |
| YAML;
 | |
|         $this->assertSame($expected, $yaml);
 | |
|     }
 | |
| 
 | |
|     public function testDumpingArrayObjectInstancesWithNumericKeysInlined()
 | |
|     {
 | |
|         $deep = new \ArrayObject(['d', 'e']);
 | |
|         $inner = new \ArrayObject(['b', 'c', $deep]);
 | |
|         $outer = new \ArrayObject(['a', $inner]);
 | |
| 
 | |
|         $yaml = $this->dumper->dump($outer, 0, 0, Yaml::DUMP_OBJECT_AS_MAP);
 | |
|         $expected = <<<YAML
 | |
| { 0: a, 1: { 0: b, 1: c, 2: { 0: d, 1: e } } }
 | |
| YAML;
 | |
|         $this->assertSame($expected, $yaml);
 | |
|     }
 | |
| 
 | |
|     public function testDumpingArrayObjectInstancesWithNumericKeysRespectsInlineLevel()
 | |
|     {
 | |
|         $deep = new \ArrayObject(['d', 'e']);
 | |
|         $inner = new \ArrayObject(['b', 'c', $deep]);
 | |
|         $outer = new \ArrayObject(['a', $inner]);
 | |
|         $yaml = $this->dumper->dump($outer, 2, 0, Yaml::DUMP_OBJECT_AS_MAP);
 | |
|         $expected = <<<YAML
 | |
| 0: a
 | |
| 1:
 | |
|     0: b
 | |
|     1: c
 | |
|     2: { 0: d, 1: e }
 | |
| 
 | |
| YAML;
 | |
|         $this->assertEquals($expected, $yaml);
 | |
|     }
 | |
| 
 | |
|     public function testDumpEmptyArrayObjectInstanceAsMap()
 | |
|     {
 | |
|         $this->assertSame('{  }', $this->dumper->dump(new \ArrayObject(), 2, 0, Yaml::DUMP_OBJECT_AS_MAP));
 | |
|     }
 | |
| 
 | |
|     public function testDumpEmptyStdClassInstanceAsMap()
 | |
|     {
 | |
|         $this->assertSame('{  }', $this->dumper->dump(new \stdClass(), 2, 0, Yaml::DUMP_OBJECT_AS_MAP));
 | |
|     }
 | |
| 
 | |
|     public function testDumpingStdClassInstancesRespectsInlineLevel()
 | |
|     {
 | |
|         $deep = new \stdClass();
 | |
|         $deep->deep1 = 'd';
 | |
|         $deep->deep2 = 'e';
 | |
| 
 | |
|         $inner = new \stdClass();
 | |
|         $inner->inner1 = 'b';
 | |
|         $inner->inner2 = 'c';
 | |
|         $inner->inner3 = $deep;
 | |
| 
 | |
|         $outer = new \stdClass();
 | |
|         $outer->outer1 = 'a';
 | |
|         $outer->outer2 = $inner;
 | |
| 
 | |
|         $yaml = $this->dumper->dump($outer, 2, 0, Yaml::DUMP_OBJECT_AS_MAP);
 | |
| 
 | |
|         $expected = <<<YAML
 | |
| outer1: a
 | |
| outer2:
 | |
|     inner1: b
 | |
|     inner2: c
 | |
|     inner3: { deep1: d, deep2: e }
 | |
| 
 | |
| YAML;
 | |
|         $this->assertSame($expected, $yaml);
 | |
|     }
 | |
| 
 | |
|     public function testDumpingTaggedValueSequenceRespectsInlineLevel()
 | |
|     {
 | |
|         $data = [
 | |
|             new TaggedValue('user', [
 | |
|                 'username' => 'jane',
 | |
|             ]),
 | |
|             new TaggedValue('user', [
 | |
|                 'username' => 'john',
 | |
|             ]),
 | |
|         ];
 | |
| 
 | |
|         $yaml = $this->dumper->dump($data, 2);
 | |
| 
 | |
|         $expected = <<<YAML
 | |
| - !user
 | |
|   username: jane
 | |
| - !user
 | |
|   username: john
 | |
| 
 | |
| YAML;
 | |
|         $this->assertSame($expected, $yaml);
 | |
|     }
 | |
| 
 | |
|     public function testDumpingTaggedValueSequenceWithInlinedTagValues()
 | |
|     {
 | |
|         $data = [
 | |
|             new TaggedValue('user', [
 | |
|                 'username' => 'jane',
 | |
|             ]),
 | |
|             new TaggedValue('user', [
 | |
|                 'username' => 'john',
 | |
|             ]),
 | |
|         ];
 | |
| 
 | |
|         $yaml = $this->dumper->dump($data, 1);
 | |
| 
 | |
|         $expected = <<<YAML
 | |
| - !user { username: jane }
 | |
| - !user { username: john }
 | |
| 
 | |
| YAML;
 | |
|         $this->assertSame($expected, $yaml);
 | |
|     }
 | |
| 
 | |
|     public function testDumpingTaggedValueMapRespectsInlineLevel()
 | |
|     {
 | |
|         $data = [
 | |
|             'user1' => new TaggedValue('user', [
 | |
|                 'username' => 'jane',
 | |
|             ]),
 | |
|             'user2' => new TaggedValue('user', [
 | |
|                 'username' => 'john',
 | |
|             ]),
 | |
|         ];
 | |
| 
 | |
|         $yaml = $this->dumper->dump($data, 2);
 | |
| 
 | |
|         $expected = <<<YAML
 | |
| user1: !user
 | |
|     username: jane
 | |
| user2: !user
 | |
|     username: john
 | |
| 
 | |
| YAML;
 | |
|         $this->assertSame($expected, $yaml);
 | |
|     }
 | |
| 
 | |
|     public function testDumpingTaggedValueMapWithInlinedTagValues()
 | |
|     {
 | |
|         $data = [
 | |
|             'user1' => new TaggedValue('user', [
 | |
|                 'username' => 'jane',
 | |
|             ]),
 | |
|             'user2' => new TaggedValue('user', [
 | |
|                 'username' => 'john',
 | |
|             ]),
 | |
|         ];
 | |
| 
 | |
|         $yaml = $this->dumper->dump($data, 1);
 | |
| 
 | |
|         $expected = <<<YAML
 | |
| user1: !user { username: jane }
 | |
| user2: !user { username: john }
 | |
| 
 | |
| YAML;
 | |
|         $this->assertSame($expected, $yaml);
 | |
|     }
 | |
| 
 | |
|     public function testDumpingNotInlinedScalarTaggedValue()
 | |
|     {
 | |
|         $data = [
 | |
|             'user1' => new TaggedValue('user', 'jane'),
 | |
|             'user2' => new TaggedValue('user', 'john'),
 | |
|         ];
 | |
|         $expected = <<<YAML
 | |
| user1: !user jane
 | |
| user2: !user john
 | |
| 
 | |
| YAML;
 | |
| 
 | |
|         $this->assertSame($expected, $this->dumper->dump($data, 2));
 | |
|     }
 | |
| 
 | |
|     public function testDumpingNotInlinedNullTaggedValue()
 | |
|     {
 | |
|         $data = [
 | |
|             'foo' => new TaggedValue('bar', null),
 | |
|         ];
 | |
|         $expected = <<<YAML
 | |
| foo: !bar null
 | |
| 
 | |
| YAML;
 | |
| 
 | |
|         $this->assertSame($expected, $this->dumper->dump($data, 2));
 | |
|     }
 | |
| 
 | |
|     public function testDumpMultiLineStringAsScalarBlock()
 | |
|     {
 | |
|         $data = [
 | |
|             'data' => [
 | |
|                 'single_line' => 'foo bar baz',
 | |
|                 'multi_line' => "foo\nline with trailing spaces:\n  \nbar\ninteger like line:\n123456789\nempty line:\n\nbaz",
 | |
|                 'multi_line_with_carriage_return' => "foo\nbar\r\nbaz",
 | |
|                 'nested_inlined_multi_line_string' => [
 | |
|                     'inlined_multi_line' => "foo\nbar\r\nempty line:\n\nbaz",
 | |
|                 ],
 | |
|             ],
 | |
|         ];
 | |
| 
 | |
|         $this->assertSame(file_get_contents(__DIR__.'/Fixtures/multiple_lines_as_literal_block.yml'), $this->dumper->dump($data, 2, 0, Yaml::DUMP_MULTI_LINE_LITERAL_BLOCK));
 | |
|     }
 | |
| 
 | |
|     public function testDumpMultiLineStringAsScalarBlockWhenFirstLineHasLeadingSpace()
 | |
|     {
 | |
|         $data = [
 | |
|             'data' => [
 | |
|                 'multi_line' => "    the first line has leading spaces\nThe second line does not.",
 | |
|             ],
 | |
|         ];
 | |
| 
 | |
|         $this->assertSame(file_get_contents(__DIR__.'/Fixtures/multiple_lines_as_literal_block_leading_space_in_first_line.yml'), $this->dumper->dump($data, 2, 0, Yaml::DUMP_MULTI_LINE_LITERAL_BLOCK));
 | |
|     }
 | |
| 
 | |
|     public function testCarriageReturnIsMaintainedWhenDumpingAsMultiLineLiteralBlock()
 | |
|     {
 | |
|         $this->assertSame("- \"a\\r\\nb\\nc\"\n", $this->dumper->dump(["a\r\nb\nc"], 2, 0, Yaml::DUMP_MULTI_LINE_LITERAL_BLOCK));
 | |
|     }
 | |
| 
 | |
|     public function testZeroIndentationThrowsException()
 | |
|     {
 | |
|         $this->expectException('InvalidArgumentException');
 | |
|         $this->expectExceptionMessage('The indentation must be greater than zero');
 | |
|         new Dumper(0);
 | |
|     }
 | |
| 
 | |
|     public function testNegativeIndentationThrowsException()
 | |
|     {
 | |
|         $this->expectException('InvalidArgumentException');
 | |
|         $this->expectExceptionMessage('The indentation must be greater than zero');
 | |
|         new Dumper(-4);
 | |
|     }
 | |
| }
 | |
| 
 | |
| class A
 | |
| {
 | |
|     public $a = 'foo';
 | |
| }
 | 
