summaryrefslogtreecommitdiff
path: root/internal/processing
diff options
context:
space:
mode:
Diffstat (limited to 'internal/processing')
-rw-r--r--internal/processing/admin/admin_test.go2
-rw-r--r--internal/processing/admin/emoji.go538
-rw-r--r--internal/processing/admin/emoji_test.go67
-rw-r--r--internal/processing/fedi/emoji.go2
4 files changed, 420 insertions, 189 deletions
diff --git a/internal/processing/admin/admin_test.go b/internal/processing/admin/admin_test.go
index 01a3a88ff..1bd355b3d 100644
--- a/internal/processing/admin/admin_test.go
+++ b/internal/processing/admin/admin_test.go
@@ -62,6 +62,7 @@ type AdminStandardTestSuite struct {
testFollows map[string]*gtsmodel.Follow
testAttachments map[string]*gtsmodel.MediaAttachment
testStatuses map[string]*gtsmodel.Status
+ testEmojis map[string]*gtsmodel.Emoji
// module being tested
adminProcessor *admin.Processor
@@ -76,6 +77,7 @@ func (suite *AdminStandardTestSuite) SetupSuite() {
suite.testFollows = testrig.NewTestFollows()
suite.testAttachments = testrig.NewTestAttachments()
suite.testStatuses = testrig.NewTestStatuses()
+ suite.testEmojis = testrig.NewTestEmojis()
}
func (suite *AdminStandardTestSuite) SetupTest() {
diff --git a/internal/processing/admin/emoji.go b/internal/processing/admin/emoji.go
index 689aad9dc..dcdf77642 100644
--- a/internal/processing/admin/emoji.go
+++ b/internal/processing/admin/emoji.go
@@ -36,37 +36,39 @@ import (
)
// EmojiCreate creates a custom emoji on this instance.
-func (p *Processor) EmojiCreate(ctx context.Context, account *gtsmodel.Account, user *gtsmodel.User, form *apimodel.EmojiCreateRequest) (*apimodel.Emoji, gtserror.WithCode) {
- if !*user.Admin {
- return nil, gtserror.NewErrorUnauthorized(fmt.Errorf("user %s not an admin", user.ID), "user is not an admin")
- }
-
+func (p *Processor) EmojiCreate(
+ ctx context.Context,
+ account *gtsmodel.Account,
+ form *apimodel.EmojiCreateRequest,
+) (*apimodel.Emoji, gtserror.WithCode) {
+ // Ensure emoji with this shortcode
+ // doesn't already exist on the instance.
maybeExisting, err := p.state.DB.GetEmojiByShortcodeDomain(ctx, form.Shortcode, "")
- if maybeExisting != nil {
- return nil, gtserror.NewErrorConflict(fmt.Errorf("emoji with shortcode %s already exists", form.Shortcode), fmt.Sprintf("emoji with shortcode %s already exists", form.Shortcode))
- }
-
- if err != nil && err != db.ErrNoEntries {
- return nil, gtserror.NewErrorInternalError(fmt.Errorf("error checking existence of emoji with shortcode %s: %s", form.Shortcode, err))
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ err := gtserror.Newf("error checking existence of emoji with shortcode %s: %w", form.Shortcode, err)
+ return nil, gtserror.NewErrorInternalError(err)
}
- emojiID, err := id.NewRandomULID()
- if err != nil {
- return nil, gtserror.NewErrorInternalError(fmt.Errorf("error creating id for new emoji: %s", err), "error creating emoji ID")
+ if maybeExisting != nil {
+ err := fmt.Errorf("emoji with shortcode %s already exists", form.Shortcode)
+ return nil, gtserror.NewErrorConflict(err, err.Error())
}
- emojiURI := uris.URIForEmoji(emojiID)
-
+ // Prepare data function for emoji processing
+ // (just read data from the submitted form).
data := func(innerCtx context.Context) (io.ReadCloser, int64, error) {
f, err := form.Image.Open()
return f, form.Image.Size, err
}
+ // If category was supplied on the form,
+ // ensure the category exists and provide
+ // it as additional info to emoji processing.
var ai *media.AdditionalEmojiInfo
if form.CategoryName != "" {
category, err := p.getOrCreateEmojiCategory(ctx, form.CategoryName)
if err != nil {
- return nil, gtserror.NewErrorInternalError(fmt.Errorf("error putting id in category: %s", err), "error putting id in category")
+ return nil, gtserror.NewErrorInternalError(err)
}
ai = &media.AdditionalEmojiInfo{
@@ -74,29 +76,92 @@ func (p *Processor) EmojiCreate(ctx context.Context, account *gtsmodel.Account,
}
}
- processingEmoji, err := p.mediaManager.PreProcessEmoji(ctx, data, form.Shortcode, emojiID, emojiURI, ai, false)
+ // Generate new emoji ID and URI.
+ emojiID, err := id.NewRandomULID()
if err != nil {
- return nil, gtserror.NewErrorInternalError(fmt.Errorf("error processing emoji: %s", err), "error processing emoji")
+ err := gtserror.Newf("error creating id for new emoji: %w", err)
+ return nil, gtserror.NewErrorInternalError(err)
}
+ emojiURI := uris.URIForEmoji(emojiID)
+
+ // Begin media processing.
+ processingEmoji, err := p.mediaManager.PreProcessEmoji(ctx,
+ data, form.Shortcode, emojiID, emojiURI, ai, false,
+ )
+ if err != nil {
+ err := gtserror.Newf("error processing emoji: %w", err)
+ return nil, gtserror.NewErrorInternalError(err)
+ }
+
+ // Complete processing immediately.
emoji, err := processingEmoji.LoadEmoji(ctx)
if err != nil {
- return nil, gtserror.NewErrorInternalError(fmt.Errorf("error loading emoji: %s", err), "error loading emoji")
+ err := gtserror.Newf("error loading emoji: %w", err)
+ return nil, gtserror.NewErrorInternalError(err)
}
apiEmoji, err := p.converter.EmojiToAPIEmoji(ctx, emoji)
if err != nil {
- return nil, gtserror.NewErrorInternalError(fmt.Errorf("error converting emoji: %s", err), "error converting emoji to api representation")
+ err := gtserror.Newf("error converting emoji: %w", err)
+ return nil, gtserror.NewErrorInternalError(err)
}
return &apiEmoji, nil
}
-// EmojisGet returns an admin view of custom emojis, filtered with the given parameters.
+// emojisGetFilterParams builds extra
+// query parameters to return as part
+// of an Emojis pageable response.
+//
+// The returned string will look like:
+//
+// "filter=domain:all,enabled,shortcode:example"
+func emojisGetFilterParams(
+ shortcode string,
+ domain string,
+ includeDisabled bool,
+ includeEnabled bool,
+) string {
+ var filterBuilder strings.Builder
+ filterBuilder.WriteString("filter=")
+
+ switch domain {
+ case "", "local":
+ // Local emojis only.
+ filterBuilder.WriteString("domain:local")
+
+ case db.EmojiAllDomains:
+ // Local or remote.
+ filterBuilder.WriteString("domain:all")
+
+ default:
+ // Specific domain only.
+ filterBuilder.WriteString("domain:" + domain)
+ }
+
+ if includeDisabled != includeEnabled {
+ if includeDisabled {
+ filterBuilder.WriteString(",disabled")
+ }
+ if includeEnabled {
+ filterBuilder.WriteString(",enabled")
+ }
+ }
+
+ if shortcode != "" {
+ // Specific shortcode only.
+ filterBuilder.WriteString(",shortcode:" + shortcode)
+ }
+
+ return filterBuilder.String()
+}
+
+// EmojisGet returns an admin view of custom
+// emojis, filtered with the given parameters.
func (p *Processor) EmojisGet(
ctx context.Context,
account *gtsmodel.Account,
- user *gtsmodel.User,
domain string,
includeDisabled bool,
includeEnabled bool,
@@ -105,13 +170,17 @@ func (p *Processor) EmojisGet(
minShortcodeDomain string,
limit int,
) (*apimodel.PageableResponse, gtserror.WithCode) {
- if !*user.Admin {
- return nil, gtserror.NewErrorUnauthorized(fmt.Errorf("user %s not an admin", user.ID), "user is not an admin")
- }
-
- emojis, err := p.state.DB.GetEmojisBy(ctx, domain, includeDisabled, includeEnabled, shortcode, maxShortcodeDomain, minShortcodeDomain, limit)
+ emojis, err := p.state.DB.GetEmojisBy(ctx,
+ domain,
+ includeDisabled,
+ includeEnabled,
+ shortcode,
+ maxShortcodeDomain,
+ minShortcodeDomain,
+ limit,
+ )
if err != nil && !errors.Is(err, db.ErrNoEntries) {
- err := fmt.Errorf("EmojisGet: db error: %s", err)
+ err := gtserror.Newf("db error: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
@@ -124,137 +193,139 @@ func (p *Processor) EmojisGet(
for _, emoji := range emojis {
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
if err != nil {
- err := fmt.Errorf("EmojisGet: error converting emoji to admin model emoji: %s", err)
+ err := gtserror.Newf("error converting emoji to admin model emoji: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
items = append(items, adminEmoji)
}
- filterBuilder := strings.Builder{}
- filterBuilder.WriteString("filter=")
-
- switch domain {
- case "", "local":
- filterBuilder.WriteString("domain:local")
- case db.EmojiAllDomains:
- filterBuilder.WriteString("domain:all")
- default:
- filterBuilder.WriteString("domain:")
- filterBuilder.WriteString(domain)
- }
-
- if includeDisabled != includeEnabled {
- if includeDisabled {
- filterBuilder.WriteString(",disabled")
- }
- if includeEnabled {
- filterBuilder.WriteString(",enabled")
- }
- }
-
- if shortcode != "" {
- filterBuilder.WriteString(",shortcode:")
- filterBuilder.WriteString(shortcode)
- }
-
return util.PackagePageableResponse(util.PageableResponseParams{
- Items: items,
- Path: "api/v1/admin/custom_emojis",
- NextMaxIDKey: "max_shortcode_domain",
- NextMaxIDValue: util.ShortcodeDomain(emojis[count-1]),
- PrevMinIDKey: "min_shortcode_domain",
- PrevMinIDValue: util.ShortcodeDomain(emojis[0]),
- Limit: limit,
- ExtraQueryParams: []string{filterBuilder.String()},
+ Items: items,
+ Path: "api/v1/admin/custom_emojis",
+ NextMaxIDKey: "max_shortcode_domain",
+ NextMaxIDValue: util.ShortcodeDomain(emojis[count-1]),
+ PrevMinIDKey: "min_shortcode_domain",
+ PrevMinIDValue: util.ShortcodeDomain(emojis[0]),
+ Limit: limit,
+ ExtraQueryParams: []string{
+ emojisGetFilterParams(
+ shortcode,
+ domain,
+ includeDisabled,
+ includeEnabled,
+ ),
+ },
})
}
-// EmojiGet returns the admin view of one custom emoji with the given id.
-func (p *Processor) EmojiGet(ctx context.Context, account *gtsmodel.Account, user *gtsmodel.User, id string) (*apimodel.AdminEmoji, gtserror.WithCode) {
- if !*user.Admin {
- return nil, gtserror.NewErrorUnauthorized(fmt.Errorf("user %s not an admin", user.ID), "user is not an admin")
- }
-
+// EmojiGet returns the admin view of
+// one custom emoji with the given id.
+func (p *Processor) EmojiGet(
+ ctx context.Context,
+ account *gtsmodel.Account,
+ id string,
+) (*apimodel.AdminEmoji, gtserror.WithCode) {
emoji, err := p.state.DB.GetEmojiByID(ctx, id)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- err = fmt.Errorf("EmojiGet: no emoji with id %s found in the db", id)
- return nil, gtserror.NewErrorNotFound(err)
- }
- err := fmt.Errorf("EmojiGet: db error: %s", err)
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ err := gtserror.Newf("db error: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
+ if emoji == nil {
+ err := gtserror.Newf("no emoji with id %s found in the db", id)
+ return nil, gtserror.NewErrorNotFound(err)
+ }
+
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
if err != nil {
- err = fmt.Errorf("EmojiGet: error converting emoji to admin api emoji: %s", err)
+ err := gtserror.Newf("error converting emoji to admin api emoji: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
return adminEmoji, nil
}
-// EmojiDelete deletes one emoji from the database, with the given id.
-func (p *Processor) EmojiDelete(ctx context.Context, id string) (*apimodel.AdminEmoji, gtserror.WithCode) {
+// EmojiDelete deletes one *local* emoji
+// from the database, with the given id.
+func (p *Processor) EmojiDelete(
+ ctx context.Context,
+ id string,
+) (*apimodel.AdminEmoji, gtserror.WithCode) {
emoji, err := p.state.DB.GetEmojiByID(ctx, id)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- err = fmt.Errorf("EmojiDelete: no emoji with id %s found in the db", id)
- return nil, gtserror.NewErrorNotFound(err)
- }
- err := fmt.Errorf("EmojiDelete: db error: %s", err)
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ err := gtserror.Newf("db error: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
- if emoji.Domain != "" {
- err = fmt.Errorf("EmojiDelete: emoji with id %s was not a local emoji, will not delete", id)
+ if emoji == nil {
+ err := gtserror.Newf("no emoji with id %s found in the db", id)
+ return nil, gtserror.NewErrorNotFound(err)
+ }
+
+ if !emoji.IsLocal() {
+ err := fmt.Errorf("emoji with id %s was not a local emoji, will not delete", id)
return nil, gtserror.NewErrorBadRequest(err, err.Error())
}
+ // Convert to admin emoji before deletion,
+ // so we can return the deleted emoji.
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
if err != nil {
- err = fmt.Errorf("EmojiDelete: error converting emoji to admin api emoji: %s", err)
+ err := gtserror.Newf("error converting emoji to admin api emoji: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
if err := p.state.DB.DeleteEmojiByID(ctx, id); err != nil {
- err := fmt.Errorf("EmojiDelete: db error: %s", err)
+ err := gtserror.Newf("db error deleting emoji %s: %w", id, err)
return nil, gtserror.NewErrorInternalError(err)
}
return adminEmoji, nil
}
-// EmojiUpdate updates one emoji with the given id, using the provided form parameters.
-func (p *Processor) EmojiUpdate(ctx context.Context, id string, form *apimodel.EmojiUpdateRequest) (*apimodel.AdminEmoji, gtserror.WithCode) {
+// EmojiUpdate updates one emoji with the
+// given id, using the provided form parameters.
+func (p *Processor) EmojiUpdate(
+ ctx context.Context,
+ id string,
+ form *apimodel.EmojiUpdateRequest,
+) (*apimodel.AdminEmoji, gtserror.WithCode) {
emoji, err := p.state.DB.GetEmojiByID(ctx, id)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- err = fmt.Errorf("EmojiUpdate: no emoji with id %s found in the db", id)
- return nil, gtserror.NewErrorNotFound(err)
- }
- err := fmt.Errorf("EmojiUpdate: db error: %s", err)
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ err := gtserror.Newf("db error: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
- switch form.Type {
+ if emoji == nil {
+ err := gtserror.Newf("no emoji with id %s found in the db", id)
+ return nil, gtserror.NewErrorNotFound(err)
+ }
+
+ switch t := form.Type; t {
+
case apimodel.EmojiUpdateCopy:
return p.emojiUpdateCopy(ctx, emoji, form.Shortcode, form.CategoryName)
+
case apimodel.EmojiUpdateDisable:
return p.emojiUpdateDisable(ctx, emoji)
+
case apimodel.EmojiUpdateModify:
return p.emojiUpdateModify(ctx, emoji, form.Image, form.CategoryName)
+
default:
- err := errors.New("unrecognized emoji action type")
+ err := fmt.Errorf("unrecognized emoji action type %s", t)
return nil, gtserror.NewErrorBadRequest(err, err.Error())
}
}
-// EmojiCategoriesGet returns all custom emoji categories that exist on this instance.
-func (p *Processor) EmojiCategoriesGet(ctx context.Context) ([]*apimodel.EmojiCategory, gtserror.WithCode) {
+// EmojiCategoriesGet returns all custom emoji
+// categories that exist on this instance.
+func (p *Processor) EmojiCategoriesGet(
+ ctx context.Context,
+) ([]*apimodel.EmojiCategory, gtserror.WithCode) {
categories, err := p.state.DB.GetEmojiCategories(ctx)
if err != nil {
- err := fmt.Errorf("EmojiCategoriesGet: db error: %s", err)
+ err := gtserror.Newf("db error getting emoji categories: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
@@ -262,7 +333,7 @@ func (p *Processor) EmojiCategoriesGet(ctx context.Context) ([]*apimodel.EmojiCa
for _, category := range categories {
apiCategory, err := p.converter.EmojiCategoryToAPIEmojiCategory(ctx, category)
if err != nil {
- err := fmt.Errorf("EmojiCategoriesGet: error converting emoji category to api emoji category: %s", err)
+ err := gtserror.Newf("error converting emoji category to api emoji category: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
apiCategories = append(apiCategories, apiCategory)
@@ -275,22 +346,35 @@ func (p *Processor) EmojiCategoriesGet(ctx context.Context) ([]*apimodel.EmojiCa
UTIL FUNCTIONS
*/
-func (p *Processor) getOrCreateEmojiCategory(ctx context.Context, name string) (*gtsmodel.EmojiCategory, error) {
+// getOrCreateEmojiCategory either gets an existing
+// category with the given name from the database,
+// or, if the category doesn't yet exist, it creates
+// the category and then returns it.
+func (p *Processor) getOrCreateEmojiCategory(
+ ctx context.Context,
+ name string,
+) (*gtsmodel.EmojiCategory, error) {
category, err := p.state.DB.GetEmojiCategoryByName(ctx, name)
- if err == nil {
- return category, nil
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ return nil, gtserror.Newf(
+ "database error trying get emoji category %s: %w",
+ name, err,
+ )
}
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- err = fmt.Errorf("GetOrCreateEmojiCategory: database error trying get emoji category by name: %s", err)
- return nil, err
+ if category != nil {
+ // We had it already.
+ return category, nil
}
- // we don't have the category yet, just create it with the given name
+ // We don't have the category yet,
+ // create it with the given name.
categoryID, err := id.NewRandomULID()
if err != nil {
- err = fmt.Errorf("GetOrCreateEmojiCategory: error generating id for new emoji category: %s", err)
- return nil, err
+ return nil, gtserror.Newf(
+ "error generating id for new emoji category %s: %w",
+ name, err,
+ )
}
category = &gtsmodel.EmojiCategory{
@@ -299,54 +383,85 @@ func (p *Processor) getOrCreateEmojiCategory(ctx context.Context, name string) (
}
if err := p.state.DB.PutEmojiCategory(ctx, category); err != nil {
- err = fmt.Errorf("GetOrCreateEmojiCategory: error putting new emoji category in the database: %s", err)
- return nil, err
+ return nil, gtserror.Newf(
+ "db error putting new emoji category %s: %w",
+ name, err,
+ )
}
return category, nil
}
-// copy an emoji from remote to local
-func (p *Processor) emojiUpdateCopy(ctx context.Context, emoji *gtsmodel.Emoji, shortcode *string, categoryName *string) (*apimodel.AdminEmoji, gtserror.WithCode) {
- if emoji.Domain == "" {
- err := fmt.Errorf("emojiUpdateCopy: emoji %s is not a remote emoji, cannot copy it to local", emoji.ID)
+// emojiUpdateCopy copies and stores the given
+// *remote* emoji as a *local* emoji, preserving
+// the same image, and using the provided shortcode.
+//
+// The provided emoji model must correspond to an
+// emoji already stored in the database + storage.
+func (p *Processor) emojiUpdateCopy(
+ ctx context.Context,
+ targetEmoji *gtsmodel.Emoji,
+ shortcode *string,
+ category *string,
+) (*apimodel.AdminEmoji, gtserror.WithCode) {
+ if targetEmoji.IsLocal() {
+ err := fmt.Errorf("emoji %s is not a remote emoji, cannot copy it to local", targetEmoji.ID)
return nil, gtserror.NewErrorBadRequest(err, err.Error())
}
if shortcode == nil {
- err := fmt.Errorf("emojiUpdateCopy: emoji %s could not be copied, no shortcode provided", emoji.ID)
+ err := errors.New("no shortcode provided")
return nil, gtserror.NewErrorBadRequest(err, err.Error())
}
- maybeExisting, err := p.state.DB.GetEmojiByShortcodeDomain(ctx, *shortcode, "")
+ sc := *shortcode
+ if sc == "" {
+ err := errors.New("empty shortcode provided")
+ return nil, gtserror.NewErrorBadRequest(err, err.Error())
+ }
+
+ // Ensure we don't already have an emoji
+ // stored locally with this shortcode.
+ maybeExisting, err := p.state.DB.GetEmojiByShortcodeDomain(ctx, sc, "")
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ err := gtserror.Newf("db error checking for emoji with shortcode %s: %w", sc, err)
+ return nil, gtserror.NewErrorInternalError(err)
+ }
+
if maybeExisting != nil {
- err := fmt.Errorf("emojiUpdateCopy: emoji %s could not be copied, emoji with shortcode %s already exists on this instance", emoji.ID, *shortcode)
+ err := fmt.Errorf("emoji with shortcode %s already exists on this instance", sc)
return nil, gtserror.NewErrorConflict(err, err.Error())
}
- if err != nil && err != db.ErrNoEntries {
- err := fmt.Errorf("emojiUpdateCopy: emoji %s could not be copied, error checking existence of emoji with shortcode %s: %s", emoji.ID, *shortcode, err)
- return nil, gtserror.NewErrorInternalError(err)
+ // We don't have an emoji with this
+ // shortcode yet! Prepare to create it.
+
+ // Data function for copying just streams media
+ // out of storage into an additional location.
+ //
+ // This means that data for the copy persists even
+ // if the remote copied emoji gets deleted at some point.
+ data := func(ctx context.Context) (io.ReadCloser, int64, error) {
+ rc, err := p.state.Storage.GetStream(ctx, targetEmoji.ImagePath)
+ return rc, int64(targetEmoji.ImageFileSize), err
}
- newEmojiID, err := id.NewRandomULID()
+ // Generate new emoji ID and URI.
+ emojiID, err := id.NewRandomULID()
if err != nil {
- err := fmt.Errorf("emojiUpdateCopy: emoji %s could not be copied, error creating id for new emoji: %s", emoji.ID, err)
+ err := gtserror.Newf("error creating id for new emoji: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
- newEmojiURI := uris.URIForEmoji(newEmojiID)
-
- data := func(ctx context.Context) (reader io.ReadCloser, fileSize int64, err error) {
- rc, err := p.state.Storage.GetStream(ctx, emoji.ImagePath)
- return rc, int64(emoji.ImageFileSize), err
- }
+ emojiURI := uris.URIForEmoji(emojiID)
+ // If category was supplied, ensure the
+ // category exists and provide it as
+ // additional info to emoji processing.
var ai *media.AdditionalEmojiInfo
- if categoryName != nil {
- category, err := p.getOrCreateEmojiCategory(ctx, *categoryName)
+ if category != nil && *category != "" {
+ category, err := p.getOrCreateEmojiCategory(ctx, *category)
if err != nil {
- err = fmt.Errorf("emojiUpdateCopy: error getting or creating category: %s", err)
return nil, gtserror.NewErrorInternalError(err)
}
@@ -355,126 +470,173 @@ func (p *Processor) emojiUpdateCopy(ctx context.Context, emoji *gtsmodel.Emoji,
}
}
- processingEmoji, err := p.mediaManager.PreProcessEmoji(ctx, data, *shortcode, newEmojiID, newEmojiURI, ai, false)
+ // Begin media processing.
+ processingEmoji, err := p.mediaManager.PreProcessEmoji(ctx,
+ data, sc, emojiID, emojiURI, ai, false,
+ )
if err != nil {
- err = fmt.Errorf("emojiUpdateCopy: error processing emoji %s: %s", emoji.ID, err)
+ err := gtserror.Newf("error processing emoji: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
+ // Complete processing immediately.
newEmoji, err := processingEmoji.LoadEmoji(ctx)
if err != nil {
- err = fmt.Errorf("emojiUpdateCopy: error loading processed emoji %s: %s", emoji.ID, err)
+ err := gtserror.Newf("error loading emoji: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, newEmoji)
if err != nil {
- err = fmt.Errorf("emojiUpdateCopy: error converting updated emoji %s to admin emoji: %s", emoji.ID, err)
+ err := gtserror.Newf("error converting emoji %s to admin emoji: %w", newEmoji.ID, err)
return nil, gtserror.NewErrorInternalError(err)
}
return adminEmoji, nil
}
-// disable a remote emoji
-func (p *Processor) emojiUpdateDisable(ctx context.Context, emoji *gtsmodel.Emoji) (*apimodel.AdminEmoji, gtserror.WithCode) {
- if emoji.Domain == "" {
- err := fmt.Errorf("emojiUpdateDisable: emoji %s is not a remote emoji, cannot disable it via this endpoint", emoji.ID)
+// emojiUpdateDisable marks the given *remote*
+// emoji as disabled by setting disabled = true.
+//
+// The provided emoji model must correspond to an
+// emoji already stored in the database + storage.
+func (p *Processor) emojiUpdateDisable(
+ ctx context.Context,
+ emoji *gtsmodel.Emoji,
+) (*apimodel.AdminEmoji, gtserror.WithCode) {
+ if emoji.IsLocal() {
+ err := fmt.Errorf("emoji %s is not a remote emoji, cannot disable it via this endpoint", emoji.ID)
return nil, gtserror.NewErrorBadRequest(err, err.Error())
}
- emojiDisabled := true
- emoji.Disabled = &emojiDisabled
- err := p.state.DB.UpdateEmoji(ctx, emoji, "disabled")
- if err != nil {
- err = fmt.Errorf("emojiUpdateDisable: error updating emoji %s: %s", emoji.ID, err)
- return nil, gtserror.NewErrorInternalError(err)
+ // Only bother with a db call
+ // if emoji not already disabled.
+ if !*emoji.Disabled {
+ emoji.Disabled = util.Ptr(true)
+ if err := p.state.DB.UpdateEmoji(ctx, emoji, "disabled"); err != nil {
+ err := gtserror.Newf("db error updating emoji %s: %w", emoji.ID, err)
+ return nil, gtserror.NewErrorInternalError(err)
+ }
}
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
if err != nil {
- err = fmt.Errorf("emojiUpdateDisable: error converting updated emoji %s to admin emoji: %s", emoji.ID, err)
+ err := gtserror.Newf("error converting emoji %s to admin emoji: %w", emoji.ID, err)
return nil, gtserror.NewErrorInternalError(err)
}
return adminEmoji, nil
}
-// modify a local emoji
-func (p *Processor) emojiUpdateModify(ctx context.Context, emoji *gtsmodel.Emoji, image *multipart.FileHeader, categoryName *string) (*apimodel.AdminEmoji, gtserror.WithCode) {
- if emoji.Domain != "" {
- err := fmt.Errorf("emojiUpdateModify: emoji %s is not a local emoji, cannot do a modify action on it", emoji.ID)
+// emojiUpdateModify modifies the given *local* emoji.
+//
+// Either one of image or category must be non-nil,
+// otherwise there's nothing to modify. If category
+// is non-nil and dereferences to an empty string,
+// category will be cleared.
+//
+// The provided emoji model must correspond to an
+// emoji already stored in the database + storage.
+func (p *Processor) emojiUpdateModify(
+ ctx context.Context,
+ emoji *gtsmodel.Emoji,
+ image *multipart.FileHeader,
+ category *string,
+) (*apimodel.AdminEmoji, gtserror.WithCode) {
+ if !emoji.IsLocal() {
+ err := fmt.Errorf("emoji %s is not a local emoji, cannot update it via this endpoint", emoji.ID)
+ return nil, gtserror.NewErrorBadRequest(err, err.Error())
+ }
+
+ // Ensure there's actually something to update.
+ if image == nil && category == nil {
+ err := errors.New("neither new category nor new image set, cannot update")
return nil, gtserror.NewErrorBadRequest(err, err.Error())
}
- // keep existing categoryID unless a new one is defined
+ // Only update category
+ // if it's changed.
var (
- updatedCategoryID = emoji.CategoryID
- updateCategoryID bool
+ newCategory *gtsmodel.EmojiCategory
+ newCategoryID string
+ updateCategoryID bool
)
- if categoryName != nil {
- category, err := p.getOrCreateEmojiCategory(ctx, *categoryName)
- if err != nil {
- err = fmt.Errorf("emojiUpdateModify: error getting or creating category: %s", err)
- return nil, gtserror.NewErrorInternalError(err)
+
+ if category != nil {
+ catName := *category
+ if catName != "" {
+ // Set new category.
+ var err error
+ newCategory, err = p.getOrCreateEmojiCategory(ctx, catName)
+ if err != nil {
+ err := gtserror.Newf("error getting or creating category: %w", err)
+ return nil, gtserror.NewErrorInternalError(err)
+ }
+
+ newCategoryID = newCategory.ID
+ } else {
+ // Clear existing category.
+ newCategoryID = ""
}
- updatedCategoryID = category.ID
- updateCategoryID = true
+ updateCategoryID = emoji.CategoryID != newCategoryID
}
- // only update image if provided with one
+ // Only update image
+ // if one is provided.
var updateImage bool
if image != nil && image.Size != 0 {
updateImage = true
}
- if !updateImage {
- // only updating fields, we only need
- // to do a database update for this
- var columns []string
-
- if updateCategoryID {
- emoji.CategoryID = updatedCategoryID
- columns = append(columns, "category_id")
- }
-
- var err error
- err = p.state.DB.UpdateEmoji(ctx, emoji, columns...)
- if err != nil {
- err = fmt.Errorf("emojiUpdateModify: error updating emoji %s: %s", emoji.ID, err)
+ if updateCategoryID && !updateImage {
+ // Only updating category; we only
+ // need to do a db update for this.
+ emoji.CategoryID = newCategoryID
+ emoji.Category = newCategory
+ if err := p.state.DB.UpdateEmoji(ctx, emoji, "category_id"); err != nil {
+ err := gtserror.Newf("db error updating emoji %s: %w", emoji.ID, err)
return nil, gtserror.NewErrorInternalError(err)
}
- } else {
- // new image, so we need to reprocess the emoji
- data := func(ctx context.Context) (reader io.ReadCloser, fileSize int64, err error) {
+ } else if updateImage {
+ // Updating image and maybe categoryID.
+ // We can do both at the same time :)
+
+ // Set data function to provided image.
+ data := func(ctx context.Context) (io.ReadCloser, int64, error) {
i, err := image.Open()
return i, image.Size, err
}
+ // If necessary, include
+ // update to categoryID too.
var ai *media.AdditionalEmojiInfo
if updateCategoryID {
ai = &media.AdditionalEmojiInfo{
- CategoryID: &updatedCategoryID,
+ CategoryID: &newCategoryID,
}
}
- processingEmoji, err := p.mediaManager.PreProcessEmoji(ctx, data, emoji.Shortcode, emoji.ID, emoji.URI, ai, true)
+ // Begin media processing.
+ processingEmoji, err := p.mediaManager.PreProcessEmoji(ctx,
+ data, emoji.Shortcode, emoji.ID, emoji.URI, ai, false,
+ )
if err != nil {
- err = fmt.Errorf("emojiUpdateModify: error processing emoji %s: %s", emoji.ID, err)
+ err := gtserror.Newf("error processing emoji: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
+ // Replace emoji ptr with newly-processed version.
emoji, err = processingEmoji.LoadEmoji(ctx)
if err != nil {
- err = fmt.Errorf("emojiUpdateModify: error loading processed emoji %s: %s", emoji.ID, err)
+ err := gtserror.Newf("error loading emoji: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
}
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
if err != nil {
- err = fmt.Errorf("emojiUpdateModify: error converting updated emoji %s to admin emoji: %s", emoji.ID, err)
+ err := gtserror.Newf("error converting emoji %s to admin emoji: %w", emoji.ID, err)
return nil, gtserror.NewErrorInternalError(err)
}
diff --git a/internal/processing/admin/emoji_test.go b/internal/processing/admin/emoji_test.go
new file mode 100644
index 000000000..17f5fc864
--- /dev/null
+++ b/internal/processing/admin/emoji_test.go
@@ -0,0 +1,67 @@
+// 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/>.
+
+package admin_test
+
+import (
+ "context"
+ "testing"
+
+ "github.com/stretchr/testify/suite"
+ apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model"
+ "github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
+ "github.com/superseriousbusiness/gotosocial/internal/util"
+)
+
+type EmojiTestSuite struct {
+ AdminStandardTestSuite
+}
+
+func (suite *EmojiTestSuite) TestUpdateEmojiCategory() {
+ ctx := context.Background()
+ testEmoji := new(gtsmodel.Emoji)
+ *testEmoji = *suite.testEmojis["rainbow"]
+
+ // Toggle the emoji category around.
+ for _, categoryName := range []string{
+ "",
+ "newCategory",
+ "newCategory",
+ "newCategory2",
+ "",
+ "reactions",
+ "",
+ "",
+ } {
+ emoji, err := suite.adminProcessor.EmojiUpdate(ctx,
+ testEmoji.ID,
+ &apimodel.EmojiUpdateRequest{
+ Type: apimodel.EmojiUpdateModify,
+ CategoryName: util.Ptr(categoryName),
+ },
+ )
+ if err != nil {
+ suite.FailNow(err.Error())
+ }
+
+ suite.Equal(categoryName, emoji.Category)
+ }
+}
+
+func TestEmojiTestSuite(t *testing.T) {
+ suite.Run(t, new(EmojiTestSuite))
+}
diff --git a/internal/processing/fedi/emoji.go b/internal/processing/fedi/emoji.go
index 4acf8c671..9ac0ea244 100644
--- a/internal/processing/fedi/emoji.go
+++ b/internal/processing/fedi/emoji.go
@@ -36,7 +36,7 @@ func (p *Processor) EmojiGet(ctx context.Context, requestedEmojiID string) (inte
return nil, gtserror.NewErrorNotFound(fmt.Errorf("database error getting emoji with id %s: %s", requestedEmojiID, err))
}
- if requestedEmoji.Domain != "" {
+ if !requestedEmoji.IsLocal() {
return nil, gtserror.NewErrorNotFound(fmt.Errorf("emoji with id %s doesn't belong to this instance (domain %s)", requestedEmojiID, requestedEmoji.Domain))
}