mirror of
https://github.com/superseriousbusiness/gotosocial.git
synced 2025-01-14 18:25:32 +00:00
[bugfix] notification types missing from link header (#3571)
* ensure notification types get included in link header query for notifications * fix type query keys
This commit is contained in:
parent
c454b1b488
commit
a444adee97
6 changed files with 87 additions and 103 deletions
|
@ -18,14 +18,13 @@
|
||||||
package notifications
|
package notifications
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"strconv"
|
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
apiutil "github.com/superseriousbusiness/gotosocial/internal/api/util"
|
apiutil "github.com/superseriousbusiness/gotosocial/internal/api/util"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
|
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/oauth"
|
"github.com/superseriousbusiness/gotosocial/internal/oauth"
|
||||||
|
"github.com/superseriousbusiness/gotosocial/internal/paging"
|
||||||
)
|
)
|
||||||
|
|
||||||
// NotificationsGETHandler swagger:operation GET /api/v1/notifications notifications
|
// NotificationsGETHandler swagger:operation GET /api/v1/notifications notifications
|
||||||
|
@ -152,18 +151,6 @@ func (m *Module) NotificationsGETHandler(c *gin.Context) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
limit := 20
|
|
||||||
limitString := c.Query(LimitKey)
|
|
||||||
if limitString != "" {
|
|
||||||
i, err := strconv.ParseInt(limitString, 10, 32)
|
|
||||||
if err != nil {
|
|
||||||
err := fmt.Errorf("error parsing %s: %s", LimitKey, err)
|
|
||||||
apiutil.ErrorHandler(c, gtserror.NewErrorBadRequest(err, err.Error()), m.processor.InstanceGetV1)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
limit = int(i)
|
|
||||||
}
|
|
||||||
|
|
||||||
types, errWithCode := apiutil.ParseNotificationTypes(c.QueryArray(TypesKey))
|
types, errWithCode := apiutil.ParseNotificationTypes(c.QueryArray(TypesKey))
|
||||||
if errWithCode != nil {
|
if errWithCode != nil {
|
||||||
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
|
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
|
||||||
|
@ -176,13 +163,20 @@ func (m *Module) NotificationsGETHandler(c *gin.Context) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
page, errWithCode := paging.ParseIDPage(c,
|
||||||
|
1, // min limit
|
||||||
|
80, // max limit
|
||||||
|
20, // no limit
|
||||||
|
)
|
||||||
|
if errWithCode != nil {
|
||||||
|
apiutil.ErrorHandler(c, errWithCode, m.processor.InstanceGetV1)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
resp, errWithCode := m.processor.Timeline().NotificationsGet(
|
resp, errWithCode := m.processor.Timeline().NotificationsGet(
|
||||||
c.Request.Context(),
|
c.Request.Context(),
|
||||||
authed,
|
authed,
|
||||||
c.Query(MaxIDKey),
|
page,
|
||||||
c.Query(SinceIDKey),
|
|
||||||
c.Query(MinIDKey),
|
|
||||||
limit,
|
|
||||||
types,
|
types,
|
||||||
exclTypes,
|
exclTypes,
|
||||||
)
|
)
|
||||||
|
|
|
@ -26,8 +26,8 @@ import (
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/gtscontext"
|
"github.com/superseriousbusiness/gotosocial/internal/gtscontext"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
|
"github.com/superseriousbusiness/gotosocial/internal/gtserror"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/id"
|
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/log"
|
"github.com/superseriousbusiness/gotosocial/internal/log"
|
||||||
|
"github.com/superseriousbusiness/gotosocial/internal/paging"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/state"
|
"github.com/superseriousbusiness/gotosocial/internal/state"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/util/xslices"
|
"github.com/superseriousbusiness/gotosocial/internal/util/xslices"
|
||||||
"github.com/uptrace/bun"
|
"github.com/uptrace/bun"
|
||||||
|
@ -192,22 +192,19 @@ func (n *notificationDB) PopulateNotification(ctx context.Context, notif *gtsmod
|
||||||
func (n *notificationDB) GetAccountNotifications(
|
func (n *notificationDB) GetAccountNotifications(
|
||||||
ctx context.Context,
|
ctx context.Context,
|
||||||
accountID string,
|
accountID string,
|
||||||
maxID string,
|
page *paging.Page,
|
||||||
sinceID string,
|
|
||||||
minID string,
|
|
||||||
limit int,
|
|
||||||
types []gtsmodel.NotificationType,
|
types []gtsmodel.NotificationType,
|
||||||
excludeTypes []gtsmodel.NotificationType,
|
excludeTypes []gtsmodel.NotificationType,
|
||||||
) ([]*gtsmodel.Notification, error) {
|
) ([]*gtsmodel.Notification, error) {
|
||||||
// Ensure reasonable
|
|
||||||
if limit < 0 {
|
|
||||||
limit = 0
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make educated guess for slice size
|
|
||||||
var (
|
var (
|
||||||
notifIDs = make([]string, 0, limit)
|
// Get paging params.
|
||||||
frontToBack = true
|
minID = page.GetMin()
|
||||||
|
maxID = page.GetMax()
|
||||||
|
limit = page.GetLimit()
|
||||||
|
order = page.GetOrder()
|
||||||
|
|
||||||
|
// Make educated guess for slice size
|
||||||
|
notifIDs = make([]string, 0, limit)
|
||||||
)
|
)
|
||||||
|
|
||||||
q := n.db.
|
q := n.db.
|
||||||
|
@ -215,23 +212,14 @@ func (n *notificationDB) GetAccountNotifications(
|
||||||
TableExpr("? AS ?", bun.Ident("notifications"), bun.Ident("notification")).
|
TableExpr("? AS ?", bun.Ident("notifications"), bun.Ident("notification")).
|
||||||
Column("notification.id")
|
Column("notification.id")
|
||||||
|
|
||||||
if maxID == "" {
|
if maxID != "" {
|
||||||
maxID = id.Highest
|
// Return only notifs LOWER (ie., older) than maxID.
|
||||||
}
|
q = q.Where("? < ?", bun.Ident("notification.id"), maxID)
|
||||||
|
|
||||||
// Return only notifs LOWER (ie., older) than maxID.
|
|
||||||
q = q.Where("? < ?", bun.Ident("notification.id"), maxID)
|
|
||||||
|
|
||||||
if sinceID != "" {
|
|
||||||
// Return only notifs HIGHER (ie., newer) than sinceID.
|
|
||||||
q = q.Where("? > ?", bun.Ident("notification.id"), sinceID)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if minID != "" {
|
if minID != "" {
|
||||||
// Return only notifs HIGHER (ie., newer) than minID.
|
// Return only notifs HIGHER (ie., newer) than minID.
|
||||||
q = q.Where("? > ?", bun.Ident("notification.id"), minID)
|
q = q.Where("? > ?", bun.Ident("notification.id"), minID)
|
||||||
|
|
||||||
frontToBack = false // page up
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(types) > 0 {
|
if len(types) > 0 {
|
||||||
|
@ -251,12 +239,12 @@ func (n *notificationDB) GetAccountNotifications(
|
||||||
q = q.Limit(limit)
|
q = q.Limit(limit)
|
||||||
}
|
}
|
||||||
|
|
||||||
if frontToBack {
|
if order == paging.OrderAscending {
|
||||||
// Page down.
|
|
||||||
q = q.Order("notification.id DESC")
|
|
||||||
} else {
|
|
||||||
// Page up.
|
// Page up.
|
||||||
q = q.Order("notification.id ASC")
|
q = q.Order("notification.id ASC")
|
||||||
|
} else {
|
||||||
|
// Page down.
|
||||||
|
q = q.Order("notification.id DESC")
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := q.Scan(ctx, ¬ifIDs); err != nil {
|
if err := q.Scan(ctx, ¬ifIDs); err != nil {
|
||||||
|
@ -269,11 +257,8 @@ func (n *notificationDB) GetAccountNotifications(
|
||||||
|
|
||||||
// If we're paging up, we still want notifications
|
// If we're paging up, we still want notifications
|
||||||
// to be sorted by ID desc, so reverse ids slice.
|
// to be sorted by ID desc, so reverse ids slice.
|
||||||
// https://zchee.github.io/golang-wiki/SliceTricks/#reversing
|
if order == paging.OrderAscending {
|
||||||
if !frontToBack {
|
slices.Reverse(notifIDs)
|
||||||
for l, r := 0, len(notifIDs)-1; l < r; l, r = l+1, r-1 {
|
|
||||||
notifIDs[l], notifIDs[r] = notifIDs[r], notifIDs[l]
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fetch notification models by their IDs.
|
// Fetch notification models by their IDs.
|
||||||
|
|
|
@ -28,6 +28,7 @@ import (
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/gtscontext"
|
"github.com/superseriousbusiness/gotosocial/internal/gtscontext"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/id"
|
"github.com/superseriousbusiness/gotosocial/internal/id"
|
||||||
|
"github.com/superseriousbusiness/gotosocial/internal/paging"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/util"
|
"github.com/superseriousbusiness/gotosocial/internal/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -92,10 +93,11 @@ func (suite *NotificationTestSuite) TestGetAccountNotificationsWithSpam() {
|
||||||
notifications, err := suite.db.GetAccountNotifications(
|
notifications, err := suite.db.GetAccountNotifications(
|
||||||
gtscontext.SetBarebones(context.Background()),
|
gtscontext.SetBarebones(context.Background()),
|
||||||
testAccount.ID,
|
testAccount.ID,
|
||||||
id.Highest,
|
&paging.Page{
|
||||||
id.Lowest,
|
Min: paging.EitherMinID("", id.Lowest),
|
||||||
"",
|
Max: paging.MaxID(id.Highest),
|
||||||
20,
|
Limit: 20,
|
||||||
|
},
|
||||||
nil,
|
nil,
|
||||||
nil,
|
nil,
|
||||||
)
|
)
|
||||||
|
@ -115,10 +117,11 @@ func (suite *NotificationTestSuite) TestGetAccountNotificationsWithoutSpam() {
|
||||||
notifications, err := suite.db.GetAccountNotifications(
|
notifications, err := suite.db.GetAccountNotifications(
|
||||||
gtscontext.SetBarebones(context.Background()),
|
gtscontext.SetBarebones(context.Background()),
|
||||||
testAccount.ID,
|
testAccount.ID,
|
||||||
id.Highest,
|
&paging.Page{
|
||||||
id.Lowest,
|
Min: paging.EitherMinID("", id.Lowest),
|
||||||
"",
|
Max: paging.MaxID(id.Highest),
|
||||||
20,
|
Limit: 20,
|
||||||
|
},
|
||||||
nil,
|
nil,
|
||||||
nil,
|
nil,
|
||||||
)
|
)
|
||||||
|
@ -140,10 +143,11 @@ func (suite *NotificationTestSuite) TestDeleteNotificationsWithSpam() {
|
||||||
notifications, err := suite.db.GetAccountNotifications(
|
notifications, err := suite.db.GetAccountNotifications(
|
||||||
gtscontext.SetBarebones(context.Background()),
|
gtscontext.SetBarebones(context.Background()),
|
||||||
testAccount.ID,
|
testAccount.ID,
|
||||||
id.Highest,
|
&paging.Page{
|
||||||
id.Lowest,
|
Min: paging.EitherMinID("", id.Lowest),
|
||||||
"",
|
Max: paging.MaxID(id.Highest),
|
||||||
20,
|
Limit: 20,
|
||||||
|
},
|
||||||
nil,
|
nil,
|
||||||
nil,
|
nil,
|
||||||
)
|
)
|
||||||
|
@ -161,10 +165,11 @@ func (suite *NotificationTestSuite) TestDeleteNotificationsWithSpam() {
|
||||||
notifications, err = suite.db.GetAccountNotifications(
|
notifications, err = suite.db.GetAccountNotifications(
|
||||||
gtscontext.SetBarebones(context.Background()),
|
gtscontext.SetBarebones(context.Background()),
|
||||||
testAccount.ID,
|
testAccount.ID,
|
||||||
id.Highest,
|
&paging.Page{
|
||||||
id.Lowest,
|
Min: paging.EitherMinID("", id.Lowest),
|
||||||
"",
|
Max: paging.MaxID(id.Highest),
|
||||||
20,
|
Limit: 20,
|
||||||
|
},
|
||||||
nil,
|
nil,
|
||||||
nil,
|
nil,
|
||||||
)
|
)
|
||||||
|
@ -183,10 +188,11 @@ func (suite *NotificationTestSuite) TestDeleteNotificationsWithTwoAccounts() {
|
||||||
notifications, err := suite.db.GetAccountNotifications(
|
notifications, err := suite.db.GetAccountNotifications(
|
||||||
gtscontext.SetBarebones(context.Background()),
|
gtscontext.SetBarebones(context.Background()),
|
||||||
testAccount.ID,
|
testAccount.ID,
|
||||||
id.Highest,
|
&paging.Page{
|
||||||
id.Lowest,
|
Min: paging.EitherMinID("", id.Lowest),
|
||||||
"",
|
Max: paging.MaxID(id.Highest),
|
||||||
20,
|
Limit: 20,
|
||||||
|
},
|
||||||
nil,
|
nil,
|
||||||
nil,
|
nil,
|
||||||
)
|
)
|
||||||
|
|
|
@ -21,6 +21,7 @@ import (
|
||||||
"context"
|
"context"
|
||||||
|
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
||||||
|
"github.com/superseriousbusiness/gotosocial/internal/paging"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Notification contains functions for creating and getting notifications.
|
// Notification contains functions for creating and getting notifications.
|
||||||
|
@ -29,7 +30,7 @@ type Notification interface {
|
||||||
//
|
//
|
||||||
// Returned notifications will be ordered ID descending (ie., highest/newest to lowest/oldest).
|
// Returned notifications will be ordered ID descending (ie., highest/newest to lowest/oldest).
|
||||||
// If types is empty, *all* notification types will be included.
|
// If types is empty, *all* notification types will be included.
|
||||||
GetAccountNotifications(ctx context.Context, accountID string, maxID string, sinceID string, minID string, limit int, types []gtsmodel.NotificationType, excludeTypes []gtsmodel.NotificationType) ([]*gtsmodel.Notification, error)
|
GetAccountNotifications(ctx context.Context, accountID string, page *paging.Page, types []gtsmodel.NotificationType, excludeTypes []gtsmodel.NotificationType) ([]*gtsmodel.Notification, error)
|
||||||
|
|
||||||
// GetNotificationByID returns one notification according to its id.
|
// GetNotificationByID returns one notification according to its id.
|
||||||
GetNotificationByID(ctx context.Context, id string) (*gtsmodel.Notification, error)
|
GetNotificationByID(ctx context.Context, id string) (*gtsmodel.Notification, error)
|
||||||
|
|
|
@ -21,6 +21,7 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"net/url"
|
||||||
|
|
||||||
apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model"
|
apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/db"
|
"github.com/superseriousbusiness/gotosocial/internal/db"
|
||||||
|
@ -31,26 +32,21 @@ import (
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
"github.com/superseriousbusiness/gotosocial/internal/gtsmodel"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/log"
|
"github.com/superseriousbusiness/gotosocial/internal/log"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/oauth"
|
"github.com/superseriousbusiness/gotosocial/internal/oauth"
|
||||||
|
"github.com/superseriousbusiness/gotosocial/internal/paging"
|
||||||
"github.com/superseriousbusiness/gotosocial/internal/util"
|
"github.com/superseriousbusiness/gotosocial/internal/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (p *Processor) NotificationsGet(
|
func (p *Processor) NotificationsGet(
|
||||||
ctx context.Context,
|
ctx context.Context,
|
||||||
authed *oauth.Auth,
|
authed *oauth.Auth,
|
||||||
maxID string,
|
page *paging.Page,
|
||||||
sinceID string,
|
|
||||||
minID string,
|
|
||||||
limit int,
|
|
||||||
types []gtsmodel.NotificationType,
|
types []gtsmodel.NotificationType,
|
||||||
excludeTypes []gtsmodel.NotificationType,
|
excludeTypes []gtsmodel.NotificationType,
|
||||||
) (*apimodel.PageableResponse, gtserror.WithCode) {
|
) (*apimodel.PageableResponse, gtserror.WithCode) {
|
||||||
notifs, err := p.state.DB.GetAccountNotifications(
|
notifs, err := p.state.DB.GetAccountNotifications(
|
||||||
ctx,
|
ctx,
|
||||||
authed.Account.ID,
|
authed.Account.ID,
|
||||||
maxID,
|
page,
|
||||||
sinceID,
|
|
||||||
minID,
|
|
||||||
limit,
|
|
||||||
types,
|
types,
|
||||||
excludeTypes,
|
excludeTypes,
|
||||||
)
|
)
|
||||||
|
@ -78,22 +74,15 @@ func (p *Processor) NotificationsGet(
|
||||||
compiledMutes := usermute.NewCompiledUserMuteList(mutes)
|
compiledMutes := usermute.NewCompiledUserMuteList(mutes)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
items = make([]interface{}, 0, count)
|
items = make([]interface{}, 0, count)
|
||||||
nextMaxIDValue string
|
|
||||||
prevMinIDValue string
|
// Get the lowest and highest
|
||||||
|
// ID values, used for paging.
|
||||||
|
lo = notifs[count-1].ID
|
||||||
|
hi = notifs[0].ID
|
||||||
)
|
)
|
||||||
|
|
||||||
for i, n := range notifs {
|
for _, n := range notifs {
|
||||||
// Set next + prev values before filtering and API
|
|
||||||
// converting, so caller can still page properly.
|
|
||||||
if i == count-1 {
|
|
||||||
nextMaxIDValue = n.ID
|
|
||||||
}
|
|
||||||
|
|
||||||
if i == 0 {
|
|
||||||
prevMinIDValue = n.ID
|
|
||||||
}
|
|
||||||
|
|
||||||
visible, err := p.notifVisible(ctx, n, authed.Account)
|
visible, err := p.notifVisible(ctx, n, authed.Account)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Debugf(ctx, "skipping notification %s because of an error checking notification visibility: %v", n.ID, err)
|
log.Debugf(ctx, "skipping notification %s because of an error checking notification visibility: %v", n.ID, err)
|
||||||
|
@ -115,13 +104,22 @@ func (p *Processor) NotificationsGet(
|
||||||
items = append(items, item)
|
items = append(items, item)
|
||||||
}
|
}
|
||||||
|
|
||||||
return util.PackagePageableResponse(util.PageableResponseParams{
|
// Build type query string.
|
||||||
Items: items,
|
query := make(url.Values)
|
||||||
Path: "api/v1/notifications",
|
for _, typ := range types {
|
||||||
NextMaxIDValue: nextMaxIDValue,
|
query.Add("types[]", typ.String())
|
||||||
PrevMinIDValue: prevMinIDValue,
|
}
|
||||||
Limit: limit,
|
for _, typ := range excludeTypes {
|
||||||
})
|
query.Add("exclude_types[]", typ.String())
|
||||||
|
}
|
||||||
|
|
||||||
|
return paging.PackageResponse(paging.ResponseParams{
|
||||||
|
Items: items,
|
||||||
|
Path: "/api/v1/notifications",
|
||||||
|
Next: page.Next(lo, hi),
|
||||||
|
Prev: page.Prev(lo, hi),
|
||||||
|
Query: query,
|
||||||
|
}), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Processor) NotificationGet(ctx context.Context, account *gtsmodel.Account, targetNotifID string) (*apimodel.Notification, gtserror.WithCode) {
|
func (p *Processor) NotificationGet(ctx context.Context, account *gtsmodel.Account, targetNotifID string) (*apimodel.Notification, gtserror.WithCode) {
|
||||||
|
|
|
@ -89,7 +89,7 @@ func (suite *SurfaceNotifyTestSuite) TestSpamNotifs() {
|
||||||
notifs, err := testStructs.State.DB.GetAccountNotifications(
|
notifs, err := testStructs.State.DB.GetAccountNotifications(
|
||||||
gtscontext.SetBarebones(ctx),
|
gtscontext.SetBarebones(ctx),
|
||||||
targetAccount.ID,
|
targetAccount.ID,
|
||||||
"", "", "", 0, nil, nil,
|
nil, nil, nil,
|
||||||
)
|
)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
suite.FailNow(err.Error())
|
suite.FailNow(err.Error())
|
||||||
|
|
Loading…
Reference in a new issue