1
0
Fork 0
mirror of https://github.com/Eggbertx/gochan.git synced 2025-09-04 05:56:23 -07:00
gochan/pkg/gcsql/uploads.go

156 lines
4.7 KiB
Go

package gcsql
import (
"database/sql"
"errors"
"fmt"
"strings"
"github.com/gochan-org/gochan/pkg/events"
)
const (
selectFilesBaseSQL = `SELECT
id, post_id, file_order, original_filename, filename, checksum,
file_size, is_spoilered, thumbnail_width, thumbnail_height, width, height
FROM DBPREFIXfiles `
)
var (
ErrUploadAlreadyAttached = errors.New("upload already processed")
ErrEmbedAlreadyAttached = errors.New("embed already processed")
)
// GetThreadFiles gets a list of the files owned by posts in the thread, including thumbnails for convenience.
// It does not include deleted file entries or embeds
func GetThreadFiles(post *Post) ([]Upload, error) {
query := selectFilesBaseSQL + `WHERE post_id IN (
SELECT id FROM DBPREFIXposts WHERE thread_id = (
SELECT thread_id FROM DBPREFIXposts WHERE id = ?)) AND filename != 'deleted' AND filename NOT LIKE 'embed:%'`
rows, err := Query(nil, query, post.ID)
if err != nil {
return nil, err
}
defer rows.Close()
var uploads []Upload
for rows.Next() {
var upload Upload
if err = rows.Scan(
&upload.ID, &upload.PostID, &upload.FileOrder, &upload.OriginalFilename, &upload.Filename, &upload.Checksum,
&upload.FileSize, &upload.IsSpoilered, &upload.ThumbnailWidth, &upload.ThumbnailHeight, &upload.Width, &upload.Height,
); err != nil {
return uploads, err
}
uploads = append(uploads, upload)
}
return uploads, nil
}
// NextFileOrder gets what would be the next file_order value (not particularly useful until multi-file posting is implemented)
func (p *Post) NextFileOrder(requestOpts ...*RequestOptions) (int, error) {
opts := setupOptions(requestOpts...)
const query = `SELECT COALESCE(MAX(file_order) + 1, 0) FROM DBPREFIXfiles WHERE post_id = ?`
var next int
err := QueryRow(opts, query, []any{p.ID}, []any{&next})
return next, err
}
// AddAttachment attaches an upload or an embed to a post, returning an error if the post already has an attachment
func (p *Post) AddAttachment(upload *Upload, requestOpts ...*RequestOptions) error {
if upload == nil {
return nil // no upload to attach, so no error
}
if upload.ID > 0 {
if upload.IsEmbed() {
return ErrEmbedAlreadyAttached
}
return ErrUploadAlreadyAttached
}
uploadOrEmbed := "upload"
if upload.IsEmbed() {
uploadOrEmbed = "embed"
}
opts := setupOptions(requestOpts...)
shouldCommit := opts.Tx == nil
var err error
if shouldCommit {
opts.Tx, err = BeginTx()
if err != nil {
return err
}
defer opts.Tx.Rollback()
}
filename, _, err := GetUploadFilenameAndBoard(p.ID)
if err != nil {
return fmt.Errorf("failed to check for existing %s: %w", uploadOrEmbed, err)
}
if strings.HasPrefix(filename, "embed:") {
return ErrEmbedAlreadyAttached
}
if filename != "" {
return ErrUploadAlreadyAttached
}
var recovered bool
_, err, recovered = events.TriggerEvent("incoming-"+uploadOrEmbed, upload)
if recovered {
return errors.New("recovered from a panic in an event handler (incoming-" + uploadOrEmbed + ")")
}
if err != nil {
return fmt.Errorf("unable to attach %s to post: %w", uploadOrEmbed, err)
}
const insertSQL = `INSERT INTO DBPREFIXfiles (
post_id, file_order, original_filename, filename, checksum, file_size,
is_spoilered, thumbnail_width, thumbnail_height, width, height)
VALUES(?,?,?,?,?,?,?,?,?,?,?)`
if upload.FileOrder < 1 {
upload.FileOrder, err = p.NextFileOrder(opts)
if err != nil {
return err
}
}
upload.PostID = p.ID
if _, err = Exec(opts, insertSQL,
&upload.PostID, &upload.FileOrder, &upload.OriginalFilename, &upload.Filename, &upload.Checksum, &upload.FileSize,
&upload.IsSpoilered, &upload.ThumbnailWidth, &upload.ThumbnailHeight, &upload.Width, &upload.Height,
); err != nil {
return err
}
upload.ID, err = getLatestID(opts, "DBPREFIXfiles")
if err != nil {
return err
}
if shouldCommit {
if err = opts.Tx.Commit(); err != nil {
return err
}
}
return nil
}
// AttachFile is an alias for AddAttachment
func (p *Post) AttachFile(upload *Upload, requestOpts ...*RequestOptions) error {
return p.AddAttachment(upload, requestOpts...)
}
// GetUploadFilenameAndBoard returns the filename (or an empty string) and
// the board of the given post ID
func GetUploadFilenameAndBoard(postID int) (string, string, error) {
const query = `SELECT filename, dir FROM DBPREFIXfiles
JOIN DBPREFIXposts ON post_id = DBPREFIXposts.id
JOIN DBPREFIXthreads ON thread_id = DBPREFIXthreads.id
JOIN DBPREFIXboards ON DBPREFIXboards.id = board_id
WHERE DBPREFIXposts.id = ?`
var filename, dir string
err := QueryRow(nil, query, []any{postID}, []any{&filename, &dir})
if errors.Is(err, sql.ErrNoRows) {
return "", "", nil
} else if err != nil {
return "", "", err
}
return filename, dir, nil
}