summaryrefslogtreecommitdiff
path: root/deps/node/deps/npm/node_modules/fs-vacuum/test/racy-entries.js
blob: c0ed53243de07aed21b4e62ad2a2a914867e60ca (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
var path = require('path')

var test = require('tap').test

var readdir = require('graceful-fs').readdir
var readdirSync = require('graceful-fs').readdirSync
var statSync = require('graceful-fs').statSync
var writeFile = require('graceful-fs').writeFile
var mkdirp = require('mkdirp')
var mkdtemp = require('tmp').dir
var tmpFile = require('tmp').file

var requireInject = require('require-inject')
var vacuum = requireInject('../vacuum.js', {
  'graceful-fs': {
    'lstat': require('graceful-fs').lstat,
    'readdir': function (dir, cb) {
      readdir(dir, function (err, files) {
        // Simulate racy removal by creating a file after vacuum
        // thinks the directory is empty
        if (dir === partialPath && files.length === 0) {
          tmpFile({dir: dir}, function (err, path, fd) {
            if (err) throw err
            cb(err, files)
          })
        } else {
          cb(err, files)
        }
      })
    },
    'rmdir': require('graceful-fs').rmdir,
    'unlink': require('graceful-fs').unlink
  }
})

// CONSTANTS
var TEMP_OPTIONS = {
  unsafeCleanup: true,
  mode: '0700'
}
var SHORT_PATH = path.join('i', 'am', 'a', 'path')
var PARTIAL_PATH = path.join(SHORT_PATH, 'that', 'ends', 'at', 'a')
var FULL_PATH = path.join(PARTIAL_PATH, 'file')

var messages = []
function log () { messages.push(Array.prototype.slice.call(arguments).join(' ')) }

var testBase, partialPath, fullPath
test('xXx setup xXx', function (t) {
  mkdtemp(TEMP_OPTIONS, function (er, tmpdir) {
    t.ifError(er, 'temp directory exists')

    testBase = path.resolve(tmpdir, SHORT_PATH)
    partialPath = path.resolve(tmpdir, PARTIAL_PATH)
    fullPath = path.resolve(tmpdir, FULL_PATH)

    mkdirp(partialPath, function (er) {
      t.ifError(er, 'made test path')

      writeFile(fullPath, new Buffer('hi'), function (er) {
        t.ifError(er, 'made file')

        t.end()
      })
    })
  })
})

test('racy removal should quit gracefully', function (t) {
  vacuum(fullPath, {purge: false, base: testBase, log: log}, function (er) {
    t.ifError(er, 'cleaned up to base')

    t.equal(messages.length, 3, 'got 2 removal & 1 quit message')
    t.equal(messages[2], 'quitting because new (racy) entries in ' + partialPath)

    var stat
    var verifyPath = fullPath

    function verify () { stat = statSync(verifyPath) }

    // handle the file separately
    t.throws(verify, verifyPath + ' cannot be statted')
    t.notOk(stat && stat.isFile(), verifyPath + ' is totally gone')
    verifyPath = path.dirname(verifyPath)

    for (var i = 0; i < 4; i++) {
      t.doesNotThrow(function () {
        stat = statSync(verifyPath)
      }, verifyPath + ' can be statted')
      t.ok(stat && stat.isDirectory(), verifyPath + ' is still a directory')
      verifyPath = path.dirname(verifyPath)
    }

    t.doesNotThrow(function () {
      stat = statSync(testBase)
    }, testBase + ' can be statted')
    t.ok(stat && stat.isDirectory(), testBase + ' is still a directory')

    var files = readdirSync(testBase)
    t.equal(files.length, 1, 'base directory untouched')

    t.end()
  })
})