2014-05-09 10:49:22 +04:00
|
|
|
package ledis
|
2014-05-06 13:37:58 +04:00
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
|
|
|
"errors"
|
2014-05-14 12:35:49 +04:00
|
|
|
"github.com/siddontang/go-leveldb/leveldb"
|
2014-05-06 13:37:58 +04:00
|
|
|
)
|
|
|
|
|
2014-05-16 04:56:32 +04:00
|
|
|
type FVPair struct {
|
|
|
|
Field []byte
|
|
|
|
Value []byte
|
|
|
|
}
|
|
|
|
|
2014-05-06 13:37:58 +04:00
|
|
|
var errHashKey = errors.New("invalid hash key")
|
|
|
|
var errHSizeKey = errors.New("invalid hsize key")
|
|
|
|
|
|
|
|
const (
|
|
|
|
hashStartSep byte = ':'
|
2014-05-08 06:54:33 +04:00
|
|
|
hashStopSep byte = hashStartSep + 1
|
2014-05-06 13:37:58 +04:00
|
|
|
)
|
|
|
|
|
|
|
|
func encode_hsize_key(key []byte) []byte {
|
|
|
|
buf := make([]byte, len(key)+1)
|
2014-05-16 11:03:23 +04:00
|
|
|
buf[0] = hSizeType
|
2014-05-06 13:37:58 +04:00
|
|
|
|
|
|
|
copy(buf[1:], key)
|
|
|
|
return buf
|
|
|
|
}
|
|
|
|
|
|
|
|
func decode_hsize_key(ek []byte) ([]byte, error) {
|
2014-05-16 11:03:23 +04:00
|
|
|
if len(ek) == 0 || ek[0] != hSizeType {
|
2014-05-06 13:37:58 +04:00
|
|
|
return nil, errHSizeKey
|
|
|
|
}
|
|
|
|
|
|
|
|
return ek[1:], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func encode_hash_key(key []byte, field []byte) []byte {
|
|
|
|
buf := make([]byte, len(key)+len(field)+1+4+1)
|
|
|
|
|
|
|
|
pos := 0
|
2014-05-16 11:03:23 +04:00
|
|
|
buf[pos] = hashType
|
2014-05-06 13:37:58 +04:00
|
|
|
pos++
|
|
|
|
binary.BigEndian.PutUint32(buf[pos:], uint32(len(key)))
|
|
|
|
pos += 4
|
|
|
|
|
|
|
|
copy(buf[pos:], key)
|
|
|
|
pos += len(key)
|
|
|
|
|
|
|
|
buf[pos] = hashStartSep
|
|
|
|
pos++
|
|
|
|
copy(buf[pos:], field)
|
|
|
|
|
|
|
|
return buf
|
|
|
|
}
|
|
|
|
|
|
|
|
func encode_hash_start_key(key []byte) []byte {
|
2014-05-08 06:54:33 +04:00
|
|
|
k := encode_hash_key(key, nil)
|
|
|
|
return k
|
2014-05-06 13:37:58 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
func encode_hash_stop_key(key []byte) []byte {
|
2014-05-08 06:54:33 +04:00
|
|
|
k := encode_hash_key(key, nil)
|
2014-05-06 13:37:58 +04:00
|
|
|
|
2014-05-08 06:54:33 +04:00
|
|
|
k[len(k)-1] = hashStopSep
|
2014-05-06 13:37:58 +04:00
|
|
|
|
2014-05-08 06:54:33 +04:00
|
|
|
return k
|
2014-05-06 13:37:58 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
func decode_hash_key(ek []byte) ([]byte, []byte, error) {
|
2014-05-16 11:03:23 +04:00
|
|
|
if len(ek) < 6 || ek[0] != hashType {
|
2014-05-06 13:37:58 +04:00
|
|
|
return nil, nil, errHashKey
|
|
|
|
}
|
|
|
|
|
|
|
|
pos := 1
|
|
|
|
keyLen := int(binary.BigEndian.Uint32(ek[pos:]))
|
|
|
|
pos += 4
|
|
|
|
|
|
|
|
if keyLen+6 > len(ek) {
|
|
|
|
return nil, nil, errHashKey
|
|
|
|
}
|
|
|
|
|
|
|
|
key := ek[pos : pos+keyLen]
|
|
|
|
pos += keyLen
|
|
|
|
|
|
|
|
if ek[pos] != hashStartSep {
|
|
|
|
return nil, nil, errHashKey
|
|
|
|
}
|
|
|
|
|
|
|
|
pos++
|
|
|
|
field := ek[pos:]
|
|
|
|
return key, field, nil
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) HLen(key []byte) (int64, error) {
|
|
|
|
return Int64(db.db.Get(encode_hsize_key(key)))
|
2014-05-06 13:37:58 +04:00
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) hSetItem(key []byte, field []byte, value []byte) (int64, error) {
|
|
|
|
t := db.hashTx
|
2014-05-06 13:37:58 +04:00
|
|
|
|
|
|
|
ek := encode_hash_key(key, field)
|
|
|
|
|
|
|
|
var n int64 = 1
|
2014-05-15 10:19:48 +04:00
|
|
|
if v, _ := db.db.Get(ek); v != nil {
|
2014-05-06 13:37:58 +04:00
|
|
|
n = 0
|
|
|
|
} else {
|
2014-05-15 10:19:48 +04:00
|
|
|
if _, err := db.hIncrSize(key, 1); err != nil {
|
2014-05-14 12:35:49 +04:00
|
|
|
return 0, err
|
|
|
|
}
|
2014-05-06 13:37:58 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
t.Put(ek, value)
|
|
|
|
return n, nil
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) HSet(key []byte, field []byte, value []byte) (int64, error) {
|
|
|
|
t := db.hashTx
|
2014-05-06 13:37:58 +04:00
|
|
|
t.Lock()
|
|
|
|
defer t.Unlock()
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
n, err := db.hSetItem(key, field, value)
|
2014-05-06 13:37:58 +04:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//todo add binlog
|
|
|
|
|
|
|
|
err = t.Commit()
|
|
|
|
return n, err
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) HGet(key []byte, field []byte) ([]byte, error) {
|
|
|
|
return db.db.Get(encode_hash_key(key, field))
|
2014-05-06 13:37:58 +04:00
|
|
|
}
|
|
|
|
|
2014-05-16 04:56:32 +04:00
|
|
|
func (db *DB) HMset(key []byte, args ...FVPair) error {
|
2014-05-15 10:19:48 +04:00
|
|
|
t := db.hashTx
|
2014-05-06 13:37:58 +04:00
|
|
|
t.Lock()
|
|
|
|
defer t.Unlock()
|
|
|
|
|
2014-05-14 12:35:49 +04:00
|
|
|
var num int64 = 0
|
2014-05-16 04:56:32 +04:00
|
|
|
for i := 0; i < len(args); i++ {
|
|
|
|
ek := encode_hash_key(key, args[i].Field)
|
2014-05-15 10:19:48 +04:00
|
|
|
if v, _ := db.db.Get(ek); v == nil {
|
2014-05-14 12:35:49 +04:00
|
|
|
num++
|
2014-05-06 13:37:58 +04:00
|
|
|
}
|
|
|
|
|
2014-05-16 04:56:32 +04:00
|
|
|
t.Put(ek, args[i].Value)
|
2014-05-06 13:37:58 +04:00
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
if _, err := db.hIncrSize(key, num); err != nil {
|
2014-05-14 12:35:49 +04:00
|
|
|
return err
|
|
|
|
}
|
2014-05-06 13:37:58 +04:00
|
|
|
|
|
|
|
//todo add binglog
|
2014-05-14 12:35:49 +04:00
|
|
|
err := t.Commit()
|
2014-05-06 13:37:58 +04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) HMget(key []byte, args [][]byte) ([]interface{}, error) {
|
2014-05-06 13:37:58 +04:00
|
|
|
r := make([]interface{}, len(args))
|
|
|
|
for i := 0; i < len(args); i++ {
|
2014-05-15 10:19:48 +04:00
|
|
|
v, err := db.db.Get(encode_hash_key(key, args[i]))
|
2014-05-06 13:37:58 +04:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
r[i] = v
|
|
|
|
}
|
|
|
|
|
|
|
|
return r, nil
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) HDel(key []byte, args [][]byte) (int64, error) {
|
|
|
|
t := db.hashTx
|
2014-05-06 13:37:58 +04:00
|
|
|
t.Lock()
|
|
|
|
defer t.Unlock()
|
|
|
|
|
|
|
|
var num int64 = 0
|
|
|
|
for i := 0; i < len(args); i++ {
|
|
|
|
ek := encode_hash_key(key, args[i])
|
2014-05-15 10:19:48 +04:00
|
|
|
if v, err := db.db.Get(ek); err != nil {
|
2014-05-06 13:37:58 +04:00
|
|
|
return 0, err
|
|
|
|
} else if v == nil {
|
|
|
|
continue
|
|
|
|
} else {
|
|
|
|
num++
|
|
|
|
t.Delete(ek)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
if _, err := db.hIncrSize(key, -num); err != nil {
|
2014-05-14 12:35:49 +04:00
|
|
|
return 0, err
|
2014-05-06 13:37:58 +04:00
|
|
|
}
|
|
|
|
|
2014-05-14 12:35:49 +04:00
|
|
|
err := t.Commit()
|
2014-05-06 13:37:58 +04:00
|
|
|
|
|
|
|
return num, err
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) hIncrSize(key []byte, delta int64) (int64, error) {
|
|
|
|
t := db.hashTx
|
2014-05-14 12:35:49 +04:00
|
|
|
sk := encode_hsize_key(key)
|
2014-05-15 10:19:48 +04:00
|
|
|
size, err := Int64(db.db.Get(sk))
|
2014-05-14 12:35:49 +04:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
} else {
|
|
|
|
size += delta
|
|
|
|
if size <= 0 {
|
|
|
|
size = 0
|
|
|
|
t.Delete(sk)
|
|
|
|
} else {
|
|
|
|
t.Put(sk, PutInt64(size))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return size, nil
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) HIncrBy(key []byte, field []byte, delta int64) (int64, error) {
|
|
|
|
t := db.hashTx
|
2014-05-06 13:37:58 +04:00
|
|
|
t.Lock()
|
|
|
|
defer t.Unlock()
|
|
|
|
|
|
|
|
ek := encode_hash_key(key, field)
|
|
|
|
|
|
|
|
var n int64 = 0
|
2014-05-15 10:19:48 +04:00
|
|
|
n, err := StrInt64(db.db.Get(ek))
|
2014-05-06 13:37:58 +04:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
n += delta
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
_, err = db.hSetItem(key, field, StrPutInt64(n))
|
2014-05-06 13:37:58 +04:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = t.Commit()
|
|
|
|
|
|
|
|
return n, err
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) HGetAll(key []byte) ([]interface{}, error) {
|
2014-05-06 13:37:58 +04:00
|
|
|
start := encode_hash_start_key(key)
|
|
|
|
stop := encode_hash_stop_key(key)
|
|
|
|
|
|
|
|
v := make([]interface{}, 0, 16)
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
it := db.db.Iterator(start, stop, leveldb.RangeROpen, 0, -1)
|
2014-05-06 13:37:58 +04:00
|
|
|
for ; it.Valid(); it.Next() {
|
|
|
|
_, k, err := decode_hash_key(it.Key())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
v = append(v, k)
|
|
|
|
v = append(v, it.Value())
|
|
|
|
}
|
|
|
|
|
2014-05-08 06:54:33 +04:00
|
|
|
it.Close()
|
|
|
|
|
2014-05-06 13:37:58 +04:00
|
|
|
return v, nil
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) HKeys(key []byte) ([]interface{}, error) {
|
2014-05-06 13:37:58 +04:00
|
|
|
start := encode_hash_start_key(key)
|
|
|
|
stop := encode_hash_stop_key(key)
|
|
|
|
|
|
|
|
v := make([]interface{}, 0, 16)
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
it := db.db.Iterator(start, stop, leveldb.RangeROpen, 0, -1)
|
2014-05-06 13:37:58 +04:00
|
|
|
for ; it.Valid(); it.Next() {
|
|
|
|
_, k, err := decode_hash_key(it.Key())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
v = append(v, k)
|
|
|
|
}
|
|
|
|
|
2014-05-08 06:54:33 +04:00
|
|
|
it.Close()
|
|
|
|
|
2014-05-06 13:37:58 +04:00
|
|
|
return v, nil
|
|
|
|
}
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) HValues(key []byte) ([]interface{}, error) {
|
2014-05-06 13:37:58 +04:00
|
|
|
start := encode_hash_start_key(key)
|
|
|
|
stop := encode_hash_stop_key(key)
|
|
|
|
|
|
|
|
v := make([]interface{}, 0, 16)
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
it := db.db.Iterator(start, stop, leveldb.RangeROpen, 0, -1)
|
2014-05-06 13:37:58 +04:00
|
|
|
for ; it.Valid(); it.Next() {
|
|
|
|
v = append(v, it.Value())
|
|
|
|
}
|
|
|
|
|
2014-05-08 06:54:33 +04:00
|
|
|
it.Close()
|
|
|
|
|
2014-05-06 13:37:58 +04:00
|
|
|
return v, nil
|
|
|
|
}
|
2014-05-12 11:08:59 +04:00
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
func (db *DB) HClear(key []byte) (int64, error) {
|
2014-05-12 11:08:59 +04:00
|
|
|
sk := encode_hsize_key(key)
|
|
|
|
|
2014-05-15 10:19:48 +04:00
|
|
|
t := db.hashTx
|
2014-05-12 11:08:59 +04:00
|
|
|
t.Lock()
|
|
|
|
defer t.Unlock()
|
|
|
|
|
|
|
|
start := encode_hash_start_key(key)
|
|
|
|
stop := encode_hash_stop_key(key)
|
|
|
|
|
|
|
|
var num int64 = 0
|
2014-05-15 10:19:48 +04:00
|
|
|
it := db.db.Iterator(start, stop, leveldb.RangeROpen, 0, -1)
|
2014-05-12 11:08:59 +04:00
|
|
|
for ; it.Valid(); it.Next() {
|
|
|
|
t.Delete(it.Key())
|
|
|
|
num++
|
|
|
|
}
|
|
|
|
|
|
|
|
it.Close()
|
|
|
|
|
|
|
|
t.Delete(sk)
|
|
|
|
|
|
|
|
err := t.Commit()
|
|
|
|
return num, err
|
|
|
|
}
|