2019-08-22 11:27:18 +02:00
|
|
|
package opentsdbhttp
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2019-12-13 23:29:14 +01:00
|
|
|
"net"
|
2019-08-22 11:27:18 +02:00
|
|
|
"net/http"
|
2019-12-13 23:29:14 +01:00
|
|
|
"sync"
|
2019-08-22 11:27:18 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/httpserver"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/logger"
|
2019-12-13 23:29:14 +01:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/netutil"
|
2019-08-22 11:27:18 +02:00
|
|
|
"github.com/VictoriaMetrics/metrics"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2020-02-23 12:35:47 +01:00
|
|
|
writeRequests = metrics.NewCounter(`vm_opentsdbhttp_requests_total{name="write", net="tcp"}`)
|
|
|
|
writeErrors = metrics.NewCounter(`vm_opentsdbhttp_request_errors_total{name="write", net="tcp"}`)
|
2019-08-22 11:27:18 +02:00
|
|
|
)
|
|
|
|
|
2019-12-13 23:29:14 +01:00
|
|
|
// Server represents HTTP OpenTSDB server.
|
|
|
|
type Server struct {
|
|
|
|
s *http.Server
|
|
|
|
ln net.Listener
|
|
|
|
wg sync.WaitGroup
|
|
|
|
}
|
2019-08-22 11:27:18 +02:00
|
|
|
|
2019-12-13 23:29:14 +01:00
|
|
|
// MustStart starts HTTP OpenTSDB server on the given addr.
|
|
|
|
//
|
|
|
|
// MustStop must be called on the returned server when it is no longer needed.
|
2020-02-23 12:35:47 +01:00
|
|
|
func MustStart(addr string, insertHandler func(req *http.Request) error) *Server {
|
2019-08-22 11:27:18 +02:00
|
|
|
logger.Infof("starting HTTP OpenTSDB server at %q", addr)
|
2019-12-13 23:29:14 +01:00
|
|
|
lnTCP, err := netutil.NewTCPListener("opentsdbhttp", addr)
|
|
|
|
if err != nil {
|
|
|
|
logger.Fatalf("cannot start HTTP OpenTSDB collector at %q: %s", addr, err)
|
|
|
|
}
|
2020-02-23 12:35:47 +01:00
|
|
|
return MustServe(lnTCP, insertHandler)
|
2019-12-13 23:29:14 +01:00
|
|
|
}
|
|
|
|
|
2020-01-16 12:03:28 +01:00
|
|
|
// MustServe serves OpenTSDB HTTP put requests from ln.
|
2019-12-13 23:29:14 +01:00
|
|
|
//
|
|
|
|
// MustStop must be called on the returned server when it is no longer needed.
|
2020-02-23 12:35:47 +01:00
|
|
|
func MustServe(ln net.Listener, insertHandler func(req *http.Request) error) *Server {
|
|
|
|
h := newRequestHandler(insertHandler)
|
2019-12-13 23:29:14 +01:00
|
|
|
hs := &http.Server{
|
|
|
|
Handler: h,
|
2019-08-22 11:27:18 +02:00
|
|
|
ReadTimeout: 30 * time.Second,
|
|
|
|
WriteTimeout: 10 * time.Second,
|
|
|
|
}
|
2019-12-13 23:29:14 +01:00
|
|
|
s := &Server{
|
|
|
|
s: hs,
|
|
|
|
ln: ln,
|
|
|
|
}
|
|
|
|
s.wg.Add(1)
|
2019-08-22 11:27:18 +02:00
|
|
|
go func() {
|
2019-12-13 23:29:14 +01:00
|
|
|
defer s.wg.Done()
|
|
|
|
err := s.s.Serve(s.ln)
|
2019-08-22 11:27:18 +02:00
|
|
|
if err == http.ErrServerClosed {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if err != nil {
|
2019-12-13 23:29:14 +01:00
|
|
|
logger.Fatalf("error serving HTTP OpenTSDB at %q: %s", s.ln.Addr(), err)
|
2019-08-22 11:27:18 +02:00
|
|
|
}
|
|
|
|
}()
|
2019-12-13 23:29:14 +01:00
|
|
|
return s
|
2019-08-22 11:27:18 +02:00
|
|
|
}
|
|
|
|
|
2019-12-13 23:29:14 +01:00
|
|
|
// Wait waits until the server is stopped with MustStop.
|
|
|
|
func (s *Server) Wait() {
|
|
|
|
s.wg.Wait()
|
2019-08-22 11:27:18 +02:00
|
|
|
}
|
|
|
|
|
2019-12-13 23:29:14 +01:00
|
|
|
// MustStop stops HTTP OpenTSDB server.
|
|
|
|
func (s *Server) MustStop() {
|
|
|
|
logger.Infof("stopping HTTP OpenTSDB server at %q...", s.ln.Addr())
|
2019-08-22 11:27:18 +02:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
|
|
|
|
defer cancel()
|
2019-12-13 23:29:14 +01:00
|
|
|
if err := s.s.Shutdown(ctx); err != nil {
|
|
|
|
logger.Fatalf("cannot close HTTP OpenTSDB server at %q: %s", s.ln.Addr(), err)
|
|
|
|
}
|
|
|
|
s.wg.Wait()
|
|
|
|
logger.Infof("OpenTSDB HTTP server at %q has been stopped", s.ln.Addr())
|
|
|
|
}
|
|
|
|
|
2020-02-23 12:35:47 +01:00
|
|
|
func newRequestHandler(insertHandler func(req *http.Request) error) http.Handler {
|
|
|
|
rh := func(w http.ResponseWriter, req *http.Request) {
|
|
|
|
writeRequests.Inc()
|
|
|
|
if err := insertHandler(req); err != nil {
|
|
|
|
writeErrors.Inc()
|
|
|
|
httpserver.Errorf(w, "error in %q: %s", req.URL.Path, err)
|
|
|
|
return
|
2019-12-13 23:29:14 +01:00
|
|
|
}
|
2020-02-23 12:35:47 +01:00
|
|
|
w.WriteHeader(http.StatusNoContent)
|
2019-08-22 11:27:18 +02:00
|
|
|
}
|
2019-12-13 23:29:14 +01:00
|
|
|
return http.HandlerFunc(rh)
|
2019-08-22 11:27:18 +02:00
|
|
|
}
|