mirror of
https://github.com/refraction-networking/utls.git
synced 2025-04-03 20:17:36 +03:00
crypto/tls: add CloseWrite method to Conn
The CloseWrite method sends a close_notify alert record to the other side of the connection. This record indicates that the sender has finished sending on the connection. Unlike the Close method, the sender may still read from the connection until it recieves a close_notify record (or the underlying connection is closed). This is analogous to a TCP half-close. This is a rework of CL 25159 with fixes for the unstable test. Updates #8579 Change-Id: I47608d2f82a88baff07a90fd64c280ed16a60d5e Reviewed-on: https://go-review.googlesource.com/31318 Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org> Run-TryBot: Brad Fitzpatrick <bradfitz@golang.org> TryBot-Result: Gobot Gobot <gobot@golang.org>
This commit is contained in:
parent
3fac26a27b
commit
d313432832
2 changed files with 149 additions and 2 deletions
107
tls_test.go
107
tls_test.go
|
@ -11,6 +11,7 @@ import (
|
|||
"fmt"
|
||||
"internal/testenv"
|
||||
"io"
|
||||
"io/ioutil"
|
||||
"math"
|
||||
"math/rand"
|
||||
"net"
|
||||
|
@ -458,6 +459,112 @@ func TestConnCloseBreakingWrite(t *testing.T) {
|
|||
}
|
||||
}
|
||||
|
||||
func TestConnCloseWrite(t *testing.T) {
|
||||
ln := newLocalListener(t)
|
||||
defer ln.Close()
|
||||
|
||||
clientDoneChan := make(chan struct{})
|
||||
|
||||
serverCloseWrite := func() error {
|
||||
sconn, err := ln.Accept()
|
||||
if err != nil {
|
||||
return fmt.Errorf("accept: %v", err)
|
||||
}
|
||||
defer sconn.Close()
|
||||
|
||||
serverConfig := testConfig.Clone()
|
||||
srv := Server(sconn, serverConfig)
|
||||
if err := srv.Handshake(); err != nil {
|
||||
return fmt.Errorf("handshake: %v", err)
|
||||
}
|
||||
defer srv.Close()
|
||||
|
||||
data, err := ioutil.ReadAll(srv)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if len(data) > 0 {
|
||||
return fmt.Errorf("Read data = %q; want nothing", data)
|
||||
}
|
||||
|
||||
if err := srv.CloseWrite(); err != nil {
|
||||
return fmt.Errorf("server CloseWrite: %v", err)
|
||||
}
|
||||
|
||||
// Wait for clientCloseWrite to finish, so we know we
|
||||
// tested the CloseWrite before we defer the
|
||||
// sconn.Close above, which would also cause the
|
||||
// client to unblock like CloseWrite.
|
||||
<-clientDoneChan
|
||||
return nil
|
||||
}
|
||||
|
||||
clientCloseWrite := func() error {
|
||||
defer close(clientDoneChan)
|
||||
|
||||
clientConfig := testConfig.Clone()
|
||||
conn, err := Dial("tcp", ln.Addr().String(), clientConfig)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if err := conn.Handshake(); err != nil {
|
||||
return err
|
||||
}
|
||||
defer conn.Close()
|
||||
|
||||
if err := conn.CloseWrite(); err != nil {
|
||||
return fmt.Errorf("client CloseWrite: %v", err)
|
||||
}
|
||||
|
||||
if _, err := conn.Write([]byte{0}); err != errShutdown {
|
||||
return fmt.Errorf("CloseWrite error = %v; want errShutdown", err)
|
||||
}
|
||||
|
||||
data, err := ioutil.ReadAll(conn)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if len(data) > 0 {
|
||||
return fmt.Errorf("Read data = %q; want nothing", data)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
errChan := make(chan error, 2)
|
||||
|
||||
go func() { errChan <- serverCloseWrite() }()
|
||||
go func() { errChan <- clientCloseWrite() }()
|
||||
|
||||
for i := 0; i < 2; i++ {
|
||||
select {
|
||||
case err := <-errChan:
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
case <-time.After(10 * time.Second):
|
||||
t.Fatal("deadlock")
|
||||
}
|
||||
}
|
||||
|
||||
// Also test CloseWrite being called before the handshake is
|
||||
// finished:
|
||||
{
|
||||
ln2 := newLocalListener(t)
|
||||
defer ln2.Close()
|
||||
|
||||
netConn, err := net.Dial("tcp", ln2.Addr().String())
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
defer netConn.Close()
|
||||
conn := Client(netConn, testConfig.Clone())
|
||||
|
||||
if err := conn.CloseWrite(); err != errEarlyCloseWrite {
|
||||
t.Errorf("CloseWrite error = %v; want errEarlyCloseWrite", err)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestClone(t *testing.T) {
|
||||
var c1 Config
|
||||
v := reflect.ValueOf(&c1).Elem()
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue