2019-08-01 12:24:01 +00:00
|
|
|
const tape = require('tape')
|
|
|
|
const fs = require('fs')
|
|
|
|
const path = require('path')
|
|
|
|
const concat = require('concat-stream')
|
|
|
|
|
|
|
|
const Fuse = require('../')
|
2020-01-29 12:32:31 +00:00
|
|
|
const createMountpoint = require('./fixtures/mnt')
|
2019-08-01 12:24:01 +00:00
|
|
|
const stat = require('./fixtures/stat')
|
2019-11-19 13:04:46 +00:00
|
|
|
const { unmount } = require('./helpers')
|
2019-08-01 12:24:01 +00:00
|
|
|
|
2020-01-29 12:32:31 +00:00
|
|
|
const mnt = createMountpoint()
|
|
|
|
|
2019-08-01 12:24:01 +00:00
|
|
|
tape('read', function (t) {
|
|
|
|
var ops = {
|
|
|
|
force: true,
|
|
|
|
readdir: function (path, cb) {
|
2019-08-06 08:55:08 +00:00
|
|
|
if (path === '/') return process.nextTick(cb, null, ['test'])
|
|
|
|
return process.nextTick(cb, Fuse.ENOENT)
|
2019-08-01 12:24:01 +00:00
|
|
|
},
|
|
|
|
getattr: function (path, cb) {
|
2019-08-06 08:55:08 +00:00
|
|
|
if (path === '/') return process.nextTick(cb, null, stat({ mode: 'dir', size: 4096 }))
|
|
|
|
if (path === '/test') return process.nextTick(cb, null, stat({ mode: 'file', size: 11 }))
|
|
|
|
return process.nextTick(cb, Fuse.ENOENT)
|
2019-08-01 12:24:01 +00:00
|
|
|
},
|
|
|
|
open: function (path, flags, cb) {
|
2019-08-06 08:55:08 +00:00
|
|
|
return process.nextTick(cb, 0, 42)
|
2019-08-01 12:24:01 +00:00
|
|
|
},
|
|
|
|
release: function (path, fd, cb) {
|
|
|
|
t.same(fd, 42, 'fd was passed to release')
|
2019-08-06 08:55:08 +00:00
|
|
|
return process.nextTick(cb, 0)
|
2019-08-01 12:24:01 +00:00
|
|
|
},
|
|
|
|
read: function (path, fd, buf, len, pos, cb) {
|
|
|
|
var str = 'hello world'.slice(pos, pos + len)
|
2019-08-06 08:55:08 +00:00
|
|
|
if (!str) return process.nextTick(cb, 0)
|
2019-08-01 12:24:01 +00:00
|
|
|
buf.write(str)
|
2019-08-06 08:55:08 +00:00
|
|
|
return process.nextTick(cb, str.length)
|
2019-08-01 12:24:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const fuse = new Fuse(mnt, ops, { debug: true })
|
|
|
|
fuse.mount(function (err) {
|
|
|
|
t.error(err, 'no error')
|
|
|
|
|
|
|
|
fs.readFile(path.join(mnt, 'test'), function (err, buf) {
|
|
|
|
t.error(err, 'no error')
|
2019-08-06 08:55:08 +00:00
|
|
|
t.same(buf, Buffer.from('hello world'), 'read file')
|
2019-08-01 12:24:01 +00:00
|
|
|
|
|
|
|
fs.readFile(path.join(mnt, 'test'), function (err, buf) {
|
|
|
|
t.error(err, 'no error')
|
2019-08-06 08:55:08 +00:00
|
|
|
t.same(buf, Buffer.from('hello world'), 'read file again')
|
2019-08-01 12:24:01 +00:00
|
|
|
|
|
|
|
fs.createReadStream(path.join(mnt, 'test'), { start: 0, end: 4 }).pipe(concat(function (buf) {
|
2019-08-06 08:55:08 +00:00
|
|
|
t.same(buf, Buffer.from('hello'), 'partial read file')
|
2019-08-01 12:24:01 +00:00
|
|
|
|
|
|
|
fs.createReadStream(path.join(mnt, 'test'), { start: 6, end: 10 }).pipe(concat(function (buf) {
|
2019-08-06 08:55:08 +00:00
|
|
|
t.same(buf, Buffer.from('world'), 'partial read file + start offset')
|
2019-08-01 12:24:01 +00:00
|
|
|
|
2019-11-19 13:04:46 +00:00
|
|
|
unmount(fuse, function () {
|
2019-08-01 12:24:01 +00:00
|
|
|
t.end()
|
|
|
|
})
|
|
|
|
}))
|
|
|
|
}))
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
2020-01-24 09:31:51 +00:00
|
|
|
|
|
|
|
// Skipped because this test takes 2 minutes to run.
|
|
|
|
tape.skip('read timeout does not force unmount', function (t) {
|
|
|
|
var ops = {
|
|
|
|
force: true,
|
|
|
|
readdir: function (path, cb) {
|
|
|
|
if (path === '/') return process.nextTick(cb, null, ['test'])
|
|
|
|
return process.nextTick(cb, Fuse.ENOENT)
|
|
|
|
},
|
|
|
|
getattr: function (path, cb) {
|
|
|
|
if (path === '/') return process.nextTick(cb, null, stat({ mode: 'dir', size: 4096 }))
|
|
|
|
if (path === '/test') return process.nextTick(cb, null, stat({ mode: 'file', size: 11 }))
|
|
|
|
if (path === '/timeout') return process.nextTick(cb, null, stat({ mode: 'file', size: 11 }))
|
|
|
|
return process.nextTick(cb, Fuse.ENOENT)
|
|
|
|
},
|
|
|
|
open: function (path, flags, cb) {
|
|
|
|
return process.nextTick(cb, 0, 42)
|
|
|
|
},
|
|
|
|
release: function (path, fd, cb) {
|
|
|
|
t.same(fd, 42, 'fd was passed to release')
|
|
|
|
return process.nextTick(cb, 0)
|
|
|
|
},
|
|
|
|
read: function (path, fd, buf, len, pos, cb) {
|
|
|
|
if (path === '/test') {
|
|
|
|
var str = 'hello world'.slice(pos, pos + len)
|
|
|
|
if (!str) return process.nextTick(cb, 0)
|
|
|
|
buf.write(str)
|
|
|
|
return process.nextTick(cb, str.length)
|
|
|
|
} else if (path === '/timeout') {
|
|
|
|
console.log('read is gonna time out')
|
|
|
|
// Just let this one timeout
|
2020-01-24 10:19:17 +00:00
|
|
|
setTimeout(cb, 20 * 1000, -2)
|
2020-01-24 09:31:51 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
return cb(-2)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const fuse = new Fuse(mnt, ops, { debug: false })
|
|
|
|
fuse.mount(function (err) {
|
|
|
|
t.error(err, 'no error')
|
|
|
|
|
|
|
|
fs.readFile(path.join(mnt, 'test'), function (err, buf) {
|
|
|
|
t.error(err, 'no error')
|
|
|
|
t.same(buf, Buffer.from('hello world'), 'read file')
|
|
|
|
|
|
|
|
// Start the read that will time out, wait a bit, then ensure that the second read works.
|
|
|
|
console.time('timeout')
|
|
|
|
fs.readFile(path.join(mnt, 'timeout'), function (err, buf) {
|
|
|
|
console.timeEnd('timeout')
|
|
|
|
console.log('the read timed out')
|
|
|
|
t.true(err)
|
|
|
|
})
|
|
|
|
|
|
|
|
// The default FUSE timeout is 2 minutes, so wait another second after the timeout.
|
|
|
|
setTimeout(function () {
|
|
|
|
console.log('reading from test')
|
|
|
|
fs.readFile(path.join(mnt, 'test'), function (err, buf) {
|
|
|
|
t.error(err, 'no error')
|
|
|
|
t.same(buf, Buffer.from('hello world'), 'read file')
|
|
|
|
unmount(fuse, function () {
|
|
|
|
t.end()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}, 1000 * 121)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
|