Reduce polling (#2465)
* Move profile and preference polling to polls-on-foreground * Refetch prefs on feeds screen refresh since polling no longer occurs * Reduce notifications polling by 50% if there's already an unread * Disable feed polling if we know we have content * Disable the hard refresh after 1 hour in case it's the cause of the random feed refresh bug * Fix typeszio/stable
parent
0442dcc1a0
commit
7ab4be6f7d
|
@ -1,11 +1,39 @@
|
||||||
import {QueryClient} from '@tanstack/react-query'
|
import {AppState, AppStateStatus} from 'react-native'
|
||||||
|
import {QueryClient, focusManager} from '@tanstack/react-query'
|
||||||
|
import {isNative} from '#/platform/detection'
|
||||||
|
|
||||||
|
focusManager.setEventListener(onFocus => {
|
||||||
|
if (isNative) {
|
||||||
|
const subscription = AppState.addEventListener(
|
||||||
|
'change',
|
||||||
|
(status: AppStateStatus) => {
|
||||||
|
focusManager.setFocused(status === 'active')
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
return () => subscription.remove()
|
||||||
|
} else if (typeof window !== 'undefined' && window.addEventListener) {
|
||||||
|
// these handlers are a bit redundant but focus catches when the browser window
|
||||||
|
// is blurred/focused while visibilitychange seems to only handle when the
|
||||||
|
// window minimizes (both of them catch tab changes)
|
||||||
|
// there's no harm to redundant fires because refetchOnWindowFocus is only
|
||||||
|
// used with queries that employ stale data times
|
||||||
|
const handler = () => onFocus()
|
||||||
|
window.addEventListener('focus', handler, false)
|
||||||
|
window.addEventListener('visibilitychange', handler, false)
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('visibilitychange', handler)
|
||||||
|
window.removeEventListener('focus', handler)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
export const queryClient = new QueryClient({
|
export const queryClient = new QueryClient({
|
||||||
defaultOptions: {
|
defaultOptions: {
|
||||||
queries: {
|
queries: {
|
||||||
// NOTE
|
// NOTE
|
||||||
// refetchOnWindowFocus breaks some UIs (like feeds)
|
// refetchOnWindowFocus breaks some UIs (like feeds)
|
||||||
// so we NEVER want to enable this
|
// so we only selectively want to enable this
|
||||||
// -prf
|
// -prf
|
||||||
refetchOnWindowFocus: false,
|
refetchOnWindowFocus: false,
|
||||||
// Structural sharing between responses makes it impossible to rely on
|
// Structural sharing between responses makes it impossible to rely on
|
||||||
|
|
|
@ -35,4 +35,5 @@ export interface CachedFeedPage {
|
||||||
usableInFeed: boolean
|
usableInFeed: boolean
|
||||||
syncedAt: Date
|
syncedAt: Date
|
||||||
data: FeedPage | undefined
|
data: FeedPage | undefined
|
||||||
|
unreadCount: number
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,7 +25,10 @@ type StateContext = string
|
||||||
|
|
||||||
interface ApiContext {
|
interface ApiContext {
|
||||||
markAllRead: () => Promise<void>
|
markAllRead: () => Promise<void>
|
||||||
checkUnread: (opts?: {invalidate?: boolean}) => Promise<void>
|
checkUnread: (opts?: {
|
||||||
|
invalidate?: boolean
|
||||||
|
isPoll?: boolean
|
||||||
|
}) => Promise<void>
|
||||||
getCachedUnreadPage: () => FeedPage | undefined
|
getCachedUnreadPage: () => FeedPage | undefined
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -50,6 +53,7 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
|
||||||
usableInFeed: false,
|
usableInFeed: false,
|
||||||
syncedAt: new Date(),
|
syncedAt: new Date(),
|
||||||
data: undefined,
|
data: undefined,
|
||||||
|
unreadCount: 0,
|
||||||
})
|
})
|
||||||
|
|
||||||
// periodic sync
|
// periodic sync
|
||||||
|
@ -58,7 +62,10 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
checkUnreadRef.current() // fire on init
|
checkUnreadRef.current() // fire on init
|
||||||
const interval = setInterval(checkUnreadRef.current, UPDATE_INTERVAL)
|
const interval = setInterval(
|
||||||
|
() => checkUnreadRef.current?.({isPoll: true}),
|
||||||
|
UPDATE_INTERVAL,
|
||||||
|
)
|
||||||
return () => clearInterval(interval)
|
return () => clearInterval(interval)
|
||||||
}, [hasSession])
|
}, [hasSession])
|
||||||
|
|
||||||
|
@ -69,6 +76,12 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
|
||||||
usableInFeed: false,
|
usableInFeed: false,
|
||||||
syncedAt: new Date(),
|
syncedAt: new Date(),
|
||||||
data: undefined,
|
data: undefined,
|
||||||
|
unreadCount:
|
||||||
|
data.event === '30+'
|
||||||
|
? 30
|
||||||
|
: data.event === ''
|
||||||
|
? 0
|
||||||
|
: parseInt(data.event, 10) || 1,
|
||||||
}
|
}
|
||||||
setNumUnread(data.event)
|
setNumUnread(data.event)
|
||||||
}
|
}
|
||||||
|
@ -95,13 +108,24 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
async checkUnread({invalidate}: {invalidate?: boolean} = {}) {
|
async checkUnread({
|
||||||
|
invalidate,
|
||||||
|
isPoll,
|
||||||
|
}: {invalidate?: boolean; isPoll?: boolean} = {}) {
|
||||||
try {
|
try {
|
||||||
if (!getAgent().session) return
|
if (!getAgent().session) return
|
||||||
if (AppState.currentState !== 'active') {
|
if (AppState.currentState !== 'active') {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// reduce polling if unread count is set
|
||||||
|
if (isPoll && cacheRef.current?.unreadCount !== 0) {
|
||||||
|
// if hit 30+ then don't poll, otherwise reduce polling by 50%
|
||||||
|
if (cacheRef.current?.unreadCount >= 30 || Math.random() >= 0.5) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// count
|
// count
|
||||||
const page = await fetchPage({
|
const page = await fetchPage({
|
||||||
cursor: undefined,
|
cursor: undefined,
|
||||||
|
@ -133,6 +157,7 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
|
||||||
usableInFeed: !!invalidate, // will be used immediately
|
usableInFeed: !!invalidate, // will be used immediately
|
||||||
data: page,
|
data: page,
|
||||||
syncedAt: !lastIndexed || now > lastIndexed ? now : lastIndexed,
|
syncedAt: !lastIndexed || now > lastIndexed ? now : lastIndexed,
|
||||||
|
unreadCount,
|
||||||
}
|
}
|
||||||
|
|
||||||
// update & broadcast
|
// update & broadcast
|
||||||
|
|
|
@ -31,7 +31,7 @@ export function usePreferencesQuery() {
|
||||||
return useQuery({
|
return useQuery({
|
||||||
staleTime: STALE.SECONDS.FIFTEEN,
|
staleTime: STALE.SECONDS.FIFTEEN,
|
||||||
structuralSharing: true,
|
structuralSharing: true,
|
||||||
refetchInterval: STALE.SECONDS.FIFTEEN,
|
refetchOnWindowFocus: true,
|
||||||
queryKey: preferencesQueryKey,
|
queryKey: preferencesQueryKey,
|
||||||
queryFn: async () => {
|
queryFn: async () => {
|
||||||
const agent = getAgent()
|
const agent = getAgent()
|
||||||
|
|
|
@ -35,7 +35,7 @@ export function useProfileQuery({did}: {did: string | undefined}) {
|
||||||
// if you remove it, the UI infinite-loops
|
// if you remove it, the UI infinite-loops
|
||||||
// -prf
|
// -prf
|
||||||
staleTime: isCurrentAccount ? STALE.SECONDS.THIRTY : STALE.MINUTES.FIVE,
|
staleTime: isCurrentAccount ? STALE.SECONDS.THIRTY : STALE.MINUTES.FIVE,
|
||||||
refetchInterval: STALE.MINUTES.FIVE,
|
refetchOnWindowFocus: true,
|
||||||
queryKey: RQKEY(did || ''),
|
queryKey: RQKEY(did || ''),
|
||||||
queryFn: async () => {
|
queryFn: async () => {
|
||||||
const res = await getAgent().getProfile({actor: did || ''})
|
const res = await getAgent().getProfile({actor: did || ''})
|
||||||
|
|
|
@ -174,6 +174,7 @@ export function FeedPage({
|
||||||
feed={feed}
|
feed={feed}
|
||||||
feedParams={feedParams}
|
feedParams={feedParams}
|
||||||
pollInterval={POLL_FREQ}
|
pollInterval={POLL_FREQ}
|
||||||
|
disablePoll={hasNew}
|
||||||
scrollElRef={scrollElRef}
|
scrollElRef={scrollElRef}
|
||||||
onScrolledDownChange={setIsScrolledDown}
|
onScrolledDownChange={setIsScrolledDown}
|
||||||
onHasNew={setHasNew}
|
onHasNew={setHasNew}
|
||||||
|
|
|
@ -36,7 +36,8 @@ const EMPTY_FEED_ITEM = {_reactKey: '__empty__'}
|
||||||
const ERROR_ITEM = {_reactKey: '__error__'}
|
const ERROR_ITEM = {_reactKey: '__error__'}
|
||||||
const LOAD_MORE_ERROR_ITEM = {_reactKey: '__load_more_error__'}
|
const LOAD_MORE_ERROR_ITEM = {_reactKey: '__load_more_error__'}
|
||||||
|
|
||||||
const REFRESH_AFTER = STALE.HOURS.ONE
|
// DISABLED need to check if this is causing random feed refreshes -prf
|
||||||
|
// const REFRESH_AFTER = STALE.HOURS.ONE
|
||||||
const CHECK_LATEST_AFTER = STALE.SECONDS.THIRTY
|
const CHECK_LATEST_AFTER = STALE.SECONDS.THIRTY
|
||||||
|
|
||||||
let Feed = ({
|
let Feed = ({
|
||||||
|
@ -46,6 +47,7 @@ let Feed = ({
|
||||||
style,
|
style,
|
||||||
enabled,
|
enabled,
|
||||||
pollInterval,
|
pollInterval,
|
||||||
|
disablePoll,
|
||||||
scrollElRef,
|
scrollElRef,
|
||||||
onScrolledDownChange,
|
onScrolledDownChange,
|
||||||
onHasNew,
|
onHasNew,
|
||||||
|
@ -63,6 +65,7 @@ let Feed = ({
|
||||||
style?: StyleProp<ViewStyle>
|
style?: StyleProp<ViewStyle>
|
||||||
enabled?: boolean
|
enabled?: boolean
|
||||||
pollInterval?: number
|
pollInterval?: number
|
||||||
|
disablePoll?: boolean
|
||||||
scrollElRef?: ListRef
|
scrollElRef?: ListRef
|
||||||
onHasNew?: (v: boolean) => void
|
onHasNew?: (v: boolean) => void
|
||||||
onScrolledDownChange?: (isScrolledDown: boolean) => void
|
onScrolledDownChange?: (isScrolledDown: boolean) => void
|
||||||
|
@ -107,7 +110,7 @@ let Feed = ({
|
||||||
)
|
)
|
||||||
|
|
||||||
const checkForNew = React.useCallback(async () => {
|
const checkForNew = React.useCallback(async () => {
|
||||||
if (!data?.pages[0] || isFetching || !onHasNew || !enabled) {
|
if (!data?.pages[0] || isFetching || !onHasNew || !enabled || disablePoll) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
|
@ -117,7 +120,7 @@ let Feed = ({
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
logger.error('Poll latest failed', {feed, error: String(e)})
|
logger.error('Poll latest failed', {feed, error: String(e)})
|
||||||
}
|
}
|
||||||
}, [feed, data, isFetching, onHasNew, enabled])
|
}, [feed, data, isFetching, onHasNew, enabled, disablePoll])
|
||||||
|
|
||||||
const myDid = currentAccount?.did || ''
|
const myDid = currentAccount?.did || ''
|
||||||
const onPostCreated = React.useCallback(() => {
|
const onPostCreated = React.useCallback(() => {
|
||||||
|
@ -146,11 +149,12 @@ let Feed = ({
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
if (enabled) {
|
if (enabled) {
|
||||||
const timeSinceFirstLoad = Date.now() - lastFetchRef.current
|
const timeSinceFirstLoad = Date.now() - lastFetchRef.current
|
||||||
if (timeSinceFirstLoad > REFRESH_AFTER) {
|
// DISABLED need to check if this is causing random feed refreshes -prf
|
||||||
|
/*if (timeSinceFirstLoad > REFRESH_AFTER) {
|
||||||
// do a full refresh
|
// do a full refresh
|
||||||
scrollElRef?.current?.scrollToOffset({offset: 0, animated: false})
|
scrollElRef?.current?.scrollToOffset({offset: 0, animated: false})
|
||||||
queryClient.resetQueries({queryKey: RQKEY(feed)})
|
queryClient.resetQueries({queryKey: RQKEY(feed)})
|
||||||
} else if (
|
} else*/ if (
|
||||||
timeSinceFirstLoad > CHECK_LATEST_AFTER &&
|
timeSinceFirstLoad > CHECK_LATEST_AFTER &&
|
||||||
checkForNewRef.current
|
checkForNewRef.current
|
||||||
) {
|
) {
|
||||||
|
|
|
@ -97,6 +97,7 @@ export function FeedsScreen(_props: Props) {
|
||||||
data: preferences,
|
data: preferences,
|
||||||
isLoading: isPreferencesLoading,
|
isLoading: isPreferencesLoading,
|
||||||
error: preferencesError,
|
error: preferencesError,
|
||||||
|
refetch: refetchPreferences,
|
||||||
} = usePreferencesQuery()
|
} = usePreferencesQuery()
|
||||||
const {
|
const {
|
||||||
data: popularFeeds,
|
data: popularFeeds,
|
||||||
|
@ -151,9 +152,12 @@ export function FeedsScreen(_props: Props) {
|
||||||
}, [query, debouncedSearch])
|
}, [query, debouncedSearch])
|
||||||
const onPullToRefresh = React.useCallback(async () => {
|
const onPullToRefresh = React.useCallback(async () => {
|
||||||
setIsPTR(true)
|
setIsPTR(true)
|
||||||
await refetchPopularFeeds()
|
await Promise.all([
|
||||||
|
refetchPreferences().catch(_e => undefined),
|
||||||
|
refetchPopularFeeds().catch(_e => undefined),
|
||||||
|
])
|
||||||
setIsPTR(false)
|
setIsPTR(false)
|
||||||
}, [setIsPTR, refetchPopularFeeds])
|
}, [setIsPTR, refetchPreferences, refetchPopularFeeds])
|
||||||
const onEndReached = React.useCallback(() => {
|
const onEndReached = React.useCallback(() => {
|
||||||
if (
|
if (
|
||||||
isPopularFeedsFetching ||
|
isPopularFeedsFetching ||
|
||||||
|
|
|
@ -490,6 +490,7 @@ const FeedSection = React.forwardRef<SectionRef, FeedSectionProps>(
|
||||||
enabled={isFocused}
|
enabled={isFocused}
|
||||||
feed={feed}
|
feed={feed}
|
||||||
pollInterval={60e3}
|
pollInterval={60e3}
|
||||||
|
disablePoll={hasNew}
|
||||||
scrollElRef={scrollElRef}
|
scrollElRef={scrollElRef}
|
||||||
onHasNew={setHasNew}
|
onHasNew={setHasNew}
|
||||||
onScrolledDownChange={setIsScrolledDown}
|
onScrolledDownChange={setIsScrolledDown}
|
||||||
|
|
|
@ -646,6 +646,7 @@ const FeedSection = React.forwardRef<SectionRef, FeedSectionProps>(
|
||||||
enabled={isFocused}
|
enabled={isFocused}
|
||||||
feed={feed}
|
feed={feed}
|
||||||
pollInterval={60e3}
|
pollInterval={60e3}
|
||||||
|
disablePoll={hasNew}
|
||||||
scrollElRef={scrollElRef}
|
scrollElRef={scrollElRef}
|
||||||
onHasNew={setHasNew}
|
onHasNew={setHasNew}
|
||||||
onScrolledDownChange={setIsScrolledDown}
|
onScrolledDownChange={setIsScrolledDown}
|
||||||
|
|
Loading…
Reference in New Issue