123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322 |
- package daemon
- import (
- "context"
- "strconv"
- "strings"
- "time"
- "github.com/Sirupsen/logrus"
- "github.com/docker/docker/api/types/events"
- "github.com/docker/docker/api/types/filters"
- "github.com/docker/docker/container"
- daemonevents "github.com/docker/docker/daemon/events"
- "github.com/docker/libnetwork"
- swarmapi "github.com/docker/swarmkit/api"
- gogotypes "github.com/gogo/protobuf/types"
- )
- var (
- clusterEventAction = map[swarmapi.WatchActionKind]string{
- swarmapi.WatchActionKindCreate: "create",
- swarmapi.WatchActionKindUpdate: "update",
- swarmapi.WatchActionKindRemove: "remove",
- }
- )
- // LogContainerEvent generates an event related to a container with only the default attributes.
- func (daemon *Daemon) LogContainerEvent(container *container.Container, action string) {
- daemon.LogContainerEventWithAttributes(container, action, map[string]string{})
- }
- // LogContainerEventWithAttributes generates an event related to a container with specific given attributes.
- func (daemon *Daemon) LogContainerEventWithAttributes(container *container.Container, action string, attributes map[string]string) {
- copyAttributes(attributes, container.Config.Labels)
- if container.Config.Image != "" {
- attributes["image"] = container.Config.Image
- }
- attributes["name"] = strings.TrimLeft(container.Name, "/")
- actor := events.Actor{
- ID: container.ID,
- Attributes: attributes,
- }
- daemon.EventsService.Log(action, events.ContainerEventType, actor)
- }
- // LogImageEvent generates an event related to an image with only the default attributes.
- func (daemon *Daemon) LogImageEvent(imageID, refName, action string) {
- daemon.LogImageEventWithAttributes(imageID, refName, action, map[string]string{})
- }
- // LogImageEventWithAttributes generates an event related to an image with specific given attributes.
- func (daemon *Daemon) LogImageEventWithAttributes(imageID, refName, action string, attributes map[string]string) {
- img, err := daemon.GetImage(imageID)
- if err == nil && img.Config != nil {
- // image has not been removed yet.
- // it could be missing if the event is `delete`.
- copyAttributes(attributes, img.Config.Labels)
- }
- if refName != "" {
- attributes["name"] = refName
- }
- actor := events.Actor{
- ID: imageID,
- Attributes: attributes,
- }
- daemon.EventsService.Log(action, events.ImageEventType, actor)
- }
- // LogPluginEvent generates an event related to a plugin with only the default attributes.
- func (daemon *Daemon) LogPluginEvent(pluginID, refName, action string) {
- daemon.LogPluginEventWithAttributes(pluginID, refName, action, map[string]string{})
- }
- // LogPluginEventWithAttributes generates an event related to a plugin with specific given attributes.
- func (daemon *Daemon) LogPluginEventWithAttributes(pluginID, refName, action string, attributes map[string]string) {
- attributes["name"] = refName
- actor := events.Actor{
- ID: pluginID,
- Attributes: attributes,
- }
- daemon.EventsService.Log(action, events.PluginEventType, actor)
- }
- // LogVolumeEvent generates an event related to a volume.
- func (daemon *Daemon) LogVolumeEvent(volumeID, action string, attributes map[string]string) {
- actor := events.Actor{
- ID: volumeID,
- Attributes: attributes,
- }
- daemon.EventsService.Log(action, events.VolumeEventType, actor)
- }
- // LogNetworkEvent generates an event related to a network with only the default attributes.
- func (daemon *Daemon) LogNetworkEvent(nw libnetwork.Network, action string) {
- daemon.LogNetworkEventWithAttributes(nw, action, map[string]string{})
- }
- // LogNetworkEventWithAttributes generates an event related to a network with specific given attributes.
- func (daemon *Daemon) LogNetworkEventWithAttributes(nw libnetwork.Network, action string, attributes map[string]string) {
- attributes["name"] = nw.Name()
- attributes["type"] = nw.Type()
- actor := events.Actor{
- ID: nw.ID(),
- Attributes: attributes,
- }
- daemon.EventsService.Log(action, events.NetworkEventType, actor)
- }
- // LogDaemonEventWithAttributes generates an event related to the daemon itself with specific given attributes.
- func (daemon *Daemon) LogDaemonEventWithAttributes(action string, attributes map[string]string) {
- if daemon.EventsService != nil {
- if info, err := daemon.SystemInfo(); err == nil && info.Name != "" {
- attributes["name"] = info.Name
- }
- actor := events.Actor{
- ID: daemon.ID,
- Attributes: attributes,
- }
- daemon.EventsService.Log(action, events.DaemonEventType, actor)
- }
- }
- // SubscribeToEvents returns the currently record of events, a channel to stream new events from, and a function to cancel the stream of events.
- func (daemon *Daemon) SubscribeToEvents(since, until time.Time, filter filters.Args) ([]events.Message, chan interface{}) {
- ef := daemonevents.NewFilter(filter)
- return daemon.EventsService.SubscribeTopic(since, until, ef)
- }
- // UnsubscribeFromEvents stops the event subscription for a client by closing the
- // channel where the daemon sends events to.
- func (daemon *Daemon) UnsubscribeFromEvents(listener chan interface{}) {
- daemon.EventsService.Evict(listener)
- }
- // copyAttributes guarantees that labels are not mutated by event triggers.
- func copyAttributes(attributes, labels map[string]string) {
- if labels == nil {
- return
- }
- for k, v := range labels {
- attributes[k] = v
- }
- }
- // ProcessClusterNotifications gets changes from store and add them to event list
- func (daemon *Daemon) ProcessClusterNotifications(ctx context.Context, watchStream chan *swarmapi.WatchMessage) {
- for {
- select {
- case <-ctx.Done():
- return
- case message, ok := <-watchStream:
- if !ok {
- logrus.Debug("cluster event channel has stopped")
- return
- }
- daemon.generateClusterEvent(message)
- }
- }
- }
- func (daemon *Daemon) generateClusterEvent(msg *swarmapi.WatchMessage) {
- for _, event := range msg.Events {
- if event.Object == nil {
- logrus.Errorf("event without object: %v", event)
- continue
- }
- switch v := event.Object.GetObject().(type) {
- case *swarmapi.Object_Node:
- daemon.logNodeEvent(event.Action, v.Node, event.OldObject.GetNode())
- case *swarmapi.Object_Service:
- daemon.logServiceEvent(event.Action, v.Service, event.OldObject.GetService())
- case *swarmapi.Object_Network:
- daemon.logNetworkEvent(event.Action, v.Network, event.OldObject.GetNetwork())
- case *swarmapi.Object_Secret:
- daemon.logSecretEvent(event.Action, v.Secret, event.OldObject.GetSecret())
- default:
- logrus.Warnf("unrecognized event: %v", event)
- }
- }
- }
- func (daemon *Daemon) logNetworkEvent(action swarmapi.WatchActionKind, net *swarmapi.Network, oldNet *swarmapi.Network) {
- attributes := map[string]string{
- "name": net.Spec.Annotations.Name,
- }
- eventTime := eventTimestamp(net.Meta, action)
- daemon.logClusterEvent(action, net.ID, "network", attributes, eventTime)
- }
- func (daemon *Daemon) logSecretEvent(action swarmapi.WatchActionKind, secret *swarmapi.Secret, oldSecret *swarmapi.Secret) {
- attributes := map[string]string{
- "name": secret.Spec.Annotations.Name,
- }
- eventTime := eventTimestamp(secret.Meta, action)
- daemon.logClusterEvent(action, secret.ID, "secret", attributes, eventTime)
- }
- func (daemon *Daemon) logNodeEvent(action swarmapi.WatchActionKind, node *swarmapi.Node, oldNode *swarmapi.Node) {
- name := node.Spec.Annotations.Name
- if name == "" && node.Description != nil {
- name = node.Description.Hostname
- }
- attributes := map[string]string{
- "name": name,
- }
- eventTime := eventTimestamp(node.Meta, action)
- // In an update event, display the changes in attributes
- if action == swarmapi.WatchActionKindUpdate && oldNode != nil {
- if node.Spec.Availability != oldNode.Spec.Availability {
- attributes["availability.old"] = strings.ToLower(oldNode.Spec.Availability.String())
- attributes["availability.new"] = strings.ToLower(node.Spec.Availability.String())
- }
- if node.Role != oldNode.Role {
- attributes["role.old"] = strings.ToLower(oldNode.Role.String())
- attributes["role.new"] = strings.ToLower(node.Role.String())
- }
- if node.Status.State != oldNode.Status.State {
- attributes["state.old"] = strings.ToLower(oldNode.Status.State.String())
- attributes["state.new"] = strings.ToLower(node.Status.State.String())
- }
- // This handles change within manager role
- if node.ManagerStatus != nil && oldNode.ManagerStatus != nil {
- // leader change
- if node.ManagerStatus.Leader != oldNode.ManagerStatus.Leader {
- if node.ManagerStatus.Leader {
- attributes["leader.old"] = "false"
- attributes["leader.new"] = "true"
- } else {
- attributes["leader.old"] = "true"
- attributes["leader.new"] = "false"
- }
- }
- if node.ManagerStatus.Reachability != oldNode.ManagerStatus.Reachability {
- attributes["reachability.old"] = strings.ToLower(oldNode.ManagerStatus.Reachability.String())
- attributes["reachability.new"] = strings.ToLower(node.ManagerStatus.Reachability.String())
- }
- }
- }
- daemon.logClusterEvent(action, node.ID, "node", attributes, eventTime)
- }
- func (daemon *Daemon) logServiceEvent(action swarmapi.WatchActionKind, service *swarmapi.Service, oldService *swarmapi.Service) {
- attributes := map[string]string{
- "name": service.Spec.Annotations.Name,
- }
- eventTime := eventTimestamp(service.Meta, action)
- if action == swarmapi.WatchActionKindUpdate && oldService != nil {
- // check image
- if x, ok := service.Spec.Task.GetRuntime().(*swarmapi.TaskSpec_Container); ok {
- containerSpec := x.Container
- if y, ok := oldService.Spec.Task.GetRuntime().(*swarmapi.TaskSpec_Container); ok {
- oldContainerSpec := y.Container
- if containerSpec.Image != oldContainerSpec.Image {
- attributes["image.old"] = oldContainerSpec.Image
- attributes["image.new"] = containerSpec.Image
- }
- } else {
- // This should not happen.
- logrus.Errorf("service %s runtime changed from %T to %T", service.Spec.Annotations.Name, oldService.Spec.Task.GetRuntime(), service.Spec.Task.GetRuntime())
- }
- }
- // check replicated count change
- if x, ok := service.Spec.GetMode().(*swarmapi.ServiceSpec_Replicated); ok {
- replicas := x.Replicated.Replicas
- if y, ok := oldService.Spec.GetMode().(*swarmapi.ServiceSpec_Replicated); ok {
- oldReplicas := y.Replicated.Replicas
- if replicas != oldReplicas {
- attributes["replicas.old"] = strconv.FormatUint(oldReplicas, 10)
- attributes["replicas.new"] = strconv.FormatUint(replicas, 10)
- }
- } else {
- // This should not happen.
- logrus.Errorf("service %s mode changed from %T to %T", service.Spec.Annotations.Name, oldService.Spec.GetMode(), service.Spec.GetMode())
- }
- }
- if service.UpdateStatus != nil {
- if oldService.UpdateStatus == nil {
- attributes["updatestate.new"] = strings.ToLower(service.UpdateStatus.State.String())
- } else if service.UpdateStatus.State != oldService.UpdateStatus.State {
- attributes["updatestate.old"] = strings.ToLower(oldService.UpdateStatus.State.String())
- attributes["updatestate.new"] = strings.ToLower(service.UpdateStatus.State.String())
- }
- }
- }
- daemon.logClusterEvent(action, service.ID, "service", attributes, eventTime)
- }
- func (daemon *Daemon) logClusterEvent(action swarmapi.WatchActionKind, id, eventType string, attributes map[string]string, eventTime time.Time) {
- actor := events.Actor{
- ID: id,
- Attributes: attributes,
- }
- jm := events.Message{
- Action: clusterEventAction[action],
- Type: eventType,
- Actor: actor,
- Scope: "swarm",
- Time: eventTime.UTC().Unix(),
- TimeNano: eventTime.UTC().UnixNano(),
- }
- daemon.EventsService.PublishMessage(jm)
- }
- func eventTimestamp(meta swarmapi.Meta, action swarmapi.WatchActionKind) time.Time {
- var eventTime time.Time
- switch action {
- case swarmapi.WatchActionKindCreate:
- eventTime, _ = gogotypes.TimestampFromProto(meta.CreatedAt)
- case swarmapi.WatchActionKindUpdate:
- eventTime, _ = gogotypes.TimestampFromProto(meta.UpdatedAt)
- case swarmapi.WatchActionKindRemove:
- // There is no timestamp from store message for remove operations.
- // Use current time.
- eventTime = time.Now()
- }
- return eventTime
- }
|