2022-09-21 15:09:50 +00:00
|
|
|
// Package s3 implements a fake s3 server for rclone
|
|
|
|
package s3
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"math/rand"
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/go-chi/chi/v5"
|
2024-04-13 16:50:11 +00:00
|
|
|
"github.com/rclone/gofakes3"
|
2022-09-21 15:09:50 +00:00
|
|
|
"github.com/rclone/rclone/fs"
|
|
|
|
"github.com/rclone/rclone/fs/hash"
|
|
|
|
httplib "github.com/rclone/rclone/lib/http"
|
|
|
|
"github.com/rclone/rclone/vfs"
|
2024-07-03 10:34:29 +00:00
|
|
|
"github.com/rclone/rclone/vfs/vfscommon"
|
2022-09-21 15:09:50 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Options contains options for the http Server
|
|
|
|
type Options struct {
|
|
|
|
//TODO add more options
|
|
|
|
pathBucketMode bool
|
|
|
|
hashName string
|
|
|
|
hashType hash.Type
|
|
|
|
authPair []string
|
|
|
|
noCleanup bool
|
|
|
|
HTTP httplib.Config
|
|
|
|
}
|
|
|
|
|
|
|
|
// Server is a s3.FileSystem interface
|
|
|
|
type Server struct {
|
|
|
|
*httplib.Server
|
|
|
|
f fs.Fs
|
|
|
|
vfs *vfs.VFS
|
|
|
|
faker *gofakes3.GoFakeS3
|
|
|
|
handler http.Handler
|
|
|
|
ctx context.Context // for global config
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make a new S3 Server to serve the remote
|
|
|
|
func newServer(ctx context.Context, f fs.Fs, opt *Options) (s *Server, err error) {
|
|
|
|
w := &Server{
|
|
|
|
f: f,
|
|
|
|
ctx: ctx,
|
2024-07-03 10:34:29 +00:00
|
|
|
vfs: vfs.New(f, &vfscommon.Opt),
|
2022-09-21 15:09:50 +00:00
|
|
|
}
|
|
|
|
|
2023-11-13 16:40:34 +00:00
|
|
|
if len(opt.authPair) == 0 {
|
|
|
|
fs.Logf("serve s3", "No auth provided so allowing anonymous access")
|
|
|
|
}
|
|
|
|
|
2022-09-21 15:09:50 +00:00
|
|
|
var newLogger logger
|
|
|
|
w.faker = gofakes3.New(
|
|
|
|
newBackend(w.vfs, opt),
|
|
|
|
gofakes3.WithHostBucket(!opt.pathBucketMode),
|
|
|
|
gofakes3.WithLogger(newLogger),
|
|
|
|
gofakes3.WithRequestID(rand.Uint64()),
|
|
|
|
gofakes3.WithoutVersioning(),
|
|
|
|
gofakes3.WithV4Auth(authlistResolver(opt.authPair)),
|
2023-11-13 16:40:34 +00:00
|
|
|
gofakes3.WithIntegrityCheck(true), // Check Content-MD5 if supplied
|
2022-09-21 15:09:50 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
w.Server, err = httplib.NewServer(ctx,
|
|
|
|
httplib.WithConfig(opt.HTTP),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to init server: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
w.handler = w.faker.Server()
|
|
|
|
return w, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Bind register the handler to http.Router
|
|
|
|
func (w *Server) Bind(router chi.Router) {
|
|
|
|
router.Handle("/*", w.handler)
|
|
|
|
}
|
2023-09-12 10:50:13 +00:00
|
|
|
|
|
|
|
func (w *Server) serve() error {
|
|
|
|
w.Serve()
|
|
|
|
fs.Logf(w.f, "Starting s3 server on %s", w.URLs())
|
|
|
|
return nil
|
|
|
|
}
|