Skip to content

Commit

Permalink
Improve service discovery, implement SD for CalDav
Browse files Browse the repository at this point in the history
  • Loading branch information
dpeukert committed Sep 6, 2020
1 parent 9e23289 commit ec9206a
Show file tree
Hide file tree
Showing 3 changed files with 95 additions and 33 deletions.
9 changes: 7 additions & 2 deletions caldav/client.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,12 @@ import (
"github.com/emersion/go-webdav/internal"
)

// Client provides access to a remote CardDAV server.
func Discover(host string) (string, error) {
url, err := internal.Discover("caldav", host)
return url, err
}

// Client provides access to a remote CalDAV server.
type Client struct {
*webdav.Client

Expand Down Expand Up @@ -91,7 +96,7 @@ func (c *Client) FindCalendars(calendarHomeSet string) ([]Calendar, error) {
return nil, err
}
if maxResSize.Size < 0 {
return nil, fmt.Errorf("carddav: max-resource-size must be a positive integer")
return nil, fmt.Errorf("caldav: max-resource-size must be a positive integer")
}

l = append(l, Calendar{
Expand Down
34 changes: 3 additions & 31 deletions carddav/client.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@ import (
"bytes"
"fmt"
"mime"
"net"
"net/http"
"net/url"
"strconv"
Expand All @@ -16,36 +15,9 @@ import (
"github.com/emersion/go-webdav/internal"
)

// Discover performs a DNS-based CardDAV service discovery as described in
// RFC 6352 section 11. It returns the URL to the CardDAV server.
func Discover(domain string) (string, error) {
// Only lookup carddavs (not carddav), plaintext connections are insecure
_, addrs, err := net.LookupSRV("carddavs", "tcp", domain)
if dnsErr, ok := err.(*net.DNSError); ok {
if dnsErr.IsTemporary {
return "", err
}
} else if err != nil {
return "", err
}

if len(addrs) == 0 {
return "", fmt.Errorf("carddav: domain doesn't have an SRV record")
}
addr := addrs[0]

target := strings.TrimSuffix(addr.Target, ".")
if target == "" {
return "", fmt.Errorf("carddav: empty target in SRV record")
}

u := url.URL{Scheme: "https"}
if addr.Port == 443 {
u.Host = target
} else {
u.Host = fmt.Sprintf("%v:%v", target, addr.Port)
}
return u.String(), nil
func Discover(host string) (string, error) {
url, err := internal.Discover("carddav", host)
return url, err
}

// Client provides access to a remote CardDAV server.
Expand Down
85 changes: 85 additions & 0 deletions internal/client.go
Original file line number Diff line number Diff line change
Expand Up @@ -6,13 +6,98 @@ import (
"fmt"
"io"
"mime"
"net"
"net/http"
"net/url"
"path"
"strings"
"unicode"
)

// Discover performs a DNS-based CalDAV/CardDAV service discovery as described in
// RFC 6764 section 6. It returns the URL to the CalDAV/CardDAV server.
func Discover(service string, host string) (string, error) {
if service != "caldav" && service != "carddav" {
return "", fmt.Errorf("webdav: service discovery of type %v not supported", service)
}

path := ""

// Check for SRV records for the service we want, only lookup secure versions
// (caldavs, carddavs), plaintext connections are insecure
_, addrs, err := net.LookupSRV(fmt.Sprintf("%vs", service), "tcp", host)
if dnsErr, ok := err.(*net.DNSError); ok {
if dnsErr.IsTemporary {
return "", err
}
} else if err != nil {
return "", err
}

if len(addrs) > 0 {
srvTarget := strings.TrimSuffix(addrs[0].Target, ".")

// If we found one, check for TXT records specifying the path
if srvTarget != "" {
txtRecs, err := net.LookupTXT(fmt.Sprintf("_%vs._tcp.%v", service, host))
if dnsErr, ok := err.(*net.DNSError); ok {
if dnsErr.IsTemporary {
return "", err
}
} else if err != nil {
return "", err
}

for _, txtRec := range txtRecs {
// This is not correct according to RFC 6763, but LookupTXT merges all constituent strings together
for _, txtRecKeyVal := range strings.Split(txtRec, " ") {
if strings.HasPrefix(txtRecKeyVal, "path=") {
path = strings.TrimPrefix(txtRecKeyVal, "path=")
break
}
}

if path != "" {
break
}
}

if addrs[0].Port == 443 {
host = srvTarget
} else {
host = fmt.Sprintf("%v:%v", srvTarget, addrs[0].Port)
}
}
}

// If we didn't get a path from TXT records, use the default well-known location
if path == "" {
path = fmt.Sprintf("/.well-known/%v", service)
}

u := url.URL{Scheme: "https", Host: host, Path: path}
serviceUrl := u.String()

// Check if the resulting URL hosts a service
req, err := http.NewRequest(http.MethodOptions, serviceUrl, nil)
if err != nil {
return "", err
}

resp, err := http.DefaultClient.Do(req)
if err != nil {
return "", err
}
resp.Body.Close()

// Servers might require authentication to perform an OPTIONS request
if resp.StatusCode/100 != 2 && resp.StatusCode != http.StatusUnauthorized {
return "", fmt.Errorf("HTTP request to %v failed: %v %v", serviceUrl, resp.StatusCode, resp.Status)
}

return serviceUrl, nil
}

// HTTPClient performs HTTP requests. It's implemented by *http.Client.
type HTTPClient interface {
Do(req *http.Request) (*http.Response, error)
Expand Down

0 comments on commit ec9206a

Please sign in to comment.