tile38/controller/crud.go

1030 lines
22 KiB
Go
Raw Normal View History

2016-03-05 02:08:16 +03:00
package controller
import (
"bytes"
2016-03-28 18:57:41 +03:00
"sort"
2016-03-05 02:08:16 +03:00
"strconv"
"strings"
"time"
2016-07-10 05:44:28 +03:00
"github.com/tidwall/btree"
2016-03-28 18:57:41 +03:00
"github.com/tidwall/resp"
2016-03-06 17:55:00 +03:00
"github.com/tidwall/tile38/controller/collection"
"github.com/tidwall/tile38/controller/glob"
2016-03-28 18:57:41 +03:00
"github.com/tidwall/tile38/controller/server"
2016-03-05 02:08:16 +03:00
"github.com/tidwall/tile38/geojson"
"github.com/tidwall/tile38/geojson/geohash"
)
2016-03-28 18:57:41 +03:00
type fvt struct {
field string
value float64
}
type byField []fvt
func (a byField) Len() int {
return len(a)
}
func (a byField) Less(i, j int) bool {
return a[i].field < a[j].field
}
func (a byField) Swap(i, j int) {
a[i], a[j] = a[j], a[i]
}
func orderFields(fmap map[string]int, fields []float64) []fvt {
var fv fvt
fvs := make([]fvt, 0, len(fmap))
for field, idx := range fmap {
if idx < len(fields) {
fv.field = field
fv.value = fields[idx]
2016-03-30 19:32:38 +03:00
if fv.value != 0 {
2016-03-28 18:57:41 +03:00
fvs = append(fvs, fv)
}
}
}
sort.Sort(byField(fvs))
return fvs
}
func (c *Controller) cmdBounds(msg *server.Message) (string, error) {
start := time.Now()
vs := msg.Values[1:]
var ok bool
var key string
if vs, key, ok = tokenval(vs); !ok || key == "" {
return "", errInvalidNumberOfArguments
}
if len(vs) != 0 {
return "", errInvalidNumberOfArguments
}
col := c.getCol(key)
if col == nil {
if msg.OutputType == server.RESP {
return "$-1\r\n", nil
}
return "", errKeyNotFound
}
vals := make([]resp.Value, 0, 2)
var buf bytes.Buffer
if msg.OutputType == server.JSON {
buf.WriteString(`{"ok":true`)
}
2016-10-03 21:37:16 +03:00
minX, minY, minZ, maxX, maxY, maxZ := col.Bounds()
bbox := geojson.New2DBBox(minX, minY, maxX, maxY)
if msg.OutputType == server.JSON {
buf.WriteString(`,"bounds":`)
buf.WriteString(bbox.ExternalJSON())
} else {
vals = append(vals, resp.ArrayValue([]resp.Value{
resp.ArrayValue([]resp.Value{
2016-10-03 21:37:16 +03:00
resp.FloatValue(minX),
resp.FloatValue(minY),
resp.FloatValue(minZ),
}),
resp.ArrayValue([]resp.Value{
2016-10-03 21:37:16 +03:00
resp.FloatValue(maxX),
resp.FloatValue(maxY),
resp.FloatValue(maxZ),
}),
}))
}
switch msg.OutputType {
case server.JSON:
buf.WriteString(`,"elapsed":"` + time.Now().Sub(start).String() + "\"}")
return buf.String(), nil
case server.RESP:
var oval resp.Value
oval = vals[0]
data, err := oval.MarshalRESP()
if err != nil {
return "", err
}
return string(data), nil
}
return "", nil
}
2016-08-26 23:42:52 +03:00
func (c *Controller) cmdType(msg *server.Message) (string, error) {
start := time.Now()
vs := msg.Values[1:]
var ok bool
var key string
if vs, key, ok = tokenval(vs); !ok || key == "" {
return "", errInvalidNumberOfArguments
}
col := c.getCol(key)
if col == nil {
if msg.OutputType == server.RESP {
return "+none\r\n", nil
}
return "", errKeyNotFound
}
typ := "hash"
switch msg.OutputType {
case server.JSON:
return `{"ok":true,"type":` + string(typ) + `,"elapsed":"` + time.Now().Sub(start).String() + "\"}", nil
case server.RESP:
return "+" + typ + "\r\n", nil
}
return "", nil
}
2016-03-28 18:57:41 +03:00
func (c *Controller) cmdGet(msg *server.Message) (string, error) {
2016-03-05 02:08:16 +03:00
start := time.Now()
2016-03-28 18:57:41 +03:00
vs := msg.Values[1:]
var ok bool
2016-03-05 02:08:16 +03:00
var key, id, typ, sprecision string
2016-03-28 18:57:41 +03:00
if vs, key, ok = tokenval(vs); !ok || key == "" {
2016-03-05 02:08:16 +03:00
return "", errInvalidNumberOfArguments
}
2016-03-28 18:57:41 +03:00
if vs, id, ok = tokenval(vs); !ok || id == "" {
2016-03-05 02:08:16 +03:00
return "", errInvalidNumberOfArguments
}
2016-04-01 22:46:39 +03:00
withfields := false
if _, peek, ok := tokenval(vs); ok && strings.ToLower(peek) == "withfields" {
withfields = true
vs = vs[1:]
}
2016-03-05 02:08:16 +03:00
col := c.getCol(key)
if col == nil {
2016-03-28 18:57:41 +03:00
if msg.OutputType == server.RESP {
return "$-1\r\n", nil
}
2016-03-05 02:08:16 +03:00
return "", errKeyNotFound
}
o, fields, ok := col.Get(id)
ok = ok && !c.hasExpired(key, id)
2016-03-05 02:08:16 +03:00
if !ok {
2016-03-28 18:57:41 +03:00
if msg.OutputType == server.RESP {
return "$-1\r\n", nil
}
2016-03-05 02:08:16 +03:00
return "", errIDNotFound
}
2016-03-28 18:57:41 +03:00
vals := make([]resp.Value, 0, 2)
2016-03-05 02:08:16 +03:00
var buf bytes.Buffer
2016-03-28 18:57:41 +03:00
if msg.OutputType == server.JSON {
buf.WriteString(`{"ok":true`)
}
2016-07-10 23:23:50 +03:00
vs, typ, ok = tokenval(vs)
typ = strings.ToLower(typ)
if !ok {
typ = "object"
}
switch typ {
default:
return "", errInvalidArgument(typ)
case "object":
2016-03-28 18:57:41 +03:00
if msg.OutputType == server.JSON {
buf.WriteString(`,"object":`)
buf.WriteString(o.JSON())
} else {
2016-07-10 05:44:28 +03:00
vals = append(vals, resp.StringValue(o.String()))
2016-03-28 18:57:41 +03:00
}
2016-07-10 23:23:50 +03:00
case "point":
point := o.CalculatedPoint()
if msg.OutputType == server.JSON {
buf.WriteString(`,"point":`)
buf.WriteString(point.ExternalJSON())
} else {
if point.Z != 0 {
vals = append(vals, resp.ArrayValue([]resp.Value{
resp.StringValue(strconv.FormatFloat(point.Y, 'f', -1, 64)),
resp.StringValue(strconv.FormatFloat(point.X, 'f', -1, 64)),
resp.StringValue(strconv.FormatFloat(point.Z, 'f', -1, 64)),
}))
2016-03-28 18:57:41 +03:00
} else {
vals = append(vals, resp.ArrayValue([]resp.Value{
2016-07-10 23:23:50 +03:00
resp.StringValue(strconv.FormatFloat(point.Y, 'f', -1, 64)),
resp.StringValue(strconv.FormatFloat(point.X, 'f', -1, 64)),
2016-03-28 18:57:41 +03:00
}))
}
2016-03-05 02:08:16 +03:00
}
2016-07-10 23:23:50 +03:00
case "hash":
if vs, sprecision, ok = tokenval(vs); !ok || sprecision == "" {
return "", errInvalidNumberOfArguments
}
if msg.OutputType == server.JSON {
buf.WriteString(`,"hash":`)
}
precision, err := strconv.ParseInt(sprecision, 10, 64)
if err != nil || precision < 1 || precision > 64 {
return "", errInvalidArgument(sprecision)
}
p, err := o.Geohash(int(precision))
if err != nil {
return "", err
}
if msg.OutputType == server.JSON {
buf.WriteString(`"` + p + `"`)
} else {
vals = append(vals, resp.StringValue(p))
}
case "bounds":
bbox := o.CalculatedBBox()
if msg.OutputType == server.JSON {
buf.WriteString(`,"bounds":`)
buf.WriteString(bbox.ExternalJSON())
} else {
vals = append(vals, resp.ArrayValue([]resp.Value{
resp.ArrayValue([]resp.Value{
resp.FloatValue(bbox.Min.Y),
resp.FloatValue(bbox.Min.X),
}),
resp.ArrayValue([]resp.Value{
resp.FloatValue(bbox.Max.Y),
resp.FloatValue(bbox.Max.X),
}),
}))
}
2016-03-05 02:08:16 +03:00
}
2016-07-10 23:23:50 +03:00
2016-03-28 18:57:41 +03:00
if len(vs) != 0 {
2016-03-05 02:08:16 +03:00
return "", errInvalidNumberOfArguments
}
2016-04-01 22:46:39 +03:00
if withfields {
fvs := orderFields(col.FieldMap(), fields)
if len(fvs) > 0 {
fvals := make([]resp.Value, 0, len(fvs)*2)
2016-03-28 18:57:41 +03:00
if msg.OutputType == server.JSON {
2016-04-01 22:46:39 +03:00
buf.WriteString(`,"fields":{`)
}
for i, fv := range fvs {
if msg.OutputType == server.JSON {
if i > 0 {
buf.WriteString(`,`)
}
buf.WriteString(jsonString(fv.field) + ":" + strconv.FormatFloat(fv.value, 'f', -1, 64))
} else {
fvals = append(fvals, resp.StringValue(fv.field), resp.StringValue(strconv.FormatFloat(fv.value, 'f', -1, 64)))
2016-03-05 02:08:16 +03:00
}
2016-04-01 22:46:39 +03:00
i++
}
if msg.OutputType == server.JSON {
buf.WriteString(`}`)
2016-03-28 18:57:41 +03:00
} else {
2016-04-01 22:46:39 +03:00
vals = append(vals, resp.ArrayValue(fvals))
2016-03-05 02:08:16 +03:00
}
2016-03-28 18:57:41 +03:00
}
}
2016-04-01 22:46:39 +03:00
switch msg.OutputType {
case server.JSON:
2016-03-28 18:57:41 +03:00
buf.WriteString(`,"elapsed":"` + time.Now().Sub(start).String() + "\"}")
return buf.String(), nil
2016-04-01 22:46:39 +03:00
case server.RESP:
var oval resp.Value
if withfields {
oval = resp.ArrayValue(vals)
} else {
oval = vals[0]
}
data, err := oval.MarshalRESP()
if err != nil {
return "", err
}
return string(data), nil
2016-03-05 02:08:16 +03:00
}
2016-04-01 22:46:39 +03:00
return "", nil
2016-03-05 02:08:16 +03:00
}
2016-03-28 18:57:41 +03:00
func (c *Controller) cmdDel(msg *server.Message) (res string, d commandDetailsT, err error) {
start := time.Now()
vs := msg.Values[1:]
var ok bool
if vs, d.key, ok = tokenval(vs); !ok || d.key == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if vs, d.id, ok = tokenval(vs); !ok || d.id == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if len(vs) != 0 {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
found := false
2016-03-05 02:08:16 +03:00
col := c.getCol(d.key)
if col != nil {
2016-03-28 18:57:41 +03:00
d.obj, d.fields, ok = col.Remove(d.id)
if ok {
2016-07-13 07:59:36 +03:00
if col.Count() == 0 {
2016-03-28 18:57:41 +03:00
c.deleteCol(d.key)
}
found = true
2016-03-05 02:08:16 +03:00
}
}
2016-07-15 22:22:48 +03:00
c.clearIDExpires(d.key, d.id)
2016-03-05 02:08:16 +03:00
d.command = "del"
2016-03-28 18:57:41 +03:00
d.updated = found
2016-04-02 17:20:30 +03:00
d.timestamp = time.Now()
2016-03-28 18:57:41 +03:00
switch msg.OutputType {
case server.JSON:
res = `{"ok":true,"elapsed":"` + time.Now().Sub(start).String() + "\"}"
case server.RESP:
if d.updated {
res = ":1\r\n"
} else {
res = ":0\r\n"
}
}
2016-03-05 02:08:16 +03:00
return
}
func (c *Controller) cmdPdel(msg *server.Message) (res string, d commandDetailsT, err error) {
start := time.Now()
vs := msg.Values[1:]
var ok bool
if vs, d.key, ok = tokenval(vs); !ok || d.key == "" {
err = errInvalidNumberOfArguments
return
}
if vs, d.pattern, ok = tokenval(vs); !ok || d.pattern == "" {
err = errInvalidNumberOfArguments
return
}
if len(vs) != 0 {
err = errInvalidNumberOfArguments
return
}
now := time.Now()
iter := func(id string, o geojson.Object, fields []float64) bool {
if match, _ := glob.Match(d.pattern, id); match {
d.children = append(d.children, &commandDetailsT{
command: "del",
updated: true,
timestamp: now,
key: d.key,
id: id,
})
}
return true
}
var expired int
col := c.getCol(d.key)
if col != nil {
g := glob.Parse(d.pattern, false)
if g.Limits[0] == "" && g.Limits[1] == "" {
col.Scan(false, iter)
} else {
col.ScanRange(g.Limits[0], g.Limits[1], false, iter)
}
var atLeastOneNotDeleted bool
for i, dc := range d.children {
dc.obj, dc.fields, ok = col.Remove(dc.id)
if !ok {
d.children[i].command = "?"
atLeastOneNotDeleted = true
} else {
d.children[i] = dc
}
c.clearIDExpires(d.key, dc.id)
}
if atLeastOneNotDeleted {
var nchildren []*commandDetailsT
for _, dc := range d.children {
if dc.command == "del" {
nchildren = append(nchildren, dc)
}
}
d.children = nchildren
}
if col.Count() == 0 {
c.deleteCol(d.key)
}
}
d.command = "pdel"
d.updated = len(d.children) > 0
d.timestamp = now
d.parent = true
switch msg.OutputType {
case server.JSON:
res = `{"ok":true,"elapsed":"` + time.Now().Sub(start).String() + "\"}"
case server.RESP:
total := len(d.children) - expired
if total < 0 {
total = 0
}
res = ":" + strconv.FormatInt(int64(total), 10) + "\r\n"
}
return
}
2016-03-28 18:57:41 +03:00
func (c *Controller) cmdDrop(msg *server.Message) (res string, d commandDetailsT, err error) {
start := time.Now()
vs := msg.Values[1:]
var ok bool
if vs, d.key, ok = tokenval(vs); !ok || d.key == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if len(vs) != 0 {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
col := c.getCol(d.key)
if col != nil {
c.deleteCol(d.key)
2016-03-28 18:57:41 +03:00
d.updated = true
2016-03-05 02:08:16 +03:00
} else {
d.key = "" // ignore the details
2016-03-28 18:57:41 +03:00
d.updated = false
2016-03-05 02:08:16 +03:00
}
d.command = "drop"
2016-04-02 17:20:30 +03:00
d.timestamp = time.Now()
2016-07-15 22:22:48 +03:00
c.clearKeyExpires(d.key)
2016-03-28 18:57:41 +03:00
switch msg.OutputType {
case server.JSON:
res = `{"ok":true,"elapsed":"` + time.Now().Sub(start).String() + "\"}"
case server.RESP:
if d.updated {
res = ":1\r\n"
} else {
res = ":0\r\n"
}
}
2016-03-05 02:08:16 +03:00
return
}
2016-03-28 18:57:41 +03:00
func (c *Controller) cmdFlushDB(msg *server.Message) (res string, d commandDetailsT, err error) {
start := time.Now()
vs := msg.Values[1:]
if len(vs) != 0 {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-07-10 05:44:28 +03:00
c.cols = btree.New(16, 0)
2016-07-15 22:22:48 +03:00
c.clearAllExpires()
2016-03-19 17:16:19 +03:00
c.hooks = make(map[string]*Hook)
c.hookcols = make(map[string]map[string]*Hook)
2016-03-05 02:08:16 +03:00
d.command = "flushdb"
2016-03-28 18:57:41 +03:00
d.updated = true
2016-04-02 17:20:30 +03:00
d.timestamp = time.Now()
2016-03-28 18:57:41 +03:00
switch msg.OutputType {
case server.JSON:
res = `{"ok":true,"elapsed":"` + time.Now().Sub(start).String() + "\"}"
case server.RESP:
res = "+OK\r\n"
}
2016-03-05 02:08:16 +03:00
return
}
2016-07-15 22:22:48 +03:00
func (c *Controller) parseSetArgs(vs []resp.Value) (
d commandDetailsT, fields []string, values []float64,
2016-10-03 18:31:13 +03:00
xx, nx bool,
expires *float64, etype []byte, evs []resp.Value, err error,
2016-07-15 22:22:48 +03:00
) {
2016-03-28 18:57:41 +03:00
var ok bool
var typ []byte
2016-03-28 18:57:41 +03:00
if vs, d.key, ok = tokenval(vs); !ok || d.key == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if vs, d.id, ok = tokenval(vs); !ok || d.id == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
var arg []byte
2016-03-28 18:57:41 +03:00
var nvs []resp.Value
2016-03-05 02:08:16 +03:00
for {
if nvs, arg, ok = tokenvalbytes(vs); !ok || len(arg) == 0 {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
if lcb(arg, "field") {
2016-03-28 18:57:41 +03:00
vs = nvs
2016-03-05 02:08:16 +03:00
var name string
var svalue string
var value float64
2016-03-28 18:57:41 +03:00
if vs, name, ok = tokenval(vs); !ok || name == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
if isReservedFieldName(name) {
err = errInvalidArgument(name)
return
}
2016-03-28 18:57:41 +03:00
if vs, svalue, ok = tokenval(vs); !ok || svalue == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
value, err = strconv.ParseFloat(svalue, 64)
if err != nil {
err = errInvalidArgument(svalue)
return
}
fields = append(fields, name)
values = append(values, value)
continue
}
if lcb(arg, "ex") {
2016-07-15 22:22:48 +03:00
vs = nvs
if expires != nil {
err = errInvalidArgument(string(arg))
2016-07-15 22:22:48 +03:00
return
}
var s string
var v float64
if vs, s, ok = tokenval(vs); !ok || s == "" {
err = errInvalidNumberOfArguments
return
}
v, err = strconv.ParseFloat(s, 64)
if err != nil {
err = errInvalidArgument(s)
return
}
expires = &v
continue
}
if lcb(arg, "xx") {
2016-10-03 18:31:13 +03:00
vs = nvs
if nx {
err = errInvalidArgument(string(arg))
2016-10-03 18:31:13 +03:00
return
}
xx = true
continue
}
if lcb(arg, "nx") {
2016-10-03 18:31:13 +03:00
vs = nvs
if xx {
err = errInvalidArgument(string(arg))
2016-10-03 18:31:13 +03:00
return
}
nx = true
continue
}
2016-03-05 02:08:16 +03:00
break
}
if vs, typ, ok = tokenvalbytes(vs); !ok || len(typ) == 0 {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if len(vs) == 0 {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
etype = typ
2016-03-28 18:57:41 +03:00
evs = vs
2016-03-05 02:08:16 +03:00
switch {
default:
err = errInvalidArgument(string(typ))
2016-03-05 02:08:16 +03:00
return
case lcb(typ, "string"):
2016-07-10 05:44:28 +03:00
var str string
if vs, str, ok = tokenval(vs); !ok {
err = errInvalidNumberOfArguments
return
}
d.obj = geojson.String(str)
case lcb(typ, "point"):
2016-03-05 02:08:16 +03:00
var slat, slon, sz string
2016-03-28 18:57:41 +03:00
if vs, slat, ok = tokenval(vs); !ok || slat == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if vs, slon, ok = tokenval(vs); !ok || slon == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
vs, sz, ok = tokenval(vs)
if !ok || sz == "" {
2016-03-05 02:08:16 +03:00
var sp geojson.SimplePoint
sp.Y, err = strconv.ParseFloat(slat, 64)
if err != nil {
err = errInvalidArgument(slat)
return
}
sp.X, err = strconv.ParseFloat(slon, 64)
if err != nil {
err = errInvalidArgument(slon)
return
}
d.obj = sp
} else {
var sp geojson.Point
sp.Coordinates.Y, err = strconv.ParseFloat(slat, 64)
if err != nil {
err = errInvalidArgument(slat)
return
}
sp.Coordinates.X, err = strconv.ParseFloat(slon, 64)
if err != nil {
err = errInvalidArgument(slon)
return
}
sp.Coordinates.Z, err = strconv.ParseFloat(sz, 64)
if err != nil {
err = errInvalidArgument(sz)
return
}
d.obj = sp
}
case lcb(typ, "bounds"):
2016-03-05 02:08:16 +03:00
var sminlat, sminlon, smaxlat, smaxlon string
2016-03-28 18:57:41 +03:00
if vs, sminlat, ok = tokenval(vs); !ok || sminlat == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if vs, sminlon, ok = tokenval(vs); !ok || sminlon == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if vs, smaxlat, ok = tokenval(vs); !ok || smaxlat == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if vs, smaxlon, ok = tokenval(vs); !ok || smaxlon == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
var minlat, minlon, maxlat, maxlon float64
minlat, err = strconv.ParseFloat(sminlat, 64)
if err != nil {
err = errInvalidArgument(sminlat)
return
}
minlon, err = strconv.ParseFloat(sminlon, 64)
if err != nil {
err = errInvalidArgument(sminlon)
return
}
maxlat, err = strconv.ParseFloat(smaxlat, 64)
if err != nil {
err = errInvalidArgument(smaxlat)
return
}
maxlon, err = strconv.ParseFloat(smaxlon, 64)
if err != nil {
err = errInvalidArgument(smaxlon)
return
}
g := geojson.Polygon{
Coordinates: [][]geojson.Position{
2016-04-03 05:19:43 +03:00
{
{X: minlon, Y: minlat, Z: 0},
{X: minlon, Y: maxlat, Z: 0},
{X: maxlon, Y: maxlat, Z: 0},
{X: maxlon, Y: minlat, Z: 0},
{X: minlon, Y: minlat, Z: 0},
2016-03-05 02:08:16 +03:00
},
},
}
d.obj = g
case lcb(typ, "hash"):
2016-03-05 02:08:16 +03:00
var sp geojson.SimplePoint
var shash string
2016-03-28 18:57:41 +03:00
if vs, shash, ok = tokenval(vs); !ok || shash == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
var lat, lon float64
lat, lon, err = geohash.Decode(shash)
if err != nil {
return
}
sp.X = lon
sp.Y = lat
d.obj = sp
case lcb(typ, "object"):
2016-03-28 18:57:41 +03:00
var object string
if vs, object, ok = tokenval(vs); !ok || object == "" {
err = errInvalidNumberOfArguments
return
}
d.obj, err = geojson.ObjectJSON(object)
2016-03-05 02:08:16 +03:00
if err != nil {
return
}
}
2016-03-28 18:57:41 +03:00
if len(vs) != 0 {
err = errInvalidNumberOfArguments
}
2016-03-05 02:08:16 +03:00
return
}
2016-03-28 18:57:41 +03:00
func (c *Controller) cmdSet(msg *server.Message) (res string, d commandDetailsT, err error) {
2016-05-24 05:44:25 +03:00
if c.config.MaxMemory > 0 && c.outOfMemory {
err = errOOM
return
}
2016-03-28 18:57:41 +03:00
start := time.Now()
vs := msg.Values[1:]
2016-10-03 18:31:13 +03:00
var fmap map[string]int
2016-03-05 02:08:16 +03:00
var fields []string
var values []float64
2016-10-03 18:31:13 +03:00
var xx, nx bool
2016-07-15 22:22:48 +03:00
var ex *float64
2016-10-03 18:31:13 +03:00
d, fields, values, xx, nx, ex, _, _, err = c.parseSetArgs(vs)
2016-03-05 02:08:16 +03:00
if err != nil {
return
}
2016-07-15 22:22:48 +03:00
ex = ex
2016-03-05 02:08:16 +03:00
col := c.getCol(d.key)
if col == nil {
2016-10-03 18:31:13 +03:00
if xx {
goto notok
}
2016-03-05 02:08:16 +03:00
col = collection.New()
c.setCol(d.key, col)
}
2016-10-31 20:17:30 +03:00
if xx || nx {
2016-10-03 18:31:13 +03:00
_, _, ok := col.Get(d.id)
2016-10-31 20:17:30 +03:00
if (nx && ok) || (xx && !ok) {
2016-10-03 18:31:13 +03:00
goto notok
}
}
2016-07-15 22:22:48 +03:00
c.clearIDExpires(d.key, d.id)
2016-03-05 02:08:16 +03:00
d.oldObj, d.oldFields, d.fields = col.ReplaceOrInsert(d.id, d.obj, fields, values)
d.command = "set"
2016-03-28 18:57:41 +03:00
d.updated = true // perhaps we should do a diff on the previous object?
2016-04-02 17:20:30 +03:00
d.timestamp = time.Now()
if msg.ConnType != server.Null || msg.OutputType != server.Null {
// likely loaded from aof at server startup, ignore field remapping.
fmap = col.FieldMap()
d.fmap = make(map[string]int)
for key, idx := range fmap {
d.fmap[key] = idx
}
}
2016-07-15 22:22:48 +03:00
if ex != nil {
c.expireAt(d.key, d.id, d.timestamp.Add(time.Duration(float64(time.Second)*(*ex))))
}
2016-03-28 18:57:41 +03:00
switch msg.OutputType {
default:
2016-03-28 18:57:41 +03:00
case server.JSON:
res = `{"ok":true,"elapsed":"` + time.Now().Sub(start).String() + "\"}"
case server.RESP:
res = "+OK\r\n"
}
2016-03-05 02:08:16 +03:00
return
2016-10-03 18:31:13 +03:00
notok:
switch msg.OutputType {
default:
2016-10-03 18:31:13 +03:00
case server.JSON:
2016-10-31 20:17:30 +03:00
if nx {
err = errIDAlreadyExists
} else {
err = errIDNotFound
}
return
2016-10-03 18:31:13 +03:00
case server.RESP:
res = "$-1\r\n"
}
return
2016-03-05 02:08:16 +03:00
}
2016-03-28 18:57:41 +03:00
func (c *Controller) parseFSetArgs(vs []resp.Value) (d commandDetailsT, err error) {
2016-03-05 02:08:16 +03:00
var svalue string
2016-03-28 18:57:41 +03:00
var ok bool
if vs, d.key, ok = tokenval(vs); !ok || d.key == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if vs, d.id, ok = tokenval(vs); !ok || d.id == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if vs, d.field, ok = tokenval(vs); !ok || d.field == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
if isReservedFieldName(d.field) {
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if vs, svalue, ok = tokenval(vs); !ok || svalue == "" {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
2016-03-28 18:57:41 +03:00
if len(vs) != 0 {
2016-03-05 02:08:16 +03:00
err = errInvalidNumberOfArguments
return
}
d.value, err = strconv.ParseFloat(svalue, 64)
if err != nil {
err = errInvalidArgument(svalue)
return
}
return
}
2016-03-28 18:57:41 +03:00
func (c *Controller) cmdFset(msg *server.Message) (res string, d commandDetailsT, err error) {
start := time.Now()
vs := msg.Values[1:]
d, err = c.parseFSetArgs(vs)
2016-03-05 02:08:16 +03:00
col := c.getCol(d.key)
if col == nil {
err = errKeyNotFound
return
}
var ok bool
2016-04-02 17:20:30 +03:00
d.obj, d.fields, d.updated, ok = col.SetField(d.id, d.field, d.value)
2016-03-05 02:08:16 +03:00
if !ok {
err = errIDNotFound
return
}
d.command = "fset"
2016-04-02 17:20:30 +03:00
d.timestamp = time.Now()
fmap := col.FieldMap()
d.fmap = make(map[string]int)
for key, idx := range fmap {
d.fmap[key] = idx
}
2016-03-28 18:57:41 +03:00
switch msg.OutputType {
case server.JSON:
res = `{"ok":true,"elapsed":"` + time.Now().Sub(start).String() + "\"}"
case server.RESP:
2016-04-02 17:20:30 +03:00
if d.updated {
2016-03-28 18:57:41 +03:00
res = ":1\r\n"
} else {
res = ":0\r\n"
}
}
2016-03-05 02:08:16 +03:00
return
}
2016-07-15 22:22:48 +03:00
func (c *Controller) cmdExpire(msg *server.Message) (res string, d commandDetailsT, err error) {
start := time.Now()
vs := msg.Values[1:]
var key, id, svalue string
var ok bool
if vs, key, ok = tokenval(vs); !ok || key == "" {
err = errInvalidNumberOfArguments
return
}
if vs, id, ok = tokenval(vs); !ok || id == "" {
err = errInvalidNumberOfArguments
return
}
if vs, svalue, ok = tokenval(vs); !ok || svalue == "" {
err = errInvalidNumberOfArguments
return
}
if len(vs) != 0 {
err = errInvalidNumberOfArguments
return
}
var value float64
value, err = strconv.ParseFloat(svalue, 64)
if err != nil {
err = errInvalidArgument(svalue)
return
}
ok = false
col := c.getCol(key)
if col != nil {
_, _, ok = col.Get(id)
ok = ok && !c.hasExpired(key, id)
}
if ok {
c.expireAt(key, id, time.Now().Add(time.Duration(float64(time.Second)*value)))
d.updated = true
2016-07-15 22:22:48 +03:00
}
switch msg.OutputType {
case server.JSON:
if ok {
res = `{"ok":true,"elapsed":"` + time.Now().Sub(start).String() + "\"}"
} else {
return "", d, errIDNotFound
}
2016-07-15 22:22:48 +03:00
case server.RESP:
if ok {
res = ":1\r\n"
} else {
res = ":0\r\n"
}
}
return
}
func (c *Controller) cmdPersist(msg *server.Message) (res string, d commandDetailsT, err error) {
start := time.Now()
vs := msg.Values[1:]
var key, id string
var ok bool
if vs, key, ok = tokenval(vs); !ok || key == "" {
err = errInvalidNumberOfArguments
return
}
if vs, id, ok = tokenval(vs); !ok || id == "" {
err = errInvalidNumberOfArguments
return
}
if len(vs) != 0 {
err = errInvalidNumberOfArguments
return
}
var cleared bool
2016-07-15 22:22:48 +03:00
ok = false
col := c.getCol(key)
if col != nil {
_, _, ok = col.Get(id)
ok = ok && !c.hasExpired(key, id)
2016-07-15 22:22:48 +03:00
if ok {
cleared = c.clearIDExpires(key, id)
}
}
if !ok {
if msg.OutputType == server.RESP {
return ":0\r\n", d, nil
2016-07-15 22:22:48 +03:00
}
return "", d, errIDNotFound
2016-07-15 22:22:48 +03:00
}
d.command = "persist"
d.updated = cleared
d.timestamp = time.Now()
2016-07-15 22:22:48 +03:00
switch msg.OutputType {
case server.JSON:
res = `{"ok":true,"elapsed":"` + time.Now().Sub(start).String() + "\"}"
case server.RESP:
if cleared {
2016-07-15 22:22:48 +03:00
res = ":1\r\n"
} else {
res = ":0\r\n"
}
}
return
}
func (c *Controller) cmdTTL(msg *server.Message) (res string, err error) {
2016-07-15 22:22:48 +03:00
start := time.Now()
vs := msg.Values[1:]
var key, id string
var ok bool
if vs, key, ok = tokenval(vs); !ok || key == "" {
err = errInvalidNumberOfArguments
return
}
if vs, id, ok = tokenval(vs); !ok || id == "" {
err = errInvalidNumberOfArguments
return
}
if len(vs) != 0 {
err = errInvalidNumberOfArguments
return
}
var v float64
ok = false
var ok2 bool
col := c.getCol(key)
if col != nil {
_, _, ok = col.Get(id)
ok = ok && !c.hasExpired(key, id)
2016-07-15 22:22:48 +03:00
if ok {
var at time.Time
at, ok2 = c.getExpires(key, id)
if ok2 {
if time.Now().After(at) {
ok2 = false
} else {
v = float64(at.Sub(time.Now())) / float64(time.Second)
if v < 0 {
v = 0
}
2016-07-15 22:22:48 +03:00
}
}
}
}
switch msg.OutputType {
case server.JSON:
if ok {
var ttl string
if ok2 {
ttl = strconv.FormatFloat(v, 'f', -1, 64)
} else {
ttl = "-1"
}
res = `{"ok":true,"ttl":` + ttl + `,"elapsed":"` + time.Now().Sub(start).String() + "\"}"
} else {
return "", errIDNotFound
}
2016-07-15 22:22:48 +03:00
case server.RESP:
if ok {
if ok2 {
2016-12-02 19:14:34 +03:00
res = ":" + strconv.FormatInt(int64(v), 10) + "\r\n"
2016-07-15 22:22:48 +03:00
} else {
res = ":-1\r\n"
}
} else {
res = ":-2\r\n"
}
}
return
}