diff options
| author | 2022-06-08 20:22:49 +0200 | |
|---|---|---|
| committer | 2022-06-08 20:22:49 +0200 | |
| commit | 6f6e89e2715c9ecbadda6b8dbe5227995348dae8 (patch) | |
| tree | da867633cdd72981460baf76615a81ab3390ce9c /internal/api | |
| parent | [frontend] linkify header mascot+title (#633) (diff) | |
| download | gotosocial-6f6e89e2715c9ecbadda6b8dbe5227995348dae8.tar.xz | |
[feature] Add paging via `Link` header for notifications and account statuses (#629)
* test link headers
* page get account statuses properly
* page get notifications
* add util func for packaging timeline responses
* return timelined stuff from accountstatusesget
* rename timeline response
* use new convenience function
* go fmt
Diffstat (limited to 'internal/api')
| -rw-r--r-- | internal/api/client/account/statuses.go | 7 | ||||
| -rw-r--r-- | internal/api/client/account/statuses_test.go | 44 | ||||
| -rw-r--r-- | internal/api/client/favourites/favouritesget.go | 2 | ||||
| -rw-r--r-- | internal/api/client/notification/notificationsget.go | 7 | ||||
| -rw-r--r-- | internal/api/client/timeline/home.go | 2 | ||||
| -rw-r--r-- | internal/api/client/timeline/public.go | 2 | ||||
| -rw-r--r-- | internal/api/model/notification.go | 21 | ||||
| -rw-r--r-- | internal/api/model/timeline.go | 8 | 
8 files changed, 82 insertions, 11 deletions
| diff --git a/internal/api/client/account/statuses.go b/internal/api/client/account/statuses.go index b440e582a..18b551fcc 100644 --- a/internal/api/client/account/statuses.go +++ b/internal/api/client/account/statuses.go @@ -222,12 +222,15 @@ func (m *Module) AccountStatusesGETHandler(c *gin.Context) {  		publicOnly = i  	} -	statuses, errWithCode := m.processor.AccountStatusesGet(c.Request.Context(), authed, targetAcctID, limit, excludeReplies, excludeReblogs, maxID, minID, pinnedOnly, mediaOnly, publicOnly) +	resp, errWithCode := m.processor.AccountStatusesGet(c.Request.Context(), authed, targetAcctID, limit, excludeReplies, excludeReblogs, maxID, minID, pinnedOnly, mediaOnly, publicOnly)  	if errWithCode != nil {  		l.Debugf("error from processor account statuses get: %s", errWithCode)  		c.JSON(errWithCode.Code(), gin.H{"error": errWithCode.Safe()})  		return  	} -	c.JSON(http.StatusOK, statuses) +	if resp.LinkHeader != "" { +		c.Header("Link", resp.LinkHeader) +	} +	c.JSON(http.StatusOK, resp.Items)  } diff --git a/internal/api/client/account/statuses_test.go b/internal/api/client/account/statuses_test.go index 0e95d47fc..1f935896c 100644 --- a/internal/api/client/account/statuses_test.go +++ b/internal/api/client/account/statuses_test.go @@ -37,7 +37,47 @@ type AccountStatusesTestSuite struct {  	AccountStandardTestSuite  } -func (suite *AccountStatusesTestSuite) TestGetStatusesMediaOnly() { +func (suite *AccountStatusesTestSuite) TestGetStatusesPublicOnly() { +	// set up the request +	// we're getting statuses of admin +	targetAccount := suite.testAccounts["admin_account"] +	recorder := httptest.NewRecorder() +	ctx := suite.newContext(recorder, http.MethodGet, nil, fmt.Sprintf("/api/v1/accounts/%s/statuses?limit=20&only_media=false&only_public=true", targetAccount.ID), "") +	ctx.Params = gin.Params{ +		gin.Param{ +			Key:   account.IDKey, +			Value: targetAccount.ID, +		}, +	} + +	// call the handler +	suite.accountModule.AccountStatusesGETHandler(ctx) + +	// 1. we should have OK because our request was valid +	suite.Equal(http.StatusOK, recorder.Code) + +	// 2. we should have no error message in the result body +	result := recorder.Result() +	defer result.Body.Close() + +	// check the response +	b, err := ioutil.ReadAll(result.Body) +	assert.NoError(suite.T(), err) + +	// unmarshal the returned statuses +	apimodelStatuses := []*apimodel.Status{} +	err = json.Unmarshal(b, &apimodelStatuses) +	suite.NoError(err) +	suite.NotEmpty(apimodelStatuses) + +	for _, s := range apimodelStatuses { +		suite.Equal(apimodel.VisibilityPublic, s.Visibility) +	} + +	suite.Equal(`<http://localhost:8080/api/v1/accounts/01F8MH17FWEB39HZJ76B6VXSKF/statuses?limit=20&max_id=01F8MH75CBF9JFX4ZAD54N0W0R&exclude_replies=false&exclude_reblogs=false&pinned_only=false&only_media=false&only_public=true>; rel="next", <http://localhost:8080/api/v1/accounts/01F8MH17FWEB39HZJ76B6VXSKF/statuses?limit=20&min_id=01G36SF3V6Y6V5BF9P4R7PQG7G&exclude_replies=false&exclude_reblogs=false&pinned_only=false&only_media=false&only_public=true>; rel="prev"`, result.Header.Get("link")) +} + +func (suite *AccountStatusesTestSuite) TestGetStatusesPublicOnlyMediaOnly() {  	// set up the request  	// we're getting statuses of admin  	targetAccount := suite.testAccounts["admin_account"] @@ -74,6 +114,8 @@ func (suite *AccountStatusesTestSuite) TestGetStatusesMediaOnly() {  		suite.NotEmpty(s.MediaAttachments)  		suite.Equal(apimodel.VisibilityPublic, s.Visibility)  	} + +	suite.Equal(`<http://localhost:8080/api/v1/accounts/01F8MH17FWEB39HZJ76B6VXSKF/statuses?limit=20&max_id=01F8MH75CBF9JFX4ZAD54N0W0R&exclude_replies=false&exclude_reblogs=false&pinned_only=false&only_media=true&only_public=true>; rel="next", <http://localhost:8080/api/v1/accounts/01F8MH17FWEB39HZJ76B6VXSKF/statuses?limit=20&min_id=01F8MH75CBF9JFX4ZAD54N0W0R&exclude_replies=false&exclude_reblogs=false&pinned_only=false&only_media=true&only_public=true>; rel="prev"`, result.Header.Get("link"))  }  func TestAccountStatusesTestSuite(t *testing.T) { diff --git a/internal/api/client/favourites/favouritesget.go b/internal/api/client/favourites/favouritesget.go index d112e9b95..5a317b2ea 100644 --- a/internal/api/client/favourites/favouritesget.go +++ b/internal/api/client/favourites/favouritesget.go @@ -61,5 +61,5 @@ func (m *Module) FavouritesGETHandler(c *gin.Context) {  	if resp.LinkHeader != "" {  		c.Header("Link", resp.LinkHeader)  	} -	c.JSON(http.StatusOK, resp.Statuses) +	c.JSON(http.StatusOK, resp.Items)  } diff --git a/internal/api/client/notification/notificationsget.go b/internal/api/client/notification/notificationsget.go index 0a56ee80b..b6f7cdd01 100644 --- a/internal/api/client/notification/notificationsget.go +++ b/internal/api/client/notification/notificationsget.go @@ -74,12 +74,15 @@ func (m *Module) NotificationsGETHandler(c *gin.Context) {  		sinceID = sinceIDString  	} -	notifs, errWithCode := m.processor.NotificationsGet(c.Request.Context(), authed, limit, maxID, sinceID) +	resp, errWithCode := m.processor.NotificationsGet(c.Request.Context(), authed, limit, maxID, sinceID)  	if errWithCode != nil {  		l.Debugf("error processing notifications get: %s", errWithCode.Error())  		c.JSON(errWithCode.Code(), gin.H{"error": errWithCode.Safe()})  		return  	} -	c.JSON(http.StatusOK, notifs) +	if resp.LinkHeader != "" { +		c.Header("Link", resp.LinkHeader) +	} +	c.JSON(http.StatusOK, resp.Items)  } diff --git a/internal/api/client/timeline/home.go b/internal/api/client/timeline/home.go index dfe1a0db0..47bc2943e 100644 --- a/internal/api/client/timeline/home.go +++ b/internal/api/client/timeline/home.go @@ -171,5 +171,5 @@ func (m *Module) HomeTimelineGETHandler(c *gin.Context) {  	if resp.LinkHeader != "" {  		c.Header("Link", resp.LinkHeader)  	} -	c.JSON(http.StatusOK, resp.Statuses) +	c.JSON(http.StatusOK, resp.Items)  } diff --git a/internal/api/client/timeline/public.go b/internal/api/client/timeline/public.go index 075d9267c..06da6e947 100644 --- a/internal/api/client/timeline/public.go +++ b/internal/api/client/timeline/public.go @@ -171,5 +171,5 @@ func (m *Module) PublicTimelineGETHandler(c *gin.Context) {  	if resp.LinkHeader != "" {  		c.Header("Link", resp.LinkHeader)  	} -	c.JSON(http.StatusOK, resp.Statuses) +	c.JSON(http.StatusOK, resp.Items)  } diff --git a/internal/api/model/notification.go b/internal/api/model/notification.go index 8f929cab4..efcd2431b 100644 --- a/internal/api/model/notification.go +++ b/internal/api/model/notification.go @@ -43,3 +43,24 @@ type Notification struct {  	// Status that was the object of the notification, e.g. in mentions, reblogs, favourites, or polls.  	Status *Status `json:"status,omitempty"`  } + +/* +	The below functions are added onto the apimodel notification so that it satisfies +	the Timelineable interface in internal/timeline. +*/ + +func (n *Notification) GetID() string { +	return n.ID +} + +func (n *Notification) GetAccountID() string { +	return "" +} + +func (n *Notification) GetBoostOfID() string { +	return "" +} + +func (n *Notification) GetBoostOfAccountID() string { +	return "" +} diff --git a/internal/api/model/timeline.go b/internal/api/model/timeline.go index 66a5263b8..71d839ed2 100644 --- a/internal/api/model/timeline.go +++ b/internal/api/model/timeline.go @@ -18,9 +18,11 @@  package model -// StatusTimelineResponse wraps a slice of statuses, ready to be serialized, along with the Link +import "github.com/superseriousbusiness/gotosocial/internal/timeline" + +// TimelineResponse wraps a slice of timelineables, ready to be serialized, along with the Link  // header for the previous and next queries, to be returned to the client. -type StatusTimelineResponse struct { -	Statuses   []*Status +type TimelineResponse struct { +	Items      []timeline.Timelineable  	LinkHeader string  } | 
