Spaces:
Running
Running
// Copyright 2014 The Go Authors. All rights reserved. | |
// Use of this source code is governed by a BSD-style | |
// license that can be found in the LICENSE file. | |
package webdav | |
import ( | |
"bytes" | |
"encoding/xml" | |
"fmt" | |
"io" | |
"net/http" | |
"net/http/httptest" | |
"reflect" | |
"sort" | |
"strings" | |
"testing" | |
ixml "github.com/alist-org/alist/v3/server/webdav/internal/xml" | |
) | |
func TestReadLockInfo(t *testing.T) { | |
// The "section x.y.z" test cases come from section x.y.z of the spec at | |
// http://www.webdav.org/specs/rfc4918.html | |
testCases := []struct { | |
desc string | |
input string | |
wantLI lockInfo | |
wantStatus int | |
}{{ | |
"bad: junk", | |
"xxx", | |
lockInfo{}, | |
http.StatusBadRequest, | |
}, { | |
"bad: invalid owner XML", | |
"" + | |
"<D:lockinfo xmlns:D='DAV:'>\n" + | |
" <D:lockscope><D:exclusive/></D:lockscope>\n" + | |
" <D:locktype><D:write/></D:locktype>\n" + | |
" <D:owner>\n" + | |
" <D:href> no end tag \n" + | |
" </D:owner>\n" + | |
"</D:lockinfo>", | |
lockInfo{}, | |
http.StatusBadRequest, | |
}, { | |
"bad: invalid UTF-8", | |
"" + | |
"<D:lockinfo xmlns:D='DAV:'>\n" + | |
" <D:lockscope><D:exclusive/></D:lockscope>\n" + | |
" <D:locktype><D:write/></D:locktype>\n" + | |
" <D:owner>\n" + | |
" <D:href> \xff </D:href>\n" + | |
" </D:owner>\n" + | |
"</D:lockinfo>", | |
lockInfo{}, | |
http.StatusBadRequest, | |
}, { | |
"bad: unfinished XML #1", | |
"" + | |
"<D:lockinfo xmlns:D='DAV:'>\n" + | |
" <D:lockscope><D:exclusive/></D:lockscope>\n" + | |
" <D:locktype><D:write/></D:locktype>\n", | |
lockInfo{}, | |
http.StatusBadRequest, | |
}, { | |
"bad: unfinished XML #2", | |
"" + | |
"<D:lockinfo xmlns:D='DAV:'>\n" + | |
" <D:lockscope><D:exclusive/></D:lockscope>\n" + | |
" <D:locktype><D:write/></D:locktype>\n" + | |
" <D:owner>\n", | |
lockInfo{}, | |
http.StatusBadRequest, | |
}, { | |
"good: empty", | |
"", | |
lockInfo{}, | |
0, | |
}, { | |
"good: plain-text owner", | |
"" + | |
"<D:lockinfo xmlns:D='DAV:'>\n" + | |
" <D:lockscope><D:exclusive/></D:lockscope>\n" + | |
" <D:locktype><D:write/></D:locktype>\n" + | |
" <D:owner>gopher</D:owner>\n" + | |
"</D:lockinfo>", | |
lockInfo{ | |
XMLName: ixml.Name{Space: "DAV:", Local: "lockinfo"}, | |
Exclusive: new(struct{}), | |
Write: new(struct{}), | |
Owner: owner{ | |
InnerXML: "gopher", | |
}, | |
}, | |
0, | |
}, { | |
"section 9.10.7", | |
"" + | |
"<D:lockinfo xmlns:D='DAV:'>\n" + | |
" <D:lockscope><D:exclusive/></D:lockscope>\n" + | |
" <D:locktype><D:write/></D:locktype>\n" + | |
" <D:owner>\n" + | |
" <D:href>http://example.org/~ejw/contact.html</D:href>\n" + | |
" </D:owner>\n" + | |
"</D:lockinfo>", | |
lockInfo{ | |
XMLName: ixml.Name{Space: "DAV:", Local: "lockinfo"}, | |
Exclusive: new(struct{}), | |
Write: new(struct{}), | |
Owner: owner{ | |
InnerXML: "\n <D:href>http://example.org/~ejw/contact.html</D:href>\n ", | |
}, | |
}, | |
0, | |
}} | |
for _, tc := range testCases { | |
li, status, err := readLockInfo(strings.NewReader(tc.input)) | |
if tc.wantStatus != 0 { | |
if err == nil { | |
t.Errorf("%s: got nil error, want non-nil", tc.desc) | |
continue | |
} | |
} else if err != nil { | |
t.Errorf("%s: %v", tc.desc, err) | |
continue | |
} | |
if !reflect.DeepEqual(li, tc.wantLI) || status != tc.wantStatus { | |
t.Errorf("%s:\ngot lockInfo=%v, status=%v\nwant lockInfo=%v, status=%v", | |
tc.desc, li, status, tc.wantLI, tc.wantStatus) | |
continue | |
} | |
} | |
} | |
func TestReadPropfind(t *testing.T) { | |
testCases := []struct { | |
desc string | |
input string | |
wantPF propfind | |
wantStatus int | |
}{{ | |
desc: "propfind: propname", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:propname/>\n" + | |
"</A:propfind>", | |
wantPF: propfind{ | |
XMLName: ixml.Name{Space: "DAV:", Local: "propfind"}, | |
Propname: new(struct{}), | |
}, | |
}, { | |
desc: "propfind: empty body means allprop", | |
input: "", | |
wantPF: propfind{ | |
Allprop: new(struct{}), | |
}, | |
}, { | |
desc: "propfind: allprop", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:allprop/>\n" + | |
"</A:propfind>", | |
wantPF: propfind{ | |
XMLName: ixml.Name{Space: "DAV:", Local: "propfind"}, | |
Allprop: new(struct{}), | |
}, | |
}, { | |
desc: "propfind: allprop followed by include", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:allprop/>\n" + | |
" <A:include><A:displayname/></A:include>\n" + | |
"</A:propfind>", | |
wantPF: propfind{ | |
XMLName: ixml.Name{Space: "DAV:", Local: "propfind"}, | |
Allprop: new(struct{}), | |
Include: propfindProps{xml.Name{Space: "DAV:", Local: "displayname"}}, | |
}, | |
}, { | |
desc: "propfind: include followed by allprop", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:include><A:displayname/></A:include>\n" + | |
" <A:allprop/>\n" + | |
"</A:propfind>", | |
wantPF: propfind{ | |
XMLName: ixml.Name{Space: "DAV:", Local: "propfind"}, | |
Allprop: new(struct{}), | |
Include: propfindProps{xml.Name{Space: "DAV:", Local: "displayname"}}, | |
}, | |
}, { | |
desc: "propfind: propfind", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:prop><A:displayname/></A:prop>\n" + | |
"</A:propfind>", | |
wantPF: propfind{ | |
XMLName: ixml.Name{Space: "DAV:", Local: "propfind"}, | |
Prop: propfindProps{xml.Name{Space: "DAV:", Local: "displayname"}}, | |
}, | |
}, { | |
desc: "propfind: prop with ignored comments", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:prop>\n" + | |
" <!-- ignore -->\n" + | |
" <A:displayname><!-- ignore --></A:displayname>\n" + | |
" </A:prop>\n" + | |
"</A:propfind>", | |
wantPF: propfind{ | |
XMLName: ixml.Name{Space: "DAV:", Local: "propfind"}, | |
Prop: propfindProps{xml.Name{Space: "DAV:", Local: "displayname"}}, | |
}, | |
}, { | |
desc: "propfind: propfind with ignored whitespace", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:prop> <A:displayname/></A:prop>\n" + | |
"</A:propfind>", | |
wantPF: propfind{ | |
XMLName: ixml.Name{Space: "DAV:", Local: "propfind"}, | |
Prop: propfindProps{xml.Name{Space: "DAV:", Local: "displayname"}}, | |
}, | |
}, { | |
desc: "propfind: propfind with ignored mixed-content", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:prop>foo<A:displayname/>bar</A:prop>\n" + | |
"</A:propfind>", | |
wantPF: propfind{ | |
XMLName: ixml.Name{Space: "DAV:", Local: "propfind"}, | |
Prop: propfindProps{xml.Name{Space: "DAV:", Local: "displayname"}}, | |
}, | |
}, { | |
desc: "propfind: propname with ignored element (section A.4)", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:propname/>\n" + | |
" <E:leave-out xmlns:E='E:'>*boss*</E:leave-out>\n" + | |
"</A:propfind>", | |
wantPF: propfind{ | |
XMLName: ixml.Name{Space: "DAV:", Local: "propfind"}, | |
Propname: new(struct{}), | |
}, | |
}, { | |
desc: "propfind: bad: junk", | |
input: "xxx", | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "propfind: bad: propname and allprop (section A.3)", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:propname/>" + | |
" <A:allprop/>" + | |
"</A:propfind>", | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "propfind: bad: propname and prop", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:prop><A:displayname/></A:prop>\n" + | |
" <A:propname/>\n" + | |
"</A:propfind>", | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "propfind: bad: allprop and prop", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:allprop/>\n" + | |
" <A:prop><A:foo/><A:/prop>\n" + | |
"</A:propfind>", | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "propfind: bad: empty propfind with ignored element (section A.4)", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <E:expired-props/>\n" + | |
"</A:propfind>", | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "propfind: bad: empty prop", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:prop/>\n" + | |
"</A:propfind>", | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "propfind: bad: prop with just chardata", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:prop>foo</A:prop>\n" + | |
"</A:propfind>", | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "bad: interrupted prop", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:prop><A:foo></A:prop>\n", | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "bad: malformed end element prop", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:prop><A:foo/></A:bar></A:prop>\n", | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "propfind: bad: property with chardata value", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:prop><A:foo>bar</A:foo></A:prop>\n" + | |
"</A:propfind>", | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "propfind: bad: property with whitespace value", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:prop><A:foo> </A:foo></A:prop>\n" + | |
"</A:propfind>", | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "propfind: bad: include without allprop", | |
input: "" + | |
"<A:propfind xmlns:A='DAV:'>\n" + | |
" <A:include><A:foo/></A:include>\n" + | |
"</A:propfind>", | |
wantStatus: http.StatusBadRequest, | |
}} | |
for _, tc := range testCases { | |
pf, status, err := readPropfind(strings.NewReader(tc.input)) | |
if tc.wantStatus != 0 { | |
if err == nil { | |
t.Errorf("%s: got nil error, want non-nil", tc.desc) | |
continue | |
} | |
} else if err != nil { | |
t.Errorf("%s: %v", tc.desc, err) | |
continue | |
} | |
if !reflect.DeepEqual(pf, tc.wantPF) || status != tc.wantStatus { | |
t.Errorf("%s:\ngot propfind=%v, status=%v\nwant propfind=%v, status=%v", | |
tc.desc, pf, status, tc.wantPF, tc.wantStatus) | |
continue | |
} | |
} | |
} | |
func TestMultistatusWriter(t *testing.T) { | |
///The "section x.y.z" test cases come from section x.y.z of the spec at | |
// http://www.webdav.org/specs/rfc4918.html | |
testCases := []struct { | |
desc string | |
responses []response | |
respdesc string | |
writeHeader bool | |
wantXML string | |
wantCode int | |
wantErr error | |
}{{ | |
desc: "section 9.2.2 (failed dependency)", | |
responses: []response{{ | |
Href: []string{"http://example.com/foo"}, | |
Propstat: []propstat{{ | |
Prop: []Property{{ | |
XMLName: xml.Name{ | |
Space: "http://ns.example.com/", | |
Local: "Authors", | |
}, | |
}}, | |
Status: "HTTP/1.1 424 Failed Dependency", | |
}, { | |
Prop: []Property{{ | |
XMLName: xml.Name{ | |
Space: "http://ns.example.com/", | |
Local: "Copyright-Owner", | |
}, | |
}}, | |
Status: "HTTP/1.1 409 Conflict", | |
}}, | |
ResponseDescription: "Copyright Owner cannot be deleted or altered.", | |
}}, | |
wantXML: `` + | |
`<?xml version="1.0" encoding="UTF-8"?>` + | |
`<multistatus xmlns="DAV:">` + | |
` <response>` + | |
` <href>http://example.com/foo</href>` + | |
` <propstat>` + | |
` <prop>` + | |
` <Authors xmlns="http://ns.example.com/"></Authors>` + | |
` </prop>` + | |
` <status>HTTP/1.1 424 Failed Dependency</status>` + | |
` </propstat>` + | |
` <propstat xmlns="DAV:">` + | |
` <prop>` + | |
` <Copyright-Owner xmlns="http://ns.example.com/"></Copyright-Owner>` + | |
` </prop>` + | |
` <status>HTTP/1.1 409 Conflict</status>` + | |
` </propstat>` + | |
` <responsedescription>Copyright Owner cannot be deleted or altered.</responsedescription>` + | |
`</response>` + | |
`</multistatus>`, | |
wantCode: StatusMulti, | |
}, { | |
desc: "section 9.6.2 (lock-token-submitted)", | |
responses: []response{{ | |
Href: []string{"http://example.com/foo"}, | |
Status: "HTTP/1.1 423 Locked", | |
Error: &xmlError{ | |
InnerXML: []byte(`<lock-token-submitted xmlns="DAV:"/>`), | |
}, | |
}}, | |
wantXML: `` + | |
`<?xml version="1.0" encoding="UTF-8"?>` + | |
`<multistatus xmlns="DAV:">` + | |
` <response>` + | |
` <href>http://example.com/foo</href>` + | |
` <status>HTTP/1.1 423 Locked</status>` + | |
` <error><lock-token-submitted xmlns="DAV:"/></error>` + | |
` </response>` + | |
`</multistatus>`, | |
wantCode: StatusMulti, | |
}, { | |
desc: "section 9.1.3", | |
responses: []response{{ | |
Href: []string{"http://example.com/foo"}, | |
Propstat: []propstat{{ | |
Prop: []Property{{ | |
XMLName: xml.Name{Space: "http://ns.example.com/boxschema/", Local: "bigbox"}, | |
InnerXML: []byte(`` + | |
`<BoxType xmlns="http://ns.example.com/boxschema/">` + | |
`Box type A` + | |
`</BoxType>`), | |
}, { | |
XMLName: xml.Name{Space: "http://ns.example.com/boxschema/", Local: "author"}, | |
InnerXML: []byte(`` + | |
`<Name xmlns="http://ns.example.com/boxschema/">` + | |
`J.J. Johnson` + | |
`</Name>`), | |
}}, | |
Status: "HTTP/1.1 200 OK", | |
}, { | |
Prop: []Property{{ | |
XMLName: xml.Name{Space: "http://ns.example.com/boxschema/", Local: "DingALing"}, | |
}, { | |
XMLName: xml.Name{Space: "http://ns.example.com/boxschema/", Local: "Random"}, | |
}}, | |
Status: "HTTP/1.1 403 Forbidden", | |
ResponseDescription: "The user does not have access to the DingALing property.", | |
}}, | |
}}, | |
respdesc: "There has been an access violation error.", | |
wantXML: `` + | |
`<?xml version="1.0" encoding="UTF-8"?>` + | |
`<multistatus xmlns="DAV:" xmlns:B="http://ns.example.com/boxschema/">` + | |
` <response>` + | |
` <href>http://example.com/foo</href>` + | |
` <propstat>` + | |
` <prop>` + | |
` <B:bigbox><B:BoxType>Box type A</B:BoxType></B:bigbox>` + | |
` <B:author><B:Name>J.J. Johnson</B:Name></B:author>` + | |
` </prop>` + | |
` <status>HTTP/1.1 200 OK</status>` + | |
` </propstat>` + | |
` <propstat>` + | |
` <prop>` + | |
` <B:DingALing/>` + | |
` <B:Random/>` + | |
` </prop>` + | |
` <status>HTTP/1.1 403 Forbidden</status>` + | |
` <responsedescription>The user does not have access to the DingALing property.</responsedescription>` + | |
` </propstat>` + | |
` </response>` + | |
` <responsedescription>There has been an access violation error.</responsedescription>` + | |
`</multistatus>`, | |
wantCode: StatusMulti, | |
}, { | |
desc: "no response written", | |
// default of http.responseWriter | |
wantCode: http.StatusOK, | |
}, { | |
desc: "no response written (with description)", | |
respdesc: "too bad", | |
// default of http.responseWriter | |
wantCode: http.StatusOK, | |
}, { | |
desc: "empty multistatus with header", | |
writeHeader: true, | |
wantXML: `<multistatus xmlns="DAV:"></multistatus>`, | |
wantCode: StatusMulti, | |
}, { | |
desc: "bad: no href", | |
responses: []response{{ | |
Propstat: []propstat{{ | |
Prop: []Property{{ | |
XMLName: xml.Name{ | |
Space: "http://example.com/", | |
Local: "foo", | |
}, | |
}}, | |
Status: "HTTP/1.1 200 OK", | |
}}, | |
}}, | |
wantErr: errInvalidResponse, | |
// default of http.responseWriter | |
wantCode: http.StatusOK, | |
}, { | |
desc: "bad: multiple hrefs and no status", | |
responses: []response{{ | |
Href: []string{"http://example.com/foo", "http://example.com/bar"}, | |
}}, | |
wantErr: errInvalidResponse, | |
// default of http.responseWriter | |
wantCode: http.StatusOK, | |
}, { | |
desc: "bad: one href and no propstat", | |
responses: []response{{ | |
Href: []string{"http://example.com/foo"}, | |
}}, | |
wantErr: errInvalidResponse, | |
// default of http.responseWriter | |
wantCode: http.StatusOK, | |
}, { | |
desc: "bad: status with one href and propstat", | |
responses: []response{{ | |
Href: []string{"http://example.com/foo"}, | |
Propstat: []propstat{{ | |
Prop: []Property{{ | |
XMLName: xml.Name{ | |
Space: "http://example.com/", | |
Local: "foo", | |
}, | |
}}, | |
Status: "HTTP/1.1 200 OK", | |
}}, | |
Status: "HTTP/1.1 200 OK", | |
}}, | |
wantErr: errInvalidResponse, | |
// default of http.responseWriter | |
wantCode: http.StatusOK, | |
}, { | |
desc: "bad: multiple hrefs and propstat", | |
responses: []response{{ | |
Href: []string{ | |
"http://example.com/foo", | |
"http://example.com/bar", | |
}, | |
Propstat: []propstat{{ | |
Prop: []Property{{ | |
XMLName: xml.Name{ | |
Space: "http://example.com/", | |
Local: "foo", | |
}, | |
}}, | |
Status: "HTTP/1.1 200 OK", | |
}}, | |
}}, | |
wantErr: errInvalidResponse, | |
// default of http.responseWriter | |
wantCode: http.StatusOK, | |
}} | |
n := xmlNormalizer{omitWhitespace: true} | |
loop: | |
for _, tc := range testCases { | |
rec := httptest.NewRecorder() | |
w := multistatusWriter{w: rec, responseDescription: tc.respdesc} | |
if tc.writeHeader { | |
if err := w.writeHeader(); err != nil { | |
t.Errorf("%s: got writeHeader error %v, want nil", tc.desc, err) | |
continue | |
} | |
} | |
for _, r := range tc.responses { | |
if err := w.write(&r); err != nil { | |
if err != tc.wantErr { | |
t.Errorf("%s: got write error %v, want %v", | |
tc.desc, err, tc.wantErr) | |
} | |
continue loop | |
} | |
} | |
if err := w.close(); err != tc.wantErr { | |
t.Errorf("%s: got close error %v, want %v", | |
tc.desc, err, tc.wantErr) | |
continue | |
} | |
if rec.Code != tc.wantCode { | |
t.Errorf("%s: got HTTP status code %d, want %d\n", | |
tc.desc, rec.Code, tc.wantCode) | |
continue | |
} | |
gotXML := rec.Body.String() | |
eq, err := n.equalXML(strings.NewReader(gotXML), strings.NewReader(tc.wantXML)) | |
if err != nil { | |
t.Errorf("%s: equalXML: %v", tc.desc, err) | |
continue | |
} | |
if !eq { | |
t.Errorf("%s: XML body\ngot %s\nwant %s", tc.desc, gotXML, tc.wantXML) | |
} | |
} | |
} | |
func TestReadProppatch(t *testing.T) { | |
ppStr := func(pps []Proppatch) string { | |
var outer []string | |
for _, pp := range pps { | |
var inner []string | |
for _, p := range pp.Props { | |
inner = append(inner, fmt.Sprintf("{XMLName: %q, Lang: %q, InnerXML: %q}", | |
p.XMLName, p.Lang, p.InnerXML)) | |
} | |
outer = append(outer, fmt.Sprintf("{Remove: %t, Props: [%s]}", | |
pp.Remove, strings.Join(inner, ", "))) | |
} | |
return "[" + strings.Join(outer, ", ") + "]" | |
} | |
testCases := []struct { | |
desc string | |
input string | |
wantPP []Proppatch | |
wantStatus int | |
}{{ | |
desc: "proppatch: section 9.2 (with simple property value)", | |
input: `` + | |
`<?xml version="1.0" encoding="utf-8" ?>` + | |
`<D:propertyupdate xmlns:D="DAV:"` + | |
` xmlns:Z="http://ns.example.com/z/">` + | |
` <D:set>` + | |
` <D:prop><Z:Authors>somevalue</Z:Authors></D:prop>` + | |
` </D:set>` + | |
` <D:remove>` + | |
` <D:prop><Z:Copyright-Owner/></D:prop>` + | |
` </D:remove>` + | |
`</D:propertyupdate>`, | |
wantPP: []Proppatch{{ | |
Props: []Property{{ | |
xml.Name{Space: "http://ns.example.com/z/", Local: "Authors"}, | |
"", | |
[]byte(`somevalue`), | |
}}, | |
}, { | |
Remove: true, | |
Props: []Property{{ | |
xml.Name{Space: "http://ns.example.com/z/", Local: "Copyright-Owner"}, | |
"", | |
nil, | |
}}, | |
}}, | |
}, { | |
desc: "proppatch: lang attribute on prop", | |
input: `` + | |
`<?xml version="1.0" encoding="utf-8" ?>` + | |
`<D:propertyupdate xmlns:D="DAV:">` + | |
` <D:set>` + | |
` <D:prop xml:lang="en">` + | |
` <foo xmlns="http://example.com/ns"/>` + | |
` </D:prop>` + | |
` </D:set>` + | |
`</D:propertyupdate>`, | |
wantPP: []Proppatch{{ | |
Props: []Property{{ | |
xml.Name{Space: "http://example.com/ns", Local: "foo"}, | |
"en", | |
nil, | |
}}, | |
}}, | |
}, { | |
desc: "bad: remove with value", | |
input: `` + | |
`<?xml version="1.0" encoding="utf-8" ?>` + | |
`<D:propertyupdate xmlns:D="DAV:"` + | |
` xmlns:Z="http://ns.example.com/z/">` + | |
` <D:remove>` + | |
` <D:prop>` + | |
` <Z:Authors>` + | |
` <Z:Author>Jim Whitehead</Z:Author>` + | |
` </Z:Authors>` + | |
` </D:prop>` + | |
` </D:remove>` + | |
`</D:propertyupdate>`, | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "bad: empty propertyupdate", | |
input: `` + | |
`<?xml version="1.0" encoding="utf-8" ?>` + | |
`<D:propertyupdate xmlns:D="DAV:"` + | |
`</D:propertyupdate>`, | |
wantStatus: http.StatusBadRequest, | |
}, { | |
desc: "bad: empty prop", | |
input: `` + | |
`<?xml version="1.0" encoding="utf-8" ?>` + | |
`<D:propertyupdate xmlns:D="DAV:"` + | |
` xmlns:Z="http://ns.example.com/z/">` + | |
` <D:remove>` + | |
` <D:prop/>` + | |
` </D:remove>` + | |
`</D:propertyupdate>`, | |
wantStatus: http.StatusBadRequest, | |
}} | |
for _, tc := range testCases { | |
pp, status, err := readProppatch(strings.NewReader(tc.input)) | |
if tc.wantStatus != 0 { | |
if err == nil { | |
t.Errorf("%s: got nil error, want non-nil", tc.desc) | |
continue | |
} | |
} else if err != nil { | |
t.Errorf("%s: %v", tc.desc, err) | |
continue | |
} | |
if status != tc.wantStatus { | |
t.Errorf("%s: got status %d, want %d", tc.desc, status, tc.wantStatus) | |
continue | |
} | |
if !reflect.DeepEqual(pp, tc.wantPP) || status != tc.wantStatus { | |
t.Errorf("%s: proppatch\ngot %v\nwant %v", tc.desc, ppStr(pp), ppStr(tc.wantPP)) | |
} | |
} | |
} | |
func TestUnmarshalXMLValue(t *testing.T) { | |
testCases := []struct { | |
desc string | |
input string | |
wantVal string | |
}{{ | |
desc: "simple char data", | |
input: "<root>foo</root>", | |
wantVal: "foo", | |
}, { | |
desc: "empty element", | |
input: "<root><foo/></root>", | |
wantVal: "<foo/>", | |
}, { | |
desc: "preserve namespace", | |
input: `<root><foo xmlns="bar"/></root>`, | |
wantVal: `<foo xmlns="bar"/>`, | |
}, { | |
desc: "preserve root element namespace", | |
input: `<root xmlns:bar="bar"><bar:foo/></root>`, | |
wantVal: `<foo xmlns="bar"/>`, | |
}, { | |
desc: "preserve whitespace", | |
input: "<root> \t </root>", | |
wantVal: " \t ", | |
}, { | |
desc: "preserve mixed content", | |
input: `<root xmlns="bar"> <foo>a<bam xmlns="baz"/> </foo> </root>`, | |
wantVal: ` <foo xmlns="bar">a<bam xmlns="baz"/> </foo> `, | |
}, { | |
desc: "section 9.2", | |
input: `` + | |
`<Z:Authors xmlns:Z="http://ns.example.com/z/">` + | |
` <Z:Author>Jim Whitehead</Z:Author>` + | |
` <Z:Author>Roy Fielding</Z:Author>` + | |
`</Z:Authors>`, | |
wantVal: `` + | |
` <Author xmlns="http://ns.example.com/z/">Jim Whitehead</Author>` + | |
` <Author xmlns="http://ns.example.com/z/">Roy Fielding</Author>`, | |
}, { | |
desc: "section 4.3.1 (mixed content)", | |
input: `` + | |
`<x:author ` + | |
` xmlns:x='http://example.com/ns' ` + | |
` xmlns:D="DAV:">` + | |
` <x:name>Jane Doe</x:name>` + | |
` <!-- Jane's contact info -->` + | |
` <x:uri type='email'` + | |
` added='2005-11-26'>mailto:jane.doe@example.com</x:uri>` + | |
` <x:uri type='web'` + | |
` added='2005-11-27'>http://www.example.com</x:uri>` + | |
` <x:notes xmlns:h='http://www.w3.org/1999/xhtml'>` + | |
` Jane has been working way <h:em>too</h:em> long on the` + | |
` long-awaited revision of <![CDATA[<RFC2518>]]>.` + | |
` </x:notes>` + | |
`</x:author>`, | |
wantVal: `` + | |
` <name xmlns="http://example.com/ns">Jane Doe</name>` + | |
` ` + | |
` <uri type='email'` + | |
` xmlns="http://example.com/ns" ` + | |
` added='2005-11-26'>mailto:jane.doe@example.com</uri>` + | |
` <uri added='2005-11-27'` + | |
` type='web'` + | |
` xmlns="http://example.com/ns">http://www.example.com</uri>` + | |
` <notes xmlns="http://example.com/ns" ` + | |
` xmlns:h="http://www.w3.org/1999/xhtml">` + | |
` Jane has been working way <h:em>too</h:em> long on the` + | |
` long-awaited revision of <RFC2518>.` + | |
` </notes>`, | |
}} | |
var n xmlNormalizer | |
for _, tc := range testCases { | |
d := ixml.NewDecoder(strings.NewReader(tc.input)) | |
var v xmlValue | |
if err := d.Decode(&v); err != nil { | |
t.Errorf("%s: got error %v, want nil", tc.desc, err) | |
continue | |
} | |
eq, err := n.equalXML(bytes.NewReader(v), strings.NewReader(tc.wantVal)) | |
if err != nil { | |
t.Errorf("%s: equalXML: %v", tc.desc, err) | |
continue | |
} | |
if !eq { | |
t.Errorf("%s:\ngot %s\nwant %s", tc.desc, string(v), tc.wantVal) | |
} | |
} | |
} | |
// xmlNormalizer normalizes XML. | |
type xmlNormalizer struct { | |
// omitWhitespace instructs to ignore whitespace between element tags. | |
omitWhitespace bool | |
// omitComments instructs to ignore XML comments. | |
omitComments bool | |
} | |
// normalize writes the normalized XML content of r to w. It applies the | |
// following rules | |
// | |
// - Rename namespace prefixes according to an internal heuristic. | |
// - Remove unnecessary namespace declarations. | |
// - Sort attributes in XML start elements in lexical order of their | |
// fully qualified name. | |
// - Remove XML directives and processing instructions. | |
// - Remove CDATA between XML tags that only contains whitespace, if | |
// instructed to do so. | |
// - Remove comments, if instructed to do so. | |
func (n *xmlNormalizer) normalize(w io.Writer, r io.Reader) error { | |
d := ixml.NewDecoder(r) | |
e := ixml.NewEncoder(w) | |
for { | |
t, err := d.Token() | |
if err != nil { | |
if t == nil && err == io.EOF { | |
break | |
} | |
return err | |
} | |
switch val := t.(type) { | |
case ixml.Directive, ixml.ProcInst: | |
continue | |
case ixml.Comment: | |
if n.omitComments { | |
continue | |
} | |
case ixml.CharData: | |
if n.omitWhitespace && len(bytes.TrimSpace(val)) == 0 { | |
continue | |
} | |
case ixml.StartElement: | |
start, _ := ixml.CopyToken(val).(ixml.StartElement) | |
attr := start.Attr[:0] | |
for _, a := range start.Attr { | |
if a.Name.Space == "xmlns" || a.Name.Local == "xmlns" { | |
continue | |
} | |
attr = append(attr, a) | |
} | |
sort.Sort(byName(attr)) | |
start.Attr = attr | |
t = start | |
} | |
err = e.EncodeToken(t) | |
if err != nil { | |
return err | |
} | |
} | |
return e.Flush() | |
} | |
// equalXML tests for equality of the normalized XML contents of a and b. | |
func (n *xmlNormalizer) equalXML(a, b io.Reader) (bool, error) { | |
var buf bytes.Buffer | |
if err := n.normalize(&buf, a); err != nil { | |
return false, err | |
} | |
normA := buf.String() | |
buf.Reset() | |
if err := n.normalize(&buf, b); err != nil { | |
return false, err | |
} | |
normB := buf.String() | |
return normA == normB, nil | |
} | |
type byName []ixml.Attr | |
func (a byName) Len() int { return len(a) } | |
func (a byName) Swap(i, j int) { a[i], a[j] = a[j], a[i] } | |
func (a byName) Less(i, j int) bool { | |
if a[i].Name.Space != a[j].Name.Space { | |
return a[i].Name.Space < a[j].Name.Space | |
} | |
return a[i].Name.Local < a[j].Name.Local | |
} | |