ledisdb/server/http/cmd_zset.go

528 lines
11 KiB
Go

package http
import (
"errors"
"fmt"
"github.com/siddontang/ledisdb/ledis"
"math"
"strconv"
"strings"
)
var errScoreOverflow = errors.New("zset score overflow")
func zaddCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) < 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zadd")
}
if len(args[1:])%2 != 0 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zadd")
}
key := []byte(args[0])
args = args[1:]
params := make([]ledis.ScorePair, len(args)/2)
for i := 0; i < len(params); i++ {
score, err := strconv.ParseInt(args[2*i], 10, 64)
if err != nil {
return nil, ErrValue
}
params[i].Score = score
params[i].Member = []byte(args[2*i+1])
}
if n, err := db.ZAdd(key, params...); err != nil {
return nil, err
} else {
return n, nil
}
}
func zcardCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 1 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zcard")
}
key := []byte(args[0])
if n, err := db.ZCard(key); err != nil {
return nil, err
} else {
return n, nil
}
}
func zscoreCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 2 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zscore")
}
key := []byte(args[0])
member := []byte(args[1])
if s, err := db.ZScore(key, member); err != nil {
if err == ledis.ErrScoreMiss {
return nil, nil
} else {
return nil, err
}
} else {
return strconv.FormatInt(s, 10), nil
}
}
func zremCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) < 2 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zrem")
}
key := []byte(args[0])
members := make([][]byte, len(args[1:]))
for i, arg := range args[1:] {
members[i] = []byte(arg)
}
if n, err := db.ZRem(key, members...); err != nil {
return nil, err
} else {
return n, nil
}
}
func zincrbyCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zincrby")
}
key := []byte(args[0])
delta, err := strconv.ParseInt(args[1], 10, 64)
if err != nil {
return nil, ErrValue
}
member := []byte(args[2])
if v, err := db.ZIncrBy(key, delta, member); err != nil {
return nil, err
} else {
return strconv.FormatInt(v, 10), nil
}
}
func zparseScoreRange(minBuf string, maxBuf string) (min int64, max int64, err error) {
if strings.ToLower(minBuf) == "-inf" {
min = math.MinInt64
} else {
var lopen bool = false
if len(minBuf) == 0 {
err = ErrValue
return
}
if minBuf[0] == '(' {
lopen = true
minBuf = minBuf[1:]
}
min, err = strconv.ParseInt(minBuf, 10, 64)
if err != nil {
err = ErrValue
return
}
if min <= ledis.MinScore || min >= ledis.MaxScore {
err = errScoreOverflow
return
}
if lopen {
min++
}
}
if strings.ToLower(maxBuf) == "+inf" {
max = math.MaxInt64
} else {
var ropen = false
if len(maxBuf) == 0 {
err = ErrValue
return
}
if maxBuf[0] == '(' {
ropen = true
maxBuf = maxBuf[1:]
}
max, err = strconv.ParseInt(maxBuf, 10, 64)
if err != nil {
err = ErrValue
return
}
if max <= ledis.MinScore || max >= ledis.MaxScore {
err = errScoreOverflow
return
}
if ropen {
max--
}
}
return
}
func zcountCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zcount")
}
min, max, err := zparseScoreRange(args[1], args[2])
if err != nil {
return nil, err
}
if min > max {
return 0, nil
}
key := []byte(args[0])
if n, err := db.ZCount(key, min, max); err != nil {
return nil, err
} else {
return n, nil
}
}
func zrankCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 2 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zrank")
}
key := []byte(args[0])
member := []byte(args[1])
if n, err := db.ZRank(key, member); err != nil {
return nil, err
} else if n == -1 {
return nil, nil
} else {
return n, nil
}
}
func zrevrankCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 2 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zrevrank")
}
key := []byte(args[0])
member := []byte(args[1])
if n, err := db.ZRevRank(key, member); err != nil {
return nil, err
} else if n == -1 {
return nil, nil
} else {
return n, nil
}
}
func zremrangebyrankCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zremrangebyrank")
}
key := []byte(args[0])
start, err := strconv.Atoi(args[1])
if err != nil {
return nil, ErrValue
}
stop, err := strconv.Atoi(args[2])
if err != nil {
return nil, ErrValue
}
if n, err := db.ZRemRangeByRank(key, start, stop); err != nil {
return nil, err
} else {
return n, nil
}
}
func zremrangebyscoreCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zremrangebyscore")
}
key := []byte(args[0])
min, max, err := zparseScoreRange(args[1], args[2])
if err != nil {
return nil, err
}
if n, err := db.ZRemRangeByScore(key, min, max); err != nil {
return nil, err
} else {
return n, nil
}
}
func zrangeGeneric(db *ledis.DB, reverse bool, args ...string) (interface{}, error) {
key := []byte(args[0])
start, err := strconv.Atoi(args[1])
if err != nil {
return nil, ErrValue
}
stop, err := strconv.Atoi(args[2])
if err != nil {
return nil, ErrValue
}
args = args[3:]
var withScores bool = false
if len(args) > 0 {
if len(args) != 1 {
return nil, ErrSyntax
}
if strings.ToLower(args[0]) == "withscores" {
withScores = true
} else {
return nil, ErrSyntax
}
}
if datas, err := db.ZRangeGeneric(key, start, stop, reverse); err != nil {
return nil, err
} else {
return makeScorePairArray(datas, withScores), nil
}
}
func makeScorePairArray(datas []ledis.ScorePair, withScores bool) []string {
var arr []string
if withScores {
arr = make([]string, 2*len(datas))
for i, data := range datas {
arr[2*i] = ledis.String(data.Member)
arr[2*i+1] = strconv.FormatInt(data.Score, 10)
}
} else {
arr = make([]string, len(datas))
for i, data := range datas {
arr[i] = ledis.String(data.Member)
}
}
return arr
}
func zrangeCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) < 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zrange")
}
return zrangeGeneric(db, false, args...)
}
func zrevrangeCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) < 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zrevrange")
}
return zrangeGeneric(db, true, args...)
}
func zrangebyscoreGeneric(db *ledis.DB, reverse bool, args ...string) (interface{}, error) {
key := []byte(args[0])
var minScore, maxScore string
if !reverse {
minScore, maxScore = args[1], args[2]
} else {
minScore, maxScore = args[2], args[1]
}
min, max, err := zparseScoreRange(minScore, maxScore)
if err != nil {
return nil, err
}
args = args[3:]
var withScores bool = false
if len(args) > 0 && strings.ToLower(args[0]) == "withscores" {
withScores = true
args = args[1:]
}
var offset int = 0
var count int = -1
if len(args) > 0 {
if len(args) != 3 {
return nil, ErrSyntax
}
if strings.ToLower(args[0]) != "limit" {
return nil, ErrSyntax
}
if offset, err = strconv.Atoi(args[1]); err != nil {
return nil, ErrValue
}
if count, err = strconv.Atoi(args[2]); err != nil {
return nil, ErrValue
}
}
if offset < 0 {
return []interface{}{}, nil
}
if datas, err := db.ZRangeByScoreGeneric(key, min, max, offset, count, reverse); err != nil {
return nil, err
} else {
return makeScorePairArray(datas, withScores), nil
}
}
func zrangebyscoreCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) < 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zrangebyscore")
}
return zrangebyscoreGeneric(db, false, args...)
}
func zrevrangebyscoreCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) < 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zrevrangebyscore")
}
return zrangebyscoreGeneric(db, true, args...)
}
func zclearCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 1 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zclear")
}
key := []byte(args[0])
if n, err := db.ZClear(key); err != nil {
return nil, err
} else {
return n, nil
}
}
func zmclearCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) < 1 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zmclear")
}
keys := make([][]byte, len(args))
for i, arg := range args {
keys[i] = []byte(arg)
}
if n, err := db.ZMclear(keys...); err != nil {
return nil, err
} else {
return n, nil
}
}
func zexpireCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 2 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zexpire")
}
duration, err := strconv.ParseInt(args[1], 10, 64)
if err != nil {
return nil, ErrValue
}
key := []byte(args[0])
if v, err := db.ZExpire(key, duration); err != nil {
return nil, err
} else {
return v, nil
}
}
func zexpireAtCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 2 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zexpireat")
}
when, err := strconv.ParseInt(args[1], 10, 64)
if err != nil {
return nil, ErrValue
}
key := []byte(args[0])
if v, err := db.ZExpireAt(key, when); err != nil {
return nil, err
} else {
return v, nil
}
}
func zttlCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 1 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zttl")
}
key := []byte(args[0])
if v, err := db.ZTTL(key); err != nil {
return nil, err
} else {
return v, nil
}
}
func zpersistCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 1 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "zpersist")
}
key := []byte(args[0])
if n, err := db.ZPersist(key); err != nil {
return nil, err
} else {
return n, nil
}
}
func init() {
register("zadd", zaddCommand)
register("zcard", zcardCommand)
register("zcount", zcountCommand)
register("zincrby", zincrbyCommand)
register("zrange", zrangeCommand)
register("zrangebyscore", zrangebyscoreCommand)
register("zrank", zrankCommand)
register("zrem", zremCommand)
register("zremrangebyrank", zremrangebyrankCommand)
register("zremrangebyscore", zremrangebyscoreCommand)
register("zrevrange", zrevrangeCommand)
register("zrevrank", zrevrankCommand)
register("zrevrangebyscore", zrevrangebyscoreCommand)
register("zscore", zscoreCommand)
//ledisdb special command
register("zclear", zclearCommand)
register("zmclear", zmclearCommand)
register("zexpire", zexpireCommand)
register("zexpireat", zexpireAtCommand)
register("zttl", zttlCommand)
register("zpersist", zpersistCommand)
}