2023-08-02 22:56:08 +00:00
|
|
|
/** @module mwActionApiSearchClient */
|
|
|
|
|
|
|
|
const fetchJson = require( '../fetch.js' );
|
|
|
|
const urlGenerator = require( '../urlGenerator.js' );
|
|
|
|
|
2024-09-27 07:15:55 +00:00
|
|
|
// Based on mediawiki.searchSuggest
|
|
|
|
// eslint-disable-next-line array-callback-return
|
|
|
|
const searchNS = Object.entries( mw.config.get( 'wgFormattedNamespaces' ) ).map( ( [ nsID ] ) => {
|
|
|
|
if ( nsID >= 0 && mw.user.options.get( 'searchNs' + nsID ) ) {
|
|
|
|
// Cast string key to number
|
|
|
|
return Number( nsID );
|
|
|
|
}
|
|
|
|
} ).filter( ( item ) => item !== undefined );
|
|
|
|
|
2023-08-02 22:56:08 +00:00
|
|
|
/**
|
|
|
|
* @typedef {Object} ActionResponse
|
|
|
|
* @property {ActionQuery[]} query
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {Object} ActionQuery
|
|
|
|
* @property {ActionRedirects[] | null} redirects
|
|
|
|
* @property {ActionResult[]} pages
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {Object} ActionRedirects
|
|
|
|
* @property {string} from
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {Object} ActionResult
|
|
|
|
* @property {number} pageid
|
|
|
|
* @property {number} index
|
|
|
|
* @property {string} title
|
|
|
|
* @property {ActionThumbnail | null} [thumbnail]
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {Object} ActionThumbnail
|
|
|
|
* @property {string} source
|
|
|
|
* @property {number | null} [width]
|
|
|
|
* @property {number | null} [height]
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {Object} SearchResponse
|
|
|
|
* @property {string} query
|
|
|
|
* @property {SearchResult[]} results
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {MwMap} config
|
|
|
|
* @param {string} query
|
|
|
|
* @param {Object} response
|
|
|
|
* @param {boolean} showDescription
|
|
|
|
* @return {SearchResponse}
|
|
|
|
*/
|
|
|
|
function adaptApiResponse( config, query, response, showDescription ) {
|
|
|
|
const urlGeneratorInstance = urlGenerator( config );
|
|
|
|
const getDescription = ( page ) => {
|
|
|
|
switch ( config.wgCitizenSearchDescriptionSource ) {
|
|
|
|
case 'wikidata':
|
|
|
|
return page?.description;
|
|
|
|
case 'textextracts':
|
|
|
|
return page?.extract;
|
|
|
|
case 'pagedescription':
|
2023-08-24 22:30:03 +00:00
|
|
|
return page?.pageprops?.description?.slice( 0, 100 );
|
2023-08-02 22:56:08 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-08-03 01:28:39 +00:00
|
|
|
// Early exit with there are no query
|
|
|
|
if ( !response.query ) {
|
|
|
|
return { query, results: {} };
|
|
|
|
}
|
|
|
|
|
2023-08-02 22:56:08 +00:00
|
|
|
response = response.query;
|
|
|
|
|
|
|
|
// Merge redirects array into pages array if avaliable
|
|
|
|
// So the from key can be used for matched title
|
|
|
|
if ( response.redirects ) {
|
2023-08-04 01:20:47 +00:00
|
|
|
const pageCount = response.pages.length;
|
|
|
|
|
2023-08-02 22:56:08 +00:00
|
|
|
response.pages = Object.values(
|
|
|
|
[ ...response.redirects, ...response.pages ].reduce( ( acc, curr ) => {
|
|
|
|
const index = curr.index;
|
|
|
|
acc[ index ] = { ...acc[ index ], ...curr };
|
|
|
|
return acc;
|
|
|
|
}, [] )
|
|
|
|
);
|
2023-08-04 01:20:47 +00:00
|
|
|
|
|
|
|
// Sometimes there can be multiple redirect object for the same page, only take the one with lower index
|
|
|
|
if ( response.pages.length !== pageCount ) {
|
2024-06-07 19:01:54 +00:00
|
|
|
response.pages = response.pages.filter( ( obj ) => Object.prototype.hasOwnProperty.call( obj, 'title' ) );
|
2023-08-04 01:20:47 +00:00
|
|
|
}
|
2023-08-02 22:56:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Sort pages by index key instead of page id
|
|
|
|
response.pages.sort( ( a, b ) => a.index - b.index );
|
|
|
|
|
|
|
|
return {
|
|
|
|
query,
|
|
|
|
results: response.pages.map( ( page ) => {
|
|
|
|
const thumbnail = page.thumbnail;
|
|
|
|
return {
|
|
|
|
id: page.pageid,
|
2023-08-04 01:01:21 +00:00
|
|
|
label: page.from || page.title,
|
2023-08-02 22:56:08 +00:00
|
|
|
key: page.title.replace( / /g, '_' ),
|
|
|
|
title: page.title,
|
|
|
|
description: showDescription ? getDescription( page ) : undefined,
|
|
|
|
url: urlGeneratorInstance.generateUrl( page ),
|
|
|
|
thumbnail: thumbnail ? {
|
|
|
|
url: thumbnail.source,
|
|
|
|
width: thumbnail.width ?? undefined,
|
|
|
|
height: thumbnail.height ?? undefined
|
|
|
|
} : undefined
|
|
|
|
};
|
|
|
|
} )
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {Object} AbortableSearchFetch
|
|
|
|
* @property {Promise<SearchResponse>} fetch
|
|
|
|
* @property {Function} abort
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @callback fetchByTitle
|
|
|
|
* @param {string} query The search term.
|
|
|
|
* @param {number} [limit] Maximum number of results.
|
|
|
|
* @return {AbortableSearchFetch}
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @callback loadMore
|
|
|
|
* @param {string} query The search term.
|
|
|
|
* @param {number} offset The number of search results that were already loaded.
|
|
|
|
* @param {number} [limit] How many further search results to load (at most).
|
|
|
|
* @return {AbortableSearchFetch}
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {Object} SearchClient
|
|
|
|
* @property {fetchByTitle} fetchByTitle
|
|
|
|
* @property {loadMore} [loadMore]
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {MwMap} config
|
|
|
|
* @return {SearchClient}
|
|
|
|
*/
|
|
|
|
function mwActionApiSearchClient( config ) {
|
|
|
|
return {
|
|
|
|
/**
|
|
|
|
* @type {fetchByTitle}
|
|
|
|
*/
|
|
|
|
fetchByTitle: ( q, limit = config.wgCitizenMaxSearchResults, showDescription = true ) => {
|
|
|
|
const cacheExpiry = config.wgSearchSuggestCacheExpiry;
|
|
|
|
const descriptionSource = config.wgCitizenSearchDescriptionSource;
|
|
|
|
|
|
|
|
const searchApiUrl = config.wgScriptPath + '/api.php';
|
2024-09-27 07:15:55 +00:00
|
|
|
|
2023-08-02 22:56:08 +00:00
|
|
|
const params = {
|
|
|
|
format: 'json',
|
|
|
|
formatversion: '2',
|
|
|
|
action: 'query',
|
|
|
|
smaxage: cacheExpiry,
|
|
|
|
maxage: cacheExpiry,
|
2024-09-27 07:15:55 +00:00
|
|
|
namespace: searchNS,
|
2023-08-02 22:56:08 +00:00
|
|
|
generator: 'prefixsearch',
|
2023-08-30 14:32:32 +00:00
|
|
|
gpssearch: q,
|
|
|
|
gpslimit: limit.toString(),
|
2023-08-02 22:56:08 +00:00
|
|
|
redirects: '',
|
2023-08-30 14:32:32 +00:00
|
|
|
prop: 'pageprops|pageimages',
|
2023-08-02 22:56:08 +00:00
|
|
|
ppprop: 'displaytitle',
|
|
|
|
piprop: 'thumbnail',
|
2023-08-30 14:32:32 +00:00
|
|
|
pilicense: 'any',
|
2023-08-02 22:56:08 +00:00
|
|
|
pithumbsize: 200,
|
2023-08-30 14:32:32 +00:00
|
|
|
pilimit: limit.toString()
|
2023-08-02 22:56:08 +00:00
|
|
|
};
|
|
|
|
switch ( descriptionSource ) {
|
|
|
|
case 'wikidata':
|
|
|
|
params.prop += '|description';
|
|
|
|
params.descprefersource = 'local';
|
|
|
|
break;
|
|
|
|
case 'textextracts':
|
|
|
|
params.prop += '|extracts';
|
2023-08-24 22:30:03 +00:00
|
|
|
params.exchars = '100';
|
2023-08-02 22:56:08 +00:00
|
|
|
params.exintro = '1';
|
|
|
|
params.exlimit = limit.toString();
|
|
|
|
params.explaintext = '1';
|
|
|
|
break;
|
|
|
|
case 'pagedescription':
|
|
|
|
params.prop += '|pageprops';
|
|
|
|
params.ppprop = 'description';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
const search = new URLSearchParams( params );
|
2024-01-27 02:51:04 +00:00
|
|
|
const url = `${ searchApiUrl }?${ search.toString() }`;
|
2023-08-02 22:56:08 +00:00
|
|
|
const result = fetchJson( url, {
|
|
|
|
headers: {
|
|
|
|
accept: 'application/json'
|
|
|
|
}
|
|
|
|
} );
|
|
|
|
const searchResponsePromise = result.fetch
|
2024-06-07 19:01:54 +00:00
|
|
|
.then( ( /** @type {ActionResponse} */ res ) => adaptApiResponse( config, q, res, showDescription ) );
|
2023-08-02 22:56:08 +00:00
|
|
|
return {
|
|
|
|
abort: result.abort,
|
|
|
|
fetch: searchResponsePromise
|
|
|
|
};
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = mwActionApiSearchClient;
|