mirror of
https://github.com/superseriousbusiness/gotosocial.git
synced 2025-01-12 09:15:33 +00:00
[bugfix] Fix Postgres emoji delete, emoji category change (#2570)
* [bugfix] Fix Postgres emoji delete, emoji category change * revert trace logging * caching issue * update tests
This commit is contained in:
parent
14b684b2b5
commit
aa8bbe6ad2
15 changed files with 500 additions and 233 deletions
|
@ -125,7 +125,7 @@ func (m *Module) EmojiCreatePOSTHandler(c *gin.Context) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
apiEmoji, errWithCode := m.processor.Admin().EmojiCreate(c.Request.Context(), authed.Account, authed.User, form)
|
apiEmoji, errWithCode := m.processor.Admin().EmojiCreate(c.Request.Context(), authed.Account, form)
|
||||||
if errWithCode != nil {
|
if errWithCode != nil {
|
||||||
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
|
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
|
||||||
return
|
return
|
||||||
|
|
|
@ -87,7 +87,7 @@ func (suite *EmojiDeleteTestSuite) TestEmojiDelete2() {
|
||||||
suite.NoError(err)
|
suite.NoError(err)
|
||||||
suite.NotNil(b)
|
suite.NotNil(b)
|
||||||
|
|
||||||
suite.Equal(`{"error":"Bad Request: EmojiDelete: emoji with id 01GD5KP5CQEE1R3X43Y1EHS2CW was not a local emoji, will not delete"}`, string(b))
|
suite.Equal(`{"error":"Bad Request: emoji with id 01GD5KP5CQEE1R3X43Y1EHS2CW was not a local emoji, will not delete"}`, string(b))
|
||||||
|
|
||||||
// emoji should still be in the db
|
// emoji should still be in the db
|
||||||
dbEmoji, err := suite.db.GetEmojiByID(context.Background(), testEmoji.ID)
|
dbEmoji, err := suite.db.GetEmojiByID(context.Background(), testEmoji.ID)
|
||||||
|
|
|
@ -89,7 +89,7 @@ func (m *Module) EmojiGETHandler(c *gin.Context) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
emoji, errWithCode := m.processor.Admin().EmojiGet(c.Request.Context(), authed.Account, authed.User, emojiID)
|
emoji, errWithCode := m.processor.Admin().EmojiGet(c.Request.Context(), authed.Account, emojiID)
|
||||||
if errWithCode != nil {
|
if errWithCode != nil {
|
||||||
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
|
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
|
||||||
return
|
return
|
||||||
|
|
|
@ -197,7 +197,17 @@ func (m *Module) EmojisGETHandler(c *gin.Context) {
|
||||||
includeEnabled = true
|
includeEnabled = true
|
||||||
}
|
}
|
||||||
|
|
||||||
resp, errWithCode := m.processor.Admin().EmojisGet(c.Request.Context(), authed.Account, authed.User, domain, includeDisabled, includeEnabled, shortcode, maxShortcodeDomain, minShortcodeDomain, limit)
|
resp, errWithCode := m.processor.Admin().EmojisGet(
|
||||||
|
c.Request.Context(),
|
||||||
|
authed.Account,
|
||||||
|
domain,
|
||||||
|
includeDisabled,
|
||||||
|
includeEnabled,
|
||||||
|
shortcode,
|
||||||
|
maxShortcodeDomain,
|
||||||
|
minShortcodeDomain,
|
||||||
|
limit,
|
||||||
|
)
|
||||||
if errWithCode != nil {
|
if errWithCode != nil {
|
||||||
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
|
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
|
||||||
return
|
return
|
||||||
|
|
|
@ -339,7 +339,7 @@ func (suite *EmojiUpdateTestSuite) TestEmojiUpdateDisableLocalEmoji() {
|
||||||
b, err := ioutil.ReadAll(result.Body)
|
b, err := ioutil.ReadAll(result.Body)
|
||||||
suite.NoError(err)
|
suite.NoError(err)
|
||||||
|
|
||||||
suite.Equal(`{"error":"Bad Request: emojiUpdateDisable: emoji 01F8MH9H8E4VG3KDYJR9EGPXCQ is not a remote emoji, cannot disable it via this endpoint"}`, string(b))
|
suite.Equal(`{"error":"Bad Request: emoji 01F8MH9H8E4VG3KDYJR9EGPXCQ is not a remote emoji, cannot disable it via this endpoint"}`, string(b))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (suite *EmojiUpdateTestSuite) TestEmojiUpdateModifyRemoteEmoji() {
|
func (suite *EmojiUpdateTestSuite) TestEmojiUpdateModifyRemoteEmoji() {
|
||||||
|
@ -372,7 +372,7 @@ func (suite *EmojiUpdateTestSuite) TestEmojiUpdateModifyRemoteEmoji() {
|
||||||
b, err := ioutil.ReadAll(result.Body)
|
b, err := ioutil.ReadAll(result.Body)
|
||||||
suite.NoError(err)
|
suite.NoError(err)
|
||||||
|
|
||||||
suite.Equal(`{"error":"Bad Request: emojiUpdateModify: emoji 01GD5KP5CQEE1R3X43Y1EHS2CW is not a local emoji, cannot do a modify action on it"}`, string(b))
|
suite.Equal(`{"error":"Bad Request: emoji 01GD5KP5CQEE1R3X43Y1EHS2CW is not a local emoji, cannot update it via this endpoint"}`, string(b))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (suite *EmojiUpdateTestSuite) TestEmojiUpdateModifyNoParams() {
|
func (suite *EmojiUpdateTestSuite) TestEmojiUpdateModifyNoParams() {
|
||||||
|
@ -439,7 +439,7 @@ func (suite *EmojiUpdateTestSuite) TestEmojiUpdateCopyLocalToLocal() {
|
||||||
b, err := ioutil.ReadAll(result.Body)
|
b, err := ioutil.ReadAll(result.Body)
|
||||||
suite.NoError(err)
|
suite.NoError(err)
|
||||||
|
|
||||||
suite.Equal(`{"error":"Bad Request: emojiUpdateCopy: emoji 01F8MH9H8E4VG3KDYJR9EGPXCQ is not a remote emoji, cannot copy it to local"}`, string(b))
|
suite.Equal(`{"error":"Bad Request: emoji 01F8MH9H8E4VG3KDYJR9EGPXCQ is not a remote emoji, cannot copy it to local"}`, string(b))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (suite *EmojiUpdateTestSuite) TestEmojiUpdateCopyEmptyShortcode() {
|
func (suite *EmojiUpdateTestSuite) TestEmojiUpdateCopyEmptyShortcode() {
|
||||||
|
@ -540,7 +540,7 @@ func (suite *EmojiUpdateTestSuite) TestEmojiUpdateCopyShortcodeAlreadyInUse() {
|
||||||
b, err := ioutil.ReadAll(result.Body)
|
b, err := ioutil.ReadAll(result.Body)
|
||||||
suite.NoError(err)
|
suite.NoError(err)
|
||||||
|
|
||||||
suite.Equal(`{"error":"Conflict: emojiUpdateCopy: emoji 01GD5KP5CQEE1R3X43Y1EHS2CW could not be copied, emoji with shortcode rainbow already exists on this instance"}`, string(b))
|
suite.Equal(`{"error":"Conflict: emoji with shortcode rainbow already exists on this instance"}`, string(b))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestEmojiUpdateTestSuite(t *testing.T) {
|
func TestEmojiUpdateTestSuite(t *testing.T) {
|
||||||
|
|
|
@ -132,28 +132,32 @@ func (e *emojiDB) DeleteEmojiByID(ctx context.Context, id string) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, statusID := range statusIDs {
|
for _, statusID := range statusIDs {
|
||||||
var emojiIDs []string
|
status := new(gtsmodel.Status)
|
||||||
|
|
||||||
// Select statuses with ID.
|
// Select status emoji IDs.
|
||||||
if _, err := tx.NewSelect().
|
if err := tx.NewSelect().
|
||||||
Table("statuses").
|
Model(status).
|
||||||
Column("emojis").
|
Column("emojis").
|
||||||
Where("? = ?", bun.Ident("id"), statusID).
|
Where("? = ?", bun.Ident("id"), statusID).
|
||||||
Exec(ctx); err != nil &&
|
Scan(ctx); err != nil &&
|
||||||
err != sql.ErrNoRows {
|
err != sql.ErrNoRows {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete all instances of this emoji ID from status emojis.
|
// Delete all instances of this
|
||||||
emojiIDs = slices.DeleteFunc(emojiIDs, func(emojiID string) bool {
|
// emoji ID from status emoji IDs.
|
||||||
return emojiID == id
|
status.EmojiIDs = slices.DeleteFunc(
|
||||||
})
|
status.EmojiIDs,
|
||||||
|
func(emojiID string) bool {
|
||||||
|
return emojiID == id
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
// Update status emoji IDs.
|
// Update status emoji IDs.
|
||||||
if _, err := tx.NewUpdate().
|
if _, err := tx.NewUpdate().
|
||||||
Table("statuses").
|
Model(status).
|
||||||
Where("? = ?", bun.Ident("id"), statusID).
|
Where("? = ?", bun.Ident("id"), statusID).
|
||||||
Set("emojis = ?", emojiIDs).
|
Column("emojis").
|
||||||
Exec(ctx); err != nil &&
|
Exec(ctx); err != nil &&
|
||||||
err != sql.ErrNoRows {
|
err != sql.ErrNoRows {
|
||||||
return err
|
return err
|
||||||
|
@ -161,35 +165,39 @@ func (e *emojiDB) DeleteEmojiByID(ctx context.Context, id string) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, accountID := range accountIDs {
|
for _, accountID := range accountIDs {
|
||||||
var emojiIDs []string
|
account := new(gtsmodel.Account)
|
||||||
|
|
||||||
// Select account with ID.
|
// Select account emoji IDs.
|
||||||
if _, err := tx.NewSelect().
|
if err := tx.NewSelect().
|
||||||
Table("accounts").
|
Model(account).
|
||||||
Column("emojis").
|
Column("emojis").
|
||||||
Where("? = ?", bun.Ident("id"), accountID).
|
Where("? = ?", bun.Ident("id"), accountID).
|
||||||
Exec(ctx); err != nil &&
|
Scan(ctx); err != nil &&
|
||||||
err != sql.ErrNoRows {
|
err != sql.ErrNoRows {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete all instances of this emoji ID from account emojis.
|
// Delete all instances of this
|
||||||
emojiIDs = slices.DeleteFunc(emojiIDs, func(emojiID string) bool {
|
// emoji ID from account emoji IDs.
|
||||||
return emojiID == id
|
account.EmojiIDs = slices.DeleteFunc(
|
||||||
})
|
account.EmojiIDs,
|
||||||
|
func(emojiID string) bool {
|
||||||
|
return emojiID == id
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
// Update account emoji IDs.
|
// Update account emoji IDs.
|
||||||
if _, err := tx.NewUpdate().
|
if _, err := tx.NewUpdate().
|
||||||
Table("accounts").
|
Model(account).
|
||||||
Where("? = ?", bun.Ident("id"), accountID).
|
Where("? = ?", bun.Ident("id"), accountID).
|
||||||
Set("emojis = ?", emojiIDs).
|
Column("emojis").
|
||||||
Exec(ctx); err != nil &&
|
Exec(ctx); err != nil &&
|
||||||
err != sql.ErrNoRows {
|
err != sql.ErrNoRows {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delete emoji from database.
|
// Finally, delete emoji from database.
|
||||||
if _, err := tx.NewDelete().
|
if _, err := tx.NewDelete().
|
||||||
Table("emojis").
|
Table("emojis").
|
||||||
Where("? = ?", bun.Ident("id"), id).
|
Where("? = ?", bun.Ident("id"), id).
|
||||||
|
|
|
@ -23,6 +23,7 @@ import (
|
||||||
|
|
||||||
"github.com/stretchr/testify/suite"
|
"github.com/stretchr/testify/suite"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/db"
|
"github.com/superseriousbusiness/gotosocial/internal/db"
|
||||||
|
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
||||||
"github.com/superseriousbusiness/gotosocial/testrig"
|
"github.com/superseriousbusiness/gotosocial/testrig"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -160,6 +161,16 @@ func (suite *EmojiTestSuite) TestGetEmojiCategory() {
|
||||||
suite.NotNil(category)
|
suite.NotNil(category)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (suite *EmojiTestSuite) TestUpdateEmojiCategory() {
|
||||||
|
testEmoji := new(gtsmodel.Emoji)
|
||||||
|
*testEmoji = *suite.testEmojis["rainbow"]
|
||||||
|
|
||||||
|
testEmoji.CategoryID = ""
|
||||||
|
|
||||||
|
err := suite.db.UpdateEmoji(context.Background(), testEmoji, "category_id")
|
||||||
|
suite.NoError(err)
|
||||||
|
}
|
||||||
|
|
||||||
func TestEmojiTestSuite(t *testing.T) {
|
func TestEmojiTestSuite(t *testing.T) {
|
||||||
suite.Run(t, new(EmojiTestSuite))
|
suite.Run(t, new(EmojiTestSuite))
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,3 +44,10 @@ type Emoji struct {
|
||||||
CategoryID string `bun:"type:CHAR(26),nullzero"` // ID of the category this emoji belongs to.
|
CategoryID string `bun:"type:CHAR(26),nullzero"` // ID of the category this emoji belongs to.
|
||||||
Cached *bool `bun:",nullzero,notnull,default:false"`
|
Cached *bool `bun:",nullzero,notnull,default:false"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// IsLocal returns true if the emoji is
|
||||||
|
// local to this instance., ie., it did
|
||||||
|
// not originate from a remote instance.
|
||||||
|
func (e *Emoji) IsLocal() bool {
|
||||||
|
return e.Domain == ""
|
||||||
|
}
|
||||||
|
|
|
@ -158,7 +158,7 @@ func (p *ProcessingEmoji) store(ctx context.Context) error {
|
||||||
|
|
||||||
var maxSize bytesize.Size
|
var maxSize bytesize.Size
|
||||||
|
|
||||||
if p.emoji.Domain == "" {
|
if p.emoji.IsLocal() {
|
||||||
// this is a local emoji upload
|
// this is a local emoji upload
|
||||||
maxSize = config.GetMediaEmojiLocalMaxSize()
|
maxSize = config.GetMediaEmojiLocalMaxSize()
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -61,7 +61,7 @@ func (m *Manager) RefetchEmojis(ctx context.Context, domain string, dereferenceM
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, emoji := range emojis {
|
for _, emoji := range emojis {
|
||||||
if emoji.Domain == "" {
|
if emoji.IsLocal() {
|
||||||
// never try to refetch local emojis
|
// never try to refetch local emojis
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
|
@ -62,6 +62,7 @@ type AdminStandardTestSuite struct {
|
||||||
testFollows map[string]*gtsmodel.Follow
|
testFollows map[string]*gtsmodel.Follow
|
||||||
testAttachments map[string]*gtsmodel.MediaAttachment
|
testAttachments map[string]*gtsmodel.MediaAttachment
|
||||||
testStatuses map[string]*gtsmodel.Status
|
testStatuses map[string]*gtsmodel.Status
|
||||||
|
testEmojis map[string]*gtsmodel.Emoji
|
||||||
|
|
||||||
// module being tested
|
// module being tested
|
||||||
adminProcessor *admin.Processor
|
adminProcessor *admin.Processor
|
||||||
|
@ -76,6 +77,7 @@ func (suite *AdminStandardTestSuite) SetupSuite() {
|
||||||
suite.testFollows = testrig.NewTestFollows()
|
suite.testFollows = testrig.NewTestFollows()
|
||||||
suite.testAttachments = testrig.NewTestAttachments()
|
suite.testAttachments = testrig.NewTestAttachments()
|
||||||
suite.testStatuses = testrig.NewTestStatuses()
|
suite.testStatuses = testrig.NewTestStatuses()
|
||||||
|
suite.testEmojis = testrig.NewTestEmojis()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (suite *AdminStandardTestSuite) SetupTest() {
|
func (suite *AdminStandardTestSuite) SetupTest() {
|
||||||
|
|
|
@ -36,37 +36,39 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
// EmojiCreate creates a custom emoji on this instance.
|
// 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) {
|
func (p *Processor) EmojiCreate(
|
||||||
if !*user.Admin {
|
ctx context.Context,
|
||||||
return nil, gtserror.NewErrorUnauthorized(fmt.Errorf("user %s not an admin", user.ID), "user is not an admin")
|
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, "")
|
maybeExisting, err := p.state.DB.GetEmojiByShortcodeDomain(ctx, form.Shortcode, "")
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
|
||||||
if maybeExisting != nil {
|
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))
|
err := fmt.Errorf("emoji with shortcode %s already exists", form.Shortcode)
|
||||||
|
return nil, gtserror.NewErrorConflict(err, err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
if err != nil && err != db.ErrNoEntries {
|
// Prepare data function for emoji processing
|
||||||
return nil, gtserror.NewErrorInternalError(fmt.Errorf("error checking existence of emoji with shortcode %s: %s", form.Shortcode, err))
|
// (just read data from the submitted form).
|
||||||
}
|
|
||||||
|
|
||||||
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")
|
|
||||||
}
|
|
||||||
|
|
||||||
emojiURI := uris.URIForEmoji(emojiID)
|
|
||||||
|
|
||||||
data := func(innerCtx context.Context) (io.ReadCloser, int64, error) {
|
data := func(innerCtx context.Context) (io.ReadCloser, int64, error) {
|
||||||
f, err := form.Image.Open()
|
f, err := form.Image.Open()
|
||||||
return f, form.Image.Size, err
|
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
|
var ai *media.AdditionalEmojiInfo
|
||||||
if form.CategoryName != "" {
|
if form.CategoryName != "" {
|
||||||
category, err := p.getOrCreateEmojiCategory(ctx, form.CategoryName)
|
category, err := p.getOrCreateEmojiCategory(ctx, form.CategoryName)
|
||||||
if err != nil {
|
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{
|
ai = &media.AdditionalEmojiInfo{
|
||||||
|
@ -74,73 +76,68 @@ 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 {
|
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)
|
emoji, err := processingEmoji.LoadEmoji(ctx)
|
||||||
if err != nil {
|
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)
|
apiEmoji, err := p.converter.EmojiToAPIEmoji(ctx, emoji)
|
||||||
if err != nil {
|
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
|
return &apiEmoji, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// EmojisGet returns an admin view of custom emojis, filtered with the given parameters.
|
// emojisGetFilterParams builds extra
|
||||||
func (p *Processor) EmojisGet(
|
// query parameters to return as part
|
||||||
ctx context.Context,
|
// of an Emojis pageable response.
|
||||||
account *gtsmodel.Account,
|
//
|
||||||
user *gtsmodel.User,
|
// The returned string will look like:
|
||||||
|
//
|
||||||
|
// "filter=domain:all,enabled,shortcode:example"
|
||||||
|
func emojisGetFilterParams(
|
||||||
|
shortcode string,
|
||||||
domain string,
|
domain string,
|
||||||
includeDisabled bool,
|
includeDisabled bool,
|
||||||
includeEnabled bool,
|
includeEnabled bool,
|
||||||
shortcode string,
|
) string {
|
||||||
maxShortcodeDomain string,
|
var filterBuilder strings.Builder
|
||||||
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)
|
|
||||||
if err != nil && !errors.Is(err, db.ErrNoEntries) {
|
|
||||||
err := fmt.Errorf("EmojisGet: db error: %s", err)
|
|
||||||
return nil, gtserror.NewErrorInternalError(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
count := len(emojis)
|
|
||||||
if count == 0 {
|
|
||||||
return util.EmptyPageableResponse(), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
items := make([]interface{}, 0, count)
|
|
||||||
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)
|
|
||||||
return nil, gtserror.NewErrorInternalError(err)
|
|
||||||
}
|
|
||||||
items = append(items, adminEmoji)
|
|
||||||
}
|
|
||||||
|
|
||||||
filterBuilder := strings.Builder{}
|
|
||||||
filterBuilder.WriteString("filter=")
|
filterBuilder.WriteString("filter=")
|
||||||
|
|
||||||
switch domain {
|
switch domain {
|
||||||
case "", "local":
|
case "", "local":
|
||||||
|
// Local emojis only.
|
||||||
filterBuilder.WriteString("domain:local")
|
filterBuilder.WriteString("domain:local")
|
||||||
|
|
||||||
case db.EmojiAllDomains:
|
case db.EmojiAllDomains:
|
||||||
|
// Local or remote.
|
||||||
filterBuilder.WriteString("domain:all")
|
filterBuilder.WriteString("domain:all")
|
||||||
|
|
||||||
default:
|
default:
|
||||||
filterBuilder.WriteString("domain:")
|
// Specific domain only.
|
||||||
filterBuilder.WriteString(domain)
|
filterBuilder.WriteString("domain:" + domain)
|
||||||
}
|
}
|
||||||
|
|
||||||
if includeDisabled != includeEnabled {
|
if includeDisabled != includeEnabled {
|
||||||
|
@ -153,108 +150,182 @@ func (p *Processor) EmojisGet(
|
||||||
}
|
}
|
||||||
|
|
||||||
if shortcode != "" {
|
if shortcode != "" {
|
||||||
filterBuilder.WriteString(",shortcode:")
|
// Specific shortcode only.
|
||||||
filterBuilder.WriteString(shortcode)
|
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,
|
||||||
|
domain string,
|
||||||
|
includeDisabled bool,
|
||||||
|
includeEnabled bool,
|
||||||
|
shortcode string,
|
||||||
|
maxShortcodeDomain string,
|
||||||
|
minShortcodeDomain string,
|
||||||
|
limit int,
|
||||||
|
) (*apimodel.PageableResponse, gtserror.WithCode) {
|
||||||
|
emojis, err := p.state.DB.GetEmojisBy(ctx,
|
||||||
|
domain,
|
||||||
|
includeDisabled,
|
||||||
|
includeEnabled,
|
||||||
|
shortcode,
|
||||||
|
maxShortcodeDomain,
|
||||||
|
minShortcodeDomain,
|
||||||
|
limit,
|
||||||
|
)
|
||||||
|
if err != nil && !errors.Is(err, db.ErrNoEntries) {
|
||||||
|
err := gtserror.Newf("db error: %w", err)
|
||||||
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
count := len(emojis)
|
||||||
|
if count == 0 {
|
||||||
|
return util.EmptyPageableResponse(), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
items := make([]interface{}, 0, count)
|
||||||
|
for _, emoji := range emojis {
|
||||||
|
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
|
||||||
|
if err != nil {
|
||||||
|
err := gtserror.Newf("error converting emoji to admin model emoji: %w", err)
|
||||||
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
|
}
|
||||||
|
items = append(items, adminEmoji)
|
||||||
}
|
}
|
||||||
|
|
||||||
return util.PackagePageableResponse(util.PageableResponseParams{
|
return util.PackagePageableResponse(util.PageableResponseParams{
|
||||||
Items: items,
|
Items: items,
|
||||||
Path: "api/v1/admin/custom_emojis",
|
Path: "api/v1/admin/custom_emojis",
|
||||||
NextMaxIDKey: "max_shortcode_domain",
|
NextMaxIDKey: "max_shortcode_domain",
|
||||||
NextMaxIDValue: util.ShortcodeDomain(emojis[count-1]),
|
NextMaxIDValue: util.ShortcodeDomain(emojis[count-1]),
|
||||||
PrevMinIDKey: "min_shortcode_domain",
|
PrevMinIDKey: "min_shortcode_domain",
|
||||||
PrevMinIDValue: util.ShortcodeDomain(emojis[0]),
|
PrevMinIDValue: util.ShortcodeDomain(emojis[0]),
|
||||||
Limit: limit,
|
Limit: limit,
|
||||||
ExtraQueryParams: []string{filterBuilder.String()},
|
ExtraQueryParams: []string{
|
||||||
|
emojisGetFilterParams(
|
||||||
|
shortcode,
|
||||||
|
domain,
|
||||||
|
includeDisabled,
|
||||||
|
includeEnabled,
|
||||||
|
),
|
||||||
|
},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// EmojiGet returns the admin view of one custom emoji with the given id.
|
// EmojiGet returns the admin view of
|
||||||
func (p *Processor) EmojiGet(ctx context.Context, account *gtsmodel.Account, user *gtsmodel.User, id string) (*apimodel.AdminEmoji, gtserror.WithCode) {
|
// one custom emoji with the given id.
|
||||||
if !*user.Admin {
|
func (p *Processor) EmojiGet(
|
||||||
return nil, gtserror.NewErrorUnauthorized(fmt.Errorf("user %s not an admin", user.ID), "user is not an admin")
|
ctx context.Context,
|
||||||
|
account *gtsmodel.Account,
|
||||||
|
id string,
|
||||||
|
) (*apimodel.AdminEmoji, gtserror.WithCode) {
|
||||||
|
emoji, err := p.state.DB.GetEmojiByID(ctx, id)
|
||||||
|
if err != nil && !errors.Is(err, db.ErrNoEntries) {
|
||||||
|
err := gtserror.Newf("db error: %w", err)
|
||||||
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
emoji, err := p.state.DB.GetEmojiByID(ctx, id)
|
if emoji == nil {
|
||||||
if err != nil {
|
err := gtserror.Newf("no emoji with id %s found in the db", id)
|
||||||
if errors.Is(err, db.ErrNoEntries) {
|
return nil, gtserror.NewErrorNotFound(err)
|
||||||
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)
|
|
||||||
return nil, gtserror.NewErrorInternalError(err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
|
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
|
||||||
if err != nil {
|
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 nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
return adminEmoji, nil
|
return adminEmoji, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// EmojiDelete deletes one emoji from the database, with the given id.
|
// EmojiDelete deletes one *local* emoji
|
||||||
func (p *Processor) EmojiDelete(ctx context.Context, id string) (*apimodel.AdminEmoji, gtserror.WithCode) {
|
// 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)
|
emoji, err := p.state.DB.GetEmojiByID(ctx, id)
|
||||||
if err != nil {
|
if err != nil && !errors.Is(err, db.ErrNoEntries) {
|
||||||
if errors.Is(err, db.ErrNoEntries) {
|
err := gtserror.Newf("db error: %w", err)
|
||||||
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)
|
|
||||||
return nil, gtserror.NewErrorInternalError(err)
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if emoji.Domain != "" {
|
if emoji == nil {
|
||||||
err = fmt.Errorf("EmojiDelete: emoji with id %s was not a local emoji, will not delete", id)
|
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())
|
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)
|
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
|
||||||
if err != nil {
|
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)
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := p.state.DB.DeleteEmojiByID(ctx, id); err != nil {
|
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 nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
return adminEmoji, nil
|
return adminEmoji, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// EmojiUpdate updates one emoji with the given id, using the provided form parameters.
|
// EmojiUpdate updates one emoji with the
|
||||||
func (p *Processor) EmojiUpdate(ctx context.Context, id string, form *apimodel.EmojiUpdateRequest) (*apimodel.AdminEmoji, gtserror.WithCode) {
|
// 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)
|
emoji, err := p.state.DB.GetEmojiByID(ctx, id)
|
||||||
if err != nil {
|
if err != nil && !errors.Is(err, db.ErrNoEntries) {
|
||||||
if errors.Is(err, db.ErrNoEntries) {
|
err := gtserror.Newf("db error: %w", err)
|
||||||
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)
|
|
||||||
return nil, gtserror.NewErrorInternalError(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:
|
case apimodel.EmojiUpdateCopy:
|
||||||
return p.emojiUpdateCopy(ctx, emoji, form.Shortcode, form.CategoryName)
|
return p.emojiUpdateCopy(ctx, emoji, form.Shortcode, form.CategoryName)
|
||||||
|
|
||||||
case apimodel.EmojiUpdateDisable:
|
case apimodel.EmojiUpdateDisable:
|
||||||
return p.emojiUpdateDisable(ctx, emoji)
|
return p.emojiUpdateDisable(ctx, emoji)
|
||||||
|
|
||||||
case apimodel.EmojiUpdateModify:
|
case apimodel.EmojiUpdateModify:
|
||||||
return p.emojiUpdateModify(ctx, emoji, form.Image, form.CategoryName)
|
return p.emojiUpdateModify(ctx, emoji, form.Image, form.CategoryName)
|
||||||
|
|
||||||
default:
|
default:
|
||||||
err := errors.New("unrecognized emoji action type")
|
err := fmt.Errorf("unrecognized emoji action type %s", t)
|
||||||
return nil, gtserror.NewErrorBadRequest(err, err.Error())
|
return nil, gtserror.NewErrorBadRequest(err, err.Error())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// EmojiCategoriesGet returns all custom emoji categories that exist on this instance.
|
// EmojiCategoriesGet returns all custom emoji
|
||||||
func (p *Processor) EmojiCategoriesGet(ctx context.Context) ([]*apimodel.EmojiCategory, gtserror.WithCode) {
|
// categories that exist on this instance.
|
||||||
|
func (p *Processor) EmojiCategoriesGet(
|
||||||
|
ctx context.Context,
|
||||||
|
) ([]*apimodel.EmojiCategory, gtserror.WithCode) {
|
||||||
categories, err := p.state.DB.GetEmojiCategories(ctx)
|
categories, err := p.state.DB.GetEmojiCategories(ctx)
|
||||||
if err != nil {
|
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)
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -262,7 +333,7 @@ func (p *Processor) EmojiCategoriesGet(ctx context.Context) ([]*apimodel.EmojiCa
|
||||||
for _, category := range categories {
|
for _, category := range categories {
|
||||||
apiCategory, err := p.converter.EmojiCategoryToAPIEmojiCategory(ctx, category)
|
apiCategory, err := p.converter.EmojiCategoryToAPIEmojiCategory(ctx, category)
|
||||||
if err != nil {
|
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)
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
apiCategories = append(apiCategories, apiCategory)
|
apiCategories = append(apiCategories, apiCategory)
|
||||||
|
@ -275,22 +346,35 @@ func (p *Processor) EmojiCategoriesGet(ctx context.Context) ([]*apimodel.EmojiCa
|
||||||
UTIL FUNCTIONS
|
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)
|
category, err := p.state.DB.GetEmojiCategoryByName(ctx, name)
|
||||||
if err == nil {
|
if err != nil && !errors.Is(err, db.ErrNoEntries) {
|
||||||
|
return nil, gtserror.Newf(
|
||||||
|
"database error trying get emoji category %s: %w",
|
||||||
|
name, err,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
if category != nil {
|
||||||
|
// We had it already.
|
||||||
return category, nil
|
return category, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
if err != nil && !errors.Is(err, db.ErrNoEntries) {
|
// We don't have the category yet,
|
||||||
err = fmt.Errorf("GetOrCreateEmojiCategory: database error trying get emoji category by name: %s", err)
|
// create it with the given name.
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// we don't have the category yet, just create it with the given name
|
|
||||||
categoryID, err := id.NewRandomULID()
|
categoryID, err := id.NewRandomULID()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
err = fmt.Errorf("GetOrCreateEmojiCategory: error generating id for new emoji category: %s", err)
|
return nil, gtserror.Newf(
|
||||||
return nil, err
|
"error generating id for new emoji category %s: %w",
|
||||||
|
name, err,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
category = >smodel.EmojiCategory{
|
category = >smodel.EmojiCategory{
|
||||||
|
@ -299,54 +383,85 @@ func (p *Processor) getOrCreateEmojiCategory(ctx context.Context, name string) (
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := p.state.DB.PutEmojiCategory(ctx, category); err != nil {
|
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, gtserror.Newf(
|
||||||
return nil, err
|
"db error putting new emoji category %s: %w",
|
||||||
|
name, err,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
return category, nil
|
return category, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// copy an emoji from remote to local
|
// emojiUpdateCopy copies and stores the given
|
||||||
func (p *Processor) emojiUpdateCopy(ctx context.Context, emoji *gtsmodel.Emoji, shortcode *string, categoryName *string) (*apimodel.AdminEmoji, gtserror.WithCode) {
|
// *remote* emoji as a *local* emoji, preserving
|
||||||
if emoji.Domain == "" {
|
// the same image, and using the provided shortcode.
|
||||||
err := fmt.Errorf("emojiUpdateCopy: emoji %s is not a remote emoji, cannot copy it to local", emoji.ID)
|
//
|
||||||
|
// 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())
|
return nil, gtserror.NewErrorBadRequest(err, err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
if shortcode == nil {
|
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())
|
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 {
|
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())
|
return nil, gtserror.NewErrorConflict(err, err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
if err != nil && err != db.ErrNoEntries {
|
// We don't have an emoji with this
|
||||||
err := fmt.Errorf("emojiUpdateCopy: emoji %s could not be copied, error checking existence of emoji with shortcode %s: %s", emoji.ID, *shortcode, err)
|
// shortcode yet! Prepare to create it.
|
||||||
return nil, gtserror.NewErrorInternalError(err)
|
|
||||||
|
// 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 {
|
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)
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
newEmojiURI := uris.URIForEmoji(newEmojiID)
|
emojiURI := uris.URIForEmoji(emojiID)
|
||||||
|
|
||||||
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
|
|
||||||
}
|
|
||||||
|
|
||||||
|
// If category was supplied, ensure the
|
||||||
|
// category exists and provide it as
|
||||||
|
// additional info to emoji processing.
|
||||||
var ai *media.AdditionalEmojiInfo
|
var ai *media.AdditionalEmojiInfo
|
||||||
if categoryName != nil {
|
if category != nil && *category != "" {
|
||||||
category, err := p.getOrCreateEmojiCategory(ctx, *categoryName)
|
category, err := p.getOrCreateEmojiCategory(ctx, *category)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
err = fmt.Errorf("emojiUpdateCopy: error getting or creating category: %s", err)
|
|
||||||
return nil, gtserror.NewErrorInternalError(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 {
|
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)
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Complete processing immediately.
|
||||||
newEmoji, err := processingEmoji.LoadEmoji(ctx)
|
newEmoji, err := processingEmoji.LoadEmoji(ctx)
|
||||||
if err != nil {
|
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)
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, newEmoji)
|
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, newEmoji)
|
||||||
if err != nil {
|
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 nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
return adminEmoji, nil
|
return adminEmoji, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// disable a remote emoji
|
// emojiUpdateDisable marks the given *remote*
|
||||||
func (p *Processor) emojiUpdateDisable(ctx context.Context, emoji *gtsmodel.Emoji) (*apimodel.AdminEmoji, gtserror.WithCode) {
|
// emoji as disabled by setting disabled = true.
|
||||||
if emoji.Domain == "" {
|
//
|
||||||
err := fmt.Errorf("emojiUpdateDisable: emoji %s is not a remote emoji, cannot disable it via this endpoint", emoji.ID)
|
// 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())
|
return nil, gtserror.NewErrorBadRequest(err, err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
emojiDisabled := true
|
// Only bother with a db call
|
||||||
emoji.Disabled = &emojiDisabled
|
// if emoji not already disabled.
|
||||||
err := p.state.DB.UpdateEmoji(ctx, emoji, "disabled")
|
if !*emoji.Disabled {
|
||||||
if err != nil {
|
emoji.Disabled = util.Ptr(true)
|
||||||
err = fmt.Errorf("emojiUpdateDisable: error updating emoji %s: %s", emoji.ID, err)
|
if err := p.state.DB.UpdateEmoji(ctx, emoji, "disabled"); err != nil {
|
||||||
return nil, gtserror.NewErrorInternalError(err)
|
err := gtserror.Newf("db error updating emoji %s: %w", emoji.ID, err)
|
||||||
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
|
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
|
||||||
if err != nil {
|
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 nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
return adminEmoji, nil
|
return adminEmoji, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// modify a local emoji
|
// emojiUpdateModify modifies the given *local* emoji.
|
||||||
func (p *Processor) emojiUpdateModify(ctx context.Context, emoji *gtsmodel.Emoji, image *multipart.FileHeader, categoryName *string) (*apimodel.AdminEmoji, gtserror.WithCode) {
|
//
|
||||||
if emoji.Domain != "" {
|
// Either one of image or category must be non-nil,
|
||||||
err := fmt.Errorf("emojiUpdateModify: emoji %s is not a local emoji, cannot do a modify action on it", emoji.ID)
|
// 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())
|
return nil, gtserror.NewErrorBadRequest(err, err.Error())
|
||||||
}
|
}
|
||||||
|
|
||||||
// keep existing categoryID unless a new one is defined
|
// Ensure there's actually something to update.
|
||||||
var (
|
if image == nil && category == nil {
|
||||||
updatedCategoryID = emoji.CategoryID
|
err := errors.New("neither new category nor new image set, cannot update")
|
||||||
updateCategoryID bool
|
return nil, gtserror.NewErrorBadRequest(err, err.Error())
|
||||||
)
|
|
||||||
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)
|
|
||||||
}
|
|
||||||
|
|
||||||
updatedCategoryID = category.ID
|
|
||||||
updateCategoryID = true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// only update image if provided with one
|
// Only update category
|
||||||
|
// if it's changed.
|
||||||
|
var (
|
||||||
|
newCategory *gtsmodel.EmojiCategory
|
||||||
|
newCategoryID string
|
||||||
|
updateCategoryID bool
|
||||||
|
)
|
||||||
|
|
||||||
|
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 = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
updateCategoryID = emoji.CategoryID != newCategoryID
|
||||||
|
}
|
||||||
|
|
||||||
|
// Only update image
|
||||||
|
// if one is provided.
|
||||||
var updateImage bool
|
var updateImage bool
|
||||||
if image != nil && image.Size != 0 {
|
if image != nil && image.Size != 0 {
|
||||||
updateImage = true
|
updateImage = true
|
||||||
}
|
}
|
||||||
|
|
||||||
if !updateImage {
|
if updateCategoryID && !updateImage {
|
||||||
// only updating fields, we only need
|
// Only updating category; we only
|
||||||
// to do a database update for this
|
// need to do a db update for this.
|
||||||
var columns []string
|
emoji.CategoryID = newCategoryID
|
||||||
|
emoji.Category = newCategory
|
||||||
if updateCategoryID {
|
if err := p.state.DB.UpdateEmoji(ctx, emoji, "category_id"); err != nil {
|
||||||
emoji.CategoryID = updatedCategoryID
|
err := gtserror.Newf("db error updating emoji %s: %w", emoji.ID, err)
|
||||||
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)
|
|
||||||
return nil, gtserror.NewErrorInternalError(err)
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
} else {
|
} else if updateImage {
|
||||||
// new image, so we need to reprocess the emoji
|
// Updating image and maybe categoryID.
|
||||||
data := func(ctx context.Context) (reader io.ReadCloser, fileSize int64, err error) {
|
// 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()
|
i, err := image.Open()
|
||||||
return i, image.Size, err
|
return i, image.Size, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If necessary, include
|
||||||
|
// update to categoryID too.
|
||||||
var ai *media.AdditionalEmojiInfo
|
var ai *media.AdditionalEmojiInfo
|
||||||
if updateCategoryID {
|
if updateCategoryID {
|
||||||
ai = &media.AdditionalEmojiInfo{
|
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 {
|
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)
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Replace emoji ptr with newly-processed version.
|
||||||
emoji, err = processingEmoji.LoadEmoji(ctx)
|
emoji, err = processingEmoji.LoadEmoji(ctx)
|
||||||
if err != nil {
|
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)
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
|
adminEmoji, err := p.converter.EmojiToAdminAPIEmoji(ctx, emoji)
|
||||||
if err != nil {
|
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)
|
return nil, gtserror.NewErrorInternalError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
67
internal/processing/admin/emoji_test.go
Normal file
67
internal/processing/admin/emoji_test.go
Normal file
|
@ -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))
|
||||||
|
}
|
|
@ -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))
|
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))
|
return nil, gtserror.NewErrorNotFound(fmt.Errorf("emoji with id %s doesn't belong to this instance (domain %s)", requestedEmojiID, requestedEmoji.Domain))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -612,7 +612,7 @@ func (c *Converter) EmojiToAdminAPIEmoji(ctx context.Context, e *gtsmodel.Emoji)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if e.Domain != "" {
|
if !e.IsLocal() {
|
||||||
// Domain may be in Punycode,
|
// Domain may be in Punycode,
|
||||||
// de-punify it just in case.
|
// de-punify it just in case.
|
||||||
var err error
|
var err error
|
||||||
|
|
Loading…
Reference in a new issue