summaryrefslogtreecommitdiffstats
path: root/api/post.go
diff options
context:
space:
mode:
authorJoram Wilander <jwawilander@gmail.com>2017-01-25 09:32:42 -0500
committerHarrison Healey <harrisonmhealey@gmail.com>2017-01-25 09:32:42 -0500
commitd245b29f82a03f1aff966a2fb2100a5703d82f32 (patch)
treecf204039af74fa2579e0291e9e595f08411b0a7a /api/post.go
parent8ed665cb76c0763e83a2949c4bdd70153baf72f7 (diff)
downloadchat-d245b29f82a03f1aff966a2fb2100a5703d82f32.tar.gz
chat-d245b29f82a03f1aff966a2fb2100a5703d82f32.tar.bz2
chat-d245b29f82a03f1aff966a2fb2100a5703d82f32.zip
More app code migration (#5170)
* Migrate admin functions into app package * More user function refactoring * Move post functions into app package
Diffstat (limited to 'api/post.go')
-rw-r--r--api/post.go356
1 files changed, 57 insertions, 299 deletions
diff --git a/api/post.go b/api/post.go
index 13cfecfdf..4d1425c18 100644
--- a/api/post.go
+++ b/api/post.go
@@ -8,11 +8,9 @@ import (
"strconv"
l4g "github.com/alecthomas/log4go"
- "github.com/dyatlov/go-opengraph/opengraph"
"github.com/gorilla/mux"
"github.com/mattermost/platform/app"
"github.com/mattermost/platform/model"
- "github.com/mattermost/platform/store"
"github.com/mattermost/platform/utils"
)
@@ -44,67 +42,28 @@ func createPost(c *Context, w http.ResponseWriter, r *http.Request) {
c.SetInvalidParam("createPost", "post")
return
}
- post.UserId = c.Session.UserId
- cchan := app.Srv.Store.Channel().Get(post.ChannelId, true)
+ post.UserId = c.Session.UserId
if !app.SessionHasPermissionToChannel(c.Session, post.ChannelId, model.PERMISSION_CREATE_POST) {
c.SetPermissionError(model.PERMISSION_CREATE_POST)
return
}
- // Check that channel has not been deleted
- var channel *model.Channel
- if result := <-cchan; result.Err != nil {
- c.SetInvalidParam("createPost", "post.channelId")
- return
- } else {
- channel = result.Data.(*model.Channel)
- }
-
- if channel.DeleteAt != 0 {
- c.Err = model.NewLocAppError("createPost", "api.post.create_post.can_not_post_to_deleted.error", nil, "")
- c.Err.StatusCode = http.StatusBadRequest
- return
- }
-
if post.CreateAt != 0 && !app.SessionHasPermissionTo(c.Session, model.PERMISSION_MANAGE_SYSTEM) {
post.CreateAt = 0
}
- if rp, err := app.CreatePost(post, c.TeamId, true); err != nil {
+ rp, err := app.CreatePostAsUser(post, c.TeamId)
+ if err != nil {
c.Err = err
-
- if c.Err.Id == "api.post.create_post.root_id.app_error" ||
- c.Err.Id == "api.post.create_post.channel_root_id.app_error" ||
- c.Err.Id == "api.post.create_post.parent_id.app_error" {
- c.Err.StatusCode = http.StatusBadRequest
- }
-
return
- } else {
- // Update the LastViewAt only if the post does not have from_webhook prop set (eg. Zapier app)
- if _, ok := post.Props["from_webhook"]; !ok {
- if result := <-app.Srv.Store.Channel().UpdateLastViewedAt([]string{post.ChannelId}, c.Session.UserId); result.Err != nil {
- l4g.Error(utils.T("api.post.create_post.last_viewed.error"), post.ChannelId, c.Session.UserId, result.Err)
- }
- }
-
- w.Write([]byte(rp.ToJson()))
}
+
+ w.Write([]byte(rp.ToJson()))
}
func updatePost(c *Context, w http.ResponseWriter, r *http.Request) {
-
- if utils.IsLicensed {
- if *utils.Cfg.ServiceSettings.AllowEditPost == model.ALLOW_EDIT_POST_NEVER {
- c.Err = model.NewLocAppError("updatePost", "api.post.update_post.permissions.app_error", nil,
- c.T("api.post.update_post.permissions_denied.app_error"))
- c.Err.StatusCode = http.StatusForbidden
- return
- }
- }
-
post := model.PostFromJson(r.Body)
if post == nil {
@@ -112,77 +71,20 @@ func updatePost(c *Context, w http.ResponseWriter, r *http.Request) {
return
}
- pchan := app.Srv.Store.Post().Get(post.Id)
-
if !app.SessionHasPermissionToChannel(c.Session, post.ChannelId, model.PERMISSION_EDIT_POST) {
c.SetPermissionError(model.PERMISSION_EDIT_POST)
return
}
- var oldPost *model.Post
- if result := <-pchan; result.Err != nil {
- c.Err = result.Err
- return
- } else {
- oldPost = result.Data.(*model.PostList).Posts[post.Id]
-
- if oldPost == nil {
- c.Err = model.NewLocAppError("updatePost", "api.post.update_post.find.app_error", nil, "id="+post.Id)
- c.Err.StatusCode = http.StatusBadRequest
- return
- }
-
- if oldPost.UserId != c.Session.UserId {
- c.Err = model.NewLocAppError("updatePost", "api.post.update_post.permissions.app_error", nil, "oldUserId="+oldPost.UserId)
- c.Err.StatusCode = http.StatusForbidden
- return
- }
-
- if oldPost.DeleteAt != 0 {
- c.Err = model.NewLocAppError("updatePost", "api.post.update_post.permissions.app_error", nil,
- c.T("api.post.update_post.permissions_details.app_error", map[string]interface{}{"PostId": post.Id}))
- c.Err.StatusCode = http.StatusForbidden
- return
- }
-
- if oldPost.IsSystemMessage() {
- c.Err = model.NewLocAppError("updatePost", "api.post.update_post.system_message.app_error", nil, "id="+post.Id)
- c.Err.StatusCode = http.StatusForbidden
- return
- }
-
- if utils.IsLicensed {
- if *utils.Cfg.ServiceSettings.AllowEditPost == model.ALLOW_EDIT_POST_TIME_LIMIT && model.GetMillis() > oldPost.CreateAt+int64(*utils.Cfg.ServiceSettings.PostEditTimeLimit*1000) {
- c.Err = model.NewLocAppError("updatePost", "api.post.update_post.permissions.app_error", nil,
- c.T("api.post.update_post.permissions_time_limit.app_error", map[string]interface{}{"timeLimit": *utils.Cfg.ServiceSettings.PostEditTimeLimit}))
- c.Err.StatusCode = http.StatusForbidden
- return
- }
- }
- }
-
- newPost := &model.Post{}
- *newPost = *oldPost
-
- newPost.Message = post.Message
- newPost.EditAt = model.GetMillis()
- newPost.Hashtags, _ = model.ParseHashtags(post.Message)
+ post.UserId = c.Session.UserId
- if result := <-app.Srv.Store.Post().Update(newPost, oldPost); result.Err != nil {
- c.Err = result.Err
+ rpost, err := app.UpdatePost(post)
+ if err != nil {
+ c.Err = err
return
- } else {
- rpost := result.Data.(*model.Post)
-
- message := model.NewWebSocketEvent(model.WEBSOCKET_EVENT_POST_EDITED, "", rpost.ChannelId, "", nil)
- message.Add("post", rpost.ToJson())
-
- go app.Publish(message)
-
- app.InvalidateCacheForChannelPosts(rpost.ChannelId)
-
- w.Write([]byte(rpost.ToJson()))
}
+
+ w.Write([]byte(rpost.ToJson()))
}
func getFlaggedPosts(c *Context, w http.ResponseWriter, r *http.Request) {
@@ -200,16 +102,12 @@ func getFlaggedPosts(c *Context, w http.ResponseWriter, r *http.Request) {
return
}
- posts := &model.PostList{}
-
- if result := <-app.Srv.Store.Post().GetFlaggedPosts(c.Session.UserId, offset, limit); result.Err != nil {
- c.Err = result.Err
+ if posts, err := app.GetFlaggedPosts(c.Session.UserId, offset, limit); err != nil {
+ c.Err = err
return
} else {
- posts = result.Data.(*model.PostList)
+ w.Write([]byte(posts.ToJson()))
}
-
- w.Write([]byte(posts.ToJson()))
}
func getPosts(c *Context, w http.ResponseWriter, r *http.Request) {
@@ -233,27 +131,21 @@ func getPosts(c *Context, w http.ResponseWriter, r *http.Request) {
return
}
- etagChan := app.Srv.Store.Post().GetEtag(id, true)
-
if !app.SessionHasPermissionToChannel(c.Session, id, model.PERMISSION_CREATE_POST) {
c.SetPermissionError(model.PERMISSION_CREATE_POST)
return
}
- etag := (<-etagChan).Data.(string)
+ etag := app.GetPostsEtag(id)
if HandleEtag(etag, "Get Posts", w, r) {
return
}
- pchan := app.Srv.Store.Post().GetPosts(id, offset, limit, true)
-
- if result := <-pchan; result.Err != nil {
- c.Err = result.Err
+ if list, err := app.GetPosts(id, offset, limit); err != nil {
+ c.Err = err
return
} else {
- list := result.Data.(*model.PostList)
-
w.Header().Set(model.HEADER_ETAG_SERVER, etag)
w.Write([]byte(list.ToJson()))
}
@@ -275,19 +167,15 @@ func getPostsSince(c *Context, w http.ResponseWriter, r *http.Request) {
return
}
- pchan := app.Srv.Store.Post().GetPostsSince(id, time, true)
-
if !app.SessionHasPermissionToChannel(c.Session, id, model.PERMISSION_READ_CHANNEL) {
c.SetPermissionError(model.PERMISSION_READ_CHANNEL)
return
}
- if result := <-pchan; result.Err != nil {
- c.Err = result.Err
+ if list, err := app.GetPostsSince(id, time); err != nil {
+ c.Err = err
return
} else {
- list := result.Data.(*model.PostList)
-
w.Write([]byte(list.ToJson()))
}
@@ -308,21 +196,17 @@ func getPost(c *Context, w http.ResponseWriter, r *http.Request) {
return
}
- pchan := app.Srv.Store.Post().Get(postId)
-
if !app.SessionHasPermissionToChannel(c.Session, channelId, model.PERMISSION_READ_CHANNEL) {
c.SetPermissionError(model.PERMISSION_READ_CHANNEL)
return
}
- if result := <-pchan; result.Err != nil {
- c.Err = result.Err
+ if list, err := app.GetPostThread(postId); err != nil {
+ c.Err = err
return
- } else if HandleEtag(result.Data.(*model.PostList).Etag(), "Get Post", w, r) {
+ } else if HandleEtag(list.Etag(), "Get Post", w, r) {
return
} else {
- list := result.Data.(*model.PostList)
-
if !list.IsChannelId(channelId) {
c.Err = model.NewLocAppError("getPost", "api.post.get_post.permissions.app_error", nil, "")
c.Err.StatusCode = http.StatusForbidden
@@ -343,12 +227,10 @@ func getPostById(c *Context, w http.ResponseWriter, r *http.Request) {
return
}
- if result := <-app.Srv.Store.Post().Get(postId); result.Err != nil {
- c.Err = result.Err
+ if list, err := app.GetPostThread(postId); err != nil {
+ c.Err = err
return
} else {
- list := result.Data.(*model.PostList)
-
if len(list.Order) != 1 {
c.Err = model.NewLocAppError("getPostById", "api.post_get_post_by_id.get.app_error", nil, "")
return
@@ -378,39 +260,17 @@ func getPermalinkTmp(c *Context, w http.ResponseWriter, r *http.Request) {
return
}
- if result := <-app.Srv.Store.Post().Get(postId); result.Err != nil {
- c.Err = result.Err
+ if !app.HasPermissionToChannelByPost(c.Session.UserId, postId, model.PERMISSION_JOIN_PUBLIC_CHANNELS) {
+ c.SetPermissionError(model.PERMISSION_JOIN_PUBLIC_CHANNELS)
return
- } else {
- list := result.Data.(*model.PostList)
-
- if len(list.Order) != 1 {
- c.Err = model.NewLocAppError("getPermalinkTmp", "api.post_get_post_by_id.get.app_error", nil, "")
- return
- }
- post := list.Posts[list.Order[0]]
-
- var channel *model.Channel
- var err *model.AppError
- if channel, err = app.GetChannel(post.ChannelId); err != nil {
- c.Err = err
- return
- }
-
- if !app.SessionHasPermissionToTeam(c.Session, channel.TeamId, model.PERMISSION_JOIN_PUBLIC_CHANNELS) {
- c.SetPermissionError(model.PERMISSION_JOIN_PUBLIC_CHANNELS)
- return
- }
-
- if err = app.JoinChannel(channel, c.Session.UserId); err != nil {
- c.Err = err
- return
- }
-
- if HandleEtag(list.Etag(), "Get Permalink TMP", w, r) {
- return
- }
+ }
+ if list, err := app.GetPermalinkPost(postId, c.Session.UserId); err != nil {
+ c.Err = err
+ return
+ } else if HandleEtag(list.Etag(), "Get Permalink TMP", w, r) {
+ return
+ } else {
w.Header().Set(model.HEADER_ETAG_SERVER, list.Etag())
w.Write([]byte(list.ToJson()))
}
@@ -436,69 +296,27 @@ func deletePost(c *Context, w http.ResponseWriter, r *http.Request) {
return
}
- pchan := app.Srv.Store.Post().Get(postId)
+ if !app.SessionHasPermissionToPost(c.Session, postId, model.PERMISSION_DELETE_OTHERS_POSTS) {
+ c.SetPermissionError(model.PERMISSION_DELETE_OTHERS_POSTS)
+ return
+ }
- if result := <-pchan; result.Err != nil {
- c.Err = result.Err
+ if post, err := app.DeletePost(postId); err != nil {
+ c.Err = err
return
} else {
-
- post := result.Data.(*model.PostList).Posts[postId]
-
- if post == nil {
- c.SetInvalidParam("deletePost", "postId")
- return
- }
-
if post.ChannelId != channelId {
c.Err = model.NewLocAppError("deletePost", "api.post.delete_post.permissions.app_error", nil, "")
c.Err.StatusCode = http.StatusForbidden
return
}
- if post.UserId != c.Session.UserId && !app.SessionHasPermissionToChannel(c.Session, post.ChannelId, model.PERMISSION_DELETE_OTHERS_POSTS) {
- c.Err = model.NewLocAppError("deletePost", "api.post.delete_post.permissions.app_error", nil, "")
- c.Err.StatusCode = http.StatusForbidden
- return
- }
-
- if dresult := <-app.Srv.Store.Post().Delete(postId, model.GetMillis()); dresult.Err != nil {
- c.Err = dresult.Err
- return
- }
-
- message := model.NewWebSocketEvent(model.WEBSOCKET_EVENT_POST_DELETED, "", post.ChannelId, "", nil)
- message.Add("post", post.ToJson())
-
- go app.Publish(message)
- go DeletePostFiles(post)
- go DeleteFlaggedPost(c.Session.UserId, post)
-
- app.InvalidateCacheForChannelPosts(post.ChannelId)
-
result := make(map[string]string)
result["id"] = postId
w.Write([]byte(model.MapToJson(result)))
}
}
-func DeleteFlaggedPost(userId string, post *model.Post) {
- if result := <-app.Srv.Store.Preference().Delete(userId, model.PREFERENCE_CATEGORY_FLAGGED_POST, post.Id); result.Err != nil {
- l4g.Warn(utils.T("api.post.delete_flagged_post.app_error.warn"), result.Err)
- return
- }
-}
-
-func DeletePostFiles(post *model.Post) {
- if len(post.FileIds) != 0 {
- return
- }
-
- if result := <-app.Srv.Store.FileInfo().DeleteForPost(post.Id); result.Err != nil {
- l4g.Warn(utils.T("api.post.delete_post_files.app_error.warn"), post.Id, result.Err)
- }
-}
-
func getPostsBefore(c *Context, w http.ResponseWriter, r *http.Request) {
getPostsBeforeOrAfter(c, w, r, true)
}
@@ -534,32 +352,22 @@ func getPostsBeforeOrAfter(c *Context, w http.ResponseWriter, r *http.Request, b
return
}
- // We can do better than this etag in this situation
- etagChan := app.Srv.Store.Post().GetEtag(id, true)
-
if !app.SessionHasPermissionToChannel(c.Session, id, model.PERMISSION_READ_CHANNEL) {
c.SetPermissionError(model.PERMISSION_READ_CHANNEL)
return
}
- etag := (<-etagChan).Data.(string)
+ // We can do better than this etag in this situation
+ etag := app.GetPostsEtag(id)
+
if HandleEtag(etag, "Get Posts Before or After", w, r) {
return
}
- var pchan store.StoreChannel
- if before {
- pchan = app.Srv.Store.Post().GetPostsBefore(id, postId, numPosts, offset)
- } else {
- pchan = app.Srv.Store.Post().GetPostsAfter(id, postId, numPosts, offset)
- }
-
- if result := <-pchan; result.Err != nil {
- c.Err = result.Err
+ if list, err := app.GetPostsAroundPost(postId, id, offset, numPosts, before); err != nil {
+ c.Err = err
return
} else {
- list := result.Data.(*model.PostList)
-
w.Header().Set(model.HEADER_ETAG_SERVER, etag)
w.Write([]byte(list.ToJson()))
}
@@ -579,26 +387,10 @@ func searchPosts(c *Context, w http.ResponseWriter, r *http.Request) {
isOrSearch = val.(bool)
}
- paramsList := model.ParseSearchParams(terms)
- channels := []store.StoreChannel{}
-
- for _, params := range paramsList {
- params.OrTerms = isOrSearch
- // don't allow users to search for everything
- if params.Terms != "*" {
- channels = append(channels, app.Srv.Store.Post().Search(c.TeamId, c.Session.UserId, params))
- }
- }
-
- posts := &model.PostList{}
- for _, channel := range channels {
- if result := <-channel; result.Err != nil {
- c.Err = result.Err
- return
- } else {
- data := result.Data.(*model.PostList)
- posts.Extend(data)
- }
+ posts, err := app.SearchPostsInTeam(terms, c.Session.UserId, c.TeamId, isOrSearch)
+ if err != nil {
+ c.Err = err
+ return
}
w.Header().Set("Cache-Control", "no-cache, no-store, must-revalidate")
@@ -620,69 +412,35 @@ func getFileInfosForPost(c *Context, w http.ResponseWriter, r *http.Request) {
return
}
- pchan := app.Srv.Store.Post().Get(postId)
- fchan := app.Srv.Store.FileInfo().GetForPost(postId)
-
if !app.SessionHasPermissionToChannel(c.Session, channelId, model.PERMISSION_READ_CHANNEL) {
c.SetPermissionError(model.PERMISSION_READ_CHANNEL)
return
}
- var infos []*model.FileInfo
- if result := <-fchan; result.Err != nil {
- c.Err = result.Err
+ if infos, err := app.GetFileInfosForPost(postId); err != nil {
+ c.Err = err
return
- } else {
- infos = result.Data.([]*model.FileInfo)
- }
-
- if len(infos) == 0 {
- // No FileInfos were returned so check if they need to be created for this post
- var post *model.Post
- if result := <-pchan; result.Err != nil {
- c.Err = result.Err
- return
- } else {
- post = result.Data.(*model.PostList).Posts[postId]
- }
-
- if len(post.Filenames) > 0 {
- // The post has Filenames that need to be replaced with FileInfos
- infos = app.MigrateFilenamesToFileInfos(post)
- }
- }
-
- etag := model.GetEtagForFileInfos(infos)
-
- if HandleEtag(etag, "Get File Infos For Post", w, r) {
+ } else if HandleEtag(model.GetEtagForFileInfos(infos), "Get File Infos For Post", w, r) {
return
} else {
w.Header().Set("Cache-Control", "max-age=2592000, public")
- w.Header().Set(model.HEADER_ETAG_SERVER, etag)
+ w.Header().Set(model.HEADER_ETAG_SERVER, model.GetEtagForFileInfos(infos))
w.Write([]byte(model.FileInfosToJson(infos)))
}
}
func getOpenGraphMetadata(c *Context, w http.ResponseWriter, r *http.Request) {
props := model.StringInterfaceFromJson(r.Body)
- og := opengraph.NewOpenGraph()
- res, err := http.Get(props["url"].(string))
- defer closeBody(res)
- if err != nil {
- writeOpenGraphToResponse(w, og)
+ url := ""
+ ok := false
+ if url, ok = props["url"].(string); len(url) == 0 || !ok {
+ c.SetInvalidParam("getOpenGraphMetadata", "url")
return
}
- if err := og.ProcessHTML(res.Body); err != nil {
- writeOpenGraphToResponse(w, og)
- return
- }
-
- writeOpenGraphToResponse(w, og)
-}
+ og := app.GetOpenGraphMetadata(url)
-func writeOpenGraphToResponse(w http.ResponseWriter, og *opengraph.OpenGraph) {
ogJson, err := og.ToJSON()
if err != nil {
w.Write([]byte(`{"url": ""}`))