ledisdb/ledis/t_list.go

670 lines
12 KiB
Go
Raw Permalink Normal View History

2014-05-09 10:49:22 +04:00
package ledis
2014-05-06 09:32:38 +04:00
import (
2014-10-17 05:55:43 +04:00
"container/list"
2014-05-06 09:32:38 +04:00
"encoding/binary"
"errors"
2014-10-16 13:51:52 +04:00
"github.com/siddontang/go/hack"
2014-07-25 13:58:00 +04:00
"github.com/siddontang/ledisdb/store"
2014-10-16 13:51:52 +04:00
"sync"
"time"
2014-05-06 09:32:38 +04:00
)
const (
2014-05-09 05:17:28 +04:00
listHeadSeq int32 = 1
listTailSeq int32 = 2
2014-05-06 09:32:38 +04:00
2014-05-09 05:17:28 +04:00
listMinSeq int32 = 1000
listMaxSeq int32 = 1<<31 - 1000
listInitialSeq int32 = listMinSeq + (listMaxSeq-listMinSeq)/2
2014-05-06 09:32:38 +04:00
)
2014-05-09 05:17:28 +04:00
var errLMetaKey = errors.New("invalid lmeta key")
2014-05-06 09:32:38 +04:00
var errListKey = errors.New("invalid list key")
var errListSeq = errors.New("invalid list sequence, overflow")
2014-05-20 04:41:24 +04:00
func (db *DB) lEncodeMetaKey(key []byte) []byte {
buf := make([]byte, len(key)+2)
buf[0] = db.index
buf[1] = LMetaType
2014-05-06 09:32:38 +04:00
2014-05-20 04:41:24 +04:00
copy(buf[2:], key)
2014-05-06 09:32:38 +04:00
return buf
}
2014-05-20 04:41:24 +04:00
func (db *DB) lDecodeMetaKey(ek []byte) ([]byte, error) {
if len(ek) < 2 || ek[0] != db.index || ek[1] != LMetaType {
2014-05-09 05:17:28 +04:00
return nil, errLMetaKey
2014-05-06 09:32:38 +04:00
}
2014-05-20 04:41:24 +04:00
return ek[2:], nil
2014-05-06 09:32:38 +04:00
}
2014-05-20 04:41:24 +04:00
func (db *DB) lEncodeListKey(key []byte, seq int32) []byte {
buf := make([]byte, len(key)+8)
2014-05-06 09:32:38 +04:00
pos := 0
2014-05-20 04:41:24 +04:00
buf[pos] = db.index
pos++
buf[pos] = ListType
2014-05-06 09:32:38 +04:00
pos++
2014-05-20 04:41:24 +04:00
binary.BigEndian.PutUint16(buf[pos:], uint16(len(key)))
pos += 2
2014-05-06 09:32:38 +04:00
copy(buf[pos:], key)
pos += len(key)
2014-05-09 05:17:28 +04:00
binary.BigEndian.PutUint32(buf[pos:], uint32(seq))
2014-05-06 09:32:38 +04:00
return buf
}
2014-05-20 04:41:24 +04:00
func (db *DB) lDecodeListKey(ek []byte) (key []byte, seq int32, err error) {
if len(ek) < 8 || ek[0] != db.index || ek[1] != ListType {
2014-05-06 09:32:38 +04:00
err = errListKey
return
}
2014-05-20 04:41:24 +04:00
keyLen := int(binary.BigEndian.Uint16(ek[2:]))
if keyLen+8 != len(ek) {
2014-05-06 09:32:38 +04:00
err = errListKey
return
}
2014-05-20 04:41:24 +04:00
key = ek[4 : 4+keyLen]
seq = int32(binary.BigEndian.Uint32(ek[4+keyLen:]))
2014-05-06 09:32:38 +04:00
return
}
2014-05-16 04:56:32 +04:00
func (db *DB) lpush(key []byte, whereSeq int32, args ...[]byte) (int64, error) {
2014-05-20 04:41:24 +04:00
if err := checkKeySize(key); err != nil {
return 0, err
}
var headSeq int32
var tailSeq int32
var size int32
var err error
2014-08-25 10:18:23 +04:00
t := db.listBatch
t.Lock()
defer t.Unlock()
2014-05-20 04:41:24 +04:00
metaKey := db.lEncodeMetaKey(key)
headSeq, tailSeq, size, err = db.lGetMeta(nil, metaKey)
if err != nil {
return 0, err
2014-05-16 04:56:32 +04:00
}
var pushCnt int = len(args)
if pushCnt == 0 {
return int64(size), nil
2014-05-06 09:32:38 +04:00
}
var seq int32 = headSeq
var delta int32 = -1
if whereSeq == listTailSeq {
2014-05-09 05:17:28 +04:00
seq = tailSeq
delta = 1
2014-05-06 09:32:38 +04:00
}
// append elements
if size > 0 {
2014-05-06 09:32:38 +04:00
seq += delta
}
for i := 0; i < pushCnt; i++ {
2014-05-20 04:41:24 +04:00
ek := db.lEncodeListKey(key, seq+int32(i)*delta)
t.Put(ek, args[i])
2014-05-06 09:32:38 +04:00
}
seq += int32(pushCnt-1) * delta
2014-05-06 09:32:38 +04:00
if seq <= listMinSeq || seq >= listMaxSeq {
return 0, errListSeq
}
// set meta info
2014-05-09 05:17:28 +04:00
if whereSeq == listHeadSeq {
headSeq = seq
} else {
tailSeq = seq
}
db.lSetMeta(metaKey, headSeq, tailSeq)
2014-05-06 09:32:38 +04:00
err = t.Commit()
2014-10-17 05:55:43 +04:00
if err == nil {
db.lSignalAsReady(key, pushCnt)
}
return int64(size) + int64(pushCnt), err
2014-05-06 09:32:38 +04:00
}
func (db *DB) lpop(key []byte, whereSeq int32) ([]byte, error) {
2014-05-20 04:41:24 +04:00
if err := checkKeySize(key); err != nil {
return nil, err
}
2014-08-25 10:18:23 +04:00
t := db.listBatch
2014-05-06 09:32:38 +04:00
t.Lock()
defer t.Unlock()
2014-05-20 04:41:24 +04:00
var headSeq int32
var tailSeq int32
var err error
metaKey := db.lEncodeMetaKey(key)
headSeq, tailSeq, _, err = db.lGetMeta(nil, metaKey)
2014-05-06 09:32:38 +04:00
if err != nil {
return nil, err
}
var value []byte
var seq int32 = headSeq
if whereSeq == listTailSeq {
2014-05-09 05:17:28 +04:00
seq = tailSeq
}
2014-05-20 04:41:24 +04:00
itemKey := db.lEncodeListKey(key, seq)
2014-08-25 10:18:23 +04:00
value, err = db.bucket.Get(itemKey)
2014-05-06 09:32:38 +04:00
if err != nil {
return nil, err
}
if whereSeq == listHeadSeq {
headSeq += 1
2014-05-06 09:32:38 +04:00
} else {
tailSeq -= 1
2014-05-06 09:32:38 +04:00
}
t.Delete(itemKey)
size := db.lSetMeta(metaKey, headSeq, tailSeq)
if size == 0 {
db.rmExpire(t, HashType, key)
}
2014-05-06 09:32:38 +04:00
err = t.Commit()
return value, err
}
// ps : here just focus on deleting the list data,
// any other likes expire is ignore.
2014-08-25 10:18:23 +04:00
func (db *DB) lDelete(t *batch, key []byte) int64 {
mk := db.lEncodeMetaKey(key)
var headSeq int32
var tailSeq int32
var err error
2014-08-25 10:18:23 +04:00
it := db.bucket.NewIterator()
defer it.Close()
headSeq, tailSeq, _, err = db.lGetMeta(it, mk)
if err != nil {
return 0
}
var num int64 = 0
startKey := db.lEncodeListKey(key, headSeq)
stopKey := db.lEncodeListKey(key, tailSeq)
2014-07-25 13:58:00 +04:00
rit := store.NewRangeIterator(it, &store.Range{startKey, stopKey, store.RangeClose})
for ; rit.Valid(); rit.Next() {
t.Delete(rit.RawKey())
num++
}
t.Delete(mk)
return num
}
2014-07-25 13:58:00 +04:00
func (db *DB) lGetMeta(it *store.Iterator, ek []byte) (headSeq int32, tailSeq int32, size int32, err error) {
var v []byte
if it != nil {
v = it.Find(ek)
} else {
2014-08-25 10:18:23 +04:00
v, err = db.bucket.Get(ek)
}
if err != nil {
return
} else if v == nil {
headSeq = listInitialSeq
tailSeq = listInitialSeq
size = 0
return
} else {
headSeq = int32(binary.LittleEndian.Uint32(v[0:4]))
tailSeq = int32(binary.LittleEndian.Uint32(v[4:8]))
size = tailSeq - headSeq + 1
}
return
}
func (db *DB) lSetMeta(ek []byte, headSeq int32, tailSeq int32) int32 {
2014-08-25 10:18:23 +04:00
t := db.listBatch
var size int32 = tailSeq - headSeq + 1
if size < 0 {
// todo : log error + panic
} else if size == 0 {
t.Delete(ek)
} else {
buf := make([]byte, 8)
binary.LittleEndian.PutUint32(buf[0:4], uint32(headSeq))
binary.LittleEndian.PutUint32(buf[4:8], uint32(tailSeq))
t.Put(ek, buf)
}
return size
}
func (db *DB) lExpireAt(key []byte, when int64) (int64, error) {
2014-08-25 10:18:23 +04:00
t := db.listBatch
t.Lock()
defer t.Unlock()
if llen, err := db.LLen(key); err != nil || llen == 0 {
return 0, err
} else {
db.expireAt(t, ListType, key, when)
if err := t.Commit(); err != nil {
return 0, err
}
}
return 1, nil
}
func (db *DB) LIndex(key []byte, index int32) ([]byte, error) {
2014-05-20 04:41:24 +04:00
if err := checkKeySize(key); err != nil {
return nil, err
}
var seq int32
2014-05-20 04:41:24 +04:00
var headSeq int32
var tailSeq int32
var err error
metaKey := db.lEncodeMetaKey(key)
2014-08-25 10:18:23 +04:00
it := db.bucket.NewIterator()
defer it.Close()
headSeq, tailSeq, _, err = db.lGetMeta(it, metaKey)
if err != nil {
return nil, err
}
if index >= 0 {
seq = headSeq + index
} else {
seq = tailSeq + index + 1
}
2014-05-20 04:41:24 +04:00
sk := db.lEncodeListKey(key, seq)
v := it.Find(sk)
return v, nil
}
func (db *DB) LLen(key []byte) (int64, error) {
2014-05-20 04:41:24 +04:00
if err := checkKeySize(key); err != nil {
return 0, err
}
ek := db.lEncodeMetaKey(key)
_, _, size, err := db.lGetMeta(nil, ek)
return int64(size), err
}
func (db *DB) LPop(key []byte) ([]byte, error) {
return db.lpop(key, listHeadSeq)
}
2014-05-16 04:56:32 +04:00
func (db *DB) LPush(key []byte, args ...[]byte) (int64, error) {
return db.lpush(key, listHeadSeq, args...)
}
func (db *DB) LRange(key []byte, start int32, stop int32) ([][]byte, error) {
2014-05-20 04:41:24 +04:00
if err := checkKeySize(key); err != nil {
return nil, err
}
var headSeq int32
var llen int32
2014-05-20 04:41:24 +04:00
var err error
metaKey := db.lEncodeMetaKey(key)
2014-08-25 10:18:23 +04:00
it := db.bucket.NewIterator()
defer it.Close()
if headSeq, _, llen, err = db.lGetMeta(it, metaKey); err != nil {
2014-05-09 05:17:28 +04:00
return nil, err
}
2014-05-06 09:32:38 +04:00
if start < 0 {
start = llen + start
}
if stop < 0 {
stop = llen + stop
}
if start < 0 {
start = 0
2014-05-06 09:32:38 +04:00
}
if start > stop || start >= llen {
return [][]byte{}, nil
2014-05-06 09:32:38 +04:00
}
if stop >= llen {
stop = llen - 1
}
limit := (stop - start) + 1
headSeq += start
v := make([][]byte, 0, limit)
startKey := db.lEncodeListKey(key, headSeq)
2014-07-25 13:58:00 +04:00
rit := store.NewRangeLimitIterator(it,
&store.Range{
Min: startKey,
Max: nil,
2014-07-25 13:58:00 +04:00
Type: store.RangeClose},
&store.Limit{
Offset: 0,
Count: int(limit)})
for ; rit.Valid(); rit.Next() {
v = append(v, rit.Value())
2014-05-06 09:32:38 +04:00
}
return v, nil
}
func (db *DB) RPop(key []byte) ([]byte, error) {
return db.lpop(key, listTailSeq)
}
2014-05-06 09:32:38 +04:00
2014-05-16 04:56:32 +04:00
func (db *DB) RPush(key []byte, args ...[]byte) (int64, error) {
return db.lpush(key, listTailSeq, args...)
2014-05-06 09:32:38 +04:00
}
2014-05-12 11:08:59 +04:00
func (db *DB) LClear(key []byte) (int64, error) {
2014-05-20 04:41:24 +04:00
if err := checkKeySize(key); err != nil {
return 0, err
}
2014-08-25 10:18:23 +04:00
t := db.listBatch
2014-05-12 11:08:59 +04:00
t.Lock()
defer t.Unlock()
num := db.lDelete(t, key)
db.rmExpire(t, ListType, key)
2014-05-12 11:08:59 +04:00
err := t.Commit()
2014-05-12 11:08:59 +04:00
return num, err
}
func (db *DB) LMclear(keys ...[]byte) (int64, error) {
2014-08-25 10:18:23 +04:00
t := db.listBatch
t.Lock()
defer t.Unlock()
for _, key := range keys {
if err := checkKeySize(key); err != nil {
return 0, err
}
db.lDelete(t, key)
db.rmExpire(t, ListType, key)
}
err := t.Commit()
return int64(len(keys)), err
}
func (db *DB) lFlush() (drop int64, err error) {
2014-08-25 10:18:23 +04:00
t := db.listBatch
2014-07-03 07:38:52 +04:00
t.Lock()
defer t.Unlock()
2014-08-16 11:35:05 +04:00
return db.flushType(t, ListType)
}
func (db *DB) LExpire(key []byte, duration int64) (int64, error) {
if duration <= 0 {
return 0, errExpireValue
}
return db.lExpireAt(key, time.Now().Unix()+duration)
}
func (db *DB) LExpireAt(key []byte, when int64) (int64, error) {
if when <= time.Now().Unix() {
return 0, errExpireValue
}
return db.lExpireAt(key, when)
}
func (db *DB) LTTL(key []byte) (int64, error) {
if err := checkKeySize(key); err != nil {
return -1, err
}
return db.ttl(ListType, key)
}
2014-06-23 07:12:20 +04:00
func (db *DB) LPersist(key []byte) (int64, error) {
if err := checkKeySize(key); err != nil {
return 0, err
}
2014-08-25 10:18:23 +04:00
t := db.listBatch
2014-06-23 07:12:20 +04:00
t.Lock()
defer t.Unlock()
n, err := db.rmExpire(t, ListType, key)
2014-06-23 07:12:20 +04:00
if err != nil {
return 0, err
}
err = t.Commit()
return n, err
}
2014-08-14 11:24:41 +04:00
func (db *DB) lEncodeMinKey() []byte {
return db.lEncodeMetaKey(nil)
}
func (db *DB) lEncodeMaxKey() []byte {
ek := db.lEncodeMetaKey(nil)
ek[len(ek)-1] = LMetaType + 1
return ek
}
2014-10-16 13:51:52 +04:00
func (db *DB) BLPop(keys [][]byte, timeout time.Duration) ([]interface{}, error) {
2014-10-16 13:51:52 +04:00
return db.lblockPop(keys, listHeadSeq, timeout)
}
func (db *DB) BRPop(keys [][]byte, timeout time.Duration) ([]interface{}, error) {
2014-10-16 13:51:52 +04:00
return db.lblockPop(keys, listTailSeq, timeout)
}
func (db *DB) LKeyExists(key []byte) (int64, error) {
2015-02-01 12:41:28 +03:00
if err := checkKeySize(key); err != nil {
return 0, err
}
sk := db.lEncodeMetaKey(key)
v, err := db.bucket.Get(sk)
if v != nil && err == nil {
return 1, nil
}
return 0, err
}
func (db *DB) lblockPop(keys [][]byte, whereSeq int32, timeout time.Duration) ([]interface{}, error) {
2014-10-16 13:51:52 +04:00
ch := make(chan []byte)
bkeys := [][]byte{}
for _, key := range keys {
v, err := db.lpop(key, whereSeq)
if err != nil {
return nil, err
} else if v != nil {
return []interface{}{key, v}, nil
} else {
if db.IsAutoCommit() {
//block wait can not be supported in transaction and multi
db.lbkeys.wait(key, ch)
bkeys = append(bkeys, key)
}
}
}
if len(bkeys) == 0 {
return nil, nil
}
defer func() {
for _, key := range bkeys {
db.lbkeys.unwait(key, ch)
}
}()
deadT := time.Now().Add(timeout)
2014-10-16 13:51:52 +04:00
for {
if timeout == 0 {
key := <-ch
if v, err := db.lpop(key, whereSeq); err != nil {
return nil, err
} else if v == nil {
continue
} else {
return []interface{}{key, v}, nil
}
} else {
d := deadT.Sub(time.Now())
if d < 0 {
return nil, nil
}
select {
case key := <-ch:
if v, err := db.lpop(key, whereSeq); err != nil {
return nil, err
} else if v == nil {
2014-10-17 05:55:43 +04:00
db.lbkeys.wait(key, ch)
2014-10-16 13:51:52 +04:00
continue
} else {
return []interface{}{key, v}, nil
}
case <-time.After(d):
return nil, nil
}
}
}
}
2014-10-17 05:55:43 +04:00
func (db *DB) lSignalAsReady(key []byte, num int) {
2014-10-16 13:51:52 +04:00
if db.status == DBInTransaction {
//for transaction, only data can be pushed after tx commit and it is hard to signal
//so we don't handle it now
return
}
2014-10-17 05:55:43 +04:00
db.lbkeys.signal(key, num)
2014-10-16 13:51:52 +04:00
}
type lbKeyCh chan<- []byte
type lBlockKeys struct {
sync.Mutex
2014-10-17 05:55:43 +04:00
keys map[string]*list.List
2014-10-16 13:51:52 +04:00
}
func newLBlockKeys() *lBlockKeys {
l := new(lBlockKeys)
2014-10-17 05:55:43 +04:00
l.keys = make(map[string]*list.List)
2014-10-16 13:51:52 +04:00
return l
}
2014-10-17 05:55:43 +04:00
func (l *lBlockKeys) signal(key []byte, num int) {
2014-10-16 13:51:52 +04:00
l.Lock()
defer l.Unlock()
s := hack.String(key)
chs, ok := l.keys[s]
if !ok {
return
}
2014-10-17 05:55:43 +04:00
var n *list.Element
2014-10-16 13:51:52 +04:00
2014-10-17 05:55:43 +04:00
i := 0
for e := chs.Front(); e != nil && i < num; e = n {
ch := e.Value.(lbKeyCh)
n = e.Next()
2014-10-16 13:51:52 +04:00
select {
case ch <- key:
2014-10-17 05:55:43 +04:00
chs.Remove(e)
i++
2014-10-16 13:51:52 +04:00
default:
//waiter unwait
2014-10-17 05:55:43 +04:00
chs.Remove(e)
2014-10-16 13:51:52 +04:00
}
}
2014-10-17 05:55:43 +04:00
if chs.Len() == 0 {
2014-10-16 13:51:52 +04:00
delete(l.keys, s)
}
}
func (l *lBlockKeys) wait(key []byte, ch lbKeyCh) {
l.Lock()
defer l.Unlock()
s := hack.String(key)
chs, ok := l.keys[s]
if !ok {
2014-10-17 05:55:43 +04:00
chs = list.New()
2014-10-16 13:51:52 +04:00
l.keys[s] = chs
}
2014-10-17 05:55:43 +04:00
chs.PushBack(ch)
2014-10-16 13:51:52 +04:00
}
func (l *lBlockKeys) unwait(key []byte, ch lbKeyCh) {
l.Lock()
defer l.Unlock()
s := hack.String(key)
chs, ok := l.keys[s]
if !ok {
return
} else {
2014-10-17 05:55:43 +04:00
var n *list.Element
for e := chs.Front(); e != nil; e = n {
c := e.Value.(lbKeyCh)
n = e.Next()
if c == ch {
chs.Remove(e)
}
2014-10-16 13:51:52 +04:00
}
2014-10-17 05:55:43 +04:00
if chs.Len() == 0 {
delete(l.keys, s)
2014-10-16 13:51:52 +04:00
}
}
}