| 
									
										
										
										
											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-09-12 13:03:23 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | package dereferencing | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io" | 
					
						
							|  |  |  | 	"net/url" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-26 11:56:01 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/db" | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel" | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/id" | 
					
						
							|  |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/log" | 
					
						
							| 
									
										
										
										
											2022-09-12 13:03:23 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/media" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | func (d *deref) GetRemoteEmoji(ctx context.Context, requestingUsername string, remoteURL string, shortcode string, domain string, id string, emojiURI string, ai *media.AdditionalEmojiInfo, refresh bool) (*media.ProcessingEmoji, error) { | 
					
						
							|  |  |  | 	var ( | 
					
						
							|  |  |  | 		shortcodeDomain = shortcode + "@" + domain | 
					
						
							|  |  |  | 		processingEmoji *media.ProcessingEmoji | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 20:15:23 +00:00
										 |  |  | 	// Acquire lock for derefs map. | 
					
						
							|  |  |  | 	unlock := d.derefEmojisMu.Lock() | 
					
						
							|  |  |  | 	defer unlock() | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// first check if we're already processing this emoji | 
					
						
							| 
									
										
										
										
											2023-02-10 20:15:23 +00:00
										 |  |  | 	if alreadyProcessing, ok := d.derefEmojis[shortcodeDomain]; ok { | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 		// we're already on it, no worries | 
					
						
							|  |  |  | 		processingEmoji = alreadyProcessing | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		// not processing it yet, let's start | 
					
						
							|  |  |  | 		t, err := d.transportController.NewTransportForUsername(ctx, requestingUsername) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, fmt.Errorf("GetRemoteEmoji: error creating transport to fetch emoji %s: %s", shortcodeDomain, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		derefURI, err := url.Parse(remoteURL) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, fmt.Errorf("GetRemoteEmoji: error parsing url for emoji %s: %s", shortcodeDomain, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		dataFunc := func(innerCtx context.Context) (io.ReadCloser, int64, error) { | 
					
						
							|  |  |  | 			return t.DereferenceMedia(innerCtx, derefURI) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-13 18:40:48 +00:00
										 |  |  | 		newProcessing, err := d.mediaManager.PreProcessEmoji(ctx, dataFunc, nil, shortcode, id, emojiURI, ai, refresh) | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return nil, fmt.Errorf("GetRemoteEmoji: error processing emoji %s: %s", shortcodeDomain, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// store it in our map to indicate it's in process | 
					
						
							| 
									
										
										
										
											2023-02-10 20:15:23 +00:00
										 |  |  | 		d.derefEmojis[shortcodeDomain] = newProcessing | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 		processingEmoji = newProcessing | 
					
						
							| 
									
										
										
										
											2022-09-12 13:03:23 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 20:15:23 +00:00
										 |  |  | 	// Unlock map. | 
					
						
							|  |  |  | 	unlock() | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 20:15:23 +00:00
										 |  |  | 	defer func() { | 
					
						
							|  |  |  | 		// On exit safely remove emoji from map. | 
					
						
							|  |  |  | 		unlock := d.derefEmojisMu.Lock() | 
					
						
							|  |  |  | 		delete(d.derefEmojis, shortcodeDomain) | 
					
						
							|  |  |  | 		unlock() | 
					
						
							|  |  |  | 	}() | 
					
						
							| 
									
										
										
										
											2022-09-12 13:03:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-10 20:15:23 +00:00
										 |  |  | 	// Start emoji attachment loading (blocking call). | 
					
						
							|  |  |  | 	if _, err := processingEmoji.LoadEmoji(ctx); err != nil { | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 		return nil, err | 
					
						
							| 
									
										
										
										
											2022-09-12 13:03:23 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 	return processingEmoji, nil | 
					
						
							| 
									
										
										
										
											2022-09-12 13:03:23 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-09-26 11:56:01 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | func (d *deref) populateEmojis(ctx context.Context, rawEmojis []*gtsmodel.Emoji, requestingUsername string) ([]*gtsmodel.Emoji, error) { | 
					
						
							|  |  |  | 	// At this point we should know: | 
					
						
							|  |  |  | 	// * the AP uri of the emoji | 
					
						
							|  |  |  | 	// * the domain of the emoji | 
					
						
							|  |  |  | 	// * the shortcode of the emoji | 
					
						
							|  |  |  | 	// * the remote URL of the image | 
					
						
							|  |  |  | 	// This should be enough to dereference the emoji | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	gotEmojis := make([]*gtsmodel.Emoji, 0, len(rawEmojis)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for _, e := range rawEmojis { | 
					
						
							|  |  |  | 		var gotEmoji *gtsmodel.Emoji | 
					
						
							|  |  |  | 		var err error | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 		shortcodeDomain := e.Shortcode + "@" + e.Domain | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// check if we already know this emoji | 
					
						
							|  |  |  | 		if e.ID != "" { | 
					
						
							|  |  |  | 			// we had an ID for this emoji already, which means | 
					
						
							|  |  |  | 			// it should be fleshed out already and we won't | 
					
						
							|  |  |  | 			// have to get it from the database again | 
					
						
							|  |  |  | 			gotEmoji = e | 
					
						
							|  |  |  | 		} else if gotEmoji, err = d.db.GetEmojiByShortcodeDomain(ctx, e.Shortcode, e.Domain); err != nil && err != db.ErrNoEntries { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 			log.Errorf(ctx, "error checking database for emoji %s: %s", shortcodeDomain, err) | 
					
						
							| 
									
										
										
										
											2022-09-26 11:56:01 +02:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 		var refresh bool | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-13 15:16:24 +02:00
										 |  |  | 		if gotEmoji != nil { | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 			// we had the emoji already, but refresh it if necessary | 
					
						
							|  |  |  | 			if e.UpdatedAt.Unix() > gotEmoji.ImageUpdatedAt.Unix() { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 				log.Tracef(ctx, "emoji %s was updated since we last saw it, will refresh", shortcodeDomain) | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 				refresh = true | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if !refresh && (e.URI != gotEmoji.URI) { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 				log.Tracef(ctx, "emoji %s changed URI since we last saw it, will refresh", shortcodeDomain) | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 				refresh = true | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if !refresh && (e.ImageRemoteURL != gotEmoji.ImageRemoteURL) { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 				log.Tracef(ctx, "emoji %s changed image URL since we last saw it, will refresh", shortcodeDomain) | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 				refresh = true | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if !refresh { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 				log.Tracef(ctx, "emoji %s is up to date, will not refresh", shortcodeDomain) | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 				log.Tracef(ctx, "refreshing emoji %s", shortcodeDomain) | 
					
						
							| 
									
										
										
										
											2022-10-13 15:16:24 +02:00
										 |  |  | 				emojiID := gotEmoji.ID // use existing ID | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 				processingEmoji, err := d.GetRemoteEmoji(ctx, requestingUsername, e.ImageRemoteURL, e.Shortcode, e.Domain, emojiID, e.URI, &media.AdditionalEmojiInfo{ | 
					
						
							| 
									
										
										
										
											2022-10-13 15:16:24 +02:00
										 |  |  | 					Domain:               &e.Domain, | 
					
						
							|  |  |  | 					ImageRemoteURL:       &e.ImageRemoteURL, | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 					ImageStaticRemoteURL: &e.ImageStaticRemoteURL, | 
					
						
							| 
									
										
										
										
											2022-10-13 15:16:24 +02:00
										 |  |  | 					Disabled:             gotEmoji.Disabled, | 
					
						
							|  |  |  | 					VisibleInPicker:      gotEmoji.VisibleInPicker, | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 				}, refresh) | 
					
						
							| 
									
										
										
										
											2022-10-13 15:16:24 +02:00
										 |  |  | 				if err != nil { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 					log.Errorf(ctx, "couldn't refresh remote emoji %s: %s", shortcodeDomain, err) | 
					
						
							| 
									
										
										
										
											2022-10-13 15:16:24 +02:00
										 |  |  | 					continue | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if gotEmoji, err = processingEmoji.LoadEmoji(ctx); err != nil { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 					log.Errorf(ctx, "couldn't load refreshed remote emoji %s: %s", shortcodeDomain, err) | 
					
						
							| 
									
										
										
										
											2022-10-13 15:16:24 +02:00
										 |  |  | 					continue | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2022-09-26 11:56:01 +02:00
										 |  |  | 			// it's new! go get it! | 
					
						
							|  |  |  | 			newEmojiID, err := id.NewRandomULID() | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 				log.Errorf(ctx, "error generating id for remote emoji %s: %s", shortcodeDomain, err) | 
					
						
							| 
									
										
										
										
											2022-09-26 11:56:01 +02:00
										 |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 			processingEmoji, err := d.GetRemoteEmoji(ctx, requestingUsername, e.ImageRemoteURL, e.Shortcode, e.Domain, newEmojiID, e.URI, &media.AdditionalEmojiInfo{ | 
					
						
							| 
									
										
										
										
											2022-09-26 11:56:01 +02:00
										 |  |  | 				Domain:               &e.Domain, | 
					
						
							|  |  |  | 				ImageRemoteURL:       &e.ImageRemoteURL, | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 				ImageStaticRemoteURL: &e.ImageStaticRemoteURL, | 
					
						
							| 
									
										
										
										
											2022-09-26 11:56:01 +02:00
										 |  |  | 				Disabled:             e.Disabled, | 
					
						
							|  |  |  | 				VisibleInPicker:      e.VisibleInPicker, | 
					
						
							| 
									
										
										
										
											2022-11-11 20:27:37 +01:00
										 |  |  | 			}, refresh) | 
					
						
							| 
									
										
										
										
											2022-09-26 11:56:01 +02:00
										 |  |  | 			if err != nil { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 				log.Errorf(ctx, "couldn't get remote emoji %s: %s", shortcodeDomain, err) | 
					
						
							| 
									
										
										
										
											2022-09-26 11:56:01 +02:00
										 |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if gotEmoji, err = processingEmoji.LoadEmoji(ctx); err != nil { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 				log.Errorf(ctx, "couldn't load remote emoji %s: %s", shortcodeDomain, err) | 
					
						
							| 
									
										
										
										
											2022-09-26 11:56:01 +02:00
										 |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// if we get here, we either had the emoji already or we successfully fetched it | 
					
						
							|  |  |  | 		gotEmojis = append(gotEmojis, gotEmoji) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return gotEmojis, nil | 
					
						
							|  |  |  | } |