163 lines
4.8 KiB
Go
163 lines
4.8 KiB
Go
package main
|
|
|
|
import (
|
|
"flag"
|
|
"fmt"
|
|
"math/rand"
|
|
"net"
|
|
"net/http"
|
|
"os"
|
|
"time"
|
|
|
|
"github.com/golang/glog"
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
|
)
|
|
|
|
var (
|
|
numCurrentClients int64
|
|
numTotalClients int64
|
|
numTotalClientsClosed int64
|
|
numTotalBytes int64
|
|
numTotalMilliseconds int64
|
|
totalClients prometheus.CounterFunc
|
|
totalClientsClosed prometheus.CounterFunc
|
|
totalBytes prometheus.CounterFunc
|
|
totalSeconds prometheus.CounterFunc
|
|
clientIP *prometheus.CounterVec
|
|
clientSeconds *prometheus.CounterVec
|
|
)
|
|
|
|
func initPrometheus(connHost, prometheusPort, prometheusEntry string) {
|
|
totalClients = prometheus.NewCounterFunc(
|
|
prometheus.CounterOpts{
|
|
Name: "endlessh_client_open_count_total",
|
|
Help: "Total number of clients that tried to connect to this host.",
|
|
}, func() float64 {
|
|
return float64(numTotalClients)
|
|
},
|
|
)
|
|
totalClientsClosed = prometheus.NewCounterFunc(
|
|
prometheus.CounterOpts{
|
|
Name: "endlessh_client_closed_count_total",
|
|
Help: "Total number of clients that stopped connecting to this host.",
|
|
}, func() float64 {
|
|
return float64(numTotalClientsClosed)
|
|
},
|
|
)
|
|
totalBytes = prometheus.NewCounterFunc(
|
|
prometheus.CounterOpts{
|
|
Name: "endlessh_sent_bytes_total",
|
|
Help: "Total bytes sent to clients that tried to connect to this host.",
|
|
}, func() float64 {
|
|
return float64(numTotalBytes)
|
|
},
|
|
)
|
|
totalSeconds = prometheus.NewCounterFunc(
|
|
prometheus.CounterOpts{
|
|
Name: "endlessh_trapped_time_seconds_total",
|
|
Help: "Total seconds clients spent on endlessh.",
|
|
}, func() float64 {
|
|
return float64(numTotalMilliseconds) / 1000
|
|
},
|
|
)
|
|
clientIP = prometheus.NewCounterVec(
|
|
prometheus.CounterOpts{
|
|
Name: "endlessh_client_open_count",
|
|
Help: "Number of connections of clients.",
|
|
},
|
|
[]string{"ip", "geohash", "country", "location"},
|
|
)
|
|
clientSeconds = prometheus.NewCounterVec(
|
|
prometheus.CounterOpts{
|
|
Name: "endlessh_client_trapped_time_seconds",
|
|
Help: "Seconds a client spends on endlessh.",
|
|
},
|
|
[]string{"ip"},
|
|
)
|
|
prometheus.MustRegister(totalClients)
|
|
prometheus.MustRegister(totalClientsClosed)
|
|
prometheus.MustRegister(totalBytes)
|
|
prometheus.MustRegister(totalSeconds)
|
|
prometheus.MustRegister(clientIP)
|
|
prometheus.MustRegister(clientSeconds)
|
|
http.Handle("/"+prometheusEntry, promhttp.Handler())
|
|
go func() {
|
|
glog.Infof("Starting Prometheus on %v:%v, entry point is /%v", connHost, prometheusPort, prometheusEntry)
|
|
http.ListenAndServe(connHost+":"+prometheusPort, nil)
|
|
}()
|
|
}
|
|
|
|
func main() {
|
|
intervalMs := flag.Int("interval_ms", 1000, "Message millisecond delay")
|
|
bannerMaxLength := flag.Int64("line_length", 32, "Maximum banner line length")
|
|
maxClients := flag.Int64("max_clients", 4096, "Maximum number of clients")
|
|
connType := flag.String("conn_type", "tcp", "Connection type. Possible values are tcp, tcp4, tcp6")
|
|
connHost := flag.String("host", "0.0.0.0", "Listening address")
|
|
connPort := flag.String("port", "2222", "Listening port")
|
|
enablePrometheus := flag.Bool("enable_prometheus", false, "Enable prometheus")
|
|
prometheusPort := flag.String("prometheus_port", "2112", "The port for prometheus")
|
|
prometheusEntry := flag.String("prometheus_entry", "metrics", "Entry point for prometheus")
|
|
geoipSupplier := flag.String("geoip_supplier", "ip-api", "Supplier to obtain Geohash of IPs. Possible values are \"ip-api\", \"freegeoip\"")
|
|
|
|
flag.Usage = func() {
|
|
fmt.Fprintf(flag.CommandLine.Output(), "Usage of %v \n", os.Args[0])
|
|
flag.PrintDefaults()
|
|
}
|
|
flag.Parse()
|
|
|
|
if *enablePrometheus {
|
|
initPrometheus(*connHost, *prometheusPort, *prometheusEntry)
|
|
}
|
|
|
|
rand.Seed(time.Now().UnixNano())
|
|
interval := time.Duration(*intervalMs) * time.Millisecond
|
|
// Listen for incoming connections.
|
|
if *connType == "tcp6" && *connHost == "0.0.0.0" {
|
|
*connHost = "[::]"
|
|
}
|
|
l, err := net.Listen(*connType, *connHost+":"+*connPort)
|
|
if err != nil {
|
|
glog.Errorf("Error listening: %v", err)
|
|
os.Exit(1)
|
|
}
|
|
// Close the listener when the application closes.
|
|
defer l.Close()
|
|
glog.Infof("Listening on %v:%v", *connHost, *connPort)
|
|
|
|
clients := make(chan *client, *maxClients)
|
|
go func() {
|
|
for {
|
|
c, more := <-clients
|
|
if !more {
|
|
return
|
|
}
|
|
if time.Now().Before(c.next) {
|
|
time.Sleep(c.next.Sub(time.Now()))
|
|
}
|
|
err := c.Send(*bannerMaxLength)
|
|
if err != nil {
|
|
c.Close()
|
|
continue
|
|
}
|
|
go func() { clients <- c }()
|
|
}
|
|
}()
|
|
listener := func() {
|
|
for {
|
|
// Listen for an incoming connection.
|
|
conn, err := l.Accept()
|
|
if err != nil {
|
|
glog.Errorf("Error accepting: %v", err)
|
|
os.Exit(1)
|
|
}
|
|
// Handle connections in a new goroutine.
|
|
for numCurrentClients >= *maxClients {
|
|
time.Sleep(interval)
|
|
}
|
|
clients <- NewClient(conn, interval, *maxClients, *geoipSupplier)
|
|
}
|
|
}
|
|
listener()
|
|
}
|