package queue

import (
	"context"
	"os"
	"os/exec"
	"testing"
	"time"

	"code.gitea.io/gitea/modules/nosql"
	"code.gitea.io/gitea/modules/setting"

	"github.com/stretchr/testify/suite"
)

const defaultTestRedisServer = "127.0.0.1:6379"

type baseRedisWithServerTestSuite struct {
	suite.Suite
}

func TestBaseRedisWithServer(t *testing.T) {
	suite.Run(t, &baseRedisWithServerTestSuite{})
}

func (suite *baseRedisWithServerTestSuite) TestNormal() {
	redisAddress := "redis://" + suite.testRedisHost() + "/0"
	queueSettings := setting.QueueSettings{
		Length:  10,
		ConnStr: redisAddress,
	}

	redisServer, accessible := suite.startRedisServer(redisAddress)

	// If it's accessible, but redisServer command is nil, that means we are using
	// an already running redis server.
	if redisServer == nil && !accessible {
		suite.T().Skip("redis-server not found in Forgejo test yet")

		return
	}

	defer func() {
		if redisServer != nil {
			_ = redisServer.Process.Signal(os.Interrupt)
			_ = redisServer.Wait()
		}
	}()

	testQueueBasic(suite.T(), newBaseRedisSimple, toBaseConfig("baseRedis", queueSettings), false)
	testQueueBasic(suite.T(), newBaseRedisUnique, toBaseConfig("baseRedisUnique", queueSettings), true)
}

func (suite *baseRedisWithServerTestSuite) TestWithPrefix() {
	redisAddress := "redis://" + suite.testRedisHost() + "/0?prefix=forgejo:queue:"
	queueSettings := setting.QueueSettings{
		Length:  10,
		ConnStr: redisAddress,
	}

	redisServer, accessible := suite.startRedisServer(redisAddress)

	// If it's accessible, but redisServer command is nil, that means we are using
	// an already running redis server.
	if redisServer == nil && !accessible {
		suite.T().Skip("redis-server not found in Forgejo test yet")

		return
	}

	defer func() {
		if redisServer != nil {
			_ = redisServer.Process.Signal(os.Interrupt)
			_ = redisServer.Wait()
		}
	}()

	testQueueBasic(suite.T(), newBaseRedisSimple, toBaseConfig("baseRedis", queueSettings), false)
	testQueueBasic(suite.T(), newBaseRedisUnique, toBaseConfig("baseRedisUnique", queueSettings), true)
}

func (suite *baseRedisWithServerTestSuite) startRedisServer(address string) (*exec.Cmd, bool) {
	var redisServer *exec.Cmd

	if !suite.waitRedisReady(address, 0) {
		redisServerProg, err := exec.LookPath("redis-server")
		if err != nil {
			return nil, false
		}
		redisServer = &exec.Cmd{
			Path:   redisServerProg,
			Args:   []string{redisServerProg, "--bind", "127.0.0.1", "--port", "6379"},
			Dir:    suite.T().TempDir(),
			Stdin:  os.Stdin,
			Stdout: os.Stdout,
			Stderr: os.Stderr,
		}

		suite.Require().NoError(redisServer.Start())

		if !suite.True(suite.waitRedisReady(address, 5*time.Second), "start redis-server") {
			// Return with redis server even if it's not available. It was started,
			// even if it's not reachable for any reasons, it's still started, the
			// parent will close it.
			return redisServer, false
		}
	}

	return redisServer, true
}

func (suite *baseRedisWithServerTestSuite) waitRedisReady(conn string, dur time.Duration) (ready bool) {
	ctxTimed, cancel := context.WithTimeout(context.Background(), time.Second*5)
	defer cancel()
	for t := time.Now(); ; time.Sleep(50 * time.Millisecond) {
		ret := nosql.GetManager().GetRedisClient(conn).Ping(ctxTimed)
		if ret.Err() == nil {
			return true
		}
		if time.Since(t) > dur {
			return false
		}
	}
}

func (suite *baseRedisWithServerTestSuite) testRedisHost() string {
	value := os.Getenv("TEST_REDIS_SERVER")
	if value != "" {
		return value
	}

	return defaultTestRedisServer
}