74 lines
		
	
	
		
			1.3 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			74 lines
		
	
	
		
			1.3 KiB
		
	
	
	
		
			Go
		
	
	
	
| package nodb
 | |
| 
 | |
| 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()
 | |
| 	m.DB.binBatch = m.newBatch()
 | |
| 	m.DB.setBatch = m.newBatch()
 | |
| 
 | |
| 	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 {
 | |
| 	if index < 0 || index >= int(MaxDBNumber) {
 | |
| 		return fmt.Errorf("invalid db index %d", index)
 | |
| 	}
 | |
| 
 | |
| 	m.DB.index = uint8(index)
 | |
| 	return nil
 | |
| }
 |