2017-05-27 13:30:11 -06:00
package quic
import (
"bytes"
2018-09-03 04:18:54 +07:00
"context"
2017-07-27 16:11:56 -06:00
"crypto/tls"
2017-05-27 13:30:11 -06:00
"errors"
"fmt"
"net"
"sync"
2018-09-03 04:18:54 +07:00
"github.com/bifurcation/mint"
2018-02-17 13:29:53 +08:00
"github.com/lucas-clemente/quic-go/internal/handshake"
"github.com/lucas-clemente/quic-go/internal/protocol"
2017-07-27 16:11:56 -06:00
"github.com/lucas-clemente/quic-go/internal/utils"
2018-02-17 13:29:53 +08:00
"github.com/lucas-clemente/quic-go/internal/wire"
2017-05-27 13:30:11 -06:00
"github.com/lucas-clemente/quic-go/qerr"
)
type client struct {
2018-02-17 13:29:53 +08:00
mutex sync . Mutex
2017-05-27 13:30:11 -06:00
2018-09-03 04:18:54 +07:00
conn connection
// If the client is created with DialAddr, we create a packet conn.
// If it is started with Dial, we take a packet conn as a parameter.
createdPacketConn bool
2017-05-27 13:30:11 -06:00
hostname string
2018-09-03 04:18:54 +07:00
packetHandlers packetHandlerManager
token [ ] byte
numRetries int
versionNegotiated bool // has the server accepted our version
2018-02-17 13:29:53 +08:00
receivedVersionNegotiationPacket bool
negotiatedVersions [ ] protocol . VersionNumber // the list of versions from the version negotiation packet
2017-05-27 13:30:11 -06:00
2018-09-03 04:18:54 +07:00
tlsConf * tls . Config
mintConf * mint . Config
config * Config
2017-05-27 13:30:11 -06:00
2018-09-03 04:18:54 +07:00
srcConnID protocol . ConnectionID
destConnID protocol . ConnectionID
2018-02-17 13:29:53 +08:00
initialVersion protocol . VersionNumber
version protocol . VersionNumber
2017-05-27 13:30:11 -06:00
2018-09-03 04:18:54 +07:00
handshakeChan chan struct { }
closeCallback func ( protocol . ConnectionID )
session quicSession
2018-04-18 15:48:08 -06:00
logger utils . Logger
2017-05-27 13:30:11 -06:00
}
2018-09-03 04:18:54 +07:00
var _ packetHandler = & client { }
2017-05-27 13:30:11 -06:00
var (
2018-02-17 13:29:53 +08:00
// make it possible to mock connection ID generation in the tests
2018-09-03 04:18:54 +07:00
generateConnectionID = protocol . GenerateConnectionID
generateConnectionIDForInitial = protocol . GenerateConnectionIDForInitial
errCloseSessionForNewVersion = errors . New ( "closing session in order to recreate it with a new version" )
errCloseSessionForRetry = errors . New ( "closing session in response to a stateless retry" )
2017-05-27 13:30:11 -06:00
)
// DialAddr establishes a new QUIC connection to a server.
// The hostname for SNI is taken from the given address.
2018-09-03 04:18:54 +07:00
func DialAddr (
addr string ,
tlsConf * tls . Config ,
config * Config ,
) ( Session , error ) {
return DialAddrContext ( context . Background ( ) , addr , tlsConf , config )
}
// DialAddrContext establishes a new QUIC connection to a server using the provided context.
// The hostname for SNI is taken from the given address.
func DialAddrContext (
ctx context . Context ,
addr string ,
tlsConf * tls . Config ,
config * Config ,
) ( Session , error ) {
2017-05-27 13:30:11 -06:00
udpAddr , err := net . ResolveUDPAddr ( "udp" , addr )
if err != nil {
return nil , err
}
udpConn , err := net . ListenUDP ( "udp" , & net . UDPAddr { IP : net . IPv4zero , Port : 0 } )
if err != nil {
return nil , err
}
2018-09-03 04:18:54 +07:00
return dialContext ( ctx , udpConn , udpAddr , addr , tlsConf , config , true )
2017-05-27 13:30:11 -06:00
}
2018-02-17 13:29:53 +08:00
// Dial establishes a new QUIC connection to a server using a net.PacketConn.
2017-05-27 13:30:11 -06:00
// The host parameter is used for SNI.
2018-02-17 13:29:53 +08:00
func Dial (
2017-07-27 16:11:56 -06:00
pconn net . PacketConn ,
remoteAddr net . Addr ,
host string ,
tlsConf * tls . Config ,
config * Config ,
2018-02-17 13:29:53 +08:00
) ( Session , error ) {
2018-09-03 04:18:54 +07:00
return DialContext ( context . Background ( ) , pconn , remoteAddr , host , tlsConf , config )
}
// DialContext establishes a new QUIC connection to a server using a net.PacketConn using the provided context.
// The host parameter is used for SNI.
func DialContext (
ctx context . Context ,
pconn net . PacketConn ,
remoteAddr net . Addr ,
host string ,
tlsConf * tls . Config ,
config * Config ,
) ( Session , error ) {
return dialContext ( ctx , pconn , remoteAddr , host , tlsConf , config , false )
}
func dialContext (
ctx context . Context ,
pconn net . PacketConn ,
remoteAddr net . Addr ,
host string ,
tlsConf * tls . Config ,
config * Config ,
createdPacketConn bool ,
) ( Session , error ) {
config = populateClientConfig ( config , createdPacketConn )
if ! createdPacketConn {
for _ , v := range config . Versions {
if v == protocol . Version44 {
return nil , errors . New ( "Cannot multiplex connections using gQUIC 44, see https://groups.google.com/a/chromium.org/forum/#!topic/proto-quic/pE9NlLLjizE. Please disable gQUIC 44 in the quic.Config, or use DialAddr" )
}
}
}
packetHandlers , err := getMultiplexer ( ) . AddConn ( pconn , config . ConnectionIDLength )
if err != nil {
return nil , err
}
c , err := newClient ( pconn , remoteAddr , config , tlsConf , host , packetHandlers . Remove , createdPacketConn )
2017-05-27 13:30:11 -06:00
if err != nil {
return nil , err
}
2018-09-03 04:18:54 +07:00
c . packetHandlers = packetHandlers
if err := c . dial ( ctx ) ; err != nil {
return nil , err
}
return c . session , nil
}
2017-05-27 13:30:11 -06:00
2018-09-03 04:18:54 +07:00
func newClient (
pconn net . PacketConn ,
remoteAddr net . Addr ,
config * Config ,
tlsConf * tls . Config ,
host string ,
closeCallback func ( protocol . ConnectionID ) ,
createdPacketConn bool ,
) ( * client , error ) {
2017-07-27 16:11:56 -06:00
var hostname string
if tlsConf != nil {
hostname = tlsConf . ServerName
}
if hostname == "" {
2018-09-03 04:18:54 +07:00
var err error
2017-07-27 16:11:56 -06:00
hostname , _ , err = net . SplitHostPort ( host )
if err != nil {
return nil , err
}
2017-05-27 13:30:11 -06:00
}
2018-03-25 22:37:41 -06:00
// check that all versions are actually supported
if config != nil {
for _ , v := range config . Versions {
if ! protocol . IsValidVersion ( v ) {
return nil , fmt . Errorf ( "%s is not a valid QUIC version" , v )
}
}
}
2018-09-03 04:18:54 +07:00
onClose := func ( protocol . ConnectionID ) { }
if closeCallback != nil {
onClose = closeCallback
2017-05-27 13:30:11 -06:00
}
2018-09-03 04:18:54 +07:00
c := & client {
conn : & conn { pconn : pconn , currentAddr : remoteAddr } ,
createdPacketConn : createdPacketConn ,
hostname : hostname ,
tlsConf : tlsConf ,
config : config ,
version : config . Versions [ 0 ] ,
handshakeChan : make ( chan struct { } ) ,
closeCallback : onClose ,
logger : utils . DefaultLogger . WithPrefix ( "client" ) ,
}
return c , c . generateConnectionIDs ( )
2017-05-27 13:30:11 -06:00
}
2017-07-27 16:11:56 -06:00
// populateClientConfig populates fields in the quic.Config with their default values, if none are set
// it may be called with nil
2018-09-03 04:18:54 +07:00
func populateClientConfig ( config * Config , createdPacketConn bool ) * Config {
2017-07-27 16:11:56 -06:00
if config == nil {
config = & Config { }
}
2017-05-27 13:30:11 -06:00
versions := config . Versions
if len ( versions ) == 0 {
versions = protocol . SupportedVersions
}
2017-07-27 16:11:56 -06:00
handshakeTimeout := protocol . DefaultHandshakeTimeout
if config . HandshakeTimeout != 0 {
handshakeTimeout = config . HandshakeTimeout
}
2018-02-17 13:29:53 +08:00
idleTimeout := protocol . DefaultIdleTimeout
if config . IdleTimeout != 0 {
idleTimeout = config . IdleTimeout
}
2017-07-27 16:11:56 -06:00
maxReceiveStreamFlowControlWindow := config . MaxReceiveStreamFlowControlWindow
if maxReceiveStreamFlowControlWindow == 0 {
maxReceiveStreamFlowControlWindow = protocol . DefaultMaxReceiveStreamFlowControlWindowClient
}
maxReceiveConnectionFlowControlWindow := config . MaxReceiveConnectionFlowControlWindow
if maxReceiveConnectionFlowControlWindow == 0 {
maxReceiveConnectionFlowControlWindow = protocol . DefaultMaxReceiveConnectionFlowControlWindowClient
}
2018-03-25 22:37:41 -06:00
maxIncomingStreams := config . MaxIncomingStreams
if maxIncomingStreams == 0 {
maxIncomingStreams = protocol . DefaultMaxIncomingStreams
} else if maxIncomingStreams < 0 {
maxIncomingStreams = 0
}
maxIncomingUniStreams := config . MaxIncomingUniStreams
if maxIncomingUniStreams == 0 {
maxIncomingUniStreams = protocol . DefaultMaxIncomingUniStreams
} else if maxIncomingUniStreams < 0 {
maxIncomingUniStreams = 0
}
2018-09-03 04:18:54 +07:00
connIDLen := config . ConnectionIDLength
if connIDLen == 0 && ! createdPacketConn {
connIDLen = protocol . DefaultConnectionIDLength
}
for _ , v := range versions {
if v == protocol . Version44 {
connIDLen = 0
}
}
2017-07-27 16:11:56 -06:00
2017-05-27 13:30:11 -06:00
return & Config {
2017-07-27 16:11:56 -06:00
Versions : versions ,
HandshakeTimeout : handshakeTimeout ,
2018-02-17 13:29:53 +08:00
IdleTimeout : idleTimeout ,
RequestConnectionIDOmission : config . RequestConnectionIDOmission ,
2018-09-03 04:18:54 +07:00
ConnectionIDLength : connIDLen ,
2017-07-27 16:11:56 -06:00
MaxReceiveStreamFlowControlWindow : maxReceiveStreamFlowControlWindow ,
MaxReceiveConnectionFlowControlWindow : maxReceiveConnectionFlowControlWindow ,
2018-03-25 22:37:41 -06:00
MaxIncomingStreams : maxIncomingStreams ,
MaxIncomingUniStreams : maxIncomingUniStreams ,
KeepAlive : config . KeepAlive ,
2017-05-27 13:30:11 -06:00
}
}
2018-09-03 04:18:54 +07:00
func ( c * client ) generateConnectionIDs ( ) error {
connIDLen := protocol . ConnectionIDLenGQUIC
if c . version . UsesTLS ( ) {
connIDLen = c . config . ConnectionIDLength
}
srcConnID , err := generateConnectionID ( connIDLen )
if err != nil {
return err
}
destConnID := srcConnID
if c . version . UsesTLS ( ) {
destConnID , err = generateConnectionIDForInitial ( )
if err != nil {
return err
}
}
c . srcConnID = srcConnID
c . destConnID = destConnID
if c . version == protocol . Version44 {
c . srcConnID = nil
}
return nil
}
func ( c * client ) dial ( ctx context . Context ) error {
c . logger . Infof ( "Starting new connection to %s (%s -> %s), source connection ID %s, destination connection ID %s, version %s" , c . hostname , c . conn . LocalAddr ( ) , c . conn . RemoteAddr ( ) , c . srcConnID , c . destConnID , c . version )
2018-02-17 13:29:53 +08:00
var err error
if c . version . UsesTLS ( ) {
2018-09-03 04:18:54 +07:00
err = c . dialTLS ( ctx )
2018-02-17 13:29:53 +08:00
} else {
2018-09-03 04:18:54 +07:00
err = c . dialGQUIC ( ctx )
2018-02-17 13:29:53 +08:00
}
return err
}
2018-09-03 04:18:54 +07:00
func ( c * client ) dialGQUIC ( ctx context . Context ) error {
2018-02-17 13:29:53 +08:00
if err := c . createNewGQUICSession ( ) ; err != nil {
return err
}
2018-09-03 04:18:54 +07:00
err := c . establishSecureConnection ( ctx )
if err == errCloseSessionForNewVersion {
return c . dial ( ctx )
}
return err
2018-02-17 13:29:53 +08:00
}
2017-05-27 13:30:11 -06:00
2018-09-03 04:18:54 +07:00
func ( c * client ) dialTLS ( ctx context . Context ) error {
2018-02-17 13:29:53 +08:00
params := & handshake . TransportParameters {
StreamFlowControlWindow : protocol . ReceiveStreamFlowControlWindow ,
ConnectionFlowControlWindow : protocol . ReceiveConnectionFlowControlWindow ,
IdleTimeout : c . config . IdleTimeout ,
OmitConnectionID : c . config . RequestConnectionIDOmission ,
2018-03-25 22:37:41 -06:00
MaxBidiStreams : uint16 ( c . config . MaxIncomingStreams ) ,
MaxUniStreams : uint16 ( c . config . MaxIncomingUniStreams ) ,
2018-09-03 04:18:54 +07:00
DisableMigration : true ,
2018-02-17 13:29:53 +08:00
}
2018-04-18 15:48:08 -06:00
extHandler := handshake . NewExtensionHandlerClient ( params , c . initialVersion , c . config . Versions , c . version , c . logger )
2018-02-17 13:29:53 +08:00
mintConf , err := tlsToMintConfig ( c . tlsConf , protocol . PerspectiveClient )
if err != nil {
return err
}
mintConf . ExtensionHandler = extHandler
mintConf . ServerName = c . hostname
2018-09-03 04:18:54 +07:00
c . mintConf = mintConf
2018-02-17 13:29:53 +08:00
if err := c . createNewTLSSession ( extHandler . GetPeerParams ( ) , c . version ) ; err != nil {
return err
}
2018-09-03 04:18:54 +07:00
err = c . establishSecureConnection ( ctx )
if err == errCloseSessionForRetry || err == errCloseSessionForNewVersion {
return c . dial ( ctx )
2018-02-17 13:29:53 +08:00
}
2018-09-03 04:18:54 +07:00
return err
2018-02-17 13:29:53 +08:00
}
// establishSecureConnection runs the session, and tries to establish a secure connection
// It returns:
// - errCloseSessionForNewVersion when the server sends a version negotiation packet
// - handshake.ErrCloseSessionForRetry when the server performs a stateless retry (for IETF QUIC)
// - any other error that might occur
// - when the connection is secure (for gQUIC), or forward-secure (for IETF QUIC)
2018-09-03 04:18:54 +07:00
func ( c * client ) establishSecureConnection ( ctx context . Context ) error {
errorChan := make ( chan error , 1 )
2018-02-17 13:29:53 +08:00
go func ( ) {
2018-09-03 04:18:54 +07:00
err := c . session . run ( ) // returns as soon as the session is closed
if err != errCloseSessionForRetry && err != errCloseSessionForNewVersion && c . createdPacketConn {
2018-02-17 13:29:53 +08:00
c . conn . Close ( )
}
2018-09-03 04:18:54 +07:00
errorChan <- err
2018-02-17 13:29:53 +08:00
} ( )
select {
2018-09-03 04:18:54 +07:00
case <- ctx . Done ( ) :
// The session will send a PeerGoingAway error to the server.
c . session . Close ( )
return ctx . Err ( )
case err := <- errorChan :
2018-02-17 13:29:53 +08:00
return err
2018-09-03 04:18:54 +07:00
case <- c . handshakeChan :
// handshake successfully completed
return nil
2017-05-27 13:30:11 -06:00
}
}
2018-09-03 04:18:54 +07:00
func ( c * client ) handlePacket ( p * receivedPacket ) {
if err := c . handlePacketImpl ( p ) ; err != nil {
c . logger . Errorf ( "error handling packet: %s" , err )
2017-05-27 13:30:11 -06:00
}
}
2018-09-03 04:18:54 +07:00
func ( c * client ) handlePacketImpl ( p * receivedPacket ) error {
2017-05-27 13:30:11 -06:00
c . mutex . Lock ( )
defer c . mutex . Unlock ( )
2018-09-03 04:18:54 +07:00
// handle Version Negotiation Packets
if p . header . IsVersionNegotiation {
err := c . handleVersionNegotiationPacket ( p . header )
2017-07-27 16:11:56 -06:00
if err != nil {
2018-09-03 04:18:54 +07:00
c . session . destroy ( err )
2017-07-27 16:11:56 -06:00
}
2018-09-03 04:18:54 +07:00
// version negotiation packets have no payload
return err
2017-07-27 16:11:56 -06:00
}
2018-09-03 04:18:54 +07:00
if ! c . version . UsesIETFHeaderFormat ( ) {
connID := p . header . DestConnectionID
// reject packets with truncated connection id if we didn't request truncation
if ! c . config . RequestConnectionIDOmission && connID . Len ( ) == 0 {
return errors . New ( "received packet with truncated connection ID, but didn't request truncation" )
2018-02-17 13:29:53 +08:00
}
2018-09-03 04:18:54 +07:00
// reject packets with the wrong connection ID
if connID . Len ( ) > 0 && ! connID . Equal ( c . srcConnID ) {
return fmt . Errorf ( "received a packet with an unexpected connection ID (%s, expected %s)" , connID , c . srcConnID )
}
if p . header . ResetFlag {
return c . handlePublicReset ( p )
}
} else {
// reject packets with the wrong connection ID
if ! p . header . DestConnectionID . Equal ( c . srcConnID ) {
return fmt . Errorf ( "received a packet with an unexpected connection ID (%s, expected %s)" , p . header . DestConnectionID , c . srcConnID )
}
}
2017-05-27 13:30:11 -06:00
2018-09-03 04:18:54 +07:00
if p . header . IsLongHeader {
switch p . header . Type {
case protocol . PacketTypeRetry :
c . handleRetryPacket ( p . header )
return nil
case protocol . PacketTypeHandshake , protocol . PacketType0RTT :
default :
return fmt . Errorf ( "Received unsupported packet type: %s" , p . header . Type )
2017-07-27 16:11:56 -06:00
}
2017-05-27 13:30:11 -06:00
}
2018-02-17 13:29:53 +08:00
// this is the first packet we are receiving
// since it is not a Version Negotiation Packet, this means the server supports the suggested version
if ! c . versionNegotiated {
c . versionNegotiated = true
}
2018-09-03 04:18:54 +07:00
c . session . handlePacket ( p )
return nil
}
2018-02-17 13:29:53 +08:00
2018-09-03 04:18:54 +07:00
func ( c * client ) handlePublicReset ( p * receivedPacket ) error {
cr := c . conn . RemoteAddr ( )
// check if the remote address and the connection ID match
// otherwise this might be an attacker trying to inject a PUBLIC_RESET to kill the connection
if cr . Network ( ) != p . remoteAddr . Network ( ) || cr . String ( ) != p . remoteAddr . String ( ) || ! p . header . DestConnectionID . Equal ( c . srcConnID ) {
return errors . New ( "Received a spoofed Public Reset" )
}
pr , err := wire . ParsePublicReset ( bytes . NewReader ( p . data ) )
if err != nil {
return fmt . Errorf ( "Received a Public Reset. An error occurred parsing the packet: %s" , err )
}
c . session . closeRemote ( qerr . Error ( qerr . PublicReset , fmt . Sprintf ( "Received a Public Reset for packet number %#x" , pr . RejectedPacketNumber ) ) )
c . logger . Infof ( "Received Public Reset, rejected packet number: %#x" , pr . RejectedPacketNumber )
return nil
2017-05-27 13:30:11 -06:00
}
2018-02-17 13:29:53 +08:00
func ( c * client ) handleVersionNegotiationPacket ( hdr * wire . Header ) error {
2018-09-03 04:18:54 +07:00
// ignore delayed / duplicated version negotiation packets
if c . receivedVersionNegotiationPacket || c . versionNegotiated {
c . logger . Debugf ( "Received a delayed Version Negotiation Packet." )
return nil
}
2017-05-27 13:30:11 -06:00
for _ , v := range hdr . SupportedVersions {
if v == c . version {
// the version negotiation packet contains the version that we offered
// this might be a packet sent by an attacker (or by a terribly broken server implementation)
// ignore it
return nil
}
}
2018-04-18 15:48:08 -06:00
c . logger . Infof ( "Received a Version Negotiation Packet. Supported Versions: %s" , hdr . SupportedVersions )
2018-02-17 13:29:53 +08:00
newVersion , ok := protocol . ChooseSupportedVersion ( c . config . Versions , hdr . SupportedVersions )
if ! ok {
2017-05-27 13:30:11 -06:00
return qerr . InvalidVersion
}
2018-02-17 13:29:53 +08:00
c . receivedVersionNegotiationPacket = true
c . negotiatedVersions = hdr . SupportedVersions
2017-05-27 13:30:11 -06:00
// switch to negotiated version
2018-02-17 13:29:53 +08:00
c . initialVersion = c . version
2017-05-27 13:30:11 -06:00
c . version = newVersion
2018-09-03 04:18:54 +07:00
if err := c . generateConnectionIDs ( ) ; err != nil {
2017-05-27 13:30:11 -06:00
return err
}
2018-09-03 04:18:54 +07:00
c . logger . Infof ( "Switching to QUIC version %s. New connection ID: %s" , newVersion , c . destConnID )
c . session . destroy ( errCloseSessionForNewVersion )
2018-02-17 13:29:53 +08:00
return nil
2017-05-27 13:30:11 -06:00
}
2018-09-03 04:18:54 +07:00
func ( c * client ) handleRetryPacket ( hdr * wire . Header ) {
c . logger . Debugf ( "<- Received Retry" )
hdr . Log ( c . logger )
// A server that performs multiple retries must use a source connection ID of at least 8 bytes.
// Only a server that won't send additional Retries can use shorter connection IDs.
if hdr . OrigDestConnectionID . Len ( ) < protocol . MinConnectionIDLenInitial {
c . logger . Debugf ( "Received a Retry with a too short Original Destination Connection ID: %d bytes, must have at least %d bytes." , hdr . OrigDestConnectionID . Len ( ) , protocol . MinConnectionIDLenInitial )
return
}
if ! hdr . OrigDestConnectionID . Equal ( c . destConnID ) {
c . logger . Debugf ( "Received spoofed Retry. Original Destination Connection ID: %s, expected: %s" , hdr . OrigDestConnectionID , c . destConnID )
return
}
c . numRetries ++
if c . numRetries > protocol . MaxRetries {
c . session . destroy ( qerr . CryptoTooManyRejects )
return
}
c . destConnID = hdr . SrcConnectionID
c . token = hdr . Token
c . session . destroy ( errCloseSessionForRetry )
}
func ( c * client ) createNewGQUICSession ( ) error {
2018-02-17 13:29:53 +08:00
c . mutex . Lock ( )
defer c . mutex . Unlock ( )
2018-09-03 04:18:54 +07:00
runner := & runner {
onHandshakeCompleteImpl : func ( _ Session ) { close ( c . handshakeChan ) } ,
removeConnectionIDImpl : c . closeCallback ,
}
sess , err := newClientSession (
2017-05-27 13:30:11 -06:00
c . conn ,
2018-09-03 04:18:54 +07:00
runner ,
2017-05-27 13:30:11 -06:00
c . hostname ,
c . version ,
2018-09-03 04:18:54 +07:00
c . destConnID ,
c . srcConnID ,
2017-07-27 16:11:56 -06:00
c . tlsConf ,
2017-05-27 13:30:11 -06:00
c . config ,
2018-02-17 13:29:53 +08:00
c . initialVersion ,
c . negotiatedVersions ,
2018-04-18 15:48:08 -06:00
c . logger ,
2017-05-27 13:30:11 -06:00
)
2018-09-03 04:18:54 +07:00
if err != nil {
return err
}
c . session = sess
c . packetHandlers . Add ( c . srcConnID , c )
if c . config . RequestConnectionIDOmission {
c . packetHandlers . Add ( protocol . ConnectionID { } , c )
}
return nil
2018-02-17 13:29:53 +08:00
}
2017-05-27 13:30:11 -06:00
2018-02-17 13:29:53 +08:00
func ( c * client ) createNewTLSSession (
paramsChan <- chan handshake . TransportParameters ,
version protocol . VersionNumber ,
2018-09-03 04:18:54 +07:00
) error {
2018-02-17 13:29:53 +08:00
c . mutex . Lock ( )
defer c . mutex . Unlock ( )
2018-09-03 04:18:54 +07:00
runner := & runner {
onHandshakeCompleteImpl : func ( _ Session ) { close ( c . handshakeChan ) } ,
removeConnectionIDImpl : c . closeCallback ,
}
sess , err := newTLSClientSession (
2018-02-17 13:29:53 +08:00
c . conn ,
2018-09-03 04:18:54 +07:00
runner ,
c . token ,
c . destConnID ,
c . srcConnID ,
2018-02-17 13:29:53 +08:00
c . config ,
2018-09-03 04:18:54 +07:00
c . mintConf ,
2018-02-17 13:29:53 +08:00
paramsChan ,
1 ,
2018-04-18 15:48:08 -06:00
c . logger ,
2018-09-03 04:18:54 +07:00
c . version ,
2018-02-17 13:29:53 +08:00
)
2018-09-03 04:18:54 +07:00
if err != nil {
return err
}
c . session = sess
c . packetHandlers . Add ( c . srcConnID , c )
return nil
}
func ( c * client ) Close ( ) error {
c . mutex . Lock ( )
defer c . mutex . Unlock ( )
if c . session == nil {
return nil
}
return c . session . Close ( )
}
func ( c * client ) destroy ( e error ) {
c . mutex . Lock ( )
defer c . mutex . Unlock ( )
if c . session == nil {
return
}
c . session . destroy ( e )
}
func ( c * client ) GetVersion ( ) protocol . VersionNumber {
c . mutex . Lock ( )
v := c . version
c . mutex . Unlock ( )
return v
}
func ( c * client ) GetPerspective ( ) protocol . Perspective {
return protocol . PerspectiveClient
2017-05-27 13:30:11 -06:00
}