mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-23 11:21:26 +01:00
e3cc329d85
The v0.1.0 points to the last verified changes made by me.
I'm afraid that releases after v0.1.0 may contain completely broken changes like
996610f021
449 lines
12 KiB
Go
449 lines
12 KiB
Go
package fasthttp
|
|
|
|
import (
|
|
"errors"
|
|
"net"
|
|
"strconv"
|
|
"sync"
|
|
"sync/atomic"
|
|
"time"
|
|
)
|
|
|
|
// Dial dials the given TCP addr using tcp4.
|
|
//
|
|
// This function has the following additional features comparing to net.Dial:
|
|
//
|
|
// * It reduces load on DNS resolver by caching resolved TCP addressed
|
|
// for DefaultDNSCacheDuration.
|
|
// * It dials all the resolved TCP addresses in round-robin manner until
|
|
// connection is established. This may be useful if certain addresses
|
|
// are temporarily unreachable.
|
|
// * It returns ErrDialTimeout if connection cannot be established during
|
|
// DefaultDialTimeout seconds. Use DialTimeout for customizing dial timeout.
|
|
//
|
|
// This dialer is intended for custom code wrapping before passing
|
|
// to Client.Dial or HostClient.Dial.
|
|
//
|
|
// For instance, per-host counters and/or limits may be implemented
|
|
// by such wrappers.
|
|
//
|
|
// The addr passed to the function must contain port. Example addr values:
|
|
//
|
|
// * foobar.baz:443
|
|
// * foo.bar:80
|
|
// * aaa.com:8080
|
|
func Dial(addr string) (net.Conn, error) {
|
|
return defaultDialer.Dial(addr)
|
|
}
|
|
|
|
// DialTimeout dials the given TCP addr using tcp4 using the given timeout.
|
|
//
|
|
// This function has the following additional features comparing to net.Dial:
|
|
//
|
|
// * It reduces load on DNS resolver by caching resolved TCP addressed
|
|
// for DefaultDNSCacheDuration.
|
|
// * It dials all the resolved TCP addresses in round-robin manner until
|
|
// connection is established. This may be useful if certain addresses
|
|
// are temporarily unreachable.
|
|
//
|
|
// This dialer is intended for custom code wrapping before passing
|
|
// to Client.Dial or HostClient.Dial.
|
|
//
|
|
// For instance, per-host counters and/or limits may be implemented
|
|
// by such wrappers.
|
|
//
|
|
// The addr passed to the function must contain port. Example addr values:
|
|
//
|
|
// * foobar.baz:443
|
|
// * foo.bar:80
|
|
// * aaa.com:8080
|
|
func DialTimeout(addr string, timeout time.Duration) (net.Conn, error) {
|
|
return defaultDialer.DialTimeout(addr, timeout)
|
|
}
|
|
|
|
// DialDualStack dials the given TCP addr using both tcp4 and tcp6.
|
|
//
|
|
// This function has the following additional features comparing to net.Dial:
|
|
//
|
|
// * It reduces load on DNS resolver by caching resolved TCP addressed
|
|
// for DefaultDNSCacheDuration.
|
|
// * It dials all the resolved TCP addresses in round-robin manner until
|
|
// connection is established. This may be useful if certain addresses
|
|
// are temporarily unreachable.
|
|
// * It returns ErrDialTimeout if connection cannot be established during
|
|
// DefaultDialTimeout seconds. Use DialDualStackTimeout for custom dial
|
|
// timeout.
|
|
//
|
|
// This dialer is intended for custom code wrapping before passing
|
|
// to Client.Dial or HostClient.Dial.
|
|
//
|
|
// For instance, per-host counters and/or limits may be implemented
|
|
// by such wrappers.
|
|
//
|
|
// The addr passed to the function must contain port. Example addr values:
|
|
//
|
|
// * foobar.baz:443
|
|
// * foo.bar:80
|
|
// * aaa.com:8080
|
|
func DialDualStack(addr string) (net.Conn, error) {
|
|
return defaultDialer.DialDualStack(addr)
|
|
}
|
|
|
|
// DialDualStackTimeout dials the given TCP addr using both tcp4 and tcp6
|
|
// using the given timeout.
|
|
//
|
|
// This function has the following additional features comparing to net.Dial:
|
|
//
|
|
// * It reduces load on DNS resolver by caching resolved TCP addressed
|
|
// for DefaultDNSCacheDuration.
|
|
// * It dials all the resolved TCP addresses in round-robin manner until
|
|
// connection is established. This may be useful if certain addresses
|
|
// are temporarily unreachable.
|
|
//
|
|
// This dialer is intended for custom code wrapping before passing
|
|
// to Client.Dial or HostClient.Dial.
|
|
//
|
|
// For instance, per-host counters and/or limits may be implemented
|
|
// by such wrappers.
|
|
//
|
|
// The addr passed to the function must contain port. Example addr values:
|
|
//
|
|
// * foobar.baz:443
|
|
// * foo.bar:80
|
|
// * aaa.com:8080
|
|
func DialDualStackTimeout(addr string, timeout time.Duration) (net.Conn, error) {
|
|
return defaultDialer.DialDualStackTimeout(addr, timeout)
|
|
}
|
|
|
|
var (
|
|
defaultDialer = &TCPDialer{Concurrency: 1000}
|
|
)
|
|
|
|
// TCPDialer contains options to control a group of Dial calls.
|
|
type TCPDialer struct {
|
|
// Concurrency controls the maximum number of concurrent Dails
|
|
// that can be performed using this object.
|
|
// Setting this to 0 means unlimited.
|
|
//
|
|
// WARNING: This can only be changed before the first Dial.
|
|
// Changes made after the first Dial will not affect anything.
|
|
Concurrency int
|
|
|
|
tcpAddrsLock sync.Mutex
|
|
tcpAddrsMap map[string]*tcpAddrEntry
|
|
|
|
concurrencyCh chan struct{}
|
|
|
|
once sync.Once
|
|
}
|
|
|
|
// Dial dials the given TCP addr using tcp4.
|
|
//
|
|
// This function has the following additional features comparing to net.Dial:
|
|
//
|
|
// * It reduces load on DNS resolver by caching resolved TCP addressed
|
|
// for DefaultDNSCacheDuration.
|
|
// * It dials all the resolved TCP addresses in round-robin manner until
|
|
// connection is established. This may be useful if certain addresses
|
|
// are temporarily unreachable.
|
|
// * It returns ErrDialTimeout if connection cannot be established during
|
|
// DefaultDialTimeout seconds. Use DialTimeout for customizing dial timeout.
|
|
//
|
|
// This dialer is intended for custom code wrapping before passing
|
|
// to Client.Dial or HostClient.Dial.
|
|
//
|
|
// For instance, per-host counters and/or limits may be implemented
|
|
// by such wrappers.
|
|
//
|
|
// The addr passed to the function must contain port. Example addr values:
|
|
//
|
|
// * foobar.baz:443
|
|
// * foo.bar:80
|
|
// * aaa.com:8080
|
|
func (d *TCPDialer) Dial(addr string) (net.Conn, error) {
|
|
return d.dial(addr, false, DefaultDialTimeout)
|
|
}
|
|
|
|
// DialTimeout dials the given TCP addr using tcp4 using the given timeout.
|
|
//
|
|
// This function has the following additional features comparing to net.Dial:
|
|
//
|
|
// * It reduces load on DNS resolver by caching resolved TCP addressed
|
|
// for DefaultDNSCacheDuration.
|
|
// * It dials all the resolved TCP addresses in round-robin manner until
|
|
// connection is established. This may be useful if certain addresses
|
|
// are temporarily unreachable.
|
|
//
|
|
// This dialer is intended for custom code wrapping before passing
|
|
// to Client.Dial or HostClient.Dial.
|
|
//
|
|
// For instance, per-host counters and/or limits may be implemented
|
|
// by such wrappers.
|
|
//
|
|
// The addr passed to the function must contain port. Example addr values:
|
|
//
|
|
// * foobar.baz:443
|
|
// * foo.bar:80
|
|
// * aaa.com:8080
|
|
func (d *TCPDialer) DialTimeout(addr string, timeout time.Duration) (net.Conn, error) {
|
|
return d.dial(addr, false, timeout)
|
|
}
|
|
|
|
// DialDualStack dials the given TCP addr using both tcp4 and tcp6.
|
|
//
|
|
// This function has the following additional features comparing to net.Dial:
|
|
//
|
|
// * It reduces load on DNS resolver by caching resolved TCP addressed
|
|
// for DefaultDNSCacheDuration.
|
|
// * It dials all the resolved TCP addresses in round-robin manner until
|
|
// connection is established. This may be useful if certain addresses
|
|
// are temporarily unreachable.
|
|
// * It returns ErrDialTimeout if connection cannot be established during
|
|
// DefaultDialTimeout seconds. Use DialDualStackTimeout for custom dial
|
|
// timeout.
|
|
//
|
|
// This dialer is intended for custom code wrapping before passing
|
|
// to Client.Dial or HostClient.Dial.
|
|
//
|
|
// For instance, per-host counters and/or limits may be implemented
|
|
// by such wrappers.
|
|
//
|
|
// The addr passed to the function must contain port. Example addr values:
|
|
//
|
|
// * foobar.baz:443
|
|
// * foo.bar:80
|
|
// * aaa.com:8080
|
|
func (d *TCPDialer) DialDualStack(addr string) (net.Conn, error) {
|
|
return d.dial(addr, true, DefaultDialTimeout)
|
|
}
|
|
|
|
// DialDualStackTimeout dials the given TCP addr using both tcp4 and tcp6
|
|
// using the given timeout.
|
|
//
|
|
// This function has the following additional features comparing to net.Dial:
|
|
//
|
|
// * It reduces load on DNS resolver by caching resolved TCP addressed
|
|
// for DefaultDNSCacheDuration.
|
|
// * It dials all the resolved TCP addresses in round-robin manner until
|
|
// connection is established. This may be useful if certain addresses
|
|
// are temporarily unreachable.
|
|
//
|
|
// This dialer is intended for custom code wrapping before passing
|
|
// to Client.Dial or HostClient.Dial.
|
|
//
|
|
// For instance, per-host counters and/or limits may be implemented
|
|
// by such wrappers.
|
|
//
|
|
// The addr passed to the function must contain port. Example addr values:
|
|
//
|
|
// * foobar.baz:443
|
|
// * foo.bar:80
|
|
// * aaa.com:8080
|
|
func (d *TCPDialer) DialDualStackTimeout(addr string, timeout time.Duration) (net.Conn, error) {
|
|
return d.dial(addr, true, timeout)
|
|
}
|
|
|
|
func (d *TCPDialer) dial(addr string, dualStack bool, timeout time.Duration) (net.Conn, error) {
|
|
d.once.Do(func() {
|
|
if d.Concurrency > 0 {
|
|
d.concurrencyCh = make(chan struct{}, d.Concurrency)
|
|
}
|
|
d.tcpAddrsMap = make(map[string]*tcpAddrEntry)
|
|
go d.tcpAddrsClean()
|
|
})
|
|
|
|
addrs, idx, err := d.getTCPAddrs(addr, dualStack)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
network := "tcp4"
|
|
if dualStack {
|
|
network = "tcp"
|
|
}
|
|
|
|
var conn net.Conn
|
|
n := uint32(len(addrs))
|
|
deadline := time.Now().Add(timeout)
|
|
for n > 0 {
|
|
conn, err = tryDial(network, &addrs[idx%n], deadline, d.concurrencyCh)
|
|
if err == nil {
|
|
return conn, nil
|
|
}
|
|
if err == ErrDialTimeout {
|
|
return nil, err
|
|
}
|
|
idx++
|
|
n--
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
func tryDial(network string, addr *net.TCPAddr, deadline time.Time, concurrencyCh chan struct{}) (net.Conn, error) {
|
|
timeout := -time.Since(deadline)
|
|
if timeout <= 0 {
|
|
return nil, ErrDialTimeout
|
|
}
|
|
|
|
if concurrencyCh != nil {
|
|
select {
|
|
case concurrencyCh <- struct{}{}:
|
|
default:
|
|
tc := AcquireTimer(timeout)
|
|
isTimeout := false
|
|
select {
|
|
case concurrencyCh <- struct{}{}:
|
|
case <-tc.C:
|
|
isTimeout = true
|
|
}
|
|
ReleaseTimer(tc)
|
|
if isTimeout {
|
|
return nil, ErrDialTimeout
|
|
}
|
|
}
|
|
}
|
|
|
|
chv := dialResultChanPool.Get()
|
|
if chv == nil {
|
|
chv = make(chan dialResult, 1)
|
|
}
|
|
ch := chv.(chan dialResult)
|
|
go func() {
|
|
var dr dialResult
|
|
dr.conn, dr.err = net.DialTCP(network, nil, addr)
|
|
ch <- dr
|
|
if concurrencyCh != nil {
|
|
<-concurrencyCh
|
|
}
|
|
}()
|
|
|
|
var (
|
|
conn net.Conn
|
|
err error
|
|
)
|
|
|
|
tc := AcquireTimer(timeout)
|
|
select {
|
|
case dr := <-ch:
|
|
conn = dr.conn
|
|
err = dr.err
|
|
dialResultChanPool.Put(ch)
|
|
case <-tc.C:
|
|
err = ErrDialTimeout
|
|
}
|
|
ReleaseTimer(tc)
|
|
|
|
return conn, err
|
|
}
|
|
|
|
var dialResultChanPool sync.Pool
|
|
|
|
type dialResult struct {
|
|
conn net.Conn
|
|
err error
|
|
}
|
|
|
|
// ErrDialTimeout is returned when TCP dialing is timed out.
|
|
var ErrDialTimeout = errors.New("dialing to the given TCP address timed out")
|
|
|
|
// DefaultDialTimeout is timeout used by Dial and DialDualStack
|
|
// for establishing TCP connections.
|
|
const DefaultDialTimeout = 3 * time.Second
|
|
|
|
type tcpAddrEntry struct {
|
|
addrs []net.TCPAddr
|
|
addrsIdx uint32
|
|
|
|
resolveTime time.Time
|
|
pending bool
|
|
}
|
|
|
|
// DefaultDNSCacheDuration is the duration for caching resolved TCP addresses
|
|
// by Dial* functions.
|
|
const DefaultDNSCacheDuration = time.Minute
|
|
|
|
func (d *TCPDialer) tcpAddrsClean() {
|
|
expireDuration := 2 * DefaultDNSCacheDuration
|
|
for {
|
|
time.Sleep(time.Second)
|
|
t := time.Now()
|
|
|
|
d.tcpAddrsLock.Lock()
|
|
for k, e := range d.tcpAddrsMap {
|
|
if t.Sub(e.resolveTime) > expireDuration {
|
|
delete(d.tcpAddrsMap, k)
|
|
}
|
|
}
|
|
d.tcpAddrsLock.Unlock()
|
|
}
|
|
}
|
|
|
|
func (d *TCPDialer) getTCPAddrs(addr string, dualStack bool) ([]net.TCPAddr, uint32, error) {
|
|
d.tcpAddrsLock.Lock()
|
|
e := d.tcpAddrsMap[addr]
|
|
if e != nil && !e.pending && time.Since(e.resolveTime) > DefaultDNSCacheDuration {
|
|
e.pending = true
|
|
e = nil
|
|
}
|
|
d.tcpAddrsLock.Unlock()
|
|
|
|
if e == nil {
|
|
addrs, err := resolveTCPAddrs(addr, dualStack)
|
|
if err != nil {
|
|
d.tcpAddrsLock.Lock()
|
|
e = d.tcpAddrsMap[addr]
|
|
if e != nil && e.pending {
|
|
e.pending = false
|
|
}
|
|
d.tcpAddrsLock.Unlock()
|
|
return nil, 0, err
|
|
}
|
|
|
|
e = &tcpAddrEntry{
|
|
addrs: addrs,
|
|
resolveTime: time.Now(),
|
|
}
|
|
|
|
d.tcpAddrsLock.Lock()
|
|
d.tcpAddrsMap[addr] = e
|
|
d.tcpAddrsLock.Unlock()
|
|
}
|
|
|
|
idx := atomic.AddUint32(&e.addrsIdx, 1)
|
|
return e.addrs, idx, nil
|
|
}
|
|
|
|
func resolveTCPAddrs(addr string, dualStack bool) ([]net.TCPAddr, error) {
|
|
host, portS, err := net.SplitHostPort(addr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
port, err := strconv.Atoi(portS)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ips, err := net.LookupIP(host)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
n := len(ips)
|
|
addrs := make([]net.TCPAddr, 0, n)
|
|
for i := 0; i < n; i++ {
|
|
ip := ips[i]
|
|
if !dualStack && ip.To4() == nil {
|
|
continue
|
|
}
|
|
addrs = append(addrs, net.TCPAddr{
|
|
IP: ip,
|
|
Port: port,
|
|
})
|
|
}
|
|
if len(addrs) == 0 {
|
|
return nil, errNoDNSEntries
|
|
}
|
|
return addrs, nil
|
|
}
|
|
|
|
var errNoDNSEntries = errors.New("couldn't find DNS entries for the given domain. Try using DialDualStack")
|