mirror of https://github.com/etcd-io/bbolt.git
commit
6b378d844e
8
Makefile
8
Makefile
|
@ -7,6 +7,14 @@ default: build
|
||||||
race:
|
race:
|
||||||
@go test -v -race -test.run="TestSimulate_(100op|1000op)"
|
@go test -v -race -test.run="TestSimulate_(100op|1000op)"
|
||||||
|
|
||||||
|
# go get honnef.co/go/tools/simple
|
||||||
|
# go get honnef.co/go/tools/unused
|
||||||
|
fmt:
|
||||||
|
gosimple ./...
|
||||||
|
unused ./...
|
||||||
|
gofmt -l -s -d $(find -name \*.go)
|
||||||
|
|
||||||
|
|
||||||
# go get github.com/kisielk/errcheck
|
# go get github.com/kisielk/errcheck
|
||||||
errcheck:
|
errcheck:
|
||||||
@errcheck -ignorepkg=bytes -ignore=os:Remove github.com/coreos/bbolt
|
@errcheck -ignorepkg=bytes -ignore=os:Remove github.com/coreos/bbolt
|
||||||
|
|
|
@ -14,13 +14,6 @@ const (
|
||||||
MaxValueSize = (1 << 31) - 2
|
MaxValueSize = (1 << 31) - 2
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
|
||||||
maxUint = ^uint(0)
|
|
||||||
minUint = 0
|
|
||||||
maxInt = int(^uint(0) >> 1)
|
|
||||||
minInt = -maxInt - 1
|
|
||||||
)
|
|
||||||
|
|
||||||
const bucketHeaderSize = int(unsafe.Sizeof(bucket{}))
|
const bucketHeaderSize = int(unsafe.Sizeof(bucket{}))
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
|
|
@ -48,12 +48,6 @@ var (
|
||||||
|
|
||||||
// ErrPageIDRequired is returned when a required page id is not specified.
|
// ErrPageIDRequired is returned when a required page id is not specified.
|
||||||
ErrPageIDRequired = errors.New("page id required")
|
ErrPageIDRequired = errors.New("page id required")
|
||||||
|
|
||||||
// ErrPageNotFound is returned when specifying a page above the high water mark.
|
|
||||||
ErrPageNotFound = errors.New("page not found")
|
|
||||||
|
|
||||||
// ErrPageFreed is returned when reading a page that has already been freed.
|
|
||||||
ErrPageFreed = errors.New("page freed")
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// PageHeaderSize represents the size of the bolt.page header.
|
// PageHeaderSize represents the size of the bolt.page header.
|
||||||
|
@ -188,17 +182,9 @@ func (cmd *CheckCommand) Run(args ...string) error {
|
||||||
// Perform consistency check.
|
// Perform consistency check.
|
||||||
return db.View(func(tx *bolt.Tx) error {
|
return db.View(func(tx *bolt.Tx) error {
|
||||||
var count int
|
var count int
|
||||||
ch := tx.Check()
|
for err := range tx.Check() {
|
||||||
loop:
|
fmt.Fprintln(cmd.Stdout, err)
|
||||||
for {
|
count++
|
||||||
select {
|
|
||||||
case err, ok := <-ch:
|
|
||||||
if !ok {
|
|
||||||
break loop
|
|
||||||
}
|
|
||||||
fmt.Fprintln(cmd.Stdout, err)
|
|
||||||
count++
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Print summary of errors.
|
// Print summary of errors.
|
||||||
|
@ -1031,12 +1017,12 @@ func (cmd *BenchCommand) runWritesRandom(db *bolt.DB, options *BenchOptions, res
|
||||||
|
|
||||||
func (cmd *BenchCommand) runWritesSequentialNested(db *bolt.DB, options *BenchOptions, results *BenchResults) error {
|
func (cmd *BenchCommand) runWritesSequentialNested(db *bolt.DB, options *BenchOptions, results *BenchResults) error {
|
||||||
var i = uint32(0)
|
var i = uint32(0)
|
||||||
return cmd.runWritesWithSource(db, options, results, func() uint32 { i++; return i })
|
return cmd.runWritesNestedWithSource(db, options, results, func() uint32 { i++; return i })
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cmd *BenchCommand) runWritesRandomNested(db *bolt.DB, options *BenchOptions, results *BenchResults) error {
|
func (cmd *BenchCommand) runWritesRandomNested(db *bolt.DB, options *BenchOptions, results *BenchResults) error {
|
||||||
r := rand.New(rand.NewSource(time.Now().UnixNano()))
|
r := rand.New(rand.NewSource(time.Now().UnixNano()))
|
||||||
return cmd.runWritesWithSource(db, options, results, func() uint32 { return r.Uint32() })
|
return cmd.runWritesNestedWithSource(db, options, results, func() uint32 { return r.Uint32() })
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cmd *BenchCommand) runWritesWithSource(db *bolt.DB, options *BenchOptions, results *BenchResults, keySource func() uint32) error {
|
func (cmd *BenchCommand) runWritesWithSource(db *bolt.DB, options *BenchOptions, results *BenchResults, keySource func() uint32) error {
|
||||||
|
|
15
db.go
15
db.go
|
@ -7,9 +7,7 @@ import (
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"runtime"
|
"runtime"
|
||||||
"runtime/debug"
|
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
|
@ -193,6 +191,7 @@ func Open(path string, mode os.FileMode, options *Options) (*DB, error) {
|
||||||
// The database file is locked using the shared lock (more than one process may
|
// The database file is locked using the shared lock (more than one process may
|
||||||
// hold a lock at the same time) otherwise (options.ReadOnly is set).
|
// hold a lock at the same time) otherwise (options.ReadOnly is set).
|
||||||
if err := flock(db, mode, !db.readOnly, options.Timeout); err != nil {
|
if err := flock(db, mode, !db.readOnly, options.Timeout); err != nil {
|
||||||
|
db.lockfile = nil // make 'unused' happy. TODO: rework locks
|
||||||
_ = db.close()
|
_ = db.close()
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -1040,10 +1039,6 @@ func (s *Stats) Sub(other *Stats) Stats {
|
||||||
return diff
|
return diff
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Stats) add(other *Stats) {
|
|
||||||
s.TxStats.add(&other.TxStats)
|
|
||||||
}
|
|
||||||
|
|
||||||
type Info struct {
|
type Info struct {
|
||||||
Data uintptr
|
Data uintptr
|
||||||
PageSize int
|
PageSize int
|
||||||
|
@ -1110,11 +1105,3 @@ func _assert(condition bool, msg string, v ...interface{}) {
|
||||||
panic(fmt.Sprintf("assertion failed: "+msg, v...))
|
panic(fmt.Sprintf("assertion failed: "+msg, v...))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func warn(v ...interface{}) { fmt.Fprintln(os.Stderr, v...) }
|
|
||||||
func warnf(msg string, v ...interface{}) { fmt.Fprintf(os.Stderr, msg+"\n", v...) }
|
|
||||||
|
|
||||||
func printstack() {
|
|
||||||
stack := strings.Join(strings.Split(string(debug.Stack()), "\n")[2:], "\n")
|
|
||||||
fmt.Fprintln(os.Stderr, stack)
|
|
||||||
}
|
|
||||||
|
|
48
db_test.go
48
db_test.go
|
@ -12,8 +12,6 @@ import (
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"regexp"
|
"regexp"
|
||||||
"sort"
|
|
||||||
"strings"
|
|
||||||
"sync"
|
"sync"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
@ -24,12 +22,6 @@ import (
|
||||||
|
|
||||||
var statsFlag = flag.Bool("stats", false, "show performance stats")
|
var statsFlag = flag.Bool("stats", false, "show performance stats")
|
||||||
|
|
||||||
// version is the data file format version.
|
|
||||||
const version = 2
|
|
||||||
|
|
||||||
// magic is the marker value to indicate that a file is a Bolt DB.
|
|
||||||
const magic uint32 = 0xED0CDAED
|
|
||||||
|
|
||||||
// pageSize is the size of one page in the data file.
|
// pageSize is the size of one page in the data file.
|
||||||
const pageSize = 4096
|
const pageSize = 4096
|
||||||
|
|
||||||
|
@ -1575,40 +1567,6 @@ func tempfile() string {
|
||||||
return f.Name()
|
return f.Name()
|
||||||
}
|
}
|
||||||
|
|
||||||
// mustContainKeys checks that a bucket contains a given set of keys.
|
|
||||||
func mustContainKeys(b *bolt.Bucket, m map[string]string) {
|
|
||||||
found := make(map[string]string)
|
|
||||||
if err := b.ForEach(func(k, _ []byte) error {
|
|
||||||
found[string(k)] = ""
|
|
||||||
return nil
|
|
||||||
}); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check for keys found in bucket that shouldn't be there.
|
|
||||||
var keys []string
|
|
||||||
for k := range found {
|
|
||||||
if _, ok := m[string(k)]; !ok {
|
|
||||||
keys = append(keys, k)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if len(keys) > 0 {
|
|
||||||
sort.Strings(keys)
|
|
||||||
panic(fmt.Sprintf("keys found(%d): %s", len(keys), strings.Join(keys, ",")))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check for keys not found in bucket that should be there.
|
|
||||||
for k := range m {
|
|
||||||
if _, ok := found[string(k)]; !ok {
|
|
||||||
keys = append(keys, k)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if len(keys) > 0 {
|
|
||||||
sort.Strings(keys)
|
|
||||||
panic(fmt.Sprintf("keys not found(%d): %s", len(keys), strings.Join(keys, ",")))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func trunc(b []byte, length int) []byte {
|
func trunc(b []byte, length int) []byte {
|
||||||
if length < len(b) {
|
if length < len(b) {
|
||||||
return b[:length]
|
return b[:length]
|
||||||
|
@ -1628,15 +1586,9 @@ func fileSize(path string) int64 {
|
||||||
return fi.Size()
|
return fi.Size()
|
||||||
}
|
}
|
||||||
|
|
||||||
func warn(v ...interface{}) { fmt.Fprintln(os.Stderr, v...) }
|
|
||||||
func warnf(msg string, v ...interface{}) { fmt.Fprintf(os.Stderr, msg+"\n", v...) }
|
|
||||||
|
|
||||||
// u64tob converts a uint64 into an 8-byte slice.
|
// u64tob converts a uint64 into an 8-byte slice.
|
||||||
func u64tob(v uint64) []byte {
|
func u64tob(v uint64) []byte {
|
||||||
b := make([]byte, 8)
|
b := make([]byte, 8)
|
||||||
binary.BigEndian.PutUint64(b, v)
|
binary.BigEndian.PutUint64(b, v)
|
||||||
return b
|
return b
|
||||||
}
|
}
|
||||||
|
|
||||||
// btou64 converts an 8-byte slice into an uint64.
|
|
||||||
func btou64(b []byte) uint64 { return binary.BigEndian.Uint64(b) }
|
|
||||||
|
|
|
@ -245,7 +245,7 @@ func (f *freelist) read(p *page) {
|
||||||
if count == 0 {
|
if count == 0 {
|
||||||
f.ids = nil
|
f.ids = nil
|
||||||
} else {
|
} else {
|
||||||
ids := ((*[maxAllocSize]pgid)(unsafe.Pointer(&p.ptr)))[idx:idx+count]
|
ids := ((*[maxAllocSize]pgid)(unsafe.Pointer(&p.ptr)))[idx : idx+count]
|
||||||
f.ids = make([]pgid, len(ids))
|
f.ids = make([]pgid, len(ids))
|
||||||
copy(f.ids, ids)
|
copy(f.ids, ids)
|
||||||
|
|
||||||
|
|
5
tx.go
5
tx.go
|
@ -126,10 +126,7 @@ func (tx *Tx) DeleteBucket(name []byte) error {
|
||||||
// the error is returned to the caller.
|
// the error is returned to the caller.
|
||||||
func (tx *Tx) ForEach(fn func(name []byte, b *Bucket) error) error {
|
func (tx *Tx) ForEach(fn func(name []byte, b *Bucket) error) error {
|
||||||
return tx.root.ForEach(func(k, v []byte) error {
|
return tx.root.ForEach(func(k, v []byte) error {
|
||||||
if err := fn(k, tx.root.Bucket(k)); err != nil {
|
return fn(k, tx.root.Bucket(k))
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue