2014-06-30 20:03:27 -05:00
|
|
|
package resource
|
|
|
|
|
|
|
|
import (
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2014-10-07 23:44:51 -05:00
|
|
|
// RetryFunc is the function retried until it succeeds.
|
|
|
|
type RetryFunc func() error
|
|
|
|
|
|
|
|
// Retry is a basic wrapper around StateChangeConf that will just retry
|
|
|
|
// a function until it no longer returns an error.
|
|
|
|
func Retry(timeout time.Duration, f RetryFunc) error {
|
|
|
|
c := &StateChangeConf{
|
|
|
|
Pending: []string{"error"},
|
|
|
|
Target: "success",
|
|
|
|
Timeout: timeout,
|
|
|
|
MinTimeout: 500 * time.Millisecond,
|
|
|
|
Refresh: func() (interface{}, string, error) {
|
2016-01-16 12:09:01 -06:00
|
|
|
err := f()
|
2014-10-17 20:28:03 -05:00
|
|
|
if err == nil {
|
|
|
|
return 42, "success", nil
|
2014-07-07 17:21:54 -05:00
|
|
|
}
|
|
|
|
|
2014-10-17 20:28:03 -05:00
|
|
|
if rerr, ok := err.(RetryError); ok {
|
|
|
|
err = rerr.Err
|
|
|
|
return nil, "quit", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return 42, "error", nil
|
2014-10-07 23:44:51 -05:00
|
|
|
},
|
2014-06-30 20:03:27 -05:00
|
|
|
}
|
2014-10-07 23:44:51 -05:00
|
|
|
|
2016-01-16 12:09:01 -06:00
|
|
|
_, err := c.WaitForState()
|
2014-10-07 23:44:51 -05:00
|
|
|
return err
|
2014-06-30 20:03:27 -05:00
|
|
|
}
|
2014-10-17 20:28:03 -05:00
|
|
|
|
|
|
|
// RetryError, if returned, will quit the retry immediately with the
|
|
|
|
// Err.
|
|
|
|
type RetryError struct {
|
|
|
|
Err error
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e RetryError) Error() string {
|
|
|
|
return e.Err.Error()
|
|
|
|
}
|