2011-01-18 02:11:11 +11:00
|
|
|
/*
|
|
|
|
* A name server which sends back the IP address of its client, the
|
|
|
|
* recursive resolver. When queried for type TXT, it sends back the text
|
|
|
|
* form of the address. When queried for type A (resp. AAAA), it sends
|
|
|
|
* back the IPv4 (resp. v6) address.
|
2011-02-12 06:54:54 +11:00
|
|
|
*
|
2011-01-18 02:11:11 +11:00
|
|
|
* Similar services: whoami.ultradns.net, whoami.akamai.net. Also (but it
|
|
|
|
* is not their normal goal): rs.dns-oarc.net, porttest.dns-oarc.net,
|
|
|
|
* amiopen.openresolvers.org.
|
2011-02-12 06:54:54 +11:00
|
|
|
*
|
2011-01-18 02:11:11 +11:00
|
|
|
* Stephane Bortzmeyer <stephane+grong@bortzmeyer.org>
|
|
|
|
*
|
|
|
|
* Adapted to Go DNS (i.e. completely rewritten)
|
|
|
|
* Miek Gieben <miek@miek.nl>
|
|
|
|
*/
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2011-01-28 06:16:58 +11:00
|
|
|
"os"
|
2011-01-18 02:11:11 +11:00
|
|
|
"net"
|
|
|
|
"dns"
|
2011-01-28 08:09:24 +11:00
|
|
|
"fmt"
|
2011-01-28 06:16:58 +11:00
|
|
|
"strconv"
|
|
|
|
"runtime"
|
|
|
|
"os/signal"
|
2011-01-18 02:11:11 +11:00
|
|
|
)
|
|
|
|
|
2011-02-12 06:54:54 +11:00
|
|
|
func reply(a net.Addr, in *dns.Msg, tcp bool) *dns.Msg {
|
|
|
|
if in.MsgHdr.Response == true {
|
2011-01-28 06:16:58 +11:00
|
|
|
return nil // Don't answer responses
|
|
|
|
}
|
|
|
|
m := new(dns.Msg)
|
2011-02-12 06:54:54 +11:00
|
|
|
m.MsgHdr.Id = in.MsgHdr.Id
|
2011-01-28 06:16:58 +11:00
|
|
|
m.MsgHdr.Authoritative = true
|
|
|
|
m.MsgHdr.Response = true
|
|
|
|
m.MsgHdr.Opcode = dns.OpcodeQuery
|
2011-01-18 02:11:11 +11:00
|
|
|
|
2011-01-28 06:16:58 +11:00
|
|
|
m.MsgHdr.Rcode = dns.RcodeSuccess
|
|
|
|
m.Question = make([]dns.Question, 1)
|
|
|
|
m.Answer = make([]dns.RR, 1)
|
|
|
|
m.Extra = make([]dns.RR, 1)
|
2011-01-18 02:11:11 +11:00
|
|
|
|
2011-01-28 06:16:58 +11:00
|
|
|
r := new(dns.RR_A)
|
|
|
|
r.Hdr = dns.RR_Header{Name: "whoami.miek.nl.", Rrtype: dns.TypeA, Class: dns.ClassINET, Ttl: 0}
|
|
|
|
ip, _ := net.ResolveUDPAddr(a.String()) // No general variant for both upd and tcp
|
|
|
|
r.A = ip.IP.To4() // To4 very important
|
2011-01-18 02:11:11 +11:00
|
|
|
|
2011-01-28 06:16:58 +11:00
|
|
|
t := new(dns.RR_TXT)
|
|
|
|
t.Hdr = dns.RR_Header{Name: "whoami.miek.nl.", Rrtype: dns.TypeTXT, Class: dns.ClassINET, Ttl: 0}
|
|
|
|
if tcp {
|
|
|
|
t.Txt = "Port: " + strconv.Itoa(ip.Port) + " (tcp)"
|
|
|
|
} else {
|
|
|
|
t.Txt = "Port: " + strconv.Itoa(ip.Port) + " (udp)"
|
|
|
|
}
|
2011-01-18 02:11:11 +11:00
|
|
|
|
2011-02-12 06:54:54 +11:00
|
|
|
m.Question[0] = in.Question[0]
|
2011-01-28 06:16:58 +11:00
|
|
|
m.Answer[0] = r
|
|
|
|
m.Extra[0] = t
|
|
|
|
return m
|
2011-01-19 07:34:22 +11:00
|
|
|
}
|
|
|
|
|
2011-02-12 06:54:54 +11:00
|
|
|
func replyUDP(c *net.UDPConn, a net.Addr, in *dns.Msg) {
|
2011-01-28 06:16:58 +11:00
|
|
|
m := reply(a, in, false)
|
|
|
|
if m == nil {
|
|
|
|
return
|
|
|
|
}
|
2011-01-28 08:09:24 +11:00
|
|
|
fmt.Fprintf(os.Stderr, "%v\n", m)
|
2011-01-28 06:16:58 +11:00
|
|
|
out, ok := m.Pack()
|
|
|
|
if !ok {
|
|
|
|
println("Failed to pack")
|
|
|
|
return
|
|
|
|
}
|
2011-02-10 04:06:37 +11:00
|
|
|
dns.SendUDP(out, c, a)
|
2011-01-18 02:11:11 +11:00
|
|
|
}
|
|
|
|
|
2011-02-12 06:54:54 +11:00
|
|
|
func replyTCP(c *net.TCPConn, a net.Addr, in *dns.Msg) {
|
|
|
|
m := reply(a, in, true)
|
2011-01-28 06:16:58 +11:00
|
|
|
if m == nil {
|
|
|
|
return
|
|
|
|
}
|
2011-01-28 08:09:24 +11:00
|
|
|
fmt.Fprintf(os.Stderr, "%v\n", m)
|
2011-01-28 06:16:58 +11:00
|
|
|
out, ok := m.Pack()
|
|
|
|
if !ok {
|
|
|
|
println("Failed to pack")
|
|
|
|
return
|
|
|
|
}
|
2011-02-10 04:06:37 +11:00
|
|
|
dns.SendTCP(out, c, a)
|
2011-01-18 02:11:11 +11:00
|
|
|
}
|
|
|
|
|
2011-01-19 07:34:22 +11:00
|
|
|
|
2011-02-12 06:54:54 +11:00
|
|
|
func main() {
|
|
|
|
dns.ListenAndServeUDP("127.0.0.1:8053", replyUDP)
|
2011-01-18 02:11:11 +11:00
|
|
|
}
|