bolt/db_test.go

197 lines
6.8 KiB
Go
Raw Normal View History

2014-01-12 05:51:01 +00:00
package bolt
import (
2014-01-13 15:25:56 +00:00
"io"
2014-01-12 05:51:01 +00:00
"io/ioutil"
"os"
2014-01-13 15:25:56 +00:00
"syscall"
2014-01-12 05:51:01 +00:00
"testing"
2014-01-13 15:25:56 +00:00
"time"
"unsafe"
2014-01-12 05:51:01 +00:00
"github.com/stretchr/testify/assert"
2014-01-13 15:25:56 +00:00
"github.com/stretchr/testify/mock"
2014-01-12 05:51:01 +00:00
)
2014-01-12 22:30:09 +00:00
// Ensure that a database can be opened without error.
2014-01-12 05:51:01 +00:00
func TestDBOpen(t *testing.T) {
withDB(func(db *DB, path string) {
err := db.Open(path, 0666)
assert.NoError(t, err)
2014-01-12 22:30:09 +00:00
assert.Equal(t, db.Path(), path)
2014-01-12 05:51:01 +00:00
})
}
2014-01-12 22:30:09 +00:00
// Ensure that the database returns an error if already open.
func TestDBReopen(t *testing.T) {
withDB(func(db *DB, path string) {
db.Open(path, 0666)
err := db.Open(path, 0666)
assert.Equal(t, err, DatabaseAlreadyOpenedError)
})
}
// Ensure that the database returns an error if the file handle cannot be open.
func TestDBOpenFileError(t *testing.T) {
2014-01-13 15:25:56 +00:00
withMockDB(func(db *DB, mockos *mockos, mocksyscall *mocksyscall, path string) {
2014-01-12 22:30:09 +00:00
exp := &os.PathError{}
2014-01-13 15:25:56 +00:00
mockos.On("OpenFile", path, os.O_RDWR|os.O_CREATE, os.FileMode(0666)).Return((*mockfile)(nil), exp)
2014-01-12 22:30:09 +00:00
err := db.Open(path, 0666)
assert.Equal(t, err, exp)
})
}
// Ensure that the database returns an error if the meta file handle cannot be open.
func TestDBOpenMetaFileError(t *testing.T) {
2014-01-13 15:25:56 +00:00
withMockDB(func(db *DB, mockos *mockos, mocksyscall *mocksyscall, path string) {
2014-01-12 22:30:09 +00:00
exp := &os.PathError{}
2014-01-13 15:25:56 +00:00
mockos.On("OpenFile", path, os.O_RDWR|os.O_CREATE, os.FileMode(0666)).Return(&mockfile{}, nil)
mockos.On("OpenFile", path, os.O_RDWR|os.O_SYNC, os.FileMode(0666)).Return((*mockfile)(nil), exp)
2014-01-12 22:30:09 +00:00
err := db.Open(path, 0666)
assert.Equal(t, err, exp)
})
}
2014-01-13 15:25:56 +00:00
// Ensure that write errors to the meta file handler during initialization are returned.
func TestDBMetaInitWriteError(t *testing.T) {
withMockDB(func(db *DB, mockos *mockos, mocksyscall *mocksyscall, path string) {
2014-01-28 19:52:09 +00:00
// Mock the file system.
2014-01-13 15:25:56 +00:00
file, metafile := &mockfile{}, &mockfile{}
mockos.On("OpenFile", path, os.O_RDWR|os.O_CREATE, os.FileMode(0666)).Return(file, nil)
mockos.On("OpenFile", path, os.O_RDWR|os.O_SYNC, os.FileMode(0666)).Return(metafile, nil)
mockos.On("Getpagesize").Return(0x10000)
2014-01-28 19:52:09 +00:00
file.On("Stat").Return(&mockfileinfo{"", 0, 0666, time.Now(), false, nil}, nil)
2014-01-13 15:25:56 +00:00
metafile.On("WriteAt", mock.Anything, int64(0)).Return(0, io.ErrShortWrite)
2014-01-28 19:52:09 +00:00
// Open the database.
2014-01-13 15:25:56 +00:00
err := db.Open(path, 0666)
assert.Equal(t, err, io.ErrShortWrite)
})
}
// Ensure that a database that is too small returns an error.
func TestDBFileTooSmall(t *testing.T) {
withMockDB(func(db *DB, mockos *mockos, mocksyscall *mocksyscall, path string) {
file, metafile := &mockfile{}, &mockfile{}
mockos.On("OpenFile", path, os.O_RDWR|os.O_CREATE, os.FileMode(0666)).Return(file, nil)
mockos.On("OpenFile", path, os.O_RDWR|os.O_SYNC, os.FileMode(0666)).Return(metafile, nil)
mockos.On("Getpagesize").Return(0x1000)
2014-01-28 19:52:09 +00:00
file.On("Stat").Return(&mockfileinfo{"", 0, 0666, time.Now(), false, nil}, nil)
2014-01-13 15:25:56 +00:00
metafile.On("WriteAt", mock.Anything, int64(0)).Return(0, nil)
err := db.Open(path, 0666)
assert.Equal(t, err, &Error{"file size too small", nil})
})
}
// Ensure that stat errors during mmap get returned.
func TestDBMmapStatError(t *testing.T) {
withMockDB(func(db *DB, mockos *mockos, mocksyscall *mocksyscall, path string) {
exp := &os.PathError{}
file, metafile := &mockfile{}, &mockfile{}
mockos.On("OpenFile", path, os.O_RDWR|os.O_CREATE, os.FileMode(0666)).Return(file, nil)
mockos.On("OpenFile", path, os.O_RDWR|os.O_SYNC, os.FileMode(0666)).Return(metafile, nil)
mockos.On("Getpagesize").Return(0x1000)
file.On("ReadAt", mock.Anything, int64(0)).Return(0, nil)
file.On("Stat").Return((*mockfileinfo)(nil), exp)
metafile.On("WriteAt", mock.Anything, int64(0)).Return(0, nil)
2014-01-12 22:50:35 +00:00
err := db.Open(path, 0666)
2014-01-28 19:52:09 +00:00
assert.Equal(t, err, &Error{"stat error", exp})
2014-01-12 22:50:35 +00:00
})
}
2014-01-13 15:25:56 +00:00
// Ensure that mmap errors get returned.
2014-01-28 19:52:09 +00:00
/*
2014-01-13 15:25:56 +00:00
func TestDBMmapError(t *testing.T) {
withMockDB(func(db *DB, mockos *mockos, mocksyscall *mocksyscall, path string) {
exp := errors.New("")
file, metafile := &mockfile{}, &mockfile{}
mockos.On("OpenFile", path, os.O_RDWR|os.O_CREATE, os.FileMode(0666)).Return(file, nil)
mockos.On("OpenFile", path, os.O_RDWR|os.O_SYNC, os.FileMode(0666)).Return(metafile, nil)
mockos.On("Getpagesize").Return(0x1000)
file.On("ReadAt", mock.Anything, int64(0)).Return(0, nil)
file.On("Stat").Return(&mockfileinfo{"", 0x2000, 0666, time.Now(), false, nil}, nil)
metafile.On("WriteAt", mock.Anything, int64(0)).Return(0, nil)
mocksyscall.On("Mmap", 0, int64(0), 0x2000, syscall.PROT_READ, syscall.MAP_SHARED).Return(([]byte)(nil), exp)
err := db.Open(path, 0666)
assert.Equal(t, err, exp)
})
}
2014-01-28 19:52:09 +00:00
*/
2014-01-13 15:25:56 +00:00
// Ensure that corrupt meta0 page errors get returned.
func TestDBCorruptMeta0(t *testing.T) {
withMockDB(func(db *DB, mockos *mockos, mocksyscall *mocksyscall, path string) {
2014-01-30 03:35:58 +00:00
var m meta
m.magic = magic
2014-01-31 17:22:58 +00:00
m.version = version
2014-01-30 03:35:58 +00:00
m.pageSize = 0x8000
2014-01-28 19:52:09 +00:00
// Create a file with bad magic.
2014-01-13 15:25:56 +00:00
b := make([]byte, 0x10000)
p0, p1 := (*page)(unsafe.Pointer(&b[0x0000])), (*page)(unsafe.Pointer(&b[0x8000]))
2014-01-30 03:35:58 +00:00
p0.meta().magic = 0
2014-01-31 17:22:58 +00:00
p0.meta().version = version
2014-01-30 03:35:58 +00:00
p1.meta().magic = magic
2014-01-31 17:22:58 +00:00
p1.meta().version = version
2014-01-13 15:25:56 +00:00
2014-01-28 19:52:09 +00:00
// Mock file access.
2014-01-13 15:25:56 +00:00
file, metafile := &mockfile{}, &mockfile{}
mockos.On("OpenFile", path, os.O_RDWR|os.O_CREATE, os.FileMode(0666)).Return(file, nil)
mockos.On("OpenFile", path, os.O_RDWR|os.O_SYNC, os.FileMode(0666)).Return(metafile, nil)
mockos.On("Getpagesize").Return(0x10000)
file.On("ReadAt", mock.Anything, int64(0)).Return(0, nil)
file.On("Stat").Return(&mockfileinfo{"", 0x10000, 0666, time.Now(), false, nil}, nil)
metafile.On("WriteAt", mock.Anything, int64(0)).Return(0, nil)
mocksyscall.On("Mmap", 0, int64(0), 0x10000, syscall.PROT_READ, syscall.MAP_SHARED).Return(b, nil)
2014-01-28 19:52:09 +00:00
// Open the database.
2014-01-13 15:25:56 +00:00
err := db.Open(path, 0666)
2014-01-30 03:35:58 +00:00
assert.Equal(t, err, &Error{"meta error", InvalidError})
2014-01-13 15:25:56 +00:00
})
}
2014-01-12 22:50:35 +00:00
2014-01-13 17:35:04 +00:00
//--------------------------------------
// Transaction()
//--------------------------------------
// Ensure that a database cannot open a transaction when it's not open.
func TestDBTransactionDatabaseNotOpenError(t *testing.T) {
withDB(func(db *DB, path string) {
2014-01-28 03:22:37 +00:00
txn, err := db.Transaction()
2014-01-13 17:35:04 +00:00
assert.Nil(t, txn)
assert.Equal(t, err, DatabaseNotOpenError)
})
}
2014-01-12 22:30:09 +00:00
// withDB executes a function with a database reference.
2014-01-12 05:51:01 +00:00
func withDB(fn func(*DB, string)) {
f, _ := ioutil.TempFile("", "bolt-")
path := f.Name()
f.Close()
os.Remove(path)
defer os.RemoveAll(path)
db := NewDB()
fn(db, path)
}
2014-01-12 22:30:09 +00:00
// withMockDB executes a function with a database reference and a mock filesystem.
2014-01-13 15:25:56 +00:00
func withMockDB(fn func(*DB, *mockos, *mocksyscall, string)) {
os, syscall := &mockos{}, &mocksyscall{}
2014-01-12 22:30:09 +00:00
db := NewDB()
db.os = os
2014-01-13 15:25:56 +00:00
db.syscall = syscall
fn(db, os, syscall, "/mock/db")
2014-01-12 22:30:09 +00:00
}
2014-01-13 17:35:04 +00:00
// withOpenDB executes a function with an already opened database.
func withOpenDB(fn func(*DB, string)) {
withDB(func(db *DB, path string) {
if err := db.Open(path, 0666); err != nil {
panic("cannot open db: " + err.Error())
}
defer db.Close()
fn(db, path)
})
}