summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--go.mod2
-rw-r--r--go.sum4
-rw-r--r--internal/api/client/exports/exports_test.go4
-rw-r--r--internal/api/client/lists/listaccounts.go36
-rw-r--r--internal/api/client/lists/listaccounts_test.go13
-rw-r--r--internal/api/client/lists/listaccountsadd_test.go7
-rw-r--r--internal/cache/cache.go12
-rw-r--r--internal/cache/db.go151
-rw-r--r--internal/cache/invalidate.go82
-rw-r--r--internal/cache/size.go21
-rw-r--r--internal/config/config.go6
-rw-r--r--internal/config/defaults.go6
-rw-r--r--internal/config/helpers.gen.go136
-rw-r--r--internal/db/account.go3
-rw-r--r--internal/db/bundb/account.go68
-rw-r--r--internal/db/bundb/application.go9
-rw-r--r--internal/db/bundb/conversation.go43
-rw-r--r--internal/db/bundb/emoji.go57
-rw-r--r--internal/db/bundb/filter.go9
-rw-r--r--internal/db/bundb/filterkeyword.go8
-rw-r--r--internal/db/bundb/filterstatus.go9
-rw-r--r--internal/db/bundb/list.go569
-rw-r--r--internal/db/bundb/list_test.go145
-rw-r--r--internal/db/bundb/media.go75
-rw-r--r--internal/db/bundb/mention.go37
-rw-r--r--internal/db/bundb/move.go16
-rw-r--r--internal/db/bundb/notification.go15
-rw-r--r--internal/db/bundb/poll.go226
-rw-r--r--internal/db/bundb/poll_test.go36
-rw-r--r--internal/db/bundb/relationship_block.go136
-rw-r--r--internal/db/bundb/relationship_follow.go214
-rw-r--r--internal/db/bundb/relationship_follow_req.go190
-rw-r--r--internal/db/bundb/relationship_mute.go95
-rw-r--r--internal/db/bundb/relationship_test.go12
-rw-r--r--internal/db/bundb/report.go49
-rw-r--r--internal/db/bundb/report_test.go4
-rw-r--r--internal/db/bundb/sinbinstatus.go19
-rw-r--r--internal/db/bundb/status.go62
-rw-r--r--internal/db/bundb/statusbookmark.go75
-rw-r--r--internal/db/bundb/statusfave.go9
-rw-r--r--internal/db/bundb/tag.go80
-rw-r--r--internal/db/bundb/timeline.go34
-rw-r--r--internal/db/bundb/timeline_test.go4
-rw-r--r--internal/db/bundb/tombstone.go6
-rw-r--r--internal/db/bundb/user.go36
-rw-r--r--internal/db/list.go53
-rw-r--r--internal/db/poll.go6
-rw-r--r--internal/db/relationship.go3
-rw-r--r--internal/db/report.go2
-rw-r--r--internal/federation/dereferencing/status.go3
-rw-r--r--internal/gtsmodel/list.go1
-rw-r--r--internal/processing/account/export.go2
-rw-r--r--internal/processing/account/lists.go43
-rw-r--r--internal/processing/admin/report.go4
-rw-r--r--internal/processing/common/status.go2
-rw-r--r--internal/processing/list/get.go138
-rw-r--r--internal/processing/list/updateentries.go177
-rw-r--r--internal/processing/list/util.go46
-rw-r--r--internal/processing/workers/fromclientapi_test.go3
-rw-r--r--internal/processing/workers/surfacetimeline.go343
-rw-r--r--internal/processing/workers/util.go5
-rw-r--r--internal/typeutils/csv.go71
-rw-r--r--internal/util/unique.go30
-rwxr-xr-xtest/envparsing.sh6
-rw-r--r--vendor/codeberg.org/gruf/go-structr/cache.go5
-rw-r--r--vendor/codeberg.org/gruf/go-structr/item.go1
-rw-r--r--vendor/codeberg.org/gruf/go-structr/runtime.go2
-rw-r--r--vendor/modules.txt2
68 files changed, 1663 insertions, 2115 deletions
diff --git a/go.mod b/go.mod
index a9a4afa84..58a93278e 100644
--- a/go.mod
+++ b/go.mod
@@ -23,7 +23,7 @@ require (
codeberg.org/gruf/go-runners v1.6.2
codeberg.org/gruf/go-sched v1.2.3
codeberg.org/gruf/go-storage v0.1.2
- codeberg.org/gruf/go-structr v0.8.8
+ codeberg.org/gruf/go-structr v0.8.9
codeberg.org/superseriousbusiness/exif-terminator v0.9.0
github.com/DmitriyVTitov/size v1.5.0
github.com/KimMachineGun/automemlimit v0.6.1
diff --git a/go.sum b/go.sum
index 8940bb3b2..2d4608b0f 100644
--- a/go.sum
+++ b/go.sum
@@ -80,8 +80,8 @@ codeberg.org/gruf/go-sched v1.2.3 h1:H5ViDxxzOBR3uIyGBCf0eH8b1L8wMybOXcdtUUTXZHk
codeberg.org/gruf/go-sched v1.2.3/go.mod h1:vT9uB6KWFIIwnG9vcPY2a0alYNoqdL1mSzRM8I+PK7A=
codeberg.org/gruf/go-storage v0.1.2 h1:dIOVOKq1CJpRmuhbB8Zok3mmo8V6VV/nX5GLIm6hywA=
codeberg.org/gruf/go-storage v0.1.2/go.mod h1:LRDpFHqRJi0f+35c3ltBH2e/pGfwY5dGlNlgCJ/R1DA=
-codeberg.org/gruf/go-structr v0.8.8 h1:lRPpyTmLKvQCkkQiSUbOAh6jtL2wncEO8DwksMqQXM8=
-codeberg.org/gruf/go-structr v0.8.8/go.mod h1:zkoXVrAnKosh8VFAsbP/Hhs8FmLBjbVVy5w/Ngm8ApM=
+codeberg.org/gruf/go-structr v0.8.9 h1:OyiSspWYCeJOm356fFPd+bDRumPrard2VAUXAPqZiJ0=
+codeberg.org/gruf/go-structr v0.8.9/go.mod h1:zkoXVrAnKosh8VFAsbP/Hhs8FmLBjbVVy5w/Ngm8ApM=
codeberg.org/superseriousbusiness/exif-terminator v0.9.0 h1:/EfyGI6HIrbkhFwgXGSjZ9o1kr/+k8v4mKdfXTH02Go=
codeberg.org/superseriousbusiness/exif-terminator v0.9.0/go.mod h1:gCWKduudUWFzsnixoMzu0FYVdxHWG+AbXnZ50DqxsUE=
dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU=
diff --git a/internal/api/client/exports/exports_test.go b/internal/api/client/exports/exports_test.go
index 1943f2582..303361b86 100644
--- a/internal/api/client/exports/exports_test.go
+++ b/internal/api/client/exports/exports_test.go
@@ -188,8 +188,8 @@ admin@localhost:8080
token: suite.testTokens["local_account_1"],
user: suite.testUsers["local_account_1"],
account: suite.testAccounts["local_account_1"],
- expect: `Cool Ass Posters From This Instance,admin@localhost:8080
-Cool Ass Posters From This Instance,1happyturtle@localhost:8080
+ expect: `Cool Ass Posters From This Instance,1happyturtle@localhost:8080
+Cool Ass Posters From This Instance,admin@localhost:8080
`,
},
// Export Mutes.
diff --git a/internal/api/client/lists/listaccounts.go b/internal/api/client/lists/listaccounts.go
index e1d340ebb..d609251f7 100644
--- a/internal/api/client/lists/listaccounts.go
+++ b/internal/api/client/lists/listaccounts.go
@@ -25,6 +25,7 @@ import (
apiutil "github.com/superseriousbusiness/gotosocial/internal/api/util"
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
"github.com/superseriousbusiness/gotosocial/internal/oauth"
+ "github.com/superseriousbusiness/gotosocial/internal/paging"
)
// ListAccountsGETHandler swagger:operation GET /api/v1/lists/{id}/accounts listAccounts
@@ -129,42 +130,27 @@ func (m *Module) ListAccountsGETHandler(c *gin.Context) {
targetListID := c.Param(IDKey)
if targetListID == "" {
- err := errors.New("no list id specified")
- apiutil.ErrorHandler(c, gtserror.NewErrorBadRequest(err, err.Error()), m.processor.InstanceGetV1)
- return
- }
-
- limit, errWithCode := apiutil.ParseLimit(c.Query(apiutil.LimitKey), 40, 80, 0)
- if errWithCode != nil {
+ const text = "no list id specified"
+ errWithCode := gtserror.NewErrorBadRequest(errors.New(text), text)
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
return
}
- var (
- ctx = c.Request.Context()
+ page, errWithCode := paging.ParseIDPage(c,
+ 1, // min limit
+ 80, // max limit
+ 0, // default = paging disabled
)
-
- if limit == 0 {
- // Return all accounts in the list without pagination.
- accounts, errWithCode := m.processor.List().GetAllListAccounts(ctx, authed.Account, targetListID)
- if errWithCode != nil {
- apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
- return
- }
-
- c.JSON(http.StatusOK, accounts)
+ if errWithCode != nil {
+ apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
return
}
- // Return subset of accounts in the list with pagination.
resp, errWithCode := m.processor.List().GetListAccounts(
- ctx,
+ c.Request.Context(),
authed.Account,
targetListID,
- c.Query(MaxIDKey),
- c.Query(SinceIDKey),
- c.Query(MinIDKey),
- limit,
+ page,
)
if errWithCode != nil {
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
diff --git a/internal/api/client/lists/listaccounts_test.go b/internal/api/client/lists/listaccounts_test.go
index bbd187f7d..e0a16e29f 100644
--- a/internal/api/client/lists/listaccounts_test.go
+++ b/internal/api/client/lists/listaccounts_test.go
@@ -19,7 +19,7 @@ package lists_test
import (
"encoding/json"
- "io/ioutil"
+ "io"
"net/http"
"net/http/httptest"
"strconv"
@@ -97,7 +97,7 @@ func (suite *ListAccountsTestSuite) getListAccounts(
result := recorder.Result()
defer result.Body.Close()
- b, err := ioutil.ReadAll(result.Body)
+ b, err := io.ReadAll(result.Body)
if err != nil {
return nil, "", err
}
@@ -151,8 +151,7 @@ func (suite *ListAccountsTestSuite) TestGetListAccountsPaginatedDefaultLimit() {
suite.Len(accounts, 2)
suite.Equal(
- `<http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=40&max_id=01H0G89MWVQE0M58VD2HQYMQWH>; rel="next", `+
- `<http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=40&min_id=01H0G8FFM1AGQDRNGBGGX8CYJQ>; rel="prev"`,
+ "<http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=40&max_id=01F8MH5NBDF2MV7CTC4Q5128HF>; rel=\"next\", <http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=40&min_id=01F8MH17FWEB39HZJ76B6VXSKF>; rel=\"prev\"",
link,
)
}
@@ -184,8 +183,7 @@ func (suite *ListAccountsTestSuite) TestGetListAccountsPaginatedNextPage() {
suite.Len(accounts, 1)
suite.Equal(
- `<http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=1&max_id=01H0G8FFM1AGQDRNGBGGX8CYJQ>; rel="next", `+
- `<http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=1&min_id=01H0G8FFM1AGQDRNGBGGX8CYJQ>; rel="prev"`,
+ "<http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=1&max_id=01F8MH17FWEB39HZJ76B6VXSKF>; rel=\"next\", <http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=1&min_id=01F8MH17FWEB39HZJ76B6VXSKF>; rel=\"prev\"",
link,
)
@@ -206,8 +204,7 @@ func (suite *ListAccountsTestSuite) TestGetListAccountsPaginatedNextPage() {
suite.Len(accounts, 1)
suite.Equal(
- `<http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=1&max_id=01H0G89MWVQE0M58VD2HQYMQWH>; rel="next", `+
- `<http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=1&min_id=01H0G89MWVQE0M58VD2HQYMQWH>; rel="prev"`,
+ "<http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=1&max_id=01F8MH17FWEB39HZJ76B6VXSKF>; rel=\"next\", <http://localhost:8080/api/v1/lists/01H0G8E4Q2J3FE3JDWJVWEDCD1/accounts?limit=1&min_id=01F8MH17FWEB39HZJ76B6VXSKF>; rel=\"prev\"",
link,
)
}
diff --git a/internal/api/client/lists/listaccountsadd_test.go b/internal/api/client/lists/listaccountsadd_test.go
index 7e44eeed3..e71cf0992 100644
--- a/internal/api/client/lists/listaccountsadd_test.go
+++ b/internal/api/client/lists/listaccountsadd_test.go
@@ -98,14 +98,17 @@ func (suite *ListAccountsAddTestSuite) TestPostListAccountNotFollowed() {
resp, err := suite.postListAccounts(http.StatusNotFound, listID, accountIDs)
suite.NoError(err)
- suite.Equal(`{"error":"Not Found: you do not follow account 01F8MH5ZK5VRH73AKHQM6Y9VNX"}`, string(resp))
+ suite.Equal(`{"error":"Not Found: account 01F8MH5ZK5VRH73AKHQM6Y9VNX not currently followed"}`, string(resp))
}
func (suite *ListAccountsAddTestSuite) TestPostListAccountOK() {
+ entry := suite.testListEntries["local_account_1_list_1_entry_1"]
+
// Remove turtle from the list.
if err := suite.db.DeleteListEntry(
context.Background(),
- suite.testListEntries["local_account_1_list_1_entry_1"].ID,
+ entry.ListID,
+ entry.FollowID,
); err != nil {
suite.FailNow(err.Error())
}
diff --git a/internal/cache/cache.go b/internal/cache/cache.go
index 8291dec5a..09e505ff5 100644
--- a/internal/cache/cache.go
+++ b/internal/cache/cache.go
@@ -62,7 +62,6 @@ func (c *Caches) Init() {
log.Infof(nil, "init: %p", c)
c.initAccount()
- c.initAccountIDsFollowingTag()
c.initAccountNote()
c.initAccountSettings()
c.initAccountStats()
@@ -84,11 +83,13 @@ func (c *Caches) Init() {
c.initFollowIDs()
c.initFollowRequest()
c.initFollowRequestIDs()
+ c.initFollowingTagIDs()
c.initInReplyToIDs()
c.initInstance()
c.initInteractionRequest()
c.initList()
- c.initListEntry()
+ c.initListIDs()
+ c.initListedIDs()
c.initMarker()
c.initMedia()
c.initMention()
@@ -105,7 +106,6 @@ func (c *Caches) Init() {
c.initStatusFave()
c.initStatusFaveIDs()
c.initTag()
- c.initTagIDsFollowedByAccount()
c.initThreadMute()
c.initToken()
c.initTombstone()
@@ -148,7 +148,6 @@ func (c *Caches) Stop() {
// significant overhead to all cache writes.
func (c *Caches) Sweep(threshold float64) {
c.DB.Account.Trim(threshold)
- c.DB.AccountIDsFollowingTag.Trim(threshold)
c.DB.AccountNote.Trim(threshold)
c.DB.AccountSettings.Trim(threshold)
c.DB.AccountStats.Trim(threshold)
@@ -168,11 +167,13 @@ func (c *Caches) Sweep(threshold float64) {
c.DB.FollowIDs.Trim(threshold)
c.DB.FollowRequest.Trim(threshold)
c.DB.FollowRequestIDs.Trim(threshold)
+ c.DB.FollowingTagIDs.Trim(threshold)
c.DB.InReplyToIDs.Trim(threshold)
c.DB.Instance.Trim(threshold)
c.DB.InteractionRequest.Trim(threshold)
c.DB.List.Trim(threshold)
- c.DB.ListEntry.Trim(threshold)
+ c.DB.ListIDs.Trim(threshold)
+ c.DB.ListedIDs.Trim(threshold)
c.DB.Marker.Trim(threshold)
c.DB.Media.Trim(threshold)
c.DB.Mention.Trim(threshold)
@@ -189,7 +190,6 @@ func (c *Caches) Sweep(threshold float64) {
c.DB.StatusFave.Trim(threshold)
c.DB.StatusFaveIDs.Trim(threshold)
c.DB.Tag.Trim(threshold)
- c.DB.TagIDsFollowedByAccount.Trim(threshold)
c.DB.ThreadMute.Trim(threshold)
c.DB.Token.Trim(threshold)
c.DB.Tombstone.Trim(threshold)
diff --git a/internal/cache/db.go b/internal/cache/db.go
index 7f54ee8c5..fe9085613 100644
--- a/internal/cache/db.go
+++ b/internal/cache/db.go
@@ -29,9 +29,6 @@ type DBCaches struct {
// Account provides access to the gtsmodel Account database cache.
Account StructCache[*gtsmodel.Account]
- // AccountIDsFollowingTag caches account IDs following a given tag ID.
- AccountIDsFollowingTag SliceCache[string]
-
// AccountNote provides access to the gtsmodel Note database cache.
AccountNote StructCache[*gtsmodel.AccountNote]
@@ -88,10 +85,23 @@ type DBCaches struct {
// FollowIDs provides access to the follower / following IDs database cache.
// THIS CACHE IS KEYED AS THE FOLLOWING {prefix}{accountID} WHERE PREFIX IS:
- // - '>' for following IDs
- // - 'l>' for local following IDs
- // - '<' for follower IDs
- // - 'l<' for local follower IDs
+ //
+ // - '>{$accountID}' for following IDs
+ // e.g. FollowIDs.Load(">" + account.ID, func() {})
+ // which will load a slice of follows IDs FROM account.
+ //
+ // - 'l>{$accountID}' for local following IDs
+ // e.g. FollowIDs.Load("l>" + account.ID, func() {})
+ // which will load a slice of LOCAL follows IDs FROM account.
+ //
+ // - '<{$accountID}' for follower IDs
+ // e.g. FollowIDs.Load("<" + account.ID, func() {})
+ // which will load a slice of follows IDs TARGETTING account.
+ //
+ // - 'l<{$accountID}' for local follower IDs
+ // e.g. FollowIDs.Load("l<" + account.ID, func() {})
+ // which will load a slice of LOCAL follows IDs TARGETTING account.
+ //
FollowIDs SliceCache[string]
// FollowRequest provides access to the gtsmodel FollowRequest database cache.
@@ -99,10 +109,30 @@ type DBCaches struct {
// FollowRequestIDs provides access to the follow requester / requesting IDs database
// cache. THIS CACHE IS KEYED AS THE FOLLOWING {prefix}{accountID} WHERE PREFIX IS:
- // - '>' for following IDs
- // - '<' for follower IDs
+ //
+ // - '>{$accountID}' for follow request IDs
+ // e.g. FollowRequestIDs.Load(">" + account.ID, func() {})
+ // which will load a slice of follow request IDs TARGETTING account.
+ //
+ // - '<{$accountID}' for follow request IDs
+ // e.g. FollowRequestIDs.Load("<" + account.ID, func() {})
+ // which will load a slice of follow request IDs FROM account.
+ //
FollowRequestIDs SliceCache[string]
+ // FollowingTagIDs provides access to account IDs following / tag IDs followed by
+ // account db cache. THIS CACHE IS KEYED AS THE FOLLOWING {prefix}{id} WHERE:
+ //
+ // - '>{$accountID}' for tag IDs followed by account
+ // e.g. FollowingTagIDs.Load(">" + account.ID, func() {})
+ // which will load a slice of tag IDs followed by account.
+ //
+ // - '<{$tagIDs}' for account IDs following tag
+ // e.g. FollowingTagIDs.Load("<" + tag.ID, func() {})
+ // which will load a slice of account IDs following tag.
+ //
+ FollowingTagIDs SliceCache[string]
+
// Instance provides access to the gtsmodel Instance database cache.
Instance StructCache[*gtsmodel.Instance]
@@ -115,8 +145,31 @@ type DBCaches struct {
// List provides access to the gtsmodel List database cache.
List StructCache[*gtsmodel.List]
- // ListEntry provides access to the gtsmodel ListEntry database cache.
- ListEntry StructCache[*gtsmodel.ListEntry]
+ // ListIDs provides access to the list IDs owned by account / list IDs follow
+ // contained in db cache. THIS CACHE IS KEYED AS FOLLOWING {prefix}{id} WHERE:
+ //
+ // - 'a{$accountID}' for list IDs owned by account
+ // e.g. ListIDs.Load("a" + account.ID, func() {})
+ // which will load a slice of list IDs owned by account.
+ //
+ // - 'f{$followID}' for list IDs follow contained in
+ // e.g. ListIDs.Load("f" + follow.ID, func() {})
+ // which will load a slice of list IDs containing follow.
+ //
+ ListIDs SliceCache[string]
+
+ // ListedIDs provides access to the account IDs in list / follow IDs in
+ // list db cache. THIS CACHE IS KEYED AS FOLLOWING {prefix}{id} WHERE:
+ //
+ // - 'a{listID}' for account IDs in list ID
+ // e.g. ListedIDs.Load("a" + list.ID, func() {})
+ // which will load a slice of account IDs in list.
+ //
+ // - 'f{listID}' for follow IDs in list ID
+ // e.g. ListedIDs.Load("f" + list.ID, func() {})
+ // which will load a slice of follow IDs in list.
+ //
+ ListedIDs SliceCache[string]
// Marker provides access to the gtsmodel Marker database cache.
Marker StructCache[*gtsmodel.Marker]
@@ -151,10 +204,10 @@ type DBCaches struct {
// Status provides access to the gtsmodel Status database cache.
Status StructCache[*gtsmodel.Status]
- // StatusBookmark ...
+ // StatusBookmark provides access to the gtsmodel StatusBookmark database cache.
StatusBookmark StructCache[*gtsmodel.StatusBookmark]
- // StatusBookmarkIDs ...
+ // StatusBookmarkIDs provides access to the status bookmark IDs list database cache.
StatusBookmarkIDs SliceCache[string]
// StatusFave provides access to the gtsmodel StatusFave database cache.
@@ -166,9 +219,6 @@ type DBCaches struct {
// Tag provides access to the gtsmodel Tag database cache.
Tag StructCache[*gtsmodel.Tag]
- // TagIDsFollowedByAccount caches tag IDs followed by a given account ID.
- TagIDsFollowedByAccount SliceCache[string]
-
// ThreadMute provides access to the gtsmodel ThreadMute database cache.
ThreadMute StructCache[*gtsmodel.ThreadMute]
@@ -243,17 +293,6 @@ func (c *Caches) initAccount() {
})
}
-func (c *Caches) initAccountIDsFollowingTag() {
- // Calculate maximum cache size.
- cap := calculateSliceCacheMax(
- config.GetCacheAccountIDsFollowingTagMemRatio(),
- )
-
- log.Infof(nil, "cache size = %d", cap)
-
- c.DB.AccountIDsFollowingTag.Init(0, cap)
-}
-
func (c *Caches) initAccountNote() {
// Calculate maximum cache size.
cap := calculateResultCacheMax(
@@ -761,6 +800,17 @@ func (c *Caches) initFollowRequestIDs() {
c.DB.FollowRequestIDs.Init(0, cap)
}
+func (c *Caches) initFollowingTagIDs() {
+ // Calculate maximum cache size.
+ cap := calculateSliceCacheMax(
+ config.GetCacheFollowingTagIDsMemRatio(),
+ )
+
+ log.Infof(nil, "cache size = %d", cap)
+
+ c.DB.FollowingTagIDs.Init(0, cap)
+}
+
func (c *Caches) initInReplyToIDs() {
// Calculate maximum cache size.
cap := calculateSliceCacheMax(
@@ -860,7 +910,6 @@ func (c *Caches) initList() {
// will be populated separately.
// See internal/db/bundb/list.go.
l2.Account = nil
- l2.ListEntries = nil
return l2
}
@@ -876,37 +925,26 @@ func (c *Caches) initList() {
})
}
-func (c *Caches) initListEntry() {
+func (c *Caches) initListIDs() {
// Calculate maximum cache size.
- cap := calculateResultCacheMax(
- sizeofListEntry(), // model in-mem size.
- config.GetCacheListEntryMemRatio(),
+ cap := calculateSliceCacheMax(
+ config.GetCacheListIDsMemRatio(),
)
log.Infof(nil, "cache size = %d", cap)
- copyF := func(l1 *gtsmodel.ListEntry) *gtsmodel.ListEntry {
- l2 := new(gtsmodel.ListEntry)
- *l2 = *l1
+ c.DB.ListIDs.Init(0, cap)
+}
- // Don't include ptr fields that
- // will be populated separately.
- // See internal/db/bundb/list.go.
- l2.Follow = nil
+func (c *Caches) initListedIDs() {
+ // Calculate maximum cache size.
+ cap := calculateSliceCacheMax(
+ config.GetCacheListedIDsMemRatio(),
+ )
- return l2
- }
+ log.Infof(nil, "cache size = %d", cap)
- c.DB.ListEntry.Init(structr.CacheConfig[*gtsmodel.ListEntry]{
- Indices: []structr.IndexConfig{
- {Fields: "ID"},
- {Fields: "ListID", Multiple: true},
- {Fields: "FollowID", Multiple: true},
- },
- MaxSize: cap,
- IgnoreErr: ignoreErrors,
- Copy: copyF,
- })
+ c.DB.ListedIDs.Init(0, cap)
}
func (c *Caches) initMarker() {
@@ -1368,17 +1406,6 @@ func (c *Caches) initTag() {
})
}
-func (c *Caches) initTagIDsFollowedByAccount() {
- // Calculate maximum cache size.
- cap := calculateSliceCacheMax(
- config.GetCacheTagIDsFollowedByAccountMemRatio(),
- )
-
- log.Infof(nil, "cache size = %d", cap)
-
- c.DB.TagIDsFollowedByAccount.Init(0, cap)
-}
-
func (c *Caches) initThreadMute() {
cap := calculateResultCacheMax(
sizeofThreadMute(), // model in-mem size.
diff --git a/internal/cache/invalidate.go b/internal/cache/invalidate.go
index ac326eda3..ca12e412c 100644
--- a/internal/cache/invalidate.go
+++ b/internal/cache/invalidate.go
@@ -97,9 +97,6 @@ func (c *Caches) OnInvalidateFollow(follow *gtsmodel.Follow) {
// Invalidate follow request with this same ID.
c.DB.FollowRequest.Invalidate("ID", follow.ID)
- // Invalidate any related list entries.
- c.DB.ListEntry.Invalidate("FollowID", follow.ID)
-
// Invalidate follow origin account ID cached visibility.
c.Visibility.Invalidate("ItemID", follow.AccountID)
c.Visibility.Invalidate("RequesterID", follow.AccountID)
@@ -108,18 +105,47 @@ func (c *Caches) OnInvalidateFollow(follow *gtsmodel.Follow) {
c.Visibility.Invalidate("ItemID", follow.TargetAccountID)
c.Visibility.Invalidate("RequesterID", follow.TargetAccountID)
- // Invalidate source account's following
- // lists, and destination's follwer lists.
- // (see FollowIDs() comment for details).
+ // Invalidate ID slice cache.
c.DB.FollowIDs.Invalidate(
+
+ // Invalidate follow ID lists
+ // TARGETTING origin account
+ // (including local-only follows).
">"+follow.AccountID,
"l>"+follow.AccountID,
+
+ // Invalidate follow ID lists
+ // FROM the origin account
+ // (including local-only follows).
"<"+follow.AccountID,
"l<"+follow.AccountID,
- "<"+follow.TargetAccountID,
- "l<"+follow.TargetAccountID,
+
+ // Invalidate follow ID lists
+ // TARGETTING the target account
+ // (including local-only follows).
">"+follow.TargetAccountID,
"l>"+follow.TargetAccountID,
+
+ // Invalidate follow ID lists
+ // FROM the target account
+ // (including local-only follows).
+ "<"+follow.TargetAccountID,
+ "l<"+follow.TargetAccountID,
+ )
+
+ // Invalidate ID slice cache.
+ c.DB.ListIDs.Invalidate(
+
+ // Invalidate source
+ // account's owned lists.
+ "a"+follow.AccountID,
+
+ // Invalidate target account's.
+ "a"+follow.TargetAccountID,
+
+ // Invalidate lists containing
+ // list entries for follow.
+ "f"+follow.ID,
)
}
@@ -127,20 +153,48 @@ func (c *Caches) OnInvalidateFollowRequest(followReq *gtsmodel.FollowRequest) {
// Invalidate follow with this same ID.
c.DB.Follow.Invalidate("ID", followReq.ID)
- // Invalidate source account's followreq
- // lists, and destinations follow req lists.
- // (see FollowRequestIDs() comment for details).
+ // Invalidate ID slice cache.
c.DB.FollowRequestIDs.Invalidate(
+
+ // Invalidate follow request ID
+ // lists TARGETTING origin account
+ // (including local-only follows).
">"+followReq.AccountID,
+
+ // Invalidate follow request ID
+ // lists FROM the origin account
+ // (including local-only follows).
"<"+followReq.AccountID,
+
+ // Invalidate follow request ID
+ // lists TARGETTING target account
+ // (including local-only follows).
">"+followReq.TargetAccountID,
+
+ // Invalidate follow request ID
+ // lists FROM the target account
+ // (including local-only follows).
"<"+followReq.TargetAccountID,
)
}
func (c *Caches) OnInvalidateList(list *gtsmodel.List) {
- // Invalidate all cached entries of this list.
- c.DB.ListEntry.Invalidate("ListID", list.ID)
+ // Invalidate list IDs cache.
+ c.DB.ListIDs.Invalidate(
+ "a" + list.AccountID,
+ )
+
+ // Invalidate ID slice cache.
+ c.DB.ListedIDs.Invalidate(
+
+ // Invalidate list of
+ // account IDs in list.
+ "a"+list.ID,
+
+ // Invalidate list of
+ // follow IDs in list.
+ "f"+list.ID,
+ )
}
func (c *Caches) OnInvalidateMedia(media *gtsmodel.MediaAttachment) {
@@ -184,7 +238,7 @@ func (c *Caches) OnInvalidateStatus(status *gtsmodel.Status) {
// the media IDs in use before the media table is
// aware of the status ID they are linked to.
//
- // c.DB.Media().Invalidate("StatusID") will not work.
+ // c.DB.Media.Invalidate("StatusID") will not work.
c.DB.Media.InvalidateIDs("ID", status.AttachmentIDs)
if status.BoostOfID != "" {
diff --git a/internal/cache/size.go b/internal/cache/size.go
index 49c2f4318..8367e4c46 100644
--- a/internal/cache/size.go
+++ b/internal/cache/size.go
@@ -166,6 +166,7 @@ func calculateCacheMax(keySz, valSz uintptr, ratio float64) int {
// totalOfRatios returns the total of all cache ratios added together.
func totalOfRatios() float64 {
+
// NOTE: this is not performant calculating
// this every damn time (mainly the mutex unlocks
// required to access each config var). fortunately
@@ -189,11 +190,13 @@ func totalOfRatios() float64 {
config.GetCacheFollowIDsMemRatio() +
config.GetCacheFollowRequestMemRatio() +
config.GetCacheFollowRequestIDsMemRatio() +
+ config.GetCacheFollowingTagIDsMemRatio() +
+ config.GetCacheInReplyToIDsMemRatio() +
config.GetCacheInstanceMemRatio() +
config.GetCacheInteractionRequestMemRatio() +
- config.GetCacheInReplyToIDsMemRatio() +
config.GetCacheListMemRatio() +
- config.GetCacheListEntryMemRatio() +
+ config.GetCacheListIDsMemRatio() +
+ config.GetCacheListedIDsMemRatio() +
config.GetCacheMarkerMemRatio() +
config.GetCacheMediaMemRatio() +
config.GetCacheMentionMemRatio() +
@@ -201,7 +204,9 @@ func totalOfRatios() float64 {
config.GetCacheNotificationMemRatio() +
config.GetCachePollMemRatio() +
config.GetCachePollVoteMemRatio() +
+ config.GetCachePollVoteIDsMemRatio() +
config.GetCacheReportMemRatio() +
+ config.GetCacheSinBinStatusMemRatio() +
config.GetCacheStatusMemRatio() +
config.GetCacheStatusBookmarkMemRatio() +
config.GetCacheStatusBookmarkIDsMemRatio() +
@@ -212,6 +217,8 @@ func totalOfRatios() float64 {
config.GetCacheTokenMemRatio() +
config.GetCacheTombstoneMemRatio() +
config.GetCacheUserMemRatio() +
+ config.GetCacheUserMuteMemRatio() +
+ config.GetCacheUserMuteIDsMemRatio() +
config.GetCacheWebfingerMemRatio() +
config.GetCacheVisibilityMemRatio()
}
@@ -466,16 +473,6 @@ func sizeofList() uintptr {
}))
}
-func sizeofListEntry() uintptr {
- return uintptr(size.Of(&gtsmodel.ListEntry{
- ID: exampleID,
- CreatedAt: exampleTime,
- UpdatedAt: exampleTime,
- ListID: exampleID,
- FollowID: exampleID,
- }))
-}
-
func sizeofMarker() uintptr {
return uintptr(size.Of(&gtsmodel.Marker{
AccountID: exampleID,
diff --git a/internal/config/config.go b/internal/config/config.go
index e24cb639b..4a40e9c13 100644
--- a/internal/config/config.go
+++ b/internal/config/config.go
@@ -196,7 +196,6 @@ type HTTPClientConfiguration struct {
type CacheConfiguration struct {
MemoryTarget bytesize.Size `name:"memory-target"`
AccountMemRatio float64 `name:"account-mem-ratio"`
- AccountIDsFollowingTagMemRatio float64 `name:"account-ids-following-tag-mem-ratio"`
AccountNoteMemRatio float64 `name:"account-note-mem-ratio"`
AccountSettingsMemRatio float64 `name:"account-settings-mem-ratio"`
AccountStatsMemRatio float64 `name:"account-stats-mem-ratio"`
@@ -216,11 +215,13 @@ type CacheConfiguration struct {
FollowIDsMemRatio float64 `name:"follow-ids-mem-ratio"`
FollowRequestMemRatio float64 `name:"follow-request-mem-ratio"`
FollowRequestIDsMemRatio float64 `name:"follow-request-ids-mem-ratio"`
+ FollowingTagIDsMemRatio float64 `name:"following-tag-ids-mem-ratio"`
InReplyToIDsMemRatio float64 `name:"in-reply-to-ids-mem-ratio"`
InstanceMemRatio float64 `name:"instance-mem-ratio"`
InteractionRequestMemRatio float64 `name:"interaction-request-mem-ratio"`
ListMemRatio float64 `name:"list-mem-ratio"`
- ListEntryMemRatio float64 `name:"list-entry-mem-ratio"`
+ ListIDsMemRatio float64 `name:"list-ids-mem-ratio"`
+ ListedIDsMemRatio float64 `name:"listed-ids-mem-ratio"`
MarkerMemRatio float64 `name:"marker-mem-ratio"`
MediaMemRatio float64 `name:"media-mem-ratio"`
MentionMemRatio float64 `name:"mention-mem-ratio"`
@@ -237,7 +238,6 @@ type CacheConfiguration struct {
StatusFaveMemRatio float64 `name:"status-fave-mem-ratio"`
StatusFaveIDsMemRatio float64 `name:"status-fave-ids-mem-ratio"`
TagMemRatio float64 `name:"tag-mem-ratio"`
- TagIDsFollowedByAccountMemRatio float64 `name:"tag-ids-followed-by-account-mem-ratio"`
ThreadMuteMemRatio float64 `name:"thread-mute-mem-ratio"`
TokenMemRatio float64 `name:"token-mem-ratio"`
TombstoneMemRatio float64 `name:"tombstone-mem-ratio"`
diff --git a/internal/config/defaults.go b/internal/config/defaults.go
index 58e11a292..48d880e1b 100644
--- a/internal/config/defaults.go
+++ b/internal/config/defaults.go
@@ -159,7 +159,6 @@ var Defaults = Configuration{
// file have been addressed, these should
// be able to make some more sense :D
AccountMemRatio: 5,
- AccountIDsFollowingTagMemRatio: 1,
AccountNoteMemRatio: 1,
AccountSettingsMemRatio: 0.1,
AccountStatsMemRatio: 2,
@@ -179,11 +178,13 @@ var Defaults = Configuration{
FollowIDsMemRatio: 4,
FollowRequestMemRatio: 2,
FollowRequestIDsMemRatio: 2,
+ FollowingTagIDsMemRatio: 2,
InReplyToIDsMemRatio: 3,
InstanceMemRatio: 1,
InteractionRequestMemRatio: 1,
ListMemRatio: 1,
- ListEntryMemRatio: 2,
+ ListIDsMemRatio: 2,
+ ListedIDsMemRatio: 2,
MarkerMemRatio: 0.5,
MediaMemRatio: 4,
MentionMemRatio: 2,
@@ -200,7 +201,6 @@ var Defaults = Configuration{
StatusFaveMemRatio: 2,
StatusFaveIDsMemRatio: 3,
TagMemRatio: 2,
- TagIDsFollowedByAccountMemRatio: 1,
ThreadMuteMemRatio: 0.2,
TokenMemRatio: 0.75,
TombstoneMemRatio: 0.5,
diff --git a/internal/config/helpers.gen.go b/internal/config/helpers.gen.go
index 75231d37b..d25d6cca8 100644
--- a/internal/config/helpers.gen.go
+++ b/internal/config/helpers.gen.go
@@ -2850,37 +2850,6 @@ func GetCacheAccountMemRatio() float64 { return global.GetCacheAccountMemRatio()
// SetCacheAccountMemRatio safely sets the value for global configuration 'Cache.AccountMemRatio' field
func SetCacheAccountMemRatio(v float64) { global.SetCacheAccountMemRatio(v) }
-// GetCacheAccountIDsFollowingTagMemRatio safely fetches the Configuration value for state's 'Cache.AccountIDsFollowingTagMemRatio' field
-func (st *ConfigState) GetCacheAccountIDsFollowingTagMemRatio() (v float64) {
- st.mutex.RLock()
- v = st.config.Cache.AccountIDsFollowingTagMemRatio
- st.mutex.RUnlock()
- return
-}
-
-// SetCacheAccountIDsFollowingTagMemRatio safely sets the Configuration value for state's 'Cache.AccountIDsFollowingTagMemRatio' field
-func (st *ConfigState) SetCacheAccountIDsFollowingTagMemRatio(v float64) {
- st.mutex.Lock()
- defer st.mutex.Unlock()
- st.config.Cache.AccountIDsFollowingTagMemRatio = v
- st.reloadToViper()
-}
-
-// CacheAccountIDsFollowingTagMemRatioFlag returns the flag name for the 'Cache.AccountIDsFollowingTagMemRatio' field
-func CacheAccountIDsFollowingTagMemRatioFlag() string {
- return "cache-account-ids-following-tag-mem-ratio"
-}
-
-// GetCacheAccountIDsFollowingTagMemRatio safely fetches the value for global configuration 'Cache.AccountIDsFollowingTagMemRatio' field
-func GetCacheAccountIDsFollowingTagMemRatio() float64 {
- return global.GetCacheAccountIDsFollowingTagMemRatio()
-}
-
-// SetCacheAccountIDsFollowingTagMemRatio safely sets the value for global configuration 'Cache.AccountIDsFollowingTagMemRatio' field
-func SetCacheAccountIDsFollowingTagMemRatio(v float64) {
- global.SetCacheAccountIDsFollowingTagMemRatio(v)
-}
-
// GetCacheAccountNoteMemRatio safely fetches the Configuration value for state's 'Cache.AccountNoteMemRatio' field
func (st *ConfigState) GetCacheAccountNoteMemRatio() (v float64) {
st.mutex.RLock()
@@ -3362,6 +3331,31 @@ func GetCacheFollowRequestIDsMemRatio() float64 { return global.GetCacheFollowRe
// SetCacheFollowRequestIDsMemRatio safely sets the value for global configuration 'Cache.FollowRequestIDsMemRatio' field
func SetCacheFollowRequestIDsMemRatio(v float64) { global.SetCacheFollowRequestIDsMemRatio(v) }
+// GetCacheFollowingTagIDsMemRatio safely fetches the Configuration value for state's 'Cache.FollowingTagIDsMemRatio' field
+func (st *ConfigState) GetCacheFollowingTagIDsMemRatio() (v float64) {
+ st.mutex.RLock()
+ v = st.config.Cache.FollowingTagIDsMemRatio
+ st.mutex.RUnlock()
+ return
+}
+
+// SetCacheFollowingTagIDsMemRatio safely sets the Configuration value for state's 'Cache.FollowingTagIDsMemRatio' field
+func (st *ConfigState) SetCacheFollowingTagIDsMemRatio(v float64) {
+ st.mutex.Lock()
+ defer st.mutex.Unlock()
+ st.config.Cache.FollowingTagIDsMemRatio = v
+ st.reloadToViper()
+}
+
+// CacheFollowingTagIDsMemRatioFlag returns the flag name for the 'Cache.FollowingTagIDsMemRatio' field
+func CacheFollowingTagIDsMemRatioFlag() string { return "cache-following-tag-ids-mem-ratio" }
+
+// GetCacheFollowingTagIDsMemRatio safely fetches the value for global configuration 'Cache.FollowingTagIDsMemRatio' field
+func GetCacheFollowingTagIDsMemRatio() float64 { return global.GetCacheFollowingTagIDsMemRatio() }
+
+// SetCacheFollowingTagIDsMemRatio safely sets the value for global configuration 'Cache.FollowingTagIDsMemRatio' field
+func SetCacheFollowingTagIDsMemRatio(v float64) { global.SetCacheFollowingTagIDsMemRatio(v) }
+
// GetCacheInReplyToIDsMemRatio safely fetches the Configuration value for state's 'Cache.InReplyToIDsMemRatio' field
func (st *ConfigState) GetCacheInReplyToIDsMemRatio() (v float64) {
st.mutex.RLock()
@@ -3462,30 +3456,55 @@ func GetCacheListMemRatio() float64 { return global.GetCacheListMemRatio() }
// SetCacheListMemRatio safely sets the value for global configuration 'Cache.ListMemRatio' field
func SetCacheListMemRatio(v float64) { global.SetCacheListMemRatio(v) }
-// GetCacheListEntryMemRatio safely fetches the Configuration value for state's 'Cache.ListEntryMemRatio' field
-func (st *ConfigState) GetCacheListEntryMemRatio() (v float64) {
+// GetCacheListIDsMemRatio safely fetches the Configuration value for state's 'Cache.ListIDsMemRatio' field
+func (st *ConfigState) GetCacheListIDsMemRatio() (v float64) {
st.mutex.RLock()
- v = st.config.Cache.ListEntryMemRatio
+ v = st.config.Cache.ListIDsMemRatio
st.mutex.RUnlock()
return
}
-// SetCacheListEntryMemRatio safely sets the Configuration value for state's 'Cache.ListEntryMemRatio' field
-func (st *ConfigState) SetCacheListEntryMemRatio(v float64) {
+// SetCacheListIDsMemRatio safely sets the Configuration value for state's 'Cache.ListIDsMemRatio' field
+func (st *ConfigState) SetCacheListIDsMemRatio(v float64) {
st.mutex.Lock()
defer st.mutex.Unlock()
- st.config.Cache.ListEntryMemRatio = v
+ st.config.Cache.ListIDsMemRatio = v
st.reloadToViper()
}
-// CacheListEntryMemRatioFlag returns the flag name for the 'Cache.ListEntryMemRatio' field
-func CacheListEntryMemRatioFlag() string { return "cache-list-entry-mem-ratio" }
+// CacheListIDsMemRatioFlag returns the flag name for the 'Cache.ListIDsMemRatio' field
+func CacheListIDsMemRatioFlag() string { return "cache-list-ids-mem-ratio" }
-// GetCacheListEntryMemRatio safely fetches the value for global configuration 'Cache.ListEntryMemRatio' field
-func GetCacheListEntryMemRatio() float64 { return global.GetCacheListEntryMemRatio() }
+// GetCacheListIDsMemRatio safely fetches the value for global configuration 'Cache.ListIDsMemRatio' field
+func GetCacheListIDsMemRatio() float64 { return global.GetCacheListIDsMemRatio() }
-// SetCacheListEntryMemRatio safely sets the value for global configuration 'Cache.ListEntryMemRatio' field
-func SetCacheListEntryMemRatio(v float64) { global.SetCacheListEntryMemRatio(v) }
+// SetCacheListIDsMemRatio safely sets the value for global configuration 'Cache.ListIDsMemRatio' field
+func SetCacheListIDsMemRatio(v float64) { global.SetCacheListIDsMemRatio(v) }
+
+// GetCacheListedIDsMemRatio safely fetches the Configuration value for state's 'Cache.ListedIDsMemRatio' field
+func (st *ConfigState) GetCacheListedIDsMemRatio() (v float64) {
+ st.mutex.RLock()
+ v = st.config.Cache.ListedIDsMemRatio
+ st.mutex.RUnlock()
+ return
+}
+
+// SetCacheListedIDsMemRatio safely sets the Configuration value for state's 'Cache.ListedIDsMemRatio' field
+func (st *ConfigState) SetCacheListedIDsMemRatio(v float64) {
+ st.mutex.Lock()
+ defer st.mutex.Unlock()
+ st.config.Cache.ListedIDsMemRatio = v
+ st.reloadToViper()
+}
+
+// CacheListedIDsMemRatioFlag returns the flag name for the 'Cache.ListedIDsMemRatio' field
+func CacheListedIDsMemRatioFlag() string { return "cache-listed-ids-mem-ratio" }
+
+// GetCacheListedIDsMemRatio safely fetches the value for global configuration 'Cache.ListedIDsMemRatio' field
+func GetCacheListedIDsMemRatio() float64 { return global.GetCacheListedIDsMemRatio() }
+
+// SetCacheListedIDsMemRatio safely sets the value for global configuration 'Cache.ListedIDsMemRatio' field
+func SetCacheListedIDsMemRatio(v float64) { global.SetCacheListedIDsMemRatio(v) }
// GetCacheMarkerMemRatio safely fetches the Configuration value for state's 'Cache.MarkerMemRatio' field
func (st *ConfigState) GetCacheMarkerMemRatio() (v float64) {
@@ -3887,37 +3906,6 @@ func GetCacheTagMemRatio() float64 { return global.GetCacheTagMemRatio() }
// SetCacheTagMemRatio safely sets the value for global configuration 'Cache.TagMemRatio' field
func SetCacheTagMemRatio(v float64) { global.SetCacheTagMemRatio(v) }
-// GetCacheTagIDsFollowedByAccountMemRatio safely fetches the Configuration value for state's 'Cache.TagIDsFollowedByAccountMemRatio' field
-func (st *ConfigState) GetCacheTagIDsFollowedByAccountMemRatio() (v float64) {
- st.mutex.RLock()
- v = st.config.Cache.TagIDsFollowedByAccountMemRatio
- st.mutex.RUnlock()
- return
-}
-
-// SetCacheTagIDsFollowedByAccountMemRatio safely sets the Configuration value for state's 'Cache.TagIDsFollowedByAccountMemRatio' field
-func (st *ConfigState) SetCacheTagIDsFollowedByAccountMemRatio(v float64) {
- st.mutex.Lock()
- defer st.mutex.Unlock()
- st.config.Cache.TagIDsFollowedByAccountMemRatio = v
- st.reloadToViper()
-}
-
-// CacheTagIDsFollowedByAccountMemRatioFlag returns the flag name for the 'Cache.TagIDsFollowedByAccountMemRatio' field
-func CacheTagIDsFollowedByAccountMemRatioFlag() string {
- return "cache-tag-ids-followed-by-account-mem-ratio"
-}
-
-// GetCacheTagIDsFollowedByAccountMemRatio safely fetches the value for global configuration 'Cache.TagIDsFollowedByAccountMemRatio' field
-func GetCacheTagIDsFollowedByAccountMemRatio() float64 {
- return global.GetCacheTagIDsFollowedByAccountMemRatio()
-}
-
-// SetCacheTagIDsFollowedByAccountMemRatio safely sets the value for global configuration 'Cache.TagIDsFollowedByAccountMemRatio' field
-func SetCacheTagIDsFollowedByAccountMemRatio(v float64) {
- global.SetCacheTagIDsFollowedByAccountMemRatio(v)
-}
-
// GetCacheThreadMuteMemRatio safely fetches the Configuration value for state's 'Cache.ThreadMuteMemRatio' field
func (st *ConfigState) GetCacheThreadMuteMemRatio() (v float64) {
st.mutex.RLock()
diff --git a/internal/db/account.go b/internal/db/account.go
index 225c8e1d2..aa0dfd985 100644
--- a/internal/db/account.go
+++ b/internal/db/account.go
@@ -123,9 +123,6 @@ type Account interface {
// In the case of no statuses, this function will return db.ErrNoEntries.
GetAccountWebStatuses(ctx context.Context, account *gtsmodel.Account, limit int, maxID string) ([]*gtsmodel.Status, error)
- // SetAccountHeaderOrAvatar sets the header or avatar for the given accountID to the given media attachment.
- SetAccountHeaderOrAvatar(ctx context.Context, mediaAttachment *gtsmodel.MediaAttachment, accountID string) error
-
// GetInstanceAccount returns the instance account for the given domain.
// If domain is empty, this instance account will be returned.
GetInstanceAccount(ctx context.Context, domain string) (*gtsmodel.Account, error)
diff --git a/internal/db/bundb/account.go b/internal/db/bundb/account.go
index 1569af9cb..16c82c08f 100644
--- a/internal/db/bundb/account.go
+++ b/internal/db/bundb/account.go
@@ -64,15 +64,8 @@ func (a *accountDB) GetAccountsByIDs(ctx context.Context, ids []string) ([]*gtsm
accounts, err := a.state.Caches.DB.Account.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.Account, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached accounts.
- accounts := make([]*gtsmodel.Account, 0, count)
+ accounts := make([]*gtsmodel.Account, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) account IDs.
@@ -796,20 +789,14 @@ func (a *accountDB) UpdateAccount(ctx context.Context, account *gtsmodel.Account
}
func (a *accountDB) DeleteAccount(ctx context.Context, id string) error {
- defer a.state.Caches.DB.Account.Invalidate("ID", id)
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.Account
+ deleted.ID = id
- // Load account into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := a.GetAccountByID(gtscontext.SetBarebones(ctx), id)
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- // NOTE: even if db.ErrNoEntries is returned, we
- // still run the below transaction to ensure related
- // objects are appropriately deleted.
- return err
- }
+ // Delete account from database and any related links in a transaction.
+ if err := a.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
- return a.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
// clear out any emoji links
if _, err := tx.
NewDelete().
@@ -822,44 +809,19 @@ func (a *accountDB) DeleteAccount(ctx context.Context, id string) error {
// delete the account
_, err := tx.
NewDelete().
- TableExpr("? AS ?", bun.Ident("accounts"), bun.Ident("account")).
- Where("? = ?", bun.Ident("account.id"), id).
+ Model(&deleted).
+ Where("? = ?", bun.Ident("id"), id).
+ Returning("?", bun.Ident("uri")).
Exec(ctx)
return err
- })
-}
-
-func (a *accountDB) SetAccountHeaderOrAvatar(ctx context.Context, mediaAttachment *gtsmodel.MediaAttachment, accountID string) error {
- if *mediaAttachment.Avatar && *mediaAttachment.Header {
- return errors.New("one media attachment cannot be both header and avatar")
- }
-
- var column bun.Ident
- switch {
- case *mediaAttachment.Avatar:
- column = bun.Ident("account.avatar_media_attachment_id")
- case *mediaAttachment.Header:
- column = bun.Ident("account.header_media_attachment_id")
- default:
- return errors.New("given media attachment was neither a header nor an avatar")
- }
-
- // TODO: there are probably more side effects here that need to be handled
- if _, err := a.db.
- NewInsert().
- Model(mediaAttachment).
- Exec(ctx); err != nil {
+ }); err != nil {
return err
}
- if _, err := a.db.
- NewUpdate().
- TableExpr("? AS ?", bun.Ident("accounts"), bun.Ident("account")).
- Set("? = ?", column, mediaAttachment.ID).
- Where("? = ?", bun.Ident("account.id"), accountID).
- Exec(ctx); err != nil {
- return err
- }
+ // Invalidate cached account by its ID, manually
+ // call invalidate hook in case not cached.
+ a.state.Caches.DB.Account.Invalidate("ID", id)
+ a.state.Caches.OnInvalidateAccount(&deleted)
return nil
}
diff --git a/internal/db/bundb/application.go b/internal/db/bundb/application.go
index 72c4ec206..fda0ba602 100644
--- a/internal/db/bundb/application.go
+++ b/internal/db/bundb/application.go
@@ -147,15 +147,8 @@ func (a *applicationDB) GetAllTokens(ctx context.Context) ([]*gtsmodel.Token, er
tokens, err := a.state.Caches.DB.Token.LoadIDs("ID",
tokenIDs,
func(uncached []string) ([]*gtsmodel.Token, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached tokens.
- tokens := make([]*gtsmodel.Token, 0, count)
+ tokens := make([]*gtsmodel.Token, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) token IDs.
diff --git a/internal/db/bundb/conversation.go b/internal/db/bundb/conversation.go
index 053b23e31..22ff4fd79 100644
--- a/internal/db/bundb/conversation.go
+++ b/internal/db/bundb/conversation.go
@@ -188,15 +188,8 @@ func (c *conversationDB) getConversationsByLastStatusIDs(
accountID,
conversationLastStatusIDs,
func(accountID string, uncached []string) ([]*gtsmodel.Conversation, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached conversations.
- conversations := make([]*gtsmodel.Conversation, 0, count)
+ conversations := make([]*gtsmodel.Conversation, 0, len(uncached))
// Perform database query scanning the remaining (uncached) IDs.
if err := c.db.NewSelect().
@@ -267,27 +260,27 @@ func (c *conversationDB) LinkConversationToStatus(ctx context.Context, conversat
}
func (c *conversationDB) DeleteConversationByID(ctx context.Context, id string) error {
- // Load conversation into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := c.GetConversationByID(gtscontext.SetBarebones(ctx), id)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // not an issue.
- err = nil
- }
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.Conversation
+ deleted.ID = id
+
+ // Delete conversation from DB.
+ if _, err := c.db.NewDelete().
+ Model(&deleted).
+ Where("? = ?", bun.Ident("id"), id).
+ Returning("?", bun.Ident("account_id")).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Drop this now-cached conversation on return after delete.
- defer c.state.Caches.DB.Conversation.Invalidate("ID", id)
+ // Invalidate cached conversation by ID,
+ // manually invalidate hook in case not cached.
+ c.state.Caches.DB.Conversation.Invalidate("ID", id)
+ c.state.Caches.OnInvalidateConversation(&deleted)
- // Finally delete conversation from DB.
- _, err = c.db.NewDelete().
- Model((*gtsmodel.Conversation)(nil)).
- Where("? = ?", bun.Ident("id"), id).
- Exec(ctx)
- return err
+ return nil
}
func (c *conversationDB) DeleteConversationsByOwnerAccountID(ctx context.Context, accountID string) error {
diff --git a/internal/db/bundb/emoji.go b/internal/db/bundb/emoji.go
index 6e4b5f36b..db9daf0aa 100644
--- a/internal/db/bundb/emoji.go
+++ b/internal/db/bundb/emoji.go
@@ -20,7 +20,6 @@ package bundb
import (
"context"
"database/sql"
- "errors"
"slices"
"strings"
"time"
@@ -70,34 +69,15 @@ func (e *emojiDB) UpdateEmoji(ctx context.Context, emoji *gtsmodel.Emoji, column
func (e *emojiDB) DeleteEmojiByID(ctx context.Context, id string) error {
var (
+ // Gather necessary fields from
+ // deleted for cache invaliation.
accountIDs []string
statusIDs []string
)
- defer func() {
- // Invalidate cached emoji.
- e.state.Caches.DB.Emoji.Invalidate("ID", id)
+ // Delete the emoji and all related links to it in a singular transaction.
+ if err := e.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
- // Invalidate cached account and status IDs.
- e.state.Caches.DB.Account.InvalidateIDs("ID", accountIDs)
- e.state.Caches.DB.Status.InvalidateIDs("ID", statusIDs)
- }()
-
- // Load emoji into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := e.GetEmojiByID(
- gtscontext.SetBarebones(ctx),
- id,
- )
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- // NOTE: even if db.ErrNoEntries is returned, we
- // still run the below transaction to ensure related
- // objects are appropriately deleted.
- return err
- }
-
- return e.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
// Delete relational links between this emoji
// and any statuses using it, returning the
// status IDs so we can later update them.
@@ -195,7 +175,16 @@ func (e *emojiDB) DeleteEmojiByID(ctx context.Context, id string) error {
}
return nil
- })
+ }); err != nil {
+ return err
+ }
+
+ // Invalidate emoji, and any effected statuses / accounts.
+ e.state.Caches.DB.Emoji.Invalidate("ID", id)
+ e.state.Caches.DB.Account.InvalidateIDs("ID", accountIDs)
+ e.state.Caches.DB.Status.InvalidateIDs("ID", statusIDs)
+
+ return nil
}
func (e *emojiDB) GetEmojisBy(ctx context.Context, domain string, includeDisabled bool, includeEnabled bool, shortcode string, maxShortcodeDomain string, minShortcodeDomain string, limit int) ([]*gtsmodel.Emoji, error) {
@@ -586,15 +575,8 @@ func (e *emojiDB) GetEmojisByIDs(ctx context.Context, ids []string) ([]*gtsmodel
emojis, err := e.state.Caches.DB.Emoji.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.Emoji, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached emojis.
- emojis := make([]*gtsmodel.Emoji, 0, count)
+ emojis := make([]*gtsmodel.Emoji, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
@@ -657,15 +639,8 @@ func (e *emojiDB) GetEmojiCategoriesByIDs(ctx context.Context, ids []string) ([]
categories, err := e.state.Caches.DB.EmojiCategory.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.EmojiCategory, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached categories.
- categories := make([]*gtsmodel.EmojiCategory, 0, count)
+ categories := make([]*gtsmodel.EmojiCategory, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
diff --git a/internal/db/bundb/filter.go b/internal/db/bundb/filter.go
index b84091cd6..e68a0bcd0 100644
--- a/internal/db/bundb/filter.go
+++ b/internal/db/bundb/filter.go
@@ -83,14 +83,7 @@ func (f *filterDB) GetFiltersForAccountID(ctx context.Context, accountID string)
filters, err := f.state.Caches.DB.Filter.LoadIDs("ID",
filterIDs,
func(uncached []string) ([]*gtsmodel.Filter, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
- filters := make([]*gtsmodel.Filter, 0, count)
+ filters := make([]*gtsmodel.Filter, 0, len(uncached))
if err := f.db.
NewSelect().
Model(&filters).
diff --git a/internal/db/bundb/filterkeyword.go b/internal/db/bundb/filterkeyword.go
index cb9958b81..8a006d10f 100644
--- a/internal/db/bundb/filterkeyword.go
+++ b/internal/db/bundb/filterkeyword.go
@@ -113,14 +113,8 @@ func (f *filterDB) getFilterKeywords(ctx context.Context, idColumn string, id st
filterKeywords, err := f.state.Caches.DB.FilterKeyword.LoadIDs("ID",
filterKeywordIDs,
func(uncached []string) ([]*gtsmodel.FilterKeyword, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
+ filterKeywords := make([]*gtsmodel.FilterKeyword, 0, len(uncached))
- filterKeywords := make([]*gtsmodel.FilterKeyword, 0, count)
if err := f.db.
NewSelect().
Model(&filterKeywords).
diff --git a/internal/db/bundb/filterstatus.go b/internal/db/bundb/filterstatus.go
index 8256cd401..95919bd2c 100644
--- a/internal/db/bundb/filterstatus.go
+++ b/internal/db/bundb/filterstatus.go
@@ -100,14 +100,7 @@ func (f *filterDB) getFilterStatuses(ctx context.Context, idColumn string, id st
filterStatuses, err := f.state.Caches.DB.FilterStatus.LoadIDs("ID",
filterStatusIDs,
func(uncached []string) ([]*gtsmodel.FilterStatus, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
- filterStatuses := make([]*gtsmodel.FilterStatus, 0, count)
+ filterStatuses := make([]*gtsmodel.FilterStatus, 0, len(uncached))
if err := f.db.
NewSelect().
Model(&filterStatuses).
diff --git a/internal/db/bundb/list.go b/internal/db/bundb/list.go
index 0ed0f1b15..03dff95e3 100644
--- a/internal/db/bundb/list.go
+++ b/internal/db/bundb/list.go
@@ -29,6 +29,7 @@ import (
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
"github.com/superseriousbusiness/gotosocial/internal/log"
+ "github.com/superseriousbusiness/gotosocial/internal/paging"
"github.com/superseriousbusiness/gotosocial/internal/state"
"github.com/superseriousbusiness/gotosocial/internal/util"
"github.com/uptrace/bun"
@@ -85,39 +86,52 @@ func (l *listDB) getList(ctx context.Context, lookup string, dbQuery func(*gtsmo
return list, nil
}
-func (l *listDB) GetListsForAccountID(ctx context.Context, accountID string) ([]*gtsmodel.List, error) {
- // Fetch IDs of all lists owned by this account.
- var listIDs []string
- if err := l.db.
- NewSelect().
- TableExpr("? AS ?", bun.Ident("lists"), bun.Ident("list")).
- Column("list.id").
- Where("? = ?", bun.Ident("list.account_id"), accountID).
- Order("list.id DESC").
- Scan(ctx, &listIDs); err != nil {
+func (l *listDB) GetListsByAccountID(ctx context.Context, accountID string) ([]*gtsmodel.List, error) {
+ listIDs, err := l.getListIDsByAccountID(ctx, accountID)
+ if err != nil {
return nil, err
}
+ return l.GetListsByIDs(ctx, listIDs)
+}
- if len(listIDs) == 0 {
- return nil, nil
- }
+func (l *listDB) CountListsByAccountID(ctx context.Context, accountID string) (int, error) {
+ listIDs, err := l.getListIDsByAccountID(ctx, accountID)
+ return len(listIDs), err
+}
- // Return lists by their IDs.
+func (l *listDB) GetListsContainingFollowID(ctx context.Context, followID string) ([]*gtsmodel.List, error) {
+ listIDs, err := l.getListIDsWithFollowID(ctx, followID)
+ if err != nil {
+ return nil, err
+ }
return l.GetListsByIDs(ctx, listIDs)
}
-func (l *listDB) CountListsForAccountID(ctx context.Context, accountID string) (int, error) {
- return l.db.
- NewSelect().
- Table("lists").
- Where("? = ?", bun.Ident("account_id"), accountID).
- Count(ctx)
+func (l *listDB) GetFollowsInList(ctx context.Context, listID string, page *paging.Page) ([]*gtsmodel.Follow, error) {
+ followIDs, err := l.GetFollowIDsInList(ctx, listID, page)
+ if err != nil {
+ return nil, err
+ }
+ return l.state.DB.GetFollowsByIDs(ctx, followIDs)
+}
+
+func (l *listDB) GetAccountsInList(ctx context.Context, listID string, page *paging.Page) ([]*gtsmodel.Account, error) {
+ accountIDs, err := l.GetAccountIDsInList(ctx, listID, page)
+ if err != nil {
+ return nil, err
+ }
+ return l.state.DB.GetAccountsByIDs(ctx, accountIDs)
+}
+
+func (l *listDB) IsAccountInList(ctx context.Context, listID string, accountID string) (bool, error) {
+ accountIDs, err := l.GetAccountIDsInList(ctx, listID, nil)
+ return slices.Contains(accountIDs, accountID), err
}
func (l *listDB) PopulateList(ctx context.Context, list *gtsmodel.List) error {
var (
err error
- errs = gtserror.NewMultiError(2)
+ errs gtserror.MultiError
)
if list.Account == nil {
@@ -131,22 +145,12 @@ func (l *listDB) PopulateList(ctx context.Context, list *gtsmodel.List) error {
}
}
- if list.ListEntries == nil {
- // List entries are not set, fetch from the database.
- list.ListEntries, err = l.state.DB.GetListEntries(
- gtscontext.SetBarebones(ctx),
- list.ID,
- "", "", "", 0,
- )
- if err != nil {
- errs.Appendf("error populating list entries: %w", err)
- }
- }
-
return errs.Combine()
}
func (l *listDB) PutList(ctx context.Context, list *gtsmodel.List) error {
+ // note that inserting list will call OnInvalidateList()
+ // which will handle clearing caches other than List cache.
return l.state.Caches.DB.List.Store(list, func() error {
_, err := l.db.NewInsert().Model(list).Exec(ctx)
return err
@@ -160,192 +164,146 @@ func (l *listDB) UpdateList(ctx context.Context, list *gtsmodel.List, columns ..
columns = append(columns, "updated_at")
}
- defer func() {
- // Invalidate all entries for this list ID.
- l.state.Caches.DB.ListEntry.Invalidate("ListID", list.ID)
-
- // Invalidate this entire list's timeline.
- if err := l.state.Timelines.List.RemoveTimeline(ctx, list.ID); err != nil {
- log.Errorf(ctx, "error invalidating list timeline: %q", err)
- }
- }()
-
- return l.state.Caches.DB.List.Store(list, func() error {
+ // Update list in the database, invalidating main list cache.
+ if err := l.state.Caches.DB.List.Store(list, func() error {
_, err := l.db.NewUpdate().
Model(list).
Where("? = ?", bun.Ident("list.id"), list.ID).
Column(columns...).
Exec(ctx)
return err
- })
-}
-
-func (l *listDB) DeleteListByID(ctx context.Context, id string) error {
- // Load list by ID into cache to ensure we can perform
- // all necessary cache invalidation hooks on removal.
- _, err := l.GetListByID(
- // Don't populate the entry;
- // we only want the list ID.
- gtscontext.SetBarebones(ctx),
- id,
- )
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- // NOTE: even if db.ErrNoEntries is returned, we
- // still run the below transaction to ensure related
- // objects are appropriately deleted.
+ }); err != nil {
return err
}
- defer func() {
- // Invalidate this list from cache.
- l.state.Caches.DB.List.Invalidate("ID", id)
+ // Invalidate this entire list's timeline.
+ if err := l.state.Timelines.List.RemoveTimeline(ctx, list.ID); err != nil {
+ log.Errorf(ctx, "error invalidating list timeline: %q", err)
+ }
- // Invalidate this entire list's timeline.
- if err := l.state.Timelines.List.RemoveTimeline(ctx, id); err != nil {
- log.Errorf(ctx, "error invalidating list timeline: %q", err)
- }
- }()
+ return nil
+}
- return l.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
- // Delete all entries attached to list.
+func (l *listDB) DeleteListByID(ctx context.Context, id string) error {
+ // Acquire list owner ID.
+ var accountID string
+
+ // Gather follow IDs of all
+ // entries contained in list.
+ var followIDs []string
+
+ // Delete all list entries associated with list, and list itself in transaction.
+ if err := l.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
if _, err := tx.NewDelete().
Table("list_entries").
Where("? = ?", bun.Ident("list_id"), id).
- Exec(ctx); err != nil {
+ Returning("?", bun.Ident("follow_id")).
+ Exec(ctx, &followIDs); err != nil {
return err
}
- // Delete the list itself.
_, err := tx.NewDelete().
Table("lists").
Where("? = ?", bun.Ident("id"), id).
- Exec(ctx)
+ Returning("?", bun.Ident("account_id")).
+ Exec(ctx, &accountID)
return err
- })
-}
+ }); err != nil {
+ return err
+ }
-/*
- LIST ENTRY functions
-*/
+ // Invalidate the main list database cache.
+ l.state.Caches.DB.List.Invalidate("ID", id)
-func (l *listDB) GetListEntryByID(ctx context.Context, id string) (*gtsmodel.ListEntry, error) {
- return l.getListEntry(
- ctx,
- "ID",
- func(listEntry *gtsmodel.ListEntry) error {
- return l.db.NewSelect().
- Model(listEntry).
- Where("? = ?", bun.Ident("list_entry.id"), id).
- Scan(ctx)
- },
- id,
- )
+ // Invalidate cache of list IDs owned by account.
+ l.state.Caches.DB.ListIDs.Invalidate("a" + accountID)
+
+ // Invalidate all related entry caches for this list.
+ l.invalidateEntryCaches(ctx, []string{id}, followIDs)
+
+ return nil
}
-func (l *listDB) getListEntry(ctx context.Context, lookup string, dbQuery func(*gtsmodel.ListEntry) error, keyParts ...any) (*gtsmodel.ListEntry, error) {
- listEntry, err := l.state.Caches.DB.ListEntry.LoadOne(lookup, func() (*gtsmodel.ListEntry, error) {
- var listEntry gtsmodel.ListEntry
+func (l *listDB) getListIDsByAccountID(ctx context.Context, accountID string) ([]string, error) {
+ return l.state.Caches.DB.ListIDs.Load("a"+accountID, func() ([]string, error) {
+ var listIDs []string
- // Not cached! Perform database query.
- if err := dbQuery(&listEntry); err != nil {
+ // List IDs not in cache.
+ // Perform the DB query.
+ if _, err := l.db.NewSelect().
+ Table("lists").
+ Column("id").
+ Where("? = ?", bun.Ident("account_id"), accountID).
+ OrderExpr("? DESC", bun.Ident("created_at")).
+ Exec(ctx, &listIDs); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return nil, err
}
- return &listEntry, nil
- }, keyParts...)
- if err != nil {
- return nil, err // already processed
- }
-
- if gtscontext.Barebones(ctx) {
- // Only a barebones model was requested.
- return listEntry, nil
- }
-
- // Further populate the list entry fields where applicable.
- if err := l.state.DB.PopulateListEntry(ctx, listEntry); err != nil {
- return nil, err
- }
-
- return listEntry, nil
+ return listIDs, nil
+ })
}
-func (l *listDB) GetListEntries(ctx context.Context,
- listID string,
- maxID string,
- sinceID string,
- minID string,
- limit int,
-) ([]*gtsmodel.ListEntry, error) {
- // Ensure reasonable
- if limit < 0 {
- limit = 0
- }
-
- // Make educated guess for slice size
- var (
- entryIDs = make([]string, 0, limit)
- frontToBack = true
- )
-
- q := l.db.
- NewSelect().
- TableExpr("? AS ?", bun.Ident("list_entries"), bun.Ident("entry")).
- // Select only IDs from table
- Column("entry.id").
- // Select only entries belonging to listID.
- Where("? = ?", bun.Ident("entry.list_id"), listID)
-
- if maxID != "" {
- // return only entries LOWER (ie., older) than maxID
- q = q.Where("? < ?", bun.Ident("entry.id"), maxID)
- }
-
- if sinceID != "" {
- // return only entries HIGHER (ie., newer) than sinceID
- q = q.Where("? > ?", bun.Ident("entry.id"), sinceID)
- }
+func (l *listDB) getListIDsWithFollowID(ctx context.Context, followID string) ([]string, error) {
+ return l.state.Caches.DB.ListIDs.Load("f"+followID, func() ([]string, error) {
+ var listIDs []string
- if minID != "" {
- // return only entries HIGHER (ie., newer) than minID
- q = q.Where("? > ?", bun.Ident("entry.id"), minID)
-
- // page up
- frontToBack = false
- }
+ // List IDs not in cache.
+ // Perform the DB query.
+ if _, err := l.db.NewSelect().
+ Table("list_entries").
+ Column("list_id").
+ Where("? = ?", bun.Ident("follow_id"), followID).
+ OrderExpr("? DESC", bun.Ident("created_at")).
+ Exec(ctx, &listIDs); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
+ return nil, err
+ }
- if limit > 0 {
- // limit amount of entries returned
- q = q.Limit(limit)
- }
+ return listIDs, nil
+ })
+}
- if frontToBack {
- // Page down.
- q = q.Order("entry.id DESC")
- } else {
- // Page up.
- q = q.Order("entry.id ASC")
- }
+func (l *listDB) GetFollowIDsInList(ctx context.Context, listID string, page *paging.Page) ([]string, error) {
+ return loadPagedIDs(&l.state.Caches.DB.ListedIDs, "f"+listID, page, func() ([]string, error) {
+ var followIDs []string
- if err := q.Scan(ctx, &entryIDs); err != nil {
- return nil, err
- }
+ // Follow IDs not in cache.
+ // Perform the DB query.
+ _, err := l.db.NewSelect().
+ Table("list_entries").
+ Column("follow_id").
+ Where("? = ?", bun.Ident("list_id"), listID).
+ OrderExpr("? DESC", bun.Ident("created_at")).
+ Exec(ctx, &followIDs)
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ return nil, err
+ }
- if len(entryIDs) == 0 {
- return nil, nil
- }
+ return followIDs, nil
+ })
+}
- // If we're paging up, we still want entries
- // to be sorted by ID desc, so reverse ids slice.
- // https://zchee.github.io/golang-wiki/SliceTricks/#reversing
- if !frontToBack {
- for l, r := 0, len(entryIDs)-1; l < r; l, r = l+1, r-1 {
- entryIDs[l], entryIDs[r] = entryIDs[r], entryIDs[l]
+func (l *listDB) GetAccountIDsInList(ctx context.Context, listID string, page *paging.Page) ([]string, error) {
+ return loadPagedIDs(&l.state.Caches.DB.ListedIDs, "a"+listID, page, func() ([]string, error) {
+ var accountIDs []string
+
+ // Account IDs not in cache.
+ // Perform the DB query.
+ _, err := l.db.NewSelect().
+ Table("follows").
+ Column("follows.target_account_id").
+ Join("INNER JOIN ?", bun.Ident("list_entries")).
+ JoinOn("? = ?", bun.Ident("follows.id"), bun.Ident("list_entries.follow_id")).
+ Where("? = ?", bun.Ident("list_entries.list_id"), listID).
+ OrderExpr("? DESC", bun.Ident("list_entries.id")).
+ Exec(ctx, &accountIDs)
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ return nil, err
}
- }
- // Return list entries by their IDs.
- return l.GetListEntriesByIDs(ctx, entryIDs)
+ return accountIDs, nil
+ })
}
func (l *listDB) GetListsByIDs(ctx context.Context, ids []string) ([]*gtsmodel.List, error) {
@@ -353,15 +311,8 @@ func (l *listDB) GetListsByIDs(ctx context.Context, ids []string) ([]*gtsmodel.L
lists, err := l.state.Caches.DB.List.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.List, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached lists.
- lists := make([]*gtsmodel.List, 0, count)
+ lists := make([]*gtsmodel.List, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
@@ -402,82 +353,6 @@ func (l *listDB) GetListsByIDs(ctx context.Context, ids []string) ([]*gtsmodel.L
return lists, nil
}
-func (l *listDB) GetListEntriesByIDs(ctx context.Context, ids []string) ([]*gtsmodel.ListEntry, error) {
- // Load all entry IDs via cache loader callbacks.
- entries, err := l.state.Caches.DB.ListEntry.LoadIDs("ID",
- ids,
- func(uncached []string) ([]*gtsmodel.ListEntry, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
- // Preallocate expected length of uncached entries.
- entries := make([]*gtsmodel.ListEntry, 0, count)
-
- // Perform database query scanning
- // the remaining (uncached) IDs.
- if err := l.db.NewSelect().
- Model(&entries).
- Where("? IN (?)", bun.Ident("id"), bun.In(uncached)).
- Scan(ctx); err != nil {
- return nil, err
- }
-
- return entries, nil
- },
- )
- if err != nil {
- return nil, err
- }
-
- // Reorder the entries by their
- // IDs to ensure in correct order.
- getID := func(e *gtsmodel.ListEntry) string { return e.ID }
- util.OrderBy(entries, ids, getID)
-
- if gtscontext.Barebones(ctx) {
- // no need to fully populate.
- return entries, nil
- }
-
- // Populate all loaded entries, removing those we fail to
- // populate (removes needing so many nil checks everywhere).
- entries = slices.DeleteFunc(entries, func(entry *gtsmodel.ListEntry) bool {
- if err := l.PopulateListEntry(ctx, entry); err != nil {
- log.Errorf(ctx, "error populating entry %s: %v", entry.ID, err)
- return true
- }
- return false
- })
-
- return entries, nil
-}
-
-func (l *listDB) GetListEntriesForFollowID(ctx context.Context, followID string) ([]*gtsmodel.ListEntry, error) {
- var entryIDs []string
-
- if err := l.db.
- NewSelect().
- TableExpr("? AS ?", bun.Ident("list_entries"), bun.Ident("entry")).
- // Select only IDs from table
- Column("entry.id").
- // Select only entries belonging with given followID.
- Where("? = ?", bun.Ident("entry.follow_id"), followID).
- Scan(ctx, &entryIDs); err != nil {
- return nil, err
- }
-
- if len(entryIDs) == 0 {
- return nil, nil
- }
-
- // Return list entries by their IDs.
- return l.GetListEntriesByIDs(ctx, entryIDs)
-}
-
func (l *listDB) PopulateListEntry(ctx context.Context, listEntry *gtsmodel.ListEntry) error {
var err error
@@ -496,109 +371,111 @@ func (l *listDB) PopulateListEntry(ctx context.Context, listEntry *gtsmodel.List
}
func (l *listDB) PutListEntries(ctx context.Context, entries []*gtsmodel.ListEntry) error {
- defer func() {
- // Collect unique list IDs from the provided entries.
- listIDs := util.Collate(entries, func(e *gtsmodel.ListEntry) string {
- return e.ListID
- })
-
- for _, id := range listIDs {
- // Invalidate the timeline for the list this entry belongs to.
- if err := l.state.Timelines.List.RemoveTimeline(ctx, id); err != nil {
- log.Errorf(ctx, "error invalidating list timeline: %q", err)
- }
- }
- }()
-
- // Finally, insert each list entry into the database.
- return l.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
+ // Insert all entries into the database in a single transaction (all or nothing!).
+ if err := l.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
for _, entry := range entries {
- entry := entry // rescope
- if err := l.state.Caches.DB.ListEntry.Store(entry, func() error {
- _, err := tx.
- NewInsert().
- Model(entry).
- Exec(ctx)
- return err
- }); err != nil {
+ if _, err := tx.
+ NewInsert().
+ Model(entry).
+ Exec(ctx); err != nil {
return err
}
}
return nil
+ }); err != nil {
+ return err
+ }
+
+ // Collect unique list IDs from the provided list entries.
+ listIDs := util.Collate(entries, func(e *gtsmodel.ListEntry) string {
+ return e.ListID
+ })
+
+ // Collect unique follow IDs from the provided list entries.
+ followIDs := util.Collate(entries, func(e *gtsmodel.ListEntry) string {
+ return e.FollowID
})
+
+ // Invalidate all related list entry caches.
+ l.invalidateEntryCaches(ctx, listIDs, followIDs)
+
+ return nil
}
-func (l *listDB) DeleteListEntry(ctx context.Context, id string) error {
- // Load list entry into cache to ensure we can perform
- // all necessary cache invalidation hooks on removal.
- entry, err := l.GetListEntryByID(
- // Don't populate the entry;
- // we only want the list ID.
- gtscontext.SetBarebones(ctx),
- id,
- )
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // Already gone.
- return nil
- }
+func (l *listDB) DeleteListEntry(ctx context.Context, listID string, followID string) error {
+ // Delete list entry with given
+ // ID, returning its list ID.
+ if _, err := l.db.NewDelete().
+ Table("list_entries").
+ Where("? = ?", bun.Ident("list_id"), listID).
+ Where("? = ?", bun.Ident("follow_id"), followID).
+ Exec(ctx, &listID); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- defer func() {
- // Invalidate this list entry upon delete.
- l.state.Caches.DB.ListEntry.Invalidate("ID", id)
+ // Invalidate all related list entry caches.
+ l.invalidateEntryCaches(ctx, []string{listID},
+ []string{followID})
- // Invalidate the timeline for the list this entry belongs to.
- if err := l.state.Timelines.List.RemoveTimeline(ctx, entry.ListID); err != nil {
- log.Errorf(ctx, "error invalidating list timeline: %q", err)
- }
- }()
-
- // Finally delete the list entry.
- _, err = l.db.NewDelete().
- Table("list_entries").
- Where("? = ?", bun.Ident("id"), id).
- Exec(ctx)
- return err
+ return nil
}
-func (l *listDB) DeleteListEntriesForFollowID(ctx context.Context, followID string) error {
- var entryIDs []string
+func (l *listDB) DeleteAllListEntriesByFollows(ctx context.Context, followIDs ...string) error {
+ var listIDs []string
+
+ // Check for empty list.
+ if len(followIDs) == 0 {
+ return nil
+ }
- // Fetch entry IDs for follow ID.
- if err := l.db.
- NewSelect().
+ // Delete all entries with follow
+ // ID, returning IDs and list IDs.
+ if _, err := l.db.NewDelete().
Table("list_entries").
- Column("id").
- Where("? = ?", bun.Ident("follow_id"), followID).
- Order("id DESC").
- Scan(ctx, &entryIDs); err != nil {
+ Where("? IN (?)", bun.Ident("follow_id"), bun.In(followIDs)).
+ Returning("?", bun.Ident("list_id")).
+ Exec(ctx, &listIDs); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- for _, id := range entryIDs {
- // Delete each separately to trigger cache invalidations.
- if err := l.DeleteListEntry(ctx, id); err != nil {
- return err
- }
- }
+ // Deduplicate IDs before invalidate.
+ listIDs = util.Deduplicate(listIDs)
+
+ // Invalidate all related list entry caches.
+ l.invalidateEntryCaches(ctx, listIDs, followIDs)
return nil
}
-func (l *listDB) ListIncludesAccount(ctx context.Context, listID string, accountID string) (bool, error) {
- exists, err := l.db.
- NewSelect().
- TableExpr("? AS ?", bun.Ident("list_entries"), bun.Ident("list_entry")).
- Join(
- "JOIN ? AS ? ON ? = ?",
- bun.Ident("follows"), bun.Ident("follow"),
- bun.Ident("list_entry.follow_id"), bun.Ident("follow.id"),
- ).
- Where("? = ?", bun.Ident("list_entry.list_id"), listID).
- Where("? = ?", bun.Ident("follow.target_account_id"), accountID).
- Exists(ctx)
-
- return exists, err
+// invalidateEntryCaches will invalidate all related ListEntry caches for given list IDs and follow IDs, including timelines.
+func (l *listDB) invalidateEntryCaches(ctx context.Context, listIDs, followIDs []string) {
+ var keys []string
+
+ // Generate ListedID keys to invalidate.
+ keys = slices.Grow(keys[:0], 2*len(listIDs))
+ for _, listID := range listIDs {
+ keys = append(keys,
+ "a"+listID,
+ "f"+listID,
+ )
+
+ // Invalidate the timeline for the list this entry belongs to.
+ if err := l.state.Timelines.List.RemoveTimeline(ctx, listID); err != nil {
+ log.Errorf(ctx, "error invalidating list timeline: %q", err)
+ }
+ }
+
+ // Invalidate ListedID slice cache entries.
+ l.state.Caches.DB.ListedIDs.Invalidate(keys...)
+
+ // Generate ListID keys to invalidate.
+ keys = slices.Grow(keys[:0], len(followIDs))
+ for _, followID := range followIDs {
+ keys = append(keys, "f"+followID)
+ }
+
+ // Invalidate ListID slice cache entries.
+ l.state.Caches.DB.ListIDs.Invalidate(keys...)
}
diff --git a/internal/db/bundb/list_test.go b/internal/db/bundb/list_test.go
index 9c5fb2c76..3952a87c0 100644
--- a/internal/db/bundb/list_test.go
+++ b/internal/db/bundb/list_test.go
@@ -24,7 +24,6 @@ import (
"github.com/stretchr/testify/suite"
"github.com/superseriousbusiness/gotosocial/internal/db"
- "github.com/superseriousbusiness/gotosocial/internal/gtscontext"
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
)
@@ -32,7 +31,7 @@ type ListTestSuite struct {
BunDBStandardTestSuite
}
-func (suite *ListTestSuite) testStructs() (*gtsmodel.List, *gtsmodel.Account) {
+func (suite *ListTestSuite) testStructs() (*gtsmodel.List, []*gtsmodel.ListEntry, *gtsmodel.Account) {
testList := &gtsmodel.List{}
*testList = *suite.testLists["local_account_1_list_1"]
@@ -55,12 +54,10 @@ func (suite *ListTestSuite) testStructs() (*gtsmodel.List, *gtsmodel.Account) {
}
})
- testList.ListEntries = entries
-
testAccount := &gtsmodel.Account{}
*testAccount = *suite.testAccounts["local_account_1"]
- return testList, testAccount
+ return testList, entries, testAccount
}
func (suite *ListTestSuite) checkList(expected *gtsmodel.List, actual *gtsmodel.List) {
@@ -103,7 +100,7 @@ func (suite *ListTestSuite) checkListEntries(expected []*gtsmodel.ListEntry, act
}
func (suite *ListTestSuite) TestGetListByID() {
- testList, _ := suite.testStructs()
+ testList, _, _ := suite.testStructs()
dbList, err := suite.db.GetListByID(context.Background(), testList.ID)
if err != nil {
@@ -111,13 +108,12 @@ func (suite *ListTestSuite) TestGetListByID() {
}
suite.checkList(testList, dbList)
- suite.checkListEntries(testList.ListEntries, dbList.ListEntries)
}
func (suite *ListTestSuite) TestGetListsForAccountID() {
- testList, testAccount := suite.testStructs()
+ testList, _, testAccount := suite.testStructs()
- dbLists, err := suite.db.GetListsForAccountID(context.Background(), testAccount.ID)
+ dbLists, err := suite.db.GetListsByAccountID(context.Background(), testAccount.ID)
if err != nil {
suite.FailNow(err.Error())
}
@@ -129,20 +125,9 @@ func (suite *ListTestSuite) TestGetListsForAccountID() {
suite.checkList(testList, dbLists[0])
}
-func (suite *ListTestSuite) TestGetListEntries() {
- testList, _ := suite.testStructs()
-
- dbListEntries, err := suite.db.GetListEntries(context.Background(), testList.ID, "", "", "", 0)
- if err != nil {
- suite.FailNow(err.Error())
- }
-
- suite.checkListEntries(testList.ListEntries, dbListEntries)
-}
-
func (suite *ListTestSuite) TestPutList() {
ctx := context.Background()
- _, testAccount := suite.testStructs()
+ _, _, testAccount := suite.testStructs()
testList := &gtsmodel.List{
ID: "01H0J2PMYM54618VCV8Y8QYAT4",
@@ -166,7 +151,7 @@ func (suite *ListTestSuite) TestPutList() {
func (suite *ListTestSuite) TestUpdateList() {
ctx := context.Background()
- testList, _ := suite.testStructs()
+ testList, _, _ := suite.testStructs()
// Get List in the cache first.
dbList, err := suite.db.GetListByID(ctx, testList.ID)
@@ -192,7 +177,7 @@ func (suite *ListTestSuite) TestUpdateList() {
func (suite *ListTestSuite) TestDeleteList() {
ctx := context.Background()
- testList, _ := suite.testStructs()
+ testList, _, _ := suite.testStructs()
// Get List in the cache first.
if _, err := suite.db.GetListByID(ctx, testList.ID); err != nil {
@@ -209,18 +194,19 @@ func (suite *ListTestSuite) TestDeleteList() {
_, err := suite.db.GetListByID(ctx, testList.ID)
suite.ErrorIs(err, db.ErrNoEntries)
- // All entries belonging to this
- // list should now be deleted.
- listEntries, err := suite.db.GetListEntries(ctx, testList.ID, "", "", "", 0)
- if err != nil {
- suite.FailNow(err.Error())
- }
- suite.Empty(listEntries)
+ // All accounts / follows attached to this
+ // list should now be return empty values.
+ listAccounts, err1 := suite.db.GetAccountsInList(ctx, testList.ID, nil)
+ listFollows, err2 := suite.db.GetFollowsInList(ctx, testList.ID, nil)
+ suite.NoError(err1)
+ suite.NoError(err2)
+ suite.Empty(listAccounts)
+ suite.Empty(listFollows)
}
func (suite *ListTestSuite) TestPutListEntries() {
ctx := context.Background()
- testList, _ := suite.testStructs()
+ testList, testEntries, _ := suite.testStructs()
listEntries := []*gtsmodel.ListEntry{
{
@@ -244,91 +230,58 @@ func (suite *ListTestSuite) TestPutListEntries() {
suite.FailNow(err.Error())
}
- // Add these entries to the test list, sort it again
- // to reflect what we'd expect to get from the db.
- testList.ListEntries = append(testList.ListEntries, listEntries...)
- slices.SortFunc(testList.ListEntries, func(a, b *gtsmodel.ListEntry) int {
- const k = -1
- switch {
- case a.ID > b.ID:
- return +k
- case a.ID < b.ID:
- return -k
- default:
- return 0
- }
- })
-
- // Now get all list entries from the db.
- // Use barebones for this because the ones
- // we just added will fail if we try to get
- // the nonexistent follows.
- dbListEntries, err := suite.db.GetListEntries(
- gtscontext.SetBarebones(ctx),
- testList.ID,
- "", "", "", 0)
- if err != nil {
- suite.FailNow(err.Error())
- }
-
- suite.checkListEntries(testList.ListEntries, dbListEntries)
+ // Get all follows stored under this list ID, to ensure
+ // the newly added list entry follows are among these.
+ followIDs, err := suite.db.GetFollowIDsInList(ctx, testList.ID, nil)
+ suite.NoError(err)
+ suite.Len(followIDs, len(testEntries)+len(listEntries))
+ suite.Contains(followIDs, "01H0MKNFRFZS8R9WV6DBX31Y03")
+ suite.Contains(followIDs, "01H0MKP6RR8VEHN3GVWFBP2H30")
+ suite.Contains(followIDs, "01H0MKQ0KA29C6NFJ27GTZD16J")
}
func (suite *ListTestSuite) TestDeleteListEntry() {
ctx := context.Background()
- testList, _ := suite.testStructs()
-
- // Get List in the cache first.
- if _, err := suite.db.GetListByID(ctx, testList.ID); err != nil {
- suite.FailNow(err.Error())
- }
+ testList, testEntries, _ := suite.testStructs()
// Delete the first entry.
- if err := suite.db.DeleteListEntry(ctx, testList.ListEntries[0].ID); err != nil {
- suite.FailNow(err.Error())
- }
-
- // Get list from the db again.
- dbList, err := suite.db.GetListByID(ctx, testList.ID)
- if err != nil {
+ if err := suite.db.DeleteListEntry(ctx,
+ testEntries[0].ListID,
+ testEntries[0].FollowID,
+ ); err != nil {
suite.FailNow(err.Error())
}
- // Bodge the testlist as though
- // we'd removed the first entry.
- testList.ListEntries = testList.ListEntries[1:]
- suite.checkList(testList, dbList)
+ // Get all follows stored under this list ID, to ensure
+ // the newly removed list entry follow is now missing.
+ followIDs, err := suite.db.GetFollowIDsInList(ctx, testList.ID, nil)
+ suite.NoError(err)
+ suite.Len(followIDs, len(testEntries)-1)
+ suite.NotContains(followIDs, testEntries[0].FollowID)
}
-func (suite *ListTestSuite) TestDeleteListEntriesForFollowID() {
+func (suite *ListTestSuite) TestDeleteAllListEntriesByFollows() {
ctx := context.Background()
- testList, _ := suite.testStructs()
-
- // Get List in the cache first.
- if _, err := suite.db.GetListByID(ctx, testList.ID); err != nil {
- suite.FailNow(err.Error())
- }
+ testList, testEntries, _ := suite.testStructs()
// Delete the first entry.
- if err := suite.db.DeleteListEntriesForFollowID(ctx, testList.ListEntries[0].FollowID); err != nil {
+ if err := suite.db.DeleteAllListEntriesByFollows(ctx,
+ testEntries[0].FollowID,
+ ); err != nil {
suite.FailNow(err.Error())
}
- // Get list from the db again.
- dbList, err := suite.db.GetListByID(ctx, testList.ID)
- if err != nil {
- suite.FailNow(err.Error())
- }
-
- // Bodge the testlist as though
- // we'd removed the first entry.
- testList.ListEntries = testList.ListEntries[1:]
- suite.checkList(testList, dbList)
+ // Get all follows stored under this list ID, to ensure
+ // the newly removed list entry follow is now missing.
+ followIDs, err := suite.db.GetFollowIDsInList(ctx, testList.ID, nil)
+ suite.NoError(err)
+ suite.Len(followIDs, len(testEntries)-1)
+ suite.NotContains(followIDs, testEntries[0].FollowID)
}
func (suite *ListTestSuite) TestListIncludesAccount() {
ctx := context.Background()
- testList, _ := suite.testStructs()
+ testList, _, _ := suite.testStructs()
for accountID, expected := range map[string]bool{
suite.testAccounts["admin_account"].ID: true,
@@ -336,7 +289,7 @@ func (suite *ListTestSuite) TestListIncludesAccount() {
suite.testAccounts["local_account_2"].ID: true,
"01H7074GEZJ56J5C86PFB0V2CT": false,
} {
- includes, err := suite.db.ListIncludesAccount(ctx, testList.ID, accountID)
+ includes, err := suite.db.IsAccountInList(ctx, testList.ID, accountID)
if err != nil {
suite.FailNow(err.Error())
}
diff --git a/internal/db/bundb/media.go b/internal/db/bundb/media.go
index 65348733c..de980a16a 100644
--- a/internal/db/bundb/media.go
+++ b/internal/db/bundb/media.go
@@ -24,7 +24,6 @@ import (
"time"
"github.com/superseriousbusiness/gotosocial/internal/db"
- "github.com/superseriousbusiness/gotosocial/internal/gtscontext"
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
"github.com/superseriousbusiness/gotosocial/internal/paging"
@@ -57,15 +56,8 @@ func (m *mediaDB) GetAttachmentsByIDs(ctx context.Context, ids []string) ([]*gts
media, err := m.state.Caches.DB.Media.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.MediaAttachment, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached media attachments.
- media := make([]*gtsmodel.MediaAttachment, 0, count)
+ media := make([]*gtsmodel.MediaAttachment, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
@@ -129,30 +121,38 @@ func (m *mediaDB) UpdateAttachment(ctx context.Context, media *gtsmodel.MediaAtt
}
func (m *mediaDB) DeleteAttachment(ctx context.Context, id string) error {
- // Load media into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- media, err := m.GetAttachmentByID(gtscontext.SetBarebones(ctx), id)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // not an issue.
- err = nil
- }
- return err
- }
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.MediaAttachment
+ deleted.ID = id
- // On return, ensure that media with ID is invalidated.
- defer m.state.Caches.DB.Media.Invalidate("ID", id)
+ // Delete media attachment and update related models in new transaction.
+ err := m.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
- // Delete media attachment in new transaction.
- err = m.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
- if media.AccountID != "" {
+ // Initially, delete the media model,
+ // returning the required fields we need.
+ if _, err := tx.NewDelete().
+ Model(&deleted).
+ Where("? = ?", bun.Ident("id"), id).
+ Returning("?, ?, ?, ?",
+ bun.Ident("account_id"),
+ bun.Ident("status_id"),
+ bun.Ident("avatar"),
+ bun.Ident("header"),
+ ).
+ Exec(ctx); err != nil {
+ return gtserror.Newf("error deleting media: %w", err)
+ }
+
+ // If media was attached to account,
+ // we need to remove link from account.
+ if deleted.AccountID != "" {
var account gtsmodel.Account
// Get related account model.
if _, err := tx.NewSelect().
Model(&account).
- Where("? = ?", bun.Ident("id"), media.AccountID).
+ Where("? = ?", bun.Ident("id"), deleted.AccountID).
Exec(ctx); err != nil && !errors.Is(err, db.ErrNoEntries) {
return gtserror.Newf("error selecting account: %w", err)
}
@@ -160,11 +160,11 @@ func (m *mediaDB) DeleteAttachment(ctx context.Context, id string) error {
var set func(*bun.UpdateQuery) *bun.UpdateQuery
switch {
- case *media.Avatar && account.AvatarMediaAttachmentID == id:
+ case *deleted.Avatar && account.AvatarMediaAttachmentID == id:
set = func(q *bun.UpdateQuery) *bun.UpdateQuery {
return q.Set("? = NULL", bun.Ident("avatar_media_attachment_id"))
}
- case *media.Header && account.HeaderMediaAttachmentID == id:
+ case *deleted.Header && account.HeaderMediaAttachmentID == id:
set = func(q *bun.UpdateQuery) *bun.UpdateQuery {
return q.Set("? = NULL", bun.Ident("header_media_attachment_id"))
}
@@ -183,13 +183,15 @@ func (m *mediaDB) DeleteAttachment(ctx context.Context, id string) error {
}
}
- if media.StatusID != "" {
+ // If media was attached to a status,
+ // we need to remove link from status.
+ if deleted.StatusID != "" {
var status gtsmodel.Status
// Get related status model.
if _, err := tx.NewSelect().
Model(&status).
- Where("? = ?", bun.Ident("id"), media.StatusID).
+ Where("? = ?", bun.Ident("id"), deleted.StatusID).
Exec(ctx); err != nil && !errors.Is(err, db.ErrNoEntries) {
return gtserror.Newf("error selecting status: %w", err)
}
@@ -213,17 +215,14 @@ func (m *mediaDB) DeleteAttachment(ctx context.Context, id string) error {
}
}
- // Finally delete this media.
- if _, err := tx.NewDelete().
- Table("media_attachments").
- Where("? = ?", bun.Ident("id"), id).
- Exec(ctx); err != nil {
- return gtserror.Newf("error deleting media: %w", err)
- }
-
return nil
})
+ // Invalidate cached media with ID, manually
+ // call invalidate hook in case not in cache.
+ m.state.Caches.DB.Media.Invalidate("ID", id)
+ m.state.Caches.OnInvalidateMedia(&deleted)
+
return err
}
diff --git a/internal/db/bundb/mention.go b/internal/db/bundb/mention.go
index e56300367..ba8c0ba11 100644
--- a/internal/db/bundb/mention.go
+++ b/internal/db/bundb/mention.go
@@ -69,15 +69,8 @@ func (m *mentionDB) GetMentions(ctx context.Context, ids []string) ([]*gtsmodel.
mentions, err := m.state.Caches.DB.Mention.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.Mention, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached mentions.
- mentions := make([]*gtsmodel.Mention, 0, count)
+ mentions := make([]*gtsmodel.Mention, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
@@ -166,24 +159,18 @@ func (m *mentionDB) PutMention(ctx context.Context, mention *gtsmodel.Mention) e
}
func (m *mentionDB) DeleteMentionByID(ctx context.Context, id string) error {
- defer m.state.Caches.DB.Mention.Invalidate("ID", id)
-
- // Load mention into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := m.GetMention(gtscontext.SetBarebones(ctx), id)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // not an issue.
- err = nil
- }
+ // Delete mention with given ID,
+ // returning the deleted models.
+ if _, err := m.db.NewDelete().
+ Table("mentions").
+ Where("? = ?", bun.Ident("id"), id).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Finally delete mention from DB.
- _, err = m.db.NewDelete().
- Table("mentions").
- Where("? = ?", bun.Ident("id"), id).
- Exec(ctx)
- return err
+ // Invalidate the cached mention with ID.
+ m.state.Caches.DB.Mention.Invalidate("ID", id)
+
+ return nil
}
diff --git a/internal/db/bundb/move.go b/internal/db/bundb/move.go
index cccef5872..23e5c6d27 100644
--- a/internal/db/bundb/move.go
+++ b/internal/db/bundb/move.go
@@ -234,13 +234,17 @@ func (m *moveDB) UpdateMove(ctx context.Context, move *gtsmodel.Move, columns ..
}
func (m *moveDB) DeleteMoveByID(ctx context.Context, id string) error {
- defer m.state.Caches.DB.Move.Invalidate("ID", id)
-
- _, err := m.db.
- NewDelete().
+ // Delete move with given ID.
+ if _, err := m.db.NewDelete().
TableExpr("? AS ?", bun.Ident("moves"), bun.Ident("move")).
Where("? = ?", bun.Ident("move.id"), id).
- Exec(ctx)
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
+ return nil
+ }
- return err
+ // Invalidate the cached move model with ID.
+ m.state.Caches.DB.Move.Invalidate("ID", id)
+
+ return nil
}
diff --git a/internal/db/bundb/notification.go b/internal/db/bundb/notification.go
index 9959b160e..770e84c5c 100644
--- a/internal/db/bundb/notification.go
+++ b/internal/db/bundb/notification.go
@@ -22,6 +22,7 @@ import (
"errors"
"slices"
+ "github.com/superseriousbusiness/gotosocial/internal/db"
"github.com/superseriousbusiness/gotosocial/internal/gtscontext"
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
@@ -107,15 +108,8 @@ func (n *notificationDB) GetNotificationsByIDs(ctx context.Context, ids []string
notifs, err := n.state.Caches.DB.Notification.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.Notification, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached notifications.
- notifs := make([]*gtsmodel.Notification, 0, count)
+ notifs := make([]*gtsmodel.Notification, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
@@ -299,7 +293,8 @@ func (n *notificationDB) DeleteNotificationByID(ctx context.Context, id string)
NewDelete().
Table("notifications").
Where("? = ?", bun.Ident("id"), id).
- Exec(ctx); err != nil {
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
@@ -310,7 +305,7 @@ func (n *notificationDB) DeleteNotificationByID(ctx context.Context, id string)
func (n *notificationDB) DeleteNotifications(ctx context.Context, types []string, targetAccountID string, originAccountID string) error {
if targetAccountID == "" && originAccountID == "" {
- return errors.New("DeleteNotifications: one of targetAccountID or originAccountID must be set")
+ return gtserror.New("one of targetAccountID or originAccountID must be set")
}
q := n.db.
diff --git a/internal/db/bundb/poll.go b/internal/db/bundb/poll.go
index 5c1d9c6dd..f5c33ce9b 100644
--- a/internal/db/bundb/poll.go
+++ b/internal/db/bundb/poll.go
@@ -177,17 +177,36 @@ func (p *pollDB) UpdatePoll(ctx context.Context, poll *gtsmodel.Poll, cols ...st
}
func (p *pollDB) DeletePollByID(ctx context.Context, id string) error {
- // Delete poll by ID from database.
- if _, err := p.db.NewDelete().
- Table("polls").
- Where("? = ?", bun.Ident("id"), id).
- Exec(ctx); err != nil {
+ // Delete poll vote with ID, and its associated votes from the database.
+ if err := p.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
+
+ // Delete poll from database.
+ if _, err := tx.NewDelete().
+ Table("polls").
+ Where("? = ?", bun.Ident("id"), id).
+ Exec(ctx); err != nil {
+ return err
+ }
+
+ // Delete the poll votes.
+ _, err := tx.NewDelete().
+ Table("poll_votes").
+ Where("? = ?", bun.Ident("poll_id"), id).
+ Exec(ctx)
+ return err
+ }); err != nil {
return err
}
- // Invalidate poll by ID from cache.
+ // Wrap provided ID in a poll
+ // model for calling cache hook.
+ var deleted gtsmodel.Poll
+ deleted.ID = id
+
+ // Invalidate cached poll with ID, manually
+ // call invalidate hook in case not cached.
p.state.Caches.DB.Poll.Invalidate("ID", id)
- p.state.Caches.DB.PollVoteIDs.Invalidate(id)
+ p.state.Caches.OnInvalidatePoll(&deleted)
return nil
}
@@ -274,15 +293,8 @@ func (p *pollDB) GetPollVotes(ctx context.Context, pollID string) ([]*gtsmodel.P
votes, err := p.state.Caches.DB.PollVote.LoadIDs("ID",
voteIDs,
func(uncached []string) ([]*gtsmodel.PollVote, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached votes.
- votes := make([]*gtsmodel.PollVote, 0, count)
+ votes := make([]*gtsmodel.PollVote, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
@@ -391,148 +403,44 @@ func (p *pollDB) PutPollVote(ctx context.Context, vote *gtsmodel.PollVote) error
})
}
-func (p *pollDB) DeletePollVotes(ctx context.Context, pollID string) error {
- err := p.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
- // Delete all votes in poll.
- res, err := tx.NewDelete().
- Table("poll_votes").
- Where("? = ?", bun.Ident("poll_id"), pollID).
- Exec(ctx)
- if err != nil {
- // irrecoverable
- return err
- }
-
- ra, err := res.RowsAffected()
- if err != nil {
- // irrecoverable
- return err
- }
-
- if ra == 0 {
- // No poll votes deleted,
- // nothing to update.
- return nil
- }
-
- // Select current poll counts from DB,
- // taking minimal columns needed to
- // increment/decrement votes.
- var poll gtsmodel.Poll
- switch err := tx.NewSelect().
- Model(&poll).
- Column("options", "votes", "voters").
- Where("? = ?", bun.Ident("id"), pollID).
- Scan(ctx); {
-
- case err == nil:
- // no issue.
-
- case errors.Is(err, db.ErrNoEntries):
- // no votes found,
- // return here.
- return nil
-
- default:
- // irrecoverable.
- return err
- }
-
- // Zero all counts.
- poll.ResetVotes()
-
- // Finally, update the poll entry.
- _, err = tx.NewUpdate().
- Model(&poll).
- Column("votes", "voters").
- Where("? = ?", bun.Ident("id"), pollID).
- Exec(ctx)
- return err
- })
-
- if err != nil {
- return err
- }
-
- // Invalidate poll vote and poll entry from caches.
- p.state.Caches.DB.Poll.Invalidate("ID", pollID)
- p.state.Caches.DB.PollVote.Invalidate("PollID", pollID)
- p.state.Caches.DB.PollVoteIDs.Invalidate(pollID)
-
- return nil
-}
-
func (p *pollDB) DeletePollVoteBy(ctx context.Context, pollID string, accountID string) error {
- err := p.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
- // Slice should only ever be of length
- // 0 or 1; it's a slice of slices only
- // because we can't LIMIT deletes to 1.
- var choicesSlice [][]int
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.PollVote
+ deleted.AccountID = accountID
+ deleted.PollID = pollID
+
+ // Delete the poll vote with given poll and account IDs, and update vote counts.
+ if err := p.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
// Delete vote in poll by account,
- // returning the ID + choices of the vote.
- if err := tx.NewDelete().
- Table("poll_votes").
+ // returning deleted model info.
+ switch _, err := tx.NewDelete().
+ Model(&deleted).
Where("? = ?", bun.Ident("poll_id"), pollID).
Where("? = ?", bun.Ident("account_id"), accountID).
Returning("?", bun.Ident("choices")).
- Scan(ctx, &choicesSlice); err != nil {
- // irrecoverable.
- return err
- }
-
- if len(choicesSlice) != 1 {
- // No poll votes by this
- // acct on this poll.
- return nil
- }
-
- // Extract the *actual* choices.
- choices := choicesSlice[0]
-
- // Select current poll counts from DB,
- // taking minimal columns needed to
- // increment/decrement votes.
- var poll gtsmodel.Poll
- switch err := tx.NewSelect().
- Model(&poll).
- Column("options", "votes", "voters").
- Where("? = ?", bun.Ident("id"), pollID).
- Scan(ctx); {
+ Exec(ctx); {
case err == nil:
- // no issue.
-
+ // no issue
case errors.Is(err, db.ErrNoEntries):
- // no poll found,
- // return here.
return nil
-
default:
- // irrecoverable.
return err
}
- // Decrement votes for choices.
- poll.DecrementVotes(choices)
-
- // Finally, update the poll entry.
- _, err := tx.NewUpdate().
- Model(&poll).
- Column("votes", "voters").
- Where("? = ?", bun.Ident("id"), pollID).
- Exec(ctx)
+ // Update the votes for this deleted poll.
+ err := updatePollCounts(ctx, tx, &deleted)
return err
- })
-
- if err != nil {
+ }); err != nil {
return err
}
- // Invalidate poll vote and poll entry from caches.
- p.state.Caches.DB.Poll.Invalidate("ID", pollID)
+ // Invalidate the poll vote cache by given poll + account IDs, also
+ // manually call invalidation hook in case not actually stored in cache.
p.state.Caches.DB.PollVote.Invalidate("PollID,AccountID", pollID, accountID)
- p.state.Caches.DB.PollVoteIDs.Invalidate(pollID)
+ p.state.Caches.OnInvalidatePollVote(&deleted)
return nil
}
@@ -562,6 +470,48 @@ func (p *pollDB) DeletePollVotesByAccountID(ctx context.Context, accountID strin
return nil
}
+// updatePollCounts updates the vote counts on a poll for the given deleted PollVote model.
+func updatePollCounts(ctx context.Context, tx bun.Tx, deleted *gtsmodel.PollVote) error {
+
+ // Select current poll counts from DB,
+ // taking minimal columns needed to
+ // increment/decrement votes.
+ var poll gtsmodel.Poll
+ switch err := tx.NewSelect().
+ Model(&poll).
+ Column("options", "votes", "voters").
+ Where("? = ?", bun.Ident("id"), deleted.PollID).
+ Scan(ctx); {
+
+ case err == nil:
+ // no issue.
+
+ case errors.Is(err, db.ErrNoEntries):
+ // no poll found,
+ // return here.
+ return nil
+
+ default:
+ // irrecoverable.
+ return err
+ }
+
+ // Decrement votes for these choices.
+ poll.DecrementVotes(deleted.Choices)
+
+ // Finally, update the poll entry.
+ if _, err := tx.NewUpdate().
+ Model(&poll).
+ Column("votes", "voters").
+ Where("? = ?", bun.Ident("id"), deleted.PollID).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
+ return err
+ }
+
+ return nil
+}
+
// newSelectPollVotes returns a new select query for all rows in the poll_votes table with poll_id = pollID.
func newSelectPollVotes(db *bun.DB, pollID string) *bun.SelectQuery {
return db.NewSelect().
diff --git a/internal/db/bundb/poll_test.go b/internal/db/bundb/poll_test.go
index 6bdbdb983..8af9295d9 100644
--- a/internal/db/bundb/poll_test.go
+++ b/internal/db/bundb/poll_test.go
@@ -26,7 +26,6 @@ import (
"github.com/stretchr/testify/suite"
"github.com/superseriousbusiness/gotosocial/internal/db"
- "github.com/superseriousbusiness/gotosocial/internal/gtscontext"
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
"github.com/superseriousbusiness/gotosocial/internal/id"
"github.com/superseriousbusiness/gotosocial/internal/util"
@@ -286,41 +285,6 @@ func (suite *PollTestSuite) TestDeletePoll() {
}
}
-func (suite *PollTestSuite) TestDeletePollVotes() {
- // Create a new context for this test.
- ctx, cncl := context.WithCancel(context.Background())
- defer cncl()
-
- for _, poll := range suite.testPolls {
- // Delete votes associated with poll from database.
- err := suite.db.DeletePollVotes(ctx, poll.ID)
- suite.NoError(err)
-
- // Fetch latest version of poll from database.
- poll, err = suite.db.GetPollByID(
- gtscontext.SetBarebones(ctx),
- poll.ID,
- )
- suite.NoError(err)
-
- // Check that poll counts are all zero.
- suite.Equal(*poll.Voters, 0)
- suite.Equal(make([]int, len(poll.Options)), poll.Votes)
- }
-}
-
-func (suite *PollTestSuite) TestDeletePollVotesNoPoll() {
- // Create a new context for this test.
- ctx, cncl := context.WithCancel(context.Background())
- defer cncl()
-
- // Try to delete votes of nonexistent poll.
- nonPollID := "01HF6V4XWTSZWJ80JNPPDTD4DB"
-
- err := suite.db.DeletePollVotes(ctx, nonPollID)
- suite.NoError(err)
-}
-
func (suite *PollTestSuite) TestDeletePollVotesBy() {
ctx, cncl := context.WithCancel(context.Background())
defer cncl()
diff --git a/internal/db/bundb/relationship_block.go b/internal/db/bundb/relationship_block.go
index 4093bad07..9738970e5 100644
--- a/internal/db/bundb/relationship_block.go
+++ b/internal/db/bundb/relationship_block.go
@@ -105,15 +105,8 @@ func (r *relationshipDB) GetBlocksByIDs(ctx context.Context, ids []string) ([]*g
blocks, err := r.state.Caches.DB.Block.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.Block, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached blocks.
- blocks := make([]*gtsmodel.Block, 0, count)
+ blocks := make([]*gtsmodel.Block, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
@@ -222,94 +215,93 @@ func (r *relationshipDB) PutBlock(ctx context.Context, block *gtsmodel.Block) er
}
func (r *relationshipDB) DeleteBlockByID(ctx context.Context, id string) error {
- // Load block into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := r.GetBlockByID(gtscontext.SetBarebones(ctx), id)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // not an issue.
- err = nil
- }
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.Block
+
+ // Delete block with given ID,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
+ Where("? = ?", bun.Ident("id"), id).
+ Returning("?, ?",
+ bun.Ident("account_id"),
+ bun.Ident("target_account_id"),
+ ).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Drop this now-cached block on return after delete.
- defer r.state.Caches.DB.Block.Invalidate("ID", id)
+ // Invalidate cached block with ID, manually
+ // call invalidate hook in case not cached.
+ r.state.Caches.DB.Block.Invalidate("ID", id)
+ r.state.Caches.OnInvalidateBlock(&deleted)
- // Finally delete block from DB.
- _, err = r.db.NewDelete().
- Table("blocks").
- Where("? = ?", bun.Ident("id"), id).
- Exec(ctx)
- return err
+ return nil
}
func (r *relationshipDB) DeleteBlockByURI(ctx context.Context, uri string) error {
- // Load block into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := r.GetBlockByURI(gtscontext.SetBarebones(ctx), uri)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // not an issue.
- err = nil
- }
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.Block
+
+ // Delete block with given URI,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
+ Where("? = ?", bun.Ident("uri"), uri).
+ Returning("?, ?",
+ bun.Ident("account_id"),
+ bun.Ident("target_account_id"),
+ ).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Drop this now-cached block on return after delete.
- defer r.state.Caches.DB.Block.Invalidate("URI", uri)
+ // Invalidate cached block with URI, manually
+ // call invalidate hook in case not cached.
+ r.state.Caches.DB.Block.Invalidate("URI", uri)
+ r.state.Caches.OnInvalidateBlock(&deleted)
- // Finally delete block from DB.
- _, err = r.db.NewDelete().
- Table("blocks").
- Where("? = ?", bun.Ident("uri"), uri).
- Exec(ctx)
- return err
+ return nil
}
func (r *relationshipDB) DeleteAccountBlocks(ctx context.Context, accountID string) error {
- var blockIDs []string
-
- // Get full list of IDs.
- if err := r.db.NewSelect().
- Column("id").
- Table("blocks").
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted []*gtsmodel.Block
+
+ // Delete all blocks either from
+ // account, or targeting account,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
WhereOr("? = ? OR ? = ?",
bun.Ident("account_id"),
accountID,
bun.Ident("target_account_id"),
accountID,
).
- Scan(ctx, &blockIDs); err != nil {
+ Returning("?, ?",
+ bun.Ident("account_id"),
+ bun.Ident("target_account_id"),
+ ).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- if len(blockIDs) == 0 {
- // Nothing
- // to delete.
- return nil
- }
-
- defer func() {
- // Invalidate all account's incoming / outoing blocks on return.
- r.state.Caches.DB.Block.Invalidate("AccountID", accountID)
- r.state.Caches.DB.Block.Invalidate("TargetAccountID", accountID)
- }()
+ // Invalidate all account's incoming / outoing blocks.
+ r.state.Caches.DB.Block.Invalidate("AccountID", accountID)
+ r.state.Caches.DB.Block.Invalidate("TargetAccountID", accountID)
- // Load all blocks into cache, this *really* isn't great
- // but it is the only way we can ensure we invalidate all
- // related caches correctly (e.g. visibility).
- _, err := r.GetAccountBlocks(ctx, accountID, nil)
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- return err
+ // In case not all blocks were in
+ // cache, manually call invalidate hooks.
+ for _, block := range deleted {
+ r.state.Caches.OnInvalidateBlock(block)
}
- // Finally delete all from DB.
- _, err = r.db.NewDelete().
- Table("blocks").
- Where("? IN (?)", bun.Ident("id"), bun.In(blockIDs)).
- Exec(ctx)
- return err
+ return nil
}
diff --git a/internal/db/bundb/relationship_follow.go b/internal/db/bundb/relationship_follow.go
index 413f3a2af..042d12f37 100644
--- a/internal/db/bundb/relationship_follow.go
+++ b/internal/db/bundb/relationship_follow.go
@@ -20,7 +20,6 @@ package bundb
import (
"context"
"errors"
- "fmt"
"slices"
"time"
@@ -82,15 +81,8 @@ func (r *relationshipDB) GetFollowsByIDs(ctx context.Context, ids []string) ([]*
follows, err := r.state.Caches.DB.Follow.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.Follow, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached follows.
- follows := make([]*gtsmodel.Follow, 0, count)
+ follows := make([]*gtsmodel.Follow, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
@@ -252,139 +244,155 @@ func (r *relationshipDB) UpdateFollow(ctx context.Context, follow *gtsmodel.Foll
})
}
-func (r *relationshipDB) deleteFollow(ctx context.Context, id string) error {
- // Delete the follow itself using the given ID.
+func (r *relationshipDB) DeleteFollow(
+ ctx context.Context,
+ sourceAccountID string,
+ targetAccountID string,
+) error {
+
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.Follow
+ deleted.AccountID = sourceAccountID
+ deleted.TargetAccountID = targetAccountID
+
+ // Delete follow from origin
+ // account, to targeting account,
+ // returning the deleted models.
if _, err := r.db.NewDelete().
- Table("follows").
- Where("? = ?", bun.Ident("id"), id).
- Exec(ctx); err != nil {
+ Model(&deleted).
+ Where("? = ?", bun.Ident("account_id"), sourceAccountID).
+ Where("? = ?", bun.Ident("target_account_id"), targetAccountID).
+ Returning("?", bun.Ident("id")).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Delete every list entry that used this followID.
- if err := r.state.DB.DeleteListEntriesForFollowID(ctx, id); err != nil {
- return fmt.Errorf("deleteFollow: error deleting list entries: %w", err)
+ // Invalidate cached follow with source / target account IDs,
+ // manually calling invalidate hook in case it isn't cached.
+ r.state.Caches.DB.Follow.Invalidate("AccountID,TargetAccountID",
+ sourceAccountID, targetAccountID)
+ r.state.Caches.OnInvalidateFollow(&deleted)
+
+ // Delete every list entry that was created targetting this follow ID.
+ if err := r.state.DB.DeleteAllListEntriesByFollows(ctx, deleted.ID); err != nil {
+ return gtserror.Newf("error deleting list entries: %w", err)
}
return nil
}
-func (r *relationshipDB) DeleteFollow(ctx context.Context, sourceAccountID string, targetAccountID string) error {
- // Load follow into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- follow, err := r.GetFollow(
- gtscontext.SetBarebones(ctx),
- sourceAccountID,
- targetAccountID,
- )
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // Already gone.
- return nil
- }
+func (r *relationshipDB) DeleteFollowByID(ctx context.Context, id string) error {
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.Follow
+ deleted.ID = id
+
+ // Delete follow with given ID,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
+ Where("? = ?", bun.Ident("id"), id).
+ Returning("?, ?",
+ bun.Ident("account_id"),
+ bun.Ident("target_account_id"),
+ ).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Drop this now-cached follow on return after delete.
- defer r.state.Caches.DB.Follow.Invalidate("AccountID,TargetAccountID", sourceAccountID, targetAccountID)
-
- // Finally delete follow from DB.
- return r.deleteFollow(ctx, follow.ID)
-}
+ // Invalidate cached follow with ID, manually
+ // call invalidate hook in case not cached.
+ r.state.Caches.DB.Follow.Invalidate("ID", id)
+ r.state.Caches.OnInvalidateFollow(&deleted)
-func (r *relationshipDB) DeleteFollowByID(ctx context.Context, id string) error {
- // Load follow into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- follow, err := r.GetFollowByID(gtscontext.SetBarebones(ctx), id)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // Already gone.
- return nil
- }
- return err
+ // Delete every list entry that was created targetting this follow ID.
+ if err := r.state.DB.DeleteAllListEntriesByFollows(ctx, id); err != nil {
+ return gtserror.Newf("error deleting list entries: %w", err)
}
- // Drop this now-cached follow on return after delete.
- defer r.state.Caches.DB.Follow.Invalidate("ID", id)
-
- // Finally delete follow from DB.
- return r.deleteFollow(ctx, follow.ID)
+ return nil
}
func (r *relationshipDB) DeleteFollowByURI(ctx context.Context, uri string) error {
- // Load follow into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- follow, err := r.GetFollowByURI(gtscontext.SetBarebones(ctx), uri)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // Already gone.
- return nil
- }
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.Follow
+
+ // Delete follow with given URI,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
+ Where("? = ?", bun.Ident("uri"), uri).
+ Returning("?, ?, ?",
+ bun.Ident("id"),
+ bun.Ident("account_id"),
+ bun.Ident("target_account_id"),
+ ).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Drop this now-cached follow on return after delete.
- defer r.state.Caches.DB.Follow.Invalidate("URI", uri)
+ // Invalidate cached follow with URI, manually
+ // call invalidate hook in case not cached.
+ r.state.Caches.DB.Follow.Invalidate("URI", uri)
+ r.state.Caches.OnInvalidateFollow(&deleted)
- // Finally delete follow from DB.
- return r.deleteFollow(ctx, follow.ID)
+ // Delete every list entry that was created targetting this follow ID.
+ if err := r.state.DB.DeleteAllListEntriesByFollows(ctx, deleted.ID); err != nil {
+ return gtserror.Newf("error deleting list entries: %w", err)
+ }
+
+ return nil
}
func (r *relationshipDB) DeleteAccountFollows(ctx context.Context, accountID string) error {
- var followIDs []string
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted []*gtsmodel.Follow
- // Get full list of IDs.
- if _, err := r.db.
- NewSelect().
- Column("id").
- Table("follows").
+ // Delete all follows either from
+ // account, or targeting account,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
WhereOr("? = ? OR ? = ?",
bun.Ident("account_id"),
accountID,
bun.Ident("target_account_id"),
accountID,
).
- Exec(ctx, &followIDs); err != nil {
+ Returning("?, ?, ?",
+ bun.Ident("id"),
+ bun.Ident("account_id"),
+ bun.Ident("target_account_id"),
+ ).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- if len(followIDs) == 0 {
- // Nothing
- // to delete.
- return nil
- }
-
- defer func() {
- // Invalidate all account's incoming / outoing follows on return.
- r.state.Caches.DB.Follow.Invalidate("AccountID", accountID)
- r.state.Caches.DB.Follow.Invalidate("TargetAccountID", accountID)
- }()
+ // Gather the follow IDs that were deleted for removing related list entries.
+ followIDs := util.Gather(nil, deleted, func(follow *gtsmodel.Follow) string {
+ return follow.ID
+ })
- // Load all follows into cache, this *really* isn't great
- // but it is the only way we can ensure we invalidate all
- // related caches correctly (e.g. visibility).
- _, err := r.GetAccountFollows(ctx, accountID, nil)
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- return err
+ // Delete every list entry that was created targetting any of these follow IDs.
+ if err := r.state.DB.DeleteAllListEntriesByFollows(ctx, followIDs...); err != nil {
+ return gtserror.Newf("error deleting list entries: %w", err)
}
- // Delete all follows from DB.
- _, err = r.db.NewDelete().
- Table("follows").
- Where("? IN (?)", bun.Ident("id"), bun.In(followIDs)).
- Exec(ctx)
- if err != nil {
- return err
- }
+ // Invalidate all account's incoming / outoing follows.
+ r.state.Caches.DB.Follow.Invalidate("AccountID", accountID)
+ r.state.Caches.DB.Follow.Invalidate("TargetAccountID", accountID)
- for _, id := range followIDs {
- // Finally, delete all list entries associated with each follow ID.
- if err := r.state.DB.DeleteListEntriesForFollowID(ctx, id); err != nil {
- return err
- }
+ // In case not all follow were in
+ // cache, manually call invalidate hooks.
+ for _, follow := range deleted {
+ r.state.Caches.OnInvalidateFollow(follow)
}
return nil
diff --git a/internal/db/bundb/relationship_follow_req.go b/internal/db/bundb/relationship_follow_req.go
index 2e058fbbb..fc0ca5c0a 100644
--- a/internal/db/bundb/relationship_follow_req.go
+++ b/internal/db/bundb/relationship_follow_req.go
@@ -81,15 +81,8 @@ func (r *relationshipDB) GetFollowRequestsByIDs(ctx context.Context, ids []strin
follows, err := r.state.Caches.DB.FollowRequest.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.FollowRequest, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached followReqs.
- follows := make([]*gtsmodel.FollowRequest, 0, count)
+ follows := make([]*gtsmodel.FollowRequest, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
@@ -293,124 +286,131 @@ func (r *relationshipDB) RejectFollowRequest(ctx context.Context, sourceAccountI
}, targetAccountID, sourceAccountID)
}
-func (r *relationshipDB) DeleteFollowRequest(ctx context.Context, sourceAccountID string, targetAccountID string) error {
- // Load followreq into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- follow, err := r.GetFollowRequest(
- gtscontext.SetBarebones(ctx),
- sourceAccountID,
- targetAccountID,
- )
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // Already gone.
- return nil
- }
+func (r *relationshipDB) DeleteFollowRequest(
+ ctx context.Context,
+ sourceAccountID string,
+ targetAccountID string,
+) error {
+
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.FollowRequest
+ deleted.AccountID = sourceAccountID
+ deleted.TargetAccountID = targetAccountID
+
+ // Delete all follow reqs either
+ // from account, or targeting account,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
+ Where("? = ?", bun.Ident("account_id"), sourceAccountID).
+ Where("? = ?", bun.Ident("target_account_id"), targetAccountID).
+ Returning("?", bun.Ident("id")).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Drop this now-cached follow request on return after delete.
- defer r.state.Caches.DB.FollowRequest.Invalidate("AccountID,TargetAccountID", sourceAccountID, targetAccountID)
+ // Invalidate cached follow with source / target account IDs,
+ // manually calling invalidate hook in case it isn't cached.
+ r.state.Caches.DB.FollowRequest.Invalidate("AccountID,TargetAccountID",
+ sourceAccountID, targetAccountID)
+ r.state.Caches.OnInvalidateFollowRequest(&deleted)
- // Finally delete followreq from DB.
- _, err = r.db.NewDelete().
- Table("follow_requests").
- Where("? = ?", bun.Ident("id"), follow.ID).
- Exec(ctx)
- return err
+ return nil
}
func (r *relationshipDB) DeleteFollowRequestByID(ctx context.Context, id string) error {
- // Load followreq into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := r.GetFollowRequestByID(gtscontext.SetBarebones(ctx), id)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // not an issue.
- err = nil
- }
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.FollowRequest
+ deleted.ID = id
+
+ // Delete follow with given URI,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
+ Where("? = ?", bun.Ident("id"), id).
+ Returning("?, ?",
+ bun.Ident("account_id"),
+ bun.Ident("target_account_id"),
+ ).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Drop this now-cached follow request on return after delete.
- defer r.state.Caches.DB.FollowRequest.Invalidate("ID", id)
+ // Invalidate cached follow with URI, manually
+ // call invalidate hook in case not cached.
+ r.state.Caches.DB.FollowRequest.Invalidate("ID", id)
+ r.state.Caches.OnInvalidateFollowRequest(&deleted)
- // Finally delete followreq from DB.
- _, err = r.db.NewDelete().
- Table("follow_requests").
- Where("? = ?", bun.Ident("id"), id).
- Exec(ctx)
- return err
+ return nil
}
func (r *relationshipDB) DeleteFollowRequestByURI(ctx context.Context, uri string) error {
- // Load followreq into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := r.GetFollowRequestByURI(gtscontext.SetBarebones(ctx), uri)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // not an issue.
- err = nil
- }
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.FollowRequest
+
+ // Delete follow with given URI,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
+ Where("? = ?", bun.Ident("uri"), uri).
+ Returning("?, ?, ?",
+ bun.Ident("id"),
+ bun.Ident("account_id"),
+ bun.Ident("target_account_id"),
+ ).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Drop this now-cached follow request on return after delete.
- defer r.state.Caches.DB.FollowRequest.Invalidate("URI", uri)
+ // Invalidate cached follow with URI, manually
+ // call invalidate hook in case not cached.
+ r.state.Caches.DB.FollowRequest.Invalidate("URI", uri)
+ r.state.Caches.OnInvalidateFollowRequest(&deleted)
- // Finally delete followreq from DB.
- _, err = r.db.NewDelete().
- Table("follow_requests").
- Where("? = ?", bun.Ident("uri"), uri).
- Exec(ctx)
- return err
+ return nil
}
func (r *relationshipDB) DeleteAccountFollowRequests(ctx context.Context, accountID string) error {
- var followReqIDs []string
-
- // Get full list of IDs.
- if _, err := r.db.
- NewSelect().
- Column("id").
- Table("follow_requests").
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted []*gtsmodel.FollowRequest
+
+ // Delete all follows either from
+ // account, or targeting account,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
WhereOr("? = ? OR ? = ?",
bun.Ident("account_id"),
accountID,
bun.Ident("target_account_id"),
accountID,
).
- Exec(ctx, &followReqIDs); err != nil {
+ Returning("?, ?, ?",
+ bun.Ident("id"),
+ bun.Ident("account_id"),
+ bun.Ident("target_account_id"),
+ ).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- if len(followReqIDs) == 0 {
- // Nothing
- // to delete.
- return nil
- }
+ // Invalidate all account's incoming / outoing follows requests.
+ r.state.Caches.DB.FollowRequest.Invalidate("AccountID", accountID)
+ r.state.Caches.DB.FollowRequest.Invalidate("TargetAccountID", accountID)
- defer func() {
- // Invalidate all account's incoming / outoing follow requests on return.
- r.state.Caches.DB.FollowRequest.Invalidate("AccountID", accountID)
- r.state.Caches.DB.FollowRequest.Invalidate("TargetAccountID", accountID)
- }()
-
- // Load all followreqs into cache, this *really* isn't
- // great but it is the only way we can ensure we invalidate
- // all related caches correctly (e.g. visibility).
- _, err := r.GetAccountFollowRequests(ctx, accountID, nil)
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- return err
+ // In case not all follow were in
+ // cache, manually call invalidate hooks.
+ for _, followReq := range deleted {
+ r.state.Caches.OnInvalidateFollowRequest(followReq)
}
- // Finally delete all from DB.
- _, err = r.db.NewDelete().
- Table("follow_requests").
- Where("? IN (?)", bun.Ident("id"), bun.In(followReqIDs)).
- Exec(ctx)
- return err
+ return nil
}
diff --git a/internal/db/bundb/relationship_mute.go b/internal/db/bundb/relationship_mute.go
index 61b89d323..37d97a64f 100644
--- a/internal/db/bundb/relationship_mute.go
+++ b/internal/db/bundb/relationship_mute.go
@@ -87,15 +87,8 @@ func (r *relationshipDB) getMutesByIDs(ctx context.Context, ids []string) ([]*gt
mutes, err := r.state.Caches.DB.UserMute.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.UserMute, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached mutes.
- mutes := make([]*gtsmodel.UserMute, 0, count)
+ mutes := make([]*gtsmodel.UserMute, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
@@ -209,72 +202,64 @@ func (r *relationshipDB) PutMute(ctx context.Context, mute *gtsmodel.UserMute) e
}
func (r *relationshipDB) DeleteMuteByID(ctx context.Context, id string) error {
- // Load mute into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := r.GetMuteByID(gtscontext.SetBarebones(ctx), id)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // not an issue.
- err = nil
- }
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.UserMute
+
+ // Delete mute with given ID,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
+ Where("? = ?", bun.Ident("id"), id).
+ Returning("?", bun.Ident("account_id")).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Drop this now-cached mute on return after delete.
- defer r.state.Caches.DB.UserMute.Invalidate("ID", id)
+ // Invalidate cached mute with ID, manually
+ // call invalidate hook in case not cached.
+ r.state.Caches.DB.UserMute.Invalidate("ID", id)
+ r.state.Caches.OnInvalidateUserMute(&deleted)
- // Finally delete mute from DB.
- _, err = r.db.NewDelete().
- Table("user_mutes").
- Where("? = ?", bun.Ident("id"), id).
- Exec(ctx)
- return err
+ return nil
}
func (r *relationshipDB) DeleteAccountMutes(ctx context.Context, accountID string) error {
- var muteIDs []string
-
- // Get full list of IDs.
- if err := r.db.NewSelect().
- Column("id").
- Table("user_mutes").
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted []*gtsmodel.UserMute
+
+ // Delete all mutes either from
+ // account, or targeting account,
+ // returning the deleted models.
+ if _, err := r.db.NewDelete().
+ Model(&deleted).
WhereOr("? = ? OR ? = ?",
bun.Ident("account_id"),
accountID,
bun.Ident("target_account_id"),
accountID,
).
- Scan(ctx, &muteIDs); err != nil {
+ Returning("?",
+ bun.Ident("account_id"),
+ ).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- if len(muteIDs) == 0 {
- // Nothing
- // to delete.
- return nil
- }
-
- defer func() {
- // Invalidate all account's incoming / outoing mutes on return.
- r.state.Caches.DB.UserMute.Invalidate("AccountID", accountID)
- r.state.Caches.DB.UserMute.Invalidate("TargetAccountID", accountID)
- }()
+ // Invalidate all account's incoming / outoing user mutes.
+ r.state.Caches.DB.UserMute.Invalidate("AccountID", accountID)
+ r.state.Caches.DB.UserMute.Invalidate("TargetAccountID", accountID)
- // Load all mutes into cache, this *really* isn't great
- // but it is the only way we can ensure we invalidate all
- // related caches correctly (e.g. visibility).
- _, err := r.GetAccountMutes(ctx, accountID, nil)
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- return err
+ // In case not all user mutes were in
+ // cache, manually call invalidate hooks.
+ for _, block := range deleted {
+ r.state.Caches.OnInvalidateUserMute(block)
}
- // Finally delete all from DB.
- _, err = r.db.NewDelete().
- Table("user_mutes").
- Where("? IN (?)", bun.Ident("id"), bun.In(muteIDs)).
- Exec(ctx)
- return err
+ return nil
}
func (r *relationshipDB) GetAccountMutes(
diff --git a/internal/db/bundb/relationship_test.go b/internal/db/bundb/relationship_test.go
index 46a4f1f25..7aa749c90 100644
--- a/internal/db/bundb/relationship_test.go
+++ b/internal/db/bundb/relationship_test.go
@@ -826,10 +826,10 @@ func (suite *RelationshipTestSuite) TestUnfollowExisting() {
suite.NotNil(follow)
followID := follow.ID
- // We should have list entries for this follow.
- listEntries, err := suite.db.GetListEntriesForFollowID(context.Background(), followID)
+ // We should have lists that this follow is a part of.
+ lists, err := suite.db.GetListsContainingFollowID(context.Background(), followID)
suite.NoError(err)
- suite.NotEmpty(listEntries)
+ suite.NotEmpty(lists)
err = suite.db.DeleteFollowByID(context.Background(), followID)
suite.NoError(err)
@@ -838,10 +838,10 @@ func (suite *RelationshipTestSuite) TestUnfollowExisting() {
suite.EqualError(err, db.ErrNoEntries.Error())
suite.Nil(follow)
- // ListEntries pertaining to this follow should be deleted too.
- listEntries, err = suite.db.GetListEntriesForFollowID(context.Background(), followID)
+ // Lists containing this follow should return empty too.
+ lists, err = suite.db.GetListsContainingFollowID(context.Background(), followID)
suite.NoError(err)
- suite.Empty(listEntries)
+ suite.Empty(lists)
}
func (suite *RelationshipTestSuite) TestGetFollowNotExisting() {
diff --git a/internal/db/bundb/report.go b/internal/db/bundb/report.go
index d2096a78a..582584988 100644
--- a/internal/db/bundb/report.go
+++ b/internal/db/bundb/report.go
@@ -248,45 +248,36 @@ func (r *reportDB) PutReport(ctx context.Context, report *gtsmodel.Report) error
})
}
-func (r *reportDB) UpdateReport(ctx context.Context, report *gtsmodel.Report, columns ...string) (*gtsmodel.Report, error) {
+func (r *reportDB) UpdateReport(ctx context.Context, report *gtsmodel.Report, columns ...string) error {
// Update the report's last-updated
report.UpdatedAt = time.Now()
if len(columns) != 0 {
columns = append(columns, "updated_at")
}
- if _, err := r.db.
- NewUpdate().
- Model(report).
- Where("? = ?", bun.Ident("report.id"), report.ID).
- Column(columns...).
- Exec(ctx); err != nil {
- return nil, err
- }
-
- r.state.Caches.DB.Report.Invalidate("ID", report.ID)
- return report, nil
+ return r.state.Caches.DB.Report.Store(report, func() error {
+ _, err := r.db.
+ NewUpdate().
+ Model(report).
+ Where("? = ?", bun.Ident("report.id"), report.ID).
+ Column(columns...).
+ Exec(ctx)
+ return err
+ })
}
func (r *reportDB) DeleteReportByID(ctx context.Context, id string) error {
- defer r.state.Caches.DB.Report.Invalidate("ID", id)
-
- // Load status into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := r.GetReportByID(gtscontext.SetBarebones(ctx), id)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // not an issue.
- err = nil
- }
+ // Delete the report from DB.
+ if _, err := r.db.NewDelete().
+ TableExpr("? AS ?", bun.Ident("reports"), bun.Ident("report")).
+ Where("? = ?", bun.Ident("report.id"), id).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- // Finally delete report from DB.
- _, err = r.db.NewDelete().
- TableExpr("? AS ?", bun.Ident("reports"), bun.Ident("report")).
- Where("? = ?", bun.Ident("report.id"), id).
- Exec(ctx)
- return err
+ // Invalidate any cached report model by ID.
+ r.state.Caches.DB.Report.Invalidate("ID", id)
+
+ return nil
}
diff --git a/internal/db/bundb/report_test.go b/internal/db/bundb/report_test.go
index 1a488c729..57828890d 100644
--- a/internal/db/bundb/report_test.go
+++ b/internal/db/bundb/report_test.go
@@ -202,7 +202,7 @@ func (suite *ReportTestSuite) TestUpdateReport() {
report.ActionTakenByAccountID = suite.testAccounts["admin_account"].ID
report.ActionTakenAt = testrig.TimeMustParse("2022-05-14T12:20:03+02:00")
- if _, err := suite.db.UpdateReport(ctx, report, "action_taken", "action_taken_by_account_id", "action_taken_at"); err != nil {
+ if err := suite.db.UpdateReport(ctx, report, "action_taken", "action_taken_by_account_id", "action_taken_at"); err != nil {
suite.FailNow(err.Error())
}
@@ -228,7 +228,7 @@ func (suite *ReportTestSuite) TestUpdateReportAllColumns() {
report.ActionTakenByAccountID = suite.testAccounts["admin_account"].ID
report.ActionTakenAt = testrig.TimeMustParse("2022-05-14T12:20:03+02:00")
- if _, err := suite.db.UpdateReport(ctx, report); err != nil {
+ if err := suite.db.UpdateReport(ctx, report); err != nil {
suite.FailNow(err.Error())
}
diff --git a/internal/db/bundb/sinbinstatus.go b/internal/db/bundb/sinbinstatus.go
index 5fc368022..dd2c17f67 100644
--- a/internal/db/bundb/sinbinstatus.go
+++ b/internal/db/bundb/sinbinstatus.go
@@ -19,8 +19,10 @@ package bundb
import (
"context"
+ "errors"
"time"
+ "github.com/superseriousbusiness/gotosocial/internal/db"
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
"github.com/superseriousbusiness/gotosocial/internal/state"
"github.com/uptrace/bun"
@@ -110,13 +112,18 @@ func (s *sinBinStatusDB) UpdateSinBinStatus(
}
func (s *sinBinStatusDB) DeleteSinBinStatusByID(ctx context.Context, id string) error {
- // On return ensure status invalidated from cache.
- defer s.state.Caches.DB.SinBinStatus.Invalidate("ID", id)
-
- _, err := s.db.
+ // Delete the status from DB.
+ if _, err := s.db.
NewDelete().
TableExpr("? AS ?", bun.Ident("sin_bin_statuses"), bun.Ident("sin_bin_status")).
Where("? = ?", bun.Ident("sin_bin_status.id"), id).
- Exec(ctx)
- return err
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
+ return err
+ }
+
+ // Invalidate any cached sinbin status model by ID.
+ s.state.Caches.DB.SinBinStatus.Invalidate("ID", id)
+
+ return nil
}
diff --git a/internal/db/bundb/status.go b/internal/db/bundb/status.go
index d0befd52f..5340b63cd 100644
--- a/internal/db/bundb/status.go
+++ b/internal/db/bundb/status.go
@@ -54,15 +54,8 @@ func (s *statusDB) GetStatusesByIDs(ctx context.Context, ids []string) ([]*gtsmo
statuses, err := s.state.Caches.DB.Status.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.Status, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached statuses.
- statuses := make([]*gtsmodel.Status, 0, count)
+ statuses := make([]*gtsmodel.Status, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) status IDs.
@@ -486,24 +479,13 @@ func (s *statusDB) UpdateStatus(ctx context.Context, status *gtsmodel.Status, co
}
func (s *statusDB) DeleteStatusByID(ctx context.Context, id string) error {
- // Load status into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := s.GetStatusByID(
- gtscontext.SetBarebones(ctx),
- id,
- )
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- // NOTE: even if db.ErrNoEntries is returned, we
- // still run the below transaction to ensure related
- // objects are appropriately deleted.
- return err
- }
-
- // On return ensure status invalidated from cache.
- defer s.state.Caches.DB.Status.Invalidate("ID", id)
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.Status
+ deleted.ID = id
- return s.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
+ // Delete status from database and any related links in a transaction.
+ if err := s.db.RunInTx(ctx, nil, func(ctx context.Context, tx bun.Tx) error {
// delete links between this status and any emojis it uses
if _, err := tx.
NewDelete().
@@ -524,26 +506,42 @@ func (s *statusDB) DeleteStatusByID(ctx context.Context, id string) error {
// Delete links between this status
// and any threads it was a part of.
- _, err = tx.
+ if _, err := tx.
NewDelete().
TableExpr("? AS ?", bun.Ident("thread_to_statuses"), bun.Ident("thread_to_status")).
Where("? = ?", bun.Ident("thread_to_status.status_id"), id).
- Exec(ctx)
- if err != nil {
+ Exec(ctx); err != nil {
return err
}
// delete the status itself
if _, err := tx.
NewDelete().
- TableExpr("? AS ?", bun.Ident("statuses"), bun.Ident("status")).
- Where("? = ?", bun.Ident("status.id"), id).
- Exec(ctx); err != nil {
+ Model(&deleted).
+ Where("? = ?", bun.Ident("id"), id).
+ Returning("?, ?, ?, ?, ?",
+ bun.Ident("account_id"),
+ bun.Ident("boost_of_id"),
+ bun.Ident("in_reply_to_id"),
+ bun.Ident("attachments"),
+ bun.Ident("poll_id"),
+ ).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
return nil
- })
+ }); err != nil {
+ return err
+ }
+
+ // Invalidate cached status by its ID, manually
+ // call the invalidate hook in case not cached.
+ s.state.Caches.DB.Status.Invalidate("ID", id)
+ s.state.Caches.OnInvalidateStatus(&deleted)
+
+ return nil
}
func (s *statusDB) GetStatusesUsingEmoji(ctx context.Context, emojiID string) ([]*gtsmodel.Status, error) {
diff --git a/internal/db/bundb/statusbookmark.go b/internal/db/bundb/statusbookmark.go
index 87fb17351..1534050da 100644
--- a/internal/db/bundb/statusbookmark.go
+++ b/internal/db/bundb/statusbookmark.go
@@ -73,15 +73,8 @@ func (s *statusBookmarkDB) GetStatusBookmarksByIDs(ctx context.Context, ids []st
bookmarks, err := s.state.Caches.DB.StatusBookmark.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.StatusBookmark, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached bookmarks.
- bookmarks := make([]*gtsmodel.StatusBookmark, 0, count)
+ bookmarks := make([]*gtsmodel.StatusBookmark, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) bookmarks.
@@ -264,60 +257,86 @@ func (s *statusBookmarkDB) PutStatusBookmark(ctx context.Context, bookmark *gtsm
}
func (s *statusBookmarkDB) DeleteStatusBookmarkByID(ctx context.Context, id string) error {
- _, err := s.db.
- NewDelete().
- Table("status_bookmarks").
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.StatusBookmark
+ deleted.ID = id
+
+ // Delete block with given URI,
+ // returning the deleted models.
+ if _, err := s.db.NewDelete().
+ Model(&deleted).
Where("? = ?", bun.Ident("id"), id).
- Exec(ctx)
- if err != nil {
+ Returning("?", bun.Ident("status_id")).
+ Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
+
+ // Invalidate cached status bookmark by its ID,
+ // manually call invalidate hook in case not cached.
s.state.Caches.DB.StatusBookmark.Invalidate("ID", id)
+ s.state.Caches.OnInvalidateStatusBookmark(&deleted)
+
return nil
}
func (s *statusBookmarkDB) DeleteStatusBookmarks(ctx context.Context, targetAccountID string, originAccountID string) error {
if targetAccountID == "" && originAccountID == "" {
- return errors.New("DeleteBookmarks: one of targetAccountID or originAccountID must be set")
+ return gtserror.New("one of targetAccountID or originAccountID must be set")
}
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted []*gtsmodel.StatusBookmark
+
q := s.db.
NewDelete().
- TableExpr("? AS ?", bun.Ident("status_bookmarks"), bun.Ident("status_bookmark"))
+ Model(&deleted).
+ Returning("?", bun.Ident("status_id"))
if targetAccountID != "" {
- q = q.Where("? = ?", bun.Ident("status_bookmark.target_account_id"), targetAccountID)
- defer s.state.Caches.DB.StatusBookmark.Invalidate("TargetAccountID", targetAccountID)
+ q = q.Where("? = ?", bun.Ident("target_account_id"), targetAccountID)
}
if originAccountID != "" {
- q = q.Where("? = ?", bun.Ident("status_bookmark.account_id"), originAccountID)
- defer s.state.Caches.DB.StatusBookmark.Invalidate("AccountID", originAccountID)
+ q = q.Where("? = ?", bun.Ident("account_id"), originAccountID)
}
- if _, err := q.Exec(ctx); err != nil {
+ if _, err := q.Exec(ctx); err != nil &&
+ !errors.Is(err, db.ErrNoEntries) {
return err
}
- if targetAccountID != "" {
- s.state.Caches.DB.StatusBookmark.Invalidate("TargetAccountID", targetAccountID)
- }
-
- if originAccountID != "" {
- s.state.Caches.DB.StatusBookmark.Invalidate("AccountID", originAccountID)
+ for _, deleted := range deleted {
+ // Invalidate cached status bookmark by status ID,
+ // manually call invalidate hook in case not cached.
+ s.state.Caches.DB.StatusBookmark.Invalidate("StatusID", deleted.StatusID)
+ s.state.Caches.OnInvalidateStatusBookmark(deleted)
}
return nil
}
func (s *statusBookmarkDB) DeleteStatusBookmarksForStatus(ctx context.Context, statusID string) error {
- q := s.db.
- NewDelete().
+ // Delete status bookmarks
+ // from database by status ID.
+ q := s.db.NewDelete().
TableExpr("? AS ?", bun.Ident("status_bookmarks"), bun.Ident("status_bookmark")).
Where("? = ?", bun.Ident("status_bookmark.status_id"), statusID)
if _, err := q.Exec(ctx); err != nil {
return err
}
+
+ // Wrap provided ID in a bookmark
+ // model for calling cache hook.
+ var deleted gtsmodel.StatusBookmark
+ deleted.StatusID = statusID
+
+ // Invalidate cached status bookmark by status ID,
+ // manually call invalidate hook in case not cached.
s.state.Caches.DB.StatusBookmark.Invalidate("StatusID", statusID)
+ s.state.Caches.OnInvalidateStatusBookmark(&deleted)
+
return nil
}
diff --git a/internal/db/bundb/statusfave.go b/internal/db/bundb/statusfave.go
index eb372c24b..cf20fbba3 100644
--- a/internal/db/bundb/statusfave.go
+++ b/internal/db/bundb/statusfave.go
@@ -133,15 +133,8 @@ func (s *statusFaveDB) GetStatusFaves(ctx context.Context, statusID string) ([]*
faves, err := s.state.Caches.DB.StatusFave.LoadIDs("ID",
faveIDs,
func(uncached []string) ([]*gtsmodel.StatusFave, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached faves.
- faves := make([]*gtsmodel.StatusFave, 0, count)
+ faves := make([]*gtsmodel.StatusFave, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) fave IDs.
diff --git a/internal/db/bundb/tag.go b/internal/db/bundb/tag.go
index e6a14c97e..6c3d870f6 100644
--- a/internal/db/bundb/tag.go
+++ b/internal/db/bundb/tag.go
@@ -20,6 +20,7 @@ package bundb
import (
"context"
"errors"
+ "slices"
"strings"
"github.com/superseriousbusiness/gotosocial/internal/db"
@@ -79,15 +80,8 @@ func (t *tagDB) GetTags(ctx context.Context, ids []string) ([]*gtsmodel.Tag, err
tags, err := t.state.Caches.DB.Tag.LoadIDs("ID",
ids,
func(uncached []string) ([]*gtsmodel.Tag, error) {
- // Avoid querying
- // if none uncached.
- count := len(uncached)
- if count == 0 {
- return nil, nil
- }
-
// Preallocate expected length of uncached tags.
- tags := make([]*gtsmodel.Tag, 0, count)
+ tags := make([]*gtsmodel.Tag, 0, len(uncached))
// Perform database query scanning
// the remaining (uncached) IDs.
@@ -148,17 +142,11 @@ func (t *tagDB) GetFollowedTags(ctx context.Context, accountID string, page *pag
if err != nil {
return nil, err
}
-
- tags, err := t.GetTags(ctx, tagIDs)
- if err != nil {
- return nil, err
- }
-
- return tags, nil
+ return t.GetTags(ctx, tagIDs)
}
func (t *tagDB) getTagIDsFollowedByAccount(ctx context.Context, accountID string, page *paging.Page) ([]string, error) {
- return loadPagedIDs(&t.state.Caches.DB.TagIDsFollowedByAccount, accountID, page, func() ([]string, error) {
+ return loadPagedIDs(&t.state.Caches.DB.FollowingTagIDs, ">"+accountID, page, func() ([]string, error) {
var tagIDs []string
// Tag IDs not in cache. Perform DB query.
@@ -178,7 +166,7 @@ func (t *tagDB) getTagIDsFollowedByAccount(ctx context.Context, accountID string
}
func (t *tagDB) getAccountIDsFollowingTag(ctx context.Context, tagID string) ([]string, error) {
- return loadPagedIDs(&t.state.Caches.DB.AccountIDsFollowingTag, tagID, nil, func() ([]string, error) {
+ return loadPagedIDs(&t.state.Caches.DB.FollowingTagIDs, "<"+tagID, nil, func() ([]string, error) {
var accountIDs []string
// Account IDs not in cache. Perform DB query.
@@ -198,18 +186,11 @@ func (t *tagDB) getAccountIDsFollowingTag(ctx context.Context, tagID string) ([]
}
func (t *tagDB) IsAccountFollowingTag(ctx context.Context, accountID string, tagID string) (bool, error) {
- accountTagIDs, err := t.getTagIDsFollowedByAccount(ctx, accountID, nil)
+ followingTagIDs, err := t.getTagIDsFollowedByAccount(ctx, accountID, nil)
if err != nil {
return false, err
}
-
- for _, accountTagID := range accountTagIDs {
- if accountTagID == tagID {
- return true, nil
- }
- }
-
- return false, nil
+ return slices.Contains(followingTagIDs, tagID), nil
}
func (t *tagDB) PutFollowedTag(ctx context.Context, accountID string, tagID string) error {
@@ -234,9 +215,15 @@ func (t *tagDB) PutFollowedTag(ctx context.Context, accountID string, tagID stri
return nil
}
- // Otherwise, this is a new followed tag, so we invalidate caches related to it.
- t.state.Caches.DB.AccountIDsFollowingTag.Invalidate(tagID)
- t.state.Caches.DB.TagIDsFollowedByAccount.Invalidate(accountID)
+ // We updated something, invalidate caches.
+ t.state.Caches.DB.FollowingTagIDs.Invalidate(
+
+ // tag IDs followed by account
+ ">"+accountID,
+
+ // account IDs following tag
+ "<"+tagID,
+ )
return nil
}
@@ -259,9 +246,15 @@ func (t *tagDB) DeleteFollowedTag(ctx context.Context, accountID string, tagID s
return nil
}
- // If we deleted anything, invalidate caches related to it.
- t.state.Caches.DB.AccountIDsFollowingTag.Invalidate(tagID)
- t.state.Caches.DB.TagIDsFollowedByAccount.Invalidate(accountID)
+ // We deleted something, invalidate caches.
+ t.state.Caches.DB.FollowingTagIDs.Invalidate(
+
+ // tag IDs followed by account
+ ">"+accountID,
+
+ // account IDs following tag
+ "<"+tagID,
+ )
return err
}
@@ -278,16 +271,26 @@ func (t *tagDB) DeleteFollowedTagsByAccountID(ctx context.Context, accountID str
return gtserror.Newf("error deleting followed tags for account %s: %w", accountID, err)
}
- // Invalidate account ID caches for the account and those tags.
- t.state.Caches.DB.TagIDsFollowedByAccount.Invalidate(accountID)
- t.state.Caches.DB.AccountIDsFollowingTag.Invalidate(tagIDs...)
+ // Convert tag IDs to the keys
+ // we use for caching tag follow
+ // and following IDs.
+ keys := tagIDs
+ for i := range keys {
+ keys[i] = "<" + keys[i]
+ }
+ keys = append(keys, ">"+accountID)
+
+ // If we deleted anything, invalidate caches with keys.
+ t.state.Caches.DB.FollowingTagIDs.Invalidate(keys...)
return nil
}
func (t *tagDB) GetAccountIDsFollowingTagIDs(ctx context.Context, tagIDs []string) ([]string, error) {
- // Accounts might be following multiple tags in this list, but we only want to return each account once.
- accountIDs := []string{}
+ // Make conservative estimate for no. accounts.
+ accountIDs := make([]string, 0, len(tagIDs))
+
+ // Gather all accounts following tags.
for _, tagID := range tagIDs {
tagAccountIDs, err := t.getAccountIDsFollowingTag(ctx, tagID)
if err != nil {
@@ -295,5 +298,8 @@ func (t *tagDB) GetAccountIDsFollowingTagIDs(ctx context.Context, tagIDs []strin
}
accountIDs = append(accountIDs, tagAccountIDs...)
}
+
+ // Accounts might be following multiple tags in list,
+ // but we only want to return each account once.
return util.Deduplicate(accountIDs), nil
}
diff --git a/internal/db/bundb/timeline.go b/internal/db/bundb/timeline.go
index b2af5583f..bcb7953d4 100644
--- a/internal/db/bundb/timeline.go
+++ b/internal/db/bundb/timeline.go
@@ -70,7 +70,7 @@ func (t *timelineDB) GetHomeTimeline(ctx context.Context, accountID string, maxI
// To take account of exclusive lists, get all of
// this account's lists, so we can filter out follows
// that are in contained in exclusive lists.
- lists, err := t.state.DB.GetListsForAccountID(ctx, accountID)
+ lists, err := t.state.DB.GetListsByAccountID(ctx, accountID)
if err != nil && !errors.Is(err, db.ErrNoEntries) {
return nil, gtserror.Newf("db error getting lists for account %s: %w", accountID, err)
}
@@ -84,9 +84,15 @@ func (t *timelineDB) GetHomeTimeline(ctx context.Context, accountID string, maxI
continue
}
+ // Fetch all follow IDs of the entries ccontained in this list.
+ listFollowIDs, err := t.state.DB.GetFollowIDsInList(ctx, list.ID, nil)
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ return nil, gtserror.Newf("db error getting list entry follow ids: %w", err)
+ }
+
// Exclusive list, index all its follow IDs.
- for _, listEntry := range list.ListEntries {
- ignoreFollowIDs[listEntry.FollowID] = struct{}{}
+ for _, followID := range listFollowIDs {
+ ignoreFollowIDs[followID] = struct{}{}
}
}
@@ -370,30 +376,20 @@ func (t *timelineDB) GetListTimeline(
frontToBack = true
)
- // Fetch all listEntries entries from the database.
- listEntries, err := t.state.DB.GetListEntries(
- // Don't need actual follows
- // for this, just the IDs.
- gtscontext.SetBarebones(ctx),
- listID,
- "", "", "", 0,
+ // Fetch all follow IDs contained in list from DB.
+ followIDs, err := t.state.DB.GetFollowIDsInList(
+ ctx, listID, nil,
)
if err != nil {
- return nil, fmt.Errorf("error getting entries for list %s: %w", listID, err)
+ return nil, fmt.Errorf("error getting follows in list: %w", err)
}
- // If there's no list entries we can't
+ // If there's no list follows we can't
// possibly return anything for this list.
- if len(listEntries) == 0 {
+ if len(followIDs) == 0 {
return make([]*gtsmodel.Status, 0), nil
}
- // Extract just the IDs of each follow.
- followIDs := make([]string, 0, len(listEntries))
- for _, listEntry := range listEntries {
- followIDs = append(followIDs, listEntry.FollowID)
- }
-
// Select target account IDs from follows.
subQ := t.db.
NewSelect().
diff --git a/internal/db/bundb/timeline_test.go b/internal/db/bundb/timeline_test.go
index 4874c2b35..50747b50d 100644
--- a/internal/db/bundb/timeline_test.go
+++ b/internal/db/bundb/timeline_test.go
@@ -184,8 +184,8 @@ func (suite *TimelineTestSuite) TestGetHomeTimelineIgnoreExclusive() {
suite.checkStatuses(s, id.Highest, id.Lowest, 8)
// Remove admin account from the exclusive list.
- listEntryID := suite.testListEntries["local_account_1_list_1_entry_2"].ID
- if err := suite.db.DeleteListEntry(ctx, listEntryID); err != nil {
+ listEntry := suite.testListEntries["local_account_1_list_1_entry_2"]
+ if err := suite.db.DeleteListEntry(ctx, listEntry.ListID, listEntry.FollowID); err != nil {
suite.FailNow(err.Error())
}
diff --git a/internal/db/bundb/tombstone.go b/internal/db/bundb/tombstone.go
index bff4ad839..773702323 100644
--- a/internal/db/bundb/tombstone.go
+++ b/internal/db/bundb/tombstone.go
@@ -67,12 +67,14 @@ func (t *tombstoneDB) PutTombstone(ctx context.Context, tombstone *gtsmodel.Tomb
}
func (t *tombstoneDB) DeleteTombstone(ctx context.Context, id string) error {
- defer t.state.Caches.DB.Tombstone.Invalidate("ID", id)
-
// Delete tombstone from DB.
_, err := t.db.NewDelete().
TableExpr("? AS ?", bun.Ident("tombstones"), bun.Ident("tombstone")).
Where("? = ?", bun.Ident("tombstone.id"), id).
Exec(ctx)
+
+ // Invalidate any cached tombstone by given ID.
+ t.state.Caches.DB.Tombstone.Invalidate("ID", id)
+
return err
}
diff --git a/internal/db/bundb/user.go b/internal/db/bundb/user.go
index 1ca65f016..fc8effa91 100644
--- a/internal/db/bundb/user.go
+++ b/internal/db/bundb/user.go
@@ -19,10 +19,8 @@ package bundb
import (
"context"
- "errors"
"time"
- "github.com/superseriousbusiness/gotosocial/internal/db"
"github.com/superseriousbusiness/gotosocial/internal/gtscontext"
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
@@ -209,26 +207,26 @@ func (u *userDB) UpdateUser(ctx context.Context, user *gtsmodel.User, columns ..
}
func (u *userDB) DeleteUserByID(ctx context.Context, userID string) error {
- defer u.state.Caches.DB.User.Invalidate("ID", userID)
-
- // Load user into cache before attempting a delete,
- // as we need it cached in order to trigger the invalidate
- // callback. This in turn invalidates others.
- _, err := u.GetUserByID(gtscontext.SetBarebones(ctx), userID)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // not an issue.
- err = nil
- }
+ // Gather necessary fields from
+ // deleted for cache invaliation.
+ var deleted gtsmodel.User
+ deleted.ID = userID
+
+ // Delete user from DB.
+ if _, err := u.db.NewDelete().
+ Model(&deleted).
+ Where("? = ?", bun.Ident("id"), userID).
+ Returning("?", bun.Ident("account_id")).
+ Exec(ctx); err != nil {
return err
}
- // Finally delete user from DB.
- _, err = u.db.NewDelete().
- TableExpr("? AS ?", bun.Ident("users"), bun.Ident("user")).
- Where("? = ?", bun.Ident("user.id"), userID).
- Exec(ctx)
- return err
+ // Invalidate cached user by ID, manually
+ // call invalidate hook in case not cached.
+ u.state.Caches.DB.User.Invalidate("ID", userID)
+ u.state.Caches.OnInvalidateUser(&deleted)
+
+ return nil
}
func (u *userDB) PutDeniedUser(ctx context.Context, deniedUser *gtsmodel.DeniedUser) error {
diff --git a/internal/db/list.go b/internal/db/list.go
index a57f0ed23..4ce0ff988 100644
--- a/internal/db/list.go
+++ b/internal/db/list.go
@@ -21,6 +21,7 @@ import (
"context"
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
+ "github.com/superseriousbusiness/gotosocial/internal/paging"
)
type List interface {
@@ -30,11 +31,29 @@ type List interface {
// GetListsByIDs fetches all lists with the provided IDs.
GetListsByIDs(ctx context.Context, ids []string) ([]*gtsmodel.List, error)
- // GetListsForAccountID gets all lists owned by the given accountID.
- GetListsForAccountID(ctx context.Context, accountID string) ([]*gtsmodel.List, error)
+ // GetListsByAccountID gets all lists owned by the given accountID.
+ GetListsByAccountID(ctx context.Context, accountID string) ([]*gtsmodel.List, error)
- // CountListsForAccountID counts the number of lists owned by the given accountID.
- CountListsForAccountID(ctx context.Context, accountID string) (int, error)
+ // CountListsByAccountID counts the number of lists owned by the given accountID.
+ CountListsByAccountID(ctx context.Context, accountID string) (int, error)
+
+ // GetListsContainingFollowID gets all lists that contain the given follow with ID.
+ GetListsContainingFollowID(ctx context.Context, followID string) ([]*gtsmodel.List, error)
+
+ // GetFollowIDsInList returns all the follow IDs contained within given list ID.
+ GetFollowIDsInList(ctx context.Context, listID string, page *paging.Page) ([]string, error)
+
+ // GetFollowsInList returns all the follows contained within given list ID.
+ GetFollowsInList(ctx context.Context, listID string, page *paging.Page) ([]*gtsmodel.Follow, error)
+
+ // GetAccountIDsInList return all the account IDs (follow targets) contained within given list ID.
+ GetAccountIDsInList(ctx context.Context, listID string, page *paging.Page) ([]string, error)
+
+ // GetAccountsInList return all the accounts (follow targets) contained within given list ID.
+ GetAccountsInList(ctx context.Context, listID string, page *paging.Page) ([]*gtsmodel.Account, error)
+
+ // IsAccountInListID returns whether given account with ID is in the list with ID.
+ IsAccountInList(ctx context.Context, listID string, accountID string) (bool, error)
// PopulateList ensures that the list's struct fields are populated.
PopulateList(ctx context.Context, list *gtsmodel.List) error
@@ -49,31 +68,13 @@ type List interface {
// DeleteListByID deletes one list with the given ID.
DeleteListByID(ctx context.Context, id string) error
- // GetListEntryByID gets one list entry with the given ID.
- GetListEntryByID(ctx context.Context, id string) (*gtsmodel.ListEntry, error)
-
- // GetListEntriesyIDs fetches all list entries with the provided IDs.
- GetListEntriesByIDs(ctx context.Context, ids []string) ([]*gtsmodel.ListEntry, error)
-
- // GetListEntries gets list entries from the given listID, using the given parameters.
- GetListEntries(ctx context.Context, listID string, maxID string, sinceID string, minID string, limit int) ([]*gtsmodel.ListEntry, error)
-
- // GetListEntriesForFollowID returns all listEntries that pertain to the given followID.
- GetListEntriesForFollowID(ctx context.Context, followID string) ([]*gtsmodel.ListEntry, error)
-
- // PopulateListEntry ensures that the listEntry's struct fields are populated.
- PopulateListEntry(ctx context.Context, listEntry *gtsmodel.ListEntry) error
-
// PutListEntries inserts a slice of listEntries into the database.
// It uses a transaction to ensure no partial updates.
PutListEntries(ctx context.Context, listEntries []*gtsmodel.ListEntry) error
- // DeleteListEntry deletes one list entry with the given id.
- DeleteListEntry(ctx context.Context, id string) error
-
- // DeleteListEntryForFollowID deletes all list entries with the given followID.
- DeleteListEntriesForFollowID(ctx context.Context, followID string) error
+ // DeleteListEntry deletes the list entry with given list ID and follow ID.
+ DeleteListEntry(ctx context.Context, listID string, followID string) error
- // ListIncludesAccount returns true if the given listID includes the given accountID.
- ListIncludesAccount(ctx context.Context, listID string, accountID string) (bool, error)
+ // DeleteAllListEntryByFollow deletes all list entries with the given followIDs.
+ DeleteAllListEntriesByFollows(ctx context.Context, followIDs ...string) error
}
diff --git a/internal/db/poll.go b/internal/db/poll.go
index ac0229855..88de6bfcd 100644
--- a/internal/db/poll.go
+++ b/internal/db/poll.go
@@ -39,7 +39,8 @@ type Poll interface {
// UpdatePoll updates the Poll in the database, only on selected columns if provided (else, all).
UpdatePoll(ctx context.Context, poll *gtsmodel.Poll, cols ...string) error
- // DeletePollByID deletes the Poll with given ID from the database.
+ // DeletePollByID deletes the Poll with given ID from the
+ // database, along with all its associated poll votes.
DeletePollByID(ctx context.Context, id string) error
// GetPollVoteByID gets the PollVote with given ID from the database.
@@ -57,9 +58,6 @@ type Poll interface {
// PutPollVote puts the given PollVote in the database.
PutPollVote(ctx context.Context, vote *gtsmodel.PollVote) error
- // DeletePollVotes deletes all PollVotes in Poll with given ID from the database.
- DeletePollVotes(ctx context.Context, pollID string) error
-
// DeletePollVoteBy deletes the PollVote in Poll with ID, by account ID, from the database.
DeletePollVoteBy(ctx context.Context, pollID string, accountID string) error
diff --git a/internal/db/relationship.go b/internal/db/relationship.go
index ddc09d67b..e121f07bd 100644
--- a/internal/db/relationship.go
+++ b/internal/db/relationship.go
@@ -68,6 +68,9 @@ type Relationship interface {
// GetFollow retrieves a follow if it exists between source and target accounts.
GetFollow(ctx context.Context, sourceAccountID string, targetAccountID string) (*gtsmodel.Follow, error)
+ // GetFollowsByIDs fetches all follows from database with given IDs.
+ GetFollowsByIDs(ctx context.Context, ids []string) ([]*gtsmodel.Follow, error)
+
// PopulateFollow populates the struct pointers on the given follow.
PopulateFollow(ctx context.Context, follow *gtsmodel.Follow) error
diff --git a/internal/db/report.go b/internal/db/report.go
index 91b368106..605d6d80b 100644
--- a/internal/db/report.go
+++ b/internal/db/report.go
@@ -44,7 +44,7 @@ type Report interface {
// provided, then all columns will be updated.
// updated_at will also be updated, no need to pass this
// as a specific column.
- UpdateReport(ctx context.Context, report *gtsmodel.Report, columns ...string) (*gtsmodel.Report, error)
+ UpdateReport(ctx context.Context, report *gtsmodel.Report, columns ...string) error
// DeleteReportByID deletes report with the given id.
DeleteReportByID(ctx context.Context, id string) error
diff --git a/internal/federation/dereferencing/status.go b/internal/federation/dereferencing/status.go
index a3c1b7371..28e9d0196 100644
--- a/internal/federation/dereferencing/status.go
+++ b/internal/federation/dereferencing/status.go
@@ -826,9 +826,6 @@ func (d *Dereferencer) fetchStatusPoll(
if err := d.state.DB.DeletePollByID(ctx, pollID); err != nil {
return gtserror.Newf("error deleting existing poll from database: %w", err)
}
- if err := d.state.DB.DeletePollVotes(ctx, pollID); err != nil {
- return gtserror.Newf("error deleting existing votes from database: %w", err)
- }
return nil
}
)
diff --git a/internal/gtsmodel/list.go b/internal/gtsmodel/list.go
index f99531ce8..e3b9f9a30 100644
--- a/internal/gtsmodel/list.go
+++ b/internal/gtsmodel/list.go
@@ -27,7 +27,6 @@ type List struct {
Title string `bun:",nullzero,notnull,unique:listaccounttitle"` // Title of this list.
AccountID string `bun:"type:CHAR(26),notnull,nullzero,unique:listaccounttitle"` // Account that created/owns the list
Account *Account `bun:"-"` // Account corresponding to accountID
- ListEntries []*ListEntry `bun:"-"` // Entries contained by this list.
RepliesPolicy RepliesPolicy `bun:",nullzero,notnull,default:'followed'"` // RepliesPolicy for this list.
Exclusive *bool `bun:",nullzero,notnull,default:false"` // Hide posts from members of this list from your home timeline.
}
diff --git a/internal/processing/account/export.go b/internal/processing/account/export.go
index 9954ea225..68cc17b6d 100644
--- a/internal/processing/account/export.go
+++ b/internal/processing/account/export.go
@@ -98,7 +98,7 @@ func (p *Processor) ExportLists(
ctx context.Context,
requester *gtsmodel.Account,
) ([][]string, gtserror.WithCode) {
- lists, err := p.state.DB.GetListsForAccountID(ctx, requester.ID)
+ lists, err := p.state.DB.GetListsByAccountID(ctx, requester.ID)
if err != nil && !errors.Is(err, db.ErrNoEntries) {
err = gtserror.Newf("db error getting lists: %w", err)
return nil, gtserror.NewErrorInternalError(err)
diff --git a/internal/processing/account/lists.go b/internal/processing/account/lists.go
index 1d92bee82..04cf4ca73 100644
--- a/internal/processing/account/lists.go
+++ b/internal/processing/account/lists.go
@@ -30,8 +30,6 @@ import (
"github.com/superseriousbusiness/gotosocial/internal/log"
)
-var noLists = make([]*apimodel.List, 0)
-
// ListsGet returns all lists owned by requestingAccount, which contain a follow for targetAccountID.
func (p *Processor) ListsGet(ctx context.Context, requestingAccount *gtsmodel.Account, targetAccountID string) ([]*apimodel.List, gtserror.WithCode) {
targetAccount, err := p.state.DB.GetAccountByID(ctx, targetAccountID)
@@ -54,52 +52,35 @@ func (p *Processor) ListsGet(ctx context.Context, requestingAccount *gtsmodel.Ac
// Requester has to follow targetAccount
// for them to be in any of their lists.
follow, err := p.state.DB.GetFollow(
+
// Don't populate follow.
gtscontext.SetBarebones(ctx),
requestingAccount.ID,
targetAccountID,
)
if err != nil && !errors.Is(err, db.ErrNoEntries) {
- return nil, gtserror.NewErrorInternalError(fmt.Errorf("db error: %w", err))
+ err := gtserror.Newf("error getting follow: %w", err)
+ return nil, gtserror.NewErrorInternalError(err)
}
if follow == nil {
- return noLists, nil // by definition we know they're in no lists
- }
-
- listEntries, err := p.state.DB.GetListEntriesForFollowID(
- // Don't populate entries.
- gtscontext.SetBarebones(ctx),
- follow.ID,
- )
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- return nil, gtserror.NewErrorInternalError(fmt.Errorf("db error: %w", err))
+ return []*apimodel.List{}, nil
}
- count := len(listEntries)
- if count == 0 {
- return noLists, nil
+ // Get all lists that this follow is an entry within.
+ lists, err := p.state.DB.GetListsContainingFollowID(ctx, follow.ID)
+ if err != nil {
+ err := gtserror.Newf("error getting lists for follow: %w", err)
+ return nil, gtserror.NewErrorInternalError(err)
}
- apiLists := make([]*apimodel.List, 0, count)
- for _, listEntry := range listEntries {
- list, err := p.state.DB.GetListByID(
- // Don't populate list.
- gtscontext.SetBarebones(ctx),
- listEntry.ListID,
- )
-
- if err != nil {
- log.Debugf(ctx, "skipping list %s due to error %q", listEntry.ListID, err)
- continue
- }
-
+ apiLists := make([]*apimodel.List, 0, len(lists))
+ for _, list := range lists {
apiList, err := p.converter.ListToAPIList(ctx, list)
if err != nil {
- log.Debugf(ctx, "skipping list %s due to error %q", listEntry.ListID, err)
+ log.Errorf(ctx, "error converting list: %v", err)
continue
}
-
apiLists = append(apiLists, apiList)
}
diff --git a/internal/processing/admin/report.go b/internal/processing/admin/report.go
index 13b5a9d86..ed34a4e83 100644
--- a/internal/processing/admin/report.go
+++ b/internal/processing/admin/report.go
@@ -142,7 +142,7 @@ func (p *Processor) ReportResolve(ctx context.Context, account *gtsmodel.Account
columns = append(columns, "action_taken")
}
- updatedReport, err := p.state.DB.UpdateReport(ctx, report, columns...)
+ err = p.state.DB.UpdateReport(ctx, report, columns...)
if err != nil {
return nil, gtserror.NewErrorInternalError(err)
}
@@ -156,7 +156,7 @@ func (p *Processor) ReportResolve(ctx context.Context, account *gtsmodel.Account
Target: report.Account,
})
- apimodelReport, err := p.converter.ReportToAdminAPIReport(ctx, updatedReport, account)
+ apimodelReport, err := p.converter.ReportToAdminAPIReport(ctx, report, account)
if err != nil {
return nil, gtserror.NewErrorInternalError(err)
}
diff --git a/internal/processing/common/status.go b/internal/processing/common/status.go
index 3ef643292..a1d432eb0 100644
--- a/internal/processing/common/status.go
+++ b/internal/processing/common/status.go
@@ -189,7 +189,7 @@ func (p *Processor) GetAPIStatus(
// such invalidation will, in that case, be handled by the processor instead.
func (p *Processor) InvalidateTimelinedStatus(ctx context.Context, accountID string, statusID string) error {
// Get lists first + bail if this fails.
- lists, err := p.state.DB.GetListsForAccountID(ctx, accountID)
+ lists, err := p.state.DB.GetListsByAccountID(ctx, accountID)
if err != nil {
return gtserror.Newf("db error getting lists for account %s: %w", accountID, err)
}
diff --git a/internal/processing/list/get.go b/internal/processing/list/get.go
index cdd3c6e0c..b98678eef 100644
--- a/internal/processing/list/get.go
+++ b/internal/processing/list/get.go
@@ -20,7 +20,6 @@ package list
import (
"context"
"errors"
- "fmt"
apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model"
"github.com/superseriousbusiness/gotosocial/internal/db"
@@ -28,7 +27,7 @@ import (
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
"github.com/superseriousbusiness/gotosocial/internal/log"
- "github.com/superseriousbusiness/gotosocial/internal/util"
+ "github.com/superseriousbusiness/gotosocial/internal/paging"
)
// Get returns the api model of one list with the given ID.
@@ -49,16 +48,14 @@ func (p *Processor) Get(ctx context.Context, account *gtsmodel.Account, id strin
// GetAll returns multiple lists created by the given account, sorted by list ID DESC (newest first).
func (p *Processor) GetAll(ctx context.Context, account *gtsmodel.Account) ([]*apimodel.List, gtserror.WithCode) {
- lists, err := p.state.DB.GetListsForAccountID(
+ lists, err := p.state.DB.GetListsByAccountID(
+
// Use barebones ctx; no embedded
// structs necessary for simple GET.
gtscontext.SetBarebones(ctx),
account.ID,
)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- return nil, nil
- }
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
return nil, gtserror.NewErrorInternalError(err)
}
@@ -68,66 +65,23 @@ func (p *Processor) GetAll(ctx context.Context, account *gtsmodel.Account) ([]*a
if errWithCode != nil {
return nil, errWithCode
}
-
apiLists = append(apiLists, apiList)
}
return apiLists, nil
}
-// GetAllListAccounts returns all accounts that are in the given list,
-// owned by the given account. There's no pagination for this endpoint.
-//
-// See https://docs.joinmastodon.org/methods/lists/#query-parameters:
-//
-// Limit: Integer. Maximum number of results. Defaults to 40 accounts.
-// Max 80 accounts. Set to 0 in order to get all accounts without pagination.
-func (p *Processor) GetAllListAccounts(
- ctx context.Context,
- account *gtsmodel.Account,
- listID string,
-) ([]*apimodel.Account, gtserror.WithCode) {
- // Ensure list exists + is owned by requesting account.
- _, errWithCode := p.getList(
- // Use barebones ctx; no embedded
- // structs necessary for this call.
- gtscontext.SetBarebones(ctx),
- account.ID,
- listID,
- )
- if errWithCode != nil {
- return nil, errWithCode
- }
-
- // Get all entries for this list.
- listEntries, err := p.state.DB.GetListEntries(ctx, listID, "", "", "", 0)
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- err = gtserror.Newf("error getting list entries: %w", err)
- return nil, gtserror.NewErrorInternalError(err)
- }
-
- // Extract accounts from list entries + add them to response.
- accounts := make([]*apimodel.Account, 0, len(listEntries))
- p.accountsFromListEntries(ctx, listEntries, func(acc *apimodel.Account) {
- accounts = append(accounts, acc)
- })
-
- return accounts, nil
-}
-
// GetListAccounts returns accounts that are in the given list, owned by the given account.
-// The additional parameters can be used for paging.
+// The additional parameters can be used for paging. Nil page param returns all accounts.
func (p *Processor) GetListAccounts(
ctx context.Context,
account *gtsmodel.Account,
listID string,
- maxID string,
- sinceID string,
- minID string,
- limit int,
+ page *paging.Page,
) (*apimodel.PageableResponse, gtserror.WithCode) {
// Ensure list exists + is owned by requesting account.
_, errWithCode := p.getList(
+
// Use barebones ctx; no embedded
// structs necessary for this call.
gtscontext.SetBarebones(ctx),
@@ -138,71 +92,45 @@ func (p *Processor) GetListAccounts(
return nil, errWithCode
}
- // To know which accounts are in the list,
- // we need to first get requested list entries.
- listEntries, err := p.state.DB.GetListEntries(ctx, listID, maxID, sinceID, minID, limit)
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- err = fmt.Errorf("GetListAccounts: error getting list entries: %w", err)
+ // Get all accounts contained within list.
+ accounts, err := p.state.DB.GetAccountsInList(ctx,
+ listID,
+ page,
+ )
+ if err != nil {
+ err := gtserror.Newf("db error getting accounts in list: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
- count := len(listEntries)
+ // Check for any accounts.
+ count := len(accounts)
if count == 0 {
- // No list entries means no accounts.
- return util.EmptyPageableResponse(), nil
+ return paging.EmptyResponse(), nil
}
var (
+ // Preallocate expected frontend items.
items = make([]interface{}, 0, count)
- // Set next + prev values before filtering and API
- // converting, so caller can still page properly.
- nextMaxIDValue = listEntries[count-1].ID
- prevMinIDValue = listEntries[0].ID
+ // Set paging low / high IDs.
+ lo = accounts[count-1].ID
+ hi = accounts[0].ID
)
- // Extract accounts from list entries + add them to response.
- p.accountsFromListEntries(ctx, listEntries, func(acc *apimodel.Account) {
- items = append(items, acc)
- })
-
- return util.PackagePageableResponse(util.PageableResponseParams{
- Items: items,
- Path: "/api/v1/lists/" + listID + "/accounts",
- NextMaxIDValue: nextMaxIDValue,
- PrevMinIDValue: prevMinIDValue,
- Limit: limit,
- })
-}
-
-func (p *Processor) accountsFromListEntries(
- ctx context.Context,
- listEntries []*gtsmodel.ListEntry,
- appendAcc func(*apimodel.Account),
-) {
- // For each list entry, we want the account it points to.
- // To get this, we need to first get the follow that the
- // list entry pertains to, then extract the target account
- // from that follow.
- //
- // We do paging not by account ID, but by list entry ID.
- for _, listEntry := range listEntries {
- if err := p.state.DB.PopulateListEntry(ctx, listEntry); err != nil {
- log.Errorf(ctx, "error populating list entry: %v", err)
- continue
- }
-
- if err := p.state.DB.PopulateFollow(ctx, listEntry.Follow); err != nil {
- log.Errorf(ctx, "error populating follow: %v", err)
- continue
- }
-
- apiAccount, err := p.converter.AccountToAPIAccountPublic(ctx, listEntry.Follow.TargetAccount)
+ // Convert accounts to frontend.
+ for _, account := range accounts {
+ apiAccount, err := p.converter.AccountToAPIAccountPublic(ctx, account)
if err != nil {
- log.Errorf(ctx, "error converting to public api account: %v", err)
+ log.Errorf(ctx, "error converting to api account: %v", err)
continue
}
-
- appendAcc(apiAccount)
+ items = append(items, apiAccount)
}
+
+ return paging.PackageResponse(paging.ResponseParams{
+ Items: items,
+ Path: "/api/v1/lists/" + listID + "/accounts",
+ Next: page.Next(lo, hi),
+ Prev: page.Prev(lo, hi),
+ }), nil
}
diff --git a/internal/processing/list/updateentries.go b/internal/processing/list/updateentries.go
index 6dcb951a7..c15248f39 100644
--- a/internal/processing/list/updateentries.go
+++ b/internal/processing/list/updateentries.go
@@ -23,73 +23,90 @@ import (
"fmt"
"github.com/superseriousbusiness/gotosocial/internal/db"
+ "github.com/superseriousbusiness/gotosocial/internal/gtscontext"
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
"github.com/superseriousbusiness/gotosocial/internal/id"
+ "github.com/superseriousbusiness/gotosocial/internal/util"
)
// AddToList adds targetAccountIDs to the given list, if valid.
func (p *Processor) AddToList(ctx context.Context, account *gtsmodel.Account, listID string, targetAccountIDs []string) gtserror.WithCode {
+
// Ensure this list exists + account owns it.
- list, errWithCode := p.getList(ctx, account.ID, listID)
+ _, errWithCode := p.getList(ctx, account.ID, listID)
if errWithCode != nil {
return errWithCode
}
- // Pre-assemble list of entries to add. We *could* add these
- // one by one as we iterate through accountIDs, but according
- // to the Mastodon API we should only add them all once we know
- // they're all valid, no partial updates.
- listEntries := make([]*gtsmodel.ListEntry, 0, len(targetAccountIDs))
+ // Get all follows that are entries in list.
+ follows, err := p.state.DB.GetFollowsInList(
+
+ // We only need barebones model.
+ gtscontext.SetBarebones(ctx),
+ listID,
+ nil,
+ )
+ if err != nil {
+ err := gtserror.Newf("error getting list follows: %w", err)
+ return gtserror.NewErrorInternalError(err)
+ }
+
+ // Convert the follows to a hash set containing the target account IDs.
+ inFollows := util.ToSetFunc(follows, func(follow *gtsmodel.Follow) string {
+ return follow.TargetAccountID
+ })
- // Check each targetAccountID is valid.
- // - Follow must exist.
- // - Follow must not already be in the given list.
+ // Preallocate a slice of expected list entries, we specifically
+ // gather and add all the target accounts in one go rather than
+ // individually, to ensure we don't end up with partial updates.
+ entries := make([]*gtsmodel.ListEntry, 0, len(targetAccountIDs))
+
+ // Iterate all the account IDs in given target list.
for _, targetAccountID := range targetAccountIDs {
- // Ensure follow exists.
- follow, err := p.state.DB.GetFollow(ctx, account.ID, targetAccountID)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- err = fmt.Errorf("you do not follow account %s", targetAccountID)
- return gtserror.NewErrorNotFound(err, err.Error())
- }
- return gtserror.NewErrorInternalError(err)
+
+ // Look for follow to target account.
+ if inFollows.Has(targetAccountID) {
+ text := fmt.Sprintf("account %s is already in list %s", targetAccountID, listID)
+ return gtserror.NewErrorUnprocessableEntity(errors.New(text), text)
}
- // Ensure followID not already in list.
- // This particular call to isInList will
- // never error, so just check entryID.
- entryID, _ := isInList(
- list,
- follow.ID,
- func(listEntry *gtsmodel.ListEntry) (string, error) {
- // Looking for the listEntry follow ID.
- return listEntry.FollowID, nil
- },
+ // Get the actual follow to target.
+ follow, err := p.state.DB.GetFollow(
+
+ // We don't need any sub-models.
+ gtscontext.SetBarebones(ctx),
+ account.ID,
+ targetAccountID,
)
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ err := gtserror.Newf("db error getting follow: %w", err)
+ return gtserror.NewErrorInternalError(err)
+ }
- // Empty entryID means entry with given
- // followID wasn't found in the list.
- if entryID != "" {
- err = fmt.Errorf("account with id %s is already in list %s with entryID %s", targetAccountID, listID, entryID)
- return gtserror.NewErrorUnprocessableEntity(err, err.Error())
+ if follow == nil {
+ text := fmt.Sprintf("account %s not currently followed", targetAccountID)
+ return gtserror.NewErrorNotFound(errors.New(text), text)
}
- // Entry wasn't in the list, we can add it.
- listEntries = append(listEntries, &gtsmodel.ListEntry{
+ // Generate new entry for this follow in list.
+ entries = append(entries, &gtsmodel.ListEntry{
ID: id.NewULID(),
ListID: listID,
FollowID: follow.ID,
})
}
- // If we get to here we can assume all
- // entries are valid, so try to add them.
- if err := p.state.DB.PutListEntries(ctx, listEntries); err != nil {
- if errors.Is(err, db.ErrAlreadyExists) {
- err = fmt.Errorf("one or more errors inserting list entries: %w", err)
- return gtserror.NewErrorUnprocessableEntity(err, err.Error())
- }
+ // Add all of the gathered list entries to the database.
+ switch err := p.state.DB.PutListEntries(ctx, entries); {
+ case err == nil:
+
+ case errors.Is(err, db.ErrAlreadyExists):
+ err := gtserror.Newf("conflict adding list entry: %w", err)
+ return gtserror.NewErrorUnprocessableEntity(err)
+
+ default:
+ err := gtserror.Newf("db error inserting list entries: %w", err)
return gtserror.NewErrorInternalError(err)
}
@@ -97,55 +114,61 @@ func (p *Processor) AddToList(ctx context.Context, account *gtsmodel.Account, li
}
// RemoveFromList removes targetAccountIDs from the given list, if valid.
-func (p *Processor) RemoveFromList(ctx context.Context, account *gtsmodel.Account, listID string, targetAccountIDs []string) gtserror.WithCode {
+func (p *Processor) RemoveFromList(
+ ctx context.Context,
+ account *gtsmodel.Account,
+ listID string,
+ targetAccountIDs []string,
+) gtserror.WithCode {
// Ensure this list exists + account owns it.
- list, errWithCode := p.getList(ctx, account.ID, listID)
+ _, errWithCode := p.getList(ctx, account.ID, listID)
if errWithCode != nil {
return errWithCode
}
- // For each targetAccountID, we want to check if
- // a follow with that targetAccountID is in the
- // given list. If it is in there, we want to remove
- // it from the list.
+ // Get all follows that are entries in list.
+ follows, err := p.state.DB.GetFollowsInList(
+
+ // We only need barebones model.
+ gtscontext.SetBarebones(ctx),
+ listID,
+ nil,
+ )
+ if err != nil {
+ err := gtserror.Newf("error getting list follows: %w", err)
+ return gtserror.NewErrorInternalError(err)
+ }
+
+ // Convert the follows to a map keyed by the target account ID.
+ followsMap := util.KeyBy(follows, func(follow *gtsmodel.Follow) string {
+ return follow.TargetAccountID
+ })
+
+ var errs gtserror.MultiError
+
+ // Iterate all the account IDs in given target list.
for _, targetAccountID := range targetAccountIDs {
- // Check if targetAccountID is
- // on a follow in the list.
- entryID, err := isInList(
- list,
- targetAccountID,
- func(listEntry *gtsmodel.ListEntry) (string, error) {
- // We need the follow so populate this
- // entry, if it's not already populated.
- if err := p.state.DB.PopulateListEntry(ctx, listEntry); err != nil {
- return "", err
- }
-
- // Looking for the list entry targetAccountID.
- return listEntry.Follow.TargetAccountID, nil
- },
- )
- // Error may be returned here if there was an issue
- // populating the list entry. We only return on proper
- // DB errors, we can just skip no entry errors.
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- err = fmt.Errorf("error checking if targetAccountID %s was in list %s: %w", targetAccountID, listID, err)
- return gtserror.NewErrorInternalError(err)
- }
+ // Look for follow targetting this account.
+ follow, ok := followsMap[targetAccountID]
- if entryID == "" {
- // There was an errNoEntries or targetAccount
- // wasn't in this list anyway, so we can skip it.
+ if !ok {
+ // not in list.
continue
}
- // TargetAccount was in the list, remove the entry.
- if err := p.state.DB.DeleteListEntry(ctx, entryID); err != nil && !errors.Is(err, db.ErrNoEntries) {
- err = fmt.Errorf("error removing list entry %s from list %s: %w", entryID, listID, err)
- return gtserror.NewErrorInternalError(err)
+ // Delete the list entry containing follow ID in list.
+ err := p.state.DB.DeleteListEntry(ctx, listID, follow.ID)
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ errs.Appendf("error removing list entry: %w", err)
+ continue
}
}
+ // Wrap errors in errWithCode if set.
+ if err := errs.Combine(); err != nil {
+ return gtserror.NewErrorInternalError(err)
+ }
+
return nil
}
diff --git a/internal/processing/list/util.go b/internal/processing/list/util.go
index c5b1e5081..74d148704 100644
--- a/internal/processing/list/util.go
+++ b/internal/processing/list/util.go
@@ -33,18 +33,25 @@ import (
// appropriate errors so caller doesn't need to bother.
func (p *Processor) getList(ctx context.Context, accountID string, listID string) (*gtsmodel.List, gtserror.WithCode) {
list, err := p.state.DB.GetListByID(ctx, listID)
- if err != nil {
- if errors.Is(err, db.ErrNoEntries) {
- // List doesn't seem to exist.
- return nil, gtserror.NewErrorNotFound(err)
- }
- // Real database error.
+ if err != nil && !errors.Is(err, db.ErrNoEntries) {
+ err := gtserror.Newf("db error getting list: %w", err)
return nil, gtserror.NewErrorInternalError(err)
}
+ if list == nil {
+ const text = "list not found"
+ return nil, gtserror.NewErrorNotFound(
+ errors.New(text),
+ text,
+ )
+ }
+
if list.AccountID != accountID {
- err = fmt.Errorf("list with id %s does not belong to account %s", list.ID, accountID)
- return nil, gtserror.NewErrorNotFound(err)
+ const text = "list not found"
+ return nil, gtserror.NewErrorNotFound(
+ errors.New("list does not belong to account"),
+ text,
+ )
}
return list, nil
@@ -60,26 +67,3 @@ func (p *Processor) apiList(ctx context.Context, list *gtsmodel.List) (*apimodel
return apiList, nil
}
-
-// isInList check if thisID is equal to the result of thatID
-// for any entry in the given list.
-//
-// Will return the id of the listEntry if true, empty if false,
-// or an error if the result of thatID returns an error.
-func isInList(
- list *gtsmodel.List,
- thisID string,
- getThatID func(listEntry *gtsmodel.ListEntry) (string, error),
-) (string, error) {
- for _, listEntry := range list.ListEntries {
- thatID, err := getThatID(listEntry)
- if err != nil {
- return "", err
- }
-
- if thisID == thatID {
- return listEntry.ID, nil
- }
- }
- return "", nil
-}
diff --git a/internal/processing/workers/fromclientapi_test.go b/internal/processing/workers/fromclientapi_test.go
index cc8801e1c..d955f0529 100644
--- a/internal/processing/workers/fromclientapi_test.go
+++ b/internal/processing/workers/fromclientapi_test.go
@@ -649,7 +649,8 @@ func (suite *FromClientAPITestSuite) TestProcessCreateStatusListRepliesPolicyLis
}
// Remove turtle from the list.
- if err := testStructs.State.DB.DeleteListEntry(ctx, suite.testListEntries["local_account_1_list_1_entry_1"].ID); err != nil {
+ testEntry := suite.testListEntries["local_account_1_list_1_entry_1"]
+ if err := testStructs.State.DB.DeleteListEntry(ctx, testEntry.ListID, testEntry.FollowID); err != nil {
suite.FailNow(err.Error())
}
diff --git a/internal/processing/workers/surfacetimeline.go b/internal/processing/workers/surfacetimeline.go
index 81544d928..90cb1fed3 100644
--- a/internal/processing/workers/surfacetimeline.go
+++ b/internal/processing/workers/surfacetimeline.go
@@ -21,7 +21,6 @@ import (
"context"
"errors"
- "github.com/superseriousbusiness/gotosocial/internal/db"
statusfilter "github.com/superseriousbusiness/gotosocial/internal/filter/status"
"github.com/superseriousbusiness/gotosocial/internal/filter/usermute"
"github.com/superseriousbusiness/gotosocial/internal/gtscontext"
@@ -63,13 +62,9 @@ func (s *Surface) timelineAndNotifyStatus(ctx context.Context, status *gtsmodel.
})
}
- // Timeline the status for each local follower of this account.
- // This will also handle notifying any followers with notify
- // set to true on their follow.
- homeTimelinedAccountIDs, err := s.timelineAndNotifyStatusForFollowers(ctx, status, follows)
- if err != nil {
- return gtserror.Newf("error timelining status %s for followers: %w", status.ID, err)
- }
+ // Timeline the status for each local follower of this account. This will
+ // also handle notifying any followers with notify set to true on their follow.
+ homeTimelinedAccountIDs := s.timelineAndNotifyStatusForFollowers(ctx, status, follows)
// Timeline the status for each local account who follows a tag used by this status.
if err := s.timelineAndNotifyStatusForTagFollowers(ctx, status, homeTimelinedAccountIDs); err != nil {
@@ -105,12 +100,10 @@ func (s *Surface) timelineAndNotifyStatusForFollowers(
ctx context.Context,
status *gtsmodel.Status,
follows []*gtsmodel.Follow,
-) ([]string, error) {
+) (homeTimelinedAccountIDs []string) {
var (
- errs gtserror.MultiError
- boost = status.BoostOfID != ""
- reply = status.InReplyToURI != ""
- homeTimelinedAccountIDs = []string{}
+ boost = (status.BoostOfID != "")
+ reply = (status.InReplyToURI != "")
)
for _, follow := range follows {
@@ -130,7 +123,7 @@ func (s *Surface) timelineAndNotifyStatusForFollowers(
ctx, follow.Account, status,
)
if err != nil {
- errs.Appendf("error checking status %s hometimelineability: %w", status.ID, err)
+ log.Errorf(ctx, "error checking status home visibility for follow: %v", err)
continue
}
@@ -139,29 +132,36 @@ func (s *Surface) timelineAndNotifyStatusForFollowers(
continue
}
+ // Get relevant filters and mutes for this follow's account.
+ // (note the origin account of the follow is receiver of status).
filters, mutes, err := s.getFiltersAndMutes(ctx, follow.AccountID)
if err != nil {
- errs.Append(err)
+ log.Error(ctx, err)
continue
}
- // Add status to any relevant lists
- // for this follow, if applicable.
- exclusive, listTimelined := s.listTimelineStatusForFollow(
- ctx,
+ // Add status to any relevant lists for this follow, if applicable.
+ listTimelined, exclusive, err := s.listTimelineStatusForFollow(ctx,
status,
follow,
- &errs,
filters,
mutes,
)
+ if err != nil {
+ log.Errorf(ctx, "error list timelining status: %v", err)
+ continue
+ }
- // Add status to home timeline for owner
- // of this follow, if applicable.
- homeTimelined := false
+ var homeTimelined bool
+
+ // If this was timelined into
+ // list with exclusive flag set,
+ // don't add to home timeline.
if !exclusive {
- homeTimelined, err = s.timelineStatus(
- ctx,
+
+ // Add status to home timeline for owner of
+ // this follow (origin account), if applicable.
+ homeTimelined, err = s.timelineStatus(ctx,
s.State.Timelines.Home.IngestOne,
follow.AccountID, // home timelines are keyed by account ID
follow.Account,
@@ -171,10 +171,12 @@ func (s *Surface) timelineAndNotifyStatusForFollowers(
mutes,
)
if err != nil {
- errs.Appendf("error home timelining status: %w", err)
+ log.Errorf(ctx, "error home timelining status: %v", err)
continue
}
+
if homeTimelined {
+ // If hometimelined, add to list of returned account IDs.
homeTimelinedAccountIDs = append(homeTimelinedAccountIDs, follow.AccountID)
}
}
@@ -210,11 +212,12 @@ func (s *Surface) timelineAndNotifyStatusForFollowers(
status.Account,
status.ID,
); err != nil {
- errs.Appendf("error notifying account %s about new status: %w", follow.AccountID, err)
+ log.Errorf(ctx, "error notifying status for account: %v", err)
+ continue
}
}
- return homeTimelinedAccountIDs, errs.Combine()
+ return homeTimelinedAccountIDs
}
// listTimelineStatusForFollow puts the given status
@@ -227,107 +230,59 @@ func (s *Surface) listTimelineStatusForFollow(
ctx context.Context,
status *gtsmodel.Status,
follow *gtsmodel.Follow,
- errs *gtserror.MultiError,
filters []*gtsmodel.Filter,
mutes *usermute.CompiledUserMuteList,
-) (bool, bool) {
- // To put this status in appropriate list timelines,
- // we need to get each listEntry that pertains to
- // this follow. Then, we want to iterate through all
- // those list entries, and add the status to the list
- // that the entry belongs to if it meets criteria for
- // inclusion in the list.
-
- listEntries, err := s.getListEntries(ctx, follow)
- if err != nil {
- errs.Append(err)
- return false, false
- }
- exclusive, err := s.isAnyListExclusive(ctx, listEntries)
+) (timelined bool, exclusive bool, err error) {
+
+ // Get all lists that contain this given follow.
+ lists, err := s.State.DB.GetListsContainingFollowID(
+
+ // We don't need list sub-models.
+ gtscontext.SetBarebones(ctx),
+ follow.ID,
+ )
if err != nil {
- errs.Append(err)
- return false, false
+ return false, false, gtserror.Newf("error getting lists for follow: %w", err)
}
- // Check eligibility for each list entry (if any).
- listTimelined := false
- for _, listEntry := range listEntries {
- eligible, err := s.listEligible(ctx, listEntry, status)
+ for _, list := range lists {
+ // Check whether list is eligible for this status.
+ eligible, err := s.listEligible(ctx, list, status)
if err != nil {
- errs.Appendf("error checking list eligibility: %w", err)
+ log.Errorf(ctx, "error checking list eligibility: %v", err)
continue
}
if !eligible {
- // Don't add this.
continue
}
+ // Update exclusive flag if list is so.
+ exclusive = exclusive || *list.Exclusive
+
// At this point we are certain this status
// should be included in the timeline of the
// list that this list entry belongs to.
- timelined, err := s.timelineStatus(
+ listTimelined, err := s.timelineStatus(
ctx,
s.State.Timelines.List.IngestOne,
- listEntry.ListID, // list timelines are keyed by list ID
+ list.ID, // list timelines are keyed by list ID
follow.Account,
status,
- stream.TimelineList+":"+listEntry.ListID, // key streamType to this specific list
+ stream.TimelineList+":"+list.ID, // key streamType to this specific list
filters,
mutes,
)
if err != nil {
- errs.Appendf("error adding status to timeline for list %s: %w", listEntry.ListID, err)
- // implicit continue
+ log.Errorf(ctx, "error adding status to list timeline: %v", err)
+ continue
}
- listTimelined = listTimelined || timelined
- }
-
- return exclusive, listTimelined
-}
-// getListEntries returns list entries for a given follow.
-func (s *Surface) getListEntries(ctx context.Context, follow *gtsmodel.Follow) ([]*gtsmodel.ListEntry, error) {
- // Get every list entry that targets this follow's ID.
- listEntries, err := s.State.DB.GetListEntriesForFollowID(
- // We only need the list IDs.
- gtscontext.SetBarebones(ctx),
- follow.ID,
- )
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- return nil, gtserror.Newf("DB error getting list entries: %v", err)
+ // Update flag based on if timelined.
+ timelined = timelined || listTimelined
}
- return listEntries, nil
-}
-// isAnyListExclusive determines whether any provided list entry corresponds to an exclusive list.
-func (s *Surface) isAnyListExclusive(ctx context.Context, listEntries []*gtsmodel.ListEntry) (bool, error) {
- if len(listEntries) == 0 {
- return false, nil
- }
-
- listIDs := make([]string, 0, len(listEntries))
- for _, listEntry := range listEntries {
- listIDs = append(listIDs, listEntry.ListID)
- }
- lists, err := s.State.DB.GetListsByIDs(
- // We only need the list exclusive flags.
- gtscontext.SetBarebones(ctx),
- listIDs,
- )
- if err != nil && !errors.Is(err, db.ErrNoEntries) {
- return false, gtserror.Newf("DB error getting lists for list entries: %v", err)
- }
-
- if len(lists) == 0 {
- return false, nil
- }
- for _, list := range lists {
- if *list.Exclusive {
- return true, nil
- }
- }
- return false, nil
+ return timelined, exclusive, nil
}
// getFiltersAndMutes returns an account's filters and mutes.
@@ -341,8 +296,8 @@ func (s *Surface) getFiltersAndMutes(ctx context.Context, accountID string) ([]*
if err != nil {
return nil, nil, gtserror.Newf("couldn't retrieve mutes for account %s: %w", accountID, err)
}
- compiledMutes := usermute.NewCompiledUserMuteList(mutes)
+ compiledMutes := usermute.NewCompiledUserMuteList(mutes)
return filters, compiledMutes, err
}
@@ -351,7 +306,7 @@ func (s *Surface) getFiltersAndMutes(ctx context.Context, accountID string) ([]*
// belongs to, based on the replies policy of the list.
func (s *Surface) listEligible(
ctx context.Context,
- listEntry *gtsmodel.ListEntry,
+ list *gtsmodel.List,
status *gtsmodel.Status,
) (bool, error) {
if status.InReplyToURI == "" {
@@ -366,18 +321,6 @@ func (s *Surface) listEligible(
return false, nil
}
- // Status is a reply to a known account.
- // We need to fetch the list that this
- // entry belongs to, in order to check
- // the list's replies policy.
- list, err := s.State.DB.GetListByID(
- ctx, listEntry.ListID,
- )
- if err != nil {
- err := gtserror.Newf("db error getting list %s: %w", listEntry.ListID, err)
- return false, err
- }
-
switch list.RepliesPolicy {
case gtsmodel.RepliesPolicyNone:
// This list should not show
@@ -390,20 +333,15 @@ func (s *Surface) listEligible(
//
// Check if replied-to account is
// also included in this list.
- includes, err := s.State.DB.ListIncludesAccount(
- ctx,
+ in, err := s.State.DB.IsAccountInList(ctx,
list.ID,
status.InReplyToAccountID,
)
if err != nil {
- err := gtserror.Newf(
- "db error checking if account %s in list %s: %w",
- status.InReplyToAccountID, listEntry.ListID, err,
- )
+ err := gtserror.Newf("db error checking if account in list: %w", err)
return false, err
}
-
- return includes, nil
+ return in, nil
case gtsmodel.RepliesPolicyFollowed:
// This list should show replies
@@ -418,22 +356,14 @@ func (s *Surface) listEligible(
status.InReplyToAccountID,
)
if err != nil {
- err := gtserror.Newf(
- "db error checking if account %s is followed by %s: %w",
- status.InReplyToAccountID, list.AccountID, err,
- )
+ err := gtserror.Newf("db error checking if account followed: %w", err)
return false, err
}
-
return follows, nil
default:
- // HUH??
- err := gtserror.Newf(
- "reply policy '%s' not recognized on list %s",
- list.RepliesPolicy, list.ID,
- )
- return false, err
+ log.Panicf(ctx, "unknown reply policy: %s", list.RepliesPolicy)
+ return false, nil // unreachable code
}
}
@@ -452,6 +382,7 @@ func (s *Surface) timelineStatus(
filters []*gtsmodel.Filter,
mutes *usermute.CompiledUserMuteList,
) (bool, error) {
+
// Ingest status into given timeline using provided function.
if inserted, err := ingest(ctx, timelineID, status); err != nil {
err = gtserror.Newf("error ingesting status %s: %w", status.ID, err)
@@ -461,7 +392,7 @@ func (s *Surface) timelineStatus(
return false, nil
}
- // The status was inserted so stream it to the user.
+ // Convert updated database model to frontend model.
apiStatus, err := s.Converter.StatusToAPIStatus(ctx,
status,
account,
@@ -473,6 +404,8 @@ func (s *Surface) timelineStatus(
err = gtserror.Newf("error converting status %s to frontend representation: %w", status.ID, err)
return true, err
}
+
+ // The status was inserted so stream it to the user.
s.Stream.Update(ctx, account, apiStatus, streamType)
return true, nil
@@ -492,7 +425,8 @@ func (s *Surface) timelineAndNotifyStatusForTagFollowers(
}
if status.BoostOf != nil {
- // Unwrap boost and work with the original status.
+ // Unwrap boost and work
+ // with the original status.
status = status.BoostOf
}
@@ -523,6 +457,7 @@ func (s *Surface) timelineAndNotifyStatusForTagFollowers(
)
}
}
+
return errs.Combine()
}
@@ -667,17 +602,15 @@ func (s *Surface) timelineStatusUpdate(ctx context.Context, status *gtsmodel.Sta
follows = append(follows, &gtsmodel.Follow{
AccountID: status.AccountID,
Account: status.Account,
- Notify: func() *bool { b := false; return &b }(), // Account shouldn't notify itself.
- ShowReblogs: func() *bool { b := true; return &b }(), // Account should show own reblogs.
+ Notify: util.Ptr(false), // Account shouldn't notify itself.
+ ShowReblogs: util.Ptr(true), // Account should show own reblogs.
})
}
- // Push to streams for each local follower of this account.
- homeTimelinedAccountIDs, err := s.timelineStatusUpdateForFollowers(ctx, status, follows)
- if err != nil {
- return gtserror.Newf("error timelining status %s for followers: %w", status.ID, err)
- }
+ // Push updated status to streams for each local follower of this account.
+ homeTimelinedAccountIDs := s.timelineStatusUpdateForFollowers(ctx, status, follows)
+ // Push updated status to streams for each local follower of tags in status, if applicable.
if err := s.timelineStatusUpdateForTagFollowers(ctx, status, homeTimelinedAccountIDs); err != nil {
return gtserror.Newf("error timelining status %s for tag followers: %w", status.ID, err)
}
@@ -695,12 +628,7 @@ func (s *Surface) timelineStatusUpdateForFollowers(
ctx context.Context,
status *gtsmodel.Status,
follows []*gtsmodel.Follow,
-) ([]string, error) {
- var (
- errs gtserror.MultiError
- homeTimelinedAccountIDs = []string{}
- )
-
+) (homeTimelinedAccountIDs []string) {
for _, follow := range follows {
// Check to see if the status is timelineable for this follower,
// taking account of its visibility, who it replies to, and, if
@@ -718,7 +646,7 @@ func (s *Surface) timelineStatusUpdateForFollowers(
ctx, follow.Account, status,
)
if err != nil {
- errs.Appendf("error checking status %s hometimelineability: %w", status.ID, err)
+ log.Errorf(ctx, "error checking status home visibility for follow: %v", err)
continue
}
@@ -727,31 +655,36 @@ func (s *Surface) timelineStatusUpdateForFollowers(
continue
}
+ // Get relevant filters and mutes for this follow's account.
+ // (note the origin account of the follow is receiver of status).
filters, mutes, err := s.getFiltersAndMutes(ctx, follow.AccountID)
if err != nil {
- errs.Append(err)
+ log.Error(ctx, err)
continue
}
- // Add status to any relevant lists
- // for this follow, if applicable.
- exclusive := s.listTimelineStatusUpdateForFollow(
- ctx,
+ // Add status to relevant lists for this follow, if applicable.
+ _, exclusive, err := s.listTimelineStatusUpdateForFollow(ctx,
status,
follow,
- &errs,
filters,
mutes,
)
+ if err != nil {
+ log.Errorf(ctx, "error list timelining status: %v", err)
+ continue
+ }
+ // If this was timelined into
+ // list with exclusive flag set,
+ // don't add to home timeline.
if exclusive {
continue
}
- // Add status to home timeline for owner
- // of this follow, if applicable.
- homeTimelined, err := s.timelineStreamStatusUpdate(
- ctx,
+ // Add status to home timeline for owner of
+ // this follow (origin account), if applicable.
+ homeTimelined, err := s.timelineStreamStatusUpdate(ctx,
follow.Account,
status,
stream.TimelineHome,
@@ -759,15 +692,17 @@ func (s *Surface) timelineStatusUpdateForFollowers(
mutes,
)
if err != nil {
- errs.Appendf("error home timelining status: %w", err)
+ log.Errorf(ctx, "error home timelining status: %v", err)
continue
}
+
if homeTimelined {
+ // If hometimelined, add to list of returned account IDs.
homeTimelinedAccountIDs = append(homeTimelinedAccountIDs, follow.AccountID)
}
}
- return homeTimelinedAccountIDs, errs.Combine()
+ return homeTimelinedAccountIDs
}
// listTimelineStatusUpdateForFollow pushes edits of the given status
@@ -779,58 +714,59 @@ func (s *Surface) listTimelineStatusUpdateForFollow(
ctx context.Context,
status *gtsmodel.Status,
follow *gtsmodel.Follow,
- errs *gtserror.MultiError,
filters []*gtsmodel.Filter,
mutes *usermute.CompiledUserMuteList,
-) bool {
- // To put this status in appropriate list timelines,
- // we need to get each listEntry that pertains to
- // this follow. Then, we want to iterate through all
- // those list entries, and add the status to the list
- // that the entry belongs to if it meets criteria for
- // inclusion in the list.
-
- listEntries, err := s.getListEntries(ctx, follow)
- if err != nil {
- errs.Append(err)
- return false
- }
- exclusive, err := s.isAnyListExclusive(ctx, listEntries)
+) (bool, bool, error) {
+
+ // Get all lists that contain this given follow.
+ lists, err := s.State.DB.GetListsContainingFollowID(
+
+ // We don't need list sub-models.
+ gtscontext.SetBarebones(ctx),
+ follow.ID,
+ )
if err != nil {
- errs.Append(err)
- return false
+ return false, false, gtserror.Newf("error getting lists for follow: %w", err)
}
- // Check eligibility for each list entry (if any).
- for _, listEntry := range listEntries {
- eligible, err := s.listEligible(ctx, listEntry, status)
+ var exclusive, timelined bool
+ for _, list := range lists {
+
+ // Check whether list is eligible for this status.
+ eligible, err := s.listEligible(ctx, list, status)
if err != nil {
- errs.Appendf("error checking list eligibility: %w", err)
+ log.Errorf(ctx, "error checking list eligibility: %v", err)
continue
}
if !eligible {
- // Don't add this.
continue
}
+ // Update exclusive flag if list is so.
+ exclusive = exclusive || *list.Exclusive
+
// At this point we are certain this status
// should be included in the timeline of the
// list that this list entry belongs to.
- if _, err := s.timelineStreamStatusUpdate(
+ listTimelined, err := s.timelineStreamStatusUpdate(
ctx,
follow.Account,
status,
- stream.TimelineList+":"+listEntry.ListID, // key streamType to this specific list
+ stream.TimelineList+":"+list.ID, // key streamType to this specific list
filters,
mutes,
- ); err != nil {
- errs.Appendf("error adding status to timeline for list %s: %w", listEntry.ListID, err)
- // implicit continue
+ )
+ if err != nil {
+ log.Errorf(ctx, "error adding status to list timeline: %v", err)
+ continue
}
+
+ // Update flag based on if timelined.
+ timelined = timelined || listTimelined
}
- return exclusive
+ return timelined, exclusive, nil
}
// timelineStatusUpdate streams the edited status to the user using the
@@ -845,16 +781,31 @@ func (s *Surface) timelineStreamStatusUpdate(
filters []*gtsmodel.Filter,
mutes *usermute.CompiledUserMuteList,
) (bool, error) {
- apiStatus, err := s.Converter.StatusToAPIStatus(ctx, status, account, statusfilter.FilterContextHome, filters, mutes)
- if errors.Is(err, statusfilter.ErrHideStatus) {
+
+ // Convert updated database model to frontend model.
+ apiStatus, err := s.Converter.StatusToAPIStatus(ctx,
+ status,
+ account,
+ statusfilter.FilterContextHome,
+ filters,
+ mutes,
+ )
+
+ switch {
+ case err == nil:
+ // no issue.
+
+ case errors.Is(err, statusfilter.ErrHideStatus):
// Don't put this status in the stream.
return false, nil
+
+ default:
+ return false, gtserror.Newf("error converting status: %w", err)
}
- if err != nil {
- err = gtserror.Newf("error converting status %s to frontend representation: %w", status.ID, err)
- return false, err
- }
+
+ // The status was updated so stream it to the user.
s.Stream.StatusUpdate(ctx, account, apiStatus, streamType)
+
return true, nil
}
diff --git a/internal/processing/workers/util.go b/internal/processing/workers/util.go
index 042f4827c..62ea6c95c 100644
--- a/internal/processing/workers/util.go
+++ b/internal/processing/workers/util.go
@@ -126,11 +126,6 @@ func (u *utils) wipeStatus(
errs.Appendf("error deleting status poll: %w", err)
}
- // Delete any poll votes pointing to this poll ID.
- if err := u.state.DB.DeletePollVotes(ctx, pollID); err != nil {
- errs.Appendf("error deleting status poll votes: %w", err)
- }
-
// Cancel any scheduled expiry task for poll.
_ = u.state.Workers.Scheduler.Cancel(pollID)
}
diff --git a/internal/typeutils/csv.go b/internal/typeutils/csv.go
index 063e31d54..b1e35ef1a 100644
--- a/internal/typeutils/csv.go
+++ b/internal/typeutils/csv.go
@@ -34,16 +34,14 @@ func (c *Converter) AccountToExportStats(
a *gtsmodel.Account,
) (*apimodel.AccountExportStats, error) {
// Ensure account stats populated.
- if a.Stats == nil {
- if err := c.state.DB.PopulateAccountStats(ctx, a); err != nil {
- return nil, gtserror.Newf(
- "error getting stats for account %s: %w",
- a.ID, err,
- )
- }
+ if err := c.state.DB.PopulateAccountStats(ctx, a); err != nil {
+ return nil, gtserror.Newf(
+ "error getting stats for account %s: %w",
+ a.ID, err,
+ )
}
- listsCount, err := c.state.DB.CountListsForAccountID(ctx, a.ID)
+ listsCount, err := c.state.DB.CountListsByAccountID(ctx, a.ID)
if err != nil {
return nil, gtserror.Newf(
"error counting lists for account %s: %w",
@@ -202,6 +200,7 @@ func (c *Converter) ListsToCSV(
ctx context.Context,
lists []*gtsmodel.List,
) ([][]string, error) {
+
// We need to know our own domain for this.
// Try account domain, fall back to host.
thisDomain := config.GetAccountDomain()
@@ -215,41 +214,23 @@ func (c *Converter) ListsToCSV(
// For each item, add a record.
for _, list := range lists {
- for _, entry := range list.ListEntries {
- if entry.Follow == nil {
- // Retrieve follow.
- var err error
- entry.Follow, err = c.state.DB.GetFollowByID(
- ctx,
- entry.FollowID,
- )
- if err != nil {
- return nil, gtserror.Newf(
- "db error getting follow for list entry %s: %w",
- entry.ID, err,
- )
- }
- }
- if entry.Follow.TargetAccount == nil {
- // Retrieve account.
- var err error
- entry.Follow.TargetAccount, err = c.state.DB.GetAccountByID(
- // Barebones is fine here.
- gtscontext.SetBarebones(ctx),
- entry.Follow.TargetAccountID,
- )
- if err != nil {
- return nil, gtserror.Newf(
- "db error getting target account for list entry %s: %w",
- entry.ID, err,
- )
- }
- }
+ // Get all follows contained with this list.
+ follows, err := c.state.DB.GetFollowsInList(ctx,
+ list.ID,
+ nil,
+ )
+ if err != nil {
+ err := gtserror.Newf("db error getting follows for list: %w", err)
+ return nil, err
+ }
+ // Append each follow as CSV record.
+ for _, follow := range follows {
var (
- username = entry.Follow.TargetAccount.Username
- domain = entry.Follow.TargetAccount.Domain
+ // Extract username / domain from target.
+ username = follow.TargetAccount.Username
+ domain = follow.TargetAccount.Domain
)
if domain == "" {
@@ -259,14 +240,16 @@ func (c *Converter) ListsToCSV(
}
records = append(records, []string{
- // List title: eg., Very cool list
+ // List title: e.g.
+ // Very cool list
list.Title,
- // Account address: eg., someone@example.org
- // -- NOTE: without the leading '@'!
+
+ // Account address: e.g.,
+ // someone@example.org
+ // NOTE: without the leading '@'!
username + "@" + domain,
})
}
-
}
return records, nil
diff --git a/internal/util/unique.go b/internal/util/unique.go
index bad553d3f..68c1d1235 100644
--- a/internal/util/unique.go
+++ b/internal/util/unique.go
@@ -17,12 +17,23 @@
package util
+// KeyBy creates a map of T->S, keyed by value returned from key func.
+func KeyBy[S any, T comparable](in []S, key func(S) T) map[T]S {
+ if key == nil {
+ panic("nil func")
+ }
+ m := make(map[T]S, len(in))
+ for _, v := range in {
+ m[key(v)] = v
+ }
+ return m
+}
+
// Set represents a hashmap of only keys,
// useful for deduplication / key checking.
type Set[T comparable] map[T]struct{}
-// ToSet creates a Set[T] from given values,
-// noting that this does not maintain any order.
+// ToSet creates a Set[T] from given values.
func ToSet[T comparable](in []T) Set[T] {
set := make(Set[T], len(in))
for _, v := range in {
@@ -31,8 +42,19 @@ func ToSet[T comparable](in []T) Set[T] {
return set
}
-// FromSet extracts the values from set to slice,
-// noting that this does not maintain any order.
+// ToSetFunc creates a Set[T] from input slice, keys provided by func.
+func ToSetFunc[S any, T comparable](in []S, key func(S) T) Set[T] {
+ if key == nil {
+ panic("nil func")
+ }
+ set := make(Set[T], len(in))
+ for _, v := range in {
+ set[key(v)] = struct{}{}
+ }
+ return set
+}
+
+// FromSet extracts the values from set to slice.
func FromSet[T comparable](in Set[T]) []T {
out := make([]T, len(in))
var i int
diff --git a/test/envparsing.sh b/test/envparsing.sh
index ab01578d6..ac6c2edc0 100755
--- a/test/envparsing.sh
+++ b/test/envparsing.sh
@@ -23,7 +23,6 @@ EXPECT=$(cat << "EOF"
"application-name": "gts",
"bind-address": "127.0.0.1",
"cache": {
- "account-ids-following-tag-mem-ratio": 1,
"account-mem-ratio": 5,
"account-note-mem-ratio": 1,
"account-settings-mem-ratio": 0.1,
@@ -44,11 +43,13 @@ EXPECT=$(cat << "EOF"
"follow-mem-ratio": 2,
"follow-request-ids-mem-ratio": 2,
"follow-request-mem-ratio": 2,
+ "following-tag-ids-mem-ratio": 2,
"in-reply-to-ids-mem-ratio": 3,
"instance-mem-ratio": 1,
"interaction-request-mem-ratio": 1,
- "list-entry-mem-ratio": 2,
+ "list-ids-mem-ratio": 2,
"list-mem-ratio": 1,
+ "listed-ids-mem-ratio": 2,
"marker-mem-ratio": 0.5,
"media-mem-ratio": 4,
"memory-target": 104857600,
@@ -65,7 +66,6 @@ EXPECT=$(cat << "EOF"
"status-fave-ids-mem-ratio": 3,
"status-fave-mem-ratio": 2,
"status-mem-ratio": 5,
- "tag-ids-followed-by-account-mem-ratio": 1,
"tag-mem-ratio": 2,
"thread-mute-mem-ratio": 0.2,
"token-mem-ratio": 0.75,
diff --git a/vendor/codeberg.org/gruf/go-structr/cache.go b/vendor/codeberg.org/gruf/go-structr/cache.go
index 8fcd4fec4..e73db58f8 100644
--- a/vendor/codeberg.org/gruf/go-structr/cache.go
+++ b/vendor/codeberg.org/gruf/go-structr/cache.go
@@ -375,6 +375,11 @@ func (c *Cache[T]) Load(index *Index, keys []Key, load func([]Key) ([]T, error))
// the lock.
unlock()
+ if len(keys) == 0 {
+ // We loaded everything!
+ return values, nil
+ }
+
// Load uncached values.
uncached, err := load(keys)
if err != nil {
diff --git a/vendor/codeberg.org/gruf/go-structr/item.go b/vendor/codeberg.org/gruf/go-structr/item.go
index 97079c378..bf83f1444 100644
--- a/vendor/codeberg.org/gruf/go-structr/item.go
+++ b/vendor/codeberg.org/gruf/go-structr/item.go
@@ -41,7 +41,6 @@ func free_indexed_item(item *indexed_item) {
}
// drop_index will drop the given index entry from item's indexed.
-// note this also handles freeing the index_entry memory (e.g. to pool)
func (i *indexed_item) drop_index(entry *index_entry) {
for x := 0; x < len(i.indexed); x++ {
if i.indexed[x] != entry {
diff --git a/vendor/codeberg.org/gruf/go-structr/runtime.go b/vendor/codeberg.org/gruf/go-structr/runtime.go
index 44fdd74a7..d2bdba380 100644
--- a/vendor/codeberg.org/gruf/go-structr/runtime.go
+++ b/vendor/codeberg.org/gruf/go-structr/runtime.go
@@ -152,8 +152,10 @@ func extract_fields(ptr unsafe.Pointer, fields []struct_field) []unsafe.Pointer
fptr = field.zero
}
+ // Set field ptr.
ptrs[i] = fptr
}
+
return ptrs
}
diff --git a/vendor/modules.txt b/vendor/modules.txt
index adcfcef52..19e346a1d 100644
--- a/vendor/modules.txt
+++ b/vendor/modules.txt
@@ -75,7 +75,7 @@ codeberg.org/gruf/go-storage/disk
codeberg.org/gruf/go-storage/internal
codeberg.org/gruf/go-storage/memory
codeberg.org/gruf/go-storage/s3
-# codeberg.org/gruf/go-structr v0.8.8
+# codeberg.org/gruf/go-structr v0.8.9
## explicit; go 1.21
codeberg.org/gruf/go-structr
# codeberg.org/superseriousbusiness/exif-terminator v0.9.0