tracker/middleware/middleware.go

120 lines
3.3 KiB
Go
Raw Normal View History

2016-08-10 02:08:15 +02:00
// Package middleware implements the TrackerLogic interface by executing
// a series of middleware hooks.
package middleware
import (
2016-08-17 04:32:15 +02:00
"context"
2016-08-10 02:08:15 +02:00
"time"
log "github.com/Sirupsen/logrus"
2016-08-17 03:42:08 +02:00
"github.com/chihaya/chihaya/bittorrent"
"github.com/chihaya/chihaya/frontend"
2016-09-24 19:38:05 +02:00
"github.com/chihaya/chihaya/stopper"
2016-08-17 03:42:08 +02:00
"github.com/chihaya/chihaya/storage"
2016-08-10 02:08:15 +02:00
)
// Config holds the configuration common across all middleware.
2016-08-10 02:08:15 +02:00
type Config struct {
AnnounceInterval time.Duration `yaml:"announce_interval"`
}
var _ frontend.TrackerLogic = &Logic{}
// NewLogic creates a new instance of a TrackerLogic that executes the provided
// middleware hooks.
func NewLogic(cfg Config, peerStore storage.PeerStore, preHooks, postHooks []Hook) *Logic {
2016-08-10 02:08:15 +02:00
l := &Logic{
announceInterval: cfg.AnnounceInterval,
peerStore: peerStore,
2016-09-08 16:46:16 +02:00
preHooks: append(preHooks, &responseHook{store: peerStore}),
2016-09-08 02:25:10 +02:00
postHooks: append(postHooks, &swarmInteractionHook{store: peerStore}),
2016-08-10 02:08:15 +02:00
}
return l
}
// Logic is an implementation of the TrackerLogic that functions by
// executing a series of middleware hooks.
type Logic struct {
announceInterval time.Duration
peerStore storage.PeerStore
preHooks []Hook
postHooks []Hook
2016-08-10 02:08:15 +02:00
}
// HandleAnnounce generates a response for an Announce.
func (l *Logic) HandleAnnounce(ctx context.Context, req *bittorrent.AnnounceRequest) (resp *bittorrent.AnnounceResponse, err error) {
resp = &bittorrent.AnnounceResponse{
2016-09-08 16:46:16 +02:00
Interval: l.announceInterval,
MinInterval: l.announceInterval,
Compact: req.Compact,
2016-08-10 02:08:15 +02:00
}
for _, h := range l.preHooks {
if ctx, err = h.HandleAnnounce(ctx, req, resp); err != nil {
2016-08-10 02:08:15 +02:00
return nil, err
}
}
return resp, nil
}
// AfterAnnounce does something with the results of an Announce after it has
// been completed.
func (l *Logic) AfterAnnounce(ctx context.Context, req *bittorrent.AnnounceRequest, resp *bittorrent.AnnounceResponse) {
var err error
for _, h := range l.postHooks {
if ctx, err = h.HandleAnnounce(ctx, req, resp); err != nil {
log.Errorln("chihaya: post-announce hooks failed:", err.Error())
2016-08-10 02:08:15 +02:00
return
}
}
}
// HandleScrape generates a response for a Scrape.
func (l *Logic) HandleScrape(ctx context.Context, req *bittorrent.ScrapeRequest) (resp *bittorrent.ScrapeResponse, err error) {
resp = &bittorrent.ScrapeResponse{
2016-08-10 02:08:15 +02:00
Files: make(map[bittorrent.InfoHash]bittorrent.Scrape),
}
for _, h := range l.preHooks {
if ctx, err = h.HandleScrape(ctx, req, resp); err != nil {
2016-08-10 02:08:15 +02:00
return nil, err
}
}
return resp, nil
}
// AfterScrape does something with the results of a Scrape after it has been
// completed.
func (l *Logic) AfterScrape(ctx context.Context, req *bittorrent.ScrapeRequest, resp *bittorrent.ScrapeResponse) {
var err error
for _, h := range l.postHooks {
if ctx, err = h.HandleScrape(ctx, req, resp); err != nil {
log.Errorln("chihaya: post-scrape hooks failed:", err.Error())
2016-08-10 02:08:15 +02:00
return
}
}
}
2016-09-24 19:38:05 +02:00
// Stop stops the Logic.
//
// This stops any hooks that implement stopper.Stopper.
func (l *Logic) Stop() []error {
stopGroup := stopper.NewStopGroup()
for _, hook := range l.preHooks {
stoppable, ok := hook.(stopper.Stopper)
if ok {
stopGroup.Add(stoppable)
}
}
for _, hook := range l.postHooks {
stoppable, ok := hook.(stopper.Stopper)
if ok {
stopGroup.Add(stoppable)
}
}
return stopGroup.Stop()
}