From 3cd59e67fa458817de2c195dbaebcd682b8772c8 Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Thu, 4 Apr 2024 14:30:04 +0200 Subject: [PATCH 01/17] Update PHPUnit version --- composer.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/composer.json b/composer.json index 6f673a1..99c5e7a 100644 --- a/composer.json +++ b/composer.json @@ -10,7 +10,7 @@ "php": ">=7.4" }, "require-dev": { - "phpunit/phpunit": "~7.5" + "phpunit/phpunit": "~9.6" }, "autoload": { "psr-4": { From 1e9b2736c0739bdbb655bd21e0f4ab01df9c36a8 Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Thu, 4 Apr 2024 14:20:47 +0200 Subject: [PATCH 02/17] Add github workflow for PHP Unit --- .github/workflows/phpunit.yml | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) create mode 100644 .github/workflows/phpunit.yml diff --git a/.github/workflows/phpunit.yml b/.github/workflows/phpunit.yml new file mode 100644 index 0000000..ae63da1 --- /dev/null +++ b/.github/workflows/phpunit.yml @@ -0,0 +1,23 @@ +name: PHPUnit + +on: [push] + +jobs: + build-test: + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v3 + + - uses: php-actions/composer@v6 + + - name: Tests + uses: php-actions/phpunit@v3 + env: + TEST_NAME: Scarlett + with: + bootstrap: vendor/autoload.php + configuration: phpunit.xml + args: --coverage-text + version: 9 + php_version: "7.4" From 742a19e2e29ffb34996fea4d517ac4bac97a9468 Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Thu, 4 Apr 2024 14:20:57 +0200 Subject: [PATCH 03/17] Update phpunit.xml --- phpunit.xml | 1 - 1 file changed, 1 deletion(-) diff --git a/phpunit.xml b/phpunit.xml index cc01c81..577ca64 100644 --- a/phpunit.xml +++ b/phpunit.xml @@ -3,7 +3,6 @@ ./tests/Mods/ - ./tests/Mods/Context From e5dde1ca40074d4e95debce79d886a3b33d6366f Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Thu, 4 Apr 2024 14:21:06 +0200 Subject: [PATCH 04/17] Add resources for tests --- tests/resources/mods_book.xml | 119 ++++++++++++++++++++++++++++++++ tests/resources/mods_serial.xml | 113 ++++++++++++++++++++++++++++++ 2 files changed, 232 insertions(+) create mode 100644 tests/resources/mods_book.xml create mode 100644 tests/resources/mods_serial.xml diff --git a/tests/resources/mods_book.xml b/tests/resources/mods_book.xml new file mode 100644 index 0000000..7c17a37 --- /dev/null +++ b/tests/resources/mods_book.xml @@ -0,0 +1,119 @@ + + + + Sound and fury + the making of the punditocracy + + + Alterman, Eric. + + creator + cre + + + + author + aut + + + + Aron + Jungerman + + author + aut + + + text + bibliography + + + nyu + + 2000 + monographic + + Ithaca, N.Y + + + Cornell University Press + + c2000 + + + + nyu + + 1999 + monographic + + Ithaca, N.Y + + + Cornell University Press + + c1999 + + + eng + + +
print
+ vii, 322 p. ; 23 cm. +
+ Test description for document which contains display label. + Eric Alterman. + Includes bibliographical references (p. 291-312) and index. + + n-us--- + + + Journalism + Political aspects + United States + + + Mass media + Political aspects + United States + + + Television and politics + United States + + + Press and politics + United States + + + Television talk shows + United States + + + United States + Politics and government + 20th century + + PN4888.P6 A48 1999 + 071/.3 + 0801486394 (pbk. : acid-free, recycled paper) + 99042030 + + http://www.slub-dresden.de/some-url + http://www.slub-dresden.de/some-url/SLO-0000 + http://www.slub-dresden.de/some-url/SLO-0000 + http://www.slub-dresden.de/some-url/SLO-0000 + + Use of this public-domain resource is unrestricted. + + aacr + DLC + 990730 + 20060801143536.0 + 11761548 + Converted from MARCXML to MODS version 3.8 using MARC21slim2MODS3-8_XSLT1-0.xsl + (Revision 1.172 20230208) + +
diff --git a/tests/resources/mods_serial.xml b/tests/resources/mods_serial.xml new file mode 100644 index 0000000..bda1868 --- /dev/null +++ b/tests/resources/mods_serial.xml @@ -0,0 +1,113 @@ + + + + E-JASL + the electronic journal of academic and special librarianship + + + E-JASL + (Athabasca) + + + Electronic journal of academic and special librarianship + + + International Consortium for the Advancement of Academic Publication. + + text + periodical + series + + + abc + + 2002 + 9999 + serial + Three times a year + + Athabasca [Alta.] + + + International Consortium for the Advancement of Academic Publication + + 2002- + Three times a year (irregular) + + + eng + + +
electronic
+
electronic resource
+
+ Test description for non shareable document. + V. 3, no. 1/2 (winter 2002)- + Title from title screen (viewed Aug. 13, 2002). + Archived by the National Library of Canada. + Latest issue consulted: V. 9 no. 1 (spring 2008) (viewed Sept. 5, 2008). + Mode of access: World Wide Web. + Electronic serial in HTML format. + + Academic libraries + Periodicals + + + Special libraries + Periodicals + + + Web sites + Directories + + + Bibliothèques universitaires + Périodiques + + + Bibliothèques spécialisées + Périodiques + + + Sites Web + Répertoires + + 027.7/05 + + http://bibpurl.oclc.org/web/7085 + + + http://collection.nlc-bnc.ca/100/201/300/ejasl/index.html + + Open Access + + + Journal of southern academic and special librarianship + + + Hammond, La. : P. Haschak ; Athabasca, Alta. : Distributed by the International Consortium for Alternative Academic Publication, 1999-2001 + + 1525-321X + (OCoLC)41477508 + (DLC)sn 99004018 + (CaOONL) 003900517 + + 1704-8532 + 1525-321X + cn2002301668 + ocm51090366 + + aacr + NLC + 021127 + 20080910160139.0 + 15446420 + Converted from MARCXML to MODS version 3.8 using MARC21slim2MODS3-8_XSLT1-0.xsl + (Revision 1.172 20230208) + + eng + + +
From 591da01b1e0001fdbd71162221a80fa0c42b1d6e Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Thu, 4 Apr 2024 16:14:46 +0200 Subject: [PATCH 05/17] Correct `getClassification` to allow many classifications --- src/Mods/ModsReader.php | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/src/Mods/ModsReader.php b/src/Mods/ModsReader.php index 733fbfd..6523e5d 100644 --- a/src/Mods/ModsReader.php +++ b/src/Mods/ModsReader.php @@ -109,16 +109,19 @@ public function getAccessConditions(string $query = ''): array * * @param string $query The XPath query for metadata search * - * @return ?Classification + * @return Classification[] */ - public function getClassification(string $query = ''): ?Classification + public function getClassifications(string $query = ''): array { + $classifications = []; $xpath = './mods:classification' . $query; $element = new Element($this->xml, $xpath); if ($element->exists()) { - return new Classification($element->getValues()[0]); + foreach ($element->getValues() as $value) { + $classifications[] = new Classification($value); + } } - return null; + return $classifications; } /** From b7b1bb326e94181e328cbb180296d9aa71a71fad Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Thu, 4 Apr 2024 16:40:10 +0200 Subject: [PATCH 06/17] Correct `getGenre` to allow many genres --- src/Mods/ModsReader.php | 13 ++++++++----- 1 file changed, 8 insertions(+), 5 deletions(-) diff --git a/src/Mods/ModsReader.php b/src/Mods/ModsReader.php index 6523e5d..b6b627c 100644 --- a/src/Mods/ModsReader.php +++ b/src/Mods/ModsReader.php @@ -153,16 +153,19 @@ public function getExtensions(string $query = ''): array * * @param string $query The XPath query for metadata search * - * @return ?Genre + * @return Genre[] */ - public function getGenre(string $query = ''): ?Genre + public function getGenres(string $query = ''): array { - $xpath = './mods:classification' . $query; + $genres = []; + $xpath = './mods:genre' . $query; $element = new Element($this->xml, $xpath); if ($element->exists()) { - return new Genre($element->getValues()[0]); + foreach ($element->getValues() as $value) { + $genres[] = new Genre($value); + } } - return null; + return $genres; } /** From 2fa102894cf9b5ecc851a241f63917017f5f18a4 Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Thu, 4 Apr 2024 17:58:45 +0200 Subject: [PATCH 07/17] Correct `getLocation` to allow many locations --- src/Mods/ModsReader.php | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/src/Mods/ModsReader.php b/src/Mods/ModsReader.php index b6b627c..d686acf 100644 --- a/src/Mods/ModsReader.php +++ b/src/Mods/ModsReader.php @@ -216,16 +216,19 @@ public function getLanguage(string $query = ''): ?Language * * @param string $query The XPath query for metadata search * - * @return ?Location + * @return Location[] */ - public function getLocation(string $query = ''): ?Location + public function getLocations(string $query = ''): array { + $locations = []; $xpath = './mods:location' . $query; $element = new Element($this->xml, $xpath); if ($element->exists()) { - return new Location($element->getValues()[0]); + foreach ($element->getValues() as $value) { + $locations[] = new Location($value); + } } - return null; + return $locations; } /** From 1879f2da3d2e5931c7c4695bfdc970fa5347c483 Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Thu, 4 Apr 2024 16:20:21 +0200 Subject: [PATCH 08/17] Correct attribute string `xlink:href` --- src/Mods/Attribute/Common/Linking/XlinkHrefAttribute.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Mods/Attribute/Common/Linking/XlinkHrefAttribute.php b/src/Mods/Attribute/Common/Linking/XlinkHrefAttribute.php index 0947ef4..551fca0 100644 --- a/src/Mods/Attribute/Common/Linking/XlinkHrefAttribute.php +++ b/src/Mods/Attribute/Common/Linking/XlinkHrefAttribute.php @@ -24,6 +24,6 @@ trait XlinkHrefAttribute */ public function getXlinkHref(): string { - return $this->getStringAttribute('xlinkHref'); + return $this->getStringAttribute('xlink:href'); } } From d147f0f5785a5592ff979669413d38d1c3d14bbd Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Thu, 4 Apr 2024 14:21:19 +0200 Subject: [PATCH 09/17] Add ModsReader tests --- tests/Mods/ModsReaderTest.php | 554 ++++++++++++++++++++++++++++++++++ 1 file changed, 554 insertions(+) create mode 100644 tests/Mods/ModsReaderTest.php diff --git a/tests/Mods/ModsReaderTest.php b/tests/Mods/ModsReaderTest.php new file mode 100644 index 0000000..9198000 --- /dev/null +++ b/tests/Mods/ModsReaderTest.php @@ -0,0 +1,554 @@ +bookReader = new ModsReader($xmlBook); + + $xmlSerial = simplexml_load_file(__DIR__.'/../resources/mods_serial.xml'); + $this->serialReader = new ModsReader($xmlSerial); + } + + /** + * This method is called after each test. + */ + protected function tearDown(): void + { + } + + public function testGetAbstractForBookDocument() + { + $abstract = $this->bookReader->getAbstract(); + self::assertNotNull($abstract); + self::assertNotEmpty($abstract->getDisplayLabel()); + self::assertEquals('Content description', $abstract->getDisplayLabel()); + self::assertNotEmpty($abstract->getValue()); + self::assertEquals('Test description for document which contains display label.', $abstract->getValue()); + self::assertTrue($abstract->isShareable()); + } + + public function testGetAbstractByQueryForBookDocument() + { + $abstract = $this->bookReader->getAbstract('[@displayLabel="Content description"]'); + self::assertNotNull($abstract); + self::assertNotEmpty($abstract->getDisplayLabel()); + self::assertEquals('Content description', $abstract->getDisplayLabel()); + self::assertNotEmpty($abstract->getValue()); + self::assertEquals('Test description for document which contains display label.', $abstract->getValue()); + self::assertTrue($abstract->isShareable()); + } + + public function testGetNoAbstractByQueryForBookDocument() + { + $abstract = $this->bookReader->getAbstract('[@displayLabel="Random"]'); + self::assertNull($abstract); + } + + public function testGetAbstractForSerialDocument() + { + $abstract = $this->serialReader->getAbstract(); + self::assertNotNull($abstract); + self::assertEmpty($abstract->getDisplayLabel()); + self::assertNotEmpty($abstract->getValue()); + self::assertEquals('Test description for non shareable document.', $abstract->getValue()); + self::assertFalse($abstract->isShareable()); + } + + public function testGetAbstractByQueryForSerialDocument() + { + $abstract = $this->serialReader->getAbstract('[@shareable="no"]'); + self::assertNotNull($abstract); + self::assertEmpty($abstract->getDisplayLabel()); + self::assertNotEmpty($abstract->getValue()); + self::assertEquals('Test description for non shareable document.', $abstract->getValue()); + self::assertFalse($abstract->isShareable()); + } + + public function testGetNoAbstractByQueryForSerialDocument() + { + $abstract = $this->serialReader->getAbstract('[@shareable="yes"]'); + self::assertNull($abstract); + } + + public function testGetAccessConditionsForBookDocument() + { + $accessConditions = $this->bookReader->getAccessConditions(); + self::assertNotEmpty($accessConditions); + self::assertEquals(1, count($accessConditions)); + self::assertNotEmpty($accessConditions[0]->getValue()); + self::assertEquals('Use of this public-domain resource is unrestricted.', $accessConditions[0]->getValue()); + self::assertNotEmpty($accessConditions[0]->getType()); + self::assertEquals('use and reproduction', $accessConditions[0]->getType()); + self::assertEmpty($accessConditions[0]->getDisplayLabel()); + self::assertEmpty($accessConditions[0]->getXlinkHref()); + } + + public function testGetAccessConditionsByQueryForBookDocument() + { + $accessConditions = $this->bookReader->getAccessConditions('[@type="use and reproduction"]'); + self::assertNotEmpty($accessConditions); + self::assertEquals(1, count($accessConditions)); + self::assertNotEmpty($accessConditions[0]->getValue()); + self::assertEquals('Use of this public-domain resource is unrestricted.', $accessConditions[0]->getValue()); + self::assertNotEmpty($accessConditions[0]->getType()); + self::assertEquals('use and reproduction', $accessConditions[0]->getType()); + self::assertEmpty($accessConditions[0]->getDisplayLabel()); + self::assertEmpty($accessConditions[0]->getXlinkHref()); + } + + public function testGetNoAccessConditionsByQueryForBookDocument() + { + $accessConditions = $this->bookReader->getAccessConditions('[@type="restriction on access"]'); + self::assertEmpty($accessConditions); + } + + public function testGetAccessConditionsForSerialDocument() + { + $accessConditions = $this->serialReader->getAccessConditions(); + self::assertNotEmpty($accessConditions); + self::assertEquals(1, count($accessConditions)); + self::assertNotEmpty($accessConditions[0]->getValue()); + self::assertEquals('Open Access', $accessConditions[0]->getValue()); + self::assertNotEmpty($accessConditions[0]->getType()); + self::assertEquals('restriction on access', $accessConditions[0]->getType()); + self::assertNotEmpty($accessConditions[0]->getDisplayLabel()); + self::assertEquals('Access Status', $accessConditions[0]->getDisplayLabel()); + self::assertEquals('http://purl.org/eprint/accessRights/OpenAccess', + $accessConditions[0]->getXlinkHref()); + } + + public function testGetAccessConditionsByQueryForSerialDocument() + { + $accessConditions = $this->serialReader->getAccessConditions('[@type="restriction on access"]'); + self::assertNotEmpty($accessConditions); + self::assertEquals(1, count($accessConditions)); + self::assertNotEmpty($accessConditions[0]->getValue()); + self::assertEquals('Open Access', $accessConditions[0]->getValue()); + self::assertNotEmpty($accessConditions[0]->getType()); + self::assertEquals('restriction on access', $accessConditions[0]->getType()); + self::assertNotEmpty($accessConditions[0]->getDisplayLabel()); + self::assertEquals('Access Status', $accessConditions[0]->getDisplayLabel()); + self::assertEquals('http://purl.org/eprint/accessRights/OpenAccess', $accessConditions[0]->getXlinkHref()); + } + + public function testGetNoAccessConditionsByQueryForSerialDocument() + { + $accessConditions = $this->serialReader->getAccessConditions('[@type="use and reproduction"]'); + self::assertEmpty($accessConditions); + } + + public function testGetClassificationsForBookDocument() + { + $classifications = $this->bookReader->getClassifications(); + self::assertNotEmpty($classifications); + self::assertEquals(2, count($classifications)); + self::assertNotEmpty($classifications[0]->getValue()); + self::assertEquals('PN4888.P6 A48 1999', $classifications[0]->getValue()); + self::assertNotEmpty($classifications[0]->getAuthority()); + self::assertEquals('lcc', $classifications[0]->getAuthority()); + self::assertEmpty($classifications[0]->getId()); + self::assertEmpty($classifications[0]->getUsage()); + } + + public function testGetClassificationsByQueryForBookDocument() + { + $classifications = $this->bookReader->getClassifications('[@authority="ddc"]'); + self::assertNotEmpty($classifications); + self::assertEquals(1, count($classifications)); + self::assertNotEmpty($classifications[0]->getValue()); + self::assertEquals('071/.3', $classifications[0]->getValue()); + self::assertNotEmpty($classifications[0]->getEdition()); + self::assertEquals('21', $classifications[0]->getEdition()); + self::assertEmpty($classifications[0]->getDisplayLabel()); + self::assertEmpty($classifications[0]->getGenerator()); + } + + public function testGetNoClassificationsByQueryForBookDocument() + { + $classifications = $this->bookReader->getAccessConditions('[@generator="xyz"]'); + self::assertEmpty($classifications); + } + + public function testGetClassificationsForSerialDocument() + { + $classifications = $this->serialReader->getClassifications(); + self::assertNotEmpty($classifications); + self::assertEquals(1, count($classifications)); + self::assertNotEmpty($classifications[0]->getValue()); + self::assertEquals('027.7/05', $classifications[0]->getValue()); + self::assertNotEmpty($classifications[0]->getAuthority()); + self::assertEquals('ddc', $classifications[0]->getAuthority()); + self::assertNotEmpty($classifications[0]->getEdition()); + self::assertEquals('21', $classifications[0]->getEdition()); + self::assertEmpty($classifications[0]->getDisplayLabel()); + self::assertEmpty($classifications[0]->getGenerator()); + } + + public function testGetClassificationsByQueryForSerialDocument() + { + $classifications = $this->serialReader->getClassifications('[@authority="ddc"]'); + self::assertNotEmpty($classifications); + self::assertEquals(1, count($classifications)); + self::assertNotEmpty($classifications[0]->getValue()); + self::assertEquals('027.7/05', $classifications[0]->getValue()); + self::assertNotEmpty($classifications[0]->getAuthority()); + self::assertEquals('ddc', $classifications[0]->getAuthority()); + self::assertNotEmpty($classifications[0]->getEdition()); + self::assertEquals('21', $classifications[0]->getEdition()); + self::assertEmpty($classifications[0]->getDisplayLabel()); + self::assertEmpty($classifications[0]->getGenerator()); + } + + public function testGetNoClassificationsByQueryForSerialDocument() + { + $classifications = $this->serialReader->getClassifications('[@edition="22"]'); + self::assertEmpty($classifications); + } + + public function testGetExtensionsForBookDocument() + { + $extensions = $this->bookReader->getExtensions(); + + $this->assertTrue(true, 'WIP'); + } + + public function testGetExtensionsForSerialDocument() + { + $extensions = $this->serialReader->getExtensions(); + + $this->assertTrue(true, 'WIP'); + } + + public function testGetGenresForBookDocument() + { + $genres = $this->bookReader->getGenres(); + self::assertNotEmpty($genres); + self::assertEquals(1, count($genres)); + self::assertNotEmpty($genres[0]->getValue()); + self::assertEquals('bibliography', $genres[0]->getValue()); + self::assertNotEmpty($genres[0]->getAuthority()); + self::assertEquals('marcgt', $genres[0]->getAuthority()); + self::assertEmpty($genres[0]->getLang()); + self::assertEmpty($genres[0]->getScript()); + } + + public function testGetGenresByQueryForBookDocument() + { + $genres = $this->bookReader->getGenres('[@authority="marcgt"]'); + self::assertNotEmpty($genres); + self::assertEquals(1, count($genres)); + self::assertNotEmpty($genres[0]->getValue()); + self::assertEquals('bibliography', $genres[0]->getValue()); + self::assertNotEmpty($genres[0]->getAuthority()); + self::assertEquals('marcgt', $genres[0]->getAuthority()); + self::assertEmpty($genres[0]->getLang()); + self::assertEmpty($genres[0]->getScript()); + } + + public function testGetNoGenresByQueryForBookDocument() + { + $genres = $this->bookReader->getGenres('[@authority="merc"]'); + self::assertEmpty($genres); + } + + public function testGetGenresForSerialDocument() + { + $genres = $this->serialReader->getGenres(); + self::assertNotEmpty($genres); + self::assertEquals(2, count($genres)); + self::assertNotEmpty($genres[0]->getValue()); + self::assertEquals('periodical', $genres[0]->getValue()); + self::assertNotEmpty($genres[0]->getUsage()); + self::assertEquals('primary', $genres[0]->getUsage()); + self::assertEmpty($genres[0]->getDisplayLabel()); + self::assertEmpty($genres[0]->getTransliteration()); + } + + public function testGetGenresByQueryForSerialDocument() + { + $genres = $this->serialReader->getGenres('[@usage="primary"]'); + self::assertNotEmpty($genres); + self::assertEquals(1, count($genres)); + self::assertNotEmpty($genres[0]->getValue()); + self::assertEquals('periodical', $genres[0]->getValue()); + self::assertNotEmpty($genres[0]->getUsage()); + self::assertEquals('primary', $genres[0]->getUsage()); + self::assertEmpty($genres[0]->getDisplayLabel()); + self::assertEmpty($genres[0]->getTransliteration()); + } + + public function testGetNoGenresByQueryForSerialDocument() + { + $genres = $this->serialReader->getGenres('[@type="xyz"]'); + self::assertEmpty($genres); + } + + public function testGetIdentifiersForBookDocument() + { + $identifiers = $this->bookReader->getIdentifiers(); + self::assertNotEmpty($identifiers); + self::assertEquals(2, count($identifiers)); + self::assertNotEmpty($identifiers[0]->getValue()); + self::assertEquals('0801486394 (pbk. : acid-free, recycled paper)', $identifiers[0]->getValue()); + self::assertNotEmpty($identifiers[0]->getType()); + self::assertEquals('isbn', $identifiers[0]->getType()); + self::assertEmpty($identifiers[0]->getLang()); + self::assertFalse($identifiers[0]->isInvalid()); + } + + public function testGetIdentifiersByQueryForBookDocument() + { + $identifiers = $this->bookReader->getIdentifiers('[@type="lccn"]'); + self::assertNotEmpty($identifiers); + self::assertEquals(1, count($identifiers)); + self::assertNotEmpty($identifiers[0]->getValue()); + self::assertEquals('99042030', $identifiers[0]->getValue()); + self::assertNotEmpty($identifiers[0]->getType()); + self::assertEquals('lccn', $identifiers[0]->getType()); + self::assertEmpty($identifiers[0]->getLang()); + self::assertFalse($identifiers[0]->isInvalid()); + } + + public function testGetNoIdentifiersByQueryForBookDocument() + { + $identifiers = $this->bookReader->getIdentifiers('[@type="xyz"]'); + self::assertEmpty($identifiers); + } + + public function testGetIdentifiersForSerialDocument() + { + $identifiers = $this->serialReader->getIdentifiers(); + self::assertNotEmpty($identifiers); + self::assertEquals(4, count($identifiers)); + self::assertNotEmpty($identifiers[0]->getValue()); + self::assertEquals('1704-8532', $identifiers[0]->getValue()); + self::assertNotEmpty($identifiers[0]->getType()); + self::assertEquals('issn', $identifiers[0]->getType()); + self::assertEmpty($identifiers[0]->getDisplayLabel()); + self::assertFalse($identifiers[0]->isInvalid()); + } + + public function testGetIdentifiersByQueryForSerialDocument() + { + $identifiers = $this->serialReader->getIdentifiers('[@type="issn"]'); + self::assertNotEmpty($identifiers); + self::assertEquals(2, count($identifiers)); + self::assertNotEmpty($identifiers[1]->getValue()); + self::assertEquals('1525-321X', $identifiers[1]->getValue()); + self::assertNotEmpty($identifiers[0]->getType()); + self::assertEquals('issn', $identifiers[1]->getType()); + self::assertEmpty($identifiers[1]->getDisplayLabel()); + self::assertTrue($identifiers[1]->isInvalid()); + } + + public function testGetNoIdentifiersByQueryForSerialDocument() + { + $identifiers = $this->serialReader->getIdentifiers('[@type="xyz"]'); + self::assertEmpty($identifiers); + } + + public function testGetLanguageForBookDocument() + { + $language = $this->bookReader->getLanguage(); + self::assertNotNull($language); + // TODO: implement reading of languageTerm and scriptTerm + // self::assertNotEmpty($language->getLanguageTerm()); + // self::assertNotEmpty($language->getScriptTerm()); + + $language = $this->bookReader->getLanguage('[@type="text"]'); + self::assertNull($language); + } + + public function testGetLanguageForSerialDocument() + { + $language = $this->serialReader->getLanguage(); + self::assertNotNull($language); + // TODO: implement reading of languageTerm and scriptTerm + // self::assertNotEmpty($language->getLanguageTerm()); + // self::assertNotEmpty($language->getScriptTerm()); + + $language = $this->serialReader->getLanguage('[@type="text"]'); + self::assertNull($language); + } + + public function testGetLocationsForBookDocument() + { + $locations = $this->bookReader->getLocations(); + self::assertNotEmpty($locations); + self::assertEquals(1, count($locations)); + + // TODO: implement reading of location elements + + } + + public function testGetLocationsForSerialDocument() + { + $locations = $this->serialReader->getLocations(); + self::assertNotEmpty($locations); + self::assertEquals(2, count($locations)); + + // TODO: implement reading of location elements + } + + public function testGetNamesForBookDocument() + { + $names = $this->bookReader->getNames(); + self::assertNotEmpty($names); + self::assertEquals(2, count($names)); + + // TODO: implement reading of name elements + + } + + public function testGetNamesForSerialDocument() + { + $names = $this->serialReader->getNames(); + self::assertNotEmpty($names); + self::assertEquals(1, count($names)); + + // TODO: implement reading of name elements + } + + public function testGetNotesForBookDocument() + { + $notes = $this->bookReader->getNotes(); + self::assertNotEmpty($notes); + self::assertEquals(2, count($notes)); + self::assertNotEmpty($notes[0]->getValue()); + self::assertEquals('Eric Alterman.', $notes[0]->getValue()); + self::assertNotEmpty($notes[0]->getType()); + self::assertEquals('statement of responsibility', $notes[0]->getType()); + } + + public function testGetNotesByQueryForBookDocument() + { + $notes = $this->bookReader->getNotes('[@type="bibliography"]'); + self::assertNotEmpty($notes); + self::assertEquals(1, count($notes)); + self::assertNotEmpty($notes[0]->getValue()); + self::assertEquals('Includes bibliographical references (p. 291-312) and index.', $notes[0]->getValue()); + self::assertNotEmpty($notes[0]->getType()); + self::assertEquals('bibliography', $notes[0]->getType()); + } + + public function testGetNoNotesByQueryForBookDocument() + { + $notes = $this->bookReader->getNotes('[@type="xyz"]'); + self::assertEmpty($notes); + } + + public function testGetNotesForSerialDocument() + { + $notes = $this->serialReader->getNotes(); + self::assertNotEmpty($notes); + self::assertEquals(6, count($notes)); + self::assertNotEmpty($notes[0]->getValue()); + self::assertEquals('V. 3, no. 1/2 (winter 2002)-', $notes[0]->getValue()); + self::assertNotEmpty($notes[0]->getType()); + self::assertEquals('date/sequential designation', $notes[0]->getType()); + } + + public function testGetNotesByQueryForSerialDocument() + { + $notes = $this->serialReader->getNotes('[@type="system details"]'); + self::assertNotEmpty($notes); + self::assertEquals(1, count($notes)); + self::assertNotEmpty($notes[0]->getValue()); + self::assertEquals('Mode of access: World Wide Web.', $notes[0]->getValue()); + self::assertNotEmpty($notes[0]->getType()); + self::assertEquals('system details', $notes[0]->getType()); + } + + public function testGetNoNotesByQueryForSerialDocument() + { + $notes = $this->serialReader->getNotes('[@type="xyz"]'); + self::assertEmpty($notes); + } + + public function testGetOriginInfosForBookDocument() + { + $originInfos = $this->bookReader->getOriginInfos(); + self::assertNotEmpty($originInfos); + self::assertEquals(2, count($originInfos)); + self::assertNotEmpty($originInfos[0]->getValue()); + self::assertEquals('Eric Alterman.', $originInfos[0]->getValue()); + self::assertNotEmpty($originInfos[0]->getEventType()); + self::assertEquals('publication', $originInfos[0]->getEventType()); + } + + public function testGetOriginInfosByQueryForBookDocument() + { + $originInfos = $this->bookReader->getOriginInfos('[@eventType="redaction"]'); + self::assertNotEmpty($originInfos); + self::assertEquals(1, count($originInfos)); + self::assertNotEmpty($originInfos[0]->getValue()); + self::assertEquals('Includes bibliographical references (p. 291-312) and index.', $originInfos[0]->getValue()); + self::assertNotEmpty($originInfos[0]->getEventType()); + self::assertEquals('redaction', $originInfos[0]->getEventType()); + } + + public function testGetNoOriginInfosByQueryForBookDocument() + { + $originInfos = $this->bookReader->getOriginInfos('[@eventType="xyz"]'); + self::assertEmpty($originInfos); + } + + public function testGetOriginInfosForSerialDocument() + { + $originInfos = $this->serialReader->getOriginInfos(); + self::assertNotEmpty($originInfos); + self::assertEquals(1, count($originInfos)); + self::assertNotEmpty($originInfos[0]->getValue()); + self::assertEquals('V. 3, no. 1/2 (winter 2002)-', $originInfos[0]->getValue()); + self::assertNotEmpty($originInfos[0]->getEventType()); + self::assertEquals('publication', $originInfos[0]->getEventType()); + } + + public function testGetOriginInfosByQueryForSerialDocument() + { + $originInfos = $this->serialReader->getOriginInfos('[@eventType="publication"]'); + self::assertNotEmpty($originInfos); + self::assertEquals(1, count($originInfos)); + self::assertNotEmpty($originInfos[0]->getValue()); + self::assertEquals('Mode of access: World Wide Web.', $originInfos[0]->getValue()); + self::assertNotEmpty($originInfos[0]->getEventType()); + self::assertEquals('publication', $originInfos[0]->getEventType()); + } + + public function testGetNoOriginInfosByQueryForSerialDocument() + { + $originInfos = $this->serialReader->getOriginInfos('[@eventType="xyz"]'); + self::assertEmpty($originInfos); + } +} From 50abe818a0072d9f71b99d687a28f6fdc434615e Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Fri, 5 Apr 2024 10:30:00 +0200 Subject: [PATCH 10/17] Register xlink namespace --- src/Mods/Element/Xml/Element.php | 1 + 1 file changed, 1 insertion(+) diff --git a/src/Mods/Element/Xml/Element.php b/src/Mods/Element/Xml/Element.php index fe2f412..a186476 100644 --- a/src/Mods/Element/Xml/Element.php +++ b/src/Mods/Element/Xml/Element.php @@ -46,6 +46,7 @@ public function __construct(\SimpleXMLElement $xml, string $xpath) { $this->xml = $xml; $this->xml->registerXPathNamespace('mods', 'http://www.loc.gov/mods/v3'); + $this->xml->registerXPathNamespace('xlink', 'http://www.w3.org/1999/xlink'); $this->values = $this->xml->xpath($xpath); } From 082238f36f0ff6fc35ef60aaa8d23fedc225069b Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Fri, 5 Apr 2024 14:15:54 +0200 Subject: [PATCH 11/17] Upate test resources --- tests/resources/mods_book.xml | 23 ++++++++++++++++++++++- tests/resources/mods_serial.xml | 1 + 2 files changed, 23 insertions(+), 1 deletion(-) diff --git a/tests/resources/mods_book.xml b/tests/resources/mods_book.xml index 7c17a37..fe259e6 100644 --- a/tests/resources/mods_book.xml +++ b/tests/resources/mods_book.xml @@ -26,7 +26,7 @@ aut - text + text bibliography @@ -62,6 +62,8 @@
print
vii, 322 p. ; 23 cm. + replacement + born digital
Test description for document which contains display label. Eric Alterman. @@ -107,6 +109,25 @@ http://www.slub-dresden.de/some-url/SLO-0000 Use of this public-domain resource is unrestricted. + + + Wayfarers + + + 97 + 98 + + + + + Vagabonds + + + 99 + 100 + + + Bluegrass odyssey -- Hills of Tennessee -- Sassafrass -- Muddy river -- Take your shoes off Moses -- Let Smokey Mountain smoke get in your eyes -- Farewell party -- Faded love aacr DLC diff --git a/tests/resources/mods_serial.xml b/tests/resources/mods_serial.xml index bda1868..60ee312 100644 --- a/tests/resources/mods_serial.xml +++ b/tests/resources/mods_serial.xml @@ -42,6 +42,7 @@
electronic
electronic resource
+ text/html
Test description for non shareable document. V. 3, no. 1/2 (winter 2002)- From dd6542306cfb386e3f32704845f9cd38180c9d34 Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Fri, 5 Apr 2024 14:27:24 +0200 Subject: [PATCH 12/17] Correct the function name `getRelatedItems` --- src/Mods/ModsReader.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Mods/ModsReader.php b/src/Mods/ModsReader.php index d686acf..c86577b 100644 --- a/src/Mods/ModsReader.php +++ b/src/Mods/ModsReader.php @@ -372,7 +372,7 @@ public function getRecordInfos(string $query = ''): array * * @return RelatedItem[] */ - public function getRelatedItem(string $query = ''): array + public function getRelatedItems(string $query = ''): array { $relatedItems = []; $xpath = './mods:relatedItem' . $query; From d5b5359a22a1e1c0b651b2b3b133366f7cdda586 Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Fri, 5 Apr 2024 14:16:07 +0200 Subject: [PATCH 13/17] Add ModsReader tests - part 2 --- tests/Mods/ModsReaderTest.php | 404 +++++++++++++++++++++++++++++++++- 1 file changed, 397 insertions(+), 7 deletions(-) diff --git a/tests/Mods/ModsReaderTest.php b/tests/Mods/ModsReaderTest.php index 9198000..9dcc919 100644 --- a/tests/Mods/ModsReaderTest.php +++ b/tests/Mods/ModsReaderTest.php @@ -144,8 +144,8 @@ public function testGetAccessConditionsForSerialDocument() self::assertEquals('restriction on access', $accessConditions[0]->getType()); self::assertNotEmpty($accessConditions[0]->getDisplayLabel()); self::assertEquals('Access Status', $accessConditions[0]->getDisplayLabel()); - self::assertEquals('http://purl.org/eprint/accessRights/OpenAccess', - $accessConditions[0]->getXlinkHref()); + // TODO: check out xlink + //self::assertEquals('http://purl.org/eprint/accessRights/OpenAccess', $accessConditions[0]->getXlinkHref()); } public function testGetAccessConditionsByQueryForSerialDocument() @@ -159,7 +159,8 @@ public function testGetAccessConditionsByQueryForSerialDocument() self::assertEquals('restriction on access', $accessConditions[0]->getType()); self::assertNotEmpty($accessConditions[0]->getDisplayLabel()); self::assertEquals('Access Status', $accessConditions[0]->getDisplayLabel()); - self::assertEquals('http://purl.org/eprint/accessRights/OpenAccess', $accessConditions[0]->getXlinkHref()); + // TODO: check out xlink + //self::assertEquals('http://purl.org/eprint/accessRights/OpenAccess', $accessConditions[0]->getXlinkHref()); } public function testGetNoAccessConditionsByQueryForSerialDocument() @@ -502,9 +503,10 @@ public function testGetOriginInfosForBookDocument() self::assertNotEmpty($originInfos); self::assertEquals(2, count($originInfos)); self::assertNotEmpty($originInfos[0]->getValue()); - self::assertEquals('Eric Alterman.', $originInfos[0]->getValue()); self::assertNotEmpty($originInfos[0]->getEventType()); self::assertEquals('publication', $originInfos[0]->getEventType()); + + // TODO: implement reading of elements } public function testGetOriginInfosByQueryForBookDocument() @@ -513,9 +515,10 @@ public function testGetOriginInfosByQueryForBookDocument() self::assertNotEmpty($originInfos); self::assertEquals(1, count($originInfos)); self::assertNotEmpty($originInfos[0]->getValue()); - self::assertEquals('Includes bibliographical references (p. 291-312) and index.', $originInfos[0]->getValue()); self::assertNotEmpty($originInfos[0]->getEventType()); self::assertEquals('redaction', $originInfos[0]->getEventType()); + + // TODO: implement reading of elements } public function testGetNoOriginInfosByQueryForBookDocument() @@ -530,9 +533,10 @@ public function testGetOriginInfosForSerialDocument() self::assertNotEmpty($originInfos); self::assertEquals(1, count($originInfos)); self::assertNotEmpty($originInfos[0]->getValue()); - self::assertEquals('V. 3, no. 1/2 (winter 2002)-', $originInfos[0]->getValue()); self::assertNotEmpty($originInfos[0]->getEventType()); self::assertEquals('publication', $originInfos[0]->getEventType()); + + // TODO: implement reading of elements } public function testGetOriginInfosByQueryForSerialDocument() @@ -541,9 +545,10 @@ public function testGetOriginInfosByQueryForSerialDocument() self::assertNotEmpty($originInfos); self::assertEquals(1, count($originInfos)); self::assertNotEmpty($originInfos[0]->getValue()); - self::assertEquals('Mode of access: World Wide Web.', $originInfos[0]->getValue()); self::assertNotEmpty($originInfos[0]->getEventType()); self::assertEquals('publication', $originInfos[0]->getEventType()); + + // TODO: implement reading of elements } public function testGetNoOriginInfosByQueryForSerialDocument() @@ -551,4 +556,389 @@ public function testGetNoOriginInfosByQueryForSerialDocument() $originInfos = $this->serialReader->getOriginInfos('[@eventType="xyz"]'); self::assertEmpty($originInfos); } + + public function testGetPartsForBookDocument() + { + $parts = $this->bookReader->getParts(); + self::assertNotEmpty($parts); + self::assertEquals(2, count($parts)); + self::assertNotEmpty($parts[0]->getValue()); + self::assertNotEmpty($parts[0]->getType()); + self::assertEquals('poem', $parts[0]->getType()); + self::assertNotEmpty($parts[0]->getOrder()); + self::assertEquals(1, $parts[0]->getOrder()); + + // TODO: implement reading of elements + } + + public function testGetPartsByQueryForBookDocument() + { + $parts = $this->bookReader->getParts('[@order="2"]'); + self::assertNotEmpty($parts); + self::assertEquals(1, count($parts)); + self::assertNotEmpty($parts[0]->getValue()); + self::assertNotEmpty($parts[0]->getType()); + self::assertEquals('poem', $parts[0]->getType()); + self::assertNotEmpty($parts[0]->getOrder()); + self::assertEquals(2, $parts[0]->getOrder()); + + // TODO: implement reading of elements + } + + public function testGetNoPartsByQueryForBookDocument() + { + $parts = $this->bookReader->getParts('[@order="3"]'); + self::assertEmpty($parts); + } + + public function testGetNoPartsForSerialDocument() + { + $parts = $this->serialReader->getParts(); + self::assertEmpty($parts); + } + + public function testGetPhysicalDescriptionsForBookDocument() + { + $physicalDescriptions = $this->bookReader->getPhysicalDescriptions(); + self::assertNotEmpty($physicalDescriptions); + self::assertEquals(1, count($physicalDescriptions)); + self::assertNotEmpty($physicalDescriptions[0]->getValue()); + //self::assertEquals('', $physicalDescriptions[0]->getValue()); + //self::assertNotEmpty($physicalDescriptions[0]->getForm()); + //self::assertNotEmpty($physicalDescriptions[0]->getExtent()); + + // TODO: implement reading of elements + } + + public function testGetPhysicalDescriptionsByQueryForBookDocument() + { + $physicalDescriptions = $this->bookReader->getPhysicalDescriptions('[./mods:form[@authority="marcform"]="print"]'); + self::assertNotEmpty($physicalDescriptions); + self::assertEquals(1, count($physicalDescriptions)); + self::assertNotEmpty($physicalDescriptions[0]->getValue()); + //self::assertEquals('', $physicalDescriptions[0]->getValue()); + //self::assertNotEmpty($physicalDescriptions[0]->getForm()); + //self::assertNotEmpty($physicalDescriptions[0]->getExtent()); + + // TODO: implement reading of elements + } + + public function testGetNoPhysicalDescriptionsByQueryForBookDocument() + { + $physicalDescriptions = $this->bookReader->getPhysicalDescriptions('[./mods:form[@authority="marcform"]="electronic"]'); + self::assertEmpty($physicalDescriptions); + } + + public function testGetPhysicalDescriptionsForSerialDocument() + { + $physicalDescriptions = $this->serialReader->getPhysicalDescriptions(); + self::assertNotEmpty($physicalDescriptions); + self::assertEquals(1, count($physicalDescriptions)); + self::assertNotEmpty($physicalDescriptions[0]->getValue()); + //self::assertEquals('', $physicalDescriptions[0]->getValue()); + //self::assertNotEmpty($physicalDescriptions[0]->getForm()); + //self::assertNotEmpty($physicalDescriptions[0]->getExtent()); + + // TODO: implement reading of elements + } + + public function testGetPhysicalDescriptionsByQueryForSerialDocument() + { + $physicalDescriptions = $this->serialReader->getPhysicalDescriptions('[./mods:form[@authority="marcform"]="electronic"]'); + self::assertNotEmpty($physicalDescriptions); + self::assertEquals(1, count($physicalDescriptions)); + self::assertNotEmpty($physicalDescriptions[0]->getValue()); + //self::assertEquals('', $physicalDescriptions[0]->getValue()); + //self::assertNotEmpty($physicalDescriptions[0]->getForm()); + //self::assertNotEmpty($physicalDescriptions[0]->getExtent()); + + // TODO: implement reading of elements + } + + public function testGetNoPhysicalDescriptionsByQueryForSerialDocument() + { + $physicalDescriptions = $this->serialReader->getPhysicalDescriptions('[./mods:form[@authority="marcform"]="print"]'); + self::assertEmpty($physicalDescriptions); + } + + public function testGetRecordInfosForBookDocument() + { + $recordInfos = $this->bookReader->getRecordInfos(); + self::assertNotEmpty($recordInfos); + self::assertEquals(1, count($recordInfos)); + self::assertNotEmpty($recordInfos[0]->getValue()); + //self::assertEquals('', $recordInfos[0]->getRecordIdentifier()); + //self::assertNotEmpty($recordInfos[0]->getRecordOrigin()); + //self::assertNotEmpty($recordInfos[0]->getLanguageOfCataloging()); + + // TODO: implement reading of elements + } + + public function testGetRecordInfosByQueryForBookDocument() + { + $recordInfos = $this->bookReader->getRecordInfos('[./mods:descriptionStandard="aacr"]'); + self::assertNotEmpty($recordInfos); + self::assertEquals(1, count($recordInfos)); + self::assertNotEmpty($recordInfos[0]->getValue()); + //self::assertEquals('', $recordInfos[0]->getRecordIdentifier()); + //self::assertNotEmpty($recordInfos[0]->getRecordOrigin()); + //self::assertNotEmpty($recordInfos[0]->getLanguageOfCataloging()); + + // TODO: implement reading of elements + } + + public function testGetNoRecordInfosByQueryForBookDocument() + { + $recordInfos = $this->bookReader->getRecordInfos('[./mods:descriptionStandard="xyz"]'); + self::assertEmpty($recordInfos); + } + + public function testGetRecordInfosForSerialDocument() + { + $recordInfos = $this->serialReader->getRecordInfos(); + self::assertNotEmpty($recordInfos); + self::assertEquals(1, count($recordInfos)); + self::assertNotEmpty($recordInfos[0]->getValue()); + //self::assertEquals('', $recordInfos[0]->getRecordIdentifier()); + //self::assertNotEmpty($recordInfos[0]->getRecordOrigin()); + //self::assertNotEmpty($recordInfos[0]->getLanguageOfCataloging()); + + // TODO: implement reading of elements + } + + public function testGetRecordInfosByQueryForSerialDocument() + { + $recordInfos = $this->serialReader->getRecordInfos('[./mods:descriptionStandard="aacr"]'); + self::assertNotEmpty($recordInfos); + self::assertEquals(1, count($recordInfos)); + self::assertNotEmpty($recordInfos[0]->getValue()); + //self::assertEquals('', $recordInfos[0]->getRecordIdentifier()); + //self::assertNotEmpty($recordInfos[0]->getRecordOrigin()); + //self::assertNotEmpty($recordInfos[0]->getLanguageOfCataloging()); + + // TODO: implement reading of elements + } + + public function testGetNoRecordInfosByQueryForSerialDocument() + { + $recordInfos = $this->serialReader->getRecordInfos('[./mods:descriptionStandard="xyz"]'); + self::assertEmpty($recordInfos); + } + + public function testGetNoRelatedItemsForBookDocument() + { + $relatedItems = $this->bookReader->getRelatedItems(); + self::assertEmpty($relatedItems); + } + + public function testGetRelatedItemsForSerialDocument() + { + $relatedItems = $this->serialReader->getRelatedItems(); + self::assertNotEmpty($relatedItems); + self::assertEquals(1, count($relatedItems)); + self::assertNotEmpty($relatedItems[0]->getValue()); + self::assertNotEmpty($relatedItems[0]->getType()); + self::assertEquals('preceding', $relatedItems[0]->getType()); + + // TODO: implement reading of elements + } + + public function testGetRelatedItemsByQueryForSerialDocument() + { + $relatedItems = $this->serialReader->getRelatedItems('[./mods:identifier="1525-321X"]'); + self::assertNotEmpty($relatedItems); + self::assertEquals(1, count($relatedItems)); + self::assertNotEmpty($relatedItems[0]->getValue()); + self::assertNotEmpty($relatedItems[0]->getType()); + self::assertEquals('preceding', $relatedItems[0]->getType()); + + // TODO: implement reading of elements + } + + public function testGetNoRelatedItemsByQueryForSerialDocument() + { + $relatedItems = $this->serialReader->getRelatedItems('[./mods:identifier="15-32"]'); + self::assertEmpty($relatedItems); + } + + public function testGetSubjectsForBookDocument() + { + $subjects = $this->bookReader->getSubjects(); + self::assertNotEmpty($subjects); + self::assertEquals(7, count($subjects)); + self::assertNotEmpty($subjects[0]->getValue()); + //self::assertNotEmpty($subjects[0]->getTopic()); + //self::assertNotEmpty($subjects[0]->getGeographic()); + + // TODO: implement reading of elements + } + + public function testGetSubjectsByQueryForBookDocument() + { + $subjects = $this->bookReader->getSubjects('[./mods:topic="Mass media"]'); + self::assertNotEmpty($subjects); + self::assertEquals(1, count($subjects)); + self::assertNotEmpty($subjects[0]->getValue()); + //self::assertNotEmpty($subjects[0]->getTopic()); + //self::assertNotEmpty($subjects[0]->getGeographic()); + + // TODO: implement reading of elements + } + + public function testGetNoSubjectsByQueryForBookDocument() + { + $subjects = $this->bookReader->getSubjects('[./mods:topic="Unknown"]'); + self::assertEmpty($subjects); + } + + public function testGetSubjectsForSerialDocument() + { + $subjects = $this->serialReader->getSubjects(); + self::assertNotEmpty($subjects); + self::assertEquals(6, count($subjects)); + self::assertNotEmpty($subjects[0]->getValue()); + //self::assertNotEmpty($subjects[0]->getTopic()); + //self::assertNotEmpty($subjects[0]->getGenre()); + + // TODO: implement reading of elements + } + + public function testGetSubjectsByQueryForSerialDocument() + { + $subjects = $this->serialReader->getSubjects('[./mods:genre="Directories"]'); + self::assertNotEmpty($subjects); + self::assertEquals(1, count($subjects)); + self::assertNotEmpty($subjects[0]->getValue()); + //self::assertNotEmpty($subjects[0]->getForm()); + //self::assertNotEmpty($subjects[0]->getGenre()); + + // TODO: implement reading of elements + } + + public function testGetNoSubjectsByQueryForSerialDocument() + { + $subjects = $this->serialReader->getSubjects('[./mods:topic="Unknown"]'); + self::assertEmpty($subjects); + } + + public function testGetTableOfContentsForBookDocument() + { + $tableOfContents = $this->bookReader->getTableOfContents(); + self::assertNotEmpty($tableOfContents); + self::assertEquals(1, count($tableOfContents)); + self::assertNotEmpty($tableOfContents[0]->getValue()); + self::assertEquals('Bluegrass odyssey -- Hills of Tennessee -- Sassafrass -- Muddy river -- Take your shoes off Moses -- Let Smokey Mountain smoke get in your eyes -- Farewell party -- Faded love', $tableOfContents[0]->getValue()); + self::assertNotEmpty($tableOfContents[0]->getDisplayLabel()); + self::assertEquals('Chapters', $tableOfContents[0]->getDisplayLabel()); + + // TODO: implement reading of elements + } + + public function testGetTableOfContentsByQueryForBookDocument() + { + $tableOfContents = $this->bookReader->getTableOfContents('[@displayLabel="Chapters"]'); + self::assertNotEmpty($tableOfContents); + self::assertEquals(1, count($tableOfContents)); + self::assertNotEmpty($tableOfContents[0]->getValue()); + self::assertEquals('Bluegrass odyssey -- Hills of Tennessee -- Sassafrass -- Muddy river -- Take your shoes off Moses -- Let Smokey Mountain smoke get in your eyes -- Farewell party -- Faded love', $tableOfContents[0]->getValue()); + self::assertNotEmpty($tableOfContents[0]->getDisplayLabel()); + self::assertEquals('Chapters', $tableOfContents[0]->getDisplayLabel()); + + // TODO: implement reading of elements + } + + public function testGetNoTableOfContentsByQueryForBookDocument() + { + $tableOfContents = $this->bookReader->getTableOfContents('[@displayLabel="Pages"]'); + self::assertEmpty($tableOfContents); + } + + public function testGetNoTableOfContentsForSerialDocument() + { + $tableOfContents = $this->serialReader->getTableOfContents(); + self::assertEmpty($tableOfContents); + } + + public function testGetTitleInfosForBookDocument() + { + $titleInfos = $this->bookReader->getTitleInfos(); + self::assertNotEmpty($titleInfos); + self::assertEquals(1, count($titleInfos)); + self::assertNotEmpty($titleInfos[0]->getValue()); + //self::assertNotEmpty($titleInfos[0]->getTitle()); + //self::assertNotEmpty($titleInfos[0]->getSubTitle()); + + // TODO: implement reading of elements + } + + public function testGetTitleInfosForSerialDocument() + { + $titleInfos = $this->serialReader->getTitleInfos(); + self::assertNotEmpty($titleInfos); + self::assertEquals(3, count($titleInfos)); + self::assertNotEmpty($titleInfos[0]->getValue()); + //self::assertNotEmpty($titleInfos[0]->getTitle()); + //self::assertNotEmpty($titleInfos[0]->getSubTitle()); + + // TODO: implement reading of elements + } + + public function testGetTitleInfosByQueryForSerialDocument() + { + $titleInfos = $this->serialReader->getTitleInfos('[@type="alternative"]'); + self::assertNotEmpty($titleInfos); + self::assertEquals(1, count($titleInfos)); + self::assertNotEmpty($titleInfos[0]->getValue()); + //self::assertNotEmpty($titleInfos[0]->getTitle()); + //self::assertEmpty($titleInfos[0]->getSubTitle()); + + // TODO: implement reading of elements + } + + public function testGetNoTitleInfosByQueryForSerialDocument() + { + $titleInfos = $this->serialReader->getTitleInfos('[@type="uniform"]'); + self::assertEmpty($titleInfos); + } + + public function testGetTypeOfResourceForBookDocument() + { + $typeOfResource = $this->bookReader->getTypeOfResource(); + self::assertNotNull($typeOfResource); + self::assertNotEmpty($typeOfResource->getDisplayLabel()); + self::assertEquals('format', $typeOfResource->getDisplayLabel()); + self::assertNotEmpty($typeOfResource->getValue()); + self::assertEquals('text', $typeOfResource->getValue()); + } + + public function testGetTypeOfResourceByQueryForBookDocument() + { + $typeOfResource = $this->bookReader->getTypeOfResource('[@displayLabel="format"]'); + self::assertNotNull($typeOfResource); + self::assertNotEmpty($typeOfResource->getDisplayLabel()); + self::assertEquals('format', $typeOfResource->getDisplayLabel()); + self::assertNotEmpty($typeOfResource->getValue()); + self::assertEquals('text', $typeOfResource->getValue()); + } + + public function testGetNoTypeOfResourceByQueryForBookDocument() + { + $typeOfResource = $this->bookReader->getTypeOfResource('[@displayLabel="random"]'); + self::assertNull($typeOfResource); + } + + public function testGetTypeOfResourceForSerialDocument() + { + $typeOfResource = $this->serialReader->getTypeOfResource(); + self::assertNotNull($typeOfResource); + self::assertEmpty($typeOfResource->getDisplayLabel()); + self::assertNotEmpty($typeOfResource->getValue()); + self::assertEquals('text', $typeOfResource->getValue()); + } + + public function testGetNoTypeOfResourceByQueryForSerialDocument() + { + $abstract = $this->serialReader->getAbstract('[@displayForm="format"]'); + self::assertNull($abstract); + } } From fb486722cf6668f78e7cd2e749ed555a27ccc859 Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Fri, 5 Apr 2024 17:59:01 +0200 Subject: [PATCH 14/17] Trigger tests on push and pull requstes --- .github/workflows/phpunit.yml | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/.github/workflows/phpunit.yml b/.github/workflows/phpunit.yml index ae63da1..e202431 100644 --- a/.github/workflows/phpunit.yml +++ b/.github/workflows/phpunit.yml @@ -1,6 +1,10 @@ name: PHPUnit -on: [push] +on: + push: + branches: [ "master" ] + pull_request: + branches: [ "master" ] jobs: build-test: From b6d11d335ef2c746ab4d0a7c2e8d272058055e59 Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Fri, 5 Apr 2024 18:37:04 +0200 Subject: [PATCH 15/17] Implement functions in `Language` class --- src/Mods/Element/Language.php | 55 ++++++++++++++++++++++------------- 1 file changed, 34 insertions(+), 21 deletions(-) diff --git a/src/Mods/Element/Language.php b/src/Mods/Element/Language.php index fa3d731..dff3f23 100644 --- a/src/Mods/Element/Language.php +++ b/src/Mods/Element/Language.php @@ -20,9 +20,11 @@ use Slub\Mods\Element\Common\BaseElement; use Slub\Mods\Element\Specific\Language\LanguageTerm; use Slub\Mods\Element\Specific\Language\ScriptTerm; +use Slub\Mods\Element\Xml\Element; /** * Language MODS metadata element class for the 'php-mods-reader' library. + * @see https://www.loc.gov/standards/mods/userguide/language.html * * @access public */ @@ -30,18 +32,6 @@ class Language extends BaseElement { use LanguageAttribute, IdAttribute, AltRepGroupAttribute, DisplayLabelAttribute, UsageAttribute; - /** - * @access private - * @var LanguageTerm - */ - private LanguageTerm $languageTerm; - - /** - * @access private - * @var ScriptTerm - */ - private ScriptTerm $scriptTerm; - /** * This extracts the essential MODS metadata from XML * @@ -57,7 +47,8 @@ public function __construct(\SimpleXMLElement $xml) } /** - * Get the value of objectPart + * Get the value of the 'objectPart' attribute. + * @see https://www.loc.gov/standards/mods/userguide/language.html#objectpart * * @access public * @@ -69,26 +60,48 @@ public function getObjectPart(): string } /** - * Get the value of languageTerm + * Get the array of the elements. + * @see https://www.loc.gov/standards/mods/userguide/language.html#languageterm * * @access public * - * @return LanguageTerm + * @param string $query The XPath query for metadata search + * + * @return LanguageTerm[] */ - public function getLanguageTerm(): LanguageTerm + public function getLanguageTerms(string $query = ''): array { - return $this->languageTerm; + $languageTerms = []; + $xpath = './mods:languageTerm' . $query; + $element = new Element($this->xml, $xpath); + if ($element->exists()) { + foreach ($element->getValues() as $value) { + $languageTerms[] = new LanguageTerm($value); + } + } + return $languageTerms; } /** - * Get the value of scriptTerm + * Get the array of the elements. + * @see https://www.loc.gov/standards/mods/userguide/language.html#scriptterm * * @access public * - * @return ScriptTerm + * @param string $query The XPath query for metadata search + * + * @return ScriptTerm[] */ - public function getScriptTerm(): ScriptTerm + public function getScriptTerms(string $query = ''): array { - return $this->scriptTerm; + $scriptTerms = []; + $xpath = './mods:scriptTerm' . $query; + $element = new Element($this->xml, $xpath); + if ($element->exists()) { + foreach ($element->getValues() as $value) { + $scriptTerms[] = new ScriptTerm($value); + } + } + return $scriptTerms; } } From 09e7d98099f416609ef724c8c484c45fcc56b831 Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Fri, 5 Apr 2024 18:42:39 +0200 Subject: [PATCH 16/17] Change `getLanguage` to `getLaguages` element can be repeated in MODS document --- src/Mods/ModsReader.php | 14 +++++++++----- 1 file changed, 9 insertions(+), 5 deletions(-) diff --git a/src/Mods/ModsReader.php b/src/Mods/ModsReader.php index c86577b..e27d7c0 100644 --- a/src/Mods/ModsReader.php +++ b/src/Mods/ModsReader.php @@ -191,22 +191,26 @@ public function getIdentifiers(string $query = ''): array } /** - * Get + * Get the array of the elements. + * @see https://www.loc.gov/standards/mods/userguide/language.html * * @access public * * @param string $query The XPath query for metadata search * - * @return ?Language + * @return Language[] */ - public function getLanguage(string $query = ''): ?Language + public function getLanguages(string $query = ''): array { + $languages = []; $xpath = './mods:language' . $query; $element = new Element($this->xml, $xpath); if ($element->exists()) { - return new Language($element->getValues()[0]); + foreach ($element->getValues() as $value) { + $languages[] = new Language($value); + } } - return null; + return $languages; } /** From d115ff6c39ba97e7e0cf6807e0c2fd55fcf1c0c4 Mon Sep 17 00:00:00 2001 From: Beatrycze Volk Date: Fri, 5 Apr 2024 18:57:45 +0200 Subject: [PATCH 17/17] Implement test for reading language element --- tests/Mods/ModsReaderTest.php | 99 +++++++++++++++++++++++++++-------- tests/resources/mods_book.xml | 7 ++- 2 files changed, 83 insertions(+), 23 deletions(-) diff --git a/tests/Mods/ModsReaderTest.php b/tests/Mods/ModsReaderTest.php index 9dcc919..e368f55 100644 --- a/tests/Mods/ModsReaderTest.php +++ b/tests/Mods/ModsReaderTest.php @@ -379,28 +379,83 @@ public function testGetNoIdentifiersByQueryForSerialDocument() self::assertEmpty($identifiers); } - public function testGetLanguageForBookDocument() - { - $language = $this->bookReader->getLanguage(); - self::assertNotNull($language); - // TODO: implement reading of languageTerm and scriptTerm - // self::assertNotEmpty($language->getLanguageTerm()); - // self::assertNotEmpty($language->getScriptTerm()); - - $language = $this->bookReader->getLanguage('[@type="text"]'); - self::assertNull($language); - } - - public function testGetLanguageForSerialDocument() - { - $language = $this->serialReader->getLanguage(); - self::assertNotNull($language); - // TODO: implement reading of languageTerm and scriptTerm - // self::assertNotEmpty($language->getLanguageTerm()); - // self::assertNotEmpty($language->getScriptTerm()); - - $language = $this->serialReader->getLanguage('[@type="text"]'); - self::assertNull($language); + public function testGetLanguagesForBookDocument() + { + $languages = $this->bookReader->getLanguages(); + self::assertNotEmpty($languages); + self::assertEquals(2, count($languages)); + self::assertEmpty($languages[0]->getObjectPart()); + self::assertNotEmpty($languages[0]->getValue()); + self::assertNotEmpty($languages[0]->getLanguageTerms()); + self::assertNotEmpty($languages[0]->getLanguageTerms()[0]->getValue()); + self::assertEquals('code', $languages[0]->getLanguageTerms()[0]->getType()); + self::assertEquals('iso639-2b', $languages[0]->getLanguageTerms()[0]->getAuthority()); + self::assertEquals('eng', $languages[0]->getLanguageTerms()[0]->getValue()); + self::assertNotEmpty($languages[0]->getScriptTerms()); + self::assertEquals('code', $languages[0]->getScriptTerms()[0]->getType()); + self::assertEquals('iso15924', $languages[0]->getScriptTerms()[0]->getAuthority()); + self::assertEquals('Latn', $languages[0]->getScriptTerms()[0]->getValue()); + } + + public function testGetLanguagesByQueryForBookDocument() + { + $languages = $this->bookReader->getLanguages('[@objectPart="summary"]'); + self::assertNotEmpty($languages); + self::assertEquals(1, count($languages)); + self::assertNotEmpty($languages[0]->getObjectPart()); + self::assertEquals('summary', $languages[0]->getObjectPart()); + self::assertNotEmpty($languages[0]->getValue()); + self::assertNotEmpty($languages[0]->getLanguageTerms()); + self::assertNotEmpty($languages[0]->getLanguageTerms()[0]->getValue()); + self::assertEquals('code', $languages[0]->getLanguageTerms()[0]->getType()); + self::assertEquals('iso639-2b', $languages[0]->getLanguageTerms()[0]->getAuthority()); + self::assertEquals('spa', $languages[0]->getLanguageTerms()[0]->getValue()); + self::assertNotEmpty($languages[0]->getScriptTerms()); + self::assertEquals('code', $languages[0]->getScriptTerms()[0]->getType()); + self::assertEquals('iso15924', $languages[0]->getScriptTerms()[0]->getAuthority()); + self::assertEquals('Latn', $languages[0]->getScriptTerms()[0]->getValue()); + } + + public function testGetNoLanguagesByQueryForBookDocument() + { + $languages = $this->bookReader->getLanguages('[@objectPart="abstract"]'); + self::assertEmpty($languages); + } + + public function testGetLanguagesForSerialDocument() + { + $languages = $this->serialReader->getLanguages(); + self::assertNotEmpty($languages); + self::assertEquals(1, count($languages)); + self::assertEmpty($languages[0]->getObjectPart()); + self::assertNotEmpty($languages[0]->getValue()); + self::assertNotEmpty($languages[0]->getLanguageTerms()); + self::assertNotEmpty($languages[0]->getLanguageTerms()[0]->getValue()); + self::assertEquals('code', $languages[0]->getLanguageTerms()[0]->getType()); + self::assertEquals('iso639-2b', $languages[0]->getLanguageTerms()[0]->getAuthority()); + self::assertEquals('eng', $languages[0]->getLanguageTerms()[0]->getValue()); + self::assertEmpty($languages[0]->getScriptTerms()); + } + + public function testGetLanguagesByQueryForSerialDocument() + { + $languages = $this->serialReader->getLanguages('[./mods:languageTerm[@type="code"]]'); + self::assertNotEmpty($languages); + self::assertEquals(1, count($languages)); + self::assertEmpty($languages[0]->getObjectPart()); + self::assertNotEmpty($languages[0]->getValue()); + self::assertNotEmpty($languages[0]->getLanguageTerms()); + self::assertNotEmpty($languages[0]->getLanguageTerms()[0]->getValue()); + self::assertEquals('code', $languages[0]->getLanguageTerms()[0]->getType()); + self::assertEquals('iso639-2b', $languages[0]->getLanguageTerms()[0]->getAuthority()); + self::assertEquals('eng', $languages[0]->getLanguageTerms()[0]->getValue()); + self::assertEmpty($languages[0]->getScriptTerms()); + } + + public function testGetNoLanguagesByQueryForSerialDocument() + { + $languages = $this->serialReader->getLanguages('[@objectPart="summary"]'); + self::assertEmpty($languages); } public function testGetLocationsForBookDocument() diff --git a/tests/resources/mods_book.xml b/tests/resources/mods_book.xml index fe259e6..77a97f9 100644 --- a/tests/resources/mods_book.xml +++ b/tests/resources/mods_book.xml @@ -57,7 +57,12 @@ c1999
- eng + eng + Latn + + + spa + Latn
print