@@ -10,6 +10,8 @@ import (
10
10
"crypto/tls"
11
11
"net/http"
12
12
"sync"
13
+ "math/rand"
14
+ "time"
13
15
)
14
16
15
17
// ClientConnPool manages a pool of HTTP/2 client connections.
@@ -84,27 +86,52 @@ func (p *clientConnPool) getClientConn(req *http.Request, addr string, dialOnMis
84
86
}
85
87
return cc , nil
86
88
}
89
+
87
90
p .mu .Lock ()
88
- for _ , cc := range p .conns [addr ] {
89
- if st := cc .idleState (); st .canTakeNewRequest {
90
- if p .shouldTraceGetConn (st ) {
91
- traceGetConn (req , addr )
91
+ defer p .mu .Unlock ()
92
+
93
+ ccList := p .conns [addr ]
94
+ minCon := int (p .t .MinConcurrentConns )
95
+
96
+ if len (ccList ) >= minCon {
97
+ if minCon > 0 {
98
+ rndSrc := rand .NewSource (time .Now ().UnixNano ())
99
+ rnd := rand .New (rndSrc )
100
+ startOffset := rnd .Intn (minCon )
101
+ // find an available connection randomly among MinConcurrentConns connections for spreading streams
102
+ for i := 0 ; i < minCon ; i ++ {
103
+ idx := (i + startOffset ) % minCon
104
+ if p .takeNewRequest (req , addr , ccList [idx ]) {
105
+ return ccList [idx ], nil
106
+ }
107
+ }
108
+ }
109
+ // find an available connection sequentially among the rests
110
+ for _ , cc := range ccList [minCon :] {
111
+ if p .takeNewRequest (req , addr , cc ) {
112
+ return cc , nil
92
113
}
93
- p .mu .Unlock ()
94
- return cc , nil
95
114
}
96
115
}
97
116
if ! dialOnMiss {
98
- p .mu .Unlock ()
99
117
return nil , ErrNoCachedConn
100
118
}
101
119
traceGetConn (req , addr )
102
120
call := p .getStartDialLocked (addr )
103
- p .mu .Unlock ()
104
121
<- call .done
105
122
return call .res , call .err
106
123
}
107
124
125
+ func (p * clientConnPool ) takeNewRequest (req * http.Request , addr string , cc * ClientConn ) bool {
126
+ if st := cc .idleState (); st .canTakeNewRequest {
127
+ if p .shouldTraceGetConn (st ) {
128
+ traceGetConn (req , addr )
129
+ }
130
+ return true
131
+ }
132
+ return false
133
+ }
134
+
108
135
// dialCall is an in-flight Transport dial call to a host.
109
136
type dialCall struct {
110
137
p * clientConnPool
0 commit comments