Created
April 10, 2015 02:39
-
-
Save cmelbye/3b42461c23a37dcf5e09 to your computer and use it in GitHub Desktop.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// Copyright 2014 Google Inc. All rights reserved. | |
// Use of this source code is governed by the Apache 2.0 | |
// license that can be found in the LICENSE file. | |
package internal | |
// This file implements a network dialer that limits the number of concurrent connections. | |
// It is only used for API calls. | |
import ( | |
"log" | |
"net" | |
"runtime" | |
"sync" | |
"time" | |
) | |
var limitSem = make(chan int, 100) // TODO(dsymonds): Use environment variable. | |
func limitRelease() { | |
// non-blocking | |
select { | |
case <-limitSem: | |
default: | |
// This should not normally happen. | |
log.Print("appengine: unbalanced limitSem release!") | |
} | |
} | |
func limitDial(network, addr string) (net.Conn, error) { | |
limitSem <- 1 | |
// Dial with a timeout in case the API host is MIA. | |
// The connection should normally be very fast. | |
conn, err := net.DialTimeout(network, addr, 500*time.Millisecond) | |
if err != nil { | |
limitRelease() | |
return nil, err | |
} | |
lc := &limitConn{Conn: conn} | |
runtime.SetFinalizer(lc, (*limitConn).Close) // shouldn't usually be required | |
return lc, nil | |
} | |
type limitConn struct { | |
relonce sync.Once | |
net.Conn | |
} | |
func (lc *limitConn) Close() error { | |
defer lc.relonce.Do(func() { | |
limitRelease() | |
runtime.SetFinalizer(lc, nil) | |
}) | |
return lc.Conn.Close() | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment