221 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			221 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package ledis
 | 
						|
 | 
						|
import (
 | 
						|
	"bufio"
 | 
						|
	"bytes"
 | 
						|
	"encoding/binary"
 | 
						|
	"io"
 | 
						|
	"os"
 | 
						|
 | 
						|
	"github.com/siddontang/go/snappy"
 | 
						|
	"github.com/siddontang/ledisdb/store"
 | 
						|
)
 | 
						|
 | 
						|
// DumpHead is the head of a dump.
 | 
						|
type DumpHead struct {
 | 
						|
	CommitID uint64
 | 
						|
}
 | 
						|
 | 
						|
// Read reads meta from the Reader.
 | 
						|
func (h *DumpHead) Read(r io.Reader) error {
 | 
						|
	return binary.Read(r, binary.BigEndian, &h.CommitID)
 | 
						|
}
 | 
						|
 | 
						|
// Write writes meta to the Writer
 | 
						|
func (h *DumpHead) Write(w io.Writer) error {
 | 
						|
	return binary.Write(w, binary.BigEndian, h.CommitID)
 | 
						|
}
 | 
						|
 | 
						|
// DumpFile dumps data to the file
 | 
						|
func (l *Ledis) DumpFile(path string) error {
 | 
						|
	f, err := os.Create(path)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	defer f.Close()
 | 
						|
 | 
						|
	return l.Dump(f)
 | 
						|
}
 | 
						|
 | 
						|
// Dump dumps data to the Writer.
 | 
						|
func (l *Ledis) Dump(w io.Writer) error {
 | 
						|
	var err error
 | 
						|
 | 
						|
	var commitID uint64
 | 
						|
	var snap *store.Snapshot
 | 
						|
 | 
						|
	l.wLock.Lock()
 | 
						|
 | 
						|
	if l.r != nil {
 | 
						|
		if commitID, err = l.r.LastCommitID(); err != nil {
 | 
						|
			l.wLock.Unlock()
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if snap, err = l.ldb.NewSnapshot(); err != nil {
 | 
						|
		l.wLock.Unlock()
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	defer snap.Close()
 | 
						|
 | 
						|
	l.wLock.Unlock()
 | 
						|
 | 
						|
	wb := bufio.NewWriterSize(w, 4096)
 | 
						|
 | 
						|
	h := &DumpHead{commitID}
 | 
						|
 | 
						|
	if err = h.Write(wb); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	it := snap.NewIterator()
 | 
						|
	defer it.Close()
 | 
						|
	it.SeekToFirst()
 | 
						|
 | 
						|
	compressBuf := make([]byte, 4096)
 | 
						|
 | 
						|
	var key []byte
 | 
						|
	var value []byte
 | 
						|
	for ; it.Valid(); it.Next() {
 | 
						|
		key = it.RawKey()
 | 
						|
		value = it.RawValue()
 | 
						|
 | 
						|
		if key, err = snappy.Encode(compressBuf, key); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		if err = binary.Write(wb, binary.BigEndian, uint16(len(key))); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		if _, err = wb.Write(key); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		if value, err = snappy.Encode(compressBuf, value); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		if err = binary.Write(wb, binary.BigEndian, uint32(len(value))); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		if _, err = wb.Write(value); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if err = wb.Flush(); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	compressBuf = nil
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// LoadDumpFile clears all data and loads dump file to db
 | 
						|
func (l *Ledis) LoadDumpFile(path string) (*DumpHead, error) {
 | 
						|
	f, err := os.Open(path)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	defer f.Close()
 | 
						|
 | 
						|
	return l.LoadDump(f)
 | 
						|
}
 | 
						|
 | 
						|
// LoadDump clears all data and loads dump file to db
 | 
						|
func (l *Ledis) LoadDump(r io.Reader) (*DumpHead, error) {
 | 
						|
	l.wLock.Lock()
 | 
						|
	defer l.wLock.Unlock()
 | 
						|
 | 
						|
	var err error
 | 
						|
	if err = l.flushAll(); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	rb := bufio.NewReaderSize(r, 4096)
 | 
						|
 | 
						|
	h := new(DumpHead)
 | 
						|
 | 
						|
	if err = h.Read(rb); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	var keyLen uint16
 | 
						|
	var valueLen uint32
 | 
						|
 | 
						|
	var keyBuf bytes.Buffer
 | 
						|
	var valueBuf bytes.Buffer
 | 
						|
 | 
						|
	deKeyBuf := make([]byte, 4096)
 | 
						|
	deValueBuf := make([]byte, 4096)
 | 
						|
 | 
						|
	var key, value []byte
 | 
						|
 | 
						|
	wb := l.ldb.NewWriteBatch()
 | 
						|
	defer wb.Close()
 | 
						|
 | 
						|
	n := 0
 | 
						|
 | 
						|
	for {
 | 
						|
		if err = binary.Read(rb, binary.BigEndian, &keyLen); err != nil && err != io.EOF {
 | 
						|
			return nil, err
 | 
						|
		} else if err == io.EOF {
 | 
						|
			break
 | 
						|
		}
 | 
						|
 | 
						|
		if _, err = io.CopyN(&keyBuf, rb, int64(keyLen)); err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
 | 
						|
		if key, err = snappy.Decode(deKeyBuf, keyBuf.Bytes()); err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
 | 
						|
		if err = binary.Read(rb, binary.BigEndian, &valueLen); err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
 | 
						|
		if _, err = io.CopyN(&valueBuf, rb, int64(valueLen)); err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
 | 
						|
		if value, err = snappy.Decode(deValueBuf, valueBuf.Bytes()); err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
 | 
						|
		wb.Put(key, value)
 | 
						|
		n++
 | 
						|
		if n%1024 == 0 {
 | 
						|
			if err = wb.Commit(); err != nil {
 | 
						|
				return nil, err
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		// if err = l.ldb.Put(key, value); err != nil {
 | 
						|
		// 	return nil, err
 | 
						|
		// }
 | 
						|
 | 
						|
		keyBuf.Reset()
 | 
						|
		valueBuf.Reset()
 | 
						|
	}
 | 
						|
 | 
						|
	if err = wb.Commit(); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	deKeyBuf = nil
 | 
						|
	deValueBuf = nil
 | 
						|
 | 
						|
	if l.r != nil {
 | 
						|
		if err := l.r.UpdateCommitID(h.CommitID); err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return h, nil
 | 
						|
}
 |