-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
2 changed files
with
89 additions
and
88 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,89 @@ | ||
/* | ||
Package chainable provides an easy and convenient way of chaining function calls in Golang. | ||
Basics | ||
It's common to find Go programs similar to the one below: | ||
x, err := sum2(2) | ||
if err != nil { | ||
return 0, err | ||
} | ||
y, err = mul2(x) | ||
if err != nil { | ||
return 0, err | ||
} | ||
z, err = div2(y) | ||
if err != nil { | ||
return 0, err | ||
} | ||
return z, nil | ||
Error handling can make the codebase messy. At the same time, chaining function calls can obfuscate | ||
errors and cascading calls can increase the cognitive complexity. | ||
Elixir, F# and other languages solve this problem with the support to pipes, but Golang hasn't such | ||
feature. | ||
Chainable provides a clearer way to chain function calls, using the output of the previous function | ||
as the input of the next one. The example above could be re-written using Chainable as follow: | ||
import (chainable "github.com/mauricioklein/go-chainable") | ||
res, err := chainable.New(). | ||
From(2). | ||
Chain( | ||
sum2, | ||
mul2, | ||
div2, | ||
). | ||
Unwrap() | ||
// "Unwrap" returns the result as a slice of Argument, matching the return values | ||
// of the last function. So, we just need to cast them to the correct type | ||
z := res[0].(int) | ||
Another advantage is that Chainable automatically handle errors in a chain. | ||
Thus, if one of the methods returns an error as the last argument, the chain is broken | ||
and the error is returned by the "Unwrap" method: | ||
raiseError := func () (int, error) { return 0, errors.New("generic error") } | ||
plus2 := func (x int) { return x + 2 } | ||
chainable.New(). | ||
Chain(raiseError). // breakes the chain | ||
Chain(plus2). // never called | ||
Unwrap() // returns nil, "a generic error" | ||
If automatic error handling isn't desired (i.e. the error should be chained along with the other arguments), | ||
the method "ChainDummy" should be used instead of "Chain". Pay attention that the next function in the chain | ||
must be able to receive the error generated by the previous one: | ||
raiseError := func () (int, error) { return 2, errors.New("generic error") } | ||
plus2AndError := func (x int, e error) (int) { return x + 2 } | ||
chainable.New(). | ||
ChainDummy(raiseError). | ||
Chain(plus2AndError). | ||
Unwrap() // returns []Argument{4}, nil | ||
"Chain" and "DummyChain" methods are variadics, and can be used in conjunction: | ||
chainable.New(). | ||
Chain(f1). // error handling enabled for f1 | ||
ChainDummy(f2). // error handling disabled for f2 | ||
Chain(f3, f4). // error handling enabled for f3 and f4 | ||
Unwrap() | ||
Finally, to reset a chain and make it ready to be reused, just call the method "Reset": | ||
chain := chainable.New() | ||
... use of chain | ||
chain.Reset() | ||
*/ | ||
package chainable |