// Copyright 2015 The Chihaya Authors. All rights reserved.
// Use of this source code is governed by the BSD 2-Clause license,
// which can be found in the LICENSE file.

// Package udp implements a UDP BitTorrent tracker per BEP 15 and BEP 41.
// IPv6 is currently unsupported as there is no widely-implemented standard.
package udp

import (
	"net"

	"github.com/golang/glog"
	"github.com/pushrax/bufferpool"

	"github.com/chihaya/chihaya/config"
	"github.com/chihaya/chihaya/tracker"
)

// Server represents a UDP torrent tracker.
type Server struct {
	config  *config.Config
	tracker *tracker.Tracker
}

func (srv *Server) ListenAndServe() error {
	listenAddr, err := net.ResolveUDPAddr("udp", srv.config.UDPListenAddr)
	if err != nil {
		return err
	}

	sock, err := net.ListenUDP("udp", listenAddr)
	defer sock.Close()
	if err != nil {
		return err
	}

	if srv.config.UDPReadBufferSize > 0 {
		sock.SetReadBuffer(srv.config.UDPReadBufferSize)
	}

	pool := bufferpool.New(1000, 2048)

	for {
		buffer := pool.TakeSlice()
		n, addr, err := sock.ReadFromUDP(buffer)
		if err != nil {
			return err
		}

		go func() {
			response := srv.handlePacket(buffer[:n], addr)
			if response != nil {
				sock.WriteToUDP(response, addr)
			}
			pool.GiveSlice(buffer)
		}()
	}
}

func Serve(cfg *config.Config, tkr *tracker.Tracker) {
	srv := &Server{
		config:  cfg,
		tracker: tkr,
	}

	glog.V(0).Info("Starting UDP on ", cfg.UDPListenAddr)
	if err := srv.ListenAndServe(); err != nil {
		glog.Errorf("Failed to run UDP server: %s", err.Error())
	}
}