2021-07-22 23:26:21 +03:00
|
|
|
package v2
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/fasthttp/router"
|
|
|
|
"github.com/lus/pasty/internal/config"
|
|
|
|
"github.com/lus/pasty/internal/report"
|
|
|
|
"github.com/lus/pasty/internal/shared"
|
|
|
|
"github.com/lus/pasty/internal/storage"
|
|
|
|
"github.com/lus/pasty/internal/utils"
|
|
|
|
limitFasthttp "github.com/ulule/limiter/v3/drivers/middleware/fasthttp"
|
|
|
|
"github.com/valyala/fasthttp"
|
|
|
|
)
|
|
|
|
|
|
|
|
// InitializePastesController initializes the '/v2/pastes/*' controller
|
|
|
|
func InitializePastesController(group *router.Group, rateLimiterMiddleware *limitFasthttp.Middleware) {
|
|
|
|
// moms spaghetti
|
|
|
|
group.GET("/{id}", rateLimiterMiddleware.Handle(middlewareInjectPaste(endpointGetPaste)))
|
2021-10-15 18:45:17 +03:00
|
|
|
group.POST("", rateLimiterMiddleware.Handle(endpointCreatePaste))
|
2021-07-22 23:26:21 +03:00
|
|
|
group.PATCH("/{id}", rateLimiterMiddleware.Handle(middlewareInjectPaste(middlewareValidateModificationToken(endpointModifyPaste))))
|
|
|
|
group.DELETE("/{id}", rateLimiterMiddleware.Handle(middlewareInjectPaste(middlewareValidateModificationToken(endpointDeletePaste))))
|
|
|
|
|
|
|
|
if config.Current.Reports.Reports {
|
|
|
|
group.POST("/{id}/report", rateLimiterMiddleware.Handle(middlewareInjectPaste(endpointReportPaste)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// middlewareInjectPaste retrieves and injects the paste with the specified ID
|
|
|
|
func middlewareInjectPaste(next fasthttp.RequestHandler) fasthttp.RequestHandler {
|
|
|
|
return func(ctx *fasthttp.RequestCtx) {
|
|
|
|
pasteID := ctx.UserValue("id").(string)
|
|
|
|
|
|
|
|
paste, err := storage.Current.Get(pasteID)
|
|
|
|
if err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if paste == nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusNotFound)
|
|
|
|
ctx.SetBodyString("paste not found")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if paste.Metadata == nil {
|
|
|
|
paste.Metadata = map[string]interface{}{}
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.SetUserValue("_paste", paste)
|
|
|
|
|
|
|
|
next(ctx)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// middlewareValidateModificationToken extracts and validates a given modification token for an injected paste
|
|
|
|
func middlewareValidateModificationToken(next fasthttp.RequestHandler) fasthttp.RequestHandler {
|
|
|
|
return func(ctx *fasthttp.RequestCtx) {
|
|
|
|
paste := ctx.UserValue("_paste").(*shared.Paste)
|
|
|
|
|
|
|
|
authHeaderSplit := strings.SplitN(string(ctx.Request.Header.Peek("Authorization")), " ", 2)
|
|
|
|
if len(authHeaderSplit) < 2 || authHeaderSplit[0] != "Bearer" {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusUnauthorized)
|
|
|
|
ctx.SetBodyString("unauthorized")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
modificationToken := authHeaderSplit[1]
|
|
|
|
if config.Current.ModificationTokenMaster != "" && modificationToken == config.Current.ModificationTokenMaster {
|
|
|
|
next(ctx)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
valid := paste.CheckModificationToken(modificationToken)
|
|
|
|
if !valid {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusUnauthorized)
|
|
|
|
ctx.SetBodyString("unauthorized")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
next(ctx)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// endpointGetPaste handles the 'GET /v2/pastes/{id}' endpoint
|
|
|
|
func endpointGetPaste(ctx *fasthttp.RequestCtx) {
|
|
|
|
paste := ctx.UserValue("_paste").(*shared.Paste)
|
|
|
|
paste.DeletionToken = ""
|
|
|
|
paste.ModificationToken = ""
|
|
|
|
|
|
|
|
jsonData, err := json.Marshal(paste)
|
|
|
|
if err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.SetBody(jsonData)
|
|
|
|
}
|
|
|
|
|
|
|
|
type endpointCreatePastePayload struct {
|
|
|
|
Content string `json:"content"`
|
|
|
|
Metadata map[string]interface{} `json:"metadata"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// endpointCreatePaste handles the 'POST /v2/pastes' endpoint
|
|
|
|
func endpointCreatePaste(ctx *fasthttp.RequestCtx) {
|
|
|
|
// Read, parse and validate the request payload
|
|
|
|
payload := new(endpointCreatePastePayload)
|
|
|
|
if err := json.Unmarshal(ctx.PostBody(), payload); err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if payload.Content == "" {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusBadRequest)
|
|
|
|
ctx.SetBodyString("missing paste content")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if config.Current.LengthCap > 0 && len(payload.Content) > config.Current.LengthCap {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusBadRequest)
|
|
|
|
ctx.SetBodyString("too large paste content")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Acquire a new paste ID
|
|
|
|
id, err := storage.AcquireID()
|
|
|
|
if err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prepare the paste object
|
|
|
|
if payload.Metadata == nil {
|
|
|
|
payload.Metadata = map[string]interface{}{}
|
|
|
|
}
|
|
|
|
paste := &shared.Paste{
|
2021-07-30 22:51:33 +03:00
|
|
|
ID: id,
|
|
|
|
Content: payload.Content,
|
|
|
|
Created: time.Now().Unix(),
|
|
|
|
Metadata: payload.Metadata,
|
2021-07-22 23:26:21 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create a new modification token if enabled
|
|
|
|
modificationToken := ""
|
|
|
|
if config.Current.ModificationTokens {
|
2021-07-24 18:00:29 +03:00
|
|
|
modificationToken = utils.RandomString(config.Current.ModificationTokenCharacters, config.Current.ModificationTokenLength)
|
2021-07-22 23:26:21 +03:00
|
|
|
paste.ModificationToken = modificationToken
|
|
|
|
|
|
|
|
err = paste.HashModificationToken()
|
|
|
|
if err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save the paste
|
|
|
|
err = storage.Current.Save(paste)
|
|
|
|
if err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Respond with the paste
|
|
|
|
pasteCopy := *paste
|
|
|
|
pasteCopy.ModificationToken = modificationToken
|
|
|
|
jsonData, err := json.Marshal(pasteCopy)
|
|
|
|
if err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusCreated)
|
|
|
|
ctx.SetBody(jsonData)
|
|
|
|
}
|
|
|
|
|
|
|
|
type endpointModifyPastePayload struct {
|
|
|
|
Content *string `json:"content"`
|
|
|
|
Metadata map[string]interface{} `json:"metadata"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// endpointModifyPaste handles the 'PATCH /v2/pastes/{id}' endpoint
|
|
|
|
func endpointModifyPaste(ctx *fasthttp.RequestCtx) {
|
|
|
|
// Read, parse and validate the request payload
|
|
|
|
payload := new(endpointModifyPastePayload)
|
|
|
|
if err := json.Unmarshal(ctx.PostBody(), payload); err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if payload.Content != nil && *payload.Content == "" {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusBadRequest)
|
|
|
|
ctx.SetBodyString("missing paste content")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if payload.Content != nil && config.Current.LengthCap > 0 && len(*payload.Content) > config.Current.LengthCap {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusBadRequest)
|
|
|
|
ctx.SetBodyString("too large paste content")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Modify the paste itself
|
|
|
|
paste := ctx.UserValue("_paste").(*shared.Paste)
|
|
|
|
if payload.Content != nil {
|
|
|
|
paste.Content = *payload.Content
|
|
|
|
}
|
|
|
|
if payload.Metadata != nil {
|
|
|
|
for key, value := range payload.Metadata {
|
|
|
|
if value == nil {
|
|
|
|
delete(paste.Metadata, key)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
paste.Metadata[key] = value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save the modified paste
|
|
|
|
if err := storage.Current.Save(paste); err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// endpointDeletePaste handles the 'DELETE /v2/pastes/{id}' endpoint
|
|
|
|
func endpointDeletePaste(ctx *fasthttp.RequestCtx) {
|
|
|
|
paste := ctx.UserValue("_paste").(*shared.Paste)
|
|
|
|
if err := storage.Current.Delete(paste.ID); err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type endpointReportPastePayload struct {
|
|
|
|
Reason string `json:"reason"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func endpointReportPaste(ctx *fasthttp.RequestCtx) {
|
|
|
|
// Read, parse and validate the request payload
|
|
|
|
payload := new(endpointReportPastePayload)
|
|
|
|
if err := json.Unmarshal(ctx.PostBody(), payload); err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if payload.Reason == "" {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusBadRequest)
|
|
|
|
ctx.SetBodyString("missing report reason")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
request := &report.ReportRequest{
|
2021-07-24 18:03:40 +03:00
|
|
|
Paste: ctx.UserValue("_paste").(*shared.Paste).ID,
|
|
|
|
Reason: payload.Reason,
|
2021-07-22 23:26:21 +03:00
|
|
|
}
|
|
|
|
response, err := report.SendReport(request)
|
|
|
|
if err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
jsonData, err := json.Marshal(response)
|
|
|
|
if err != nil {
|
|
|
|
ctx.SetStatusCode(fasthttp.StatusInternalServerError)
|
|
|
|
ctx.SetBodyString(err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.SetBody(jsonData)
|
|
|
|
}
|