2014-11-04 04:15:58 +00:00
|
|
|
package influxdb
|
|
|
|
|
|
|
|
import (
|
2015-01-10 15:48:50 +00:00
|
|
|
"encoding/binary"
|
2014-11-10 02:55:53 +00:00
|
|
|
"errors"
|
2014-11-04 04:15:58 +00:00
|
|
|
"fmt"
|
2015-04-23 21:23:00 +00:00
|
|
|
"os"
|
2015-03-14 20:47:20 +00:00
|
|
|
"sync"
|
2014-11-04 04:15:58 +00:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/boltdb/bolt"
|
2015-03-14 20:47:20 +00:00
|
|
|
"github.com/influxdb/influxdb/messaging"
|
2014-11-04 04:15:58 +00:00
|
|
|
)
|
|
|
|
|
2015-01-10 15:48:50 +00:00
|
|
|
// ShardGroup represents a group of shards created for a single time range.
|
|
|
|
type ShardGroup struct {
|
2014-11-10 02:55:53 +00:00
|
|
|
ID uint64 `json:"id,omitempty"`
|
2014-11-06 06:03:35 +00:00
|
|
|
StartTime time.Time `json:"startTime,omitempty"`
|
|
|
|
EndTime time.Time `json:"endTime,omitempty"`
|
2015-01-10 15:48:50 +00:00
|
|
|
Shards []*Shard `json:"shards,omitempty"`
|
|
|
|
}
|
|
|
|
|
2015-03-09 21:47:41 +00:00
|
|
|
// newShardGroup returns a new initialized ShardGroup instance.
|
2015-04-23 21:23:00 +00:00
|
|
|
func newShardGroup(t time.Time, d time.Duration) *ShardGroup {
|
|
|
|
sg := ShardGroup{}
|
|
|
|
sg.StartTime = t.Truncate(d).UTC()
|
|
|
|
sg.EndTime = sg.StartTime.Add(d).UTC()
|
2015-03-09 21:47:41 +00:00
|
|
|
|
2015-04-23 21:23:00 +00:00
|
|
|
return &sg
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sg *ShardGroup) initialize(index uint64, shardN, replicaN int, db *database, rp *RetentionPolicy, nodes []*DataNode, meta *metastore) error {
|
|
|
|
sg.Shards = make([]*Shard, shardN)
|
|
|
|
|
|
|
|
// Persist to metastore if a shard was created.
|
|
|
|
return meta.mustUpdate(index, func(tx *metatx) error {
|
|
|
|
// Generate an ID for the group.
|
|
|
|
sg.ID = tx.nextShardGroupID()
|
|
|
|
|
|
|
|
// Generate an ID for each shard.
|
|
|
|
for i := range sg.Shards {
|
|
|
|
sg.Shards[i] = newShard(tx.nextShardID())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assign data nodes to shards via round robin.
|
|
|
|
// Start from a repeatably "random" place in the node list.
|
|
|
|
nodeIndex := int(index % uint64(len(nodes)))
|
|
|
|
for _, sh := range sg.Shards {
|
|
|
|
for i := 0; i < replicaN; i++ {
|
|
|
|
node := nodes[nodeIndex%len(nodes)]
|
|
|
|
sh.DataNodeIDs = append(sh.DataNodeIDs, node.ID)
|
|
|
|
nodeIndex++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Retention policy has a new shard group, so update the policy.
|
|
|
|
rp.shardGroups = append(rp.shardGroups, sg)
|
|
|
|
|
|
|
|
return tx.saveDatabase(db)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sg *ShardGroup) close(id uint64) error {
|
|
|
|
for _, shard := range sg.Shards {
|
|
|
|
// Ignore shards not on this server.
|
|
|
|
if !shard.HasDataNodeID(id) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
path := shard.store.Path()
|
|
|
|
shard.close()
|
|
|
|
if err := os.Remove(path); err != nil {
|
|
|
|
return fmt.Errorf("shard id %d, path %s : %s", shard.ID, path, err)
|
|
|
|
}
|
2015-01-10 15:48:50 +00:00
|
|
|
}
|
2015-04-23 21:23:00 +00:00
|
|
|
return nil
|
2015-01-10 15:48:50 +00:00
|
|
|
}
|
2014-11-06 06:03:35 +00:00
|
|
|
|
2015-01-14 23:44:09 +00:00
|
|
|
// ShardBySeriesID returns the shard that a series is assigned to in the group.
|
2015-04-23 21:23:00 +00:00
|
|
|
func (sg *ShardGroup) ShardBySeriesID(seriesID uint64) *Shard {
|
|
|
|
return sg.Shards[int(seriesID)%len(sg.Shards)]
|
2015-01-14 23:44:09 +00:00
|
|
|
}
|
|
|
|
|
2015-01-10 15:48:50 +00:00
|
|
|
// Duration returns the duration between the shard group's start and end time.
|
2015-04-23 21:23:00 +00:00
|
|
|
func (sg *ShardGroup) Duration() time.Duration { return sg.EndTime.Sub(sg.StartTime) }
|
2015-01-10 15:48:50 +00:00
|
|
|
|
2015-05-19 16:35:44 +00:00
|
|
|
// Contains return whether the shard group contains data for the timestamp
|
|
|
|
func (sg *ShardGroup) Contains(timestamp time.Time) bool {
|
|
|
|
return !sg.StartTime.After(timestamp) && sg.EndTime.After(timestamp)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Overlaps return whether the shard group contains data for the time range between min and max
|
|
|
|
func (sg *ShardGroup) Overlaps(min, max time.Time) bool {
|
|
|
|
return !sg.StartTime.After(max) && sg.EndTime.After(min)
|
2015-02-27 21:34:35 +00:00
|
|
|
}
|
|
|
|
|
2015-02-21 01:36:15 +00:00
|
|
|
// dropSeries will delete all data with the seriesID
|
2015-04-23 21:23:00 +00:00
|
|
|
func (sg *ShardGroup) dropSeries(seriesIDs ...uint64) error {
|
|
|
|
for _, s := range sg.Shards {
|
2015-03-17 17:44:08 +00:00
|
|
|
err := s.dropSeries(seriesIDs...)
|
2015-02-20 17:58:21 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-04-22 04:45:55 +00:00
|
|
|
// shardtx represents a shard transaction.
|
|
|
|
type shardtx struct {
|
|
|
|
*bolt.Tx
|
|
|
|
}
|
|
|
|
|
2015-03-09 21:47:41 +00:00
|
|
|
// Shard represents the logical storage for a given time range.
|
|
|
|
// The instance on a local server may contain the raw data in "store" if the
|
|
|
|
// shard is assigned to the server's data node id.
|
|
|
|
type Shard struct {
|
|
|
|
ID uint64 `json:"id,omitempty"`
|
|
|
|
DataNodeIDs []uint64 `json:"nodeIDs,omitempty"` // owners
|
|
|
|
|
2015-03-30 20:44:28 +00:00
|
|
|
mu sync.RWMutex
|
2015-03-09 21:47:41 +00:00
|
|
|
index uint64 // highest replicated index
|
|
|
|
store *bolt.DB // underlying data store
|
|
|
|
conn MessagingConn // streaming connection to broker
|
2015-03-14 20:47:20 +00:00
|
|
|
|
2015-03-18 22:02:25 +00:00
|
|
|
stats *Stats // In-memory stats
|
|
|
|
|
2015-03-14 20:47:20 +00:00
|
|
|
wg sync.WaitGroup // pending goroutines
|
|
|
|
closing chan struct{} // close notification
|
2015-03-09 21:47:41 +00:00
|
|
|
}
|
|
|
|
|
2014-11-10 02:55:53 +00:00
|
|
|
// newShard returns a new initialized Shard instance.
|
2015-04-23 21:23:00 +00:00
|
|
|
func newShard(id uint64) *Shard {
|
|
|
|
s := &Shard{}
|
|
|
|
s.ID = id
|
|
|
|
s.stats = NewStats(fmt.Sprintf("shard %d", id))
|
|
|
|
s.stats.Inc("initialize")
|
|
|
|
|
|
|
|
return s
|
|
|
|
}
|
2014-11-10 02:55:53 +00:00
|
|
|
|
|
|
|
// open initializes and opens the shard's store.
|
2015-03-09 21:47:41 +00:00
|
|
|
func (s *Shard) open(path string, conn MessagingConn) error {
|
2015-03-30 20:44:28 +00:00
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
2015-04-22 21:37:59 +00:00
|
|
|
if s.stats == nil {
|
2015-04-22 23:14:37 +00:00
|
|
|
s.stats = NewStats(fmt.Sprintf("shard %d", s.ID))
|
2015-04-22 21:37:59 +00:00
|
|
|
}
|
2015-04-22 22:03:37 +00:00
|
|
|
s.stats.Inc("open")
|
2015-04-22 21:37:59 +00:00
|
|
|
|
2014-11-10 02:55:53 +00:00
|
|
|
// Return an error if the shard is already open.
|
|
|
|
if s.store != nil {
|
2015-04-22 21:37:59 +00:00
|
|
|
s.stats.Inc("errAlreadyOpen")
|
2014-11-10 02:55:53 +00:00
|
|
|
return errors.New("shard already open")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Open store on shard.
|
2015-03-14 19:49:25 +00:00
|
|
|
store, err := bolt.Open(path, 0666, &bolt.Options{Timeout: 1 * time.Second})
|
2014-11-10 02:55:53 +00:00
|
|
|
if err != nil {
|
2015-04-22 21:37:59 +00:00
|
|
|
s.stats.Inc("errBoltOpenFailure")
|
2014-11-10 02:55:53 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
s.store = store
|
|
|
|
|
|
|
|
// Initialize store.
|
2015-01-10 15:48:50 +00:00
|
|
|
if err := s.store.Update(func(tx *bolt.Tx) error {
|
2015-03-20 04:23:52 +00:00
|
|
|
_, _ = tx.CreateBucketIfNotExists([]byte("meta"))
|
2015-01-10 15:48:50 +00:00
|
|
|
_, _ = tx.CreateBucketIfNotExists([]byte("values"))
|
2015-03-09 21:47:41 +00:00
|
|
|
|
|
|
|
// Find highest replicated index.
|
2015-03-20 04:23:52 +00:00
|
|
|
s.index = shardMetaIndex(tx)
|
2015-03-09 21:47:41 +00:00
|
|
|
|
2015-03-14 20:47:20 +00:00
|
|
|
// Open connection.
|
|
|
|
if err := conn.Open(s.index, true); err != nil {
|
|
|
|
return fmt.Errorf("open shard conn: id=%d, idx=%d, err=%s", s.ID, s.index, err)
|
|
|
|
}
|
|
|
|
|
2015-01-10 15:48:50 +00:00
|
|
|
return nil
|
|
|
|
}); err != nil {
|
2015-04-22 21:37:59 +00:00
|
|
|
s.stats.Inc("errBoltStoreUpdateFailure")
|
2014-11-10 02:55:53 +00:00
|
|
|
_ = s.close()
|
|
|
|
return fmt.Errorf("init: %s", err)
|
|
|
|
}
|
|
|
|
|
2015-03-09 21:47:41 +00:00
|
|
|
// Start importing from connection.
|
2015-03-14 20:47:20 +00:00
|
|
|
s.closing = make(chan struct{})
|
|
|
|
s.wg.Add(1)
|
|
|
|
go s.processor(conn, s.closing)
|
2015-03-09 21:47:41 +00:00
|
|
|
|
2014-11-10 02:55:53 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-03-20 04:23:52 +00:00
|
|
|
// shardMetaIndex returns the index from the "meta" bucket on a transaction.
|
|
|
|
func shardMetaIndex(tx *bolt.Tx) uint64 {
|
|
|
|
var index uint64
|
|
|
|
if buf := tx.Bucket([]byte("meta")).Get([]byte("index")); len(buf) > 0 {
|
|
|
|
index = btou64(buf)
|
|
|
|
}
|
|
|
|
return index
|
|
|
|
}
|
|
|
|
|
2015-04-22 04:45:55 +00:00
|
|
|
// view executes a function in the context of a read-only transaction.
|
|
|
|
func (s *Shard) view(fn func(*shardtx) error) error {
|
|
|
|
return s.store.View(func(tx *bolt.Tx) error { return fn(&shardtx{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 (s *Shard) mustView(fn func(*shardtx) error) (err error) {
|
|
|
|
if e := s.view(func(tx *shardtx) error {
|
|
|
|
err = fn(tx)
|
|
|
|
return nil
|
|
|
|
}); e != nil {
|
|
|
|
panic("shard view: " + e.Error())
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-04-20 20:48:23 +00:00
|
|
|
// Index returns the highest Raft index processed by this shard. Shard RLock
|
|
|
|
// held during execution.
|
|
|
|
func (s *Shard) Index() uint64 {
|
|
|
|
s.mu.RLock()
|
|
|
|
defer s.mu.RUnlock()
|
|
|
|
return s.index
|
|
|
|
}
|
|
|
|
|
2014-11-10 02:55:53 +00:00
|
|
|
// close shuts down the shard's store.
|
|
|
|
func (s *Shard) close() error {
|
2015-03-14 20:47:20 +00:00
|
|
|
// Wait for goroutines to stop.
|
|
|
|
if s.closing != nil {
|
|
|
|
close(s.closing)
|
|
|
|
s.closing = nil
|
|
|
|
}
|
|
|
|
|
|
|
|
s.wg.Wait()
|
|
|
|
|
2015-03-09 21:47:41 +00:00
|
|
|
if s.store != nil {
|
|
|
|
_ = s.store.Close()
|
2015-01-10 15:48:50 +00:00
|
|
|
}
|
2015-04-29 15:29:04 +00:00
|
|
|
if s.stats != nil {
|
|
|
|
s.stats.Inc("close")
|
|
|
|
}
|
2015-03-09 21:47:41 +00:00
|
|
|
return nil
|
2014-11-10 02:55:53 +00:00
|
|
|
}
|
|
|
|
|
2015-03-22 20:58:40 +00:00
|
|
|
// sync returns after a given index has been reached.
|
|
|
|
func (s *Shard) sync(index uint64) error {
|
|
|
|
for {
|
|
|
|
// Check if index has occurred.
|
2015-03-30 20:44:28 +00:00
|
|
|
s.mu.RLock()
|
|
|
|
i := s.index
|
|
|
|
s.mu.RUnlock()
|
|
|
|
if i >= index {
|
2015-03-22 20:58:40 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise wait momentarily and check again.
|
|
|
|
time.Sleep(1 * time.Millisecond)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-10 15:48:50 +00:00
|
|
|
// HasDataNodeID return true if the data node owns the shard.
|
|
|
|
func (s *Shard) HasDataNodeID(id uint64) bool {
|
|
|
|
for _, dataNodeID := range s.DataNodeIDs {
|
|
|
|
if dataNodeID == id {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2015-01-10 20:22:57 +00:00
|
|
|
// readSeries reads encoded series data from a shard.
|
2015-04-08 22:46:56 +00:00
|
|
|
func (s *Shard) readSeries(seriesID uint64, timestamp int64) (values []byte, err error) {
|
2015-01-10 20:22:57 +00:00
|
|
|
err = s.store.View(func(tx *bolt.Tx) error {
|
|
|
|
// Find series bucket.
|
2015-04-08 22:46:56 +00:00
|
|
|
b := tx.Bucket(u64tob(seriesID))
|
2015-01-10 20:22:57 +00:00
|
|
|
if b == nil {
|
|
|
|
return nil
|
|
|
|
}
|
2014-11-30 23:00:50 +00:00
|
|
|
|
2015-01-10 20:22:57 +00:00
|
|
|
// Retrieve encoded series data.
|
|
|
|
values = b.Get(u64tob(uint64(timestamp)))
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-02-20 08:45:35 +00:00
|
|
|
// writeSeries writes series batch to a shard.
|
2015-03-09 21:47:41 +00:00
|
|
|
func (s *Shard) writeSeries(index uint64, batch []byte) error {
|
2014-11-04 04:15:58 +00:00
|
|
|
return s.store.Update(func(tx *bolt.Tx) error {
|
2015-02-20 08:45:35 +00:00
|
|
|
for {
|
|
|
|
if pointHeaderSize > len(batch) {
|
|
|
|
return ErrInvalidPointBuffer
|
|
|
|
}
|
|
|
|
seriesID, payloadLength, timestamp := unmarshalPointHeader(batch[:pointHeaderSize])
|
|
|
|
batch = batch[pointHeaderSize:]
|
|
|
|
|
|
|
|
if payloadLength > uint32(len(batch)) {
|
|
|
|
return ErrInvalidPointBuffer
|
|
|
|
}
|
|
|
|
data := batch[:payloadLength]
|
|
|
|
|
|
|
|
// Create a bucket for the series.
|
2015-04-08 22:46:56 +00:00
|
|
|
b, err := tx.CreateBucketIfNotExists(u64tob(seriesID))
|
2015-02-20 08:45:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert the values by timestamp.
|
|
|
|
if err := b.Put(u64tob(uint64(timestamp)), data); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-03-18 22:02:25 +00:00
|
|
|
s.stats.Add("shardBytes", int64(len(data))+8) // Payload plus timestamp
|
|
|
|
s.stats.Inc("shardWrite")
|
2015-02-20 08:45:35 +00:00
|
|
|
|
|
|
|
// Push the buffer forward and check if we're done.
|
|
|
|
batch = batch[payloadLength:]
|
|
|
|
if len(batch) == 0 {
|
|
|
|
break
|
|
|
|
}
|
2015-01-10 15:48:50 +00:00
|
|
|
}
|
|
|
|
|
2015-03-09 21:47:41 +00:00
|
|
|
// Set index.
|
|
|
|
if err := tx.Bucket([]byte("meta")).Put([]byte("index"), u64tob(index)); err != nil {
|
|
|
|
return fmt.Errorf("write shard index: %s", err)
|
|
|
|
}
|
|
|
|
|
2015-01-10 15:48:50 +00:00
|
|
|
return nil
|
2014-11-04 04:15:58 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2015-04-08 22:46:56 +00:00
|
|
|
func (s *Shard) dropSeries(seriesIDs ...uint64) error {
|
2015-02-22 05:32:07 +00:00
|
|
|
if s.store == nil {
|
|
|
|
return nil
|
|
|
|
}
|
2015-02-20 17:58:21 +00:00
|
|
|
return s.store.Update(func(tx *bolt.Tx) error {
|
2015-03-17 17:44:08 +00:00
|
|
|
for _, seriesID := range seriesIDs {
|
2015-04-08 22:46:56 +00:00
|
|
|
err := tx.DeleteBucket(u64tob(seriesID))
|
2015-03-17 17:44:08 +00:00
|
|
|
if err != bolt.ErrBucketNotFound {
|
|
|
|
return err
|
|
|
|
}
|
2015-02-26 23:48:07 +00:00
|
|
|
}
|
|
|
|
return nil
|
2015-02-20 17:58:21 +00:00
|
|
|
})
|
2014-11-04 04:15:58 +00:00
|
|
|
}
|
|
|
|
|
2015-03-09 21:47:41 +00:00
|
|
|
// processor runs in a separate goroutine and processes all incoming broker messages.
|
2015-03-14 20:47:20 +00:00
|
|
|
func (s *Shard) processor(conn MessagingConn, closing <-chan struct{}) {
|
|
|
|
defer s.wg.Done()
|
|
|
|
|
2015-03-09 21:47:41 +00:00
|
|
|
for {
|
|
|
|
// Read incoming message.
|
2015-03-14 20:47:20 +00:00
|
|
|
// Exit if the connection has been closed or if shard is closing.
|
|
|
|
var ok bool
|
|
|
|
var m *messaging.Message
|
|
|
|
select {
|
|
|
|
case m, ok = <-conn.C():
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
case <-closing:
|
2015-03-09 21:47:41 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ignore any writes that are from an old index.
|
2015-03-30 20:44:28 +00:00
|
|
|
s.mu.RLock()
|
|
|
|
i := s.index
|
|
|
|
s.mu.RUnlock()
|
|
|
|
if m.Index < i {
|
2015-03-09 21:47:41 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle write series separately so we don't lock server during shard writes.
|
|
|
|
switch m.Type {
|
|
|
|
case writeRawSeriesMessageType:
|
2015-03-18 22:02:25 +00:00
|
|
|
s.stats.Inc("writeSeriesMessageRx")
|
2015-03-09 21:47:41 +00:00
|
|
|
if err := s.writeSeries(m.Index, m.Data); err != nil {
|
|
|
|
panic(fmt.Errorf("apply shard: id=%d, idx=%d, err=%s", s.ID, m.Index, err))
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
panic(fmt.Sprintf("invalid shard message type: %d", m.Type))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Track last index.
|
2015-03-30 20:44:28 +00:00
|
|
|
s.mu.Lock()
|
2015-03-09 21:47:41 +00:00
|
|
|
s.index = m.Index
|
2015-03-30 20:44:28 +00:00
|
|
|
s.mu.Unlock()
|
2015-05-11 22:26:56 +00:00
|
|
|
|
|
|
|
// Update the connection with the high index.
|
|
|
|
conn.SetIndex(m.Index)
|
2015-03-09 21:47:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-07 01:18:36 +00:00
|
|
|
// Shards represents a list of shards.
|
|
|
|
type Shards []*Shard
|
|
|
|
|
2015-01-10 15:48:50 +00:00
|
|
|
// pointHeaderSize represents the size of a point header, in bytes.
|
2015-04-08 22:46:56 +00:00
|
|
|
const pointHeaderSize = 8 + 4 + 8 // seriesID + payload length + timestamp
|
2015-01-10 15:48:50 +00:00
|
|
|
|
2015-02-19 23:21:51 +00:00
|
|
|
// marshalPointHeader encodes a series id, payload length, timestamp, & flagset into a byte slice.
|
2015-04-08 22:46:56 +00:00
|
|
|
func marshalPointHeader(seriesID uint64, payloadLength uint32, timestamp int64) []byte {
|
2015-02-19 23:21:51 +00:00
|
|
|
b := make([]byte, pointHeaderSize)
|
2015-04-08 22:46:56 +00:00
|
|
|
binary.BigEndian.PutUint64(b[0:8], seriesID)
|
|
|
|
binary.BigEndian.PutUint32(b[8:12], payloadLength)
|
|
|
|
binary.BigEndian.PutUint64(b[12:20], uint64(timestamp))
|
2015-01-10 15:48:50 +00:00
|
|
|
return b
|
2014-11-07 01:18:36 +00:00
|
|
|
}
|
2014-12-23 06:18:05 +00:00
|
|
|
|
2015-01-10 15:48:50 +00:00
|
|
|
// unmarshalPointHeader decodes a byte slice into a series id, timestamp & flagset.
|
2015-04-08 22:46:56 +00:00
|
|
|
func unmarshalPointHeader(b []byte) (seriesID uint64, payloadLength uint32, timestamp int64) {
|
|
|
|
seriesID = binary.BigEndian.Uint64(b[0:8])
|
|
|
|
payloadLength = binary.BigEndian.Uint32(b[8:12])
|
|
|
|
timestamp = int64(binary.BigEndian.Uint64(b[12:20]))
|
2015-01-10 15:48:50 +00:00
|
|
|
return
|
|
|
|
}
|
2014-12-23 06:18:05 +00:00
|
|
|
|
2015-01-10 15:48:50 +00:00
|
|
|
type uint8Slice []uint8
|
2014-12-23 06:18:05 +00:00
|
|
|
|
2015-01-10 15:48:50 +00:00
|
|
|
func (p uint8Slice) Len() int { return len(p) }
|
|
|
|
func (p uint8Slice) Less(i, j int) bool { return p[i] < p[j] }
|
|
|
|
func (p uint8Slice) Swap(i, j int) { p[i], p[j] = p[j], p[i] }
|