From f9352b3daf75185456bfe3c377802080fc9c855d Mon Sep 17 00:00:00 2001 From: Jeromy Date: Sat, 31 Oct 2015 19:09:39 -0700 Subject: [PATCH] cache resolve cache to lru to avoid leaking memory License: MIT Signed-off-by: Jeromy --- core/commands/ipns.go | 4 ++-- core/core.go | 34 +++++++++++++++++++++++----------- namesys/namesys.go | 4 ++-- namesys/routing.go | 23 +++++++++++++++++------ repo/config/ipns.go | 1 + routing/dht/routing.go | 5 +---- 6 files changed, 46 insertions(+), 25 deletions(-) diff --git a/core/commands/ipns.go b/core/commands/ipns.go index 59128691402f..5123945619ca 100644 --- a/core/commands/ipns.go +++ b/core/commands/ipns.go @@ -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 diff --git a/core/core.go b/core/core.go index 4f46b534564d..d2351575f2cf 100644 --- a/core/core.go +++ b/core/core.go @@ -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() @@ -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 { @@ -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 } diff --git a/namesys/namesys.go b/namesys/namesys.go index 17815f1f3600..d2299f402b0a 100644 --- a/namesys/namesys.go +++ b/namesys/namesys.go @@ -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), diff --git a/namesys/routing.go b/namesys/routing.go index aebac546c12b..7df127ca7535 100644 --- a/namesys/routing.go +++ b/namesys/routing.go @@ -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" @@ -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 } @@ -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() } @@ -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, } } diff --git a/repo/config/ipns.go b/repo/config/ipns.go index 2dcbbea16233..7862131ed238 100644 --- a/repo/config/ipns.go +++ b/repo/config/ipns.go @@ -5,4 +5,5 @@ type Ipns struct { RecordLifetime string ResolveCacheTime string + ResolveCacheSize int } diff --git a/routing/dht/routing.go b/routing/dht/routing.go index 5f0206199302..df93396ce379 100644 --- a/routing/dht/routing.go +++ b/routing/dht/routing.go @@ -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 }