index.js 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. 'use strict'
  2. // most of this code was written by Andrew Kelley
  3. // licensed under the BSD license: see
  4. // https://github.com/andrewrk/node-mv/blob/master/package.json
  5. // this needs a cleanup
  6. const u = require('universalify').fromCallback
  7. const fs = require('graceful-fs')
  8. const ncp = require('../copy/ncp')
  9. const path = require('path')
  10. const remove = require('../remove').remove
  11. const mkdirp = require('../mkdirs').mkdirs
  12. function move (source, dest, options, callback) {
  13. if (typeof options === 'function') {
  14. callback = options
  15. options = {}
  16. }
  17. const shouldMkdirp = ('mkdirp' in options) ? options.mkdirp : true
  18. const overwrite = options.overwrite || options.clobber || false
  19. if (shouldMkdirp) {
  20. mkdirs()
  21. } else {
  22. doRename()
  23. }
  24. function mkdirs () {
  25. mkdirp(path.dirname(dest), err => {
  26. if (err) return callback(err)
  27. doRename()
  28. })
  29. }
  30. function doRename () {
  31. if (path.resolve(source) === path.resolve(dest)) {
  32. fs.access(source, callback)
  33. } else if (overwrite) {
  34. fs.rename(source, dest, err => {
  35. if (!err) return callback()
  36. if (err.code === 'ENOTEMPTY' || err.code === 'EEXIST') {
  37. remove(dest, err => {
  38. if (err) return callback(err)
  39. options.overwrite = false // just overwriteed it, no need to do it again
  40. move(source, dest, options, callback)
  41. })
  42. return
  43. }
  44. // weird Windows shit
  45. if (err.code === 'EPERM') {
  46. setTimeout(() => {
  47. remove(dest, err => {
  48. if (err) return callback(err)
  49. options.overwrite = false
  50. move(source, dest, options, callback)
  51. })
  52. }, 200)
  53. return
  54. }
  55. if (err.code !== 'EXDEV') return callback(err)
  56. moveAcrossDevice(source, dest, overwrite, callback)
  57. })
  58. } else {
  59. fs.link(source, dest, err => {
  60. if (err) {
  61. if (err.code === 'EXDEV' || err.code === 'EISDIR' || err.code === 'EPERM' || err.code === 'ENOTSUP') {
  62. moveAcrossDevice(source, dest, overwrite, callback)
  63. return
  64. }
  65. callback(err)
  66. return
  67. }
  68. fs.unlink(source, callback)
  69. })
  70. }
  71. }
  72. }
  73. function moveAcrossDevice (source, dest, overwrite, callback) {
  74. fs.stat(source, (err, stat) => {
  75. if (err) {
  76. callback(err)
  77. return
  78. }
  79. if (stat.isDirectory()) {
  80. moveDirAcrossDevice(source, dest, overwrite, callback)
  81. } else {
  82. moveFileAcrossDevice(source, dest, overwrite, callback)
  83. }
  84. })
  85. }
  86. function moveFileAcrossDevice (source, dest, overwrite, callback) {
  87. const flags = overwrite ? 'w' : 'wx'
  88. const ins = fs.createReadStream(source)
  89. const outs = fs.createWriteStream(dest, { flags })
  90. ins.on('error', err => {
  91. ins.destroy()
  92. outs.destroy()
  93. outs.removeListener('close', onClose)
  94. // may want to create a directory but `out` line above
  95. // creates an empty file for us: See #108
  96. // don't care about error here
  97. fs.unlink(dest, () => {
  98. // note: `err` here is from the input stream errror
  99. if (err.code === 'EISDIR' || err.code === 'EPERM') {
  100. moveDirAcrossDevice(source, dest, overwrite, callback)
  101. } else {
  102. callback(err)
  103. }
  104. })
  105. })
  106. outs.on('error', err => {
  107. ins.destroy()
  108. outs.destroy()
  109. outs.removeListener('close', onClose)
  110. callback(err)
  111. })
  112. outs.once('close', onClose)
  113. ins.pipe(outs)
  114. function onClose () {
  115. fs.unlink(source, callback)
  116. }
  117. }
  118. function moveDirAcrossDevice (source, dest, overwrite, callback) {
  119. const options = {
  120. overwrite: false
  121. }
  122. if (overwrite) {
  123. remove(dest, err => {
  124. if (err) return callback(err)
  125. startNcp()
  126. })
  127. } else {
  128. startNcp()
  129. }
  130. function startNcp () {
  131. ncp(source, dest, options, err => {
  132. if (err) return callback(err)
  133. remove(source, callback)
  134. })
  135. }
  136. }
  137. module.exports = {
  138. move: u(move)
  139. }