XMLHttpRequest
ObjectCopyright © 2007 W3C® (MIT, ERCIM, Keio), All Rights Reserved. W3C liability, trademark and document use rules apply.
The XMLHttpRequest
Object specification defines an
API that provides
scripted client functionality for transferring data between a client and a
server.
This section describes the status of this document at the time of its publication. Other documents may supersede this document. A list of current W3C publications and the latest revision of this technical report can be found in the W3C technical reports index at http://www.w3.org/TR/.
This is the 18 June 2007 Working Draft
of The XMLHttpRequest
Object specification. Given
that the previous Last Call Working Draft got feedback that required
extensive changes this is a normal Working Draft again. It is expected
that this document will become a second Last Call Working Draft with no
more than editorial changes. Please send comments to public-webapi@w3.org (archived)
with either [XHR] or [XMLHttpRequest]
at the start of the subject line.
This document is produced by the Web API Working Group, part of the Rich Web Clients Activity in the W3C Interaction Domain. Changes made to this document can be found in the W3C public CVS server.
Publication as a Working Draft does not imply endorsement by the W3C Membership. This is a draft document and may be updated, replaced or obsoleted by other documents at any time. It is inappropriate to cite this document as other than work in progress.
This document was produced by a group operating under the 5 February 2004 W3C Patent Policy. W3C maintains a public list of any patent disclosures made in connection with the deliverables of the group; that page also includes instructions for disclosing a patent. An individual who has actual knowledge of a patent which the individual believes contains Essential Claim(s) must disclose the information in accordance with section 6 of the W3C Patent Policy.
This section is non-normative.
The XMLHttpRequest
object implements an interface exposed by a scripting engine that allows
scripts to perform HTTP client functionality, such as submitting form data
or loading data from a server.
The name of the object is XMLHttpRequest
for compatibility
with the web, though each component of this name is potentially
misleading. First, the object supports any text based format, including
XML. Second, it can be used to make requests over both HTTP and HTTPS
(some implementations support protocols in addition to HTTP and HTTPS, but
that functionality is not covered by this specification). Finally, it
supports "requests" in a broad sense of the term as it pertains to HTTP;
namely all activity involved with HTTP requests or responses for the
defined HTTP methods.
This section is non-normative.
Some [ECMAScript] examples are listed in the specification. In addition, you can find some below.
Some simple code to do something with data from an XML document fetched over the network:
function test(data) {
// taking care of data
}
function handler() {
if(this.readyState == 4 && this.status == 200) {
// so far so good
if(this.responseXML != null && this.responseXML.getElementById('test').firstChild.data)
// success!
test(this.responseXML.getElementById('test').firstChild.data);
else
test(null);
} else if (this.readyState == 4 && this.status != 200) {
// fetched the wrong page or network error...
test(null);
}
}
var client = new XMLHttpRequest();
client.onreadystatechange = handler;
client.open("GET", "test.xml");
client.send();
If you just want to log a message to the server:
function log(message) {
var client = new XMLHttpRequest();
client.open("POST", "/log");
client.setRequestHeader("Content-Type", "text/plain;charset=UTF-8");
client.send(message);
}
Or if you want to check the status of a document on the server:
function fetchStatus(address) {
var client = new XMLHttpRequest();
client.onreadystatechange = function() {
// in case of network errors this might not give reliable results
if(this.readyState == 4)
returnStatus(this.status);
}
client.open("HEAD", address);
client.send();
}
Everything in this specification is normative except for diagrams, examples, notes and sections marked non-normative.
The key words must, must not, should and may in this document are to be interpreted as described in RFC 2119. [RFC2119]
This specification defines the following classes of products:
A user agent must behave as described in this specification in order to be considered conformant.
If the user agent does not support XML (including support for
namespaces) the XML response entity
body must (always) be null
.
User agents may optimize any algorithm given in this specification, so long as the end result is indistinguishable from the result that would be obtained by the specification's algorithms. (The algorithms in this specification are generally written with more concern for clarity than efficiency.)
This specification uses both the terms "conforming user agent(s)" and "user agent(s)" to refer to this product class.
A user agent that is a conforming user agent and also supports XML (including support for namespaces).
This specification relies on several underlying specifications.
Conforming user agents must support some version of DOM Events and DOM Core, because this specification uses some of the features defined in those specifications. [DOM3Events] [DOM3Core]
It must also support some version of the Window Object because some of the functionality in this specification relies on it. [Window]
Conforming user agents must support some
version of the HTTP protocol. The user agent should
support any HTTP method that matches the Method
production and must at least support the following methods:
GET
POST
HEAD
PUT
DELETE
OPTIONS
Other requirements regarding HTTP are made throughout the specification. [RFC2616]
Conforming XML user agents must support XML (including support for namespaces). [XML] [XMLNS]
There is a case-insensitive match of strings s1 and s2 if after uppercasing both strings (by mapping a-z to A-Z) they are identical.
Extensions of the API defined by this specification are strongly discouraged. User agents, Working Groups and other interested parties should discuss extensions on a relevant public forum, preferably public-webapi@w3.org.
XMLHttpRequest
ObjectThe XMLHttpRequest
object can be used by scripts to programmatically connect to their
originating server via HTTP.
Objects implementing the XMLHttpRequest
interface must also implement the EventTarget
interface.
[DOM3Events]
Objects implementing the Window
interface must provide an XMLHttpRequest()
constructor. [Window]
In ECMAScript this can be used as follows:
var client = new XMLHttpRequest();
When the XMLHttpRequest()
constructor is invoked a
persistent pointer to the associated Window
object
must be stored on the newly created object. This is the
Window
pointer. The associated Window
object is the
one of which the XMLHttpRequest
constructor was
invoked. This pointer
must persist even if the browsing context in which the
Window
is located is destroyed (by removing it from
a parent browsing context, for instance).
The term browsing context is defined by the Window Object 1.0 specification. [Window]
If win
is a Window
object
client
will have a pointer to
win
in the following example:
var client = new win.XMLHttpRequest()
interface XMLHttpRequest { // event handler attribute EventListener onreadystatechange; // state const unsigned short UNSENT = 0; const unsigned short OPEN = 1; const unsigned short SENT = 2; const unsigned short LOADING = 3; const unsigned short DONE = 4; readonly attribute unsigned short readyState; // request void open(in DOMString method, in DOMString url); void open(in DOMString method, in DOMString url, in boolean async); void open(in DOMString method, in DOMString url, in boolean async, in DOMString user); void open(in DOMString method, in DOMString url, in boolean async, in DOMString user, in DOMString password); void setRequestHeader(in DOMString header, in DOMString value); void send(); void send(in DOMString data); void send(in Document data); void abort(); // response DOMString getAllResponseHeaders(); DOMString getResponseHeader(in DOMString header); readonly attribute DOMString responseText; readonly attribute Document responseXML; readonly attribute unsigned short status; readonly attribute DOMString statusText; };
The XMLHttpRequest
object can be in five states: UNSENT, OPEN, SENT, LOADING and DONE. The current state is exposed through the readyState
attribute. The method
definitions below define when a state transition takes place.
When constructed, the XMLHttpRequest
object must be in the UNSENT state. This state is represented by
the UNSENT
constant, whose value is 0
.
The OPEN state is the state of the object when the open()
method has been successfully invoked.
During this state request headers can be set using setRequestHeader()
and the request
can be made using send()
. This state
is represented by the OPEN
constant, whose value is 1
.
During the OPEN state there is a send()
flag which can be either "true" or "false". The initial value of the
send()
flag is "false".
The SENT state is the state of the object when the user agent
successfully acknowledged the request. This state is represented by the
SENT
constant, whose
value is 2
.
The LOADING state is the state of the object when all HTTP headers have
been received. The object typically remains in this state until the
complete message body (if any) has been received. This state is
represented by the LOADING
constant, whose value is 3
.
The DONE state is the state of the object when either the data transfer
has been completed or something went wrong during the transfer (infinite
redirects for instance). This state is represented by the DONE
constant, whose value is
4
.
The response entity body is the fragment of the entity body received so far (LOADING state) or the complete entity body (DONE state). If there is no entity body the response entity body is "null".
The text response entity body is
either a DOMString
representing the response entity body or
null
. The value of the text response entity body must be determined by running the following algorithm:
If the response entity body is "null" return null
and
terminate these steps.
Let charset be "null".
If there is no Content-Type
header or there is a
Content-Type
header which contains a MIME type that is
text/xml
, application/xml
or ends in +xml
(ignoring any parameters) use the rules set forth
in the XML specification to determine the character encoding. Let
charset be the determined character encoding.
If there is a Content-Type
header contains a
text/html
MIME type follow the rules set forth in the HTML
specification to determine the character encoding. Let
charset be the determined character encoding.
If the Content-Type
MIME type contains a
charset
parameter and charset is "null" let
charset be the value of that parameter.
The algorithms described by the XML and HTML specifications
already take Content-Type
into account.
If charset is "null" then, for each of the rows in the following table, starting with the first one and going down, if the start of the entity body (bytes) is equal to or greater than the number of bytes in the first column, and the first bytes of bytes match the bytes given in the first column, then let charset be the encoding given in the cell in the second column of that row. When an encoding is determined or when no encoding is determined go to the next step:
Bytes in Hexadecimal | Description |
---|---|
00 00 FE FF | UTF-32BE BOM |
FF FE 00 00 | UTF-32LE BOM |
FE FF | UTF-16BE BOM |
FF FE | UTF-16LE BOM |
EF BB BF | UTF-8 BOM |
If charset is "null" let charset be UTF-8.
Return the result of decoding the response entity body using
charset. Or, if that fails, return null
.
The XML response entity body is
either a Document
representing the response entity body or null
and is determined by running the following algorithm (the return value is
its value):
If the response entity body is "null" terminate these steps and return
null
.
If a Content-Type
is present and it does not contain a
MIME type (ignoring any parameters) that is text/xml
,
application/xml
or ends in +xml
terminate these steps and return null
. (Do not terminate
these steps if there is no Content-Type
header at all.)
Parse the response entity body following the rules from the XML
specifications. Let the result be parsed document. If this
fails (unsupported character encoding, namespace well-formedness error
et cetera) terminate these steps return null
. [XML] [XMLNS]
Return an object implementing the Document
interface
representing the parsed document.
onreadystatechange
of type EventListener
An attribute that takes an EventListener
as value that
must be invoked along with any other appropriate event
listeners that are registered on this object when a readystatechange
event is
dispatched on it during the bubbling phase. Its initial value must be null
.
readyState
of type
unsigned short
, readonly
The attribute must be the value of the constant corresponding to the object's current state.
open(method,
url, async, user,
password)
, method
When invoked, the user agent must follow the following steps (unless otherwise indicated):
If the method argument does not match the Method
production defined in section
5.1.1 of RFC 2616 raise a SYNTAX_ERR
exception and
terminate these steps. [RFC2616]
If the given method is not supported for security reasons
the user agent should raise a
SECURITY_ERR
exception and terminate these steps.
Let stored method be method.
If method case-insensitively matches
GET
, POST
, HEAD
,
PUT
, DELETE
or OPTIONS
convert
it to its uppercase equivalent (by mapping a-z to A-Z) and let
stored method be the result.
Drop the fragment identifier (if any) from url and let stored url be the result of that operation.
If stored url is a relative reference resolve it using
the current value of the baseURI
attribute of the
Document
object currently associated with the Window
pointer. If this fails
raise a SYNTAX_ERR
exception and terminate these steps.
If stored url contains an unsupported scheme raise a
NOT_SUPPORTED_ERR
and terminate these steps.
If the "user:password"
format in the
userinfo
production defined in section 3.2.1 of RFC 3986
is not supported for the relevant scheme and stored url
contains this format raise a SYNTAX_ERR
and terminate
these steps. [RFC3986]
If stored url contains the "user:password"
format let stored user be the user part and stored
password be the password part.
If stored url just contains the "user"
format let stored user be the user part.
If stored url is non same-origin the user agent should raise a SECURITY_ERR
exception and
terminate these steps.
This specification does not define what constitutes as non same-origin. It is expected that what constitutes as non same-origin will be defined before this specification eventually reaches the W3C Recommendation status by way of referencing a specification that defines it.
Let async be the value of the async argument
or true
if it was omitted.
If the user argument was not omitted and its syntax does
not match that specified by the relevant authentication scheme raise a
SYNTAX_ERR
exception and terminate these steps.
If the user argument was not omitted and is not
null
let stored user be user
encoded using the encoding specified in the relevant authentication
scheme or UTF-8 if the scheme fails to specify an encoding.
This step overrides any user that may have been set by the url argument.
If the user argument was not omitted and is
null
remove stored user.
If the password argument was not omitted and its syntax
does not match that specified by the relevant authentication scheme
raise a SYNTAX_ERR
exception and terminate these steps.
If the password argument was not omitted and is not
null
let stored password be
password encoded using the encoding specified in the
relevant authentication scheme or UTF-8 if the scheme fails to specify
an encoding.
If the password argument was not omitted and is
null
remove stored password.
Abort the
send()
algorithm, set response entity body to "null" and
reset the list of request headers.
The user agent should cancel any network activity for which the object is responsible.
Switch the object to the OPEN
state, set the send()
flag to
"false" and then synchronously dispatch a readystatechange
event on the
object and return the method call.
A future version or extension of this specification will most likely define a way of doing cross-site requests.
setRequestHeader(header,
value)
, method
Each request has a list of request headers with associated values. This method can be used to manipulate those values and set new request headers.
When invoked, the user agent must follow the following steps (unless otherwise indicated):
If the state of the object is not OPEN raise an INVALID_STATE_ERR
exception and
terminate these steps.
If the send()
flag is "true"
raise an INVALID_STATE_ERR
exception and terminate these
steps.
If the header argument does not match the field-name
production as defined by
section 4.2 of RFC 2616 or is null
raise a
SYNTAX_ERR
exception and terminate these steps. [RFC2616]
If the value argument is null
terminate
these steps. (Do not raise an exception.)
If the value argument does not match the field-value
production as defined by
section 4.2 of RFC 2616 raise a SYNTAX_ERR
and terminate
these steps. [RFC2616]
For security reasons these steps should be terminated if the header argument case-insensitively matches one of the following headers:
Accept-Charset
Accept-Encoding
Connection
Content-Length
Content-Transfer-Encoding
Date
Expect
Host
Keep-Alive
Referer
TE
Trailer
Transfer-Encoding
Upgrade
Via
If the header argument is not in the list of request headers append the header with its associated value to the list and terminate these steps.
If the header argument is in the list of request headers either use multiple headers, combine the values or use a combination of those (section 4.2, RFC 2616). [RFC2616]
See also the send()
method regarding user agent header handling for caching, authentication,
proxies, and cookies.
// The following script:
var client = new XMLHttpRequest();
client.open('GET', 'demo.cgi');
client.setRequestHeader('X-Test', 'one');
client.setRequestHeader('X-Test', 'two');
client.send();
// ...would result in the following header being sent:
...
X-Test: one, two
...
send(data)
,
method
The send()
method initiates the
request and its optional argument provides the entity body. Authors are strongly encouraged
to ensure that they have specified the Content-Type
header
via setRequestHeader()
before invoking send()
with a
non-null
data argument.
When invoked, the user agent must follow the
following steps (unless otherwise noted). Note that this algorithm might
get aborted if the open()
or
abort()
is invoked. When the send()
algorithm is aborted the user agent must
terminate the algorithm after finishing the step it is on.
The following algorithm can not be aborted through script
when async is false
. It can only be aborted when
async is true
and only after the method call has
returned.
If the state of the object is not OPEN raise an INVALID_STATE_ERR
exception and
terminate these steps.
If the send()
flag is "true"
raise an INVALID_STATE_ERR
exception and terminate these
steps.
If async is true
set the send()
flag to "true".
If the data argument has not been omitted and is not
null
use it for the entity
body as defined by section 7.2 of RFC 2616 observing the
following rules: [RFC2616]
DOMString
Encode data using UTF-8 for transmission.
Document
Serialize data into a namespace well-formed XML
document and encoded using the encoding given by
data.xmlEncoding
, if specified, or UTF-8
otherwise. Or, if this fails because the Document
cannot be serialized act as if data is null
.
If no Content-Type
header is in the list of request
headers append it to the list of request headers with a value of
application/xml
.
Subsequent changes to the Document
have no
effect on what is submitted.
DOMString
or
Document
Use the stringification mechanisms of the host language on
data and treat the result as if data is a
DOMString
. Or, if this fails, act as if data
is null
.
If the data argument has been omitted or is null
no
entity body is used in the request.
Make a request to stored url, using HTTP method stored method, user stored user (if provided) and password stored password (if provided), taking into account the entity body, list of request headers and the rules listed directly after this set of steps.
Synchronously dispatch a readystatechange
event on the
object.
The state of the object does not change. The event is dispatched for historical reasons.
If async is true
return the send()
method call. (Do not terminate the
steps in the algorithm though.)
While downloading the resource the following is to be observed.
If the redirect does not violate security or infinite loop precautions and the scheme is supported transparently follow it and go to the start of this step.
HTTP places requirements on the user agent regarding the preservation of the request method and entity body during redirects, and also requires users to be notified of certain kinds of automatic redirections.
Otherwise, follow the following set of steps:
Set response entity body to "null" and reset the list of request headers.
Synchronously switch the state to DONE.
If async is set to false
raise a
NETWORK_ERR
exception and
terminate the overall algorithm.
Synchronously dispatch a readystatechange
event on
the object.
Terminate the overall algorithm.
It is likely that a future version of the XMLHttpRequest
object will
dispatch an error
event here as well.
Run the following set of steps:
Set response entity body to "null" and reset the list of request headers.
Synchronously switch the state to DONE.
If async is set to false
raise an
ABORT_ERR
exception and
terminate the overall algorithm.
Synchronously dispatch a readystatechange
event on
the object.
Terminate the overall algorithm.
It is likely that a future version of the XMLHttpRequest
object will
dispatch an abort
event here as well.
In case of DNS errors or other type of networks run the following set of steps. This does not include HTTP responses that indicate some type of error, such as HTTP status code 410.
Set response entity body to "null" and reset the list of request headers.
Synchronously switch the state to DONE.
If async is set to false
raise a
NETWORK_ERR
exception and
terminate the overall algorithm.
Synchronously dispatch a readystatechange
event on
the object.
Terminate the overall algorithm.
It is likely that a future version of the XMLHttpRequest
object will
dispatch an error
event here as well.
If all HTTP headers have been received, before receiving the message body (if any), follow the following steps:
Synchronously switch the state to SENT.
Synchronously dispatch a readystatechange
event on
the object.
Synchronously switch the state to LOADING.
Synchronously dispatch a readystatechange
event on
the object.
Finally, if the complete resource has been downloaded go to the next step.
When the request has successfully completed loading, synchronously
switch the state to DONE and
then synchronously dispatch a readystatechange
event on the
object and return the method call in case of async being
false
.
If the user agent allows the user to configure a proxy it should modify the request appropriately; i.e., connect to the proxy host instead of the
origin server, modify the Request-Line
and send
Proxy-Authorization
headers as specified.
If the user agent supports HTTP Authentication it should consider requests originating from this object to
be part of the protection space that includes the accessed URIs and send
Authorization
headers and handle 401
Unauthorised
requests appropriately. If authentication fails,
user agents should prompt the users for credentials.
[RFC2617]
If the user agent supports HTTP State Management it should persist, discard and send cookies (as received in
the Set-Cookie
and Set-Cookie2
response
headers, and sent in the Cookie
header) as applicable.
[RFC2965]
If the user agent implements a HTTP cache it should
respect Cache-Control
request headers set by the script
(e.g., Cache-Control:
no-cache
bypasses the cache). It must not send
Cache-Control
or Pragma
request headers
automatically unless the user explicitly requests such behaviour
(e.g., by (force-)reloading the page). 304 Not
Modified
responses that are a result of a user agent generated
conditional request must be presented as 200
OK
responses with the appropriate content. The user agent must allow scripts to override automatic cache validation
by setting request headers (e.g., If-None-Match
,
If-Modified-Since
), in which case 304 Not
Modified
responses must be passed through.
[RFC2616]
If the user agent implements server-driven content-negotiation it should set Accept-Language
,
Accept-Encoding
and Accept-Charset
headers as
appropriate; it must not automatically set the
Accept
header. Responses to such requests must have the content-codings automatically decoded.
[RFC2616]
abort()
, method
When invoked, the user agent must run the following steps (unless otherwise noted):
Abort the
send()
algorithm, set the response entity body to "null", set
the send()
flag to "false" and
remove any registered request headers.
The user agent should cancel any network activity for which the object is responsible.
If the state is UNSENT,
OPEN and the send()
flag is "false", or DONE go to the next step.
Otherwise, switch the state to DONE and synchronously dispatch a readystatechange
event on the
object.
Switch the state to UNSENT. (Do not dispatch the readystatechange
event.)
It is likely that a future version of the XMLHttpRequest
object will
dispatch an abort
event here as well.
getAllResponseHeaders()
,
method
When invoked, the user agent must run the following steps:
If the state is not LOADING or DONE
raise an INVALID_STATE_ERR
exception and terminate these
steps.
Return all the HTTP headers, as a single string, with each header line separated by a U+000D CR U+000A LF pair excluding the status line.
// The following script:
var client = new XMLHttpRequest();
client.open("GET", "test.txt", true);
client.send();
client.onreadystatechange = function() {
if(this.readyState == 3) {
print(this.getAllResponseHeaders());
}
}
// ...should output something similar to the following text:
Date: Sun, 24 Oct 2004 04:58:38 GMT
Server: Apache/1.3.31 (Unix)
Keep-Alive: timeout=15, max=99
Connection: Keep-Alive
Transfer-Encoding: chunked
Content-Type: text/plain; charset=utf-8
getResponseHeader(header)
,
method
When the method is invoked, the user agent must run the following steps:
If the state is not LOADING or DONE
raise an INVALID_STATE_ERR
exception and terminate these
steps.
If the header argument does not match the field-name
production raise a
SYNTAX_ERR
exception and terminate these steps.
If the header argument case-insensitively matches multiple HTTP headers for the last request sent return the values of these headers as a single concatenated string separated from each other by an U+002C COMMA followed by an U+0020 SPACE and terminate these steps.
If the header argument case-insensitively matches a single HTTP header for the last request sent return the value of that header and terminate these steps.
Return null
.
// The following script:
var client = new XMLHttpRequest();
client.open("GET", "test.txt", true);
client.send();
client.onreadystatechange = function() {
if(this.readyState == 3) {
print(client.getResponseHeader("Content-Type"));
}
}
// ...should output something similar to the following text:
Content-Type: text/plain; charset=utf-8
responseText
of type
DOMString
, readonly
On getting, the user agent must run the following steps:
If the state is not LOADING or DONE
raise an INVALID_STATE_ERR
exception and terminate these
steps.
Return the text response entity body.
responseXML
of type
Document
, readonly
On getting, the user agent must run the following steps:
If the state is not DONE
raise an INVALID_STATE_ERR
exception and terminate these
steps.
Return the XML response entity body.
status
of type unsigned
short
, readonly
On getting, if available, it must return the HTTP
status code sent by the server (typically 200
for a
successful request). Otherwise, if not available, the user agent must raise an INVALID_STATE_ERR
exception.
statusText
of type
DOMString
, readonly
On getting, if available, it must return the HTTP
status text sent by the server (appears after the status code).
Otherwise, if not available, the user agent must raise
an INVALID_STATE_ERR
exception.
XMLHttpRequest
ObjectThese sections describe the various events that can be dispatched on the
object implementing the XMLHttpRequest
interface. For
this version of the specification only one event is defined.
readystatechange
readystatechange
event (as indicated above) it must not bubble, must not be cancelable
and must implement the Event
interface.
Its namespaceURI
attribute must be
null
. [DOM3Events]
XMLHttpRequest
Objectexception XMLHttpRequestException { unsigned short code; }; const unsigned short NETWORK_ERR = 101; const unsigned short ABORT_ERR = 102;
The NETWORK_ERR
exception is
thrown when a network error occurs in synchronous requests.
The ABORT_ERR
exception is thrown
when the user aborts a request in synchronous requests.
See the section on send()
for more
details.
This section is non normative.
This specification does not include the following features which are being considered for a future version of this specification:
load
event and onload
attribute;
error
event and onerror
attribute;
progress
event and onprogress
attribute;
abort
event and onabort
attribute;
ontimeout
attribute;
responseXML
for text/html
documents;
XMLHttpRequest
;
responseBody
to deal with byte streams;
overrideMimeType
to fix up MIME types;
getRequestHeader()
and
removeRequestHeader()
.
This section is non-normative
The editor would like to thank to the following people who have contributed to this specification (ordered by first name):
Special thanks to the Microsoft employees who first implemented the
XMLHttpRequest
interface, which was first widely deployed by the Windows Internet
Explorer browser.
Special thanks also to the WHATWG for drafting an initial version of this specification in their Web Applications 1.0 document.
Thanks also to all those who have helped to improve this specification by sending suggestions and corrections. (Please, keep bugging us with your issues!)