Convert Figma logo to code with AI

tidwall logobuntdb

BuntDB is an embeddable, in-memory key/value database for Go with custom indexing and geospatial support

4,598
290
4,598
31

Top Related Projects

8,368

An embedded key/value database for Go.

14,089

Fast key-value DB in Go.

LevelDB key/value database in Go.

5,035

RocksDB/LevelDB inspired key-value database in Go

14,262

An embedded key/value database for Go.

1,570

Modern embedded SQL database

Quick Overview

BuntDB is a low-level, in-memory key/value database for Go with custom indexing and geospatial support. It's designed to be fast, simple, and reliable, with a focus on performance and ease of use.

Pros

  • Fast performance due to in-memory storage
  • Supports custom indexing and geospatial queries
  • ACID compliant with support for transactions
  • Simple API and easy to integrate into Go projects

Cons

  • Limited to in-memory storage, which may not be suitable for large datasets
  • Lacks some advanced features found in more complex databases
  • May require additional effort for persistence and data recovery

Code Examples

  1. Opening a database and setting a key-value pair:
db, _ := buntdb.Open(":memory:")
defer db.Close()

db.Update(func(tx *buntdb.Tx) error {
    _, _, err := tx.Set("mykey", "myvalue", nil)
    return err
})
  1. Creating an index and querying data:
db.CreateIndex("last_name", "*", buntdb.IndexString)

db.View(func(tx *buntdb.Tx) error {
    tx.Ascend("last_name", func(key, value string) bool {
        fmt.Printf("%s: %s\n", key, value)
        return true
    })
    return nil
})
  1. Performing a geospatial query:
db.CreateSpatialIndex("fleet", "fleet:*", buntdb.IndexRect)

db.View(func(tx *buntdb.Tx) error {
    tx.Intersects("fleet", "[[-122.4,37.7],[-122.3,37.8]]", func(key, value string) bool {
        fmt.Printf("%s: %s\n", key, value)
        return true
    })
    return nil
})

Getting Started

To use BuntDB in your Go project, first install it:

go get github.com/tidwall/buntdb

Then, import it in your Go code:

import "github.com/tidwall/buntdb"

Create a new in-memory database:

db, err := buntdb.Open(":memory:")
if err != nil {
    log.Fatal(err)
}
defer db.Close()

Now you can start using the database with methods like Update(), View(), and CreateIndex().

Competitor Comparisons

8,368

An embedded key/value database for Go.

Pros of bbolt

  • Mature and battle-tested, used in production by etcd
  • ACID compliant with full transaction support
  • Designed for high read performance

Cons of bbolt

  • Limited to key-value operations, no built-in indexing or querying
  • Single-writer, multiple-reader model may limit write concurrency

Code Comparison

bbolt:

db, _ := bbolt.Open("my.db", 0600, nil)
defer db.Close()

db.Update(func(tx *bbolt.Tx) error {
    b, _ := tx.CreateBucketIfNotExists([]byte("MyBucket"))
    return b.Put([]byte("key"), []byte("value"))
})

BuntDB:

db, _ := buntdb.Open("my.db")
defer db.Close()

db.Update(func(tx *buntdb.Tx) error {
    _, _, _ = tx.Set("key", "value", nil)
    return nil
})

Both libraries offer similar APIs for basic key-value operations, but BuntDB provides additional features like spatial indexing and custom indexing functions. bbolt focuses on simplicity and robustness, while BuntDB offers more flexibility and advanced querying capabilities.

14,089

Fast key-value DB in Go.

Pros of Badger

  • Designed for high performance on SSDs with optimized read and write operations
  • Supports ACID transactions with snapshot isolation
  • Offers better scalability for large datasets due to its LSM tree structure

Cons of Badger

  • Higher memory usage compared to BuntDB
  • More complex setup and configuration process
  • Lacks some of the built-in features found in BuntDB, such as spatial indexing

Code Comparison

Badger:

db, err := badger.Open(badger.DefaultOptions("/tmp/badger"))
defer db.Close()

err = db.Update(func(txn *badger.Txn) error {
    return txn.Set([]byte("key"), []byte("value"))
})

BuntDB:

db, err := buntdb.Open(":memory:")
defer db.Close()

err = db.Update(func(tx *buntdb.Tx) error {
    _, _, err := tx.Set("key", "value", nil)
    return err
})

Both databases offer similar API structures for basic operations, but Badger's API is more focused on byte slices, while BuntDB uses string keys and values. Badger's transactions are more explicit about their read or write nature, whereas BuntDB uses a single transaction type for both operations.

LevelDB key/value database in Go.

Pros of goleveldb

  • Implements the LevelDB key-value storage engine, providing better performance for large datasets
  • Supports advanced features like snapshots and iterators
  • More mature and widely used in production environments

Cons of goleveldb

  • More complex API compared to BuntDB's simpler interface
  • Lacks some of BuntDB's built-in features like spatial indexing and TTL support
  • Requires manual compaction management for optimal performance

Code Comparison

BuntDB example:

db, _ := buntdb.Open("data.db")
db.Update(func(tx *buntdb.Tx) error {
    tx.Set("key", "value", nil)
    return nil
})

goleveldb example:

db, _ := leveldb.OpenFile("data.db", nil)
db.Put([]byte("key"), []byte("value"), nil)

Both libraries provide key-value storage capabilities, but BuntDB offers a more straightforward API with transaction support built into its core operations. goleveldb, on the other hand, provides lower-level access to the database, requiring manual byte conversions for keys and values.

BuntDB is designed for simplicity and ease of use, making it suitable for smaller projects or those requiring specific features like spatial indexing. goleveldb is better suited for larger-scale applications that need high performance and advanced features like snapshots and iterators.

5,035

RocksDB/LevelDB inspired key-value database in Go

Pros of Pebble

  • Designed for high performance and scalability, suitable for large-scale distributed systems
  • Implements advanced features like bloom filters and compression for improved efficiency
  • Actively maintained and backed by the CockroachDB team, ensuring ongoing development and support

Cons of Pebble

  • More complex to set up and use compared to BuntDB's simpler API
  • Requires more system resources due to its advanced features and optimizations
  • May be overkill for smaller projects or simple key-value storage needs

Code Comparison

BuntDB example:

db, _ := buntdb.Open("data.db")
db.Update(func(tx *buntdb.Tx) error {
    _, _, _ = tx.Set("mykey", "myvalue", nil)
    return nil
})

Pebble example:

db, _ := pebble.Open("data", &pebble.Options{})
_ = db.Set([]byte("mykey"), []byte("myvalue"), pebble.Sync)

Both libraries provide key-value storage functionality, but Pebble offers more advanced features and is designed for larger-scale applications. BuntDB is simpler to use and may be more suitable for smaller projects or those requiring a lightweight database solution. The choice between the two depends on the specific requirements of your project, including scalability needs, performance demands, and complexity tolerance.

14,262

An embedded key/value database for Go.

Pros of Bolt

  • Mature and battle-tested, with a longer history of production use
  • Supports nested buckets for more complex data structures
  • Offers ACID compliance with full transaction support

Cons of Bolt

  • Read-only transactions are not supported, potentially impacting performance
  • Limited built-in indexing capabilities
  • Less active development and maintenance compared to BuntDB

Code Comparison

BuntDB:

db, _ := buntdb.Open("data.db")
db.Update(func(tx *buntdb.Tx) error {
    tx.Set("key", "value", nil)
    return nil
})

Bolt:

db, _ := bolt.Open("data.db", 0600, nil)
db.Update(func(tx *bolt.Tx) error {
    b := tx.Bucket([]byte("MyBucket"))
    b.Put([]byte("key"), []byte("value"))
    return nil
})

Both BuntDB and Bolt are embedded key-value stores for Go, but they have different focuses. BuntDB is designed for simplicity and speed, with features like spatial indexing and TTL support. Bolt, on the other hand, emphasizes durability and ACID compliance, making it suitable for applications requiring strong consistency guarantees. The choice between them depends on specific project requirements, such as data structure complexity, performance needs, and consistency requirements.

1,570

Modern embedded SQL database

Pros of Chai

  • Supports SQL-like query language for more complex data operations
  • Offers ACID transactions for data integrity
  • Provides indexing capabilities for improved query performance

Cons of Chai

  • Less mature project with potentially fewer community contributions
  • May have higher memory usage due to more complex features
  • Potentially slower for simple key-value operations

Code Comparison

BuntDB:

db, _ := buntdb.Open("data.db")
db.Update(func(tx *buntdb.Tx) error {
    tx.Set("key", "value", nil)
    return nil
})

Chai:

db, _ := chai.Open("data.db")
db.Exec("CREATE TABLE users (id INT, name TEXT)")
db.Exec("INSERT INTO users (id, name) VALUES (?, ?)", 1, "John")

Key Differences

  • BuntDB is a simple key-value store, while Chai offers more advanced database features
  • Chai provides a SQL-like interface, making it easier for developers familiar with SQL
  • BuntDB focuses on performance for basic operations, while Chai offers more flexibility
  • Chai supports structured data and complex queries, whereas BuntDB is better suited for simpler data storage needs

Both databases have their strengths, and the choice between them depends on the specific requirements of your project. BuntDB is ideal for lightweight, high-performance key-value storage, while Chai is better suited for applications requiring more advanced database features and SQL-like querying capabilities.

Convert Figma logo designs to code with AI

Visual Copilot

Introducing Visual Copilot: A new AI model to turn Figma designs to high quality code using your components.

Try Visual Copilot

README

BuntDB
Godoc LICENSE

BuntDB is a low-level, in-memory, key/value store in pure Go. It persists to disk, is ACID compliant, and uses locking for multiple readers and a single writer. It supports custom indexes and geospatial data. It's ideal for projects that need a dependable database and favor speed over data size.

Features

Getting Started

Installing

To start using BuntDB, install Go and run go get:

$ go get -u github.com/tidwall/buntdb

This will retrieve the library.

Opening a database

The primary object in BuntDB is a DB. To open or create your database, use the buntdb.Open() function:

package main

import (
	"log"

	"github.com/tidwall/buntdb"
)

func main() {
	// Open the data.db file. It will be created if it doesn't exist.
	db, err := buntdb.Open("data.db")
	if err != nil {
		log.Fatal(err)
	}
	defer db.Close()

	...
}

It's also possible to open a database that does not persist to disk by using :memory: as the path of the file.

buntdb.Open(":memory:") // Open a file that does not persist to disk.

Transactions

All reads and writes must be performed from inside a transaction. BuntDB can have one write transaction opened at a time, but can have many concurrent read transactions. Each transaction maintains a stable view of the database. In other words, once a transaction has begun, the data for that transaction cannot be changed by other transactions.

Transactions run in a function that exposes a Tx object, which represents the transaction state. While inside a transaction, all database operations should be performed using this object. You should never access the origin DB object while inside a transaction. Doing so may have side-effects, such as blocking your application.

When a transaction fails, it will roll back, and revert all changes that occurred to the database during that transaction. There's a single return value that you can use to close the transaction. For read/write transactions, returning an error this way will force the transaction to roll back. When a read/write transaction succeeds all changes are persisted to disk.

Read-only Transactions

A read-only transaction should be used when you don't need to make changes to the data. The advantage of a read-only transaction is that there can be many running concurrently.

err := db.View(func(tx *buntdb.Tx) error {
	...
	return nil
})

Read/write Transactions

A read/write transaction is used when you need to make changes to your data. There can only be one read/write transaction running at a time. So make sure you close it as soon as you are done with it.

err := db.Update(func(tx *buntdb.Tx) error {
	...
	return nil
})

Setting and getting key/values

To set a value you must open a read/write transaction:

err := db.Update(func(tx *buntdb.Tx) error {
	_, _, err := tx.Set("mykey", "myvalue", nil)
	return err
})

To get the value:

err := db.View(func(tx *buntdb.Tx) error {
	val, err := tx.Get("mykey")
	if err != nil{
		return err
	}
	fmt.Printf("value is %s\n", val)
	return nil
})

Getting non-existent values will cause an ErrNotFound error.

Iterating

All keys/value pairs are ordered in the database by the key. To iterate over the keys:

err := db.View(func(tx *buntdb.Tx) error {
	err := tx.Ascend("", func(key, value string) bool {
		fmt.Printf("key: %s, value: %s\n", key, value)
		return true // continue iteration
	})
	return err
})

There is also AscendGreaterOrEqual, AscendLessThan, AscendRange, AscendEqual, Descend, DescendLessOrEqual, DescendGreaterThan, DescendRange, and DescendEqual. Please see the documentation for more information on these functions.

Custom Indexes

Initially all data is stored in a single B-tree with each item having one key and one value. All of these items are ordered by the key. This is great for quickly getting a value from a key or iterating over the keys. Feel free to peruse the B-tree implementation.

You can also create custom indexes that allow for ordering and iterating over values. A custom index also uses a B-tree, but it's more flexible because it allows for custom ordering.

For example, let's say you want to create an index for ordering names:

db.CreateIndex("names", "*", buntdb.IndexString)

This will create an index named names which stores and sorts all values. The second parameter is a pattern that is used to filter on keys. A * wildcard argument means that we want to accept all keys. IndexString is a built-in function that performs case-insensitive ordering on the values

Now you can add various names:

db.Update(func(tx *buntdb.Tx) error {
	tx.Set("user:0:name", "tom", nil)
	tx.Set("user:1:name", "Randi", nil)
	tx.Set("user:2:name", "jane", nil)
	tx.Set("user:4:name", "Janet", nil)
	tx.Set("user:5:name", "Paula", nil)
	tx.Set("user:6:name", "peter", nil)
	tx.Set("user:7:name", "Terri", nil)
	return nil
})

Finally you can iterate over the index:

db.View(func(tx *buntdb.Tx) error {
	tx.Ascend("names", func(key, val string) bool {
	fmt.Printf(buf, "%s %s\n", key, val)
		return true
	})
	return nil
})

The output should be:

user:2:name jane
user:4:name Janet
user:5:name Paula
user:6:name peter
user:1:name Randi
user:7:name Terri
user:0:name tom

The pattern parameter can be used to filter on keys like this:

db.CreateIndex("names", "user:*", buntdb.IndexString)

Now only items with keys that have the prefix user: will be added to the names index.

Built-in types

Along with IndexString, there is also IndexInt, IndexUint, and IndexFloat. These are built-in types for indexing. You can choose to use these or create your own.

So to create an index that is numerically ordered on an age key, we could use:

db.CreateIndex("ages", "user:*:age", buntdb.IndexInt)

And then add values:

db.Update(func(tx *buntdb.Tx) error {
	tx.Set("user:0:age", "35", nil)
	tx.Set("user:1:age", "49", nil)
	tx.Set("user:2:age", "13", nil)
	tx.Set("user:4:age", "63", nil)
	tx.Set("user:5:age", "8", nil)
	tx.Set("user:6:age", "3", nil)
	tx.Set("user:7:age", "16", nil)
	return nil
})
db.View(func(tx *buntdb.Tx) error {
	tx.Ascend("ages", func(key, val string) bool {
	fmt.Printf(buf, "%s %s\n", key, val)
		return true
	})
	return nil
})

The output should be:

user:6:age 3
user:5:age 8
user:2:age 13
user:7:age 16
user:0:age 35
user:1:age 49
user:4:age 63

Spatial Indexes

BuntDB has support for spatial indexes by storing rectangles in an R-tree. An R-tree is organized in a similar manner as a B-tree, and both are balanced trees. But, an R-tree is special because it can operate on data that is in multiple dimensions. This is super handy for Geospatial applications.

To create a spatial index use the CreateSpatialIndex function:

db.CreateSpatialIndex("fleet", "fleet:*:pos", buntdb.IndexRect)

Then IndexRect is a built-in function that converts rect strings to a format that the R-tree can use. It's easy to use this function out of the box, but you might find it better to create a custom one that renders from a different format, such as Well-known text or GeoJSON.

To add some lon,lat points to the fleet index:

db.Update(func(tx *buntdb.Tx) error {
	tx.Set("fleet:0:pos", "[-115.567 33.532]", nil)
	tx.Set("fleet:1:pos", "[-116.671 35.735]", nil)
	tx.Set("fleet:2:pos", "[-113.902 31.234]", nil)
	return nil
})

And then you can run the Intersects function on the index:

db.View(func(tx *buntdb.Tx) error {
	tx.Intersects("fleet", "[-117 30],[-112 36]", func(key, val string) bool {
		...
		return true
	})
	return nil
})

This will get all three positions.

k-Nearest Neighbors

Use the Nearby function to get all the positions in order of nearest to farthest :

db.View(func(tx *buntdb.Tx) error {
	tx.Nearby("fleet", "[-113 33]", func(key, val string, dist float64) bool {
		...
		return true
	})
	return nil
})

Spatial bracket syntax

The bracket syntax [-117 30],[-112 36] is unique to BuntDB, and it's how the built-in rectangles are processed. But, you are not limited to this syntax. Whatever Rect function you choose to use during CreateSpatialIndex will be used to process the parameter, in this case it's IndexRect.

  • 2D rectangle: [10 15],[20 25] Min XY: "10x15", Max XY: "20x25"

  • 3D rectangle: [10 15 12],[20 25 18] Min XYZ: "10x15x12", Max XYZ: "20x25x18"

  • 2D point: [10 15] XY: "10x15"

  • LonLat point: [-112.2693 33.5123] LatLon: "33.5123 -112.2693"

  • LonLat bounding box: [-112.26 33.51],[-112.18 33.67] Min LatLon: "33.51 -112.26", Max LatLon: "33.67 -112.18"

Notice: The longitude is the Y axis and is on the left, and latitude is the X axis and is on the right.

You can also represent Infinity by using -inf and +inf. For example, you might have the following points ([X Y M] where XY is a point and M is a timestamp):

[3 9 1]
[3 8 2]
[4 8 3]
[4 7 4]
[5 7 5]
[5 6 6]

You can then do a search for all points with M between 2-4 by calling Intersects.

tx.Intersects("points", "[-inf -inf 2],[+inf +inf 4]", func(key, val string) bool {
	println(val)
	return true
})

Which will return:

[3 8 2]
[4 8 3]
[4 7 4]

JSON Indexes

Indexes can be created on individual fields inside JSON documents. BuntDB uses GJSON under the hood.

For example:

package main

import (
	"fmt"

	"github.com/tidwall/buntdb"
)

func main() {
	db, _ := buntdb.Open(":memory:")
	db.CreateIndex("last_name", "*", buntdb.IndexJSON("name.last"))
	db.CreateIndex("age", "*", buntdb.IndexJSON("age"))
	db.Update(func(tx *buntdb.Tx) error {
		tx.Set("1", `{"name":{"first":"Tom","last":"Johnson"},"age":38}`, nil)
		tx.Set("2", `{"name":{"first":"Janet","last":"Prichard"},"age":47}`, nil)
		tx.Set("3", `{"name":{"first":"Carol","last":"Anderson"},"age":52}`, nil)
		tx.Set("4", `{"name":{"first":"Alan","last":"Cooper"},"age":28}`, nil)
		return nil
	})
	db.View(func(tx *buntdb.Tx) error {
		fmt.Println("Order by last name")
		tx.Ascend("last_name", func(key, value string) bool {
			fmt.Printf("%s: %s\n", key, value)
			return true
		})
		fmt.Println("Order by age")
		tx.Ascend("age", func(key, value string) bool {
			fmt.Printf("%s: %s\n", key, value)
			return true
		})
		fmt.Println("Order by age range 30-50")
		tx.AscendRange("age", `{"age":30}`, `{"age":50}`, func(key, value string) bool {
			fmt.Printf("%s: %s\n", key, value)
			return true
		})
		return nil
	})
}

Results:

Order by last name
3: {"name":{"first":"Carol","last":"Anderson"},"age":52}
4: {"name":{"first":"Alan","last":"Cooper"},"age":28}
1: {"name":{"first":"Tom","last":"Johnson"},"age":38}
2: {"name":{"first":"Janet","last":"Prichard"},"age":47}

Order by age
4: {"name":{"first":"Alan","last":"Cooper"},"age":28}
1: {"name":{"first":"Tom","last":"Johnson"},"age":38}
2: {"name":{"first":"Janet","last":"Prichard"},"age":47}
3: {"name":{"first":"Carol","last":"Anderson"},"age":52}

Order by age range 30-50
1: {"name":{"first":"Tom","last":"Johnson"},"age":38}
2: {"name":{"first":"Janet","last":"Prichard"},"age":47}

Multi Value Index

With BuntDB it's possible to join multiple values on a single index. This is similar to a multi column index in a traditional SQL database.

In this example we are creating a multi value index on "name.last" and "age":

db, _ := buntdb.Open(":memory:")
db.CreateIndex("last_name_age", "*", buntdb.IndexJSON("name.last"), buntdb.IndexJSON("age"))
db.Update(func(tx *buntdb.Tx) error {
	tx.Set("1", `{"name":{"first":"Tom","last":"Johnson"},"age":38}`, nil)
	tx.Set("2", `{"name":{"first":"Janet","last":"Prichard"},"age":47}`, nil)
	tx.Set("3", `{"name":{"first":"Carol","last":"Anderson"},"age":52}`, nil)
	tx.Set("4", `{"name":{"first":"Alan","last":"Cooper"},"age":28}`, nil)
	tx.Set("5", `{"name":{"first":"Sam","last":"Anderson"},"age":51}`, nil)
	tx.Set("6", `{"name":{"first":"Melinda","last":"Prichard"},"age":44}`, nil)
	return nil
})
db.View(func(tx *buntdb.Tx) error {
	tx.Ascend("last_name_age", func(key, value string) bool {
		fmt.Printf("%s: %s\n", key, value)
		return true
	})
	return nil
})

// Output:
// 5: {"name":{"first":"Sam","last":"Anderson"},"age":51}
// 3: {"name":{"first":"Carol","last":"Anderson"},"age":52}
// 4: {"name":{"first":"Alan","last":"Cooper"},"age":28}
// 1: {"name":{"first":"Tom","last":"Johnson"},"age":38}
// 6: {"name":{"first":"Melinda","last":"Prichard"},"age":44}
// 2: {"name":{"first":"Janet","last":"Prichard"},"age":47}

Descending Ordered Index

Any index can be put in descending order by wrapping it's less function with buntdb.Desc.

db.CreateIndex("last_name_age", "*",
    buntdb.IndexJSON("name.last"),
    buntdb.Desc(buntdb.IndexJSON("age")),
)

This will create a multi value index where the last name is ascending and the age is descending.

Collate i18n Indexes

Using the external collate package it's possible to create indexes that are sorted by the specified language. This is similar to the SQL COLLATE keyword found in traditional databases.

To install:

go get -u github.com/tidwall/collate

For example:

import "github.com/tidwall/collate"

// To sort case-insensitive in French.
db.CreateIndex("name", "*", collate.IndexString("FRENCH_CI"))

// To specify that numbers should sort numerically ("2" < "12")
// and use a comma to represent a decimal point.
db.CreateIndex("amount", "*", collate.IndexString("FRENCH_NUM"))

There's also support for Collation on JSON indexes:

db.CreateIndex("last_name", "*", collate.IndexJSON("CHINESE_CI", "name.last"))

Check out the collate project for more information.

Data Expiration

Items can be automatically evicted by using the SetOptions object in the Set function to set a TTL.

db.Update(func(tx *buntdb.Tx) error {
	tx.Set("mykey", "myval", &buntdb.SetOptions{Expires:true, TTL:time.Second})
	return nil
})

Now mykey will automatically be deleted after one second. You can remove the TTL by setting the value again with the same key/value, but with the options parameter set to nil.

Delete while iterating

BuntDB does not currently support deleting a key while in the process of iterating. As a workaround you'll need to delete keys following the completion of the iterator.

var delkeys []string
tx.AscendKeys("object:*", func(k, v string) bool {
	if someCondition(k) == true {
		delkeys = append(delkeys, k)
	}
	return true // continue
})
for _, k := range delkeys {
	if _, err = tx.Delete(k); err != nil {
		return err
	}
}

Append-only File

BuntDB uses an AOF (append-only file) which is a log of all database changes that occur from operations like Set() and Delete().

The format of this file looks like:

set key:1 value1
set key:2 value2
set key:1 value3
del key:2
...

When the database opens again, it will read back the aof file and process each command in exact order. This read process happens one time when the database opens. From there on the file is only appended.

As you may guess this log file can grow large over time. There's a background routine that automatically shrinks the log file when it gets too large. There is also a Shrink() function which will rewrite the aof file so that it contains only the items in the database. The shrink operation does not lock up the database so read and write transactions can continue while shrinking is in process.

Durability and fsync

By default BuntDB executes an fsync once every second on the aof file. Which simply means that there's a chance that up to one second of data might be lost. If you need higher durability then there's an optional database config setting Config.SyncPolicy which can be set to Always.

The Config.SyncPolicy has the following options:

  • Never - fsync is managed by the operating system, less safe
  • EverySecond - fsync every second, fast and safer, this is the default
  • Always - fsync after every write, very durable, slower

Config

Here are some configuration options that can be use to change various behaviors of the database.

  • SyncPolicy adjusts how often the data is synced to disk. This value can be Never, EverySecond, or Always. Default is EverySecond.
  • AutoShrinkPercentage is used by the background process to trigger a shrink of the aof file when the size of the file is larger than the percentage of the result of the previous shrunk file. For example, if this value is 100, and the last shrink process resulted in a 100mb file, then the new aof file must be 200mb before a shrink is triggered. Default is 100.
  • AutoShrinkMinSize defines the minimum size of the aof file before an automatic shrink can occur. Default is 32MB.
  • AutoShrinkDisabled turns off automatic background shrinking. Default is false.

To update the configuration you should call ReadConfig followed by SetConfig. For example:


var config buntdb.Config
if err := db.ReadConfig(&config); err != nil{
	log.Fatal(err)
}
if err := db.SetConfig(config); err != nil{
	log.Fatal(err)
}

Performance

How fast is BuntDB?

Here are some example benchmarks when using BuntDB in a Raft Store implementation.

You can also run the standard Go benchmark tool from the project root directory:

go test --bench=.

BuntDB-Benchmark

There's a custom utility that was created specifically for benchmarking BuntDB.

These are the results from running the benchmarks on a MacBook Pro 15" 2.8 GHz Intel Core i7:

$ buntdb-benchmark -q
GET: 4609604.74 operations per second
SET: 248500.33 operations per second
ASCEND_100: 2268998.79 operations per second
ASCEND_200: 1178388.14 operations per second
ASCEND_400: 679134.20 operations per second
ASCEND_800: 348445.55 operations per second
DESCEND_100: 2313821.69 operations per second
DESCEND_200: 1292738.38 operations per second
DESCEND_400: 675258.76 operations per second
DESCEND_800: 337481.67 operations per second
SPATIAL_SET: 134824.60 operations per second
SPATIAL_INTERSECTS_100: 939491.47 operations per second
SPATIAL_INTERSECTS_200: 561590.40 operations per second
SPATIAL_INTERSECTS_400: 306951.15 operations per second
SPATIAL_INTERSECTS_800: 159673.91 operations per second

To install this utility:

go get github.com/tidwall/buntdb-benchmark

Contact

Josh Baker @tidwall

License

BuntDB source code is available under the MIT License.