| 
									
										
										
										
											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-11 17:49:14 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | package media | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2024-07-28 08:31:49 +00:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-26 15:34:10 +02:00
										 |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/gtserror" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/gtsmodel" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/log" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/storage" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/uris" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/util" | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 	errorsv2 "codeberg.org/gruf/go-errors/v2" | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	"codeberg.org/gruf/go-runners" | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ProcessingEmoji represents an emoji currently processing. It exposes | 
					
						
							|  |  |  | // various functions for retrieving data from the process. | 
					
						
							|  |  |  | type ProcessingEmoji struct { | 
					
						
							| 
									
										
										
										
											2023-05-28 13:08:35 +01:00
										 |  |  | 	emoji     *gtsmodel.Emoji   // processing emoji details | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 	instAccID string            // instance account ID | 
					
						
							| 
									
										
										
										
											2023-07-24 13:14:13 +01:00
										 |  |  | 	newPathID string            // new emoji path ID to use when being refreshed | 
					
						
							| 
									
										
										
										
											2023-05-28 13:08:35 +01:00
										 |  |  | 	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-11 17:49:14 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | // ID returns the ID of the underlying emoji. | 
					
						
							|  |  |  | func (p *ProcessingEmoji) ID() string { | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	return p.emoji.ID // immutable, safe outside mutex. | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | // LoadEmoji blocks until the static and fullsize image has been processed, and then returns the completed emoji. | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | func (p *ProcessingEmoji) Load(ctx context.Context) (*gtsmodel.Emoji, error) { | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	emoji, 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 emoji: %v", err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 	return emoji, 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 *ProcessingEmoji) load(ctx context.Context) ( | 
					
						
							|  |  |  | 	emoji *gtsmodel.Emoji, | 
					
						
							|  |  |  | 	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 | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											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-07-22 18:45:48 +01:00
										 |  |  | 			if done = (err == nil || !errorsv2.IsV2(err, | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 				context.Canceled, | 
					
						
							|  |  |  | 				context.DeadlineExceeded, | 
					
						
							| 
									
										
										
										
											2024-07-22 18:45:48 +01:00
										 |  |  | 			)); done { | 
					
						
							|  |  |  | 				// Processing finished, | 
					
						
							|  |  |  | 				// whether error or not! | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// Anything from here, we | 
					
						
							|  |  |  | 				// need to ensure happens | 
					
						
							|  |  |  | 				// (i.e. no ctx canceled). | 
					
						
							|  |  |  | 				ctx = context.WithoutCancel(ctx) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// On error, clean | 
					
						
							|  |  |  | 				// downloaded files. | 
					
						
							|  |  |  | 				if err != nil { | 
					
						
							|  |  |  | 					p.cleanup(ctx) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// Update with latest details, whatever happened. | 
					
						
							|  |  |  | 				e := p.mgr.state.DB.UpdateEmoji(ctx, p.emoji) | 
					
						
							|  |  |  | 				if e != nil { | 
					
						
							|  |  |  | 					log.Errorf(ctx, "error updating emoji in db: %v", e) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// Store values. | 
					
						
							|  |  |  | 				p.done = true | 
					
						
							|  |  |  | 				p.err = err | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		}() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Attempt to store media and calculate | 
					
						
							|  |  |  | 		// full-size media attachment details. | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 		// | 
					
						
							|  |  |  | 		// This will update p.emoji 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-07-22 18:45:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Return a copy of emoji. | 
					
						
							|  |  |  | 	emoji = new(gtsmodel.Emoji) | 
					
						
							|  |  |  | 	*emoji = *p.emoji | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 	return | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-16 18:52:55 +01: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 *ProcessingEmoji) store(ctx context.Context) error { | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Load media from data func. | 
					
						
							|  |  |  | 	rc, err := p.dataFn(ctx) | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 		return gtserror.Newf("error executing data function: %w", err) | 
					
						
							| 
									
										
										
										
											2022-01-16 18:52:55 +01: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 | 
					
						
							|  |  |  | 		staticpath string | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-21 13:41:44 +01:00
										 |  |  | 	defer func() { | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		if err := remove(temppath, staticpath); err != nil { | 
					
						
							|  |  |  | 			log.Errorf(ctx, "error(s) cleaning up files: %v", err) | 
					
						
							| 
									
										
										
										
											2022-03-21 13:41:44 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Drain reader to tmp file | 
					
						
							|  |  |  | 	// (this reader handles close). | 
					
						
							|  |  |  | 	temppath, err = drainToTmp(rc) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return gtserror.Newf("error draining data to tmp: %w", err) | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Pass input file through ffprobe to | 
					
						
							|  |  |  | 	// parse further metadata information. | 
					
						
							| 
									
										
										
										
											2024-08-19 13:38:10 +00:00
										 |  |  | 	result, err := probe(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-10-28 13:09:21 +00:00
										 |  |  | 		log.Warnf(ctx, "unsupported data type by ffprobe: %v", err) | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		return nil | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	var ext string | 
					
						
							| 
									
										
										
										
											2024-11-04 13:58:15 +00:00
										 |  |  | 	var fileType gtsmodel.FileType | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-04 13:58:15 +00:00
										 |  |  | 	// Get abstract file type, mimetype and ext from ffprobe data. | 
					
						
							|  |  |  | 	fileType, p.emoji.ImageContentType, ext = result.GetFileType() | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	if fileType != gtsmodel.FileTypeImage { | 
					
						
							|  |  |  | 		return gtserror.Newf("unsupported emoji filetype: %s (%s)", fileType, ext) | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-28 08:31:49 +00:00
										 |  |  | 	// Add file extension to path. | 
					
						
							|  |  |  | 	newpath := temppath + "." + ext | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Before ffmpeg processing, rename to set file ext. | 
					
						
							|  |  |  | 	if err := os.Rename(temppath, newpath); err != nil { | 
					
						
							|  |  |  | 		return gtserror.Newf("error renaming to %s - >%s: %w", temppath, newpath, err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Update path var | 
					
						
							|  |  |  | 	// AFTER successful. | 
					
						
							|  |  |  | 	temppath = newpath | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Generate a static image from input emoji path. | 
					
						
							|  |  |  | 	staticpath, err = ffmpegGenerateStatic(ctx, temppath) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return gtserror.Newf("error generating emoji static: %w", err) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-13 15:16:24 +02:00
										 |  |  | 	var pathID string | 
					
						
							| 
									
										
										
										
											2023-07-24 13:14:13 +01:00
										 |  |  | 	if p.newPathID != "" { | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		// This is a refreshed emoji with a new | 
					
						
							|  |  |  | 		// path ID that this will be stored under. | 
					
						
							| 
									
										
										
										
											2022-10-13 15:16:24 +02:00
										 |  |  | 		pathID = p.newPathID | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		// This is a new emoji, simply use provided ID. | 
					
						
							| 
									
										
										
										
											2022-10-13 15:16:24 +02:00
										 |  |  | 		pathID = p.emoji.ID | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 	// Calculate final emoji media file path. | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	p.emoji.ImagePath = uris.StoragePathForAttachment( | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 		p.instAccID, | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 		string(TypeEmoji), | 
					
						
							|  |  |  | 		string(SizeOriginal), | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		pathID, | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		ext, | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 	// Calculate final emoji static media file path. | 
					
						
							|  |  |  | 	p.emoji.ImageStaticPath = uris.StoragePathForAttachment( | 
					
						
							|  |  |  | 		p.instAccID, | 
					
						
							|  |  |  | 		string(TypeEmoji), | 
					
						
							|  |  |  | 		string(SizeStatic), | 
					
						
							|  |  |  | 		pathID, | 
					
						
							|  |  |  | 		"png", | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-26 12:43:10 +00:00
										 |  |  | 	// Set the known emoji static content type. | 
					
						
							|  |  |  | 	p.emoji.ImageStaticContentType = "image/png" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Copy temporary file into storage at path. | 
					
						
							|  |  |  | 	filesz, err := p.mgr.state.Storage.PutFile(ctx, | 
					
						
							|  |  |  | 		p.emoji.ImagePath, | 
					
						
							|  |  |  | 		temppath, | 
					
						
							| 
									
										
										
										
											2024-09-26 12:43:10 +00:00
										 |  |  | 		p.emoji.ImageContentType, | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	) | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-06-22 20:46:36 +01:00
										 |  |  | 		return gtserror.Newf("error writing emoji to storage: %w", err) | 
					
						
							| 
									
										
										
										
											2022-09-24 11:11:47 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Copy static emoji file into storage at path. | 
					
						
							|  |  |  | 	staticsz, err := p.mgr.state.Storage.PutFile(ctx, | 
					
						
							|  |  |  | 		p.emoji.ImageStaticPath, | 
					
						
							|  |  |  | 		staticpath, | 
					
						
							| 
									
										
										
										
											2024-09-26 12:43:10 +00:00
										 |  |  | 		p.emoji.ImageStaticContentType, | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return gtserror.Newf("error writing static to storage: %w", err) | 
					
						
							| 
									
										
										
										
											2022-09-24 11:11:47 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-01-16 18:52:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// Set final determined file sizes. | 
					
						
							|  |  |  | 	p.emoji.ImageFileSize = int(filesz) | 
					
						
							|  |  |  | 	p.emoji.ImageStaticFileSize = int(staticsz) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 	// Generate an emoji media static URL. | 
					
						
							| 
									
										
										
										
											2023-11-10 19:29:26 +01:00
										 |  |  | 	p.emoji.ImageURL = uris.URIForAttachment( | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 		p.instAccID, | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 		string(TypeEmoji), | 
					
						
							|  |  |  | 		string(SizeOriginal), | 
					
						
							|  |  |  | 		pathID, | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 		ext, | 
					
						
							| 
									
										
										
										
											2023-01-11 11:13:13 +00:00
										 |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 	// Generate an emoji image static URL. | 
					
						
							|  |  |  | 	p.emoji.ImageStaticURL = uris.URIForAttachment( | 
					
						
							|  |  |  | 		p.instAccID, | 
					
						
							|  |  |  | 		string(TypeEmoji), | 
					
						
							|  |  |  | 		string(SizeStatic), | 
					
						
							|  |  |  | 		pathID, | 
					
						
							|  |  |  | 		"png", | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-12 09:39:47 +00:00
										 |  |  | 	// We can now consider this cached. | 
					
						
							|  |  |  | 	p.emoji.Cached = util.Ptr(true) | 
					
						
							| 
									
										
										
										
											2022-02-22 13:50:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-11 17:49:14 +01:00
										 |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | // cleanup will remove any traces of processing emoji from storage, | 
					
						
							|  |  |  | // and perform any other necessary cleanup steps after failure. | 
					
						
							|  |  |  | func (p *ProcessingEmoji) cleanup(ctx context.Context) { | 
					
						
							| 
									
										
										
										
											2024-07-22 18:45:48 +01:00
										 |  |  | 	log.Debugf(ctx, "running cleanup of emoji %s", p.emoji.ID) | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if p.emoji.ImagePath != "" { | 
					
						
							|  |  |  | 		// Ensure emoji file at path is deleted from storage. | 
					
						
							| 
									
										
										
										
											2024-07-22 18:45:48 +01:00
										 |  |  | 		err := p.mgr.state.Storage.Delete(ctx, p.emoji.ImagePath) | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 		if err != nil && !storage.IsNotFound(err) { | 
					
						
							|  |  |  | 			log.Errorf(ctx, "error deleting %s: %v", p.emoji.ImagePath, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if p.emoji.ImageStaticPath != "" { | 
					
						
							|  |  |  | 		// Ensure emoji static file at path is deleted from storage. | 
					
						
							| 
									
										
										
										
											2024-07-22 18:45:48 +01:00
										 |  |  | 		err := p.mgr.state.Storage.Delete(ctx, p.emoji.ImageStaticPath) | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 		if err != nil && !storage.IsNotFound(err) { | 
					
						
							|  |  |  | 			log.Errorf(ctx, "error deleting %s: %v", p.emoji.ImageStaticPath, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-17 15:26:33 +00:00
										 |  |  | 	// Unset processor-calculated fields. | 
					
						
							|  |  |  | 	p.emoji.ImageStaticContentType = "" | 
					
						
							|  |  |  | 	p.emoji.ImageStaticFileSize = 0 | 
					
						
							|  |  |  | 	p.emoji.ImageStaticPath = "" | 
					
						
							|  |  |  | 	p.emoji.ImageStaticURL = "" | 
					
						
							|  |  |  | 	p.emoji.ImageContentType = "" | 
					
						
							|  |  |  | 	p.emoji.ImageFileSize = 0 | 
					
						
							|  |  |  | 	p.emoji.ImagePath = "" | 
					
						
							|  |  |  | 	p.emoji.ImageURL = "" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-26 15:01:16 +00:00
										 |  |  | 	// Ensure marked as not cached. | 
					
						
							|  |  |  | 	p.emoji.Cached = util.Ptr(false) | 
					
						
							|  |  |  | } |