mirror of https://github.com/nocodb/nocodb
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
249 lines
7.0 KiB
249 lines
7.0 KiB
import { defineStore } from 'pinia' |
|
import type { NotificationType } from 'nocodb-sdk' |
|
import axios, { type CancelTokenSource } from 'axios' |
|
import { useStorage } from '@vueuse/core' |
|
|
|
const CancelToken = axios.CancelToken |
|
|
|
export const useNotification = defineStore('notificationStore', () => { |
|
const isTokenRefreshInProgress = useStorage(TOKEN_REFRESH_PROGRESS_KEY, false) |
|
|
|
const readNotifications = ref<NotificationType[]>([]) |
|
|
|
const unreadNotifications = ref<NotificationType[]>([]) |
|
|
|
const readPageInfo = ref() |
|
|
|
const unreadPageInfo = ref() |
|
|
|
const unreadCount = ref(0) |
|
|
|
const notificationTab = ref<'read' | 'unread'>('unread') |
|
|
|
const { api, isLoading } = useApi() |
|
|
|
const { token } = useGlobal() |
|
|
|
let timeOutId: number | null = null |
|
|
|
let cancelTokenSource: CancelTokenSource | null |
|
|
|
const pollNotificationsApiCall = async () => { |
|
// set up cancel token for polling to cancel when token changes/token is removed |
|
cancelTokenSource = CancelToken.source() |
|
|
|
return await api.notification.poll({ |
|
cancelToken: cancelTokenSource.token, |
|
}) |
|
} |
|
|
|
const sharedExecutionPollNotificationsApiCall = useSharedExecutionFn('notification', pollNotificationsApiCall, { |
|
timeout: 30000, |
|
}) |
|
|
|
const pollNotifications = async () => { |
|
try { |
|
if (!token.value) return |
|
|
|
const res = await sharedExecutionPollNotificationsApiCall() |
|
|
|
if (res.status === 'success') { |
|
if (notificationTab.value === 'unread') { |
|
unreadNotifications.value = [JSON.parse(res.data), ...unreadNotifications.value] |
|
} |
|
|
|
unreadCount.value = unreadCount.value + 1 |
|
} |
|
|
|
timeOutId = setTimeout(pollNotifications, 0) |
|
} catch (e) { |
|
// If request is cancelled, do nothing |
|
if (axios.isCancel(e)) return |
|
// If network error, retry after 2 seconds |
|
timeOutId = setTimeout(pollNotifications, 2000) |
|
} |
|
} |
|
|
|
const loadReadNotifications = async (loadMore?: boolean) => { |
|
try { |
|
const response = await api.notification.list({ |
|
is_read: true, |
|
limit: 10, |
|
offset: loadMore ? readNotifications.value.length : 0, |
|
}) |
|
|
|
if (loadMore) { |
|
readNotifications.value = [...readNotifications.value, ...response.list] |
|
} else { |
|
readNotifications.value = response.list |
|
} |
|
|
|
readPageInfo.value = response.pageInfo |
|
|
|
unreadCount.value = Number(response.unreadCount) |
|
} catch (e) { |
|
console.log(e) |
|
} |
|
} |
|
|
|
const loadUnReadNotifications = async (loadMore?: boolean) => { |
|
try { |
|
const response = await api.notification.list({ |
|
is_read: false, |
|
limit: 10, |
|
offset: loadMore ? unreadNotifications.value.length : 0, |
|
}) |
|
|
|
if (loadMore) { |
|
unreadNotifications.value = [...unreadNotifications.value, ...response.list] |
|
} else { |
|
unreadNotifications.value = response.list |
|
} |
|
|
|
unreadPageInfo.value = response.pageInfo |
|
|
|
unreadCount.value = Number(response.unreadCount) |
|
} catch (e) { |
|
console.log(e) |
|
} |
|
} |
|
|
|
const insertAndSort = (notification: NotificationType, oldState?: boolean) => { |
|
if (oldState) { |
|
readNotifications.value = readNotifications.value.filter((n) => n.id !== notification.id) |
|
|
|
unreadNotifications.value = [notification, ...unreadNotifications.value].sort((a, b) => { |
|
return new Date(b.created_at).getTime() - new Date(a.created_at).getTime() |
|
}) |
|
|
|
unreadCount.value = unreadCount.value + 1 |
|
} else { |
|
readNotifications.value = [notification, ...readNotifications.value].sort((a, b) => { |
|
return new Date(b.created_at).getTime() - new Date(a.created_at).getTime() |
|
}) |
|
|
|
unreadNotifications.value = unreadNotifications.value.filter((n) => n.id !== notification.id) |
|
|
|
unreadCount.value = unreadCount.value - 1 |
|
} |
|
} |
|
|
|
const toggleRead = async (notification: NotificationType, ignoreTrigger?: boolean) => { |
|
if (ignoreTrigger) return |
|
|
|
const currState = notification.is_read |
|
|
|
try { |
|
await api.notification.update(notification.id!, { |
|
is_read: !currState, |
|
}) |
|
notification.is_read = !currState |
|
|
|
insertAndSort(notification, currState) |
|
} catch (e) { |
|
message.error( |
|
`Failed to update Notification: ${await extractSdkResponseErrorMsgv2(e as Error & { response: { data: string } })}`, |
|
) |
|
} |
|
} |
|
|
|
const markAllAsRead = async (notification: NotificationType) => { |
|
if (notification.is_read) return |
|
|
|
await api.notification.markAllAsRead() |
|
|
|
await Promise.allSettled([loadReadNotifications(), loadUnReadNotifications()]) |
|
} |
|
|
|
const deleteNotification = async (notification: NotificationType) => { |
|
try { |
|
readNotifications.value = readNotifications.value.filter((n) => n.id !== notification.id) |
|
|
|
await api.notification.delete(notification.id!) |
|
} catch (e) { |
|
readNotifications.value = [notification, ...readNotifications.value] |
|
|
|
message.error( |
|
`Failed to delete Notification: ${await extractSdkResponseErrorMsgv2( |
|
e as Error & { |
|
response: { |
|
data: string |
|
} |
|
}, |
|
)}`, |
|
) |
|
} |
|
} |
|
|
|
watch(notificationTab, async (tab) => { |
|
if (tab === 'read') { |
|
await loadReadNotifications() |
|
} else { |
|
await loadUnReadNotifications() |
|
} |
|
}) |
|
|
|
// function to clear polling and cancel any pending requests |
|
const clearPolling = async () => { |
|
if (timeOutId) { |
|
clearTimeout(timeOutId) |
|
timeOutId = null |
|
} |
|
// take a reference of the cancel token source and set the current one to null |
|
// so that we can cancel the polling request even if token changes |
|
const source = cancelTokenSource |
|
cancelTokenSource = null |
|
// wait if refresh token generation is in progress and cancel the polling after that |
|
// set a timeout of 10 seconds to avoid hanging |
|
await until(isTokenRefreshInProgress).toMatch((v) => !v, { timeout: 10000 }) |
|
source?.cancel() |
|
} |
|
|
|
const init = async () => { |
|
await Promise.allSettled([loadReadNotifications(), loadUnReadNotifications()]) |
|
// For playwright, polling will cause the test to hang indefinitely |
|
// as we wait for the networkidle event. So, we disable polling for playwright |
|
if (!(window as any).isPlaywright) { |
|
clearPolling().catch((e) => console.log(e)) |
|
pollNotifications().catch((e) => console.log(e)) |
|
} |
|
} |
|
|
|
// watch for token changes and re-init notifications if token changes |
|
watch( |
|
token, |
|
async (newToken, oldToken) => { |
|
try { |
|
if (newToken && newToken !== oldToken) { |
|
await init() |
|
} else if (!newToken) { |
|
clearPolling().catch((e) => console.log(e)) |
|
} |
|
} catch (e) { |
|
console.error(e) |
|
} |
|
}, |
|
{ |
|
immediate: true, |
|
}, |
|
) |
|
|
|
return { |
|
unreadNotifications, |
|
readNotifications, |
|
loadUnReadNotifications, |
|
loadReadNotifications, |
|
deleteNotification, |
|
readPageInfo, |
|
unreadPageInfo, |
|
isLoading, |
|
notificationTab, |
|
toggleRead, |
|
markAllAsRead, |
|
unreadCount, |
|
} |
|
}) |
|
|
|
if (import.meta.hot) { |
|
import.meta.hot.accept(acceptHMRUpdate(useNotification, import.meta.hot)) |
|
}
|
|
|