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

add support for tiered caches

The -cache flag can be specified multiple times to create a tier of
caches.  Typically this would be used to be put a small in-memory cache
in front of a slower on-disk cache.

Fixes #111
This commit is contained in:
Will Norris 2017-09-17 09:55:41 +00:00
parent a57047ff22
commit 13409fd7c6
2 changed files with 47 additions and 13 deletions

View file

@ -129,6 +129,16 @@ version.
[codercat URL]: http://localhost:8080/500/https://octodex.github.com/images/codercat.jpg [codercat URL]: http://localhost:8080/500/https://octodex.github.com/images/codercat.jpg
If the `-cache` flag is specified multiple times, multiple caches will be
created in a [tiered fashion][]. Typically this is used to put a smaller and
faster in-memory cache in front of a larger but slower on-disk cache. For
example, the following will first check an in-memory cache for an image,
followed by a gcs bucket:
imageproxy -cache memory -cache gcs://my-bucket/
[tiered fashion]: https://godoc.org/github.com/die-net/lrucache/twotier
### Referrer Whitelist ### ### Referrer Whitelist ###
You can limit images to only be accessible for certain hosts in the HTTP You can limit images to only be accessible for certain hosts in the HTTP

View file

@ -29,6 +29,7 @@ import (
"github.com/PaulARoy/azurestoragecache" "github.com/PaulARoy/azurestoragecache"
"github.com/die-net/lrucache" "github.com/die-net/lrucache"
"github.com/die-net/lrucache/twotier"
"github.com/diegomarangoni/gcscache" "github.com/diegomarangoni/gcscache"
"github.com/garyburd/redigo/redis" "github.com/garyburd/redigo/redis"
"github.com/gregjones/httpcache/diskcache" "github.com/gregjones/httpcache/diskcache"
@ -44,22 +45,21 @@ var addr = flag.String("addr", "localhost:8080", "TCP address to listen on")
var whitelist = flag.String("whitelist", "", "comma separated list of allowed remote hosts") var whitelist = flag.String("whitelist", "", "comma separated list of allowed remote hosts")
var referrers = flag.String("referrers", "", "comma separated list of allowed referring hosts") var referrers = flag.String("referrers", "", "comma separated list of allowed referring hosts")
var baseURL = flag.String("baseURL", "", "default base URL for relative remote URLs") var baseURL = flag.String("baseURL", "", "default base URL for relative remote URLs")
var cache = flag.String("cache", "", "location to cache images (see https://github.com/willnorris/imageproxy#cache)") var cache tieredCache
var signatureKey = flag.String("signatureKey", "", "HMAC key used in calculating request signatures") var signatureKey = flag.String("signatureKey", "", "HMAC key used in calculating request signatures")
var scaleUp = flag.Bool("scaleUp", false, "allow images to scale beyond their original dimensions") var scaleUp = flag.Bool("scaleUp", false, "allow images to scale beyond their original dimensions")
var timeout = flag.Duration("timeout", 0, "time limit for requests served by this proxy") var timeout = flag.Duration("timeout", 0, "time limit for requests served by this proxy")
var verbose = flag.Bool("verbose", false, "print verbose logging messages") var verbose = flag.Bool("verbose", false, "print verbose logging messages")
var version = flag.Bool("version", false, "Deprecated: this flag does nothing") var version = flag.Bool("version", false, "Deprecated: this flag does nothing")
func init() {
flag.Var(&cache, "cache", "location to cache images (see https://github.com/willnorris/imageproxy#cache)")
}
func main() { func main() {
flag.Parse() flag.Parse()
c, err := parseCache() p := imageproxy.NewProxy(nil, cache.Cache)
if err != nil {
log.Fatal(err)
}
p := imageproxy.NewProxy(nil, c)
if *whitelist != "" { if *whitelist != "" {
p.Whitelist = strings.Split(*whitelist, ",") p.Whitelist = strings.Split(*whitelist, ",")
} }
@ -99,17 +99,41 @@ func main() {
log.Fatal(server.ListenAndServe()) log.Fatal(server.ListenAndServe())
} }
// parseCache parses the cache-related flags and returns the specified Cache implementation. // tieredCache allows specifying multiple caches via flags, which will create
func parseCache() (imageproxy.Cache, error) { // tiered caches using the twotier package.
if *cache == "" { type tieredCache struct {
imageproxy.Cache
}
func (tc *tieredCache) String() string {
return fmt.Sprint(*tc)
}
func (tc *tieredCache) Set(value string) error {
c, err := parseCache(value)
if err != nil {
return err
}
if tc.Cache == nil {
tc.Cache = c
} else {
tc.Cache = twotier.New(tc.Cache, c)
}
return nil
}
// parseCache parses c returns the specified Cache implementation.
func parseCache(c string) (imageproxy.Cache, error) {
if c == "" {
return nil, nil return nil, nil
} }
if *cache == "memory" { if c == "memory" {
*cache = fmt.Sprintf("memory:%d", defaultMemorySize) c = fmt.Sprintf("memory:%d", defaultMemorySize)
} }
u, err := url.Parse(*cache) u, err := url.Parse(c)
if err != nil { if err != nil {
return nil, fmt.Errorf("error parsing cache flag: %v", err) return nil, fmt.Errorf("error parsing cache flag: %v", err)
} }