| 
									
										
										
										
											2023-03-12 16:00:57 +01:00
										 |  |  | // GoToSocial | 
					
						
							|  |  |  | // Copyright (C) GoToSocial Authors admin@gotosocial.org | 
					
						
							|  |  |  | // SPDX-License-Identifier: AGPL-3.0-or-later | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // This program is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // it under the terms of the GNU Affero General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  | // GNU Affero General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Affero General Public License | 
					
						
							|  |  |  | // along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2022-01-08 17:17:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-02 15:00:53 +01:00
										 |  |  | package media | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-03 17:37:38 +01:00
										 |  |  | import ( | 
					
						
							| 
									
										
										
										
											2022-01-04 17:37:54 +01:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2022-01-08 17:17:01 +01:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2022-01-03 17:37:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	errorsv2 "codeberg.org/gruf/go-errors/v2" | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	"codeberg.org/gruf/go-runners" | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/gtscontext" | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/gtserror" | 
					
						
							| 
									
										
										
										
											2022-01-03 17:37:38 +01:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel" | 
					
						
							| 
									
										
										
										
											2022-07-19 09:47:55 +01:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/log" | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/storage" | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/uris" | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/util" | 
					
						
							| 
									
										
										
										
											2022-01-03 17:37:38 +01:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2022-01-02 15:00:53 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | // ProcessingMedia represents a piece of media | 
					
						
							|  |  |  | // currently being processed. It exposes functions | 
					
						
							|  |  |  | // for retrieving data from the process. | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | type ProcessingMedia struct { | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 	media  *gtsmodel.MediaAttachment // processing media attachment details | 
					
						
							|  |  |  | 	dataFn DataFunc                  // load-data function, returns media stream | 
					
						
							|  |  |  | 	done   bool                      // done is set when process finishes with non ctx canceled type error | 
					
						
							|  |  |  | 	proc   runners.Processor         // proc helps synchronize only a singular running processing instance | 
					
						
							|  |  |  | 	err    error                     // error stores permanent error value when done | 
					
						
							|  |  |  | 	mgr    *Manager                  // mgr instance (access to db / storage) | 
					
						
							| 
									
										
										
										
											2022-01-03 17:37:38 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | // ID returns the ID of the underlying media. | 
					
						
							|  |  |  | func (p *ProcessingMedia) ID() string { | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	return p.media.ID // immutable, safe outside mutex. | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | // LoadAttachment blocks until the thumbnail and | 
					
						
							|  |  |  | // fullsize content has been processed, and then | 
					
						
							|  |  |  | // returns the attachment. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // If processing could not be completed fully | 
					
						
							|  |  |  | // then an error will be returned. The attachment | 
					
						
							|  |  |  | // will still be returned in that case, but it will | 
					
						
							|  |  |  | // only be partially complete and should be treated | 
					
						
							|  |  |  | // as a placeholder. | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | func (p *ProcessingMedia) Load(ctx context.Context) (*gtsmodel.MediaAttachment, error) { | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	media, done, err := p.load(ctx) | 
					
						
							|  |  |  | 	if !done { | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 		// On a context-canceled error (marked as !done), requeue for loading. | 
					
						
							|  |  |  | 		p.mgr.state.Workers.Dereference.Queue.Push(func(ctx context.Context) { | 
					
						
							|  |  |  | 			if _, _, err := p.load(ctx); err != nil { | 
					
						
							|  |  |  | 				log.Errorf(ctx, "error loading media: %v", err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	return media, err | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | // load is the package private form of load() that is wrapped to catch context canceled. | 
					
						
							|  |  |  | func (p *ProcessingMedia) load(ctx context.Context) ( | 
					
						
							|  |  |  | 	media *gtsmodel.MediaAttachment, | 
					
						
							|  |  |  | 	done bool, | 
					
						
							|  |  |  | 	err error, | 
					
						
							|  |  |  | ) { | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	err = p.proc.Process(func() error { | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 		if done = p.done; done { | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 			// Already proc'd. | 
					
						
							|  |  |  | 			return p.err | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-01-16 18:52:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		defer func() { | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 			// This is only done when ctx NOT cancelled. | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 			done = (err == nil || !errorsv2.IsV2(err, | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 				context.Canceled, | 
					
						
							|  |  |  | 				context.DeadlineExceeded, | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 			)) | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if !done { | 
					
						
							|  |  |  | 				return | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 			// Anything from here, we | 
					
						
							|  |  |  | 			// need to ensure happens | 
					
						
							|  |  |  | 			// (i.e. no ctx canceled). | 
					
						
							|  |  |  | 			ctx = gtscontext.WithValues( | 
					
						
							|  |  |  | 				context.Background(), | 
					
						
							|  |  |  | 				ctx, // values | 
					
						
							|  |  |  | 			) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// On error or unknown media types, perform error cleanup. | 
					
						
							|  |  |  | 			if err != nil || p.media.Type == gtsmodel.FileTypeUnknown { | 
					
						
							|  |  |  | 				p.cleanup(ctx) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// Update with latest details, whatever happened. | 
					
						
							|  |  |  | 			e := p.mgr.state.DB.UpdateAttachment(ctx, p.media) | 
					
						
							|  |  |  | 			if e != nil { | 
					
						
							|  |  |  | 				log.Errorf(ctx, "error updating media in db: %v", e) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 			// Store final values. | 
					
						
							|  |  |  | 			p.done = true | 
					
						
							|  |  |  | 			p.err = err | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		}() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 		// TODO: in time update this | 
					
						
							|  |  |  | 		// to perhaps follow a similar | 
					
						
							|  |  |  | 		// freshness window to statuses | 
					
						
							|  |  |  | 		// / accounts? But that's a big | 
					
						
							|  |  |  | 		// maybe, media don't change in | 
					
						
							|  |  |  | 		// the same way so this is largely | 
					
						
							|  |  |  | 		// just to slow down fail retries. | 
					
						
							|  |  |  | 		const maxfreq = 6 * time.Hour | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Check whether media is uncached but repeatedly failing, | 
					
						
							|  |  |  | 		// specifically limit the frequency at which we allow this. | 
					
						
							|  |  |  | 		if !p.media.UpdatedAt.Equal(p.media.CreatedAt) && // i.e. not new | 
					
						
							|  |  |  | 			p.media.UpdatedAt.Add(maxfreq).Before(time.Now()) { | 
					
						
							|  |  |  | 			return nil | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		// Attempt to store media and calculate | 
					
						
							|  |  |  | 		// full-size media attachment details. | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// | 
					
						
							|  |  |  | 		// This will update p.media as it goes. | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		err = p.store(ctx) | 
					
						
							|  |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 	media = p.media | 
					
						
							|  |  |  | 	return | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | // store calls the data function attached to p if it hasn't been called yet, | 
					
						
							|  |  |  | // and updates the underlying attachment fields as necessary. It will then stream | 
					
						
							|  |  |  | // bytes from p's reader directly into storage so that it can be retrieved later. | 
					
						
							|  |  |  | func (p *ProcessingMedia) store(ctx context.Context) error { | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Load media from data func. | 
					
						
							|  |  |  | 	rc, err := p.dataFn(ctx) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 		return gtserror.Newf("error executing data function: %w", err) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	var ( | 
					
						
							|  |  |  | 		// predfine temporary media | 
					
						
							|  |  |  | 		// file path variables so we | 
					
						
							|  |  |  | 		// can remove them on error. | 
					
						
							|  |  |  | 		temppath  string | 
					
						
							|  |  |  | 		thumbpath string | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	defer func() { | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		if err := remove(temppath, thumbpath); err != nil { | 
					
						
							|  |  |  | 			log.Errorf(ctx, "error(s) cleaning up files: %v", err) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							| 
									
										
										
										
											2022-03-22 12:42:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Drain reader to tmp file | 
					
						
							|  |  |  | 	// (this reader handles close). | 
					
						
							|  |  |  | 	temppath, err = drainToTmp(rc) | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		return gtserror.Newf("error draining data to tmp: %w", err) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-12-17 05:38:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Pass input file through ffprobe to | 
					
						
							|  |  |  | 	// parse further metadata information. | 
					
						
							|  |  |  | 	result, err := ffprobe(ctx, temppath) | 
					
						
							| 
									
										
										
										
											2024-07-15 14:24:53 +00:00
										 |  |  | 	if err != nil && !isUnsupportedTypeErr(err) { | 
					
						
							|  |  |  | 		return gtserror.Newf("ffprobe error: %w", err) | 
					
						
							|  |  |  | 	} else if result == nil { | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		log.Warn(ctx, "unsupported data type") | 
					
						
							|  |  |  | 		return nil | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-01-04 17:37:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	var ext string | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-15 14:24:53 +00:00
										 |  |  | 	// Extract any video stream metadata from media. | 
					
						
							|  |  |  | 	// This will always be used regardless of type, | 
					
						
							|  |  |  | 	// as even audio files may contain embedded album art. | 
					
						
							|  |  |  | 	width, height, framerate := result.ImageMeta() | 
					
						
							|  |  |  | 	p.media.FileMeta.Original.Width = width | 
					
						
							|  |  |  | 	p.media.FileMeta.Original.Height = height | 
					
						
							|  |  |  | 	p.media.FileMeta.Original.Size = (width * height) | 
					
						
							|  |  |  | 	p.media.FileMeta.Original.Aspect = util.Div(float32(width), float32(height)) | 
					
						
							|  |  |  | 	p.media.FileMeta.Original.Framerate = util.PtrIf(framerate) | 
					
						
							|  |  |  | 	p.media.FileMeta.Original.Duration = util.PtrIf(float32(result.duration)) | 
					
						
							|  |  |  | 	p.media.FileMeta.Original.Bitrate = util.PtrIf(result.bitrate) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Set media type from ffprobe format data. | 
					
						
							|  |  |  | 	p.media.Type, ext = result.GetFileType() | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	switch p.media.Type { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-15 14:24:53 +00:00
										 |  |  | 	case gtsmodel.FileTypeImage, | 
					
						
							|  |  |  | 		gtsmodel.FileTypeVideo: | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		// Pass file through ffmpeg clearing | 
					
						
							|  |  |  | 		// any excess metadata (e.g. EXIF). | 
					
						
							|  |  |  | 		if err := ffmpegClearMetadata(ctx, | 
					
						
							|  |  |  | 			temppath, ext, | 
					
						
							|  |  |  | 		); err != nil { | 
					
						
							|  |  |  | 			return gtserror.Newf("error cleaning metadata: %w", err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-15 14:24:53 +00:00
										 |  |  | 	case gtsmodel.FileTypeAudio: | 
					
						
							|  |  |  | 		// NOTE: we do not clean audio file | 
					
						
							|  |  |  | 		// metadata, in order to keep tags. | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-15 14:24:53 +00:00
										 |  |  | 	default: | 
					
						
							|  |  |  | 		log.Warn(ctx, "unsupported data type: %s", result.format) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-15 14:24:53 +00:00
										 |  |  | 	if width > 0 && height > 0 { | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		// Determine thumbnail dimensions to use. | 
					
						
							|  |  |  | 		thumbWidth, thumbHeight := thumbSize(width, height) | 
					
						
							|  |  |  | 		p.media.FileMeta.Small.Width = thumbWidth | 
					
						
							|  |  |  | 		p.media.FileMeta.Small.Height = thumbHeight | 
					
						
							|  |  |  | 		p.media.FileMeta.Small.Size = (thumbWidth * thumbHeight) | 
					
						
							|  |  |  | 		p.media.FileMeta.Small.Aspect = float32(thumbWidth) / float32(thumbHeight) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-15 14:24:53 +00:00
										 |  |  | 		// Generate a thumbnail image from input image path. | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		thumbpath, err = ffmpegGenerateThumb(ctx, temppath, | 
					
						
							|  |  |  | 			thumbWidth, | 
					
						
							|  |  |  | 			thumbHeight, | 
					
						
							|  |  |  | 		) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2024-07-15 14:24:53 +00:00
										 |  |  | 			return gtserror.Newf("error generating image thumb: %w", err) | 
					
						
							| 
									
										
										
										
											2022-12-22 11:48:28 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-01-08 17:17:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-15 14:24:53 +00:00
										 |  |  | 		if p.media.Blurhash == "" { | 
					
						
							|  |  |  | 			// Generate blurhash (if not already) from thumbnail. | 
					
						
							|  |  |  | 			p.media.Blurhash, err = generateBlurhash(thumbpath) | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 			if err != nil { | 
					
						
							| 
									
										
										
										
											2024-07-15 14:24:53 +00:00
										 |  |  | 				return gtserror.Newf("error generating thumb blurhash: %w", err) | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Calculate final media attachment thumbnail path. | 
					
						
							|  |  |  | 		p.media.Thumbnail.Path = uris.StoragePathForAttachment( | 
					
						
							|  |  |  | 			p.media.AccountID, | 
					
						
							|  |  |  | 			string(TypeAttachment), | 
					
						
							|  |  |  | 			string(SizeSmall), | 
					
						
							|  |  |  | 			p.media.ID, | 
					
						
							|  |  |  | 			"jpeg", | 
					
						
							|  |  |  | 		) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-01-03 17:37:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Calculate final media attachment file path. | 
					
						
							|  |  |  | 	p.media.File.Path = uris.StoragePathForAttachment( | 
					
						
							|  |  |  | 		p.media.AccountID, | 
					
						
							|  |  |  | 		string(TypeAttachment), | 
					
						
							|  |  |  | 		string(SizeOriginal), | 
					
						
							|  |  |  | 		p.media.ID, | 
					
						
							|  |  |  | 		ext, | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Copy temporary file into storage at path. | 
					
						
							|  |  |  | 	filesz, err := p.mgr.state.Storage.PutFile(ctx, | 
					
						
							|  |  |  | 		p.media.File.Path, | 
					
						
							|  |  |  | 		temppath, | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return gtserror.Newf("error writing media to storage: %w", err) | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-01-08 17:17:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Set final determined file size. | 
					
						
							|  |  |  | 	p.media.File.FileSize = int(filesz) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	if thumbpath != "" { | 
					
						
							|  |  |  | 		// Copy thumbnail file into storage at path. | 
					
						
							|  |  |  | 		thumbsz, err := p.mgr.state.Storage.PutFile(ctx, | 
					
						
							|  |  |  | 			p.media.Thumbnail.Path, | 
					
						
							|  |  |  | 			thumbpath, | 
					
						
							|  |  |  | 		) | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 			return gtserror.Newf("error writing thumb to storage: %w", err) | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-01-16 18:52:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		// Set final determined thumbnail size. | 
					
						
							|  |  |  | 		p.media.Thumbnail.FileSize = int(thumbsz) | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-11-03 15:03:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 	// Generate a media attachment URL. | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	p.media.URL = uris.URIForAttachment( | 
					
						
							|  |  |  | 		p.media.AccountID, | 
					
						
							|  |  |  | 		string(TypeAttachment), | 
					
						
							|  |  |  | 		string(SizeOriginal), | 
					
						
							|  |  |  | 		p.media.ID, | 
					
						
							|  |  |  | 		ext, | 
					
						
							|  |  |  | 	) | 
					
						
							| 
									
										
										
										
											2022-01-15 17:41:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 	// Generate a media attachment thumbnail URL. | 
					
						
							|  |  |  | 	p.media.Thumbnail.URL = uris.URIForAttachment( | 
					
						
							|  |  |  | 		p.media.AccountID, | 
					
						
							|  |  |  | 		string(TypeAttachment), | 
					
						
							|  |  |  | 		string(SizeSmall), | 
					
						
							|  |  |  | 		p.media.ID, | 
					
						
							|  |  |  | 		"jpeg", | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Get mimetype for the file container | 
					
						
							|  |  |  | 	// type, falling back to generic data. | 
					
						
							|  |  |  | 	p.media.File.ContentType = getMimeType(ext) | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 	// Set the known thumbnail content type. | 
					
						
							|  |  |  | 	p.media.Thumbnail.ContentType = "image/jpeg" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// We can now consider this cached. | 
					
						
							|  |  |  | 	p.media.Cached = util.Ptr(true) | 
					
						
							| 
									
										
										
										
											2022-01-16 18:52:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Finally set the attachment as finished processing. | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	p.media.Processing = gtsmodel.ProcessingStatusProcessed | 
					
						
							| 
									
										
										
										
											2022-02-22 13:50:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2022-01-10 18:36:09 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | // cleanup will remove any traces of processing media from storage. | 
					
						
							|  |  |  | // and perform any other necessary cleanup steps after failure. | 
					
						
							|  |  |  | func (p *ProcessingMedia) cleanup(ctx context.Context) { | 
					
						
							|  |  |  | 	var err error | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if p.media.File.Path != "" { | 
					
						
							|  |  |  | 		// Ensure media file at path is deleted from storage. | 
					
						
							|  |  |  | 		err = p.mgr.state.Storage.Delete(ctx, p.media.File.Path) | 
					
						
							|  |  |  | 		if err != nil && !storage.IsNotFound(err) { | 
					
						
							|  |  |  | 			log.Errorf(ctx, "error deleting %s: %v", p.media.File.Path, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if p.media.Thumbnail.Path != "" { | 
					
						
							|  |  |  | 		// Ensure media thumbnail at path is deleted from storage. | 
					
						
							|  |  |  | 		err = p.mgr.state.Storage.Delete(ctx, p.media.Thumbnail.Path) | 
					
						
							|  |  |  | 		if err != nil && !storage.IsNotFound(err) { | 
					
						
							|  |  |  | 			log.Errorf(ctx, "error deleting %s: %v", p.media.Thumbnail.Path, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 	// Unset all processor-calculated media fields. | 
					
						
							|  |  |  | 	p.media.FileMeta.Original = gtsmodel.Original{} | 
					
						
							|  |  |  | 	p.media.FileMeta.Small = gtsmodel.Small{} | 
					
						
							|  |  |  | 	p.media.File.ContentType = "" | 
					
						
							|  |  |  | 	p.media.File.FileSize = 0 | 
					
						
							|  |  |  | 	p.media.File.Path = "" | 
					
						
							|  |  |  | 	p.media.Thumbnail.FileSize = 0 | 
					
						
							|  |  |  | 	p.media.Thumbnail.ContentType = "" | 
					
						
							|  |  |  | 	p.media.Thumbnail.Path = "" | 
					
						
							|  |  |  | 	p.media.Thumbnail.URL = "" | 
					
						
							|  |  |  | 	p.media.URL = "" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 	// Also ensure marked as unknown and finished | 
					
						
							|  |  |  | 	// processing so gets inserted as placeholder URL. | 
					
						
							|  |  |  | 	p.media.Processing = gtsmodel.ProcessingStatusProcessed | 
					
						
							|  |  |  | 	p.media.Type = gtsmodel.FileTypeUnknown | 
					
						
							|  |  |  | 	p.media.Cached = util.Ptr(false) | 
					
						
							|  |  |  | } |