mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2024-11-21 14:44:00 +00:00
9bd9f67718
* adds dockerswarm service discovery https://github.com/VictoriaMetrics/VictoriaMetrics/issues/656 Following roles supported: services, tasks and nodes. Basic, token and tls auth supported. Added tests for labels generation. * added unix socket support to discovery utils Co-authored-by: Aliaksandr Valialkin <valyala@gmail.com>
139 lines
3.7 KiB
Go
139 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 := getNetworksLabels(cfg)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return addServicesLabels(services, networksLabels, cfg.port), nil
|
|
}
|
|
|
|
func getServices(cfg *apiConfig) ([]service, error) {
|
|
data, err := cfg.client.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]string, port int) []map[string]string {
|
|
var ms []map[string]string
|
|
for _, service := range services {
|
|
m := map[string]string{
|
|
"__meta_dockerswarm_service_id": service.ID,
|
|
"__meta_dockerswarm_service_name": service.Spec.Name,
|
|
"__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_mode": getServiceMode(service),
|
|
}
|
|
if service.UpdateStatus != nil {
|
|
m["__meta_dockerswarm_service_updating_status"] = service.UpdateStatus.State
|
|
}
|
|
for k, v := range service.Spec.Labels {
|
|
m["__meta_dockerswarm_service_label_"+discoveryutils.SanitizeLabelName(k)] = v
|
|
}
|
|
for _, vip := range service.Endpoint.VirtualIPs {
|
|
var added bool
|
|
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
|
|
}
|
|
for _, ep := range service.Endpoint.Ports {
|
|
if ep.Protocol != "tcp" {
|
|
continue
|
|
}
|
|
lbls := map[string]string{
|
|
"__meta_dockerswarm_service_endpoint_port_name": ep.Name,
|
|
"__meta_dockerswarm_service_endpoint_port_publish_mode": ep.PublishMode,
|
|
"__address__": discoveryutils.JoinHostPort(ip.String(), ep.PublishedPort),
|
|
}
|
|
for k, v := range m {
|
|
lbls[k] = v
|
|
}
|
|
lbls = joinLabels(networksLabels, lbls, "__meta_dockerswarm_network_id", vip.NetworkID)
|
|
added = true
|
|
ms = append(ms, lbls)
|
|
}
|
|
if !added {
|
|
lbls := make(map[string]string, len(m))
|
|
for k, v := range m {
|
|
lbls[k] = v
|
|
}
|
|
lbls = joinLabels(networksLabels, lbls, "__meta_dockerswarm_network_id", vip.NetworkID)
|
|
lbls["__address__"] = discoveryutils.JoinHostPort(ip.String(), port)
|
|
ms = append(ms, lbls)
|
|
}
|
|
}
|
|
|
|
}
|
|
return ms
|
|
}
|