2019-10-08 09:04:13 +01:00
|
|
|
// Package dns resolves names to dns records
|
2021-10-02 19:55:07 +03:00
|
|
|
package dns // import "go.unistack.org/micro/v3/resolver/dns"
|
2019-07-28 12:14:40 +01:00
|
|
|
|
|
|
|
import (
|
2019-11-12 15:46:30 +00:00
|
|
|
"context"
|
2019-07-28 12:14:40 +01:00
|
|
|
"net"
|
2021-02-13 15:35:56 +03:00
|
|
|
"sync"
|
|
|
|
"time"
|
2019-07-28 12:14:40 +01:00
|
|
|
|
2021-10-02 19:55:07 +03:00
|
|
|
"go.unistack.org/micro/v3/resolver"
|
2019-07-28 12:14:40 +01:00
|
|
|
)
|
|
|
|
|
2019-08-20 12:48:51 +01:00
|
|
|
// Resolver is a DNS network resolve
|
2019-11-12 15:46:30 +00:00
|
|
|
type Resolver struct {
|
2021-02-13 15:35:56 +03:00
|
|
|
goresolver *net.Resolver
|
2021-03-06 19:45:13 +03:00
|
|
|
// Address of resolver to use
|
|
|
|
Address string
|
2021-02-13 15:35:56 +03:00
|
|
|
sync.RWMutex
|
2019-11-12 15:46:30 +00:00
|
|
|
}
|
2019-07-28 12:14:40 +01:00
|
|
|
|
2021-02-14 11:28:50 +03:00
|
|
|
// Resolve tries to resolve endpoint address
|
2019-07-28 20:00:09 +01:00
|
|
|
func (r *Resolver) Resolve(name string) ([]*resolver.Record, error) {
|
2019-10-08 09:04:13 +01:00
|
|
|
host, port, err := net.SplitHostPort(name)
|
|
|
|
if err != nil {
|
|
|
|
host = name
|
|
|
|
port = "8085"
|
|
|
|
}
|
|
|
|
|
2019-10-13 18:36:22 +01:00
|
|
|
if len(host) == 0 {
|
|
|
|
host = "localhost"
|
|
|
|
}
|
|
|
|
|
2019-11-12 15:46:30 +00:00
|
|
|
if len(r.Address) == 0 {
|
2021-02-13 15:35:56 +03:00
|
|
|
r.Address = "1.1.1.1:53"
|
2019-11-12 15:46:30 +00:00
|
|
|
}
|
|
|
|
|
2019-12-07 23:28:39 +00:00
|
|
|
// parsed an actual ip
|
|
|
|
if v := net.ParseIP(host); v != nil {
|
2021-02-13 15:35:56 +03:00
|
|
|
rec := &resolver.Record{Address: net.JoinHostPort(host, port)}
|
|
|
|
return []*resolver.Record{rec}, nil
|
2019-12-07 23:28:39 +00:00
|
|
|
}
|
|
|
|
|
2021-02-13 15:35:56 +03:00
|
|
|
r.RLock()
|
|
|
|
goresolver := r.goresolver
|
|
|
|
r.RUnlock()
|
|
|
|
|
|
|
|
if goresolver == nil {
|
|
|
|
r.Lock()
|
|
|
|
r.goresolver = &net.Resolver{
|
|
|
|
Dial: func(ctx context.Context, network, address string) (net.Conn, error) {
|
|
|
|
d := net.Dialer{
|
|
|
|
Timeout: time.Millisecond * time.Duration(100),
|
|
|
|
}
|
|
|
|
return d.DialContext(ctx, "udp", r.Address)
|
|
|
|
},
|
2019-11-12 15:46:30 +00:00
|
|
|
}
|
2021-02-13 15:35:56 +03:00
|
|
|
r.Unlock()
|
|
|
|
}
|
2019-11-12 15:46:30 +00:00
|
|
|
|
2021-02-13 15:35:56 +03:00
|
|
|
addrs, err := goresolver.LookupIP(context.TODO(), "ip", host)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-10-08 09:04:13 +01:00
|
|
|
|
2021-02-13 15:35:56 +03:00
|
|
|
if len(addrs) == 0 {
|
|
|
|
rec := &resolver.Record{Address: net.JoinHostPort(host, port)}
|
|
|
|
return []*resolver.Record{rec}, nil
|
2019-07-28 12:14:40 +01:00
|
|
|
}
|
2019-10-08 09:04:13 +01:00
|
|
|
|
2021-02-13 15:35:56 +03:00
|
|
|
records := make([]*resolver.Record, 0, len(addrs))
|
|
|
|
for _, addr := range addrs {
|
2019-12-07 23:28:39 +00:00
|
|
|
records = append(records, &resolver.Record{
|
2021-02-13 15:35:56 +03:00
|
|
|
Address: net.JoinHostPort(addr.String(), port),
|
2019-12-07 23:28:39 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-07-28 12:14:40 +01:00
|
|
|
return records, nil
|
|
|
|
}
|