2014-05-16 11:03:23 +04:00
|
|
|
package server
|
2014-05-03 10:55:12 +04:00
|
|
|
|
2014-05-08 06:54:33 +04:00
|
|
|
import (
|
|
|
|
"errors"
|
2014-05-16 11:03:23 +04:00
|
|
|
"github.com/siddontang/ledisdb/ledis"
|
2014-05-08 06:54:33 +04:00
|
|
|
"math"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
//for simple implementation, we only support int64 score
|
|
|
|
|
|
|
|
var errScoreOverflow = errors.New("zset score overflow")
|
|
|
|
|
|
|
|
func zaddCommand(c *client) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) < 3 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
|
|
|
key := args[0]
|
|
|
|
if len(args[1:])%2 != 0 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
|
|
|
args = args[1:]
|
2014-05-16 04:56:32 +04:00
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
params := make([]ledis.ScorePair, len(args)/2)
|
2014-05-16 04:56:32 +04:00
|
|
|
for i := 0; i < len(params); i++ {
|
2014-05-16 11:03:23 +04:00
|
|
|
score, err := ledis.StrInt64(args[2*i], nil)
|
2014-05-08 06:54:33 +04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2014-05-16 04:56:32 +04:00
|
|
|
params[i].Score = score
|
|
|
|
params[i].Member = args[2*i+1]
|
2014-05-08 06:54:33 +04:00
|
|
|
}
|
|
|
|
|
2014-05-16 04:56:32 +04:00
|
|
|
if n, err := c.db.ZAdd(key, params...); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
c.writeInteger(n)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func zcardCommand(c *client) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) != 1 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
if n, err := c.db.ZCard(args[0]); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
c.writeInteger(n)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func zscoreCommand(c *client) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) != 2 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
2014-06-05 10:21:50 +04:00
|
|
|
if s, err := c.db.ZScore(args[0], args[1]); err != nil {
|
|
|
|
if err == ledis.ErrScoreMiss {
|
|
|
|
c.writeBulk(nil)
|
|
|
|
} else {
|
|
|
|
return err
|
|
|
|
}
|
2014-05-08 06:54:33 +04:00
|
|
|
} else {
|
2014-06-05 10:21:50 +04:00
|
|
|
c.writeBulk(ledis.StrPutInt64(s))
|
2014-05-08 06:54:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func zremCommand(c *client) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) < 2 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
2014-05-16 04:56:32 +04:00
|
|
|
if n, err := c.db.ZRem(args[0], args[1:]...); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
c.writeInteger(n)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func zincrbyCommand(c *client) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) != 3 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
|
|
|
key := args[0]
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
delta, err := ledis.StrInt64(args[1], nil)
|
2014-05-08 06:54:33 +04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
if v, err := c.db.ZIncrBy(key, delta, args[2]); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return err
|
|
|
|
} else {
|
2014-06-05 10:21:50 +04:00
|
|
|
c.writeBulk(ledis.StrPutInt64(v))
|
2014-05-08 06:54:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func zparseScoreRange(minBuf []byte, maxBuf []byte) (min int64, max int64, err error) {
|
2014-05-16 11:03:23 +04:00
|
|
|
if strings.ToLower(ledis.String(minBuf)) == "-inf" {
|
2014-05-08 06:54:33 +04:00
|
|
|
min = math.MinInt64
|
|
|
|
} else {
|
|
|
|
var lopen bool = false
|
|
|
|
if minBuf[0] == '(' {
|
|
|
|
lopen = true
|
|
|
|
minBuf = minBuf[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(minBuf) == 0 {
|
|
|
|
err = ErrCmdParams
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
min, err = ledis.StrInt64(minBuf, nil)
|
2014-05-08 06:54:33 +04:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
if min <= ledis.MinScore || min >= ledis.MaxScore {
|
2014-05-08 06:54:33 +04:00
|
|
|
err = errScoreOverflow
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if lopen {
|
|
|
|
min++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
if strings.ToLower(ledis.String(maxBuf)) == "+inf" {
|
2014-05-08 06:54:33 +04:00
|
|
|
max = math.MaxInt64
|
|
|
|
} else {
|
|
|
|
var ropen = false
|
|
|
|
if maxBuf[0] == '(' {
|
|
|
|
ropen = true
|
|
|
|
maxBuf = maxBuf[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(maxBuf) == 0 {
|
|
|
|
err = ErrCmdParams
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
max, err = ledis.StrInt64(maxBuf, nil)
|
2014-05-08 06:54:33 +04:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
if max <= ledis.MinScore || max >= ledis.MaxScore {
|
2014-05-08 06:54:33 +04:00
|
|
|
err = errScoreOverflow
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if ropen {
|
|
|
|
max--
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func zcountCommand(c *client) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) != 3 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
|
|
|
min, max, err := zparseScoreRange(args[1], args[2])
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if min > max {
|
|
|
|
c.writeInteger(0)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
if n, err := c.db.ZCount(args[0], min, max); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
c.writeInteger(n)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func zrankCommand(c *client) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) != 2 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
if n, err := c.db.ZRank(args[0], args[1]); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return err
|
|
|
|
} else if n == -1 {
|
|
|
|
c.writeBulk(nil)
|
|
|
|
} else {
|
|
|
|
c.writeInteger(n)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func zrevrankCommand(c *client) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) != 2 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
if n, err := c.db.ZRevRank(args[0], args[1]); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return err
|
|
|
|
} else if n == -1 {
|
|
|
|
c.writeBulk(nil)
|
|
|
|
} else {
|
|
|
|
c.writeInteger(n)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func zremrangebyrankCommand(c *client) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) != 3 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
|
|
|
key := args[0]
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
start, stop, err := zparseRange(c, args[1], args[2])
|
2014-05-08 06:54:33 +04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
if n, err := c.db.ZRemRangeByRank(key, start, stop); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
c.writeInteger(n)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func zremrangebyscoreCommand(c *client) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) != 3 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
|
|
|
key := args[0]
|
|
|
|
min, max, err := zparseScoreRange(args[1], args[2])
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
if n, err := c.db.ZRemRangeByScore(key, min, max); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
c.writeInteger(n)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func zparseRange(c *client, a1 []byte, a2 []byte) (start int, stop int, err error) {
|
2014-05-16 11:03:23 +04:00
|
|
|
if start, err = strconv.Atoi(ledis.String(a1)); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
if stop, err = strconv.Atoi(ledis.String(a2)); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func zrangeGeneric(c *client, reverse bool) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) < 3 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
|
|
|
key := args[0]
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
start, stop, err := zparseRange(c, args[1], args[2])
|
2014-05-08 06:54:33 +04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
args = args[3:]
|
|
|
|
var withScores bool = false
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
if len(args) > 0 && strings.ToLower(ledis.String(args[0])) == "withscores" {
|
2014-05-08 06:54:33 +04:00
|
|
|
withScores = true
|
|
|
|
}
|
|
|
|
|
2014-06-05 10:21:50 +04:00
|
|
|
if datas, err := c.db.ZRangeGeneric(key, start, stop, withScores, reverse); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return err
|
|
|
|
} else {
|
2014-06-05 10:21:50 +04:00
|
|
|
if withScores {
|
|
|
|
for i := len(datas) - 1; i > 0; i -= 2 {
|
|
|
|
v, _ := datas[i].(int64)
|
|
|
|
datas[i] = ledis.StrPutInt64(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c.writeArray(datas)
|
2014-05-08 06:54:33 +04:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func zrangeCommand(c *client) error {
|
|
|
|
return zrangeGeneric(c, false)
|
|
|
|
}
|
|
|
|
|
|
|
|
func zrevrangeCommand(c *client) error {
|
|
|
|
return zrangeGeneric(c, true)
|
|
|
|
}
|
|
|
|
|
|
|
|
func zrangebyscoreGeneric(c *client, reverse bool) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) < 3 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
|
|
|
key := args[0]
|
|
|
|
min, max, err := zparseScoreRange(args[1], args[2])
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
args = args[3:]
|
|
|
|
|
|
|
|
var withScores bool = false
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
if len(args) > 0 && strings.ToLower(ledis.String(args[0])) == "withscores" {
|
2014-05-08 06:54:33 +04:00
|
|
|
withScores = true
|
|
|
|
args = args[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
var offset int = 0
|
2014-05-15 10:19:48 +04:00
|
|
|
var count int = -1
|
2014-05-08 06:54:33 +04:00
|
|
|
|
|
|
|
if len(args) > 0 {
|
|
|
|
if len(args) != 3 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
if strings.ToLower(ledis.String(args[0])) != "limit" {
|
2014-05-08 06:54:33 +04:00
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
if offset, err = strconv.Atoi(ledis.String(args[1])); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
2014-05-16 11:03:23 +04:00
|
|
|
if count, err = strconv.Atoi(ledis.String(args[2])); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if offset < 0 {
|
|
|
|
//for redis, if offset < 0, a empty will return
|
|
|
|
//so here we directly return a empty array
|
|
|
|
c.writeArray([]interface{}{})
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-06-05 10:21:50 +04:00
|
|
|
if datas, err := c.db.ZRangeByScoreGeneric(key, min, max, withScores, offset, count, reverse); err != nil {
|
2014-05-08 06:54:33 +04:00
|
|
|
return err
|
|
|
|
} else {
|
2014-06-05 10:21:50 +04:00
|
|
|
if withScores {
|
|
|
|
for i := len(datas) - 1; i > 0; i -= 2 {
|
|
|
|
v, _ := datas[i].(int64)
|
|
|
|
datas[i] = ledis.StrPutInt64(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c.writeArray(datas)
|
2014-05-08 06:54:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func zrangebyscoreCommand(c *client) error {
|
|
|
|
return zrangebyscoreGeneric(c, false)
|
|
|
|
}
|
|
|
|
|
|
|
|
func zrevrangebyscoreCommand(c *client) error {
|
|
|
|
return zrangebyscoreGeneric(c, true)
|
|
|
|
}
|
|
|
|
|
2014-05-12 11:08:59 +04:00
|
|
|
func zclearCommand(c *client) error {
|
|
|
|
args := c.args
|
|
|
|
if len(args) != 1 {
|
|
|
|
return ErrCmdParams
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
if n, err := c.db.ZClear(args[0]); err != nil {
|
2014-05-12 11:08:59 +04:00
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
c.writeInteger(n)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-05-03 10:55:12 +04:00
|
|
|
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)
|
2014-05-12 11:08:59 +04:00
|
|
|
|
|
|
|
//ledisdb special command
|
|
|
|
register("zclear", zclearCommand)
|
|
|
|
|
2014-05-03 10:55:12 +04:00
|
|
|
}
|