tracker/http/http.go

138 lines
3.2 KiB
Go
Raw Normal View History

// Copyright 2014 The Chihaya Authors. All rights reserved.
2014-07-01 21:40:29 -04:00
// Use of this source code is governed by the BSD 2-Clause license,
// which can be found in the LICENSE file.
2014-07-16 04:40:17 -04:00
// Package http implements an http-serving BitTorrent tracker.
2014-07-01 21:40:29 -04:00
package http
import (
"net"
2014-07-01 21:40:29 -04:00
"net/http"
"time"
"github.com/golang/glog"
"github.com/julienschmidt/httprouter"
"github.com/stretchr/graceful"
"github.com/chihaya/chihaya/config"
"github.com/chihaya/chihaya/stats"
"github.com/chihaya/chihaya/tracker"
2014-07-01 21:40:29 -04:00
)
type ResponseHandler func(http.ResponseWriter, *http.Request, httprouter.Params) (int, error)
2014-07-01 21:40:29 -04:00
type Server struct {
config *config.Config
tracker *tracker.Tracker
2014-07-01 21:40:29 -04:00
}
func makeHandler(handler ResponseHandler) httprouter.Handle {
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
start := time.Now()
httpCode, err := handler(w, r, p)
2014-07-23 01:20:48 -04:00
duration := time.Since(start)
if err != nil {
stats.RecordEvent(stats.ErroredRequest)
http.Error(w, err.Error(), httpCode)
2014-07-23 01:31:22 -04:00
glog.Errorf(
"Failed (%v:%s) %s with %s in %s",
httpCode,
http.StatusText(httpCode),
r.URL.Path,
err.Error(),
duration,
)
2014-07-23 01:20:48 -04:00
} else if glog.V(2) {
2014-07-23 02:25:15 -04:00
reqString := r.URL.Path
if glog.V(3) {
reqString = r.URL.RequestURI() + " for " + r.RemoteAddr
}
glog.Infof(
2014-07-23 01:20:48 -04:00
"Completed (%v:%s) %s in %v",
httpCode,
http.StatusText(httpCode),
2014-07-23 02:25:15 -04:00
reqString,
2014-07-22 14:57:36 -04:00
duration,
)
}
2014-07-23 01:31:22 -04:00
stats.RecordEvent(stats.HandledRequest)
stats.RecordTiming(stats.ResponseTime, duration)
2014-07-01 21:40:29 -04:00
}
}
func newRouter(s *Server) *httprouter.Router {
2014-07-01 21:40:29 -04:00
r := httprouter.New()
2014-07-09 00:53:57 -04:00
if s.config.Private {
r.GET("/users/:passkey/announce", makeHandler(s.serveAnnounce))
r.GET("/users/:passkey/scrape", makeHandler(s.serveScrape))
2014-07-09 00:53:57 -04:00
r.PUT("/users/:passkey", makeHandler(s.putUser))
r.DELETE("/users/:passkey", makeHandler(s.delUser))
2014-07-01 21:40:29 -04:00
} else {
r.GET("/announce", makeHandler(s.serveAnnounce))
r.GET("/scrape", makeHandler(s.serveScrape))
2014-07-01 21:40:29 -04:00
}
if s.config.Whitelist {
r.PUT("/clients/:clientID", makeHandler(s.putClient))
r.DELETE("/clients/:clientID", makeHandler(s.delClient))
2014-07-09 00:53:57 -04:00
}
r.GET("/torrents/:infohash", makeHandler(s.getTorrent))
r.PUT("/torrents/:infohash", makeHandler(s.putTorrent))
r.DELETE("/torrents/:infohash", makeHandler(s.delTorrent))
r.GET("/check", makeHandler(s.check))
2014-07-22 01:19:09 -04:00
r.GET("/stats", makeHandler(s.stats))
2014-07-09 01:34:34 -04:00
2014-07-01 21:40:29 -04:00
return r
}
func (s *Server) connState(conn net.Conn, state http.ConnState) {
switch state {
case http.StateNew:
stats.RecordEvent(stats.AcceptedConnection)
case http.StateClosed:
stats.RecordEvent(stats.ClosedConnection)
case http.StateHijacked:
panic("connection impossibly hijacked")
case http.StateActive: // Ignore.
case http.StateIdle: // Ignore.
default:
glog.Errorf("Connection transitioned to unknown state %s (%d)", state, state)
}
}
func Serve(cfg *config.Config, tkr *tracker.Tracker) {
srv := &Server{
config: cfg,
tracker: tkr,
2014-07-01 21:40:29 -04:00
}
glog.V(0).Info("Starting on ", cfg.Addr)
grace := graceful.Server{
Timeout: cfg.RequestTimeout.Duration,
ConnState: srv.connState,
Server: &http.Server{
Addr: cfg.Addr,
Handler: newRouter(srv),
},
}
grace.ListenAndServe()
2014-07-21 03:54:22 -04:00
err := srv.tracker.Close()
if err != nil {
glog.Errorf("Failed to shutdown tracker cleanly: %s", err.Error())
}
2014-07-01 21:40:29 -04:00
}