mirror of
https://github.com/dutchcoders/transfer.sh.git
synced 2025-01-12 11:50:18 +01:00
server: propagate context to storage layer (#455)
This change propagates the context of the actual user request through. Additionally it configures the Storj Backend to do in memory buffering instead of relying on temporary files in /tmp/ folders. Fixes #448
This commit is contained in:
parent
f062af9fc5
commit
492731e31f
3 changed files with 86 additions and 97 deletions
|
@ -27,13 +27,12 @@ THE SOFTWARE.
|
||||||
package server
|
package server
|
||||||
|
|
||||||
import (
|
import (
|
||||||
// _ "transfer.sh/app/handlers"
|
|
||||||
// _ "transfer.sh/app/utils"
|
|
||||||
|
|
||||||
"archive/tar"
|
"archive/tar"
|
||||||
"archive/zip"
|
"archive/zip"
|
||||||
"bytes"
|
"bytes"
|
||||||
"compress/gzip"
|
"compress/gzip"
|
||||||
|
"context"
|
||||||
|
"encoding/base64"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
@ -42,6 +41,7 @@ import (
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"mime"
|
"mime"
|
||||||
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
|
@ -53,15 +53,10 @@ import (
|
||||||
text_template "text/template"
|
text_template "text/template"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
blackfriday "github.com/russross/blackfriday/v2"
|
|
||||||
|
|
||||||
"net"
|
|
||||||
|
|
||||||
"encoding/base64"
|
|
||||||
|
|
||||||
web "github.com/dutchcoders/transfer.sh-web"
|
web "github.com/dutchcoders/transfer.sh-web"
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
"github.com/microcosm-cc/bluemonday"
|
"github.com/microcosm-cc/bluemonday"
|
||||||
|
blackfriday "github.com/russross/blackfriday/v2"
|
||||||
"github.com/skip2/go-qrcode"
|
"github.com/skip2/go-qrcode"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -125,7 +120,7 @@ func (s *Server) previewHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
token := vars["token"]
|
token := vars["token"]
|
||||||
filename := vars["filename"]
|
filename := vars["filename"]
|
||||||
|
|
||||||
metadata, err := s.checkMetadata(token, filename, false)
|
metadata, err := s.checkMetadata(r.Context(), token, filename, false)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
s.logger.Printf("Error metadata: %s", err.Error())
|
s.logger.Printf("Error metadata: %s", err.Error())
|
||||||
|
@ -134,7 +129,7 @@ func (s *Server) previewHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
}
|
}
|
||||||
|
|
||||||
contentType := metadata.ContentType
|
contentType := metadata.ContentType
|
||||||
contentLength, err := s.storage.Head(token, filename)
|
contentLength, err := s.storage.Head(r.Context(), token, filename)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
http.Error(w, http.StatusText(404), 404)
|
http.Error(w, http.StatusText(404), 404)
|
||||||
return
|
return
|
||||||
|
@ -154,7 +149,7 @@ func (s *Server) previewHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
templatePath = "download.markdown.html"
|
templatePath = "download.markdown.html"
|
||||||
|
|
||||||
var reader io.ReadCloser
|
var reader io.ReadCloser
|
||||||
if reader, _, err = s.storage.Get(token, filename); err != nil {
|
if reader, _, err = s.storage.Get(r.Context(), token, filename); err != nil {
|
||||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -370,7 +365,7 @@ func (s *Server) postHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
http.Error(w, "Could not encode metadata", 500)
|
http.Error(w, "Could not encode metadata", 500)
|
||||||
|
|
||||||
return
|
return
|
||||||
} else if err := s.storage.Put(token, fmt.Sprintf("%s.metadata", filename), buffer, "text/json", uint64(buffer.Len())); err != nil {
|
} else if err := s.storage.Put(r.Context(), token, fmt.Sprintf("%s.metadata", filename), buffer, "text/json", uint64(buffer.Len())); err != nil {
|
||||||
s.logger.Printf("%s", err.Error())
|
s.logger.Printf("%s", err.Error())
|
||||||
http.Error(w, "Could not save metadata", 500)
|
http.Error(w, "Could not save metadata", 500)
|
||||||
|
|
||||||
|
@ -379,7 +374,7 @@ func (s *Server) postHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|
||||||
s.logger.Printf("Uploading %s %s %d %s", token, filename, contentLength, contentType)
|
s.logger.Printf("Uploading %s %s %d %s", token, filename, contentLength, contentType)
|
||||||
|
|
||||||
if err = s.storage.Put(token, filename, reader, contentType, uint64(contentLength)); err != nil {
|
if err = s.storage.Put(r.Context(), token, filename, reader, contentType, uint64(contentLength)); err != nil {
|
||||||
s.logger.Printf("Backend storage error: %s", err.Error())
|
s.logger.Printf("Backend storage error: %s", err.Error())
|
||||||
http.Error(w, err.Error(), 500)
|
http.Error(w, err.Error(), 500)
|
||||||
return
|
return
|
||||||
|
@ -533,7 +528,7 @@ func (s *Server) putHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
s.logger.Print("Invalid MaxDate")
|
s.logger.Print("Invalid MaxDate")
|
||||||
http.Error(w, "Invalid MaxDate, make sure Max-Days is smaller than 290 years", 400)
|
http.Error(w, "Invalid MaxDate, make sure Max-Days is smaller than 290 years", 400)
|
||||||
return
|
return
|
||||||
} else if err := s.storage.Put(token, fmt.Sprintf("%s.metadata", filename), buffer, "text/json", uint64(buffer.Len())); err != nil {
|
} else if err := s.storage.Put(r.Context(), token, fmt.Sprintf("%s.metadata", filename), buffer, "text/json", uint64(buffer.Len())); err != nil {
|
||||||
s.logger.Printf("%s", err.Error())
|
s.logger.Printf("%s", err.Error())
|
||||||
http.Error(w, "Could not save metadata", 500)
|
http.Error(w, "Could not save metadata", 500)
|
||||||
return
|
return
|
||||||
|
@ -543,7 +538,7 @@ func (s *Server) putHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
if err = s.storage.Put(token, filename, reader, contentType, uint64(contentLength)); err != nil {
|
if err = s.storage.Put(r.Context(), token, filename, reader, contentType, uint64(contentLength)); err != nil {
|
||||||
s.logger.Printf("Error putting new file: %s", err.Error())
|
s.logger.Printf("Error putting new file: %s", err.Error())
|
||||||
http.Error(w, "Could not save file", 500)
|
http.Error(w, "Could not save file", 500)
|
||||||
return
|
return
|
||||||
|
@ -679,13 +674,13 @@ func (s *Server) unlock(token, filename string) {
|
||||||
lock.(*sync.Mutex).Unlock()
|
lock.(*sync.Mutex).Unlock()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Server) checkMetadata(token, filename string, increaseDownload bool) (metadata, error) {
|
func (s *Server) checkMetadata(ctx context.Context, token, filename string, increaseDownload bool) (metadata, error) {
|
||||||
s.lock(token, filename)
|
s.lock(token, filename)
|
||||||
defer s.unlock(token, filename)
|
defer s.unlock(token, filename)
|
||||||
|
|
||||||
var metadata metadata
|
var metadata metadata
|
||||||
|
|
||||||
r, _, err := s.storage.Get(token, fmt.Sprintf("%s.metadata", filename))
|
r, _, err := s.storage.Get(ctx, token, fmt.Sprintf("%s.metadata", filename))
|
||||||
defer CloseCheck(r.Close)
|
defer CloseCheck(r.Close)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -707,7 +702,7 @@ func (s *Server) checkMetadata(token, filename string, increaseDownload bool) (m
|
||||||
buffer := &bytes.Buffer{}
|
buffer := &bytes.Buffer{}
|
||||||
if err := json.NewEncoder(buffer).Encode(metadata); err != nil {
|
if err := json.NewEncoder(buffer).Encode(metadata); err != nil {
|
||||||
return metadata, errors.New("could not encode metadata")
|
return metadata, errors.New("could not encode metadata")
|
||||||
} else if err := s.storage.Put(token, fmt.Sprintf("%s.metadata", filename), buffer, "text/json", uint64(buffer.Len())); err != nil {
|
} else if err := s.storage.Put(ctx, token, fmt.Sprintf("%s.metadata", filename), buffer, "text/json", uint64(buffer.Len())); err != nil {
|
||||||
return metadata, errors.New("could not save metadata")
|
return metadata, errors.New("could not save metadata")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -715,13 +710,13 @@ func (s *Server) checkMetadata(token, filename string, increaseDownload bool) (m
|
||||||
return metadata, nil
|
return metadata, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Server) checkDeletionToken(deletionToken, token, filename string) error {
|
func (s *Server) checkDeletionToken(ctx context.Context, deletionToken, token, filename string) error {
|
||||||
s.lock(token, filename)
|
s.lock(token, filename)
|
||||||
defer s.unlock(token, filename)
|
defer s.unlock(token, filename)
|
||||||
|
|
||||||
var metadata metadata
|
var metadata metadata
|
||||||
|
|
||||||
r, _, err := s.storage.Get(token, fmt.Sprintf("%s.metadata", filename))
|
r, _, err := s.storage.Get(ctx, token, fmt.Sprintf("%s.metadata", filename))
|
||||||
defer CloseCheck(r.Close)
|
defer CloseCheck(r.Close)
|
||||||
|
|
||||||
if s.storage.IsNotExist(err) {
|
if s.storage.IsNotExist(err) {
|
||||||
|
@ -744,7 +739,7 @@ func (s *Server) purgeHandler() {
|
||||||
go func() {
|
go func() {
|
||||||
for {
|
for {
|
||||||
<-ticker.C
|
<-ticker.C
|
||||||
err := s.storage.Purge(s.purgeDays)
|
err := s.storage.Purge(context.TODO(), s.purgeDays)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
s.logger.Printf("error cleaning up expired files: %v", err)
|
s.logger.Printf("error cleaning up expired files: %v", err)
|
||||||
}
|
}
|
||||||
|
@ -759,13 +754,13 @@ func (s *Server) deleteHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
filename := vars["filename"]
|
filename := vars["filename"]
|
||||||
deletionToken := vars["deletionToken"]
|
deletionToken := vars["deletionToken"]
|
||||||
|
|
||||||
if err := s.checkDeletionToken(deletionToken, token, filename); err != nil {
|
if err := s.checkDeletionToken(r.Context(), deletionToken, token, filename); err != nil {
|
||||||
s.logger.Printf("Error metadata: %s", err.Error())
|
s.logger.Printf("Error metadata: %s", err.Error())
|
||||||
http.Error(w, http.StatusText(http.StatusNotFound), http.StatusNotFound)
|
http.Error(w, http.StatusText(http.StatusNotFound), http.StatusNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
err := s.storage.Delete(token, filename)
|
err := s.storage.Delete(r.Context(), token, filename)
|
||||||
if s.storage.IsNotExist(err) {
|
if s.storage.IsNotExist(err) {
|
||||||
http.Error(w, http.StatusText(http.StatusNotFound), http.StatusNotFound)
|
http.Error(w, http.StatusText(http.StatusNotFound), http.StatusNotFound)
|
||||||
return
|
return
|
||||||
|
@ -795,12 +790,12 @@ func (s *Server) zipHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
token := strings.Split(key, "/")[0]
|
token := strings.Split(key, "/")[0]
|
||||||
filename := sanitize(strings.Split(key, "/")[1])
|
filename := sanitize(strings.Split(key, "/")[1])
|
||||||
|
|
||||||
if _, err := s.checkMetadata(token, filename, true); err != nil {
|
if _, err := s.checkMetadata(r.Context(), token, filename, true); err != nil {
|
||||||
s.logger.Printf("Error metadata: %s", err.Error())
|
s.logger.Printf("Error metadata: %s", err.Error())
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
reader, _, err := s.storage.Get(token, filename)
|
reader, _, err := s.storage.Get(r.Context(), token, filename)
|
||||||
defer CloseCheck(reader.Close)
|
defer CloseCheck(reader.Close)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -866,12 +861,12 @@ func (s *Server) tarGzHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
token := strings.Split(key, "/")[0]
|
token := strings.Split(key, "/")[0]
|
||||||
filename := sanitize(strings.Split(key, "/")[1])
|
filename := sanitize(strings.Split(key, "/")[1])
|
||||||
|
|
||||||
if _, err := s.checkMetadata(token, filename, true); err != nil {
|
if _, err := s.checkMetadata(r.Context(), token, filename, true); err != nil {
|
||||||
s.logger.Printf("Error metadata: %s", err.Error())
|
s.logger.Printf("Error metadata: %s", err.Error())
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
reader, contentLength, err := s.storage.Get(token, filename)
|
reader, contentLength, err := s.storage.Get(r.Context(), token, filename)
|
||||||
defer CloseCheck(reader.Close)
|
defer CloseCheck(reader.Close)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -925,12 +920,12 @@ func (s *Server) tarHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
token := strings.Split(key, "/")[0]
|
token := strings.Split(key, "/")[0]
|
||||||
filename := strings.Split(key, "/")[1]
|
filename := strings.Split(key, "/")[1]
|
||||||
|
|
||||||
if _, err := s.checkMetadata(token, filename, true); err != nil {
|
if _, err := s.checkMetadata(r.Context(), token, filename, true); err != nil {
|
||||||
s.logger.Printf("Error metadata: %s", err.Error())
|
s.logger.Printf("Error metadata: %s", err.Error())
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
reader, contentLength, err := s.storage.Get(token, filename)
|
reader, contentLength, err := s.storage.Get(r.Context(), token, filename)
|
||||||
defer CloseCheck(reader.Close)
|
defer CloseCheck(reader.Close)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -970,7 +965,7 @@ func (s *Server) headHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
token := vars["token"]
|
token := vars["token"]
|
||||||
filename := vars["filename"]
|
filename := vars["filename"]
|
||||||
|
|
||||||
metadata, err := s.checkMetadata(token, filename, false)
|
metadata, err := s.checkMetadata(r.Context(), token, filename, false)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
s.logger.Printf("Error metadata: %s", err.Error())
|
s.logger.Printf("Error metadata: %s", err.Error())
|
||||||
|
@ -979,7 +974,7 @@ func (s *Server) headHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
}
|
}
|
||||||
|
|
||||||
contentType := metadata.ContentType
|
contentType := metadata.ContentType
|
||||||
contentLength, err := s.storage.Head(token, filename)
|
contentLength, err := s.storage.Head(r.Context(), token, filename)
|
||||||
if s.storage.IsNotExist(err) {
|
if s.storage.IsNotExist(err) {
|
||||||
http.Error(w, http.StatusText(http.StatusNotFound), http.StatusNotFound)
|
http.Error(w, http.StatusText(http.StatusNotFound), http.StatusNotFound)
|
||||||
return
|
return
|
||||||
|
@ -1005,7 +1000,7 @@ func (s *Server) getHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
token := vars["token"]
|
token := vars["token"]
|
||||||
filename := vars["filename"]
|
filename := vars["filename"]
|
||||||
|
|
||||||
metadata, err := s.checkMetadata(token, filename, true)
|
metadata, err := s.checkMetadata(r.Context(), token, filename, true)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
s.logger.Printf("Error metadata: %s", err.Error())
|
s.logger.Printf("Error metadata: %s", err.Error())
|
||||||
|
@ -1014,7 +1009,7 @@ func (s *Server) getHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
}
|
}
|
||||||
|
|
||||||
contentType := metadata.ContentType
|
contentType := metadata.ContentType
|
||||||
reader, contentLength, err := s.storage.Get(token, filename)
|
reader, contentLength, err := s.storage.Get(r.Context(), token, filename)
|
||||||
defer CloseCheck(reader.Close)
|
defer CloseCheck(reader.Close)
|
||||||
|
|
||||||
if s.storage.IsNotExist(err) {
|
if s.storage.IsNotExist(err) {
|
||||||
|
|
|
@ -426,11 +426,11 @@ func (s *Server) Run() {
|
||||||
|
|
||||||
_, err = htmlTemplates.New(stripPrefix(path)).Parse(string(bytes))
|
_, err = htmlTemplates.New(stripPrefix(path)).Parse(string(bytes))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
s.logger.Panicln("Unable to parse template")
|
s.logger.Println("Unable to parse html template", err)
|
||||||
}
|
}
|
||||||
_, err = textTemplates.New(stripPrefix(path)).Parse(string(bytes))
|
_, err = textTemplates.New(stripPrefix(path)).Parse(string(bytes))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
s.logger.Panicln("Unable to parse template")
|
s.logger.Println("Unable to parse text template", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,19 +1,10 @@
|
||||||
package server
|
package server
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"github.com/aws/aws-sdk-go/aws"
|
|
||||||
"github.com/aws/aws-sdk-go/aws/awserr"
|
|
||||||
"github.com/aws/aws-sdk-go/aws/session"
|
|
||||||
"github.com/aws/aws-sdk-go/service/s3"
|
|
||||||
"github.com/aws/aws-sdk-go/service/s3/s3manager"
|
|
||||||
"golang.org/x/net/context"
|
|
||||||
"golang.org/x/oauth2"
|
|
||||||
"golang.org/x/oauth2/google"
|
|
||||||
"google.golang.org/api/drive/v3"
|
|
||||||
"google.golang.org/api/googleapi"
|
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
"log"
|
||||||
|
@ -23,6 +14,17 @@ import (
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/aws/aws-sdk-go/aws"
|
||||||
|
"github.com/aws/aws-sdk-go/aws/awserr"
|
||||||
|
"github.com/aws/aws-sdk-go/aws/session"
|
||||||
|
"github.com/aws/aws-sdk-go/service/s3"
|
||||||
|
"github.com/aws/aws-sdk-go/service/s3/s3manager"
|
||||||
|
"golang.org/x/oauth2"
|
||||||
|
"golang.org/x/oauth2/google"
|
||||||
|
"google.golang.org/api/drive/v3"
|
||||||
|
"google.golang.org/api/googleapi"
|
||||||
|
|
||||||
|
"storj.io/common/fpath"
|
||||||
"storj.io/common/storj"
|
"storj.io/common/storj"
|
||||||
"storj.io/uplink"
|
"storj.io/uplink"
|
||||||
)
|
)
|
||||||
|
@ -30,17 +32,17 @@ import (
|
||||||
// Storage is the interface for storage operation
|
// Storage is the interface for storage operation
|
||||||
type Storage interface {
|
type Storage interface {
|
||||||
// Get retrieves a file from storage
|
// Get retrieves a file from storage
|
||||||
Get(token string, filename string) (reader io.ReadCloser, contentLength uint64, err error)
|
Get(ctx context.Context, token string, filename string) (reader io.ReadCloser, contentLength uint64, err error)
|
||||||
// Head retrieves content length of a file from storage
|
// Head retrieves content length of a file from storage
|
||||||
Head(token string, filename string) (contentLength uint64, err error)
|
Head(ctx context.Context, token string, filename string) (contentLength uint64, err error)
|
||||||
// Put saves a file on storage
|
// Put saves a file on storage
|
||||||
Put(token string, filename string, reader io.Reader, contentType string, contentLength uint64) error
|
Put(ctx context.Context, token string, filename string, reader io.Reader, contentType string, contentLength uint64) error
|
||||||
// Delete removes a file from storage
|
// Delete removes a file from storage
|
||||||
Delete(token string, filename string) error
|
Delete(ctx context.Context, token string, filename string) error
|
||||||
// IsNotExist indicates if a file doesn't exist on storage
|
// IsNotExist indicates if a file doesn't exist on storage
|
||||||
IsNotExist(err error) bool
|
IsNotExist(err error) bool
|
||||||
// Purge cleans up the storage
|
// Purge cleans up the storage
|
||||||
Purge(days time.Duration) error
|
Purge(ctx context.Context, days time.Duration) error
|
||||||
|
|
||||||
// Type returns the storage type
|
// Type returns the storage type
|
||||||
Type() string
|
Type() string
|
||||||
|
@ -64,7 +66,7 @@ func (s *LocalStorage) Type() string {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Head retrieves content length of a file from storage
|
// Head retrieves content length of a file from storage
|
||||||
func (s *LocalStorage) Head(token string, filename string) (contentLength uint64, err error) {
|
func (s *LocalStorage) Head(ctx context.Context, token string, filename string) (contentLength uint64, err error) {
|
||||||
path := filepath.Join(s.basedir, token, filename)
|
path := filepath.Join(s.basedir, token, filename)
|
||||||
|
|
||||||
var fi os.FileInfo
|
var fi os.FileInfo
|
||||||
|
@ -78,7 +80,7 @@ func (s *LocalStorage) Head(token string, filename string) (contentLength uint64
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get retrieves a file from storage
|
// Get retrieves a file from storage
|
||||||
func (s *LocalStorage) Get(token string, filename string) (reader io.ReadCloser, contentLength uint64, err error) {
|
func (s *LocalStorage) Get(ctx context.Context, token string, filename string) (reader io.ReadCloser, contentLength uint64, err error) {
|
||||||
path := filepath.Join(s.basedir, token, filename)
|
path := filepath.Join(s.basedir, token, filename)
|
||||||
|
|
||||||
// content type , content length
|
// content type , content length
|
||||||
|
@ -97,7 +99,7 @@ func (s *LocalStorage) Get(token string, filename string) (reader io.ReadCloser,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete removes a file from storage
|
// Delete removes a file from storage
|
||||||
func (s *LocalStorage) Delete(token string, filename string) (err error) {
|
func (s *LocalStorage) Delete(ctx context.Context, token string, filename string) (err error) {
|
||||||
metadata := filepath.Join(s.basedir, token, fmt.Sprintf("%s.metadata", filename))
|
metadata := filepath.Join(s.basedir, token, fmt.Sprintf("%s.metadata", filename))
|
||||||
_ = os.Remove(metadata)
|
_ = os.Remove(metadata)
|
||||||
|
|
||||||
|
@ -107,7 +109,7 @@ func (s *LocalStorage) Delete(token string, filename string) (err error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Purge cleans up the storage
|
// Purge cleans up the storage
|
||||||
func (s *LocalStorage) Purge(days time.Duration) (err error) {
|
func (s *LocalStorage) Purge(ctx context.Context, days time.Duration) (err error) {
|
||||||
err = filepath.Walk(s.basedir,
|
err = filepath.Walk(s.basedir,
|
||||||
func(path string, info os.FileInfo, err error) error {
|
func(path string, info os.FileInfo, err error) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -138,7 +140,7 @@ func (s *LocalStorage) IsNotExist(err error) bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Put saves a file on storage
|
// Put saves a file on storage
|
||||||
func (s *LocalStorage) Put(token string, filename string, reader io.Reader, contentType string, contentLength uint64) error {
|
func (s *LocalStorage) Put(ctx context.Context, token string, filename string, reader io.Reader, contentType string, contentLength uint64) error {
|
||||||
var f io.WriteCloser
|
var f io.WriteCloser
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
|
@ -193,7 +195,7 @@ func (s *S3Storage) Type() string {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Head retrieves content length of a file from storage
|
// Head retrieves content length of a file from storage
|
||||||
func (s *S3Storage) Head(token string, filename string) (contentLength uint64, err error) {
|
func (s *S3Storage) Head(ctx context.Context, token string, filename string) (contentLength uint64, err error) {
|
||||||
key := fmt.Sprintf("%s/%s", token, filename)
|
key := fmt.Sprintf("%s/%s", token, filename)
|
||||||
|
|
||||||
headRequest := &s3.HeadObjectInput{
|
headRequest := &s3.HeadObjectInput{
|
||||||
|
@ -202,7 +204,7 @@ func (s *S3Storage) Head(token string, filename string) (contentLength uint64, e
|
||||||
}
|
}
|
||||||
|
|
||||||
// content type , content length
|
// content type , content length
|
||||||
response, err := s.s3.HeadObject(headRequest)
|
response, err := s.s3.HeadObjectWithContext(ctx, headRequest)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -215,7 +217,7 @@ func (s *S3Storage) Head(token string, filename string) (contentLength uint64, e
|
||||||
}
|
}
|
||||||
|
|
||||||
// Purge cleans up the storage
|
// Purge cleans up the storage
|
||||||
func (s *S3Storage) Purge(days time.Duration) (err error) {
|
func (s *S3Storage) Purge(ctx context.Context, days time.Duration) (err error) {
|
||||||
// NOOP expiration is set at upload time
|
// NOOP expiration is set at upload time
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -237,7 +239,7 @@ func (s *S3Storage) IsNotExist(err error) bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get retrieves a file from storage
|
// Get retrieves a file from storage
|
||||||
func (s *S3Storage) Get(token string, filename string) (reader io.ReadCloser, contentLength uint64, err error) {
|
func (s *S3Storage) Get(ctx context.Context, token string, filename string) (reader io.ReadCloser, contentLength uint64, err error) {
|
||||||
key := fmt.Sprintf("%s/%s", token, filename)
|
key := fmt.Sprintf("%s/%s", token, filename)
|
||||||
|
|
||||||
getRequest := &s3.GetObjectInput{
|
getRequest := &s3.GetObjectInput{
|
||||||
|
@ -245,7 +247,7 @@ func (s *S3Storage) Get(token string, filename string) (reader io.ReadCloser, co
|
||||||
Key: aws.String(key),
|
Key: aws.String(key),
|
||||||
}
|
}
|
||||||
|
|
||||||
response, err := s.s3.GetObject(getRequest)
|
response, err := s.s3.GetObjectWithContext(ctx, getRequest)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -259,14 +261,14 @@ func (s *S3Storage) Get(token string, filename string) (reader io.ReadCloser, co
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete removes a file from storage
|
// Delete removes a file from storage
|
||||||
func (s *S3Storage) Delete(token string, filename string) (err error) {
|
func (s *S3Storage) Delete(ctx context.Context, token string, filename string) (err error) {
|
||||||
metadata := fmt.Sprintf("%s/%s.metadata", token, filename)
|
metadata := fmt.Sprintf("%s/%s.metadata", token, filename)
|
||||||
deleteRequest := &s3.DeleteObjectInput{
|
deleteRequest := &s3.DeleteObjectInput{
|
||||||
Bucket: aws.String(s.bucket),
|
Bucket: aws.String(s.bucket),
|
||||||
Key: aws.String(metadata),
|
Key: aws.String(metadata),
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = s.s3.DeleteObject(deleteRequest)
|
_, err = s.s3.DeleteObjectWithContext(ctx, deleteRequest)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -277,13 +279,13 @@ func (s *S3Storage) Delete(token string, filename string) (err error) {
|
||||||
Key: aws.String(key),
|
Key: aws.String(key),
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = s.s3.DeleteObject(deleteRequest)
|
_, err = s.s3.DeleteObjectWithContext(ctx, deleteRequest)
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Put saves a file on storage
|
// Put saves a file on storage
|
||||||
func (s *S3Storage) Put(token string, filename string, reader io.Reader, contentType string, contentLength uint64) (err error) {
|
func (s *S3Storage) Put(ctx context.Context, token string, filename string, reader io.Reader, contentType string, contentLength uint64) (err error) {
|
||||||
key := fmt.Sprintf("%s/%s", token, filename)
|
key := fmt.Sprintf("%s/%s", token, filename)
|
||||||
|
|
||||||
s.logger.Printf("Uploading file %s to S3 Bucket", filename)
|
s.logger.Printf("Uploading file %s to S3 Bucket", filename)
|
||||||
|
@ -305,7 +307,7 @@ func (s *S3Storage) Put(token string, filename string, reader io.Reader, content
|
||||||
expire = aws.Time(time.Now().Add(s.purgeDays))
|
expire = aws.Time(time.Now().Add(s.purgeDays))
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = uploader.Upload(&s3manager.UploadInput{
|
_, err = uploader.UploadWithContext(ctx, &s3manager.UploadInput{
|
||||||
Bucket: aws.String(s.bucket),
|
Bucket: aws.String(s.bucket),
|
||||||
Key: aws.String(key),
|
Key: aws.String(key),
|
||||||
Body: reader,
|
Body: reader,
|
||||||
|
@ -339,7 +341,7 @@ func NewGDriveStorage(clientJSONFilepath string, localConfigPath string, basedir
|
||||||
}
|
}
|
||||||
|
|
||||||
// ToDo: Upgrade deprecated version
|
// ToDo: Upgrade deprecated version
|
||||||
srv, err := drive.New(getGDriveClient(config, localConfigPath, logger)) // nolint: staticcheck
|
srv, err := drive.New(getGDriveClient(context.TODO(), config, localConfigPath, logger)) // nolint: staticcheck
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -468,7 +470,7 @@ func (s *GDrive) Type() string {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Head retrieves content length of a file from storage
|
// Head retrieves content length of a file from storage
|
||||||
func (s *GDrive) Head(token string, filename string) (contentLength uint64, err error) {
|
func (s *GDrive) Head(ctx context.Context, token string, filename string) (contentLength uint64, err error) {
|
||||||
var fileID string
|
var fileID string
|
||||||
fileID, err = s.findID(filename, token)
|
fileID, err = s.findID(filename, token)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -486,7 +488,7 @@ func (s *GDrive) Head(token string, filename string) (contentLength uint64, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get retrieves a file from storage
|
// Get retrieves a file from storage
|
||||||
func (s *GDrive) Get(token string, filename string) (reader io.ReadCloser, contentLength uint64, err error) {
|
func (s *GDrive) Get(ctx context.Context, token string, filename string) (reader io.ReadCloser, contentLength uint64, err error) {
|
||||||
var fileID string
|
var fileID string
|
||||||
fileID, err = s.findID(filename, token)
|
fileID, err = s.findID(filename, token)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -505,7 +507,6 @@ func (s *GDrive) Get(token string, filename string) (reader io.ReadCloser, conte
|
||||||
|
|
||||||
contentLength = uint64(fi.Size)
|
contentLength = uint64(fi.Size)
|
||||||
|
|
||||||
ctx := context.Background()
|
|
||||||
var res *http.Response
|
var res *http.Response
|
||||||
res, err = s.service.Files.Get(fileID).Context(ctx).Download()
|
res, err = s.service.Files.Get(fileID).Context(ctx).Download()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -518,7 +519,7 @@ func (s *GDrive) Get(token string, filename string) (reader io.ReadCloser, conte
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete removes a file from storage
|
// Delete removes a file from storage
|
||||||
func (s *GDrive) Delete(token string, filename string) (err error) {
|
func (s *GDrive) Delete(ctx context.Context, token string, filename string) (err error) {
|
||||||
metadata, _ := s.findID(fmt.Sprintf("%s.metadata", filename), token)
|
metadata, _ := s.findID(fmt.Sprintf("%s.metadata", filename), token)
|
||||||
_ = s.service.Files.Delete(metadata).Do()
|
_ = s.service.Files.Delete(metadata).Do()
|
||||||
|
|
||||||
|
@ -533,7 +534,7 @@ func (s *GDrive) Delete(token string, filename string) (err error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Purge cleans up the storage
|
// Purge cleans up the storage
|
||||||
func (s *GDrive) Purge(days time.Duration) (err error) {
|
func (s *GDrive) Purge(ctx context.Context, days time.Duration) (err error) {
|
||||||
nextPageToken := ""
|
nextPageToken := ""
|
||||||
|
|
||||||
expirationDate := time.Now().Add(-1 * days).Format(time.RFC3339)
|
expirationDate := time.Now().Add(-1 * days).Format(time.RFC3339)
|
||||||
|
@ -578,7 +579,7 @@ func (s *GDrive) IsNotExist(err error) bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Put saves a file on storage
|
// Put saves a file on storage
|
||||||
func (s *GDrive) Put(token string, filename string, reader io.Reader, contentType string, contentLength uint64) error {
|
func (s *GDrive) Put(ctx context.Context, token string, filename string, reader io.Reader, contentType string, contentLength uint64) error {
|
||||||
dirID, err := s.findID("", token)
|
dirID, err := s.findID("", token)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -607,7 +608,6 @@ func (s *GDrive) Put(token string, filename string, reader io.Reader, contentTyp
|
||||||
MimeType: contentType,
|
MimeType: contentType,
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx := context.Background()
|
|
||||||
_, err = s.service.Files.Create(dst).Context(ctx).Media(reader, googleapi.ChunkSize(s.chunkSize)).Do()
|
_, err = s.service.Files.Create(dst).Context(ctx).Media(reader, googleapi.ChunkSize(s.chunkSize)).Do()
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -618,19 +618,19 @@ func (s *GDrive) Put(token string, filename string, reader io.Reader, contentTyp
|
||||||
}
|
}
|
||||||
|
|
||||||
// Retrieve a token, saves the token, then returns the generated client.
|
// Retrieve a token, saves the token, then returns the generated client.
|
||||||
func getGDriveClient(config *oauth2.Config, localConfigPath string, logger *log.Logger) *http.Client {
|
func getGDriveClient(ctx context.Context, config *oauth2.Config, localConfigPath string, logger *log.Logger) *http.Client {
|
||||||
tokenFile := filepath.Join(localConfigPath, gdriveTokenJSONFile)
|
tokenFile := filepath.Join(localConfigPath, gdriveTokenJSONFile)
|
||||||
tok, err := gDriveTokenFromFile(tokenFile)
|
tok, err := gDriveTokenFromFile(tokenFile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
tok = getGDriveTokenFromWeb(config, logger)
|
tok = getGDriveTokenFromWeb(ctx, config, logger)
|
||||||
saveGDriveToken(tokenFile, tok, logger)
|
saveGDriveToken(tokenFile, tok, logger)
|
||||||
}
|
}
|
||||||
|
|
||||||
return config.Client(context.Background(), tok)
|
return config.Client(ctx, tok)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Request a token from the web, then returns the retrieved token.
|
// Request a token from the web, then returns the retrieved token.
|
||||||
func getGDriveTokenFromWeb(config *oauth2.Config, logger *log.Logger) *oauth2.Token {
|
func getGDriveTokenFromWeb(ctx context.Context, config *oauth2.Config, logger *log.Logger) *oauth2.Token {
|
||||||
authURL := config.AuthCodeURL("state-token", oauth2.AccessTypeOffline)
|
authURL := config.AuthCodeURL("state-token", oauth2.AccessTypeOffline)
|
||||||
fmt.Printf("Go to the following link in your browser then type the "+
|
fmt.Printf("Go to the following link in your browser then type the "+
|
||||||
"authorization code: \n%v\n", authURL)
|
"authorization code: \n%v\n", authURL)
|
||||||
|
@ -640,7 +640,7 @@ func getGDriveTokenFromWeb(config *oauth2.Config, logger *log.Logger) *oauth2.To
|
||||||
logger.Fatalf("Unable to read authorization code %v", err)
|
logger.Fatalf("Unable to read authorization code %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
tok, err := config.Exchange(context.TODO(), authCode)
|
tok, err := config.Exchange(ctx, authCode)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.Fatalf("Unable to retrieve token from web %v", err)
|
logger.Fatalf("Unable to retrieve token from web %v", err)
|
||||||
}
|
}
|
||||||
|
@ -688,7 +688,9 @@ func NewStorjStorage(access, bucket string, purgeDays int, logger *log.Logger) (
|
||||||
var instance StorjStorage
|
var instance StorjStorage
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
ctx := context.TODO()
|
pCtx := context.TODO()
|
||||||
|
|
||||||
|
ctx := fpath.WithTempData(pCtx, "", true)
|
||||||
|
|
||||||
parsedAccess, err := uplink.ParseAccess(access)
|
parsedAccess, err := uplink.ParseAccess(access)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -720,12 +722,10 @@ func (s *StorjStorage) Type() string {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Head retrieves content length of a file from storage
|
// Head retrieves content length of a file from storage
|
||||||
func (s *StorjStorage) Head(token string, filename string) (contentLength uint64, err error) {
|
func (s *StorjStorage) Head(ctx context.Context, token string, filename string) (contentLength uint64, err error) {
|
||||||
key := storj.JoinPaths(token, filename)
|
key := storj.JoinPaths(token, filename)
|
||||||
|
|
||||||
ctx := context.TODO()
|
obj, err := s.project.StatObject(fpath.WithTempData(ctx, "", true), s.bucket.Name, key)
|
||||||
|
|
||||||
obj, err := s.project.StatObject(ctx, s.bucket.Name, key)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, err
|
return 0, err
|
||||||
}
|
}
|
||||||
|
@ -736,14 +736,12 @@ func (s *StorjStorage) Head(token string, filename string) (contentLength uint64
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get retrieves a file from storage
|
// Get retrieves a file from storage
|
||||||
func (s *StorjStorage) Get(token string, filename string) (reader io.ReadCloser, contentLength uint64, err error) {
|
func (s *StorjStorage) Get(ctx context.Context, token string, filename string) (reader io.ReadCloser, contentLength uint64, err error) {
|
||||||
key := storj.JoinPaths(token, filename)
|
key := storj.JoinPaths(token, filename)
|
||||||
|
|
||||||
s.logger.Printf("Getting file %s from Storj Bucket", filename)
|
s.logger.Printf("Getting file %s from Storj Bucket", filename)
|
||||||
|
|
||||||
ctx := context.TODO()
|
download, err := s.project.DownloadObject(fpath.WithTempData(ctx, "", true), s.bucket.Name, key, nil)
|
||||||
|
|
||||||
download, err := s.project.DownloadObject(ctx, s.bucket.Name, key, nil)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, 0, err
|
return nil, 0, err
|
||||||
}
|
}
|
||||||
|
@ -755,38 +753,34 @@ func (s *StorjStorage) Get(token string, filename string) (reader io.ReadCloser,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete removes a file from storage
|
// Delete removes a file from storage
|
||||||
func (s *StorjStorage) Delete(token string, filename string) (err error) {
|
func (s *StorjStorage) Delete(ctx context.Context, token string, filename string) (err error) {
|
||||||
key := storj.JoinPaths(token, filename)
|
key := storj.JoinPaths(token, filename)
|
||||||
|
|
||||||
s.logger.Printf("Deleting file %s from Storj Bucket", filename)
|
s.logger.Printf("Deleting file %s from Storj Bucket", filename)
|
||||||
|
|
||||||
ctx := context.TODO()
|
_, err = s.project.DeleteObject(fpath.WithTempData(ctx, "", true), s.bucket.Name, key)
|
||||||
|
|
||||||
_, err = s.project.DeleteObject(ctx, s.bucket.Name, key)
|
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Purge cleans up the storage
|
// Purge cleans up the storage
|
||||||
func (s *StorjStorage) Purge(days time.Duration) (err error) {
|
func (s *StorjStorage) Purge(ctx context.Context, days time.Duration) (err error) {
|
||||||
// NOOP expiration is set at upload time
|
// NOOP expiration is set at upload time
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Put saves a file on storage
|
// Put saves a file on storage
|
||||||
func (s *StorjStorage) Put(token string, filename string, reader io.Reader, contentType string, contentLength uint64) (err error) {
|
func (s *StorjStorage) Put(ctx context.Context, token string, filename string, reader io.Reader, contentType string, contentLength uint64) (err error) {
|
||||||
key := storj.JoinPaths(token, filename)
|
key := storj.JoinPaths(token, filename)
|
||||||
|
|
||||||
s.logger.Printf("Uploading file %s to Storj Bucket", filename)
|
s.logger.Printf("Uploading file %s to Storj Bucket", filename)
|
||||||
|
|
||||||
ctx := context.TODO()
|
|
||||||
|
|
||||||
var uploadOptions *uplink.UploadOptions
|
var uploadOptions *uplink.UploadOptions
|
||||||
if s.purgeDays.Hours() > 0 {
|
if s.purgeDays.Hours() > 0 {
|
||||||
uploadOptions = &uplink.UploadOptions{Expires: time.Now().Add(s.purgeDays)}
|
uploadOptions = &uplink.UploadOptions{Expires: time.Now().Add(s.purgeDays)}
|
||||||
}
|
}
|
||||||
|
|
||||||
writer, err := s.project.UploadObject(ctx, s.bucket.Name, key, uploadOptions)
|
writer, err := s.project.UploadObject(fpath.WithTempData(ctx, "", true), s.bucket.Name, key, uploadOptions)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue