| | var CombinedStream = require('combined-stream'); |
| | var util = require('util'); |
| | var path = require('path'); |
| | var http = require('http'); |
| | var https = require('https'); |
| | var parseUrl = require('url').parse; |
| | var fs = require('fs'); |
| | var Stream = require('stream').Stream; |
| | var mime = require('mime-types'); |
| | var asynckit = require('asynckit'); |
| | var populate = require('./populate.js'); |
| |
|
| | |
| | module.exports = FormData; |
| |
|
| | |
| | util.inherits(FormData, CombinedStream); |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | function FormData(options) { |
| | if (!(this instanceof FormData)) { |
| | return new FormData(options); |
| | } |
| |
|
| | this._overheadLength = 0; |
| | this._valueLength = 0; |
| | this._valuesToMeasure = []; |
| |
|
| | CombinedStream.call(this); |
| |
|
| | options = options || {}; |
| | for (var option in options) { |
| | this[option] = options[option]; |
| | } |
| | } |
| |
|
| | FormData.LINE_BREAK = '\r\n'; |
| | FormData.DEFAULT_CONTENT_TYPE = 'application/octet-stream'; |
| |
|
| | FormData.prototype.append = function(field, value, options) { |
| |
|
| | options = options || {}; |
| |
|
| | |
| | if (typeof options == 'string') { |
| | options = {filename: options}; |
| | } |
| |
|
| | var append = CombinedStream.prototype.append.bind(this); |
| |
|
| | |
| | if (typeof value == 'number') { |
| | value = '' + value; |
| | } |
| |
|
| | |
| | if (Array.isArray(value)) { |
| | |
| | |
| | this._error(new Error('Arrays are not supported.')); |
| | return; |
| | } |
| |
|
| | var header = this._multiPartHeader(field, value, options); |
| | var footer = this._multiPartFooter(); |
| |
|
| | append(header); |
| | append(value); |
| | append(footer); |
| |
|
| | |
| | this._trackLength(header, value, options); |
| | }; |
| |
|
| | FormData.prototype._trackLength = function(header, value, options) { |
| | var valueLength = 0; |
| |
|
| | |
| | |
| | |
| | |
| | if (options.knownLength != null) { |
| | valueLength += +options.knownLength; |
| | } else if (Buffer.isBuffer(value)) { |
| | valueLength = value.length; |
| | } else if (typeof value === 'string') { |
| | valueLength = Buffer.byteLength(value); |
| | } |
| |
|
| | this._valueLength += valueLength; |
| |
|
| | |
| | this._overheadLength += |
| | Buffer.byteLength(header) + |
| | FormData.LINE_BREAK.length; |
| |
|
| | |
| | if (!value || ( !value.path && !(value.readable && value.hasOwnProperty('httpVersion')) && !(value instanceof Stream))) { |
| | return; |
| | } |
| |
|
| | |
| | if (!options.knownLength) { |
| | this._valuesToMeasure.push(value); |
| | } |
| | }; |
| |
|
| | FormData.prototype._lengthRetriever = function(value, callback) { |
| |
|
| | if (value.hasOwnProperty('fd')) { |
| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | if (value.end != undefined && value.end != Infinity && value.start != undefined) { |
| |
|
| | |
| | |
| | |
| | callback(null, value.end + 1 - (value.start ? value.start : 0)); |
| |
|
| | |
| | } else { |
| | |
| | fs.stat(value.path, function(err, stat) { |
| |
|
| | var fileSize; |
| |
|
| | if (err) { |
| | callback(err); |
| | return; |
| | } |
| |
|
| | |
| | fileSize = stat.size - (value.start ? value.start : 0); |
| | callback(null, fileSize); |
| | }); |
| | } |
| |
|
| | |
| | } else if (value.hasOwnProperty('httpVersion')) { |
| | callback(null, +value.headers['content-length']); |
| |
|
| | |
| | } else if (value.hasOwnProperty('httpModule')) { |
| | |
| | value.on('response', function(response) { |
| | value.pause(); |
| | callback(null, +response.headers['content-length']); |
| | }); |
| | value.resume(); |
| |
|
| | |
| | } else { |
| | callback('Unknown stream'); |
| | } |
| | }; |
| |
|
| | FormData.prototype._multiPartHeader = function(field, value, options) { |
| | |
| | |
| | |
| | if (typeof options.header == 'string') { |
| | return options.header; |
| | } |
| |
|
| | var contentDisposition = this._getContentDisposition(value, options); |
| | var contentType = this._getContentType(value, options); |
| |
|
| | var contents = ''; |
| | var headers = { |
| | |
| | 'Content-Disposition': ['form-data', 'name="' + field + '"'].concat(contentDisposition || []), |
| | |
| | 'Content-Type': [].concat(contentType || []) |
| | }; |
| |
|
| | |
| | if (typeof options.header == 'object') { |
| | populate(headers, options.header); |
| | } |
| |
|
| | var header; |
| | for (var prop in headers) { |
| | if (!headers.hasOwnProperty(prop)) continue; |
| | header = headers[prop]; |
| |
|
| | |
| | if (header == null) { |
| | continue; |
| | } |
| |
|
| | |
| | if (!Array.isArray(header)) { |
| | header = [header]; |
| | } |
| |
|
| | |
| | if (header.length) { |
| | contents += prop + ': ' + header.join('; ') + FormData.LINE_BREAK; |
| | } |
| | } |
| |
|
| | return '--' + this.getBoundary() + FormData.LINE_BREAK + contents + FormData.LINE_BREAK; |
| | }; |
| |
|
| | FormData.prototype._getContentDisposition = function(value, options) { |
| |
|
| | var filename |
| | , contentDisposition |
| | ; |
| |
|
| | if (typeof options.filepath === 'string') { |
| | |
| | filename = path.normalize(options.filepath).replace(/\\/g, '/'); |
| | } else if (options.filename || value.name || value.path) { |
| | |
| | |
| | |
| | filename = path.basename(options.filename || value.name || value.path); |
| | } else if (value.readable && value.hasOwnProperty('httpVersion')) { |
| | |
| | filename = path.basename(value.client._httpMessage.path || ''); |
| | } |
| |
|
| | if (filename) { |
| | contentDisposition = 'filename="' + filename + '"'; |
| | } |
| |
|
| | return contentDisposition; |
| | }; |
| |
|
| | FormData.prototype._getContentType = function(value, options) { |
| |
|
| | |
| | var contentType = options.contentType; |
| |
|
| | |
| | if (!contentType && value.name) { |
| | contentType = mime.lookup(value.name); |
| | } |
| |
|
| | |
| | if (!contentType && value.path) { |
| | contentType = mime.lookup(value.path); |
| | } |
| |
|
| | |
| | if (!contentType && value.readable && value.hasOwnProperty('httpVersion')) { |
| | contentType = value.headers['content-type']; |
| | } |
| |
|
| | |
| | if (!contentType && (options.filepath || options.filename)) { |
| | contentType = mime.lookup(options.filepath || options.filename); |
| | } |
| |
|
| | |
| | if (!contentType && typeof value == 'object') { |
| | contentType = FormData.DEFAULT_CONTENT_TYPE; |
| | } |
| |
|
| | return contentType; |
| | }; |
| |
|
| | FormData.prototype._multiPartFooter = function() { |
| | return function(next) { |
| | var footer = FormData.LINE_BREAK; |
| |
|
| | var lastPart = (this._streams.length === 0); |
| | if (lastPart) { |
| | footer += this._lastBoundary(); |
| | } |
| |
|
| | next(footer); |
| | }.bind(this); |
| | }; |
| |
|
| | FormData.prototype._lastBoundary = function() { |
| | return '--' + this.getBoundary() + '--' + FormData.LINE_BREAK; |
| | }; |
| |
|
| | FormData.prototype.getHeaders = function(userHeaders) { |
| | var header; |
| | var formHeaders = { |
| | 'content-type': 'multipart/form-data; boundary=' + this.getBoundary() |
| | }; |
| |
|
| | for (header in userHeaders) { |
| | if (userHeaders.hasOwnProperty(header)) { |
| | formHeaders[header.toLowerCase()] = userHeaders[header]; |
| | } |
| | } |
| |
|
| | return formHeaders; |
| | }; |
| |
|
| | FormData.prototype.setBoundary = function(boundary) { |
| | this._boundary = boundary; |
| | }; |
| |
|
| | FormData.prototype.getBoundary = function() { |
| | if (!this._boundary) { |
| | this._generateBoundary(); |
| | } |
| |
|
| | return this._boundary; |
| | }; |
| |
|
| | FormData.prototype.getBuffer = function() { |
| | var dataBuffer = new Buffer.alloc( 0 ); |
| | var boundary = this.getBoundary(); |
| |
|
| | |
| | for (var i = 0, len = this._streams.length; i < len; i++) { |
| | if (typeof this._streams[i] !== 'function') { |
| |
|
| | |
| | if(Buffer.isBuffer(this._streams[i])) { |
| | dataBuffer = Buffer.concat( [dataBuffer, this._streams[i]]); |
| | }else { |
| | dataBuffer = Buffer.concat( [dataBuffer, Buffer.from(this._streams[i])]); |
| | } |
| |
|
| | |
| | if (typeof this._streams[i] !== 'string' || this._streams[i].substring( 2, boundary.length + 2 ) !== boundary) { |
| | dataBuffer = Buffer.concat( [dataBuffer, Buffer.from(FormData.LINE_BREAK)] ); |
| | } |
| | } |
| | } |
| |
|
| | |
| | return Buffer.concat( [dataBuffer, Buffer.from(this._lastBoundary())] ); |
| | }; |
| |
|
| | FormData.prototype._generateBoundary = function() { |
| | |
| | |
| | var boundary = '--------------------------'; |
| | for (var i = 0; i < 24; i++) { |
| | boundary += Math.floor(Math.random() * 10).toString(16); |
| | } |
| |
|
| | this._boundary = boundary; |
| | }; |
| |
|
| | |
| | |
| | |
| | FormData.prototype.getLengthSync = function() { |
| | var knownLength = this._overheadLength + this._valueLength; |
| |
|
| | |
| | |
| | if (this._streams.length) { |
| | knownLength += this._lastBoundary().length; |
| | } |
| |
|
| | |
| | if (!this.hasKnownLength()) { |
| | |
| | |
| | |
| | this._error(new Error('Cannot calculate proper length in synchronous way.')); |
| | } |
| |
|
| | return knownLength; |
| | }; |
| |
|
| | |
| | |
| | |
| | FormData.prototype.hasKnownLength = function() { |
| | var hasKnownLength = true; |
| |
|
| | if (this._valuesToMeasure.length) { |
| | hasKnownLength = false; |
| | } |
| |
|
| | return hasKnownLength; |
| | }; |
| |
|
| | FormData.prototype.getLength = function(cb) { |
| | var knownLength = this._overheadLength + this._valueLength; |
| |
|
| | if (this._streams.length) { |
| | knownLength += this._lastBoundary().length; |
| | } |
| |
|
| | if (!this._valuesToMeasure.length) { |
| | process.nextTick(cb.bind(this, null, knownLength)); |
| | return; |
| | } |
| |
|
| | asynckit.parallel(this._valuesToMeasure, this._lengthRetriever, function(err, values) { |
| | if (err) { |
| | cb(err); |
| | return; |
| | } |
| |
|
| | values.forEach(function(length) { |
| | knownLength += length; |
| | }); |
| |
|
| | cb(null, knownLength); |
| | }); |
| | }; |
| |
|
| | FormData.prototype.submit = function(params, cb) { |
| | var request |
| | , options |
| | , defaults = {method: 'post'} |
| | ; |
| |
|
| | |
| | |
| | if (typeof params == 'string') { |
| |
|
| | params = parseUrl(params); |
| | options = populate({ |
| | port: params.port, |
| | path: params.pathname, |
| | host: params.hostname, |
| | protocol: params.protocol |
| | }, defaults); |
| |
|
| | |
| | } else { |
| |
|
| | options = populate(params, defaults); |
| | |
| | if (!options.port) { |
| | options.port = options.protocol == 'https:' ? 443 : 80; |
| | } |
| | } |
| |
|
| | |
| | options.headers = this.getHeaders(params.headers); |
| |
|
| | |
| | if (options.protocol == 'https:') { |
| | request = https.request(options); |
| | } else { |
| | request = http.request(options); |
| | } |
| |
|
| | |
| | this.getLength(function(err, length) { |
| | if (err && err !== 'Unknown stream') { |
| | this._error(err); |
| | return; |
| | } |
| |
|
| | |
| | if (length) { |
| | request.setHeader('Content-Length', length); |
| | } |
| |
|
| | this.pipe(request); |
| | if (cb) { |
| | var onResponse; |
| |
|
| | var callback = function (error, responce) { |
| | request.removeListener('error', callback); |
| | request.removeListener('response', onResponse); |
| |
|
| | return cb.call(this, error, responce); |
| | }; |
| |
|
| | onResponse = callback.bind(this, null); |
| |
|
| | request.on('error', callback); |
| | request.on('response', onResponse); |
| | } |
| | }.bind(this)); |
| |
|
| | return request; |
| | }; |
| |
|
| | FormData.prototype._error = function(err) { |
| | if (!this.error) { |
| | this.error = err; |
| | this.pause(); |
| | this.emit('error', err); |
| | } |
| | }; |
| |
|
| | FormData.prototype.toString = function () { |
| | return '[object FormData]'; |
| | }; |
| |
|