12 _ "github.com/anacrolix/envpprof"
14 "github.com/anacrolix/torrent/dht"
15 _ "github.com/anacrolix/torrent/util/profile"
19 tableFileName = flag.String("tableFile", "", "name of file for storing node info")
20 serveAddr = flag.String("serveAddr", ":0", "local UDP address")
21 infoHash = flag.String("infoHash", "", "torrent infohash")
22 once = flag.Bool("once", false, "only do one scrape iteration")
25 quitting = make(chan struct{})
28 func loadTable() error {
29 if *tableFileName == "" {
32 f, err := os.Open(*tableFileName)
33 if os.IsNotExist(err) {
37 return fmt.Errorf("error opening table file: %s", err)
42 b := make([]byte, dht.CompactIPv4NodeInfoLen)
43 _, err := io.ReadFull(f, b)
48 return fmt.Errorf("error reading table file: %s", err)
51 err = ni.UnmarshalCompactIPv4(b)
53 return fmt.Errorf("error unmarshaling compact node info: %s", err)
58 log.Printf("loaded %d nodes from table file", added)
63 log.SetFlags(log.LstdFlags | log.Lshortfile)
65 switch len(*infoHash) {
68 _, err := fmt.Sscanf(*infoHash, "%x", infoHash)
73 log.Fatal("require 20 byte infohash")
76 s, err = dht.NewServer(&dht.ServerConfig{
84 log.Fatalf("error loading table: %s", err)
86 log.Printf("dht server on %s, ID is %x", s.Addr(), s.ID())
90 func saveTable() error {
91 goodNodes := s.Nodes()
92 if *tableFileName == "" {
93 if len(goodNodes) != 0 {
94 log.Print("good nodes were discarded because you didn't specify a table file")
98 f, err := os.OpenFile(*tableFileName, os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0666)
100 return fmt.Errorf("error opening table file: %s", err)
103 for _, nodeInfo := range goodNodes {
104 var b [dht.CompactIPv4NodeInfoLen]byte
105 err := nodeInfo.PutCompact(b[:])
107 return fmt.Errorf("error compacting node info: %s", err)
109 _, err = f.Write(b[:])
111 return fmt.Errorf("error writing compact node info: %s", err)
114 log.Printf("saved %d nodes to table file", len(goodNodes))
118 func setupSignals() {
119 ch := make(chan os.Signal)
120 signal.Notify(ch, os.Interrupt)
128 seen := make(map[string]struct{})
131 ps, err := s.Announce(*infoHash, 0, false)
138 case v, ok := <-ps.Peers:
142 log.Printf("received %d peers from %x", len(v.Peers), v.NodeInfo.ID)
143 for _, p := range v.Peers {
144 if _, ok := seen[p.String()]; ok {
147 seen[p.String()] = struct{}{}
148 fmt.Println((&net.UDPAddr{
161 if err := saveTable(); err != nil {
162 log.Printf("error saving node table: %s", err)