| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773 | <?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 Symfony\Component\Yaml\Yaml;use Symfony\Component\Yaml\Parser;class ParserTest extends \PHPUnit_Framework_TestCase{    protected $parser;    protected function setUp()    {        $this->parser = new Parser();    }    protected function tearDown()    {        $this->parser = null;    }    /**     * @dataProvider getDataFormSpecifications     */    public function testSpecifications($file, $expected, $yaml, $comment)    {        $this->assertEquals($expected, var_export($this->parser->parse($yaml), true), $comment);    }    public function getDataFormSpecifications()    {        $parser = new Parser();        $path = __DIR__.'/Fixtures';        $tests = array();        $files = $parser->parse(file_get_contents($path.'/index.yml'));        foreach ($files as $file) {            $yamls = file_get_contents($path.'/'.$file.'.yml');            // split YAMLs documents            foreach (preg_split('/^---( %YAML\:1\.0)?/m', $yamls) as $yaml) {                if (!$yaml) {                    continue;                }                $test = $parser->parse($yaml);                if (isset($test['todo']) && $test['todo']) {                    // TODO                } else {                    eval('$expected = '.trim($test['php']).';');                    $tests[] = array($file, var_export($expected, true), $test['yaml'], $test['test']);                }            }        }        return $tests;    }    public function testTabsInYaml()    {        // test tabs in YAML        $yamls = array(            "foo:\n	bar",            "foo:\n 	bar",            "foo:\n	 bar",            "foo:\n 	 bar",        );        foreach ($yamls as $yaml) {            try {                $content = $this->parser->parse($yaml);                $this->fail('YAML files must not contain tabs');            } catch (\Exception $e) {                $this->assertInstanceOf('\Exception', $e, 'YAML files must not contain tabs');                $this->assertEquals('A YAML file cannot contain tabs as indentation at line 2 (near "'.strpbrk($yaml, "\t").'").', $e->getMessage(), 'YAML files must not contain tabs');            }        }    }    public function testEndOfTheDocumentMarker()    {        $yaml = <<<EOF--- %YAML:1.0foo...EOF;        $this->assertEquals('foo', $this->parser->parse($yaml));    }    public function getBlockChompingTests()    {        $tests = array();        $yaml = <<<'EOF'foo: |-    one    twobar: |-    one    twoEOF;        $expected = array(            'foo' => "one\ntwo",            'bar' => "one\ntwo",        );        $tests['Literal block chomping strip with single trailing newline'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: |-    one    twobar: |-    one    twoEOF;        $expected = array(            'foo' => "one\ntwo",            'bar' => "one\ntwo",        );        $tests['Literal block chomping strip with multiple trailing newlines'] = array($expected, $yaml);        $yaml = <<<'EOF'{}EOF;        $expected = array();        $tests['Literal block chomping strip with multiple trailing newlines after a 1-liner'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: |-    one    twobar: |-    one    twoEOF;        $expected = array(            'foo' => "one\ntwo",            'bar' => "one\ntwo",        );        $tests['Literal block chomping strip without trailing newline'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: |    one    twobar: |    one    twoEOF;        $expected = array(            'foo' => "one\ntwo\n",            'bar' => "one\ntwo\n",        );        $tests['Literal block chomping clip with single trailing newline'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: |    one    twobar: |    one    twoEOF;        $expected = array(            'foo' => "one\ntwo\n",            'bar' => "one\ntwo\n",        );        $tests['Literal block chomping clip with multiple trailing newlines'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: |    one    twobar: |    one    twoEOF;        $expected = array(            'foo' => "one\ntwo\n",            'bar' => "one\ntwo",        );        $tests['Literal block chomping clip without trailing newline'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: |+    one    twobar: |+    one    twoEOF;        $expected = array(            'foo' => "one\ntwo\n",            'bar' => "one\ntwo\n",        );        $tests['Literal block chomping keep with single trailing newline'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: |+    one    twobar: |+    one    twoEOF;        $expected = array(            'foo' => "one\ntwo\n\n",            'bar' => "one\ntwo\n\n",        );        $tests['Literal block chomping keep with multiple trailing newlines'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: |+    one    twobar: |+    one    twoEOF;        $expected = array(            'foo' => "one\ntwo\n",            'bar' => "one\ntwo",        );        $tests['Literal block chomping keep without trailing newline'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: >-    one    twobar: >-    one    twoEOF;        $expected = array(            'foo' => 'one two',            'bar' => 'one two',        );        $tests['Folded block chomping strip with single trailing newline'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: >-    one    twobar: >-    one    twoEOF;        $expected = array(            'foo' => 'one two',            'bar' => 'one two',        );        $tests['Folded block chomping strip with multiple trailing newlines'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: >-    one    twobar: >-    one    twoEOF;        $expected = array(            'foo' => 'one two',            'bar' => 'one two',        );        $tests['Folded block chomping strip without trailing newline'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: >    one    twobar: >    one    twoEOF;        $expected = array(            'foo' => "one two\n",            'bar' => "one two\n",        );        $tests['Folded block chomping clip with single trailing newline'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: >    one    twobar: >    one    twoEOF;        $expected = array(            'foo' => "one two\n",            'bar' => "one two\n",        );        $tests['Folded block chomping clip with multiple trailing newlines'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: >    one    twobar: >    one    twoEOF;        $expected = array(            'foo' => "one two\n",            'bar' => 'one two',        );        $tests['Folded block chomping clip without trailing newline'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: >+    one    twobar: >+    one    twoEOF;        $expected = array(            'foo' => "one two\n",            'bar' => "one two\n",        );        $tests['Folded block chomping keep with single trailing newline'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: >+    one    twobar: >+    one    twoEOF;        $expected = array(            'foo' => "one two\n\n",            'bar' => "one two\n\n",        );        $tests['Folded block chomping keep with multiple trailing newlines'] = array($expected, $yaml);        $yaml = <<<'EOF'foo: >+    one    twobar: >+    one    twoEOF;        $expected = array(            'foo' => "one two\n",            'bar' => 'one two',        );        $tests['Folded block chomping keep without trailing newline'] = array($expected, $yaml);        return $tests;    }    /**     * @dataProvider getBlockChompingTests     */    public function testBlockChomping($expected, $yaml)    {        $this->assertSame($expected, $this->parser->parse($yaml));    }    /**     * Regression test for issue #7989.     *     * @see https://github.com/symfony/symfony/issues/7989     */    public function testBlockLiteralWithLeadingNewlines()    {        $yaml = <<<'EOF'foo: |-    barEOF;        $expected = array(            'foo' => "\n\nbar",        );        $this->assertSame($expected, $this->parser->parse($yaml));    }    public function testObjectSupportEnabled()    {        $input = <<<EOFfoo: !!php/object:O:30:"Symfony\Component\Yaml\Tests\B":1:{s:1:"b";s:3:"foo";}bar: 1EOF;        $this->assertEquals(array('foo' => new B(), 'bar' => 1), $this->parser->parse($input, false, true), '->parse() is able to parse objects');    }    public function testObjectSupportDisabledButNoExceptions()    {        $input = <<<EOFfoo: !!php/object:O:30:"Symfony\Tests\Component\Yaml\B":1:{s:1:"b";s:3:"foo";}bar: 1EOF;        $this->assertEquals(array('foo' => null, 'bar' => 1), $this->parser->parse($input), '->parse() does not parse objects');    }    /**     * @expectedException \Symfony\Component\Yaml\Exception\ParseException     */    public function testObjectsSupportDisabledWithExceptions()    {        $this->parser->parse('foo: !!php/object:O:30:"Symfony\Tests\Component\Yaml\B":1:{s:1:"b";s:3:"foo";}', true, false);    }    public function testNonUtf8Exception()    {        if (!function_exists('iconv')) {            $this->markTestSkipped('Exceptions for non-utf8 charsets require the iconv() function.');            return;        }        $yamls = array(            iconv('UTF-8', 'ISO-8859-1', "foo: 'äöüß'"),            iconv('UTF-8', 'ISO-8859-15', "euro: '€'"),            iconv('UTF-8', 'CP1252', "cp1252: '©ÉÇáñ'"),        );        foreach ($yamls as $yaml) {            try {                $this->parser->parse($yaml);                $this->fail('charsets other than UTF-8 are rejected.');            } catch (\Exception $e) {                $this->assertInstanceOf('Symfony\Component\Yaml\Exception\ParseException', $e, 'charsets other than UTF-8 are rejected.');            }        }    }    /**     * @expectedException \Symfony\Component\Yaml\Exception\ParseException     */    public function testUnindentedCollectionException()    {        $yaml = <<<EOFcollection:-item1-item2-item3EOF;        $this->parser->parse($yaml);    }    /**     * @expectedException \Symfony\Component\Yaml\Exception\ParseException     */    public function testShortcutKeyUnindentedCollectionException()    {        $yaml = <<<EOFcollection:-  key: foo  foo: barEOF;        $this->parser->parse($yaml);    }    /**     * @expectedException \Symfony\Component\Yaml\Exception\ParseException     * @expectedExceptionMessage Multiple documents are not supported.     */    public function testMultipleDocumentsNotSupportedException()    {        Yaml::parse(<<<EOL# Ranking of 1998 home runs---- Mark McGwire- Sammy Sosa- Ken Griffey# Team ranking---- Chicago Cubs- St Louis CardinalsEOL        );    }    /**     * @expectedException \Symfony\Component\Yaml\Exception\ParseException     */    public function testSequenceInAMapping()    {        Yaml::parse(<<<EOFyaml:  hash: me  - array stuffEOF        );    }    /**     * @expectedException \Symfony\Component\Yaml\Exception\ParseException     */    public function testMappingInASequence()    {        Yaml::parse(<<<EOFyaml:  - array stuff  hash: meEOF        );    }    /**     * @expectedException \Symfony\Component\Yaml\Exception\ParseException     * @expectedExceptionMessage missing colon     */    public function testScalarInSequence()    {        Yaml::parse(<<<EOFfoo:    - bar"missing colon"    foo: barEOF        );    }    /**     * > It is an error for two equal keys to appear in the same mapping node.     * > In such a case the YAML processor may continue, ignoring the second     * > `key: value` pair and issuing an appropriate warning. This strategy     * > preserves a consistent information model for one-pass and random access     * > applications.     *     * @see http://yaml.org/spec/1.2/spec.html#id2759572     * @see http://yaml.org/spec/1.1/#id932806     *     * @covers \Symfony\Component\Yaml\Parser::parse     */    public function testMappingDuplicateKeyBlock()    {        $input = <<<EODparent:    child: first    child: duplicateparent:    child: duplicate    child: duplicateEOD;        $expected = array(            'parent' => array(                'child' => 'first',            ),        );        $this->assertSame($expected, Yaml::parse($input));    }    /**     * @covers \Symfony\Component\Yaml\Inline::parseMapping     */    public function testMappingDuplicateKeyFlow()    {        $input = <<<EODparent: { child: first, child: duplicate }parent: { child: duplicate, child: duplicate }EOD;        $expected = array(            'parent' => array(                'child' => 'first',            ),        );        $this->assertSame($expected, Yaml::parse($input));    }    public function testEmptyValue()    {        $input = <<<EOFhash:EOF;        $this->assertEquals(array('hash' => null), Yaml::parse($input));    }    public function testStringBlockWithComments()    {        $this->assertEquals(array('content' => <<<EOT# comment 1header    # comment 2    <body>        <h1>title</h1>    </body>footer # comment3EOT        ), Yaml::parse(<<<EOFcontent: |    # comment 1    header        # comment 2        <body>            <h1>title</h1>        </body>    footer # comment3EOF        ));    }    public function testFoldedStringBlockWithComments()    {        $this->assertEquals(array(array('content' => <<<EOT# comment 1header    # comment 2    <body>        <h1>title</h1>    </body>footer # comment3EOT        )), Yaml::parse(<<<EOF-    content: |        # comment 1        header            # comment 2            <body>                <h1>title</h1>            </body>        footer # comment3EOF        ));    }    public function testNestedFoldedStringBlockWithComments()    {        $this->assertEquals(array(array(            'title' => 'some title',            'content' => <<<EOT# comment 1header    # comment 2    <body>        <h1>title</h1>    </body>footer # comment3EOT        )), Yaml::parse(<<<EOF-    title: some title    content: |        # comment 1        header            # comment 2            <body>                <h1>title</h1>            </body>        footer # comment3EOF        ));    }    public function testReferenceResolvingInInlineStrings()    {        $this->assertEquals(array(            'var' => 'var-value',            'scalar' => 'var-value',            'list' => array('var-value'),            'list_in_list' => array(array('var-value')),            'map_in_list' => array(array('key' => 'var-value')),            'embedded_mapping' => array(array('key' => 'var-value')),            'map' => array('key' => 'var-value'),            'list_in_map' => array('key' => array('var-value')),            'map_in_map' => array('foo' => array('bar' => 'var-value')),        ), Yaml::parse(<<<EOFvar:  &var var-valuescalar: *varlist: [ *var ]list_in_list: [[ *var ]]map_in_list: [ { key: *var } ]embedded_mapping: [ key: *var ]map: { key: *var }list_in_map: { key: [*var] }map_in_map: { foo: { bar: *var } }EOF        ));    }    public function testYamlDirective()    {        $yaml = <<<EOF%YAML 1.2---foo: 1bar: 2EOF;        $this->assertEquals(array('foo' => 1, 'bar' => 2), $this->parser->parse($yaml));    }    public function testFloatKeys()    {        $yaml = <<<EOFfoo:    1.2: "bar"    1.3: "baz"EOF;        $expected = array(            'foo' => array(                '1.2' => 'bar',                '1.3' => 'baz',            ),        );        $this->assertEquals($expected, $this->parser->parse($yaml));    }}class B{    public $b = 'foo';}
 |