@stream-io/feeds-client 0.1.10 → 0.1.11

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (152) hide show
  1. package/CHANGELOG.md +13 -0
  2. package/dist/@react-bindings/contexts/StreamFeedContext.d.ts +1 -1
  3. package/dist/@react-bindings/contexts/StreamFeedsContext.d.ts +1 -1
  4. package/dist/@react-bindings/hooks/feed-state-hooks/useComments.d.ts +1 -1
  5. package/dist/@react-bindings/hooks/feed-state-hooks/useFeedActivities.d.ts +1 -1
  6. package/dist/@react-bindings/hooks/feed-state-hooks/useFeedMetadata.d.ts +1 -1
  7. package/dist/@react-bindings/hooks/feed-state-hooks/useFollowers.d.ts +1 -1
  8. package/dist/@react-bindings/hooks/feed-state-hooks/useFollowing.d.ts +1 -1
  9. package/dist/@react-bindings/hooks/feed-state-hooks/useOwnCapabilities.d.ts +1 -1
  10. package/dist/@react-bindings/hooks/feed-state-hooks/useOwnFollows.d.ts +1 -1
  11. package/dist/@react-bindings/hooks/useCreateFeedsClient.d.ts +1 -1
  12. package/dist/@react-bindings/wrappers/StreamFeed.d.ts +1 -1
  13. package/dist/index-react-bindings.browser.cjs +1589 -1518
  14. package/dist/index-react-bindings.browser.cjs.map +1 -1
  15. package/dist/index-react-bindings.browser.js +1589 -1518
  16. package/dist/index-react-bindings.browser.js.map +1 -1
  17. package/dist/index-react-bindings.node.cjs +1589 -1518
  18. package/dist/index-react-bindings.node.cjs.map +1 -1
  19. package/dist/index-react-bindings.node.js +1589 -1518
  20. package/dist/index-react-bindings.node.js.map +1 -1
  21. package/dist/index.browser.cjs +1607 -1533
  22. package/dist/index.browser.cjs.map +1 -1
  23. package/dist/index.browser.js +1605 -1534
  24. package/dist/index.browser.js.map +1 -1
  25. package/dist/index.d.ts +2 -2
  26. package/dist/index.node.cjs +1607 -1533
  27. package/dist/index.node.cjs.map +1 -1
  28. package/dist/index.node.js +1605 -1534
  29. package/dist/index.node.js.map +1 -1
  30. package/dist/src/common/ActivitySearchSource.d.ts +1 -1
  31. package/dist/src/common/FeedSearchSource.d.ts +2 -2
  32. package/dist/src/common/Poll.d.ts +1 -1
  33. package/dist/src/common/UserSearchSource.d.ts +1 -1
  34. package/dist/src/common/real-time/StableWSConnection.d.ts +3 -3
  35. package/dist/src/feed/event-handlers/activity/handle-activity-added.d.ts +7 -0
  36. package/dist/src/feed/event-handlers/activity/handle-activity-deleted.d.ts +8 -0
  37. package/dist/src/feed/event-handlers/activity/handle-activity-reaction-added.d.ts +8 -0
  38. package/dist/src/feed/event-handlers/activity/handle-activity-reaction-deleted.d.ts +8 -0
  39. package/dist/src/feed/event-handlers/activity/handle-activity-removed-from-feed.d.ts +3 -0
  40. package/dist/src/feed/event-handlers/activity/handle-activity-updated.d.ts +8 -0
  41. package/dist/src/feed/event-handlers/activity/index.d.ts +6 -0
  42. package/dist/src/feed/event-handlers/bookmark/handle-bookmark-added.d.ts +8 -0
  43. package/dist/src/feed/event-handlers/bookmark/handle-bookmark-deleted.d.ts +9 -0
  44. package/dist/src/feed/event-handlers/bookmark/handle-bookmark-updated.d.ts +8 -0
  45. package/dist/src/feed/event-handlers/bookmark/index.d.ts +3 -0
  46. package/dist/src/feed/event-handlers/comment/handle-comment-added.d.ts +3 -0
  47. package/dist/src/feed/event-handlers/comment/handle-comment-deleted.d.ts +3 -0
  48. package/dist/src/feed/event-handlers/comment/handle-comment-reaction.d.ts +3 -0
  49. package/dist/src/feed/event-handlers/comment/handle-comment-updated.d.ts +3 -0
  50. package/dist/src/feed/event-handlers/comment/index.d.ts +4 -0
  51. package/dist/src/feed/event-handlers/feed/handle-feed-updated.d.ts +3 -0
  52. package/dist/src/feed/event-handlers/feed/index.d.ts +1 -0
  53. package/dist/src/feed/event-handlers/feed-member/handle-feed-member-added.d.ts +3 -0
  54. package/dist/src/feed/event-handlers/feed-member/handle-feed-member-removed.d.ts +3 -0
  55. package/dist/src/feed/event-handlers/feed-member/handle-feed-member-updated.d.ts +3 -0
  56. package/dist/src/feed/event-handlers/feed-member/index.d.ts +3 -0
  57. package/dist/src/feed/event-handlers/follow/handle-follow-created.d.ts +7 -0
  58. package/dist/src/feed/event-handlers/follow/handle-follow-deleted.d.ts +7 -0
  59. package/dist/src/feed/event-handlers/follow/handle-follow-updated.d.ts +3 -0
  60. package/dist/src/feed/event-handlers/follow/index.d.ts +3 -0
  61. package/dist/src/feed/event-handlers/index.d.ts +7 -0
  62. package/dist/src/feed/event-handlers/notification-feed/handle-notification-feed-updated.d.ts +3 -0
  63. package/dist/src/feed/event-handlers/notification-feed/index.d.ts +1 -0
  64. package/dist/src/{Feed.d.ts → feed/feed.d.ts} +15 -34
  65. package/dist/src/feed/index.d.ts +2 -0
  66. package/dist/src/{FeedsClient.d.ts → feeds-client.d.ts} +6 -5
  67. package/dist/src/gen/models/index.d.ts +5 -0
  68. package/dist/src/gen-imports.d.ts +1 -1
  69. package/dist/src/test-utils/index.d.ts +1 -0
  70. package/dist/src/test-utils/response-generators.d.ts +9 -0
  71. package/dist/src/types-internal.d.ts +7 -0
  72. package/dist/src/types.d.ts +1 -1
  73. package/dist/src/utils/check-has-another-page.d.ts +1 -0
  74. package/dist/src/utils/constants.d.ts +3 -0
  75. package/dist/src/utils/index.d.ts +5 -0
  76. package/dist/src/utils/state-update-queue.d.ts +6 -0
  77. package/dist/src/utils/type-assertions.d.ts +7 -0
  78. package/dist/src/utils/unique-array-merge.d.ts +1 -0
  79. package/dist/tsconfig.tsbuildinfo +1 -1
  80. package/index.ts +2 -2
  81. package/package.json +2 -1
  82. package/src/common/ActivitySearchSource.ts +1 -1
  83. package/src/common/FeedSearchSource.ts +2 -2
  84. package/src/common/Poll.ts +1 -1
  85. package/src/common/UserSearchSource.ts +1 -1
  86. package/src/{state-updates → feed/event-handlers/activity}/activity-reaction-utils.test.ts +12 -2
  87. package/src/{state-updates → feed/event-handlers/activity}/activity-utils.test.ts +3 -2
  88. package/src/{state-updates/activity-utils.ts → feed/event-handlers/activity/handle-activity-added.ts} +16 -36
  89. package/src/feed/event-handlers/activity/handle-activity-deleted.ts +30 -0
  90. package/src/feed/event-handlers/activity/handle-activity-reaction-added.ts +67 -0
  91. package/src/feed/event-handlers/activity/handle-activity-reaction-deleted.ts +75 -0
  92. package/src/feed/event-handlers/activity/handle-activity-removed-from-feed.ts +16 -0
  93. package/src/feed/event-handlers/activity/handle-activity-updated.ts +47 -0
  94. package/src/feed/event-handlers/activity/index.ts +6 -0
  95. package/src/{state-updates → feed/event-handlers/bookmark}/bookmark-utils.test.ts +2 -2
  96. package/src/feed/event-handlers/bookmark/handle-bookmark-added.ts +63 -0
  97. package/src/feed/event-handlers/bookmark/handle-bookmark-deleted.ts +84 -0
  98. package/src/feed/event-handlers/bookmark/handle-bookmark-updated.ts +76 -0
  99. package/src/feed/event-handlers/bookmark/index.ts +3 -0
  100. package/src/feed/event-handlers/comment/handle-comment-added.ts +38 -0
  101. package/src/feed/event-handlers/comment/handle-comment-deleted.ts +35 -0
  102. package/src/feed/event-handlers/comment/handle-comment-reaction.ts +61 -0
  103. package/src/feed/event-handlers/comment/handle-comment-updated.ts +35 -0
  104. package/src/feed/event-handlers/comment/index.ts +4 -0
  105. package/src/feed/event-handlers/feed/handle-feed-updated.ts +9 -0
  106. package/src/feed/event-handlers/feed/index.ts +1 -0
  107. package/src/feed/event-handlers/feed-member/handle-feed-member-added.ts +31 -0
  108. package/src/feed/event-handlers/feed-member/handle-feed-member-removed.ts +24 -0
  109. package/src/feed/event-handlers/feed-member/handle-feed-member-updated.ts +40 -0
  110. package/src/feed/event-handlers/feed-member/index.ts +3 -0
  111. package/src/feed/event-handlers/follow/handle-follow-created.test.ts +246 -0
  112. package/src/feed/event-handlers/follow/handle-follow-created.ts +93 -0
  113. package/src/feed/event-handlers/follow/handle-follow-deleted.test.ts +264 -0
  114. package/src/feed/event-handlers/follow/handle-follow-deleted.ts +95 -0
  115. package/src/feed/event-handlers/follow/handle-follow-updated.test.ts +174 -0
  116. package/src/feed/event-handlers/follow/handle-follow-updated.ts +88 -0
  117. package/src/feed/event-handlers/follow/index.ts +3 -0
  118. package/src/feed/event-handlers/index.ts +7 -0
  119. package/src/feed/event-handlers/notification-feed/handle-notification-feed-updated.ts +10 -0
  120. package/src/feed/event-handlers/notification-feed/index.ts +1 -0
  121. package/src/{Feed.ts → feed/feed.ts} +72 -483
  122. package/src/feed/index.ts +2 -0
  123. package/src/{FeedsClient.ts → feeds-client.ts} +26 -8
  124. package/src/gen/model-decoders/decoders.ts +7 -0
  125. package/src/gen/models/index.ts +10 -0
  126. package/src/gen-imports.ts +1 -1
  127. package/src/test-utils/index.ts +1 -0
  128. package/src/test-utils/response-generators.ts +102 -0
  129. package/src/types-internal.ts +11 -0
  130. package/src/types.ts +1 -1
  131. package/src/utils/check-has-another-page.ts +6 -0
  132. package/src/utils/constants.ts +3 -0
  133. package/src/utils/index.ts +5 -0
  134. package/src/{state-updates → utils}/state-update-queue.test.ts +6 -6
  135. package/src/utils/state-update-queue.ts +42 -0
  136. package/src/utils/type-assertions.ts +22 -0
  137. package/src/{utils.test.ts → utils/unique-array-merge.test.ts} +7 -3
  138. package/src/utils/unique-array-merge.ts +19 -0
  139. package/dist/src/state-updates/activity-reaction-utils.d.ts +0 -10
  140. package/dist/src/state-updates/activity-utils.d.ts +0 -13
  141. package/dist/src/state-updates/bookmark-utils.d.ts +0 -14
  142. package/dist/src/state-updates/follow-utils.d.ts +0 -19
  143. package/dist/src/state-updates/state-update-queue.d.ts +0 -15
  144. package/dist/src/utils.d.ts +0 -10
  145. package/src/state-updates/activity-reaction-utils.ts +0 -107
  146. package/src/state-updates/bookmark-utils.ts +0 -167
  147. package/src/state-updates/follow-utils.test.ts +0 -552
  148. package/src/state-updates/follow-utils.ts +0 -126
  149. package/src/state-updates/state-update-queue.ts +0 -35
  150. package/src/utils.ts +0 -48
  151. /package/dist/src/{ModerationClient.d.ts → moderation-client.d.ts} +0 -0
  152. /package/src/{ModerationClient.ts → moderation-client.ts} +0 -0
@@ -0,0 +1,76 @@
1
+ import type { Feed } from '../../../feed';
2
+ import type {
3
+ ActivityResponse,
4
+ BookmarkUpdatedEvent,
5
+ } from '../../../gen/models';
6
+ import type { EventPayload, UpdateStateResult } from '../../../types-internal';
7
+
8
+ import { updateActivityInState } from '../activity';
9
+ import { isSameBookmark } from './handle-bookmark-deleted';
10
+
11
+ export const updateBookmarkInActivity = (
12
+ event: BookmarkUpdatedEvent,
13
+ activity: ActivityResponse,
14
+ isCurrentUser: boolean,
15
+ ): UpdateStateResult<ActivityResponse> => {
16
+ // Update own_bookmarks if the bookmark is from the current user
17
+ let ownBookmarks = activity.own_bookmarks || [];
18
+ if (isCurrentUser) {
19
+ const bookmarkIndex = ownBookmarks.findIndex((bookmark) =>
20
+ isSameBookmark(bookmark, event.bookmark),
21
+ );
22
+ if (bookmarkIndex !== -1) {
23
+ ownBookmarks = [...ownBookmarks];
24
+ ownBookmarks[bookmarkIndex] = event.bookmark;
25
+ }
26
+ }
27
+
28
+ return {
29
+ ...activity,
30
+ own_bookmarks: ownBookmarks,
31
+ changed: true,
32
+ };
33
+ };
34
+
35
+ export const updateBookmarkInActivities = (
36
+ event: BookmarkUpdatedEvent,
37
+ activities: ActivityResponse[] | undefined,
38
+ isCurrentUser: boolean,
39
+ ): UpdateStateResult<{ activities: ActivityResponse[] }> => {
40
+ if (!activities) {
41
+ return { changed: false, activities: [] };
42
+ }
43
+
44
+ const activityIndex = activities.findIndex(
45
+ (a) => a.id === event.bookmark.activity.id,
46
+ );
47
+ if (activityIndex === -1) {
48
+ return { changed: false, activities };
49
+ }
50
+
51
+ const activity = activities[activityIndex];
52
+ const updatedActivity = updateBookmarkInActivity(
53
+ event,
54
+ activity,
55
+ isCurrentUser,
56
+ );
57
+ return updateActivityInState(updatedActivity, activities, true);
58
+ };
59
+
60
+ export function handleBookmarkUpdated(
61
+ this: Feed,
62
+ event: EventPayload<'feeds.bookmark.updated'>,
63
+ ) {
64
+ const currentActivities = this.currentState.activities;
65
+ const { connected_user: connectedUser } = this.client.state.getLatestValue();
66
+ const isCurrentUser = event.bookmark.user.id === connectedUser?.id;
67
+
68
+ const result = updateBookmarkInActivities(
69
+ event,
70
+ currentActivities,
71
+ isCurrentUser,
72
+ );
73
+ if (result.changed) {
74
+ this.state.partialNext({ activities: result.activities });
75
+ }
76
+ }
@@ -0,0 +1,3 @@
1
+ export * from './handle-bookmark-added';
2
+ export * from './handle-bookmark-deleted';
3
+ export * from './handle-bookmark-updated';
@@ -0,0 +1,38 @@
1
+ import type { Feed } from '../../../feed';
2
+ import type { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleCommentAdded(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.comment.added'>,
7
+ ) {
8
+ const { comment } = event;
9
+ const entityId = comment.parent_id ?? comment.object_id;
10
+
11
+ this.state.next((currentState) => {
12
+ const entityState = currentState.comments_by_entity_id[entityId];
13
+
14
+ if (typeof entityState?.comments === 'undefined') {
15
+ return currentState;
16
+ }
17
+
18
+ const newComments = entityState?.comments ? [...entityState.comments] : [];
19
+
20
+ if (entityState.pagination?.sort === 'last') {
21
+ newComments.unshift(comment);
22
+ } else {
23
+ // 'first' and other sort options
24
+ newComments.push(comment);
25
+ }
26
+
27
+ return {
28
+ ...currentState,
29
+ comments_by_entity_id: {
30
+ ...currentState.comments_by_entity_id,
31
+ [entityId]: {
32
+ ...currentState.comments_by_entity_id[entityId],
33
+ comments: newComments,
34
+ },
35
+ },
36
+ };
37
+ });
38
+ }
@@ -0,0 +1,35 @@
1
+ import { Feed } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleCommentDeleted(
5
+ this: Feed,
6
+ { comment }: EventPayload<'feeds.comment.deleted'>,
7
+ ) {
8
+ const entityId = comment.parent_id ?? comment.object_id;
9
+
10
+ this.state.next((currentState) => {
11
+ const newCommentsByEntityId = {
12
+ ...currentState.comments_by_entity_id,
13
+ [entityId]: {
14
+ ...currentState.comments_by_entity_id[entityId],
15
+ },
16
+ };
17
+
18
+ const index = this.getCommentIndex(comment, currentState);
19
+
20
+ if (newCommentsByEntityId?.[entityId]?.comments?.length && index !== -1) {
21
+ newCommentsByEntityId[entityId].comments = [
22
+ ...newCommentsByEntityId[entityId].comments,
23
+ ];
24
+
25
+ newCommentsByEntityId[entityId]?.comments?.splice(index, 1);
26
+ }
27
+
28
+ delete newCommentsByEntityId[comment.id];
29
+
30
+ return {
31
+ ...currentState,
32
+ comments_by_entity_id: newCommentsByEntityId,
33
+ };
34
+ });
35
+ }
@@ -0,0 +1,61 @@
1
+ import { Feed } from '../../../feed';
2
+ import { CommentResponse } from '../../../gen/models';
3
+ import { EventPayload } from '../../../types-internal';
4
+
5
+ export function handleCommentReaction(
6
+ this: Feed,
7
+ event: EventPayload<
8
+ 'feeds.comment.reaction.added' | 'feeds.comment.reaction.deleted'
9
+ >,
10
+ ) {
11
+ const { comment, reaction } = event;
12
+ const connectedUser = this.client.state.getLatestValue().connected_user;
13
+
14
+ this.state.next((currentState) => {
15
+ const forId = comment.parent_id ?? comment.object_id;
16
+ const entityState = currentState.comments_by_entity_id[forId];
17
+
18
+ const commentIndex = this.getCommentIndex(comment, currentState);
19
+
20
+ if (commentIndex === -1) return currentState;
21
+
22
+ const newComments = entityState?.comments?.concat([]) ?? [];
23
+
24
+ const commentCopy: Partial<CommentResponse> = { ...comment };
25
+
26
+ delete commentCopy.own_reactions;
27
+
28
+ const newComment: CommentResponse = {
29
+ ...newComments[commentIndex],
30
+ ...commentCopy,
31
+ // TODO: FIXME this should be handled by the backend
32
+ latest_reactions: commentCopy.latest_reactions ?? [],
33
+ reaction_groups: commentCopy.reaction_groups ?? {},
34
+ };
35
+
36
+ newComments[commentIndex] = newComment;
37
+
38
+ if (reaction.user.id === connectedUser?.id) {
39
+ if (event.type === 'feeds.comment.reaction.added') {
40
+ newComment.own_reactions = newComment.own_reactions.concat(
41
+ reaction,
42
+ ) ?? [reaction];
43
+ } else if (event.type === 'feeds.comment.reaction.deleted') {
44
+ newComment.own_reactions = newComment.own_reactions.filter(
45
+ (r) => r.type !== reaction.type,
46
+ );
47
+ }
48
+ }
49
+
50
+ return {
51
+ ...currentState,
52
+ comments_by_entity_id: {
53
+ ...currentState.comments_by_entity_id,
54
+ [forId]: {
55
+ ...entityState,
56
+ comments: newComments,
57
+ },
58
+ },
59
+ };
60
+ });
61
+ }
@@ -0,0 +1,35 @@
1
+ import { Feed } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleCommentUpdated(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.comment.updated'>,
7
+ ) {
8
+ const { comment } = event;
9
+ const entityId = comment.parent_id ?? comment.object_id;
10
+
11
+ this.state.next((currentState) => {
12
+ const entityState = currentState.comments_by_entity_id[entityId];
13
+
14
+ if (!entityState?.comments?.length) return currentState;
15
+
16
+ const index = this.getCommentIndex(comment, currentState);
17
+
18
+ if (index === -1) return currentState;
19
+
20
+ const newComments = [...entityState.comments];
21
+
22
+ newComments[index] = comment;
23
+
24
+ return {
25
+ ...currentState,
26
+ comments_by_entity_id: {
27
+ ...currentState.comments_by_entity_id,
28
+ [entityId]: {
29
+ ...currentState.comments_by_entity_id[entityId],
30
+ comments: newComments,
31
+ },
32
+ },
33
+ };
34
+ });
35
+ }
@@ -0,0 +1,4 @@
1
+ export * from './handle-comment-added';
2
+ export * from './handle-comment-deleted';
3
+ export * from './handle-comment-updated';
4
+ export * from './handle-comment-reaction';
@@ -0,0 +1,9 @@
1
+ import { Feed } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleFeedUpdated(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.feed.updated'>,
7
+ ) {
8
+ this.state.partialNext({ ...event.feed });
9
+ }
@@ -0,0 +1 @@
1
+ export * from './handle-feed-updated';
@@ -0,0 +1,31 @@
1
+ import { Feed, FeedState } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleFeedMemberAdded(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.feed_member.added'>,
7
+ ) {
8
+ const { connected_user: connectedUser } = this.client.state.getLatestValue();
9
+
10
+ this.state.next((currentState) => {
11
+ let newState: FeedState | undefined;
12
+
13
+ if (typeof currentState.members !== 'undefined') {
14
+ newState ??= {
15
+ ...currentState,
16
+ };
17
+
18
+ newState.members = [event.member, ...currentState.members];
19
+ }
20
+
21
+ if (connectedUser?.id === event.member.user.id) {
22
+ newState ??= {
23
+ ...currentState,
24
+ };
25
+
26
+ newState.own_membership = event.member;
27
+ }
28
+
29
+ return newState ?? currentState;
30
+ });
31
+ }
@@ -0,0 +1,24 @@
1
+ import { Feed } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleFeedMemberRemoved(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.feed_member.removed'>,
7
+ ) {
8
+ const { connected_user: connectedUser } = this.client.state.getLatestValue();
9
+
10
+ this.state.next((currentState) => {
11
+ const newState = {
12
+ ...currentState,
13
+ members: currentState.members?.filter(
14
+ (member) => member.user.id !== event.user?.id,
15
+ ),
16
+ };
17
+
18
+ if (connectedUser?.id === event.member_id) {
19
+ delete newState.own_membership;
20
+ }
21
+
22
+ return newState;
23
+ });
24
+ }
@@ -0,0 +1,40 @@
1
+ import { Feed, FeedState } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleFeedMemberUpdated(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.feed_member.updated'>,
7
+ ) {
8
+ const { connected_user: connectedUser } = this.client.state.getLatestValue();
9
+
10
+ this.state.next((currentState) => {
11
+ const memberIndex =
12
+ currentState.members?.findIndex(
13
+ (member) => member.user.id === event.member.user.id,
14
+ ) ?? -1;
15
+
16
+ let newState: FeedState | undefined;
17
+
18
+ if (memberIndex !== -1) {
19
+ // if there's an index, there's a member to update
20
+ const newMembers = [...currentState.members!];
21
+ newMembers[memberIndex] = event.member;
22
+
23
+ newState ??= {
24
+ ...currentState,
25
+ };
26
+
27
+ newState.members = newMembers;
28
+ }
29
+
30
+ if (connectedUser?.id === event.member.user.id) {
31
+ newState ??= {
32
+ ...currentState,
33
+ };
34
+
35
+ newState.own_membership = event.member;
36
+ }
37
+
38
+ return newState ?? currentState;
39
+ });
40
+ }
@@ -0,0 +1,3 @@
1
+ export * from './handle-feed-member-added';
2
+ export * from './handle-feed-member-updated';
3
+ export * from './handle-feed-member-removed';
@@ -0,0 +1,246 @@
1
+ import { FeedResponse, FollowResponse, UserResponse } from '../../../gen/models';
2
+ import { generateFollowResponse } from '../../../test-utils';
3
+ import { updateStateFollowCreated } from './handle-follow-created';
4
+
5
+ import { describe, it, expect, beforeEach } from 'vitest';
6
+
7
+ describe('handle-follow-created', () => {
8
+ describe(updateStateFollowCreated.name, () => {
9
+ let mockFollow: FollowResponse;
10
+ let mockFeed: FeedResponse;
11
+ let mockUser: UserResponse;
12
+
13
+ beforeEach(() => {
14
+ mockFollow = generateFollowResponse();
15
+ mockFeed = mockFollow.source_feed;
16
+ mockUser = mockFeed.created_by;
17
+ });
18
+
19
+ it('should return unchanged state for non-accepted follows', () => {
20
+ const follow: FollowResponse = {
21
+ ...mockFollow,
22
+ status: 'pending',
23
+ };
24
+
25
+ // @ts-expect-error - we're not testing the full state here
26
+ const currentState: FeedState = {
27
+ followers: [],
28
+ following: [],
29
+ };
30
+
31
+ const result = updateStateFollowCreated(
32
+ follow,
33
+ currentState,
34
+ 'user:feed-1',
35
+ 'user-1',
36
+ );
37
+
38
+ expect(result.changed).toBe(false);
39
+ });
40
+
41
+ it('should handle when this feed follows someone', () => {
42
+ const follow: FollowResponse = {
43
+ ...mockFollow,
44
+ source_feed: {
45
+ ...mockFeed,
46
+ id: 'feed-x',
47
+ fid: 'user:feed-x',
48
+ created_by: {
49
+ ...mockUser,
50
+ id: 'user-x',
51
+ },
52
+ following_count: 1,
53
+ },
54
+ target_feed: {
55
+ ...mockFeed,
56
+ id: 'other-feed',
57
+ fid: 'user:other-feed',
58
+ created_by: mockUser,
59
+ },
60
+ };
61
+
62
+ // @ts-expect-error - we're not testing the full state here
63
+ const currentState: FeedState = {
64
+ following: [],
65
+ following_count: 0,
66
+ };
67
+
68
+ const result = updateStateFollowCreated(
69
+ follow,
70
+ currentState,
71
+ 'user:feed-x',
72
+ 'user-1',
73
+ );
74
+
75
+ expect(result.changed).toBe(true);
76
+ expect(result.data.following).toHaveLength(1);
77
+ expect(result.data.following?.[0]).toEqual(follow);
78
+ expect(result.data).toMatchObject(follow.source_feed);
79
+ expect(result.data.own_follows).toBeUndefined();
80
+ expect(result.data.following_count).toEqual(1);
81
+ });
82
+
83
+ it('should handle when someone follows this feed', () => {
84
+ const follow: FollowResponse = {
85
+ ...mockFollow,
86
+ source_feed: {
87
+ ...mockFeed,
88
+ id: 'other-feed',
89
+ fid: 'user:other-feed',
90
+ created_by: {
91
+ ...mockUser,
92
+ id: 'other-user',
93
+ },
94
+ },
95
+ target_feed: {
96
+ ...mockFeed,
97
+ id: 'feed-1',
98
+ fid: 'user:feed-1',
99
+ created_by: mockUser,
100
+ follower_count: 1,
101
+ },
102
+ };
103
+
104
+ // @ts-expect-error - we're not testing the full state here
105
+ const currentState: FeedState = {
106
+ followers: [],
107
+ follower_count: 0,
108
+ };
109
+
110
+ const result = updateStateFollowCreated(
111
+ follow,
112
+ currentState,
113
+ 'user:feed-1',
114
+ 'user-1',
115
+ );
116
+
117
+ expect(result.changed).toBe(true);
118
+ expect(result.data.followers).toHaveLength(1);
119
+ expect(result.data.followers?.[0]).toEqual(follow);
120
+ expect(result.data).toMatchObject(follow.target_feed);
121
+ expect(result.data.own_follows).toBeUndefined();
122
+ expect(result.data.follower_count).toEqual(1);
123
+ });
124
+
125
+ it('should add to own_follows when connected user is the source', () => {
126
+ const follow: FollowResponse = {
127
+ ...mockFollow,
128
+ source_feed: {
129
+ ...mockFeed,
130
+ id: 'feed-1',
131
+ fid: 'user:feed-1',
132
+ created_by: { ...mockUser, id: 'user-1' },
133
+ },
134
+ target_feed: {
135
+ ...mockFeed,
136
+ id: 'feed-x',
137
+ fid: 'user:feed-x',
138
+ created_by: {
139
+ ...mockUser,
140
+ id: 'user-x',
141
+ },
142
+ },
143
+ };
144
+
145
+ // @ts-expect-error - we're not testing the full state here
146
+ const currentState: FeedState = {
147
+ followers: [],
148
+ own_follows: [],
149
+ };
150
+
151
+ const result = updateStateFollowCreated(
152
+ follow,
153
+ currentState,
154
+ 'user:feed-x',
155
+ 'user-1',
156
+ );
157
+
158
+ expect(result.changed).toBe(true);
159
+ expect(result.data.own_follows).toHaveLength(1);
160
+ expect(result.data.own_follows?.[0]).toEqual(follow);
161
+ });
162
+
163
+ it('should not update followers/following when they are undefined', () => {
164
+ const follow: FollowResponse = {
165
+ ...mockFollow,
166
+ source_feed: {
167
+ ...mockFeed,
168
+ id: 'other-feed',
169
+ fid: 'user:other-feed',
170
+ created_by: mockUser,
171
+ },
172
+ target_feed: {
173
+ ...mockFeed,
174
+ id: 'feed-1',
175
+ fid: 'user:feed-1',
176
+ created_by: mockUser,
177
+ },
178
+ };
179
+
180
+ // @ts-expect-error - we're not testing the full state here
181
+ const currentState: FeedState = {
182
+ followers: undefined,
183
+ following: undefined,
184
+ own_follows: undefined,
185
+ };
186
+
187
+ const result = updateStateFollowCreated(
188
+ follow,
189
+ currentState,
190
+ 'user:feed-1',
191
+ 'user-1',
192
+ );
193
+
194
+ expect(result.changed).toBe(true);
195
+ expect(result.data.followers).toBeUndefined();
196
+ expect(result.data).toMatchObject(follow.target_feed);
197
+ });
198
+
199
+ it('should add new followers to the top of existing arrays', () => {
200
+ const existingFollow: FollowResponse = {
201
+ ...mockFollow,
202
+ source_feed: {
203
+ ...mockFeed,
204
+ id: 'existing-feed',
205
+ fid: 'user:existing-feed',
206
+ created_by: mockUser,
207
+ },
208
+ };
209
+
210
+ const follow: FollowResponse = {
211
+ ...mockFollow,
212
+ source_feed: {
213
+ ...mockFeed,
214
+ id: 'other-feed',
215
+ fid: 'user:other-feed',
216
+ created_by: mockUser,
217
+ },
218
+ target_feed: {
219
+ ...mockFeed,
220
+ id: 'feed-1',
221
+ fid: 'user:feed-1',
222
+ created_by: mockUser,
223
+ },
224
+ };
225
+
226
+ // @ts-expect-error - we're not testing the full state here
227
+ const currentState: FeedState = {
228
+ followers: [existingFollow],
229
+ following: undefined,
230
+ own_follows: undefined,
231
+ };
232
+
233
+ const result = updateStateFollowCreated(
234
+ follow,
235
+ currentState,
236
+ 'user:feed-1',
237
+ 'user-1',
238
+ );
239
+
240
+ expect(result.changed).toBe(true);
241
+ expect(result.data.followers).toHaveLength(2);
242
+ expect(result.data.followers?.[0]).toEqual(follow);
243
+ expect(result.data.followers?.[1]).toEqual(existingFollow);
244
+ });
245
+ });
246
+ });