Categorygithub.com/hashicorp/go-memdb
modulepackage
1.3.5
Repository: https://github.com/hashicorp/go-memdb.git
Documentation: pkg.go.dev

# README

go-memdb CircleCI

Provides the memdb package that implements a simple in-memory database built on immutable radix trees. The database provides Atomicity, Consistency and Isolation from ACID. Being that it is in-memory, it does not provide durability. The database is instantiated with a schema that specifies the tables and indices that exist and allows transactions to be executed.

The database provides the following:

  • Multi-Version Concurrency Control (MVCC) - By leveraging immutable radix trees the database is able to support any number of concurrent readers without locking, and allows a writer to make progress.

  • Transaction Support - The database allows for rich transactions, in which multiple objects are inserted, updated or deleted. The transactions can span multiple tables, and are applied atomically. The database provides atomicity and isolation in ACID terminology, such that until commit the updates are not visible.

  • Rich Indexing - Tables can support any number of indexes, which can be simple like a single field index, or more advanced compound field indexes. Certain types like UUID can be efficiently compressed from strings into byte indexes for reduced storage requirements.

  • Watches - Callers can populate a watch set as part of a query, which can be used to detect when a modification has been made to the database which affects the query results. This lets callers easily watch for changes in the database in a very general way.

For the underlying immutable radix trees, see go-immutable-radix.

Documentation

The full documentation is available on Godoc.

Example

Below is a simple example of usage

// Create a sample struct
type Person struct {
	Email string
	Name  string
	Age   int
}

// Create the DB schema
schema := &memdb.DBSchema{
	Tables: map[string]*memdb.TableSchema{
		"person": &memdb.TableSchema{
			Name: "person",
			Indexes: map[string]*memdb.IndexSchema{
				"id": &memdb.IndexSchema{
					Name:    "id",
					Unique:  true,
					Indexer: &memdb.StringFieldIndex{Field: "Email"},
				},
				"age": &memdb.IndexSchema{
					Name:    "age",
					Unique:  false,
					Indexer: &memdb.IntFieldIndex{Field: "Age"},
				},
			},
		},
	},
}

// Create a new data base
db, err := memdb.NewMemDB(schema)
if err != nil {
	panic(err)
}

// Create a write transaction
txn := db.Txn(true)

// Insert some people
people := []*Person{
	&Person{"joe@aol.com", "Joe", 30},
	&Person{"lucy@aol.com", "Lucy", 35},
	&Person{"tariq@aol.com", "Tariq", 21},
	&Person{"dorothy@aol.com", "Dorothy", 53},
}
for _, p := range people {
	if err := txn.Insert("person", p); err != nil {
		panic(err)
	}
}

// Commit the transaction
txn.Commit()

// Create read-only transaction
txn = db.Txn(false)
defer txn.Abort()

// Lookup by email
raw, err := txn.First("person", "id", "joe@aol.com")
if err != nil {
	panic(err)
}

// Say hi!
fmt.Printf("Hello %s!\n", raw.(*Person).Name)

// List all the people
it, err := txn.Get("person", "id")
if err != nil {
	panic(err)
}

fmt.Println("All the people:")
for obj := it.Next(); obj != nil; obj = it.Next() {
	p := obj.(*Person)
	fmt.Printf("  %s\n", p.Name)
}

// Range scan over people with ages between 25 and 35 inclusive
it, err = txn.LowerBound("person", "age", 25)
if err != nil {
	panic(err)
}

fmt.Println("People aged 25 - 35:")
for obj := it.Next(); obj != nil; obj = it.Next() {
	p := obj.(*Person)
	if p.Age > 35 {
		break
	}
	fmt.Printf("  %s is aged %d\n", p.Name, p.Age)
}
// Output:
// Hello Joe!
// All the people:
//   Dorothy
//   Joe
//   Lucy
//   Tariq
// People aged 25 - 35:
//   Joe is aged 30
//   Lucy is aged 35

# Packages

This tool generates the special-case code for a small number of watchers which runs all the watches in a single select vs.

# Functions

IsIntType returns whether the passed type is a type of int and the number of bytes needed to encode the type.
IsUintType returns whether the passed type is a type of uint and the number of bytes needed to encode the type.
NewFilterIterator wraps a ResultIterator.
NewMemDB creates a new MemDB with the given schema.
NewWatchSet constructs a new watch set.

# Variables

ErrNotFound is returned when the requested item is not found.

# Structs

BoolFieldIndex is used to extract an boolean field from an object using reflection and builds an index on that field.
Change describes a mutation to an object in a table.
CompoundIndex is used to build an index using multiple sub-indexes Prefix based iteration is supported as long as the appropriate prefix of indexers support it.
CompoundMultiIndex is used to build an index using multiple sub-indexes.
ConditionalIndex builds an index based on a condition specified by a passed user function.
DBSchema is the schema to use for the full database with a MemDB instance.
FieldSetIndex is used to extract a field from an object using reflection and builds an index on whether the field is set by comparing it against its type's nil value.
FilterIterator is used to wrap a ResultIterator and apply a filter over it.
IndexSchema is the schema for an index.
IntFieldIndex is used to extract an int field from an object using reflection and builds an index on that field.
MemDB is an in-memory database providing Atomicity, Consistency, and Isolation from ACID.
StringFieldIndex is used to extract a field from an object using reflection and builds an index on that field.
StringMapFieldIndex is used to extract a field of type map[string]string from an object using reflection and builds an index on that field.
StringSliceFieldIndex builds an index from a field on an object that is a string slice ([]string).
TableSchema is the schema for a single table.
Txn is a transaction against a MemDB.
UintFieldIndex is used to extract a uint field from an object using reflection and builds an index on that field.
UUIDFieldIndex is used to extract a field from an object using reflection and builds an index on that field by treating it as a UUID.

# Interfaces

Indexer is an interface used for defining indexes.
MultiIndexer is an interface used for defining indexes that generate multiple values per object.
PrefixIndexer is an optional interface on top of an Indexer that allows indexes to support prefix-based iteration.
ResultIterator is used to iterate over a list of results from a query on a table.
SingleIndexer is an interface used for defining indexes that generate a single value per object.

# Type aliases

Changes describes a set of mutations to memDB tables performed during a transaction.
ConditionalIndexFunc is the required function interface for a ConditionalIndex.
FilterFunc is a function that takes the results of an iterator and returns whether the result should be filtered out.
WatchSet is a collection of watch channels.