Skip to content

Commit 654454c

Browse files
committed
refactor(replies): useReplies should use a comment argument to not create infinite comment updates
1 parent 701c3aa commit 654454c

12 files changed

+152
-93
lines changed

README.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ useNotifications(): {notifications: Notification[], markAsRead: Function}
2727
#### Comments Hooks
2828
```
2929
useComment({commentCid: string, onlyIfCached?: boolean}): Comment
30-
useReplies({commentCid: string, sortType?: string, flat?: boolean, repliesPerPage?: number, filter>: CommentsFilter}): {replies: Comment[], hasMore: boolean, loadMore: function, reset: function, updatedReplies: Comment[], bufferedReplies: Comment[]}
30+
useReplies({comment: Comment, sortType?: string, flat?: boolean, repliesPerPage?: number, filter>: CommentsFilter}): {replies: Comment[], hasMore: boolean, loadMore: function, reset: function, updatedReplies: Comment[], bufferedReplies: Comment[]}
3131
useComments({commentCids: string[], onlyIfCached?: boolean}): {comments: Comment[]}
3232
useEditedComment({comment: Comment}): {editedComment: Comment | undefined}
3333
useValidateComment({comment: Comment, validateReplies?: boolean}): {valid: boolean}
@@ -144,7 +144,7 @@ const {authorAddress, shortAuthorAddress} = useAuthorAddress({comment: post})
144144
const post = useComment({commentCid, onlyIfCached: true})
145145

146146
// post.replies are not validated, to show replies
147-
const {replies, hasMore, loadMore} = useReplies({commentCid})
147+
const {replies, hasMore, loadMore} = useReplies({comment: post})
148148

149149
// to show a preloaded reply without rerenders, validate manually
150150
const {valid} = useValidateComment({comment: post.replies.pages.best[0]})

src/hooks/comments.test.ts

+57-39
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import {act, renderHook} from '@testing-library/react-hooks'
22
import testUtils from '../lib/test-utils'
33
import {useComment, useComments, useReplies, useValidateComment, setPlebbitJs} from '..'
44
import commentsStore from '../stores/comments'
5+
import repliesCommentsStore from '../stores/replies/replies-comments-store'
56
import subplebbitsPagesStore from '../stores/subplebbits-pages'
67
import PlebbitJsMock, {Plebbit, Comment, Pages, simulateLoadingTime} from '../lib/plebbit-js/plebbit-js-mock'
78
import utils from '../lib/utils'
@@ -339,7 +340,11 @@ describe('comment replies', () => {
339340
let rendered, waitFor, scrollOnePage
340341

341342
beforeEach(() => {
342-
rendered = renderHook<any, any>((useRepliesOptions) => useReplies(useRepliesOptions))
343+
rendered = renderHook<any, any>((useRepliesOptions) => {
344+
// useReplies accepts only 'comment', but for ease of use also accept a 'commentCid' in the tests
345+
const comment = useComment({commentCid: useRepliesOptions?.commentCid})
346+
return useReplies({...useRepliesOptions, commentCid: undefined, comment: useRepliesOptions?.comment || comment})
347+
})
343348
waitFor = testUtils.createWaitFor(rendered)
344349
scrollOnePage = scrollOnePage = async () => {
345350
const nextFeedLength = (rendered.result.current.replies?.length || 0) + repliesPerPage
@@ -383,6 +388,48 @@ describe('comment replies', () => {
383388
expect(rendered.result.current.hasMore).toBe(true)
384389
})
385390

391+
test('sort type new, switch to sort type old, switch to different comment (comment argument)', async () => {
392+
expect(rendered.result.current.replies).toEqual([])
393+
394+
const createMockComment = (comment) => {
395+
comment.timestamp = 1
396+
comment.updatedAt = 2
397+
comment.replies = {
398+
pages: {best: new Pages({comment}).pageToGet(comment.cid + ' page cid best')},
399+
pageCids: {
400+
new: comment.cid + ' page cid new',
401+
newFlat: comment.cid + ' page cid newFlat',
402+
old: comment.cid + ' page cid old',
403+
oldFlat: comment.cid + ' page cid oldFlat',
404+
},
405+
}
406+
return comment
407+
}
408+
409+
const commentAbc = createMockComment({cid: 'comment cid abc', subplebbitAddress: 'subplebbit address 1'})
410+
rendered.rerender({comment: commentAbc, sortType: 'new'})
411+
await waitFor(() => rendered.result.current.replies.length === repliesPerPage)
412+
expect(rendered.result.current.replies.length).toBe(repliesPerPage)
413+
expect(rendered.result.current.replies[0].cid).toBe('comment cid abc page cid new comment cid 100')
414+
expect(rendered.result.current.replies[0].timestamp).toBeGreaterThan(rendered.result.current.replies[repliesPerPage - 1].timestamp)
415+
expect(rendered.result.current.hasMore).toBe(true)
416+
417+
rendered.rerender({comment: commentAbc, sortType: 'old'})
418+
await waitFor(() => rendered.result.current.replies[0].cid === 'comment cid abc page cid old comment cid 1')
419+
expect(rendered.result.current.replies.length).toBe(repliesPerPage)
420+
expect(rendered.result.current.replies[0].cid).toBe('comment cid abc page cid old comment cid 1')
421+
expect(rendered.result.current.replies[repliesPerPage - 1].timestamp).toBeGreaterThan(rendered.result.current.replies[0].timestamp)
422+
expect(rendered.result.current.hasMore).toBe(true)
423+
424+
const commentXyz = createMockComment({cid: 'comment cid xyz', subplebbitAddress: 'subplebbit address 1'})
425+
rendered.rerender({comment: commentXyz, sortType: 'old'})
426+
await waitFor(() => rendered.result.current.replies[0].cid === 'comment cid xyz page cid old comment cid 1')
427+
expect(rendered.result.current.replies.length).toBe(repliesPerPage)
428+
expect(rendered.result.current.replies[0].cid).toBe('comment cid xyz page cid old comment cid 1')
429+
expect(rendered.result.current.replies[repliesPerPage - 1].timestamp).toBeGreaterThan(rendered.result.current.replies[0].timestamp)
430+
expect(rendered.result.current.hasMore).toBe(true)
431+
})
432+
386433
test('scroll pages', async () => {
387434
expect(rendered.result.current.replies).toEqual([])
388435

@@ -603,40 +650,6 @@ describe('comment replies', () => {
603650
expect(Object.keys(repliesStore.getState().feedsOptions).length).toBe(2)
604651
})
605652

606-
test('can useComment({onlyIfCached: true}) on a comment from replies pages', async () => {
607-
rendered = renderHook<any, any>(({useCommentOptions, useRepliesOptions}) => {
608-
const comment = useComment(useCommentOptions)
609-
const replies = useReplies(useRepliesOptions)
610-
return {comment, replies}
611-
})
612-
waitFor = testUtils.createWaitFor(rendered)
613-
614-
const useRepliesOptions = {
615-
commentCid: 'comment cid 1',
616-
// the preloaded page is best
617-
sortType: 'best',
618-
// the preloaded page size is 100
619-
repliesPerPage: 200,
620-
}
621-
rendered.rerender({useRepliesOptions})
622-
await waitFor(() => rendered.result.current.replies.replies.length === useRepliesOptions.repliesPerPage)
623-
expect(rendered.result.current.replies.replies.length).toBe(useRepliesOptions.repliesPerPage)
624-
625-
// first reply (preloaded) is defined
626-
const firstCommentCid = rendered.result.current.replies.replies[0].cid
627-
rendered.rerender({useRepliesOptions, useCommentOptions: {commentCid: firstCommentCid, onlyIfCached: true}})
628-
await waitFor(() => typeof rendered.result.current.comment.updatedAt === 'number')
629-
expect(typeof rendered.result.current.comment.updatedAt).toBe('number')
630-
expect(rendered.result.current.comment.cid).toBe(firstCommentCid)
631-
632-
// last reply (from a page) is defined
633-
const lastCommentCid = rendered.result.current.replies.replies[useRepliesOptions.repliesPerPage - 1].cid
634-
rendered.rerender({useRepliesOptions, useCommentOptions: {commentCid: lastCommentCid, onlyIfCached: true}})
635-
await waitFor(() => typeof rendered.result.current.comment.updatedAt === 'number')
636-
expect(typeof rendered.result.current.comment.updatedAt).toBe('number')
637-
expect(rendered.result.current.comment.cid).toBe(lastCommentCid)
638-
})
639-
640653
test.todo('has account comments', async () => {})
641654

642655
test.todo('nested scroll pages', async () => {})
@@ -718,7 +731,12 @@ describe('comment replies', () => {
718731
let rendered, waitFor, scrollOnePage
719732

720733
beforeEach(() => {
721-
rendered = renderHook<any, any>((useRepliesOptions) => useReplies(useRepliesOptions))
734+
rendered = renderHook<any, any>((useRepliesOptions) => {
735+
// useReplies accepts only 'comment', but for ease of use also accept a 'commentCid' in the tests
736+
const comment = useComment({commentCid: useRepliesOptions?.commentCid})
737+
return useReplies({...useRepliesOptions, commentCid: undefined, comment: useRepliesOptions?.comment || comment})
738+
})
739+
722740
waitFor = testUtils.createWaitFor(rendered)
723741
scrollOnePage = scrollOnePage = async () => {
724742
const nextFeedLength = (rendered.result.current.replies?.length || 0) + repliesPerPage
@@ -1019,11 +1037,11 @@ describe('comment replies', () => {
10191037

10201038
// second comment update (updatedAt doesn't change, so shouldn't update)
10211039
comment.simulateUpdateEvent()
1022-
await waitFor(() => commentsStore.getState().comments['comment cid 1'].replies.pages.best.comments[0].upvoteCount === 2)
1040+
await waitFor(() => repliesCommentsStore.getState().comments['comment cid 1'].replies.pages.best.comments[0].upvoteCount === 2)
10231041
expect(pages.length).toBe(2)
10241042
// comment in store updated, but the updatedAt didn't change so no change in useReplies().updatedReplies
1025-
expect(commentsStore.getState().comments['comment cid 1'].replies.pages.best.comments[0].updatedAt).toBe(1)
1026-
expect(commentsStore.getState().comments['comment cid 1'].replies.pages.best.comments[0].upvoteCount).toBe(2)
1043+
expect(repliesCommentsStore.getState().comments['comment cid 1'].replies.pages.best.comments[0].updatedAt).toBe(1)
1044+
expect(repliesCommentsStore.getState().comments['comment cid 1'].replies.pages.best.comments[0].upvoteCount).toBe(2)
10271045
expect(rendered.result.current.replies.length).toBe(1)
10281046
expect(rendered.result.current.updatedReplies.length).toBe(1)
10291047
expect(rendered.result.current.replies[0].cid).toBe('comment cid 1')

src/hooks/comments.ts

+14-14
Original file line numberDiff line numberDiff line change
@@ -187,7 +187,7 @@ export function useComments(options?: UseCommentsOptions): UseCommentsResult {
187187

188188
export function useReplies(options?: UseRepliesOptions): UseRepliesResult {
189189
assert(!options || typeof options === 'object', `useReplies options argument '${options}' not an object`)
190-
let {commentCid, sortType, accountName, flat, accountComments, repliesPerPage, filter} = options || {}
190+
let {comment, sortType, accountName, flat, accountComments, repliesPerPage, filter} = options || {}
191191
if (!sortType) {
192192
sortType = 'best'
193193
}
@@ -198,23 +198,23 @@ export function useReplies(options?: UseRepliesOptions): UseRepliesResult {
198198
accountComments = true
199199
}
200200

201-
validator.validateUseRepliesArguments(commentCid, sortType, accountName, flat, accountComments, repliesPerPage, filter)
201+
validator.validateUseRepliesArguments(comment?.cid, sortType, accountName, flat, accountComments, repliesPerPage, filter)
202202
const account = useAccount({accountName})
203-
const addFeedToStore = useRepliesStore((state: RepliesState) => state.addFeedToStore)
203+
const addFeedToStoreOrUpdateComment = useRepliesStore((state: RepliesState) => state.addFeedToStoreOrUpdateComment)
204204
const incrementFeedPageNumber = useRepliesStore((state: RepliesState) => state.incrementFeedPageNumber)
205205
const resetFeed = useRepliesStore((state: RepliesState) => state.resetFeed)
206-
const repliesFeedName = useRepliesFeedName(account?.id, commentCid, sortType, flat, accountComments, repliesPerPage, filter)
206+
const repliesFeedName = useRepliesFeedName(account?.id, comment?.cid, sortType, flat, accountComments, repliesPerPage, filter)
207207
const [errors, setErrors] = useState<Error[]>([])
208208

209209
// add replies to store
210210
useEffect(() => {
211-
if (!commentCid || !account) {
211+
if (!comment?.cid || !account || !comment) {
212212
return
213213
}
214-
addFeedToStore(repliesFeedName, commentCid, sortType, account, flat, accountComments, repliesPerPage, filter).catch((error: unknown) =>
215-
log.error('useReplies addFeedToStore error', {repliesFeedName, error})
214+
addFeedToStoreOrUpdateComment(repliesFeedName, comment, sortType, account, flat, accountComments, repliesPerPage, filter).catch((error: unknown) =>
215+
log.error('useReplies addFeedToStoreOrUpdateComment error', {repliesFeedName, error})
216216
)
217-
}, [repliesFeedName])
217+
}, [repliesFeedName, comment])
218218

219219
const replies = useRepliesStore((state: RepliesState) => state.loadedFeeds[repliesFeedName || ''])
220220
const bufferedReplies = useRepliesStore((state: RepliesState) => state.bufferedFeeds[repliesFeedName || ''])
@@ -224,14 +224,14 @@ export function useReplies(options?: UseRepliesOptions): UseRepliesResult {
224224
if (!repliesFeedName || typeof hasMore !== 'boolean') {
225225
hasMore = true
226226
}
227-
// if the replies is not yet defined, but no comment cid, doesn't have more
228-
if (!commentCid) {
227+
// if the replies is not yet defined, but no comment, doesn't have more
228+
if (!comment) {
229229
hasMore = false
230230
}
231231

232232
const loadMore = async () => {
233233
try {
234-
if (!commentCid || !account) {
234+
if (!comment?.cid || !account) {
235235
throw Error('useReplies cannot load more replies not initalized yet')
236236
}
237237
incrementFeedPageNumber(repliesFeedName)
@@ -244,7 +244,7 @@ export function useReplies(options?: UseRepliesOptions): UseRepliesResult {
244244

245245
const reset = async () => {
246246
try {
247-
if (!commentCid || !account) {
247+
if (!comment?.cid || !account) {
248248
throw Error('useReplies cannot reset replies not initalized yet')
249249
}
250250
resetFeed(repliesFeedName)
@@ -255,11 +255,11 @@ export function useReplies(options?: UseRepliesOptions): UseRepliesResult {
255255
}
256256
}
257257

258-
if (account && commentCid) {
258+
if (account && comment?.cid) {
259259
log('useReplies', {
260260
repliesLength: replies?.length || 0,
261261
hasMore,
262-
commentCid,
262+
commentCid: comment.cid,
263263
sortType,
264264
account,
265265
repliesStoreOptions: useRepliesStore.getState().feedsOptions,

src/hooks/feeds/feeds.test.ts

+7-7
Original file line numberDiff line numberDiff line change
@@ -592,7 +592,7 @@ describe('feeds', () => {
592592

593593
test('dynamic filter', async () => {
594594
const createCidMatchFilter = (cid: string) => ({
595-
filter: (comment: Comment) => !!comment.cid.match(cid),
595+
filter: (comment: Comment) => !!comment?.cid?.match(cid),
596596
key: `cid-match-${cid}`,
597597
})
598598

@@ -601,19 +601,19 @@ describe('feeds', () => {
601601
filter: createCidMatchFilter('13'),
602602
})
603603
await waitFor(() => rendered.result.current.feed?.length > 0)
604-
expect(rendered.result.current.feed[0].cid).match(/13$/)
605-
expect(rendered.result.current.feed[1].cid).match(/13$/)
606-
expect(rendered.result.current.feed[2].cid).match(/13$/)
604+
expect(rendered.result.current.feed[0].cid).toMatch(/13$/)
605+
expect(rendered.result.current.feed[1].cid).toMatch(/13$/)
606+
expect(rendered.result.current.feed[2].cid).toMatch(/13$/)
607607
expect(Object.keys(feedsStore.getState().feedsOptions).length).toBe(1)
608608

609609
rendered.rerender({
610610
subplebbitAddresses: ['subplebbit address 1', 'subplebbit address 2', 'subplebbit address 3'],
611611
filter: createCidMatchFilter('14'),
612612
})
613613
await waitFor(() => rendered.result.current.feed?.length > 0)
614-
expect(rendered.result.current.feed[0].cid).match(/14$/)
615-
expect(rendered.result.current.feed[1].cid).match(/14$/)
616-
expect(rendered.result.current.feed[2].cid).match(/14$/)
614+
expect(rendered.result.current.feed[0].cid).toMatch(/14$/)
615+
expect(rendered.result.current.feed[1].cid).toMatch(/14$/)
616+
expect(rendered.result.current.feed[2].cid).toMatch(/14$/)
617617
expect(Object.keys(feedsStore.getState().feedsOptions).length).toBe(2)
618618
})
619619

src/lib/utils/utils.ts

+1
Original file line numberDiff line numberDiff line change
@@ -95,6 +95,7 @@ export const flattenCommentsPages = (pageInstanceOrPagesInstance: any) => {
9595
// @ts-ignore
9696
uniqueFlattened.push(flattenedCommentsObject[cid])
9797
}
98+
9899
return uniqueFlattened
99100
}
100101

src/lib/validator.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -282,7 +282,7 @@ export const validateRepliesSortType = (sortType: any) => {
282282
}
283283
export const validateUseRepliesArguments = (commentCid?: any, sortType?: any, accountName?: any, flat?: any, accountComments?: any, postsPerPage?: any, filter?: any) => {
284284
if (commentCid) {
285-
assert(typeof commentCid === 'string', `useReplies commentCid argument '${commentCid}' not a string`)
285+
assert(typeof commentCid === 'string', `useReplies comment.cid argument '${commentCid}' not a string`)
286286
}
287287
assert(repliesSortTypes.has(sortType), `useReplies sortType argument '${sortType}' invalid`)
288288
if (accountName) {

src/stores/feeds/feeds-store.ts

+1-6
Original file line numberDiff line numberDiff line change
@@ -36,9 +36,6 @@ export const defaultPostsPerPage = 25
3636
// keep large buffer because fetching cids is slow
3737
export const subplebbitPostsLeftBeforeNextPage = 50
3838

39-
// reset all event listeners in between tests
40-
export const listeners: any = []
41-
4239
export type FeedsState = {
4340
feedsOptions: FeedsOptions
4441
bufferedFeeds: Feeds
@@ -92,7 +89,7 @@ const feedsStore = createStore<FeedsState>((setState: Function, getState: Functi
9289

9390
const {feedsOptions, updateFeeds} = getState()
9491
// feed is in store already, do nothing
95-
// if the feed already exist but is at page 1, reset it to page 1
92+
// if the feed already exist but is at page 0, reset it to page 1
9693
if (feedsOptions[feedName] && feedsOptions[feedName].pageNumber !== 0) {
9794
return
9895
}
@@ -446,8 +443,6 @@ export const resetFeedsStore = async () => {
446443
previousFeedsSubplebbitsPostsPagesFirstUpdatedAts = ''
447444
previousSubplebbitsPages = {}
448445
updateFeedsPending = false
449-
// remove all event listeners
450-
listeners.forEach((listener: any) => listener.removeAllListeners())
451446
// destroy all component subscriptions to the store
452447
feedsStore.destroy()
453448
// restore original state

src/stores/replies-pages/replies-pages-store.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ import Logger from '@plebbit/plebbit-logger'
44
const log = Logger('plebbit-react-hooks:replies:stores')
55
import {RepliesPage, RepliesPages, Account, Comment, Comments} from '../../types'
66
import accountsStore from '../accounts'
7-
import commentsStore, {CommentsState} from '../comments'
7+
import repliesCommentsStore, {RepliesCommentsState} from '../replies/replies-comments-store'
88
import localForageLru from '../../lib/localforage-lru'
99
import createStore from 'zustand'
1010
import assert from 'assert'
@@ -149,7 +149,7 @@ const repliesPagesStore = createStore<RepliesPagesState>((setState: Function, ge
149149

150150
// set clients states on comments store so the frontend can display it, dont persist in db because a reload cancels updating
151151
const onCommentRepliesClientsStateChange = (commentCid: string) => (clientState: string, clientType: string, sortType: string, clientUrl: string) => {
152-
commentsStore.setState((state: CommentsState) => {
152+
repliesCommentsStore.setState((state: RepliesCommentsState) => {
153153
// make sure not undefined, sometimes happens in e2e tests
154154
if (!state.comments[commentCid]) {
155155
return {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
import {Comment, Comments} from '../../types'
2+
import createStore from 'zustand'
3+
4+
export type RepliesCommentsState = {
5+
comments: Comments
6+
addCommentToStoreOrUpdateComment: Function
7+
}
8+
9+
const repliesCommentsStore = createStore<RepliesCommentsState>((setState: Function, getState: Function) => ({
10+
comments: {},
11+
addCommentToStoreOrUpdateComment(comment: Comment) {
12+
setState((state: RepliesCommentsState) => {
13+
// updatedAt hasn't changed so no need to update the comment
14+
if (state.comments[comment.cid] && comment.updatedAt <= state.comments[comment.cid].updatedAt) {
15+
return
16+
}
17+
return {comments: {...state.comments, [comment.cid]: comment}}
18+
})
19+
},
20+
}))
21+
22+
export default repliesCommentsStore

0 commit comments

Comments
 (0)