Files
redka/redsrv/internal/command/list/lrange_test.go
2025-07-13 22:37:43 +05:00

140 lines
3.7 KiB
Go

package list
import (
"testing"
"github.com/nalgeon/be"
"github.com/nalgeon/redka/internal/core"
"github.com/nalgeon/redka/redsrv/internal/redis"
)
func TestLRangeParse(t *testing.T) {
tests := []struct {
cmd string
want LRange
err error
}{
{
cmd: "lrange",
want: LRange{},
err: redis.ErrInvalidArgNum,
},
{
cmd: "lrange key",
want: LRange{},
err: redis.ErrInvalidArgNum,
},
{
cmd: "lrange key 0",
want: LRange{},
err: redis.ErrInvalidArgNum,
},
{
cmd: "lrange key 0 -1",
want: LRange{key: "key", start: 0, stop: -1},
err: nil,
},
}
for _, test := range tests {
t.Run(test.cmd, func(t *testing.T) {
cmd, err := redis.Parse(ParseLRange, test.cmd)
be.Equal(t, err, test.err)
if err == nil {
be.Equal(t, cmd.key, test.want.key)
be.Equal(t, cmd.start, test.want.start)
be.Equal(t, cmd.stop, test.want.stop)
} else {
be.Equal(t, cmd, test.want)
}
})
}
}
func TestLRangeExec(t *testing.T) {
t.Run("empty list", func(t *testing.T) {
red := getRedka(t)
cmd := redis.MustParse(ParseLRange, "lrange key 0 0")
conn := redis.NewFakeConn()
res, err := cmd.Run(conn, red)
be.Err(t, err, nil)
be.Equal(t, res.([]core.Value), []core.Value(nil))
be.Equal(t, conn.Out(), "0")
})
t.Run("single elem", func(t *testing.T) {
red := getRedka(t)
_, _ = red.List().PushBack("key", "elem")
cmd := redis.MustParse(ParseLRange, "lrange key 0 0")
conn := redis.NewFakeConn()
res, err := cmd.Run(conn, red)
be.Err(t, err, nil)
be.Equal(t, res.([]core.Value), []core.Value{core.Value("elem")})
be.Equal(t, conn.Out(), "1,elem")
})
t.Run("multiple elems", func(t *testing.T) {
red := getRedka(t)
_, _ = red.List().PushBack("key", "one")
_, _ = red.List().PushBack("key", "two")
_, _ = red.List().PushBack("key", "thr")
cmd := redis.MustParse(ParseLRange, "lrange key 0 1")
conn := redis.NewFakeConn()
res, err := cmd.Run(conn, red)
be.Err(t, err, nil)
be.Equal(t, res.([]core.Value), []core.Value{core.Value("one"), core.Value("two")})
be.Equal(t, conn.Out(), "2,one,two")
})
t.Run("negative indexes", func(t *testing.T) {
red := getRedka(t)
_, _ = red.List().PushBack("key", "one")
_, _ = red.List().PushBack("key", "two")
_, _ = red.List().PushBack("key", "thr")
cmd := redis.MustParse(ParseLRange, "lrange key -2 -1")
conn := redis.NewFakeConn()
res, err := cmd.Run(conn, red)
be.Err(t, err, nil)
be.Equal(t, res.([]core.Value), []core.Value{core.Value("two"), core.Value("thr")})
be.Equal(t, conn.Out(), "2,two,thr")
})
t.Run("out of bounds", func(t *testing.T) {
red := getRedka(t)
_, _ = red.List().PushBack("key", "one")
_, _ = red.List().PushBack("key", "two")
_, _ = red.List().PushBack("key", "thr")
cmd := redis.MustParse(ParseLRange, "lrange key 3 5")
conn := redis.NewFakeConn()
res, err := cmd.Run(conn, red)
be.Err(t, err, nil)
be.Equal(t, res.([]core.Value), []core.Value(nil))
be.Equal(t, conn.Out(), "0")
})
t.Run("start < stop", func(t *testing.T) {
red := getRedka(t)
_, _ = red.List().PushBack("key", "one")
_, _ = red.List().PushBack("key", "two")
_, _ = red.List().PushBack("key", "thr")
cmd := redis.MustParse(ParseLRange, "lrange key 2 1")
conn := redis.NewFakeConn()
res, err := cmd.Run(conn, red)
be.Err(t, err, nil)
be.Equal(t, res.([]core.Value), []core.Value(nil))
be.Equal(t, conn.Out(), "0")
})
t.Run("key type mismatch", func(t *testing.T) {
red := getRedka(t)
_ = red.Str().Set("key", "str")
cmd := redis.MustParse(ParseLRange, "lrange key 0 0")
conn := redis.NewFakeConn()
res, err := cmd.Run(conn, red)
be.Err(t, err, nil)
be.Equal(t, res.([]core.Value), []core.Value(nil))
be.Equal(t, conn.Out(), "0")
})
}