mirror of
https://github.com/meteor/meteor.git
synced 2026-05-02 03:01:46 -04:00
179 lines
6.1 KiB
JavaScript
179 lines
6.1 KiB
JavaScript
/**
|
|
* @summary Perform an outbound HTTP request.
|
|
* @locus Anywhere
|
|
* @param {String} method The [HTTP method](http://en.wikipedia.org/wiki/HTTP_method) to use, such as "`GET`", "`POST`", or "`HEAD`".
|
|
* @param {String} url The URL to retrieve.
|
|
* @param {Object} [options]
|
|
* @param {String} options.content String to use as the HTTP request body.
|
|
* @param {Object} options.data JSON-able object to stringify and use as the HTTP request body. Overwrites `content`.
|
|
* @param {String} options.query Query string to go in the URL. Overwrites any query string in `url`.
|
|
* @param {Object} options.params Dictionary of request parameters to be encoded and placed in the URL (for GETs) or request body (for POSTs). If `content` or `data` is specified, `params` will always be placed in the URL.
|
|
* @param {String} options.auth HTTP basic authentication string of the form `"username:password"`
|
|
* @param {Object} options.headers Dictionary of strings, headers to add to the HTTP request.
|
|
* @param {Number} options.timeout Maximum time in milliseconds to wait for the request before failing. There is no timeout by default.
|
|
* @param {Boolean} options.followRedirects If `true`, transparently follow HTTP redirects. Cannot be set to `false` on the client. Default `true`.
|
|
* @param {Function} [asyncCallback] Optional callback. If passed, the method runs asynchronously, instead of synchronously, and calls asyncCallback. On the client, this callback is required.
|
|
*/
|
|
HTTP.call = function(method, url, options, callback) {
|
|
|
|
////////// Process arguments //////////
|
|
|
|
if (! callback && typeof options === "function") {
|
|
// support (method, url, callback) argument list
|
|
callback = options;
|
|
options = null;
|
|
}
|
|
|
|
options = options || {};
|
|
|
|
if (typeof callback !== "function")
|
|
throw new Error(
|
|
"Can't make a blocking HTTP call from the client; callback required.");
|
|
|
|
method = (method || "").toUpperCase();
|
|
|
|
var headers = {};
|
|
|
|
var content = options.content;
|
|
if (options.data) {
|
|
content = JSON.stringify(options.data);
|
|
headers['Content-Type'] = 'application/json';
|
|
}
|
|
|
|
var params_for_url, params_for_body;
|
|
if (content || method === "GET" || method === "HEAD")
|
|
params_for_url = options.params;
|
|
else
|
|
params_for_body = options.params;
|
|
|
|
url = URL._constructUrl(url, options.query, params_for_url);
|
|
|
|
if (options.followRedirects === false)
|
|
throw new Error("Option followRedirects:false not supported on client.");
|
|
|
|
var username, password;
|
|
if (options.auth) {
|
|
var colonLoc = options.auth.indexOf(':');
|
|
if (colonLoc < 0)
|
|
throw new Error('auth option should be of the form "username:password"');
|
|
username = options.auth.substring(0, colonLoc);
|
|
password = options.auth.substring(colonLoc+1);
|
|
}
|
|
|
|
if (params_for_body) {
|
|
content = URL._encodeParams(params_for_body);
|
|
}
|
|
|
|
_.extend(headers, options.headers || {});
|
|
|
|
////////// Callback wrapping //////////
|
|
|
|
// wrap callback to add a 'response' property on an error, in case
|
|
// we have both (http 4xx/5xx error, which has a response payload)
|
|
callback = (function(callback) {
|
|
return function(error, response) {
|
|
if (error && response)
|
|
error.response = response;
|
|
callback(error, response);
|
|
};
|
|
})(callback);
|
|
|
|
// safety belt: only call the callback once.
|
|
callback = _.once(callback);
|
|
|
|
|
|
////////// Kickoff! //////////
|
|
|
|
// from this point on, errors are because of something remote, not
|
|
// something we should check in advance. Turn exceptions into error
|
|
// results.
|
|
try {
|
|
// setup XHR object
|
|
var xhr;
|
|
if (typeof XMLHttpRequest !== "undefined")
|
|
xhr = new XMLHttpRequest();
|
|
else if (typeof ActiveXObject !== "undefined")
|
|
xhr = new ActiveXObject("Microsoft.XMLHttp"); // IE6
|
|
else
|
|
throw new Error("Can't create XMLHttpRequest"); // ???
|
|
|
|
xhr.open(method, url, true, username, password);
|
|
|
|
for (var k in headers)
|
|
xhr.setRequestHeader(k, headers[k]);
|
|
|
|
|
|
// setup timeout
|
|
var timed_out = false;
|
|
var timer;
|
|
if (options.timeout) {
|
|
timer = Meteor.setTimeout(function() {
|
|
timed_out = true;
|
|
xhr.abort();
|
|
}, options.timeout);
|
|
};
|
|
|
|
// callback on complete
|
|
xhr.onreadystatechange = function(evt) {
|
|
if (xhr.readyState === 4) { // COMPLETE
|
|
if (timer)
|
|
Meteor.clearTimeout(timer);
|
|
|
|
if (timed_out) {
|
|
callback(new Error("timeout"));
|
|
} else if (! xhr.status) {
|
|
// no HTTP response
|
|
callback(new Error("network"));
|
|
} else {
|
|
|
|
var response = {};
|
|
response.statusCode = xhr.status;
|
|
response.content = xhr.responseText;
|
|
|
|
response.headers = {};
|
|
var header_str = xhr.getAllResponseHeaders();
|
|
|
|
// https://github.com/meteor/meteor/issues/553
|
|
//
|
|
// In Firefox there is a weird issue, sometimes
|
|
// getAllResponseHeaders returns the empty string, but
|
|
// getResponseHeader returns correct results. Possibly this
|
|
// issue:
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=608735
|
|
//
|
|
// If this happens we can't get a full list of headers, but
|
|
// at least get content-type so our JSON decoding happens
|
|
// correctly. In theory, we could try and rescue more header
|
|
// values with a list of common headers, but content-type is
|
|
// the only vital one for now.
|
|
if ("" === header_str && xhr.getResponseHeader("content-type"))
|
|
header_str =
|
|
"content-type: " + xhr.getResponseHeader("content-type");
|
|
|
|
var headers_raw = header_str.split(/\r?\n/);
|
|
_.each(headers_raw, function (h) {
|
|
var m = /^(.*?):(?:\s+)(.*)$/.exec(h);
|
|
if (m && m.length === 3)
|
|
response.headers[m[1].toLowerCase()] = m[2];
|
|
});
|
|
|
|
populateData(response);
|
|
|
|
var error = null;
|
|
if (response.statusCode >= 400)
|
|
error = makeErrorByStatus(response.statusCode, response.content);
|
|
|
|
callback(error, response);
|
|
}
|
|
}
|
|
};
|
|
|
|
// send it on its way
|
|
xhr.send(content);
|
|
|
|
} catch (err) {
|
|
callback(err);
|
|
}
|
|
|
|
};
|