2020-10-19 21:40:00 +00:00
|
|
|
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
|
|
// See LICENSE.txt for license information.
|
|
|
|
|
|
|
|
package model
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
)
|
|
|
|
|
|
|
|
// UploadType defines the type of an upload.
|
|
|
|
type UploadType string
|
|
|
|
|
|
|
|
const (
|
|
|
|
UploadTypeAttachment UploadType = "attachment"
|
|
|
|
UploadTypeImport UploadType = "import"
|
|
|
|
)
|
|
|
|
|
2021-10-16 22:47:22 +00:00
|
|
|
// UploadNoUserID is a "fake" user id used by the API layer when in local mode.
|
|
|
|
const UploadNoUserID = "nouser"
|
|
|
|
|
2020-10-19 21:40:00 +00:00
|
|
|
// UploadSession contains information used to keep track of a file upload.
|
|
|
|
type UploadSession struct {
|
|
|
|
// The unique identifier for the session.
|
|
|
|
Id string `json:"id"`
|
|
|
|
// The type of the upload.
|
|
|
|
Type UploadType `json:"type"`
|
|
|
|
// The timestamp of creation.
|
|
|
|
CreateAt int64 `json:"create_at"`
|
|
|
|
// The id of the user performing the upload.
|
|
|
|
UserId string `json:"user_id"`
|
|
|
|
// The id of the channel to upload to.
|
2021-10-16 22:47:22 +00:00
|
|
|
ChannelId string `json:"channel_id,omitempty"`
|
2020-10-19 21:40:00 +00:00
|
|
|
// The name of the file to upload.
|
|
|
|
Filename string `json:"filename"`
|
|
|
|
// The path where the file is stored.
|
|
|
|
Path string `json:"-"`
|
|
|
|
// The size of the file to upload.
|
|
|
|
FileSize int64 `json:"file_size"`
|
|
|
|
// The amount of received data in bytes. If equal to FileSize it means the
|
|
|
|
// upload has finished.
|
|
|
|
FileOffset int64 `json:"file_offset"`
|
2021-10-16 22:47:22 +00:00
|
|
|
// Id of remote cluster if uploading for shared channel
|
|
|
|
RemoteId string `json:"remote_id"`
|
|
|
|
// Requested file id if uploading for shared channel
|
|
|
|
ReqFileId string `json:"req_file_id"`
|
2020-10-19 21:40:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ToJson serializes the UploadSession into JSON and returns it as string.
|
|
|
|
func (us *UploadSession) ToJson() string {
|
|
|
|
b, _ := json.Marshal(us)
|
|
|
|
return string(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UploadSessionsToJson serializes a list of UploadSession into JSON and
|
|
|
|
// returns it as string.
|
|
|
|
func UploadSessionsToJson(uss []*UploadSession) string {
|
|
|
|
b, _ := json.Marshal(uss)
|
|
|
|
return string(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UploadSessionsFromJson deserializes a list of UploadSession from JSON data.
|
|
|
|
func UploadSessionsFromJson(data io.Reader) []*UploadSession {
|
|
|
|
decoder := json.NewDecoder(data)
|
|
|
|
var uss []*UploadSession
|
|
|
|
if err := decoder.Decode(&uss); err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return uss
|
|
|
|
}
|
|
|
|
|
|
|
|
// UploadSessionFromJson deserializes the UploadSession from JSON data.
|
|
|
|
func UploadSessionFromJson(data io.Reader) *UploadSession {
|
|
|
|
decoder := json.NewDecoder(data)
|
|
|
|
var us UploadSession
|
|
|
|
if err := decoder.Decode(&us); err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return &us
|
|
|
|
}
|
|
|
|
|
|
|
|
// PreSave is a utility function used to fill required information.
|
|
|
|
func (us *UploadSession) PreSave() {
|
|
|
|
if us.Id == "" {
|
|
|
|
us.Id = NewId()
|
|
|
|
}
|
|
|
|
|
|
|
|
if us.CreateAt == 0 {
|
|
|
|
us.CreateAt = GetMillis()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsValid validates an UploadType. It returns an error in case of
|
|
|
|
// failure.
|
|
|
|
func (t UploadType) IsValid() error {
|
|
|
|
switch t {
|
|
|
|
case UploadTypeAttachment:
|
|
|
|
return nil
|
|
|
|
case UploadTypeImport:
|
|
|
|
return nil
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
return fmt.Errorf("invalid UploadType %s", t)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsValid validates an UploadSession. It returns an error in case of
|
|
|
|
// failure.
|
|
|
|
func (us *UploadSession) IsValid() *AppError {
|
|
|
|
if !IsValidId(us.Id) {
|
|
|
|
return NewAppError("UploadSession.IsValid", "model.upload_session.is_valid.id.app_error", nil, "", http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := us.Type.IsValid(); err != nil {
|
|
|
|
return NewAppError("UploadSession.IsValid", "model.upload_session.is_valid.type.app_error", nil, err.Error(), http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
|
2021-10-16 22:47:22 +00:00
|
|
|
if !IsValidId(us.UserId) && us.UserId != UploadNoUserID {
|
2020-10-19 21:40:00 +00:00
|
|
|
return NewAppError("UploadSession.IsValid", "model.upload_session.is_valid.user_id.app_error", nil, "id="+us.Id, http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
|
|
|
|
if us.Type == UploadTypeAttachment && !IsValidId(us.ChannelId) {
|
|
|
|
return NewAppError("UploadSession.IsValid", "model.upload_session.is_valid.channel_id.app_error", nil, "id="+us.Id, http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
|
|
|
|
if us.CreateAt == 0 {
|
|
|
|
return NewAppError("UploadSession.IsValid", "model.upload_session.is_valid.create_at.app_error", nil, "id="+us.Id, http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
|
|
|
|
if us.Filename == "" {
|
|
|
|
return NewAppError("UploadSession.IsValid", "model.upload_session.is_valid.filename.app_error", nil, "id="+us.Id, http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
|
|
|
|
if us.FileSize <= 0 {
|
|
|
|
return NewAppError("UploadSession.IsValid", "model.upload_session.is_valid.file_size.app_error", nil, "id="+us.Id, http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
|
|
|
|
if us.FileOffset < 0 || us.FileOffset > us.FileSize {
|
|
|
|
return NewAppError("UploadSession.IsValid", "model.upload_session.is_valid.file_offset.app_error", nil, "id="+us.Id, http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
|
|
|
|
if us.Path == "" {
|
|
|
|
return NewAppError("UploadSession.IsValid", "model.upload_session.is_valid.path.app_error", nil, "id="+us.Id, http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|