2020-01-22 09:06:00 +00:00
|
|
|
// Package internal provides low-level helpers for WebDAV clients and servers.
|
|
|
|
package internal
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2020-05-13 17:24:29 +01:00
|
|
|
"net/http"
|
2020-01-22 09:06:00 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Depth indicates whether a request applies to the resource's members. It's
|
|
|
|
// defined in RFC 4918 section 10.2.
|
|
|
|
type Depth int
|
|
|
|
|
|
|
|
const (
|
|
|
|
// DepthZero indicates that the request applies only to the resource.
|
|
|
|
DepthZero Depth = 0
|
|
|
|
// DepthOne indicates that the request applies to the resource and its
|
|
|
|
// internal members only.
|
|
|
|
DepthOne Depth = 1
|
|
|
|
// DepthInfinity indicates that the request applies to the resource and all
|
|
|
|
// of its members.
|
|
|
|
DepthInfinity Depth = -1
|
|
|
|
)
|
|
|
|
|
|
|
|
// ParseDepth parses a Depth header.
|
|
|
|
func ParseDepth(s string) (Depth, error) {
|
|
|
|
switch s {
|
|
|
|
case "0":
|
|
|
|
return DepthZero, nil
|
|
|
|
case "1":
|
|
|
|
return DepthOne, nil
|
|
|
|
case "infinity":
|
|
|
|
return DepthInfinity, nil
|
|
|
|
}
|
|
|
|
return 0, fmt.Errorf("webdav: invalid Depth value")
|
|
|
|
}
|
|
|
|
|
|
|
|
// String formats the depth.
|
|
|
|
func (d Depth) String() string {
|
|
|
|
switch d {
|
|
|
|
case DepthZero:
|
|
|
|
return "0"
|
|
|
|
case DepthOne:
|
|
|
|
return "1"
|
|
|
|
case DepthInfinity:
|
|
|
|
return "infinity"
|
|
|
|
}
|
|
|
|
panic("webdav: invalid Depth value")
|
|
|
|
}
|
2020-01-22 09:09:51 +00:00
|
|
|
|
|
|
|
// ParseOverwrite parses an Overwrite header.
|
|
|
|
func ParseOverwrite(s string) (bool, error) {
|
|
|
|
switch s {
|
|
|
|
case "T":
|
|
|
|
return true, nil
|
|
|
|
case "F":
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
return false, fmt.Errorf("webdav: invalid Overwrite value")
|
|
|
|
}
|
|
|
|
|
|
|
|
// FormatOverwrite formats an Overwrite header.
|
|
|
|
func FormatOverwrite(overwrite bool) string {
|
|
|
|
if overwrite {
|
|
|
|
return "T"
|
|
|
|
} else {
|
|
|
|
return "F"
|
|
|
|
}
|
|
|
|
}
|
2020-05-13 17:24:29 +01:00
|
|
|
|
|
|
|
type HTTPError struct {
|
|
|
|
Code int
|
|
|
|
Err error
|
|
|
|
}
|
|
|
|
|
|
|
|
func HTTPErrorFromError(err error) *HTTPError {
|
|
|
|
if err == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if httpErr, ok := err.(*HTTPError); ok {
|
|
|
|
return httpErr
|
|
|
|
} else {
|
|
|
|
return &HTTPError{http.StatusInternalServerError, err}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func IsNotFound(err error) bool {
|
|
|
|
return HTTPErrorFromError(err).Code == http.StatusNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
func HTTPErrorf(code int, format string, a ...interface{}) *HTTPError {
|
|
|
|
return &HTTPError{code, fmt.Errorf(format, a...)}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (err *HTTPError) Error() string {
|
|
|
|
s := fmt.Sprintf("%v %v", err.Code, http.StatusText(err.Code))
|
|
|
|
if err.Err != nil {
|
|
|
|
return fmt.Sprintf("%v: %v", s, err.Err)
|
|
|
|
} else {
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
}
|
2022-03-10 14:35:39 +00:00
|
|
|
|
2022-05-02 10:23:06 +01:00
|
|
|
func (err *HTTPError) Unwrap() error {
|
|
|
|
return err.Err
|
|
|
|
}
|