a7cf89a977
We were already limiting the call to once-per-5-minutes (for the succesful case), so just need to put failure cases into the mix. Retry 3 times before stopping.
221 lines
7.1 KiB
JavaScript
221 lines
7.1 KiB
JavaScript
// @flow
|
|
import * as ACTIONS from 'constants/action_types';
|
|
import { doClaimSearch } from 'redux/actions/claims';
|
|
import {
|
|
LiveStatus,
|
|
fetchLiveChannel,
|
|
fetchLiveChannels,
|
|
determineLiveClaim,
|
|
filterUpcomingLiveStreamClaims,
|
|
} from 'util/livestream';
|
|
import moment from 'moment';
|
|
import { isLocalStorageAvailable } from 'util/storage';
|
|
import { isEmpty } from 'util/object';
|
|
|
|
const localStorageAvailable = isLocalStorageAvailable();
|
|
|
|
const FETCH_ACTIVE_LIVESTREAMS_MIN_INTERVAL_MS = 5 * 60 * 1000;
|
|
|
|
export const doFetchNoSourceClaims = (channelId: string) => async (dispatch: Dispatch, getState: GetState) => {
|
|
dispatch({
|
|
type: ACTIONS.FETCH_NO_SOURCE_CLAIMS_STARTED,
|
|
data: channelId,
|
|
});
|
|
|
|
try {
|
|
await dispatch(
|
|
doClaimSearch({
|
|
channel_ids: [channelId],
|
|
has_no_source: true,
|
|
claim_type: ['stream'],
|
|
no_totals: true,
|
|
page_size: 20,
|
|
page: 1,
|
|
include_is_my_output: true,
|
|
})
|
|
);
|
|
|
|
dispatch({
|
|
type: ACTIONS.FETCH_NO_SOURCE_CLAIMS_COMPLETED,
|
|
data: channelId,
|
|
});
|
|
} catch (error) {
|
|
dispatch({
|
|
type: ACTIONS.FETCH_NO_SOURCE_CLAIMS_FAILED,
|
|
data: channelId,
|
|
});
|
|
}
|
|
};
|
|
|
|
const fetchUpcomingLivestreamClaims = (channelIds: Array<string>, lang: ?Array<string> = null) =>
|
|
doClaimSearch(
|
|
{
|
|
page: 1,
|
|
page_size: 50,
|
|
has_no_source: true,
|
|
channel_ids: channelIds,
|
|
claim_type: ['stream'],
|
|
order_by: ['^release_time'],
|
|
release_time: `>${moment().subtract(5, 'minutes').unix()}`,
|
|
limit_claims_per_channel: 1,
|
|
no_totals: true,
|
|
...(lang ? { any_languages: lang } : {}),
|
|
},
|
|
{
|
|
useAutoPagination: true,
|
|
}
|
|
);
|
|
|
|
const fetchMostRecentLivestreamClaims = (
|
|
channelIds: Array<string>,
|
|
orderBy: Array<string> = ['release_time'],
|
|
lang: ?Array<string> = null
|
|
) =>
|
|
doClaimSearch(
|
|
{
|
|
page: 1,
|
|
page_size: 50,
|
|
has_no_source: true,
|
|
channel_ids: channelIds,
|
|
claim_type: ['stream'],
|
|
order_by: orderBy,
|
|
release_time: `<${moment().unix()}`,
|
|
limit_claims_per_channel: 2,
|
|
no_totals: true,
|
|
...(lang ? { any_languages: lang } : {}),
|
|
},
|
|
{
|
|
useAutoPagination: true,
|
|
}
|
|
);
|
|
|
|
const findActiveStreams = async (
|
|
channelIDs: Array<string>,
|
|
orderBy: Array<string>,
|
|
liveChannels: any,
|
|
dispatch,
|
|
lang: ?Array<string> = null
|
|
) => {
|
|
// @Note: This can likely be simplified down to one query, but first we'll need to address the query limit / pagination issue.
|
|
|
|
// Find the two most recent claims for the channels that are actively broadcasting a stream.
|
|
const mostRecentClaims = await dispatch(fetchMostRecentLivestreamClaims(channelIDs, orderBy, lang));
|
|
|
|
// Find the first upcoming claim (if one exists) for each channel that's actively broadcasting a stream.
|
|
const upcomingClaims = await dispatch(fetchUpcomingLivestreamClaims(channelIDs, lang));
|
|
|
|
// Filter out any of those claims that aren't scheduled to start within the configured "soon" buffer time (ex. next 15 min).
|
|
const startingSoonClaims = filterUpcomingLiveStreamClaims(upcomingClaims);
|
|
|
|
// Reduce the claim list to one "live" claim per channel, based on how close each claim's
|
|
// release time is to the time the channels stream started.
|
|
const allClaims = Object.assign(
|
|
{},
|
|
mostRecentClaims,
|
|
!isEmpty(startingSoonClaims) ? startingSoonClaims : upcomingClaims
|
|
);
|
|
|
|
return determineLiveClaim(allClaims, liveChannels);
|
|
};
|
|
|
|
export const doFetchChannelLiveStatus = (channelId: string) => async (dispatch: Dispatch) => {
|
|
const statusForId = `channel-live-status`;
|
|
const localStatus = localStorageAvailable && window.localStorage.getItem(statusForId);
|
|
|
|
try {
|
|
const { channelStatus, channelData } = await fetchLiveChannel(channelId);
|
|
// store live state locally, and force 2 non-live statuses before returninig NOT LIVE. This allows for the stream to finish before disposing player.
|
|
if (localStatus === LiveStatus.LIVE && channelStatus === LiveStatus.NOT_LIVE) {
|
|
localStorageAvailable && window.localStorage.removeItem(statusForId);
|
|
return;
|
|
}
|
|
|
|
if (channelStatus === LiveStatus.NOT_LIVE && !localStatus) {
|
|
dispatch({ type: ACTIONS.REMOVE_CHANNEL_FROM_ACTIVE_LIVESTREAMS, data: { channelId } });
|
|
localStorageAvailable && window.localStorage.removeItem(statusForId);
|
|
return;
|
|
}
|
|
|
|
if (channelStatus === LiveStatus.UNKNOWN) {
|
|
return;
|
|
}
|
|
|
|
const currentlyLiveClaims = await findActiveStreams([channelId], ['release_time'], channelData, dispatch);
|
|
const liveClaim = currentlyLiveClaims[channelId];
|
|
|
|
if (channelData && liveClaim) {
|
|
channelData[channelId].claimId = liveClaim.stream.claim_id;
|
|
channelData[channelId].claimUri = liveClaim.stream.canonical_url;
|
|
dispatch({ type: ACTIONS.ADD_CHANNEL_TO_ACTIVE_LIVESTREAMS, data: { ...channelData } });
|
|
}
|
|
|
|
localStorageAvailable && window.localStorage.setItem(statusForId, channelStatus);
|
|
} catch (err) {
|
|
dispatch({ type: ACTIONS.REMOVE_CHANNEL_FROM_ACTIVE_LIVESTREAMS, data: { channelId } });
|
|
localStorageAvailable && window.localStorage.removeItem(statusForId);
|
|
}
|
|
};
|
|
|
|
export const doFetchActiveLivestreams = (
|
|
orderBy: Array<string> = ['release_time'],
|
|
lang: ?Array<string> = null
|
|
) => async (dispatch: Dispatch, getState: GetState) => {
|
|
const state = getState();
|
|
const now = Date.now();
|
|
const timeDelta = now - state.livestream.activeLivestreamsLastFetchedDate;
|
|
|
|
const prevOptions = state.livestream.activeLivestreamsLastFetchedOptions;
|
|
const nextOptions = { order_by: orderBy, ...(lang ? { any_languages: lang } : {}) };
|
|
const sameOptions = JSON.stringify(prevOptions) === JSON.stringify(nextOptions);
|
|
|
|
if (sameOptions && timeDelta < FETCH_ACTIVE_LIVESTREAMS_MIN_INTERVAL_MS) {
|
|
const failCount = state.livestream.activeLivestreamsLastFetchedFailCount;
|
|
if (failCount === 0 || failCount > 3) {
|
|
// Just fetched successfully, or failed 3 times. Skip for FETCH_ACTIVE_LIVESTREAMS_MIN_INTERVAL_MS.
|
|
dispatch({ type: ACTIONS.FETCH_ACTIVE_LIVESTREAMS_SKIPPED });
|
|
return;
|
|
}
|
|
}
|
|
|
|
// start fetching livestreams
|
|
dispatch({ type: ACTIONS.FETCH_ACTIVE_LIVESTREAMS_STARTED });
|
|
|
|
try {
|
|
const liveChannels = await fetchLiveChannels();
|
|
const liveChannelIds = Object.keys(liveChannels);
|
|
|
|
const currentlyLiveClaims = await findActiveStreams(
|
|
liveChannelIds,
|
|
nextOptions.order_by,
|
|
liveChannels,
|
|
dispatch,
|
|
nextOptions.any_languages
|
|
);
|
|
Object.values(currentlyLiveClaims).forEach((claim: any) => {
|
|
const channelId = claim.stream.signing_channel.claim_id;
|
|
|
|
liveChannels[channelId] = {
|
|
...liveChannels[channelId],
|
|
claimId: claim.stream.claim_id,
|
|
claimUri: claim.stream.canonical_url,
|
|
};
|
|
});
|
|
|
|
dispatch({
|
|
type: ACTIONS.FETCH_ACTIVE_LIVESTREAMS_COMPLETED,
|
|
data: {
|
|
activeLivestreams: liveChannels,
|
|
activeLivestreamsLastFetchedDate: now,
|
|
activeLivestreamsLastFetchedOptions: nextOptions,
|
|
},
|
|
});
|
|
} catch (err) {
|
|
dispatch({
|
|
type: ACTIONS.FETCH_ACTIVE_LIVESTREAMS_FAILED,
|
|
data: {
|
|
activeLivestreamsLastFetchedDate: now,
|
|
activeLivestreamsLastFetchedOptions: nextOptions,
|
|
},
|
|
});
|
|
}
|
|
};
|