summaryrefslogtreecommitdiffstats
path: root/store
diff options
context:
space:
mode:
authorDerrick Anderson <derrick@andersonwebstudio.com>2018-04-30 16:06:06 -0400
committerJesse Hallam <jesse.hallam@gmail.com>2018-04-30 16:06:06 -0400
commita5f006b8a94bdadf5343aacd2b58c5bad4485153 (patch)
tree7d4f6a91b3c74217b0745e9d8ec4198fe5afecb1 /store
parente73f1d73143ebba9c7e80d21c45bba9b61f2611c (diff)
downloadchat-a5f006b8a94bdadf5343aacd2b58c5bad4485153.tar.gz
chat-a5f006b8a94bdadf5343aacd2b58c5bad4485153.tar.bz2
chat-a5f006b8a94bdadf5343aacd2b58c5bad4485153.zip
Revert "MM-9770: rewrite getParentsPosts to improve performance (#8467)" (#8659) (#8694)
This reverts commit 4b675b347b5241def7807fab5e01ce9b98531815.
Diffstat (limited to 'store')
-rw-r--r--store/sqlstore/post_store.go85
-rw-r--r--store/storetest/post_store.go248
2 files changed, 118 insertions, 215 deletions
diff --git a/store/sqlstore/post_store.go b/store/sqlstore/post_store.go
index 75154791c..e4872fe11 100644
--- a/store/sqlstore/post_store.go
+++ b/store/sqlstore/post_store.go
@@ -715,70 +715,31 @@ func (s *SqlPostStore) getRootPosts(channelId string, offset int, limit int) sto
func (s *SqlPostStore) getParentsPosts(channelId string, offset int, limit int) store.StoreChannel {
return store.Do(func(result *store.StoreResult) {
var posts []*model.Post
- _, err := s.GetReplica().Select(&posts, `
- SELECT
- *
+ _, err := s.GetReplica().Select(&posts,
+ `SELECT
+ q2.*
FROM
- Posts
+ Posts q2
+ INNER JOIN
+ (SELECT DISTINCT
+ q3.RootId
+ FROM
+ (SELECT
+ RootId
+ FROM
+ Posts
+ WHERE
+ ChannelId = :ChannelId1
+ AND DeleteAt = 0
+ ORDER BY CreateAt DESC
+ LIMIT :Limit OFFSET :Offset) q3
+ WHERE q3.RootId != '') q1
+ ON q1.RootId = q2.Id OR q1.RootId = q2.RootId
WHERE
- Id IN (SELECT * FROM (
- -- The root post of any replies in the window
- (SELECT * FROM (
- SELECT
- CASE RootId
- WHEN '' THEN NULL
- ELSE RootId
- END
- FROM
- Posts
- WHERE
- ChannelId = :ChannelId1
- AND DeleteAt = 0
- ORDER BY
- CreateAt DESC
- LIMIT :Limit1 OFFSET :Offset1
- ) x )
-
- UNION
-
- -- The reply posts to all threads intersecting with the window, including replies
- -- to root posts in the window itself.
- (
- SELECT
- Id
- FROM
- Posts
- WHERE RootId IN (SELECT * FROM (
- SELECT
- CASE RootId
- -- If there is no RootId, return the post id itself to be considered
- -- as a root post.
- WHEN '' THEN Id
- -- If there is a RootId, this post isn't a root post and return its
- -- root to be considered as a root post.
- ELSE RootId
- END
- FROM
- Posts
- WHERE
- ChannelId = :ChannelId2
- AND DeleteAt = 0
- ORDER BY
- CreateAt DESC
- LIMIT :Limit2 OFFSET :Offset2
- ) x )
- )
- ) x )
- AND
- DeleteAt = 0
- `, map[string]interface{}{
- "ChannelId1": channelId,
- "ChannelId2": channelId,
- "Offset1": offset,
- "Offset2": offset,
- "Limit1": limit,
- "Limit2": limit,
- })
+ ChannelId = :ChannelId2
+ AND DeleteAt = 0
+ ORDER BY CreateAt`,
+ map[string]interface{}{"ChannelId1": channelId, "Offset": offset, "Limit": limit, "ChannelId2": channelId})
if err != nil {
result.Err = model.NewAppError("SqlPostStore.GetLinearPosts", "store.sql_post.get_parents_posts.app_error", nil, "channelId="+channelId+" err="+err.Error(), http.StatusInternalServerError)
} else {
diff --git a/store/storetest/post_store.go b/store/storetest/post_store.go
index 44ce47d9d..6ebb875bf 100644
--- a/store/storetest/post_store.go
+++ b/store/storetest/post_store.go
@@ -5,7 +5,6 @@ package storetest
import (
"fmt"
- "sort"
"strings"
"testing"
"time"
@@ -493,182 +492,125 @@ func testPostStoreGetWithChildren(t *testing.T, ss store.Store) {
}
func testPostStoreGetPostsWithDetails(t *testing.T, ss store.Store) {
- assertPosts := func(expected []*model.Post, actual map[string]*model.Post) {
- expectedIds := make([]string, 0, len(expected))
- expectedMessages := make([]string, 0, len(expected))
- for _, post := range expected {
- expectedIds = append(expectedIds, post.Id)
- expectedMessages = append(expectedMessages, post.Message)
- }
- sort.Strings(expectedIds)
- sort.Strings(expectedMessages)
-
- actualIds := make([]string, 0, len(actual))
- actualMessages := make([]string, 0, len(actual))
- for _, post := range actual {
- actualIds = append(actualIds, post.Id)
- actualMessages = append(actualMessages, post.Message)
- }
- sort.Strings(actualIds)
- sort.Strings(actualMessages)
-
- if assert.Equal(t, expectedIds, actualIds) {
- assert.Equal(t, expectedMessages, actualMessages)
- }
- }
-
- root1 := &model.Post{}
- root1.ChannelId = model.NewId()
- root1.UserId = model.NewId()
- root1.Message = "zz" + model.NewId() + "b"
- root1 = (<-ss.Post().Save(root1)).Data.(*model.Post)
+ o1 := &model.Post{}
+ o1.ChannelId = model.NewId()
+ o1.UserId = model.NewId()
+ o1.Message = "zz" + model.NewId() + "b"
+ o1 = (<-ss.Post().Save(o1)).Data.(*model.Post)
time.Sleep(2 * time.Millisecond)
- root1Reply1 := &model.Post{}
- root1Reply1.ChannelId = root1.ChannelId
- root1Reply1.UserId = model.NewId()
- root1Reply1.Message = "zz" + model.NewId() + "b"
- root1Reply1.ParentId = root1.Id
- root1Reply1.RootId = root1.Id
- root1Reply1 = (<-ss.Post().Save(root1Reply1)).Data.(*model.Post)
+ o2 := &model.Post{}
+ o2.ChannelId = o1.ChannelId
+ o2.UserId = model.NewId()
+ o2.Message = "zz" + model.NewId() + "b"
+ o2.ParentId = o1.Id
+ o2.RootId = o1.Id
+ o2 = (<-ss.Post().Save(o2)).Data.(*model.Post)
time.Sleep(2 * time.Millisecond)
- root1Reply2 := &model.Post{}
- root1Reply2.ChannelId = root1.ChannelId
- root1Reply2.UserId = model.NewId()
- root1Reply2.Message = "zz" + model.NewId() + "b"
- root1Reply2.ParentId = root1.Id
- root1Reply2.RootId = root1.Id
- root1Reply2 = (<-ss.Post().Save(root1Reply2)).Data.(*model.Post)
+ o2a := &model.Post{}
+ o2a.ChannelId = o1.ChannelId
+ o2a.UserId = model.NewId()
+ o2a.Message = "zz" + model.NewId() + "b"
+ o2a.ParentId = o1.Id
+ o2a.RootId = o1.Id
+ o2a = (<-ss.Post().Save(o2a)).Data.(*model.Post)
time.Sleep(2 * time.Millisecond)
- root1Reply3 := &model.Post{}
- root1Reply3.ChannelId = root1.ChannelId
- root1Reply3.UserId = model.NewId()
- root1Reply3.Message = "zz" + model.NewId() + "b"
- root1Reply3.ParentId = root1.Id
- root1Reply3.RootId = root1.Id
- root1Reply3 = (<-ss.Post().Save(root1Reply3)).Data.(*model.Post)
+ o3 := &model.Post{}
+ o3.ChannelId = o1.ChannelId
+ o3.UserId = model.NewId()
+ o3.Message = "zz" + model.NewId() + "b"
+ o3.ParentId = o1.Id
+ o3.RootId = o1.Id
+ o3 = (<-ss.Post().Save(o3)).Data.(*model.Post)
time.Sleep(2 * time.Millisecond)
- root2 := &model.Post{}
- root2.ChannelId = root1.ChannelId
- root2.UserId = model.NewId()
- root2.Message = "zz" + model.NewId() + "b"
- root2 = (<-ss.Post().Save(root2)).Data.(*model.Post)
+ o4 := &model.Post{}
+ o4.ChannelId = o1.ChannelId
+ o4.UserId = model.NewId()
+ o4.Message = "zz" + model.NewId() + "b"
+ o4 = (<-ss.Post().Save(o4)).Data.(*model.Post)
time.Sleep(2 * time.Millisecond)
- root2Reply1 := &model.Post{}
- root2Reply1.ChannelId = root1.ChannelId
- root2Reply1.UserId = model.NewId()
- root2Reply1.Message = "zz" + model.NewId() + "b"
- root2Reply1.ParentId = root2.Id
- root2Reply1.RootId = root2.Id
- root2Reply1 = (<-ss.Post().Save(root2Reply1)).Data.(*model.Post)
+ o5 := &model.Post{}
+ o5.ChannelId = o1.ChannelId
+ o5.UserId = model.NewId()
+ o5.Message = "zz" + model.NewId() + "b"
+ o5.ParentId = o4.Id
+ o5.RootId = o4.Id
+ o5 = (<-ss.Post().Save(o5)).Data.(*model.Post)
+
+ r1 := (<-ss.Post().GetPosts(o1.ChannelId, 0, 4, false)).Data.(*model.PostList)
- r1 := (<-ss.Post().GetPosts(root1.ChannelId, 0, 4, false)).Data.(*model.PostList)
+ if r1.Order[0] != o5.Id {
+ t.Fatal("invalid order")
+ }
- expectedOrder := []string{
- root2Reply1.Id,
- root2.Id,
- root1Reply3.Id,
- root1Reply2.Id,
+ if r1.Order[1] != o4.Id {
+ t.Fatal("invalid order")
}
- expectedPosts := []*model.Post{
- root1,
- root1Reply1,
- root1Reply2,
- root1Reply3,
- root2,
- root2Reply1,
+ if r1.Order[2] != o3.Id {
+ t.Fatal("invalid order")
}
- assert.Equal(t, expectedOrder, r1.Order)
- assertPosts(expectedPosts, r1.Posts)
+ if r1.Order[3] != o2a.Id {
+ t.Fatal("invalid order")
+ }
- r2 := (<-ss.Post().GetPosts(root1.ChannelId, 0, 4, true)).Data.(*model.PostList)
- assert.Equal(t, expectedOrder, r2.Order)
- assertPosts(expectedPosts, r2.Posts)
+ if len(r1.Posts) != 6 { //the last 4, + o1 (o2a and o3's parent) + o2 (in same thread as o2a and o3)
+ t.Fatal("wrong size")
+ }
- // Run once to fill cache
- <-ss.Post().GetPosts(root1.ChannelId, 0, 30, true)
- expectedOrder = []string{
- root2Reply1.Id,
- root2.Id,
- root1Reply3.Id,
- root1Reply2.Id,
- root1Reply1.Id,
- root1.Id,
+ if r1.Posts[o1.Id].Message != o1.Message {
+ t.Fatal("Missing parent")
}
- root3 := &model.Post{}
- root3.ChannelId = root1.ChannelId
- root3.UserId = model.NewId()
- root3.Message = "zz" + model.NewId() + "b"
- root3 = (<-ss.Post().Save(root3)).Data.(*model.Post)
+ r2 := (<-ss.Post().GetPosts(o1.ChannelId, 0, 4, true)).Data.(*model.PostList)
- // Response should be the same despite the new post since we hit the cache
- r3 := (<-ss.Post().GetPosts(root1.ChannelId, 0, 30, true)).Data.(*model.PostList)
- assert.Equal(t, expectedOrder, r3.Order)
- assertPosts(expectedPosts, r3.Posts)
+ if r2.Order[0] != o5.Id {
+ t.Fatal("invalid order")
+ }
- ss.Post().InvalidateLastPostTimeCache(root1.ChannelId)
+ if r2.Order[1] != o4.Id {
+ t.Fatal("invalid order")
+ }
+
+ if r2.Order[2] != o3.Id {
+ t.Fatal("invalid order")
+ }
+
+ if r2.Order[3] != o2a.Id {
+ t.Fatal("invalid order")
+ }
+
+ if len(r2.Posts) != 6 { //the last 4, + o1 (o2a and o3's parent) + o2 (in same thread as o2a and o3)
+ t.Fatal("wrong size")
+ }
+
+ if r2.Posts[o1.Id].Message != o1.Message {
+ t.Fatal("Missing parent")
+ }
+
+ // Run once to fill cache
+ <-ss.Post().GetPosts(o1.ChannelId, 0, 30, true)
+
+ o6 := &model.Post{}
+ o6.ChannelId = o1.ChannelId
+ o6.UserId = model.NewId()
+ o6.Message = "zz" + model.NewId() + "b"
+ o6 = (<-ss.Post().Save(o6)).Data.(*model.Post)
+
+ // Should only be 6 since we hit the cache
+ r3 := (<-ss.Post().GetPosts(o1.ChannelId, 0, 30, true)).Data.(*model.PostList)
+ assert.Equal(t, 6, len(r3.Order))
+
+ ss.Post().InvalidateLastPostTimeCache(o1.ChannelId)
// Cache was invalidated, we should get all the posts
- r4 := (<-ss.Post().GetPosts(root1.ChannelId, 0, 30, true)).Data.(*model.PostList)
- expectedOrder = []string{
- root3.Id,
- root2Reply1.Id,
- root2.Id,
- root1Reply3.Id,
- root1Reply2.Id,
- root1Reply1.Id,
- root1.Id,
- }
- expectedPosts = []*model.Post{
- root1,
- root1Reply1,
- root1Reply2,
- root1Reply3,
- root2,
- root2Reply1,
- root3,
- }
-
- assert.Equal(t, expectedOrder, r4.Order)
- assertPosts(expectedPosts, r4.Posts)
-
- // Replies past the window should be included if the root post itself is in the window
- root3Reply1 := &model.Post{}
- root3Reply1.ChannelId = root1.ChannelId
- root3Reply1.UserId = model.NewId()
- root3Reply1.Message = "zz" + model.NewId() + "b"
- root3Reply1.ParentId = root3.Id
- root3Reply1.RootId = root3.Id
- root3Reply1 = (<-ss.Post().Save(root3Reply1)).Data.(*model.Post)
-
- r5 := (<-ss.Post().GetPosts(root1.ChannelId, 1, 5, false)).Data.(*model.PostList)
- expectedOrder = []string{
- root3.Id,
- root2Reply1.Id,
- root2.Id,
- root1Reply3.Id,
- root1Reply2.Id,
- }
- expectedPosts = []*model.Post{
- root1,
- root1Reply1,
- root1Reply2,
- root1Reply3,
- root2,
- root2Reply1,
- root3,
- root3Reply1,
- }
-
- assert.Equal(t, expectedOrder, r5.Order)
- assertPosts(expectedPosts, r5.Posts)
+ r4 := (<-ss.Post().GetPosts(o1.ChannelId, 0, 30, true)).Data.(*model.PostList)
+ assert.Equal(t, 7, len(r4.Order))
}
func testPostStoreGetPostsBeforeAfter(t *testing.T, ss store.Store) {