mirror of
https://github.com/1f349/go-webdav.git
synced 2025-01-21 23:06:23 +00:00
webdav: add very basic Client
This commit is contained in:
parent
055a297f6e
commit
3beb076950
63
client.go
Normal file
63
client.go
Normal file
@ -0,0 +1,63 @@
|
||||
package webdav
|
||||
|
||||
import (
|
||||
"encoding/xml"
|
||||
"fmt"
|
||||
"net/http"
|
||||
"strings"
|
||||
|
||||
"github.com/emersion/go-webdav/internal"
|
||||
)
|
||||
|
||||
type Client struct {
|
||||
c *internal.Client
|
||||
}
|
||||
|
||||
func NewClient(c *http.Client, endpoint string) (*Client, error) {
|
||||
ic, err := internal.NewClient(c, endpoint)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &Client{ic}, nil
|
||||
}
|
||||
|
||||
func (c *Client) FindCurrentUserPrincipal() (string, error) {
|
||||
r := strings.NewReader(`<?xml version="1.0" encoding="utf-8"?>
|
||||
<D:propfind xmlns:D="DAV:">
|
||||
<D:prop>
|
||||
<D:current-user-principal/>
|
||||
</D:prop>
|
||||
</D:propfind>
|
||||
`)
|
||||
|
||||
req, err := c.c.NewRequest("PROPFIND", "", r)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
req.Header.Add("Depth", "0")
|
||||
req.Header.Add("Content-Type", "text/xml; charset=\"utf-8\"")
|
||||
|
||||
resps, err := c.c.DoMultiStatus(req)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
if len(resps) != 1 {
|
||||
return "", fmt.Errorf("expected exactly one response in multistatus, got %v", len(resps))
|
||||
}
|
||||
resp := &resps[0]
|
||||
|
||||
// TODO: handle propstats with errors
|
||||
if len(resp.Propstats) != 1 {
|
||||
return "", fmt.Errorf("expected exactly one propstat in response")
|
||||
}
|
||||
propstat := &resp.Propstats[0]
|
||||
|
||||
var prop currentUserPrincipalProp
|
||||
if err := xml.NewTokenDecoder(propstat.Prop.TokenReader()).Decode(&prop); err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
return prop.Href, nil
|
||||
}
|
5
elements.go
Normal file
5
elements.go
Normal file
@ -0,0 +1,5 @@
|
||||
package webdav
|
||||
|
||||
type currentUserPrincipalProp struct {
|
||||
Href string `xml:"current-user-principal>href"`
|
||||
}
|
63
internal/client.go
Normal file
63
internal/client.go
Normal file
@ -0,0 +1,63 @@
|
||||
package internal
|
||||
|
||||
import (
|
||||
"encoding/xml"
|
||||
"fmt"
|
||||
"io"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"path"
|
||||
)
|
||||
|
||||
type Client struct {
|
||||
http *http.Client
|
||||
endpoint *url.URL
|
||||
}
|
||||
|
||||
func NewClient(c *http.Client, endpoint string) (*Client, error) {
|
||||
if c == nil {
|
||||
c = http.DefaultClient
|
||||
}
|
||||
|
||||
u, err := url.Parse(endpoint)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &Client{c, u}, nil
|
||||
}
|
||||
|
||||
func (c *Client) NewRequest(method string, p string, body io.Reader) (*http.Request, error) {
|
||||
u := url.URL{
|
||||
Scheme: c.endpoint.Scheme,
|
||||
User: c.endpoint.User,
|
||||
Host: c.endpoint.Host,
|
||||
Path: path.Join(c.endpoint.Path, p),
|
||||
}
|
||||
return http.NewRequest(method, u.String(), body)
|
||||
}
|
||||
|
||||
func (c *Client) Do(req *http.Request) (*http.Response, error) {
|
||||
// TODO: remove this quirk
|
||||
req.SetBasicAuth("simon", "")
|
||||
return c.http.Do(req)
|
||||
}
|
||||
|
||||
func (c *Client) DoMultiStatus(req *http.Request) ([]Response, error) {
|
||||
resp, err := c.Do(req)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer resp.Body.Close()
|
||||
|
||||
if resp.StatusCode != http.StatusMultiStatus {
|
||||
return nil, fmt.Errorf("HTTP multi-status request failed: %v", resp.Status)
|
||||
}
|
||||
|
||||
// TODO: the response can be quite large, support streaming Response elements
|
||||
var ms multistatus
|
||||
if err := xml.NewDecoder(resp.Body).Decode(&ms); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return ms.Responses, nil
|
||||
}
|
29
internal/elements.go
Normal file
29
internal/elements.go
Normal file
@ -0,0 +1,29 @@
|
||||
package internal
|
||||
|
||||
import (
|
||||
"encoding/xml"
|
||||
)
|
||||
|
||||
// https://tools.ietf.org/html/rfc4918#section-14.16
|
||||
type multistatus struct {
|
||||
XMLName xml.Name `xml:"DAV: multistatus"`
|
||||
Responses []Response `xml:"DAV: response"`
|
||||
// TODO: responsedescription?
|
||||
}
|
||||
|
||||
// https://tools.ietf.org/html/rfc4918#section-14.24
|
||||
type Response struct {
|
||||
XMLName xml.Name `xml:"DAV: response"`
|
||||
Href string `xml:"DAV: href"`
|
||||
Propstats []Propstat `xml:"DAV: propstat"`
|
||||
// TODO: (href*, status)
|
||||
// TODO: error?, responsedescription? , location?
|
||||
}
|
||||
|
||||
// https://tools.ietf.org/html/rfc4918#section-14.22
|
||||
type Propstat struct {
|
||||
XMLName xml.Name `xml:"DAV: propstat"`
|
||||
Prop RawXMLValue `xml:"DAV: prop"`
|
||||
Status string `xml:"DAV: status"`
|
||||
// TODO: error?, responsedescription?
|
||||
}
|
Loading…
Reference in New Issue
Block a user