pleroma-fe/src/modules/api.js

361 lines
13 KiB
JavaScript
Raw Normal View History

import backendInteractorService from '../services/backend_interactor_service/backend_interactor_service.js'
2020-05-07 16:10:53 +03:00
import { WSConnectionStatus } from '../services/api/api.service.js'
2020-07-13 00:06:45 +03:00
import { maybeShowChatNotification } from '../services/chat_utils/chat_utils.js'
2017-12-05 11:47:10 +01:00
import { Socket } from 'phoenix'
2025-02-03 13:02:14 +02:00
import { useShoutStore } from 'src/stores/shout.js'
import { useInterfaceStore } from 'src/stores/interface.js'
const retryTimeout = (multiplier) => 1000 * multiplier
const api = {
state: {
retryMultiplier: 1,
backendInteractor: backendInteractorService(),
2017-12-05 11:47:10 +01:00
fetchers: {},
2017-12-07 18:20:44 +02:00
socket: null,
mastoUserSocket: null,
2020-05-07 16:10:53 +03:00
mastoUserSocketStatus: null,
2026-01-06 16:22:52 +02:00
followRequests: [],
},
getters: {
2026-01-06 16:22:52 +02:00
followRequestCount: (state) => state.followRequests.length,
},
mutations: {
2026-01-06 16:22:52 +02:00
setBackendInteractor(state, backendInteractor) {
state.backendInteractor = backendInteractor
},
2026-01-06 16:22:52 +02:00
addFetcher(state, { fetcherName, fetcher }) {
2019-04-04 09:03:56 -07:00
state.fetchers[fetcherName] = fetcher
},
2026-01-06 16:22:52 +02:00
removeFetcher(state, { fetcherName }) {
2020-09-04 11:19:53 +03:00
state.fetchers[fetcherName].stop()
2019-04-04 09:03:56 -07:00
delete state.fetchers[fetcherName]
2017-12-05 11:47:10 +01:00
},
2026-01-06 16:22:52 +02:00
setWsToken(state, token) {
state.wsToken = token
},
2026-01-06 16:22:52 +02:00
setSocket(state, socket) {
2017-12-05 11:47:10 +01:00
state.socket = socket
2017-12-07 18:20:44 +02:00
},
2026-01-06 16:22:52 +02:00
setFollowRequests(state, value) {
state.followRequests = value
2020-05-07 16:10:53 +03:00
},
2026-01-06 16:22:52 +02:00
setMastoUserSocketStatus(state, value) {
2020-05-07 16:10:53 +03:00
state.mastoUserSocketStatus = value
},
2026-01-06 16:22:52 +02:00
incrementRetryMultiplier(state) {
state.retryMultiplier = Math.max(++state.retryMultiplier, 3)
},
2026-01-06 16:22:52 +02:00
resetRetryMultiplier(state) {
state.retryMultiplier = 1
2026-01-06 16:22:52 +02:00
},
},
actions: {
/**
* Global MastoAPI socket control, in future should disable ALL sockets/(re)start relevant sockets
*
* @param {Boolean} [initial] - whether this enabling happened at boot time or not
*/
2026-01-06 16:22:52 +02:00
enableMastoSockets(store, initial) {
const { state, dispatch, commit } = store
// Do not initialize unless nonexistent or closed
if (
state.mastoUserSocket &&
2026-01-06 16:22:52 +02:00
![WebSocket.CLOSED, WebSocket.CLOSING].includes(
state.mastoUserSocket.getState(),
)
) {
return
}
if (initial) {
commit('setMastoUserSocketStatus', WSConnectionStatus.STARTING_INITIAL)
} else {
commit('setMastoUserSocketStatus', WSConnectionStatus.STARTING)
}
2019-12-26 14:12:35 +02:00
return dispatch('startMastoUserSocket')
2019-12-10 21:30:27 +02:00
},
2026-01-06 16:22:52 +02:00
disableMastoSockets(store) {
const { state, dispatch, commit } = store
2019-12-10 21:30:27 +02:00
if (!state.mastoUserSocket) return
commit('setMastoUserSocketStatus', WSConnectionStatus.DISABLED)
2019-12-26 14:12:35 +02:00
return dispatch('stopMastoUserSocket')
2019-12-10 21:30:27 +02:00
},
// MastoAPI 'User' sockets
2026-01-06 16:22:52 +02:00
startMastoUserSocket(store) {
2019-12-26 14:12:35 +02:00
return new Promise((resolve, reject) => {
try {
2020-05-07 16:10:53 +03:00
const { state, commit, dispatch, rootState } = store
2019-12-26 14:35:46 +02:00
const timelineData = rootState.statuses.timelines.friends
2026-01-06 16:22:52 +02:00
state.mastoUserSocket = state.backendInteractor.startUserSocket({
store,
})
2026-01-06 16:22:52 +02:00
state.mastoUserSocket.addEventListener(
'pleroma:authenticated',
() => {
state.mastoUserSocket.subscribe('user')
},
)
2019-12-26 14:12:35 +02:00
state.mastoUserSocket.addEventListener(
'message',
({ detail: message }) => {
if (!message) return // pings
if (message.event === 'notification') {
dispatch('addNewNotifications', {
notifications: [message.notification],
2026-01-06 16:22:52 +02:00
older: false,
2019-12-26 14:12:35 +02:00
})
} else if (message.event === 'update') {
dispatch('addNewStatuses', {
statuses: [message.status],
userId: false,
2019-12-26 14:35:46 +02:00
showImmediately: timelineData.visibleStatuses.length === 0,
2026-01-06 16:22:52 +02:00
timeline: 'friends',
2019-12-26 14:12:35 +02:00
})
2022-06-07 21:31:48 -06:00
} else if (message.event === 'status.update') {
2022-06-11 12:20:11 -04:00
dispatch('addNewStatuses', {
statuses: [message.status],
userId: false,
2026-01-06 16:22:52 +02:00
showImmediately:
message.status.id in timelineData.visibleStatusesObject,
timeline: 'friends',
2022-06-11 12:20:11 -04:00
})
} else if (message.event === 'delete') {
dispatch('deleteStatusById', message.id)
2020-05-07 16:10:53 +03:00
} else if (message.event === 'pleroma:chat_update') {
2020-10-29 13:33:06 +03:00
// The setTimeout wrapper is a temporary band-aid to avoid duplicates for the user's own messages when doing optimistic sending.
// The cause of the duplicates is the WS event arriving earlier than the HTTP response.
// This setTimeout wrapper can be removed once the commit `8e41baff` is in the stable Pleroma release.
// (`8e41baff` adds the idempotency key to the chat message entity, which PleromaFE uses when it's available, and it makes this artificial delay unnecessary).
setTimeout(() => {
dispatch('addChatMessages', {
chatId: message.chatUpdate.id,
2026-01-06 16:22:52 +02:00
messages: [message.chatUpdate.lastMessage],
2020-10-29 13:33:06 +03:00
})
dispatch('updateChat', { chat: message.chatUpdate })
maybeShowChatNotification(store, message.chatUpdate)
}, 100)
2019-12-26 14:12:35 +02:00
}
2026-01-06 16:22:52 +02:00
},
2019-12-26 14:12:35 +02:00
)
2020-05-07 16:10:53 +03:00
state.mastoUserSocket.addEventListener('open', () => {
2021-01-13 21:31:57 +02:00
// Do not show notification when we just opened up the page
2026-01-06 16:22:52 +02:00
if (
state.mastoUserSocketStatus !==
WSConnectionStatus.STARTING_INITIAL
) {
2023-04-05 21:06:37 -06:00
useInterfaceStore().pushGlobalNotice({
2021-01-13 21:31:57 +02:00
level: 'success',
messageKey: 'timeline.socket_reconnected',
2026-01-06 16:22:52 +02:00
timeout: 5000,
2021-01-13 21:31:57 +02:00
})
}
// Stop polling if we were errored or disabled
2026-01-06 16:22:52 +02:00
if (
new Set([
WSConnectionStatus.ERROR,
WSConnectionStatus.DISABLED,
]).has(state.mastoUserSocketStatus)
) {
dispatch('stopFetchingTimeline', { timeline: 'friends' })
dispatch('stopFetchingNotifications')
dispatch('stopFetchingChats')
}
commit('resetRetryMultiplier')
2020-05-07 16:10:53 +03:00
commit('setMastoUserSocketStatus', WSConnectionStatus.JOINED)
})
2026-01-06 16:22:52 +02:00
state.mastoUserSocket.addEventListener(
'error',
({ detail: error }) => {
console.error('Error in MastoAPI websocket:', error)
// TODO is this needed?
dispatch('clearOpenedChats')
},
)
state.mastoUserSocket.addEventListener(
'close',
({ detail: closeEvent }) => {
const ignoreCodes = new Set([
1000, // Normal (intended) closure
1001, // Going away
])
const { code } = closeEvent
if (ignoreCodes.has(code)) {
console.debug(
`Not restarting socket becasue of closure code ${code} is in ignore list`,
)
commit('setMastoUserSocketStatus', WSConnectionStatus.CLOSED)
} else {
console.warn(
`MastoAPI websocket disconnected, restarting. CloseEvent code: ${code}`,
)
setTimeout(() => {
dispatch('startMastoUserSocket')
}, retryTimeout(state.retryMultiplier))
commit('incrementRetryMultiplier')
if (state.mastoUserSocketStatus !== WSConnectionStatus.ERROR) {
dispatch('startFetchingTimeline', { timeline: 'friends' })
dispatch('startFetchingNotifications')
dispatch('startFetchingChats')
useInterfaceStore().pushGlobalNotice({
level: 'error',
messageKey: 'timeline.socket_broke',
messageArgs: [code],
timeout: 5000,
})
}
commit('setMastoUserSocketStatus', WSConnectionStatus.ERROR)
}
2026-01-06 16:22:52 +02:00
dispatch('clearOpenedChats')
},
)
2019-12-26 14:12:35 +02:00
resolve()
} catch (e) {
reject(e)
}
})
2019-11-24 18:50:28 +02:00
},
2026-01-06 16:22:52 +02:00
stopMastoUserSocket({ state, dispatch }) {
2019-12-10 21:30:27 +02:00
dispatch('startFetchingTimeline', { timeline: 'friends' })
dispatch('startFetchingNotifications')
2020-05-07 16:10:53 +03:00
dispatch('startFetchingChats')
2019-12-10 21:30:27 +02:00
state.mastoUserSocket.close()
},
// Timelines
2026-01-06 16:22:52 +02:00
startFetchingTimeline(
store,
{
timeline = 'friends',
tag = false,
userId = false,
listId = false,
statusId = false,
bookmarkFolderId = false,
},
) {
if (
timeline === 'favourites' &&
!store.rootState.instance.pleromaPublicFavouritesAvailable
)
return
2019-02-07 16:23:18 -07:00
if (store.state.fetchers[timeline]) return
const fetcher = store.state.backendInteractor.startFetchingTimeline({
2026-01-06 16:22:52 +02:00
timeline,
store,
userId,
listId,
statusId,
bookmarkFolderId,
tag,
})
2019-04-04 09:03:56 -07:00
store.commit('addFetcher', { fetcherName: timeline, fetcher })
},
2026-01-06 16:22:52 +02:00
stopFetchingTimeline(store, timeline) {
const fetcher = store.state.fetchers[timeline]
if (!fetcher) return
store.commit('removeFetcher', { fetcherName: timeline, fetcher })
},
2026-01-06 16:22:52 +02:00
fetchTimeline(store, { timeline, ...rest }) {
store.state.backendInteractor.fetchTimeline({
store,
timeline,
2026-01-06 16:22:52 +02:00
...rest,
})
},
2019-04-04 09:03:56 -07:00
// Notifications
2026-01-06 16:22:52 +02:00
startFetchingNotifications(store) {
if (store.state.fetchers.notifications) return
2026-01-06 16:22:52 +02:00
const fetcher = store.state.backendInteractor.startFetchingNotifications({
store,
})
2019-04-04 09:03:56 -07:00
store.commit('addFetcher', { fetcherName: 'notifications', fetcher })
},
2026-01-06 16:22:52 +02:00
stopFetchingNotifications(store) {
const fetcher = store.state.fetchers.notifications
if (!fetcher) return
store.commit('removeFetcher', { fetcherName: 'notifications', fetcher })
},
2026-01-06 16:22:52 +02:00
fetchNotifications(store, { ...rest }) {
store.state.backendInteractor.fetchNotifications({
store,
2026-01-06 16:22:52 +02:00
...rest,
})
},
// Follow requests
2026-01-06 16:22:52 +02:00
startFetchingFollowRequests(store) {
2022-07-31 12:35:48 +03:00
if (store.state.fetchers.followRequests) return
2026-01-06 16:22:52 +02:00
const fetcher = store.state.backendInteractor.startFetchingFollowRequests(
{ store },
)
2020-01-21 16:51:49 +01:00
store.commit('addFetcher', { fetcherName: 'followRequests', fetcher })
},
2026-01-06 16:22:52 +02:00
stopFetchingFollowRequests(store) {
const fetcher = store.state.fetchers.followRequests
if (!fetcher) return
store.commit('removeFetcher', { fetcherName: 'followRequests', fetcher })
},
2026-01-06 16:22:52 +02:00
removeFollowRequest(store, request) {
2022-07-31 12:35:48 +03:00
const requests = store.state.followRequests.filter((it) => it !== request)
store.commit('setFollowRequests', requests)
2017-12-05 11:47:10 +01:00
},
2022-08-06 17:26:43 +03:00
// Lists
2026-01-06 16:22:52 +02:00
startFetchingLists(store) {
2022-08-06 17:26:43 +03:00
if (store.state.fetchers.lists) return
2026-01-06 16:22:52 +02:00
const fetcher = store.state.backendInteractor.startFetchingLists({
store,
})
2022-08-06 17:26:43 +03:00
store.commit('addFetcher', { fetcherName: 'lists', fetcher })
},
2026-01-06 16:22:52 +02:00
stopFetchingLists(store) {
2022-08-06 17:26:43 +03:00
const fetcher = store.state.fetchers.lists
if (!fetcher) return
store.commit('removeFetcher', { fetcherName: 'lists', fetcher })
},
// Bookmark folders
2026-01-06 16:22:52 +02:00
startFetchingBookmarkFolders(store) {
if (store.state.fetchers.bookmarkFolders) return
if (!store.rootState.instance.pleromaBookmarkFoldersAvailable) return
2026-01-06 16:22:52 +02:00
const fetcher =
store.state.backendInteractor.startFetchingBookmarkFolders({ store })
store.commit('addFetcher', { fetcherName: 'bookmarkFolders', fetcher })
},
2026-01-06 16:22:52 +02:00
stopFetchingBookmarkFolders(store) {
const fetcher = store.state.fetchers.bookmarkFolders
if (!fetcher) return
store.commit('removeFetcher', { fetcherName: 'bookmarkFolders', fetcher })
},
// Pleroma websocket
2026-01-06 16:22:52 +02:00
setWsToken(store, token) {
store.commit('setWsToken', token)
},
2026-01-06 16:22:52 +02:00
initializeSocket({ commit, state, rootState }) {
2017-12-05 11:47:10 +01:00
// Set up websocket connection
const token = state.wsToken
2026-01-06 16:22:52 +02:00
if (
rootState.instance.shoutAvailable &&
typeof token !== 'undefined' &&
state.socket === null
) {
2019-06-18 20:28:31 +00:00
const socket = new Socket('/socket', { params: { token } })
2017-12-07 18:20:44 +02:00
socket.connect()
commit('setSocket', socket)
2023-04-04 21:17:54 -06:00
useShoutStore().initializeShout(socket)
2017-12-07 18:20:44 +02:00
}
},
2026-01-06 16:22:52 +02:00
disconnectFromSocket({ commit, state }) {
state.socket && state.socket.disconnect()
commit('setSocket', null)
2026-01-06 16:22:52 +02:00
},
},
}
export default api