diff options
Diffstat (limited to 'internal/api/client')
| -rw-r--r-- | internal/api/client/fileserver/fileserver_test.go | 109 | ||||
| -rw-r--r-- | internal/api/client/fileserver/servefile.go | 13 | ||||
| -rw-r--r-- | internal/api/client/fileserver/servefile_test.go | 347 | 
3 files changed, 322 insertions, 147 deletions
| diff --git a/internal/api/client/fileserver/fileserver_test.go b/internal/api/client/fileserver/fileserver_test.go new file mode 100644 index 000000000..f1fab5672 --- /dev/null +++ b/internal/api/client/fileserver/fileserver_test.go @@ -0,0 +1,109 @@ +/* +   GoToSocial +   Copyright (C) 2021-2022 GoToSocial Authors admin@gotosocial.org + +   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 fileserver_test + +import ( +	"context" + +	"github.com/stretchr/testify/suite" +	"github.com/superseriousbusiness/gotosocial/internal/api/client/fileserver" +	"github.com/superseriousbusiness/gotosocial/internal/concurrency" +	"github.com/superseriousbusiness/gotosocial/internal/db" +	"github.com/superseriousbusiness/gotosocial/internal/email" +	"github.com/superseriousbusiness/gotosocial/internal/federation" +	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel" +	"github.com/superseriousbusiness/gotosocial/internal/log" +	"github.com/superseriousbusiness/gotosocial/internal/media" +	"github.com/superseriousbusiness/gotosocial/internal/messages" +	"github.com/superseriousbusiness/gotosocial/internal/oauth" +	"github.com/superseriousbusiness/gotosocial/internal/processing" +	"github.com/superseriousbusiness/gotosocial/internal/storage" +	"github.com/superseriousbusiness/gotosocial/internal/typeutils" +	"github.com/superseriousbusiness/gotosocial/testrig" +) + +type FileserverTestSuite struct { +	// standard suite interfaces +	suite.Suite +	db           db.DB +	storage      *storage.Driver +	federator    federation.Federator +	tc           typeutils.TypeConverter +	processor    processing.Processor +	mediaManager media.Manager +	oauthServer  oauth.Server +	emailSender  email.Sender + +	// standard suite models +	testTokens       map[string]*gtsmodel.Token +	testClients      map[string]*gtsmodel.Client +	testApplications map[string]*gtsmodel.Application +	testUsers        map[string]*gtsmodel.User +	testAccounts     map[string]*gtsmodel.Account +	testAttachments  map[string]*gtsmodel.MediaAttachment + +	// item being tested +	fileServer *fileserver.FileServer +} + +/* +	TEST INFRASTRUCTURE +*/ + +func (suite *FileserverTestSuite) SetupSuite() { +	testrig.InitTestConfig() +	testrig.InitTestLog() + +	fedWorker := concurrency.NewWorkerPool[messages.FromFederator](-1, -1) +	clientWorker := concurrency.NewWorkerPool[messages.FromClientAPI](-1, -1) + +	suite.db = testrig.NewTestDB() +	suite.storage = testrig.NewInMemoryStorage() +	suite.federator = testrig.NewTestFederator(suite.db, testrig.NewTestTransportController(testrig.NewMockHTTPClient(nil, "../../../../testrig/media"), suite.db, fedWorker), suite.storage, suite.mediaManager, fedWorker) +	suite.emailSender = testrig.NewEmailSender("../../../../web/template/", nil) + +	suite.processor = testrig.NewTestProcessor(suite.db, suite.storage, suite.federator, suite.emailSender, testrig.NewTestMediaManager(suite.db, suite.storage), clientWorker, fedWorker) +	suite.tc = testrig.NewTestTypeConverter(suite.db) +	suite.mediaManager = testrig.NewTestMediaManager(suite.db, suite.storage) +	suite.oauthServer = testrig.NewTestOauthServer(suite.db) + +	suite.fileServer = fileserver.New(suite.processor).(*fileserver.FileServer) +} + +func (suite *FileserverTestSuite) SetupTest() { +	testrig.StandardDBSetup(suite.db, nil) +	testrig.StandardStorageSetup(suite.storage, "../../../../testrig/media") +	suite.testTokens = testrig.NewTestTokens() +	suite.testClients = testrig.NewTestClients() +	suite.testApplications = testrig.NewTestApplications() +	suite.testUsers = testrig.NewTestUsers() +	suite.testAccounts = testrig.NewTestAccounts() +	suite.testAttachments = testrig.NewTestAttachments() +} + +func (suite *FileserverTestSuite) TearDownSuite() { +	if err := suite.db.Stop(context.Background()); err != nil { +		log.Panicf("error closing db connection: %s", err) +	} +} + +func (suite *FileserverTestSuite) TearDownTest() { +	testrig.StandardDBTeardown(suite.db) +	testrig.StandardStorageTeardown(suite.storage) +} diff --git a/internal/api/client/fileserver/servefile.go b/internal/api/client/fileserver/servefile.go index e4eca770f..d2328a5fc 100644 --- a/internal/api/client/fileserver/servefile.go +++ b/internal/api/client/fileserver/servefile.go @@ -19,7 +19,9 @@  package fileserver  import ( +	"bytes"  	"fmt" +	"io"  	"net/http"  	"strconv" @@ -120,5 +122,14 @@ func (m *FileServer) ServeFile(c *gin.Context) {  		return  	} -	c.DataFromReader(http.StatusOK, content.ContentLength, format, content.Content, nil) +	// try to slurp the first few bytes to make sure we have something +	b := bytes.NewBuffer(make([]byte, 0, 64)) +	if _, err := io.CopyN(b, content.Content, 64); err != nil { +		err = fmt.Errorf("ServeFile: error reading from content: %w", err) +		api.ErrorHandler(c, gtserror.NewErrorNotFound(err, err.Error()), m.processor.InstanceGet) +		return +	} + +	// we're good, return the slurped bytes + the rest of the content +	c.DataFromReader(http.StatusOK, content.ContentLength, format, io.MultiReader(b, content.Content), nil)  } diff --git a/internal/api/client/fileserver/servefile_test.go b/internal/api/client/fileserver/servefile_test.go index a6c46e23f..1ca0c60d6 100644 --- a/internal/api/client/fileserver/servefile_test.go +++ b/internal/api/client/fileserver/servefile_test.go @@ -20,196 +20,251 @@ package fileserver_test  import (  	"context" -	"fmt"  	"io/ioutil"  	"net/http"  	"net/http/httptest"  	"testing" -	"github.com/gin-gonic/gin"  	"github.com/stretchr/testify/suite"  	"github.com/superseriousbusiness/gotosocial/internal/api/client/fileserver" -	"github.com/superseriousbusiness/gotosocial/internal/concurrency" -	"github.com/superseriousbusiness/gotosocial/internal/db" -	"github.com/superseriousbusiness/gotosocial/internal/email" -	"github.com/superseriousbusiness/gotosocial/internal/federation"  	"github.com/superseriousbusiness/gotosocial/internal/gtsmodel" -	"github.com/superseriousbusiness/gotosocial/internal/log"  	"github.com/superseriousbusiness/gotosocial/internal/media" -	"github.com/superseriousbusiness/gotosocial/internal/messages" -	"github.com/superseriousbusiness/gotosocial/internal/oauth" -	"github.com/superseriousbusiness/gotosocial/internal/processing" -	"github.com/superseriousbusiness/gotosocial/internal/storage" -	"github.com/superseriousbusiness/gotosocial/internal/typeutils"  	"github.com/superseriousbusiness/gotosocial/testrig"  )  type ServeFileTestSuite struct { -	// standard suite interfaces -	suite.Suite -	db           db.DB -	storage      *storage.Driver -	federator    federation.Federator -	tc           typeutils.TypeConverter -	processor    processing.Processor -	mediaManager media.Manager -	oauthServer  oauth.Server -	emailSender  email.Sender - -	// standard suite models -	testTokens       map[string]*gtsmodel.Token -	testClients      map[string]*gtsmodel.Client -	testApplications map[string]*gtsmodel.Application -	testUsers        map[string]*gtsmodel.User -	testAccounts     map[string]*gtsmodel.Account -	testAttachments  map[string]*gtsmodel.MediaAttachment - -	// item being tested -	fileServer *fileserver.FileServer +	FileserverTestSuite  } -/* -	TEST INFRASTRUCTURE -*/ +// GetFile is just a convenience function to save repetition in this test suite. +// It takes the required params to serve a file, calls the handler, and returns +// the http status code, the response headers, and the parsed body bytes. +func (suite *ServeFileTestSuite) GetFile( +	accountID string, +	mediaType media.Type, +	mediaSize media.Size, +	filename string, +) (code int, headers http.Header, body []byte) { +	recorder := httptest.NewRecorder() + +	ctx, _ := testrig.CreateGinTestContext(recorder, nil) +	ctx.Request = httptest.NewRequest(http.MethodGet, "http://localhost:8080/whatever", nil) +	ctx.Request.Header.Set("accept", "*/*") +	ctx.AddParam(fileserver.AccountIDKey, accountID) +	ctx.AddParam(fileserver.MediaTypeKey, string(mediaType)) +	ctx.AddParam(fileserver.MediaSizeKey, string(mediaSize)) +	ctx.AddParam(fileserver.FileNameKey, filename) -func (suite *ServeFileTestSuite) SetupSuite() { -	// setup standard items -	testrig.InitTestConfig() -	testrig.InitTestLog() +	suite.fileServer.ServeFile(ctx) +	code = recorder.Code +	headers = recorder.Result().Header + +	var err error +	body, err = ioutil.ReadAll(recorder.Body) +	if err != nil { +		suite.FailNow(err.Error()) +	} -	fedWorker := concurrency.NewWorkerPool[messages.FromFederator](-1, -1) -	clientWorker := concurrency.NewWorkerPool[messages.FromClientAPI](-1, -1) +	return +} -	suite.db = testrig.NewTestDB() -	suite.storage = testrig.NewInMemoryStorage() -	suite.federator = testrig.NewTestFederator(suite.db, testrig.NewTestTransportController(testrig.NewMockHTTPClient(nil, "../../../../testrig/media"), suite.db, fedWorker), suite.storage, suite.mediaManager, fedWorker) -	suite.emailSender = testrig.NewEmailSender("../../../../web/template/", nil) +// UncacheAttachment is a convenience function that uncaches the targetAttachment by +// removing its associated files from storage, and updating the database. +func (suite *ServeFileTestSuite) UncacheAttachment(targetAttachment *gtsmodel.MediaAttachment) { +	ctx := context.Background() -	suite.processor = testrig.NewTestProcessor(suite.db, suite.storage, suite.federator, suite.emailSender, testrig.NewTestMediaManager(suite.db, suite.storage), clientWorker, fedWorker) -	suite.tc = testrig.NewTestTypeConverter(suite.db) -	suite.mediaManager = testrig.NewTestMediaManager(suite.db, suite.storage) -	suite.oauthServer = testrig.NewTestOauthServer(suite.db) +	cached := false +	targetAttachment.Cached = &cached -	// setup module being tested -	suite.fileServer = fileserver.New(suite.processor).(*fileserver.FileServer) +	if err := suite.db.UpdateByID(ctx, targetAttachment, targetAttachment.ID, "cached"); err != nil { +		suite.FailNow(err.Error()) +	} +	if err := suite.storage.Delete(ctx, targetAttachment.File.Path); err != nil { +		suite.FailNow(err.Error()) +	} +	if err := suite.storage.Delete(ctx, targetAttachment.Thumbnail.Path); err != nil { +		suite.FailNow(err.Error()) +	}  } -func (suite *ServeFileTestSuite) TearDownSuite() { -	if err := suite.db.Stop(context.Background()); err != nil { -		log.Panicf("error closing db connection: %s", err) +func (suite *ServeFileTestSuite) TestServeOriginalLocalFileOK() { +	targetAttachment := >smodel.MediaAttachment{} +	*targetAttachment = *suite.testAttachments["admin_account_status_1_attachment_1"] +	fileInStorage, err := suite.storage.Get(context.Background(), targetAttachment.File.Path) +	if err != nil { +		suite.FailNow(err.Error())  	} + +	code, headers, body := suite.GetFile( +		targetAttachment.AccountID, +		media.TypeAttachment, +		media.SizeOriginal, +		targetAttachment.ID+".jpeg", +	) + +	suite.Equal(http.StatusOK, code) +	suite.Equal("image/jpeg", headers.Get("content-type")) +	suite.Equal(fileInStorage, body)  } -func (suite *ServeFileTestSuite) SetupTest() { -	testrig.StandardDBSetup(suite.db, nil) -	testrig.StandardStorageSetup(suite.storage, "../../../../testrig/media") -	suite.testTokens = testrig.NewTestTokens() -	suite.testClients = testrig.NewTestClients() -	suite.testApplications = testrig.NewTestApplications() -	suite.testUsers = testrig.NewTestUsers() -	suite.testAccounts = testrig.NewTestAccounts() -	suite.testAttachments = testrig.NewTestAttachments() +func (suite *ServeFileTestSuite) TestServeSmallLocalFileOK() { +	targetAttachment := >smodel.MediaAttachment{} +	*targetAttachment = *suite.testAttachments["admin_account_status_1_attachment_1"] +	fileInStorage, err := suite.storage.Get(context.Background(), targetAttachment.Thumbnail.Path) +	if err != nil { +		suite.FailNow(err.Error()) +	} + +	code, headers, body := suite.GetFile( +		targetAttachment.AccountID, +		media.TypeAttachment, +		media.SizeSmall, +		targetAttachment.ID+".jpeg", +	) + +	suite.Equal(http.StatusOK, code) +	suite.Equal("image/jpeg", headers.Get("content-type")) +	suite.Equal(fileInStorage, body)  } -func (suite *ServeFileTestSuite) TearDownTest() { -	testrig.StandardDBTeardown(suite.db) -	testrig.StandardStorageTeardown(suite.storage) +func (suite *ServeFileTestSuite) TestServeOriginalRemoteFileOK() { +	targetAttachment := >smodel.MediaAttachment{} +	*targetAttachment = *suite.testAttachments["remote_account_1_status_1_attachment_1"] +	fileInStorage, err := suite.storage.Get(context.Background(), targetAttachment.File.Path) +	if err != nil { +		suite.FailNow(err.Error()) +	} + +	code, headers, body := suite.GetFile( +		targetAttachment.AccountID, +		media.TypeAttachment, +		media.SizeOriginal, +		targetAttachment.ID+".jpeg", +	) + +	suite.Equal(http.StatusOK, code) +	suite.Equal("image/jpeg", headers.Get("content-type")) +	suite.Equal(fileInStorage, body)  } -/* -	ACTUAL TESTS -*/ +func (suite *ServeFileTestSuite) TestServeSmallRemoteFileOK() { +	targetAttachment := >smodel.MediaAttachment{} +	*targetAttachment = *suite.testAttachments["remote_account_1_status_1_attachment_1"] +	fileInStorage, err := suite.storage.Get(context.Background(), targetAttachment.Thumbnail.Path) +	if err != nil { +		suite.FailNow(err.Error()) +	} -func (suite *ServeFileTestSuite) TestServeOriginalFileSuccessful() { -	targetAttachment, ok := suite.testAttachments["admin_account_status_1_attachment_1"] -	suite.True(ok) -	suite.NotNil(targetAttachment) +	code, headers, body := suite.GetFile( +		targetAttachment.AccountID, +		media.TypeAttachment, +		media.SizeSmall, +		targetAttachment.ID+".jpeg", +	) -	recorder := httptest.NewRecorder() -	ctx, _ := testrig.CreateGinTestContext(recorder, nil) -	ctx.Request = httptest.NewRequest(http.MethodGet, targetAttachment.URL, nil) -	ctx.Request.Header.Set("accept", "*/*") +	suite.Equal(http.StatusOK, code) +	suite.Equal("image/jpeg", headers.Get("content-type")) +	suite.Equal(fileInStorage, body) +} -	// normally the router would populate these params from the path values, -	// but because we're calling the ServeFile function directly, we need to set them manually. -	ctx.Params = gin.Params{ -		gin.Param{ -			Key:   fileserver.AccountIDKey, -			Value: targetAttachment.AccountID, -		}, -		gin.Param{ -			Key:   fileserver.MediaTypeKey, -			Value: string(media.TypeAttachment), -		}, -		gin.Param{ -			Key:   fileserver.MediaSizeKey, -			Value: string(media.SizeOriginal), -		}, -		gin.Param{ -			Key:   fileserver.FileNameKey, -			Value: fmt.Sprintf("%s.jpeg", targetAttachment.ID), -		}, +func (suite *ServeFileTestSuite) TestServeOriginalRemoteFileRecache() { +	targetAttachment := >smodel.MediaAttachment{} +	*targetAttachment = *suite.testAttachments["remote_account_1_status_1_attachment_1"] +	fileInStorage, err := suite.storage.Get(context.Background(), targetAttachment.File.Path) +	if err != nil { +		suite.FailNow(err.Error())  	} -	// call the function we're testing and check status code -	suite.fileServer.ServeFile(ctx) -	suite.EqualValues(http.StatusOK, recorder.Code) -	suite.EqualValues("image/jpeg", recorder.Header().Get("content-type")) +	// uncache the attachment so we'll have to refetch it from the 'remote' instance +	suite.UncacheAttachment(targetAttachment) -	b, err := ioutil.ReadAll(recorder.Body) -	suite.NoError(err) -	suite.NotNil(b) +	code, headers, body := suite.GetFile( +		targetAttachment.AccountID, +		media.TypeAttachment, +		media.SizeOriginal, +		targetAttachment.ID+".jpeg", +	) -	fileInStorage, err := suite.storage.Get(ctx, targetAttachment.File.Path) -	suite.NoError(err) -	suite.NotNil(fileInStorage) -	suite.Equal(b, fileInStorage) +	suite.Equal(http.StatusOK, code) +	suite.Equal("image/jpeg", headers.Get("content-type")) +	suite.Equal(fileInStorage, body)  } -func (suite *ServeFileTestSuite) TestServeSmallFileSuccessful() { -	targetAttachment, ok := suite.testAttachments["admin_account_status_1_attachment_1"] -	suite.True(ok) -	suite.NotNil(targetAttachment) +func (suite *ServeFileTestSuite) TestServeSmallRemoteFileRecache() { +	targetAttachment := >smodel.MediaAttachment{} +	*targetAttachment = *suite.testAttachments["remote_account_1_status_1_attachment_1"] +	fileInStorage, err := suite.storage.Get(context.Background(), targetAttachment.Thumbnail.Path) +	if err != nil { +		suite.FailNow(err.Error()) +	} -	recorder := httptest.NewRecorder() -	ctx, _ := testrig.CreateGinTestContext(recorder, nil) -	ctx.Request = httptest.NewRequest(http.MethodGet, targetAttachment.Thumbnail.URL, nil) -	ctx.Request.Header.Set("accept", "*/*") +	// uncache the attachment so we'll have to refetch it from the 'remote' instance +	suite.UncacheAttachment(targetAttachment) + +	code, headers, body := suite.GetFile( +		targetAttachment.AccountID, +		media.TypeAttachment, +		media.SizeSmall, +		targetAttachment.ID+".jpeg", +	) -	// normally the router would populate these params from the path values, -	// but because we're calling the ServeFile function directly, we need to set them manually. -	ctx.Params = gin.Params{ -		gin.Param{ -			Key:   fileserver.AccountIDKey, -			Value: targetAttachment.AccountID, -		}, -		gin.Param{ -			Key:   fileserver.MediaTypeKey, -			Value: string(media.TypeAttachment), -		}, -		gin.Param{ -			Key:   fileserver.MediaSizeKey, -			Value: string(media.SizeSmall), -		}, -		gin.Param{ -			Key:   fileserver.FileNameKey, -			Value: fmt.Sprintf("%s.jpeg", targetAttachment.ID), -		}, +	suite.Equal(http.StatusOK, code) +	suite.Equal("image/jpeg", headers.Get("content-type")) +	suite.Equal(fileInStorage, body) +} + +func (suite *ServeFileTestSuite) TestServeOriginalRemoteFileRecacheNotFound() { +	targetAttachment := >smodel.MediaAttachment{} +	*targetAttachment = *suite.testAttachments["remote_account_1_status_1_attachment_1"] + +	// uncache the attachment *and* set the remote URL to something that will return a 404 +	suite.UncacheAttachment(targetAttachment) +	targetAttachment.RemoteURL = "http://nothing.at.this.url/weeeeeeeee" +	if err := suite.db.UpdateByID(context.Background(), targetAttachment, targetAttachment.ID, "remote_url"); err != nil { +		suite.FailNow(err.Error())  	} -	// call the function we're testing and check status code -	suite.fileServer.ServeFile(ctx) -	suite.EqualValues(http.StatusOK, recorder.Code) -	suite.EqualValues("image/jpeg", recorder.Header().Get("content-type")) +	code, _, _ := suite.GetFile( +		targetAttachment.AccountID, +		media.TypeAttachment, +		media.SizeOriginal, +		targetAttachment.ID+".jpeg", +	) + +	suite.Equal(http.StatusNotFound, code) +} + +func (suite *ServeFileTestSuite) TestServeSmallRemoteFileRecacheNotFound() { +	targetAttachment := >smodel.MediaAttachment{} +	*targetAttachment = *suite.testAttachments["remote_account_1_status_1_attachment_1"] + +	// uncache the attachment *and* set the remote URL to something that will return a 404 +	suite.UncacheAttachment(targetAttachment) +	targetAttachment.RemoteURL = "http://nothing.at.this.url/weeeeeeeee" +	if err := suite.db.UpdateByID(context.Background(), targetAttachment, targetAttachment.ID, "remote_url"); err != nil { +		suite.FailNow(err.Error()) +	} + +	code, _, _ := suite.GetFile( +		targetAttachment.AccountID, +		media.TypeAttachment, +		media.SizeSmall, +		targetAttachment.ID+".jpeg", +	) + +	suite.Equal(http.StatusNotFound, code) +} -	b, err := ioutil.ReadAll(recorder.Body) -	suite.NoError(err) -	suite.NotNil(b) +// Callers trying to get some random-ass file that doesn't exist should just get a 404 +func (suite *ServeFileTestSuite) TestServeFileNotFound() { +	code, _, _ := suite.GetFile( +		"01GMMY4G9B0QEG0PQK5Q5JGJWZ", +		media.TypeAttachment, +		media.SizeOriginal, +		"01GMMY68Y7E5DJ3CA3Y9SS8524.jpeg", +	) -	fileInStorage, err := suite.storage.Get(ctx, targetAttachment.Thumbnail.Path) -	suite.NoError(err) -	suite.NotNil(fileInStorage) -	suite.Equal(b, fileInStorage) +	suite.Equal(http.StatusNotFound, code)  }  func TestServeFileTestSuite(t *testing.T) { | 
