|
@@ -0,0 +1,103 @@
|
|
|
|
|
+// Copyright 2013 The Go Authors. All rights reserved.
|
|
|
|
|
+// Use of this source code is governed by a BSD-style
|
|
|
|
|
+// license that can be found in the LICENSE file.
|
|
|
|
|
+
|
|
|
|
|
+// +build go1.3
|
|
|
|
|
+
|
|
|
|
|
+// (We only run this test on Go 1.3 because the HTTP client timeout behavior
|
|
|
|
|
+// was bad in previous releases, causing occasional deadlocks.)
|
|
|
|
|
+
|
|
|
|
|
+package netutil
|
|
|
|
|
+
|
|
|
|
|
+import (
|
|
|
|
|
+ "errors"
|
|
|
|
|
+ "fmt"
|
|
|
|
|
+ "io"
|
|
|
|
|
+ "io/ioutil"
|
|
|
|
|
+ "net"
|
|
|
|
|
+ "net/http"
|
|
|
|
|
+ "sync"
|
|
|
|
|
+ "sync/atomic"
|
|
|
|
|
+ "testing"
|
|
|
|
|
+ "time"
|
|
|
|
|
+)
|
|
|
|
|
+
|
|
|
|
|
+func TestLimitListener(t *testing.T) {
|
|
|
|
|
+ const (
|
|
|
|
|
+ max = 5
|
|
|
|
|
+ num = 200
|
|
|
|
|
+ )
|
|
|
|
|
+
|
|
|
|
|
+ l, err := net.Listen("tcp", "127.0.0.1:0")
|
|
|
|
|
+ if err != nil {
|
|
|
|
|
+ t.Fatalf("Listen: %v", err)
|
|
|
|
|
+ }
|
|
|
|
|
+ defer l.Close()
|
|
|
|
|
+ l = LimitListener(l, max)
|
|
|
|
|
+
|
|
|
|
|
+ var open int32
|
|
|
|
|
+ go http.Serve(l, http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
+ if n := atomic.AddInt32(&open, 1); n > max {
|
|
|
|
|
+ t.Errorf("%d open connections, want <= %d", n, max)
|
|
|
|
|
+ }
|
|
|
|
|
+ defer atomic.AddInt32(&open, -1)
|
|
|
|
|
+ time.Sleep(10 * time.Millisecond)
|
|
|
|
|
+ fmt.Fprint(w, "some body")
|
|
|
|
|
+ }))
|
|
|
|
|
+
|
|
|
|
|
+ var wg sync.WaitGroup
|
|
|
|
|
+ var failed int32
|
|
|
|
|
+ for i := 0; i < num; i++ {
|
|
|
|
|
+ wg.Add(1)
|
|
|
|
|
+ go func() {
|
|
|
|
|
+ defer wg.Done()
|
|
|
|
|
+ c := http.Client{Timeout: 3 * time.Second}
|
|
|
|
|
+ r, err := c.Get("http://" + l.Addr().String())
|
|
|
|
|
+ if err != nil {
|
|
|
|
|
+ t.Logf("Get: %v", err)
|
|
|
|
|
+ atomic.AddInt32(&failed, 1)
|
|
|
|
|
+ return
|
|
|
|
|
+ }
|
|
|
|
|
+ defer r.Body.Close()
|
|
|
|
|
+ io.Copy(ioutil.Discard, r.Body)
|
|
|
|
|
+ }()
|
|
|
|
|
+ }
|
|
|
|
|
+ wg.Wait()
|
|
|
|
|
+
|
|
|
|
|
+ // We expect some Gets to fail as the kernel's accept queue is filled,
|
|
|
|
|
+ // but most should succeed.
|
|
|
|
|
+ if failed >= num/2 {
|
|
|
|
|
+ t.Errorf("too many Gets failed: %v", failed)
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+type errorListener struct {
|
|
|
|
|
+ net.Listener
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+func (errorListener) Accept() (net.Conn, error) {
|
|
|
|
|
+ return nil, errFake
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+var errFake = errors.New("fake error from errorListener")
|
|
|
|
|
+
|
|
|
|
|
+// This used to hang.
|
|
|
|
|
+func TestLimitListenerError(t *testing.T) {
|
|
|
|
|
+ donec := make(chan bool, 1)
|
|
|
|
|
+ go func() {
|
|
|
|
|
+ const n = 2
|
|
|
|
|
+ ll := LimitListener(errorListener{}, n)
|
|
|
|
|
+ for i := 0; i < n+1; i++ {
|
|
|
|
|
+ _, err := ll.Accept()
|
|
|
|
|
+ if err != errFake {
|
|
|
|
|
+ t.Fatalf("Accept error = %v; want errFake", err)
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+ donec <- true
|
|
|
|
|
+ }()
|
|
|
|
|
+ select {
|
|
|
|
|
+ case <-donec:
|
|
|
|
|
+ case <-time.After(5 * time.Second):
|
|
|
|
|
+ t.Fatal("timeout. deadlock?")
|
|
|
|
|
+ }
|
|
|
|
|
+}
|