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(res){
if (res.ok) {
alert('yay got ' + JSON.stringify(res.body));
} else {
alert('Oh no! error ' + res.text);
}
});
Test documentation
The following test documentation was generated with Mocha’s “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(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(res){
});
DELETE, HEAD, POST, PUT and other HTTP verbs may also be used, simply change the method name:
request
.head('/favicon.ico')
.end(function(res){
});
DELETE is a special-case, as it’s a reserved word, so the method is named .del():
request
.del('/user/1')
.end(function(res){
});
The HTTP method defaults to GET, so if you wish, the following is valid:
request('/search', function(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 .send() 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')
.send({ query: 'Manny' })
.send({ range: '1..5' })
.send({ order: 'desc' })
.end(function(res){
});
Or as a single object:
request
.get('/search')
.send({ query: 'Manny', range: '1..5', order: 'desc' })
.end(function(res){
});
The .send() method accepts strings as well:
request
.get('/querystring')
.send('search=Manny&range=1..5')
.end(function(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)
SuperAgent formats are extensible, however by default “json” and “form-data” are supported. To send the data as application/x-www-form-urlencoded simply invoke .type() with “form-data”, where the default is “json”. This request will POST the body “name=tj&pet=tobi”.
request.post('/user')
.type('form-data')
.send({ name: 'tj' })
.send({ pet: 'tobi' })
.end(callback)
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')
Query strings
When issuing a GET request the res.send(obj) method will invoke res.query(obj), this is a method which may be used with other HTTP methods in order 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 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 text
The res.text property is also available for a string representation of the body as illustrated by this test:
var req = request.post('local/echo');
req.write('{"name"').should.be.a('boolean');
req.write(':"tobi"}').should.be.a('boolean');
req.end(function(res){
res.text.should.equal('{"name":"tobi"}');
});
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.
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;
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('tobo', '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.createWeadStream('path/to/my.json');
var req = request.get('/some.json');
req.pipe(stream);
Multipart requests
Super Agent is also great for building multipart requests, providing a both low-level and high-level APIs.
The low-level API uses Parts to represent a file or field. The .part() method returns a new Part, which provides an API similar to the request itself.
var req = request.post('/upload');
req.part()
.set('Content-Type', 'image/png')
.set('Content-Disposition', 'attachment; filename="myimage.png"')
.write('some image data')
.write('some more image data');
req.part()
.set('Content-Disposition', 'form-data; name="name"')
.set('Content-Type', 'text/plain')
.write('tobi');
req.end(callback);
Attaching files
As mentioned a higher-level API is also provided, in the form of .attach(file[, 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('path/to/tobi.png', 'Tobi.png')
.attach('path/to/loki.png')
.attach('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('path/to/tobi.png')
.attach('path/to/loki.png')
.attach('path/to/jane.png')
.end(callback);
Compression
The Node client supports compressed responses, best of all, you don’t have to do anything! It just works.