DefinitelyTyped/types/request/request-tests.ts
Eitan Levi 4fe80e3fcc [request] Specify types allowed for Multipart body (#38190)
The request library allows anything to be put in a Multipart body as
long as it can be passed to `Buffer.from()`: https://github.com/request/request/blob/master/lib/multipart.js#L74

However the current implementation allows only `string` to be used as
the multipart body. In my use-case I needed to pass a `Buffer` instance
directly.

This change defines a type alias for `MultipartBody` that includes
`string`, `Buffer`, `ArrayBuffer` or `UInt8Array`.
2019-09-08 17:24:44 -07:00

830 lines
23 KiB
TypeScript

import FormData = require('form-data');
import fs = require('fs');
import http = require('http');
import path = require('path');
import qs = require('querystring');
import request = require('request');
import stream = require('stream');
import urlModule = require('url');
import constants = require('constants');
let value: any;
let str: string;
let strOrUndef: string | undefined;
let strOrTrueOrUndef: string | true | undefined;
const buffer: Buffer = new Buffer('foo');
let num = 0;
let bool: boolean;
let obj: object;
const dest = 'foo';
const uri = 'foo-bar';
let headers: request.Headers = {};
let write: stream.Writable = new stream.Writable();
let req: request.Request = request(uri, function callback() {});
let res: request.Response;
let form: FormData;
const bodyArr: request.RequestPart[] = [{
body: value
}, {
body: value
}, {
body: value
}];
// Defaults tests
(() => {
const githubUrl = 'https://github.com';
const defaultJarRequest = request.defaults({ jar: true });
defaultJarRequest.get(githubUrl);
// defaultJarRequest(); //this line doesn't compile (and shouldn't)
const defaultUrlRequest = request.defaults({ url: githubUrl });
defaultUrlRequest();
defaultUrlRequest.get();
const defaultBodyRequest = defaultUrlRequest.defaults({body: '{}', json: true});
defaultBodyRequest.get();
defaultBodyRequest.post();
defaultBodyRequest.put();
const defaultJarRequestCustomJar = request.defaults({ jar: request.jar() });
})();
// --- --- --- --- --- --- --- --- --- --- --- ---
obj = req.toJSON();
let cookie: request.Cookie = request.cookie('foo')!;
str = cookie.key;
str = cookie.value;
const expires: Date | 'Infinity' = cookie.expires;
const cpath: string | null = cookie.path;
str = cookie.toString();
bool = cookie.httpOnly;
let jar: request.CookieJar = request.jar();
jar.setCookie(cookie, uri);
str = jar.getCookieString(uri);
const cookies: request.Cookie[] = jar.getCookies(uri);
const aws: request.AWSOptions = { secret: 'foo' };
str = aws.secret;
strOrUndef = aws.bucket;
let oauth: request.OAuthOptions = { body_hash: 'foo' };
strOrUndef = oauth.callback;
strOrUndef = oauth.consumer_key;
strOrUndef = oauth.consumer_secret;
strOrUndef = oauth.token;
strOrUndef = oauth.token_secret;
strOrUndef = oauth.transport_method;
strOrUndef = oauth.verifier;
strOrTrueOrUndef = oauth.body_hash;
let options: request.Options = {
url: str,
uri: str,
callback: (error: any, response: any, body: any) => {},
jar: value,
form: obj,
oauth: value,
aws,
qs: obj,
json: value,
jsonReviver: (key: string, value: any) => {},
jsonReplacer: (key: string, value: any) => {},
multipart: value,
agent: new http.Agent(),
agentOptions: value,
agentClass: value,
forever: value,
host: str,
port: num,
method: str,
headers: value,
body: value,
followRedirect: bool,
followAllRedirects: bool,
maxRedirects: num,
encoding: str,
pool: value,
timeout: num,
proxy: value,
tunnel: bool,
strictSSL: bool,
rejectUnauthorized: false
};
// Below line has compile error, use OptionsWithUri or OptionsWithUrl instead. See #7979.
// options.uri = str;
const opt: request.OptionsWithUri = {
baseUrl: 'http://localhost',
uri: 'bar'
};
opt.uri = str;
// --- --- --- --- --- --- --- --- --- --- --- ---
let strOrFalse: string | false;
strOrFalse = req.setHeader(str, str);
strOrFalse = req.setHeader(str, str, bool);
req.setHeader({str});
strOrFalse = req.hasHeader(str);
strOrUndef = req.getHeader(str);
bool = req.removeHeader(str);
req = req.qs(obj);
req = req.qs(obj, bool);
req = req.form(obj);
form = req.form();
req = req.multipart(bodyArr);
req = req.json(value);
req = req.aws(aws);
req = req.aws(aws, bool);
req = req.oauth(oauth);
req = req.jar(jar);
write = req.pipe(write);
write = req.pipe(write, value);
req.pipe(req);
req.pipeDest(dest);
req.write(value);
req.end(str);
req.end(buffer);
req.pause();
req.resume();
req.abort();
req.destroy();
strOrUndef = req.host;
str = req.method;
str = req.path;
str = req.href;
str = req.uri.href;
str = req.uri.pathname;
value = req.body;
headers = req.headers;
value = req.headers['foo'];
if (req.response) {
res = req.response;
}
// --- --- --- --- --- --- --- --- --- --- --- ---
req = request(uri);
req = request(uri, options);
req = request(uri, options, callback);
req = request(uri, callback);
req = request(options);
req = request(options, callback);
req = request.get(uri);
req = request.get(uri, options);
req = request.get(uri, options, callback);
req = request.get(uri, callback);
req = request.get(options);
req = request.get(options, callback);
req = request.post(uri);
req = request.post(uri, options);
req = request.post(uri, options, callback);
req = request.post(uri, callback);
req = request.post(options);
req = request.post(options, callback);
req = request.put(uri);
req = request.put(uri, options);
req = request.put(uri, options, callback);
req = request.put(uri, callback);
req = request.put(options);
req = request.put(options, callback);
req = request.head(uri);
req = request.head(uri, options);
req = request.head(uri, options, callback);
req = request.head(uri, callback);
req = request.head(options);
req = request.head(options, callback);
req = request.patch(uri);
req = request.patch(uri, options);
req = request.patch(uri, options, callback);
req = request.patch(uri, callback);
req = request.patch(options);
req = request.patch(options, callback);
req = request.del(uri);
req = request.del(uri, options);
req = request.del(uri, options, callback);
req = request.del(uri, callback);
req = request.del(options);
req = request.del(options, callback);
req = request.delete(uri);
req = request.delete(uri, options);
req = request.delete(uri, options, callback);
req = request.delete(uri, callback);
req = request.delete(options);
req = request.delete(options, callback);
req = request.forever(value, value);
jar = request.jar();
const r = request.defaults(options);
r(str);
r.get(str);
r.post(str);
r(options);
r.get(options);
r.post(options);
request
.get('http://example.com/example.png')
.on('response', (response) => {
res = response;
num = response.statusCode;
str = response.statusMessage;
req = response.request;
value = response.body;
strOrUndef = response.caseless.get('foo');
strOrFalse = response.caseless.has('content-type');
if (response.timings) {
num = response.timings.socket;
num = response.timings.lookup;
num = response.timings.connect;
num = response.timings.response;
num = response.timings.end;
}
if (response.timingPhases) {
num = response.timingPhases.wait;
num = response.timingPhases.dns;
num = response.timingPhases.tcp;
num = response.timingPhases.firstByte;
num = response.timingPhases.download;
num = response.timingPhases.total;
}
})
.pipe(request.put('http://another.com/another.png'));
// The following examples from https://github.com/request/request
request('http://www.google.com', (error, response, body) => {
if (!error && response.statusCode === 200) {
console.log(body); // Show the HTML for the Google homepage.
}
});
request('http://google.com/doodle.png').pipe(fs.createWriteStream('doodle.png'));
fs.createReadStream('file.json').pipe(request.put('http://mysite.com/obj.json'));
request.get('http://google.com/img.png').pipe(request.put('http://mysite.com/img.png'));
request
.get('http://google.com/img.png')
.on('response', (response) => {
console.log(response.statusCode); // 200
console.log(response.headers['content-type']); // 'image/png'
})
.pipe(request.put('http://mysite.com/img.png'));
request
.get('http://mysite.com/doodle.png')
.on('error', (err: any) => {
console.log(err);
})
.pipe(fs.createWriteStream('doodle.png'));
http.createServer((req, resp) => {
if (req.url === '/doodle.png') {
switch (req.method) {
case 'PUT':
req.pipe(request.put('http://mysite.com/doodle.png'));
break;
case 'GET':
case 'HEAD':
request.get('http://mysite.com/doodle.png').pipe(resp);
}
}
});
http.createServer((req, resp) => {
if (req.url === '/doodle.png') {
const x = request('http://mysite.com/doodle.png');
req.pipe(x);
x.pipe(resp);
}
});
http.createServer((req, resp) => {
req.pipe(request('http://mysite.com/doodle.png')).pipe(resp);
});
http.createServer((req, resp) => {
if (req.url === '/doodle.png') {
r.get('http://google.com/doodle.png').pipe(resp);
}
});
request.post('http://service.com/upload', {form: {key: 'value'}});
// or
request.post('http://service.com/upload').form({key: 'value'});
// or
request.post({url: 'http://service.com/upload', form: {key: 'value'}}, (err, httpResponse, body) => { /* ... */ });
const data = {
// Pass a simple key-value pair
my_field: 'my_value',
// Pass data via Buffers
my_buffer: new Buffer([1, 2, 3]),
// Pass data via Streams
my_file: fs.createReadStream(__dirname + '/unicycle.jpg'),
// Pass multiple values /w an Array
attachments: [
fs.createReadStream(__dirname + '/attachment1.jpg'),
fs.createReadStream(__dirname + '/attachment2.jpg')
],
// Pass optional meta-data with an 'options' object with style: {value: DATA, options: OPTIONS}
// Use case: for some types of streams, you'll need to provide "file"-related information manually.
// See the `form-data` README for more information about options: https://github.com/felixge/node-form-data
custom_file: {
value: fs.createReadStream('/dev/urandom'),
options: {
filename: 'topsecret.jpg',
contentType: 'image/jpg'
}
}
};
request.post({url: 'http://service.com/upload', formData: data}, function optionalCallback(err, httpResponse, body) {
if (err) {
console.error('upload failed:', err);
return;
}
console.log('Upload successful! Server responded with:', body);
});
const requestMultipart = request.post('http://service.com/upload', function optionalCallback(err, httpResponse, body) {});
form = requestMultipart.form();
form.append('my_field', 'my_value');
form.append('my_buffer', new Buffer([1, 2, 3]));
form.append('custom_file', fs.createReadStream(__dirname + '/unicycle.jpg'), {filename: 'unicycle.jpg'});
request({
method: 'PUT',
preambleCRLF: true,
postambleCRLF: true,
uri: 'http://service.com/upload',
multipart: {
chunked: false,
data: [
{
'content-type': 'application/json',
body: JSON.stringify({foo: 'bar', _attachments: {'message.txt': {follows: true, length: 18, content_type: 'text/plain' }}})
},
{ body: 'I am an attachment' }
]
}
},
(error, response, body) => {
if (error) {
console.error('upload failed:', error);
return;
}
console.log('Upload successful! Server responded with:', body);
});
request({
method: 'PUT',
preambleCRLF: true,
postambleCRLF: true,
uri: 'http://service.com/upload',
multipart: [
{
headers: { 'content-type': 'application/json' },
body: JSON.stringify({foo: 'bar', _attachments: {'message.txt': {follows: true, length: 18, content_type: 'text/plain' }}})
},
{ body: 'I am an attachment' },
{ body: fs.createReadStream('image.png') }
]
},
(error, response, body) => {
if (error) {
console.error('upload failed:', error);
return;
}
console.log('Upload successful! Server responded with:', body);
});
request.get('http://some.server.com/').auth('username', 'password', false);
// or
request.get('http://some.server.com/', {
auth: {
user: 'username',
pass: 'password',
sendImmediately: false
}
});
// or
request.get('http://some.server.com/').auth('foo', 'bar', true, 'bearerToken');
// or
request.get('http://some.server.com/', {
auth: {
bearer: 'bearerToken'
}
});
// or
request.get('http://some.server.com/', {
auth: {
bearer: () => 'bearerToken'
}
});
const username = 'username';
const password = 'password';
let url = `http://'${username}:${password}'@some.server.com`;
request({url}, (error, response, body) => {
// Do more stuff with 'body' here
});
options = {
url: 'https://api.github.com/repos/request/request',
headers: {
'User-Agent': 'request'
}
};
function callback(error: any, response: http.IncomingMessage, body: string) {
if (!error && response.statusCode === 200) {
const info = JSON.parse(body);
console.log(info.stargazers_count + " Stars");
console.log(info.forks_count + " Forks");
}
}
request(options, callback);
// OAuth1.0 - 3-legged server side flow (Twitter example)
// step 1
const CONSUMER_KEY = 'key';
const CONSUMER_SECRET = 'secret';
oauth = {
callback: 'http://mysite.com/callback/',
consumer_key: CONSUMER_KEY,
consumer_secret: CONSUMER_SECRET,
transport_method: 'header'
};
url = 'https://api.twitter.com/oauth/request_token';
request.post({url, oauth}, (e, r, body) => {
// Ideally, you would take the body in the response
// and construct a URL that a user clicks on (like a sign in button).
// The verifier is only available in the response after a user has
// verified with twitter that they are authorizing your app.
// step 2
const req_data = qs.parse(body);
const uri = `https://api.twitter.com/oauth/authenticate?${qs.stringify({oauth_token: req_data.oauth_token})}`;
// redirect the user to the authorize uri
// step 3
// after the user is redirected back to your server
const auth_data: any = qs.parse(body);
const oauth = {
consumer_key: CONSUMER_KEY,
consumer_secret: CONSUMER_SECRET,
token: auth_data.oauth_token,
token_secret: req_data.oauth_token_secret as string,
verifier: auth_data.oauth_verifier
};
const url = 'https://api.twitter.com/oauth/access_token';
request.post({url, oauth}, (e, r, body) => {
// ready to make signed requests on behalf of the user
const perm_data: any = qs.parse(body);
const oauth = {
consumer_key: CONSUMER_KEY,
consumer_secret: CONSUMER_SECRET,
token: perm_data.oauth_token,
token_secret: perm_data.oauth_token_secret
};
const url = 'https://api.twitter.com/1.1/users/show.json';
const query = {
screen_name: perm_data.screen_name,
user_id: perm_data.user_id
};
request.get({url, oauth, qs: query, json: true}, (e, r, user) => {
console.log(user);
});
});
});
const certFile = path.resolve(__dirname, 'ssl/client.crt');
const keyFile = path.resolve(__dirname, 'ssl/client.key');
const caFile = path.resolve(__dirname, 'ssl/ca.cert.pem');
options = {
url: 'https://api.some-server.com/',
cert: fs.readFileSync(certFile),
key: fs.readFileSync(keyFile),
passphrase: 'password',
ca: fs.readFileSync(caFile)
};
request.get(options);
options = {
url: 'https://api.some-server.com/',
agentOptions: {
cert: fs.readFileSync(certFile),
key: fs.readFileSync(keyFile),
// Or use `pfx` property replacing `cert` and `key` when using private key, certificate and CA certs in PFX or PKCS12 format:
// pfx: fs.readFileSync(pfxFilePath),
passphrase: 'password',
secureOptions: constants.SSL_OP_NO_SSLv3
}
};
request.get(options);
request.get({
url: 'https://api.some-server.com/',
agentOptions: {
secureProtocol: 'SSLv3_method',
maxCachedSessions: 3,
keepAlive: true,
}
});
request.get({
url: 'https://api.some-server.com/',
agentOptions: {
ca: fs.readFileSync('ca.cert.pem')
}
});
request({
// will be ignored
method: 'GET',
uri: 'http://www.google.com',
// HTTP Archive Request Object
har: {
url: 'http://www.mockbin.com/har',
method: 'POST',
headers: [
{
name: 'content-type',
value: 'application/x-www-form-urlencoded'
}
],
postData: {
mimeType: 'application/x-www-form-urlencoded',
params: [
{
name: 'foo',
value: 'bar'
},
{
name: 'hello',
value: 'world'
}
]
}
}
});
// requests using baseRequest() will set the 'x-token' header
const baseRequest = request.defaults({
headers: {'x-token': 'my-token'}
});
// requests using specialRequest() will include the 'x-token' header set in
// baseRequest and will also include the 'special' header
const specialRequest = baseRequest.defaults({
headers: {special: 'special value'}
});
const urlRequest = specialRequest.defaults({url: 'https://github.com'});
urlRequest({}, (error, response, body) => { console.log(body); });
request.put(url);
request.patch(url);
request.post(url);
request.head(url);
request.del(url);
request.delete(url);
request.get(url);
request.cookie('key1=value1');
request.jar();
request.debug = true;
request.get('http://10.255.255.1', {timeout: 1500}, (err) => {
console.log(err.code === 'ETIMEDOUT');
// Set to `true` if the timeout was a connection timeout, `false` or
// `undefined` otherwise.
console.log(err.connect === true);
process.exit(0);
});
const rand = Math.floor(Math.random() * 100000000).toString();
request(
{ method: 'PUT'
, uri: 'http://mikeal.iriscouch.com/testjs/' + rand
, multipart:
[ { headers: { 'content-type': 'application/json' }
, body: JSON.stringify({foo: 'bar', _attachments: {'message.txt': {follows: true, length: 18, content_type: 'text/plain' }}})
}
, { body: 'I am an attachment' }
]
}
, (error, response, body) => {
if (response.statusCode === 201) {
console.log('document saved as: http://mikeal.iriscouch.com/testjs/' + rand);
} else {
console.log('error: ' + response.statusCode);
console.log(body);
}
}
);
request(
{
method: 'PUT',
uri: 'http://mikeal.iriscouch.com/testjs/' + rand,
multipart: {
data: [
{
'content-type': 'application/json; charset=utf-8',
body: JSON.stringify({
_attachments: {
'dot.png': {
follows: true,
length: 269,
content_type: 'image/png',
},
},
}),
},
{
'content-type': 'image/png',
body: Buffer.from(
'iVBORw0KGgoAAAANSUhEUgAAAAIAAAACCAIAAAD91JpzAAAAA3NCSVQICAjb4U/gAAAAX3pUWHRSYXcgcHJvZmlsZSB0eXBlIEFQUDEAAAiZ40pPzUstykxWKCjKT8vMSeVSAANjEy4TSxNL' +
'o0QDAwMLAwgwNDAwNgSSRkC2OVQo0QAFmJibpQGhuVmymSmIzwUAT7oVaBst2IwAAAAWSURBVAiZY/z//z8DAwMTAwMDAwMDACQGAwGaMKL7AAAAAElFTkSuQmCC',
),
},
],
},
},
(error, response, body) => {
if (response.statusCode === 201) {
console.log('image saved as http://mikeal.iriscouch.com/testjs/' + rand);
} else {
console.log('error: ' + response.statusCode);
console.log(body);
}
},
);
request(
{ method: 'GET'
, uri: 'http://www.google.com'
, gzip: true
}
, (error, response, body) => {
// body is the decompressed response body
console.log('server encoded the data as: ' + (response.headers['content-encoding'] || 'identity'));
console.log('the decoded data is: ' + body);
}
).on('data', (data: any) => {
// decompressed data as it is received
console.log('decoded chunk: ' + data);
})
.on('response', (response: http.IncomingMessage) => {
// unmodified http.IncomingMessage object
response.on('data', (data: any[]) => {
// compressed data as it is received
console.log(`received ${data.length} bytes of compressed data`);
});
});
let requestWithJar = request.defaults({jar: true});
requestWithJar('http://www.google.com', () => {
requestWithJar('http://images.google.com');
});
jar = request.jar();
requestWithJar = request.defaults({jar});
requestWithJar('http://www.google.com', () => {
requestWithJar('http://images.google.com');
});
jar = request.jar();
cookie = request.cookie('key1=value1')!;
url = 'http://www.google.com';
jar.setCookie(cookie, url);
request({url, jar}, () => {
request('http://images.google.com');
});
// TODO: add definitions for tough-cookie-filestore
// var FileCookieStore = require('tough-cookie-filestore');
// NOTE - currently the 'cookies.json' file must already exist!
// var j = request.jar(new FileCookieStore('cookies.json'));
requestWithJar = request.defaults({ jar });
request('http://www.google.com', () => {
request('http://images.google.com');
});
jar = request.jar();
request({url: 'http://www.google.com', jar}, () => {
const cookie_string = jar.getCookieString(url); // "key1=value1; key2=value2; ..."
const cookies = jar.getCookies(url);
// [{key: 'key1', value: 'value1', domain: "www.google.com", ...}, ...]
});
request(
{ method: 'GET'
, uri: 'http://www.google.com'
, gzip: true
}
)
.on('request', (req: http.ClientRequest) => { })
.on('response', (resp: http.IncomingMessage) => { })
.on('data', (data: Buffer | string) => { })
.on('error', (e: Error) => { })
.on('complete', (resp: http.IncomingMessage, body?: string | Buffer) => { });
// options.url / options.uri can be the Url object
request.get({
url: urlModule.parse('http://example.com')
});
request.get({
uri: urlModule.parse('http://example.com')
});
const requestWithOptionalUri = request.defaults({ uri: 'http://example.com' });
requestWithOptionalUri();
requestWithOptionalUri({});
requestWithOptionalUri({ uri: 'http://example.com' });
requestWithOptionalUri({ uri: urlModule.parse('http://example.com') });
requestWithOptionalUri({ url: 'http://example.com' });
requestWithOptionalUri({ url: urlModule.parse('http://example.com') });
requestWithOptionalUri('http://example.com');
requestWithOptionalUri(() => {});
requestWithOptionalUri.get();
requestWithOptionalUri.get(() => {});
requestWithOptionalUri.get('http://example.com');
requestWithOptionalUri.get({});
requestWithOptionalUri.get({
url: 'http://example.com'
});
requestWithOptionalUri.get({
uri: 'http://example.com'
});
requestWithOptionalUri.get({
url: urlModule.parse('http://example.com')
});
requestWithOptionalUri.get({
uri: urlModule.parse('http://example.com')
});
//// initParams
const optionsWithoutUriUrl: request.CoreOptions = {};
let optionsWithUriUrl: request.RequiredUriUrl = {
uri: 'http://example.com',
};
optionsWithUriUrl = request.initParams(uri);
optionsWithUriUrl = request.initParams(uri, optionsWithoutUriUrl);
optionsWithUriUrl = request.initParams(uri, optionsWithoutUriUrl, callback);
optionsWithUriUrl = request.initParams(uri, callback);
optionsWithUriUrl = request.initParams(optionsWithUriUrl);
optionsWithUriUrl = request.initParams(optionsWithUriUrl, callback);
const defaultsForInitParams = request.defaults({});
options = defaultsForInitParams.initParams(uri);
options = defaultsForInitParams.initParams(uri, optionsWithoutUriUrl);
options = defaultsForInitParams.initParams(uri, optionsWithoutUriUrl, callback);
options = defaultsForInitParams.initParams(uri, callback);
options = defaultsForInitParams.initParams(optionsWithUriUrl);
options = defaultsForInitParams.initParams(optionsWithUriUrl, callback);