2020-07-27 16:04:12 -04:00
|
|
|
// @flow
|
|
|
|
import * as ACTIONS from 'constants/action_types';
|
2021-03-19 23:04:12 +08:00
|
|
|
import { buildURI, doResolveUris, batchActions, isURIValid, makeSelectClaimForUri } from 'lbry-redux';
|
2021-03-26 16:33:30 +08:00
|
|
|
import {
|
|
|
|
makeSelectSearchUris,
|
|
|
|
makeSelectQueryWithOptions,
|
|
|
|
selectSearchValue,
|
2021-04-07 18:50:15 +08:00
|
|
|
selectSearchOptions,
|
|
|
|
} from 'redux/selectors/search';
|
2020-07-27 16:04:12 -04:00
|
|
|
import handleFetchResponse from 'util/handle-fetch';
|
|
|
|
|
|
|
|
type Dispatch = (action: any) => any;
|
|
|
|
type GetState = () => { search: SearchState };
|
|
|
|
|
|
|
|
type SearchOptions = {
|
|
|
|
size?: number,
|
|
|
|
from?: number,
|
|
|
|
related_to?: string,
|
|
|
|
nsfw?: boolean,
|
|
|
|
isBackgroundSearch?: boolean,
|
|
|
|
};
|
|
|
|
|
2020-12-03 12:29:47 -05:00
|
|
|
let lighthouse = {
|
|
|
|
CONNECTION_STRING: 'https://lighthouse.lbry.com/search',
|
|
|
|
search: (queryString: string) => fetch(`${lighthouse.CONNECTION_STRING}?${queryString}`).then(handleFetchResponse),
|
2020-07-27 16:04:12 -04:00
|
|
|
};
|
|
|
|
|
2020-12-03 12:29:47 -05:00
|
|
|
export const setSearchApi = (endpoint: string) => {
|
|
|
|
lighthouse.CONNECTION_STRING = endpoint.replace(/\/*$/, '/'); // exactly one slash at the end;
|
2020-07-27 16:04:12 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
export const doSearch = (rawQuery: string, searchOptions: SearchOptions) => (
|
|
|
|
dispatch: Dispatch,
|
|
|
|
getState: GetState
|
|
|
|
) => {
|
|
|
|
const query = rawQuery.replace(/^lbry:\/\//i, '').replace(/\//, ' ');
|
|
|
|
|
|
|
|
if (!query) {
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.SEARCH_FAIL,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const state = getState();
|
|
|
|
|
2021-03-26 16:33:30 +08:00
|
|
|
const mainOptions: any = selectSearchOptions(state);
|
|
|
|
const queryWithOptions = makeSelectQueryWithOptions(query, searchOptions)(state);
|
|
|
|
|
|
|
|
const size = mainOptions.size;
|
|
|
|
const from = searchOptions.from;
|
2020-07-27 16:04:12 -04:00
|
|
|
|
|
|
|
// If we have already searched for something, we don't need to do anything
|
|
|
|
const urisForQuery = makeSelectSearchUris(queryWithOptions)(state);
|
|
|
|
if (urisForQuery && !!urisForQuery.length) {
|
2021-03-26 16:33:30 +08:00
|
|
|
if (!size || !from || from + size < urisForQuery.length) {
|
|
|
|
return;
|
|
|
|
}
|
2020-07-27 16:04:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.SEARCH_START,
|
|
|
|
});
|
|
|
|
|
2020-12-03 12:29:47 -05:00
|
|
|
lighthouse
|
|
|
|
.search(queryWithOptions)
|
2020-07-27 16:04:12 -04:00
|
|
|
.then((data: Array<{ name: string, claimId: string }>) => {
|
|
|
|
const uris = [];
|
|
|
|
const actions = [];
|
|
|
|
|
2021-03-19 23:04:12 +08:00
|
|
|
data.forEach((result) => {
|
2020-07-27 16:04:12 -04:00
|
|
|
if (result) {
|
|
|
|
const { name, claimId } = result;
|
|
|
|
const urlObj: LbryUrlObj = {};
|
|
|
|
|
|
|
|
if (name.startsWith('@')) {
|
|
|
|
urlObj.channelName = name;
|
|
|
|
urlObj.channelClaimId = claimId;
|
|
|
|
} else {
|
|
|
|
urlObj.streamName = name;
|
|
|
|
urlObj.streamClaimId = claimId;
|
|
|
|
}
|
|
|
|
|
|
|
|
const url = buildURI(urlObj);
|
2021-01-28 14:03:37 -05:00
|
|
|
if (isURIValid(url)) {
|
|
|
|
uris.push(url);
|
|
|
|
}
|
2020-07-27 16:04:12 -04:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2020-08-12 13:02:19 -04:00
|
|
|
actions.push(doResolveUris(uris));
|
|
|
|
|
2020-07-27 16:04:12 -04:00
|
|
|
actions.push({
|
|
|
|
type: ACTIONS.SEARCH_SUCCESS,
|
|
|
|
data: {
|
|
|
|
query: queryWithOptions,
|
2021-04-07 18:50:15 +08:00
|
|
|
from: from,
|
2021-03-26 16:33:30 +08:00
|
|
|
size: size,
|
2020-07-27 16:04:12 -04:00
|
|
|
uris,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
dispatch(batchActions(...actions));
|
|
|
|
})
|
2021-03-19 23:04:12 +08:00
|
|
|
.catch((e) => {
|
2020-07-27 16:04:12 -04:00
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.SEARCH_FAIL,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
export const doUpdateSearchOptions = (newOptions: SearchOptions, additionalOptions: SearchOptions) => (
|
|
|
|
dispatch: Dispatch,
|
|
|
|
getState: GetState
|
|
|
|
) => {
|
|
|
|
const state = getState();
|
|
|
|
const searchValue = selectSearchValue(state);
|
|
|
|
|
|
|
|
dispatch({
|
|
|
|
type: ACTIONS.UPDATE_SEARCH_OPTIONS,
|
|
|
|
data: newOptions,
|
|
|
|
});
|
|
|
|
|
|
|
|
if (searchValue) {
|
|
|
|
// After updating, perform a search with the new options
|
|
|
|
dispatch(doSearch(searchValue, additionalOptions));
|
|
|
|
}
|
|
|
|
};
|
2020-12-03 12:29:47 -05:00
|
|
|
|
2021-03-19 23:04:12 +08:00
|
|
|
export const doFetchRecommendedContent = (uri: string, mature: boolean) => (dispatch: Dispatch, getState: GetState) => {
|
|
|
|
const state = getState();
|
|
|
|
const claim = makeSelectClaimForUri(uri)(state);
|
|
|
|
|
|
|
|
if (claim && claim.value && claim.claim_id) {
|
|
|
|
const options: SearchOptions = { size: 20, related_to: claim.claim_id, isBackgroundSearch: true };
|
|
|
|
if (!mature) {
|
|
|
|
options['nsfw'] = false;
|
|
|
|
}
|
|
|
|
const { title } = claim.value;
|
|
|
|
if (title && options) {
|
|
|
|
dispatch(doSearch(title, options));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-12-03 12:29:47 -05:00
|
|
|
export { lighthouse };
|