blob: d06dabfa4376ccbdd5bf80be9401d4f27d05460c [file] [log] [blame]
Robin Thellendac59e972014-08-19 18:26:11 -07001// Package stats implements a global repository of stats objects. Each object
2// has a name and a value.
3// Example:
4// bar1 := stats.NewInteger("foo/bar1")
5// bar2 := stats.NewFloat("foo/bar2")
6// bar3 := stats.NewCounter("foo/bar3")
7// bar1.Set(1)
8// bar2.Set(2)
9// bar3.Set(3)
10// The values can be retrieved with:
11// v, err := stats.Value("foo/bar1")
12package stats
13
14import (
15 "errors"
16 "strings"
17 "sync"
18 "time"
19)
20
21// StatsObject is the interface for objects stored in the stats repository.
22type StatsObject interface {
23 // LastUpdate is used by WatchGlob to decide which updates to send.
24 LastUpdate() time.Time
25 // Value returns the current value of the object.
26 Value() interface{}
27}
28
29type node struct {
30 object StatsObject
31 children map[string]*node
32}
33
34var (
35 lock sync.RWMutex
Robin Thellendfa70aaa2014-10-09 17:13:05 -070036 repository *node // GUARDED_BY(lock)
Robin Thellendac59e972014-08-19 18:26:11 -070037 ErrNotFound = errors.New("name not found")
38 ErrNoValue = errors.New("object has no value")
39)
40
41func init() {
42 repository = newNode()
43}
44
45// GetStatsObject returns the object with that given name, or an error if the
46// object doesn't exist.
47func GetStatsObject(name string) (StatsObject, error) {
48 lock.RLock()
49 defer lock.RUnlock()
50 node := findNodeLocked(name, false)
51 if node == nil || node.object == nil {
52 return nil, ErrNotFound
53 }
54 return node.object, nil
55}
56
57// Value returns the value of an object, or an error if the object doesn't
58// exist.
59func Value(name string) (interface{}, error) {
60 obj, err := GetStatsObject(name)
61 if err != nil {
62 return 0, err
63 }
64 if obj == nil {
65 return nil, ErrNoValue
66 }
67 return obj.Value(), nil
68}
69
Robin Thellenddf428232014-10-06 12:50:44 -070070// Delete deletes a StatsObject and all its children, if any.
71func Delete(name string) error {
72 if name == "" {
73 return ErrNotFound
74 }
75 elems := strings.Split(name, "/")
76 last := len(elems) - 1
77 dirname, basename := strings.Join(elems[:last], "/"), elems[last]
78 lock.Lock()
79 defer lock.Unlock()
80 parent := findNodeLocked(dirname, false)
81 if parent == nil {
82 return ErrNotFound
83 }
84 delete(parent.children, basename)
85 return nil
86}
87
Robin Thellendac59e972014-08-19 18:26:11 -070088func newNode() *node {
89 return &node{children: make(map[string]*node)}
90}
91
92// findNodeLocked finds a node, and optionally creates it if it doesn't already
93// exist.
94func findNodeLocked(name string, create bool) *node {
95 elems := strings.Split(name, "/")
96 node := repository
97 for {
98 if len(elems) == 0 {
99 return node
100 }
101 if len(elems[0]) == 0 {
102 elems = elems[1:]
103 continue
104 }
105 if next, ok := node.children[elems[0]]; ok {
106 node = next
107 elems = elems[1:]
108 continue
109 }
110 if create {
111 node.children[elems[0]] = newNode()
112 node = node.children[elems[0]]
113 elems = elems[1:]
114 continue
115 }
116 return nil
117 }
118}