package dockerswarm import ( "encoding/json" "fmt" "net" "github.com/VictoriaMetrics/VictoriaMetrics/lib/logger" "github.com/VictoriaMetrics/VictoriaMetrics/lib/promscrape/discoveryutils" ) // https://docs.docker.com/engine/api/v1.40/#tag/Service type service struct { ID string Spec struct { Labels map[string]string Name string TaskTemplate struct { ContainerSpec struct { Hostname string Image string } } Mode struct { Global interface{} Replicated interface{} } } UpdateStatus struct { State string } Endpoint struct { Ports []portConfig VirtualIPs []struct { NetworkID string Addr string } } } type portConfig struct { Protocol string Name string PublishMode string PublishedPort int } func getServicesLabels(cfg *apiConfig) ([]map[string]string, error) { services, err := getServices(cfg) if err != nil { return nil, err } networksLabels, err := getNetworksLabelsByNetworkID(cfg) if err != nil { return nil, err } return addServicesLabels(services, networksLabels, cfg.port), nil } func getServices(cfg *apiConfig) ([]service, error) { data, err := cfg.getAPIResponse("/services") if err != nil { return nil, fmt.Errorf("cannot query dockerswarm api for services: %w", err) } return parseServicesResponse(data) } func parseServicesResponse(data []byte) ([]service, error) { var services []service if err := json.Unmarshal(data, &services); err != nil { return nil, fmt.Errorf("cannot parse services: %w", err) } return services, nil } func getServiceMode(svc service) string { if svc.Spec.Mode.Global != nil { return "global" } if svc.Spec.Mode.Replicated != nil { return "replicated" } return "" } func addServicesLabels(services []service, networksLabels map[string]map[string]string, port int) []map[string]string { var ms []map[string]string for _, service := range services { commonLabels := map[string]string{ "__meta_dockerswarm_service_id": service.ID, "__meta_dockerswarm_service_name": service.Spec.Name, "__meta_dockerswarm_service_mode": getServiceMode(service), "__meta_dockerswarm_service_task_container_hostname": service.Spec.TaskTemplate.ContainerSpec.Hostname, "__meta_dockerswarm_service_task_container_image": service.Spec.TaskTemplate.ContainerSpec.Image, "__meta_dockerswarm_service_updating_status": service.UpdateStatus.State, } for k, v := range service.Spec.Labels { commonLabels[discoveryutils.SanitizeLabelName("__meta_dockerswarm_service_label_"+k)] = v } for _, vip := range service.Endpoint.VirtualIPs { // skip services without virtual address. // usually its host services. if vip.Addr == "" { continue } ip, _, err := net.ParseCIDR(vip.Addr) if err != nil { logger.Errorf("cannot parse: %q as cidr for service label add, err: %v", vip.Addr, err) continue } added := false for _, ep := range service.Endpoint.Ports { if ep.Protocol != "tcp" { continue } m := map[string]string{ "__address__": discoveryutils.JoinHostPort(ip.String(), ep.PublishedPort), "__meta_dockerswarm_service_endpoint_port_name": ep.Name, "__meta_dockerswarm_service_endpoint_port_publish_mode": ep.PublishMode, } for k, v := range commonLabels { m[k] = v } for k, v := range networksLabels[vip.NetworkID] { m[k] = v } added = true ms = append(ms, m) } if !added { m := map[string]string{ "__address__": discoveryutils.JoinHostPort(ip.String(), port), } for k, v := range commonLabels { m[k] = v } for k, v := range networksLabels[vip.NetworkID] { m[k] = v } ms = append(ms, m) } } } return ms }