diff --git a/.github/scripts/dev.js b/.github/scripts/dev.js index a6a7b96f49..2a3b0b4ade 100644 --- a/.github/scripts/dev.js +++ b/.github/scripts/dev.js @@ -78,7 +78,7 @@ const adminXApps = '@tryghost/admin-x-demo,@tryghost/admin-x-settings,@tryghost/ const COMMANDS_ADMINX = [{ name: 'adminXDeps', - command: 'while [ 1 ]; do nx watch --projects=apps/admin-x-design-system,apps/admin-x-framework -- nx run \\$NX_PROJECT_NAME:build; done', + command: 'while [ 1 ]; do nx watch --projects=apps/admin-x-design-system,apps/admin-x-framework,apps/shade -- nx run \\$NX_PROJECT_NAME:build; done', cwd: path.resolve(__dirname, '../..'), prefixColor: '#C72AF7', env: {} diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 7a88b03677..7c731ac3b9 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -103,6 +103,7 @@ jobs: admin-x-settings: - *shared - 'apps/admin-x-settings/**' + - 'apps/admin-x-design-system/**' announcement-bar: - *shared - 'apps/announcement-bar/**' diff --git a/.github/workflows/migration-review.yml b/.github/workflows/migration-review.yml index 885fa98f7e..9103b5d119 100644 --- a/.github/workflows/migration-review.yml +++ b/.github/workflows/migration-review.yml @@ -30,6 +30,7 @@ jobs: ### General requirements + - [ ] :warning: Tested on the staging database servers - [ ] Satisfies idempotency requirement (both `up()` and `down()`) - [ ] Does not reference models - [ ] Filename is in the correct format (and correctly ordered) diff --git a/.github/workflows/stale.yml b/.github/workflows/stale.yml index 62433c0b33..c241a75cef 100644 --- a/.github/workflows/stale.yml +++ b/.github/workflows/stale.yml @@ -21,7 +21,7 @@ jobs: If we’ve missed reviewing your PR & you’re still interested in working on it, please let us know. Otherwise this PR will be closed shortly, but can always be reopened later. Thank you for understanding 🙂 exempt-issue-labels: 'feature,pinned' exempt-pr-labels: 'feature,pinned' - days-before-stale: 120 + days-before-stale: 113 days-before-pr-stale: -1 stale-issue-label: 'stale' stale-pr-label: 'stale' diff --git a/apps/admin-x-activitypub/package.json b/apps/admin-x-activitypub/package.json index 585518a35d..5a4df2df64 100644 --- a/apps/admin-x-activitypub/package.json +++ b/apps/admin-x-activitypub/package.json @@ -1,6 +1,6 @@ { "name": "@tryghost/admin-x-activitypub", - "version": "0.3.24", + "version": "0.3.38", "license": "MIT", "repository": { "type": "git", diff --git a/apps/admin-x-activitypub/src/MainContent.tsx b/apps/admin-x-activitypub/src/MainContent.tsx index 225a9cc915..43fe5b59d0 100644 --- a/apps/admin-x-activitypub/src/MainContent.tsx +++ b/apps/admin-x-activitypub/src/MainContent.tsx @@ -2,62 +2,22 @@ import Activities from './components/Activities'; import Inbox from './components/Inbox'; import Profile from './components/Profile'; import Search from './components/Search'; -import {ActivityPubAPI} from './api/activitypub'; -import {useBrowseSite} from '@tryghost/admin-x-framework/api/site'; -import {useQuery} from '@tanstack/react-query'; import {useRouting} from '@tryghost/admin-x-framework/routing'; -export function useBrowseInboxForUser(handle: string) { - const site = useBrowseSite(); - const siteData = site.data?.site; - const siteUrl = siteData?.url ?? window.location.origin; - const api = new ActivityPubAPI( - new URL(siteUrl), - new URL('/ghost/api/admin/identities/', window.location.origin), - handle - ); - return useQuery({ - queryKey: [`inbox:${handle}`], - async queryFn() { - return api.getInbox(); - } - }); -} - -export function useFollowersForUser(handle: string) { - const site = useBrowseSite(); - const siteData = site.data?.site; - const siteUrl = siteData?.url ?? window.location.origin; - const api = new ActivityPubAPI( - new URL(siteUrl), - new URL('/ghost/api/admin/identities/', window.location.origin), - handle - ); - return useQuery({ - queryKey: [`followers:${handle}`], - async queryFn() { - return api.getFollowers(); - } - }); -} - const MainContent = () => { const {route} = useRouting(); const mainRoute = route.split('/')[0]; switch (mainRoute) { case 'search': return ; - break; case 'activity': return ; - break; case 'profile': return ; - break; default: const layout = (mainRoute === 'inbox' || mainRoute === '') ? 'inbox' : 'feed'; + return ; - break; } }; diff --git a/apps/admin-x-activitypub/src/api/activitypub.test.ts b/apps/admin-x-activitypub/src/api/activitypub.test.ts index 5c64da3623..12300e0ee7 100644 --- a/apps/admin-x-activitypub/src/api/activitypub.test.ts +++ b/apps/admin-x-activitypub/src/api/activitypub.test.ts @@ -192,13 +192,13 @@ describe('ActivityPubAPI', function () { }] }) }, - 'https://activitypub.api/.ghost/activitypub/outbox/index': { + 'https://activitypub.api/.ghost/activitypub/outbox/index?cursor=0': { async assert(_resource, init) { const headers = new Headers(init?.headers); expect(headers.get('Authorization')).toContain('fake-token'); }, response: JSONResponse({ - type: 'Collection', + type: 'OrderedCollection', orderedItems: [] }) } @@ -213,27 +213,20 @@ describe('ActivityPubAPI', function () { await api.getOutbox(); }); - test('Returns an empty array when the outbox collection is empty', async function () { + test('It returns an array of activities in the outbox collection', async function () { const fakeFetch = Fetch({ - 'https://auth.api/': { - response: JSONResponse({ - identities: [{ - token: 'fake-token' - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/outbox/index': { + [`https://activitypub.api/.ghost/activitypub/outbox/index?cursor=0`]: { response: JSONResponse({ type: 'OrderedCollection', - first: 'https://activitypub.api/.ghost/activitypub/outbox/index?cursor=0' - }) - }, - 'https://activitypub.api/.ghost/activitypub/outbox/index?cursor=0': { - response: JSONResponse({ - type: 'OrderedCollection' + orderedItems: [ + {id: 'https://example.com/activity/1'}, + {id: 'https://example.com/activity/2'} + ], + next: null }) } }); + const api = new ActivityPubAPI( new URL('https://activitypub.api'), new URL('https://auth.api'), @@ -242,49 +235,63 @@ describe('ActivityPubAPI', function () { ); const actual = await api.getOutbox(); - const expected: never[] = []; + + expect(actual.data).toEqual([ + {id: 'https://example.com/activity/1'}, + {id: 'https://example.com/activity/2'} + ]); + }); + + test('It returns next if it is present in the response', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/outbox/index?cursor=0`]: { + response: JSONResponse({ + type: 'OrderedCollection', + orderedItems: [], + next: 'https://activitypub.api/.ghost/activitypub/outbox/index?cursor=2' + }) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getOutbox(); + + expect(actual.next).toEqual('2'); + }); + + test('It returns a default return value when the response is null', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/outbox/index?cursor=0`]: { + response: JSONResponse(null) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getOutbox(); + const expected = { + data: [], + next: null + }; expect(actual).toEqual(expected); }); - test('Recursively retrieves all items and returns them when the outbox collection is not empty', async function () { + test('It returns a default return value if orderedItems is not present in the response', async function () { const fakeFetch = Fetch({ - 'https://auth.api/': { - response: JSONResponse({ - identities: [{ - token: 'fake-token' - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/outbox/index': { - response: - JSONResponse({ - type: 'OrderedCollection', - first: 'https://activitypub.api/.ghost/activitypub/outbox/index?cursor=0' - }) - }, - 'https://activitypub.api/.ghost/activitypub/outbox/index?cursor=0': { - response: JSONResponse({ - type: 'OrderedCollection', - next: 'https://activitypub.api/.ghost/activitypub/outbox/index?cursor=1', - orderedItems: [{ - type: 'Create', - object: { - type: 'Note' - } - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/outbox/index?cursor=1': { - response: JSONResponse({ - type: 'OrderedCollection', - orderedItems: [{ - type: 'Create', - object: { - type: 'Article' - } - }] - }) + [`https://activitypub.api/.ghost/activitypub/outbox/index?cursor=0`]: { + response: JSONResponse({}) } }); @@ -296,23 +303,32 @@ describe('ActivityPubAPI', function () { ); const actual = await api.getOutbox(); - const expected: Activity[] = [ - { - type: 'Create', - object: { - type: 'Note' - } - }, - { - type: 'Create', - object: { - type: 'Article' - } - } - ]; + const expected = { + data: [], + next: null + }; expect(actual).toEqual(expected); }); + + test('It returns an empty array if orderedItems in the response is not an array', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/outbox/index?cursor=0`]: { + response: JSONResponse({}) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getOutbox(); + + expect(actual.data).toEqual([]); + }); }); describe('getFollowing', function () { @@ -325,14 +341,14 @@ describe('ActivityPubAPI', function () { }] }) }, - 'https://activitypub.api/.ghost/activitypub/following/index': { + 'https://activitypub.api/.ghost/activitypub/following/index?cursor=0': { async assert(_resource, init) { const headers = new Headers(init?.headers); expect(headers.get('Authorization')).toContain('fake-token'); }, response: JSONResponse({ - type: 'Collection', - items: [] + type: 'OrderedCollection', + orderedItems: [] }) } }); @@ -346,27 +362,20 @@ describe('ActivityPubAPI', function () { await api.getFollowing(); }); - test('Returns an empty array when the following collection is empty', async function () { + test('It returns an array of actors in the following collection', async function () { const fakeFetch = Fetch({ - 'https://auth.api/': { - response: JSONResponse({ - identities: [{ - token: 'fake-token' - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/following/index': { + [`https://activitypub.api/.ghost/activitypub/following/index?cursor=0`]: { response: JSONResponse({ type: 'OrderedCollection', - first: 'https://activitypub.api/.ghost/activitypub/following/index?cursor=0' - }) - }, - 'https://activitypub.api/.ghost/activitypub/following/index?cursor=0': { - response: JSONResponse({ - type: 'OrderedCollection' + orderedItems: [ + {id: 'https://example.com/person/1'}, + {id: 'https://example.com/person/2'} + ], + next: null }) } }); + const api = new ActivityPubAPI( new URL('https://activitypub.api'), new URL('https://auth.api'), @@ -375,43 +384,63 @@ describe('ActivityPubAPI', function () { ); const actual = await api.getFollowing(); - const expected: never[] = []; + + expect(actual.data).toEqual([ + {id: 'https://example.com/person/1'}, + {id: 'https://example.com/person/2'} + ]); + }); + + test('It returns next if it is present in the response', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/following/index?cursor=0`]: { + response: JSONResponse({ + type: 'OrderedCollection', + orderedItems: [], + next: 'https://activitypub.api/.ghost/activitypub/following/index?cursor=2' + }) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getFollowing(); + + expect(actual.next).toEqual('2'); + }); + + test('It returns a default return value when the response is null', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/following/index?cursor=0`]: { + response: JSONResponse(null) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getFollowing(); + const expected = { + data: [], + next: null + }; expect(actual).toEqual(expected); }); - test('Recursively retrieves all items and returns them when the following collection is not empty', async function () { + test('It returns a default return value if orderedItems is not present in the response', async function () { const fakeFetch = Fetch({ - 'https://auth.api/': { - response: JSONResponse({ - identities: [{ - token: 'fake-token' - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/following/index': { - response: - JSONResponse({ - type: 'OrderedCollection', - first: 'https://activitypub.api/.ghost/activitypub/following/index?cursor=0' - }) - }, - 'https://activitypub.api/.ghost/activitypub/following/index?cursor=0': { - response: JSONResponse({ - type: 'OrderedCollection', - next: 'https://activitypub.api/.ghost/activitypub/following/index?cursor=1', - orderedItems: [{ - type: 'Person' - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/following/index?cursor=1': { - response: JSONResponse({ - type: 'OrderedCollection', - orderedItems: [{ - type: 'Group' - }] - }) + [`https://activitypub.api/.ghost/activitypub/following/index?cursor=0`]: { + response: JSONResponse({}) } }); @@ -423,17 +452,32 @@ describe('ActivityPubAPI', function () { ); const actual = await api.getFollowing(); - const expected: Activity[] = [ - { - type: 'Person' - }, - { - type: 'Group' - } - ]; + const expected = { + data: [], + next: null + }; expect(actual).toEqual(expected); }); + + test('It returns an empty array if orderedItems in the response is not an array', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/following/index?cursor=0`]: { + response: JSONResponse({}) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getFollowing(); + + expect(actual.data).toEqual([]); + }); }); describe('getFollowers', function () { @@ -446,13 +490,13 @@ describe('ActivityPubAPI', function () { }] }) }, - 'https://activitypub.api/.ghost/activitypub/followers/index': { + 'https://activitypub.api/.ghost/activitypub/followers/index?cursor=0': { async assert(_resource, init) { const headers = new Headers(init?.headers); expect(headers.get('Authorization')).toContain('fake-token'); }, response: JSONResponse({ - type: 'Collection', + type: 'OrderedCollection', orderedItems: [] }) } @@ -467,27 +511,20 @@ describe('ActivityPubAPI', function () { await api.getFollowers(); }); - test('Returns an empty array when the followers collection is empty', async function () { + test('It returns an array of actors in the followers collection', async function () { const fakeFetch = Fetch({ - 'https://auth.api/': { - response: JSONResponse({ - identities: [{ - token: 'fake-token' - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/followers/index': { + [`https://activitypub.api/.ghost/activitypub/followers/index?cursor=0`]: { response: JSONResponse({ type: 'OrderedCollection', - first: 'https://activitypub.api/.ghost/activitypub/followers/index?cursor=0' - }) - }, - 'https://activitypub.api/.ghost/activitypub/followers/index?cursor=0': { - response: JSONResponse({ - type: 'OrderedCollection' + orderedItems: [ + {id: 'https://example.com/person/1'}, + {id: 'https://example.com/person/2'} + ], + next: null }) } }); + const api = new ActivityPubAPI( new URL('https://activitypub.api'), new URL('https://auth.api'), @@ -496,43 +533,63 @@ describe('ActivityPubAPI', function () { ); const actual = await api.getFollowers(); - const expected: never[] = []; + + expect(actual.data).toEqual([ + {id: 'https://example.com/person/1'}, + {id: 'https://example.com/person/2'} + ]); + }); + + test('It returns next if it is present in the response', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/followers/index?cursor=0`]: { + response: JSONResponse({ + type: 'OrderedCollection', + orderedItems: [], + next: 'https://activitypub.api/.ghost/activitypub/following/index?cursor=2' + }) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getFollowers(); + + expect(actual.next).toEqual('2'); + }); + + test('It returns a default return value when the response is null', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/followers/index?cursor=0`]: { + response: JSONResponse(null) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getFollowers(); + const expected = { + data: [], + next: null + }; expect(actual).toEqual(expected); }); - test('Recursively retrieves all items and returns them when the followers collection is not empty', async function () { + test('It returns a default return value if orderedItems is not present in the response', async function () { const fakeFetch = Fetch({ - 'https://auth.api/': { - response: JSONResponse({ - identities: [{ - token: 'fake-token' - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/followers/index': { - response: - JSONResponse({ - type: 'OrderedCollection', - first: 'https://activitypub.api/.ghost/activitypub/followers/index?cursor=0' - }) - }, - 'https://activitypub.api/.ghost/activitypub/followers/index?cursor=0': { - response: JSONResponse({ - type: 'OrderedCollection', - next: 'https://activitypub.api/.ghost/activitypub/followers/index?cursor=1', - orderedItems: [{ - type: 'Person' - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/followers/index?cursor=1': { - response: JSONResponse({ - type: 'OrderedCollection', - orderedItems: [{ - type: 'Group' - }] - }) + [`https://activitypub.api/.ghost/activitypub/followers/index?cursor=0`]: { + response: JSONResponse({}) } }); @@ -544,21 +601,36 @@ describe('ActivityPubAPI', function () { ); const actual = await api.getFollowers(); - const expected: Activity[] = [ - { - type: 'Person' - }, - { - type: 'Group' - } - ]; + const expected = { + data: [], + next: null + }; expect(actual).toEqual(expected); }); + + test('It returns an empty array if orderedItems in the response is not an array', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/followers/index?cursor=0`]: { + response: JSONResponse({}) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getFollowers(); + + expect(actual.data).toEqual([]); + }); }); describe('getLiked', function () { - test('It passes the token to the liked endpoint', async function () { + test('It passes the token to the liked collection endpoint', async function () { const fakeFetch = Fetch({ 'https://auth.api/': { response: JSONResponse({ @@ -567,13 +639,13 @@ describe('ActivityPubAPI', function () { }] }) }, - 'https://activitypub.api/.ghost/activitypub/liked/index': { + 'https://activitypub.api/.ghost/activitypub/liked/index?cursor=0': { async assert(_resource, init) { const headers = new Headers(init?.headers); expect(headers.get('Authorization')).toContain('fake-token'); }, response: JSONResponse({ - type: 'Collection', + type: 'OrderedCollection', orderedItems: [] }) } @@ -588,27 +660,20 @@ describe('ActivityPubAPI', function () { await api.getLiked(); }); - test('Returns an empty array when the liked collection is empty', async function () { + test('It returns an array of liked activities in the liked collection', async function () { const fakeFetch = Fetch({ - 'https://auth.api/': { - response: JSONResponse({ - identities: [{ - token: 'fake-token' - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/liked/index': { + [`https://activitypub.api/.ghost/activitypub/liked/index?cursor=0`]: { response: JSONResponse({ type: 'OrderedCollection', - first: 'https://activitypub.api/.ghost/activitypub/liked/index?cursor=0' - }) - }, - 'https://activitypub.api/.ghost/activitypub/liked/index?cursor=0': { - response: JSONResponse({ - type: 'OrderedCollection' + orderedItems: [ + {id: 'https://example.com/activity/1'}, + {id: 'https://example.com/activity/2'} + ], + next: null }) } }); + const api = new ActivityPubAPI( new URL('https://activitypub.api'), new URL('https://auth.api'), @@ -617,49 +682,63 @@ describe('ActivityPubAPI', function () { ); const actual = await api.getLiked(); - const expected: never[] = []; + + expect(actual.data).toEqual([ + {id: 'https://example.com/activity/1'}, + {id: 'https://example.com/activity/2'} + ]); + }); + + test('It returns next if it is present in the response', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/liked/index?cursor=0`]: { + response: JSONResponse({ + type: 'OrderedCollection', + orderedItems: [], + next: 'https://activitypub.api/.ghost/activitypub/following/index?cursor=2' + }) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getLiked(); + + expect(actual.next).toEqual('2'); + }); + + test('It returns a default return value when the response is null', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/liked/index?cursor=0`]: { + response: JSONResponse(null) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getLiked(); + const expected = { + data: [], + next: null + }; expect(actual).toEqual(expected); }); - test('Recursively retrieves all items and returns them when the liked collection is not empty', async function () { + test('It returns a default return value if orderedItems is not present in the response', async function () { const fakeFetch = Fetch({ - 'https://auth.api/': { - response: JSONResponse({ - identities: [{ - token: 'fake-token' - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/liked/index': { - response: - JSONResponse({ - type: 'OrderedCollection', - first: 'https://activitypub.api/.ghost/activitypub/liked/index?cursor=0' - }) - }, - 'https://activitypub.api/.ghost/activitypub/liked/index?cursor=0': { - response: JSONResponse({ - type: 'OrderedCollection', - next: 'https://activitypub.api/.ghost/activitypub/liked/index?cursor=1', - orderedItems: [{ - type: 'Create', - object: { - type: 'Note' - } - }] - }) - }, - 'https://activitypub.api/.ghost/activitypub/liked/index?cursor=1': { - response: JSONResponse({ - type: 'OrderedCollection', - orderedItems: [{ - type: 'Create', - object: { - type: 'Article' - } - }] - }) + [`https://activitypub.api/.ghost/activitypub/liked/index?cursor=0`]: { + response: JSONResponse({}) } }); @@ -671,23 +750,32 @@ describe('ActivityPubAPI', function () { ); const actual = await api.getLiked(); - const expected: Activity[] = [ - { - type: 'Create', - object: { - type: 'Note' - } - }, - { - type: 'Create', - object: { - type: 'Article' - } - } - ]; + const expected = { + data: [], + next: null + }; expect(actual).toEqual(expected); }); + + test('It returns an empty array if orderedItems in the response is not an array', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/liked/index?cursor=0`]: { + response: JSONResponse({}) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getLiked(); + + expect(actual.data).toEqual([]); + }); }); describe('follow', function () { @@ -719,6 +807,35 @@ describe('ActivityPubAPI', function () { }); }); + describe('note', function () { + test('It creates a note and returns it', async function () { + const fakeFetch = Fetch({ + [`https://activitypub.api/.ghost/activitypub/actions/note`]: { + async assert(_resource, init) { + expect(init?.method).toEqual('POST'); + expect(init?.body).toEqual('{"content":"Hello, world!"}'); + }, + response: JSONResponse({ + id: 'https://example.com/note/abc123' + }) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const result = await api.note('Hello, world!'); + + expect(result).toEqual({ + id: 'https://example.com/note/abc123' + }); + }); + }); + describe('search', function () { test('It returns the results of the search', async function () { const handle = '@foo@bar.baz'; @@ -795,6 +912,43 @@ describe('ActivityPubAPI', function () { }); }); + describe('getProfile', function () { + test('It returns a profile', async function () { + const handle = '@foo@bar.baz'; + + const fakeFetch = Fetch({ + 'https://auth.api/': { + response: JSONResponse({ + identities: [{ + token: 'fake-token' + }] + }) + }, + [`https://activitypub.api/.ghost/activitypub/profile/${handle}`]: { + response: JSONResponse({ + handle, + name: 'Foo Bar' + }) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getProfile(handle); + const expected = { + handle, + name: 'Foo Bar' + }; + + expect(actual).toEqual(expected); + }); + }); + describe('getFollowersForProfile', function () { test('It returns an array of followers for a profile', async function () { const handle = '@foo@bar.baz'; @@ -1011,7 +1165,7 @@ describe('ActivityPubAPI', function () { }, [`https://activitypub.api/.ghost/activitypub/profile/${handle}/followers`]: { response: JSONResponse({ - followers: {} + followers: [] }) } }); @@ -1245,7 +1399,7 @@ describe('ActivityPubAPI', function () { }, [`https://activitypub.api/.ghost/activitypub/profile/${handle}/following`]: { response: JSONResponse({ - following: {} + following: [] }) } }); @@ -1263,8 +1417,8 @@ describe('ActivityPubAPI', function () { }); }); - describe('getProfile', function () { - test('It returns a profile', async function () { + describe('getPostsForProfile', function () { + test('It returns an array of posts for a profile', async function () { const handle = '@foo@bar.baz'; const fakeFetch = Fetch({ @@ -1275,10 +1429,27 @@ describe('ActivityPubAPI', function () { }] }) }, - [`https://activitypub.api/.ghost/activitypub/profile/${handle}`]: { + [`https://activitypub.api/.ghost/activitypub/profile/${handle}/posts`]: { response: JSONResponse({ - handle, - name: 'Foo Bar' + posts: [ + { + actor: { + id: 'https://example.com/users/bar' + }, + object: { + content: 'Hello, world!' + } + }, + { + actor: { + id: 'https://example.com/users/baz' + }, + object: { + content: 'Hello, world again!' + } + } + ], + next: null }) } }); @@ -1290,14 +1461,206 @@ describe('ActivityPubAPI', function () { fakeFetch ); - const actual = await api.getProfile(handle); + const actual = await api.getPostsForProfile(handle); + + expect(actual.posts).toEqual([ + { + actor: { + id: 'https://example.com/users/bar' + }, + object: { + content: 'Hello, world!' + } + }, + { + actor: { + id: 'https://example.com/users/baz' + }, + object: { + content: 'Hello, world again!' + } + } + ]); + }); + + test('It returns next if it is present in the response', async function () { + const handle = '@foo@bar.baz'; + + const fakeFetch = Fetch({ + 'https://auth.api/': { + response: JSONResponse({ + identities: [{ + token: 'fake-token' + }] + }) + }, + [`https://activitypub.api/.ghost/activitypub/profile/${handle}/posts`]: { + response: JSONResponse({ + posts: [], + next: 'abc123' + }) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getPostsForProfile(handle); + + expect(actual.next).toEqual('abc123'); + }); + + test('It includes next in the query when provided', async function () { + const handle = '@foo@bar.baz'; + const next = 'abc123'; + + const fakeFetch = Fetch({ + 'https://auth.api/': { + response: JSONResponse({ + identities: [{ + token: 'fake-token' + }] + }) + }, + [`https://activitypub.api/.ghost/activitypub/profile/${handle}/posts?next=${next}`]: { + response: JSONResponse({ + posts: [ + { + actor: { + id: 'https://example.com/users/bar' + }, + object: { + content: 'Hello, world!' + } + } + ], + next: null + }) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getPostsForProfile(handle, next); const expected = { - handle, - name: 'Foo Bar' + posts: [ + { + actor: { + id: 'https://example.com/users/bar' + }, + object: { + content: 'Hello, world!' + } + } + ], + next: null }; expect(actual).toEqual(expected); }); + + test('It returns a default return value when the response is null', async function () { + const handle = '@foo@bar.baz'; + + const fakeFetch = Fetch({ + 'https://auth.api/': { + response: JSONResponse({ + identities: [{ + token: 'fake-token' + }] + }) + }, + [`https://activitypub.api/.ghost/activitypub/profile/${handle}/posts`]: { + response: JSONResponse(null) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getPostsForProfile(handle); + const expected = { + posts: [], + next: null + }; + + expect(actual).toEqual(expected); + }); + + test('It returns a default return value if followers is not present in the response', async function () { + const handle = '@foo@bar.baz'; + + const fakeFetch = Fetch({ + 'https://auth.api/': { + response: JSONResponse({ + identities: [{ + token: 'fake-token' + }] + }) + }, + [`https://activitypub.api/.ghost/activitypub/profile/${handle}/posts`]: { + response: JSONResponse({}) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getPostsForProfile(handle); + const expected = { + posts: [], + next: null + }; + + expect(actual).toEqual(expected); + }); + + test('It returns an empty array of followers if followers in the response is not an array', async function () { + const handle = '@foo@bar.baz'; + + const fakeFetch = Fetch({ + 'https://auth.api/': { + response: JSONResponse({ + identities: [{ + token: 'fake-token' + }] + }) + }, + [`https://activitypub.api/.ghost/activitypub/profile/${handle}/posts`]: { + response: JSONResponse({ + posts: [] + }) + } + }); + + const api = new ActivityPubAPI( + new URL('https://activitypub.api'), + new URL('https://auth.api'), + 'index', + fakeFetch + ); + + const actual = await api.getPostsForProfile(handle); + + expect(actual.posts).toEqual([]); + }); }); describe('getThread', function () { @@ -1336,33 +1699,4 @@ describe('ActivityPubAPI', function () { expect(actual).toEqual(expected); }); }); - - describe('note', function () { - test('It creates a note and returns it', async function () { - const fakeFetch = Fetch({ - [`https://activitypub.api/.ghost/activitypub/actions/note`]: { - async assert(_resource, init) { - expect(init?.method).toEqual('POST'); - expect(init?.body).toEqual('{"content":"Hello, world!"}'); - }, - response: JSONResponse({ - id: 'https://example.com/note/abc123' - }) - } - }); - - const api = new ActivityPubAPI( - new URL('https://activitypub.api'), - new URL('https://auth.api'), - 'index', - fakeFetch - ); - - const result = await api.note('Hello, world!'); - - expect(result).toEqual({ - id: 'https://example.com/note/abc123' - }); - }); - }); }); diff --git a/apps/admin-x-activitypub/src/api/activitypub.ts b/apps/admin-x-activitypub/src/api/activitypub.ts index 11a31876a1..6b0a89078d 100644 --- a/apps/admin-x-activitypub/src/api/activitypub.ts +++ b/apps/admin-x-activitypub/src/api/activitypub.ts @@ -1,5 +1,6 @@ // eslint-disable-next-line @typescript-eslint/no-explicit-any export type Actor = any; + // eslint-disable-next-line @typescript-eslint/no-explicit-any export type Activity = any; @@ -9,13 +10,18 @@ export interface Profile { followerCount: number; followingCount: number; isFollowing: boolean; - posts: Activity[]; } export interface SearchResults { profiles: Profile[]; } +export interface ActivityThread { + items: Activity[]; +} + +export type ActivityPubCollectionResponse = {data: T[], next: string | null}; + export interface GetFollowersForProfileResponse { followers: { actor: Actor; @@ -32,8 +38,9 @@ export interface GetFollowingForProfileResponse { next: string | null; } -export interface ActivityThread { - items: Activity[]; +export interface GetPostsForProfileResponse { + posts: Activity[]; + next: string | null; } export class ActivityPubAPI { @@ -73,39 +80,52 @@ export class ActivityPubAPI { return json; } - private async getActivityPubCollection(collectionUrl: URL): Promise { - const fetchPage = async (pageUrl: URL): Promise => { - const json = await this.fetchJSON(pageUrl); + private async getActivityPubCollection(collectionUrl: URL, cursor?: string): Promise> { + const url = new URL(collectionUrl); + url.searchParams.set('cursor', cursor || '0'); - if (json === null) { - return []; - } + const json = await this.fetchJSON(url); - let items: T[] = []; - - if ('orderedItems' in json) { - items = Array.isArray(json.orderedItems) ? json.orderedItems : [json.orderedItems]; - } - - if ('next' in json && typeof json.next === 'string') { - const nextPageUrl = new URL(json.next); - const nextPageItems = await fetchPage(nextPageUrl); - - items = items.concat(nextPageItems); - } - - return items; - }; - - const initialJson = await this.fetchJSON(collectionUrl); - - if (initialJson === null || !('first' in initialJson) || typeof initialJson.first !== 'string') { - return []; + if (json === null) { + return { + data: [], + next: null + }; } - const firstPageUrl = new URL(initialJson.first); + if (!('orderedItems' in json)) { + return { + data: [], + next: null + }; + } - return fetchPage(firstPageUrl); + const data = Array.isArray(json.orderedItems) ? json.orderedItems : []; + let next = 'next' in json && typeof json.next === 'string' ? json.next : null; + + if (next !== null) { + const nextUrl = new URL(next); + next = nextUrl.searchParams.get('cursor') || null; + } + + return { + data, + next + }; + } + + private async getActivityPubCollectionCount(collectionUrl: URL): Promise { + const json = await this.fetchJSON(collectionUrl); + + if (json === null) { + return 0; + } + + if ('totalItems' in json && typeof json.totalItems === 'number') { + return json.totalItems; + } + + return 0; } get inboxApiUrl() { @@ -130,108 +150,32 @@ export class ActivityPubAPI { return new URL(`.ghost/activitypub/outbox/${this.handle}`, this.apiUrl); } - async getOutbox(): Promise { - return this.getActivityPubCollection(this.outboxApiUrl); + async getOutbox(cursor?: string): Promise> { + return this.getActivityPubCollection(this.outboxApiUrl, cursor); } get followingApiUrl() { return new URL(`.ghost/activitypub/following/${this.handle}`, this.apiUrl); } - async getFollowing(): Promise { - return this.getActivityPubCollection(this.followingApiUrl); + async getFollowing(cursor?: string): Promise> { + return this.getActivityPubCollection(this.followingApiUrl, cursor); } async getFollowingCount(): Promise { - const json = await this.fetchJSON(this.followingApiUrl); - if (json === null) { - return 0; - } - if ('totalItems' in json && typeof json.totalItems === 'number') { - return json.totalItems; - } - return 0; + return this.getActivityPubCollectionCount(this.followingApiUrl); } get followersApiUrl() { return new URL(`.ghost/activitypub/followers/${this.handle}`, this.apiUrl); } - async getFollowers(): Promise { - return this.getActivityPubCollection(this.followersApiUrl); + async getFollowers(cursor?: string): Promise> { + return this.getActivityPubCollection(this.followersApiUrl, cursor); } async getFollowersCount(): Promise { - const json = await this.fetchJSON(this.followersApiUrl); - if (json === null) { - return 0; - } - if ('totalItems' in json && typeof json.totalItems === 'number') { - return json.totalItems; - } - return 0; - } - - async getFollowersForProfile(handle: string, next?: string): Promise { - const url = new URL(`.ghost/activitypub/profile/${handle}/followers`, this.apiUrl); - if (next) { - url.searchParams.set('next', next); - } - - const json = await this.fetchJSON(url); - - if (json === null) { - return { - followers: [], - next: null - }; - } - - if (!('followers' in json)) { - return { - followers: [], - next: null - }; - } - - const followers = Array.isArray(json.followers) ? json.followers : []; - const nextPage = 'next' in json && typeof json.next === 'string' ? json.next : null; - - return { - followers, - next: nextPage - }; - } - - async getFollowingForProfile(handle: string, next?: string): Promise { - const url = new URL(`.ghost/activitypub/profile/${handle}/following`, this.apiUrl); - if (next) { - url.searchParams.set('next', next); - } - - const json = await this.fetchJSON(url); - - if (json === null) { - return { - following: [], - next: null - }; - } - - if (!('following' in json)) { - return { - following: [], - next: null - }; - } - - const following = Array.isArray(json.following) ? json.following : []; - const nextPage = 'next' in json && typeof json.next === 'string' ? json.next : null; - - return { - following, - next: nextPage - }; + return this.getActivityPubCollectionCount(this.followersApiUrl); } async follow(username: string): Promise { @@ -240,17 +184,16 @@ export class ActivityPubAPI { return json as Actor; } - async getActor(url: string): Promise { - const json = await this.fetchJSON(new URL(url)); - return json as Actor; - } - get likedApiUrl() { return new URL(`.ghost/activitypub/liked/${this.handle}`, this.apiUrl); } - async getLiked() { - return this.getActivityPubCollection(this.likedApiUrl); + async getLiked(cursor?: string): Promise> { + return this.getActivityPubCollection(this.likedApiUrl, cursor); + } + + async getLikedCount(): Promise { + return this.getActivityPubCollectionCount(this.likedApiUrl); } async like(id: string): Promise { @@ -270,7 +213,6 @@ export class ActivityPubAPI { async getActivities( includeOwn: boolean = false, includeReplies: boolean = false, - excludeNonFollowers: boolean = false, filter: {type?: string[]} | null = null, cursor?: string ): Promise<{data: Activity[], next: string | null}> { @@ -284,9 +226,6 @@ export class ActivityPubAPI { if (includeReplies) { url.searchParams.set('includeReplies', includeReplies.toString()); } - if (excludeNonFollowers) { - url.searchParams.set('excludeNonFollowers', excludeNonFollowers.toString()); - } if (filter) { url.searchParams.set('filter', JSON.stringify(filter)); } @@ -366,6 +305,99 @@ export class ActivityPubAPI { return json as Profile; } + async getFollowersForProfile(handle: string, next?: string): Promise { + const url = new URL(`.ghost/activitypub/profile/${handle}/followers`, this.apiUrl); + if (next) { + url.searchParams.set('next', next); + } + + const json = await this.fetchJSON(url); + + if (json === null) { + return { + followers: [], + next: null + }; + } + + if (!('followers' in json)) { + return { + followers: [], + next: null + }; + } + + const followers = Array.isArray(json.followers) ? json.followers : []; + const nextPage = 'next' in json && typeof json.next === 'string' ? json.next : null; + + return { + followers, + next: nextPage + }; + } + + async getFollowingForProfile(handle: string, next?: string): Promise { + const url = new URL(`.ghost/activitypub/profile/${handle}/following`, this.apiUrl); + if (next) { + url.searchParams.set('next', next); + } + + const json = await this.fetchJSON(url); + + if (json === null) { + return { + following: [], + next: null + }; + } + + if (!('following' in json)) { + return { + following: [], + next: null + }; + } + + const following = Array.isArray(json.following) ? json.following : []; + const nextPage = 'next' in json && typeof json.next === 'string' ? json.next : null; + + return { + following, + next: nextPage + }; + } + + async getPostsForProfile(handle: string, next?: string): Promise { + const url = new URL(`.ghost/activitypub/profile/${handle}/posts`, this.apiUrl); + if (next) { + url.searchParams.set('next', next); + } + + const json = await this.fetchJSON(url); + + if (json === null) { + return { + posts: [], + next: null + }; + } + + if (!('posts' in json)) { + return { + posts: [], + next: null + }; + } + + const posts = Array.isArray(json.posts) ? json.posts : []; + const nextPage = 'next' in json && typeof json.next === 'string' ? json.next : null; + + return { + posts, + next: nextPage + }; + } + async getThread(id: string): Promise { const url = new URL(`.ghost/activitypub/thread/${encodeURIComponent(id)}`, this.apiUrl); const json = await this.fetchJSON(url); diff --git a/apps/admin-x-activitypub/src/components/Activities.tsx b/apps/admin-x-activitypub/src/components/Activities.tsx index ed77f360fb..97e167fc53 100644 --- a/apps/admin-x-activitypub/src/components/Activities.tsx +++ b/apps/admin-x-activitypub/src/components/Activities.tsx @@ -1,95 +1,168 @@ import React, {useEffect, useRef} from 'react'; import NiceModal from '@ebay/nice-modal-react'; -import {ActorProperties} from '@tryghost/admin-x-framework/api/activitypub'; -import {LoadingIndicator, NoValueLabel} from '@tryghost/admin-x-design-system'; +import {Activity, ActorProperties, ObjectProperties} from '@tryghost/admin-x-framework/api/activitypub'; +import {Button, LoadingIndicator, NoValueLabel} from '@tryghost/admin-x-design-system'; -import APAvatar, {AvatarBadge} from './global/APAvatar'; -import ActivityItem, {type Activity} from './activities/ActivityItem'; +import APAvatar from './global/APAvatar'; import ArticleModal from './feed/ArticleModal'; import MainNavigation from './navigation/MainNavigation'; +import NotificationItem from './activities/NotificationItem'; import Separator from './global/Separator'; -import ViewProfileModal from './global/ViewProfileModal'; import getUsername from '../utils/get-username'; import stripHtml from '../utils/strip-html'; -import {useActivitiesForUser} from '../hooks/useActivityPubQueries'; +import truncate from '../utils/truncate'; +import {GET_ACTIVITIES_QUERY_KEY_NOTIFICATIONS, useActivitiesForUser} from '../hooks/useActivityPubQueries'; +import {type NotificationType} from './activities/NotificationIcon'; +import {handleProfileClick} from '../utils/handle-profile-click'; interface ActivitiesProps {} // eslint-disable-next-line no-shadow -enum ACTVITY_TYPE { +enum ACTIVITY_TYPE { CREATE = 'Create', LIKE = 'Like', FOLLOW = 'Follow' } -const getActivityDescription = (activity: Activity): string => { - switch (activity.type) { - case ACTVITY_TYPE.CREATE: - if (activity.object?.inReplyTo && typeof activity.object?.inReplyTo !== 'string') { - return `Replied to your article "${activity.object.inReplyTo.name}"`; - } +interface GroupedActivity { + type: ACTIVITY_TYPE; + actors: ActorProperties[]; + object: ObjectProperties; + id?: string; +} - return ''; - case ACTVITY_TYPE.FOLLOW: - return 'Followed you'; - case ACTVITY_TYPE.LIKE: - if (activity.object && activity.object.type === 'Article') { - return `Liked your article "${activity.object.name}"`; - } else if (activity.object && activity.object.type === 'Note') { - return `${stripHtml(activity.object.content)}`; - } - } - - return ''; -}; - -const getExtendedDescription = (activity: Activity): JSX.Element | null => { +const getExtendedDescription = (activity: GroupedActivity): JSX.Element | null => { // If the activity is a reply - if (Boolean(activity.type === ACTVITY_TYPE.CREATE && activity.object?.inReplyTo)) { + if (Boolean(activity.type === ACTIVITY_TYPE.CREATE && activity.object?.inReplyTo)) { return (
); + } else if (activity.type === ACTIVITY_TYPE.LIKE && !activity.object?.name && activity.object?.content) { + return ( +
+ ); } return null; }; -const getActivityUrl = (activity: Activity): string | null => { - if (activity.object) { - return activity.object.url || null; - } - - return null; -}; - -const getActorUrl = (activity: Activity): string | null => { - if (activity.actor) { - return activity.actor.url; - } - - return null; -}; - -const getActivityBadge = (activity: Activity): AvatarBadge => { +const getActivityBadge = (activity: GroupedActivity): NotificationType => { switch (activity.type) { - case ACTVITY_TYPE.CREATE: - return 'comment-fill'; - case ACTVITY_TYPE.FOLLOW: - return 'user-fill'; - case ACTVITY_TYPE.LIKE: + case ACTIVITY_TYPE.CREATE: + return 'reply'; + case ACTIVITY_TYPE.FOLLOW: + return 'follow'; + case ACTIVITY_TYPE.LIKE: if (activity.object) { - return 'heart-fill'; + return 'like'; } } + + return 'like'; +}; + +const groupActivities = (activities: Activity[]): GroupedActivity[] => { + const groups: {[key: string]: GroupedActivity} = {}; + + // Activities are already sorted by time from the API + activities.forEach((activity) => { + let groupKey = ''; + + switch (activity.type) { + case ACTIVITY_TYPE.FOLLOW: + // Group follows that are next to each other in the array + groupKey = `follow_${activity.type}`; + break; + case ACTIVITY_TYPE.LIKE: + if (activity.object?.id) { + // Group likes by the target object + groupKey = `like_${activity.object.id}`; + } + break; + case ACTIVITY_TYPE.CREATE: + // Don't group creates/replies + groupKey = `create_${activity.id}`; + break; + } + + if (!groups[groupKey]) { + groups[groupKey] = { + type: activity.type as ACTIVITY_TYPE, + actors: [], + object: activity.object, + id: activity.id + }; + } + + // Add actor if not already in the group + if (!groups[groupKey].actors.find(a => a.id === activity.actor.id)) { + groups[groupKey].actors.push(activity.actor); + } + }); + + // Return in same order as original activities + return Object.values(groups); +}; + +const getGroupDescription = (group: GroupedActivity): JSX.Element => { + const [firstActor, secondActor, ...otherActors] = group.actors; + const hasOthers = otherActors.length > 0; + + const actorClass = 'cursor-pointer font-semibold hover:underline'; + + const actorText = ( + <> + handleProfileClick(firstActor, e)} + >{firstActor.name} + {secondActor && ( + <> + {hasOthers ? ', ' : ' and '} + handleProfileClick(secondActor, e)} + >{secondActor.name} + + )} + {hasOthers && ' and others'} + + ); + + switch (group.type) { + case ACTIVITY_TYPE.FOLLOW: + return <>{actorText} started following you; + case ACTIVITY_TYPE.LIKE: + return <>{actorText} liked your post {group.object?.name || ''}; + case ACTIVITY_TYPE.CREATE: + if (group.object?.inReplyTo && typeof group.object?.inReplyTo !== 'string') { + const content = stripHtml(group.object.inReplyTo.name); + return <>{actorText} replied to your post {truncate(content, 80)}; + } + } + return <>; }; const Activities: React.FC = ({}) => { const user = 'index'; + const [openStates, setOpenStates] = React.useState<{[key: string]: boolean}>({}); + + const toggleOpen = (groupId: string) => { + setOpenStates(prev => ({ + ...prev, + [groupId]: !prev[groupId] + })); + }; + + const maxAvatars = 5; const {getActivitiesQuery} = useActivitiesForUser({ handle: user, @@ -97,10 +170,16 @@ const Activities: React.FC = ({}) => { includeReplies: true, filter: { type: ['Follow', 'Like', `Create:Note:isReplyToOwn`] - } + }, + key: GET_ACTIVITIES_QUERY_KEY_NOTIFICATIONS }); + const {data, fetchNextPage, hasNextPage, isFetchingNextPage, isLoading} = getActivitiesQuery; - const activities = (data?.pages.flatMap(page => page.data) ?? []); + const groupedActivities = (data?.pages.flatMap((page) => { + const filtered = page.data.filter((activity, index, self) => index === self.findIndex(a => a.id === activity.id)); + + return groupActivities(filtered); + }) ?? []); const observerRef = useRef(null); const loadMoreRef = useRef(null); @@ -127,43 +206,34 @@ const Activities: React.FC = ({}) => { }; }, [hasNextPage, isFetchingNextPage, fetchNextPage]); - // Retrieve followers for the user - // const {data: followers = []} = useFollowersForUser(user); - - // const isFollower = (id: string): boolean => { - // return followers.includes(id); - // }; - - const handleActivityClick = (activity: Activity) => { - switch (activity.type) { - case ACTVITY_TYPE.CREATE: + const handleActivityClick = (group: GroupedActivity, index: number) => { + switch (group.type) { + case ACTIVITY_TYPE.CREATE: NiceModal.show(ArticleModal, { - activityId: activity.id, - object: activity.object, - actor: activity.actor, + activityId: group.id, + object: group.object, + actor: group.actors[0], focusReplies: true, - width: typeof activity.object?.inReplyTo === 'object' && activity.object?.inReplyTo?.type === 'Article' ? 'wide' : 'narrow' + width: typeof group.object?.inReplyTo === 'object' && group.object?.inReplyTo?.type === 'Article' ? 'wide' : 'narrow' }); break; - case ACTVITY_TYPE.LIKE: + case ACTIVITY_TYPE.LIKE: NiceModal.show(ArticleModal, { - activityId: activity.id, - object: activity.object, - actor: activity.object.attributedTo as ActorProperties, - width: 'wide' + activityId: group.id, + object: group.object, + actor: group.object.attributedTo as ActorProperties, + width: group.object?.type === 'Article' ? 'wide' : 'narrow' }); break; - case ACTVITY_TYPE.FOLLOW: - NiceModal.show(ViewProfileModal, { - profile: getUsername(activity.actor), - onFollow: () => {}, - onUnfollow: () => {} - }); + case ACTIVITY_TYPE.FOLLOW: + if (group.actors.length > 1) { + toggleOpen(group.id || `${group.type}_${index}`); + } else { + handleProfileClick(group.actors[0]); + } break; - default: } }; - return ( <> @@ -174,7 +244,7 @@ const Activities: React.FC = ({}) => {
) } { - isLoading === false && activities.length === 0 && ( + isLoading === false && groupedActivities.length === 0 && (
When other Fediverse users interact with you, you'll see it here. @@ -183,26 +253,75 @@ const Activities: React.FC = ({}) => { ) } { - (isLoading === false && activities.length > 0) && ( + (isLoading === false && groupedActivities.length > 0) && ( <> -
- {activities?.map((activity, index) => ( - - handleActivityClick(activity)} +
+ {groupedActivities.map((group, index) => ( + + handleActivityClick(group, index)} > - -
-
- {activity.actor.name} - {getUsername(activity.actor)} + + +
+
+ {!openStates[group.id || `${group.type}_${index}`] && group.actors.slice(0, maxAvatars).map(actor => ( + + ))} + {group.actors.length > maxAvatars && (!openStates[group.id || `${group.type}_${index}`]) && ( +
+ {`+${group.actors.length - maxAvatars}`} +
+ )} + + {group.actors.length > 1 && ( +
+
+ {openStates[group.id || `${group.type}_${index}`] && group.actors.length > 1 && ( +
+ {group.actors.map(actor => ( +
handleProfileClick(actor, e)} + > + + {actor.name} + {getUsername(actor)} +
+ ))} +
+ )} +
-
{getActivityDescription(activity)}
- {getExtendedDescription(activity)} -
- - {index < activities.length - 1 && } + + +
+ {getGroupDescription(group)} +
+ {getExtendedDescription(group)} +
+ + {index < groupedActivities.length - 1 && } ))}
diff --git a/apps/admin-x-activitypub/src/components/Inbox.tsx b/apps/admin-x-activitypub/src/components/Inbox.tsx index 0bddd17968..4fb7fc7537 100644 --- a/apps/admin-x-activitypub/src/components/Inbox.tsx +++ b/apps/admin-x-activitypub/src/components/Inbox.tsx @@ -7,14 +7,19 @@ import NewPostModal from './modals/NewPostModal'; import NiceModal from '@ebay/nice-modal-react'; import React, {useEffect, useRef} from 'react'; import Separator from './global/Separator'; -import ViewProfileModal from './global/ViewProfileModal'; import getName from '../utils/get-name'; import getUsername from '../utils/get-username'; -import useSuggestedProfiles from '../hooks/useSuggestedProfiles'; import {ActorProperties} from '@tryghost/admin-x-framework/api/activitypub'; import {Button, Heading, LoadingIndicator} from '@tryghost/admin-x-design-system'; +import { + GET_ACTIVITIES_QUERY_KEY_FEED, + GET_ACTIVITIES_QUERY_KEY_INBOX, + useActivitiesForUser, + useSuggestedProfiles, + useUserDataForUser +} from '../hooks/useActivityPubQueries'; +import {handleProfileClick} from '../utils/handle-profile-click'; import {handleViewContent} from '../utils/content-handlers'; -import {useActivitiesForUser, useUserDataForUser} from '../hooks/useActivityPubQueries'; import {useRouting} from '@tryghost/admin-x-framework/routing'; type Layout = 'inbox' | 'feed'; @@ -24,6 +29,9 @@ interface InboxProps { } const Inbox: React.FC = ({layout}) => { + const {updateRoute} = useRouting(); + + // Initialise activities for the inbox or feed const typeFilter = layout === 'inbox' ? ['Create:Article'] : ['Create:Note', 'Announce:Note']; @@ -31,23 +39,27 @@ const Inbox: React.FC = ({layout}) => { const {getActivitiesQuery, updateActivity} = useActivitiesForUser({ handle: 'index', includeOwn: true, - excludeNonFollowers: true, filter: { type: typeFilter - } + }, + key: layout === 'inbox' ? GET_ACTIVITIES_QUERY_KEY_INBOX : GET_ACTIVITIES_QUERY_KEY_FEED }); + const {data, fetchNextPage, hasNextPage, isFetchingNextPage, isLoading} = getActivitiesQuery; - const {updateRoute} = useRouting(); + const activities = (data?.pages.flatMap(page => page.data) ?? []) + // If there somehow are duplicate activities, filter them out so the list rendering doesn't break + .filter((activity, index, self) => index === self.findIndex(a => a.id === activity.id)) + // Filter out replies + .filter((activity) => { + return !activity.object.inReplyTo; + }); - const {suggested, isLoadingSuggested} = useSuggestedProfiles(); + // Initialise suggested profiles + const {suggestedProfilesQuery} = useSuggestedProfiles('index', 3); + const {data: suggestedData, isLoading: isLoadingSuggested} = suggestedProfilesQuery; + const suggested = suggestedData || []; - const activities = (data?.pages.flatMap(page => page.data) ?? []).filter((activity) => { - return !activity.object.inReplyTo; - }); - - // Intersection observer to fetch more activities when the user scrolls - // to the bottom of the page const observerRef = useRef(null); const loadMoreRef = useRef(null); @@ -86,7 +98,7 @@ const Inbox: React.FC = ({layout}) => {
) : activities.length > 0 ? ( <> -
+
{layout === 'feed' &&
@@ -124,8 +136,7 @@ const Inbox: React.FC = ({layout}) => {
-
- {/* */} +

This is your {layout === 'inbox' ? 'inbox' : 'feed'}

You'll find {layout === 'inbox' ? 'long-form content' : 'short posts and updates'} from the accounts you follow here.

You might also like

@@ -135,28 +146,17 @@ const Inbox: React.FC = ({layout}) => {
    {suggested.map((profile, index) => { const actor = profile.actor; - // const isFollowing = profile.isFollowing; return (
  • - NiceModal.show(ViewProfileModal, { - profile: getUsername(actor), - onFollow: () => {}, - onUnfollow: () => {} - })}> + handleProfileClick(actor)} + >
    {getName(actor)} {getUsername(actor)}
    - {/* updateSuggestedProfile(actor.id!, {isFollowing: true})} - onUnfollow={() => updateSuggestedProfile(actor.id!, {isFollowing: false})} - /> */}
  • {index < suggested.length - 1 && } diff --git a/apps/admin-x-activitypub/src/components/Profile.tsx b/apps/admin-x-activitypub/src/components/Profile.tsx index 71a8614b7d..ea851516e1 100644 --- a/apps/admin-x-activitypub/src/components/Profile.tsx +++ b/apps/admin-x-activitypub/src/components/Profile.tsx @@ -1,201 +1,199 @@ -import APAvatar from './global/APAvatar'; -import ActivityItem from './activities/ActivityItem'; -import FeedItem from './feed/FeedItem'; -import MainNavigation from './navigation/MainNavigation'; -import NiceModal from '@ebay/nice-modal-react'; import React, {useEffect, useRef, useState} from 'react'; + +import NiceModal from '@ebay/nice-modal-react'; +import {ActorProperties} from '@tryghost/admin-x-framework/api/activitypub'; +import {Button, Heading, List, LoadingIndicator, NoValueLabel, Tab, TabView} from '@tryghost/admin-x-design-system'; + import getName from '../utils/get-name'; import getUsername from '../utils/get-username'; -import {ActorProperties} from '@tryghost/admin-x-framework/api/activitypub'; - -import Separator from './global/Separator'; -import ViewProfileModal from './global/ViewProfileModal'; -import {Button, Heading, List, LoadingIndicator, NoValueLabel, Tab, TabView} from '@tryghost/admin-x-design-system'; -import {handleViewContent} from '../utils/content-handlers'; import { + type ActivityPubCollectionQueryResult, useFollowersCountForUser, useFollowersForUser, useFollowingCountForUser, useFollowingForUser, + useLikedCountForUser, useLikedForUser, useOutboxForUser, useUserDataForUser } from '../hooks/useActivityPubQueries'; +import {handleViewContent} from '../utils/content-handlers'; -interface ProfileProps {} +import APAvatar from './global/APAvatar'; +import ActivityItem from './activities/ActivityItem'; +import FeedItem from './feed/FeedItem'; +import MainNavigation from './navigation/MainNavigation'; +import Separator from './global/Separator'; +import ViewProfileModal from './modals/ViewProfileModal'; +import {type Activity} from '../components/activities/ActivityItem'; -const Profile: React.FC = ({}) => { - const {data: followersCount = 0, isLoading: isLoadingFollowersCount} = useFollowersCountForUser('index'); - const {data: followingCount = 0, isLoading: isLoadingFollowingCount} = useFollowingCountForUser('index'); - const {data: following = [], isLoading: isLoadingFollowing} = useFollowingForUser('index'); - const {data: followers = [], isLoading: isLoadingFollowers} = useFollowersForUser('index'); - const {data: liked = [], isLoading: isLoadingLiked} = useLikedForUser('index'); - const {data: outboxPosts = [], isLoading: isLoadingOutbox} = useOutboxForUser('index'); - const {data: userProfile, isLoading: isLoadingProfile} = useUserDataForUser('index') as {data: ActorProperties | null, isLoading: boolean}; +interface UseInfiniteScrollTabProps { + useDataHook: (key: string) => ActivityPubCollectionQueryResult; + emptyStateLabel: string; + emptyStateIcon: string; +} - const isInitialLoading = isLoadingProfile || isLoadingOutbox; +/** + * Hook to abstract away the common logic for infinite scroll in tabs + */ +const useInfiniteScrollTab = ({useDataHook, emptyStateLabel, emptyStateIcon}: UseInfiniteScrollTabProps) => { + const { + data, + fetchNextPage, + hasNextPage, + isFetchingNextPage, + isLoading + } = useDataHook('index'); - const posts = outboxPosts.filter(post => post.type === 'Create' && !post.object.inReplyTo); + const items = (data?.pages.flatMap(page => page.data) ?? []); - type ProfileTab = 'posts' | 'likes' | 'following' | 'followers'; + const observerRef = useRef(null); + const loadMoreRef = useRef(null); - const [selectedTab, setSelectedTab] = useState('posts'); + useEffect(() => { + if (observerRef.current) { + observerRef.current.disconnect(); + } - const layout = 'feed'; - - const INCREMENT_VISIBLE_POSTS = 40; - const INCREMENT_VISIBLE_LIKES = 40; - const INCREMENT_VISIBLE_FOLLOWING = 40; - const INCREMENT_VISIBLE_FOLLOWERS = 40; - - const [visiblePosts, setVisiblePosts] = useState(INCREMENT_VISIBLE_POSTS); - const [visibleLikes, setVisibleLikes] = useState(INCREMENT_VISIBLE_LIKES); - const [visibleFollowing, setVisibleFollowing] = useState(INCREMENT_VISIBLE_FOLLOWING); - const [visibleFollowers, setVisibleFollowers] = useState(INCREMENT_VISIBLE_FOLLOWERS); - - const loadMorePosts = () => { - setVisiblePosts(prev => prev + INCREMENT_VISIBLE_POSTS); - }; - - const loadMoreLikes = () => { - setVisibleLikes(prev => prev + INCREMENT_VISIBLE_LIKES); - }; - - const loadMoreFollowing = () => { - setVisibleFollowing(prev => prev + INCREMENT_VISIBLE_FOLLOWING); - }; - - const loadMoreFollowers = () => { - setVisibleFollowers(prev => prev + INCREMENT_VISIBLE_FOLLOWERS); - }; - - const handleUserClick = (actor: ActorProperties) => { - NiceModal.show(ViewProfileModal, { - profile: getUsername(actor), - onFollow: () => {}, - onUnfollow: () => {} + observerRef.current = new IntersectionObserver((entries) => { + if (entries[0].isIntersecting && hasNextPage && !isFetchingNextPage) { + fetchNextPage(); + } }); - }; - const renderPostsTab = () => { - if (posts.length === 0) { - return ( - - You haven't posted anything yet. - - ); + if (loadMoreRef.current) { + observerRef.current.observe(loadMoreRef.current); } - return ( - <> -
      - {posts.slice(0, visiblePosts).map((activity, index) => ( -
    • - handleViewContent(activity, false)} - onCommentClick={() => handleViewContent(activity, true)} - /> - {index < posts.length - 1 && } -
    • - ))} -
    - {visiblePosts < posts.length && ( -
- )} - - containerClassName='mt-6' - selectedTab={selectedTab} - tabs={tabs} - onTabChange={setSelectedTab} - /> -
-
-
- ); - }; - return ( <> - {renderMainContent()} + {isInitialLoading ? ( +
+ +
+ ) : ( +
+
+ {userProfile?.image && ( +
+ {userProfile?.name} +
+ )} +
+
+
+ +
+
+ {userProfile?.name} + + {userProfile && getUsername(userProfile)} + + {(userProfile?.summary || attachments.length > 0) && ( +
p]:mb-3 ${isExpanded ? 'max-h-none pb-7' : 'max-h-[160px] overflow-hidden'} relative`}> +
+ {attachments.map(attachment => ( + + {attachment.name} + + + ))} + {!isExpanded && isOverflowing && ( +
+ )} + {isOverflowing &&
+ )} + + containerClassName='mt-6' + selectedTab={selectedTab} + tabs={tabs} + onTabChange={setSelectedTab} + /> +
+
+
+ )} ); }; diff --git a/apps/admin-x-activitypub/src/components/Search.tsx b/apps/admin-x-activitypub/src/components/Search.tsx index 6cc9b7013a..39451a20ea 100644 --- a/apps/admin-x-activitypub/src/components/Search.tsx +++ b/apps/admin-x-activitypub/src/components/Search.tsx @@ -1,6 +1,6 @@ import React, {useEffect, useRef, useState} from 'react'; -import {Activity, ActorProperties} from '@tryghost/admin-x-framework/api/activitypub'; +import {ActorProperties} from '@tryghost/admin-x-framework/api/activitypub'; import {Button, Icon, LoadingIndicator, NoValueLabel, TextField} from '@tryghost/admin-x-design-system'; import {useDebounce} from 'use-debounce'; @@ -10,11 +10,11 @@ import FollowButton from './global/FollowButton'; import MainNavigation from './navigation/MainNavigation'; import NiceModal from '@ebay/nice-modal-react'; -import ViewProfileModal from './global/ViewProfileModal'; +import ViewProfileModal from './modals/ViewProfileModal'; import Separator from './global/Separator'; -import useSuggestedProfiles from '../hooks/useSuggestedProfiles'; -import {useSearchForUser} from '../hooks/useActivityPubQueries'; + +import {useSearchForUser, useSuggestedProfiles} from '../hooks/useActivityPubQueries'; interface SearchResultItem { actor: ActorProperties; @@ -22,7 +22,6 @@ interface SearchResultItem { followerCount: number; followingCount: number; isFollowing: boolean; - posts: Activity[]; } interface SearchResultProps { @@ -30,8 +29,6 @@ interface SearchResultProps { update: (id: string, updated: Partial) => void; } -interface SearchProps {} - const SearchResult: React.FC = ({result, update}) => { const onFollow = () => { update(result.actor.id!, { @@ -121,9 +118,13 @@ const SuggestedAccounts: React.FC<{ ); }; +interface SearchProps {} + const Search: React.FC = ({}) => { // Initialise suggested profiles - const {suggested, isLoadingSuggested, updateSuggestedProfile} = useSuggestedProfiles(6); + const {suggestedProfilesQuery, updateSuggestedProfile} = useSuggestedProfiles('index', 6); + const {data: suggestedData, isLoading: isLoadingSuggested} = suggestedProfilesQuery; + const suggested = suggestedData || []; // Initialise search query const queryInputRef = useRef(null); @@ -151,6 +152,7 @@ const Search: React.FC = ({}) => {
= ({notificationType, className}) => { + let icon = ''; + let iconColor = ''; + let badgeColor = ''; + + switch (notificationType) { + case 'follow': + icon = 'user'; + iconColor = 'text-blue-500'; + badgeColor = 'bg-blue-100/50'; + break; + case 'like': + icon = 'heart'; + iconColor = 'text-red-500'; + badgeColor = 'bg-red-100/50'; + break; + case 'reply': + icon = 'comment'; + iconColor = 'text-purple-500'; + badgeColor = 'bg-purple-100/50'; + break; + } + + return ( +
+ +
+ ); +}; + +export default NotificationIcon; diff --git a/apps/admin-x-activitypub/src/components/activities/NotificationItem.tsx b/apps/admin-x-activitypub/src/components/activities/NotificationItem.tsx new file mode 100644 index 0000000000..1e293b6ec1 --- /dev/null +++ b/apps/admin-x-activitypub/src/components/activities/NotificationItem.tsx @@ -0,0 +1,63 @@ +import NotificationIcon, {NotificationType} from './NotificationIcon'; +import React from 'react'; + +// Context to share common props between compound components +interface NotificationContextType { + onClick?: () => void; + url?: string; +} + +const NotificationContext = React.createContext(undefined); + +// Root component +interface NotificationItemProps { + children: React.ReactNode; + onClick?: () => void; + url?: string; + className?: string; +} + +const NotificationItem = ({children, onClick, url, className}: NotificationItemProps) => { + return ( + +
+ {children} +
+
+ ); +}; + +// Sub-components +const Icon = ({type}: {type: NotificationType}) => { + return ( +
+ +
+ ); +}; + +const Avatars = ({children}: {children: React.ReactNode}) => { + return ( +
+ {children} +
+ ); +}; + +const Content = ({children}: {children: React.ReactNode}) => { + return ( +
+ {children} +
+ ); +}; + +// Attach sub-components to the main component +NotificationItem.Icon = Icon; +NotificationItem.Avatars = Avatars; +NotificationItem.Content = Content; + +export default NotificationItem; diff --git a/apps/admin-x-activitypub/src/components/articleBodyStyles.ts b/apps/admin-x-activitypub/src/components/articleBodyStyles.ts index 81f24f5181..1081426e7a 100644 --- a/apps/admin-x-activitypub/src/components/articleBodyStyles.ts +++ b/apps/admin-x-activitypub/src/components/articleBodyStyles.ts @@ -1,53 +1,5 @@ const articleBodyStyles = (siteUrl: string|undefined) => { return ` - - - -
-

${heading}

- ${excerpt ? ` -

${excerpt}

- ` : ''} - ${image ? ` -
- ${heading} -
- ` : ''} -
-
- ${html} -
- - -`; + waitForImages().then(() => { + isFullyLoaded = true; + resizeIframe(); + }); + } + + window.addEventListener('DOMContentLoaded', initializeResize); + window.addEventListener('load', resizeIframe); + window.addEventListener('resize', resizeIframe); + new MutationObserver(resizeIframe).observe(document.body, { subtree: true, childList: true }); + + window.addEventListener('message', (event) => { + if (event.data.type === 'triggerResize') { + resizeIframe(); + } + }); + + + +
+

${heading}

+ ${excerpt ? ` +

${excerpt}

+ ` : ''} + ${image ? ` +
+ ${heading} +
+ ` : ''} +
+
+ ${html} +
+ + + `; useEffect(() => { const iframe = iframeRef.current; - if (iframe) { - iframe.srcdoc = htmlContent; - - const handleMessage = (event: MessageEvent) => { - if (event.data.type === 'resize') { - setIframeHeight(`${event.data.height}px`); - iframe.style.height = `${event.data.height}px`; - } - }; - - window.addEventListener('message', handleMessage); - - return () => { - window.removeEventListener('message', handleMessage); - }; + if (!iframe) { + return; } + + if (!iframe.srcdoc) { + iframe.srcdoc = htmlContent; + } + + const handleMessage = (event: MessageEvent) => { + if (event.data.type === 'resize') { + const newHeight = `${event.data.bodyHeight + 24}px`; + setIframeHeight(newHeight); + iframe.style.height = newHeight; + + if (event.data.isLoaded) { + setIsLoading(false); + } + } + }; + + window.addEventListener('message', handleMessage); + + return () => { + window.removeEventListener('message', handleMessage); + }; }, [htmlContent]); + // Separate effect for style updates + useEffect(() => { + const iframe = iframeRef.current; + if (!iframe) { + return; + } + + const iframeDocument = iframe.contentDocument || iframe.contentWindow?.document; + if (!iframeDocument) { + return; + } + + const root = iframeDocument.documentElement; + root.style.setProperty('--font-size', fontSize); + root.style.setProperty('--line-height', lineHeight); + root.style.setProperty('--font-family', fontFamily.value); + root.style.setProperty('--letter-spacing', fontFamily.label === 'Clean sans-serif' ? '-0.013em' : '0'); + root.style.setProperty('--content-spacing-factor', SPACING_FACTORS[FONT_SIZES.indexOf(fontSize)]); + + const iframeWindow = iframe.contentWindow as IframeWindow; + if (iframeWindow && typeof iframeWindow.resizeIframe === 'function') { + iframeWindow.resizeIframe(); + } else { + // Fallback: trigger a resize event + const resizeEvent = new Event('resize'); + iframeDocument.dispatchEvent(resizeEvent); + } + }, [fontSize, lineHeight, fontFamily]); + return ( -
-