0
Fork 0
mirror of https://github.com/willnorris/imageproxy.git synced 2024-12-16 21:56:43 -05:00
imageproxy/data.go

312 lines
10 KiB
Go
Raw Normal View History

// Copyright 2013 Google Inc. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package imageproxy
import (
2013-12-26 15:50:22 -05:00
"bytes"
"fmt"
2013-12-26 17:38:15 -05:00
"net/http"
"net/url"
"regexp"
"strconv"
"strings"
)
const (
2015-05-11 21:36:42 -05:00
optFit = "fit"
optFlipVertical = "fv"
optFlipHorizontal = "fh"
optFormatJPEG = "jpeg"
optFormatPNG = "png"
2015-05-11 21:36:42 -05:00
optRotatePrefix = "r"
optQualityPrefix = "q"
optSignaturePrefix = "s"
optSizeDelimiter = "x"
optScaleUp = "scaleUp"
)
2013-12-26 21:31:20 -05:00
// URLError reports a malformed URL error.
type URLError struct {
Message string
URL *url.URL
}
func (e URLError) Error() string {
return fmt.Sprintf("malformed URL %q: %s", e.URL, e.Message)
}
// Options specifies transformations to be performed on the requested image.
2013-12-05 02:12:44 -05:00
type Options struct {
// See ParseOptions for interpretation of Width and Height values
Width float64
Height float64
// If true, resize the image to fit in the specified dimensions. Image
// will not be cropped, and aspect ratio will be maintained.
Fit bool
2013-12-06 21:03:16 -05:00
// Rotate image the specified degrees counter-clockwise. Valid values
// are 90, 180, 270.
2013-12-06 21:03:16 -05:00
Rotate int
FlipVertical bool
FlipHorizontal bool
// Quality of output image
Quality int
2015-05-11 21:36:42 -05:00
// HMAC Signature for signed requests.
Signature string
2015-08-12 13:39:38 -05:00
// Allow image to scale beyond its original dimensions. This value
// will always be overwritten by the value of Proxy.ScaleUp.
2015-08-12 13:39:38 -05:00
ScaleUp bool
// Desired image format. Valid values are "jpeg", "png".
Format string
}
2013-12-05 02:12:44 -05:00
func (o Options) String() string {
2013-12-26 15:50:22 -05:00
buf := new(bytes.Buffer)
fmt.Fprintf(buf, "%v%s%v", o.Width, optSizeDelimiter, o.Height)
2013-12-26 15:50:22 -05:00
if o.Fit {
fmt.Fprintf(buf, ",%s", optFit)
2013-12-26 15:50:22 -05:00
}
if o.Rotate != 0 {
fmt.Fprintf(buf, ",%s%d", string(optRotatePrefix), o.Rotate)
2013-12-26 15:50:22 -05:00
}
if o.FlipVertical {
fmt.Fprintf(buf, ",%s", optFlipVertical)
2013-12-26 15:50:22 -05:00
}
if o.FlipHorizontal {
fmt.Fprintf(buf, ",%s", optFlipHorizontal)
2013-12-26 15:50:22 -05:00
}
2015-05-11 21:29:25 -05:00
if o.Quality != 0 {
fmt.Fprintf(buf, ",%s%d", string(optQualityPrefix), o.Quality)
}
2015-05-11 21:36:42 -05:00
if o.Signature != "" {
fmt.Fprintf(buf, ",%s%s", string(optSignaturePrefix), o.Signature)
}
if o.ScaleUp {
fmt.Fprintf(buf, ",%s", optScaleUp)
}
if o.Format != "" {
fmt.Fprintf(buf, ",%s", o.Format)
}
2013-12-26 15:50:22 -05:00
return buf.String()
}
// transform returns whether o includes transformation options. Some fields
// are not transform related at all (like Signature), and others only apply in
// the presence of other fields (like Fit). A non-empty Format value is
// assumed to involve a transformation.
func (o Options) transform() bool {
return o.Width != 0 || o.Height != 0 || o.Rotate != 0 || o.FlipHorizontal || o.FlipVertical || o.Quality != 0 || o.Format != ""
}
// ParseOptions parses str as a list of comma separated transformation options.
// The options can be specified in in order, with duplicate options overwriting
// previous values.
//
// Size and Cropping
//
// The size option takes the general form "{width}x{height}", where width and
// height are numbers. Integer values greater than 1 are interpreted as exact
// pixel values. Floats between 0 and 1 are interpreted as percentages of the
// original image size. If either value is omitted or set to 0, it will be
// automatically set to preserve the aspect ratio based on the other dimension.
// If a single number is provided (with no "x" separator), it will be used for
// both height and width.
//
// Depending on the size options specified, an image may be cropped to fit the
// requested size. In all cases, the original aspect ratio of the image will be
// preserved; imageproxy will never stretch the original image.
//
// When no explicit crop mode is specified, the following rules are followed:
//
// - If both width and height values are specified, the image will be scaled to
// fill the space, cropping if necessary to fit the exact dimension.
//
// - If only one of the width or height values is specified, the image will be
// resized to fit the specified dimension, scaling the other dimension as
// needed to maintain the aspect ratio.
//
// If the "fit" option is specified together with a width and height value, the
// image will be resized to fit within a containing box of the specified size.
// As always, the original aspect ratio will be preserved. Specifying the "fit"
// option with only one of either width or height does the same thing as if
// "fit" had not been specified.
//
// Rotation and Flips
//
// The "r{degrees}" option will rotate the image the specified number of
// degrees, counter-clockwise. Valid degrees values are 90, 180, and 270.
//
// The "fv" option will flip the image vertically. The "fh" option will flip
// the image horizontally. Images are flipped after being rotated.
//
// Quality
//
// The "q{qualityPercentage}" option can be used to specify the quality of the
// output file (JPEG only). If not specified, the default value of "95" is used.
//
// Format
//
// The "jpeg" and "png" options can be used to specify the desired image format
// of the proxied image.
//
// Signature
//
// The "s{signature}" option specifies an optional base64 encoded HMAC used to
// sign the remote URL in the request. The HMAC key used to verify signatures is
// provided to the imageproxy server on startup.
//
// See https://github.com/willnorris/imageproxy/wiki/URL-signing
// for examples of generating signatures.
//
// Examples
//
// 0x0 - no resizing
// 200x - 200 pixels wide, proportional height
// 0.15x - 15% original width, proportional height
// x100 - 100 pixels tall, proportional width
// 100x150 - 100 by 150 pixels, cropping as needed
// 100 - 100 pixels square, cropping as needed
// 150,fit - scale to fit 150 pixels square, no cropping
// 100,r90 - 100 pixels square, rotated 90 degrees
// 100,fv,fh - 100 pixels square, flipped horizontal and vertical
// 200x,q80 - 200 pixels wide, proportional height, 80% quality
// 200x,png - 200 pixels wide, converted to PNG format
func ParseOptions(str string) Options {
2015-05-04 12:26:22 -05:00
var options Options
for _, opt := range strings.Split(str, ",") {
switch {
case len(opt) == 0:
break
case opt == optFit:
options.Fit = true
case opt == optFlipVertical:
options.FlipVertical = true
case opt == optFlipHorizontal:
options.FlipHorizontal = true
case opt == optScaleUp: // this option is intentionally not documented above
options.ScaleUp = true
case opt == optFormatJPEG, opt == optFormatPNG:
options.Format = opt
case strings.HasPrefix(opt, optRotatePrefix):
value := strings.TrimPrefix(opt, optRotatePrefix)
options.Rotate, _ = strconv.Atoi(value)
case strings.HasPrefix(opt, optQualityPrefix):
value := strings.TrimPrefix(opt, optQualityPrefix)
options.Quality, _ = strconv.Atoi(value)
2015-05-11 21:36:42 -05:00
case strings.HasPrefix(opt, optSignaturePrefix):
options.Signature = strings.TrimPrefix(opt, optSignaturePrefix)
case strings.Contains(opt, optSizeDelimiter):
size := strings.SplitN(opt, optSizeDelimiter, 2)
if w := size[0]; w != "" {
options.Width, _ = strconv.ParseFloat(w, 64)
}
if h := size[1]; h != "" {
options.Height, _ = strconv.ParseFloat(h, 64)
}
default:
if size, err := strconv.ParseFloat(opt, 64); err == nil {
options.Width = size
options.Height = size
}
}
}
return options
}
// Request is an imageproxy request which includes a remote URL of an image to
// proxy, and an optional set of transformations to perform.
type Request struct {
2015-06-14 03:26:40 -05:00
URL *url.URL // URL of the image to proxy
Options Options // Image transformation to perform
Original *http.Request // The original HTTP request
}
2015-05-11 19:12:03 -05:00
// String returns the request URL as a string, with r.Options encoded in the
// URL fragment.
func (r Request) String() string {
u := *r.URL
u.Fragment = r.Options.String()
return u.String()
}
// NewRequest parses an http.Request into an imageproxy Request. Options and
// the remote image URL are specified in the request path, formatted as:
// /{options}/{remote_url}. Options may be omitted, so a request path may
// simply contian /{remote_url}. The remote URL must be an absolute "http" or
// "https" URL, should not be URL encoded, and may contain a query string.
//
// Assuming an imageproxy server running on localhost, the following are all
// valid imageproxy requests:
//
// http://localhost/100x200/http://example.com/image.jpg
// http://localhost/100x200,r90/http://example.com/image.jpg?foo=bar
// http://localhost//http://example.com/image.jpg
// http://localhost/http://example.com/image.jpg
func NewRequest(r *http.Request, baseURL *url.URL) (*Request, error) {
2013-12-26 17:38:15 -05:00
var err error
2015-06-14 03:26:40 -05:00
req := &Request{Original: r}
2013-12-26 17:38:15 -05:00
path := r.URL.Path[1:] // strip leading slash
2016-05-26 15:45:59 -05:00
req.URL, err = parseURL(path)
2013-12-26 17:38:15 -05:00
if err != nil || !req.URL.IsAbs() {
// first segment should be options
2013-12-26 17:38:15 -05:00
parts := strings.SplitN(path, "/", 2)
if len(parts) != 2 {
return nil, URLError{"too few path segments", r.URL}
}
2015-05-04 12:26:22 -05:00
var err error
2016-05-26 15:45:59 -05:00
req.URL, err = parseURL(parts[1])
2013-12-26 17:38:15 -05:00
if err != nil {
return nil, URLError{fmt.Sprintf("unable to parse remote URL: %v", err), r.URL}
}
req.Options = ParseOptions(parts[0])
}
if baseURL != nil {
req.URL = baseURL.ResolveReference(req.URL)
}
2013-12-26 17:38:15 -05:00
if !req.URL.IsAbs() {
return nil, URLError{"must provide absolute remote URL", r.URL}
}
if req.URL.Scheme != "http" && req.URL.Scheme != "https" {
return nil, URLError{"remote URL must have http or https scheme", r.URL}
2013-12-26 17:38:15 -05:00
}
// query string is always part of the remote URL
req.URL.RawQuery = r.URL.RawQuery
return req, nil
}
2016-05-26 15:45:59 -05:00
var reCleanedURL = regexp.MustCompile(`^(https?):/+([^/])`)
// parseURL parses s as a URL, handling URLs that have been munged by
// path.Clean or a webserver that collapses multiple slashes.
func parseURL(s string) (*url.URL, error) {
s = reCleanedURL.ReplaceAllString(s, "$1://$2")
return url.Parse(s)
}