package dht import ( "sync" peer "github.com/jbenet/go-ipfs/peer" queue "github.com/jbenet/go-ipfs/peer/queue" kb "github.com/jbenet/go-ipfs/routing/kbucket" u "github.com/jbenet/go-ipfs/util" todoctr "github.com/jbenet/go-ipfs/util/todocounter" context "github.com/jbenet/go-ipfs/Godeps/_workspace/src/code.google.com/p/go.net/context" ) var maxQueryConcurrency = AlphaValue type dhtDialer interface { // DialPeer attempts to establish a connection to a given peer DialPeer(peer.Peer) error } type dhtQuery struct { // the key we're querying for key u.Key // dialer used to ensure we're connected to peers dialer dhtDialer // the function to execute per peer qfunc queryFunc // the concurrency parameter concurrency int } type dhtQueryResult struct { value []byte // GetValue peer peer.Peer // FindPeer providerPeers []peer.Peer // GetProviders closerPeers []peer.Peer // * success bool } // constructs query func newQuery(k u.Key, d dhtDialer, f queryFunc) *dhtQuery { return &dhtQuery{ key: k, dialer: d, qfunc: f, concurrency: maxQueryConcurrency, } } // QueryFunc is a function that runs a particular query with a given peer. // It returns either: // - the value // - a list of peers potentially better able to serve the query // - an error type queryFunc func(context.Context, peer.Peer) (*dhtQueryResult, error) // Run runs the query at hand. pass in a list of peers to use first. func (q *dhtQuery) Run(ctx context.Context, peers []peer.Peer) (*dhtQueryResult, error) { runner := newQueryRunner(ctx, q) return runner.Run(peers) } type dhtQueryRunner struct { // the query to run query *dhtQuery // peersToQuery is a list of peers remaining to query peersToQuery *queue.ChanQueue // peersSeen are all the peers queried. used to prevent querying same peer 2x peersSeen peer.Map // rateLimit is a channel used to rate limit our processing (semaphore) rateLimit chan struct{} // peersRemaining is a counter of peers remaining (toQuery + processing) peersRemaining todoctr.Counter // context ctx context.Context cancel context.CancelFunc // result result *dhtQueryResult // result errors errs []error // lock for concurrent access to fields sync.RWMutex } func newQueryRunner(ctx context.Context, q *dhtQuery) *dhtQueryRunner { ctx, cancel := context.WithCancel(ctx) return &dhtQueryRunner{ ctx: ctx, cancel: cancel, query: q, peersToQuery: queue.NewChanQueue(ctx, queue.NewXORDistancePQ(q.key)), peersRemaining: todoctr.NewSyncCounter(), peersSeen: peer.Map{}, rateLimit: make(chan struct{}, q.concurrency), } } func (r *dhtQueryRunner) Run(peers []peer.Peer) (*dhtQueryResult, error) { log.Debug("Run query with %d peers.", len(peers)) if len(peers) == 0 { log.Warning("Running query with no peers!") return nil, nil } // setup concurrency rate limiting for i := 0; i < r.query.concurrency; i++ { r.rateLimit <- struct{}{} } // add all the peers we got first. for _, p := range peers { r.addPeerToQuery(p, nil) // don't have access to self here... } // go do this thing. go r.spawnWorkers() // so workers are working. // wait until they're done. err := u.ErrNotFound select { case <-r.peersRemaining.Done(): r.cancel() // ran all and nothing. cancel all outstanding workers. r.RLock() defer r.RUnlock() if len(r.errs) > 0 { err = r.errs[0] } case <-r.ctx.Done(): r.RLock() defer r.RUnlock() err = r.ctx.Err() } if r.result != nil && r.result.success { return r.result, nil } return nil, err } func (r *dhtQueryRunner) addPeerToQuery(next peer.Peer, benchmark peer.Peer) { if next == nil { // wtf why are peers nil?!? log.Error("Query getting nil peers!!!\n") return } // if new peer further away than whom we got it from, bother (loops) if benchmark != nil && kb.Closer(benchmark.ID(), next.ID(), r.query.key) { return } // if already seen, no need. r.Lock() _, found := r.peersSeen[next.Key()] if found { r.Unlock() return } r.peersSeen[next.Key()] = next r.Unlock() log.Debug("adding peer to query: %v\n", next) // do this after unlocking to prevent possible deadlocks. r.peersRemaining.Increment(1) select { case r.peersToQuery.EnqChan <- next: case <-r.ctx.Done(): } } func (r *dhtQueryRunner) spawnWorkers() { for { select { case <-r.peersRemaining.Done(): return case <-r.ctx.Done(): return case p, more := <-r.peersToQuery.DeqChan: if !more { return // channel closed. } log.Debug("spawning worker for: %v\n", p) go r.queryPeer(p) } } } func (r *dhtQueryRunner) queryPeer(p peer.Peer) { log.Debug("spawned worker for: %v\n", p) // make sure we rate limit concurrency. select { case <-r.rateLimit: case <-r.ctx.Done(): r.peersRemaining.Decrement(1) return } // ok let's do this! log.Debug("running worker for: %v", p) // make sure we do this when we exit defer func() { // signal we're done proccessing peer p log.Debug("completing worker for: %v", p) r.peersRemaining.Decrement(1) r.rateLimit <- struct{}{} }() // make sure we're connected to the peer. err := r.query.dialer.DialPeer(p) if err != nil { log.Debug("ERROR worker for: %v -- err connecting: %v", p, err) r.Lock() r.errs = append(r.errs, err) r.Unlock() return } // finally, run the query against this peer res, err := r.query.qfunc(r.ctx, p) if err != nil { log.Debug("ERROR worker for: %v %v", p, err) r.Lock() r.errs = append(r.errs, err) r.Unlock() } else if res.success { log.Debug("SUCCESS worker for: %v", p, res) r.Lock() r.result = res r.Unlock() r.cancel() // signal to everyone that we're done. } else if res.closerPeers != nil { log.Debug("PEERS CLOSER -- worker for: %v\n", p) for _, next := range res.closerPeers { r.addPeerToQuery(next, p) } } }