XMLHttpRequest
XMLHttpRequest (XHR) is a built-in browser object that can be used to interact with server. XHR allows you to update data without having to reload a web page.
XMLHttpRequest
is a built-in browser object that allows to make HTTP requests in JavaScript.
Despite having the word “XML” in its name, it can operate on any data, not only in XML format. We can upload/download files, track progress and much more.
Right now, there’s another, more modern method fetch
, that somewhat deprecates XMLHttpRequest
.
In modern web-development XMLHttpRequest
is used for three reasons:
Historical reasons: we need to support existing scripts with
XMLHttpRequest
.We need to support old browsers, and don’t want polyfills (e.g. to keep scripts tiny).
We need something that
fetch
can’t do yet, e.g. to track upload progress.
Does that sound familiar? If yes, then all right, go on with XMLHttpRequest
. Otherwise, please head on to Fetch.
XMLHttpRequest has two modes of operation: synchronous and asynchronous.
Let’s see the asynchronous first, as it’s used in the majority of cases.
To do the request, we need 3 steps:
Create
XMLHttpRequest
:The constructor has no arguments.
Initialize it, usually right after
new XMLHttpRequest
:This method specifies the main parameters of the request:
method
– HTTP-method. Usually"GET"
or"POST"
.URL
– the URL to request, a string, can be URL object.async
– if explicitly set tofalse
, then the request is synchronous, we’ll cover that a bit later.user
,password
– login and password for basic HTTP auth (if required).
Please note that
open
call, contrary to its name, does not open the connection. It only configures the request, but the network activity only starts with the call ofsend
.Send it out.
This method opens the connection and sends the request to server. The optional
body
parameter contains the request body.Some request methods like
GET
do not have a body. And some of them likePOST
usebody
to send the data to the server. We’ll see examples of that later.Listen to
xhr
events for response.These three events are the most widely used:
load
– when the request is complete (even if HTTP status is like 400 or 500), and the response is fully downloaded.error
– when the request couldn’t be made, e.g. network down or invalid URL.progress
– triggers periodically while the response is being downloaded, reports how much has been downloaded.
Here’s a full example. The code below loads the URL at /article/xmlhttprequest/example/load
from the server and prints the progress:
Once the server has responded, we can receive the result in the following xhr
properties:
status
HTTP status code (a number): 200
, 404
, 403
and so on, can be 0
in case of a non-HTTP failure.statusText
HTTP status message (a string): usually OK
for 200
, Not Found
for 404
, Forbidden
for 403
and so on.response
(old scripts may use responseText
)The server response body.
We can also specify a timeout using the corresponding property:
If the request does not succeed within the given time, it gets canceled and timeout
event triggers.
URL search parameters
To add parameters to URL, like ?name=value
, and ensure the proper encoding, we can use URL object:
We can use xhr.responseType
property to set the response format:
""
(default) – get as string,"text"
– get as string,"arraybuffer"
– get asArrayBuffer
(for binary data, see chapter ArrayBuffer, binary arrays),"blob"
– get asBlob
(for binary data, see chapter Blob),"document"
– get as XML document (can use XPath and other XML methods) or HTML document (based on the MIME type of the received data),"json"
– get as JSON (parsed automatically).
For example, let’s get the response as JSON:
Please note:
In the old scripts you may also find xhr.responseText
and even xhr.responseXML
properties.
They exist for historical reasons, to get either a string or XML document. Nowadays, we should set the format in xhr.responseType
and get xhr.response
as demonstrated above.
XMLHttpRequest
changes between states as it progresses. The current state is accessible as xhr.readyState
.
All states, as in the specification:
An XMLHttpRequest
object travels them in the order 0
→ 1
→ 2
→ 3
→ … → 3
→ 4
. State 3
repeats every time a data packet is received over the network.
We can track them using readystatechange
event:
You can find readystatechange
listeners in really old code, it’s there for historical reasons, as there was a time when there were no load
and other events. Nowadays, load/error/progress
handlers deprecate it.
We can terminate the request at any time. The call to xhr.abort()
does that:
That triggers abort
event, and xhr.status
becomes 0
.
If in the open
method the third parameter async
is set to false
, the request is made synchronously.
In other words, JavaScript execution pauses at send()
and resumes when the response is received. Somewhat like alert
or prompt
commands.
Here’s the rewritten example, the 3rd parameter of open
is false
:
It might look good, but synchronous calls are used rarely, because they block in-page JavaScript till the loading is complete. In some browsers it becomes impossible to scroll. If a synchronous call takes too much time, the browser may suggest to close the “hanging” webpage.
Many advanced capabilities of XMLHttpRequest
, like requesting from another domain or specifying a timeout, are unavailable for synchronous requests. Also, as you can see, no progress indication.
Because of all that, synchronous requests are used very sparingly, almost never. We won’t talk about them any more.
XMLHttpRequest
allows both to send custom headers and read headers from the response.
There are 3 methods for HTTP-headers:
setRequestHeader(name, value)
Sets the request header with the given name
and value
.
For instance:
Headers limitations
Several headers are managed exclusively by the browser, e.g. Referer
and Host
. The full list is in the specification.
XMLHttpRequest
is not allowed to change them, for the sake of user safety and correctness of the request.
Can’t remove a header
Another peculiarity of XMLHttpRequest
is that one can’t undo setRequestHeader
.
Once the header is set, it’s set. Additional calls add information to the header, don’t overwrite it.
For instance:
getResponseHeader(name)
Gets the response header with the given name
(except Set-Cookie
and Set-Cookie2
).
For instance:
getAllResponseHeaders()
Returns all response headers, except Set-Cookie
and Set-Cookie2
.
Headers are returned as a single line, e.g.:
The line break between headers is always "\r\n"
(doesn’t depend on OS), so we can easily split it into individual headers. The separator between the name and the value is always a colon followed by a space ": "
. That’s fixed in the specification.
So, if we want to get an object with name/value pairs, we need to throw in a bit JS.
Like this (assuming that if two headers have the same name, then the latter one overwrites the former one):
To make a POST request, we can use the built-in FormData object.
The syntax:
We create it, optionally fill from a form, append
more fields if needed, and then:
xhr.open('POST', ...)
– usePOST
method.xhr.send(formData)
to submit the form to the server.
For instance:
The form is sent with multipart/form-data
encoding.
Or, if we like JSON more, then JSON.stringify
and send as a string.
Just don’t forget to set the header Content-Type: application/json
, many server-side frameworks automatically decode JSON with it:
The .send(body)
method is pretty omnivore. It can send almost any body
, including Blob
and BufferSource
objects.
The progress
event triggers only on the downloading stage.
That is: if we POST
something, XMLHttpRequest
first uploads our data (the request body), then downloads the response.
If we’re uploading something big, then we’re surely more interested in tracking the upload progress. But xhr.onprogress
doesn’t help here.
There’s another object, without methods, exclusively to track upload events: xhr.upload
.
It generates events, similar to xhr
, but xhr.upload
triggers them solely on uploading:
loadstart
– upload started.progress
– triggers periodically during the upload.abort
– upload aborted.error
– non-HTTP error.load
– upload finished successfully.timeout
– upload timed out (iftimeout
property is set).loadend
– upload finished with either success or error.
Example of handlers:
Here’s a real-life example: file upload with progress indication:
XMLHttpRequest
can make cross-origin requests, using the same CORS policy as fetch.
Just like fetch
, it doesn’t send cookies and HTTP-authorization to another origin by default. To enable them, set xhr.withCredentials
to true
:
See the chapter Fetch: Cross-Origin Requests for details about cross-origin headers.
Typical code of the GET-request with XMLHttpRequest
:
There are actually more events, the modern specification lists them (in the lifecycle order):
loadstart
– the request has started.progress
– a data packet of the response has arrived, the whole response body at the moment is inresponse
.abort
– the request was canceled by the callxhr.abort()
.error
– connection error has occurred, e.g. wrong domain name. Doesn’t happen for HTTP-errors like 404.load
– the request has finished successfully.timeout
– the request was canceled due to timeout (only happens if it was set).loadend
– triggers afterload
,error
,timeout
orabort
.
The error
, abort
, timeout
, and load
events are mutually exclusive. Only one of them may happen.
The most used events are load completion (load
), load failure (error
), or we can use a single loadend
handler and check the properties of the request object xhr
to see what happened.
We’ve already seen another event: readystatechange
. Historically, it appeared long ago, before the specification settled. Nowadays, there’s no need to use it, we can replace it with newer events, but it can often be found in older scripts.
If we need to track uploading specifically, then we should listen to same events on xhr.upload
object.
Last updated