Super Agent is light-weight progressive ajax API crafted for flexibility, readability, and a low learning curve after being frustrated with many of the existing request APIs. It also works with Node.js!
request
.post('/api/pet')
.send({ name: 'Manny', species: 'cat' })
.set('X-API-Key', 'foobar')
.set('Accept', 'application/json')
.end(function(err, res){
if (err || !res.ok) {
alert('Oh no! error');
} else {
alert('yay got ' + JSON.stringify(res.body));
}
});
The following test documentation was generated with Mocha's "doc" reporter, and directly reflects the test suite. This provides an additional source of documentation.
A request can be initiated by invoking the appropriate method on the request
object, then calling .end()
to send the request. For example a simple GET request:
request
.get('/search')
.end(function(err, res){
});
A method string may also be passed:
request('GET', '/search').end(callback);
The node client may also provide absolute urls:
request
.get('http://example.com/search')
.end(function(err, res){
});
DELETE, HEAD, POST, PUT and other HTTP verbs may also be used, simply change the method name:
request
.head('/favicon.ico')
.end(function(err, res){
});
DELETE is a special-case, as it's a reserved word, so the method is named .del()
:
request
.del('/user/1')
.end(function(err, res){
});
The HTTP method defaults to GET, so if you wish, the following is valid:
request('/search', function(err, res){
});
Setting header fields is simple, invoke .set()
with a field name and value:
request
.get('/search')
.set('API-Key', 'foobar')
.set('Accept', 'application/json')
.end(callback);
You may also pass an object to set several fields in a single call:
request
.get('/search')
.set({ 'API-Key': 'foobar', Accept: 'application/json' })
.end(callback);
The .query()
method accepts objects, which when used with the GET method will form a query-string. The following will produce the path /search?query=Manny&range=1..5&order=desc
.
request
.get('/search')
.query({ query: 'Manny' })
.query({ range: '1..5' })
.query({ order: 'desc' })
.end(function(err, res){
});
Or as a single object:
request
.get('/search')
.query({ query: 'Manny', range: '1..5', order: 'desc' })
.end(function(err, res){
});
The .query()
method accepts strings as well:
request
.get('/querystring')
.query('search=Manny&range=1..5')
.end(function(err, res){
});
Or joined:
request
.get('/querystring')
.query('search=Manny')
.query('range=1..5')
.end(function(err, res){
});
You can also use the .query()
method for HEAD requests. The following will produce the path /users?email=joe@smith.com
.
request
.head('/users')
.query({ email: 'joe@smith.com' })
.end(function(err, res){
});
A typical JSON POST request might look a little like the following, where we set the Content-Type header field appropriately, and "write" some data, in this case just a JSON string.
request.post('/user')
.set('Content-Type', 'application/json')
.send('{"name":"tj","pet":"tobi"}')
.end(callback)
Since JSON is undoubtably the most common, it's the default! The following example is equivalent to the previous.
request.post('/user')
.send({ name: 'tj', pet: 'tobi' })
.end(callback)
Or using multiple .send()
calls:
request.post('/user')
.send({ name: 'tj' })
.send({ pet: 'tobi' })
.end(callback)
By default sending strings will set the Content-Type to application/x-www-form-urlencoded
,
multiple calls will be concatenated with &
, here resulting in name=tj&pet=tobi
:
request.post('/user')
.send('name=tj')
.send('pet=tobi')
.end(callback);
SuperAgent formats are extensible, however by default "json" and "form" are supported. To send the data as application/x-www-form-urlencoded
simply invoke .type()
with "form", where the default is "json". This request will POST the body "name=tj&pet=tobi".
request.post('/user')
.type('form')
.send({ name: 'tj' })
.send({ pet: 'tobi' })
.end(callback)
Note: "form" is aliased as "form-data" and "urlencoded" for backwards compat.
The obvious solution is to use the .set()
method:
request.post('/user')
.set('Content-Type', 'application/json')
As a short-hand the .type()
method is also available, accepting
the canonicalized MIME type name complete with type/subtype, or
simply the extension name such as "xml", "json", "png", etc:
request.post('/user')
.type('application/json')
request.post('/user')
.type('json')
request.post('/user')
.type('png')
In a similar fashion to the .type()
method it is also possible to set the Accept header via the short hand method .accept()
. Which references request.types
as well allowing you to specify either the full canonicalized MIME type name as type/subtype, or the extension suffix form as "xml", "json", "png", etc for convenience:
request.get('/user')
.accept('application/json')
request.get('/user')
.accept('json')
request.post('/user')
.accept('png')
res.query(obj)
is a method which may be used to build up a query-string. For example populating ?format=json&dest=/login
on a POST:
request
.post('/')
.query({ format: 'json' })
.query({ dest: '/login' })
.send({ post: 'data', here: 'wahoo' })
.end(callback);
Super Agent will parse known response-body data for you, currently supporting application/x-www-form-urlencoded, application/json, and multipart/form-data.
The property res.body
is the parsed object, for example if a request responded with the JSON string '{"user":{"name":"tobi"}}', res.body.user.name
would be "tobi". Likewise the x-www-form-urlencoded value of "user[name]=tobi" would yield the same result.
The Node client supports multipart/form-data via the Formidable module. When parsing multipart responses, the object res.files
is also available to you. Suppose for example a request responds with the following multipart body:
--whoop
Content-Disposition: attachment; name="image"; filename="tobi.png"
Content-Type: image/png
... data here ...
--whoop
Content-Disposition: form-data; name="name"
Content-Type: text/plain
Tobi
--whoop--
You would have the values res.body.name
provided as "Tobi", and res.files.image
as a File
object containing the path on disk, filename, and other properties.
Many helpful flags and properties are set on the Response
object, ranging from the response text, parsed response body, header fields, status flags and more.
The res.text
property contains the unparsed response body string. This
property is always present for the client API, and only when the mime type
matches "text/", "/json", or "x-www-form-urlencoding" by default for node. The
reasoning is to conserve memory, as buffering text of large bodies such as multipart files or images is extremely inefficient.
To force buffering see the "Buffering responses" section.
Much like SuperAgent can auto-serialize request data, it can also automatically parse it. When a parser is defined for the Content-Type, it is parsed, which by default includes "application/json" and "application/x-www-form-urlencoded". The parsed object is then available via res.body
.
The res.header
contains an object of parsed header fields, lowercasing field names much like node does. For example res.header['content-length']
.
The Content-Type response header is special-cased, providing res.type
, which is void of the charset (if any). For example the Content-Type of "text/html; charset=utf8" will provide "text/html" as res.type
, and the res.charset
property would then contain "utf8".
The response status flags help determine if the request was a success, among other useful information, making SuperAgent ideal for interacting with RESTful web services. These flags are currently defined as:
var type = status / 100 | 0;
// status / class
res.status = status;
res.statusType = type;
// basics
res.info = 1 == type;
res.ok = 2 == type;
res.clientError = 4 == type;
res.serverError = 5 == type;
res.error = 4 == type || 5 == type;
// sugar
res.accepted = 202 == status;
res.noContent = 204 == status || 1223 == status;
res.badRequest = 400 == status;
res.unauthorized = 401 == status;
res.notAcceptable = 406 == status;
res.notFound = 404 == status;
res.forbidden = 403 == status;
To abort requests simply invoke the req.abort()
method.
A timeout can be applied by invoking req.timeout(ms)
, after which an error
will be triggered. To differentiate between other errors the err.timeout
property
is set to the ms
value. NOTE that this is a timeout applied to the request
and all subsequent redirects, not per request.
Basic auth is currently provided by the node client in two forms, first via the URL as "user:pass":
request.get('http://tobi:learnboost@local').end(callback);
As well as via the .auth()
method:
request
.get('http://local')
.auth('tobi', 'learnboost')
.end(callback);
By default up to 5 redirects will be followed, however you may specify this with the res.redirects(n)
method:
request
.get('/some.png')
.redirects(2)
.end(callback);
The Node client allows you to pipe data to and from the request. For example piping a file's contents as the request:
var request = require('superagent')
, fs = require('fs');
var stream = fs.createReadStream('path/to/my.json');
var req = request.post('/somewhere');
req.type('json');
stream.pipe(req);
Or piping the response to a file:
var request = require('superagent')
, fs = require('fs');
var stream = fs.createWriteStream('path/to/my.json');
var req = request.get('/some.json');
req.pipe(stream);
Super Agent is also great for building multipart requests for which it provides methods .attach()
and .field()
.
As mentioned a higher-level API is also provided, in the form of .attach(name, [path], [filename])
and .field(name, value)
. Attaching several files is simple, you can also provide a custom filename for the attachment, otherwise the basename of the attached file is used.
request
.post('/upload')
.attach('avatar', 'path/to/tobi.png', 'user.png')
.attach('image', 'path/to/loki.png')
.attach('file', 'path/to/jane.png')
.end(callback);
Much like form fields in HTML, you can set field values with the .field(name, value)
method. Suppose you want to upload a few images with your name and email, your request might look something like this:
request
.post('/upload')
.field('user[name]', 'Tobi')
.field('user[email]', 'tobi@learnboost.com')
.attach('image', 'path/to/tobi.png')
.end(callback);
The node client supports compressed responses, best of all, you don't have to do anything! It just works.
To force buffering of response bodies as res.text
you may invoke req.buffer()
. To undo the default of buffering for text responses such
as "text/plain", "text/html" etc you may invoke req.buffer(false)
.
When buffered the res.buffered
flag is provided, you may use this to
handle both buffered and unbuffered responses in the same callback.
The .withCredentials()
method enables the ability to send cookies
from the origin, however only when "Access-Control-Allow-Origin" is
not a wildcard ("*"), and "Access-Control-Allow-Credentials" is "true".
request
.get('http://localhost:4001/')
.withCredentials()
.end(function(err, res){
assert(200 == res.status);
assert('tobi' == res.text);
next();
})
Your callback function will always be passed two arguments: error and response. If no error occurred, the first argument will be null:
request
.post('/upload')
.attach('image', 'path/to/tobi.png')
.end(function(err, res){
});
An "error" event is also emitted, with you can listen for:
request
.post('/upload')
.attach('image', 'path/to/tobi.png')
.on('error', handle)
.end(function(err, res){
});
Note that a 4xx or 5xx response with super agent are considered an error by default. For example if you get a 500 or 403 response, this status information will be available via err.status
. Errors from such responses also contain an err.response
field with all of the properties mentioned in "Response properties". The library behaves in this way to handle the common case of wanting success responses and treating HTTP error status codes as errors while still allowing for custom logic around specific error conditions.
Network failures, timeouts, and other errors that produce no response will contain no err.status
or err.response
fields.
If you wish to handle 404 or other HTTP error responses, you can query the err.status
property.
When an HTTP error occurs (4xx or 5xx response) the res.error
property is an Error
object,
this allows you to perform checks such as:
if (err && err.status === 404) {
alert('oh no ' + res.body.message);
}
else if (err) {
// all other error types we handle generically
}
Superagent now supports easier control flow using generators. By using a generator control flow
like co or a web framework like koa,
you can yield
on any superagent method:
var res = yield request
.get('http://local')
.auth('tobi', 'learnboost')
# SuperAgent Super Agent is light-weight progressive ajax API crafted for flexibility, readability, and a low learning curve after being frustrated with many of the existing request APIs. It also works with Node.js! request .post('/api/pet') .send({ name: 'Manny', species: 'cat' }) .set('X-API-Key', 'foobar') .set('Accept', 'application/json') .end(function(err, res){ if (err || !res.ok) { alert('Oh no! error'); } else { alert('yay got ' + JSON.stringify(res.body)); } }); ## Test documentation The following [test documentation](docs/test.html) was generated with [Mocha's](http://visionmedia.github.com/mocha) "doc" reporter, and directly reflects the test suite. This provides an additional source of documentation. ## Request basics A request can be initiated by invoking the appropriate method on the `request` object, then calling `.end()` to send the request. For example a simple GET request: request .get('/search') .end(function(err, res){ }); A method string may also be passed: request('GET', '/search').end(callback); The __node__ client may also provide absolute urls: request .get('http://example.com/search') .end(function(err, res){ }); __DELETE__, __HEAD__, __POST__, __PUT__ and other __HTTP__ verbs may also be used, simply change the method name: request .head('/favicon.ico') .end(function(err, res){ }); __DELETE__ is a special-case, as it's a reserved word, so the method is named `.del()`: request .del('/user/1') .end(function(err, res){ }); The HTTP method defaults to __GET__, so if you wish, the following is valid: request('/search', function(err, res){ }); ## Setting header fields Setting header fields is simple, invoke `.set()` with a field name and value: request .get('/search') .set('API-Key', 'foobar') .set('Accept', 'application/json') .end(callback); You may also pass an object to set several fields in a single call: request .get('/search') .set({ 'API-Key': 'foobar', Accept: 'application/json' }) .end(callback); ## GET requests The `.query()` method accepts objects, which when used with the __GET__ method will form a query-string. The following will produce the path `/search?query=Manny&range=1..5&order=desc`. request .get('/search') .query({ query: 'Manny' }) .query({ range: '1..5' }) .query({ order: 'desc' }) .end(function(err, res){ }); Or as a single object: request .get('/search') .query({ query: 'Manny', range: '1..5', order: 'desc' }) .end(function(err, res){ }); The `.query()` method accepts strings as well: request .get('/querystring') .query('search=Manny&range=1..5') .end(function(err, res){ }); Or joined: request .get('/querystring') .query('search=Manny') .query('range=1..5') .end(function(err, res){ }); ## HEAD requests You can also use the `.query()` method for HEAD requests. The following will produce the path `/users?email=joe@smith.com`. request .head('/users') .query({ email: 'joe@smith.com' }) .end(function(err, res){ }); ## POST / PUT requests A typical JSON __POST__ request might look a little like the following, where we set the Content-Type header field appropriately, and "write" some data, in this case just a JSON string. request.post('/user') .set('Content-Type', 'application/json') .send('{"name":"tj","pet":"tobi"}') .end(callback) Since JSON is undoubtably the most common, it's the _default_! The following example is equivalent to the previous. request.post('/user') .send({ name: 'tj', pet: 'tobi' }) .end(callback) Or using multiple `.send()` calls: request.post('/user') .send({ name: 'tj' }) .send({ pet: 'tobi' }) .end(callback) By default sending strings will set the Content-Type to `application/x-www-form-urlencoded`, multiple calls will be concatenated with `&`, here resulting in `name=tj&pet=tobi`: request.post('/user') .send('name=tj') .send('pet=tobi') .end(callback); SuperAgent formats are extensible, however by default "json" and "form" are supported. To send the data as `application/x-www-form-urlencoded` simply invoke `.type()` with "form", where the default is "json". This request will POST the body "name=tj&pet=tobi". request.post('/user') .type('form') .send({ name: 'tj' }) .send({ pet: 'tobi' }) .end(callback) Note: "form" is aliased as "form-data" and "urlencoded" for backwards compat. ## Setting the Content-Type The obvious solution is to use the `.set()` method: request.post('/user') .set('Content-Type', 'application/json') As a short-hand the `.type()` method is also available, accepting the canonicalized MIME type name complete with type/subtype, or simply the extension name such as "xml", "json", "png", etc: request.post('/user') .type('application/json') request.post('/user') .type('json') request.post('/user') .type('png') ## Setting Accept In a similar fashion to the `.type()` method it is also possible to set the Accept header via the short hand method `.accept()`. Which references `request.types` as well allowing you to specify either the full canonicalized MIME type name as type/subtype, or the extension suffix form as "xml", "json", "png", etc for convenience: request.get('/user') .accept('application/json') request.get('/user') .accept('json') request.post('/user') .accept('png') ## Query strings `res.query(obj)` is a method which may be used to build up a query-string. For example populating `?format=json&dest=/login` on a __POST__: request .post('/') .query({ format: 'json' }) .query({ dest: '/login' }) .send({ post: 'data', here: 'wahoo' }) .end(callback); ## Parsing response bodies Super Agent will parse known response-body data for you, currently supporting _application/x-www-form-urlencoded_, _application/json_, and _multipart/form-data_. ### JSON / Urlencoded The property `res.body` is the parsed object, for example if a request responded with the JSON string '{"user":{"name":"tobi"}}', `res.body.user.name` would be "tobi". Likewise the x-www-form-urlencoded value of "user[name]=tobi" would yield the same result. ### Multipart The Node client supports _multipart/form-data_ via the [Formidable](https://github.com/felixge/node-formidable) module. When parsing multipart responses, the object `res.files` is also available to you. Suppose for example a request responds with the following multipart body: --whoop Content-Disposition: attachment; name="image"; filename="tobi.png" Content-Type: image/png ... data here ... --whoop Content-Disposition: form-data; name="name" Content-Type: text/plain Tobi --whoop-- You would have the values `res.body.name` provided as "Tobi", and `res.files.image` as a `File` object containing the path on disk, filename, and other properties. ## Response properties Many helpful flags and properties are set on the `Response` object, ranging from the response text, parsed response body, header fields, status flags and more. ### Response text The `res.text` property contains the unparsed response body string. This property is always present for the client API, and only when the mime type matches "text/*", "*/json", or "x-www-form-urlencoding" by default for node. The reasoning is to conserve memory, as buffering text of large bodies such as multipart files or images is extremely inefficient. To force buffering see the "Buffering responses" section. ### Response body Much like SuperAgent can auto-serialize request data, it can also automatically parse it. When a parser is defined for the Content-Type, it is parsed, which by default includes "application/json" and "application/x-www-form-urlencoded". The parsed object is then available via `res.body`. ### Response header fields The `res.header` contains an object of parsed header fields, lowercasing field names much like node does. For example `res.header['content-length']`. ### Response Content-Type The Content-Type response header is special-cased, providing `res.type`, which is void of the charset (if any). For example the Content-Type of "text/html; charset=utf8" will provide "text/html" as `res.type`, and the `res.charset` property would then contain "utf8". ### Response status The response status flags help determine if the request was a success, among other useful information, making SuperAgent ideal for interacting with RESTful web services. These flags are currently defined as: var type = status / 100 | 0; // status / class res.status = status; res.statusType = type; // basics res.info = 1 == type; res.ok = 2 == type; res.clientError = 4 == type; res.serverError = 5 == type; res.error = 4 == type || 5 == type; // sugar res.accepted = 202 == status; res.noContent = 204 == status || 1223 == status; res.badRequest = 400 == status; res.unauthorized = 401 == status; res.notAcceptable = 406 == status; res.notFound = 404 == status; res.forbidden = 403 == status; ## Aborting requests To abort requests simply invoke the `req.abort()` method. ## Request timeouts A timeout can be applied by invoking `req.timeout(ms)`, after which an error will be triggered. To differentiate between other errors the `err.timeout` property is set to the `ms` value. __NOTE__ that this is a timeout applied to the request and all subsequent redirects, not per request. ## Basic authentication Basic auth is currently provided by the _node_ client in two forms, first via the URL as "user:pass": request.get('http://tobi:learnboost@local').end(callback); As well as via the `.auth()` method: request .get('http://local') .auth('tobi', 'learnboost') .end(callback); ## Following redirects By default up to 5 redirects will be followed, however you may specify this with the `res.redirects(n)` method: request .get('/some.png') .redirects(2) .end(callback); ## Piping data The Node client allows you to pipe data to and from the request. For example piping a file's contents as the request: var request = require('superagent') , fs = require('fs'); var stream = fs.createReadStream('path/to/my.json'); var req = request.post('/somewhere'); req.type('json'); stream.pipe(req); Or piping the response to a file: var request = require('superagent') , fs = require('fs'); var stream = fs.createWriteStream('path/to/my.json'); var req = request.get('/some.json'); req.pipe(stream); ## Multipart requests Super Agent is also great for _building_ multipart requests for which it provides methods `.attach()` and `.field()`. ### Attaching files As mentioned a higher-level API is also provided, in the form of `.attach(name, [path], [filename])` and `.field(name, value)`. Attaching several files is simple, you can also provide a custom filename for the attachment, otherwise the basename of the attached file is used. request .post('/upload') .attach('avatar', 'path/to/tobi.png', 'user.png') .attach('image', 'path/to/loki.png') .attach('file', 'path/to/jane.png') .end(callback); ### Field values Much like form fields in HTML, you can set field values with the `.field(name, value)` method. Suppose you want to upload a few images with your name and email, your request might look something like this: request .post('/upload') .field('user[name]', 'Tobi') .field('user[email]', 'tobi@learnboost.com') .attach('image', 'path/to/tobi.png') .end(callback); ## Compression The node client supports compressed responses, best of all, you don't have to do anything! It just works. ## Buffering responses To force buffering of response bodies as `res.text` you may invoke `req.buffer()`. To undo the default of buffering for text responses such as "text/plain", "text/html" etc you may invoke `req.buffer(false)`. When buffered the `res.buffered` flag is provided, you may use this to handle both buffered and unbuffered responses in the same callback. ## CORS The `.withCredentials()` method enables the ability to send cookies from the origin, however only when "Access-Control-Allow-Origin" is _not_ a wildcard ("*"), and "Access-Control-Allow-Credentials" is "true". request .get('http://localhost:4001/') .withCredentials() .end(function(err, res){ assert(200 == res.status); assert('tobi' == res.text); next(); }) ## Error handling Your callback function will always be passed two arguments: error and response. If no error occurred, the first argument will be null: request .post('/upload') .attach('image', 'path/to/tobi.png') .end(function(err, res){ }); An "error" event is also emitted, with you can listen for: request .post('/upload') .attach('image', 'path/to/tobi.png') .on('error', handle) .end(function(err, res){ }); Note that a 4xx or 5xx response with super agent **are** considered an error by default. For example if you get a 500 or 403 response, this status information will be available via `err.status`. Errors from such responses also contain an `err.response` field with all of the properties mentioned in "Response properties". The library behaves in this way to handle the common case of wanting success responses and treating HTTP error status codes as errors while still allowing for custom logic around specific error conditions. Network failures, timeouts, and other errors that produce no response will contain no `err.status` or `err.response` fields. If you wish to handle 404 or other HTTP error responses, you can query the `err.status` property. When an HTTP error occurs (4xx or 5xx response) the `res.error` property is an `Error` object, this allows you to perform checks such as: if (err && err.status === 404) { alert('oh no ' + res.body.message); } else if (err) { // all other error types we handle generically } ## Generator support Superagent now supports easier control flow using generators. By using a generator control flow like [co](https://github.com/tj/co) or a web framework like [koa](https://github.com/koajs/koa), you can `yield` on any superagent method: var res = yield request .get('http://local') .auth('tobi', 'learnboost')
# | Change | User | Description | Committed | |
---|---|---|---|---|---|
#1 | 19553 | swellard | Move and rename clients | ||
//guest/perforce_software/helix-web-services/main/source/clients/2016.1.0/javascript/node_modules/superagent/docs/index.md | |||||
#1 | 18810 | tjuricek |
First-pass at JavaScript client SDK. JavaScript requires Node with Gulp to "browserfy" the library. It's the easiest way I found to use the swagger-js project; bundle up a wrapping method. There is no JavaScript reference guide. The swagger-js doesn't really document what they do very well, actually. Overall I'm not particularly impressed by swagger-js, it was hard to even figure out what the right method syntax was. We may want to invest time in doing it better. This required setting CORS response headers, which are currently defaulted to a fairly insecure setting. |