50 lines
1.6 KiB
Go
50 lines
1.6 KiB
Go
|
package model
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"encoding/json"
|
||
|
"errors"
|
||
|
"net"
|
||
|
)
|
||
|
|
||
|
// ErrInvalidProbeIP indicates that we're dealing with a string that
|
||
|
// is not the valid serialization of an IP address.
|
||
|
var ErrInvalidProbeIP = errors.New("model: invalid probe IP")
|
||
|
|
||
|
// Scrub scrubs the probeIP out of the measurement.
|
||
|
func (m *Measurement) Scrub(probeIP string) (err error) {
|
||
|
// We now behave like we can share everything except the
|
||
|
// probe IP, which we instead cannot ever share
|
||
|
m.ProbeIP = DefaultProbeIP
|
||
|
return m.MaybeRewriteTestKeys(probeIP, json.Marshal)
|
||
|
}
|
||
|
|
||
|
// Scrubbed is the string that replaces IP addresses.
|
||
|
const Scrubbed = `[scrubbed]`
|
||
|
|
||
|
// MaybeRewriteTestKeys is the function called by Scrub that
|
||
|
// ensures that m's serialization doesn't include the IP
|
||
|
func (m *Measurement) MaybeRewriteTestKeys(
|
||
|
currentIP string, marshal func(interface{}) ([]byte, error)) error {
|
||
|
if net.ParseIP(currentIP) == nil {
|
||
|
return ErrInvalidProbeIP
|
||
|
}
|
||
|
data, err := marshal(m.TestKeys)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
// The check using Count is to save an unnecessary copy performed by
|
||
|
// ReplaceAll when there are no matches into the body. This is what
|
||
|
// we would like the common case to be, meaning that the code has done
|
||
|
// its job correctly and has not leaked the IP.
|
||
|
bpip := []byte(currentIP)
|
||
|
if bytes.Count(data, bpip) <= 0 {
|
||
|
return nil
|
||
|
}
|
||
|
data = bytes.ReplaceAll(data, bpip, []byte(Scrubbed))
|
||
|
// We add an annotation such that hopefully later we can measure the
|
||
|
// number of cases where we failed to sanitize properly.
|
||
|
m.AddAnnotation("_probe_engine_sanitize_test_keys", "true")
|
||
|
return json.Unmarshal(data, &m.TestKeys)
|
||
|
}
|