http api of bit cmds

This commit is contained in:
wenyekui 2014-07-22 11:31:56 +08:00
parent f85da1db91
commit 1513d055f5
5 changed files with 358 additions and 24 deletions

View File

@ -3,6 +3,7 @@ package server
import ( import (
"fmt" "fmt"
"github.com/siddontang/ledisdb/ledis" "github.com/siddontang/ledisdb/ledis"
. "github.com/siddontang/ledisdb/server/http"
"net" "net"
"net/http" "net/http"
"path" "path"
@ -133,8 +134,8 @@ func (app *App) httpServe() {
mux := http.NewServeMux() mux := http.NewServeMux()
mux.Handle("/ws", &wsHandler{app}) mux.Handle("/ws", &WsHandler{app.Ledis()})
mux.Handle("/", &cmdHandler{app}) mux.Handle("/", &CmdHandler{app.Ledis()})
svr := http.Server{Handler: mux} svr := http.Server{Handler: mux}
svr.Serve(app.httpListener) svr.Serve(app.httpListener)

26
server/http/base.go Normal file
View File

@ -0,0 +1,26 @@
package http
import (
"errors"
"fmt"
"github.com/siddontang/ledisdb/ledis"
"strings"
)
const ERR_ARGUMENT_FORMAT = "ERR wrong number of arguments for '%s' command"
var ErrValue = errors.New("ERR value is not an integer or out of range")
type commondFunc func(*ledis.DB, ...string) (interface{}, error)
var regCmds = map[string]commondFunc{}
func register(name string, f commondFunc) {
if _, ok := regCmds[strings.ToLower(name)]; ok {
panic(fmt.Sprintf("%s has been registered", name))
}
regCmds[name] = f
}
func lookup(name string) commondFunc {
return regCmds[strings.ToLower(name)]
}

228
server/http/cmd_bit.go Normal file
View File

@ -0,0 +1,228 @@
package http
import (
"fmt"
"github.com/siddontang/ledisdb/ledis"
"strconv"
"strings"
)
func bgetCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 1 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bget")
}
if v, err := db.BGet(ledis.Slice(args[0])); err != nil {
return nil, err
} else {
return v, nil
}
}
func bdeleteCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 1 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bdelete")
}
if n, err := db.BDelete(ledis.Slice(args[0])); err != nil {
return nil, err
} else {
return n, err
}
}
func bsetbitCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bsetbit")
}
key := ledis.Slice(args[0])
offset, err := strconv.ParseInt(args[1], 10, 32)
if err != nil {
return nil, ErrValue
}
val, err := strconv.ParseUint(args[2], 10, 8)
if ori, err := db.BSetBit(key, int32(offset), uint8(val)); err != nil {
return nil, err
} else {
return ori, nil
}
}
func bgetbitCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 2 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bgetbit")
}
key := ledis.Slice(args[0])
offset, err := strconv.ParseInt(args[1], 10, 32)
if err != nil {
return nil, ErrValue
}
if v, err := db.BGetBit(key, int32(offset)); err != nil {
return nil, err
} else {
return v, nil
}
return nil, nil
}
func bmsetbitCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) < 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bmsetbit")
}
key := ledis.Slice(args[0])
if len(args[1:])%2 != 0 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bmsetbit")
} else {
args = args[1:]
}
pairs := make([]ledis.BitPair, len(args)/2)
for i := 0; i < len(pairs); i++ {
offset, err := strconv.ParseInt(args[i*2], 10, 32)
if err != nil {
return nil, err
}
val, err := strconv.ParseUint(args[i*2+1], 10, 8)
if err != nil {
return nil, err
}
pairs[i].Pos = int32(offset)
pairs[i].Val = uint8(val)
}
if place, err := db.BMSetBit(key, pairs...); err != nil {
return nil, err
} else {
return place, nil
}
}
func bcountCommand(db *ledis.DB, args ...string) (interface{}, error) {
argCnt := len(args)
if argCnt > 3 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bcount")
}
var err error
var start, end int64 = 0, -1
if argCnt > 1 {
if start, err = strconv.ParseInt(args[1], 10, 32); err != nil {
return nil, err
}
}
if argCnt > 2 {
if end, err = strconv.ParseInt(args[1], 10, 32); err != nil {
return nil, err
}
}
key := ledis.Slice(args[0])
if cnt, err := db.BCount(key, int32(start), int32(end)); err != nil {
return nil, err
} else {
return cnt, nil
}
}
func boptCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) < 2 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bopt")
}
opDesc := strings.ToLower(args[0])
dstKey := ledis.Slice(args[1])
var srcKeys = [][]byte{}
if len(args) >= 3 {
srcKeys = make([][]byte, len(args[2:]))
for i, arg := range args[2:] {
srcKeys[i] = ledis.Slice(arg)
}
}
var op uint8
switch opDesc {
case "and":
op = ledis.OPand
case "or":
op = ledis.OPor
case "xor":
op = ledis.OPxor
case "not":
op = ledis.OPnot
default:
return nil, fmt.Errorf("ERR invalid argument '%s' for 'bopt' command", opDesc)
}
if blen, err := db.BOperation(op, dstKey, srcKeys...); err != nil {
return nil, err
} else {
return blen, nil
}
}
func bexpireCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 2 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bexpire")
}
duration, err := strconv.ParseInt(args[1], 10, 64)
if err != nil {
return nil, err
}
key := ledis.Slice(args[0])
if v, err := db.BExpire(key, duration); err != nil {
return nil, err
} else {
return v, err
}
}
func bexpireatCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 2 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bexpireat")
}
when, err := strconv.ParseInt(args[1], 10, 64)
if err != nil {
return nil, err
}
key := ledis.Slice(args[0])
if v, err := db.BExpireAt(key, when); err != nil {
return nil, err
} else {
return v, nil
}
}
func bttlCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 1 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bttl")
}
key := ledis.Slice(args[0])
if v, err := db.BTTL(key); err != nil {
return nil, err
} else {
return v, err
}
}
func bpersistCommand(db *ledis.DB, args ...string) (interface{}, error) {
if len(args) != 1 {
return nil, fmt.Errorf(ERR_ARGUMENT_FORMAT, "bpersist")
}
key := ledis.Slice(args[0])
if n, err := db.BPersist(key); err != nil {
return nil, err
} else {
return n, nil
}
}
func init() {
register("bget", bgetCommand)
register("bdelete", bdeleteCommand)
register("bsetbit", bsetbitCommand)
register("bgetbit", bgetbitCommand)
register("bmsetbit", bmsetbitCommand)
register("bcount", bcountCommand)
register("bopt", boptCommand)
register("bexpire", bexpireCommand)
register("bexpireat", bexpireatCommand)
register("bttl", bttlCommand)
register("bpersist", bpersistCommand)
}

101
server/http/handler.go Normal file
View File

@ -0,0 +1,101 @@
package http
import (
"net/http"
//"github.com/siddontang/go-websocket/websocket"
"encoding/json"
"fmt"
"github.com/siddontang/go-log/log"
"github.com/siddontang/ledisdb/ledis"
"strconv"
"strings"
)
type CmdHandler struct {
Ldb *ledis.Ledis
}
func (h *CmdHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
if r.URL.Path == "/" {
w.WriteHeader(http.StatusForbidden)
return
}
idx, cmd, args := h.parseReqPath(r.URL.Path)
cmdFunc := lookup(cmd)
if cmdFunc == nil {
h.cmdNotFound(cmd, w)
return
}
var db *ledis.DB
var err error
if db, err = h.Ldb.Select(idx); err != nil {
h.serverError(cmd, err, w)
return
}
result, err := cmdFunc(db, args...)
if err != nil {
h.serverError(cmd, err, w)
return
}
h.write(cmd, result, w)
}
func (h *CmdHandler) parseReqPath(path string) (db int, cmd string, args []string) {
substrings := strings.Split(strings.TrimLeft(path, "/"), "/")
if len(substrings) == 1 {
return 0, substrings[0], substrings[1:]
}
db, err := strconv.Atoi(substrings[0])
if err != nil {
// db = 0
cmd = substrings[0]
args = substrings[1:]
} else {
cmd = substrings[1]
args = substrings[2:]
}
return
}
func (h *CmdHandler) cmdNotFound(cmd string, w http.ResponseWriter) {
result := [2]interface{}{
false,
fmt.Sprintf("ERR unknown command '%s'", cmd),
}
h.write(cmd, result, w)
}
func (h *CmdHandler) write(cmd string, result interface{}, w http.ResponseWriter) {
m := map[string]interface{}{
cmd: result,
}
buf, err := json.Marshal(&m)
if err != nil {
log.Error(err.Error())
return
}
w.Header().Set("Content-type", "application/json; charset=utf-8")
w.Header().Set("Content-Length", strconv.Itoa(len(buf)))
_, err = w.Write(buf)
if err != nil {
log.Error(err.Error())
}
}
func (h *CmdHandler) serverError(cmd string, err error, w http.ResponseWriter) {
result := [2]interface{}{
false,
fmt.Sprintf("ERR %s", err.Error()),
}
h.write(cmd, result, w)
}
type WsHandler struct {
Ldb *ledis.Ledis
}
func (h *WsHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("ws handler"))
}

View File

@ -1,22 +0,0 @@
package server
import (
"net/http"
//"github.com/siddontang/go-websocket/websocket"
)
type cmdHandler struct {
app *App
}
func (h *cmdHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("cmd handler"))
}
type wsHandler struct {
app *App
}
func (h *wsHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("ws handler"))
}