github.com/dschalla/mattermost-server@v4.8.1-rc1+incompatible/app/emoji.go (about)

     1  // Copyright (c) 2017-present Mattermost, Inc. All Rights Reserved.
     2  // See License.txt for license information.
     3  
     4  package app
     5  
     6  import (
     7  	"bytes"
     8  	"image"
     9  	"image/draw"
    10  	"image/gif"
    11  	_ "image/jpeg"
    12  	"image/png"
    13  	"io"
    14  	"mime/multipart"
    15  	"net/http"
    16  
    17  	l4g "github.com/alecthomas/log4go"
    18  
    19  	"image/color/palette"
    20  
    21  	"github.com/disintegration/imaging"
    22  	"github.com/mattermost/mattermost-server/model"
    23  	"github.com/mattermost/mattermost-server/utils"
    24  )
    25  
    26  const (
    27  	MaxEmojiFileSize = 1000 * 1024 // 1 MB
    28  	MaxEmojiWidth    = 128
    29  	MaxEmojiHeight   = 128
    30  )
    31  
    32  func (a *App) CreateEmoji(sessionUserId string, emoji *model.Emoji, multiPartImageData *multipart.Form) (*model.Emoji, *model.AppError) {
    33  	// wipe the emoji id so that existing emojis can't get overwritten
    34  	emoji.Id = ""
    35  
    36  	// do our best to validate the emoji before committing anything to the DB so that we don't have to clean up
    37  	// orphaned files left over when validation fails later on
    38  	emoji.PreSave()
    39  	if err := emoji.IsValid(); err != nil {
    40  		return nil, err
    41  	}
    42  
    43  	if emoji.CreatorId != sessionUserId {
    44  		return nil, model.NewAppError("createEmoji", "api.emoji.create.other_user.app_error", nil, "", http.StatusForbidden)
    45  	}
    46  
    47  	if result := <-a.Srv.Store.Emoji().GetByName(emoji.Name); result.Err == nil && result.Data != nil {
    48  		return nil, model.NewAppError("createEmoji", "api.emoji.create.duplicate.app_error", nil, "", http.StatusBadRequest)
    49  	}
    50  
    51  	if imageData := multiPartImageData.File["image"]; len(imageData) == 0 {
    52  		err := model.NewAppError("Context", "api.context.invalid_body_param.app_error", map[string]interface{}{"Name": "createEmoji"}, "", http.StatusBadRequest)
    53  		return nil, err
    54  	} else if err := a.UploadEmojiImage(emoji.Id, imageData[0]); err != nil {
    55  		return nil, err
    56  	}
    57  
    58  	if result := <-a.Srv.Store.Emoji().Save(emoji); result.Err != nil {
    59  		return nil, result.Err
    60  	} else {
    61  		message := model.NewWebSocketEvent(model.WEBSOCKET_EVENT_EMOJI_ADDED, "", "", "", nil)
    62  		message.Add("emoji", emoji.ToJson())
    63  
    64  		a.Publish(message)
    65  		return result.Data.(*model.Emoji), nil
    66  	}
    67  }
    68  
    69  func (a *App) GetEmojiList(page, perPage int, sort string) ([]*model.Emoji, *model.AppError) {
    70  	if result := <-a.Srv.Store.Emoji().GetList(page*perPage, perPage, sort); result.Err != nil {
    71  		return nil, result.Err
    72  	} else {
    73  		return result.Data.([]*model.Emoji), nil
    74  	}
    75  }
    76  
    77  func (a *App) UploadEmojiImage(id string, imageData *multipart.FileHeader) *model.AppError {
    78  	file, err := imageData.Open()
    79  	if err != nil {
    80  		return model.NewAppError("uploadEmojiImage", "api.emoji.upload.open.app_error", nil, "", http.StatusBadRequest)
    81  	}
    82  	defer file.Close()
    83  
    84  	buf := bytes.NewBuffer(nil)
    85  	io.Copy(buf, file)
    86  
    87  	// make sure the file is an image and is within the required dimensions
    88  	if config, _, err := image.DecodeConfig(bytes.NewReader(buf.Bytes())); err != nil {
    89  		return model.NewAppError("uploadEmojiImage", "api.emoji.upload.image.app_error", nil, "", http.StatusBadRequest)
    90  	} else if config.Width > MaxEmojiWidth || config.Height > MaxEmojiHeight {
    91  		data := buf.Bytes()
    92  		newbuf := bytes.NewBuffer(nil)
    93  		if info, err := model.GetInfoForBytes(imageData.Filename, data); err != nil {
    94  			return err
    95  		} else if info.MimeType == "image/gif" {
    96  			if gif_data, err := gif.DecodeAll(bytes.NewReader(data)); err != nil {
    97  				return model.NewAppError("uploadEmojiImage", "api.emoji.upload.large_image.gif_decode_error", nil, "", http.StatusBadRequest)
    98  			} else {
    99  				resized_gif := resizeEmojiGif(gif_data)
   100  				if err := gif.EncodeAll(newbuf, resized_gif); err != nil {
   101  					return model.NewAppError("uploadEmojiImage", "api.emoji.upload.large_image.gif_encode_error", nil, "", http.StatusBadRequest)
   102  				}
   103  				if err := a.WriteFile(newbuf.Bytes(), getEmojiImagePath(id)); err != nil {
   104  					return err
   105  				}
   106  			}
   107  		} else {
   108  			if img, _, err := image.Decode(bytes.NewReader(data)); err != nil {
   109  				return model.NewAppError("uploadEmojiImage", "api.emoji.upload.large_image.decode_error", nil, "", http.StatusBadRequest)
   110  			} else {
   111  				resized_image := resizeEmoji(img, config.Width, config.Height)
   112  				if err := png.Encode(newbuf, resized_image); err != nil {
   113  					return model.NewAppError("uploadEmojiImage", "api.emoji.upload.large_image.encode_error", nil, "", http.StatusBadRequest)
   114  				}
   115  				if err := a.WriteFile(newbuf.Bytes(), getEmojiImagePath(id)); err != nil {
   116  					return err
   117  				}
   118  			}
   119  		}
   120  	}
   121  
   122  	return a.WriteFile(buf.Bytes(), getEmojiImagePath(id))
   123  }
   124  
   125  func (a *App) DeleteEmoji(emoji *model.Emoji) *model.AppError {
   126  	if err := (<-a.Srv.Store.Emoji().Delete(emoji.Id, model.GetMillis())).Err; err != nil {
   127  		return err
   128  	}
   129  
   130  	a.deleteEmojiImage(emoji.Id)
   131  	a.deleteReactionsForEmoji(emoji.Name)
   132  	return nil
   133  }
   134  
   135  func (a *App) GetEmoji(emojiId string) (*model.Emoji, *model.AppError) {
   136  	if !*a.Config().ServiceSettings.EnableCustomEmoji {
   137  		return nil, model.NewAppError("GetEmoji", "api.emoji.disabled.app_error", nil, "", http.StatusNotImplemented)
   138  	}
   139  
   140  	if len(*a.Config().FileSettings.DriverName) == 0 {
   141  		return nil, model.NewAppError("GetEmoji", "api.emoji.storage.app_error", nil, "", http.StatusNotImplemented)
   142  	}
   143  
   144  	if result := <-a.Srv.Store.Emoji().Get(emojiId, false); result.Err != nil {
   145  		return nil, result.Err
   146  	} else {
   147  		return result.Data.(*model.Emoji), nil
   148  	}
   149  }
   150  
   151  func (a *App) GetEmojiByName(emojiName string) (*model.Emoji, *model.AppError) {
   152  	if !*a.Config().ServiceSettings.EnableCustomEmoji {
   153  		return nil, model.NewAppError("GetEmoji", "api.emoji.disabled.app_error", nil, "", http.StatusNotImplemented)
   154  	}
   155  
   156  	if len(*a.Config().FileSettings.DriverName) == 0 {
   157  		return nil, model.NewAppError("GetEmoji", "api.emoji.storage.app_error", nil, "", http.StatusNotImplemented)
   158  	}
   159  
   160  	if result := <-a.Srv.Store.Emoji().GetByName(emojiName); result.Err != nil {
   161  		return nil, result.Err
   162  	} else {
   163  		return result.Data.(*model.Emoji), nil
   164  	}
   165  }
   166  
   167  func (a *App) GetEmojiImage(emojiId string) (imageByte []byte, imageType string, err *model.AppError) {
   168  	if result := <-a.Srv.Store.Emoji().Get(emojiId, true); result.Err != nil {
   169  		return nil, "", result.Err
   170  	} else {
   171  		var img []byte
   172  
   173  		if data, err := a.ReadFile(getEmojiImagePath(emojiId)); err != nil {
   174  			return nil, "", model.NewAppError("getEmojiImage", "api.emoji.get_image.read.app_error", nil, err.Error(), http.StatusNotFound)
   175  		} else {
   176  			img = data
   177  		}
   178  
   179  		_, imageType, err := image.DecodeConfig(bytes.NewReader(img))
   180  		if err != nil {
   181  			return nil, "", model.NewAppError("getEmojiImage", "api.emoji.get_image.decode.app_error", nil, err.Error(), http.StatusInternalServerError)
   182  		}
   183  
   184  		return img, imageType, nil
   185  	}
   186  }
   187  
   188  func (a *App) SearchEmoji(name string, prefixOnly bool, limit int) ([]*model.Emoji, *model.AppError) {
   189  	if !*a.Config().ServiceSettings.EnableCustomEmoji {
   190  		return nil, model.NewAppError("SearchEmoji", "api.emoji.disabled.app_error", nil, "", http.StatusNotImplemented)
   191  	}
   192  
   193  	if result := <-a.Srv.Store.Emoji().Search(name, prefixOnly, limit); result.Err != nil {
   194  		return nil, result.Err
   195  	} else {
   196  		return result.Data.([]*model.Emoji), nil
   197  	}
   198  }
   199  
   200  func resizeEmojiGif(gifImg *gif.GIF) *gif.GIF {
   201  	// Create a new RGBA image to hold the incremental frames.
   202  	firstFrame := gifImg.Image[0].Bounds()
   203  	b := image.Rect(0, 0, firstFrame.Dx(), firstFrame.Dy())
   204  	img := image.NewRGBA(b)
   205  
   206  	resizedImage := image.Image(nil)
   207  	// Resize each frame.
   208  	for index, frame := range gifImg.Image {
   209  		bounds := frame.Bounds()
   210  		draw.Draw(img, bounds, frame, bounds.Min, draw.Over)
   211  		resizedImage = resizeEmoji(img, firstFrame.Dx(), firstFrame.Dy())
   212  		gifImg.Image[index] = imageToPaletted(resizedImage)
   213  	}
   214  	// Set new gif width and height
   215  	gifImg.Config.Width = resizedImage.Bounds().Dx()
   216  	gifImg.Config.Height = resizedImage.Bounds().Dy()
   217  	return gifImg
   218  }
   219  
   220  func getEmojiImagePath(id string) string {
   221  	return "emoji/" + id + "/image"
   222  }
   223  
   224  func resizeEmoji(img image.Image, width int, height int) image.Image {
   225  	emojiWidth := float64(width)
   226  	emojiHeight := float64(height)
   227  
   228  	var emoji image.Image
   229  	if emojiHeight <= MaxEmojiHeight && emojiWidth <= MaxEmojiWidth {
   230  		emoji = img
   231  	} else {
   232  		emoji = imaging.Fit(img, MaxEmojiWidth, MaxEmojiHeight, imaging.Lanczos)
   233  	}
   234  	return emoji
   235  }
   236  
   237  func imageToPaletted(img image.Image) *image.Paletted {
   238  	b := img.Bounds()
   239  	pm := image.NewPaletted(b, palette.Plan9)
   240  	draw.FloydSteinberg.Draw(pm, b, img, image.ZP)
   241  	return pm
   242  }
   243  
   244  func (a *App) deleteEmojiImage(id string) {
   245  	if err := a.MoveFile(getEmojiImagePath(id), "emoji/"+id+"/image_deleted"); err != nil {
   246  		l4g.Error("Failed to rename image when deleting emoji %v", id)
   247  	}
   248  }
   249  
   250  func (a *App) deleteReactionsForEmoji(emojiName string) {
   251  	if result := <-a.Srv.Store.Reaction().DeleteAllWithEmojiName(emojiName); result.Err != nil {
   252  		l4g.Warn(utils.T("api.emoji.delete.delete_reactions.app_error"), emojiName)
   253  		l4g.Warn(result.Err)
   254  	}
   255  }