make-middleware.js 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. var is = require('type-is')
  2. var Busboy = require('busboy')
  3. var extend = require('xtend')
  4. var appendField = require('append-field')
  5. var Counter = require('./counter')
  6. var MulterError = require('./multer-error')
  7. var FileAppender = require('./file-appender')
  8. var removeUploadedFiles = require('./remove-uploaded-files')
  9. function drainStream (stream) {
  10. stream.on('readable', () => {
  11. while (stream.read() !== null) {}
  12. })
  13. }
  14. function makeMiddleware (setup) {
  15. return function multerMiddleware (req, res, next) {
  16. if (!is(req, ['multipart'])) return next()
  17. var options = setup()
  18. var limits = options.limits
  19. var storage = options.storage
  20. var fileFilter = options.fileFilter
  21. var fileStrategy = options.fileStrategy
  22. var preservePath = options.preservePath
  23. req.body = Object.create(null)
  24. req.on('error', function (err) {
  25. abortWithError(err)
  26. })
  27. var busboy
  28. try {
  29. busboy = Busboy({ headers: req.headers, limits: limits, preservePath: preservePath })
  30. } catch (err) {
  31. return next(err)
  32. }
  33. var appender = new FileAppender(fileStrategy, req)
  34. var isDone = false
  35. var readFinished = false
  36. var errorOccured = false
  37. var pendingWrites = new Counter()
  38. var uploadedFiles = []
  39. function done (err) {
  40. if (isDone) return
  41. isDone = true
  42. req.unpipe(busboy)
  43. drainStream(req)
  44. req.resume()
  45. setImmediate(() => {
  46. busboy.removeAllListeners()
  47. })
  48. next(err)
  49. }
  50. function indicateDone () {
  51. if (readFinished && pendingWrites.isZero() && !errorOccured) done()
  52. }
  53. function abortWithError (uploadError) {
  54. if (errorOccured) return
  55. errorOccured = true
  56. pendingWrites.onceZero(function () {
  57. function remove (file, cb) {
  58. storage._removeFile(req, file, cb)
  59. }
  60. removeUploadedFiles(uploadedFiles, remove, function (err, storageErrors) {
  61. if (err) return done(err)
  62. uploadError.storageErrors = storageErrors
  63. done(uploadError)
  64. })
  65. })
  66. }
  67. function abortWithCode (code, optionalField) {
  68. abortWithError(new MulterError(code, optionalField))
  69. }
  70. // handle text field data
  71. busboy.on('field', function (fieldname, value, { nameTruncated, valueTruncated }) {
  72. if (fieldname == null) return abortWithCode('MISSING_FIELD_NAME')
  73. if (nameTruncated) return abortWithCode('LIMIT_FIELD_KEY')
  74. if (valueTruncated) return abortWithCode('LIMIT_FIELD_VALUE', fieldname)
  75. // Work around bug in Busboy (https://github.com/mscdex/busboy/issues/6)
  76. if (limits && Object.prototype.hasOwnProperty.call(limits, 'fieldNameSize')) {
  77. if (fieldname.length > limits.fieldNameSize) return abortWithCode('LIMIT_FIELD_KEY')
  78. }
  79. appendField(req.body, fieldname, value)
  80. })
  81. // handle files
  82. busboy.on('file', function (fieldname, fileStream, { filename, encoding, mimeType }) {
  83. // don't attach to the files object, if there is no file
  84. if (!filename) return fileStream.resume()
  85. // Work around bug in Busboy (https://github.com/mscdex/busboy/issues/6)
  86. if (limits && Object.prototype.hasOwnProperty.call(limits, 'fieldNameSize')) {
  87. if (fieldname.length > limits.fieldNameSize) return abortWithCode('LIMIT_FIELD_KEY')
  88. }
  89. var file = {
  90. fieldname: fieldname,
  91. originalname: filename,
  92. encoding: encoding,
  93. mimetype: mimeType
  94. }
  95. var placeholder = appender.insertPlaceholder(file)
  96. fileFilter(req, file, function (err, includeFile) {
  97. if (err) {
  98. appender.removePlaceholder(placeholder)
  99. return abortWithError(err)
  100. }
  101. if (!includeFile) {
  102. appender.removePlaceholder(placeholder)
  103. return fileStream.resume()
  104. }
  105. var aborting = false
  106. pendingWrites.increment()
  107. Object.defineProperty(file, 'stream', {
  108. configurable: true,
  109. enumerable: false,
  110. value: fileStream
  111. })
  112. fileStream.on('error', function (err) {
  113. pendingWrites.decrement()
  114. abortWithError(err)
  115. })
  116. fileStream.on('limit', function () {
  117. aborting = true
  118. abortWithCode('LIMIT_FILE_SIZE', fieldname)
  119. })
  120. storage._handleFile(req, file, function (err, info) {
  121. if (aborting) {
  122. appender.removePlaceholder(placeholder)
  123. uploadedFiles.push(extend(file, info))
  124. return pendingWrites.decrement()
  125. }
  126. if (err) {
  127. appender.removePlaceholder(placeholder)
  128. pendingWrites.decrement()
  129. return abortWithError(err)
  130. }
  131. var fileInfo = extend(file, info)
  132. appender.replacePlaceholder(placeholder, fileInfo)
  133. uploadedFiles.push(fileInfo)
  134. pendingWrites.decrement()
  135. indicateDone()
  136. })
  137. })
  138. })
  139. busboy.on('error', function (err) { abortWithError(err) })
  140. busboy.on('partsLimit', function () { abortWithCode('LIMIT_PART_COUNT') })
  141. busboy.on('filesLimit', function () { abortWithCode('LIMIT_FILE_COUNT') })
  142. busboy.on('fieldsLimit', function () { abortWithCode('LIMIT_FIELD_COUNT') })
  143. busboy.on('close', function () {
  144. readFinished = true
  145. indicateDone()
  146. })
  147. req.pipe(busboy)
  148. }
  149. }
  150. module.exports = makeMiddleware