package metadata import ( "crypto/rand" "time" "github.com/boramalper/magnetico/pkg/util" "go.uber.org/zap" "github.com/boramalper/magnetico/cmd/magneticod/dht/mainline" "github.com/boramalper/magnetico/pkg/persistence" ) type Metadata struct { InfoHash []byte // Name should be thought of "Title" of the torrent. For single-file torrents, it is the name // of the file, and for multi-file torrents, it is the name of the root directory. Name string TotalSize uint64 DiscoveredOn int64 // Files must be populated for both single-file and multi-file torrents! Files []persistence.File } type Sink struct { clientID []byte deadline time.Duration drain chan Metadata incomingInfoHashes map[[20]byte]struct{} terminated bool termination chan interface{} } func NewSink(deadline time.Duration) *Sink { ms := new(Sink) ms.clientID = make([]byte, 20) _, err := rand.Read(ms.clientID) if err != nil { zap.L().Panic("sinkMetadata couldn't read 20 random bytes for client ID!", zap.Error(err)) } ms.deadline = deadline ms.drain = make(chan Metadata) ms.incomingInfoHashes = make(map[[20]byte]struct{}) ms.termination = make(chan interface{}) return ms } func (ms *Sink) Sink(res mainline.TrawlingResult) { if ms.terminated { zap.L().Panic("Trying to Sink() an already closed Sink!") } if _, exists := ms.incomingInfoHashes[res.InfoHash]; exists { return } // BEWARE! // Although not crucial, the assumption is that Sink.Sink() will be called by only one // goroutine (i.e. it's not thread-safe), lest there might be a race condition between where we // check whether res.infoHash exists in the ms.incomingInfoHashes, and where we add the infoHash // to the incomingInfoHashes at the end of this function. zap.L().Info("Sunk!", zap.Int("leeches", len(ms.incomingInfoHashes)), util.HexField("infoHash", res.InfoHash[:])) leech := NewLeech(res.InfoHash, res.PeerAddr, LeechEventHandlers{ OnSuccess: ms.flush, OnError: ms.onLeechError, }) go leech.Do(time.Now().Add(ms.deadline)) ms.incomingInfoHashes[res.InfoHash] = struct{}{} } func (ms *Sink) Drain() <-chan Metadata { if ms.terminated { zap.L().Panic("Trying to Drain() an already closed Sink!") } return ms.drain } func (ms *Sink) Terminate() { ms.terminated = true close(ms.termination) close(ms.drain) } func (ms *Sink) flush(result Metadata) { if !ms.terminated { ms.drain <- result // Delete the infoHash from ms.incomingInfoHashes ONLY AFTER once we've flushed the // metadata! var infoHash [20]byte copy(infoHash[:], result.InfoHash) delete(ms.incomingInfoHashes, infoHash) } } func (ms *Sink) onLeechError(infoHash [20]byte, err error) { zap.L().Debug("leech error", util.HexField("infoHash", infoHash[:]), zap.Error(err)) delete(ms.incomingInfoHashes, infoHash) }