2024-07-26 22:55:02 -04:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* @author Daniel J. Summers <daniel@bitbadger.solutions>
|
|
|
|
* @license MIT
|
|
|
|
*/
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace Test;
|
|
|
|
|
|
|
|
use BitBadger\InspiredByFSharp\Result;
|
|
|
|
use InvalidArgumentException;
|
|
|
|
use PHPUnit\Framework\Attributes\TestDox;
|
|
|
|
use PHPUnit\Framework\TestCase;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unit tests for the Result class
|
|
|
|
*/
|
|
|
|
class ResultTest extends TestCase
|
|
|
|
{
|
2024-09-30 22:59:46 -04:00
|
|
|
#[TestDox('OK property succeeds for OK result')]
|
|
|
|
public function testOKPropertySucceedsForOKResult(): void
|
2024-07-26 22:55:02 -04:00
|
|
|
{
|
|
|
|
$result = Result::OK('yay');
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertEquals('yay', $result->ok, 'The OK result should have been returned');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
2024-09-30 22:59:46 -04:00
|
|
|
#[TestDox('OK property fails for Error result')]
|
|
|
|
public function testOKPropertyFailsForErrorResult(): void
|
2024-07-26 22:55:02 -04:00
|
|
|
{
|
|
|
|
$this->expectException(InvalidArgumentException::class);
|
2024-09-30 22:59:46 -04:00
|
|
|
Result::Error('whoops')->ok;
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
2024-09-30 22:59:46 -04:00
|
|
|
#[TestDox('Error property succeeds for Error result')]
|
|
|
|
public function testErrorPropertySucceedsForErrorResult(): void
|
2024-07-26 22:55:02 -04:00
|
|
|
{
|
|
|
|
$result = Result::Error('boo');
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertEquals('boo', $result->error, 'The Error result should have been returned');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
2024-09-30 22:59:46 -04:00
|
|
|
#[TestDox('Error property fails for OK result')]
|
|
|
|
public function testErrorPropertyFailsForOKResult(): void
|
2024-07-26 22:55:02 -04:00
|
|
|
{
|
|
|
|
$this->expectException(InvalidArgumentException::class);
|
2024-09-30 22:59:46 -04:00
|
|
|
Result::OK('yeah')->error;
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('IsOK succeeds for OK result')]
|
|
|
|
public function testIsOKSucceedsForOKResult(): void
|
|
|
|
{
|
|
|
|
$result = Result::OK('ok');
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($result->isOK, 'The check for "OK" should have returned true');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('IsOK succeeds for Error result')]
|
|
|
|
public function testIsOKSucceedsForErrorResult(): void
|
|
|
|
{
|
|
|
|
$result = Result::Error('error');
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertFalse($result->isOK, 'The check for "OK" should have returned false');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('IsError succeeds for Error result')]
|
|
|
|
public function testIsErrorSucceedsForErrorResult(): void
|
|
|
|
{
|
|
|
|
$result = Result::Error('not ok');
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($result->isError, 'The check for "Error" should have returned true');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('IsError succeeds for OK result')]
|
|
|
|
public function testIsErrorSucceedsForOKResult(): void
|
|
|
|
{
|
|
|
|
$result = Result::OK('fine');
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertFalse($result->isError, 'The check for "Error" should have returned false');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
2024-07-28 22:50:59 -04:00
|
|
|
#[TestDox('Bind succeeds for OK with OK')]
|
|
|
|
public function testBindSucceedsForOKWithOK(): void
|
|
|
|
{
|
|
|
|
$result = Result::OK('one')->bind(fn($it) => Result::OK("$it two"));
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($result->isOK, 'The result should have been OK');
|
|
|
|
$this->assertEquals('one two', $result->ok, 'The bound function was not called');
|
2024-07-28 22:50:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Bind succeeds for OK with Error')]
|
|
|
|
public function testBindSucceedsForOKWithError(): void
|
|
|
|
{
|
|
|
|
$result = Result::OK('three')->bind(fn($it) => Result::Error('back to two'));
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($result->isError, 'The result should have been Error');
|
|
|
|
$this->assertEquals('back to two', $result->error, 'The bound function was not called');
|
2024-07-28 22:50:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Bind succeeds for Error')]
|
|
|
|
public function testBindSucceedsForError(): void
|
|
|
|
{
|
|
|
|
$original = Result::Error('oops');
|
|
|
|
$result = $original->bind(fn($it) => Result::OK('never mind - it worked!'));
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($result->isError, 'The result should have been Error');
|
2024-07-28 22:50:59 -04:00
|
|
|
$this->assertSame($original, $result, 'The same Error result should have been returned');
|
|
|
|
}
|
|
|
|
|
2024-07-29 13:58:33 -04:00
|
|
|
|
|
|
|
#[TestDox('Contains succeeds for Error result')]
|
|
|
|
public function testContainsSucceedsForErrorResult(): void
|
|
|
|
{
|
|
|
|
$this->assertFalse(Result::Error('ouch')->contains('ouch'), '"Error" should always return false');
|
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Contains succeeds for OK result when strictly equal')]
|
|
|
|
public function testContainsSucceedsForOKResultWhenStrictlyEqual(): void
|
|
|
|
{
|
|
|
|
$this->assertTrue(Result::OK(18)->contains(18), '"OK" with strict equality should be true');
|
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Contains succeeds for OK result when strictly unequal')]
|
|
|
|
public function testContainsSucceedsForOKResultWhenStrictlyUnequal(): void
|
|
|
|
{
|
|
|
|
$this->assertFalse(Result::OK(18)->contains('18'), '"OK" with strict equality should be false');
|
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Contains succeeds for OK result when loosely equal')]
|
|
|
|
public function testContainsSucceedsForOKResultWhenLooselyEqual(): void
|
|
|
|
{
|
|
|
|
$this->assertTrue(Result::OK(18)->contains('18', strict: false), '"OK" with loose equality should be true');
|
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Contains succeeds for OK result when loosely unequal')]
|
|
|
|
public function testContainsSucceedsForOKResultWhenLooselyUnequal(): void
|
|
|
|
{
|
|
|
|
$this->assertFalse(Result::OK(18)->contains(17, strict: false), '"OK" with loose equality should be false');
|
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Exists succeeds for OK result when matching')]
|
|
|
|
public function testExistsSucceedsForOKResultWhenMatching(): void
|
|
|
|
{
|
|
|
|
$this->assertTrue(Result::OK(14)->exists(fn($it) => $it < 100), 'Exists should have returned true');
|
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Exists succeeds for OK result when not matching')]
|
|
|
|
public function testExistsSucceedsForOKResultWhenNotMatching(): void
|
|
|
|
{
|
|
|
|
$this->assertFalse(Result::OK(14)->exists(fn($it) => $it > 100), 'Exists should have returned false');
|
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Exists succeeds for Error result')]
|
|
|
|
public function testExistsSucceedsForErrorResult(): void
|
|
|
|
{
|
|
|
|
$this->assertFalse(Result::Error(true)->exists(fn($it) => true), 'Exists should have returned false');
|
|
|
|
}
|
|
|
|
|
2024-07-26 22:55:02 -04:00
|
|
|
#[TestDox('Map succeeds for OK result')]
|
|
|
|
public function testMapSucceedsForOKResult(): void
|
|
|
|
{
|
|
|
|
$ok = Result::OK('yard');
|
2024-07-28 14:15:24 -04:00
|
|
|
$mapped = $ok->map(fn($it) => strrev($it));
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($mapped->isOK, 'The mapped result should be "OK"');
|
|
|
|
$this->assertEquals('dray', $mapped->ok, 'The mapping function was not called correctly');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Map fails for OK result when mapping is null')]
|
|
|
|
public function testMapFailsForOKResultWhenMappingIsNull(): void
|
|
|
|
{
|
|
|
|
$this->expectException(InvalidArgumentException::class);
|
2024-07-28 14:15:24 -04:00
|
|
|
Result::OK('not null')->map(fn($it) => null);
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Map succeeds for Error result')]
|
|
|
|
public function testMapSucceedsForErrorResult(): void
|
|
|
|
{
|
|
|
|
$tattle = new class { public bool $called = false; };
|
|
|
|
$error = Result::Error('nope');
|
2024-07-28 14:15:24 -04:00
|
|
|
$mapped = $error->map(function () use ($tattle)
|
2024-07-26 22:55:02 -04:00
|
|
|
{
|
|
|
|
$tattle->called = true;
|
|
|
|
return 'hello';
|
2024-07-28 14:15:24 -04:00
|
|
|
});
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($mapped->isError, 'The mapped result should be "Error"');
|
2024-07-26 22:55:02 -04:00
|
|
|
$this->assertFalse($tattle->called, 'The mapping function should not have been called');
|
2024-07-28 14:15:24 -04:00
|
|
|
$this->assertSame($error, $mapped, 'The same "Error" instance should have been returned');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('MapError succeeds for OK result')]
|
|
|
|
public function testMapErrorSucceedsForOKResult(): void
|
|
|
|
{
|
|
|
|
$tattle = new class { public bool $called = false; };
|
|
|
|
$ok = Result::OK('sure');
|
2024-07-28 14:15:24 -04:00
|
|
|
$mapped = $ok->mapError(function () use ($tattle)
|
2024-07-26 22:55:02 -04:00
|
|
|
{
|
|
|
|
$tattle->called = true;
|
|
|
|
return 'hello';
|
2024-07-28 14:15:24 -04:00
|
|
|
});
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($mapped->isOK, 'The mapped result should be "OK"');
|
2024-07-26 22:55:02 -04:00
|
|
|
$this->assertFalse($tattle->called, 'The mapping function should not have been called');
|
2024-07-28 14:15:24 -04:00
|
|
|
$this->assertSame($ok, $mapped, 'The same "OK" instance should have been returned');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('MapError succeeds for Error result')]
|
|
|
|
public function testMapErrorSucceedsForErrorResult(): void
|
|
|
|
{
|
|
|
|
$error = Result::Error('taco');
|
2024-07-28 14:15:24 -04:00
|
|
|
$mapped = $error->mapError(fn($it) => str_repeat('*', strlen($it)));
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($mapped->isError, 'The mapped result should be "Error"');
|
|
|
|
$this->assertEquals('****', $mapped->error, 'The mapping function was not called correctly');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('MapError fails for Error result when mapping is null')]
|
|
|
|
public function testMapErrorFailsForErrorResultWhenMappingIsNull(): void
|
|
|
|
{
|
|
|
|
$this->expectException(InvalidArgumentException::class);
|
2024-07-28 14:15:24 -04:00
|
|
|
Result::Error('pizza')->mapError(fn($it) => null);
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Iter succeeds for OK result')]
|
|
|
|
public function testIterSucceedsForOKResult(): void
|
|
|
|
{
|
|
|
|
$target = new class { public mixed $called = null; };
|
2024-07-28 14:15:24 -04:00
|
|
|
Result::OK(77)->iter(function ($it) use ($target) { $target->called = $it; });
|
2024-07-26 22:55:02 -04:00
|
|
|
$this->assertEquals(77, $target->called, 'The function should have been called with the "OK" value');
|
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Iter succeeds for Error result')]
|
|
|
|
public function testIterSucceedsForErrorResult(): void
|
|
|
|
{
|
|
|
|
$target = new class { public mixed $called = null; };
|
2024-07-28 14:15:24 -04:00
|
|
|
Result::Error('')->iter(function () use ($target) { $target->called = 'uh oh'; });
|
2024-07-26 22:55:02 -04:00
|
|
|
$this->assertNull($target->called, 'The function should not have been called');
|
|
|
|
}
|
|
|
|
|
2024-07-29 13:58:33 -04:00
|
|
|
#[TestDox('ToArray succeeds for OK result')]
|
|
|
|
public function testToArraySucceedsForOKResult(): void
|
|
|
|
{
|
|
|
|
$arr = Result::OK('yay')->toArray();
|
|
|
|
$this->assertNotNull($arr, 'The array should not have been null');
|
|
|
|
$this->assertEquals(['yay'], $arr, 'The array was not created correctly');
|
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('ToArray succeeds for Error result')]
|
|
|
|
public function testToArraySucceedsForErrorResult(): void
|
|
|
|
{
|
|
|
|
$arr = Result::Error('oh no')->toArray();
|
|
|
|
$this->assertNotNull($arr, 'The array should not have been null');
|
|
|
|
$this->assertEmpty($arr, 'The array should have been empty');
|
|
|
|
}
|
|
|
|
|
2024-07-26 22:55:02 -04:00
|
|
|
#[TestDox('ToOption succeeds for OK result')]
|
|
|
|
public function testToOptionSucceedsForOKResult()
|
|
|
|
{
|
2024-07-28 14:15:24 -04:00
|
|
|
$value = Result::OK(99)->toOption();
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($value->isSome, 'An "OK" result should map to a "Some" option');
|
|
|
|
$this->assertEquals(99, $value->value, 'The value is not correct');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('ToOption succeeds for Error result')]
|
|
|
|
public function testToOptionSucceedsForErrorResult()
|
|
|
|
{
|
2024-07-28 14:15:24 -04:00
|
|
|
$value = Result::Error('file not found')->toOption();
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($value->isNone, 'An "Error" result should map to a "None" option');
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|
2024-07-27 13:03:54 -04:00
|
|
|
|
|
|
|
#[TestDox('Tap succeeds for OK result')]
|
|
|
|
public function testTapSucceedsForOKResult(): void
|
|
|
|
{
|
|
|
|
$value = '';
|
|
|
|
$original = Result::OK('working');
|
2024-07-28 14:15:24 -04:00
|
|
|
$tapped = $original->tap(function (Result $it) use (&$value) {
|
2024-09-30 22:59:46 -04:00
|
|
|
$value = $it->isOK ? 'OK: ' . $it->ok : 'Error: ' . $it->error;
|
2024-07-28 14:15:24 -04:00
|
|
|
});
|
2024-07-27 13:03:54 -04:00
|
|
|
$this->assertEquals('OK: working', $value, 'The tapped function was not called');
|
|
|
|
$this->assertSame($original, $tapped, 'The same result should have been returned');
|
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Tap succeeds for Error result')]
|
|
|
|
public function testTapSucceedsForErrorResult(): void
|
|
|
|
{
|
|
|
|
$value = '';
|
|
|
|
$original = Result::Error('failed');
|
2024-07-28 14:15:24 -04:00
|
|
|
$tapped = $original->tap(function (Result $it) use (&$value) {
|
2024-09-30 22:59:46 -04:00
|
|
|
$value = $it->isOK ? 'OK: ' . $it->ok : 'Error: ' . $it->error;
|
2024-07-28 14:15:24 -04:00
|
|
|
});
|
2024-07-27 13:03:54 -04:00
|
|
|
$this->assertEquals('Error: failed', $value, 'The tapped function was not called');
|
|
|
|
$this->assertSame($original, $tapped, 'The same result should have been returned');
|
|
|
|
}
|
2024-07-28 14:15:24 -04:00
|
|
|
|
|
|
|
#[TestDox('OK succeeds for non null result')]
|
|
|
|
public function testOKSucceedsForNonNullResult(): void
|
|
|
|
{
|
|
|
|
$result = Result::OK('something');
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($result->isOK, 'The result should have been "OK"');
|
|
|
|
$this->assertEquals('something', $result->ok, 'The "OK" value was incorrect');
|
2024-07-28 14:15:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('OK fails for null result')]
|
|
|
|
public function testOKFailsForNullResult(): void
|
|
|
|
{
|
|
|
|
$this->expectException(InvalidArgumentException::class);
|
|
|
|
Result::OK(null);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Error succeeds for non null result')]
|
|
|
|
public function testErrorSucceedsForNonNullResult(): void
|
|
|
|
{
|
|
|
|
$result = Result::Error('sad trombone');
|
2024-09-30 22:59:46 -04:00
|
|
|
$this->assertTrue($result->isError, 'The result should have been "Error"');
|
|
|
|
$this->assertEquals('sad trombone', $result->error, 'The "Error" value was incorrect');
|
2024-07-28 14:15:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[TestDox('Error fails for null result')]
|
|
|
|
public function testErrorFailsForNullResult(): void
|
|
|
|
{
|
|
|
|
$this->expectException(InvalidArgumentException::class);
|
|
|
|
Result::Error(null);
|
|
|
|
}
|
2024-07-26 22:55:02 -04:00
|
|
|
}
|