374 lines
11 KiB
Go
374 lines
11 KiB
Go
package influxdb
|
|
|
|
import (
|
|
"encoding/binary"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/boltdb/bolt"
|
|
"github.com/influxdb/influxdb/influxql"
|
|
"github.com/influxdb/influxdb/meta"
|
|
)
|
|
|
|
// MetaStore represents an interface access and mutate metadata.
|
|
type MetaStore interface {
|
|
CreateContinuousQuery(query string) (*meta.ContinuousQueryInfo, error)
|
|
DropContinuousQuery(query string) error
|
|
|
|
Node(id uint64) (*meta.NodeInfo, error)
|
|
NodeByHost(host string) (*meta.NodeInfo, error)
|
|
CreateNode(host string) (*meta.NodeInfo, error)
|
|
DeleteNode(id uint64) error
|
|
|
|
Database(name string) (*meta.DatabaseInfo, error)
|
|
CreateDatabase(name string) (*meta.DatabaseInfo, error)
|
|
CreateDatabaseIfNotExists(name string) (*meta.DatabaseInfo, error)
|
|
DropDatabase(name string) error
|
|
|
|
RetentionPolicy(database, name string) (*meta.RetentionPolicyInfo, error)
|
|
CreateRetentionPolicy(database string, rp *meta.RetentionPolicyInfo) (*meta.RetentionPolicyInfo, error)
|
|
CreateRetentionPolicyIfNotExists(database string, rp *meta.RetentionPolicyInfo) (*meta.RetentionPolicyInfo, error)
|
|
SetDefaultRetentionPolicy(database, name string) error
|
|
UpdateRetentionPolicy(database, name string, rpu *meta.RetentionPolicyUpdate) (*meta.RetentionPolicyInfo, error)
|
|
DeleteRetentionPolicy(database, name string) error
|
|
|
|
ShardGroup(database, policy string, timestamp time.Time) (*meta.ShardGroupInfo, error)
|
|
CreateShardGroupIfNotExists(database, policy string, timestamp time.Time) (*meta.ShardGroupInfo, error)
|
|
DeleteShardGroup(database, policy string, shardID uint64) error
|
|
|
|
User(username string) (*meta.UserInfo, error)
|
|
CreateUser(username, password string, admin bool) (*meta.UserInfo, error)
|
|
UpdateUser(username, password string) (*meta.UserInfo, error)
|
|
DeleteUser(username string) error
|
|
SetPrivilege(p influxql.Privilege, username string, dbname string) error
|
|
}
|
|
|
|
// metastore represents the low-level data store for metadata.
|
|
type metastore struct {
|
|
db *bolt.DB
|
|
}
|
|
|
|
// open initializes the metastore.
|
|
func (m *metastore) open(path string) error {
|
|
// Open the bolt-backed database.
|
|
db, err := bolt.Open(path, 0666, &bolt.Options{Timeout: 1 * time.Second})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
m.db = db
|
|
|
|
// Initialize the metastore.
|
|
if err := m.init(); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// close closes the store.
|
|
func (m *metastore) close() error {
|
|
if m.db != nil {
|
|
return m.db.Close()
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// init initializes the metastore to ensure all top-level buckets are created.
|
|
func (m *metastore) init() error {
|
|
return m.db.Update(func(tx *bolt.Tx) error {
|
|
_, _ = tx.CreateBucketIfNotExists([]byte("Meta"))
|
|
_, _ = tx.CreateBucketIfNotExists([]byte("DataNodes"))
|
|
_, _ = tx.CreateBucketIfNotExists([]byte("Databases"))
|
|
_, _ = tx.CreateBucketIfNotExists([]byte("Users"))
|
|
return nil
|
|
})
|
|
}
|
|
|
|
// view executes a function in the context of a read-only transaction.
|
|
func (m *metastore) view(fn func(*metatx) error) error {
|
|
return m.db.View(func(tx *bolt.Tx) error { return fn(&metatx{tx}) })
|
|
}
|
|
|
|
// update executes a function in the context of a read-write transaction.
|
|
func (m *metastore) update(fn func(*metatx) error) error {
|
|
return m.db.Update(func(tx *bolt.Tx) error { return fn(&metatx{tx}) })
|
|
}
|
|
|
|
// mustView executes a function in the context of a read-only transaction.
|
|
// Panics if system error occurs. Return error from the fn for validation errors.
|
|
func (m *metastore) mustView(fn func(*metatx) error) (err error) {
|
|
if e := m.view(func(tx *metatx) error {
|
|
err = fn(tx)
|
|
return nil
|
|
}); e != nil {
|
|
panic("metastore view: " + e.Error())
|
|
}
|
|
return
|
|
}
|
|
|
|
// mustUpdate executes a function in the context of a read-write transaction.
|
|
// This function requires an index so that it can track the last commit from the broker.
|
|
// Panics if a disk or system error occurs. Return error from the fn for validation errors.
|
|
func (m *metastore) mustUpdate(index uint64, fn func(*metatx) error) (err error) {
|
|
if e := m.update(func(tx *metatx) error {
|
|
// Ignore replayed commands.
|
|
curr := tx.index()
|
|
if index > 0 && index <= curr {
|
|
return nil
|
|
}
|
|
|
|
// Execute function passed in.
|
|
err = fn(tx)
|
|
|
|
// Update index, if set.
|
|
if index > 0 {
|
|
if e := tx.setIndex(index); e != nil {
|
|
return fmt.Errorf("set index: %s", e)
|
|
}
|
|
}
|
|
return nil
|
|
}); e != nil {
|
|
panic("metastore update: " + e.Error())
|
|
}
|
|
return
|
|
}
|
|
|
|
// metatx represents a metastore transaction.
|
|
type metatx struct {
|
|
*bolt.Tx
|
|
}
|
|
|
|
// id returns the server id.
|
|
func (tx *metatx) id() (id uint64) {
|
|
if v := tx.Bucket([]byte("Meta")).Get([]byte("id")); v != nil {
|
|
id = btou64(v)
|
|
}
|
|
return
|
|
}
|
|
|
|
// setID sets the server id.
|
|
func (tx *metatx) setID(v uint64) error {
|
|
return tx.Bucket([]byte("Meta")).Put([]byte("id"), u64tob(v))
|
|
}
|
|
|
|
// index returns the index stored in the meta bucket.
|
|
func (tx *metatx) index() (index uint64) {
|
|
if v := tx.Bucket([]byte("Meta")).Get([]byte("index")); v != nil {
|
|
index = btou64(v)
|
|
}
|
|
return
|
|
}
|
|
|
|
// setIndex sets the index stored in the meta bucket.
|
|
func (tx *metatx) setIndex(v uint64) error {
|
|
return tx.Bucket([]byte("Meta")).Put([]byte("index"), u64tob(v))
|
|
}
|
|
|
|
// mustNextSequence generates a new sequence for a key in the meta bucket.
|
|
func (tx *metatx) mustNextSequence(key []byte) (id uint64) {
|
|
// Retrieve the previous value, if it exists.
|
|
if v := tx.Bucket([]byte("Meta")).Get(key); v != nil {
|
|
id = btou64(v)
|
|
}
|
|
id++
|
|
|
|
// Save the new value.
|
|
// This panics on error because keys are hardcoded this shouldn't fail.
|
|
if err := tx.Bucket([]byte("Meta")).Put(key, u64tob(id)); err != nil {
|
|
panic("next seq: " + err.Error())
|
|
}
|
|
return
|
|
}
|
|
|
|
// nextShardID generates a new sequence id for a shard.
|
|
func (tx *metatx) nextShardID() (id uint64) {
|
|
return tx.mustNextSequence([]byte("shardID"))
|
|
}
|
|
|
|
// nextShardGroupID generates a new sequence id for a shard group.
|
|
func (tx *metatx) nextShardGroupID() (id uint64) {
|
|
return tx.mustNextSequence([]byte("shardGroupID"))
|
|
}
|
|
|
|
// dataNodes returns a list of all data nodes from the metastore.
|
|
func (tx *metatx) dataNodes() (a []*DataNode) {
|
|
c := tx.Bucket([]byte("DataNodes")).Cursor()
|
|
for k, v := c.First(); k != nil; k, v = c.Next() {
|
|
n := newDataNode()
|
|
mustUnmarshalJSON(v, &n)
|
|
a = append(a, n)
|
|
}
|
|
return
|
|
}
|
|
|
|
// nextDataNodeID returns a autoincrementing id.
|
|
func (tx *metatx) nextDataNodeID() uint64 {
|
|
id, _ := tx.Bucket([]byte("DataNodes")).NextSequence()
|
|
return id
|
|
}
|
|
|
|
// saveDataNode persists a data node to the metastore.
|
|
func (tx *metatx) saveDataNode(n *DataNode) error {
|
|
return tx.Bucket([]byte("DataNodes")).Put(u64tob(n.ID), mustMarshalJSON(n))
|
|
}
|
|
|
|
// deleteDataNode removes data node from the metastore.
|
|
func (tx *metatx) deleteDataNode(id uint64) error {
|
|
return tx.Bucket([]byte("DataNodes")).Delete(u64tob(id))
|
|
}
|
|
|
|
// databases returns a list of all databases from the metastore.
|
|
func (tx *metatx) databases() (a []*database) {
|
|
c := tx.Bucket([]byte("Databases")).Cursor()
|
|
for k, _ := c.First(); k != nil; k, _ = c.Next() {
|
|
b := c.Bucket().Bucket(k)
|
|
v := b.Get([]byte("meta"))
|
|
db := newDatabase()
|
|
mustUnmarshalJSON(v, &db)
|
|
a = append(a, db)
|
|
}
|
|
return
|
|
}
|
|
|
|
// saveDatabase persists a database to the metastore.
|
|
func (tx *metatx) saveDatabase(db *database) error {
|
|
b, err := tx.Bucket([]byte("Databases")).CreateBucketIfNotExists([]byte(db.name))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, _ = b.CreateBucketIfNotExists([]byte("TagBytesToID"))
|
|
_, _ = b.CreateBucketIfNotExists([]byte("Measurements"))
|
|
_, _ = b.CreateBucketIfNotExists([]byte("Series"))
|
|
return b.Put([]byte("meta"), mustMarshalJSON(db))
|
|
}
|
|
|
|
// dropDatabase removes database from the metastore.
|
|
func (tx *metatx) dropDatabase(name string) error {
|
|
return tx.Bucket([]byte("Databases")).DeleteBucket([]byte(name))
|
|
}
|
|
|
|
// dropMeasurement removes measurement from the metastore.
|
|
func (tx *metatx) dropMeasurement(database, measurement string) error {
|
|
return tx.Bucket([]byte("Databases")).Bucket([]byte(database)).Bucket([]byte("Series")).DeleteBucket([]byte(measurement))
|
|
}
|
|
|
|
// saveMeasurement persists a measurement to the metastore.
|
|
|
|
func (tx *metatx) saveMeasurement(database string, m *Measurement) error {
|
|
b := tx.Bucket([]byte("Databases")).Bucket([]byte(database)).Bucket([]byte("Measurements"))
|
|
return b.Put([]byte(m.Name), mustMarshalJSON(m))
|
|
}
|
|
|
|
// sets the series id for the database, name, and tags.
|
|
func (tx *metatx) createSeries(database, name string, tags map[string]string) (*Series, error) {
|
|
// create the buckets to store tag indexes for the series and give it a unique ID in the DB
|
|
db := tx.Bucket([]byte("Databases")).Bucket([]byte(database))
|
|
t := db.Bucket([]byte("Series"))
|
|
b, err := t.CreateBucketIfNotExists([]byte(name))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// give the series a unique ID
|
|
id, _ := t.NextSequence()
|
|
|
|
// store the tag map for the series
|
|
s := &Series{ID: uint64(id), Tags: tags}
|
|
idBytes := u64tob(uint64(id))
|
|
|
|
if err := b.Put(idBytes, mustMarshalJSON(s)); err != nil {
|
|
return nil, err
|
|
}
|
|
return s, nil
|
|
}
|
|
|
|
// dropSeries removes all seriesIDS for a given database/measurement
|
|
func (tx *metatx) dropSeries(database string, seriesByMeasurement map[string][]uint64) error {
|
|
for measurement, ids := range seriesByMeasurement {
|
|
b := tx.Bucket([]byte("Databases")).Bucket([]byte(database)).Bucket([]byte("Series")).Bucket([]byte(measurement))
|
|
if b != nil {
|
|
for _, id := range ids {
|
|
if err := b.Delete(u64tob(id)); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// loops through all the measurements and series in a database
|
|
func (tx *metatx) indexDatabase(db *database) {
|
|
// get the bucket that holds series data for the database
|
|
b := tx.Bucket([]byte("Databases")).Bucket([]byte(db.name))
|
|
|
|
// Iterate over and index series.
|
|
seriesBucket := b.Bucket([]byte("Series"))
|
|
c := seriesBucket.Cursor()
|
|
for k, _ := c.First(); k != nil; k, _ = c.Next() {
|
|
mc := seriesBucket.Bucket(k).Cursor()
|
|
name := string(k)
|
|
for id, v := mc.First(); id != nil; id, v = mc.Next() {
|
|
var s *Series
|
|
mustUnmarshalJSON(v, &s)
|
|
db.addSeriesToIndex(name, s)
|
|
}
|
|
}
|
|
|
|
// Iterate over measurement metadata.
|
|
c = b.Bucket([]byte("Measurements")).Cursor()
|
|
for k, v := c.First(); k != nil; k, v = c.Next() {
|
|
m := db.createMeasurementIfNotExists(string(k))
|
|
mustUnmarshalJSON(v, &m)
|
|
}
|
|
}
|
|
|
|
// user returns a user from the metastore by name.
|
|
func (tx *metatx) user(name string) (u *User) {
|
|
if v := tx.Bucket([]byte("Users")).Get([]byte(name)); v != nil {
|
|
mustUnmarshalJSON(v, &u)
|
|
}
|
|
return
|
|
}
|
|
|
|
// users returns a list of all users from the metastore.
|
|
func (tx *metatx) users() (a []*User) {
|
|
c := tx.Bucket([]byte("Users")).Cursor()
|
|
for k, v := c.First(); k != nil; k, v = c.Next() {
|
|
u := &User{}
|
|
mustUnmarshalJSON(v, &u)
|
|
a = append(a, u)
|
|
}
|
|
return
|
|
}
|
|
|
|
// saveUser persists a user to the metastore.
|
|
func (tx *metatx) saveUser(u *User) error {
|
|
return tx.Bucket([]byte("Users")).Put([]byte(u.Name), mustMarshalJSON(u))
|
|
}
|
|
|
|
// deleteUser removes the user from the metastore.
|
|
func (tx *metatx) deleteUser(name string) error {
|
|
return tx.Bucket([]byte("Users")).Delete([]byte(name))
|
|
}
|
|
|
|
// u64tob converts a uint64 into an 8-byte slice.
|
|
func u64tob(v uint64) []byte {
|
|
b := make([]byte, 8)
|
|
binary.BigEndian.PutUint64(b, v)
|
|
return b
|
|
}
|
|
|
|
// btou64 converts an 8-byte slice into an uint64.
|
|
func btou64(b []byte) uint64 { return binary.BigEndian.Uint64(b) }
|
|
|
|
// u32tob converts a uint32 into a 4-byte slice.
|
|
func u32tob(v uint32) []byte {
|
|
b := make([]byte, 4)
|
|
binary.BigEndian.PutUint32(b, v)
|
|
return b
|
|
}
|
|
|
|
// btou32 converts an 4-byte slice into an uint32.
|
|
func btou32(b []byte) uint32 { return binary.BigEndian.Uint32(b) }
|