mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-20 15:16:42 +00:00
86394b4179
Cache sanitized label names and return them next time. This reduces the number of allocations and speeds up the SanitizeLabelName() function for common case when the number of unique label names is smaller than 100k
146 lines
3.7 KiB
Go
146 lines
3.7 KiB
Go
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
|
|
}
|