| 
									
										
										
										
											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-16 18:52:55 +01:00
										 |  |  | 	"bytes" | 
					
						
							| 
									
										
										
										
											2022-01-04 17:37:54 +01:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	"image/jpeg" | 
					
						
							| 
									
										
										
										
											2022-01-16 18:52:55 +01:00
										 |  |  | 	"io" | 
					
						
							| 
									
										
										
										
											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-04-26 13:50:46 +01:00
										 |  |  | 	terminator "codeberg.org/superseriousbusiness/exif-terminator" | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	"github.com/disintegration/imaging" | 
					
						
							|  |  |  | 	"github.com/h2non/filetype" | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	media   *gtsmodel.MediaAttachment // processing media attachment details | 
					
						
							|  |  |  | 	dataFn  DataFunc                  // load-data function, returns media stream | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	recache bool                      // recaching existing (uncached) media | 
					
						
							|  |  |  | 	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 | 
					
						
							| 
									
										
										
										
											2023-05-28 13:08:35 +01:00
										 |  |  | 	mgr     *Manager                  // mgr instance (access to db / storage) | 
					
						
							| 
									
										
										
										
											2022-01-03 17:37:38 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | // AttachmentID returns the ID of the underlying | 
					
						
							|  |  |  | // media attachment without blocking processing. | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | func (p *ProcessingMedia) AttachmentID() 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. | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | func (p *ProcessingMedia) LoadAttachment(ctx context.Context) (*gtsmodel.MediaAttachment, error) { | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	// Attempt to load synchronously. | 
					
						
							|  |  |  | 	media, done, err := p.load(ctx) | 
					
						
							|  |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		// No issue, return media. | 
					
						
							|  |  |  | 		return media, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !done { | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// Provided context was cancelled, | 
					
						
							|  |  |  | 		// e.g. request aborted early before | 
					
						
							|  |  |  | 		// its context could be used to finish | 
					
						
							|  |  |  | 		// loading the attachment. Enqueue for | 
					
						
							|  |  |  | 		// asynchronous processing, which will | 
					
						
							|  |  |  | 		// use a background context. | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 		log.Warnf(ctx, "reprocessing media %s after canceled ctx", p.media.ID) | 
					
						
							| 
									
										
										
										
											2024-04-26 13:50:46 +01:00
										 |  |  | 		p.mgr.state.Workers.Media.Queue.Push(p.Process) | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// Media could not be retrieved FULLY, | 
					
						
							|  |  |  | 	// but partial attachment should be present. | 
					
						
							|  |  |  | 	return media, err | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | // Process allows the receiving object to fit the | 
					
						
							|  |  |  | // runners.WorkerFunc signature. It performs a | 
					
						
							|  |  |  | // (blocking) load and logs on error. | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | func (p *ProcessingMedia) Process(ctx context.Context) { | 
					
						
							|  |  |  | 	if _, _, err := p.load(ctx); err != nil { | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		log.Errorf(ctx, "error(s) processing media: %v", err) | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | // load performs a concurrency-safe load of ProcessingMedia, only | 
					
						
							|  |  |  | // marking itself as complete when returned error is NOT a context cancel. | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | func (p *ProcessingMedia) load(ctx context.Context) (*gtsmodel.MediaAttachment, bool, error) { | 
					
						
							|  |  |  | 	var ( | 
					
						
							|  |  |  | 		done bool | 
					
						
							|  |  |  | 		err  error | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	err = p.proc.Process(func() error { | 
					
						
							|  |  |  | 		if p.done { | 
					
						
							|  |  |  | 			// 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. | 
					
						
							| 
									
										
										
										
											2023-11-30 16:22:34 +00:00
										 |  |  | 			done = err == nil || !errorsv2.IsV2(err, | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 				context.Canceled, | 
					
						
							|  |  |  | 				context.DeadlineExceeded, | 
					
						
							|  |  |  | 			) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if !done { | 
					
						
							|  |  |  | 				return | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 		}() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// Gather errors as we proceed. | 
					
						
							|  |  |  | 		var errs = gtserror.NewMultiError(4) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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. | 
					
						
							|  |  |  | 		storeErr := p.store(ctx) | 
					
						
							|  |  |  | 		if storeErr != nil { | 
					
						
							|  |  |  | 			errs.Append(storeErr) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Finish processing by reloading media into | 
					
						
							|  |  |  | 		// memory to get dimension and generate a thumb. | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// | 
					
						
							|  |  |  | 		// This will update p.media as it goes. | 
					
						
							|  |  |  | 		if finishErr := p.finish(ctx); finishErr != nil { | 
					
						
							|  |  |  | 			errs.Append(finishErr) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// If this isn't a file we were able to process, | 
					
						
							|  |  |  | 		// we may have partially stored it (eg., it's a | 
					
						
							|  |  |  | 		// jpeg, which is fine, but streaming it to storage | 
					
						
							|  |  |  | 		// was interrupted halfway through and so it was | 
					
						
							|  |  |  | 		// never decoded). Try to clean up in this case. | 
					
						
							|  |  |  | 		if p.media.Type == gtsmodel.FileTypeUnknown { | 
					
						
							|  |  |  | 			deleteErr := p.mgr.state.Storage.Delete(ctx, p.media.File.Path) | 
					
						
							| 
									
										
										
										
											2024-05-22 09:46:24 +00:00
										 |  |  | 			if deleteErr != nil && !storage.IsNotFound(deleteErr) { | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 				errs.Append(deleteErr) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		var dbErr error | 
					
						
							|  |  |  | 		switch { | 
					
						
							|  |  |  | 		case !p.recache: | 
					
						
							|  |  |  | 			// First time caching this attachment, insert it. | 
					
						
							|  |  |  | 			dbErr = p.mgr.state.DB.PutAttachment(ctx, p.media) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case p.recache && len(errs) == 0: | 
					
						
							|  |  |  | 			// Existing attachment we're recaching, update it. | 
					
						
							|  |  |  | 			// | 
					
						
							|  |  |  | 			// (We only want to update if everything went OK so far, | 
					
						
							|  |  |  | 			// otherwise we'd better leave previous version alone.) | 
					
						
							|  |  |  | 			dbErr = p.mgr.state.DB.UpdateAttachment(ctx, p.media) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		if dbErr != nil { | 
					
						
							|  |  |  | 			errs.Append(dbErr) | 
					
						
							| 
									
										
										
										
											2022-01-15 17:36:15 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-12-12 11:22:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		err = errs.Combine() | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	return p.media, done, err | 
					
						
							| 
									
										
										
										
											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 { | 
					
						
							|  |  |  | 	// Load media from provided data fun | 
					
						
							|  |  |  | 	rc, sz, err := p.dataFn(ctx) | 
					
						
							|  |  |  | 	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
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	defer func() { | 
					
						
							|  |  |  | 		// Ensure data reader gets closed on return. | 
					
						
							|  |  |  | 		if err := rc.Close(); err != nil { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 			log.Errorf(ctx, "error closing data reader: %v", err) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							| 
									
										
										
										
											2022-03-22 12:42:34 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// Assume we're given correct file | 
					
						
							|  |  |  | 	// size, we can overwrite this later | 
					
						
							|  |  |  | 	// once we know THE TRUTH. | 
					
						
							|  |  |  | 	fileSize := int(sz) | 
					
						
							|  |  |  | 	p.media.File.FileSize = fileSize | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Prepare to read bytes from | 
					
						
							|  |  |  | 	// file header or magic number. | 
					
						
							|  |  |  | 	hdrBuf := newHdrBuf(fileSize) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Read into buffer as much as possible. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// UnexpectedEOF means we couldn't read up to the | 
					
						
							|  |  |  | 	// given size, but we may still have read something. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// EOF means we couldn't read anything at all. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// Any other error likely means the connection messed up. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// In other words, rather counterintuitively, we | 
					
						
							|  |  |  | 	// can only proceed on no error or unexpected error! | 
					
						
							|  |  |  | 	n, err := io.ReadFull(rc, hdrBuf) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		if err != io.ErrUnexpectedEOF { | 
					
						
							|  |  |  | 			return gtserror.Newf("error reading first bytes of incoming media: %w", err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-12-17 05:38:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// Initial file size was misreported, so we didn't read | 
					
						
							|  |  |  | 		// fully into hdrBuf. Reslice it to the size we did read. | 
					
						
							|  |  |  | 		log.Warnf(ctx, | 
					
						
							|  |  |  | 			"recovered from misreported file size; reported %d; read %d", | 
					
						
							|  |  |  | 			fileSize, n, | 
					
						
							|  |  |  | 		) | 
					
						
							|  |  |  | 		hdrBuf = hdrBuf[:n] | 
					
						
							|  |  |  | 		fileSize = n | 
					
						
							|  |  |  | 		p.media.File.FileSize = fileSize | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-12-17 05:38:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	// Parse file type info from header buffer. | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// This should only ever error if the buffer | 
					
						
							|  |  |  | 	// is empty (ie., the attachment is 0 bytes). | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	info, err := filetype.Match(hdrBuf) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 		return gtserror.Newf("error parsing file type: %w", err) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Recombine header bytes with remaining stream | 
					
						
							|  |  |  | 	r := io.MultiReader(bytes.NewReader(hdrBuf), rc) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// Assume we'll put | 
					
						
							|  |  |  | 	// this file in storage. | 
					
						
							|  |  |  | 	store := true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	switch info.Extension { | 
					
						
							|  |  |  | 	case "mp4": | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// No problem. | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	case "gif": | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// No problem | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	case "jpg", "jpeg", "png", "webp": | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		if fileSize > 0 { | 
					
						
							|  |  |  | 			// A file size was provided so we can clean | 
					
						
							|  |  |  | 			// exif data from image as we're streaming it. | 
					
						
							|  |  |  | 			r, err = terminator.Terminate(r, fileSize, info.Extension) | 
					
						
							| 
									
										
										
										
											2022-12-17 05:38:56 +01:00
										 |  |  | 			if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 				return gtserror.Newf("error cleaning exif data: %w", err) | 
					
						
							| 
									
										
										
										
											2022-12-17 05:38:56 +01:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-01-16 18:52:55 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// The file is not a supported format that | 
					
						
							|  |  |  | 		// we can process, so we can't do much with it. | 
					
						
							|  |  |  | 		log.Warnf(ctx, | 
					
						
							|  |  |  | 			"media extension '%s' not officially supported, will be processed as "+ | 
					
						
							|  |  |  | 				"type '%s' with minimal metadata, and will not be cached locally", | 
					
						
							|  |  |  | 			info.Extension, gtsmodel.FileTypeUnknown, | 
					
						
							|  |  |  | 		) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Don't bother storing this. | 
					
						
							|  |  |  | 		store = false | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-01-04 17:37:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// Fill in correct attachment | 
					
						
							|  |  |  | 	// data now we're parsed it. | 
					
						
							|  |  |  | 	p.media.URL = uris.URIForAttachment( | 
					
						
							|  |  |  | 		p.media.AccountID, | 
					
						
							|  |  |  | 		string(TypeAttachment), | 
					
						
							|  |  |  | 		string(SizeOriginal), | 
					
						
							|  |  |  | 		p.media.ID, | 
					
						
							|  |  |  | 		info.Extension, | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Prefer discovered mime type, fall back to | 
					
						
							|  |  |  | 	// generic "this contains some bytes" type. | 
					
						
							|  |  |  | 	mime := info.MIME.Value | 
					
						
							|  |  |  | 	if mime == "" { | 
					
						
							|  |  |  | 		mime = "application/octet-stream" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	p.media.File.ContentType = mime | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	// Calculate attachment file path. | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	p.media.File.Path = uris.StoragePathForAttachment( | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		p.media.AccountID, | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		string(TypeAttachment), | 
					
						
							|  |  |  | 		string(SizeOriginal), | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		p.media.ID, | 
					
						
							|  |  |  | 		info.Extension, | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// We should only try to store the file if it's | 
					
						
							|  |  |  | 	// a format we can keep processing, otherwise be | 
					
						
							|  |  |  | 	// a bit cheeky: don't store it and let users | 
					
						
							|  |  |  | 	// click through to the remote server instead. | 
					
						
							|  |  |  | 	if !store { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// File shouldn't already exist in storage at this point, | 
					
						
							|  |  |  | 	// but we do a check as it's worth logging / cleaning up. | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	if have, _ := p.mgr.state.Storage.Has(ctx, p.media.File.Path); have { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 		log.Warnf(ctx, "media already exists at storage path: %s", p.media.File.Path) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Attempt to remove existing media at storage path (might be broken / out-of-date) | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 		if err := p.mgr.state.Storage.Delete(ctx, p.media.File.Path); err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 			return gtserror.Newf("error removing media from storage: %v", err) | 
					
						
							| 
									
										
										
										
											2022-01-09 18:41:22 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-01-04 17:37:54 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// Write the final reader stream to our storage. | 
					
						
							|  |  |  | 	wroteSize, err := p.mgr.state.Storage.PutStream(ctx, p.media.File.Path, r) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 		return gtserror.Newf("error writing media to storage: %w", err) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-01-04 17:37:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// Set actual written size | 
					
						
							|  |  |  | 	// as authoritative file size. | 
					
						
							|  |  |  | 	p.media.File.FileSize = int(wroteSize) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// We can now consider this cached. | 
					
						
							|  |  |  | 	p.media.Cached = util.Ptr(true) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | func (p *ProcessingMedia) finish(ctx context.Context) error { | 
					
						
							|  |  |  | 	// Make a jolly assumption about thumbnail type. | 
					
						
							|  |  |  | 	p.media.Thumbnail.ContentType = mimeImageJpeg | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Calculate attachment thumbnail file path | 
					
						
							|  |  |  | 	p.media.Thumbnail.Path = uris.StoragePathForAttachment( | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		p.media.AccountID, | 
					
						
							|  |  |  | 		string(TypeAttachment), | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		string(SizeSmall), | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		p.media.ID, | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// Always encode attachment | 
					
						
							|  |  |  | 		// thumbnails as jpg. | 
					
						
							|  |  |  | 		"jpg", | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	) | 
					
						
							| 
									
										
										
										
											2022-01-16 18:52:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// Calculate attachment thumbnail serve path. | 
					
						
							|  |  |  | 	p.media.Thumbnail.URL = uris.URIForAttachment( | 
					
						
							|  |  |  | 		p.media.AccountID, | 
					
						
							|  |  |  | 		string(TypeAttachment), | 
					
						
							|  |  |  | 		string(SizeSmall), | 
					
						
							|  |  |  | 		p.media.ID, | 
					
						
							|  |  |  | 		// Always encode attachment | 
					
						
							|  |  |  | 		// thumbnails as jpg. | 
					
						
							|  |  |  | 		"jpg", | 
					
						
							|  |  |  | 	) | 
					
						
							| 
									
										
										
										
											2022-01-08 13:45:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// If original file hasn't been stored, there's | 
					
						
							|  |  |  | 	// likely something wrong with the data, or we | 
					
						
							|  |  |  | 	// don't want to store it. Skip everything else. | 
					
						
							|  |  |  | 	if !*p.media.Cached { | 
					
						
							|  |  |  | 		p.media.Processing = gtsmodel.ProcessingStatusProcessed | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Get a stream to the original file for further processing. | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	rc, err := p.mgr.state.Storage.GetStream(ctx, p.media.File.Path) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 		return gtserror.Newf("error loading file from storage: %w", err) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	defer rc.Close() | 
					
						
							| 
									
										
										
										
											2022-09-19 13:43:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// fullImg is the processed version of | 
					
						
							|  |  |  | 	// the original (stripped + reoriented). | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	var fullImg *gtsImage | 
					
						
							| 
									
										
										
										
											2022-01-04 17:37:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// Depending on the content type, we | 
					
						
							|  |  |  | 	// can do various types of decoding. | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	switch p.media.File.ContentType { | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	// .jpeg, .gif, .webp image type | 
					
						
							|  |  |  | 	case mimeImageJpeg, mimeImageGif, mimeImageWebp: | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		fullImg, err = decodeImage( | 
					
						
							|  |  |  | 			rc, | 
					
						
							|  |  |  | 			imaging.AutoOrientation(true), | 
					
						
							|  |  |  | 		) | 
					
						
							| 
									
										
										
										
											2022-01-08 13:45:42 +01:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 			return gtserror.Newf("error decoding image: %w", err) | 
					
						
							| 
									
										
										
										
											2022-01-08 13:45:42 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// Mark as no longer unknown type now | 
					
						
							|  |  |  | 		// we know for sure we can decode it. | 
					
						
							|  |  |  | 		p.media.Type = gtsmodel.FileTypeImage | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	// .png image (requires ancillary chunk stripping) | 
					
						
							|  |  |  | 	case mimeImagePng: | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		fullImg, err = decodeImage( | 
					
						
							|  |  |  | 			&pngAncillaryChunkStripper{Reader: rc}, | 
					
						
							|  |  |  | 			imaging.AutoOrientation(true), | 
					
						
							|  |  |  | 		) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 			return gtserror.Newf("error decoding image: %w", err) | 
					
						
							| 
									
										
										
										
											2022-01-08 17:17:01 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-12-22 11:48:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// Mark as no longer unknown type now | 
					
						
							|  |  |  | 		// we know for sure we can decode it. | 
					
						
							|  |  |  | 		p.media.Type = gtsmodel.FileTypeImage | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	// .mp4 video type | 
					
						
							|  |  |  | 	case mimeVideoMp4: | 
					
						
							|  |  |  | 		video, err := decodeVideoFrame(rc) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 			return gtserror.Newf("error decoding video: %w", err) | 
					
						
							| 
									
										
										
										
											2022-12-22 11:48:28 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-01-08 17:17:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		// Set video frame as image. | 
					
						
							|  |  |  | 		fullImg = video.frame | 
					
						
							| 
									
										
										
										
											2022-01-04 17:37:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		// Set video metadata in attachment info. | 
					
						
							|  |  |  | 		p.media.FileMeta.Original.Duration = &video.duration | 
					
						
							|  |  |  | 		p.media.FileMeta.Original.Framerate = &video.framerate | 
					
						
							|  |  |  | 		p.media.FileMeta.Original.Bitrate = &video.bitrate | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Mark as no longer unknown type now | 
					
						
							|  |  |  | 		// we know for sure we can decode it. | 
					
						
							|  |  |  | 		p.media.Type = gtsmodel.FileTypeVideo | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-01-03 17:37:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// fullImg should be in-memory by | 
					
						
							|  |  |  | 	// now so we're done with storage. | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	if err := rc.Close(); err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 		return gtserror.Newf("error closing file: %w", err) | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-01-08 17:17:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	// Set full-size dimensions in attachment info. | 
					
						
							|  |  |  | 	p.media.FileMeta.Original.Width = int(fullImg.Width()) | 
					
						
							|  |  |  | 	p.media.FileMeta.Original.Height = int(fullImg.Height()) | 
					
						
							|  |  |  | 	p.media.FileMeta.Original.Size = int(fullImg.Size()) | 
					
						
							|  |  |  | 	p.media.FileMeta.Original.Aspect = fullImg.AspectRatio() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Get smaller thumbnail image | 
					
						
							|  |  |  | 	thumbImg := fullImg.Thumbnail() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Garbage collector, you may | 
					
						
							|  |  |  | 	// now take our large son. | 
					
						
							|  |  |  | 	fullImg = nil | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// Only generate blurhash | 
					
						
							|  |  |  | 	// from thumb if necessary. | 
					
						
							|  |  |  | 	if p.media.Blurhash == "" { | 
					
						
							|  |  |  | 		hash, err := thumbImg.Blurhash() | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return gtserror.Newf("error generating blurhash: %w", err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-01-16 18:52:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// Set the attachment blurhash. | 
					
						
							|  |  |  | 		p.media.Blurhash = hash | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-11-03 15:03:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	// Thumbnail shouldn't already exist in storage at this point, | 
					
						
							|  |  |  | 	// but we do a check as it's worth logging / cleaning up. | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	if have, _ := p.mgr.state.Storage.Has(ctx, p.media.Thumbnail.Path); have { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 		log.Warnf(ctx, "thumbnail already exists at storage path: %s", p.media.Thumbnail.Path) | 
					
						
							| 
									
										
										
										
											2022-03-21 13:41:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		// Attempt to remove existing thumbnail at storage path (might be broken / out-of-date) | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 		if err := p.mgr.state.Storage.Delete(ctx, p.media.Thumbnail.Path); err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 			return gtserror.Newf("error removing thumbnail from storage: %v", err) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2022-01-08 13:45:42 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	// Create a thumbnail JPEG encoder stream. | 
					
						
							|  |  |  | 	enc := thumbImg.ToJPEG(&jpeg.Options{ | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		// Good enough for | 
					
						
							|  |  |  | 		// a thumbnail. | 
					
						
							|  |  |  | 		Quality: 70, | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2022-01-08 13:45:42 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	// Stream-encode the JPEG thumbnail image into storage. | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	sz, err := p.mgr.state.Storage.PutStream(ctx, p.media.Thumbnail.Path, enc) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 		return gtserror.Newf("error stream-encoding thumbnail to storage: %w", err) | 
					
						
							| 
									
										
										
										
											2022-01-15 17:41:18 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	// Set thumbnail dimensions in attachment info. | 
					
						
							|  |  |  | 	p.media.FileMeta.Small = gtsmodel.Small{ | 
					
						
							|  |  |  | 		Width:  int(thumbImg.Width()), | 
					
						
							|  |  |  | 		Height: int(thumbImg.Height()), | 
					
						
							|  |  |  | 		Size:   int(thumbImg.Size()), | 
					
						
							|  |  |  | 		Aspect: thumbImg.AspectRatio(), | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-01-16 18:52:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	// Set written image size. | 
					
						
							|  |  |  | 	p.media.Thumbnail.FileSize = int(sz) | 
					
						
							| 
									
										
										
										
											2022-08-15 12:35:05 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	// Finally set the attachment as processed and update time. | 
					
						
							|  |  |  | 	p.media.Processing = gtsmodel.ProcessingStatusProcessed | 
					
						
							|  |  |  | 	p.media.File.UpdatedAt = time.Now() | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } |