mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-05 13:17:36 +03:00
Application data is now retrieved and restored via two callbacks on the qtls.Config. This allows us the get rid of the rather complex wrapping of the qtls.ClientSessionCache. Furthermore, it makes sure that we only restore the application data when qtls decides to actually use the ticket.
468 lines
18 KiB
Go
468 lines
18 KiB
Go
package wire
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"math/rand"
|
|
"net"
|
|
"sort"
|
|
"time"
|
|
|
|
"github.com/lucas-clemente/quic-go/internal/qerr"
|
|
|
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
|
"github.com/lucas-clemente/quic-go/internal/utils"
|
|
)
|
|
|
|
const transportParameterMarshalingVersion = 1
|
|
|
|
func init() {
|
|
rand.Seed(time.Now().UTC().UnixNano())
|
|
}
|
|
|
|
type transportParameterID uint64
|
|
|
|
const (
|
|
originalDestinationConnectionIDParameterID transportParameterID = 0x0
|
|
maxIdleTimeoutParameterID transportParameterID = 0x1
|
|
statelessResetTokenParameterID transportParameterID = 0x2
|
|
maxUDPPayloadSizeParameterID transportParameterID = 0x3
|
|
initialMaxDataParameterID transportParameterID = 0x4
|
|
initialMaxStreamDataBidiLocalParameterID transportParameterID = 0x5
|
|
initialMaxStreamDataBidiRemoteParameterID transportParameterID = 0x6
|
|
initialMaxStreamDataUniParameterID transportParameterID = 0x7
|
|
initialMaxStreamsBidiParameterID transportParameterID = 0x8
|
|
initialMaxStreamsUniParameterID transportParameterID = 0x9
|
|
ackDelayExponentParameterID transportParameterID = 0xa
|
|
maxAckDelayParameterID transportParameterID = 0xb
|
|
disableActiveMigrationParameterID transportParameterID = 0xc
|
|
preferredAddressParameterID transportParameterID = 0xd
|
|
activeConnectionIDLimitParameterID transportParameterID = 0xe
|
|
initialSourceConnectionIDParameterID transportParameterID = 0xf
|
|
retrySourceConnectionIDParameterID transportParameterID = 0x10
|
|
)
|
|
|
|
// PreferredAddress is the value encoding in the preferred_address transport parameter
|
|
type PreferredAddress struct {
|
|
IPv4 net.IP
|
|
IPv4Port uint16
|
|
IPv6 net.IP
|
|
IPv6Port uint16
|
|
ConnectionID protocol.ConnectionID
|
|
StatelessResetToken [16]byte
|
|
}
|
|
|
|
// TransportParameters are parameters sent to the peer during the handshake
|
|
type TransportParameters struct {
|
|
InitialMaxStreamDataBidiLocal protocol.ByteCount
|
|
InitialMaxStreamDataBidiRemote protocol.ByteCount
|
|
InitialMaxStreamDataUni protocol.ByteCount
|
|
InitialMaxData protocol.ByteCount
|
|
|
|
MaxAckDelay time.Duration
|
|
AckDelayExponent uint8
|
|
|
|
DisableActiveMigration bool
|
|
|
|
MaxUDPPayloadSize protocol.ByteCount
|
|
|
|
MaxUniStreamNum protocol.StreamNum
|
|
MaxBidiStreamNum protocol.StreamNum
|
|
|
|
MaxIdleTimeout time.Duration
|
|
|
|
PreferredAddress *PreferredAddress
|
|
|
|
OriginalDestinationConnectionID protocol.ConnectionID
|
|
InitialSourceConnectionID protocol.ConnectionID
|
|
RetrySourceConnectionID *protocol.ConnectionID // use a pointer here to distinguish zero-length connection IDs from missing transport parameters
|
|
|
|
StatelessResetToken *[16]byte
|
|
ActiveConnectionIDLimit uint64
|
|
}
|
|
|
|
// Unmarshal the transport parameters
|
|
func (p *TransportParameters) Unmarshal(data []byte, sentBy protocol.Perspective) error {
|
|
if err := p.unmarshal(bytes.NewReader(data), sentBy, false); err != nil {
|
|
return qerr.NewError(qerr.TransportParameterError, err.Error())
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (p *TransportParameters) unmarshal(r *bytes.Reader, sentBy protocol.Perspective, fromSessionTicket bool) error {
|
|
// needed to check that every parameter is only sent at most once
|
|
var parameterIDs []transportParameterID
|
|
|
|
var (
|
|
readAckDelayExponent bool
|
|
readMaxAckDelay bool
|
|
readOriginalDestinationConnectionID bool
|
|
readInitialSourceConnectionID bool
|
|
)
|
|
|
|
for r.Len() > 0 {
|
|
paramIDInt, err := utils.ReadVarInt(r)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
paramID := transportParameterID(paramIDInt)
|
|
paramLen, err := utils.ReadVarInt(r)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
parameterIDs = append(parameterIDs, paramID)
|
|
switch paramID {
|
|
case ackDelayExponentParameterID:
|
|
readAckDelayExponent = true
|
|
if err := p.readNumericTransportParameter(r, paramID, int(paramLen)); err != nil {
|
|
return err
|
|
}
|
|
case maxAckDelayParameterID:
|
|
readMaxAckDelay = true
|
|
if err := p.readNumericTransportParameter(r, paramID, int(paramLen)); err != nil {
|
|
return err
|
|
}
|
|
case initialMaxStreamDataBidiLocalParameterID,
|
|
initialMaxStreamDataBidiRemoteParameterID,
|
|
initialMaxStreamDataUniParameterID,
|
|
initialMaxDataParameterID,
|
|
initialMaxStreamsBidiParameterID,
|
|
initialMaxStreamsUniParameterID,
|
|
maxIdleTimeoutParameterID,
|
|
maxUDPPayloadSizeParameterID,
|
|
activeConnectionIDLimitParameterID:
|
|
if err := p.readNumericTransportParameter(r, paramID, int(paramLen)); err != nil {
|
|
return err
|
|
}
|
|
default:
|
|
if uint64(r.Len()) < paramLen {
|
|
return fmt.Errorf("remaining length (%d) smaller than parameter length (%d)", r.Len(), paramLen)
|
|
}
|
|
switch paramID {
|
|
case preferredAddressParameterID:
|
|
if sentBy == protocol.PerspectiveClient {
|
|
return errors.New("client sent a preferred_address")
|
|
}
|
|
if err := p.readPreferredAddress(r, int(paramLen)); err != nil {
|
|
return err
|
|
}
|
|
case disableActiveMigrationParameterID:
|
|
if paramLen != 0 {
|
|
return fmt.Errorf("wrong length for disable_active_migration: %d (expected empty)", paramLen)
|
|
}
|
|
p.DisableActiveMigration = true
|
|
case statelessResetTokenParameterID:
|
|
if sentBy == protocol.PerspectiveClient {
|
|
return errors.New("client sent a stateless_reset_token")
|
|
}
|
|
if paramLen != 16 {
|
|
return fmt.Errorf("wrong length for stateless_reset_token: %d (expected 16)", paramLen)
|
|
}
|
|
var token [16]byte
|
|
r.Read(token[:])
|
|
p.StatelessResetToken = &token
|
|
case originalDestinationConnectionIDParameterID:
|
|
if sentBy == protocol.PerspectiveClient {
|
|
return errors.New("client sent an original_destination_connection_id")
|
|
}
|
|
p.OriginalDestinationConnectionID, _ = protocol.ReadConnectionID(r, int(paramLen))
|
|
readOriginalDestinationConnectionID = true
|
|
case initialSourceConnectionIDParameterID:
|
|
p.InitialSourceConnectionID, _ = protocol.ReadConnectionID(r, int(paramLen))
|
|
readInitialSourceConnectionID = true
|
|
case retrySourceConnectionIDParameterID:
|
|
if sentBy == protocol.PerspectiveClient {
|
|
return errors.New("client sent a retry_source_connection_id")
|
|
}
|
|
connID, _ := protocol.ReadConnectionID(r, int(paramLen))
|
|
p.RetrySourceConnectionID = &connID
|
|
default:
|
|
r.Seek(int64(paramLen), io.SeekCurrent)
|
|
}
|
|
}
|
|
}
|
|
|
|
if !fromSessionTicket {
|
|
if sentBy == protocol.PerspectiveServer && !readOriginalDestinationConnectionID {
|
|
return errors.New("missing original_destination_connection_id")
|
|
}
|
|
if !readAckDelayExponent {
|
|
p.AckDelayExponent = protocol.DefaultAckDelayExponent
|
|
}
|
|
if !readMaxAckDelay {
|
|
p.MaxAckDelay = protocol.DefaultMaxAckDelay
|
|
}
|
|
if p.MaxUDPPayloadSize == 0 {
|
|
p.MaxUDPPayloadSize = protocol.MaxByteCount
|
|
}
|
|
if !readInitialSourceConnectionID {
|
|
return errors.New("missing initial_source_connection_id")
|
|
}
|
|
}
|
|
|
|
// check that every transport parameter was sent at most once
|
|
sort.Slice(parameterIDs, func(i, j int) bool { return parameterIDs[i] < parameterIDs[j] })
|
|
for i := 0; i < len(parameterIDs)-1; i++ {
|
|
if parameterIDs[i] == parameterIDs[i+1] {
|
|
return fmt.Errorf("received duplicate transport parameter %#x", parameterIDs[i])
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (p *TransportParameters) readPreferredAddress(r *bytes.Reader, expectedLen int) error {
|
|
remainingLen := r.Len()
|
|
pa := &PreferredAddress{}
|
|
ipv4 := make([]byte, 4)
|
|
if _, err := io.ReadFull(r, ipv4); err != nil {
|
|
return err
|
|
}
|
|
pa.IPv4 = net.IP(ipv4)
|
|
port, err := utils.BigEndian.ReadUint16(r)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
pa.IPv4Port = port
|
|
ipv6 := make([]byte, 16)
|
|
if _, err := io.ReadFull(r, ipv6); err != nil {
|
|
return err
|
|
}
|
|
pa.IPv6 = net.IP(ipv6)
|
|
port, err = utils.BigEndian.ReadUint16(r)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
pa.IPv6Port = port
|
|
connIDLen, err := r.ReadByte()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if connIDLen == 0 || connIDLen > protocol.MaxConnIDLen {
|
|
return fmt.Errorf("invalid connection ID length: %d", connIDLen)
|
|
}
|
|
connID, err := protocol.ReadConnectionID(r, int(connIDLen))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
pa.ConnectionID = connID
|
|
if _, err := io.ReadFull(r, pa.StatelessResetToken[:]); err != nil {
|
|
return err
|
|
}
|
|
if bytesRead := remainingLen - r.Len(); bytesRead != expectedLen {
|
|
return fmt.Errorf("expected preferred_address to be %d long, read %d bytes", expectedLen, bytesRead)
|
|
}
|
|
p.PreferredAddress = pa
|
|
return nil
|
|
}
|
|
|
|
func (p *TransportParameters) readNumericTransportParameter(
|
|
r *bytes.Reader,
|
|
paramID transportParameterID,
|
|
expectedLen int,
|
|
) error {
|
|
remainingLen := r.Len()
|
|
val, err := utils.ReadVarInt(r)
|
|
if err != nil {
|
|
return fmt.Errorf("error while reading transport parameter %d: %s", paramID, err)
|
|
}
|
|
if remainingLen-r.Len() != expectedLen {
|
|
return fmt.Errorf("inconsistent transport parameter length for %d", paramID)
|
|
}
|
|
switch paramID {
|
|
case initialMaxStreamDataBidiLocalParameterID:
|
|
p.InitialMaxStreamDataBidiLocal = protocol.ByteCount(val)
|
|
case initialMaxStreamDataBidiRemoteParameterID:
|
|
p.InitialMaxStreamDataBidiRemote = protocol.ByteCount(val)
|
|
case initialMaxStreamDataUniParameterID:
|
|
p.InitialMaxStreamDataUni = protocol.ByteCount(val)
|
|
case initialMaxDataParameterID:
|
|
p.InitialMaxData = protocol.ByteCount(val)
|
|
case initialMaxStreamsBidiParameterID:
|
|
p.MaxBidiStreamNum = protocol.StreamNum(val)
|
|
case initialMaxStreamsUniParameterID:
|
|
p.MaxUniStreamNum = protocol.StreamNum(val)
|
|
case maxIdleTimeoutParameterID:
|
|
p.MaxIdleTimeout = utils.MaxDuration(protocol.MinRemoteIdleTimeout, time.Duration(val)*time.Millisecond)
|
|
case maxUDPPayloadSizeParameterID:
|
|
if val < 1200 {
|
|
return fmt.Errorf("invalid value for max_packet_size: %d (minimum 1200)", val)
|
|
}
|
|
p.MaxUDPPayloadSize = protocol.ByteCount(val)
|
|
case ackDelayExponentParameterID:
|
|
if val > protocol.MaxAckDelayExponent {
|
|
return fmt.Errorf("invalid value for ack_delay_exponent: %d (maximum %d)", val, protocol.MaxAckDelayExponent)
|
|
}
|
|
p.AckDelayExponent = uint8(val)
|
|
case maxAckDelayParameterID:
|
|
maxAckDelay := time.Duration(val) * time.Millisecond
|
|
if maxAckDelay >= protocol.MaxMaxAckDelay {
|
|
return fmt.Errorf("invalid value for max_ack_delay: %dms (maximum %dms)", maxAckDelay/time.Millisecond, (protocol.MaxMaxAckDelay-time.Millisecond)/time.Millisecond)
|
|
}
|
|
if maxAckDelay < 0 {
|
|
maxAckDelay = utils.InfDuration
|
|
}
|
|
p.MaxAckDelay = maxAckDelay
|
|
case activeConnectionIDLimitParameterID:
|
|
p.ActiveConnectionIDLimit = val
|
|
default:
|
|
return fmt.Errorf("TransportParameter BUG: transport parameter %d not found", paramID)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Marshal the transport parameters
|
|
func (p *TransportParameters) Marshal(pers protocol.Perspective) []byte {
|
|
b := &bytes.Buffer{}
|
|
|
|
//add a greased value
|
|
utils.WriteVarInt(b, uint64(27+31*rand.Intn(100)))
|
|
length := rand.Intn(16)
|
|
randomData := make([]byte, length)
|
|
rand.Read(randomData)
|
|
utils.WriteVarInt(b, uint64(length))
|
|
b.Write(randomData)
|
|
|
|
// initial_max_stream_data_bidi_local
|
|
p.marshalVarintParam(b, initialMaxStreamDataBidiLocalParameterID, uint64(p.InitialMaxStreamDataBidiLocal))
|
|
// initial_max_stream_data_bidi_remote
|
|
p.marshalVarintParam(b, initialMaxStreamDataBidiRemoteParameterID, uint64(p.InitialMaxStreamDataBidiRemote))
|
|
// initial_max_stream_data_uni
|
|
p.marshalVarintParam(b, initialMaxStreamDataUniParameterID, uint64(p.InitialMaxStreamDataUni))
|
|
// initial_max_data
|
|
p.marshalVarintParam(b, initialMaxDataParameterID, uint64(p.InitialMaxData))
|
|
// initial_max_bidi_streams
|
|
p.marshalVarintParam(b, initialMaxStreamsBidiParameterID, uint64(p.MaxBidiStreamNum))
|
|
// initial_max_uni_streams
|
|
p.marshalVarintParam(b, initialMaxStreamsUniParameterID, uint64(p.MaxUniStreamNum))
|
|
// idle_timeout
|
|
p.marshalVarintParam(b, maxIdleTimeoutParameterID, uint64(p.MaxIdleTimeout/time.Millisecond))
|
|
// max_packet_size
|
|
p.marshalVarintParam(b, maxUDPPayloadSizeParameterID, uint64(protocol.MaxReceivePacketSize))
|
|
// max_ack_delay
|
|
// Only send it if is different from the default value.
|
|
if p.MaxAckDelay != protocol.DefaultMaxAckDelay {
|
|
p.marshalVarintParam(b, maxAckDelayParameterID, uint64(p.MaxAckDelay/time.Millisecond))
|
|
}
|
|
// ack_delay_exponent
|
|
// Only send it if is different from the default value.
|
|
if p.AckDelayExponent != protocol.DefaultAckDelayExponent {
|
|
p.marshalVarintParam(b, ackDelayExponentParameterID, uint64(p.AckDelayExponent))
|
|
}
|
|
// disable_active_migration
|
|
if p.DisableActiveMigration {
|
|
utils.WriteVarInt(b, uint64(disableActiveMigrationParameterID))
|
|
utils.WriteVarInt(b, 0)
|
|
}
|
|
if pers == protocol.PerspectiveServer {
|
|
// stateless_reset_token
|
|
utils.WriteVarInt(b, uint64(statelessResetTokenParameterID))
|
|
utils.WriteVarInt(b, 16)
|
|
b.Write(p.StatelessResetToken[:])
|
|
// original_destination_connection_id
|
|
utils.WriteVarInt(b, uint64(originalDestinationConnectionIDParameterID))
|
|
utils.WriteVarInt(b, uint64(p.OriginalDestinationConnectionID.Len()))
|
|
b.Write(p.OriginalDestinationConnectionID.Bytes())
|
|
// preferred_address
|
|
if p.PreferredAddress != nil {
|
|
utils.WriteVarInt(b, uint64(preferredAddressParameterID))
|
|
utils.WriteVarInt(b, 4+2+16+2+1+uint64(p.PreferredAddress.ConnectionID.Len())+16)
|
|
ipv4 := p.PreferredAddress.IPv4
|
|
b.Write(ipv4[len(ipv4)-4:])
|
|
utils.BigEndian.WriteUint16(b, p.PreferredAddress.IPv4Port)
|
|
b.Write(p.PreferredAddress.IPv6)
|
|
utils.BigEndian.WriteUint16(b, p.PreferredAddress.IPv6Port)
|
|
b.WriteByte(uint8(p.PreferredAddress.ConnectionID.Len()))
|
|
b.Write(p.PreferredAddress.ConnectionID.Bytes())
|
|
b.Write(p.PreferredAddress.StatelessResetToken[:])
|
|
}
|
|
}
|
|
// active_connection_id_limit
|
|
p.marshalVarintParam(b, activeConnectionIDLimitParameterID, p.ActiveConnectionIDLimit)
|
|
// initial_source_connection_id
|
|
utils.WriteVarInt(b, uint64(initialSourceConnectionIDParameterID))
|
|
utils.WriteVarInt(b, uint64(p.InitialSourceConnectionID.Len()))
|
|
b.Write(p.InitialSourceConnectionID.Bytes())
|
|
// retry_source_connection_id
|
|
if pers == protocol.PerspectiveServer && p.RetrySourceConnectionID != nil {
|
|
utils.WriteVarInt(b, uint64(retrySourceConnectionIDParameterID))
|
|
utils.WriteVarInt(b, uint64(p.RetrySourceConnectionID.Len()))
|
|
b.Write(p.RetrySourceConnectionID.Bytes())
|
|
}
|
|
return b.Bytes()
|
|
}
|
|
|
|
func (p *TransportParameters) marshalVarintParam(b *bytes.Buffer, id transportParameterID, val uint64) {
|
|
utils.WriteVarInt(b, uint64(id))
|
|
utils.WriteVarInt(b, uint64(utils.VarIntLen(val)))
|
|
utils.WriteVarInt(b, val)
|
|
}
|
|
|
|
// MarshalForSessionTicket marshals the transport parameters we save in the session ticket.
|
|
// When sending a 0-RTT enabled TLS session tickets, we need to save the transport parameters.
|
|
// The client will remember the transport parameters used in the last session,
|
|
// and apply those to the 0-RTT data it sends.
|
|
// Saving the transport parameters in the ticket gives the server the option to reject 0-RTT
|
|
// if the transport parameters changed.
|
|
// Since the session ticket is encrypted, the serialization format is defined by the server.
|
|
// For convenience, we use the same format that we also use for sending the transport parameters.
|
|
func (p *TransportParameters) MarshalForSessionTicket(b *bytes.Buffer) {
|
|
utils.WriteVarInt(b, transportParameterMarshalingVersion)
|
|
|
|
// initial_max_stream_data_bidi_local
|
|
p.marshalVarintParam(b, initialMaxStreamDataBidiLocalParameterID, uint64(p.InitialMaxStreamDataBidiLocal))
|
|
// initial_max_stream_data_bidi_remote
|
|
p.marshalVarintParam(b, initialMaxStreamDataBidiRemoteParameterID, uint64(p.InitialMaxStreamDataBidiRemote))
|
|
// initial_max_stream_data_uni
|
|
p.marshalVarintParam(b, initialMaxStreamDataUniParameterID, uint64(p.InitialMaxStreamDataUni))
|
|
// initial_max_data
|
|
p.marshalVarintParam(b, initialMaxDataParameterID, uint64(p.InitialMaxData))
|
|
// initial_max_bidi_streams
|
|
p.marshalVarintParam(b, initialMaxStreamsBidiParameterID, uint64(p.MaxBidiStreamNum))
|
|
// initial_max_uni_streams
|
|
p.marshalVarintParam(b, initialMaxStreamsUniParameterID, uint64(p.MaxUniStreamNum))
|
|
// active_connection_id_limit
|
|
p.marshalVarintParam(b, activeConnectionIDLimitParameterID, p.ActiveConnectionIDLimit)
|
|
}
|
|
|
|
// UnmarshalFromSessionTicket unmarshals transport parameters from a session ticket.
|
|
func (p *TransportParameters) UnmarshalFromSessionTicket(r *bytes.Reader) error {
|
|
version, err := utils.ReadVarInt(r)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if version != transportParameterMarshalingVersion {
|
|
return fmt.Errorf("unknown transport parameter marshaling version: %d", version)
|
|
}
|
|
return p.unmarshal(r, protocol.PerspectiveServer, true)
|
|
}
|
|
|
|
// ValidFor0RTT checks if the transport parameters match those saved in the session ticket.
|
|
func (p *TransportParameters) ValidFor0RTT(tp *TransportParameters) bool {
|
|
return p.InitialMaxStreamDataBidiLocal == tp.InitialMaxStreamDataBidiLocal &&
|
|
p.InitialMaxStreamDataBidiRemote == tp.InitialMaxStreamDataBidiRemote &&
|
|
p.InitialMaxStreamDataUni == tp.InitialMaxStreamDataUni &&
|
|
p.InitialMaxData == tp.InitialMaxData &&
|
|
p.MaxBidiStreamNum == tp.MaxBidiStreamNum &&
|
|
p.MaxUniStreamNum == tp.MaxUniStreamNum
|
|
}
|
|
|
|
// String returns a string representation, intended for logging.
|
|
func (p *TransportParameters) String() string {
|
|
logString := "&wire.TransportParameters{OriginalDestinationConnectionID: %s, InitialSourceConnectionID: %s, "
|
|
logParams := []interface{}{p.OriginalDestinationConnectionID, p.InitialSourceConnectionID}
|
|
if p.RetrySourceConnectionID != nil {
|
|
logString += "RetrySourceConnectionID: %s, "
|
|
logParams = append(logParams, p.RetrySourceConnectionID)
|
|
}
|
|
logString += "InitialMaxStreamDataBidiLocal: %d, InitialMaxStreamDataBidiRemote: %d, InitialMaxStreamDataUni: %d, InitialMaxData: %d, MaxBidiStreamNum: %d, MaxUniStreamNum: %d, MaxIdleTimeout: %s, AckDelayExponent: %d, MaxAckDelay: %s, ActiveConnectionIDLimit: %d"
|
|
logParams = append(logParams, []interface{}{p.InitialMaxStreamDataBidiLocal, p.InitialMaxStreamDataBidiRemote, p.InitialMaxStreamDataUni, p.InitialMaxData, p.MaxBidiStreamNum, p.MaxUniStreamNum, p.MaxIdleTimeout, p.AckDelayExponent, p.MaxAckDelay, p.ActiveConnectionIDLimit}...)
|
|
if p.StatelessResetToken != nil { // the client never sends a stateless reset token
|
|
logString += ", StatelessResetToken: %#x"
|
|
logParams = append(logParams, *p.StatelessResetToken)
|
|
}
|
|
logString += "}"
|
|
return fmt.Sprintf(logString, logParams...)
|
|
}
|