Typically, to mark that a function return value, an argument, or a property has some value or not, you'd use the nullable type declaration. Most of the time it works just fine, but can sometimes be unpleasant when dealing directly with a null value that we weren't expecting.
Instead of playing with null values directly you can use the Option::none() and to represent some value just call Option::some($value). Later, in the , I'll show how we can use the Option to refactor a small fragment of the Result code example.
class UserService
{
public const USERNAME_ALREADY_TAKEN = 'username_already_taken';
/** @return Result<User, Error[]> */
public function registerUser(UserRegistrationData $data): Result
{
$user = $this->userRepository->find($data->username);
if ($user) {
return err([new Error(
sprintf('Username "%s" is already taken.', $data->username),
self::USERNAME_ALREADY_TAKEN
)]);
}
return User::register($data->username, $this->passwordEncoder->encode($data->password))
->andThen(function (User $user) {
$this->userRepository->save($user);
return $user;
});
}
}
It looks nice, but we can still do more here. As you can guess, it's time for the Option type to shine!
We're going to refactor the following part of the code:
$user = $this->userRepository->find($data->username);
if ($user) {
return err([new Error(
sprintf('Username "%s" is already taken.', $data->username),
self::USERNAME_ALREADY_TAKEN
)]);
}
Let's make the UserRepositoryInterface::find() method to return the Option<User>.
interface UserRepositoryInterface
{
/** @return Option<User> */
public function find(string $username): Option;
public function save(User $user): void;
}
Now, to use the new refactored version of the user repository, we can do it in two ways.
class UserService
{
public const USERNAME_ALREADY_TAKEN = 'username_already_taken';
/** @return Result<User, Error[]> */
public function registerUser(UserRegistrationData $data): Result
{
$op = $this->userRepository->find($data->username);
if ($op->isSome()) {
return err([new Error(
sprintf('Username "%s" is already taken.', $data->username),
self::USERNAME_ALREADY_TAKEN
)]);
}
return User::register($data->username, $this->passwordEncoder->encode($data->password))
->andThen(function (User $user) {
$this->userRepository->save($user);
return $user;
});
}
}
class UserService
{
public const USERNAME_ALREADY_TAKEN = 'username_already_taken';
/** @return Result<User, Error[]> */
public function registerUser(UserRegistrationData $data): Result
{
return $this->userRepository->find($data->username)->mapOrElse(
fn() => User::register($data->username, $this->passwordEncoder->encode($data->password))
->andThen(function (User $user) {
$this->userRepository->save($user);
return $user;
}),
fn() => err([new Error(
sprintf('Username "%s" is already taken.', $data->username),
self::USERNAME_ALREADY_TAKEN
)])
);
}
}
There has been a lot of discussion about the functional version as some have argued it's less readable because it has a more compact form. I think everyone can decide for themselves whether they prefer the functional version or the structural version. The library doesn't impose anything.
I hope that at least this example showed you where you can use the Option type and how easily you can switch between the Result and the Option types.