Go Promise aims to be a Promise/Future alternative of bluebird in golang. It will give all the various functionalities provided by any standard promise library along with something more. Now you may ask why should I use Promise/Futures in go-lang. The language never intended for it So why should I. The answer is to orchestrate your go routines. The go-promise here provided launches goroutines for all callbacks making it suitable for golang , but always uses mutexes and semaphores internally to synchronise between themselved. It gives you common synchronization patterns to work with. Currently the API supports much of the common use cases. More will be coming , as I find much usefull patterns in go which can be incorporated inside a promise library.
$ go get github.com/rajatkb/go-promise
- Create a single promise
import Promise "github.com/rajatkb/go-promise"
// Create a single
promise := Promise.Create(func(resolve Promise.Callback, reject Promise.Callback) {
resolve(2)
})
- Then & Catch in for same promise 🔥
w.Add(1)
Promise.Create(func(resolve Promise.Callback, reject Promise.Callback) {
reject(3)
}).Then(func(value interface{}) (interface{}, error) {
return nil, nil
}).Catch(func(value interface{}) (interface{}, error) {
tmp, _ := value.(int)
fmt.Println("got value:",tmp)
w.Done()
return nil, nil
})
w.Wait()
- Chaing your Then and Catch
var w sync.WaitGroup
w.Add(3)
Promise.Create(func(resolve Promise.Callback, reject Promise.Callback) {
fmt.Println("Promise called")
resolve(2)
}).Then(func(value interface{}) (interface{}, error) {
fmt.Println("Then called")
tmp, _ := value.(int)
fmt.Println("found value", tmp)
w.Done()
return tmp, nil
}).Then(func(value interface{}) (interface{}, error) {
fmt.Println("Then called")
tmp, _ := value.(int)
fmt.Println("found value", tmp)
w.Done()
if tmp == 2 {
return nil, fmt.Errorf("it worked")
}
return nil, nil
}).Catch(func(value interface{}) (interface{}, error) {
fmt.Println(value)
w.Done()
return nil, nil
})
w.Wait()
- Chain , but want to wait at the end ? 🕔️
var data int = 0
Promise.Create(func(resolve Promise.Callback, reject Promise.Callback) {
resolve(2)
}).Then(func(value interface{}) (interface{}, error) {
tmp, _ := value.(int)
return tmp + 1, nil
}).Then(func(value interface{}) (interface{}, error) {
tmp, _ := value.(int)
return tmp + 1, nil
}).Finally(func(value interface{}) {
tmp, _ := value.(int)
data = tmp
})
//// or u can simply return the value
value := Promise.Create(func(resolve Promise.Callback, reject Promise.Callback) {
resolve(2)
}).Then(func(value interface{}) (interface{}, error) {
tmp, _ := value.(int)
return tmp + 1, nil
}).Then(func(value interface{}) (interface{}, error) {
tmp, _ := value.(int)
return tmp + 1, nil
}).Finally(nil)
fmt.Println(data)
- I want a promise right now ? 👊
var data int = 0
Promise.Resolve(4).Finally(func(value interface{}) {
tmp, _ := value.(int)
data = tmp
})
fmt.Println(data)
Promise.Reject(5).Finally(func(value interface{}) {
tmp, _ := value.(int)
data = tmp
})
fmt.Println(data)
- Ya Go is async already with GoRoutines but how do I get a Synchronous wait ? 👐
dt, _ := Promise.Resolve(3).Finally(func(value interface{}) {}).(int)
fmt.Println(dt)
//
// as a pattern you can orchestrate many jobs together by
// combining promises only to collect one final result at one s
// ingle place using Finally
//
// You can pass nil to Finally also , instead of a function
- I want to manage my Promise , Cancel it , Inspect it ? Yes you can 😏
// IsPending
p := Promise.Create(func(resolve Promise.Callback, reject Promise.Callback) {
time.Sleep(time.Duration(3) * time.Second)
resolve(3)
}).IsPending()
fmt.Println(p)
//isFulFilled
f := Promise.Resolve(3).isFulfilled()
fmt.Println(f)
//Timeout
p := Promise.Create(func(resolve Promise.Callback, reject Promise.Callback) {
time.Sleep(time.Duration(1) * time.Second) // wait for a second
resolve(false)
// but timeout in half a second
}).Timeout(500).Catch(func(value interface{}) (interface{}, error) {
return true, nil
}).Finally(nil)
- Want bunch of Promises executed at once 👀
Note: These are position aware nothing is jumbled
promises := make([]*Promise.Promise, 20)
for i := 0; i < 10; i++ {
promises[i] = Promise.Create(
func(resolve Promise.Callback, reject Promise.Callback) {
resolve(2)
})
}
for i := 10; i < 20; i++ {
promises[i] = Promise.Create(
func(resolve Promise.Callback, reject Promise.Callback) {
reject(3)
})
}
Promise.All(promises).Then(
func(value interface{}) (interface{}, error) {
w.Done()
return nil, nil
}).Catch(func(value interface{}) (interface{}, error) {
fmt.Println(value)
w.Done()
return nil, nil
}).Finally(nil)
- What about Go Routines that race to finish and you just want the winner ? 😎
promises := make([]*Promise.Promise, 3)
for i := 0; i < len(promises); i++ {
index := i
promises[i] = Promise.Create(
func(resolve Promise.Callback, reject Promise.Callback) {
time.Sleep(time.Duration(index)*time.Second + time.Duration(10))
resolve(index + 1)
})
}
val, _ := Promise.Race(promises).Finally(nil).(int)
// Catch statements in case of collection operation
// returns a list of error for all promises that provided error
fmt.Println("Winner is :",val)
- But I want bunch of Promises returning me the fastest they can ? 😥 Worry not you have async generators
promises := make([]*Promise.Promise, 3)
for i := 0; i < len(promises); i++ {
index := i
promises[i] = Promise.Create(
func(resolve Promise.Callback, reject Promise.Callback) {
time.Sleep(time.Duration(index)*time.Second + time.Duration(10))
resolve(index + 1)
})
}
for value := range Promise.AsyncGenerator(promises) {
fmt.Println(value)
}
- My Map is all parallel , can I have a reduce, that pipelines with request delays ? 👩⚖️
Yes the Reduce provided here internally uses AsyncGenerator to grab data and quicly launch promise task to to luanch reducer for each data. Effectively pipelining with the promises passsed to it.
Note: It uses AsyncGenerator internally hence reduce will also get errors for processing , can be avoided by error handled promises
promises := make([]*Promise.Promise, 3)
for i := 0; i < len(promises); i++ {
index := i
promises[i] = Promise.Create(
func(resolve Promise.Callback, reject Promise.Callback) {
time.Sleep(time.Duration(index)*time.Second + time.Duration(10))
resolve(index + 1)
})
}
value := Promise.Reduce(promises,
func(index int, acci interface{}, valuei interface{}) interface{} {
acc, _ := acci.(int)
value, _ := valuei.(int)
acc = value + acc
return acc
}, 0).Finally(nil)
$ go test -v ./test