// -=-=- StartHere -=-=- // // # Chapter I: Using a custom UDP resolver // // In this chapter we will write together a `main.go` file that // uses a custom UDP DNS resolver to lookup domain names. // // This program is very similar to the one in the previous chapter // except that we'll be configuring a custom resolver. // // (This file is auto-generated from the corresponding source file, // so make sure you don't edit it manually.) // // ## The main.go file // // We define `main.go` file using `package main`. // // There's not much to say about the beginning of the program // since it is equal to the one in the previous chapter. // // ```Go package main import ( "context" "errors" "flag" "os" "time" "github.com/apex/log" "github.com/ooni/probe-cli/v3/internal/netxlite" ) func main() { log.SetLevel(log.DebugLevel) hostname := flag.String("hostname", "dns.google", "Hostname to resolve") timeout := flag.Duration("timeout", 60*time.Second, "Timeout") serverAddr := flag.String("server-addr", "1.1.1.1:53", "DNS server address") flag.Parse() ctx, cancel := context.WithTimeout(context.Background(), *timeout) defer cancel() // ``` // // Here's where we start to diverge. We create a dialer without a resolver, // which is going to be used by the UDP resolver. // // ```Go dialer := netxlite.NewDialerWithoutResolver(log.Log) // ``` // // Then, we create an UDP resolver. The arguments are the same as for // creating a system resolver, except that we also need to specify the // UDP endpoint address at which the server is listening. // // ```Go reso := netxlite.NewParallelUDPResolver(log.Log, dialer, *serverAddr) // ``` // // The API we invoke is the same as in the previous chapter, though, // and the rest of the program is equal to the one in the previous chapter. // // ```Go addrs, err := reso.LookupHost(ctx, *hostname) if err != nil { fatal(err) } log.Infof("resolver addrs: %+v", addrs) } func fatal(err error) { var ew *netxlite.ErrWrapper if !errors.As(err, &ew) { log.Fatal("cannot get ErrWrapper") } log.Warnf("error string : %s", err.Error()) log.Warnf("OONI failure : %s", ew.Failure) log.Warnf("failed operation: %s", ew.Operation) log.Warnf("underlying error: %+v", ew.WrappedErr) os.Exit(1) } // ``` // // ## Running the code // // ### Vanilla run // // You can now run this code as follows: // // ```bash // go run -race ./internal/tutorial/netxlite/chapter06 // ``` // // You will see debug logs describing what is happening along with timing info. // // ### NXDOMAIN // // ```bash // go run -race ./internal/tutorial/netxlite/chapter06 -hostname antani.ooni.io // ``` // // should cause a `dns_nxdomain_error`, because the domain does not exist. // // ### Timeout // // ```bash // go run -race ./internal/tutorial/netxlite/chapter06 -timeout 10us // ``` // // should cause a timeout error, because the timeout is ridicolously small. // // ```bash // go run -race ./internal/tutorial/netxlite/chapter06 -server-addr 1.1.1.1:1 // ``` // // should also cause a timeout, because 1.1.1.1:1 is not an endpoint // where a DNS-over-UDP resolver is listening. // // ## Conclusions // // We have seen how to use a custom DNS-over-UDP resolver.