mirror of
https://github.com/status-im/nim-stew.git
synced 2025-01-24 10:59:13 +00:00
add nim-result
This commit is contained in:
parent
1edeabb453
commit
f5be0ab63a
389
stew/result.nim
Normal file
389
stew/result.nim
Normal file
@ -0,0 +1,389 @@
|
|||||||
|
# nim-result is also available stand-alone from https://github.com/arnetheduck/nim-result/
|
||||||
|
|
||||||
|
# Copyright (c) 2019 Jacek Sieka
|
||||||
|
# Licensed and distributed under either of
|
||||||
|
# * MIT license (license terms in the root directory or at http://opensource.org/licenses/MIT).
|
||||||
|
# * Apache v2 license (license terms in the root directory or at http://www.apache.org/licenses/LICENSE-2.0).
|
||||||
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
||||||
|
|
||||||
|
type
|
||||||
|
ResultError*[E] = ref object of ValueError
|
||||||
|
## Error raised when trying to access value of result when error is set
|
||||||
|
## Note: If error is of exception type, it will be raised instead!
|
||||||
|
error: E
|
||||||
|
|
||||||
|
Result*[T, E] = object
|
||||||
|
## Result type that can hold either a value or an error, but not both
|
||||||
|
##
|
||||||
|
## # Example
|
||||||
|
##
|
||||||
|
## ```
|
||||||
|
## # It's convenient to create an alias - most likely, you'll do just fine
|
||||||
|
## # with strings as error!
|
||||||
|
##
|
||||||
|
## type R = Result[int, string]
|
||||||
|
##
|
||||||
|
## # Once you have a type, use `ok` and `err`:
|
||||||
|
##
|
||||||
|
## func works(): R =
|
||||||
|
## # ok says it went... ok!
|
||||||
|
## R.ok 42
|
||||||
|
## func fails(): R =
|
||||||
|
## # or type it like this, to not repeat the type!
|
||||||
|
## result.err "bad luck"
|
||||||
|
##
|
||||||
|
## if (let w = works(); w.isOk):
|
||||||
|
## echo w[], " or use value: ", w.value
|
||||||
|
##
|
||||||
|
## # In case you think your callers want to differentiate between errors:
|
||||||
|
## type
|
||||||
|
## Error = enum
|
||||||
|
## a, b, c
|
||||||
|
## type RE[T] = Result[T, Error]
|
||||||
|
##
|
||||||
|
## # In the expriments corner, you'll find the following syntax for passing
|
||||||
|
## # errors up the stack:
|
||||||
|
## func f(): R =
|
||||||
|
## let x = ?works() - ?fails()
|
||||||
|
## assert false, "will never reach"
|
||||||
|
##
|
||||||
|
## # If you provide this exception converter, this exception will be raised
|
||||||
|
## # on dereference
|
||||||
|
## func toException(v: Error): ref CatchableException = (ref CatchableException)(msg: $v)
|
||||||
|
## try:
|
||||||
|
## RE[int].err(a)[]
|
||||||
|
## except CatchableException:
|
||||||
|
## echo "in here!"
|
||||||
|
##
|
||||||
|
## ```
|
||||||
|
##
|
||||||
|
## See the tests for more practical examples, specially when working with
|
||||||
|
## back and forth with the exception world!
|
||||||
|
##
|
||||||
|
## # Potential benefits:
|
||||||
|
##
|
||||||
|
## * Handling errors becomes explicit and mandatory - goodbye "out of sight,
|
||||||
|
## out of mind"
|
||||||
|
## * Errors are a visible part of the API - when they change, so must the
|
||||||
|
## calling code and compiler will point this out - nice!
|
||||||
|
## * Errors are a visible part of the API - your fellow programmer is
|
||||||
|
## reminded that things actually can go wrong
|
||||||
|
## * Jives well with Nim `discard`
|
||||||
|
## * Jives well with the new Defect exception hierarchy, where defects
|
||||||
|
## are raised for unrecoverable errors and the rest of the API uses
|
||||||
|
## results
|
||||||
|
## * Error and value return have similar performance characteristics
|
||||||
|
## * Caller can choose to turn them into exceptions at low cost - flexible
|
||||||
|
## for libraries!
|
||||||
|
## * Mostly relies on simple Nim features - though this library is no
|
||||||
|
## exception in that compiler bugs were discovered writing it :)
|
||||||
|
##
|
||||||
|
## # Potential costs:
|
||||||
|
##
|
||||||
|
## * Handling errors becomes explicit and mandatory - if you'd rather ignore
|
||||||
|
## them or just pass them to some catch-all, this is noise
|
||||||
|
## * When composing operations, value must be lifted before funcessing,
|
||||||
|
## adding potential verbosity / noise (fancy macro, anyone?)
|
||||||
|
## * There's no call stack captured by default (see also `catch` and
|
||||||
|
## `capture`)
|
||||||
|
## * The extra branching may be more expensive for the non-error path
|
||||||
|
## (though this can be minimized with PGO)
|
||||||
|
##
|
||||||
|
## The API visibility issue of exceptions can also be solved with
|
||||||
|
## `{.raises.}` annotations - as of now, the compiler doesn't remind
|
||||||
|
## you to do so, even though it knows what the right annotation should be.
|
||||||
|
##
|
||||||
|
## Many system languages make a distinction between errors you want to
|
||||||
|
## handle and those that are simply bugs or unrealistic to deal with..
|
||||||
|
## handling the latter will often involve aborting or crashing the funcess -
|
||||||
|
## reliable systems like Erlang will try to relaunch it.
|
||||||
|
##
|
||||||
|
## On the flip side we have dynamic languages like python where there's
|
||||||
|
## nothing exceptional about exceptions (hello StopIterator). Python is
|
||||||
|
## rarely used to build reliable systems - its strengths lie elsewhere.
|
||||||
|
##
|
||||||
|
## # Other languages
|
||||||
|
##
|
||||||
|
## Result-style error handling seems pretty popular lately, specially with
|
||||||
|
## statically typed languages:
|
||||||
|
## Haskell: https://hackage.haskell.org/package/base-4.11.1.0/docs/Data-Either.html
|
||||||
|
## Rust: https://doc.rust-lang.org/std/result/enum.Result.html
|
||||||
|
## Modern C++: https://github.com/viboes/std-make/tree/master/doc/proposal/expected
|
||||||
|
## More C++: https://github.com/ned14/outcome
|
||||||
|
##
|
||||||
|
## Swift is interesting in that it uses a non-exception implementation but
|
||||||
|
## calls errors exceptions and has lots of syntactic sugar to make them feel
|
||||||
|
## that way by implicitly passing them up the call chain - with a mandatory
|
||||||
|
## annotation that function may throw:
|
||||||
|
## https://developer.apple.com/library/content/documentation/Swift/Conceptual/Swift_Programming_Language/ErrorHandling.html
|
||||||
|
##
|
||||||
|
## # Other implemenations in nim
|
||||||
|
##
|
||||||
|
## There are other implementations in nim that you might prefer:
|
||||||
|
## * Either from nimfp: https://github.com/vegansk/nimfp/blob/master/src/fp/either.nim
|
||||||
|
## * result_type: https://github.com/kapralos/result_type/
|
||||||
|
##
|
||||||
|
## # Implementation notes
|
||||||
|
##
|
||||||
|
## This implementation is mostly based on the one in rust. Compared to it,
|
||||||
|
## there are a few differences - if know of creative ways to improve things,
|
||||||
|
## I'm all ears.
|
||||||
|
##
|
||||||
|
## * Rust has the enum variants which lend themselves to nice construction
|
||||||
|
## where the full Result type isn't needed: `Err("some error")` doesn't
|
||||||
|
## need to know value type - maybe some creative converter or something
|
||||||
|
## can deal with this?
|
||||||
|
## * Nim templates allow us to fail fast without extra effort, meaning the
|
||||||
|
## other side of `and`/`or` isn't evaluated unless necessary - nice!
|
||||||
|
## * In Nim, we have exceptions - when using this library, we'll raise the
|
||||||
|
## standard crop of Nim errors when trying to access the error of a value
|
||||||
|
## and vice versa - this fits better with Nim but costs space and
|
||||||
|
## performance - need to think about this - rust will simply panic, and
|
||||||
|
## everyone seems more or less happy with that..
|
||||||
|
## * Rust uses From traits to deal with result translation as the result
|
||||||
|
## travels up the call stack - needs more tinkering - some implicit
|
||||||
|
## conversions would be nice here
|
||||||
|
|
||||||
|
case o: bool
|
||||||
|
of false:
|
||||||
|
e: E
|
||||||
|
of true:
|
||||||
|
v: T
|
||||||
|
|
||||||
|
func raiseResultError[T, E](self: Result[T, E]) {.noreturn.} =
|
||||||
|
when E is ref Exception:
|
||||||
|
if self.e.isNil: # for example Result.default()!
|
||||||
|
raise ResultError[void](msg: "Trying to access value with err (nil)")
|
||||||
|
raise self.e
|
||||||
|
elif compiles(self.e.toException()):
|
||||||
|
raise self.e.toException()
|
||||||
|
elif compiles($self.e):
|
||||||
|
raise ResultError[E](
|
||||||
|
error: self.e, msg: "Trying to access value with err: " & $self.e)
|
||||||
|
else:
|
||||||
|
raise ResultError[E](error: self.e)
|
||||||
|
|
||||||
|
template ok*(R: type Result, x: auto): auto =
|
||||||
|
## Initialize a result with a success and value
|
||||||
|
## Example: `Result[int, string].ok(42)`
|
||||||
|
R(o: true, v: x)
|
||||||
|
|
||||||
|
template ok*(self: var Result, x: auto) =
|
||||||
|
## Set the result to success and update value
|
||||||
|
## Example: `result.ok(42)`
|
||||||
|
self = ok(type self, x)
|
||||||
|
|
||||||
|
template err*(R: type Result, x: auto): auto =
|
||||||
|
## Initialize the result to an error
|
||||||
|
## Example: `Result[int, string].err("uh-oh")`
|
||||||
|
R(o: false, e: x)
|
||||||
|
|
||||||
|
template err*(self: var Result, x: auto) =
|
||||||
|
## Set the result as an error
|
||||||
|
## Example: `result.err("uh-oh")`
|
||||||
|
self = err(type self, x)
|
||||||
|
|
||||||
|
template ok*(v: auto): auto = typeof(result).ok(v)
|
||||||
|
template err*(v: auto): auto = typeof(result).err(v)
|
||||||
|
|
||||||
|
template isOk*(self: Result): bool = self.o
|
||||||
|
template isErr*(self: Result): bool = not self.o
|
||||||
|
|
||||||
|
func map*[T, E, A](
|
||||||
|
self: Result[T, E], f: proc(x: T): A): Result[A, E] {.inline.} =
|
||||||
|
## Transform value using f, or return error
|
||||||
|
if self.isOk: result.ok(f(self.v))
|
||||||
|
else: result.err(self.e)
|
||||||
|
|
||||||
|
func flatMap*[T, E, A](
|
||||||
|
self: Result[T, E], f: proc(x: T): Result[A, E]): Result[A, E] {.inline.} =
|
||||||
|
if self.isOk: f(self.v)
|
||||||
|
else: Result[A, E].err(self.e)
|
||||||
|
|
||||||
|
func mapErr*[T: not void, E, A](
|
||||||
|
self: Result[T, E], f: proc(x: E): A): Result[T, A] {.inline.} =
|
||||||
|
## Transform error using f, or return value
|
||||||
|
if self.isOk: result.ok(self.v)
|
||||||
|
else: result.err(f(self.e))
|
||||||
|
|
||||||
|
func mapConvert*[T0, E0](
|
||||||
|
self: Result[T0, E0], T1: type): Result[T1, E0] {.inline.} =
|
||||||
|
## Convert result value to A using an implicit conversion
|
||||||
|
## Would be nice if it was automatic...
|
||||||
|
if self.isOk: result.ok(self.v)
|
||||||
|
else: result.err(self.e)
|
||||||
|
|
||||||
|
func mapCast*[T0, E0](
|
||||||
|
self: Result[T0, E0], T1: type): Result[T1, E0] {.inline.} =
|
||||||
|
## Convert result value to A using a cast
|
||||||
|
## Would be nice with nicer syntax...
|
||||||
|
if self.isOk: result.ok(cast[T1](self.v))
|
||||||
|
else: result.err(self.e)
|
||||||
|
|
||||||
|
template `and`*(self: Result, other: untyped): untyped =
|
||||||
|
## Evaluate `other` iff self.isOk, else return error
|
||||||
|
## fail-fast - will not evaluate other if a is an error
|
||||||
|
if self.isOk:
|
||||||
|
other
|
||||||
|
else:
|
||||||
|
type R = type(other)
|
||||||
|
R.err(self.e)
|
||||||
|
|
||||||
|
template `or`*(self: Result, other: untyped): untyped =
|
||||||
|
## Evaluate `other` iff not self.isOk, else return self
|
||||||
|
## fail-fast - will not evaluate other if a is a value
|
||||||
|
if self.isOk: self
|
||||||
|
else: other
|
||||||
|
|
||||||
|
template catch*(body: typed): Result[type(body), ref Exception] =
|
||||||
|
## Convert a try expression into a Result
|
||||||
|
type R = Result[type(body), ref Exception]
|
||||||
|
|
||||||
|
try:
|
||||||
|
R.ok(body)
|
||||||
|
except:
|
||||||
|
R.err(getCurrentException())
|
||||||
|
|
||||||
|
template capture*(T: type, e: ref Exception): Result[T, ref Exception] =
|
||||||
|
type R = Result[T, ref Exception]
|
||||||
|
|
||||||
|
var ret: R
|
||||||
|
try:
|
||||||
|
# TODO is this needed? I think so, in order to grab a call stack, but
|
||||||
|
# haven't actually tested...
|
||||||
|
if true:
|
||||||
|
# I'm sure there's a nicer way - this just works :)
|
||||||
|
raise e
|
||||||
|
except:
|
||||||
|
ret = R.err(getCurrentException())
|
||||||
|
ret
|
||||||
|
|
||||||
|
func `==`*(lhs, rhs: Result): bool {.inline.} =
|
||||||
|
if lhs.isOk != rhs.isOk:
|
||||||
|
false
|
||||||
|
elif lhs.isOk:
|
||||||
|
lhs.v == rhs.v
|
||||||
|
else:
|
||||||
|
lhs.e == rhs.e
|
||||||
|
|
||||||
|
func get*[T: not void, E](self: Result[T, E]): T {.inline.} =
|
||||||
|
## Fetch value of result if set, or raise error as an Exception
|
||||||
|
## See also: Option.get
|
||||||
|
if self.isErr: self.raiseResultError()
|
||||||
|
|
||||||
|
self.v
|
||||||
|
|
||||||
|
func get*[T, E](self: Result[T, E], otherwise: T): T {.inline.} =
|
||||||
|
## Fetch value of result if set, or raise error as an Exception
|
||||||
|
## See also: Option.get
|
||||||
|
if self.isErr: otherwise
|
||||||
|
else: self.v
|
||||||
|
|
||||||
|
func get*[T, E](self: var Result[T, E]): var T {.inline.} =
|
||||||
|
## Fetch value of result if set, or raise error as an Exception
|
||||||
|
## See also: Option.get
|
||||||
|
if self.isErr: self.raiseResultError()
|
||||||
|
|
||||||
|
self.v
|
||||||
|
|
||||||
|
template `[]`*[T, E](self: Result[T, E]): T =
|
||||||
|
## Fetch value of result if set, or raise error as an Exception
|
||||||
|
self.get()
|
||||||
|
|
||||||
|
template `[]`*[T, E](self: var Result[T, E]): var T =
|
||||||
|
## Fetch value of result if set, or raise error as an Exception
|
||||||
|
self.get()
|
||||||
|
|
||||||
|
template unsafeGet*[T, E](self: Result[T, E]): T =
|
||||||
|
## Fetch value of result if set, undefined behavior if unset
|
||||||
|
## See also: Option.unsafeGet
|
||||||
|
assert not isErr(self)
|
||||||
|
|
||||||
|
self.v
|
||||||
|
|
||||||
|
func `$`*(self: Result): string =
|
||||||
|
## Returns string representation of `self`
|
||||||
|
if self.isOk: "Ok(" & $self.v & ")"
|
||||||
|
else: "Err(" & $self.e & ")"
|
||||||
|
|
||||||
|
func error*[T, E](self: Result[T, E]): E =
|
||||||
|
if self.isOk: raise ResultError[void](msg: "Result does not contain an error")
|
||||||
|
|
||||||
|
self.e
|
||||||
|
|
||||||
|
template value*[T, E](self: Result[T, E]): T = self.get()
|
||||||
|
template value*[T, E](self: var Result[T, E]): T = self.get()
|
||||||
|
|
||||||
|
template valueOr*[T, E](self: Result[T, E], def: T): T =
|
||||||
|
## Fetch value of result if set, or supplied default
|
||||||
|
## default will not be evaluated iff value is set
|
||||||
|
self.get(def)
|
||||||
|
|
||||||
|
# void support
|
||||||
|
|
||||||
|
template ok*[E](R: type Result[void, E]): auto =
|
||||||
|
## Initialize a result with a success and value
|
||||||
|
## Example: `Result[int, string].ok(42)`
|
||||||
|
R(o: true)
|
||||||
|
|
||||||
|
template ok*[E](self: var Result[void, E]) =
|
||||||
|
## Set the result to success and update value
|
||||||
|
## Example: `result.ok(42)`
|
||||||
|
self = (type self).ok()
|
||||||
|
|
||||||
|
func map*[E, A](
|
||||||
|
self: Result[void, E], f: proc(): A): Result[A, E] {.inline.} =
|
||||||
|
## Transform value using f, or return error
|
||||||
|
if self.isOk: result.ok(f())
|
||||||
|
else: result.err(self.e)
|
||||||
|
|
||||||
|
func flatMap*[E, A](
|
||||||
|
self: Result[void, E], f: proc(): Result[A, E]): Result[A, E] {.inline.} =
|
||||||
|
if self.isOk: f(self.v)
|
||||||
|
else: Result[A, E].err(self.e)
|
||||||
|
|
||||||
|
func mapErr*[E, A](
|
||||||
|
self: Result[void, E], f: proc(x: E): A): Result[void, A] {.inline.} =
|
||||||
|
## Transform error using f, or return value
|
||||||
|
if self.isOk: result.ok()
|
||||||
|
else: result.err(f(self.e))
|
||||||
|
|
||||||
|
func map*[T, E](
|
||||||
|
self: Result[T, E], f: proc(x: T)): Result[void, E] {.inline.} =
|
||||||
|
## Transform value using f, or return error
|
||||||
|
if self.isOk: f(self.v); result.ok()
|
||||||
|
else: result.err(self.e)
|
||||||
|
|
||||||
|
func get*[E](self: Result[void, E]) {.inline.} =
|
||||||
|
## Fetch value of result if set, or raise error as an Exception
|
||||||
|
## See also: Option.get
|
||||||
|
if self.isErr: self.raiseResultError()
|
||||||
|
|
||||||
|
template `[]`*[E](self: Result[void, E]) =
|
||||||
|
## Fetch value of result if set, or raise error as an Exception
|
||||||
|
self.get()
|
||||||
|
|
||||||
|
template unsafeGet*[E](self: Result[void, E]) =
|
||||||
|
## Fetch value of result if set, undefined behavior if unset
|
||||||
|
## See also: Option.unsafeGet
|
||||||
|
assert not self.isErr
|
||||||
|
|
||||||
|
func `$`*[E](self: Result[void, E]): string =
|
||||||
|
## Returns string representation of `self`
|
||||||
|
if self.isOk: "Ok()"
|
||||||
|
else: "Err(" & $self.e & ")"
|
||||||
|
|
||||||
|
template value*[E](self: Result[void, E]) = self.get()
|
||||||
|
template value*[E](self: var Result[void, E]) = self.get()
|
||||||
|
|
||||||
|
template `?`*[T, E](self: Result[T, E]): T =
|
||||||
|
## Early return - if self is an error, we will return from the current
|
||||||
|
## function, else we'll move on..
|
||||||
|
## Experimental
|
||||||
|
# TODO the v copy is here to prevent multiple evaluations of self - could
|
||||||
|
# probably avoid it with some fancy macro magic..
|
||||||
|
let v = self
|
||||||
|
if v.isErr: return v
|
||||||
|
|
||||||
|
v.value
|
||||||
|
|
@ -9,12 +9,13 @@
|
|||||||
|
|
||||||
import
|
import
|
||||||
ranges/all,
|
ranges/all,
|
||||||
|
test_base32,
|
||||||
|
test_base58,
|
||||||
|
test_base64,
|
||||||
test_bitops2,
|
test_bitops2,
|
||||||
test_bitseqs,
|
test_bitseqs,
|
||||||
test_byteutils,
|
test_byteutils,
|
||||||
test_endians2,
|
test_endians2,
|
||||||
test_ptrops,
|
test_ptrops,
|
||||||
test_varints,
|
test_result,
|
||||||
test_base32,
|
test_varints
|
||||||
test_base58,
|
|
||||||
test_base64
|
|
||||||
|
241
tests/test_result.nim
Normal file
241
tests/test_result.nim
Normal file
@ -0,0 +1,241 @@
|
|||||||
|
# nim-result is also available stand-alone from https://github.com/arnetheduck/nim-result/
|
||||||
|
|
||||||
|
import ../stew/result
|
||||||
|
type R = Result[int, string]
|
||||||
|
|
||||||
|
# Basic usage, producer
|
||||||
|
func works(): R = R.ok(42)
|
||||||
|
func works2(): R = result.ok(42)
|
||||||
|
func fails(): R = R.err("dummy")
|
||||||
|
func fails2(): R = result.err("dummy")
|
||||||
|
|
||||||
|
func raises(): int =
|
||||||
|
raise newException(Exception, "hello")
|
||||||
|
|
||||||
|
# Basic usage, consumer
|
||||||
|
let
|
||||||
|
rOk = works()
|
||||||
|
rOk2 = works2()
|
||||||
|
rErr = fails()
|
||||||
|
rErr2 = fails2()
|
||||||
|
|
||||||
|
doAssert rOk.isOk
|
||||||
|
doAssert rOk2.isOk
|
||||||
|
doAssert rOk.get() == 42
|
||||||
|
doAssert (not rOk.isErr)
|
||||||
|
doAssert rErr.isErr
|
||||||
|
doAssert rErr2.isErr
|
||||||
|
|
||||||
|
# Combine
|
||||||
|
doAssert (rOk and rErr).isErr
|
||||||
|
doAssert (rErr and rOk).isErr
|
||||||
|
doAssert (rOk or rErr).isOk
|
||||||
|
doAssert (rErr or rOk).isOk
|
||||||
|
|
||||||
|
# Exception on access
|
||||||
|
let va = try: discard rOk.error; false except: true
|
||||||
|
doAssert va, "not an error, should raise"
|
||||||
|
|
||||||
|
# Exception on access
|
||||||
|
let vb = try: discard rErr.value; false except: true
|
||||||
|
doAssert vb, "not an value, should raise"
|
||||||
|
|
||||||
|
var x = rOk
|
||||||
|
|
||||||
|
# Mutate
|
||||||
|
x.err("failed now")
|
||||||
|
|
||||||
|
doAssert x.isErr
|
||||||
|
|
||||||
|
# Exceptions -> results
|
||||||
|
let c = catch:
|
||||||
|
raises()
|
||||||
|
|
||||||
|
doAssert c.isErr
|
||||||
|
|
||||||
|
# De-reference
|
||||||
|
try:
|
||||||
|
echo rErr[]
|
||||||
|
doAssert false
|
||||||
|
except:
|
||||||
|
discard
|
||||||
|
|
||||||
|
doAssert rOk.valueOr(50) == rOk.value
|
||||||
|
doAssert rErr.valueOr(50) == 50
|
||||||
|
|
||||||
|
# Comparisons
|
||||||
|
doAssert (works() == works2())
|
||||||
|
doAssert (fails() == fails2())
|
||||||
|
doAssert (works() != fails())
|
||||||
|
|
||||||
|
var counter = 0
|
||||||
|
proc incCounter(): R =
|
||||||
|
counter += 1
|
||||||
|
R.ok(counter)
|
||||||
|
|
||||||
|
doAssert (rErr and incCounter()).isErr, "b fails"
|
||||||
|
doAssert counter == 0, "should fail fast on rErr"
|
||||||
|
|
||||||
|
# Mapping
|
||||||
|
doAssert (rOk.map(func(x: int): string = $x)[] == $rOk.value)
|
||||||
|
doAssert (rOk.flatMap(
|
||||||
|
proc(x: int): Result[string, string] = Result[string, string].ok($x))[] == $rOk.value)
|
||||||
|
doAssert (rErr.mapErr(func(x: string): string = x & "no!").error == (rErr.error & "no!"))
|
||||||
|
|
||||||
|
# Exception interop
|
||||||
|
let e = capture(int, newException(Exception, "test"))
|
||||||
|
doAssert e.isErr
|
||||||
|
try:
|
||||||
|
discard e[]
|
||||||
|
doAssert false, "should have raised"
|
||||||
|
except:
|
||||||
|
doAssert getCurrentException().msg == "test"
|
||||||
|
|
||||||
|
# Nice way to checks
|
||||||
|
if (let v = works(); v.isOk):
|
||||||
|
doAssert v[] == v.value
|
||||||
|
|
||||||
|
# Can formalise it into a template (https://github.com/arnetheduck/nim-result/issues/8)
|
||||||
|
template `?=`*(v: untyped{nkIdent}, vv: Result): bool =
|
||||||
|
(let vr = vv; template v: auto {.used.} = unsafeGet(vr); vr.isOk)
|
||||||
|
if f ?= works():
|
||||||
|
doAssert f == works().value
|
||||||
|
|
||||||
|
doAssert $rOk == "Ok(42)"
|
||||||
|
|
||||||
|
doAssert rOk.mapConvert(int64)[] == int64(42)
|
||||||
|
doAssert rOk.mapCast(int8)[] == int8(42)
|
||||||
|
|
||||||
|
# TODO there's a bunch of operators that one could lift through magic - this
|
||||||
|
# is mainly an example
|
||||||
|
template `+`*(self, other: Result): untyped =
|
||||||
|
## Perform `+` on the values of self and other, if both are ok
|
||||||
|
type R = type(other)
|
||||||
|
if self.isOk:
|
||||||
|
if other.isOk:
|
||||||
|
R.ok(self.value + other.value)
|
||||||
|
else:
|
||||||
|
R.err(other.error)
|
||||||
|
else:
|
||||||
|
R.err(self.error)
|
||||||
|
|
||||||
|
# Simple lifting..
|
||||||
|
doAssert (rOk + rOk)[] == rOk.value + rOk.value
|
||||||
|
|
||||||
|
iterator items[T, E](self: Result[T, E]): T =
|
||||||
|
## Iterate over result as if it were a collection of either 0 or 1 items
|
||||||
|
## TODO should a Result[seq[X]] iterate over items in seq? there are
|
||||||
|
## arguments for and against
|
||||||
|
if self.isOk:
|
||||||
|
yield self.value
|
||||||
|
|
||||||
|
# Iteration
|
||||||
|
var counter2 = 0
|
||||||
|
for v in rOk:
|
||||||
|
counter2 += 1
|
||||||
|
|
||||||
|
doAssert counter2 == 1, "one-item collection when set"
|
||||||
|
|
||||||
|
func testOk(): Result[int, string] =
|
||||||
|
ok 42
|
||||||
|
|
||||||
|
func testErr(): Result[int, string] =
|
||||||
|
err "323"
|
||||||
|
|
||||||
|
doAssert testOk()[] == 42
|
||||||
|
doAssert testErr().error == "323"
|
||||||
|
|
||||||
|
# It's also possible to use the same trick for stack capture:
|
||||||
|
template capture*(): untyped =
|
||||||
|
type R = type(result)
|
||||||
|
|
||||||
|
var ret: R
|
||||||
|
try:
|
||||||
|
# TODO is this needed? I think so, in order to grab a call stack, but
|
||||||
|
# haven't actually tested...
|
||||||
|
if true:
|
||||||
|
# I'm sure there's a nicer way - this just works :)
|
||||||
|
raise newException(Exception, "")
|
||||||
|
except:
|
||||||
|
ret = R.err(getCurrentException())
|
||||||
|
ret
|
||||||
|
|
||||||
|
proc testCapture(): Result[int, ref Exception] =
|
||||||
|
return capture()
|
||||||
|
|
||||||
|
doAssert testCapture().isErr
|
||||||
|
|
||||||
|
func testQn(): Result[int, string] =
|
||||||
|
let x = ?works() - ?works()
|
||||||
|
result.ok(x)
|
||||||
|
|
||||||
|
func testQn2(): Result[int, string] =
|
||||||
|
# looks like we can even use it creatively like this
|
||||||
|
if ?fails() == 42: raise newException(Exception, "shouldn't happen")
|
||||||
|
|
||||||
|
doAssert testQn()[] == 0
|
||||||
|
doAssert testQn2().isErr
|
||||||
|
|
||||||
|
type
|
||||||
|
AnEnum = enum
|
||||||
|
anEnumA
|
||||||
|
anEnumB
|
||||||
|
AnException = ref object of Exception
|
||||||
|
v: AnEnum
|
||||||
|
|
||||||
|
func toException(v: AnEnum): AnException = AnException(v: v)
|
||||||
|
|
||||||
|
func testToException(): int =
|
||||||
|
try:
|
||||||
|
var r = Result[int, AnEnum].err(anEnumA)
|
||||||
|
r[]
|
||||||
|
except AnException:
|
||||||
|
42
|
||||||
|
|
||||||
|
doAssert testToException() == 42
|
||||||
|
|
||||||
|
type
|
||||||
|
AnEnum2 = enum
|
||||||
|
anEnum2A
|
||||||
|
anEnum2B
|
||||||
|
|
||||||
|
func testToString(): int =
|
||||||
|
try:
|
||||||
|
var r = Result[int, AnEnum2].err(anEnum2A)
|
||||||
|
r[]
|
||||||
|
except ResultError[AnEnum2]:
|
||||||
|
42
|
||||||
|
|
||||||
|
doAssert testToString() == 42
|
||||||
|
|
||||||
|
type VoidRes = Result[void, int]
|
||||||
|
|
||||||
|
func worksVoid(): VoidRes = VoidRes.ok()
|
||||||
|
func worksVoid2(): VoidRes = result.ok()
|
||||||
|
func failsVoid(): VoidRes = VoidRes.err(42)
|
||||||
|
func failsVoid2(): VoidRes = result.err(42)
|
||||||
|
|
||||||
|
let
|
||||||
|
vOk = worksVoid()
|
||||||
|
vOk2 = worksVoid2()
|
||||||
|
vErr = failsVoid()
|
||||||
|
vErr2 = failsVoid2()
|
||||||
|
|
||||||
|
doAssert vOk.isOk
|
||||||
|
doAssert vOk2.isOk
|
||||||
|
doAssert vErr.isErr
|
||||||
|
doAssert vErr2.isErr
|
||||||
|
|
||||||
|
vOk.get()
|
||||||
|
|
||||||
|
doAssert vOk.map(proc (): int = 42).get() == 42
|
||||||
|
|
||||||
|
rOk.map(proc(x: int) = discard).get()
|
||||||
|
|
||||||
|
try:
|
||||||
|
rErr.map(proc(x: int) = discard).get()
|
||||||
|
doAssert false
|
||||||
|
except:
|
||||||
|
discard
|
||||||
|
|
||||||
|
doAssert vErr.mapErr(proc(x: int): int = 10).error() == 10
|
Loading…
x
Reference in New Issue
Block a user