123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187 |
- package report
- import (
- "bytes"
- "fmt"
- "sort"
- "time"
- "github.com/ugorji/go/codec"
- "github.com/weaveworks/ps"
- )
- // Helper functions for ps.Map, without considering what is inside
- // Return a new map containing all the elements of the two input maps
- // and where the same key is in both, pick 'b' where prefer(a,b) is true
- func mergeMaps(m, n ps.Map, prefer func(a, b interface{}) bool) ps.Map {
- switch {
- case m == nil:
- return n
- case n == nil:
- return m
- case m.Size() < n.Size():
- m, n = n, m
- }
- n.ForEach(func(key string, val interface{}) {
- if existingVal, found := m.Lookup(key); found {
- if prefer(existingVal, val) {
- m = m.Set(key, val)
- }
- } else {
- m = m.Set(key, val)
- }
- })
- return m
- }
- func mapEqual(m, n ps.Map, equalf func(a, b interface{}) bool) bool {
- var mSize, nSize int
- if m != nil {
- mSize = m.Size()
- }
- if n != nil {
- nSize = n.Size()
- }
- if mSize != nSize {
- return false
- }
- if mSize == 0 {
- return true
- }
- equal := true
- m.ForEach(func(k string, val interface{}) {
- if otherValue, ok := n.Lookup(k); !ok {
- equal = false
- } else {
- equal = equal && equalf(val, otherValue)
- }
- })
- return equal
- }
- // very similar to ps.Map.String() but with keys sorted
- func mapToString(m ps.Map) string {
- buf := bytes.NewBufferString("{")
- for _, key := range mapKeys(m) {
- val, _ := m.Lookup(key)
- fmt.Fprintf(buf, "%s: %s,\n", key, val)
- }
- fmt.Fprintf(buf, "}")
- return buf.String()
- }
- func mapKeys(m ps.Map) []string {
- if m == nil {
- return nil
- }
- keys := m.Keys()
- sort.Strings(keys)
- return keys
- }
- // constants from https://github.com/ugorji/go/blob/master/codec/helper.go#L207
- const (
- containerMapKey = 2
- containerMapValue = 3
- containerMapEnd = 4
- // from https://github.com/ugorji/go/blob/master/codec/helper.go#L152
- cUTF8 = 2
- )
- // This implementation does not use the intermediate form as that was a
- // performance issue; skipping it saved almost 10% CPU. Note this means
- // we are using undocumented, internal APIs, which could break in the future.
- // See https://github.com/weaveworks/scope/pull/1709 for more information.
- func mapRead(decoder *codec.Decoder, decodeValue func(isNil bool) interface{}) ps.Map {
- z, r := codec.GenHelperDecoder(decoder)
- if r.TryDecodeAsNil() {
- return ps.NewMap()
- }
- length := r.ReadMapStart()
- out := ps.NewMap()
- for i := 0; length < 0 || i < length; i++ {
- if length < 0 && r.CheckBreak() {
- break
- }
- var key string
- z.DecSendContainerState(containerMapKey)
- if !r.TryDecodeAsNil() {
- key = lookupCommonKey(r.DecodeStringAsBytes())
- }
- z.DecSendContainerState(containerMapValue)
- value := decodeValue(r.TryDecodeAsNil())
- out = out.UnsafeMutableSet(key, value)
- }
- z.DecSendContainerState(containerMapEnd)
- return out
- }
- // Inverse of mapRead, done for performance. Same comments about
- // undocumented internal APIs apply.
- func mapWrite(m ps.Map, encoder *codec.Encoder, encodeValue func(*codec.Encoder, interface{})) {
- z, r := codec.GenHelperEncoder(encoder)
- if m == nil || m.IsNil() {
- r.EncodeNil()
- return
- }
- r.EncodeMapStart(m.Size())
- m.ForEach(func(key string, val interface{}) {
- z.EncSendContainerState(containerMapKey)
- r.EncodeString(cUTF8, key)
- z.EncSendContainerState(containerMapValue)
- encodeValue(encoder, val)
- })
- z.EncSendContainerState(containerMapEnd)
- }
- // Now follow helpers for StringLatestMap
- // These let us sort a StringLatestMap strings by key
- func (m StringLatestMap) Len() int { return len(m) }
- func (m StringLatestMap) Swap(i, j int) { m[i], m[j] = m[j], m[i] }
- func (m StringLatestMap) Less(i, j int) bool { return m[i].key < m[j].key }
- // sort entries and shuffle down any duplicates. NOTE: may modify contents of m.
- func (m StringLatestMap) sortedAndDeduplicated() StringLatestMap {
- sort.Sort(m)
- for i := 1; i < len(m); {
- if m[i-1].key == m[i].key {
- if m[i-1].Timestamp.Before(m[i].Timestamp) {
- m = append(m[:i-1], m[i:]...)
- } else {
- m = append(m[:i], m[i+1:]...)
- }
- } else {
- i++
- }
- }
- return m
- }
- // add several entries at the same timestamp
- func (m StringLatestMap) addMapEntries(ts time.Time, n map[string]string) StringLatestMap {
- out := make(StringLatestMap, len(m), len(m)+len(n))
- copy(out, m)
- for k, v := range n {
- out = append(out, stringLatestEntry{key: k, Value: v, Timestamp: ts})
- }
- return out.sortedAndDeduplicated()
- }
- // Propagate a set of latest values from one set to another.
- func (m StringLatestMap) Propagate(from StringLatestMap, keys ...string) StringLatestMap {
- out := make(StringLatestMap, len(m), len(m)+len(keys))
- copy(out, m)
- for _, k := range keys {
- if v, ts, ok := from.LookupEntry(k); ok {
- out = append(out, stringLatestEntry{key: k, Value: v, Timestamp: ts})
- }
- }
- return out.sortedAndDeduplicated()
- }
|