feat: add url mapping value objects

This commit is contained in:
Keith Solomon
2026-04-26 14:29:53 -05:00
parent 3c254a34c9
commit 8d7abc8536
3 changed files with 100 additions and 0 deletions
+32
View File
@@ -0,0 +1,32 @@
<?php
namespace WPContentSync\Url;
final class UrlMapping {
private string $source_url;
private string $destination_url;
public function __construct( string $source_url, string $destination_url ) {
$source_url = $this->normalizeUrl( $source_url );
$destination_url = $this->normalizeUrl( $destination_url );
if ( '' === $source_url || '' === $destination_url ) {
throw new \InvalidArgumentException( 'Source and destination URLs are required.' );
}
$this->source_url = $source_url;
$this->destination_url = $destination_url;
}
public function sourceUrl(): string {
return $this->source_url;
}
public function destinationUrl(): string {
return $this->destination_url;
}
private function normalizeUrl( string $url ): string {
return rtrim( trim( $url ), '/' );
}
}
+30
View File
@@ -0,0 +1,30 @@
<?php
namespace WPContentSync\Url;
final class UrlMappingCollection {
/**
* @var array<int, UrlMapping>
*/
private array $mappings;
/**
* @param array<int, UrlMapping> $mappings URL mappings.
*/
public function __construct( array $mappings ) {
foreach ( $mappings as $mapping ) {
if ( ! $mapping instanceof UrlMapping ) {
throw new \InvalidArgumentException( 'URL mapping collections only accept UrlMapping instances.' );
}
}
$this->mappings = array_values( $mappings );
}
/**
* @return array<int, UrlMapping>
*/
public function all(): array {
return $this->mappings;
}
}
@@ -0,0 +1,38 @@
<?php
namespace WPContentSync\Tests\Unit\Url;
use PHPUnit\Framework\TestCase;
use WPContentSync\Url\UrlMapping;
use WPContentSync\Url\UrlMappingCollection;
class UrlMappingCollectionTest extends TestCase {
public function test_it_normalizes_mapping_urls_without_trailing_slashes(): void {
$mapping = new UrlMapping( 'https://example.test/', 'https://staging.example.test/' );
self::assertSame( 'https://example.test', $mapping->sourceUrl() );
self::assertSame( 'https://staging.example.test', $mapping->destinationUrl() );
}
public function test_it_rejects_empty_mapping_urls(): void {
$this->expectException( \InvalidArgumentException::class );
$this->expectExceptionMessage( 'Source and destination URLs are required.' );
new UrlMapping( '', 'https://staging.example.test' );
}
public function test_it_returns_mappings_in_order(): void {
$first = new UrlMapping( 'https://example.test', 'https://staging.example.test' );
$second = new UrlMapping( 'https://cdn.example.test', 'https://cdn.staging.example.test' );
$collection = new UrlMappingCollection( array( $first, $second ) );
self::assertSame( array( $first, $second ), $collection->all() );
}
public function test_it_rejects_non_mapping_items(): void {
$this->expectException( \InvalidArgumentException::class );
$this->expectExceptionMessage( 'URL mapping collections only accept UrlMapping instances.' );
new UrlMappingCollection( array( new \stdClass() ) );
}
}