From 6ef3febf69e481fafafbc263f481381eb7531a54 Mon Sep 17 00:00:00 2001 From: Simone Basso Date: Thu, 25 Aug 2022 13:41:26 +0200 Subject: [PATCH] feat(measurexlite): add T0 and TransactionID (#879) The T0 field is the moment when we started collecting data, while T is the moment when we finished collecting data. The TransactionID field will be repurposed for step-by-step measurements to indicate related observations collected as part of the same flow (e.g., TCP+TLS+HTTP). Note that, for now, this change will only affect measurexlite and we're not planning on changing other libraries for measuring. Part of https://github.com/ooni/probe/issues/2137 --- internal/measurexlite/conn.go | 16 ++++++------ internal/measurexlite/conn_test.go | 16 ++++++------ internal/measurexlite/dialer.go | 4 ++- internal/measurexlite/dns.go | 1 + internal/measurexlite/quic_test.go | 1 + internal/measurexlite/tls.go | 2 ++ internal/measurexlite/tls_test.go | 2 ++ internal/model/archival.go | 39 ++++++++++++++++++------------ 8 files changed, 51 insertions(+), 30 deletions(-) diff --git a/internal/measurexlite/conn.go b/internal/measurexlite/conn.go index 0236ee3..066086f 100644 --- a/internal/measurexlite/conn.go +++ b/internal/measurexlite/conn.go @@ -134,13 +134,15 @@ func addrStringIfNotNil(addr net.Addr) (out string) { func NewArchivalNetworkEvent(index int64, started time.Duration, operation string, network string, address string, count int, err error, finished time.Duration) *model.ArchivalNetworkEvent { return &model.ArchivalNetworkEvent{ - Address: address, - Failure: tracex.NewFailure(err), - NumBytes: int64(count), - Operation: operation, - Proto: network, - T: finished.Seconds(), - Tags: []string{}, + Address: address, + Failure: tracex.NewFailure(err), + NumBytes: int64(count), + Operation: operation, + Proto: network, + T0: started.Seconds(), + T: finished.Seconds(), + TransactionID: index, + Tags: []string{}, } } diff --git a/internal/measurexlite/conn_test.go b/internal/measurexlite/conn_test.go index ff59e71..26df078 100644 --- a/internal/measurexlite/conn_test.go +++ b/internal/measurexlite/conn_test.go @@ -437,13 +437,15 @@ func TestNewAnnotationArchivalNetworkEvent(t *testing.T) { operation = "tls_handshake_start" ) expect := &model.ArchivalNetworkEvent{ - Address: "", - Failure: nil, - NumBytes: 0, - Operation: operation, - Proto: "", - T: duration.Seconds(), - Tags: []string{}, + Address: "", + Failure: nil, + NumBytes: 0, + Operation: operation, + Proto: "", + T0: duration.Seconds(), + T: duration.Seconds(), + TransactionID: index, + Tags: []string{}, } got := NewAnnotationArchivalNetworkEvent( index, duration, operation, diff --git a/internal/measurexlite/dialer.go b/internal/measurexlite/dialer.go index bc7c189..7d81a5f 100644 --- a/internal/measurexlite/dialer.go +++ b/internal/measurexlite/dialer.go @@ -82,7 +82,9 @@ func NewArchivalTCPConnectResult(index int64, started time.Duration, address str Failure: tracex.NewFailure(err), Success: err == nil, }, - T: finished.Seconds(), + T0: started.Seconds(), + T: finished.Seconds(), + TransactionID: index, } } diff --git a/internal/measurexlite/dns.go b/internal/measurexlite/dns.go index 952613c..72d127e 100644 --- a/internal/measurexlite/dns.go +++ b/internal/measurexlite/dns.go @@ -127,6 +127,7 @@ func NewArchivalDNSLookupResultFromRoundTrip(index int64, started time.Duration, ResolverAddress: reso.Address(), T0: started.Seconds(), T: finished.Seconds(), + TransactionID: index, } } diff --git a/internal/measurexlite/quic_test.go b/internal/measurexlite/quic_test.go index 2292866..04e5148 100644 --- a/internal/measurexlite/quic_test.go +++ b/internal/measurexlite/quic_test.go @@ -184,6 +184,7 @@ func TestNewQUICDialerWithoutResolver(t *testing.T) { NumBytes: 0, Operation: "quic_handshake_done", Proto: "", + T0: time.Second.Seconds(), T: time.Second.Seconds(), Tags: []string{}, } diff --git a/internal/measurexlite/tls.go b/internal/measurexlite/tls.go index 0938bd9..816e69c 100644 --- a/internal/measurexlite/tls.go +++ b/internal/measurexlite/tls.go @@ -86,9 +86,11 @@ func NewArchivalTLSOrQUICHandshakeResult( NoTLSVerify: config.InsecureSkipVerify, PeerCertificates: TLSPeerCerts(state, err), ServerName: config.ServerName, + T0: started.Seconds(), T: finished.Seconds(), Tags: []string{}, TLSVersion: netxlite.TLSVersionString(state.Version), + TransactionID: index, } } diff --git a/internal/measurexlite/tls_test.go b/internal/measurexlite/tls_test.go index d1038cd..46c8c50 100644 --- a/internal/measurexlite/tls_test.go +++ b/internal/measurexlite/tls_test.go @@ -167,6 +167,7 @@ func TestNewTLSHandshakerStdlib(t *testing.T) { NumBytes: 0, Operation: "tls_handshake_done", Proto: "", + T0: time.Second.Seconds(), T: time.Second.Seconds(), Tags: []string{}, } @@ -329,6 +330,7 @@ func TestNewTLSHandshakerStdlib(t *testing.T) { NumBytes: 0, Operation: "tls_handshake_done", Proto: "", + T0: time.Second.Seconds(), T: time.Second.Seconds(), Tags: []string{}, } diff --git a/internal/model/archival.go b/internal/model/archival.go index d0c5f87..9e3ff0a 100644 --- a/internal/model/archival.go +++ b/internal/model/archival.go @@ -125,6 +125,7 @@ type ArchivalDNSLookupResult struct { ResolverAddress string `json:"resolver_address"` T0 float64 `json:"t0"` T float64 `json:"t"` + TransactionID int64 `json:"transaction_id"` } // ArchivalDNSAnswer is a DNS answer. @@ -146,10 +147,12 @@ type ArchivalDNSAnswer struct { // // See https://github.com/ooni/spec/blob/master/data-formats/df-005-tcpconnect.md. type ArchivalTCPConnectResult struct { - IP string `json:"ip"` - Port int `json:"port"` - Status ArchivalTCPConnectStatus `json:"status"` - T float64 `json:"t"` + IP string `json:"ip"` + Port int `json:"port"` + Status ArchivalTCPConnectStatus `json:"status"` + T0 float64 `json:"t0"` + T float64 `json:"t"` + TransactionID int64 `json:"transaction_id"` } // ArchivalTCPConnectStatus is the status of ArchivalTCPConnectResult. @@ -175,9 +178,11 @@ type ArchivalTLSOrQUICHandshakeResult struct { NoTLSVerify bool `json:"no_tls_verify"` PeerCertificates []ArchivalMaybeBinaryData `json:"peer_certificates"` ServerName string `json:"server_name"` + T0 float64 `json:"t0"` T float64 `json:"t"` Tags []string `json:"tags"` TLSVersion string `json:"tls_version"` + TransactionID int64 `json:"transaction_id"` } // @@ -188,10 +193,12 @@ type ArchivalTLSOrQUICHandshakeResult struct { // // See https://github.com/ooni/spec/blob/master/data-formats/df-001-httpt.md. type ArchivalHTTPRequestResult struct { - Failure *string `json:"failure"` - Request ArchivalHTTPRequest `json:"request"` - Response ArchivalHTTPResponse `json:"response"` - T float64 `json:"t"` + Failure *string `json:"failure"` + Request ArchivalHTTPRequest `json:"request"` + Response ArchivalHTTPResponse `json:"response"` + T0 float64 `json:"t0"` + T float64 `json:"t"` + TransactionID int64 `json:"transaction_id"` } // ArchivalHTTPRequest contains an HTTP request. @@ -307,11 +314,13 @@ type ArchivalHTTPTor struct { // // See https://github.com/ooni/spec/blob/master/data-formats/df-008-netevents.md. type ArchivalNetworkEvent struct { - Address string `json:"address,omitempty"` - Failure *string `json:"failure"` - NumBytes int64 `json:"num_bytes,omitempty"` - Operation string `json:"operation"` - Proto string `json:"proto,omitempty"` - T float64 `json:"t"` - Tags []string `json:"tags,omitempty"` + Address string `json:"address,omitempty"` + Failure *string `json:"failure"` + NumBytes int64 `json:"num_bytes,omitempty"` + Operation string `json:"operation"` + Proto string `json:"proto,omitempty"` + T0 float64 `json:"t0"` + T float64 `json:"t"` + TransactionID int64 `json:"transaction_id"` + Tags []string `json:"tags,omitempty"` }