From 58fdcad971d4a029ec5fefa51c26d1e464583674 Mon Sep 17 00:00:00 2001 From: Juan Batiz-Benet <juan@benet.ai> Date: Sun, 19 Oct 2014 04:01:48 -0700 Subject: [PATCH] multiconn: map + close on children close --- net/conn/multiconn.go | 8 ++++++++ net/conn/multiconn_test.go | 29 +++++++++++++++++++++++++++++ net/conn/secure_conn_test.go | 4 ++-- 3 files changed, 39 insertions(+), 2 deletions(-) diff --git a/net/conn/multiconn.go b/net/conn/multiconn.go index f4915db2b..1c4533d72 100644 --- a/net/conn/multiconn.go +++ b/net/conn/multiconn.go @@ -10,6 +10,9 @@ import ( u "github.com/jbenet/go-ipfs/util" ) +// MultiConnMap is for shorthand +type MultiConnMap map[u.Key]*MultiConn + // Duplex is a simple duplex channel type Duplex struct { In chan []byte @@ -160,10 +163,15 @@ func (c *MultiConn) fanInSingle(child Conn) { // in case it still is in the map, remove it. c.Lock() delete(c.conns, child.ID()) + connLen := len(c.conns) c.Unlock() c.Children().Done() child.Children().Done() + + if connLen == 0 { + c.Close() // close self if all underlying children are gone? + } }() for { diff --git a/net/conn/multiconn_test.go b/net/conn/multiconn_test.go index 0fde058d3..bc0b59bf2 100644 --- a/net/conn/multiconn_test.go +++ b/net/conn/multiconn_test.go @@ -293,3 +293,32 @@ func TestMulticonnSendUnderlying(t *testing.T) { msgsFrom1.CheckDone(t) msgsFrom2.CheckDone(t) } + +func TestMulticonnClose(t *testing.T) { + // t.Skip("fooo") + + log.Info("TestMulticonnSendUnderlying") + ctx := context.Background() + c1, c2 := setupMultiConns(t, ctx) + + for _, c := range c1.conns { + c.Close() + } + + for _, c := range c2.conns { + c.Close() + } + + timeout := time.After(100 * time.Millisecond) + select { + case <-c1.Closed(): + case <-timeout: + t.Fatal("timeout") + } + + select { + case <-c2.Closed(): + case <-timeout: + t.Fatal("timeout") + } +} diff --git a/net/conn/secure_conn_test.go b/net/conn/secure_conn_test.go index 5a78870d0..17c0f4bd9 100644 --- a/net/conn/secure_conn_test.go +++ b/net/conn/secure_conn_test.go @@ -50,7 +50,7 @@ func TestSecureClose(t *testing.T) { select { case <-c1.Closed(): default: - t.Fatal("not done after cancel") + t.Fatal("not done after close") } c2.Close() @@ -58,7 +58,7 @@ func TestSecureClose(t *testing.T) { select { case <-c2.Closed(): default: - t.Fatal("not done after cancel") + t.Fatal("not done after close") } cancel() // close the listener :P -- GitLab