Skip to content
GitLab
Projects
Groups
Snippets
Help
Loading...
Help
What's new
10
Help
Support
Community forum
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
Open sidebar
dms3
go-dms3
Commits
26d40b2b
Commit
26d40b2b
authored
Oct 09, 2015
by
Juan Benet
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #1802 from ipfs/fix/too-many-fd
rate limit concurrent peer dials
parents
a0f119fa
c1c9a74a
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
60 additions
and
12 deletions
+60
-12
p2p/net/swarm/simul_test.go
p2p/net/swarm/simul_test.go
+3
-1
p2p/net/swarm/swarm.go
p2p/net/swarm/swarm.go
+12
-8
p2p/net/swarm/swarm_dial.go
p2p/net/swarm/swarm_dial.go
+45
-3
No files found.
p2p/net/swarm/simul_test.go
View file @
26d40b2b
package
swarm
import
(
"runtime"
"sync"
"testing"
"time"
...
...
@@ -49,7 +50,8 @@ func TestSimultOpenMany(t *testing.T) {
addrs
:=
20
rounds
:=
10
if
ci
.
IsRunning
()
{
if
ci
.
IsRunning
()
||
runtime
.
GOOS
==
"darwin"
{
// osx has a limit of 256 file descriptors
addrs
=
10
rounds
=
5
}
...
...
p2p/net/swarm/swarm.go
View file @
26d40b2b
...
...
@@ -64,6 +64,9 @@ type Swarm struct {
// filters for addresses that shouldnt be dialed
Filters
*
filter
.
Filters
// file descriptor rate limited
fdRateLimit
chan
struct
{}
proc
goprocess
.
Process
ctx
context
.
Context
bwc
metrics
.
Reporter
...
...
@@ -79,14 +82,15 @@ func NewSwarm(ctx context.Context, listenAddrs []ma.Multiaddr,
}
s
:=
&
Swarm
{
swarm
:
ps
.
NewSwarm
(
PSTransport
),
local
:
local
,
peers
:
peers
,
ctx
:
ctx
,
dialT
:
DialTimeout
,
notifs
:
make
(
map
[
inet
.
Notifiee
]
ps
.
Notifiee
),
bwc
:
bwc
,
Filters
:
filter
.
NewFilters
(),
swarm
:
ps
.
NewSwarm
(
PSTransport
),
local
:
local
,
peers
:
peers
,
ctx
:
ctx
,
dialT
:
DialTimeout
,
notifs
:
make
(
map
[
inet
.
Notifiee
]
ps
.
Notifiee
),
bwc
:
bwc
,
fdRateLimit
:
make
(
chan
struct
{},
concurrentFdDials
),
Filters
:
filter
.
NewFilters
(),
}
// configure Swarm
...
...
p2p/net/swarm/swarm_dial.go
View file @
26d40b2b
...
...
@@ -44,6 +44,9 @@ var (
// add loop back in Dial(.)
const
dialAttempts
=
1
// number of concurrent outbound dials over transports that consume file descriptors
const
concurrentFdDials
=
160
// DialTimeout is the amount of time each dial attempt has. We can think about making
// this larger down the road, or putting more granular timeouts (i.e. within each
// subcomponent of Dial)
...
...
@@ -115,6 +118,7 @@ func (ds *dialsync) Unlock(dst peer.ID) {
if
!
found
{
panic
(
"called dialDone with no ongoing dials to peer: "
+
dst
.
Pretty
())
}
delete
(
ds
.
ongoing
,
dst
)
// remove ongoing dial
close
(
wait
)
// release everyone else
ds
.
lock
.
Unlock
()
...
...
@@ -398,7 +402,7 @@ func (s *Swarm) dialAddrs(ctx context.Context, d *conn.Dialer, p peer.ID, remote
// to end early.
go
func
()
{
// rate limiting just in case. at most 10 addrs at once.
limiter
:=
ratelimit
.
NewRateLimiter
(
process
.
Background
(),
10
)
limiter
:=
ratelimit
.
NewRateLimiter
(
process
.
Background
(),
8
)
limiter
.
Go
(
func
(
worker
process
.
Process
)
{
// permute addrs so we try different sets first each time.
for
_
,
i
:=
range
rand
.
Perm
(
len
(
remoteAddrs
))
{
...
...
@@ -411,9 +415,27 @@ func (s *Swarm) dialAddrs(ctx context.Context, d *conn.Dialer, p peer.ID, remote
}
workerAddr
:=
remoteAddrs
[
i
]
// shadow variable to avoid race
limiter
.
LimitedGo
(
func
(
worker
process
.
Process
)
{
dialSingleAddr
(
workerAddr
)
// we have to do the waiting concurrently because there are addrs
// that SHOULD NOT be rate limited (utp), nor blocked by other
// rate limited addrs (tcp).
//
// (and we need to call `limiter.Go`, instead of `go` as required
// by goproc/limiter semantics. note: limiter.Go is not LimitedGo.)
limiter
.
Go
(
func
(
p
process
.
Process
)
{
// returns whatever ratelimiting is acceptable for workerAddr.
// may not rate limit at all.
rl
:=
s
.
addrDialRateLimit
(
workerAddr
)
rl
<-
struct
{}{}
limiter
.
LimitedGo
(
func
(
worker
process
.
Process
)
{
dialSingleAddr
(
workerAddr
)
})
<-
rl
})
}
})
...
...
@@ -491,3 +513,23 @@ func dialConnSetup(ctx context.Context, s *Swarm, connC conn.Conn) (*Conn, error
return
swarmC
,
err
}
// addrDialRateLimit returns a ratelimiting channel for dialing transport
// addrs like a. for example, tcp is fd-ratelimited. utp is not ratelimited.
func
(
s
*
Swarm
)
addrDialRateLimit
(
a
ma
.
Multiaddr
)
chan
struct
{}
{
if
isFDCostlyTransport
(
a
)
{
return
s
.
fdRateLimit
}
// do not rate limit it at all
return
make
(
chan
struct
{},
1
)
}
func
isFDCostlyTransport
(
a
ma
.
Multiaddr
)
bool
{
return
isTCPMultiaddr
(
a
)
}
func
isTCPMultiaddr
(
a
ma
.
Multiaddr
)
bool
{
p
:=
a
.
Protocols
()
return
len
(
p
)
==
2
&&
(
p
[
0
]
.
Name
==
"ip4"
||
p
[
0
]
.
Name
==
"ip6"
)
&&
p
[
1
]
.
Name
==
"tcp"
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
.
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment