mediawiki-extensions-Multim.../resources/mmv/provider/mmv.provider.Api.js

167 lines
5.6 KiB
JavaScript
Raw Normal View History

/*
* This file is part of the MediaWiki extension MultimediaViewer.
*
* MultimediaViewer is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 2 of the License, or
* (at your option) any later version.
*
* MultimediaViewer is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with MultimediaViewer. If not, see <http://www.gnu.org/licenses/>.
*/
( function () {
/**
* Base class for API-based data providers.
*
* @abstract
*/
class Api {
/**
* @param {mw.Api} api
* @param {Object} [options]
* @cfg {number} [maxage] cache expiration time, in seconds
* Will be used for both client-side cache (maxage) and reverse proxies (s-maxage)
*/
constructor( api, options ) {
/**
* API object for dependency injection.
*
* @property {mw.Api}
*/
this.api = api;
/**
* Options object; the exact format and meaning is unspecified and could be different
* from subclass to subclass.
*
* @property {Object}
*/
this.options = options || {};
/**
* API call cache.
*
* @property {Object.<string, jQuery.Promise>} cache
* @protected
*/
this.cache = {};
}
/**
* Wraps a caching layer around a function returning a promise; if getCachedPromise has been
* called with the same key already, it will return the previous result.
*
* Since it is the promise and not the API response that gets cached, this method can ensure
* that there are no race conditions and multiple calls to the same resource: even if the
* request is still in progress, separate calls (with the same key) to getCachedPromise will
* share on the same promise object.
* The promise is cached even if it is rejected, so if the API request fails, all later calls
* to getCachedPromise will fail immediately without retrying the request.
*
* @param {string} key cache key
* @param {function(): jQuery.Promise} getPromise a function to get the promise on cache miss
* @return {jQuery.Promise}
*/
getCachedPromise( key, getPromise ) {
if ( !this.cache[ key ] ) {
this.cache[ key ] = getPromise();
this.cache[ key ].fail( ( error ) => {
// constructor.name is usually not reliable in inherited classes, but OOjs fixes that
mw.log( `${this.constructor.name} provider failed to load: `, error );
} );
}
return this.cache[ key ];
}
/**
* Calls mw.Api.get, with caching parameters.
*
* @param {Object} params Parameters to the API query.
* @param {Object} [ajaxOptions] ajaxOptions argument for mw.Api.get
* @param {number|null} [maxage] Cache the call for this many seconds.
* Sets both the maxage (client-side) and smaxage (proxy-side) caching parameters.
* Null means no caching. Undefined means the default caching period is used.
* @return {jQuery.Promise} the return value from mw.Api.get
*/
apiGetWithMaxAge( params, ajaxOptions, maxage ) {
if ( maxage === undefined ) {
maxage = this.options.maxage;
}
if ( maxage ) {
params.maxage = params.smaxage = maxage;
}
return this.api.get( params, ajaxOptions );
}
/**
* Pulls an error message out of an API response.
*
* @param {Object} data
* @param {Object} data.error
* @param {string} data.error.code
* @param {string} data.error.info
* @return {string} From data.error.code + ': ' + data.error.info, or 'unknown error'
*/
getErrorMessage( data ) {
const errorCode = data.error && data.error.code;
let errorMessage = data.error && data.error.info || 'unknown error';
if ( errorCode ) {
errorMessage = `${errorCode}: ${errorMessage}`;
}
return errorMessage;
}
/**
* Returns a promise with the specified field from the API result.
* This is intended to be used as a .then() callback for action=query APIs.
*
* @param {string} field
* @param {Object} data
* @return {jQuery.Promise} when successful, the first argument will be the field data,
* when unsuccessful, it will be an error message. The second argument is always
* the full API response.
*/
getQueryField( field, data ) {
if ( data && data.query && data.query[ field ] ) {
return $.Deferred().resolve( data.query[ field ], data );
} else {
return $.Deferred().reject( this.getErrorMessage( data ), data );
}
}
/**
* Returns a promise with the specified page from the API result.
* This is intended to be used as a .then() callback for action=query&prop=(...) APIs.
*
* @param {Object} data
* @return {jQuery.Promise} when successful, the first argument will be the page data,
* when unsuccessful, it will be an error message. The second argument is always
* the full API response.
*/
getQueryPage( data ) {
if ( data &&
data.query &&
Array.isArray( data.query.pages ) &&
data.query.pages.length === 1
) {
// pages is an array and the first element is always the requested title
return $.Deferred().resolve( data.query.pages[ 0 ], data );
}
// If we got to this point either the pages array is missing completely, or the
// first element is not the requested page. Neither is supposed to happen
// (if the page simply did not exist, there would still be a record for it).
return $.Deferred().reject( this.getErrorMessage( data ), data );
}
}
module.exports = Api;
}() );