( function () {

	/**
	 * @typedef {Object} mw.Rest.Options
	 * @property {Object} [ajax={ url: mw.util.wikiScript( 'rest' ), timeout: 30 * 1000 }] Default
	 *  options for [ajax()]{@link mw.Rest#ajax} calls. Can be overridden by passing `options` to
	 *  the {@link mw.Rest} constructor.
	 */

	/**
	 * @type {mw.Rest.Options}
	 * @private
	 */
	var defaultOptions = {
		ajax: {
			url: mw.util.wikiScript( 'rest' ),
			timeout: 30 * 1000 // 30 seconds
		}
	};

	/**
	 * Lower cases the key names in the provided object.
	 *
	 * @param {Object} headers
	 * @return {Object}
	 * @private
	 */
	function objectKeysToLowerCase( headers ) {
		return Object.keys( headers || {} ).reduce( function ( updatedHeaders, key ) {
			updatedHeaders[ key.toLowerCase() ] = headers[ key ];
			return updatedHeaders;
		}, {} );
	}

	/**
	 * @classdesc Interact with the REST API. mw.Rest is a client library
	 * for the [REST API](https://www.mediawiki.org/wiki/Special:MyLanguage/API:REST_API).
	 * An mw.Rest object represents the REST API of a MediaWiki site.
	 * For the action API, see {@link mw.Api}.
	 *
	 * @example
	 * var api = new mw.Rest();
	 * api.get( '/v1/page/Main_Page/html' )
	 * .then( function ( data ) {
	 *     console.log( data );
	 * } );
	 *
	 * api.post( '/v1/page/Main_Page', {
	 *      token: 'anon_token',
	 *      source: 'Lörem Ipsüm',
	 *      comment: 'tästing',
	 *      title: 'My_Page'
	 * }, {
	 *     'authorization': 'token'
	 * } )
	 * .then( function ( data ) {
	 *     console.log( data );
	 * } );
	 *
	 * @constructor
	 * @description Create an instance of `mw.Rest`.
	 * @param {mw.Rest.Options} [options] See {@link mw.Rest.Options}
	 */
	mw.Rest = function ( options ) {
		var defaults = $.extend( {}, options );
		defaults.ajax = $.extend( {}, defaultOptions.ajax, defaults.ajax );

		this.url = defaults.ajax.url;
		delete defaults.ajax.url;

		this.defaults = defaults;
		this.requests = [];
	};

	mw.Rest.prototype = {
		/**
		 * Abort all unfinished requests issued by this Api object.
		 *
		 * @method
		 */
		abort: function () {
			this.requests.forEach( function ( request ) {
				if ( request ) {
					request.abort();
				}
			} );
		},

		/**
		 * Perform REST API get request.
		 *
		 * @method
		 * @param {string} path
		 * @param {Object} query
		 * @param {Object} [headers]
		 * @return {jQuery.Promise}
		 */
		get: function ( path, query, headers ) {
			return this.ajax( path, {
				type: 'GET',
				data: query,
				headers: headers || {}
			} );
		},

		/**
		 * Perform REST API post request.
		 *
		 * Note: only sending application/json is currently supported.
		 *
		 * @method
		 * @param {string} path
		 * @param {Object} [body]
		 * @param {Object} [headers]
		 * @return {jQuery.Promise}
		 */
		post: function ( path, body, headers ) {
			if ( body === undefined ) {
				body = {};
			}

			headers = objectKeysToLowerCase( headers );
			return this.ajax( path, {
				type: 'POST',
				headers: $.extend( headers, { 'content-type': 'application/json' } ),
				data: JSON.stringify( body )
			} );
		},

		/**
		 * Perform REST API PUT request.
		 *
		 * Note: only sending `application/json` is currently supported.
		 *
		 * @method
		 * @param {string} path
		 * @param {Object} body
		 * @param {Object} [headers]
		 * @return {jQuery.Promise}
		 */
		put: function ( path, body, headers ) {
			headers = objectKeysToLowerCase( headers );
			return this.ajax( path, {
				type: 'PUT',
				headers: $.extend( headers, { 'content-type': 'application/json' } ),
				data: JSON.stringify( body )
			} );
		},

		/**
		 * Perform REST API DELETE request.
		 *
		 * Note: only sending `application/json` is currently supported.
		 *
		 * @method
		 * @param {string} path
		 * @param {Object} body
		 * @param {Object} [headers]
		 * @return {jQuery.Promise}
		 */
		delete: function ( path, body, headers ) {
			headers = objectKeysToLowerCase( headers );
			return this.ajax( path, {
				type: 'DELETE',
				headers: $.extend( headers, { 'content-type': 'application/json' } ),
				data: JSON.stringify( body )
			} );
		},

		/**
		 * Perform the API call.
		 *
		 * @method
		 * @param {string} path
		 * @param {Object} [ajaxOptions]
		 * @return {jQuery.Promise} Done: API response data and the jqXHR object.
		 *  Fail: Error code
		 */
		ajax: function ( path, ajaxOptions ) {
			var self = this,
				apiDeferred = $.Deferred(),
				xhr, requestIndex;

			ajaxOptions = $.extend( {}, this.defaults.ajax, ajaxOptions );
			ajaxOptions.url = this.url + path;

			// Make the AJAX request.
			xhr = $.ajax( ajaxOptions );

			// Save it to make it possible to abort.
			requestIndex = this.requests.length;
			this.requests.push( xhr );
			xhr.always( function () {
				self.requests[ requestIndex ] = null;
			} );

			xhr.then(
				// AJAX success just means "200 OK" response.
				function ( result, textStatus, jqXHR ) {
					apiDeferred.resolve( result, jqXHR );
				},
				// If AJAX fails, reject API call with error code 'http'
				// and details in second argument.
				function ( jqXHR, textStatus, exception ) {
					apiDeferred.reject( 'http', {
						xhr: jqXHR,
						textStatus: textStatus,
						exception: exception
					} );
				}
			);

			// Return the Promise
			return apiDeferred.promise( { abort: xhr.abort } );
		}
	};
}() );