mirror of
https://github.com/refraction-networking/utls.git
synced 2025-04-04 12:37:35 +03:00
[+] HandshakeContext
This commit is contained in:
parent
d3bc26fec3
commit
91e7e448ec
1 changed files with 62 additions and 3 deletions
65
conn.go
65
conn.go
|
@ -13,6 +13,7 @@ import (
|
||||||
"crypto/x509"
|
"crypto/x509"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"golang.org/x/net/context"
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"sync"
|
"sync"
|
||||||
|
@ -585,12 +586,14 @@ func (c *Conn) readChangeCipherSpec() error {
|
||||||
|
|
||||||
// readRecordOrCCS reads one or more TLS records from the connection and
|
// readRecordOrCCS reads one or more TLS records from the connection and
|
||||||
// updates the record layer state. Some invariants:
|
// updates the record layer state. Some invariants:
|
||||||
// * c.in must be locked
|
// - c.in must be locked
|
||||||
// * c.input must be empty
|
// - c.input must be empty
|
||||||
|
//
|
||||||
// During the handshake one and only one of the following will happen:
|
// During the handshake one and only one of the following will happen:
|
||||||
// - c.hand grows
|
// - c.hand grows
|
||||||
// - c.in.changeCipherSpec is called
|
// - c.in.changeCipherSpec is called
|
||||||
// - an error is returned
|
// - an error is returned
|
||||||
|
//
|
||||||
// After the handshake one and only one of the following will happen:
|
// After the handshake one and only one of the following will happen:
|
||||||
// - c.hand grows
|
// - c.hand grows
|
||||||
// - c.input is set
|
// - c.input is set
|
||||||
|
@ -1348,7 +1351,61 @@ func (c *Conn) closeNotify() error {
|
||||||
// protocol if it has not yet been run.
|
// protocol if it has not yet been run.
|
||||||
// Most uses of this package need not call Handshake
|
// Most uses of this package need not call Handshake
|
||||||
// explicitly: the first Read or Write will call it automatically.
|
// explicitly: the first Read or Write will call it automatically.
|
||||||
|
|
||||||
func (c *Conn) Handshake() error {
|
func (c *Conn) Handshake() error {
|
||||||
|
return c.HandshakeContext(context.Background())
|
||||||
|
}
|
||||||
|
|
||||||
|
// HandshakeContext runs the client or server handshake
|
||||||
|
// protocol if it has not yet been run.
|
||||||
|
//
|
||||||
|
// The provided Context must be non-nil. If the context is canceled before
|
||||||
|
// the handshake is complete, the handshake is interrupted and an error is returned.
|
||||||
|
// Once the handshake has completed, cancellation of the context will not affect the
|
||||||
|
// connection.
|
||||||
|
//
|
||||||
|
// Most uses of this package need not call HandshakeContext explicitly: the
|
||||||
|
// first Read or Write will call it automatically.
|
||||||
|
func (c *Conn) HandshakeContext(ctx context.Context) error {
|
||||||
|
// Delegate to unexported method for named return
|
||||||
|
// without confusing documented signature.
|
||||||
|
return c.handshakeContext(ctx)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Conn) handshakeContext(ctx context.Context) error {
|
||||||
|
handshakeCtx, cancel := context.WithCancel(ctx)
|
||||||
|
// Note: defer this before starting the "interrupter" goroutine
|
||||||
|
// so that we can tell the difference between the input being canceled and
|
||||||
|
// this cancellation. In the former case, we need to close the connection.
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
// Start the "interrupter" goroutine, if this context might be canceled.
|
||||||
|
// (The background context cannot).
|
||||||
|
//
|
||||||
|
// The interrupter goroutine waits for the input context to be done and
|
||||||
|
// closes the connection if this happens before the function returns.
|
||||||
|
if ctx.Done() != nil {
|
||||||
|
done := make(chan struct{})
|
||||||
|
interruptRes := make(chan error, 1)
|
||||||
|
defer func() {
|
||||||
|
close(done)
|
||||||
|
if ctxErr := <-interruptRes; ctxErr != nil {
|
||||||
|
// Return context error to user.
|
||||||
|
c.handshakeErr = ctxErr
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
go func() {
|
||||||
|
select {
|
||||||
|
case <-handshakeCtx.Done():
|
||||||
|
// Close the connection, discarding the error
|
||||||
|
_ = c.conn.Close()
|
||||||
|
interruptRes <- handshakeCtx.Err()
|
||||||
|
case <-done:
|
||||||
|
interruptRes <- nil
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
||||||
c.handshakeMutex.Lock()
|
c.handshakeMutex.Lock()
|
||||||
defer c.handshakeMutex.Unlock()
|
defer c.handshakeMutex.Unlock()
|
||||||
|
|
||||||
|
@ -1370,13 +1427,15 @@ func (c *Conn) Handshake() error {
|
||||||
if c.handshakeErr == nil {
|
if c.handshakeErr == nil {
|
||||||
c.handshakes++
|
c.handshakes++
|
||||||
} else {
|
} else {
|
||||||
// If an error occurred during the hadshake try to flush the
|
// If an error occurred during the handshake try to flush the
|
||||||
// alert that might be left in the buffer.
|
// alert that might be left in the buffer.
|
||||||
c.flush()
|
c.flush()
|
||||||
}
|
}
|
||||||
|
|
||||||
if c.handshakeErr == nil && !c.handshakeComplete() {
|
if c.handshakeErr == nil && !c.handshakeComplete() {
|
||||||
c.handshakeErr = errors.New("tls: internal error: handshake should have had a result")
|
c.handshakeErr = errors.New("tls: internal error: handshake should have had a result")
|
||||||
|
} else if c.handshakeErr != nil && c.handshakeComplete() {
|
||||||
|
c.handshakeErr = errors.New("tls: internal error: handshake returned an error but is marked successful")
|
||||||
}
|
}
|
||||||
|
|
||||||
return c.handshakeErr
|
return c.handshakeErr
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue