211 lines
5.6 KiB
Go
211 lines
5.6 KiB
Go
|
//go:generate ../../../tools/readme_config_includer/generator
|
||
|
package internet_speed
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
_ "embed"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"math"
|
||
|
"os"
|
||
|
"time"
|
||
|
|
||
|
"github.com/showwin/speedtest-go/speedtest"
|
||
|
"github.com/showwin/speedtest-go/speedtest/transport"
|
||
|
|
||
|
"github.com/influxdata/telegraf"
|
||
|
"github.com/influxdata/telegraf/filter"
|
||
|
"github.com/influxdata/telegraf/internal"
|
||
|
"github.com/influxdata/telegraf/plugins/inputs"
|
||
|
)
|
||
|
|
||
|
//go:embed sample.conf
|
||
|
var sampleConfig string
|
||
|
|
||
|
const (
|
||
|
measurement = "internet_speed"
|
||
|
testModeSingle = "single"
|
||
|
testModeMulti = "multi"
|
||
|
)
|
||
|
|
||
|
type InternetSpeed struct {
|
||
|
ServerIDInclude []string `toml:"server_id_include"`
|
||
|
ServerIDExclude []string `toml:"server_id_exclude"`
|
||
|
EnableFileDownload bool `toml:"enable_file_download" deprecated:"1.25.0;1.35.0;use 'memory_saving_mode' instead"`
|
||
|
MemorySavingMode bool `toml:"memory_saving_mode"`
|
||
|
Cache bool `toml:"cache"`
|
||
|
Connections int `toml:"connections"`
|
||
|
TestMode string `toml:"test_mode"`
|
||
|
|
||
|
Log telegraf.Logger `toml:"-"`
|
||
|
|
||
|
server *speedtest.Server // The main(best) server
|
||
|
servers speedtest.Servers // Auxiliary servers
|
||
|
serverFilter filter.Filter
|
||
|
}
|
||
|
|
||
|
func (*InternetSpeed) SampleConfig() string {
|
||
|
return sampleConfig
|
||
|
}
|
||
|
|
||
|
func (is *InternetSpeed) Init() error {
|
||
|
switch is.TestMode {
|
||
|
case testModeSingle, testModeMulti:
|
||
|
case "":
|
||
|
is.TestMode = testModeSingle
|
||
|
default:
|
||
|
return fmt.Errorf("unrecognized test mode: %q", is.TestMode)
|
||
|
}
|
||
|
|
||
|
is.MemorySavingMode = is.MemorySavingMode || is.EnableFileDownload
|
||
|
|
||
|
var err error
|
||
|
is.serverFilter, err = filter.NewIncludeExcludeFilterDefaults(is.ServerIDInclude, is.ServerIDExclude, false, false)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("error compiling server ID filters: %w", err)
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (is *InternetSpeed) Gather(acc telegraf.Accumulator) error {
|
||
|
// If not caching, go find the closest server each time.
|
||
|
// We will find the best server as the main server. And
|
||
|
// the remaining servers will be auxiliary candidates.
|
||
|
if !is.Cache || is.server == nil {
|
||
|
if err := is.findClosestServer(); err != nil {
|
||
|
return fmt.Errorf("unable to find closest server: %w", err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
err := is.server.PingTest(nil)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("ping test failed: %w", err)
|
||
|
}
|
||
|
|
||
|
analyzer := speedtest.NewPacketLossAnalyzer(&speedtest.PacketLossAnalyzerOptions{
|
||
|
PacketSendingInterval: time.Millisecond * 100,
|
||
|
SamplingDuration: time.Second * 15,
|
||
|
})
|
||
|
|
||
|
var pLoss *transport.PLoss
|
||
|
|
||
|
if is.TestMode == testModeMulti {
|
||
|
err = is.server.MultiDownloadTestContext(context.Background(), is.servers)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("download test failed: %w", err)
|
||
|
}
|
||
|
err = is.server.MultiUploadTestContext(context.Background(), is.servers)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("upload test failed: %w", err)
|
||
|
}
|
||
|
// Not all servers are applicable for packet loss testing.
|
||
|
// If err != nil, we skip it and just report a warning.
|
||
|
pLoss, err = analyzer.RunMulti(is.servers.Hosts())
|
||
|
if err != nil {
|
||
|
is.Log.Warnf("packet loss test failed: %s", err)
|
||
|
}
|
||
|
} else {
|
||
|
err = is.server.DownloadTest()
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("download test failed: %w", err)
|
||
|
}
|
||
|
err = is.server.UploadTest()
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("upload test failed: %w", err)
|
||
|
}
|
||
|
// Not all servers are applicable for packet loss testing.
|
||
|
// If err != nil, we skip it and just report a warning.
|
||
|
err = analyzer.Run(is.server.Host, func(pl *transport.PLoss) {
|
||
|
pLoss = pl
|
||
|
})
|
||
|
if err != nil {
|
||
|
is.Log.Warnf("packet loss test failed: %s", err)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
packetLoss := -1.0
|
||
|
if pLoss != nil {
|
||
|
packetLoss = pLoss.LossPercent()
|
||
|
}
|
||
|
|
||
|
fields := map[string]any{
|
||
|
"download": is.server.DLSpeed.Mbps(),
|
||
|
"upload": is.server.ULSpeed.Mbps(),
|
||
|
"latency": timeDurationMillisecondToFloat64(is.server.Latency),
|
||
|
"jitter": timeDurationMillisecondToFloat64(is.server.Jitter),
|
||
|
"packet_loss": packetLoss,
|
||
|
"location": is.server.Name,
|
||
|
}
|
||
|
tags := map[string]string{
|
||
|
"server_id": is.server.ID,
|
||
|
"source": is.server.Host,
|
||
|
"test_mode": is.TestMode,
|
||
|
}
|
||
|
// Recycle the history of each test to prevent data backlog.
|
||
|
is.server.Context.Reset()
|
||
|
acc.AddFields(measurement, fields, tags)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (is *InternetSpeed) findClosestServer() error {
|
||
|
proto := speedtest.HTTP
|
||
|
if os.Getegid() <= 0 {
|
||
|
proto = speedtest.ICMP
|
||
|
}
|
||
|
|
||
|
client := speedtest.New(speedtest.WithUserConfig(&speedtest.UserConfig{
|
||
|
UserAgent: internal.ProductToken(),
|
||
|
PingMode: proto,
|
||
|
SavingMode: is.MemorySavingMode,
|
||
|
}))
|
||
|
if is.Connections > 0 {
|
||
|
client.SetNThread(is.Connections)
|
||
|
}
|
||
|
|
||
|
var err error
|
||
|
is.servers, err = client.FetchServers()
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("fetching server list failed: %w", err)
|
||
|
}
|
||
|
|
||
|
if len(is.servers) < 1 {
|
||
|
return errors.New("no servers found")
|
||
|
}
|
||
|
|
||
|
// Return the first match or the server with the lowest latency
|
||
|
// when filter mismatch all servers.
|
||
|
var minLatency int64 = math.MaxInt64
|
||
|
selectIndex := -1
|
||
|
for index, server := range is.servers {
|
||
|
if is.serverFilter.Match(server.ID) {
|
||
|
selectIndex = index
|
||
|
break
|
||
|
}
|
||
|
if server.Latency > 0 {
|
||
|
if minLatency > server.Latency.Milliseconds() {
|
||
|
minLatency = server.Latency.Milliseconds()
|
||
|
selectIndex = index
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if selectIndex != -1 {
|
||
|
is.server = is.servers[selectIndex]
|
||
|
is.Log.Debugf("using server %s in %s (%s)\n", is.server.ID, is.server.Name, is.server.Host)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
return errors.New("no server set: filter excluded all servers or no available server found")
|
||
|
}
|
||
|
|
||
|
func timeDurationMillisecondToFloat64(d time.Duration) float64 {
|
||
|
return float64(d) / float64(time.Millisecond)
|
||
|
}
|
||
|
|
||
|
func init() {
|
||
|
inputs.Add("internet_speed", func() telegraf.Input {
|
||
|
return &InternetSpeed{}
|
||
|
})
|
||
|
}
|