package router import ( "fmt" "net" "strconv" ) // NewRouter returns an initialized dns router func NewRouter(opts ...Option) Router { options := NewOptions(opts...) return &dns{options} } type dns struct { options Options } func (d *dns) Init(opts ...Option) error { for _, o := range opts { o(&d.options) } return nil } func (d *dns) Options() Options { return d.options } func (d *dns) Table() Table { return nil } func (d *dns) Close() error { return nil } func (d *dns) Lookup(opts ...QueryOption) ([]Route, error) { options := NewQuery(opts...) // check to see if we have the port provided in the service, e.g. go-micro-srv-foo:8000 host, port, err := net.SplitHostPort(options.Service) if err == nil { var ips []string // lookup the service using A records ips, err = net.LookupHost(host) if err != nil { return nil, err } p, _ := strconv.Atoi(port) // convert the ip addresses to routes result := make([]Route, len(ips)) for i, ip := range ips { result[i] = Route{ Service: options.Service, Address: fmt.Sprintf("%s:%d", ip, p), } } return result, nil } // we didn't get the port so we'll lookup the service using SRV records. If we can't lookup the // service using the SRV record, we return the error. _, nodes, err := net.LookupSRV(options.Service, "tcp", d.options.Network) if err != nil { return nil, err } // convert the nodes (net services) to routes result := make([]Route, len(nodes)) for i, n := range nodes { result[i] = Route{ Service: options.Service, Address: fmt.Sprintf("%s:%d", n.Target, n.Port), Network: d.options.Network, } } return result, nil } func (d *dns) Watch(opts ...WatchOption) (Watcher, error) { return nil, nil } func (d *dns) Name() string { return d.options.Name } func (d *dns) String() string { return "dns" }