ledisdb/ledis/multi.go

76 lines
1.3 KiB
Go
Raw Permalink Normal View History

2014-09-02 13:55:12 +04:00
package ledis
import (
"errors"
"fmt"
)
var (
ErrNestMulti = errors.New("nest multi not supported")
ErrMultiDone = errors.New("multi has been closed")
)
type Multi struct {
*DB
}
func (db *DB) IsInMulti() bool {
return db.status == DBInMulti
}
// begin a mutli to execute commands,
// it will block any other write operations before you close the multi, unlike transaction, mutli can not rollback
func (db *DB) Multi() (*Multi, error) {
if db.IsInMulti() {
return nil, ErrNestMulti
}
m := new(Multi)
m.DB = new(DB)
m.DB.status = DBInMulti
m.DB.l = db.l
m.l.wLock.Lock()
m.DB.sdb = db.sdb
m.DB.bucket = db.sdb
m.DB.index = db.index
m.DB.kvBatch = m.newBatch()
m.DB.listBatch = m.newBatch()
m.DB.hashBatch = m.newBatch()
m.DB.zsetBatch = m.newBatch()
2015-03-03 04:21:00 +03:00
// m.DB.binBatch = m.newBatch()
2014-09-02 13:55:12 +04:00
m.DB.setBatch = m.newBatch()
2014-10-16 13:51:52 +04:00
m.DB.lbkeys = db.lbkeys
2014-09-02 13:55:12 +04:00
return m, nil
}
func (m *Multi) newBatch() *batch {
return m.l.newBatch(m.bucket.NewWriteBatch(), &multiBatchLocker{}, nil)
}
func (m *Multi) Close() error {
if m.bucket == nil {
return ErrMultiDone
}
m.l.wLock.Unlock()
m.bucket = nil
return nil
}
func (m *Multi) Select(index int) error {
2015-03-04 04:15:28 +03:00
if index < 0 || index >= int(m.l.cfg.Databases) {
2014-09-02 13:55:12 +04:00
return fmt.Errorf("invalid db index %d", index)
}
m.DB.index = uint8(index)
return nil
}