mirror of
				https://github.com/tailscale/tailscale.git
				synced 2025-11-04 00:55:11 +00:00 
			
		
		
		
	- Wrap each prober function into a probe class that allows associating metric labels and custom metrics with a given probe; - Make sure all existing probe classes set a `class` metric label; - Move bandwidth probe size from being a metric label to a separate gauge metric; this will make it possible to use it to calculate average used bandwidth using a PromQL query; - Also export transfer time for the bandwidth prober (more accurate than the total probe time, since it excludes connection establishment time). Updates tailscale/corp#17912 Signed-off-by: Anton Tolchanov <anton@tailscale.com>
		
			
				
	
	
		
			133 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			133 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright (c) Tailscale Inc & AUTHORS
 | 
						|
// SPDX-License-Identifier: BSD-3-Clause
 | 
						|
 | 
						|
package prober
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"fmt"
 | 
						|
	"net"
 | 
						|
	"net/netip"
 | 
						|
	"sync"
 | 
						|
 | 
						|
	"tailscale.com/types/logger"
 | 
						|
)
 | 
						|
 | 
						|
// ForEachAddrOpts contains options for ForEachAddr. The zero value for all
 | 
						|
// fields is valid unless stated otherwise.
 | 
						|
type ForEachAddrOpts struct {
 | 
						|
	// Logf is the logger to use for logging. If nil, no logging is done.
 | 
						|
	Logf logger.Logf
 | 
						|
	// Networks is the list of networks to resolve; if non-empty, it should
 | 
						|
	// contain at least one of "ip", "ip4", or "ip6".
 | 
						|
	//
 | 
						|
	// If empty, "ip" is assumed.
 | 
						|
	Networks []string
 | 
						|
	// LookupNetIP is the function to use to resolve the hostname to one or
 | 
						|
	// more IP addresses.
 | 
						|
	//
 | 
						|
	// If nil, net.DefaultResolver.LookupNetIP is used.
 | 
						|
	LookupNetIP func(context.Context, string, string) ([]netip.Addr, error)
 | 
						|
}
 | 
						|
 | 
						|
// ForEachAddr returns a Probe that resolves a given hostname into all
 | 
						|
// available IP addresses, and then calls a function to create new Probes
 | 
						|
// every time a new IP is discovered. The Probes returned will be closed if an
 | 
						|
// IP address is no longer in the DNS record for the given hostname. This can
 | 
						|
// be used to healthcheck every IP address that a hostname resolves to.
 | 
						|
func ForEachAddr(host string, makeProbes func(netip.Addr) []*Probe, opts ForEachAddrOpts) ProbeClass {
 | 
						|
	feap := makeForEachAddr(host, makeProbes, opts)
 | 
						|
	return ProbeClass{
 | 
						|
		Probe: feap.run,
 | 
						|
		Class: "dns_each_addr",
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func makeForEachAddr(host string, makeProbes func(netip.Addr) []*Probe, opts ForEachAddrOpts) *forEachAddrProbe {
 | 
						|
	if opts.Logf == nil {
 | 
						|
		opts.Logf = logger.Discard
 | 
						|
	}
 | 
						|
	if len(opts.Networks) == 0 {
 | 
						|
		opts.Networks = []string{"ip"}
 | 
						|
	}
 | 
						|
	if opts.LookupNetIP == nil {
 | 
						|
		opts.LookupNetIP = net.DefaultResolver.LookupNetIP
 | 
						|
	}
 | 
						|
 | 
						|
	return &forEachAddrProbe{
 | 
						|
		logf:        opts.Logf,
 | 
						|
		host:        host,
 | 
						|
		networks:    opts.Networks,
 | 
						|
		makeProbes:  makeProbes,
 | 
						|
		lookupNetIP: opts.LookupNetIP,
 | 
						|
		probes:      make(map[netip.Addr][]*Probe),
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
type forEachAddrProbe struct {
 | 
						|
	// inputs; immutable
 | 
						|
	logf        logger.Logf
 | 
						|
	host        string
 | 
						|
	networks    []string
 | 
						|
	makeProbes  func(netip.Addr) []*Probe
 | 
						|
	lookupNetIP func(context.Context, string, string) ([]netip.Addr, error)
 | 
						|
 | 
						|
	// state
 | 
						|
	mu     sync.Mutex // protects following
 | 
						|
	probes map[netip.Addr][]*Probe
 | 
						|
}
 | 
						|
 | 
						|
// run matches the ProbeFunc signature
 | 
						|
func (f *forEachAddrProbe) run(ctx context.Context) error {
 | 
						|
	var addrs []netip.Addr
 | 
						|
	for _, network := range f.networks {
 | 
						|
		naddrs, err := f.lookupNetIP(ctx, network, f.host)
 | 
						|
		if err != nil {
 | 
						|
			return fmt.Errorf("resolving %s addr for %q: %w", network, f.host, err)
 | 
						|
		}
 | 
						|
		addrs = append(addrs, naddrs...)
 | 
						|
	}
 | 
						|
	if len(addrs) == 0 {
 | 
						|
		return fmt.Errorf("no addrs for %q", f.host)
 | 
						|
	}
 | 
						|
 | 
						|
	// For each address, create a new probe if it doesn't already
 | 
						|
	// exist in our probe map.
 | 
						|
	f.mu.Lock()
 | 
						|
	defer f.mu.Unlock()
 | 
						|
 | 
						|
	sawIPs := make(map[netip.Addr]bool)
 | 
						|
	for _, addr := range addrs {
 | 
						|
		sawIPs[addr] = true
 | 
						|
 | 
						|
		if _, ok := f.probes[addr]; ok {
 | 
						|
			// Nothing to create
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		// Make a new probe, and add it to 'probes'; if the
 | 
						|
		// function returns an empty list, we skip it.
 | 
						|
		probes := f.makeProbes(addr)
 | 
						|
		if len(probes) == 0 {
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		f.logf("adding %d new probes for %v", len(probes), addr)
 | 
						|
		f.probes[addr] = probes
 | 
						|
	}
 | 
						|
 | 
						|
	// Remove probes that we didn't see during this address resolution.
 | 
						|
	for addr, probes := range f.probes {
 | 
						|
		if !sawIPs[addr] {
 | 
						|
			f.logf("removing %d probes for %v", len(probes), addr)
 | 
						|
 | 
						|
			// This IP is no longer in the DNS record. Close and remove all probes
 | 
						|
			for _, probe := range probes {
 | 
						|
				probe.Close()
 | 
						|
			}
 | 
						|
			delete(f.probes, addr)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 |