2021-02-02 12:05:47 +01:00
|
|
|
// Package netxlogger is a logger for netx events.
|
|
|
|
//
|
|
|
|
// This package is a fork of github.com/ooni/netx/x/logger where
|
|
|
|
// we applied ooni/probe-engine specific customisations.
|
|
|
|
package netxlogger
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/ooni/probe-cli/v3/internal/engine/legacy/netx/modelx"
|
2021-06-08 12:56:39 +02:00
|
|
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/tlsx"
|
2021-02-02 12:05:47 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// Logger is the interface we expect from a logger
|
|
|
|
type Logger interface {
|
|
|
|
Debug(msg string)
|
|
|
|
Debugf(format string, v ...interface{})
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handler is a handler that logs events.
|
|
|
|
type Handler struct {
|
|
|
|
logger Logger
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewHandler returns a new logging handler.
|
|
|
|
func NewHandler(logger Logger) *Handler {
|
|
|
|
return &Handler{logger: logger}
|
|
|
|
}
|
|
|
|
|
|
|
|
// OnMeasurement logs the specific measurement
|
|
|
|
func (h *Handler) OnMeasurement(m modelx.Measurement) {
|
|
|
|
// DNS
|
|
|
|
if m.ResolveStart != nil {
|
|
|
|
h.logger.Debugf(
|
2021-06-23 13:36:45 +02:00
|
|
|
"resolving: %s",
|
2021-02-02 12:05:47 +01:00
|
|
|
m.ResolveStart.Hostname,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
if m.ResolveDone != nil {
|
|
|
|
h.logger.Debugf(
|
2021-06-23 13:36:45 +02:00
|
|
|
"resolve done: %s, %s",
|
2021-02-02 12:05:47 +01:00
|
|
|
fmtError(m.ResolveDone.Error),
|
|
|
|
m.ResolveDone.Addresses,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Syscalls
|
|
|
|
if m.Connect != nil {
|
|
|
|
h.logger.Debugf(
|
2021-06-23 13:36:45 +02:00
|
|
|
"connect done: %s, %s (rtt=%s)",
|
2021-02-02 12:05:47 +01:00
|
|
|
fmtError(m.Connect.Error),
|
|
|
|
m.Connect.RemoteAddress,
|
|
|
|
m.Connect.SyscallDuration,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// TLS
|
|
|
|
if m.TLSHandshakeStart != nil {
|
|
|
|
h.logger.Debugf(
|
2021-06-23 13:36:45 +02:00
|
|
|
"TLS handshake: (forceSNI='%s')",
|
2021-02-02 12:05:47 +01:00
|
|
|
m.TLSHandshakeStart.SNI,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
if m.TLSHandshakeDone != nil {
|
|
|
|
h.logger.Debugf(
|
2021-06-23 13:36:45 +02:00
|
|
|
"TLS done: %s, %s (alpn='%s')",
|
2021-02-02 12:05:47 +01:00
|
|
|
fmtError(m.TLSHandshakeDone.Error),
|
|
|
|
tlsx.VersionString(m.TLSHandshakeDone.ConnectionState.Version),
|
|
|
|
m.TLSHandshakeDone.ConnectionState.NegotiatedProtocol,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// HTTP round trip
|
|
|
|
if m.HTTPRequestHeadersDone != nil {
|
|
|
|
proto := "HTTP/1.1"
|
|
|
|
for key := range m.HTTPRequestHeadersDone.Headers {
|
|
|
|
if strings.HasPrefix(key, ":") {
|
|
|
|
proto = "HTTP/2.0"
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
h.logger.Debugf(
|
2021-06-23 13:36:45 +02:00
|
|
|
"> %s %s %s",
|
2021-02-02 12:05:47 +01:00
|
|
|
m.HTTPRequestHeadersDone.Method,
|
|
|
|
m.HTTPRequestHeadersDone.URL.RequestURI(),
|
|
|
|
proto,
|
|
|
|
)
|
|
|
|
if proto == "HTTP/2.0" {
|
|
|
|
h.logger.Debugf(
|
2021-06-23 13:36:45 +02:00
|
|
|
"> Host: %s",
|
2021-02-02 12:05:47 +01:00
|
|
|
m.HTTPRequestHeadersDone.URL.Host,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
for key, values := range m.HTTPRequestHeadersDone.Headers {
|
|
|
|
if strings.HasPrefix(key, ":") {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for _, value := range values {
|
|
|
|
h.logger.Debugf(
|
2021-06-23 13:36:45 +02:00
|
|
|
"> %s: %s",
|
2021-02-02 12:05:47 +01:00
|
|
|
key, value,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
2021-06-23 13:36:45 +02:00
|
|
|
h.logger.Debug(">")
|
2021-02-02 12:05:47 +01:00
|
|
|
}
|
|
|
|
if m.HTTPRequestDone != nil {
|
2021-06-23 13:36:45 +02:00
|
|
|
h.logger.Debug("request sent; waiting for response")
|
2021-02-02 12:05:47 +01:00
|
|
|
}
|
|
|
|
if m.HTTPResponseStart != nil {
|
2021-06-23 13:36:45 +02:00
|
|
|
h.logger.Debug("start receiving response")
|
2021-02-02 12:05:47 +01:00
|
|
|
}
|
|
|
|
if m.HTTPRoundTripDone != nil && m.HTTPRoundTripDone.Error == nil {
|
|
|
|
h.logger.Debugf(
|
2021-06-23 13:36:45 +02:00
|
|
|
"< %s %d %s",
|
2021-02-02 12:05:47 +01:00
|
|
|
m.HTTPRoundTripDone.ResponseProto,
|
|
|
|
m.HTTPRoundTripDone.ResponseStatusCode,
|
|
|
|
http.StatusText(int(m.HTTPRoundTripDone.ResponseStatusCode)),
|
|
|
|
)
|
|
|
|
for key, values := range m.HTTPRoundTripDone.ResponseHeaders {
|
|
|
|
for _, value := range values {
|
|
|
|
h.logger.Debugf(
|
2021-06-23 13:36:45 +02:00
|
|
|
"< %s: %s",
|
2021-02-02 12:05:47 +01:00
|
|
|
key, value,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
2021-06-23 13:36:45 +02:00
|
|
|
h.logger.Debug("<")
|
2021-02-02 12:05:47 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// HTTP response body
|
|
|
|
if m.HTTPResponseBodyPart != nil {
|
|
|
|
h.logger.Debugf(
|
2021-06-23 13:36:45 +02:00
|
|
|
"body part: %s, %d",
|
2021-02-02 12:05:47 +01:00
|
|
|
fmtError(m.HTTPResponseBodyPart.Error),
|
|
|
|
len(m.HTTPResponseBodyPart.Data),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
if m.HTTPResponseDone != nil {
|
2021-06-23 13:36:45 +02:00
|
|
|
h.logger.Debug(
|
|
|
|
"end of response",
|
2021-02-02 12:05:47 +01:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func fmtError(err error) (s string) {
|
|
|
|
s = "success"
|
|
|
|
if err != nil {
|
|
|
|
s = err.Error()
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|