2
0
mirror of synced 2025-02-24 22:58:28 +00:00

167 lines
3.4 KiB
Go
Raw Normal View History

package main
import (
"bitbucket.org/anacrolix/go.torrent/dht"
2014-07-10 00:15:28 +10:00
"bitbucket.org/anacrolix/go.torrent/tracker"
_ "bitbucket.org/anacrolix/go.torrent/util/profile"
"flag"
"fmt"
"io"
"log"
"net"
"os"
"os/signal"
)
type pingResponse struct {
addr string
krpc dht.Msg
}
var (
tableFileName = flag.String("tableFile", "", "name of file for storing node info")
serveAddr = flag.String("serveAddr", ":0", "local UDP address")
infoHash = flag.String("infoHash", "", "torrent infohash")
s dht.Server
)
func loadTable() error {
if *tableFileName == "" {
return nil
}
f, err := os.Open(*tableFileName)
if os.IsNotExist(err) {
return nil
}
if err != nil {
return fmt.Errorf("error opening table file: %s", err)
}
defer f.Close()
added := 0
for {
b := make([]byte, dht.CompactNodeInfoLen)
_, err := io.ReadFull(f, b)
if err == io.EOF {
break
}
if err != nil {
return fmt.Errorf("error reading table file: %s", err)
}
var ni dht.NodeInfo
err = ni.UnmarshalCompact(b)
if err != nil {
return fmt.Errorf("error unmarshaling compact node info: %s", err)
}
s.AddNode(ni)
added++
}
log.Printf("loaded %d nodes from table file", added)
return nil
}
func init() {
log.SetFlags(log.LstdFlags | log.Lshortfile)
flag.Parse()
switch len(*infoHash) {
case 20:
case 40:
2014-07-10 00:15:28 +10:00
_, err := fmt.Sscanf(*infoHash, "%x", infoHash)
if err != nil {
log.Fatal(err)
}
default:
log.Fatal("require 20 byte infohash")
}
var err error
s.Socket, err = net.ListenUDP("udp4", func() *net.UDPAddr {
addr, err := net.ResolveUDPAddr("udp4", *serveAddr)
if err != nil {
log.Fatalf("error resolving serve addr: %s", err)
}
return addr
}())
if err != nil {
log.Fatal(err)
}
s.Init()
err = loadTable()
if err != nil {
log.Fatalf("error loading table: %s", err)
}
log.Printf("dht server on %s, ID is %q", s.Socket.LocalAddr(), s.IDString())
setupSignals()
}
func saveTable() error {
goodNodes := s.Nodes()
if *tableFileName == "" {
if len(goodNodes) != 0 {
log.Printf("discarding %d good nodes!", len(goodNodes))
}
return nil
}
f, err := os.OpenFile(*tableFileName, os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0666)
if err != nil {
return fmt.Errorf("error opening table file: %s", err)
}
defer f.Close()
for _, nodeInfo := range goodNodes {
var b [dht.CompactNodeInfoLen]byte
err := nodeInfo.PutCompact(b[:])
if err != nil {
return fmt.Errorf("error compacting node info: %s", err)
}
_, err = f.Write(b[:])
if err != nil {
return fmt.Errorf("error writing compact node info: %s", err)
}
}
log.Printf("saved %d nodes to table file", len(goodNodes))
return nil
}
func setupSignals() {
ch := make(chan os.Signal)
2014-07-10 00:15:28 +10:00
signal.Notify(ch, os.Interrupt)
go func() {
<-ch
s.StopServing()
}()
}
func main() {
go func() {
2014-07-10 00:15:28 +10:00
defer s.StopServing()
if err := s.Bootstrap(); err != nil {
log.Printf("error bootstrapping: %s", err)
return
}
saveTable()
ps, err := s.GetPeers(*infoHash)
if err != nil {
log.Fatal(err)
}
2014-07-10 00:15:28 +10:00
seen := make(map[tracker.CompactPeer]struct{})
for sl := range ps.Values {
for _, p := range sl {
2014-07-10 00:15:28 +10:00
if _, ok := seen[p]; ok {
continue
}
seen[p] = struct{}{}
fmt.Println((&net.UDPAddr{
IP: p.IP[:],
Port: int(p.Port),
}).String())
}
}
}()
err := s.Serve()
if err := saveTable(); err != nil {
log.Printf("error saving node table: %s", err)
}
if err != nil {
log.Fatalf("error serving dht: %s", err)
}
}