2015-01-01 18:02:25 +01:00
|
|
|
// Copyright 2015 The Chihaya Authors. All rights reserved.
|
2014-07-02 03:40:29 +02: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 10:40:17 +02:00
|
|
|
// Package http implements an http-serving BitTorrent tracker.
|
2014-07-02 03:40:29 +02:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
2014-07-22 07:38:07 +02:00
|
|
|
"net"
|
2014-07-02 03:40:29 +02:00
|
|
|
"net/http"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/golang/glog"
|
|
|
|
"github.com/julienschmidt/httprouter"
|
|
|
|
"github.com/stretchr/graceful"
|
|
|
|
|
|
|
|
"github.com/chihaya/chihaya/config"
|
2014-07-22 07:38:07 +02:00
|
|
|
"github.com/chihaya/chihaya/stats"
|
2014-07-17 06:09:56 +02:00
|
|
|
"github.com/chihaya/chihaya/tracker"
|
2014-07-02 03:40:29 +02:00
|
|
|
)
|
|
|
|
|
2014-07-17 06:09:56 +02:00
|
|
|
type ResponseHandler func(http.ResponseWriter, *http.Request, httprouter.Params) (int, error)
|
2014-07-02 03:40:29 +02:00
|
|
|
|
2014-07-17 06:09:56 +02:00
|
|
|
type Server struct {
|
|
|
|
config *config.Config
|
|
|
|
tracker *tracker.Tracker
|
2014-07-02 03:40:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func makeHandler(handler ResponseHandler) httprouter.Handle {
|
|
|
|
return func(w http.ResponseWriter, r *http.Request, p httprouter.Params) {
|
2014-07-25 09:39:02 +02:00
|
|
|
var msg string
|
|
|
|
|
2014-07-02 03:40:29 +02:00
|
|
|
start := time.Now()
|
2014-07-16 17:52:59 +02:00
|
|
|
httpCode, err := handler(w, r, p)
|
2014-07-23 07:20:48 +02:00
|
|
|
duration := time.Since(start)
|
2014-07-22 07:40:52 +02:00
|
|
|
|
2014-07-23 07:24:31 +02:00
|
|
|
if err != nil {
|
2014-07-25 09:39:02 +02:00
|
|
|
msg = err.Error()
|
|
|
|
} else if httpCode != http.StatusOK {
|
|
|
|
msg = http.StatusText(httpCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(msg) > 0 {
|
|
|
|
http.Error(w, msg, httpCode)
|
2014-07-22 07:40:52 +02:00
|
|
|
stats.RecordEvent(stats.ErroredRequest)
|
2014-07-25 09:39:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(msg) > 0 || glog.V(2) {
|
2014-07-25 09:43:06 +02:00
|
|
|
reqString := r.URL.Path + " " + r.RemoteAddr
|
2014-07-23 08:25:15 +02:00
|
|
|
if glog.V(3) {
|
2014-07-25 09:39:02 +02:00
|
|
|
reqString = r.URL.RequestURI() + " " + r.RemoteAddr
|
2014-07-23 08:25:15 +02:00
|
|
|
}
|
|
|
|
|
2014-07-25 09:39:02 +02:00
|
|
|
if len(msg) > 0 {
|
2014-07-25 09:43:06 +02:00
|
|
|
glog.Errorf("[%d - %9s] %s (%s)", httpCode, duration, reqString, msg)
|
2014-07-25 09:39:02 +02:00
|
|
|
} else {
|
2014-07-25 09:43:06 +02:00
|
|
|
glog.Infof("[%d - %9s] %s", httpCode, duration, reqString)
|
2014-07-25 09:39:02 +02:00
|
|
|
}
|
2014-07-16 17:52:59 +02:00
|
|
|
}
|
2014-07-23 07:31:22 +02:00
|
|
|
|
|
|
|
stats.RecordEvent(stats.HandledRequest)
|
|
|
|
stats.RecordTiming(stats.ResponseTime, duration)
|
2014-07-02 03:40:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-17 07:00:26 +02:00
|
|
|
func newRouter(s *Server) *httprouter.Router {
|
2014-07-02 03:40:29 +02:00
|
|
|
r := httprouter.New()
|
2014-07-09 06:53:57 +02:00
|
|
|
|
2014-07-25 22:58:26 +02:00
|
|
|
if s.config.PrivateEnabled {
|
2014-07-17 06:09:56 +02:00
|
|
|
r.GET("/users/:passkey/announce", makeHandler(s.serveAnnounce))
|
|
|
|
r.GET("/users/:passkey/scrape", makeHandler(s.serveScrape))
|
2014-07-09 06:53:57 +02:00
|
|
|
|
2014-07-17 06:09:56 +02:00
|
|
|
r.PUT("/users/:passkey", makeHandler(s.putUser))
|
|
|
|
r.DELETE("/users/:passkey", makeHandler(s.delUser))
|
2014-07-02 03:40:29 +02:00
|
|
|
} else {
|
2014-07-17 06:09:56 +02:00
|
|
|
r.GET("/announce", makeHandler(s.serveAnnounce))
|
|
|
|
r.GET("/scrape", makeHandler(s.serveScrape))
|
2014-07-02 03:40:29 +02:00
|
|
|
}
|
|
|
|
|
2014-07-25 22:58:26 +02:00
|
|
|
if s.config.ClientWhitelistEnabled {
|
2014-07-17 06:09:56 +02:00
|
|
|
r.PUT("/clients/:clientID", makeHandler(s.putClient))
|
|
|
|
r.DELETE("/clients/:clientID", makeHandler(s.delClient))
|
2014-07-09 06:53:57 +02:00
|
|
|
}
|
|
|
|
|
2014-07-17 06:09:56 +02: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 07:19:09 +02:00
|
|
|
r.GET("/stats", makeHandler(s.stats))
|
2014-07-09 07:34:34 +02:00
|
|
|
|
2014-07-02 03:40:29 +02:00
|
|
|
return r
|
|
|
|
}
|
|
|
|
|
2014-07-22 07:38:07 +02:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-17 06:09:56 +02:00
|
|
|
func Serve(cfg *config.Config, tkr *tracker.Tracker) {
|
|
|
|
srv := &Server{
|
|
|
|
config: cfg,
|
|
|
|
tracker: tkr,
|
2014-07-02 03:40:29 +02:00
|
|
|
}
|
|
|
|
|
2014-11-02 01:12:40 +01:00
|
|
|
glog.V(0).Info("Starting on ", cfg.ListenAddr)
|
2014-09-25 19:58:07 +02:00
|
|
|
if cfg.HttpListenLimit != 0 {
|
|
|
|
glog.V(0).Info("Limiting connections to ", cfg.HttpListenLimit)
|
|
|
|
}
|
2014-07-22 07:38:07 +02:00
|
|
|
|
2014-09-03 07:15:21 +02:00
|
|
|
grace := &graceful.Server{
|
2014-09-25 21:12:25 +02:00
|
|
|
Timeout: cfg.RequestTimeout.Duration,
|
|
|
|
ConnState: srv.connState,
|
2014-09-25 19:58:07 +02:00
|
|
|
ListenLimit: cfg.HttpListenLimit,
|
2014-07-22 07:38:07 +02:00
|
|
|
Server: &http.Server{
|
2014-11-02 01:12:40 +01:00
|
|
|
Addr: cfg.ListenAddr,
|
2014-09-03 16:18:26 +02:00
|
|
|
Handler: newRouter(srv),
|
|
|
|
ReadTimeout: cfg.HttpReadTimeout.Duration,
|
2014-09-03 07:15:21 +02:00
|
|
|
WriteTimeout: cfg.HttpWriteTimeout.Duration,
|
2014-07-22 07:38:07 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2014-08-27 00:03:44 +02:00
|
|
|
grace.SetKeepAlivesEnabled(false)
|
|
|
|
|
2014-07-25 09:43:06 +02:00
|
|
|
if err := grace.ListenAndServe(); err != nil {
|
2014-08-05 21:00:52 +02:00
|
|
|
if opErr, ok := err.(*net.OpError); !ok || (ok && opErr.Op != "accept") {
|
|
|
|
glog.Errorf("Failed to gracefully run HTTP server: %s", err.Error())
|
|
|
|
}
|
2014-07-25 09:43:06 +02:00
|
|
|
}
|
2014-07-22 07:38:07 +02:00
|
|
|
|
2014-07-25 09:43:06 +02:00
|
|
|
if err := srv.tracker.Close(); err != nil {
|
2014-07-21 09:54:22 +02:00
|
|
|
glog.Errorf("Failed to shutdown tracker cleanly: %s", err.Error())
|
|
|
|
}
|
2014-07-02 03:40:29 +02:00
|
|
|
}
|