2016-11-26 17:57:08 +00:00
|
|
|
import backendInteractorService from '../services/backend_interactor_service/backend_interactor_service.js'
|
2020-05-07 13:10:53 +00:00
|
|
|
import { WSConnectionStatus } from '../services/api/api.service.js'
|
2016-11-26 17:57:08 +00:00
|
|
|
|
2021-03-09 00:38:10 +00:00
|
|
|
const retryTimeout = (multiplier) => 1000 * multiplier
|
|
|
|
|
2016-11-26 17:57:08 +00:00
|
|
|
const api = {
|
|
|
|
state: {
|
2021-03-08 20:24:39 +00:00
|
|
|
retryMultiplier: 1,
|
2017-02-16 10:17:47 +00:00
|
|
|
backendInteractor: backendInteractorService(),
|
2017-12-05 10:47:10 +00:00
|
|
|
fetchers: {},
|
2017-12-07 16:20:44 +00:00
|
|
|
socket: null,
|
2019-12-08 14:05:41 +00:00
|
|
|
mastoUserSocket: null,
|
2020-05-07 13:10:53 +00:00
|
|
|
mastoUserSocketStatus: null,
|
2022-06-18 13:04:18 +00:00
|
|
|
followRequests: []
|
2016-11-26 17:57:08 +00:00
|
|
|
},
|
|
|
|
mutations: {
|
|
|
|
setBackendInteractor (state, backendInteractor) {
|
|
|
|
state.backendInteractor = backendInteractor
|
2017-02-16 10:17:47 +00:00
|
|
|
},
|
2019-04-04 16:06:53 +00:00
|
|
|
addFetcher (state, { fetcherName, fetcher }) {
|
2019-04-04 16:03:56 +00:00
|
|
|
state.fetchers[fetcherName] = fetcher
|
2017-02-16 10:17:47 +00:00
|
|
|
},
|
2019-12-08 14:05:41 +00:00
|
|
|
removeFetcher (state, { fetcherName, fetcher }) {
|
2020-09-04 08:19:53 +00:00
|
|
|
state.fetchers[fetcherName].stop()
|
2019-04-04 16:03:56 +00:00
|
|
|
delete state.fetchers[fetcherName]
|
2017-12-05 10:47:10 +00:00
|
|
|
},
|
2019-01-29 15:16:25 +00:00
|
|
|
setWsToken (state, token) {
|
|
|
|
state.wsToken = token
|
|
|
|
},
|
2017-12-05 10:47:10 +00:00
|
|
|
setSocket (state, socket) {
|
|
|
|
state.socket = socket
|
2017-12-07 16:20:44 +00:00
|
|
|
},
|
2018-06-07 01:24:31 +00:00
|
|
|
setFollowRequests (state, value) {
|
|
|
|
state.followRequests = value
|
2020-05-07 13:10:53 +00:00
|
|
|
},
|
|
|
|
setMastoUserSocketStatus (state, value) {
|
|
|
|
state.mastoUserSocketStatus = value
|
2021-03-08 20:24:39 +00:00
|
|
|
},
|
2021-03-09 00:38:10 +00:00
|
|
|
incrementRetryMultiplier (state) {
|
|
|
|
state.retryMultiplier = Math.max(++state.retryMultiplier, 3)
|
2021-03-08 20:24:39 +00:00
|
|
|
},
|
2021-03-09 00:38:10 +00:00
|
|
|
resetRetryMultiplier (state) {
|
|
|
|
state.retryMultiplier = 1
|
2017-02-16 10:17:47 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
actions: {
|
2021-03-09 00:38:10 +00:00
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
enableMastoSockets (store, initial) {
|
2021-03-08 20:24:39 +00:00
|
|
|
const { state, dispatch, commit } = store
|
2021-01-13 19:33:20 +00:00
|
|
|
// Do not initialize unless nonexistent or closed
|
|
|
|
if (
|
|
|
|
state.mastoUserSocket &&
|
|
|
|
![
|
|
|
|
WebSocket.CLOSED,
|
|
|
|
WebSocket.CLOSING
|
|
|
|
].includes(state.mastoUserSocket.getState())
|
|
|
|
) {
|
|
|
|
return
|
|
|
|
}
|
2021-03-09 00:38:10 +00:00
|
|
|
if (initial) {
|
|
|
|
commit('setMastoUserSocketStatus', WSConnectionStatus.STARTING_INITIAL)
|
|
|
|
} else {
|
|
|
|
commit('setMastoUserSocketStatus', WSConnectionStatus.STARTING)
|
|
|
|
}
|
2019-12-26 12:12:35 +00:00
|
|
|
return dispatch('startMastoUserSocket')
|
2019-12-10 19:30:27 +00:00
|
|
|
},
|
|
|
|
disableMastoSockets (store) {
|
2021-03-08 20:24:39 +00:00
|
|
|
const { state, dispatch, commit } = store
|
2019-12-10 19:30:27 +00:00
|
|
|
if (!state.mastoUserSocket) return
|
2021-03-09 00:38:10 +00:00
|
|
|
commit('setMastoUserSocketStatus', WSConnectionStatus.DISABLED)
|
2019-12-26 12:12:35 +00:00
|
|
|
return dispatch('stopMastoUserSocket')
|
2019-12-10 19:30:27 +00:00
|
|
|
},
|
|
|
|
|
2019-12-08 14:05:41 +00:00
|
|
|
// MastoAPI 'User' sockets
|
|
|
|
startMastoUserSocket (store) {
|
2019-12-26 12:12:35 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
try {
|
2020-05-07 13:10:53 +00:00
|
|
|
const { state, commit, dispatch, rootState } = store
|
2019-12-26 12:35:46 +00:00
|
|
|
const timelineData = rootState.statuses.timelines.friends
|
2019-12-26 12:12:35 +00:00
|
|
|
state.mastoUserSocket = state.backendInteractor.startUserSocket({ store })
|
|
|
|
state.mastoUserSocket.addEventListener(
|
|
|
|
'message',
|
|
|
|
({ detail: message }) => {
|
|
|
|
if (!message) return // pings
|
|
|
|
if (message.event === 'notification') {
|
|
|
|
dispatch('addNewNotifications', {
|
|
|
|
notifications: [message.notification],
|
|
|
|
older: false
|
|
|
|
})
|
|
|
|
} else if (message.event === 'update') {
|
|
|
|
dispatch('addNewStatuses', {
|
|
|
|
statuses: [message.status],
|
|
|
|
userId: false,
|
2019-12-26 12:35:46 +00:00
|
|
|
showImmediately: timelineData.visibleStatuses.length === 0,
|
2019-12-26 12:12:35 +00:00
|
|
|
timeline: 'friends'
|
|
|
|
})
|
2022-09-06 19:25:03 +00:00
|
|
|
} else if (message.event === 'status.update') {
|
|
|
|
dispatch('addNewStatuses', {
|
|
|
|
statuses: [message.status],
|
|
|
|
userId: false,
|
|
|
|
showImmediately: message.status.id in timelineData.visibleStatusesObject,
|
|
|
|
timeline: 'friends'
|
|
|
|
})
|
2020-09-06 12:28:09 +00:00
|
|
|
} else if (message.event === 'delete') {
|
|
|
|
dispatch('deleteStatusById', message.id)
|
2019-12-26 12:12:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
2020-05-07 13:10:53 +00:00
|
|
|
state.mastoUserSocket.addEventListener('open', () => {
|
2021-01-13 19:31:57 +00:00
|
|
|
// Do not show notification when we just opened up the page
|
2021-03-09 00:38:10 +00:00
|
|
|
if (state.mastoUserSocketStatus !== WSConnectionStatus.STARTING_INITIAL) {
|
2021-01-13 19:31:57 +00:00
|
|
|
dispatch('pushGlobalNotice', {
|
|
|
|
level: 'success',
|
|
|
|
messageKey: 'timeline.socket_reconnected',
|
|
|
|
timeout: 5000
|
|
|
|
})
|
|
|
|
}
|
2021-03-09 00:38:10 +00:00
|
|
|
// Stop polling if we were errored or disabled
|
|
|
|
if (new Set([
|
|
|
|
WSConnectionStatus.ERROR,
|
|
|
|
WSConnectionStatus.DISABLED
|
|
|
|
]).has(state.mastoUserSocketStatus)) {
|
|
|
|
dispatch('stopFetchingTimeline', { timeline: 'friends' })
|
|
|
|
dispatch('stopFetchingNotifications')
|
|
|
|
}
|
|
|
|
commit('resetRetryMultiplier')
|
2020-05-07 13:10:53 +00:00
|
|
|
commit('setMastoUserSocketStatus', WSConnectionStatus.JOINED)
|
|
|
|
})
|
2019-12-26 12:12:35 +00:00
|
|
|
state.mastoUserSocket.addEventListener('error', ({ detail: error }) => {
|
|
|
|
console.error('Error in MastoAPI websocket:', error)
|
|
|
|
})
|
|
|
|
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`)
|
2021-03-09 00:38:10 +00:00
|
|
|
commit('setMastoUserSocketStatus', WSConnectionStatus.CLOSED)
|
2019-12-26 12:12:35 +00:00
|
|
|
} else {
|
|
|
|
console.warn(`MastoAPI websocket disconnected, restarting. CloseEvent code: ${code}`)
|
2021-03-08 20:24:39 +00:00
|
|
|
setTimeout(() => {
|
2021-03-09 00:38:10 +00:00
|
|
|
dispatch('startMastoUserSocket')
|
|
|
|
}, retryTimeout(state.retryMultiplier))
|
|
|
|
commit('incrementRetryMultiplier')
|
|
|
|
if (state.mastoUserSocketStatus !== WSConnectionStatus.ERROR) {
|
|
|
|
dispatch('startFetchingTimeline', { timeline: 'friends' })
|
|
|
|
dispatch('startFetchingNotifications')
|
2022-07-18 13:08:50 +00:00
|
|
|
dispatch('startFetchingAnnouncements')
|
2021-03-08 20:24:39 +00:00
|
|
|
dispatch('pushGlobalNotice', {
|
|
|
|
level: 'error',
|
|
|
|
messageKey: 'timeline.socket_broke',
|
|
|
|
messageArgs: [code],
|
|
|
|
timeout: 5000
|
|
|
|
})
|
|
|
|
}
|
2021-03-09 00:38:10 +00:00
|
|
|
commit('setMastoUserSocketStatus', WSConnectionStatus.ERROR)
|
2019-12-26 12:12:35 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
resolve()
|
|
|
|
} catch (e) {
|
|
|
|
reject(e)
|
2019-12-08 14:05:41 +00:00
|
|
|
}
|
2019-11-24 20:01:12 +00:00
|
|
|
})
|
2019-11-24 16:50:28 +00:00
|
|
|
},
|
2019-12-10 19:30:27 +00:00
|
|
|
stopMastoUserSocket ({ state, dispatch }) {
|
|
|
|
dispatch('startFetchingTimeline', { timeline: 'friends' })
|
|
|
|
dispatch('startFetchingNotifications')
|
|
|
|
state.mastoUserSocket.close()
|
|
|
|
},
|
2019-12-08 14:05:41 +00:00
|
|
|
|
|
|
|
// Timelines
|
|
|
|
startFetchingTimeline (store, {
|
|
|
|
timeline = 'friends',
|
|
|
|
tag = false,
|
2022-06-15 14:12:05 +00:00
|
|
|
userId = false,
|
|
|
|
listId = false
|
2019-12-08 14:05:41 +00:00
|
|
|
}) {
|
2019-02-07 23:23:18 +00:00
|
|
|
if (store.state.fetchers[timeline]) return
|
|
|
|
|
2019-12-08 14:05:41 +00:00
|
|
|
const fetcher = store.state.backendInteractor.startFetchingTimeline({
|
2022-06-15 14:12:05 +00:00
|
|
|
timeline, store, userId, listId, tag
|
2019-12-08 14:05:41 +00:00
|
|
|
})
|
2019-04-04 16:03:56 +00:00
|
|
|
store.commit('addFetcher', { fetcherName: timeline, fetcher })
|
2017-02-16 10:17:47 +00:00
|
|
|
},
|
2019-12-08 14:05:41 +00:00
|
|
|
stopFetchingTimeline (store, timeline) {
|
|
|
|
const fetcher = store.state.fetchers[timeline]
|
|
|
|
if (!fetcher) return
|
|
|
|
store.commit('removeFetcher', { fetcherName: timeline, fetcher })
|
|
|
|
},
|
2021-01-13 20:17:10 +00:00
|
|
|
fetchTimeline (store, timeline, { ...rest }) {
|
|
|
|
store.state.backendInteractor.fetchTimeline({
|
|
|
|
store,
|
|
|
|
timeline,
|
|
|
|
...rest
|
|
|
|
})
|
|
|
|
},
|
2019-04-04 16:03:56 +00:00
|
|
|
|
2019-12-08 14:05:41 +00:00
|
|
|
// Notifications
|
|
|
|
startFetchingNotifications (store) {
|
|
|
|
if (store.state.fetchers.notifications) return
|
2019-04-04 16:03:56 +00:00
|
|
|
const fetcher = store.state.backendInteractor.startFetchingNotifications({ store })
|
|
|
|
store.commit('addFetcher', { fetcherName: 'notifications', fetcher })
|
|
|
|
},
|
2019-12-08 14:05:41 +00:00
|
|
|
stopFetchingNotifications (store) {
|
|
|
|
const fetcher = store.state.fetchers.notifications
|
|
|
|
if (!fetcher) return
|
|
|
|
store.commit('removeFetcher', { fetcherName: 'notifications', fetcher })
|
|
|
|
},
|
2021-01-13 20:17:10 +00:00
|
|
|
fetchNotifications (store, { ...rest }) {
|
|
|
|
store.state.backendInteractor.fetchNotifications({
|
|
|
|
store,
|
|
|
|
...rest
|
|
|
|
})
|
|
|
|
},
|
2019-11-19 14:07:15 +00:00
|
|
|
|
2019-12-08 14:05:41 +00:00
|
|
|
// Follow requests
|
|
|
|
startFetchingFollowRequests (store) {
|
|
|
|
if (store.state.fetchers['followRequests']) return
|
|
|
|
const fetcher = store.state.backendInteractor.startFetchingFollowRequests({ store })
|
2020-01-21 15:51:49 +00:00
|
|
|
|
2019-12-08 14:05:41 +00:00
|
|
|
store.commit('addFetcher', { fetcherName: 'followRequests', fetcher })
|
2019-11-19 14:07:15 +00:00
|
|
|
},
|
2019-12-08 14:05:41 +00:00
|
|
|
stopFetchingFollowRequests (store) {
|
|
|
|
const fetcher = store.state.fetchers.followRequests
|
|
|
|
if (!fetcher) return
|
|
|
|
store.commit('removeFetcher', { fetcherName: 'followRequests', fetcher })
|
|
|
|
},
|
|
|
|
removeFollowRequest (store, request) {
|
|
|
|
let requests = store.state.followRequests.filter((it) => it !== request)
|
|
|
|
store.commit('setFollowRequests', requests)
|
2017-12-05 10:47:10 +00:00
|
|
|
},
|
2019-12-08 14:05:41 +00:00
|
|
|
|
2022-06-15 15:34:11 +00:00
|
|
|
// Lists
|
|
|
|
startFetchingLists (store) {
|
|
|
|
if (store.state.fetchers['lists']) return
|
|
|
|
const fetcher = store.state.backendInteractor.startFetchingLists({ store })
|
|
|
|
store.commit('addFetcher', { fetcherName: 'lists', fetcher })
|
|
|
|
},
|
|
|
|
stopFetchingLists (store) {
|
|
|
|
const fetcher = store.state.fetchers.lists
|
|
|
|
if (!fetcher) return
|
|
|
|
store.commit('removeFetcher', { fetcherName: 'lists', fetcher })
|
|
|
|
},
|
|
|
|
|
2022-07-18 13:08:50 +00:00
|
|
|
// Lists
|
|
|
|
startFetchingAnnouncements (store) {
|
|
|
|
if (store.state.fetchers['announcements']) return
|
|
|
|
const fetcher = store.state.backendInteractor.startFetchingAnnouncements({ store })
|
|
|
|
store.commit('addFetcher', { fetcherName: 'announcements', fetcher })
|
|
|
|
},
|
|
|
|
stopFetchingAnnouncements (store) {
|
|
|
|
const fetcher = store.state.fetchers.announcements
|
|
|
|
if (!fetcher) return
|
|
|
|
store.commit('removeFetcher', { fetcherName: 'announcements', fetcher })
|
|
|
|
},
|
2022-08-30 13:37:36 +00:00
|
|
|
getSupportedTranslationlanguages (store) {
|
|
|
|
store.state.backendInteractor.getSupportedTranslationlanguages({ store })
|
|
|
|
.then((data) => {
|
2022-08-30 14:43:57 +00:00
|
|
|
store.dispatch('setOption', { name: 'supportedTranslationLanguages', value: data })
|
2022-08-30 13:37:36 +00:00
|
|
|
})
|
|
|
|
},
|
2022-07-18 13:08:50 +00:00
|
|
|
|
2019-12-08 14:05:41 +00:00
|
|
|
// Pleroma websocket
|
2019-01-29 15:16:25 +00:00
|
|
|
setWsToken (store, token) {
|
|
|
|
store.commit('setWsToken', token)
|
|
|
|
},
|
2019-08-17 08:18:42 +00:00
|
|
|
disconnectFromSocket ({ commit, state }) {
|
|
|
|
state.socket && state.socket.disconnect()
|
|
|
|
commit('setSocket', null)
|
2016-11-26 17:57:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default api
|