2024-08-20 09:08:40 +00:00
package api
import (
2024-08-20 14:55:58 +00:00
"encoding/json"
2024-08-20 09:08:40 +00:00
"fmt"
2024-12-26 13:53:09 +00:00
2024-10-27 05:13:40 +00:00
"git.solsynth.dev/hypernet/nexus/pkg/nex/sec"
2024-11-02 02:41:31 +00:00
"git.solsynth.dev/hypernet/paperclip/pkg/internal/database"
2024-12-28 06:07:53 +00:00
"git.solsynth.dev/hypernet/paperclip/pkg/internal/fs"
2024-11-02 02:41:31 +00:00
"git.solsynth.dev/hypernet/paperclip/pkg/internal/models"
"git.solsynth.dev/hypernet/paperclip/pkg/internal/server/exts"
"git.solsynth.dev/hypernet/paperclip/pkg/internal/services"
2024-08-20 09:08:40 +00:00
"github.com/gofiber/fiber/v2"
"github.com/spf13/viper"
)
2024-12-28 05:31:31 +00:00
func createAttachmentFragment ( c * fiber . Ctx ) error {
2024-11-02 18:14:56 +00:00
user := c . Locals ( "nex_user" ) . ( * sec . UserInfo )
2024-08-20 09:08:40 +00:00
var data struct {
Pool string ` json:"pool" validate:"required" `
Size int64 ` json:"size" validate:"required" `
2024-08-20 14:55:58 +00:00
FileName string ` json:"name" validate:"required" `
2024-08-20 09:08:40 +00:00
Alternative string ` json:"alt" `
MimeType string ` json:"mimetype" `
2024-12-28 05:31:31 +00:00
Fingerprint * string ` json:"fingerprint" `
2024-08-20 09:08:40 +00:00
Metadata map [ string ] any ` json:"metadata" `
}
2024-08-20 14:55:58 +00:00
if err := exts . BindAndValidate ( c , & data ) ; err != nil {
return err
}
2024-08-20 09:08:40 +00:00
aliasingMap := viper . GetStringMapString ( "pools.aliases" )
if val , ok := aliasingMap [ data . Pool ] ; ok {
data . Pool = val
}
pool , err := services . GetAttachmentPoolByAlias ( data . Pool )
if err != nil {
return fiber . NewError ( fiber . StatusBadRequest , fmt . Sprintf ( "unable to get attachment pool info: %v" , err ) )
}
2024-10-27 05:13:40 +00:00
if ! user . HasPermNode ( "CreateAttachments" , data . Size ) {
return fiber . NewError ( fiber . StatusForbidden , "you are not permitted to create attachments like this large" )
2024-08-20 09:08:40 +00:00
} else if pool . Config . Data ( ) . MaxFileSize != nil && * pool . Config . Data ( ) . MaxFileSize > data . Size {
return fiber . NewError ( fiber . StatusBadRequest , fmt . Sprintf ( "attachment pool %s doesn't allow file larger than %d" , pool . Alias , * pool . Config . Data ( ) . MaxFileSize ) )
}
2024-12-28 05:31:31 +00:00
metadata , err := services . NewAttachmentFragment ( database . C , user , models . AttachmentFragment {
2024-08-20 14:55:58 +00:00
Name : data . FileName ,
Size : data . Size ,
2024-08-20 09:08:40 +00:00
Alternative : data . Alternative ,
MimeType : data . MimeType ,
2024-12-26 13:53:09 +00:00
Usermeta : data . Metadata ,
2024-12-28 05:31:31 +00:00
Fingerprint : data . Fingerprint ,
2024-08-20 09:08:40 +00:00
Pool : & pool ,
PoolID : & pool . ID ,
} )
if err != nil {
return fiber . NewError ( fiber . StatusBadRequest , err . Error ( ) )
2024-12-28 05:56:25 +00:00
} else {
metadata . FileChunksMissing = services . FindFragmentMissingChunks ( metadata )
2024-08-20 09:08:40 +00:00
}
2024-08-20 11:17:43 +00:00
return c . JSON ( fiber . Map {
2024-08-20 14:55:58 +00:00
"chunk_size" : viper . GetInt64 ( "performance.file_chunk_size" ) ,
"chunk_count" : len ( metadata . FileChunks ) ,
"meta" : metadata ,
2024-08-20 11:17:43 +00:00
} )
2024-08-20 09:08:40 +00:00
}
2024-12-28 05:31:31 +00:00
func uploadFragmentChunk ( c * fiber . Ctx ) error {
2024-11-02 18:14:56 +00:00
user := c . Locals ( "nex_user" ) . ( * sec . UserInfo )
2024-08-20 09:08:40 +00:00
rid := c . Params ( "file" )
cid := c . Params ( "chunk" )
2024-11-05 12:56:53 +00:00
fileData := c . Body ( )
if len ( fileData ) == 0 {
return fiber . NewError ( fiber . StatusBadRequest , "no file data" )
} else if len ( fileData ) > viper . GetInt ( "performance.file_chunk_size" ) {
2024-08-20 09:08:40 +00:00
return fiber . NewError ( fiber . StatusBadRequest , "file is too large for one chunk" )
}
2024-12-28 05:31:31 +00:00
meta , err := services . GetFragmentByRID ( rid )
2024-08-20 09:08:40 +00:00
if err != nil {
return fiber . NewError ( fiber . StatusNotFound , fmt . Sprintf ( "attachment was not found: %v" , err ) )
} else if user . ID != meta . AccountID {
return fiber . NewError ( fiber . StatusForbidden , "you are not authorized to upload this attachment" )
}
if _ , ok := meta . FileChunks [ cid ] ; ! ok {
return fiber . NewError ( fiber . StatusNotFound , fmt . Sprintf ( "chunk %s was not found" , cid ) )
2024-12-28 05:31:31 +00:00
} else if services . CheckFragmentChunkExists ( meta , cid ) {
2024-08-20 09:08:40 +00:00
return fiber . NewError ( fiber . StatusNotFound , fmt . Sprintf ( "chunk %s was uploaded" , cid ) )
}
2024-12-28 05:31:31 +00:00
if err := services . UploadFragmentChunkBytes ( c , cid , fileData , meta ) ; err != nil {
2024-08-20 09:08:40 +00:00
return fiber . NewError ( fiber . StatusBadRequest , err . Error ( ) )
}
chunkArrange := make ( [ ] string , len ( meta . FileChunks ) )
isAllUploaded := true
for cid , idx := range meta . FileChunks {
2024-12-28 05:31:31 +00:00
if ! services . CheckFragmentChunkExists ( meta , cid ) {
2024-08-20 09:08:40 +00:00
isAllUploaded = false
break
2024-08-20 14:55:58 +00:00
} else if val , ok := idx . ( json . Number ) ; ok {
data , _ := val . Int64 ( )
chunkArrange [ data ] = cid
2024-08-20 09:08:40 +00:00
}
}
if ! isAllUploaded {
2024-12-28 08:54:30 +00:00
return c . JSON ( fiber . Map {
"is_finished" : false ,
"fragment" : meta ,
} )
2024-08-20 09:08:40 +00:00
}
2024-12-28 06:07:53 +00:00
// Merge & post-upload
attachment , err := fs . MergeFileChunks ( meta , chunkArrange )
2024-12-26 14:55:16 +00:00
if err != nil {
2024-08-20 09:08:40 +00:00
return fiber . NewError ( fiber . StatusInternalServerError , err . Error ( ) )
2024-12-28 06:07:53 +00:00
}
// Post-upload tasks
if err := database . C . Save ( & attachment ) . Error ; err != nil {
return fiber . NewError ( fiber . StatusInternalServerError , err . Error ( ) )
}
2024-12-29 15:19:26 +00:00
if c . QueryBool ( "analyzeNow" , false ) {
2024-12-28 05:31:31 +00:00
services . AnalyzeAttachment ( attachment )
2024-08-20 09:08:40 +00:00
} else {
2024-12-28 05:31:31 +00:00
services . PublishAnalyzeTask ( attachment )
2024-08-20 09:08:40 +00:00
}
2024-12-26 14:55:16 +00:00
2024-12-28 08:54:30 +00:00
return c . JSON ( fiber . Map {
"is_finished" : true ,
"attachment" : attachment ,
} )
2024-08-20 09:08:40 +00:00
}