You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							56 lines
						
					
					
						
							1.2 KiB
						
					
					
				
			
		
		
	
	
							56 lines
						
					
					
						
							1.2 KiB
						
					
					
				| // 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 {
 | |
| 	close sync.Once
 | |
| 	net.Conn
 | |
| }
 | |
| 
 | |
| func (lc *limitConn) Close() error {
 | |
| 	defer lc.close.Do(func() {
 | |
| 		limitRelease()
 | |
| 		runtime.SetFinalizer(lc, nil)
 | |
| 	})
 | |
| 	return lc.Conn.Close()
 | |
| }
 | |
| 
 |