2023-03-08 00:22:23 +00:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
|
|
|
|
// Package sockstatlog provides a logger for capturing and storing network socket stats.
|
|
|
|
package sockstatlog
|
|
|
|
|
|
|
|
import (
|
2023-03-13 23:27:41 +00:00
|
|
|
"context"
|
2023-03-22 20:51:29 +00:00
|
|
|
"crypto/sha256"
|
2023-03-08 00:22:23 +00:00
|
|
|
"encoding/json"
|
2023-03-10 22:33:26 +00:00
|
|
|
"io"
|
2023-03-22 20:51:29 +00:00
|
|
|
"net/http"
|
2023-03-08 00:22:23 +00:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"time"
|
|
|
|
|
2023-03-22 20:51:29 +00:00
|
|
|
"tailscale.com/logpolicy"
|
|
|
|
"tailscale.com/logtail"
|
2023-03-08 00:22:23 +00:00
|
|
|
"tailscale.com/logtail/filch"
|
|
|
|
"tailscale.com/net/sockstats"
|
2023-03-22 20:51:29 +00:00
|
|
|
"tailscale.com/smallzstd"
|
2023-03-10 22:33:26 +00:00
|
|
|
"tailscale.com/types/logger"
|
2023-03-22 20:51:29 +00:00
|
|
|
"tailscale.com/types/logid"
|
2023-03-08 00:22:23 +00:00
|
|
|
"tailscale.com/util/mak"
|
|
|
|
)
|
|
|
|
|
|
|
|
// pollPeriod specifies how often to poll for socket stats.
|
|
|
|
const pollPeriod = time.Second / 10
|
|
|
|
|
|
|
|
// Logger logs statistics about network sockets.
|
|
|
|
type Logger struct {
|
2023-03-13 23:27:41 +00:00
|
|
|
ctx context.Context
|
|
|
|
cancelFn context.CancelFunc
|
|
|
|
|
2023-03-22 20:51:29 +00:00
|
|
|
ticker *time.Ticker
|
|
|
|
logf logger.Logf
|
|
|
|
|
|
|
|
logger *logtail.Logger
|
2023-03-08 00:22:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// deltaStat represents the bytes transferred during a time period.
|
|
|
|
// The first element is transmitted bytes, the second element is received bytes.
|
|
|
|
type deltaStat [2]uint64
|
|
|
|
|
|
|
|
// event represents the socket stats on a specific interface during a time period.
|
|
|
|
type event struct {
|
|
|
|
// Time is when the event started as a Unix timestamp in milliseconds.
|
|
|
|
Time int64 `json:"t"`
|
|
|
|
|
|
|
|
// Duration is the duration of this event in milliseconds.
|
|
|
|
Duration int64 `json:"d"`
|
|
|
|
|
|
|
|
// IsCellularInterface is set to 1 if the traffic was sent over a cellular interface.
|
|
|
|
IsCellularInterface int `json:"c,omitempty"`
|
|
|
|
|
|
|
|
// Stats records the stats for each Label during the time period.
|
|
|
|
Stats map[sockstats.Label]deltaStat `json:"s"`
|
|
|
|
}
|
|
|
|
|
2023-03-22 20:51:29 +00:00
|
|
|
// SockstatLogID reproducibly derives a new logid.PrivateID for sockstat logging from a node's public backend log ID.
|
2023-03-23 17:49:56 +00:00
|
|
|
// The returned PrivateID is the sha256 sum of logID + "sockstat".
|
2023-03-22 20:51:29 +00:00
|
|
|
// If a node's public log ID becomes known, it is trivial to spoof sockstat logs for that node.
|
|
|
|
// Given the this is just for debugging, we're not too concerned about that.
|
2023-03-23 17:49:56 +00:00
|
|
|
func SockstatLogID(logID logid.PublicID) logid.PrivateID {
|
|
|
|
return logid.PrivateID(sha256.Sum256([]byte(logID.String() + "sockstat")))
|
2023-03-22 20:51:29 +00:00
|
|
|
}
|
|
|
|
|
2023-03-08 00:22:23 +00:00
|
|
|
// NewLogger returns a new Logger that will store stats in logdir.
|
|
|
|
// On platforms that do not support sockstat logging, a nil Logger will be returned.
|
2023-03-13 23:50:37 +00:00
|
|
|
// The returned Logger must be shut down with Shutdown when it is no longer needed.
|
2023-03-23 17:49:56 +00:00
|
|
|
func NewLogger(logdir string, logf logger.Logf, logID logid.PublicID) (*Logger, error) {
|
2023-03-08 00:22:23 +00:00
|
|
|
if !sockstats.IsAvailable {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := os.MkdirAll(logdir, 0755); err != nil && !os.IsExist(err) {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
filchPrefix := filepath.Join(logdir, "sockstats")
|
|
|
|
filch, err := filch.New(filchPrefix, filch.Options{ReplaceStderr: false})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-03-13 23:27:41 +00:00
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
2023-03-13 23:50:37 +00:00
|
|
|
logger := &Logger{
|
2023-03-22 20:51:29 +00:00
|
|
|
ctx: ctx,
|
|
|
|
cancelFn: cancel,
|
|
|
|
ticker: time.NewTicker(pollPeriod),
|
|
|
|
logf: logf,
|
2023-03-13 23:50:37 +00:00
|
|
|
}
|
2023-03-22 20:51:29 +00:00
|
|
|
logger.logger = logtail.NewLogger(logtail.Config{
|
|
|
|
BaseURL: logpolicy.LogURL(),
|
2023-03-23 17:49:56 +00:00
|
|
|
PrivateID: SockstatLogID(logID),
|
2023-03-22 20:51:29 +00:00
|
|
|
Collection: "sockstats.log.tailscale.io",
|
|
|
|
Buffer: filch,
|
|
|
|
NewZstdEncoder: func() logtail.Encoder {
|
|
|
|
w, err := smallzstd.NewEncoder(nil)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return w
|
|
|
|
},
|
|
|
|
FlushDelayFn: func() time.Duration {
|
|
|
|
// set flush delay to 100 years so it never flushes automatically
|
|
|
|
return 100 * 365 * 24 * time.Hour
|
|
|
|
},
|
|
|
|
Stderr: io.Discard, // don't log to stderr
|
|
|
|
|
|
|
|
HTTPC: &http.Client{Transport: logpolicy.NewLogtailTransport(logtail.DefaultHost)},
|
|
|
|
}, logf)
|
2023-03-13 23:50:37 +00:00
|
|
|
|
|
|
|
go logger.poll()
|
2023-03-08 00:22:23 +00:00
|
|
|
|
2023-03-13 23:50:37 +00:00
|
|
|
return logger, nil
|
2023-03-08 00:22:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// poll fetches the current socket stats at the configured time interval,
|
|
|
|
// calculates the delta since the last poll, and logs any non-zero values.
|
|
|
|
// This method does not return.
|
|
|
|
func (l *Logger) poll() {
|
|
|
|
// last is the last set of socket stats we saw.
|
|
|
|
var lastStats *sockstats.SockStats
|
|
|
|
var lastTime time.Time
|
|
|
|
|
2023-03-22 20:51:29 +00:00
|
|
|
enc := json.NewEncoder(l.logger)
|
2023-03-13 23:27:41 +00:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-l.ctx.Done():
|
|
|
|
return
|
|
|
|
case t := <-l.ticker.C:
|
|
|
|
stats := sockstats.Get()
|
|
|
|
if lastStats != nil {
|
|
|
|
diffstats := delta(lastStats, stats)
|
|
|
|
if len(diffstats) > 0 {
|
|
|
|
e := event{
|
|
|
|
Time: lastTime.UnixMilli(),
|
|
|
|
Duration: t.Sub(lastTime).Milliseconds(),
|
|
|
|
Stats: diffstats,
|
|
|
|
}
|
|
|
|
if stats.CurrentInterfaceCellular {
|
|
|
|
e.IsCellularInterface = 1
|
|
|
|
}
|
|
|
|
if err := enc.Encode(e); err != nil {
|
|
|
|
l.logf("sockstatlog: error encoding log: %v", err)
|
|
|
|
}
|
2023-03-10 22:33:26 +00:00
|
|
|
}
|
2023-03-08 00:22:23 +00:00
|
|
|
}
|
2023-03-13 23:27:41 +00:00
|
|
|
lastTime = t
|
|
|
|
lastStats = stats
|
2023-03-08 00:22:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-22 20:51:29 +00:00
|
|
|
func (l *Logger) LogID() string {
|
|
|
|
if l.logger == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
return l.logger.PrivateID().Public().String()
|
2023-03-08 00:22:23 +00:00
|
|
|
}
|
|
|
|
|
2023-03-22 20:51:29 +00:00
|
|
|
// Flush sends pending logs to the log server and flushes them from the local buffer.
|
|
|
|
func (l *Logger) Flush() {
|
|
|
|
l.logger.StartFlush()
|
|
|
|
}
|
2023-03-10 22:33:26 +00:00
|
|
|
|
2023-03-22 20:51:29 +00:00
|
|
|
func (l *Logger) Shutdown() {
|
|
|
|
l.ticker.Stop()
|
|
|
|
l.logger.Shutdown(context.Background())
|
|
|
|
l.cancelFn()
|
2023-03-10 22:33:26 +00:00
|
|
|
}
|
|
|
|
|
2023-03-08 00:22:23 +00:00
|
|
|
// delta calculates the delta stats between two SockStats snapshots.
|
|
|
|
// b is assumed to have occurred after a.
|
|
|
|
// Zero values are omitted from the returned map, and an empty map is returned if no bytes were transferred.
|
|
|
|
func delta(a, b *sockstats.SockStats) (stats map[sockstats.Label]deltaStat) {
|
|
|
|
if a == nil || b == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
for label, bs := range b.Stats {
|
|
|
|
as := a.Stats[label]
|
|
|
|
if as.TxBytes == bs.TxBytes && as.RxBytes == bs.RxBytes {
|
|
|
|
// fast path for unchanged stats
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
mak.Set(&stats, label, deltaStat{bs.TxBytes - as.TxBytes, bs.RxBytes - as.RxBytes})
|
|
|
|
}
|
|
|
|
return stats
|
|
|
|
}
|