Full Mattermost server source with integrated Community Enterprise features. Includes vendor directory for offline/air-gapped builds. Structure: - enterprise-impl/: Enterprise feature implementations - enterprise-community/: Init files that register implementations - enterprise/: Bridge imports (community_imports.go) - vendor/: All dependencies for offline builds Build (online): go build ./cmd/mattermost Build (offline/air-gapped): go build -mod=vendor ./cmd/mattermost 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
473 lines
15 KiB
Go
473 lines
15 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package api4
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"mime/multipart"
|
|
"net/http"
|
|
"os"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/mattermost/mattermost/server/public/model"
|
|
"github.com/mattermost/mattermost/server/v8/channels/utils/fileutils"
|
|
)
|
|
|
|
func TestCreateUpload(t *testing.T) {
|
|
mainHelper.Parallel(t)
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
|
|
us := &model.UploadSession{
|
|
ChannelId: th.BasicChannel.Id,
|
|
Filename: "upload",
|
|
FileSize: 8 * 1024 * 1024,
|
|
}
|
|
|
|
t.Run("file attachments disabled", func(t *testing.T) {
|
|
th.App.UpdateConfig(func(cfg *model.Config) { *cfg.FileSettings.EnableFileAttachments = false })
|
|
defer th.App.UpdateConfig(func(cfg *model.Config) { *cfg.FileSettings.EnableFileAttachments = true })
|
|
u, resp, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.Nil(t, u)
|
|
CheckErrorID(t, err, "api.file.attachments.disabled.app_error")
|
|
require.Equal(t, http.StatusNotImplemented, resp.StatusCode)
|
|
})
|
|
|
|
t.Run("no permissions", func(t *testing.T) {
|
|
us.ChannelId = th.BasicPrivateChannel2.Id
|
|
u, resp, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.Nil(t, u)
|
|
CheckErrorID(t, err, "api.context.permissions.app_error")
|
|
require.Equal(t, http.StatusForbidden, resp.StatusCode)
|
|
})
|
|
|
|
t.Run("FileSize over limit", func(t *testing.T) {
|
|
maxFileSize := *th.App.Config().FileSettings.MaxFileSize
|
|
th.App.UpdateConfig(func(cfg *model.Config) { *cfg.FileSettings.MaxFileSize = us.FileSize - 1 })
|
|
defer th.App.UpdateConfig(func(cfg *model.Config) { *cfg.FileSettings.MaxFileSize = maxFileSize })
|
|
us.ChannelId = th.BasicChannel.Id
|
|
u, resp, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.Nil(t, u)
|
|
CheckErrorID(t, err, "api.upload.create.upload_too_large.app_error")
|
|
require.Equal(t, http.StatusRequestEntityTooLarge, resp.StatusCode)
|
|
})
|
|
|
|
t.Run("not allowed in cloud", func(t *testing.T) {
|
|
th.App.Srv().SetLicense(model.NewTestLicense("cloud"))
|
|
defer func() {
|
|
appErr := th.App.Srv().RemoveLicense()
|
|
require.Nil(t, appErr)
|
|
}()
|
|
|
|
u, resp, err := th.SystemAdminClient.CreateUpload(context.Background(), &model.UploadSession{
|
|
ChannelId: th.BasicChannel.Id,
|
|
Filename: "upload",
|
|
FileSize: 8 * 1024 * 1024,
|
|
Type: model.UploadTypeImport,
|
|
})
|
|
require.Nil(t, u)
|
|
CheckErrorID(t, err, "api.file.cloud_upload.app_error")
|
|
require.Equal(t, http.StatusBadRequest, resp.StatusCode)
|
|
})
|
|
|
|
t.Run("valid", func(t *testing.T) {
|
|
us.ChannelId = th.BasicChannel.Id
|
|
u, resp, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, u)
|
|
require.Equal(t, http.StatusCreated, resp.StatusCode)
|
|
})
|
|
|
|
t.Run("import file", func(t *testing.T) {
|
|
testsDir, _ := fileutils.FindDir("tests")
|
|
|
|
importFile, err := os.Open(testsDir + "/import_test.zip")
|
|
require.NoError(t, err)
|
|
defer importFile.Close()
|
|
|
|
info, err := importFile.Stat()
|
|
require.NoError(t, err)
|
|
|
|
t.Run("permissions error", func(t *testing.T) {
|
|
us := &model.UploadSession{
|
|
Filename: info.Name(),
|
|
FileSize: info.Size(),
|
|
Type: model.UploadTypeImport,
|
|
}
|
|
u, resp, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.Nil(t, u)
|
|
CheckErrorID(t, err, "api.context.permissions.app_error")
|
|
require.Equal(t, http.StatusForbidden, resp.StatusCode)
|
|
})
|
|
|
|
t.Run("success", func(t *testing.T) {
|
|
us := &model.UploadSession{
|
|
Filename: info.Name(),
|
|
FileSize: info.Size(),
|
|
Type: model.UploadTypeImport,
|
|
}
|
|
u, _, err := th.SystemAdminClient.CreateUpload(context.Background(), us)
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, u)
|
|
})
|
|
})
|
|
|
|
t.Run("should clean filename", func(t *testing.T) {
|
|
us := &model.UploadSession{
|
|
ChannelId: th.BasicChannel.Id,
|
|
Filename: "../../../image.png",
|
|
FileSize: 8 * 1024 * 1024,
|
|
}
|
|
|
|
u, resp, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, u)
|
|
require.Equal(t, http.StatusCreated, resp.StatusCode)
|
|
|
|
require.Equal(t, "image.png", u.Filename)
|
|
|
|
rus, appErr := th.App.GetUploadSession(th.Context, u.Id)
|
|
require.Nil(t, appErr)
|
|
require.Equal(t, "image.png", rus.Filename)
|
|
require.Equal(
|
|
t,
|
|
fmt.Sprintf(
|
|
"%s/teams/noteam/channels/%s/users/%s/%s/image.png",
|
|
model.GetTimeForMillis(u.CreateAt).Format("20060102"),
|
|
u.ChannelId,
|
|
u.UserId,
|
|
u.Id,
|
|
),
|
|
rus.Path,
|
|
)
|
|
})
|
|
}
|
|
|
|
func TestGetUpload(t *testing.T) {
|
|
mainHelper.Parallel(t)
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
|
|
us := &model.UploadSession{
|
|
Id: model.NewId(),
|
|
Type: model.UploadTypeAttachment,
|
|
CreateAt: model.GetMillis(),
|
|
UserId: th.BasicUser2.Id,
|
|
ChannelId: th.BasicChannel.Id,
|
|
Filename: "upload",
|
|
FileSize: 8 * 1024 * 1024,
|
|
}
|
|
us, err := th.App.CreateUploadSession(th.Context, us)
|
|
require.Nil(t, err)
|
|
require.NotNil(t, us)
|
|
require.NotEmpty(t, us)
|
|
|
|
t.Run("upload not found", func(t *testing.T) {
|
|
u, resp, err := th.Client.GetUpload(context.Background(), model.NewId())
|
|
require.Nil(t, u)
|
|
CheckErrorID(t, err, "app.upload.get.app_error")
|
|
require.Equal(t, http.StatusNotFound, resp.StatusCode)
|
|
})
|
|
|
|
t.Run("no permissions", func(t *testing.T) {
|
|
u, _, err := th.Client.GetUpload(context.Background(), us.Id)
|
|
require.Nil(t, u)
|
|
CheckErrorID(t, err, "api.upload.get_upload.forbidden.app_error")
|
|
})
|
|
|
|
t.Run("success", func(t *testing.T) {
|
|
expected, resp, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, expected)
|
|
require.Equal(t, http.StatusCreated, resp.StatusCode)
|
|
|
|
u, _, err := th.Client.GetUpload(context.Background(), expected.Id)
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, u)
|
|
require.Equal(t, expected, u)
|
|
})
|
|
}
|
|
|
|
func TestGetUploadsForUser(t *testing.T) {
|
|
mainHelper.Parallel(t)
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
|
|
t.Run("no permissions", func(t *testing.T) {
|
|
uss, _, err := th.Client.GetUploadsForUser(context.Background(), th.BasicUser2.Id)
|
|
require.Error(t, err)
|
|
CheckErrorID(t, err, "api.user.get_uploads_for_user.forbidden.app_error")
|
|
require.Nil(t, uss)
|
|
})
|
|
|
|
t.Run("empty", func(t *testing.T) {
|
|
uss, _, err := th.Client.GetUploadsForUser(context.Background(), th.BasicUser.Id)
|
|
require.NoError(t, err)
|
|
require.Empty(t, uss)
|
|
})
|
|
|
|
t.Run("success", func(t *testing.T) {
|
|
uploads := make([]*model.UploadSession, 4)
|
|
for i := range uploads {
|
|
us := &model.UploadSession{
|
|
Id: model.NewId(),
|
|
Type: model.UploadTypeAttachment,
|
|
CreateAt: model.GetMillis(),
|
|
UserId: th.BasicUser.Id,
|
|
ChannelId: th.BasicChannel.Id,
|
|
Filename: "upload",
|
|
FileSize: 8 * 1024 * 1024,
|
|
}
|
|
us, err := th.App.CreateUploadSession(th.Context, us)
|
|
require.Nil(t, err)
|
|
require.NotNil(t, us)
|
|
require.NotEmpty(t, us)
|
|
us.Path = ""
|
|
uploads[i] = us
|
|
}
|
|
|
|
uss, _, err := th.Client.GetUploadsForUser(context.Background(), th.BasicUser.Id)
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, uss)
|
|
require.Len(t, uss, len(uploads))
|
|
for i := range uploads {
|
|
require.Contains(t, uss, uploads[i])
|
|
}
|
|
})
|
|
}
|
|
|
|
func TestUploadData(t *testing.T) {
|
|
mainHelper.Parallel(t)
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
if *th.App.Config().FileSettings.DriverName == "" {
|
|
t.Skip("skipping because no file driver is enabled")
|
|
}
|
|
|
|
us := &model.UploadSession{
|
|
Id: model.NewId(),
|
|
Type: model.UploadTypeAttachment,
|
|
CreateAt: model.GetMillis(),
|
|
UserId: th.BasicUser2.Id,
|
|
ChannelId: th.BasicChannel.Id,
|
|
Filename: "upload.zip",
|
|
FileSize: 8 * 1024 * 1024,
|
|
}
|
|
us, err := th.App.CreateUploadSession(th.Context, us)
|
|
require.Nil(t, err)
|
|
require.NotNil(t, us)
|
|
require.NotEmpty(t, us)
|
|
|
|
data := randomBytes(t, int(us.FileSize))
|
|
|
|
t.Run("file attachments disabled", func(t *testing.T) {
|
|
th.App.UpdateConfig(func(cfg *model.Config) { *cfg.FileSettings.EnableFileAttachments = false })
|
|
defer th.App.UpdateConfig(func(cfg *model.Config) { *cfg.FileSettings.EnableFileAttachments = true })
|
|
info, _, err := th.Client.UploadData(context.Background(), model.NewId(), bytes.NewReader(data))
|
|
require.Nil(t, info)
|
|
CheckErrorID(t, err, "api.file.attachments.disabled.app_error")
|
|
})
|
|
|
|
t.Run("upload not found", func(t *testing.T) {
|
|
info, resp, err := th.Client.UploadData(context.Background(), model.NewId(), bytes.NewReader(data))
|
|
require.Nil(t, info)
|
|
CheckErrorID(t, err, "app.upload.get.app_error")
|
|
require.Equal(t, http.StatusNotFound, resp.StatusCode)
|
|
})
|
|
|
|
t.Run("no permissions", func(t *testing.T) {
|
|
info, _, err := th.Client.UploadData(context.Background(), us.Id, bytes.NewReader(data))
|
|
require.Nil(t, info)
|
|
CheckErrorID(t, err, "api.context.permissions.app_error")
|
|
})
|
|
|
|
t.Run("not allowed in cloud", func(t *testing.T) {
|
|
th.App.Srv().SetLicense(model.NewTestLicense("cloud"))
|
|
defer func() {
|
|
appErr := th.App.Srv().RemoveLicense()
|
|
require.Nil(t, appErr)
|
|
}()
|
|
|
|
us2 := &model.UploadSession{
|
|
Id: model.NewId(),
|
|
Type: model.UploadTypeImport,
|
|
CreateAt: model.GetMillis(),
|
|
UserId: th.BasicUser2.Id,
|
|
ChannelId: th.BasicChannel.Id,
|
|
Filename: "upload",
|
|
FileSize: 8 * 1024 * 1024,
|
|
}
|
|
_, appErr := th.App.CreateUploadSession(th.Context, us2)
|
|
require.Nil(t, appErr)
|
|
|
|
info, resp, err := th.SystemAdminClient.UploadData(context.Background(), us2.Id, bytes.NewReader(data))
|
|
require.Nil(t, info)
|
|
CheckErrorID(t, err, "api.file.cloud_upload.app_error")
|
|
require.Equal(t, http.StatusBadRequest, resp.StatusCode)
|
|
})
|
|
|
|
t.Run("bad content-length", func(t *testing.T) {
|
|
u, resp, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, u)
|
|
require.Equal(t, http.StatusCreated, resp.StatusCode)
|
|
|
|
info, _, err := th.Client.UploadData(context.Background(), u.Id, bytes.NewReader(append(data, 0x00)))
|
|
require.Nil(t, info)
|
|
CheckErrorID(t, err, "api.upload.upload_data.invalid_content_length")
|
|
})
|
|
|
|
t.Run("success", func(t *testing.T) {
|
|
u, resp, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, u)
|
|
require.Equal(t, http.StatusCreated, resp.StatusCode)
|
|
|
|
info, _, err := th.Client.UploadData(context.Background(), u.Id, bytes.NewReader(data))
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, info)
|
|
require.Equal(t, u.Filename, info.Name)
|
|
require.Equal(t, u.FileSize, info.Size)
|
|
require.Equal(t, "zip", info.Extension)
|
|
require.Equal(t, "application/zip", info.MimeType)
|
|
|
|
file, _, err := th.Client.GetFile(context.Background(), info.Id)
|
|
require.NoError(t, err)
|
|
require.Equal(t, file, data)
|
|
})
|
|
|
|
t.Run("resume success", func(t *testing.T) {
|
|
u, resp, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, u)
|
|
require.Equal(t, http.StatusCreated, resp.StatusCode)
|
|
|
|
rd := &io.LimitedReader{
|
|
R: bytes.NewReader(data),
|
|
N: 5 * 1024 * 1024,
|
|
}
|
|
info, resp, err := th.Client.UploadData(context.Background(), u.Id, rd)
|
|
require.NoError(t, err)
|
|
require.Nil(t, info)
|
|
require.Equal(t, http.StatusNoContent, resp.StatusCode)
|
|
|
|
info, _, err = th.Client.UploadData(context.Background(), u.Id, bytes.NewReader(data[5*1024*1024:]))
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, info)
|
|
require.Equal(t, u.Filename, info.Name)
|
|
|
|
file, _, err := th.Client.GetFile(context.Background(), info.Id)
|
|
require.NoError(t, err)
|
|
require.Equal(t, file, data)
|
|
})
|
|
}
|
|
|
|
func TestUploadDataMultipart(t *testing.T) {
|
|
mainHelper.Parallel(t)
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
if *th.App.Config().FileSettings.DriverName == "" {
|
|
t.Skip("skipping because no file driver is enabled")
|
|
}
|
|
|
|
us := &model.UploadSession{
|
|
Id: model.NewId(),
|
|
Type: model.UploadTypeAttachment,
|
|
CreateAt: model.GetMillis(),
|
|
UserId: th.BasicUser.Id,
|
|
ChannelId: th.BasicChannel.Id,
|
|
Filename: "upload",
|
|
FileSize: 8 * 1024 * 1024,
|
|
}
|
|
us, _, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.NoError(t, err)
|
|
require.NotNil(t, us)
|
|
require.NotEmpty(t, us)
|
|
|
|
data := randomBytes(t, int(us.FileSize))
|
|
|
|
genMultipartData := func(t *testing.T, data []byte) (io.Reader, string) {
|
|
mpData := &bytes.Buffer{}
|
|
mpWriter := multipart.NewWriter(mpData)
|
|
part, err := mpWriter.CreateFormFile("data", us.Filename)
|
|
require.NoError(t, err)
|
|
n, err := part.Write(data)
|
|
require.NoError(t, err)
|
|
require.Equal(t, len(data), n)
|
|
err = mpWriter.Close()
|
|
require.NoError(t, err)
|
|
return mpData, mpWriter.FormDataContentType()
|
|
}
|
|
|
|
t.Run("bad content-type", func(t *testing.T) {
|
|
info, _, err := th.Client.DoUploadFile(context.Background(), "/uploads/"+us.Id, data, "multipart/form-data;")
|
|
require.Nil(t, info)
|
|
CheckErrorID(t, err, "api.upload.upload_data.invalid_content_type")
|
|
})
|
|
|
|
t.Run("success", func(t *testing.T) {
|
|
mpData, contentType := genMultipartData(t, data)
|
|
|
|
req, err := http.NewRequest("POST", th.Client.APIURL+"/uploads/"+us.Id, mpData)
|
|
require.NoError(t, err)
|
|
req.Header.Set("Content-Type", contentType)
|
|
req.Header.Set(model.HeaderAuth, th.Client.AuthType+" "+th.Client.AuthToken)
|
|
res, err := th.Client.HTTPClient.Do(req)
|
|
require.NoError(t, err)
|
|
var info model.FileInfo
|
|
err = json.NewDecoder(res.Body).Decode(&info)
|
|
res.Body.Close()
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, info)
|
|
require.Equal(t, us.Filename, info.Name)
|
|
|
|
file, _, err := th.Client.GetFile(context.Background(), info.Id)
|
|
require.NoError(t, err)
|
|
require.Equal(t, file, data)
|
|
})
|
|
|
|
t.Run("resume success", func(t *testing.T) {
|
|
mpData, contentType := genMultipartData(t, data[:5*1024*1024])
|
|
|
|
u, _, err := th.Client.CreateUpload(context.Background(), us)
|
|
require.NoError(t, err)
|
|
require.NotNil(t, u)
|
|
require.NotEmpty(t, u)
|
|
|
|
req, err := http.NewRequest("POST", th.Client.APIURL+"/uploads/"+u.Id, mpData)
|
|
require.NoError(t, err)
|
|
req.Header.Set("Content-Type", contentType)
|
|
req.Header.Set(model.HeaderAuth, th.Client.AuthType+" "+th.Client.AuthToken)
|
|
res, err := th.Client.HTTPClient.Do(req)
|
|
require.NoError(t, err)
|
|
require.Equal(t, http.StatusNoContent, res.StatusCode)
|
|
require.Equal(t, int64(0), res.ContentLength)
|
|
|
|
mpData, contentType = genMultipartData(t, data[5*1024*1024:])
|
|
|
|
req, err = http.NewRequest("POST", th.Client.APIURL+"/uploads/"+u.Id, mpData)
|
|
require.NoError(t, err)
|
|
req.Header.Set("Content-Type", contentType)
|
|
req.Header.Set(model.HeaderAuth, th.Client.AuthType+" "+th.Client.AuthToken)
|
|
res, err = th.Client.HTTPClient.Do(req)
|
|
require.NoError(t, err)
|
|
var info model.FileInfo
|
|
err = json.NewDecoder(res.Body).Decode(&info)
|
|
res.Body.Close()
|
|
require.NoError(t, err)
|
|
require.NotEmpty(t, info)
|
|
require.Equal(t, u.Filename, info.Name)
|
|
|
|
file, _, err := th.Client.GetFile(context.Background(), info.Id)
|
|
require.NoError(t, err)
|
|
require.Equal(t, file, data)
|
|
})
|
|
}
|