|
|
@@ -1,103 +0,0 @@
|
|
|
-// 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?")
|
|
|
- }
|
|
|
-}
|