Skip to content

Commit

Permalink
cache resolve cache to lru to avoid leaking memory
Browse files Browse the repository at this point in the history
License: MIT
Signed-off-by: Jeromy <jeromyj@gmail.com>
  • Loading branch information
whyrusleeping committed Nov 1, 2015
1 parent 9577fab commit f9352b3
Show file tree
Hide file tree
Showing 6 changed files with 46 additions and 25 deletions.
4 changes: 2 additions & 2 deletions core/commands/ipns.go
Original file line number Diff line number Diff line change
Expand Up @@ -76,11 +76,11 @@ Resolve the value of another name:

if local {
offroute := offline.NewOfflineRouter(n.Repo.Datastore(), n.PrivateKey)
resolver = namesys.NewRoutingResolver(offroute, 0)
resolver = namesys.NewRoutingResolver(offroute, 0, 0)
}

if nocache {
resolver = namesys.NewNameSystem(n.Routing, n.Repo.Datastore(), 0)
resolver = namesys.NewNameSystem(n.Routing, n.Repo.Datastore(), 0, 0)
}

var name string
Expand Down
34 changes: 23 additions & 11 deletions core/core.go
Original file line number Diff line number Diff line change
Expand Up @@ -226,13 +226,13 @@ func (n *IpfsNode) startOnlineServicesWithHost(ctx context.Context, host p2phost
bitswapNetwork := bsnet.NewFromIpfsHost(n.PeerHost, n.Routing)
n.Exchange = bitswap.New(ctx, n.Identity, bitswapNetwork, n.Blockstore, alwaysSendToPeer)

cachelife, err := n.getCacheLifetime()
cachelife, size, err := n.getCacheParams()
if err != nil {
return err
}

// setup name system
n.Namesys = namesys.NewNameSystem(n.Routing, n.Repo.Datastore(), cachelife)
n.Namesys = namesys.NewNameSystem(n.Routing, n.Repo.Datastore(), cachelife, size)

// setup ipns republishing
err = n.setupIpnsRepublisher()
Expand All @@ -243,21 +243,33 @@ func (n *IpfsNode) startOnlineServicesWithHost(ctx context.Context, host p2phost
return nil
}

func (n *IpfsNode) getCacheLifetime() (time.Duration, error) {
// getCacheParams returns cache life and cache size
func (n *IpfsNode) getCacheParams() (time.Duration, int, error) {
cfg, err := n.Repo.Config()
if err != nil {
return 0, err
return 0, 0, err
}

ct := cfg.Ipns.ResolveCacheTime
var d time.Duration
if ct == "" {
return namesys.DefaultResolverCacheLife, nil
d = namesys.DefaultResolverCacheLife
} else {
parsed, err := time.ParseDuration(ct)
if err != nil {
return 0, 0, fmt.Errorf("error parsing cache life from Ipns.ResolveCacheTime: %s", err)
}
d = parsed
}
d, err := time.ParseDuration(ct)
if err != nil {
return 0, fmt.Errorf("error parsing cache life from Ipns.ResolveCacheTime: %s", err)

cs := cfg.Ipns.ResolveCacheSize
if cs == 0 {
cs = 128
}
if cs < 0 {
return 0, 0, fmt.Errorf("cannot specify negative resolve cache size")
}
return d, nil
return d, cs, nil
}

func (n *IpfsNode) setupIpnsRepublisher() error {
Expand Down Expand Up @@ -478,12 +490,12 @@ func (n *IpfsNode) SetupOfflineRouting() error {

n.Routing = offroute.NewOfflineRouter(n.Repo.Datastore(), n.PrivateKey)

cachelife, err := n.getCacheLifetime()
cachelife, size, err := n.getCacheParams()
if err != nil {
return err
}

n.Namesys = namesys.NewNameSystem(n.Routing, n.Repo.Datastore(), cachelife)
n.Namesys = namesys.NewNameSystem(n.Routing, n.Repo.Datastore(), cachelife, size)

return nil
}
Expand Down
4 changes: 2 additions & 2 deletions namesys/namesys.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,12 +26,12 @@ type mpns struct {
}

// NewNameSystem will construct the IPFS naming system based on Routing
func NewNameSystem(r routing.IpfsRouting, ds ds.Datastore, cachelife time.Duration) NameSystem {
func NewNameSystem(r routing.IpfsRouting, ds ds.Datastore, cachelife time.Duration, cachesize int) NameSystem {
return &mpns{
resolvers: map[string]resolver{
"dns": newDNSResolver(),
"proquint": new(ProquintResolver),
"dht": NewRoutingResolver(r, cachelife),
"dht": NewRoutingResolver(r, cachelife, cachesize),
},
publishers: map[string]Publisher{
"/ipns/": NewRoutingPublisher(r, ds),
Expand Down
23 changes: 17 additions & 6 deletions namesys/routing.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import (
"sync"
"time"

lru "github.com/hashicorp/golang-lru"
proto "github.com/ipfs/go-ipfs/Godeps/_workspace/src/github.com/gogo/protobuf/proto"
mh "github.com/ipfs/go-ipfs/Godeps/_workspace/src/github.com/jbenet/go-multihash"
"github.com/ipfs/go-ipfs/Godeps/_workspace/src/golang.org/x/net/context"
Expand All @@ -23,15 +24,21 @@ var log = logging.Logger("namesys")
type routingResolver struct {
routing routing.IpfsRouting

cache map[string]cacheEntry
cache *lru.Cache
cachelock sync.Mutex
cachelife time.Duration
}

func (r *routingResolver) cacheGet(name string) (path.Path, bool) {
r.cachelock.Lock()
entry, ok := r.cache[name]
ientry, ok := r.cache.Get(name)
r.cachelock.Unlock()

entry, ok := ientry.(cacheEntry)
if !ok {
return "", false
}

if ok && time.Now().Before(entry.eol) {
return entry.val, true
}
Expand All @@ -56,10 +63,10 @@ func (r *routingResolver) cacheSet(name string, val path.Path, rec *pb.IpnsEntry
}

r.cachelock.Lock()
r.cache[name] = cacheEntry{
r.cache.Add(name, cacheEntry{
val: val,
eol: cacheTil,
}
})
r.cachelock.Unlock()
}

Expand All @@ -70,14 +77,18 @@ type cacheEntry struct {

// NewRoutingResolver constructs a name resolver using the IPFS Routing system
// to implement SFS-like naming on top.
func NewRoutingResolver(route routing.IpfsRouting, cachelife time.Duration) *routingResolver {
func NewRoutingResolver(route routing.IpfsRouting, cachelife time.Duration, cachesize int) *routingResolver {
if route == nil {
panic("attempt to create resolver with nil routing system")
}
cache, err := lru.New(cachesize)
if err != nil {
log.Panicf("cannot use negative value for cachesize: %s", err)
}

return &routingResolver{
routing: route,
cache: make(map[string]cacheEntry),
cache: cache,
cachelife: cachelife,
}
}
Expand Down
1 change: 1 addition & 0 deletions repo/config/ipns.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,4 +5,5 @@ type Ipns struct {
RecordLifetime string

ResolveCacheTime string
ResolveCacheSize int
}
5 changes: 1 addition & 4 deletions routing/dht/routing.go
Original file line number Diff line number Diff line change
Expand Up @@ -84,10 +84,7 @@ func (dht *IpfsDHT) GetValue(ctx context.Context, key key.Key) ([]byte, error) {
ctx, cancel := context.WithTimeout(ctx, time.Minute)
defer cancel()

// retrieve a majority of the expected record count
majority := (KValue / 2) + 1

vals, err := dht.GetValues(ctx, key, majority)
vals, err := dht.GetValues(ctx, key, 16)
if err != nil {
return nil, err
}
Expand Down

0 comments on commit f9352b3

Please sign in to comment.