You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							239 lines
						
					
					
						
							5.9 KiB
						
					
					
				
			
		
		
	
	
							239 lines
						
					
					
						
							5.9 KiB
						
					
					
				| // Copyright 2018 The Prometheus Authors
 | |
| // Licensed under the Apache License, Version 2.0 (the "License");
 | |
| // you may not use this file except in compliance with the License.
 | |
| // You may obtain a copy of the License at
 | |
| //
 | |
| // http://www.apache.org/licenses/LICENSE-2.0
 | |
| //
 | |
| // Unless required by applicable law or agreed to in writing, software
 | |
| // distributed under the License is distributed on an "AS IS" BASIS,
 | |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| // See the License for the specific language governing permissions and
 | |
| // limitations under the License.
 | |
| 
 | |
| package procfs
 | |
| 
 | |
| import (
 | |
| 	"bufio"
 | |
| 	"encoding/hex"
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"io/ioutil"
 | |
| 	"net"
 | |
| 	"os"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| )
 | |
| 
 | |
| // IPVSStats holds IPVS statistics, as exposed by the kernel in `/proc/net/ip_vs_stats`.
 | |
| type IPVSStats struct {
 | |
| 	// Total count of connections.
 | |
| 	Connections uint64
 | |
| 	// Total incoming packages processed.
 | |
| 	IncomingPackets uint64
 | |
| 	// Total outgoing packages processed.
 | |
| 	OutgoingPackets uint64
 | |
| 	// Total incoming traffic.
 | |
| 	IncomingBytes uint64
 | |
| 	// Total outgoing traffic.
 | |
| 	OutgoingBytes uint64
 | |
| }
 | |
| 
 | |
| // IPVSBackendStatus holds current metrics of one virtual / real address pair.
 | |
| type IPVSBackendStatus struct {
 | |
| 	// The local (virtual) IP address.
 | |
| 	LocalAddress net.IP
 | |
| 	// The remote (real) IP address.
 | |
| 	RemoteAddress net.IP
 | |
| 	// The local (virtual) port.
 | |
| 	LocalPort uint16
 | |
| 	// The remote (real) port.
 | |
| 	RemotePort uint16
 | |
| 	// The local firewall mark
 | |
| 	LocalMark string
 | |
| 	// The transport protocol (TCP, UDP).
 | |
| 	Proto string
 | |
| 	// The current number of active connections for this virtual/real address pair.
 | |
| 	ActiveConn uint64
 | |
| 	// The current number of inactive connections for this virtual/real address pair.
 | |
| 	InactConn uint64
 | |
| 	// The current weight of this virtual/real address pair.
 | |
| 	Weight uint64
 | |
| }
 | |
| 
 | |
| // IPVSStats reads the IPVS statistics from the specified `proc` filesystem.
 | |
| func (fs FS) IPVSStats() (IPVSStats, error) {
 | |
| 	file, err := os.Open(fs.proc.Path("net/ip_vs_stats"))
 | |
| 	if err != nil {
 | |
| 		return IPVSStats{}, err
 | |
| 	}
 | |
| 	defer file.Close()
 | |
| 
 | |
| 	return parseIPVSStats(file)
 | |
| }
 | |
| 
 | |
| // parseIPVSStats performs the actual parsing of `ip_vs_stats`.
 | |
| func parseIPVSStats(file io.Reader) (IPVSStats, error) {
 | |
| 	var (
 | |
| 		statContent []byte
 | |
| 		statLines   []string
 | |
| 		statFields  []string
 | |
| 		stats       IPVSStats
 | |
| 	)
 | |
| 
 | |
| 	statContent, err := ioutil.ReadAll(file)
 | |
| 	if err != nil {
 | |
| 		return IPVSStats{}, err
 | |
| 	}
 | |
| 
 | |
| 	statLines = strings.SplitN(string(statContent), "\n", 4)
 | |
| 	if len(statLines) != 4 {
 | |
| 		return IPVSStats{}, errors.New("ip_vs_stats corrupt: too short")
 | |
| 	}
 | |
| 
 | |
| 	statFields = strings.Fields(statLines[2])
 | |
| 	if len(statFields) != 5 {
 | |
| 		return IPVSStats{}, errors.New("ip_vs_stats corrupt: unexpected number of fields")
 | |
| 	}
 | |
| 
 | |
| 	stats.Connections, err = strconv.ParseUint(statFields[0], 16, 64)
 | |
| 	if err != nil {
 | |
| 		return IPVSStats{}, err
 | |
| 	}
 | |
| 	stats.IncomingPackets, err = strconv.ParseUint(statFields[1], 16, 64)
 | |
| 	if err != nil {
 | |
| 		return IPVSStats{}, err
 | |
| 	}
 | |
| 	stats.OutgoingPackets, err = strconv.ParseUint(statFields[2], 16, 64)
 | |
| 	if err != nil {
 | |
| 		return IPVSStats{}, err
 | |
| 	}
 | |
| 	stats.IncomingBytes, err = strconv.ParseUint(statFields[3], 16, 64)
 | |
| 	if err != nil {
 | |
| 		return IPVSStats{}, err
 | |
| 	}
 | |
| 	stats.OutgoingBytes, err = strconv.ParseUint(statFields[4], 16, 64)
 | |
| 	if err != nil {
 | |
| 		return IPVSStats{}, err
 | |
| 	}
 | |
| 
 | |
| 	return stats, nil
 | |
| }
 | |
| 
 | |
| // IPVSBackendStatus reads and returns the status of all (virtual,real) server pairs from the specified `proc` filesystem.
 | |
| func (fs FS) IPVSBackendStatus() ([]IPVSBackendStatus, error) {
 | |
| 	file, err := os.Open(fs.proc.Path("net/ip_vs"))
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	defer file.Close()
 | |
| 
 | |
| 	return parseIPVSBackendStatus(file)
 | |
| }
 | |
| 
 | |
| func parseIPVSBackendStatus(file io.Reader) ([]IPVSBackendStatus, error) {
 | |
| 	var (
 | |
| 		status       []IPVSBackendStatus
 | |
| 		scanner      = bufio.NewScanner(file)
 | |
| 		proto        string
 | |
| 		localMark    string
 | |
| 		localAddress net.IP
 | |
| 		localPort    uint16
 | |
| 		err          error
 | |
| 	)
 | |
| 
 | |
| 	for scanner.Scan() {
 | |
| 		fields := strings.Fields(scanner.Text())
 | |
| 		if len(fields) == 0 {
 | |
| 			continue
 | |
| 		}
 | |
| 		switch {
 | |
| 		case fields[0] == "IP" || fields[0] == "Prot" || fields[1] == "RemoteAddress:Port":
 | |
| 			continue
 | |
| 		case fields[0] == "TCP" || fields[0] == "UDP":
 | |
| 			if len(fields) < 2 {
 | |
| 				continue
 | |
| 			}
 | |
| 			proto = fields[0]
 | |
| 			localMark = ""
 | |
| 			localAddress, localPort, err = parseIPPort(fields[1])
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 		case fields[0] == "FWM":
 | |
| 			if len(fields) < 2 {
 | |
| 				continue
 | |
| 			}
 | |
| 			proto = fields[0]
 | |
| 			localMark = fields[1]
 | |
| 			localAddress = nil
 | |
| 			localPort = 0
 | |
| 		case fields[0] == "->":
 | |
| 			if len(fields) < 6 {
 | |
| 				continue
 | |
| 			}
 | |
| 			remoteAddress, remotePort, err := parseIPPort(fields[1])
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			weight, err := strconv.ParseUint(fields[3], 10, 64)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			activeConn, err := strconv.ParseUint(fields[4], 10, 64)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			inactConn, err := strconv.ParseUint(fields[5], 10, 64)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			status = append(status, IPVSBackendStatus{
 | |
| 				LocalAddress:  localAddress,
 | |
| 				LocalPort:     localPort,
 | |
| 				LocalMark:     localMark,
 | |
| 				RemoteAddress: remoteAddress,
 | |
| 				RemotePort:    remotePort,
 | |
| 				Proto:         proto,
 | |
| 				Weight:        weight,
 | |
| 				ActiveConn:    activeConn,
 | |
| 				InactConn:     inactConn,
 | |
| 			})
 | |
| 		}
 | |
| 	}
 | |
| 	return status, nil
 | |
| }
 | |
| 
 | |
| func parseIPPort(s string) (net.IP, uint16, error) {
 | |
| 	var (
 | |
| 		ip  net.IP
 | |
| 		err error
 | |
| 	)
 | |
| 
 | |
| 	switch len(s) {
 | |
| 	case 13:
 | |
| 		ip, err = hex.DecodeString(s[0:8])
 | |
| 		if err != nil {
 | |
| 			return nil, 0, err
 | |
| 		}
 | |
| 	case 46:
 | |
| 		ip = net.ParseIP(s[1:40])
 | |
| 		if ip == nil {
 | |
| 			return nil, 0, fmt.Errorf("invalid IPv6 address: %s", s[1:40])
 | |
| 		}
 | |
| 	default:
 | |
| 		return nil, 0, fmt.Errorf("unexpected IP:Port: %s", s)
 | |
| 	}
 | |
| 
 | |
| 	portString := s[len(s)-4:]
 | |
| 	if len(portString) != 4 {
 | |
| 		return nil, 0, fmt.Errorf("unexpected port string format: %s", portString)
 | |
| 	}
 | |
| 	port, err := strconv.ParseUint(portString, 16, 16)
 | |
| 	if err != nil {
 | |
| 		return nil, 0, err
 | |
| 	}
 | |
| 
 | |
| 	return ip, uint16(port), nil
 | |
| }
 | |
| 
 |