2016-08-04 16:27:28 -04:00
|
|
|
// Copyright 2016 Jimmy Zelinskie
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
// Package trakr implements a BitTorrent Tracker that supports multiple
|
|
|
|
// protocols and configurable Hooks that execute before and after a Response
|
2016-08-09 16:01:14 -04:00
|
|
|
// has been delivered to a BitTorrent client.
|
2016-08-06 22:41:33 -04:00
|
|
|
package backend
|
2016-08-04 16:27:28 -04:00
|
|
|
|
2016-08-05 01:47:04 -04:00
|
|
|
import (
|
2016-08-09 16:01:14 -04:00
|
|
|
"log"
|
2016-08-05 01:47:04 -04:00
|
|
|
"time"
|
|
|
|
|
2016-08-09 16:01:14 -04:00
|
|
|
"golang.org/x/net/context"
|
2016-08-09 15:21:59 -04:00
|
|
|
|
|
|
|
"github.com/jzelinskie/trakr/bittorrent"
|
2016-08-09 15:01:36 -04:00
|
|
|
"github.com/jzelinskie/trakr/frontend"
|
2016-08-05 01:47:04 -04:00
|
|
|
)
|
|
|
|
|
2016-08-07 13:40:24 -04:00
|
|
|
type BackendConfig struct {
|
2016-08-09 16:01:14 -04:00
|
|
|
AnnounceInterval time.Duration `yaml:"announce_interval"`
|
2016-08-07 13:40:24 -04:00
|
|
|
}
|
2016-08-04 16:27:28 -04:00
|
|
|
|
2016-08-09 15:21:59 -04:00
|
|
|
var _ frontend.TrackerFuncs = &Backend{}
|
|
|
|
|
2016-08-09 16:01:14 -04:00
|
|
|
func New(config BackendConfig, peerStore PeerStore, announcePreHooks, announcePostHooks, scrapePreHooks, scrapePostHooks []Hook) (*Backend, error) {
|
|
|
|
toReturn := &Backend{
|
|
|
|
announceInterval: config.AnnounceInterval,
|
|
|
|
peerStore: peerStore,
|
|
|
|
announcePreHooks: announcePreHooks,
|
|
|
|
announcePostHooks: announcePostHooks,
|
|
|
|
scrapePreHooks: scrapePreHooks,
|
|
|
|
scrapePostHooks: scrapePostHooks,
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(toReturn.announcePreHooks) == 0 {
|
|
|
|
toReturn.announcePreHooks = []Hook{nopHook{}}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(toReturn.announcePostHooks) == 0 {
|
|
|
|
toReturn.announcePostHooks = []Hook{nopHook{}}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(toReturn.scrapePreHooks) == 0 {
|
|
|
|
toReturn.scrapePreHooks = []Hook{nopHook{}}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(toReturn.scrapePostHooks) == 0 {
|
|
|
|
toReturn.scrapePostHooks = []Hook{nopHook{}}
|
|
|
|
}
|
|
|
|
|
|
|
|
return toReturn, nil
|
2016-08-07 13:40:24 -04:00
|
|
|
}
|
|
|
|
|
2016-08-09 16:01:14 -04:00
|
|
|
// Backend is a protocol-agnostic backend of a BitTorrent tracker.
|
2016-08-07 13:40:24 -04:00
|
|
|
type Backend struct {
|
2016-08-09 16:01:14 -04:00
|
|
|
announceInterval time.Duration
|
|
|
|
peerStore PeerStore
|
|
|
|
announcePreHooks []Hook
|
|
|
|
announcePostHooks []Hook
|
|
|
|
scrapePreHooks []Hook
|
|
|
|
scrapePostHooks []Hook
|
2016-08-09 15:21:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// HandleAnnounce generates a response for an Announce.
|
|
|
|
func (b *Backend) HandleAnnounce(ctx context.Context, req *bittorrent.AnnounceRequest) (*bittorrent.AnnounceResponse, error) {
|
2016-08-09 16:01:14 -04:00
|
|
|
resp := &bittorrent.AnnounceResponse{
|
|
|
|
Interval: b.announceInterval,
|
|
|
|
}
|
|
|
|
for _, h := range b.announcePreHooks {
|
|
|
|
if err := h.HandleAnnounce(ctx, req, resp); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return resp, nil
|
2016-08-09 15:21:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// AfterAnnounce does something with the results of an Announce after it
|
|
|
|
// has been completed.
|
|
|
|
func (b *Backend) AfterAnnounce(ctx context.Context, req *bittorrent.AnnounceRequest, resp *bittorrent.AnnounceResponse) {
|
2016-08-09 16:01:14 -04:00
|
|
|
for _, h := range b.announcePostHooks {
|
|
|
|
if err := h.HandleAnnounce(ctx, req, resp); err != nil {
|
|
|
|
log.Println("trakr: post-announce hooks failed:", err.Error())
|
|
|
|
return
|
|
|
|
}
|
2016-08-09 15:21:59 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// HandleScrape generates a response for a Scrape.
|
|
|
|
func (b *Backend) HandleScrape(ctx context.Context, req *bittorrent.ScrapeRequest) (*bittorrent.ScrapeResponse, error) {
|
2016-08-09 16:01:14 -04:00
|
|
|
resp := &bittorrent.ScrapeResponse{
|
|
|
|
Files: make(map[bittorrent.InfoHash]bittorrent.Scrape),
|
|
|
|
}
|
|
|
|
for _, h := range b.scrapePreHooks {
|
|
|
|
if err := h.HandleScrape(ctx, req, resp); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return resp, nil
|
2016-08-09 15:21:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// AfterScrape does something with the results of a Scrape after it has been completed.
|
|
|
|
func (b *Backend) AfterScrape(ctx context.Context, req *bittorrent.ScrapeRequest, resp *bittorrent.ScrapeResponse) {
|
2016-08-09 16:01:14 -04:00
|
|
|
for _, h := range b.scrapePostHooks {
|
|
|
|
if err := h.HandleScrape(ctx, req, resp); err != nil {
|
|
|
|
log.Println("trakr: post-scrape hooks failed:", err.Error())
|
|
|
|
return
|
|
|
|
}
|
2016-08-09 15:21:59 -04:00
|
|
|
}
|
2016-08-04 16:27:28 -04:00
|
|
|
}
|