381 lines
8.2 KiB
Go
381 lines
8.2 KiB
Go
package routes
|
|
|
|
import (
|
|
"archive/zip"
|
|
"bytes"
|
|
"filething/models"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"net/url"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/labstack/echo/v4"
|
|
)
|
|
|
|
type UploadResponse struct {
|
|
Usage int64 `json:"usage"`
|
|
File File `json:"file"`
|
|
}
|
|
|
|
func UploadFile(c echo.Context) error {
|
|
user := c.Get("user").(*models.User)
|
|
|
|
fullPath := strings.Trim(c.Param("*"), "/")
|
|
basePath := fmt.Sprintf("%s/%s/%s/", os.Getenv("STORAGE_PATH"), user.ID, fullPath)
|
|
|
|
currentUsage, err := calculateStorageUsage(fmt.Sprintf("%s/%s", os.Getenv("STORAGE_PATH"), user.ID))
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
|
|
_, err = os.Stat(basePath)
|
|
directoryExists := err == nil
|
|
|
|
// Create the directories if they don't exist
|
|
if !directoryExists {
|
|
err = os.MkdirAll(basePath, os.ModePerm)
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
}
|
|
|
|
reader, err := c.Request().MultipartReader()
|
|
if err != nil {
|
|
if err == http.ErrNotMultipart {
|
|
if directoryExists {
|
|
// Directories exist, but no file was uploaded
|
|
return c.JSON(http.StatusBadRequest, map[string]string{"message": "A folder with that name already exists"})
|
|
}
|
|
// Directories were just created, and no file was provided
|
|
entry, err := os.Stat(basePath)
|
|
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
|
|
uploadFile := &UploadResponse{
|
|
Usage: currentUsage + entry.Size(),
|
|
File: File{
|
|
Name: entry.Name(),
|
|
IsDir: entry.IsDir(),
|
|
Size: entry.Size(),
|
|
LastModified: entry.ModTime().Format("1/2/2006"),
|
|
},
|
|
}
|
|
|
|
return c.JSON(http.StatusOK, uploadFile)
|
|
}
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
|
|
part, err := reader.NextPart()
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
|
|
filepath := filepath.Join(basePath, part.FileName())
|
|
|
|
if _, err = os.Stat(filepath); err == nil {
|
|
return c.JSON(http.StatusConflict, map[string]string{"message": "File with that name already exists"})
|
|
}
|
|
|
|
dst, err := os.Create(filepath)
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
defer dst.Close()
|
|
|
|
// Read the file manually because otherwise we are limited by the arbitrarily small size of /tmp
|
|
buffer := make([]byte, 4096)
|
|
totalSize := int64(0)
|
|
|
|
for {
|
|
n, readErr := part.Read(buffer)
|
|
|
|
if readErr != nil && readErr == io.ErrUnexpectedEOF {
|
|
dst.Close()
|
|
os.Remove(filepath)
|
|
return c.JSON(http.StatusRequestTimeout, map[string]string{"message": "Upload canceled"})
|
|
}
|
|
|
|
if readErr != nil && readErr != io.EOF {
|
|
fmt.Println(err)
|
|
return readErr
|
|
}
|
|
|
|
totalSize += int64(n)
|
|
|
|
if currentUsage+totalSize > user.Plan.MaxStorage {
|
|
dst.Close()
|
|
os.Remove(filepath)
|
|
return c.JSON(http.StatusInsufficientStorage, map[string]string{"message": "Insufficient storage space"})
|
|
}
|
|
|
|
if _, err := dst.Write(buffer[:n]); err != nil {
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
|
|
if n == 0 || readErr == io.EOF {
|
|
entry, err := os.Stat(filepath)
|
|
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
|
|
uploadFile := &UploadResponse{
|
|
Usage: currentUsage + totalSize,
|
|
File: File{
|
|
Name: entry.Name(),
|
|
IsDir: entry.IsDir(),
|
|
Size: entry.Size(),
|
|
LastModified: entry.ModTime().Format("1/2/2006"),
|
|
},
|
|
}
|
|
|
|
return c.JSON(http.StatusOK, uploadFile)
|
|
}
|
|
}
|
|
}
|
|
|
|
func calculateStorageUsage(basePath string) (int64, error) {
|
|
var totalSize int64
|
|
entries, err := os.ReadDir(basePath)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
for _, entry := range entries {
|
|
if entry.IsDir() {
|
|
// Recursively calculate size of directories
|
|
dirPath := filepath.Join(basePath, entry.Name())
|
|
dirSize, err := calculateStorageUsage(dirPath)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
totalSize += dirSize
|
|
} else {
|
|
// Calculate size of file
|
|
_ = filepath.Join(basePath, entry.Name())
|
|
fileInfo, err := entry.Info()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
totalSize += fileInfo.Size()
|
|
}
|
|
}
|
|
|
|
return totalSize, nil
|
|
}
|
|
|
|
type File struct {
|
|
Name string `json:"name"`
|
|
IsDir bool `json:"is_dir"`
|
|
Size int64 `json:"size"`
|
|
LastModified string `json:"last_modified"`
|
|
}
|
|
|
|
func GetFiles(c echo.Context) error {
|
|
user := c.Get("user").(*models.User)
|
|
|
|
fullPath := strings.Trim(c.Param("*"), "/")
|
|
basePath := fmt.Sprintf("%s/%s/%s/", os.Getenv("STORAGE_PATH"), user.ID, fullPath)
|
|
|
|
f, err := os.Open(basePath)
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
|
|
files, err := f.Readdir(0)
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
|
|
jsonFiles := make([]File, 0)
|
|
|
|
for _, f := range files {
|
|
jsonFiles = append(jsonFiles, File{
|
|
Name: f.Name(),
|
|
IsDir: f.IsDir(),
|
|
Size: f.Size(),
|
|
LastModified: f.ModTime().Format("2 Jan 06"),
|
|
})
|
|
}
|
|
|
|
return c.JSON(http.StatusOK, jsonFiles)
|
|
}
|
|
|
|
func GetFile(c echo.Context) error {
|
|
user := c.Get("user").(*models.User)
|
|
|
|
fullPath := strings.Trim(c.Param("*"), "/")
|
|
|
|
fileNamesParam := c.QueryParam("filenames")
|
|
var fileNames []string
|
|
if fileNamesParam != "" {
|
|
fileNames = strings.Split(fileNamesParam, ",")
|
|
}
|
|
|
|
if fullPath == "" && len(fileNames) == 0 {
|
|
return c.JSON(http.StatusBadRequest, map[string]string{"message": "A file is required"})
|
|
}
|
|
|
|
basePath := fmt.Sprintf("%s/%s", os.Getenv("STORAGE_PATH"), user.ID)
|
|
if fullPath != "" {
|
|
basePath = filepath.Join(basePath, fullPath)
|
|
}
|
|
|
|
fileInfo, err := os.Stat(basePath)
|
|
if err != nil {
|
|
return c.JSON(http.StatusNotFound, map[string]string{"message": "No file found!"})
|
|
}
|
|
|
|
var buf bytes.Buffer
|
|
if fileInfo.IsDir() {
|
|
c.Response().Header().Set(echo.HeaderContentType, "application/zip")
|
|
|
|
if len(fileNames) != 0 {
|
|
err := zipFiles(&buf, filepath.Join(basePath, fullPath), fileNames)
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
|
|
_, err = buf.WriteTo(c.Response().Writer)
|
|
return err
|
|
}
|
|
|
|
err := zipFiles(&buf, basePath, []string{""})
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
return err
|
|
}
|
|
|
|
_, err = buf.WriteTo(c.Response().Writer)
|
|
return err
|
|
} else {
|
|
return c.File(basePath)
|
|
}
|
|
}
|
|
|
|
func zipFiles(buf *bytes.Buffer, basePath string, files []string) error {
|
|
zipWriter := zip.NewWriter(buf)
|
|
defer zipWriter.Close()
|
|
|
|
for _, filePath := range files {
|
|
unescapedFilePath, err := url.PathUnescape(filePath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = processFile(zipWriter, basePath, unescapedFilePath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func processFile(zipWriter *zip.Writer, basePath string, filePath string) error {
|
|
fullFilePath := filepath.Join(basePath, filePath)
|
|
|
|
fileInfo, err := os.Stat(fullFilePath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
header, err := zip.FileInfoHeader(fileInfo)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
header.Method = zip.Deflate
|
|
|
|
header.Name = filepath.ToSlash(filePath)
|
|
|
|
if fileInfo.IsDir() {
|
|
header.Name += "/"
|
|
}
|
|
|
|
headerWriter, err := zipWriter.CreateHeader(header)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if fileInfo.IsDir() {
|
|
files, err := os.ReadDir(fullFilePath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, file := range files {
|
|
err := processFile(zipWriter, basePath, filepath.Join(filePath, file.Name()))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
file, err := os.Open(fullFilePath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer file.Close()
|
|
|
|
_, err = io.Copy(headerWriter, file)
|
|
return err
|
|
}
|
|
|
|
type DeleteRequest struct {
|
|
Files []File `json:"files"`
|
|
}
|
|
|
|
func DeleteFiles(c echo.Context) error {
|
|
var deleteData DeleteRequest
|
|
|
|
if err := c.Bind(&deleteData); err != nil {
|
|
fmt.Println(err)
|
|
return c.JSON(http.StatusInternalServerError, map[string]string{"message": "An unknown error occoured!"})
|
|
}
|
|
|
|
if len(deleteData.Files) == 0 {
|
|
return c.JSON(http.StatusBadRequest, map[string]string{"message": "Files are required!"})
|
|
}
|
|
|
|
user := c.Get("user").(*models.User)
|
|
|
|
fullPath := strings.Trim(c.Param("*"), "/")
|
|
basePath := fmt.Sprintf("%s/%s/%s", os.Getenv("STORAGE_PATH"), user.ID, fullPath)
|
|
|
|
for _, file := range deleteData.Files {
|
|
path := filepath.Join(basePath, file.Name)
|
|
err := os.RemoveAll(path)
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
return c.JSON(http.StatusInternalServerError, map[string]string{"message": "An unknown error occoured!"})
|
|
}
|
|
}
|
|
|
|
word := "file"
|
|
fileLen := len(deleteData.Files)
|
|
|
|
if fileLen != 1 {
|
|
word = word + "s"
|
|
}
|
|
|
|
return c.JSON(http.StatusOK, map[string]string{"message": fmt.Sprintf("Successfully deleted %d %s", fileLen, word)})
|
|
}
|