Skip to content

Commit d75b190

Browse files
mattnnigeltao
authored andcommitted
webdav: fix props for directory
Fixes golang/go#13126 - GET/HEAD/POST method should not allowed for directory - Ignore EOFs while detecting the file. Change-Id: I5ce345408a5ea4ec0dc993631122e6c79fc64398 Reviewed-on: https://go-review.googlesource.com/16799 Reviewed-by: Robert Stepanek <robert.stepanek@gmail.com> Reviewed-by: Nigel Tao <nigeltao@golang.org>
1 parent b0e2f54 commit d75b190

File tree

3 files changed

+24
-31
lines changed

3 files changed

+24
-31
lines changed

webdav/prop.go

+16-12
Original file line numberDiff line numberDiff line change
@@ -115,23 +115,23 @@ var liveProps = map[xml.Name]struct {
115115
},
116116
xml.Name{Space: "DAV:", Local: "getcontentlength"}: {
117117
findFn: findContentLength,
118-
dir: true,
118+
dir: false,
119119
},
120120
xml.Name{Space: "DAV:", Local: "getlastmodified"}: {
121121
findFn: findLastModified,
122-
dir: true,
122+
dir: false,
123123
},
124124
xml.Name{Space: "DAV:", Local: "creationdate"}: {
125125
findFn: nil,
126-
dir: true,
126+
dir: false,
127127
},
128128
xml.Name{Space: "DAV:", Local: "getcontentlanguage"}: {
129129
findFn: nil,
130-
dir: true,
130+
dir: false,
131131
},
132132
xml.Name{Space: "DAV:", Local: "getcontenttype"}: {
133133
findFn: findContentType,
134-
dir: true,
134+
dir: false,
135135
},
136136
xml.Name{Space: "DAV:", Local: "getetag"}: {
137137
findFn: findETag,
@@ -358,14 +358,18 @@ func findContentType(fs FileSystem, ls LockSystem, name string, fi os.FileInfo)
358358
defer f.Close()
359359
// This implementation is based on serveContent's code in the standard net/http package.
360360
ctype := mime.TypeByExtension(filepath.Ext(name))
361-
if ctype == "" {
362-
// Read a chunk to decide between utf-8 text and binary.
363-
var buf [512]byte
364-
n, _ := io.ReadFull(f, buf[:])
365-
ctype = http.DetectContentType(buf[:n])
366-
// Rewind file.
367-
_, err = f.Seek(0, os.SEEK_SET)
361+
if ctype != "" {
362+
return ctype, nil
363+
}
364+
// Read a chunk to decide between utf-8 text and binary.
365+
var buf [512]byte
366+
n, err := io.ReadFull(f, buf[:])
367+
if err != nil && err != io.EOF && err != io.ErrUnexpectedEOF {
368+
return "", err
368369
}
370+
ctype = http.DetectContentType(buf[:n])
371+
// Rewind file.
372+
_, err = f.Seek(0, os.SEEK_SET)
369373
return ctype, err
370374
}
371375

webdav/prop_test.go

-12
Original file line numberDiff line numberDiff line change
@@ -77,9 +77,6 @@ func TestMemPS(t *testing.T) {
7777
wantPnames: []xml.Name{
7878
xml.Name{Space: "DAV:", Local: "resourcetype"},
7979
xml.Name{Space: "DAV:", Local: "displayname"},
80-
xml.Name{Space: "DAV:", Local: "getcontentlength"},
81-
xml.Name{Space: "DAV:", Local: "getlastmodified"},
82-
xml.Name{Space: "DAV:", Local: "getcontenttype"},
8380
xml.Name{Space: "DAV:", Local: "supportedlock"},
8481
},
8582
}, {
@@ -109,15 +106,6 @@ func TestMemPS(t *testing.T) {
109106
}, {
110107
XMLName: xml.Name{Space: "DAV:", Local: "displayname"},
111108
InnerXML: []byte("dir"),
112-
}, {
113-
XMLName: xml.Name{Space: "DAV:", Local: "getcontentlength"},
114-
InnerXML: []byte("0"),
115-
}, {
116-
XMLName: xml.Name{Space: "DAV:", Local: "getlastmodified"},
117-
InnerXML: nil, // Calculated during test.
118-
}, {
119-
XMLName: xml.Name{Space: "DAV:", Local: "getcontenttype"},
120-
InnerXML: []byte("text/plain; charset=utf-8"),
121109
}, {
122110
XMLName: xml.Name{Space: "DAV:", Local: "supportedlock"},
123111
InnerXML: []byte(lockEntry),

webdav/webdav.go

+8-7
Original file line numberDiff line numberDiff line change
@@ -195,7 +195,7 @@ func (h *Handler) handleOptions(w http.ResponseWriter, r *http.Request) (status
195195
allow := "OPTIONS, LOCK, PUT, MKCOL"
196196
if fi, err := h.FileSystem.Stat(reqPath); err == nil {
197197
if fi.IsDir() {
198-
allow = "OPTIONS, LOCK, GET, HEAD, POST, DELETE, PROPPATCH, COPY, MOVE, UNLOCK, PROPFIND"
198+
allow = "OPTIONS, LOCK, DELETE, PROPPATCH, COPY, MOVE, UNLOCK, PROPFIND"
199199
} else {
200200
allow = "OPTIONS, LOCK, GET, HEAD, POST, DELETE, PROPPATCH, COPY, MOVE, UNLOCK, PROPFIND, PUT"
201201
}
@@ -223,13 +223,14 @@ func (h *Handler) handleGetHeadPost(w http.ResponseWriter, r *http.Request) (sta
223223
if err != nil {
224224
return http.StatusNotFound, err
225225
}
226-
if !fi.IsDir() {
227-
etag, err := findETag(h.FileSystem, h.LockSystem, reqPath, fi)
228-
if err != nil {
229-
return http.StatusInternalServerError, err
230-
}
231-
w.Header().Set("ETag", etag)
226+
if fi.IsDir() {
227+
return http.StatusMethodNotAllowed, nil
228+
}
229+
etag, err := findETag(h.FileSystem, h.LockSystem, reqPath, fi)
230+
if err != nil {
231+
return http.StatusInternalServerError, err
232232
}
233+
w.Header().Set("ETag", etag)
233234
// Let ServeContent determine the Content-Type header.
234235
http.ServeContent(w, r, reqPath, fi.ModTime(), f)
235236
return 0, nil

0 commit comments

Comments
 (0)