package retry_test

import (
	"testing"
	"time"

	"v2ray.com/core/common/errors"
	. "v2ray.com/core/common/retry"
	. "v2ray.com/ext/assert"
)

var (
	errorTestOnly = errors.New("This is a fake error.")
)

func TestNoRetry(t *testing.T) {
	assert := With(t)

	startTime := time.Now().Unix()
	err := Timed(10, 100000).On(func() error {
		return nil
	})
	endTime := time.Now().Unix()

	assert(err, IsNil)
	assert(endTime-startTime, AtLeast, int64(0))
}

func TestRetryOnce(t *testing.T) {
	assert := With(t)

	startTime := time.Now()
	called := 0
	err := Timed(10, 1000).On(func() error {
		if called == 0 {
			called++
			return errorTestOnly
		}
		return nil
	})
	duration := time.Since(startTime)

	assert(err, IsNil)
	assert(int64(duration/time.Millisecond), AtLeast, int64(900))
}

func TestRetryMultiple(t *testing.T) {
	assert := With(t)

	startTime := time.Now()
	called := 0
	err := Timed(10, 1000).On(func() error {
		if called < 5 {
			called++
			return errorTestOnly
		}
		return nil
	})
	duration := time.Since(startTime)

	assert(err, IsNil)
	assert(int64(duration/time.Millisecond), AtLeast, int64(4900))
}

func TestRetryExhausted(t *testing.T) {
	assert := With(t)

	startTime := time.Now()
	called := 0
	err := Timed(2, 1000).On(func() error {
		called++
		return errorTestOnly
	})
	duration := time.Since(startTime)

	assert(errors.Cause(err), Equals, ErrRetryFailed)
	assert(int64(duration/time.Millisecond), AtLeast, int64(1900))
}

func TestExponentialBackoff(t *testing.T) {
	assert := With(t)

	startTime := time.Now()
	called := 0
	err := ExponentialBackoff(10, 100).On(func() error {
		called++
		return errorTestOnly
	})
	duration := time.Since(startTime)

	assert(errors.Cause(err), Equals, ErrRetryFailed)
	assert(int64(duration/time.Millisecond), AtLeast, int64(4000))
}