0
Fork 0
mirror of https://github.com/willnorris/imageproxy.git synced 2025-01-13 22:51:38 -05:00

requested refactor

This commit is contained in:
Christopher Brown 2018-06-22 16:34:46 -05:00
parent 67342bf7ac
commit b0bdbd9b5e
2 changed files with 29 additions and 42 deletions

View file

@ -168,13 +168,13 @@ func (p *Proxy) serveImage(w http.ResponseWriter, r *http.Request) {
return
}
if contentType := p.allowedContentType(resp.Header.Get("Content-Type")); contentType != "" {
w.Header().Set("Content-Type", contentType)
w.Header().Set("X-Content-Type-Options", "nosniff")
} else {
contentType, _, _ := mime.ParseMediaType(resp.Header.Get("Content-Type"))
if !validContentType(p.ContentTypes, contentType) {
http.Error(w, "forbidden content-type", http.StatusForbidden)
return
}
w.Header().Set("Content-Type", contentType)
w.Header().Set("X-Content-Type-Options", "nosniff")
copyHeader(w.Header(), resp.Header, "Content-Length")
@ -225,16 +225,11 @@ func (p *Proxy) allowed(r *Request) error {
return fmt.Errorf("request does not contain an allowed host or valid signature: %v", r)
}
// allowedContentType returns an allowed content type string to use in responses or "" if the
// content type cannot be used.
func (p *Proxy) allowedContentType(contentType string) string {
mediaType, _, _ := mime.ParseMediaType(contentType)
if mediaType == "" {
return ""
}
if len(p.ContentTypes) == 0 {
switch mediaType {
// validContentType returns whether contentType matches one of the patterns. If no patterns are
// given, validContentType returns true if contentType matches one of the whitelisted image types.
func validContentType(patterns []string, contentType string) bool {
if len(patterns) == 0 {
switch contentType {
case "image/bmp", "image/cgm", "image/g3fax", "image/gif", "image/ief", "image/jp2",
"image/jpeg", "image/jpg", "image/pict", "image/png", "image/prs.btif", "image/svg+xml",
"image/tiff", "image/vnd.adobe.photoshop", "image/vnd.djvu", "image/vnd.dwg",
@ -246,18 +241,18 @@ func (p *Proxy) allowedContentType(contentType string) string {
"image/x-portable-bitmap", "image/x-portable-graymap", "image/x-portable-pixmap",
"image/x-quicktime", "image/x-rgb", "image/x-xbitmap", "image/x-xpixmap",
"image/x-xwindowdump":
return mediaType
return true
}
return ""
return false
}
for _, pattern := range p.ContentTypes {
if ok, err := filepath.Match(pattern, mediaType); ok && err == nil {
return mediaType
for _, pattern := range patterns {
if ok, err := filepath.Match(pattern, contentType); ok && err == nil {
return true
}
}
return ""
return false
}
// validHost returns whether the host in u matches one of hosts.

View file

@ -412,36 +412,28 @@ func TestTransformingTransport(t *testing.T) {
}
}
func TestAllowedContentType(t *testing.T) {
p := &Proxy{}
for contentType, expected := range map[string]string{
"": "",
"image/png": "image/png",
"image/PNG": "image/png",
"image/PNG; foo=bar": "image/png",
"text/html": "",
func TestValidContentType(t *testing.T) {
for contentType, expected := range map[string]bool{
"": false,
"image/png": true,
"text/html": false,
} {
actual := p.allowedContentType(contentType)
actual := validContentType(nil, contentType)
if actual != expected {
t.Errorf("got %v, expected %v for content type: %v", actual, expected, contentType)
}
}
}
func TestAllowedContentType_Whitelist(t *testing.T) {
p := &Proxy{
ContentTypes: []string{"foo/*", "bar/baz"},
}
for contentType, expected := range map[string]string{
"": "",
"image/png": "",
"foo/asdf": "foo/asdf",
"bar/baz": "bar/baz",
"bar/bazz": "",
func TestValidContentType_Patterns(t *testing.T) {
for contentType, expected := range map[string]bool{
"": false,
"image/png": false,
"foo/asdf": true,
"bar/baz": true,
"bar/bazz": false,
} {
actual := p.allowedContentType(contentType)
actual := validContentType([]string{"foo/*", "bar/baz"}, contentType)
if actual != expected {
t.Errorf("got %v, expected %v for content type: %v", actual, expected, contentType)
}