mirror of
https://github.com/moby/moby.git
synced 2026-01-11 10:41:43 +00:00
Avoids extra allocations. Added in Go 1.24. Signed-off-by: Paweł Gronowski <pawel.gronowski@docker.com>
75 lines
1.9 KiB
Go
75 lines
1.9 KiB
Go
package testutils
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/moby/moby/api/types/events"
|
|
"github.com/moby/moby/v2/daemon/internal/lazyregexp"
|
|
"github.com/moby/moby/v2/daemon/internal/timestamp"
|
|
)
|
|
|
|
const (
|
|
reTimestamp = `(?P<timestamp>\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}.\d{9}(:?(:?(:?-|\+)\d{2}:\d{2})|Z))`
|
|
reEventType = `(?P<eventType>\w+)`
|
|
reAction = `(?P<action>\w+)`
|
|
reID = `(?P<id>[^\s]+)`
|
|
reAttributes = `(\s\((?P<attributes>[^\)]+)\))?`
|
|
)
|
|
|
|
// eventCliRegexp is a regular expression that matches all possible event outputs in the cli
|
|
var eventCliRegexp = lazyregexp.New(fmt.Sprintf(`\A%s\s%s\s%s\s%s%s\z`, reTimestamp, reEventType, reAction, reID, reAttributes))
|
|
|
|
// ScanMap turns an event string like the default ones formatted in the cli output
|
|
// and turns it into map.
|
|
func ScanMap(text string) map[string]string {
|
|
matches := eventCliRegexp.FindAllStringSubmatch(text, -1)
|
|
md := map[string]string{}
|
|
if len(matches) == 0 {
|
|
return md
|
|
}
|
|
|
|
names := eventCliRegexp.SubexpNames()
|
|
for i, n := range matches[0] {
|
|
md[names[i]] = n
|
|
}
|
|
return md
|
|
}
|
|
|
|
// Scan turns an event string like the default ones formatted in the cli output
|
|
// and turns it into an event message.
|
|
func Scan(text string) (*events.Message, error) {
|
|
md := ScanMap(text)
|
|
if len(md) == 0 {
|
|
return nil, fmt.Errorf("text is not an event: %s", text)
|
|
}
|
|
|
|
f, err := timestamp.GetTimestamp(md["timestamp"], time.Now())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
t, tn, err := timestamp.ParseTimestamps(f, -1)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
attrs := make(map[string]string)
|
|
for a := range strings.SplitSeq(md["attributes"], ", ") {
|
|
k, v, _ := strings.Cut(a, "=")
|
|
attrs[k] = v
|
|
}
|
|
|
|
return &events.Message{
|
|
Time: t,
|
|
TimeNano: time.Unix(t, tn).UnixNano(),
|
|
Type: events.Type(md["eventType"]),
|
|
Action: events.Action(md["action"]),
|
|
Actor: events.Actor{
|
|
ID: md["id"],
|
|
Attributes: attrs,
|
|
},
|
|
}, nil
|
|
}
|