-
Notifications
You must be signed in to change notification settings - Fork 39
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
encapsulation over inheritance: hide ItemList API on Item behind a fa…
…cade in a BC way
- Loading branch information
Showing
8 changed files
with
245 additions
and
113 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,72 @@ | ||
<?php | ||
|
||
namespace Jkphl\Micrometa\Ports\Item; | ||
|
||
use Jkphl\Micrometa\Ports\Exceptions\InvalidArgumentException; | ||
use Jkphl\Micrometa\Ports\Exceptions\OutOfBoundsException; | ||
|
||
/** | ||
* Abstract getter functionality that is applicable for implementations of ItemCollectionFacade. | ||
* | ||
* The implementation remains responsible for implementing the collection methods itself. | ||
* | ||
* Thereby is usable by both the Item for it's properties, as for an ItemList with it's items. | ||
* | ||
* @package Jkphl\Micrometa | ||
* @subpackage Jkphl\Micrometa\Ports | ||
*/ | ||
abstract class Collection implements CollectionFacade | ||
{ | ||
/** | ||
* Generic item getter | ||
* | ||
* @param string $type Item type | ||
* @param array $arguments Arguments | ||
* | ||
* @return ItemInterface Item | ||
* @throws InvalidArgumentException If the item index is invalid | ||
* @api | ||
*/ | ||
public function __call($type, $arguments) | ||
{ | ||
$index = 0; | ||
if (count($arguments)) { | ||
// If the item index is invalid | ||
if (!is_int($arguments[0]) || ($arguments[0] < 0)) { | ||
throw new InvalidArgumentException( | ||
sprintf(InvalidArgumentException::INVALID_ITEM_INDEX_STR, $arguments[0]), | ||
InvalidArgumentException::INVALID_ITEM_INDEX | ||
); | ||
} | ||
|
||
$index = $arguments[0]; | ||
} | ||
|
||
// Return the item by type and index | ||
return $this->getItemByTypeAndIndex($type, $index); | ||
} | ||
|
||
/** | ||
* Return an item by type and index | ||
* | ||
* @param string $type Item type | ||
* @param int $index Item index | ||
* | ||
* @return ItemInterface Item | ||
* @throws OutOfBoundsException If the item index is out of bounds | ||
*/ | ||
private function getItemByTypeAndIndex($type, $index) | ||
{ | ||
$typeItems = $this->getItems($type); | ||
|
||
// If the item index is out of bounds | ||
if (count($typeItems) <= $index) { | ||
throw new OutOfBoundsException( | ||
sprintf(OutOfBoundsException::INVALID_ITEM_INDEX_STR, $index), | ||
OutOfBoundsException::INVALID_ITEM_INDEX | ||
); | ||
} | ||
|
||
return $typeItems[$index]; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,46 @@ | ||
<?php | ||
|
||
namespace Jkphl\Micrometa\Ports\Item; | ||
|
||
/** | ||
* Interface to hides all other methods that ItemListInterface inherits from PHP core. | ||
* | ||
* Thereby is usable by both the Item for it's properties, as for an ItemList with it's items. | ||
* | ||
* Prefer type-hinting against this facade over the ItemList interface. | ||
* | ||
* @see ItemListInterface | ||
* | ||
* @package Jkphl\Micrometa | ||
* @subpackage Jkphl\Micrometa\Ports | ||
*/ | ||
interface CollectionFacade extends \Countable | ||
{ | ||
/** | ||
* Return an object representation of the item list | ||
* | ||
* @return \stdClass Micro information items | ||
* @api | ||
*/ | ||
public function toObject(); | ||
|
||
/** | ||
* Filter the items by item type(s) | ||
* | ||
* @param string[] $types Item types | ||
* | ||
* @return ItemInterface[] Items matching the requested types | ||
* @api | ||
*/ | ||
public function getItems(...$types); | ||
|
||
/** | ||
* Return the first item, optionally of particular types | ||
* | ||
* @param string[] $types Item types | ||
* | ||
* @return ItemInterface Item | ||
* @api | ||
*/ | ||
public function getFirstItem(...$types); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,71 @@ | ||
<?php | ||
|
||
namespace Micrometa\Ports\Item; | ||
|
||
use Jkphl\Micrometa\Infrastructure\Factory\MicroformatsFactory; | ||
use Jkphl\Micrometa\Ports\Exceptions\OutOfBoundsException; | ||
use Jkphl\Micrometa\Ports\Item\ItemInterface; | ||
use Jkphl\Micrometa\Ports\Item\ItemList; | ||
use Jkphl\Micrometa\Tests\MicroformatsFeedTrait; | ||
use PHPUnit\Framework\TestCase; | ||
|
||
class ItemListTest extends TestCase | ||
{ | ||
/** | ||
* Use the Microformats feed method | ||
*/ | ||
use MicroformatsFeedTrait; | ||
|
||
private $feedItemList; | ||
|
||
protected function setUp(): void | ||
{ | ||
$this->feedItemList = new ItemList([$this->getFeedItem(), $this->getFeedItem()]); | ||
} | ||
|
||
public function testNestedItemsCounts() | ||
{ | ||
// Test the number of nested items | ||
self::assertCount(2, $this->feedItemList); | ||
self::assertCount(2, $this->feedItemList->getItems()); | ||
} | ||
|
||
public function testNestedItemsIteration() | ||
{ | ||
foreach ($this->feedItemList->getItems() as $itemIndex => $entryItem) { | ||
self::assertInstanceOf(ItemInterface::class, $entryItem); | ||
self::assertIsInt($itemIndex); | ||
} | ||
} | ||
|
||
public function testNestedItemsRetrievalViaArrayAccess() | ||
{ | ||
$entryItems = $this->feedItemList->getFirstItem()->getItems('h-entry'); | ||
$entryItem = $entryItems[1]; | ||
|
||
self::assertInstanceOf(ItemInterface::class, $entryItem); | ||
} | ||
|
||
public function testFirstNestedItemRetrieval() | ||
{ | ||
self::assertInstanceOf(ItemInterface::class, $this->feedItemList[0]->getFirstItem('h-entry')); | ||
self::assertInstanceOf( | ||
ItemInterface::class, | ||
$this->feedItemList[0]->getFirstItem('h-entry', MicroformatsFactory::MF2_PROFILE_URI) | ||
); | ||
} | ||
|
||
public function testExistingFirstNestedItem() | ||
{ | ||
self::assertEquals('John Doe', $this->feedItemList[0]->hEntry()->author->name); | ||
self::assertEquals('John Doe', $this->feedItemList->getFirstItem()->hEntry()->author->name); | ||
} | ||
|
||
public function testNonExistingSecondNestedItem() | ||
{ | ||
$this->expectException(OutOfBoundsException::class); | ||
$this->expectExceptionCode('1492418999'); | ||
|
||
$this->feedItemList->hEntry(2); | ||
} | ||
} |
Oops, something went wrong.