Commit 25f762c7 authored by Jos Visser's avatar Jos Visser Committed by Russ Cox

expvar: add Float

R=r, rsc, dsymonds
CC=golang-dev
https://golang.org/cl/4044041
parent f49889d0
...@@ -38,7 +38,7 @@ type Var interface { ...@@ -38,7 +38,7 @@ type Var interface {
String() string String() string
} }
// Int is a 64-bit integer variable, and satisfies the Var interface. // Int is a 64-bit integer variable that satisfies the Var interface.
type Int struct { type Int struct {
i int64 i int64
mu sync.Mutex mu sync.Mutex
...@@ -58,7 +58,29 @@ func (v *Int) Set(value int64) { ...@@ -58,7 +58,29 @@ func (v *Int) Set(value int64) {
v.i = value v.i = value
} }
// Map is a string-to-Var map variable, and satisfies the Var interface. // Float is a 64-bit float variable that satisfies the Var interface.
type Float struct {
f float64
mu sync.Mutex
}
func (v *Float) String() string { return strconv.Ftoa64(v.f, 'g', -1) }
// Add adds delta to v.
func (v *Float) Add(delta float64) {
v.mu.Lock()
defer v.mu.Unlock()
v.f += delta
}
// Set sets v to value.
func (v *Float) Set(value float64) {
v.mu.Lock()
defer v.mu.Unlock()
v.f = value
}
// Map is a string-to-Var map variable that satisfies the Var interface.
type Map struct { type Map struct {
m map[string]Var m map[string]Var
mu sync.Mutex mu sync.Mutex
...@@ -119,6 +141,22 @@ func (v *Map) Add(key string, delta int64) { ...@@ -119,6 +141,22 @@ func (v *Map) Add(key string, delta int64) {
} }
} }
// AddFloat adds delta to the *Float value stored under the given map key.
func (v *Map) AddFloat(key string, delta float64) {
v.mu.Lock()
defer v.mu.Unlock()
av, ok := v.m[key]
if !ok {
av = new(Float)
v.m[key] = av
}
// Add to Float; ignore otherwise.
if iv, ok := av.(*Float); ok {
iv.Add(delta)
}
}
// TODO(rsc): Make sure map access in separate thread is safe. // TODO(rsc): Make sure map access in separate thread is safe.
func (v *Map) iterate(c chan<- KeyValue) { func (v *Map) iterate(c chan<- KeyValue) {
for k, v := range v.m { for k, v := range v.m {
...@@ -148,6 +186,12 @@ type IntFunc func() int64 ...@@ -148,6 +186,12 @@ type IntFunc func() int64
func (v IntFunc) String() string { return strconv.Itoa64(v()) } func (v IntFunc) String() string { return strconv.Itoa64(v()) }
// FloatFunc wraps a func() float64 to create a value that satisfies the Var interface.
// The function will be called each time the Var is evaluated.
type FloatFunc func() float64
func (v FloatFunc) String() string { return strconv.Ftoa64(v(), 'g', -1) }
// StringFunc wraps a func() string to create value that satisfies the Var interface. // StringFunc wraps a func() string to create value that satisfies the Var interface.
// The function will be called each time the Var is evaluated. // The function will be called each time the Var is evaluated.
type StringFunc func() string type StringFunc func() string
...@@ -192,6 +236,12 @@ func NewInt(name string) *Int { ...@@ -192,6 +236,12 @@ func NewInt(name string) *Int {
return v return v
} }
func NewFloat(name string) *Float {
v := new(Float)
Publish(name, v)
return v
}
func NewMap(name string) *Map { func NewMap(name string) *Map {
v := new(Map).Init() v := new(Map).Init()
Publish(name, v) Publish(name, v)
......
...@@ -34,6 +34,31 @@ func TestInt(t *testing.T) { ...@@ -34,6 +34,31 @@ func TestInt(t *testing.T) {
} }
} }
func TestFloat(t *testing.T) {
reqs := NewFloat("requests-float")
if reqs.f != 0.0 {
t.Errorf("reqs.f = %v, want 0", reqs.f)
}
if reqs != Get("requests-float").(*Float) {
t.Errorf("Get() failed.")
}
reqs.Add(1.5)
reqs.Add(1.25)
if reqs.f != 2.75 {
t.Errorf("reqs.f = %v, want 2.75", reqs.f)
}
if s := reqs.String(); s != "2.75" {
t.Errorf("reqs.String() = %q, want \"4.64\"", s)
}
reqs.Add(-2)
if reqs.f != 0.75 {
t.Errorf("reqs.f = %v, want 0.75", reqs.f)
}
}
func TestString(t *testing.T) { func TestString(t *testing.T) {
name := NewString("my-name") name := NewString("my-name")
if name.s != "" { if name.s != "" {
...@@ -56,12 +81,16 @@ func TestMapCounter(t *testing.T) { ...@@ -56,12 +81,16 @@ func TestMapCounter(t *testing.T) {
colours.Add("red", 1) colours.Add("red", 1)
colours.Add("red", 2) colours.Add("red", 2)
colours.Add("blue", 4) colours.Add("blue", 4)
colours.AddFloat("green", 4.125)
if x := colours.m["red"].(*Int).i; x != 3 { if x := colours.m["red"].(*Int).i; x != 3 {
t.Errorf("colours.m[\"red\"] = %v, want 3", x) t.Errorf("colours.m[\"red\"] = %v, want 3", x)
} }
if x := colours.m["blue"].(*Int).i; x != 4 { if x := colours.m["blue"].(*Int).i; x != 4 {
t.Errorf("colours.m[\"blue\"] = %v, want 4", x) t.Errorf("colours.m[\"blue\"] = %v, want 4", x)
} }
if x := colours.m["green"].(*Float).f; x != 4.125 {
t.Errorf("colours.m[\"green\"] = %v, want 3.14", x)
}
// colours.String() should be '{"red":3, "blue":4}', // colours.String() should be '{"red":3, "blue":4}',
// though the order of red and blue could vary. // though the order of red and blue could vary.
...@@ -98,6 +127,19 @@ func TestIntFunc(t *testing.T) { ...@@ -98,6 +127,19 @@ func TestIntFunc(t *testing.T) {
} }
} }
func TestFloatFunc(t *testing.T) {
x := float64(8.5)
ix := FloatFunc(func() float64 { return x })
if s := ix.String(); s != "8.5" {
t.Errorf("ix.String() = %v, want 3.14", s)
}
x -= 1.25
if s := ix.String(); s != "7.25" {
t.Errorf("ix.String() = %v, want 4.34", s)
}
}
func TestStringFunc(t *testing.T) { func TestStringFunc(t *testing.T) {
x := "hello" x := "hello"
sx := StringFunc(func() string { return x }) sx := StringFunc(func() string { return x })
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment