| 
									
										
										
										
											2023-03-12 16:00:57 +01:00
										 |  |  | // 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/>. | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | package statuses_test | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							|  |  |  | 	"io/ioutil" | 
					
						
							|  |  |  | 	"net/http" | 
					
						
							|  |  |  | 	"net/http/httptest" | 
					
						
							|  |  |  | 	"strconv" | 
					
						
							|  |  |  | 	"testing" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-26 15:34:10 +02:00
										 |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/ap" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/api/client/statuses" | 
					
						
							|  |  |  | 	apimodel "code.superseriousbusiness.org/gotosocial/internal/api/model" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/config" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/gtserror" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/gtsmodel" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/id" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/oauth" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/internal/util" | 
					
						
							|  |  |  | 	"code.superseriousbusiness.org/gotosocial/testrig" | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	"github.com/stretchr/testify/suite" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type StatusPinTestSuite struct { | 
					
						
							|  |  |  | 	StatusStandardTestSuite | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (suite *StatusPinTestSuite) createPin( | 
					
						
							|  |  |  | 	expectedHTTPStatus int, | 
					
						
							|  |  |  | 	expectedBody string, | 
					
						
							|  |  |  | 	targetStatusID string, | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 	requestingAcct *gtsmodel.Account, | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | ) (*apimodel.Status, error) { | 
					
						
							|  |  |  | 	// instantiate recorder + test context | 
					
						
							|  |  |  | 	recorder := httptest.NewRecorder() | 
					
						
							|  |  |  | 	ctx, _ := testrig.CreateGinTestContext(recorder, nil) | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 	ctx.Set(oauth.SessionAuthorizedAccount, requestingAcct) | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	ctx.Set(oauth.SessionAuthorizedToken, oauth.DBTokenToToken(suite.testTokens["local_account_1"])) | 
					
						
							|  |  |  | 	ctx.Set(oauth.SessionAuthorizedApplication, suite.testApplications["application_1"]) | 
					
						
							|  |  |  | 	ctx.Set(oauth.SessionAuthorizedUser, suite.testUsers["local_account_1"]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// create the request | 
					
						
							|  |  |  | 	ctx.Request = httptest.NewRequest(http.MethodPost, config.GetProtocol()+"://"+config.GetHost()+"/api/"+statuses.BasePath+"/"+targetStatusID+"/pin", nil) | 
					
						
							|  |  |  | 	ctx.Request.Header.Set("accept", "application/json") | 
					
						
							|  |  |  | 	ctx.AddParam(statuses.IDKey, targetStatusID) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// trigger the handler | 
					
						
							|  |  |  | 	suite.statusModule.StatusPinPOSTHandler(ctx) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// read the response | 
					
						
							|  |  |  | 	result := recorder.Result() | 
					
						
							|  |  |  | 	defer result.Body.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	b, err := ioutil.ReadAll(result.Body) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-02 17:21:46 +02:00
										 |  |  | 	errs := gtserror.NewMultiError(2) | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-02 17:21:46 +02:00
										 |  |  | 	// Check expected code + body. | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	if resultCode := recorder.Code; expectedHTTPStatus != resultCode { | 
					
						
							| 
									
										
										
										
											2023-08-02 17:21:46 +02:00
										 |  |  | 		errs.Appendf("expected %d got %d", expectedHTTPStatus, resultCode) | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-02 17:21:46 +02:00
										 |  |  | 	// If we got an expected body, return early. | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	if expectedBody != "" && string(b) != expectedBody { | 
					
						
							| 
									
										
										
										
											2023-08-02 17:21:46 +02:00
										 |  |  | 		errs.Appendf("expected %s got %s", expectedBody, string(b)) | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-02 17:21:46 +02:00
										 |  |  | 	if err := errs.Combine(); err != nil { | 
					
						
							|  |  |  | 		suite.FailNow("", "%v (body %s)", err, string(b)) | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	resp := &apimodel.Status{} | 
					
						
							|  |  |  | 	if err := json.Unmarshal(b, resp); err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return resp, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (suite *StatusPinTestSuite) TestPinStatusPublicOK() { | 
					
						
							|  |  |  | 	// Pin an unpinned public status that this account owns. | 
					
						
							|  |  |  | 	targetStatus := suite.testStatuses["local_account_1_status_1"] | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 	testAccount := new(gtsmodel.Account) | 
					
						
							|  |  |  | 	*testAccount = *suite.testAccounts["local_account_1"] | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 	resp, err := suite.createPin(http.StatusOK, "", targetStatus.ID, testAccount) | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		suite.FailNow(err.Error()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	suite.True(resp.Pinned) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (suite *StatusPinTestSuite) TestPinStatusFollowersOnlyOK() { | 
					
						
							|  |  |  | 	// Pin an unpinned followers only status that this account owns. | 
					
						
							|  |  |  | 	targetStatus := suite.testStatuses["local_account_1_status_5"] | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 	testAccount := new(gtsmodel.Account) | 
					
						
							|  |  |  | 	*testAccount = *suite.testAccounts["local_account_1"] | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 	resp, err := suite.createPin(http.StatusOK, "", targetStatus.ID, testAccount) | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		suite.FailNow(err.Error()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	suite.True(resp.Pinned) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (suite *StatusPinTestSuite) TestPinStatusTwiceError() { | 
					
						
							|  |  |  | 	// Try to pin a status that's already been pinned. | 
					
						
							|  |  |  | 	targetStatus := >smodel.Status{} | 
					
						
							|  |  |  | 	*targetStatus = *suite.testStatuses["local_account_1_status_5"] | 
					
						
							|  |  |  | 	targetStatus.PinnedAt = time.Now() | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 	testAccount := new(gtsmodel.Account) | 
					
						
							|  |  |  | 	*testAccount = *suite.testAccounts["local_account_1"] | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-01 18:52:44 +01:00
										 |  |  | 	if err := suite.db.UpdateStatus(context.Background(), targetStatus, "pinned_at"); err != nil { | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 		suite.FailNow(err.Error()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if _, err := suite.createPin( | 
					
						
							|  |  |  | 		http.StatusUnprocessableEntity, | 
					
						
							|  |  |  | 		`{"error":"Unprocessable Entity: status already pinned"}`, | 
					
						
							|  |  |  | 		targetStatus.ID, | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 		testAccount, | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	); err != nil { | 
					
						
							|  |  |  | 		suite.FailNow(err.Error()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (suite *StatusPinTestSuite) TestPinStatusOtherAccountError() { | 
					
						
							|  |  |  | 	// Try to pin a status that doesn't belong to us. | 
					
						
							|  |  |  | 	targetStatus := suite.testStatuses["admin_account_status_1"] | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 	testAccount := new(gtsmodel.Account) | 
					
						
							|  |  |  | 	*testAccount = *suite.testAccounts["local_account_1"] | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if _, err := suite.createPin( | 
					
						
							|  |  |  | 		http.StatusUnprocessableEntity, | 
					
						
							|  |  |  | 		`{"error":"Unprocessable Entity: status 01F8MH75CBF9JFX4ZAD54N0W0R does not belong to account 01F8MH1H7YV1Z7D2C8K2730QBF"}`, | 
					
						
							|  |  |  | 		targetStatus.ID, | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 		testAccount, | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	); err != nil { | 
					
						
							|  |  |  | 		suite.FailNow(err.Error()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (suite *StatusPinTestSuite) TestPinStatusTooManyPins() { | 
					
						
							|  |  |  | 	// Test pinning too many statuses. | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 	testAccount := new(gtsmodel.Account) | 
					
						
							|  |  |  | 	*testAccount = *suite.testAccounts["local_account_1"] | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Spam 10 pinned statuses into the database. | 
					
						
							|  |  |  | 	ctx := context.Background() | 
					
						
							|  |  |  | 	for i := range make([]interface{}, 10) { | 
					
						
							|  |  |  | 		status := >smodel.Status{ | 
					
						
							|  |  |  | 			ID:                  id.NewULID(), | 
					
						
							|  |  |  | 			PinnedAt:            time.Now(), | 
					
						
							|  |  |  | 			URL:                 "stub " + strconv.Itoa(i), | 
					
						
							|  |  |  | 			URI:                 "stub " + strconv.Itoa(i), | 
					
						
							| 
									
										
										
										
											2023-08-07 19:38:11 +02:00
										 |  |  | 			Local:               util.Ptr(true), | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 			AccountID:           testAccount.ID, | 
					
						
							|  |  |  | 			AccountURI:          testAccount.URI, | 
					
						
							|  |  |  | 			Visibility:          gtsmodel.VisibilityPublic, | 
					
						
							| 
									
										
										
										
											2023-08-07 19:38:11 +02:00
										 |  |  | 			Federated:           util.Ptr(true), | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 			ActivityStreamsType: ap.ObjectNote, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if err := suite.db.PutStatus(ctx, status); err != nil { | 
					
						
							|  |  |  | 			suite.FailNow(err.Error()) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 	// Regenerate account stats to set pinned count. | 
					
						
							|  |  |  | 	if err := suite.db.RegenerateAccountStats(ctx, testAccount); err != nil { | 
					
						
							|  |  |  | 		suite.FailNow(err.Error()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	// Try to pin one more status as a treat. | 
					
						
							|  |  |  | 	targetStatus := suite.testStatuses["local_account_1_status_1"] | 
					
						
							|  |  |  | 	if _, err := suite.createPin( | 
					
						
							|  |  |  | 		http.StatusUnprocessableEntity, | 
					
						
							|  |  |  | 		`{"error":"Unprocessable Entity: status pin limit exceeded, you've already pinned 10 status(es) out of 10"}`, | 
					
						
							|  |  |  | 		targetStatus.ID, | 
					
						
							| 
									
										
										
										
											2024-04-16 13:10:13 +02:00
										 |  |  | 		testAccount, | 
					
						
							| 
									
										
										
										
											2023-02-25 13:16:30 +01:00
										 |  |  | 	); err != nil { | 
					
						
							|  |  |  | 		suite.FailNow(err.Error()) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestStatusPinTestSuite(t *testing.T) { | 
					
						
							|  |  |  | 	suite.Run(t, new(StatusPinTestSuite)) | 
					
						
							|  |  |  | } |