fix(tracex): generate archival from single transaction-done event (#780)

Tracex contained some fragile code that assembled HTTP measurements
from scattered events, which worked because we were sure we were
performing a single measurement at any given time.

This diff restructures the code to emit a transaction-start and a
transaction-done events only. We have basically removed all the other
events (which we were not using). We kept the transaction-start
though, because it may be useful to see it when reading events. In
any case, what matters here is that we're now using the transaction-done
event aline to generate the archival HTTP measurement.

Hence, the original issue has been addressed. We will possibly
do more refactoring in the future, but for now this seems sufficient.

Part of https://github.com/ooni/probe/issues/2121
This commit is contained in:
Simone Basso 2022-06-01 19:27:47 +02:00 committed by GitHub
parent 153bfe1c57
commit 6212daa54a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 357 additions and 700 deletions

View File

@ -37,18 +37,15 @@ func TestGetterWithVeryShortTimeout(t *testing.T) {
if tk.Failure == nil || *tk.Failure != "generic_timeout_error" { if tk.Failure == nil || *tk.Failure != "generic_timeout_error" {
t.Fatal("not the Failure we expected") t.Fatal("not the Failure we expected")
} }
if len(tk.NetworkEvents) != 3 { if len(tk.NetworkEvents) != 2 {
t.Fatal("not the NetworkEvents we expected") t.Fatal("not the NetworkEvents we expected")
} }
if tk.NetworkEvents[0].Operation != "http_transaction_start" { if tk.NetworkEvents[0].Operation != "http_transaction_start" {
t.Fatal("not the NetworkEvents[0].Operation we expected") t.Fatal("not the NetworkEvents[0].Operation we expected")
} }
if tk.NetworkEvents[1].Operation != "http_request_metadata" { if tk.NetworkEvents[1].Operation != "http_transaction_done" {
t.Fatal("not the NetworkEvents[1].Operation we expected") t.Fatal("not the NetworkEvents[1].Operation we expected")
} }
if tk.NetworkEvents[2].Operation != "http_transaction_done" {
t.Fatal("not the NetworkEvents[2].Operation we expected")
}
if len(tk.Queries) != 0 { if len(tk.Queries) != 0 {
t.Fatal("not the Queries we expected") t.Fatal("not the Queries we expected")
} }
@ -104,18 +101,15 @@ func TestGetterWithCancelledContextVanilla(t *testing.T) {
if tk.Failure == nil || !strings.HasSuffix(*tk.Failure, "interrupted") { if tk.Failure == nil || !strings.HasSuffix(*tk.Failure, "interrupted") {
t.Fatal("not the Failure we expected") t.Fatal("not the Failure we expected")
} }
if len(tk.NetworkEvents) != 3 { if len(tk.NetworkEvents) != 2 {
t.Fatal("not the NetworkEvents we expected") t.Fatal("not the NetworkEvents we expected")
} }
if tk.NetworkEvents[0].Operation != "http_transaction_start" { if tk.NetworkEvents[0].Operation != "http_transaction_start" {
t.Fatal("not the NetworkEvents[0].Operation we expected") t.Fatal("not the NetworkEvents[0].Operation we expected")
} }
if tk.NetworkEvents[1].Operation != "http_request_metadata" { if tk.NetworkEvents[1].Operation != "http_transaction_done" {
t.Fatal("not the NetworkEvents[1].Operation we expected") t.Fatal("not the NetworkEvents[1].Operation we expected")
} }
if tk.NetworkEvents[2].Operation != "http_transaction_done" {
t.Fatal("not the NetworkEvents[2].Operation we expected")
}
if len(tk.Queries) != 0 { if len(tk.Queries) != 0 {
t.Fatal("not the Queries we expected") t.Fatal("not the Queries we expected")
} }
@ -172,18 +166,15 @@ func TestGetterWithCancelledContextAndMethod(t *testing.T) {
if tk.Failure == nil || !strings.HasSuffix(*tk.Failure, "interrupted") { if tk.Failure == nil || !strings.HasSuffix(*tk.Failure, "interrupted") {
t.Fatal("not the Failure we expected") t.Fatal("not the Failure we expected")
} }
if len(tk.NetworkEvents) != 3 { if len(tk.NetworkEvents) != 2 {
t.Fatal("not the NetworkEvents we expected") t.Fatal("not the NetworkEvents we expected")
} }
if tk.NetworkEvents[0].Operation != "http_transaction_start" { if tk.NetworkEvents[0].Operation != "http_transaction_start" {
t.Fatal("not the NetworkEvents[0].Operation we expected") t.Fatal("not the NetworkEvents[0].Operation we expected")
} }
if tk.NetworkEvents[1].Operation != "http_request_metadata" { if tk.NetworkEvents[1].Operation != "http_transaction_done" {
t.Fatal("not the NetworkEvents[1].Operation we expected") t.Fatal("not the NetworkEvents[1].Operation we expected")
} }
if tk.NetworkEvents[2].Operation != "http_transaction_done" {
t.Fatal("not the NetworkEvents[2].Operation we expected")
}
if len(tk.Queries) != 0 { if len(tk.Queries) != 0 {
t.Fatal("not the Queries we expected") t.Fatal("not the Queries we expected")
} }
@ -242,16 +233,13 @@ func TestGetterWithCancelledContextNoFollowRedirects(t *testing.T) {
if tk.Failure == nil || !strings.HasSuffix(*tk.Failure, "interrupted") { if tk.Failure == nil || !strings.HasSuffix(*tk.Failure, "interrupted") {
t.Fatal("not the Failure we expected") t.Fatal("not the Failure we expected")
} }
if len(tk.NetworkEvents) != 3 { if len(tk.NetworkEvents) != 2 {
t.Fatal("not the NetworkEvents we expected") t.Fatal("not the NetworkEvents we expected")
} }
if tk.NetworkEvents[0].Operation != "http_transaction_start" { if tk.NetworkEvents[0].Operation != "http_transaction_start" {
t.Fatal("not the NetworkEvents[0].Operation we expected") t.Fatal("not the NetworkEvents[0].Operation we expected")
} }
if tk.NetworkEvents[1].Operation != "http_request_metadata" { if tk.NetworkEvents[1].Operation != "http_transaction_done" {
t.Fatal("not the NetworkEvents[1].Operation we expected")
}
if tk.NetworkEvents[2].Operation != "http_transaction_done" {
t.Fatal("not the NetworkEvents[2].Operation we expected") t.Fatal("not the NetworkEvents[2].Operation we expected")
} }
if len(tk.Queries) != 0 { if len(tk.Queries) != 0 {
@ -422,23 +410,18 @@ func TestGetterIntegrationHTTPS(t *testing.T) {
t.Fatal("not the Failure we expected") t.Fatal("not the Failure we expected")
} }
var ( var (
httpTransactionStart bool httpTransactionStart bool
httpRequestMetadata bool resolveStart bool
resolveStart bool resolveDone bool
resolveDone bool connect bool
connect bool tlsHandshakeStart bool
tlsHandshakeStart bool tlsHandshakeDone bool
tlsHandshakeDone bool httpTransactionDone bool
httpResponseMetadata bool
httpResponseBodySnapshot bool
httpTransactionDone bool
) )
for _, ev := range tk.NetworkEvents { for _, ev := range tk.NetworkEvents {
switch ev.Operation { switch ev.Operation {
case "http_transaction_start": case "http_transaction_start":
httpTransactionStart = true httpTransactionStart = true
case "http_request_metadata":
httpRequestMetadata = true
case "resolve_start": case "resolve_start":
resolveStart = true resolveStart = true
case "resolve_done": case "resolve_done":
@ -449,24 +432,17 @@ func TestGetterIntegrationHTTPS(t *testing.T) {
tlsHandshakeStart = true tlsHandshakeStart = true
case "tls_handshake_done": case "tls_handshake_done":
tlsHandshakeDone = true tlsHandshakeDone = true
case "http_response_metadata":
httpResponseMetadata = true
case "http_response_body_snapshot":
httpResponseBodySnapshot = true
case "http_transaction_done": case "http_transaction_done":
httpTransactionDone = true httpTransactionDone = true
} }
} }
ok := true ok := true
ok = ok && httpTransactionStart ok = ok && httpTransactionStart
ok = ok && httpRequestMetadata
ok = ok && resolveStart ok = ok && resolveStart
ok = ok && resolveDone ok = ok && resolveDone
ok = ok && connect ok = ok && connect
ok = ok && tlsHandshakeStart ok = ok && tlsHandshakeStart
ok = ok && tlsHandshakeDone ok = ok && tlsHandshakeDone
ok = ok && httpResponseMetadata
ok = ok && httpResponseBodySnapshot
ok = ok && httpTransactionDone ok = ok && httpTransactionDone
if !ok { if !ok {
t.Fatal("not the NetworkEvents we expected") t.Fatal("not the NetworkEvents we expected")
@ -657,23 +633,18 @@ func TestGetterHTTPSWithTunnel(t *testing.T) {
t.Fatal("not the Failure we expected") t.Fatal("not the Failure we expected")
} }
var ( var (
httpTransactionStart bool httpTransactionStart bool
httpRequestMetadata bool resolveStart bool
resolveStart bool resolveDone bool
resolveDone bool connect bool
connect bool tlsHandshakeStart bool
tlsHandshakeStart bool tlsHandshakeDone bool
tlsHandshakeDone bool httpTransactionDone bool
httpResponseMetadata bool
httpResponseBodySnapshot bool
httpTransactionDone bool
) )
for _, ev := range tk.NetworkEvents { for _, ev := range tk.NetworkEvents {
switch ev.Operation { switch ev.Operation {
case "http_transaction_start": case "http_transaction_start":
httpTransactionStart = true httpTransactionStart = true
case "http_request_metadata":
httpRequestMetadata = true
case "resolve_start": case "resolve_start":
resolveStart = true resolveStart = true
case "resolve_done": case "resolve_done":
@ -684,24 +655,17 @@ func TestGetterHTTPSWithTunnel(t *testing.T) {
tlsHandshakeStart = true tlsHandshakeStart = true
case "tls_handshake_done": case "tls_handshake_done":
tlsHandshakeDone = true tlsHandshakeDone = true
case "http_response_metadata":
httpResponseMetadata = true
case "http_response_body_snapshot":
httpResponseBodySnapshot = true
case "http_transaction_done": case "http_transaction_done":
httpTransactionDone = true httpTransactionDone = true
} }
} }
ok := true ok := true
ok = ok && httpTransactionStart ok = ok && httpTransactionStart
ok = ok && httpRequestMetadata
ok = ok && resolveStart == false ok = ok && resolveStart == false
ok = ok && resolveDone == false ok = ok && resolveDone == false
ok = ok && connect ok = ok && connect
ok = ok && tlsHandshakeStart ok = ok && tlsHandshakeStart
ok = ok && tlsHandshakeDone ok = ok && tlsHandshakeDone
ok = ok && httpResponseMetadata
ok = ok && httpResponseBodySnapshot
ok = ok && httpTransactionDone ok = ok && httpTransactionDone
if !ok { if !ok {
t.Fatalf("not the NetworkEvents we expected: %+v", tk.NetworkEvents) t.Fatalf("not the NetworkEvents we expected: %+v", tk.NetworkEvents)

View File

@ -187,10 +187,6 @@ func NewHTTPTransport(config Config) model.HTTPTransport {
txp = &netxlite.HTTPTransportLogger{Logger: config.Logger, HTTPTransport: txp} txp = &netxlite.HTTPTransportLogger{Logger: config.Logger, HTTPTransport: txp}
} }
if config.HTTPSaver != nil { if config.HTTPSaver != nil {
txp = &tracex.SaverMetadataHTTPTransport{
HTTPTransport: txp, Saver: config.HTTPSaver}
txp = &tracex.SaverBodyHTTPTransport{
HTTPTransport: txp, Saver: config.HTTPSaver}
txp = &tracex.SaverTransactionHTTPTransport{ txp = &tracex.SaverTransactionHTTPTransport{
HTTPTransport: txp, Saver: config.HTTPSaver} HTTPTransport: txp, Saver: config.HTTPSaver}
} }

View File

@ -511,21 +511,10 @@ func TestNewWithSaver(t *testing.T) {
if stxptxp.Saver != saver { if stxptxp.Saver != saver {
t.Fatal("not the logger we expected") t.Fatal("not the logger we expected")
} }
sbtxp, ok := stxptxp.HTTPTransport.(*tracex.SaverBodyHTTPTransport) if stxptxp.Saver != saver {
if !ok {
t.Fatal("not the transport we expected")
}
if sbtxp.Saver != saver {
t.Fatal("not the logger we expected") t.Fatal("not the logger we expected")
} }
smtxp, ok := sbtxp.HTTPTransport.(*tracex.SaverMetadataHTTPTransport) if _, ok := stxptxp.HTTPTransport.(*netxlite.HTTPTransportWrapper); !ok {
if !ok {
t.Fatal("not the transport we expected")
}
if smtxp.Saver != saver {
t.Fatal("not the logger we expected")
}
if _, ok := smtxp.HTTPTransport.(*netxlite.HTTPTransportWrapper); !ok {
t.Fatal("not the transport we expected") t.Fatal("not the transport we expected")
} }
} }

View File

@ -106,6 +106,8 @@ func httpAddHeaders(
destList *[]HTTPHeader, destList *[]HTTPHeader,
destMap *map[string]MaybeBinaryValue, destMap *map[string]MaybeBinaryValue,
) { ) {
*destList = []HTTPHeader{}
*destMap = make(map[string]model.ArchivalMaybeBinaryData)
for key, values := range source { for key, values := range source {
for index, value := range values { for index, value := range values {
value := MaybeBinaryValue{Value: value} value := MaybeBinaryValue{Value: value}
@ -144,29 +146,20 @@ func newRequestList(begin time.Time, events []Event) []RequestEntry {
for _, wrapper := range events { for _, wrapper := range events {
ev := wrapper.Value() ev := wrapper.Value()
switch wrapper.(type) { switch wrapper.(type) {
case *EventHTTPTransactionStart: case *EventHTTPTransactionDone:
entry = RequestEntry{} entry = RequestEntry{}
entry.T = ev.Time.Sub(begin).Seconds() entry.T = ev.Time.Sub(begin).Seconds()
case *EventHTTPRequestBodySnapshot:
entry.Request.Body.Value = string(ev.Data)
entry.Request.BodyIsTruncated = ev.DataIsTruncated
case *EventHTTPRequestMetadata:
entry.Request.Headers = make(map[string]MaybeBinaryValue)
httpAddHeaders( httpAddHeaders(
ev.HTTPRequestHeaders, &entry.Request.HeadersList, &entry.Request.Headers) ev.HTTPRequestHeaders, &entry.Request.HeadersList, &entry.Request.Headers)
entry.Request.Method = ev.HTTPMethod entry.Request.Method = ev.HTTPMethod
entry.Request.URL = ev.HTTPURL entry.Request.URL = ev.HTTPURL
entry.Request.Transport = ev.Transport entry.Request.Transport = ev.Transport
case *EventHTTPResponseMetadata:
entry.Response.Headers = make(map[string]MaybeBinaryValue)
httpAddHeaders( httpAddHeaders(
ev.HTTPResponseHeaders, &entry.Response.HeadersList, &entry.Response.Headers) ev.HTTPResponseHeaders, &entry.Response.HeadersList, &entry.Response.Headers)
entry.Response.Code = int64(ev.HTTPStatusCode) entry.Response.Code = int64(ev.HTTPStatusCode)
entry.Response.Locations = ev.HTTPResponseHeaders.Values("Location") entry.Response.Locations = ev.HTTPResponseHeaders.Values("Location")
case *EventHTTPResponseBodySnapshot: entry.Response.Body.Value = string(ev.HTTPResponseBody)
entry.Response.Body.Value = string(ev.Data) entry.Response.BodyIsTruncated = ev.HTTPResponseBodyIsTruncated
entry.Response.BodyIsTruncated = ev.DataIsTruncated
case *EventHTTPTransactionDone:
entry.Failure = NewFailure(ev.Err) entry.Failure = NewFailure(ev.Err)
out = append(out, entry) out = append(out, entry)
} }

View File

@ -143,35 +143,27 @@ func TestNewRequestList(t *testing.T) {
name: "realistic run", name: "realistic run",
args: args{ args: args{
begin: begin, begin: begin,
events: []Event{&EventHTTPTransactionStart{&EventValue{ events: []Event{&EventHTTPTransactionDone{&EventValue{
Time: begin.Add(10 * time.Millisecond),
}}, &EventHTTPRequestBodySnapshot{&EventValue{
Data: []byte("deadbeef"),
DataIsTruncated: false,
}}, &EventHTTPRequestMetadata{&EventValue{
HTTPRequestHeaders: http.Header{ HTTPRequestHeaders: http.Header{
"User-Agent": []string{"miniooni/0.1.0-dev"}, "User-Agent": []string{"miniooni/0.1.0-dev"},
}, },
HTTPMethod: "POST", HTTPMethod: "POST",
HTTPURL: "https://www.example.com/submit", HTTPURL: "https://www.example.com/submit",
}}, &EventHTTPResponseMetadata{&EventValue{
HTTPResponseHeaders: http.Header{ HTTPResponseHeaders: http.Header{
"Server": []string{"miniooni/0.1.0-dev"}, "Server": []string{"miniooni/0.1.0-dev"},
}, },
HTTPStatusCode: 200, HTTPStatusCode: 200,
}}, &EventHTTPResponseBodySnapshot{&EventValue{ HTTPResponseBody: []byte("{}"),
Data: []byte("{}"), HTTPResponseBodyIsTruncated: false,
DataIsTruncated: false, Time: begin.Add(10 * time.Millisecond),
}}, &EventHTTPTransactionDone{&EventValue{}}, &EventHTTPTransactionStart{&EventValue{ }}, &EventHTTPTransactionDone{&EventValue{
Time: begin.Add(20 * time.Millisecond),
}}, &EventHTTPRequestMetadata{&EventValue{
HTTPRequestHeaders: http.Header{ HTTPRequestHeaders: http.Header{
"User-Agent": []string{"miniooni/0.1.0-dev"}, "User-Agent": []string{"miniooni/0.1.0-dev"},
}, },
HTTPMethod: "GET", HTTPMethod: "GET",
HTTPURL: "https://www.example.com/result", HTTPURL: "https://www.example.com/result",
}}, &EventHTTPTransactionDone{&EventValue{ Err: io.EOF,
Err: io.EOF, Time: begin.Add(20 * time.Millisecond),
}}}, }}},
}, },
want: []RequestEntry{{ want: []RequestEntry{{
@ -189,11 +181,15 @@ func TestNewRequestList(t *testing.T) {
Method: "GET", Method: "GET",
URL: "https://www.example.com/result", URL: "https://www.example.com/result",
}, },
Response: HTTPResponse{
HeadersList: []HTTPHeader{},
Headers: make(map[string]MaybeBinaryValue),
},
T: 0.02, T: 0.02,
}, { }, {
Request: HTTPRequest{ Request: HTTPRequest{
Body: MaybeBinaryValue{ Body: MaybeBinaryValue{
Value: "deadbeef", Value: "",
}, },
HeadersList: []HTTPHeader{{ HeadersList: []HTTPHeader{{
Key: "User-Agent", Key: "User-Agent",
@ -231,21 +227,19 @@ func TestNewRequestList(t *testing.T) {
name: "run with redirect and headers to sort", name: "run with redirect and headers to sort",
args: args{ args: args{
begin: begin, begin: begin,
events: []Event{&EventHTTPTransactionStart{&EventValue{ events: []Event{&EventHTTPTransactionDone{&EventValue{
Time: begin.Add(10 * time.Millisecond),
}}, &EventHTTPRequestMetadata{&EventValue{
HTTPRequestHeaders: http.Header{ HTTPRequestHeaders: http.Header{
"User-Agent": []string{"miniooni/0.1.0-dev"}, "User-Agent": []string{"miniooni/0.1.0-dev"},
}, },
HTTPMethod: "GET", HTTPMethod: "GET",
HTTPURL: "https://www.example.com/", HTTPURL: "https://www.example.com/",
}}, &EventHTTPResponseMetadata{&EventValue{
HTTPResponseHeaders: http.Header{ HTTPResponseHeaders: http.Header{
"Server": []string{"miniooni/0.1.0-dev"}, "Server": []string{"miniooni/0.1.0-dev"},
"Location": []string{"https://x.example.com", "https://y.example.com"}, "Location": []string{"https://x.example.com", "https://y.example.com"},
}, },
HTTPStatusCode: 302, HTTPStatusCode: 302,
}}, &EventHTTPTransactionDone{&EventValue{}}}, Time: begin.Add(10 * time.Millisecond),
}}},
}, },
want: []RequestEntry{{ want: []RequestEntry{{
Request: HTTPRequest{ Request: HTTPRequest{
@ -293,7 +287,7 @@ func TestNewRequestList(t *testing.T) {
for _, tt := range tests { for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) { t.Run(tt.name, func(t *testing.T) {
if got := NewRequestList(tt.args.begin, tt.args.events); !reflect.DeepEqual(got, tt.want) { if got := NewRequestList(tt.args.begin, tt.args.events); !reflect.DeepEqual(got, tt.want) {
t.Error(cmp.Diff(got, tt.want)) t.Error(cmp.Diff(tt.want, got))
} }
}) })
} }

View File

@ -147,32 +147,6 @@ func (ev *EventReadFromOperation) Name() string {
return netxlite.ReadFromOperation return netxlite.ReadFromOperation
} }
// EventHTTPRequestMetadata contains HTTP request metadata.
type EventHTTPRequestMetadata struct {
V *EventValue
}
func (ev *EventHTTPRequestMetadata) Value() *EventValue {
return ev.V
}
func (ev *EventHTTPRequestMetadata) Name() string {
return "http_request_metadata"
}
// EventHTTPResponseMetadata contains HTTP response metadata.
type EventHTTPResponseMetadata struct {
V *EventValue
}
func (ev *EventHTTPResponseMetadata) Value() *EventValue {
return ev.V
}
func (ev *EventHTTPResponseMetadata) Name() string {
return "http_response_metadata"
}
// EventHTTPTransactionStart is the beginning of an HTTP transaction. // EventHTTPTransactionStart is the beginning of an HTTP transaction.
type EventHTTPTransactionStart struct { type EventHTTPTransactionStart struct {
V *EventValue V *EventValue
@ -199,32 +173,6 @@ func (ev *EventHTTPTransactionDone) Name() string {
return "http_transaction_done" return "http_transaction_done"
} }
// EventHTTPRequestBodySnapshot contains a snapshot of the request body.
type EventHTTPRequestBodySnapshot struct {
V *EventValue
}
func (ev *EventHTTPRequestBodySnapshot) Value() *EventValue {
return ev.V
}
func (ev *EventHTTPRequestBodySnapshot) Name() string {
return "http_request_body_snapshot"
}
// EventHTTPResponseBodySnapshot contains a snapshot of the response body.
type EventHTTPResponseBodySnapshot struct {
V *EventValue
}
func (ev *EventHTTPResponseBodySnapshot) Value() *EventValue {
return ev.V
}
func (ev *EventHTTPResponseBodySnapshot) Name() string {
return "http_response_body_snapshot"
}
// EventConnectOperation contains information about the connect operation. // EventConnectOperation contains information about the connect operation.
type EventConnectOperation struct { type EventConnectOperation struct {
V *EventValue V *EventValue
@ -266,29 +214,30 @@ func (ev *EventWriteOperation) Name() string {
// Event is one of the events within a trace // Event is one of the events within a trace
type EventValue struct { type EventValue struct {
Addresses []string `json:",omitempty"` Addresses []string `json:",omitempty"`
Address string `json:",omitempty"` Address string `json:",omitempty"`
DNSQuery []byte `json:",omitempty"` DNSQuery []byte `json:",omitempty"`
DNSResponse []byte `json:",omitempty"` DNSResponse []byte `json:",omitempty"`
DataIsTruncated bool `json:",omitempty"` Data []byte `json:",omitempty"`
Data []byte `json:",omitempty"` Duration time.Duration `json:",omitempty"`
Duration time.Duration `json:",omitempty"` Err error `json:",omitempty"`
Err error `json:",omitempty"` HTTPMethod string `json:",omitempty"`
HTTPMethod string `json:",omitempty"` HTTPRequestHeaders http.Header `json:",omitempty"`
HTTPRequestHeaders http.Header `json:",omitempty"` HTTPResponseHeaders http.Header `json:",omitempty"`
HTTPResponseHeaders http.Header `json:",omitempty"` HTTPResponseBody []byte `json:",omitempty"`
HTTPStatusCode int `json:",omitempty"` HTTPResponseBodyIsTruncated bool `json:",omitempty"`
HTTPURL string `json:",omitempty"` HTTPStatusCode int `json:",omitempty"`
Hostname string `json:",omitempty"` HTTPURL string `json:",omitempty"`
NoTLSVerify bool `json:",omitempty"` Hostname string `json:",omitempty"`
NumBytes int `json:",omitempty"` NoTLSVerify bool `json:",omitempty"`
Proto string `json:",omitempty"` NumBytes int `json:",omitempty"`
TLSServerName string `json:",omitempty"` Proto string `json:",omitempty"`
TLSCipherSuite string `json:",omitempty"` TLSServerName string `json:",omitempty"`
TLSNegotiatedProto string `json:",omitempty"` TLSCipherSuite string `json:",omitempty"`
TLSNextProtos []string `json:",omitempty"` TLSNegotiatedProto string `json:",omitempty"`
TLSPeerCerts []*x509.Certificate `json:",omitempty"` TLSNextProtos []string `json:",omitempty"`
TLSVersion string `json:",omitempty"` TLSPeerCerts []*x509.Certificate `json:",omitempty"`
Time time.Time `json:",omitempty"` TLSVersion string `json:",omitempty"`
Transport string `json:",omitempty"` Time time.Time `json:",omitempty"`
Transport string `json:",omitempty"`
} }

View File

@ -6,7 +6,6 @@ package tracex
import ( import (
"bytes" "bytes"
"context"
"io" "io"
"net/http" "net/http"
"time" "time"
@ -15,34 +14,6 @@ import (
"github.com/ooni/probe-cli/v3/internal/netxlite" "github.com/ooni/probe-cli/v3/internal/netxlite"
) )
// SaverMetadataHTTPTransport is a RoundTripper that saves
// events related to HTTP request and response metadata
type SaverMetadataHTTPTransport struct {
model.HTTPTransport
Saver *Saver
}
// RoundTrip implements RoundTripper.RoundTrip
func (txp *SaverMetadataHTTPTransport) RoundTrip(req *http.Request) (*http.Response, error) {
txp.Saver.Write(&EventHTTPRequestMetadata{&EventValue{
HTTPRequestHeaders: httpCloneRequestHeaders(req),
HTTPMethod: req.Method,
HTTPURL: req.URL.String(),
Transport: txp.HTTPTransport.Network(),
Time: time.Now(),
}})
resp, err := txp.HTTPTransport.RoundTrip(req)
if err != nil {
return nil, err
}
txp.Saver.Write(&EventHTTPResponseMetadata{&EventValue{
HTTPResponseHeaders: resp.Header,
HTTPStatusCode: resp.StatusCode,
Time: time.Now(),
}})
return resp, err
}
// httpCloneRequestHeaders returns a clone of the headers where we have // httpCloneRequestHeaders returns a clone of the headers where we have
// also set the host header, which normally is not set by // also set the host header, which normally is not set by
// golang until it serializes the request itself. // golang until it serializes the request itself.
@ -59,81 +30,78 @@ func httpCloneRequestHeaders(req *http.Request) http.Header {
// SaverTransactionHTTPTransport is a RoundTripper that saves // SaverTransactionHTTPTransport is a RoundTripper that saves
// events related to the HTTP transaction // events related to the HTTP transaction
type SaverTransactionHTTPTransport struct { type SaverTransactionHTTPTransport struct {
model.HTTPTransport
Saver *Saver
}
// RoundTrip implements RoundTripper.RoundTrip
func (txp *SaverTransactionHTTPTransport) RoundTrip(req *http.Request) (*http.Response, error) {
txp.Saver.Write(&EventHTTPTransactionStart{&EventValue{
Time: time.Now(),
}})
resp, err := txp.HTTPTransport.RoundTrip(req)
txp.Saver.Write(&EventHTTPTransactionDone{&EventValue{
Err: err,
Time: time.Now(),
}})
return resp, err
}
// SaverBodyHTTPTransport is a RoundTripper that saves
// body events occurring during the round trip
type SaverBodyHTTPTransport struct {
model.HTTPTransport model.HTTPTransport
Saver *Saver Saver *Saver
SnapshotSize int SnapshotSize int64
} }
// RoundTrip implements RoundTripper.RoundTrip // HTTPRoundTrip performs the round trip with the given transport and
func (txp *SaverBodyHTTPTransport) RoundTrip(req *http.Request) (*http.Response, error) { // the given arguments and saves the results into the saver.
const defaultSnapSize = 1 << 17 //
snapsize := defaultSnapSize // The maxBodySnapshotSize argument controls the maximum size of the
if txp.SnapshotSize != 0 { // body snapshot that we collect along with the HTTP round trip.
snapsize = txp.SnapshotSize func (txp *SaverTransactionHTTPTransport) RoundTrip(req *http.Request) (*http.Response, error) {
}
if req.Body != nil { started := time.Now()
data, err := httpSaverSnapRead(req.Context(), req.Body, snapsize) txp.Saver.Write(&EventHTTPTransactionStart{&EventValue{
if err != nil { HTTPRequestHeaders: httpCloneRequestHeaders(req),
return nil, err HTTPMethod: req.Method,
} HTTPURL: req.URL.String(),
req.Body = httpSaverCompose(data, req.Body) Transport: txp.HTTPTransport.Network(),
txp.Saver.Write(&EventHTTPRequestBodySnapshot{&EventValue{ Time: started,
DataIsTruncated: len(data) >= snapsize,
Data: data,
Time: time.Now(),
}})
}
resp, err := txp.HTTPTransport.RoundTrip(req)
if err != nil {
return nil, err
}
data, err := httpSaverSnapRead(req.Context(), resp.Body, snapsize)
if err != nil {
resp.Body.Close()
return nil, err
}
resp.Body = httpSaverCompose(data, resp.Body)
txp.Saver.Write(&EventHTTPResponseBodySnapshot{&EventValue{
DataIsTruncated: len(data) >= snapsize,
Data: data,
Time: time.Now(),
}}) }})
ev := &EventValue{
HTTPRequestHeaders: httpCloneRequestHeaders(req),
HTTPMethod: req.Method,
HTTPURL: req.URL.String(),
Transport: txp.HTTPTransport.Network(),
Time: started,
}
defer txp.Saver.Write(&EventHTTPTransactionDone{ev})
resp, err := txp.HTTPTransport.RoundTrip(req)
if err != nil {
ev.Duration = time.Since(started)
ev.Err = err
return nil, err
}
ev.HTTPStatusCode = resp.StatusCode
ev.HTTPResponseHeaders = resp.Header.Clone()
maxBodySnapshotSize := txp.snapshotSize()
r := io.LimitReader(resp.Body, maxBodySnapshotSize)
body, err := netxlite.ReadAllContext(req.Context(), r)
if err != nil {
ev.Duration = time.Since(started)
ev.Err = err
return nil, err
}
resp.Body = &httpReadableAgainBody{ // allow for reading again the whole body
Reader: io.MultiReader(bytes.NewReader(body), resp.Body),
Closer: resp.Body,
}
ev.Duration = time.Since(started)
ev.HTTPResponseBody = body
ev.HTTPResponseBodyIsTruncated = int64(len(body)) >= maxBodySnapshotSize
return resp, nil return resp, nil
} }
func httpSaverSnapRead(ctx context.Context, r io.ReadCloser, snapsize int) ([]byte, error) { func (txp *SaverTransactionHTTPTransport) snapshotSize() int64 {
return netxlite.ReadAllContext(ctx, io.LimitReader(r, int64(snapsize))) if txp.SnapshotSize > 0 {
return txp.SnapshotSize
}
return 1 << 17
} }
func httpSaverCompose(data []byte, r io.ReadCloser) io.ReadCloser { type httpReadableAgainBody struct {
return httpSaverReadCloser{Closer: r, Reader: io.MultiReader(bytes.NewReader(data), r)}
}
type httpSaverReadCloser struct {
io.Closer
io.Reader io.Reader
io.Closer
} }
var _ model.HTTPTransport = &SaverMetadataHTTPTransport{}
var _ model.HTTPTransport = &SaverBodyHTTPTransport{}
var _ model.HTTPTransport = &SaverTransactionHTTPTransport{} var _ model.HTTPTransport = &SaverTransactionHTTPTransport{}

View File

@ -1,391 +1,246 @@
package tracex package tracex
import ( import (
"bytes"
"context" "context"
"errors" "errors"
"io" "io"
"net" "net"
"net/http" "net/http"
"net/url" "net/url"
"strings"
"testing" "testing"
"time"
"github.com/ooni/probe-cli/v3/internal/model" "github.com/ooni/probe-cli/v3/internal/model"
"github.com/ooni/probe-cli/v3/internal/model/mocks"
"github.com/ooni/probe-cli/v3/internal/netxlite" "github.com/ooni/probe-cli/v3/internal/netxlite"
"github.com/ooni/probe-cli/v3/internal/netxlite/filtering"
) )
func TestSaverMetadataSuccess(t *testing.T) { func TestSaverTransactionHTTPTransport(t *testing.T) {
if testing.Short() {
t.Skip("skip test in short mode")
}
saver := &Saver{}
txp := SaverMetadataHTTPTransport{
HTTPTransport: netxlite.NewHTTPTransportStdlib(model.DiscardLogger),
Saver: saver,
}
req, err := http.NewRequest("GET", "https://www.google.com", nil)
if err != nil {
t.Fatal(err)
}
req.Header.Add("User-Agent", "miniooni/0.1.0-dev")
resp, err := txp.RoundTrip(req)
if err != nil {
t.Fatal("not the error we expected")
}
if resp == nil {
t.Fatal("expected non nil response here")
}
ev := saver.Read()
if len(ev) != 2 {
t.Fatal("expected two events")
}
//
if ev[0].Value().HTTPMethod != "GET" {
t.Fatal("unexpected Method")
}
if len(ev[0].Value().HTTPRequestHeaders) <= 0 {
t.Fatal("unexpected Headers")
}
if ev[0].Value().HTTPURL != "https://www.google.com" {
t.Fatal("unexpected URL")
}
if ev[0].Name() != "http_request_metadata" {
t.Fatal("unexpected Name")
}
if !ev[0].Value().Time.Before(time.Now()) {
t.Fatal("unexpected Time")
}
//
if ev[1].Value().HTTPStatusCode != 200 {
t.Fatal("unexpected StatusCode")
}
if len(ev[1].Value().HTTPResponseHeaders) <= 0 {
t.Fatal("unexpected Headers")
}
if ev[1].Name() != "http_response_metadata" {
t.Fatal("unexpected Name")
}
if !ev[1].Value().Time.After(ev[0].Value().Time) {
t.Fatal("unexpected Time")
}
}
func TestSaverMetadataFailure(t *testing.T) { startServer := func(t *testing.T, action filtering.HTTPAction) (net.Listener, *url.URL) {
expected := errors.New("mocked error") server := &filtering.HTTPProxy{
saver := &Saver{} OnIncomingHost: func(host string) filtering.HTTPAction {
txp := SaverMetadataHTTPTransport{ return action
HTTPTransport: FakeTransport{
Err: expected,
},
Saver: saver,
}
req, err := http.NewRequest("GET", "http://www.google.com", nil)
if err != nil {
t.Fatal(err)
}
req.Header.Add("User-Agent", "miniooni/0.1.0-dev")
resp, err := txp.RoundTrip(req)
if !errors.Is(err, expected) {
t.Fatal("not the error we expected")
}
if resp != nil {
t.Fatal("expected nil response here")
}
ev := saver.Read()
if len(ev) != 1 {
t.Fatal("expected one event")
}
if ev[0].Value().HTTPMethod != "GET" {
t.Fatal("unexpected Method")
}
if len(ev[0].Value().HTTPRequestHeaders) <= 0 {
t.Fatal("unexpected Headers")
}
if ev[0].Value().HTTPURL != "http://www.google.com" {
t.Fatal("unexpected URL")
}
if ev[0].Name() != "http_request_metadata" {
t.Fatal("unexpected Name")
}
if !ev[0].Value().Time.Before(time.Now()) {
t.Fatal("unexpected Time")
}
}
func TestSaverTransactionSuccess(t *testing.T) {
if testing.Short() {
t.Skip("skip test in short mode")
}
saver := &Saver{}
txp := SaverTransactionHTTPTransport{
HTTPTransport: netxlite.NewHTTPTransportStdlib(model.DiscardLogger),
Saver: saver,
}
req, err := http.NewRequest("GET", "https://www.google.com", nil)
if err != nil {
t.Fatal(err)
}
resp, err := txp.RoundTrip(req)
if err != nil {
t.Fatal("not the error we expected")
}
if resp == nil {
t.Fatal("expected non nil response here")
}
ev := saver.Read()
if len(ev) != 2 {
t.Fatal("expected two events")
}
//
if ev[0].Name() != "http_transaction_start" {
t.Fatal("unexpected Name")
}
if !ev[0].Value().Time.Before(time.Now()) {
t.Fatal("unexpected Time")
}
//
if ev[1].Value().Err != nil {
t.Fatal("unexpected Err")
}
if ev[1].Name() != "http_transaction_done" {
t.Fatal("unexpected Name")
}
if !ev[1].Value().Time.After(ev[0].Value().Time) {
t.Fatal("unexpected Time")
}
}
func TestSaverTransactionFailure(t *testing.T) {
expected := errors.New("mocked error")
saver := &Saver{}
txp := SaverTransactionHTTPTransport{
HTTPTransport: FakeTransport{
Err: expected,
},
Saver: saver,
}
req, err := http.NewRequest("GET", "http://www.google.com", nil)
if err != nil {
t.Fatal(err)
}
resp, err := txp.RoundTrip(req)
if !errors.Is(err, expected) {
t.Fatal("not the error we expected")
}
if resp != nil {
t.Fatal("expected nil response here")
}
ev := saver.Read()
if len(ev) != 2 {
t.Fatal("expected two events")
}
if ev[0].Name() != "http_transaction_start" {
t.Fatal("unexpected Name")
}
if !ev[0].Value().Time.Before(time.Now()) {
t.Fatal("unexpected Time")
}
if ev[1].Name() != "http_transaction_done" {
t.Fatal("unexpected Name")
}
if !errors.Is(ev[1].Value().Err, expected) {
t.Fatal("unexpected Err")
}
if !ev[1].Value().Time.After(ev[0].Value().Time) {
t.Fatal("unexpected Time")
}
}
func TestSaverBodySuccess(t *testing.T) {
saver := new(Saver)
txp := SaverBodyHTTPTransport{
HTTPTransport: FakeTransport{
Func: func(req *http.Request) (*http.Response, error) {
data, err := netxlite.ReadAllContext(context.Background(), req.Body)
if err != nil {
t.Fatal(err)
}
if string(data) != "deadbeef" {
t.Fatal("invalid data")
}
return &http.Response{
StatusCode: 501,
Body: io.NopCloser(strings.NewReader("abad1dea")),
}, nil
}, },
}, }
SnapshotSize: 4, listener, err := server.Start("127.0.0.1:0")
Saver: saver, if err != nil {
t.Fatal(err)
}
URL := &url.URL{
Scheme: "http",
Host: listener.Addr().String(),
Path: "/",
}
return listener, URL
} }
body := strings.NewReader("deadbeef")
req, err := http.NewRequest("POST", "http://x.org/y", body)
if err != nil {
t.Fatal(err)
}
resp, err := txp.RoundTrip(req)
if err != nil {
t.Fatal(err)
}
if resp.StatusCode != 501 {
t.Fatal("unexpected status code")
}
defer resp.Body.Close()
data, err := netxlite.ReadAllContext(context.Background(), resp.Body)
if err != nil {
t.Fatal(err)
}
if string(data) != "abad1dea" {
t.Fatal("unexpected body")
}
ev := saver.Read()
if len(ev) != 2 {
t.Fatal("unexpected number of events")
}
if string(ev[0].Value().Data) != "dead" {
t.Fatal("invalid Data")
}
if ev[0].Value().DataIsTruncated != true {
t.Fatal("invalid DataIsTruncated")
}
if ev[0].Name() != "http_request_body_snapshot" {
t.Fatal("invalid Name")
}
if ev[0].Value().Time.After(time.Now()) {
t.Fatal("invalid Time")
}
if string(ev[1].Value().Data) != "abad" {
t.Fatal("invalid Data")
}
if ev[1].Value().DataIsTruncated != true {
t.Fatal("invalid DataIsTruncated")
}
if ev[1].Name() != "http_response_body_snapshot" {
t.Fatal("invalid Name")
}
if ev[1].Value().Time.Before(ev[0].Value().Time) {
t.Fatal("invalid Time")
}
}
func TestSaverBodyRequestReadError(t *testing.T) { measureHTTP := func(t *testing.T, URL *url.URL) (*http.Response, *Saver, error) {
saver := new(Saver) saver := &Saver{}
txp := SaverBodyHTTPTransport{ txp := &SaverTransactionHTTPTransport{
HTTPTransport: FakeTransport{ HTTPTransport: netxlite.NewHTTPTransportStdlib(model.DiscardLogger),
Func: func(req *http.Request) (*http.Response, error) { Saver: saver,
panic("should not be called") }
req, err := http.NewRequest("GET", URL.String(), nil)
if err != nil {
t.Fatal(err)
}
req.Header.Add("User-Agent", "miniooni")
resp, err := txp.RoundTrip(req)
return resp, saver, err
}
validateRequestFields := func(t *testing.T, value *EventValue, URL *url.URL) {
if value.HTTPMethod != "GET" {
t.Fatal("invalid method")
}
if value.HTTPRequestHeaders.Get("Host") != URL.Host {
t.Fatal("invalid Host header")
}
if value.HTTPRequestHeaders.Get("User-Agent") != "miniooni" {
t.Fatal("invalid User-Agent header")
}
if value.HTTPURL != URL.String() {
t.Fatal("invalid URL")
}
if value.Time.IsZero() {
t.Fatal("expected nonzero Time")
}
if value.Transport != "tcp" {
t.Fatal("expected Transport to be tcp")
}
}
validateRequest := func(t *testing.T, ev Event, URL *url.URL) {
if _, good := ev.(*EventHTTPTransactionStart); !good {
t.Fatal("invalid event type")
}
if ev.Name() != "http_transaction_start" {
t.Fatal("invalid event name")
}
value := ev.Value()
validateRequestFields(t, value, URL)
}
validateResponseSuccess := func(t *testing.T, ev Event, URL *url.URL) {
if _, good := ev.(*EventHTTPTransactionDone); !good {
t.Fatal("invalid event type")
}
if ev.Name() != "http_transaction_done" {
t.Fatal("invalid event name")
}
value := ev.Value()
validateRequestFields(t, value, URL)
if value.Duration <= 0 {
t.Fatal("expected nonzero duration")
}
if len(value.HTTPResponseHeaders) <= 0 {
t.Fatal("expected at least one response header")
}
if !bytes.Equal(value.HTTPResponseBody, filtering.HTTPBlockpage451) {
t.Fatal("unexpected value for response body")
}
if value.HTTPStatusCode != 451 {
t.Fatal("unexpected status code")
}
}
t.Run("on success", func(t *testing.T) {
listener, URL := startServer(t, filtering.HTTPAction451)
defer listener.Close()
resp, saver, err := measureHTTP(t, URL)
if err != nil {
t.Fatal(err)
}
defer resp.Body.Close()
if resp.StatusCode != 451 {
t.Fatal("unexpected status code", resp.StatusCode)
}
events := saver.Read()
if len(events) != 2 {
t.Fatal("unexpected number of events")
}
validateRequest(t, events[0], URL)
validateResponseSuccess(t, events[1], URL)
data, err := netxlite.ReadAllContext(context.Background(), resp.Body)
if err != nil {
t.Fatal(err)
}
if !bytes.Equal(data, filtering.HTTPBlockpage451) {
t.Fatal("we cannot re-read the same body")
}
})
validateResponseFailure := func(t *testing.T, ev Event, URL *url.URL) {
if _, good := ev.(*EventHTTPTransactionDone); !good {
t.Fatal("invalid event type")
}
if ev.Name() != "http_transaction_done" {
t.Fatal("invalid event name")
}
value := ev.Value()
validateRequestFields(t, value, URL)
if value.Duration <= 0 {
t.Fatal("expected nonzero duration")
}
if value.Err.Error() != "connection_reset" {
t.Fatal("unexpected Err value")
}
if len(value.HTTPResponseHeaders) > 0 {
t.Fatal("expected zero response headers")
}
if !bytes.Equal(value.HTTPResponseBody, nil) {
t.Fatal("unexpected value for response body")
}
if value.HTTPStatusCode != 0 {
t.Fatal("unexpected status code")
}
}
t.Run("on round trip failure", func(t *testing.T) {
listener, URL := startServer(t, filtering.HTTPActionReset)
defer listener.Close()
resp, saver, err := measureHTTP(t, URL)
if err == nil || err.Error() != "connection_reset" {
t.Fatal("unexpected err", err)
}
if resp != nil {
t.Fatal("expected nil response")
}
events := saver.Read()
if len(events) != 2 {
t.Fatal("unexpected number of events")
}
validateRequest(t, events[0], URL)
validateResponseFailure(t, events[1], URL)
})
// Sometimes useful for testing
/*
dumplog := func(t *testing.T, ev Event) {
data, _ := json.MarshalIndent(ev.Value(), " ", " ")
t.Log(string(data))
t.FailNow()
}
*/
t.Run("on error reading the response body", func(t *testing.T) {
saver := &Saver{}
expected := errors.New("mocked error")
txp := SaverTransactionHTTPTransport{
HTTPTransport: &mocks.HTTPTransport{
MockRoundTrip: func(req *http.Request) (*http.Response, error) {
return &http.Response{
Header: http.Header{
"Server": {"antani"},
},
StatusCode: 200,
Body: io.NopCloser(&mocks.Reader{
MockRead: func(b []byte) (int, error) {
return 0, expected
},
}),
}, nil
},
MockNetwork: func() string {
return "tcp"
},
}, },
}, SnapshotSize: 4,
SnapshotSize: 4, Saver: saver,
Saver: saver, }
} URL := &url.URL{
expected := errors.New("mocked error") Scheme: "http",
body := FakeBody{Err: expected} Host: "127.0.0.1:9050",
req, err := http.NewRequest("POST", "http://x.org/y", body) }
if err != nil { req, err := http.NewRequest("GET", URL.String(), nil)
t.Fatal(err) if err != nil {
} t.Fatal(err)
resp, err := txp.RoundTrip(req) }
if !errors.Is(err, expected) { req.Header.Add("User-Agent", "miniooni")
t.Fatal("not the error we expected") resp, err := txp.RoundTrip(req)
} if !errors.Is(err, expected) {
if resp != nil { t.Fatal("not the error we expected")
t.Fatal("expected nil response") }
} if resp != nil {
ev := saver.Read() t.Fatal("expected nil response")
if len(ev) != 0 { }
t.Fatal("unexpected number of events") ev := saver.Read()
} validateRequest(t, ev[0], URL)
if ev[1].Value().HTTPStatusCode != 200 {
t.Fatal("invalid status code")
}
if ev[1].Value().HTTPResponseHeaders.Get("Server") != "antani" {
t.Fatal("invalid Server header")
}
if ev[1].Value().Err.Error() != "unknown_failure: mocked error" {
t.Fatal("invalid error")
}
})
} }
func TestSaverBodyRoundTripError(t *testing.T) { func TestHTTPCloneRequestHeaders(t *testing.T) {
saver := new(Saver)
expected := errors.New("mocked error")
txp := SaverBodyHTTPTransport{
HTTPTransport: FakeTransport{
Err: expected,
},
SnapshotSize: 4,
Saver: saver,
}
body := strings.NewReader("deadbeef")
req, err := http.NewRequest("POST", "http://x.org/y", body)
if err != nil {
t.Fatal(err)
}
resp, err := txp.RoundTrip(req)
if !errors.Is(err, expected) {
t.Fatal("not the error we expected")
}
if resp != nil {
t.Fatal("expected nil response")
}
ev := saver.Read()
if len(ev) != 1 {
t.Fatal("unexpected number of events")
}
if string(ev[0].Value().Data) != "dead" {
t.Fatal("invalid Data")
}
if ev[0].Value().DataIsTruncated != true {
t.Fatal("invalid DataIsTruncated")
}
if ev[0].Name() != "http_request_body_snapshot" {
t.Fatal("invalid Name")
}
if ev[0].Value().Time.After(time.Now()) {
t.Fatal("invalid Time")
}
}
func TestSaverBodyResponseReadError(t *testing.T) {
saver := new(Saver)
expected := errors.New("mocked error")
txp := SaverBodyHTTPTransport{
HTTPTransport: FakeTransport{
Func: func(req *http.Request) (*http.Response, error) {
return &http.Response{
StatusCode: 200,
Body: FakeBody{
Err: expected,
},
}, nil
},
},
SnapshotSize: 4,
Saver: saver,
}
body := strings.NewReader("deadbeef")
req, err := http.NewRequest("POST", "http://x.org/y", body)
if err != nil {
t.Fatal(err)
}
resp, err := txp.RoundTrip(req)
if !errors.Is(err, expected) {
t.Fatal("not the error we expected")
}
if resp != nil {
t.Fatal("expected nil response")
}
ev := saver.Read()
if len(ev) != 1 {
t.Fatal("unexpected number of events")
}
if string(ev[0].Value().Data) != "dead" {
t.Fatal("invalid Data")
}
if ev[0].Value().DataIsTruncated != true {
t.Fatal("invalid DataIsTruncated")
}
if ev[0].Name() != "http_request_body_snapshot" {
t.Fatal("invalid Name")
}
if ev[0].Value().Time.After(time.Now()) {
t.Fatal("invalid Time")
}
}
func TestCloneHeaders(t *testing.T) {
t.Run("with req.Host set", func(t *testing.T) { t.Run("with req.Host set", func(t *testing.T) {
req := &http.Request{ req := &http.Request{
Host: "www.example.com", Host: "www.example.com",
@ -414,55 +269,3 @@ func TestCloneHeaders(t *testing.T) {
} }
}) })
} }
type FakeDialer struct {
Conn net.Conn
Err error
}
func (d FakeDialer) DialContext(ctx context.Context, network, address string) (net.Conn, error) {
time.Sleep(10 * time.Microsecond)
return d.Conn, d.Err
}
type FakeTransport struct {
Name string
Err error
Func func(*http.Request) (*http.Response, error)
Resp *http.Response
}
func (txp FakeTransport) Network() string {
return txp.Name
}
func (txp FakeTransport) RoundTrip(req *http.Request) (*http.Response, error) {
time.Sleep(10 * time.Microsecond)
if txp.Func != nil {
return txp.Func(req)
}
if req.Body != nil {
netxlite.ReadAllContext(req.Context(), req.Body)
req.Body.Close()
}
if txp.Err != nil {
return nil, txp.Err
}
txp.Resp.Request = req // non thread safe but it doesn't matter
return txp.Resp, nil
}
func (txp FakeTransport) CloseIdleConnections() {}
type FakeBody struct {
Err error
}
func (fb FakeBody) Read(p []byte) (int, error) {
time.Sleep(10 * time.Microsecond)
return 0, fb.Err
}
func (fb FakeBody) Close() error {
return nil
}

View File

@ -51,7 +51,8 @@ func (p *HTTPProxy) Start(address string) (net.Listener, error) {
return listener, nil return listener, nil
} }
var httpBlockpage451 = []byte(`<html><head> // HTTPBlockPage451 is the block page returned along with status 451
var HTTPBlockpage451 = []byte(`<html><head>
<title>451 Unavailable For Legal Reasons</title> <title>451 Unavailable For Legal Reasons</title>
</head><body> </head><body>
<center><h1>451 Unavailable For Legal Reasons</h1></center> <center><h1>451 Unavailable For Legal Reasons</h1></center>
@ -80,7 +81,7 @@ func (p *HTTPProxy) handle(w http.ResponseWriter, r *http.Request) {
p.hijack(w, r, policy) p.hijack(w, r, policy)
case HTTPAction451: case HTTPAction451:
w.WriteHeader(http.StatusUnavailableForLegalReasons) w.WriteHeader(http.StatusUnavailableForLegalReasons)
w.Write(httpBlockpage451) w.Write(HTTPBlockpage451)
default: default:
w.WriteHeader(http.StatusInternalServerError) w.WriteHeader(http.StatusInternalServerError)
} }