|
6 | 6 | package webdav // import "golang.org/x/net/webdav"
|
7 | 7 |
|
8 | 8 | import (
|
9 |
| - "encoding/xml" |
10 | 9 | "errors"
|
11 | 10 | "fmt"
|
12 | 11 | "io"
|
@@ -203,14 +202,14 @@ func (h *Handler) handleGetHeadPost(w http.ResponseWriter, r *http.Request) (sta
|
203 | 202 | if err != nil {
|
204 | 203 | return http.StatusNotFound, err
|
205 | 204 | }
|
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) |
212 | 211 | }
|
213 |
| - writeDAVHeaders(w, pstats) |
| 212 | + // Let ServeContent determine the Content-Type header. |
214 | 213 | http.ServeContent(w, r, r.URL.Path, fi.ModTime(), f)
|
215 | 214 | return 0, nil
|
216 | 215 | }
|
@@ -245,26 +244,31 @@ func (h *Handler) handlePut(w http.ResponseWriter, r *http.Request) (status int,
|
245 | 244 | return status, err
|
246 | 245 | }
|
247 | 246 | defer release()
|
| 247 | + // TODO(rost): Support the If-Match, If-None-Match headers? See bradfitz' |
| 248 | + // comments in http.checkEtag. |
248 | 249 |
|
249 | 250 | f, err := h.FileSystem.OpenFile(r.URL.Path, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0666)
|
250 | 251 | if err != nil {
|
251 | 252 | return http.StatusNotFound, err
|
252 | 253 | }
|
253 | 254 | _, copyErr := io.Copy(f, r.Body)
|
| 255 | + fi, statErr := f.Stat() |
254 | 256 | closeErr := f.Close()
|
| 257 | + // TODO(rost): Returning 405 Method Not Allowed might not be appropriate. |
255 | 258 | if copyErr != nil {
|
256 | 259 | return http.StatusMethodNotAllowed, copyErr
|
257 | 260 | }
|
| 261 | + if statErr != nil { |
| 262 | + return http.StatusMethodNotAllowed, statErr |
| 263 | + } |
258 | 264 | if closeErr != nil {
|
259 | 265 | return http.StatusMethodNotAllowed, closeErr
|
260 | 266 | }
|
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) |
264 | 268 | if err != nil {
|
265 | 269 | return http.StatusInternalServerError, err
|
266 | 270 | }
|
267 |
| - writeDAVHeaders(w, pstats) |
| 271 | + w.Header().Set("ETag", etag) |
268 | 272 | return http.StatusCreated, nil
|
269 | 273 | }
|
270 | 274 |
|
@@ -557,26 +561,6 @@ func (h *Handler) handleProppatch(w http.ResponseWriter, r *http.Request) (statu
|
557 | 561 | return 0, nil
|
558 | 562 | }
|
559 | 563 |
|
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 |
| - |
580 | 564 | func makePropstatResponse(href string, pstats []Propstat) *response {
|
581 | 565 | resp := response{
|
582 | 566 | Href: []string{href},
|
|
0 commit comments