wings/router/router_download.go

111 lines
2.9 KiB
Go
Raw Permalink Normal View History

2020-04-06 01:00:33 +00:00
package router
2020-04-06 01:56:54 +00:00
import (
"bufio"
"errors"
2020-04-06 01:56:54 +00:00
"net/http"
"os"
"strconv"
2021-01-10 01:22:39 +00:00
"github.com/gin-gonic/gin"
"github.com/google/uuid"
2021-01-26 04:28:24 +00:00
"github.com/pterodactyl/wings/router/middleware"
2021-01-10 01:22:39 +00:00
"github.com/pterodactyl/wings/router/tokens"
"github.com/pterodactyl/wings/server/backup"
2020-04-06 01:56:54 +00:00
)
// Handle a download request for a server backup.
2020-04-06 01:56:54 +00:00
func getDownloadBackup(c *gin.Context) {
2021-02-02 05:32:34 +00:00
client := middleware.ExtractApiClient(c)
2021-01-26 04:28:24 +00:00
manager := middleware.ExtractManager(c)
// Get the payload from the token.
2020-04-06 01:56:54 +00:00
token := tokens.BackupPayload{}
if err := tokens.ParseToken([]byte(c.Query("token")), &token); err != nil {
middleware.CaptureAndAbort(c, err)
2020-04-06 01:56:54 +00:00
return
}
// Get the server using the UUID from the token.
if _, ok := manager.Get(token.ServerUuid); !ok || !token.IsUniqueRequest() {
2020-04-06 01:56:54 +00:00
c.AbortWithStatusJSON(http.StatusNotFound, gin.H{
"error": "The requested resource was not found on this server.",
})
return
}
// Validate that the BackupUuid field is actually a UUID and not some random characters or a
// file path.
if _, err := uuid.Parse(token.BackupUuid); err != nil {
middleware.CaptureAndAbort(c, err)
return
}
// Locate the backup on the local disk.
2021-02-02 05:32:34 +00:00
b, st, err := backup.LocateLocal(client, token.BackupUuid)
2020-04-06 01:56:54 +00:00
if err != nil {
if errors.Is(err, os.ErrNotExist) {
c.AbortWithStatusJSON(http.StatusNotFound, gin.H{
"error": "The requested backup was not found on this server.",
})
return
}
middleware.CaptureAndAbort(c, err)
2020-04-06 01:56:54 +00:00
return
}
// The use of `os` here is safe as backups are not stored within server
// accessible directories.
f, err := os.Open(b.Path())
2020-04-06 01:56:54 +00:00
if err != nil {
middleware.CaptureAndAbort(c, err)
2020-04-06 01:56:54 +00:00
return
}
defer f.Close()
c.Header("Content-Length", strconv.Itoa(int(st.Size())))
c.Header("Content-Disposition", "attachment; filename="+strconv.Quote(st.Name()))
2020-04-06 01:56:54 +00:00
c.Header("Content-Type", "application/octet-stream")
_, _ = bufio.NewReader(f).WriteTo(c.Writer)
2020-04-06 01:56:54 +00:00
}
// Handles downloading a specific file for a server.
func getDownloadFile(c *gin.Context) {
2021-01-26 04:28:24 +00:00
manager := middleware.ExtractManager(c)
token := tokens.FilePayload{}
if err := tokens.ParseToken([]byte(c.Query("token")), &token); err != nil {
middleware.CaptureAndAbort(c, err)
return
}
2021-01-26 04:28:24 +00:00
s, ok := manager.Get(token.ServerUuid)
if !ok || !token.IsUniqueRequest() {
c.AbortWithStatusJSON(http.StatusNotFound, gin.H{
"error": "The requested resource was not found on this server.",
})
return
}
f, st, err := s.Filesystem().File(token.FilePath)
if err != nil {
middleware.CaptureAndAbort(c, err)
return
}
defer f.Close()
if st.IsDir() {
c.AbortWithStatusJSON(http.StatusNotFound, gin.H{
"error": "The requested resource was not found on this server.",
})
return
}
c.Header("Content-Length", strconv.Itoa(int(st.Size())))
c.Header("Content-Disposition", "attachment; filename="+strconv.Quote(st.Name()))
c.Header("Content-Type", "application/octet-stream")
_, _ = bufio.NewReader(f).WriteTo(c.Writer)
}