|  |  |  |  | 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 mime = require('mime-types'); | 
					
						
							|  |  |  |  | var asynckit = require('asynckit'); | 
					
						
							|  |  |  |  | var populate = require('./populate.js'); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // Public API
 | 
					
						
							|  |  |  |  | module.exports = FormData; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // make it a Stream
 | 
					
						
							|  |  |  |  | util.inherits(FormData, CombinedStream); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /** | 
					
						
							|  |  |  |  |  * Create readable "multipart/form-data" streams. | 
					
						
							|  |  |  |  |  * Can be used to submit forms | 
					
						
							|  |  |  |  |  * and file uploads to other web applications. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * @constructor | 
					
						
							|  |  |  |  |  * @param {Object} options - Properties to be added/overriden for FormData and CombinedStream | 
					
						
							|  |  |  |  |  */ | 
					
						
							|  |  |  |  | function FormData(options) { | 
					
						
							|  |  |  |  |   if (!(this instanceof FormData)) { | 
					
						
							|  |  |  |  |     return new FormData(); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   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 || {}; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // allow filename as single option
 | 
					
						
							|  |  |  |  |   if (typeof options == 'string') { | 
					
						
							|  |  |  |  |     options = {filename: options}; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   var append = CombinedStream.prototype.append.bind(this); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // all that streamy business can't handle numbers
 | 
					
						
							|  |  |  |  |   if (typeof value == 'number') { | 
					
						
							|  |  |  |  |     value = '' + value; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // https://github.com/felixge/node-form-data/issues/38
 | 
					
						
							|  |  |  |  |   if (util.isArray(value)) { | 
					
						
							|  |  |  |  |     // Please convert your array into string
 | 
					
						
							|  |  |  |  |     // the way web server expects it
 | 
					
						
							|  |  |  |  |     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); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // pass along options.knownLength
 | 
					
						
							|  |  |  |  |   this._trackLength(header, value, options); | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | FormData.prototype._trackLength = function(header, value, options) { | 
					
						
							|  |  |  |  |   var valueLength = 0; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // used w/ getLengthSync(), when length is known.
 | 
					
						
							|  |  |  |  |   // e.g. for streaming directly from a remote server,
 | 
					
						
							|  |  |  |  |   // w/ a known file a size, and not wanting to wait for
 | 
					
						
							|  |  |  |  |   // incoming file to finish to get its size.
 | 
					
						
							|  |  |  |  |   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; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // @check why add CRLF? does this account for custom/multiple CRLFs?
 | 
					
						
							|  |  |  |  |   this._overheadLength += | 
					
						
							|  |  |  |  |     Buffer.byteLength(header) + | 
					
						
							|  |  |  |  |     FormData.LINE_BREAK.length; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // empty or either doesn't have path or not an http response
 | 
					
						
							|  |  |  |  |   if (!value || ( !value.path && !(value.readable && value.hasOwnProperty('httpVersion')) )) { | 
					
						
							|  |  |  |  |     return; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // no need to bother with the length
 | 
					
						
							|  |  |  |  |   if (!options.knownLength) { | 
					
						
							|  |  |  |  |     this._valuesToMeasure.push(value); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | FormData.prototype._lengthRetriever = function(value, callback) { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (value.hasOwnProperty('fd')) { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // take read range into a account
 | 
					
						
							|  |  |  |  |     // `end` = Infinity –> read file till the end
 | 
					
						
							|  |  |  |  |     //
 | 
					
						
							|  |  |  |  |     // TODO: Looks like there is bug in Node fs.createReadStream
 | 
					
						
							|  |  |  |  |     // it doesn't respect `end` options without `start` options
 | 
					
						
							|  |  |  |  |     // Fix it when node fixes it.
 | 
					
						
							|  |  |  |  |     // https://github.com/joyent/node/issues/7819
 | 
					
						
							|  |  |  |  |     if (value.end != undefined && value.end != Infinity && value.start != undefined) { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       // when end specified
 | 
					
						
							|  |  |  |  |       // no need to calculate range
 | 
					
						
							|  |  |  |  |       // inclusive, starts with 0
 | 
					
						
							|  |  |  |  |       callback(null, value.end + 1 - (value.start ? value.start : 0)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // not that fast snoopy
 | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |       // still need to fetch file size from fs
 | 
					
						
							|  |  |  |  |       fs.stat(value.path, function(err, stat) { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         var fileSize; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (err) { | 
					
						
							|  |  |  |  |           callback(err); | 
					
						
							|  |  |  |  |           return; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         // update final size based on the range options
 | 
					
						
							|  |  |  |  |         fileSize = stat.size - (value.start ? value.start : 0); | 
					
						
							|  |  |  |  |         callback(null, fileSize); | 
					
						
							|  |  |  |  |       }); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // or http response
 | 
					
						
							|  |  |  |  |   } else if (value.hasOwnProperty('httpVersion')) { | 
					
						
							|  |  |  |  |     callback(null, +value.headers['content-length']); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // or request stream http://github.com/mikeal/request
 | 
					
						
							|  |  |  |  |   } else if (value.hasOwnProperty('httpModule')) { | 
					
						
							|  |  |  |  |     // wait till response come back
 | 
					
						
							|  |  |  |  |     value.on('response', function(response) { | 
					
						
							|  |  |  |  |       value.pause(); | 
					
						
							|  |  |  |  |       callback(null, +response.headers['content-length']); | 
					
						
							|  |  |  |  |     }); | 
					
						
							|  |  |  |  |     value.resume(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // something else
 | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     callback('Unknown stream'); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | FormData.prototype._multiPartHeader = function(field, value, options) { | 
					
						
							|  |  |  |  |   // custom header specified (as string)?
 | 
					
						
							|  |  |  |  |   // it becomes responsible for boundary
 | 
					
						
							|  |  |  |  |   // (e.g. to handle extra CRLFs on .NET servers)
 | 
					
						
							|  |  |  |  |   if (typeof options.header == 'string') { | 
					
						
							|  |  |  |  |     return options.header; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   var contentDisposition = this._getContentDisposition(value, options); | 
					
						
							|  |  |  |  |   var contentType = this._getContentType(value, options); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   var contents = ''; | 
					
						
							|  |  |  |  |   var headers  = { | 
					
						
							|  |  |  |  |     // add custom disposition as third element or keep it two elements if not
 | 
					
						
							|  |  |  |  |     'Content-Disposition': ['form-data', 'name="' + field + '"'].concat(contentDisposition || []), | 
					
						
							|  |  |  |  |     // if no content type. allow it to be empty array
 | 
					
						
							|  |  |  |  |     'Content-Type': [].concat(contentType || []) | 
					
						
							|  |  |  |  |   }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // allow custom headers.
 | 
					
						
							|  |  |  |  |   if (typeof options.header == 'object') { | 
					
						
							|  |  |  |  |     populate(headers, options.header); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   var header; | 
					
						
							|  |  |  |  |   for (var prop in headers) { | 
					
						
							|  |  |  |  |     if (!headers.hasOwnProperty(prop)) continue; | 
					
						
							|  |  |  |  |     header = headers[prop]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // skip nullish headers.
 | 
					
						
							|  |  |  |  |     if (header == null) { | 
					
						
							|  |  |  |  |       continue; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // convert all headers to arrays.
 | 
					
						
							|  |  |  |  |     if (!Array.isArray(header)) { | 
					
						
							|  |  |  |  |       header = [header]; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // add non-empty headers.
 | 
					
						
							|  |  |  |  |     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') { | 
					
						
							|  |  |  |  |     // custom filepath for relative paths
 | 
					
						
							|  |  |  |  |     filename = path.normalize(options.filepath).replace(/\\/g, '/'); | 
					
						
							|  |  |  |  |   } else if (options.filename || value.name || value.path) { | 
					
						
							|  |  |  |  |     // custom filename take precedence
 | 
					
						
							|  |  |  |  |     // formidable and the browser add a name property
 | 
					
						
							|  |  |  |  |     // fs- and request- streams have path property
 | 
					
						
							|  |  |  |  |     filename = path.basename(options.filename || value.name || value.path); | 
					
						
							|  |  |  |  |   } else if (value.readable && value.hasOwnProperty('httpVersion')) { | 
					
						
							|  |  |  |  |     // or try http response
 | 
					
						
							|  |  |  |  |     filename = path.basename(value.client._httpMessage.path || ''); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   if (filename) { | 
					
						
							|  |  |  |  |     contentDisposition = 'filename="' + filename + '"'; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return contentDisposition; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | FormData.prototype._getContentType = function(value, options) { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // use custom content-type above all
 | 
					
						
							|  |  |  |  |   var contentType = options.contentType; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // or try `name` from formidable, browser
 | 
					
						
							|  |  |  |  |   if (!contentType && value.name) { | 
					
						
							|  |  |  |  |     contentType = mime.lookup(value.name); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // or try `path` from fs-, request- streams
 | 
					
						
							|  |  |  |  |   if (!contentType && value.path) { | 
					
						
							|  |  |  |  |     contentType = mime.lookup(value.path); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // or if it's http-reponse
 | 
					
						
							|  |  |  |  |   if (!contentType && value.readable && value.hasOwnProperty('httpVersion')) { | 
					
						
							|  |  |  |  |     contentType = value.headers['content-type']; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // or guess it from the filepath or filename
 | 
					
						
							|  |  |  |  |   if (!contentType && (options.filepath || options.filename)) { | 
					
						
							|  |  |  |  |     contentType = mime.lookup(options.filepath || options.filename); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // fallback to the default content type if `value` is not simple value
 | 
					
						
							|  |  |  |  |   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.getBoundary = function() { | 
					
						
							|  |  |  |  |   if (!this._boundary) { | 
					
						
							|  |  |  |  |     this._generateBoundary(); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return this._boundary; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | FormData.prototype.getBuffer = function() { | 
					
						
							|  |  |  |  |   var dataBuffer = new Buffer.alloc( 0 ); | 
					
						
							|  |  |  |  |   var boundary = this.getBoundary(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // Create the form content. Add Line breaks to the end of data.
 | 
					
						
							|  |  |  |  |   for (var i = 0, len = this._streams.length; i < len; i++) { | 
					
						
							|  |  |  |  |     if (typeof this._streams[i] !== 'function') { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       // Add content to the buffer.
 | 
					
						
							|  |  |  |  |       if(Buffer.isBuffer(this._streams[i])) { | 
					
						
							|  |  |  |  |         dataBuffer = Buffer.concat( [dataBuffer, this._streams[i]]); | 
					
						
							|  |  |  |  |       }else { | 
					
						
							|  |  |  |  |         dataBuffer = Buffer.concat( [dataBuffer, Buffer.from(this._streams[i])]); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       // Add break after content.
 | 
					
						
							|  |  |  |  |       if (typeof this._streams[i] !== 'string' || this._streams[i].substring( 2, boundary.length + 2 ) !== boundary) { | 
					
						
							|  |  |  |  |         dataBuffer = Buffer.concat( [dataBuffer, Buffer.from(FormData.LINE_BREAK)] ); | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // Add the footer and return the Buffer object.
 | 
					
						
							|  |  |  |  |   return Buffer.concat( [dataBuffer, Buffer.from(this._lastBoundary())] ); | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | FormData.prototype._generateBoundary = function() { | 
					
						
							|  |  |  |  |   // This generates a 50 character boundary similar to those used by Firefox.
 | 
					
						
							|  |  |  |  |   // They are optimized for boyer-moore parsing.
 | 
					
						
							|  |  |  |  |   var boundary = '--------------------------'; | 
					
						
							|  |  |  |  |   for (var i = 0; i < 24; i++) { | 
					
						
							|  |  |  |  |     boundary += Math.floor(Math.random() * 10).toString(16); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   this._boundary = boundary; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // Note: getLengthSync DOESN'T calculate streams length
 | 
					
						
							|  |  |  |  | // As workaround one can calculate file size manually
 | 
					
						
							|  |  |  |  | // and add it as knownLength option
 | 
					
						
							|  |  |  |  | FormData.prototype.getLengthSync = function() { | 
					
						
							|  |  |  |  |   var knownLength = this._overheadLength + this._valueLength; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // Don't get confused, there are 3 "internal" streams for each keyval pair
 | 
					
						
							|  |  |  |  |   // so it basically checks if there is any value added to the form
 | 
					
						
							|  |  |  |  |   if (this._streams.length) { | 
					
						
							|  |  |  |  |     knownLength += this._lastBoundary().length; | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // https://github.com/form-data/form-data/issues/40
 | 
					
						
							|  |  |  |  |   if (!this.hasKnownLength()) { | 
					
						
							|  |  |  |  |     // Some async length retrievers are present
 | 
					
						
							|  |  |  |  |     // therefore synchronous length calculation is false.
 | 
					
						
							|  |  |  |  |     // Please use getLength(callback) to get proper length
 | 
					
						
							|  |  |  |  |     this._error(new Error('Cannot calculate proper length in synchronous way.')); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   return knownLength; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // Public API to check if length of added values is known
 | 
					
						
							|  |  |  |  | // https://github.com/form-data/form-data/issues/196
 | 
					
						
							|  |  |  |  | // https://github.com/form-data/form-data/issues/262
 | 
					
						
							|  |  |  |  | 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'} | 
					
						
							|  |  |  |  |     ; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // parse provided url if it's string
 | 
					
						
							|  |  |  |  |   // or treat it as options object
 | 
					
						
							|  |  |  |  |   if (typeof params == 'string') { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     params = parseUrl(params); | 
					
						
							|  |  |  |  |     options = populate({ | 
					
						
							|  |  |  |  |       port: params.port, | 
					
						
							|  |  |  |  |       path: params.pathname, | 
					
						
							|  |  |  |  |       host: params.hostname, | 
					
						
							|  |  |  |  |       protocol: params.protocol | 
					
						
							|  |  |  |  |     }, defaults); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // use custom params
 | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     options = populate(params, defaults); | 
					
						
							|  |  |  |  |     // if no port provided use default one
 | 
					
						
							|  |  |  |  |     if (!options.port) { | 
					
						
							|  |  |  |  |       options.port = options.protocol == 'https:' ? 443 : 80; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // put that good code in getHeaders to some use
 | 
					
						
							|  |  |  |  |   options.headers = this.getHeaders(params.headers); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // https if specified, fallback to http in any other case
 | 
					
						
							|  |  |  |  |   if (options.protocol == 'https:') { | 
					
						
							|  |  |  |  |     request = https.request(options); | 
					
						
							|  |  |  |  |   } else { | 
					
						
							|  |  |  |  |     request = http.request(options); | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   // get content length and fire away
 | 
					
						
							|  |  |  |  |   this.getLength(function(err, length) { | 
					
						
							|  |  |  |  |     if (err) { | 
					
						
							|  |  |  |  |       this._error(err); | 
					
						
							|  |  |  |  |       return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     // add content length
 | 
					
						
							|  |  |  |  |     request.setHeader('Content-Length', length); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     this.pipe(request); | 
					
						
							|  |  |  |  |     if (cb) { | 
					
						
							|  |  |  |  |       request.on('error', cb); | 
					
						
							|  |  |  |  |       request.on('response', cb.bind(this, null)); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   }.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]'; | 
					
						
							|  |  |  |  | }; |