jkphl/micrometa

View on GitHub
src/Micrometa/Tests/Infrastructure/ParserTest.php

Summary

Maintainability
A
0 mins
Test Coverage
<?php

/**
 * micrometa
 *
 * @category   Jkphl
 * @package    Jkphl\Micrometa
 * @subpackage Jkphl\Micrometa\Tests\Domain
 * @author     Joschi Kuphal <joschi@kuphal.net> / @jkphl
 * @copyright  Copyright © 2018 Joschi Kuphal <joschi@kuphal.net> / @jkphl
 * @license    http://opensource.org/licenses/MIT The MIT License (MIT)
 */

/***********************************************************************************
 *  The MIT License (MIT)
 *
 *  Copyright © 2018 Joschi Kuphal <joschi@kuphal.net> / @jkphl
 *
 *  Permission is hereby granted, free of charge, to any person obtaining a copy of
 *  this software and associated documentation files (the "Software"), to deal in
 *  the Software without restriction, including without limitation the rights to
 *  use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
 *  the Software, and to permit persons to whom the Software is furnished to do so,
 *  subject to the following conditions:
 *
 *  The above copyright notice and this permission notice shall be included in all
 *  copies or substantial portions of the Software.
 *
 *  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 *  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
 *  FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
 *  COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
 *  IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 *  CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 ***********************************************************************************/

namespace Jkphl\Micrometa\Tests\Infrastructure;

use Jkphl\Micrometa\Application\Contract\ParserInterface;
use Jkphl\Micrometa\Application\Item\Item;
use Jkphl\Micrometa\Application\Item\ItemInterface;
use Jkphl\Micrometa\Application\Value\StringValue;
use Jkphl\Micrometa\Domain\Item\Iri;
use Jkphl\Micrometa\Infrastructure\Parser\JsonLD;
use Jkphl\Micrometa\Infrastructure\Parser\LinkType;
use Jkphl\Micrometa\Infrastructure\Parser\Microdata;
use Jkphl\Micrometa\Infrastructure\Parser\Microformats;
use Jkphl\Micrometa\Infrastructure\Parser\RdfaLite;
use Jkphl\Micrometa\Tests\AbstractTestBase;

/**
 * Parser tests
 *
 * @package    Jkphl\Micrometa
 * @subpackage Jkphl\Micrometa\Tests
 */
class ParserTest extends AbstractTestBase
{
    /**
     * Test the JSON-LD parser with multiple languages
     */
    public function testLanguageJsonLDParser()
    {
        $items = $this->parseItems('json-ld/jsonld-languages.html', JsonLD::class);
        $this->assertTrue(is_array($items));
        $this->assertEquals(1, count($items));
        $this->assertInstanceOf(Item::class, $items[0]);
        $this->assertEquals(JsonLD::FORMAT, $items[0]->getFormat());
        $this->assertEquals('http://example.com/id1', $items[0]->getId());

        /** @var StringValue[] $propertyValues */
        $propertyValues = $items[0]->getProperty('http://example.com/term6');
        $this->assertTrue(is_array($propertyValues));
        foreach ([null, null, 'en', 'de'] as $index => $language) {
            $this->assertInstanceOf(StringValue::class, $propertyValues[$index]);
            $this->assertEquals(strval($index + 1), strval($propertyValues[$index]));
            $this->assertEquals($language, $propertyValues[$index]->getLanguage());
        }
    }

    /**
     * Parse items from fixture with a particular parser type
     *
     * @param string $fixture     Fixture
     * @param string $parser      Parser class name
     * @param int $errorThreshold Error threshold
     *
     * @return ItemInterface[] Items
     */
    protected function parseItems(string $fixture, string $parser, int $errorThreshold = 400)
    {
        list($uri, $dom) = $this->getUriFixture($fixture);
        /** @var ParserInterface $parser */
        $parser = new $parser($uri, self::getLogger($errorThreshold));

        return $parser->parseDom($dom)->getItems();
    }

    /**
     * Test the JSON-LD parser with multiple documents and file cache
     */
    public function testMultipleJsonLDParser()
    {
        $items = $this->parseItems('json-ld/jsonld-examples.html', JsonLD::class, 0);
        $this->assertTrue(is_array($items));
        $this->assertEquals(5, count($items));
        $this->assertInstanceOf(Item::class, $items[0]);
        $this->assertEquals(JsonLD::FORMAT, $items[0]->getFormat());
        $this->assertEquals('https://jsonld-examples.com/#header_website', $items[0]->getId());
    }

    /**
     * Test the JSON-LD parser with example that was throwing an InvalidArgumentException.
     * @see https://github.com/jkphl/micrometa/pull/59
     */
    public function testFixIncorrectEmptyTypeListInJsonLDParser()
    {
        $items = $this->parseItems('json-ld/jsonld-incorrect-empty-type-list.html', JsonLD::class, 0);
        $this->assertTrue(is_array($items));
        $this->assertEquals(1, count($items));

        $type = $items[0]->getType();
        $this->assertEquals('http://schema.org/Product', (string) $type[0]);
    }

    /**
     * Test the JSON-LD parser with an invalid document
     */
    public function testInvalidJsonLDParser()
    {
        $items = $this->parseItems('json-ld/jsonld-invalid.html', JsonLD::class, 0);
        $this->assertTrue(is_array($items));
        $this->assertEquals(0, count($items));
    }

    /**
     * Test the JSON-LD parser with an invalid document
     */
    public function testFixOnSemicolonForJsonLDParser()
    {
        $items = $this->parseItems('json-ld/jsonld-ending-semicolon.html', JsonLD::class, 0);
        $this->assertTrue(is_array($items));
        $this->assertEquals(1, count($items));
    }

    /**
     * Test the Microformats parser
     */
    public function testMicroformatsParser()
    {
        $items = $this->parseItems('microformats/entry.html', Microformats::class);
        $this->assertTrue(is_array($items));
        $this->assertEquals(1, count($items));
        $this->assertInstanceOf(Item::class, $items[0]);
        $this->assertEquals(Microformats::FORMAT, $items[0]->getFormat());
    }

    /**
     * Test the Microformats parser with nested items
     */
    public function testNestedMicroformatsParser()
    {
        $items = $this->parseItems('microformats/nested-events.html', Microformats::class);
        $this->assertTrue(is_array($items));
        $this->assertEquals(1, count($items));
        $this->assertInstanceOf(Item::class, $items[0]);
        $this->assertEquals(Microformats::FORMAT, $items[0]->getFormat());
        $this->assertEquals(2, count($items[0]->getChildren()));
    }

    /**
     * Test the HTML Microdata parser
     */
    public function testMicrodataParser()
    {
        $items              = $this->parseItems('html-microdata/article-microdata.html', Microdata::class);
        $expectedItemFormat = Microdata::FORMAT;
        $expectedItemIri    = new Iri('http://schema.org/', 'NewsArticle');
        $this->assertItemParsedAs($items, $expectedItemFormat, $expectedItemIri);
    }

    /**
     * Assert that items are of a particular type
     *
     * @param ItemInterface[] $items  Items
     * @param int $expectedItemFormat Expected item format
     * @param Iri $expectedItemIri    Expected item IRI
     */
    protected function assertItemParsedAs(array $items, int $expectedItemFormat, Iri $expectedItemIri)
    {
        $this->assertIsArray($items);
        $this->assertCount(1, $items);
        $this->assertInstanceOf(Item::class, $items[0]);
        $this->assertEquals($expectedItemFormat, $items[0]->getFormat());
        $this->assertEquals([$expectedItemIri], $items[0]->getType());
    }

    /**
     * Test the RDFa Lite 1.1 parser
     */
    public function testRdfaLiteParser()
    {
        $items              = $this->parseItems('rdfa-lite/article-rdfa-lite.html', RdfaLite::class);
        $expectedItemFormat = RdfaLite::FORMAT;
        $expectedItemIri    = new Iri('http://schema.org/', 'NewsArticle');
        $this->assertItemParsedAs($items, $expectedItemFormat, $expectedItemIri);
    }

    /**
     * Test the LinkType parser
     */
    public function testLinkTypeParser()
    {
        $items = $this->parseItems('link-type/valid-test.html', LinkType::class);
        $this->assertTrue(is_array($items));
        $this->assertEquals(4, count($items));
        $this->assertInstanceOf(Item::class, $items[0]);
        $this->assertEquals(LinkType::FORMAT, $items[0]->getFormat());
        $this->assertEquals([new Iri(LinkType::HTML_PROFILE_URI, 'icon')], $items[0]->getType());
    }

    /**
     * Test the JSON-LD parser with a valid recursion
     */
    public function testRecursionInJsonLDParser()
    {
        $items = $this->parseItems('json-ld/jsonld-recursion.html', JsonLD::class);
        $this->assertTrue(is_array($items));
        $this->assertInstanceOf(Item::class, $items[0]);

        $url = $items[0]->getProperty('url');
        $this->assertTrue(is_array($url));
        $this->assertEquals(1, count($url));
        $this->assertInstanceOf(StringValue::class, $url[0]);
        $this->assertEquals('https://www.example.com/', strval($url[0]));
    }
}