2019-11-26 14:20:45 +00:00
|
|
|
// Package buffer provides a simple ring buffer for storing local data
|
|
|
|
package buffer
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
2019-11-27 13:57:19 +00:00
|
|
|
"time"
|
2019-11-30 12:39:29 +00:00
|
|
|
|
|
|
|
"github.com/google/uuid"
|
2019-11-26 14:20:45 +00:00
|
|
|
)
|
|
|
|
|
2019-11-30 12:39:29 +00:00
|
|
|
type stream struct {
|
|
|
|
id string
|
|
|
|
entries chan *Entry
|
|
|
|
stop chan bool
|
|
|
|
}
|
|
|
|
|
2019-11-28 18:08:48 +00:00
|
|
|
// Buffer is ring buffer
|
2019-11-26 14:20:45 +00:00
|
|
|
type Buffer struct {
|
|
|
|
size int
|
|
|
|
sync.RWMutex
|
2019-11-30 12:39:29 +00:00
|
|
|
vals []*Entry
|
|
|
|
streams map[string]stream
|
2019-11-27 13:57:19 +00:00
|
|
|
}
|
|
|
|
|
2019-11-28 18:08:48 +00:00
|
|
|
// Entry is ring buffer data entry
|
2019-11-27 13:57:19 +00:00
|
|
|
type Entry struct {
|
|
|
|
Value interface{}
|
|
|
|
Timestamp time.Time
|
2019-11-26 14:20:45 +00:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:16 +00:00
|
|
|
// New returns a new buffer of the given size
|
|
|
|
func New(i int) *Buffer {
|
|
|
|
return &Buffer{
|
2019-11-30 12:39:29 +00:00
|
|
|
size: i,
|
|
|
|
streams: make(map[string]stream),
|
2019-11-27 16:02:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Put adds a new value to ring buffer
|
2019-11-26 14:20:45 +00:00
|
|
|
func (b *Buffer) Put(v interface{}) {
|
|
|
|
b.Lock()
|
|
|
|
defer b.Unlock()
|
|
|
|
|
|
|
|
// append to values
|
2019-11-30 12:39:29 +00:00
|
|
|
entry := &Entry{
|
2019-11-27 13:57:19 +00:00
|
|
|
Value: v,
|
|
|
|
Timestamp: time.Now(),
|
2019-11-30 12:39:29 +00:00
|
|
|
}
|
|
|
|
b.vals = append(b.vals, entry)
|
2019-11-26 14:20:45 +00:00
|
|
|
|
|
|
|
// trim if bigger than size required
|
|
|
|
if len(b.vals) > b.size {
|
|
|
|
b.vals = b.vals[1:]
|
|
|
|
}
|
2019-11-30 12:39:29 +00:00
|
|
|
|
|
|
|
// TODO: this is fucking ugly
|
|
|
|
for _, stream := range b.streams {
|
|
|
|
select {
|
|
|
|
case <-stream.stop:
|
|
|
|
delete(b.streams, stream.id)
|
|
|
|
close(stream.entries)
|
|
|
|
case stream.entries <- entry:
|
|
|
|
}
|
|
|
|
}
|
2019-11-26 14:20:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get returns the last n entries
|
2019-11-27 13:57:19 +00:00
|
|
|
func (b *Buffer) Get(n int) []*Entry {
|
2019-11-28 18:08:48 +00:00
|
|
|
b.RLock()
|
|
|
|
defer b.RUnlock()
|
|
|
|
|
2019-11-26 14:20:45 +00:00
|
|
|
// reset any invalid values
|
|
|
|
if n > b.size || n < 0 {
|
|
|
|
n = b.size
|
|
|
|
}
|
|
|
|
|
|
|
|
// create a delta
|
|
|
|
delta := b.size - n
|
|
|
|
|
|
|
|
// if all the values are less than delta
|
|
|
|
if len(b.vals) < delta {
|
|
|
|
return b.vals
|
|
|
|
}
|
|
|
|
|
|
|
|
// return the delta set
|
|
|
|
return b.vals[delta:]
|
|
|
|
}
|
|
|
|
|
2019-11-27 13:57:19 +00:00
|
|
|
// Return the entries since a specific time
|
|
|
|
func (b *Buffer) Since(t time.Time) []*Entry {
|
|
|
|
b.RLock()
|
|
|
|
defer b.RUnlock()
|
|
|
|
|
|
|
|
// return all the values
|
|
|
|
if t.IsZero() {
|
|
|
|
return b.vals
|
|
|
|
}
|
|
|
|
|
|
|
|
// if its in the future return nothing
|
|
|
|
if time.Since(t).Seconds() < 0.0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, v := range b.vals {
|
|
|
|
// find the starting point
|
|
|
|
d := v.Timestamp.Sub(t)
|
|
|
|
|
|
|
|
// return the values
|
|
|
|
if d.Seconds() > 0.0 {
|
|
|
|
return b.vals[i:]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-30 12:39:29 +00:00
|
|
|
// Stream logs from the buffer
|
|
|
|
func (b *Buffer) Stream(stop chan bool) <-chan *Entry {
|
|
|
|
b.Lock()
|
|
|
|
defer b.Unlock()
|
|
|
|
|
|
|
|
entries := make(chan *Entry, 128)
|
|
|
|
id := uuid.New().String()
|
|
|
|
b.streams[id] = stream{
|
|
|
|
id: id,
|
|
|
|
entries: entries,
|
|
|
|
stop: stop,
|
|
|
|
}
|
|
|
|
|
|
|
|
return entries
|
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:16 +00:00
|
|
|
// Size returns the size of the ring buffer
|
2019-11-26 14:20:45 +00:00
|
|
|
func (b *Buffer) Size() int {
|
|
|
|
return b.size
|
|
|
|
}
|