Skip to content

Commit 610bfee

Browse files
rstonigeltao
authored andcommitted
webdav: Simplify handling of Etag and Content-Type headers for GET, HEAD,
POST and PUT requests. Change-Id: Id7b8a0aff7af1edefc45e2441565a2261f539895 Reviewed-on: https://go-review.googlesource.com/10395 Reviewed-by: Nigel Tao <nigeltao@golang.org>
1 parent 4dbd2a1 commit 610bfee

File tree

3 files changed

+24
-41
lines changed

3 files changed

+24
-41
lines changed

webdav/prop.go

+1-6
Original file line numberDiff line numberDiff line change
@@ -359,13 +359,8 @@ func findContentType(fs FileSystem, ls LockSystem, name string, fi os.FileInfo)
359359
}
360360

361361
func findETag(fs FileSystem, ls LockSystem, name string, fi os.FileInfo) (string, error) {
362-
return detectETag(fi), nil
363-
}
364-
365-
// detectETag determines the ETag for the file described by fi.
366-
func detectETag(fi os.FileInfo) string {
367362
// The Apache http 2.4 web server by default concatenates the
368363
// modification time and size of a file. We replicate the heuristic
369364
// with nanosecond granularity.
370-
return fmt.Sprintf(`"%x%x"`, fi.ModTime().UnixNano(), fi.Size())
365+
return fmt.Sprintf(`"%x%x"`, fi.ModTime().UnixNano(), fi.Size()), nil
371366
}

webdav/prop_test.go

+7-3
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ import (
1717
func TestMemPS(t *testing.T) {
1818
// calcProps calculates the getlastmodified and getetag DAV: property
1919
// values in pstats for resource name in file-system fs.
20-
calcProps := func(name string, fs FileSystem, pstats []Propstat) error {
20+
calcProps := func(name string, fs FileSystem, ls LockSystem, pstats []Propstat) error {
2121
fi, err := fs.Stat(name)
2222
if err != nil {
2323
return err
@@ -32,7 +32,11 @@ func TestMemPS(t *testing.T) {
3232
if fi.IsDir() {
3333
continue
3434
}
35-
p.InnerXML = []byte(detectETag(fi))
35+
etag, err := findETag(fs, ls, name, fi)
36+
if err != nil {
37+
return err
38+
}
39+
p.InnerXML = []byte(etag)
3640
pst.Props[i] = p
3741
}
3842
}
@@ -494,7 +498,7 @@ func TestMemPS(t *testing.T) {
494498
ls := NewMemLS()
495499
for _, op := range tc.propOp {
496500
desc := fmt.Sprintf("%s: %s %s", tc.desc, op.op, op.name)
497-
if err = calcProps(op.name, fs, op.wantPropstats); err != nil {
501+
if err = calcProps(op.name, fs, ls, op.wantPropstats); err != nil {
498502
t.Fatalf("%s: calcProps: %v", desc, err)
499503
}
500504

webdav/webdav.go

+16-32
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,6 @@
66
package webdav // import "golang.org/x/net/webdav"
77

88
import (
9-
"encoding/xml"
109
"errors"
1110
"fmt"
1211
"io"
@@ -203,14 +202,14 @@ func (h *Handler) handleGetHeadPost(w http.ResponseWriter, r *http.Request) (sta
203202
if err != nil {
204203
return http.StatusNotFound, err
205204
}
206-
pstats, err := props(h.FileSystem, h.LockSystem, r.URL.Path, []xml.Name{
207-
{Space: "DAV:", Local: "getetag"},
208-
{Space: "DAV:", Local: "getcontenttype"},
209-
})
210-
if err != nil {
211-
return http.StatusInternalServerError, err
205+
if !fi.IsDir() {
206+
etag, err := findETag(h.FileSystem, h.LockSystem, r.URL.Path, fi)
207+
if err != nil {
208+
return http.StatusInternalServerError, err
209+
}
210+
w.Header().Set("ETag", etag)
212211
}
213-
writeDAVHeaders(w, pstats)
212+
// Let ServeContent determine the Content-Type header.
214213
http.ServeContent(w, r, r.URL.Path, fi.ModTime(), f)
215214
return 0, nil
216215
}
@@ -245,26 +244,31 @@ func (h *Handler) handlePut(w http.ResponseWriter, r *http.Request) (status int,
245244
return status, err
246245
}
247246
defer release()
247+
// TODO(rost): Support the If-Match, If-None-Match headers? See bradfitz'
248+
// comments in http.checkEtag.
248249

249250
f, err := h.FileSystem.OpenFile(r.URL.Path, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0666)
250251
if err != nil {
251252
return http.StatusNotFound, err
252253
}
253254
_, copyErr := io.Copy(f, r.Body)
255+
fi, statErr := f.Stat()
254256
closeErr := f.Close()
257+
// TODO(rost): Returning 405 Method Not Allowed might not be appropriate.
255258
if copyErr != nil {
256259
return http.StatusMethodNotAllowed, copyErr
257260
}
261+
if statErr != nil {
262+
return http.StatusMethodNotAllowed, statErr
263+
}
258264
if closeErr != nil {
259265
return http.StatusMethodNotAllowed, closeErr
260266
}
261-
pstats, err := props(h.FileSystem, h.LockSystem, r.URL.Path, []xml.Name{
262-
{Space: "DAV:", Local: "getetag"},
263-
})
267+
etag, err := findETag(h.FileSystem, h.LockSystem, r.URL.Path, fi)
264268
if err != nil {
265269
return http.StatusInternalServerError, err
266270
}
267-
writeDAVHeaders(w, pstats)
271+
w.Header().Set("ETag", etag)
268272
return http.StatusCreated, nil
269273
}
270274

@@ -557,26 +561,6 @@ func (h *Handler) handleProppatch(w http.ResponseWriter, r *http.Request) (statu
557561
return 0, nil
558562
}
559563

560-
// davHeaderNames maps the names of DAV properties to their corresponding
561-
// HTTP response headers.
562-
var davHeaderNames = map[xml.Name]string{
563-
xml.Name{Space: "DAV:", Local: "getetag"}: "ETag",
564-
xml.Name{Space: "DAV:", Local: "getcontenttype"}: "Content-Type",
565-
}
566-
567-
func writeDAVHeaders(w http.ResponseWriter, pstats []Propstat) {
568-
for _, pst := range pstats {
569-
if pst.Status == http.StatusOK {
570-
for _, p := range pst.Props {
571-
if n, ok := davHeaderNames[p.XMLName]; ok {
572-
w.Header().Set(n, string(p.InnerXML))
573-
}
574-
}
575-
break
576-
}
577-
}
578-
}
579-
580564
func makePropstatResponse(href string, pstats []Propstat) *response {
581565
resp := response{
582566
Href: []string{href},

0 commit comments

Comments
 (0)