package dnsx import ( "github.com/miekg/dns" "github.com/ooni/probe-cli/v3/internal/netxlite/dnsx/model" "github.com/ooni/probe-cli/v3/internal/netxlite/errorsx" ) // HTTPSSvc is an HTTPSSvc reply. type HTTPSSvc = model.HTTPSSvc // The DNSDecoder decodes DNS replies. type DNSDecoder interface { // DecodeLookupHost decodes an A or AAAA reply. DecodeLookupHost(qtype uint16, data []byte) ([]string, error) // DecodeHTTPS decodes an HTTPS reply. DecodeHTTPS(data []byte) (*HTTPSSvc, error) } // DNSDecoderMiekg uses github.com/miekg/dns to implement the Decoder. type DNSDecoderMiekg struct{} func (d *DNSDecoderMiekg) parseReply(data []byte) (*dns.Msg, error) { reply := new(dns.Msg) if err := reply.Unpack(data); err != nil { return nil, err } // TODO(bassosimone): map more errors to net.DNSError names // TODO(bassosimone): add support for lame referral. switch reply.Rcode { case dns.RcodeSuccess: return reply, nil case dns.RcodeNameError: return nil, errorsx.ErrOODNSNoSuchHost case dns.RcodeRefused: return nil, errorsx.ErrOODNSRefused default: return nil, errorsx.ErrOODNSMisbehaving } } func (d *DNSDecoderMiekg) DecodeHTTPS(data []byte) (*HTTPSSvc, error) { reply, err := d.parseReply(data) if err != nil { return nil, err } out := &HTTPSSvc{} for _, answer := range reply.Answer { switch avalue := answer.(type) { case *dns.HTTPS: for _, v := range avalue.Value { switch extv := v.(type) { case *dns.SVCBAlpn: out.ALPN = extv.Alpn case *dns.SVCBIPv4Hint: for _, ip := range extv.Hint { out.IPv4 = append(out.IPv4, ip.String()) } case *dns.SVCBIPv6Hint: for _, ip := range extv.Hint { out.IPv6 = append(out.IPv6, ip.String()) } } } } } if len(out.ALPN) <= 0 { return nil, errorsx.ErrOODNSNoAnswer } return out, nil } func (d *DNSDecoderMiekg) DecodeLookupHost(qtype uint16, data []byte) ([]string, error) { reply, err := d.parseReply(data) if err != nil { return nil, err } var addrs []string for _, answer := range reply.Answer { switch qtype { case dns.TypeA: if rra, ok := answer.(*dns.A); ok { ip := rra.A addrs = append(addrs, ip.String()) } case dns.TypeAAAA: if rra, ok := answer.(*dns.AAAA); ok { ip := rra.AAAA addrs = append(addrs, ip.String()) } } } if len(addrs) <= 0 { return nil, errorsx.ErrOODNSNoAnswer } return addrs, nil } var _ DNSDecoder = &DNSDecoderMiekg{}