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

89 lines
2 KiB
Go
Raw Normal View History

// Copyright 2013 The imageproxy authors.
// SPDX-License-Identifier: Apache-2.0
2018-06-02 16:44:00 +00:00
// Package gcscache provides an httpcache.Cache implementation that stores
// cached values on Google Cloud Storage.
package gcscache
import (
"context"
"crypto/md5"
"encoding/hex"
2022-01-22 10:29:37 -08:00
"errors"
"io"
"io/ioutil"
"log"
"path"
"cloud.google.com/go/storage"
)
var ctx = context.Background()
type cache struct {
bucket *storage.BucketHandle
prefix string
}
func (c *cache) Get(key string) ([]byte, bool) {
r, err := c.object(key).NewReader(ctx)
if err != nil {
2022-01-22 10:29:37 -08:00
if !errors.Is(err, storage.ErrObjectNotExist) {
log.Printf("error reading from gcs: %v", err)
}
return nil, false
}
defer r.Close()
value, err := ioutil.ReadAll(r)
if err != nil {
log.Printf("error reading from gcs: %v", err)
return nil, false
}
return value, true
}
func (c *cache) Set(key string, value []byte) {
w := c.object(key).NewWriter(ctx)
if _, err := w.Write(value); err != nil {
log.Printf("error writing to gcs: %v", err)
}
if err := w.Close(); err != nil {
log.Printf("error closing gcs object writer: %v", err)
}
}
func (c *cache) Delete(key string) {
if err := c.object(key).Delete(ctx); err != nil {
log.Printf("error deleting gcs object: %v", err)
}
}
func (c *cache) object(key string) *storage.ObjectHandle {
name := path.Join(c.prefix, keyToFilename(key))
return c.bucket.Object(name)
}
func keyToFilename(key string) string {
h := md5.New()
_, _ = io.WriteString(h, key)
return hex.EncodeToString(h.Sum(nil))
}
// New constructs a Cache storing files in the specified GCS bucket. If prefix
// is not empty, objects will be prefixed with that path. Credentials should
// be specified using one of the mechanisms supported for Application Default
// Credentials (see https://cloud.google.com/docs/authentication/production)
func New(bucket, prefix string) (*cache, error) {
client, err := storage.NewClient(ctx)
if err != nil {
return nil, err
}
return &cache{
prefix: prefix,
bucket: client.Bucket(bucket),
}, nil
}