2016-03-05 02:08:16 +03:00
|
|
|
package controller
|
|
|
|
|
2016-03-30 19:32:38 +03:00
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"math/rand"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
"sync/atomic"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/tidwall/resp"
|
|
|
|
"github.com/tidwall/tile38/controller/log"
|
|
|
|
"github.com/tidwall/tile38/controller/server"
|
|
|
|
)
|
|
|
|
|
|
|
|
const useRandField = true
|
|
|
|
|
2016-04-04 04:13:33 +03:00
|
|
|
func randMassInsertPosition(minLat, minLon, maxLat, maxLon float64) (float64, float64) {
|
|
|
|
lat, lon := (rand.Float64()*(maxLat-minLat))+minLat, (rand.Float64()*(maxLon-minLon))+minLon
|
|
|
|
return lat, lon
|
|
|
|
}
|
|
|
|
|
2016-03-30 19:32:38 +03:00
|
|
|
func (c *Controller) cmdMassInsert(msg *server.Message) (res string, err error) {
|
|
|
|
start := time.Now()
|
|
|
|
vs := msg.Values[1:]
|
|
|
|
|
2016-04-04 04:13:33 +03:00
|
|
|
minLat, minLon, maxLat, maxLon := -90.0, -180.0, 90.0, 180.0 //37.10776, -122.67145, 38.19502, -121.62775
|
|
|
|
|
2016-03-30 19:32:38 +03:00
|
|
|
var snumCols, snumPoints string
|
|
|
|
var cols, objs int
|
|
|
|
var ok bool
|
|
|
|
if vs, snumCols, ok = tokenval(vs); !ok || snumCols == "" {
|
|
|
|
return "", errInvalidNumberOfArguments
|
|
|
|
}
|
|
|
|
if vs, snumPoints, ok = tokenval(vs); !ok || snumPoints == "" {
|
|
|
|
return "", errInvalidNumberOfArguments
|
|
|
|
}
|
|
|
|
if len(vs) != 0 {
|
2016-04-04 04:13:33 +03:00
|
|
|
var sminLat, sminLon, smaxLat, smaxLon string
|
|
|
|
if vs, sminLat, ok = tokenval(vs); !ok || sminLat == "" {
|
|
|
|
return "", errInvalidNumberOfArguments
|
|
|
|
}
|
|
|
|
if vs, sminLon, ok = tokenval(vs); !ok || sminLon == "" {
|
|
|
|
return "", errInvalidNumberOfArguments
|
|
|
|
}
|
|
|
|
if vs, smaxLat, ok = tokenval(vs); !ok || smaxLat == "" {
|
|
|
|
return "", errInvalidNumberOfArguments
|
|
|
|
}
|
|
|
|
if vs, smaxLon, ok = tokenval(vs); !ok || smaxLon == "" {
|
|
|
|
return "", errInvalidNumberOfArguments
|
|
|
|
}
|
|
|
|
var err error
|
|
|
|
if minLat, err = strconv.ParseFloat(sminLat, 64); err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if minLon, err = strconv.ParseFloat(sminLon, 64); err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if maxLat, err = strconv.ParseFloat(smaxLat, 64); err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if maxLon, err = strconv.ParseFloat(smaxLon, 64); err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if len(vs) != 0 {
|
|
|
|
return "", errors.New("invalid number of arguments")
|
|
|
|
}
|
2016-03-30 19:32:38 +03:00
|
|
|
}
|
|
|
|
n, err := strconv.ParseUint(snumCols, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return "", errInvalidArgument(snumCols)
|
|
|
|
}
|
|
|
|
cols = int(n)
|
|
|
|
n, err = strconv.ParseUint(snumPoints, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return "", errInvalidArgument(snumPoints)
|
|
|
|
}
|
|
|
|
docmd := func(values []resp.Value) error {
|
|
|
|
c.mu.Lock()
|
|
|
|
defer c.mu.Unlock()
|
|
|
|
nmsg := &server.Message{}
|
|
|
|
*nmsg = *msg
|
|
|
|
nmsg.Values = values
|
|
|
|
nmsg.Command = strings.ToLower(values[0].String())
|
|
|
|
|
|
|
|
_, d, err := c.command(nmsg, nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := c.writeAOF(resp.ArrayValue(nmsg.Values), &d); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
rand.Seed(time.Now().UnixNano())
|
|
|
|
objs = int(n)
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
var k uint64
|
|
|
|
wg.Add(cols)
|
|
|
|
for i := 0; i < cols; i++ {
|
|
|
|
key := "mi:" + strconv.FormatInt(int64(i), 10)
|
|
|
|
go func(key string) {
|
|
|
|
defer func() {
|
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
|
|
|
|
for j := 0; j < objs; j++ {
|
|
|
|
id := strconv.FormatInt(int64(j), 10)
|
2016-04-04 04:13:33 +03:00
|
|
|
lat, lon := randMassInsertPosition(minLat, minLon, maxLat, maxLon)
|
2016-03-30 19:32:38 +03:00
|
|
|
values := make([]resp.Value, 0, 16)
|
|
|
|
values = append(values, resp.StringValue("set"), resp.StringValue(key), resp.StringValue(id))
|
|
|
|
if useRandField {
|
|
|
|
values = append(values, resp.StringValue("FIELD"), resp.StringValue("field"), resp.FloatValue(rand.Float64()*10))
|
|
|
|
}
|
|
|
|
values = append(values, resp.StringValue("POINT"), resp.FloatValue(lat), resp.FloatValue(lon))
|
|
|
|
if err := docmd(values); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
atomic.AddUint64(&k, 1)
|
|
|
|
if j%10000 == 10000-1 {
|
|
|
|
log.Infof("massinsert: %s %d/%d", key, atomic.LoadUint64(&k), cols*objs)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}(key)
|
|
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
log.Infof("massinsert: done %d objects", atomic.LoadUint64(&k))
|
|
|
|
return server.OKMessage(msg, start), nil
|
2016-03-05 02:08:16 +03:00
|
|
|
}
|