mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2025-01-02 00:05:11 +00:00
b9b966de6e
See I9addf43b2857de7bcac. Change-Id: Ic1998ab6f7dedf6946d7ae63a985a76031f8eecf
198 lines
5.7 KiB
JavaScript
198 lines
5.7 KiB
JavaScript
/*!
|
|
* VisualEditor MediaWiki Initialization ApiResponseCache class.
|
|
*
|
|
* @copyright See AUTHORS.txt
|
|
* @license The MIT License (MIT); see LICENSE.txt
|
|
*/
|
|
|
|
/**
|
|
* MediaWiki API batch queue.
|
|
*
|
|
* Used to queue up lists of items centrally to get information about in batches
|
|
* of requests.
|
|
*
|
|
* @class
|
|
* @extends OO.EventEmitter
|
|
* @constructor
|
|
* @param {mw.Api} [api] API object to use. Defaults to new mw.Api()
|
|
*/
|
|
ve.init.mw.ApiResponseCache = function VeInitMwApiResponseCache( api ) {
|
|
// Mixin constructor
|
|
OO.EventEmitter.call( this );
|
|
|
|
this.api = api || new mw.Api();
|
|
|
|
// Keys are titles, values are deferreds
|
|
this.deferreds = {};
|
|
|
|
// Keys are page names, values are link data objects
|
|
// This is kept for synchronous retrieval of cached values via #getCached
|
|
this.cacheValues = {};
|
|
|
|
// Array of page titles queued to be looked up
|
|
this.queue = [];
|
|
|
|
this.schedule = ve.debounce( this.processQueue.bind( this ), 0 );
|
|
};
|
|
|
|
/* Inheritance */
|
|
|
|
OO.mixinClass( ve.init.mw.ApiResponseCache, OO.EventEmitter );
|
|
|
|
/* Static methods */
|
|
|
|
/**
|
|
* Process each page in the response of an API request
|
|
*
|
|
* @abstract
|
|
* @static
|
|
* @param {Object} page
|
|
* @return {Object|undefined} Any relevant info that we want to cache and return.
|
|
*/
|
|
ve.init.mw.ApiResponseCache.static.processPage = null;
|
|
|
|
/**
|
|
* @param {string} title
|
|
* @return {string}
|
|
*/
|
|
ve.init.mw.ApiResponseCache.static.normalizeTitle = function ( title ) {
|
|
const titleObj = mw.Title.newFromText( title );
|
|
if ( !titleObj ) {
|
|
return title;
|
|
}
|
|
return titleObj.getPrefixedText();
|
|
};
|
|
|
|
/* Methods */
|
|
|
|
/**
|
|
* Look up data about a title. If the data about this title is already in the cache, this
|
|
* returns an already-resolved promise. Otherwise, it returns a pending promise and schedules
|
|
* an request to retrieve the data.
|
|
*
|
|
* @param {string} title
|
|
* @return {jQuery.Promise} Promise that will be resolved with the data once it's available
|
|
*/
|
|
ve.init.mw.ApiResponseCache.prototype.get = function ( title ) {
|
|
if ( typeof title !== 'string' ) {
|
|
// Don't bother letting things like undefined or null make it all the way through,
|
|
// just reject them here. Otherwise they'll cause problems or exceptions at random
|
|
// other points in this file.
|
|
return ve.createDeferred().reject().promise();
|
|
}
|
|
title = this.constructor.static.normalizeTitle( title );
|
|
if ( !Object.prototype.hasOwnProperty.call( this.deferreds, title ) ) {
|
|
this.deferreds[ title ] = ve.createDeferred();
|
|
this.queue.push( title );
|
|
this.schedule();
|
|
}
|
|
return this.deferreds[ title ].promise();
|
|
};
|
|
|
|
/**
|
|
* Look up data about a page in the cache. If the data about this page is already in the cache,
|
|
* this returns that data. Otherwise, it returns undefined.
|
|
*
|
|
* @param {string} name Normalized page title
|
|
* @return {Object|undefined} Cache data for this name.
|
|
*/
|
|
ve.init.mw.ApiResponseCache.prototype.getCached = function ( name ) {
|
|
if ( Object.prototype.hasOwnProperty.call( this.cacheValues, name ) ) {
|
|
return this.cacheValues[ name ];
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Fired when a new entry is added to the cache.
|
|
*
|
|
* @event ve.init.mw.ApiResponseCache#add
|
|
* @param {Object} entries Cache entries that were added. Object mapping names to data objects.
|
|
*/
|
|
|
|
/**
|
|
* Add entries to the cache. Does not overwrite already-set entries.
|
|
*
|
|
* @param {Object} entries Object keyed by page title, with the values being data objects
|
|
* @fires ve.init.mw.ApiResponseCache#add
|
|
*/
|
|
ve.init.mw.ApiResponseCache.prototype.set = function ( entries ) {
|
|
for ( const name in entries ) {
|
|
if ( !Object.prototype.hasOwnProperty.call( this.deferreds, name ) ) {
|
|
this.deferreds[ name ] = ve.createDeferred();
|
|
}
|
|
if ( this.deferreds[ name ].state() === 'pending' ) {
|
|
this.deferreds[ name ].resolve( entries[ name ] );
|
|
this.cacheValues[ name ] = entries[ name ];
|
|
}
|
|
}
|
|
this.emit( 'add', Object.keys( entries ) );
|
|
};
|
|
|
|
/**
|
|
* Get an API request promise to deal with a list of titles
|
|
*
|
|
* @abstract
|
|
* @param subqueue
|
|
* @return {jQuery.Promise}
|
|
*/
|
|
ve.init.mw.ApiResponseCache.prototype.getRequestPromise = null;
|
|
|
|
/**
|
|
* Perform any scheduled API requests.
|
|
*
|
|
* @private
|
|
* @fires ve.init.mw.ApiResponseCache#add
|
|
*/
|
|
ve.init.mw.ApiResponseCache.prototype.processQueue = function () {
|
|
const rejectSubqueue = ( rejectQueue ) => {
|
|
for ( let i = 0, len = rejectQueue.length; i < len; i++ ) {
|
|
this.deferreds[ rejectQueue[ i ] ].reject();
|
|
}
|
|
};
|
|
|
|
const processResult = ( data ) => {
|
|
const pages = ( data.query && data.query.pages ) || data.pages,
|
|
processed = {};
|
|
|
|
const mappedTitles = [];
|
|
[ 'redirects', 'normalized', 'converted' ].forEach( ( map ) => {
|
|
ve.batchPush( mappedTitles, ( data.query && data.query[ map ] ) || [] );
|
|
} );
|
|
|
|
if ( pages ) {
|
|
let page, processedPage;
|
|
for ( const pageid in pages ) {
|
|
page = pages[ pageid ];
|
|
processedPage = this.constructor.static.processPage( page );
|
|
if ( processedPage !== undefined ) {
|
|
processed[ page.title ] = processedPage;
|
|
}
|
|
}
|
|
for ( let i = 0; i < mappedTitles.length; i++ ) {
|
|
// Locate the title in mapped titles, if any.
|
|
if ( mappedTitles[ i ].to === page.title ) {
|
|
const from = mappedTitles[ i ].fromencoded === '' ?
|
|
decodeURIComponent( mappedTitles[ i ].from ) :
|
|
mappedTitles[ i ].from;
|
|
processed[ from ] = processedPage;
|
|
break;
|
|
}
|
|
}
|
|
this.set( processed );
|
|
}
|
|
};
|
|
|
|
const queue = this.queue;
|
|
this.queue = [];
|
|
while ( queue.length ) {
|
|
const subqueue = queue.splice( 0, 50 ).map( this.constructor.static.normalizeTitle );
|
|
this.getRequestPromise( subqueue )
|
|
.then( processResult )
|
|
|
|
// Reject everything in subqueue; this will only reject the ones
|
|
// that weren't already resolved above, because .reject() on an
|
|
// already resolved Deferred is a no-op.
|
|
.then( rejectSubqueue.bind( null, subqueue ) );
|
|
}
|
|
};
|