d8fba23365
Previously background contexts where created where necessary (e.g. in GCS interactions). Should I begin to use request timeouts or other context-dependent things in the future, it's useful to have the actual HTTP request context around. This threads the request context through the application to all places that need it.
82 lines
1.8 KiB
Go
82 lines
1.8 KiB
Go
// Filesystem storage backend for Nixery.
|
|
package storage
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"os"
|
|
"path"
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
type FSBackend struct {
|
|
path string
|
|
}
|
|
|
|
func NewFSBackend() (*FSBackend, error) {
|
|
p := os.Getenv("STORAGE_PATH")
|
|
if p == "" {
|
|
return nil, fmt.Errorf("STORAGE_PATH must be set for filesystem storage")
|
|
}
|
|
|
|
p = path.Clean(p)
|
|
err := os.MkdirAll(p, 0755)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to create storage dir: %s", err)
|
|
}
|
|
|
|
return &FSBackend{p}, nil
|
|
}
|
|
|
|
func (b *FSBackend) Name() string {
|
|
return fmt.Sprintf("Filesystem (%s)", b.path)
|
|
}
|
|
|
|
func (b *FSBackend) Persist(ctx context.Context, key string, f Persister) (string, int64, error) {
|
|
full := path.Join(b.path, key)
|
|
dir := path.Dir(full)
|
|
err := os.MkdirAll(dir, 0755)
|
|
if err != nil {
|
|
log.WithError(err).WithField("path", dir).Error("failed to create storage directory")
|
|
return "", 0, err
|
|
}
|
|
|
|
file, err := os.OpenFile(full, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0644)
|
|
if err != nil {
|
|
log.WithError(err).WithField("file", full).Error("failed to write file")
|
|
return "", 0, err
|
|
}
|
|
defer file.Close()
|
|
|
|
return f(file)
|
|
}
|
|
|
|
func (b *FSBackend) Fetch(ctx context.Context, key string) (io.ReadCloser, error) {
|
|
full := path.Join(b.path, key)
|
|
return os.Open(full)
|
|
}
|
|
|
|
func (b *FSBackend) Move(ctx context.Context, old, new string) error {
|
|
newpath := path.Join(b.path, new)
|
|
err := os.MkdirAll(path.Dir(newpath), 0755)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return os.Rename(path.Join(b.path, old), newpath)
|
|
}
|
|
|
|
func (b *FSBackend) ServeLayer(digest string, r *http.Request, w http.ResponseWriter) error {
|
|
p := path.Join(b.path, "layers", digest)
|
|
|
|
log.WithFields(log.Fields{
|
|
"layer": digest,
|
|
"path": p,
|
|
}).Info("serving layer from filesystem")
|
|
|
|
http.ServeFile(w, r, p)
|
|
return nil
|
|
}
|