mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-10-30 17:22:26 -05:00 
			
		
		
		
	
		
			
	
	
		
			254 lines
		
	
	
	
		
			7.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
		
		
			
		
	
	
			254 lines
		
	
	
	
		
			7.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
|  | // GoToSocial | ||
|  | // Copyright (C) GoToSocial Authors admin@gotosocial.org | ||
|  | // SPDX-License-Identifier: AGPL-3.0-or-later | ||
|  | // | ||
|  | // This program is free software: you can redistribute it and/or modify | ||
|  | // it under the terms of the GNU Affero General Public License as published by | ||
|  | // the Free Software Foundation, either version 3 of the License, or | ||
|  | // (at your option) any later version. | ||
|  | // | ||
|  | // This program is distributed in the hope that it will be useful, | ||
|  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
|  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||
|  | // GNU Affero General Public License for more details. | ||
|  | // | ||
|  | // You should have received a copy of the GNU Affero General Public License | ||
|  | // along with this program.  If not, see <http://www.gnu.org/licenses/>. | ||
|  | 
 | ||
|  | package notifications_test | ||
|  | 
 | ||
|  | import ( | ||
|  | 	"context" | ||
|  | 	"encoding/json" | ||
|  | 	"io" | ||
|  | 	"net/http" | ||
|  | 	"net/http/httptest" | ||
|  | 	"net/url" | ||
|  | 	"strconv" | ||
|  | 	"testing" | ||
|  | 
 | ||
|  | 	"github.com/stretchr/testify/suite" | ||
|  | 	"github.com/superseriousbusiness/gotosocial/internal/api/client/notifications" | ||
|  | 	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model" | ||
|  | 	"github.com/superseriousbusiness/gotosocial/internal/config" | ||
|  | 	"github.com/superseriousbusiness/gotosocial/internal/gtserror" | ||
|  | 	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel" | ||
|  | 	"github.com/superseriousbusiness/gotosocial/internal/id" | ||
|  | 	"github.com/superseriousbusiness/gotosocial/internal/oauth" | ||
|  | 	"github.com/superseriousbusiness/gotosocial/testrig" | ||
|  | ) | ||
|  | 
 | ||
|  | func (suite *NotificationsTestSuite) getNotifications( | ||
|  | 	account *gtsmodel.Account, | ||
|  | 	token *gtsmodel.Token, | ||
|  | 	user *gtsmodel.User, | ||
|  | 	maxID string, | ||
|  | 	minID string, | ||
|  | 	limit int, | ||
|  | 	includeTypes []string, | ||
|  | 	excludeTypes []string, | ||
|  | 	expectedHTTPStatus int, | ||
|  | 	expectedBody string, | ||
|  | ) ([]*apimodel.Notification, string, error) { | ||
|  | 	// instantiate recorder + test context | ||
|  | 	recorder := httptest.NewRecorder() | ||
|  | 	ctx, _ := testrig.CreateGinTestContext(recorder, nil) | ||
|  | 	ctx.Set(oauth.SessionAuthorizedAccount, account) | ||
|  | 	ctx.Set(oauth.SessionAuthorizedToken, oauth.DBTokenToToken(token)) | ||
|  | 	ctx.Set(oauth.SessionAuthorizedApplication, suite.testApplications["application_1"]) | ||
|  | 	ctx.Set(oauth.SessionAuthorizedUser, user) | ||
|  | 
 | ||
|  | 	// create the request | ||
|  | 	ctx.Request = httptest.NewRequest(http.MethodGet, config.GetProtocol()+"://"+config.GetHost()+"/api/"+notifications.BasePath, nil) | ||
|  | 	ctx.Request.Header.Set("accept", "application/json") | ||
|  | 	query := url.Values{} | ||
|  | 	if maxID != "" { | ||
|  | 		query.Set(notifications.MaxIDKey, maxID) | ||
|  | 	} | ||
|  | 	if minID != "" { | ||
|  | 		query.Set(notifications.MinIDKey, maxID) | ||
|  | 	} | ||
|  | 	if limit != 0 { | ||
|  | 		query.Set(notifications.LimitKey, strconv.Itoa(limit)) | ||
|  | 	} | ||
|  | 	if len(includeTypes) > 0 { | ||
|  | 		query[notifications.IncludeTypesKey] = includeTypes | ||
|  | 	} | ||
|  | 	if len(excludeTypes) > 0 { | ||
|  | 		query[notifications.ExcludeTypesKey] = excludeTypes | ||
|  | 	} | ||
|  | 	ctx.Request.URL.RawQuery = query.Encode() | ||
|  | 
 | ||
|  | 	// trigger the handler | ||
|  | 	suite.notificationsModule.NotificationsGETHandler(ctx) | ||
|  | 
 | ||
|  | 	// read the response | ||
|  | 	result := recorder.Result() | ||
|  | 	defer result.Body.Close() | ||
|  | 
 | ||
|  | 	b, err := io.ReadAll(result.Body) | ||
|  | 	if err != nil { | ||
|  | 		return nil, "", err | ||
|  | 	} | ||
|  | 
 | ||
|  | 	errs := gtserror.NewMultiError(2) | ||
|  | 
 | ||
|  | 	// check code | ||
|  | 	if resultCode := recorder.Code; expectedHTTPStatus != resultCode { | ||
|  | 		errs.Appendf("expected %d got %d", expectedHTTPStatus, resultCode) | ||
|  | 	} | ||
|  | 
 | ||
|  | 	// if we got an expected body, return early | ||
|  | 	if expectedBody != "" { | ||
|  | 		if string(b) != expectedBody { | ||
|  | 			errs.Appendf("expected %s got %s", expectedBody, string(b)) | ||
|  | 		} | ||
|  | 		return nil, "", errs.Combine() | ||
|  | 	} | ||
|  | 
 | ||
|  | 	resp := make([]*apimodel.Notification, 0) | ||
|  | 	if err := json.Unmarshal(b, &resp); err != nil { | ||
|  | 		return nil, "", err | ||
|  | 	} | ||
|  | 
 | ||
|  | 	return resp, result.Header.Get("Link"), nil | ||
|  | } | ||
|  | 
 | ||
|  | // Test that we can retrieve at least one notification and the expected Link header. | ||
|  | func (suite *NotificationsTestSuite) TestGetNotificationsSingle() { | ||
|  | 	testAccount := suite.testAccounts["local_account_1"] | ||
|  | 	testToken := suite.testTokens["local_account_1"] | ||
|  | 	testUser := suite.testUsers["local_account_1"] | ||
|  | 
 | ||
|  | 	maxID := "" | ||
|  | 	minID := "" | ||
|  | 	limit := 10 | ||
|  | 	includeTypes := []string(nil) | ||
|  | 	excludeTypes := []string(nil) | ||
|  | 	expectedHTTPStatus := http.StatusOK | ||
|  | 	expectedBody := "" | ||
|  | 
 | ||
|  | 	notifications, linkHeader, err := suite.getNotifications( | ||
|  | 		testAccount, | ||
|  | 		testToken, | ||
|  | 		testUser, | ||
|  | 		maxID, | ||
|  | 		minID, | ||
|  | 		limit, | ||
|  | 		includeTypes, | ||
|  | 		excludeTypes, | ||
|  | 		expectedHTTPStatus, | ||
|  | 		expectedBody, | ||
|  | 	) | ||
|  | 	if err != nil { | ||
|  | 		suite.FailNow(err.Error()) | ||
|  | 	} | ||
|  | 
 | ||
|  | 	suite.Len(notifications, 1) | ||
|  | 	suite.Equal(`<http://localhost:8080/api/v1/notifications?limit=10&max_id=01F8Q0ANPTWW10DAKTX7BRPBJP>; rel="next", <http://localhost:8080/api/v1/notifications?limit=10&min_id=01F8Q0ANPTWW10DAKTX7BRPBJP>; rel="prev"`, linkHeader) | ||
|  | } | ||
|  | 
 | ||
|  | // Add some extra notifications of different types than the fixture's single fav notification per account. | ||
|  | func (suite *NotificationsTestSuite) addMoreNotifications(testAccount *gtsmodel.Account) { | ||
|  | 	for _, b := range []*gtsmodel.Notification{ | ||
|  | 		{ | ||
|  | 			ID:               id.NewULID(), | ||
|  | 			NotificationType: gtsmodel.NotificationFollowRequest, | ||
|  | 			TargetAccountID:  testAccount.ID, | ||
|  | 			OriginAccountID:  suite.testAccounts["local_account_2"].ID, | ||
|  | 		}, | ||
|  | 		{ | ||
|  | 			ID:               id.NewULID(), | ||
|  | 			NotificationType: gtsmodel.NotificationFollow, | ||
|  | 			TargetAccountID:  testAccount.ID, | ||
|  | 			OriginAccountID:  suite.testAccounts["remote_account_2"].ID, | ||
|  | 		}, | ||
|  | 	} { | ||
|  | 		if err := suite.db.Put(context.Background(), b); err != nil { | ||
|  | 			suite.FailNow(err.Error()) | ||
|  | 		} | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | // Test that we can exclude a notification type. | ||
|  | func (suite *NotificationsTestSuite) TestGetNotificationsExcludeOneType() { | ||
|  | 	testAccount := suite.testAccounts["local_account_1"] | ||
|  | 	testToken := suite.testTokens["local_account_1"] | ||
|  | 	testUser := suite.testUsers["local_account_1"] | ||
|  | 
 | ||
|  | 	suite.addMoreNotifications(testAccount) | ||
|  | 
 | ||
|  | 	maxID := "" | ||
|  | 	minID := "" | ||
|  | 	limit := 10 | ||
|  | 	includeTypes := []string(nil) | ||
|  | 	excludeTypes := []string{"follow_request"} | ||
|  | 	expectedHTTPStatus := http.StatusOK | ||
|  | 	expectedBody := "" | ||
|  | 
 | ||
|  | 	notifications, _, err := suite.getNotifications( | ||
|  | 		testAccount, | ||
|  | 		testToken, | ||
|  | 		testUser, | ||
|  | 		maxID, | ||
|  | 		minID, | ||
|  | 		limit, | ||
|  | 		includeTypes, | ||
|  | 		excludeTypes, | ||
|  | 		expectedHTTPStatus, | ||
|  | 		expectedBody, | ||
|  | 	) | ||
|  | 	if err != nil { | ||
|  | 		suite.FailNow(err.Error()) | ||
|  | 	} | ||
|  | 
 | ||
|  | 	// This should not include the follow request notification. | ||
|  | 	suite.Len(notifications, 2) | ||
|  | 	for _, notification := range notifications { | ||
|  | 		suite.NotEqual("follow_request", notification.Type) | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | // Test that we can fetch only a single notification type. | ||
|  | func (suite *NotificationsTestSuite) TestGetNotificationsIncludeOneType() { | ||
|  | 	testAccount := suite.testAccounts["local_account_1"] | ||
|  | 	testToken := suite.testTokens["local_account_1"] | ||
|  | 	testUser := suite.testUsers["local_account_1"] | ||
|  | 
 | ||
|  | 	suite.addMoreNotifications(testAccount) | ||
|  | 
 | ||
|  | 	maxID := "" | ||
|  | 	minID := "" | ||
|  | 	limit := 10 | ||
|  | 	includeTypes := []string{"favourite"} | ||
|  | 	excludeTypes := []string(nil) | ||
|  | 	expectedHTTPStatus := http.StatusOK | ||
|  | 	expectedBody := "" | ||
|  | 
 | ||
|  | 	notifications, _, err := suite.getNotifications( | ||
|  | 		testAccount, | ||
|  | 		testToken, | ||
|  | 		testUser, | ||
|  | 		maxID, | ||
|  | 		minID, | ||
|  | 		limit, | ||
|  | 		includeTypes, | ||
|  | 		excludeTypes, | ||
|  | 		expectedHTTPStatus, | ||
|  | 		expectedBody, | ||
|  | 	) | ||
|  | 	if err != nil { | ||
|  | 		suite.FailNow(err.Error()) | ||
|  | 	} | ||
|  | 
 | ||
|  | 	// This should only include the fav notification. | ||
|  | 	suite.Len(notifications, 1) | ||
|  | 	for _, notification := range notifications { | ||
|  | 		suite.Equal("favourite", notification.Type) | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | func TestBookmarkTestSuite(t *testing.T) { | ||
|  | 	suite.Run(t, new(NotificationsTestSuite)) | ||
|  | } |