mirror of
https://github.com/nalgeon/redka.git
synced 2025-10-16 21:10:37 +08:00
232 lines
7.2 KiB
Go
232 lines
7.2 KiB
Go
package zset_test
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/nalgeon/redka/internal/command"
|
|
"github.com/nalgeon/redka/internal/command/zset"
|
|
"github.com/nalgeon/redka/internal/redis"
|
|
"github.com/nalgeon/redka/internal/rzset"
|
|
"github.com/nalgeon/redka/internal/testx"
|
|
)
|
|
|
|
func TestZRangeByScoreParse(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
args [][]byte
|
|
want zset.ZRangeByScore
|
|
err error
|
|
}{
|
|
{
|
|
name: "zrangebyscore",
|
|
args: command.BuildArgs("zrangebyscore"),
|
|
want: zset.ZRangeByScore{},
|
|
err: redis.ErrInvalidArgNum,
|
|
},
|
|
{
|
|
name: "zrangebyscore key",
|
|
args: command.BuildArgs("zrangebyscore", "key"),
|
|
want: zset.ZRangeByScore{},
|
|
err: redis.ErrInvalidArgNum,
|
|
},
|
|
{
|
|
name: "zrangebyscore key 11",
|
|
args: command.BuildArgs("zrangebyscore", "key", "11"),
|
|
want: zset.ZRangeByScore{},
|
|
err: redis.ErrInvalidArgNum,
|
|
},
|
|
{
|
|
name: "zrangebyscore key 11 22",
|
|
args: command.BuildArgs("zrangebyscore", "key", "11", "22"),
|
|
want: zset.ZRangeByScore{Key: "key", Min: 11.0, Max: 22.0},
|
|
err: nil,
|
|
},
|
|
{
|
|
name: "zrangebyscore key exclusive",
|
|
args: command.BuildArgs("zrangebyscore", "key", "(1", "(2"),
|
|
want: zset.ZRangeByScore{},
|
|
err: redis.ErrInvalidFloat,
|
|
},
|
|
{
|
|
name: "zrangebyscore key 11 22 limit 10",
|
|
args: command.BuildArgs("zrangebyscore", "key", "11", "22", "limit", "10"),
|
|
want: zset.ZRangeByScore{},
|
|
err: redis.ErrSyntaxError,
|
|
},
|
|
{
|
|
name: "zrangebyscore key 11 22 limit 10 5",
|
|
args: command.BuildArgs("zrangebyscore", "key", "11", "22", "limit", "10", "5"),
|
|
want: zset.ZRangeByScore{Key: "key", Min: 11.0, Max: 22.0, Offset: 10, Count: 5},
|
|
err: nil,
|
|
},
|
|
{
|
|
name: "zrangebyscore key 11 22 withscores",
|
|
args: command.BuildArgs("zrangebyscore", "key", "11", "22", "withscores"),
|
|
want: zset.ZRangeByScore{Key: "key", Min: 11.0, Max: 22.0, WithScores: true},
|
|
err: nil,
|
|
},
|
|
{
|
|
name: "zrangebyscore key 11 22 limit 10 5 withscores",
|
|
args: command.BuildArgs("zrangebyscore", "key", "11", "22",
|
|
"limit", "10", "5", "withscores"),
|
|
want: zset.ZRangeByScore{Key: "key", Min: 11.0, Max: 22.0,
|
|
Offset: 10, Count: 5, WithScores: true},
|
|
err: nil,
|
|
},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
t.Run(test.name, func(t *testing.T) {
|
|
cmd, err := command.Parse(test.args)
|
|
testx.AssertEqual(t, err, test.err)
|
|
if err == nil {
|
|
cm := cmd.(*zset.ZRangeByScore)
|
|
testx.AssertEqual(t, cm.Key, test.want.Key)
|
|
testx.AssertEqual(t, cm.Min, test.want.Min)
|
|
testx.AssertEqual(t, cm.Max, test.want.Max)
|
|
testx.AssertEqual(t, cm.Offset, test.want.Offset)
|
|
testx.AssertEqual(t, cm.Count, test.want.Count)
|
|
testx.AssertEqual(t, cm.WithScores, test.want.WithScores)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestZRangeByScoreExec(t *testing.T) {
|
|
t.Run("range", func(t *testing.T) {
|
|
db, red := getDB(t)
|
|
defer db.Close()
|
|
_, _ = db.ZSet().Add("key", "one", 10)
|
|
_, _ = db.ZSet().Add("key", "two", 20)
|
|
_, _ = db.ZSet().Add("key", "thr", 30)
|
|
_, _ = db.ZSet().Add("key", "2nd", 20)
|
|
|
|
{
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key 0 10")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 1)
|
|
testx.AssertEqual(t, conn.Out(), "1,one")
|
|
}
|
|
{
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key 0 50")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 4)
|
|
testx.AssertEqual(t, conn.Out(), "4,one,2nd,two,thr")
|
|
}
|
|
{
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key 30 50")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 1)
|
|
testx.AssertEqual(t, conn.Out(), "1,thr")
|
|
}
|
|
{
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key 40 50")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 0)
|
|
testx.AssertEqual(t, conn.Out(), "0")
|
|
}
|
|
})
|
|
t.Run("limit", func(t *testing.T) {
|
|
db, red := getDB(t)
|
|
defer db.Close()
|
|
_, _ = db.ZSet().Add("key", "one", 10)
|
|
_, _ = db.ZSet().Add("key", "two", 20)
|
|
_, _ = db.ZSet().Add("key", "thr", 30)
|
|
_, _ = db.ZSet().Add("key", "2nd", 20)
|
|
|
|
{
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key 0 50 limit 0 2")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 2)
|
|
testx.AssertEqual(t, conn.Out(), "2,one,2nd")
|
|
}
|
|
{
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key 0 50 limit 1 2")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 2)
|
|
testx.AssertEqual(t, conn.Out(), "2,2nd,two")
|
|
}
|
|
{
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key 0 50 limit 2 5")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 2)
|
|
testx.AssertEqual(t, conn.Out(), "2,two,thr")
|
|
}
|
|
{
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key 0 50 limit 1 -1")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 3)
|
|
testx.AssertEqual(t, conn.Out(), "3,2nd,two,thr")
|
|
}
|
|
})
|
|
t.Run("with scores", func(t *testing.T) {
|
|
db, red := getDB(t)
|
|
defer db.Close()
|
|
_, _ = db.ZSet().Add("key", "one", 10)
|
|
_, _ = db.ZSet().Add("key", "two", 20)
|
|
_, _ = db.ZSet().Add("key", "thr", 30)
|
|
_, _ = db.ZSet().Add("key", "2nd", 20)
|
|
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key 10 50 withscores")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 4)
|
|
testx.AssertEqual(t, conn.Out(), "8,one,10,2nd,20,two,20,thr,30")
|
|
})
|
|
t.Run("negative indexes", func(t *testing.T) {
|
|
db, red := getDB(t)
|
|
defer db.Close()
|
|
_, _ = db.ZSet().Add("key", "one", -10)
|
|
_, _ = db.ZSet().Add("key", "two", -20)
|
|
_, _ = db.ZSet().Add("key", "thr", -30)
|
|
_, _ = db.ZSet().Add("key", "2nd", -20)
|
|
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key -20 -10")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 3)
|
|
testx.AssertEqual(t, conn.Out(), "3,2nd,two,one")
|
|
})
|
|
t.Run("key not found", func(t *testing.T) {
|
|
db, red := getDB(t)
|
|
defer db.Close()
|
|
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key 0 1")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 0)
|
|
testx.AssertEqual(t, conn.Out(), "0")
|
|
})
|
|
t.Run("key type mismatch", func(t *testing.T) {
|
|
db, red := getDB(t)
|
|
defer db.Close()
|
|
_ = db.Str().Set("key", "value")
|
|
|
|
cmd := command.MustParse[*zset.ZRangeByScore]("zrangebyscore key 0 1")
|
|
conn := redis.NewFakeConn()
|
|
res, err := cmd.Run(conn, red)
|
|
testx.AssertNoErr(t, err)
|
|
testx.AssertEqual(t, len(res.([]rzset.SetItem)), 0)
|
|
testx.AssertEqual(t, conn.Out(), "0")
|
|
})
|
|
}
|