Typing: Result
Introduction
The Result
type, known as Result
in Rust, OCaml, and F#, and Either
in Haskell, is a powerful programming construct that handles the outcome of an operation, representing either success or error. It provides a structured and type-safe way to handle different outcomes or states. The Result
class in the Dart code includes methods like isOk()
and isError()
to determine if the result is a success or an error. This eliminates the need for manual type checks or null checks, making the code more robust and less error-prone.
The Result
type promotes safer error handling compared to traditional approaches. Explicitly defining success and error cases ensures that all possible outcomes are accounted for. This reduces the risk of unexpected errors or unhandled exceptions. Additionally, the Result
type encourages developers to handle errors gracefully by providing methods like getOrElse()
and getErrorOrElse()
to provide default values in case of errors. This prevents unexpected crashes or undefined behavior.
Using the Result
type also eliminates the need for null references. Instead of relying on null checks, the Result
class provides methods like getOk()
and getError()
that return an Option
type. This enforces the safe handling of nullable values and eliminates the possibility of null pointer exceptions. The Option
type ensures that developers explicitly handle the absence of a value, promoting more reliable and predictable code.
The Result
type’s methods, such as map()
, mapError()
, flatmap()
, and flatmapError()
, allow for seamless composition of operations. These methods enable developers to combine the value inside the result or chain results, ensuring type safety throughout the process. This reduces the likelihood of type-related errors and promotes code readability and maintainability.
The Result
type enhances code reliability and maintainability by providing a structured and type-safe approach to handling different outcomes. It encourages developers to handle errors explicitly, eliminates null references, and promotes a functional programming style. These benefits make the Result
type, known as Result
in Rust, OCaml, and F#, and Either
in Haskell, a safer alternative to traditional error handling approaches, reducing the risk of errors and improving the overall quality of the codebase.
Implementation
import 'option.dart';
sealed class Result<T, U> {
bool isError() {
switch (this) {
case (Ok _):
return false;
case (Error _):
return true;
}
}
bool isOk() {
switch (this) {
case (Ok _):
return true;
case (Error _):
return false;
}
}
bool isEqual(Result<T, U> other) {
switch ((this, other)) {
case (Ok ok, Ok other):
return ok.value == other.value;
case (Error error, Error other):
return error.value == other.value;
default:
return false;
}
}
String toString() {
switch (this) {
case (Ok ok):
return "Ok(${ok.value})";
case (Error error):
return "Error(${error.value})";
}
}
Option<T> getOk() {
switch (this) {
case (Ok ok):
return Some(ok.value);
case (Error _):
return None();
}
}
Option<U> getError() {
switch (this) {
case (Ok _):
return None();
case (Error error):
return Some(error.value);
}
}
T getOrElse(T defVal) {
switch (this) {
case (Ok ok):
return ok.value;
case (Error _):
return defVal;
}
}
U getErrorOrElse(U defVal) {
switch (this) {
case (Ok _):
return defVal;
case (Error error):
return error.value;
}
}
Result<T1, U> map<T1>(T1 Function(T) f) {
switch (this) {
case (Ok ok):
return Ok(f(ok.value));
case (Error error):
return Error(error.value);
}
}
Result<T, U1> mapError<U1>(U1 Function(U) f) {
switch (this) {
case (Ok ok):
return Ok(ok.value);
case (Error error):
return Error(f(error.value));
}
}
Result<T1, U> flatmap<T1>(Result<T1, U> Function(T) f) {
switch (this) {
case (Ok ok):
return f(ok.value);
case (Error error):
return Error(error.value);
}
}
Result<T, U1> flatmapError<U1>(Result<T, U1> Function(U) f) {
switch (this) {
case (Ok ok):
return Ok(ok.value);
case (Error error):
return f(error.value);
}
}
Result<T1, U1> bimap<T1, U1>(T1 Function(T) f, U1 Function(U) g) {
switch (this) {
case (Ok ok):
return Ok(f(ok.value));
case (Error error):
return Error(g(error.value));
}
}
Result<T, U> tap(void Function(T) f) {
switch (this) {
case (Ok ok):
f(ok.value);
return Ok(ok.value);
case (Error error):
return Error(error.value);
}
}
Result<T, U> tapError(void Function(U) f) {
switch (this) {
case (Ok ok):
return Ok(ok.value);
case (Error error):
f(error.value);
return Error(error.value);
}
}
Result<T, U> bitap(void Function(T) f, void Function(U) g) {
switch (this) {
case (Ok ok):
f(ok.value);
return Ok(ok.value);
case (Error error):
g(error.value);
return Error(error.value);
}
}
}
class Ok<T, U> extends Result<T, U> {
final T value;
Ok(this.value);
}
class Error<T, U> extends Result<T, U> {
final U value;
Error(this.value);
}
Explanation of the Dart code:
- The code defines a sealed class
Result<T, U>
that represents a result that can either be anOk
value of typeT
or anError
value of typeU
. This sealed class ensures that all possible result types are explicitly defined and cannot be extended or modified outside of the file. It provides a structured and type-safe way to handle different outcomes or states of an operation. - The
Result
class provides several utility methods to work with the result. By examining its type, theisError()
method checks if the result is an error. TheisOk()
method does the opposite, checking if the result succeeds. These methods allow developers to easily determine the outcome of an operation without resorting to manual type checks or error-prone null checks. - The
isEqual()
method compares two results for equality. It checks if both results are the same type (Ok
orError
) and their values are equal. This method is proper when comparing the results of two operations to determine if they produced the same outcome. - The
toString()
method provides a string representation of the result. It returns a formatted string that includes the type of the result (Ok
orError
) and the value it holds. This method is helpful for debugging and logging purposes, allowing developers to inspect the contents of a result quickly. - The
getOk()
andgetError()
methods retrieve the value wrapped inside the result. ThegetOk()
method returns anOption
type that represents the success value, while thegetError()
method returns anOption
type that represents the error value. These methods allow developers to safely access the value without the risk of null pointer exceptions. - The
getOrElse()
andgetErrorOrElse()
methods retrieve the value from the result or provide a default value if it is an error. ThegetOrElse()
method returns the success value if the result isOk
, or the provided default value if it is anError
. Similarly, thegetErrorOrElse()
method returns the error value if the result isError
, or the provided default value if it isOk
. These methods are helpful when handling errors and providing fallback values. - The
map()
andmapError()
methods allow mapping the value inside the result to a new value. Themap()
method takes a function that transforms the success value and returns a newResult
with the transformed value. ThemapError()
method does the same for the error value. These methods enable developers to perform operations on the result value while preserving the result type. - The
flatmap()
andflatmapError()
methods allow chaining results together. They take a function that produces a newResult
based on the value inside the current result. If the current result isOk
, the function is applied to the success value and returns a newResult
. If the current result isError
, the function is not applied, and the currentError
is returned. These methods help compose operations that depend on the outcome of previous operations. - The
bimap()
method allows simultaneous success and error values mapping. It takes two functions, one for transforming the success value and another for transforming the error value. It returns a newResult
with the transformed values. This method is proper when the success and error values must be modified simultaneously. - The
tap()
,tapError()
, andbitap()
methods allow performing side effects on the value inside the result without modifying it. Thetap()
method takes a function that performs a side effect on the success value, while thetapError()
method does the same for the error value. Thebitap()
method allows side effects on success and error values. These methods are helpful when additional actions need to be taken based on the result value without altering the result itself. - The
Ok
andError
classes are subclasses ofResult
and represent the success and error cases, respectively. They hold the actual values of typesT
andU
. These classes provide a structured way to wrap and access the success and error values within theResult
type.
This code provides a flexible and type-safe way to handle results that can be either successful or erroneous, allowing developers to handle different scenarios effectively.
For more information and examples, you can refer to the Typing: Option resource.
void main() {
Result<String, ({int status, String message})> okRes = Ok("hello world!");
Result<String, ({int status, String message})> errorRes =
Error((status: 404, message: "URL not found"));
print(okRes); // Output: Ok(hello world!)
print(errorRes); // Output: Error((message: URL not found, status: 404)))
print(okRes.isOk()); // Output: true
print(okRes.isError()); // Output: false
print(errorRes.isOk()); // Output: false
print(errorRes.isError()); // Output: true
print(okRes.isEqual(Ok("hello world!"))); // Output: true
print(errorRes
.isEqual(Error((status: 404, message: "URL not found")))); // Output: true
print(okRes.getOk()); // Output: Some(hello world!)
print(okRes.getError()); // Output: None()
print(errorRes.getOk()); // Output: None()
print(errorRes
.getError()); // Output: Some((message: URL not found, status: 404))
print(okRes.getOrElse("default value")); // Output: hello world!
print(errorRes.getOrElse("default value")); // Output: default value
print(okRes.getErrorOrElse((
status: 500,
message: "Internal server error",
))); // Output: (message: Internal server error, status: 500)
print(errorRes.getErrorOrElse((
status: 500,
message: "Internal server error",
))); // Output: (message: URL not found, status: 404)
print(okRes.map((value) => value.toUpperCase())); // Output: Ok(HELLO WORLD!)
print(errorRes.map((value) => value
.toUpperCase())); // Output: Error((message: URL not found, status: 404))
print(okRes.mapError((err) => (
status: err.status,
message: err.message.toUpperCase()
))); // Output: Ok(hello world!)
print(errorRes.mapError((err) => (
status: err.status,
message: err.message.toUpperCase()
))); // Output: Error((message: URL NOT FOUND, status: 404))
print(okRes
.flatmap((value) => Ok(value.toUpperCase()))); // Output: Ok(HELLO WORLD!)
print(errorRes.flatmap((value) => Ok(value
.toUpperCase()))); // Output: Error((message: URL not found, status: 404))
print(okRes.flatmapError((err) => Error((
status: 500,
message: err.message.toUpperCase()
)))); // Output: Ok(hello world!)
print(errorRes.flatmapError((err) => Error((
status: err.status,
message: err.message.toUpperCase()
)))); // Output: Error((message: URL NOT FOUND, status: 404))
print(okRes.bimap(
(value) => value.toUpperCase(),
(err) => (
status: err.status,
message: err.message.toUpperCase()
))); // Output: Ok(HELLO WORLD!)
print(errorRes.bimap(
(value) => value.toUpperCase(),
(err) => (
status: err.status,
message: err.message.toUpperCase()
))); // Output: Error((message: URL NOT FOUND, status: 404))
print(okRes.tap((value) =>
print(value))); // Output: print hello world! then Ok(hello world!)
print(errorRes.tap((value) =>
print(value))); // Output: Error((message: URL not found, status: 404))
print(
okRes.tapError((err) => print(err.message))); // Output: Ok(hello world!)
print(errorRes.tapError((err) => print(err
.message))); // Output: print URL not found then Error((message: URL not found, status: 404))
print(okRes.bitap(
(value) => print(value),
(err) => print(
err.message))); // Output: print hello world! then Ok(hello world!)
print(errorRes.bitap(
(value) => print(value),
(err) => print(err
.message))); // Output: print URL not found then Error((message: URL not found, status: 404)
}
Let’s break down the main
function step-by-step:
- The code demonstrates the usage of the
Result
class and its methods. It creates two instances ofResult
-okRes
anderrorRes
.okRes
represents a successful result with the value “hello world!”, whileerrorRes
represents an error result with the value{status: 404, message: "URL not found"}
. - The
print()
statements are used to output the results of various operations on theResult
instances. - The
isOk()
andisError()
methods are used to check if a result is a success or an error, respectively. The output of these methods indicates whether the result is anOk
or anError
. - The
isEqual()
method is used to compare two results for equality. It checks if the values and types of the results match. - The
getOk()
andgetError()
methods retrieve the success and error values from theResult
instances, respectively. These methods return anOption
type representing the value, allowing for safe handling of nullable values. - The
getOrElse()
andgetErrorOrElse()
methods retrieve the value from the result or provide a default value if it is an error. These methods return the value if the result isOk
, or the provided default value if it is anError
. - The
map()
,mapError()
,flatmap()
, andflatmapError()
methods are used to transform the value inside the result or chain results together. These methods apply functions to the value or error value and return a newResult
instance with the transformed value. - The
bimap()
method allows simultaneous success and error values mapping. It applies functions to both values and returns a newResult
instance with the transformed values. - The
tap()
,tapError()
, andbitap()
methods are used to perform side effects on the value inside the result without modifying it. These methods execute functions with the value or error value and return the originalResult
instance.
The output of each operation is shown as comments in the code. This code demonstrates how the Result
class can handle different outcomes or states of an operation in a structured and type-safe manner.