package api import ( "fmt" "net/url" "path/filepath" "git.solsynth.dev/hypernet/nexus/pkg/nex/sec" "git.solsynth.dev/hypernet/paperclip/pkg/internal/database" "git.solsynth.dev/hypernet/paperclip/pkg/internal/server/exts" "git.solsynth.dev/hypernet/paperclip/pkg/internal/models" "git.solsynth.dev/hypernet/paperclip/pkg/internal/services" "github.com/gofiber/fiber/v2" jsoniter "github.com/json-iterator/go" "github.com/samber/lo" "github.com/spf13/viper" ) func openAttachment(c *fiber.Ctx) error { id := c.Params("id") metadata, err := services.GetAttachmentByRID(id) if err != nil { return fiber.NewError(fiber.StatusNotFound) } else if !metadata.IsUploaded { return fiber.NewError(fiber.StatusNotFound, "file is in uploading progress, please wait until all chunk uploaded") } destMap := viper.GetStringMap(fmt.Sprintf("destinations.%d", metadata.Destination)) var dest models.BaseDestination rawDest, _ := jsoniter.Marshal(destMap) _ = jsoniter.Unmarshal(rawDest, &dest) switch dest.Type { case models.DestinationTypeLocal: var destConfigured models.LocalDestination _ = jsoniter.Unmarshal(rawDest, &destConfigured) if len(destConfigured.AccessBaseURL) > 0 && !c.QueryBool("direct", false) { // This will drop all query parameters, // for not it's okay because the openAttachment api won't take any query parameters return c.Redirect(fmt.Sprintf( "%s%s?direct=true", destConfigured.AccessBaseURL, c.Path(), ), fiber.StatusMovedPermanently) } if len(metadata.MimeType) > 0 { c.Set(fiber.HeaderContentType, metadata.MimeType) } return c.SendFile(filepath.Join(destConfigured.Path, metadata.Uuid)) case models.DestinationTypeS3: var destConfigured models.S3Destination _ = jsoniter.Unmarshal(rawDest, &destConfigured) if len(destConfigured.AccessBaseURL) > 0 { return c.Redirect(fmt.Sprintf( "%s/%s", destConfigured.AccessBaseURL, url.QueryEscape(filepath.Join(destConfigured.Path, metadata.Uuid)), ), fiber.StatusMovedPermanently) } else { protocol := lo.Ternary(destConfigured.EnableSSL, "https", "http") return c.Redirect(fmt.Sprintf( "%s://%s.%s/%s", protocol, destConfigured.Bucket, destConfigured.Endpoint, url.QueryEscape(filepath.Join(destConfigured.Path, metadata.Uuid)), ), fiber.StatusMovedPermanently) } default: return fmt.Errorf("invalid destination: unsupported protocol %s", dest.Type) } } func getAttachmentMeta(c *fiber.Ctx) error { id := c.Params("id") metadata, err := services.GetAttachmentByRID(id) if err != nil { return fiber.NewError(fiber.StatusNotFound) } return c.JSON(metadata) } func updateAttachmentMeta(c *fiber.Ctx) error { id, _ := c.ParamsInt("id", 0) user := c.Locals("nex_user").(*sec.UserInfo) var data struct { Alternative *string `json:"alt"` Thumbnail *string `json:"thumbnail"` Metadata *map[string]any `json:"metadata"` IsIndexable *bool `json:"is_indexable"` } if err := exts.BindAndValidate(c, &data); err != nil { return err } var attachment models.Attachment if err := database.C.Where("id = ? AND account_id = ?", id, user.ID).First(&attachment).Error; err != nil { return fiber.NewError(fiber.StatusBadRequest, err.Error()) } if data.Alternative != nil { attachment.Alternative = *data.Alternative } if data.Thumbnail != nil { attachment.Thumbnail = *data.Thumbnail } if data.Metadata != nil { attachment.Usermeta = *data.Metadata } if data.IsIndexable != nil { attachment.IsIndexable = *data.IsIndexable } if attachment, err := services.UpdateAttachment(attachment); err != nil { return fiber.NewError(fiber.StatusBadRequest, err.Error()) } else { return c.JSON(attachment) } } func deleteAttachment(c *fiber.Ctx) error { id, _ := c.ParamsInt("id", 0) user := c.Locals("nex_user").(*sec.UserInfo) attachment, err := services.GetAttachmentByID(uint(id)) if err != nil { return fiber.NewError(fiber.StatusNotFound, err.Error()) } else if attachment.AccountID != user.ID { return fiber.NewError(fiber.StatusNotFound, "record not created by you") } if err := services.DeleteAttachment(attachment); err != nil { return fiber.NewError(fiber.StatusInternalServerError, err.Error()) } else { return c.SendStatus(fiber.StatusOK) } }