Skip to content

Commit

Permalink
Merge bccdf81 into 742bab6
Browse files Browse the repository at this point in the history
  • Loading branch information
saturninoabril committed Apr 17, 2017
2 parents 742bab6 + bccdf81 commit 6938f66
Show file tree
Hide file tree
Showing 7 changed files with 202 additions and 8 deletions.
11 changes: 3 additions & 8 deletions api/reaction.go
Expand Up @@ -65,17 +65,12 @@ func saveReaction(c *Context, w http.ResponseWriter, r *http.Request) {
return
}

if result := <-app.Srv.Store.Reaction().Save(reaction); result.Err != nil {
c.Err = result.Err
if reaction, err := app.SaveReactionForPost(reaction); err != nil {
c.Err = err
return
} else {
go sendReactionEvent(model.WEBSOCKET_EVENT_REACTION_ADDED, channelId, reaction, post)

reaction := result.Data.(*model.Reaction)

app.InvalidateCacheForReactions(reaction.PostId)

w.Write([]byte(reaction.ToJson()))
return
}
}

Expand Down
3 changes: 3 additions & 0 deletions api4/api.go
Expand Up @@ -82,6 +82,8 @@ type Routes struct {

Public *mux.Router // 'api/v4/public'

Reactions *mux.Router // 'api/v4/reactions'

Emojis *mux.Router // 'api/v4/emoji'
Emoji *mux.Router // 'api/v4/emoji/{emoji_id:[A-Za-z0-9]+}'

Expand Down Expand Up @@ -154,6 +156,7 @@ func InitApi(full bool) {
BaseRoutes.Preferences = BaseRoutes.User.PathPrefix("/preferences").Subrouter()
BaseRoutes.License = BaseRoutes.ApiRoot.PathPrefix("/license").Subrouter()
BaseRoutes.Public = BaseRoutes.ApiRoot.PathPrefix("/public").Subrouter()
BaseRoutes.Reactions = BaseRoutes.ApiRoot.PathPrefix("/reactions").Subrouter()

BaseRoutes.Emojis = BaseRoutes.ApiRoot.PathPrefix("/emoji").Subrouter()
BaseRoutes.Emoji = BaseRoutes.Emojis.PathPrefix("/{emoji_id:[A-Za-z0-9]+}").Subrouter()
Expand Down
34 changes: 34 additions & 0 deletions api4/reaction.go
Expand Up @@ -15,9 +15,43 @@ import (
func InitReaction() {
l4g.Debug(utils.T("api.reaction.init.debug"))

BaseRoutes.Reactions.Handle("", ApiSessionRequired(saveReaction)).Methods("POST")
BaseRoutes.Post.Handle("/reactions", ApiSessionRequired(getReactions)).Methods("GET")
}

func saveReaction(c *Context, w http.ResponseWriter, r *http.Request) {
reaction := model.ReactionFromJson(r.Body)
if reaction == nil {
c.SetInvalidParam("reaction")
return
}

if len(reaction.UserId) != 26 || len(reaction.PostId) != 26 || len(reaction.EmojiName) == 0 || len(reaction.EmojiName) > 64 {
c.Err = model.NewLocAppError("saveReaction", "api.reaction.save_reaction.invalid.app_error", nil, "")
c.Err.StatusCode = http.StatusBadRequest
return
}

if reaction.UserId != c.Session.UserId {
c.Err = model.NewLocAppError("saveReaction", "api.reaction.save_reaction.user_id.app_error", nil, "")
c.Err.StatusCode = http.StatusForbidden
return
}

if !app.SessionHasPermissionToChannelByPost(c.Session, reaction.PostId, model.PERMISSION_READ_CHANNEL) {
c.SetPermissionError(model.PERMISSION_READ_CHANNEL)
return
}

if reaction, err := app.SaveReactionForPost(reaction); err != nil {
c.Err = err
return
} else {
w.Write([]byte(reaction.ToJson()))
return
}
}

func getReactions(c *Context, w http.ResponseWriter, r *http.Request) {
c.RequirePostId()
if c.Err != nil {
Expand Down
106 changes: 106 additions & 0 deletions api4/reaction_test.go
Expand Up @@ -4,6 +4,7 @@
package api4

import (
"strings"
"testing"

"reflect"
Expand All @@ -12,6 +13,111 @@ import (
"github.com/mattermost/platform/model"
)

func TestSaveReaction(t *testing.T) {
th := Setup().InitBasic().InitSystemAdmin()
defer TearDown()
Client := th.Client
userId := th.BasicUser.Id
postId := th.BasicPost.Id

reaction := &model.Reaction{
UserId: userId,
PostId: postId,
EmojiName: "smile",
}

rr, resp := Client.SaveReaction(reaction)
CheckNoError(t, resp)

if rr.UserId != reaction.UserId {
t.Fatal("UserId did not match")
}

if rr.PostId != reaction.PostId {
t.Fatal("PostId did not match")
}

if rr.EmojiName != reaction.EmojiName {
t.Fatal("EmojiName did not match")
}

if rr.CreateAt == 0 {
t.Fatal("CreateAt should exist")
}

if reactions, err := app.GetReactionsForPost(postId); err != nil && len(reactions) != 1 {
t.Fatal("didn't save reaction correctly")
}

// saving a duplicate reaction
rr, resp = Client.SaveReaction(reaction)
CheckNoError(t, resp)

if reactions, err := app.GetReactionsForPost(postId); err != nil && len(reactions) != 1 {
t.Fatal("should have not save duplicated reaction")
}

reaction.EmojiName = "sad"

rr, resp = Client.SaveReaction(reaction)
CheckNoError(t, resp)

if rr.EmojiName != reaction.EmojiName {
t.Fatal("EmojiName did not match")
}

if reactions, err := app.GetReactionsForPost(postId); err != nil && len(reactions) != 2 {
t.Fatal("should have save multiple reactions")
}

reaction.PostId = GenerateTestId()

_, resp = Client.SaveReaction(reaction)
CheckForbiddenStatus(t, resp)

reaction.PostId = "junk"

_, resp = Client.SaveReaction(reaction)
CheckBadRequestStatus(t, resp)

reaction.PostId = postId
reaction.UserId = GenerateTestId()

_, resp = Client.SaveReaction(reaction)
CheckForbiddenStatus(t, resp)

reaction.UserId = "junk"

_, resp = Client.SaveReaction(reaction)
CheckBadRequestStatus(t, resp)

reaction.UserId = userId
reaction.EmojiName = ""

_, resp = Client.SaveReaction(reaction)
CheckBadRequestStatus(t, resp)

reaction.EmojiName = strings.Repeat("a", 65)

_, resp = Client.SaveReaction(reaction)
CheckBadRequestStatus(t, resp)

reaction.EmojiName = "smile"
otherUser := th.CreateUser()
Client.Logout()
Client.Login(otherUser.Email, otherUser.Password)

_, resp = Client.SaveReaction(reaction)
CheckForbiddenStatus(t, resp)

Client.Logout()
_, resp = Client.SaveReaction(reaction)
CheckUnauthorizedStatus(t, resp)

_, resp = th.SystemAdminClient.SaveReaction(reaction)
CheckForbiddenStatus(t, resp)
}

func TestGetReactions(t *testing.T) {
th := Setup().InitBasic().InitSystemAdmin()
defer TearDown()
Expand Down
34 changes: 34 additions & 0 deletions app/reaction.go
Expand Up @@ -7,10 +7,44 @@ import (
"github.com/mattermost/platform/model"
)

func SaveReactionForPost(reaction *model.Reaction) (*model.Reaction, *model.AppError) {
post, err := GetSinglePost(reaction.PostId)
if err != nil {
return nil, err
}

if result := <-Srv.Store.Reaction().Save(reaction); result.Err != nil {
return nil, result.Err
} else {
reaction = result.Data.(*model.Reaction)

go sendReactionEvent(model.WEBSOCKET_EVENT_REACTION_ADDED, reaction, post)

InvalidateCacheForReactions(reaction.PostId)

return reaction, nil
}
}

func GetReactionsForPost(postId string) ([]*model.Reaction, *model.AppError) {
if result := <-Srv.Store.Reaction().GetForPost(postId, true); result.Err != nil {
return nil, result.Err
} else {
return result.Data.([]*model.Reaction), nil
}
}

func sendReactionEvent(event string, reaction *model.Reaction, post *model.Post) {
// send out that a reaction has been added/removed
message := model.NewWebSocketEvent(event, "", post.ChannelId, "", nil)
message.Add("reaction", reaction.ToJson())
Publish(message)

// The post is always modified since the UpdateAt always changes
InvalidateCacheForChannelPosts(post.ChannelId)
post.HasReactions = true
post.UpdateAt = model.GetMillis()
umessage := model.NewWebSocketEvent(model.WEBSOCKET_EVENT_POST_EDITED, "", post.ChannelId, "", nil)
umessage.Add("post", post.ToJson())
Publish(umessage)
}
8 changes: 8 additions & 0 deletions i18n/en.json
Expand Up @@ -1703,10 +1703,18 @@
"id": "api.reaction.list_reactions.mismatched_channel_id.app_error",
"translation": "Failed to get reactions because channel ID does not match post ID in the URL"
},
{
"id": "api.reaction.save_reaction.invalid.app_error",
"translation": "Reaction is not valid."
},
{
"id": "api.reaction.save_reaction.mismatched_channel_id.app_error",
"translation": "Failed to save reaction because channel ID does not match post ID in the URL"
},
{
"id": "api.reaction.save_reaction.user_id.app_error",
"translation": "You cannot save reaction for the other user."
},
{
"id": "api.reaction.send_reaction_event.post.app_error",
"translation": "Failed to get post when sending websocket event for reaction"
Expand Down
14 changes: 14 additions & 0 deletions model/client4.go
Expand Up @@ -238,6 +238,10 @@ func (c *Client4) GetEmojiRoute(emojiId string) string {
return fmt.Sprintf(c.GetEmojisRoute()+"/%v", emojiId)
}

func (c *Client4) GetReactionsRoute() string {
return fmt.Sprintf("/reactions")
}

func (c *Client4) DoApiGet(url string, etag string) (*http.Response, *AppError) {
return c.DoApiRequest(http.MethodGet, url, "", etag)
}
Expand Down Expand Up @@ -2374,6 +2378,16 @@ func (c *Client4) GetEmoji(emojiId string) (*Emoji, *Response) {

// Reaction Section

// SaveReaction saves an emoji reaction for a post. Returns the saved reaction if successful, otherwise an error will be returned.
func (c *Client4) SaveReaction(reaction *Reaction) (*Reaction, *Response) {
if r, err := c.DoApiPost(c.GetReactionsRoute(), reaction.ToJson()); err != nil {
return nil, &Response{StatusCode: r.StatusCode, Error: err}
} else {
defer closeBody(r)
return ReactionFromJson(r.Body), BuildResponse(r)
}
}

// GetReactions returns a list of reactions to a post.
func (c *Client4) GetReactions(postId string) ([]*Reaction, *Response) {
if r, err := c.DoApiGet(c.GetPostRoute(postId)+"/reactions", ""); err != nil {
Expand Down

0 comments on commit 6938f66

Please sign in to comment.