Skip to content

Latest commit

 

History

History
302 lines (220 loc) · 7.26 KB

README.md

File metadata and controls

302 lines (220 loc) · 7.26 KB

[Go-Promise] Library

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.

Want to try ? 🧐

$ go get github.com/rajatkb/go-promise

How to use ? 🤨

  • 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)

Development

Test

$ go test -v ./test