我智商爆棚
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

1383 lines
46 KiB

var SFTPStream = require('../lib/sftp');
var Stats = SFTPStream.Stats;
var STATUS_CODE = SFTPStream.STATUS_CODE;
var OPEN_MODE = SFTPStream.OPEN_MODE;
var constants = require('constants');
var basename = require('path').basename;
var assert = require('assert');
var group = basename(__filename, '.js') + '/';
var t = -1;
var tests = [
// successful client requests
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/tmp/foo.txt';
var handle_ = Buffer.from('node.js');
server.on('OPEN', function(id, path, pflags, attrs) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
assert(pflags === (OPEN_MODE.TRUNC
| OPEN_MODE.CREAT
| OPEN_MODE.WRITE),
makeMsg(what, 'Wrong flags: ' + flagsToHuman(pflags)));
server.handle(id, handle_);
server.end();
});
client.open(path_, 'w', function(err, handle) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected open() error: ' + err));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
});
};
},
what: 'open'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var handle_ = Buffer.from('node.js');
server.on('CLOSE', function(id, handle) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
server.status(id, STATUS_CODE.OK);
server.end();
});
client.close(handle_, function(err) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected close() error: ' + err));
});
};
},
what: 'close'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var handle_ = Buffer.from('node.js');
var expected =
Buffer.from('node.jsnode.jsnode.jsnode.jsnode.jsnode.js');
var buffer = Buffer.alloc(expected.length);
server.on('READ', function(id, handle, offset, len) {
assert(++self.state.requests <= 2,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
assert(offset === 5, makeMsg(what, 'Wrong read offset: ' + offset));
assert(len === buffer.length,
makeMsg(what, 'Wrong read len: ' + len));
server.data(id, expected);
server.end();
});
client.readData(handle_, buffer, 0, buffer.length, 5, clientReadCb);
function clientReadCb(err, nb) {
assert(++self.state.responses <= 2,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected readData() error: ' + err));
assert.deepEqual(buffer,
expected,
makeMsg(what, 'read data mismatch'));
}
};
},
what: 'readData'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var handle_ = Buffer.from('node.js');
var buf = Buffer.from('node.jsnode.jsnode.jsnode.jsnode.jsnode.js');
server.on('WRITE', function(id, handle, offset, data) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
assert(offset === 5, makeMsg(what, 'Wrong write offset: ' + offset));
assert.deepEqual(data, buf, makeMsg(what, 'write data mismatch'));
server.status(id, STATUS_CODE.OK);
server.end();
});
client.writeData(handle_, buf, 0, buf.length, 5, function(err, nb) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected writeData() error: ' + err));
assert.equal(nb, buf.length);
});
};
},
what: 'write'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var handle_ = Buffer.from('node.js');
var buf = Buffer.allocUnsafe(3 * 32 * 1024);
server.on('WRITE', function(id, handle, offset, data) {
++self.state.requests;
assert.equal(id,
self.state.requests - 1,
makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
assert.equal(offset,
(self.state.requests - 1) * 32 * 1024,
makeMsg(what, 'Wrong write offset: ' + offset));
assert((offset + data.length) <= buf.length);
assert.deepEqual(data,
buf.slice(offset, offset + data.length),
makeMsg(what, 'write data mismatch'));
server.status(id, STATUS_CODE.OK);
if (self.state.requests === 3)
server.end();
});
client.writeData(handle_, buf, 0, buf.length, 0, function(err, nb) {
++self.state.responses;
assert(!err, makeMsg(what, 'Unexpected writeData() error: ' + err));
assert.equal(nb, buf.length);
});
};
},
expected: {
requests: 3,
responses: 1
},
what: 'write (overflow)'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/foo/bar/baz';
var attrs_ = new Stats({
size: 10 * 1024,
uid: 9001,
gid: 9001,
atime: (Date.now() / 1000) | 0,
mtime: (Date.now() / 1000) | 0
});
server.on('LSTAT', function(id, path) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
server.attrs(id, attrs_);
server.end();
});
client.lstat(path_, function(err, attrs) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected lstat() error: ' + err));
assert.deepEqual(attrs, attrs_, makeMsg(what, 'attrs mismatch'));
});
};
},
what: 'lstat'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var handle_ = Buffer.from('node.js');
var attrs_ = new Stats({
size: 10 * 1024,
uid: 9001,
gid: 9001,
atime: (Date.now() / 1000) | 0,
mtime: (Date.now() / 1000) | 0
});
server.on('FSTAT', function(id, handle) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
server.attrs(id, attrs_);
server.end();
});
client.fstat(handle_, function(err, attrs) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected fstat() error: ' + err));
assert.deepEqual(attrs, attrs_, makeMsg(what, 'attrs mismatch'));
});
};
},
what: 'fstat'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/foo/bar/baz';
var attrs_ = new Stats({
uid: 9001,
gid: 9001,
atime: (Date.now() / 1000) | 0,
mtime: (Date.now() / 1000) | 0
});
server.on('SETSTAT', function(id, path, attrs) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
assert.deepEqual(attrs, attrs_, makeMsg(what, 'attrs mismatch'));
server.status(id, STATUS_CODE.OK);
server.end();
});
client.setstat(path_, attrs_, function(err) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected setstat() error: ' + err));
});
};
},
what: 'setstat'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var handle_ = Buffer.from('node.js');
var attrs_ = new Stats({
uid: 9001,
gid: 9001,
atime: (Date.now() / 1000) | 0,
mtime: (Date.now() / 1000) | 0
});
server.on('FSETSTAT', function(id, handle, attrs) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
assert.deepEqual(attrs, attrs_, makeMsg(what, 'attrs mismatch'));
server.status(id, STATUS_CODE.OK);
server.end();
});
client.fsetstat(handle_, attrs_, function(err) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected fsetstat() error: ' + err));
});
};
},
what: 'fsetstat'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var handle_ = Buffer.from('node.js');
var path_ = '/tmp';
server.on('OPENDIR', function(id, path) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
server.handle(id, handle_);
server.end();
});
client.opendir(path_, function(err, handle) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected opendir() error: ' + err));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
});
};
},
what: 'opendir'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var handle_ = Buffer.from('node.js');
var list_ = [
{ filename: '.',
longname: 'drwxr-xr-x 56 nodejs nodejs 4096 Nov 10 01:05 .',
attrs: new Stats({
mode: 0755 | constants.S_IFDIR,
size: 4096,
uid: 9001,
gid: 8001,
atime: 1415599549,
mtime: 1415599590
})
},
{ filename: '..',
longname: 'drwxr-xr-x 4 root root 4096 May 16 2013 ..',
attrs: new Stats({
mode: 0755 | constants.S_IFDIR,
size: 4096,
uid: 0,
gid: 0,
atime: 1368729954,
mtime: 1368729999
})
},
{ filename: 'foo',
longname: 'drwxrwxrwx 2 nodejs nodejs 4096 Mar 8 2009 foo',
attrs: new Stats({
mode: 0777 | constants.S_IFDIR,
size: 4096,
uid: 9001,
gid: 8001,
atime: 1368729954,
mtime: 1368729999
})
},
{ filename: 'bar',
longname: '-rw-r--r-- 1 nodejs nodejs 513901992 Dec 4 2009 bar',
attrs: new Stats({
mode: 0644 | constants.S_IFREG,
size: 513901992,
uid: 9001,
gid: 8001,
atime: 1259972199,
mtime: 1259972199
})
}
];
server.on('READDIR', function(id, handle) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
server.name(id, list_);
server.end();
});
client.readdir(handle_, function(err, list) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected readdir() error: ' + err));
assert.deepEqual(list,
list_.slice(2),
makeMsg(what, 'dir list mismatch'));
});
};
},
what: 'readdir'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var handle_ = Buffer.from('node.js');
var list_ = [
{ filename: '.',
longname: 'drwxr-xr-x 56 nodejs nodejs 4096 Nov 10 01:05 .',
attrs: new Stats({
mode: 0755 | constants.S_IFDIR,
size: 4096,
uid: 9001,
gid: 8001,
atime: 1415599549,
mtime: 1415599590
})
},
{ filename: '..',
longname: 'drwxr-xr-x 4 root root 4096 May 16 2013 ..',
attrs: new Stats({
mode: 0755 | constants.S_IFDIR,
size: 4096,
uid: 0,
gid: 0,
atime: 1368729954,
mtime: 1368729999
})
},
{ filename: 'foo',
longname: 'drwxrwxrwx 2 nodejs nodejs 4096 Mar 8 2009 foo',
attrs: new Stats({
mode: 0777 | constants.S_IFDIR,
size: 4096,
uid: 9001,
gid: 8001,
atime: 1368729954,
mtime: 1368729999
})
},
{ filename: 'bar',
longname: '-rw-r--r-- 1 nodejs nodejs 513901992 Dec 4 2009 bar',
attrs: new Stats({
mode: 0644 | constants.S_IFREG,
size: 513901992,
uid: 9001,
gid: 8001,
atime: 1259972199,
mtime: 1259972199
})
}
];
server.on('READDIR', function(id, handle) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
server.name(id, list_);
server.end();
});
client.readdir(handle_, { full: true }, function(err, list) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected readdir() error: ' + err));
assert.deepEqual(list, list_, makeMsg(what, 'dir list mismatch'));
});
};
},
what: 'readdir (full)'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/foo/bar/baz';
server.on('REMOVE', function(id, path) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
server.status(id, STATUS_CODE.OK);
server.end();
});
client.unlink(path_, function(err) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected unlink() error: ' + err));
});
};
},
what: 'remove'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/foo/bar/baz';
server.on('MKDIR', function(id, path) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
server.status(id, STATUS_CODE.OK);
server.end();
});
client.mkdir(path_, function(err) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected mkdir() error: ' + err));
});
};
},
what: 'mkdir'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/foo/bar/baz';
server.on('RMDIR', function(id, path) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
server.status(id, STATUS_CODE.OK);
server.end();
});
client.rmdir(path_, function(err) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected rmdir() error: ' + err));
});
};
},
what: 'rmdir'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/foo/bar/baz';
var name_ = { filename: '/tmp/foo' };
server.on('REALPATH', function(id, path) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
server.name(id, name_);
server.end();
});
client.realpath(path_, function(err, name) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected realpath() error: ' + err));
assert.deepEqual(name,
name_.filename,
makeMsg(what, 'name mismatch'));
});
};
},
what: 'realpath'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/foo/bar/baz';
var attrs_ = new Stats({
size: 10 * 1024,
uid: 9001,
gid: 9001,
atime: (Date.now() / 1000) | 0,
mtime: (Date.now() / 1000) | 0
});
server.on('STAT', function(id, path) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
server.attrs(id, attrs_);
server.end();
});
client.stat(path_, function(err, attrs) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected stat() error: ' + err));
assert.deepEqual(attrs, attrs_, makeMsg(what, 'attrs mismatch'));
});
};
},
what: 'stat'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var oldPath_ = '/foo/bar/baz';
var newPath_ = '/tmp/foo';
server.on('RENAME', function(id, oldPath, newPath) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(oldPath === oldPath_,
makeMsg(what, 'Wrong old path: ' + oldPath));
assert(newPath === newPath_,
makeMsg(what, 'Wrong new path: ' + newPath));
server.status(id, STATUS_CODE.OK);
server.end();
});
client.rename(oldPath_, newPath_, function(err) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected rename() error: ' + err));
});
};
},
what: 'rename'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var linkPath_ = '/foo/bar/baz';
var name = { filename: '/tmp/foo' };
server.on('READLINK', function(id, linkPath) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(linkPath === linkPath_,
makeMsg(what, 'Wrong link path: ' + linkPath));
server.name(id, name);
server.end();
});
client.readlink(linkPath_, function(err, targetPath) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected readlink() error: ' + err));
assert(targetPath === name.filename,
makeMsg(what, 'Wrong target path: ' + targetPath));
});
};
},
what: 'readlink'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var linkPath_ = '/foo/bar/baz';
var targetPath_ = '/tmp/foo';
server.on('SYMLINK', function(id, linkPath, targetPath) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(linkPath === linkPath_,
makeMsg(what, 'Wrong link path: ' + linkPath));
assert(targetPath === targetPath_,
makeMsg(what, 'Wrong target path: ' + targetPath));
server.status(id, STATUS_CODE.OK);
server.end();
});
client.symlink(targetPath_, linkPath_, function(err) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(!err, makeMsg(what, 'Unexpected symlink() error: ' + err));
});
};
},
what: 'symlink'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/foo/bar/baz';
var handle_ = Buffer.from('hi mom!');
var data_ = Buffer.from('hello world');
server.once('OPEN', function(id, path, pflags, attrs) {
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
assert(pflags === OPEN_MODE.READ,
makeMsg(what, 'Wrong flags: ' + flagsToHuman(pflags)));
server.handle(id, handle_);
}).once('FSTAT', function(id, handle) {
assert(id === 1, makeMsg(what, 'Wrong request id: ' + id));
var attrs = new Stats({
size: data_.length,
uid: 9001,
gid: 9001,
atime: (Date.now() / 1000) | 0,
mtime: (Date.now() / 1000) | 0
});
server.attrs(id, attrs);
}).once('READ', function(id, handle, offset, len) {
assert(id === 2, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
assert(offset === 0, makeMsg(what, 'Wrong read offset: ' + offset));
server.data(id, data_);
}).once('CLOSE', function(id, handle) {
++self.state.requests;
assert(id === 3, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
server.status(id, STATUS_CODE.OK);
server.end();
});
client.readFile(path_, function(err, buf) {
++self.state.responses;
assert(!err, makeMsg(what, 'Unexpected error: ' + err));
assert.deepEqual(buf, data_, makeMsg(what, 'data mismatch'));
});
};
},
what: 'readFile'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/foo/bar/baz';
var handle_ = Buffer.from('hi mom!');
var data_ = Buffer.from('hello world');
var reads = 0;
server.once('OPEN', function(id, path, pflags, attrs) {
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
assert(pflags === OPEN_MODE.READ,
makeMsg(what, 'Wrong flags: ' + flagsToHuman(pflags)));
server.handle(id, handle_);
}).once('FSTAT', function(id, handle) {
assert(id === 1, makeMsg(what, 'Wrong request id: ' + id));
var attrs = new Stats({
uid: 9001,
gid: 9001,
atime: (Date.now() / 1000) | 0,
mtime: (Date.now() / 1000) | 0
});
server.attrs(id, attrs);
}).on('READ', function(id, handle, offset, len) {
assert(++reads <= 2, makeMsg(what, 'Saw too many READs'));
assert(id === 2 || id === 3,
makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
switch(id) {
case 2:
assert(offset === 0,
makeMsg(what,
'Wrong read offset for first read: ' + offset));
server.data(id, data_);
break;
case 3:
assert(offset === data_.length,
makeMsg(what,
'Wrong read offset for second read: ' + offset));
server.status(id, STATUS_CODE.EOF);
break;
}
}).once('CLOSE', function(id, handle) {
++self.state.requests;
assert(id === 4, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
server.status(id, STATUS_CODE.OK);
server.end();
});
client.readFile(path_, function(err, buf) {
++self.state.responses;
assert(!err, makeMsg(what, 'Unexpected error: ' + err));
assert.deepEqual(buf, data_, makeMsg(what, 'data mismatch'));
});
};
},
what: 'readFile (no size from fstat)'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var opens = 0;
var reads = 0;
var closes = 0;
var path_ = '/foo/bar/baz';
var handle_ = Buffer.from('hi mom!');
var data_ = Buffer.from('hello world');
server.on('OPEN', function(id, path, pflags, attrs) {
assert(++opens === 1, makeMsg(what, 'Saw too many OPENs'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
assert(pflags === OPEN_MODE.READ,
makeMsg(what, 'Wrong flags: ' + flagsToHuman(pflags)));
server.handle(id, handle_);
}).on('READ', function(id, handle, offset, len) {
assert(++reads <= 2, makeMsg(what, 'Saw too many READs'));
assert(id === reads, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
if (reads === 1) {
assert(offset === 0, makeMsg(what, 'Wrong read offset: ' + offset));
server.data(id, data_);
} else
server.status(id, STATUS_CODE.EOF);
}).on('CLOSE', function(id, handle) {
++self.state.requests;
assert(++closes === 1, makeMsg(what, 'Saw too many CLOSEs'));
assert(id === 3, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
server.status(id, STATUS_CODE.OK);
server.end();
});
var buf = [];
client.createReadStream(path_).on('readable', function() {
var chunk;
while ((chunk = this.read()) !== null) {
buf.push(chunk);
}
}).on('end', function() {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
buf = Buffer.concat(buf);
assert.deepEqual(buf, data_, makeMsg(what, 'data mismatch'));
});
};
},
what: 'ReadStream'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/foo/bar/baz';
var handle_ = Buffer.from('hi mom!');
var data_ = Buffer.from('hello world');
server.on('OPEN', function(id, path, pflags, attrs) {
server.handle(id, handle_);
}).on('READ', function(id, handle, offset, len) {
if (offset > data_.length) {
server.status(id, STATUS_CODE.EOF);
} else {
// Only read 4 bytes at a time
server.data(id, data_.slice(offset, offset + 4));
}
}).on('CLOSE', function(id, handle) {
++self.state.requests;
server.status(id, STATUS_CODE.OK);
server.end();
});
var buf = [];
client.createReadStream(path_).on('readable', function() {
var chunk;
while ((chunk = this.read()) !== null) {
buf.push(chunk);
}
}).on('end', function() {
self.state.responses += 1;
buf = Buffer.concat(buf);
assert.deepEqual(buf, data_, makeMsg(what, 'data mismatch'));
});
};
},
what: 'ReadStream (fewer bytes than requested)'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var opens = 0;
var path_ = '/foo/bar/baz';
var error;
server.on('OPEN', function(id, path, pflags, attrs) {
++opens;
++self.state.requests;
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
assert(pflags === OPEN_MODE.READ,
makeMsg(what, 'Wrong flags: ' + flagsToHuman(pflags)));
server.status(id, STATUS_CODE.NO_SUCH_FILE);
server.end();
});
client.createReadStream(path_).on('error', function(err) {
error = err;
}).on('close', function() {
assert(opens === 1, makeMsg(what, 'Saw ' + opens + ' OPENs'));
assert(error, makeMsg(what, 'Expected error'));
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
});
};
},
what: 'ReadStream (error)'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var opens = 0;
var writes = 0;
var closes = 0;
var fsetstat = false;
var path_ = '/foo/bar/baz';
var handle_ = Buffer.from('hi mom!');
var data_ = Buffer.from('hello world');
var expFlags = OPEN_MODE.TRUNC | OPEN_MODE.CREAT | OPEN_MODE.WRITE;
server.on('OPEN', function(id, path, pflags, attrs) {
assert(++opens === 1, makeMsg(what, 'Saw too many OPENs'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert(path === path_, makeMsg(what, 'Wrong path: ' + path));
assert(pflags === expFlags,
makeMsg(what, 'Wrong flags: ' + flagsToHuman(pflags)));
server.handle(id, handle_);
}).once('FSETSTAT', function(id, handle, attrs) {
fsetstat = true;
assert(id === 1, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
assert.strictEqual(attrs.mode,
parseInt('0666', 8),
makeMsg(what, 'Wrong file mode'));
server.status(id, STATUS_CODE.OK);
}).on('WRITE', function(id, handle, offset, data) {
assert(++writes <= 3, makeMsg(what, 'Saw too many WRITEs'));
assert(id === writes + 1, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
assert(offset === ((writes - 1) * data_.length),
makeMsg(what, 'Wrong write offset: ' + offset));
assert.deepEqual(data, data_, makeMsg(what, 'Wrong data'));
server.status(id, STATUS_CODE.OK);
}).on('CLOSE', function(id, handle) {
++self.state.requests;
assert(++closes === 1, makeMsg(what, 'Saw too many CLOSEs'));
assert(id === 5, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
server.status(id, STATUS_CODE.OK);
server.end();
}).on('end', function() {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(opens === 1, makeMsg(what, 'Wrong OPEN count'));
assert(writes === 3, makeMsg(what, 'Wrong WRITE count'));
assert(closes === 1, makeMsg(what, 'Wrong CLOSE count'));
assert(fsetstat, makeMsg(what, 'Expected FSETSTAT'));
});
var writer = client.createWriteStream(path_);
if (writer.cork)
writer.cork();
writer.write(data_);
writer.write(data_);
writer.write(data_);
if (writer.uncork)
writer.uncork();
writer.end();
};
},
what: 'WriteStream'
},
// other client request scenarios
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var handle_ = Buffer.from('node.js');
server.on('READDIR', function(id, handle) {
assert(++self.state.requests === 1,
makeMsg(what, 'Saw too many requests'));
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
server.status(id, STATUS_CODE.EOF);
server.end();
});
client.readdir(handle_, function(err, list) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(err && err.code === STATUS_CODE.EOF,
makeMsg(what, 'Expected EOF, got: ' + err));
});
};
},
what: 'readdir (EOF)'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var path_ = '/tmp/foo.txt';
var reqs = 0;
var continues = 0;
client.unpipe(server);
function clientCb(err, handle) {
assert(++self.state.responses <= reqs,
makeMsg(what, 'Saw too many responses'));
if (self.state.responses === reqs) {
assert(continues === 1, makeMsg(what, 'no continue event seen'));
server.end();
}
}
client.on('continue', function() {
assert(++continues === 1, makeMsg(what, 'saw > 1 continue event'));
});
while (true) {
++reqs;
if (!client.open(path_, 'w', clientCb))
break;
}
client.pipe(server);
};
},
expected: {
requests: -1,
responses: -1
},
what: '"continue" event after push() === false'
},
{ run: function() {
var self = this;
var client = new SFTPStream();
client.once('ready', function() {
client.open('/foo/bar', 'w', function(err, handle) {
assert(err, 'Expected error');
assert.strictEqual(err.code, 4);
assert.strictEqual(err.message, 'Uh oh');
assert.strictEqual(err.lang, '');
next();
});
client.write(Buffer.from([
0, 0, 0, 18,
101,
0, 0, 0, 0,
0, 0, 0, SFTPStream.STATUS_CODE.FAILURE,
0, 0, 0, 5, 85, 104, 32, 111, 104
]));
});
client.write(Buffer.from([
0, 0, 0, 5,
2,
0, 0, 0, 3
]));
},
what: 'Can parse status response without language'
},
{ run: function() {
var self = this;
var client = new SFTPStream();
client.once('ready', function() {
client.open('/foo/bar', 'w', function(err, handle) {
assert(err, 'Expected error');
assert.strictEqual(err.code, 4);
assert.strictEqual(err.message, 'Failure');
assert.strictEqual(err.lang, '');
next();
});
client.write(Buffer.from([
0, 0, 0, 9,
101,
0, 0, 0, 0,
0, 0, 0, SFTPStream.STATUS_CODE.FAILURE
]));
});
client.write(Buffer.from([
0, 0, 0, 5,
2,
0, 0, 0, 3
]));
},
what: 'Can parse status response without message'
},
{ run: function() {
var self = this;
var err;
var client = new SFTPStream();
client.once('ready', function() {
assert(false, 'Handshake should not succeed');
}).once('error', function(err_) {
err = err_;
}).once('end', function() {
assert.strictEqual(err && err.message,
'Unexpected packet before version');
next();
});
client.write(Buffer.from([
1, 2, 3, 4,
5,
6, 7, 8, 9
]));
},
what: 'End SFTP stream on bad handshake (client)'
},
{ run: function() {
var self = this;
var err;
var client = new SFTPStream({ server: true });
client.once('ready', function() {
assert(false, 'Handshake should not succeed');
}).once('error', function(err_) {
err = err_;
}).once('end', function() {
assert.strictEqual(err && err.message,
'Unexpected packet before init');
next();
});
client.write(Buffer.from([
1, 2, 3, 4,
5,
6, 7, 8, 9
]));
},
what: 'End SFTP stream on bad handshake (server)'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
var server = this.server;
this.onReady = function() {
var handle_ = Buffer.from([0,0,0,1]);
var buffer = Buffer.allocUnsafe(4);
server.once('READ', function(id, handle, offset, len) {
++self.state.requests;
assert(id === 0, makeMsg(what, 'Wrong request id: ' + id));
assert.deepEqual(handle, handle_, makeMsg(what, 'handle mismatch'));
assert(offset === 0, makeMsg(what, 'Wrong read offset: ' + offset));
assert(len === buffer.length,
makeMsg(what, 'Wrong read length: ' + len));
server.data(id, Buffer.alloc(len + 1));
});
client.readData(handle_, buffer, 0, buffer.length, 0, clientReadCb);
function clientReadCb(err, buf) {
++self.state.responses;
assert(err && err.message.indexOf('more data than requested') !== -1,
makeMsg(what, 'Expected error'));
server.end();
}
};
},
what: 'Abort on data chunk larger than requested'
},
{ run: function() {
setup(this);
var self = this;
var what = this.what;
var client = this.client;
client._state.extensions['statvfs@openssh.com'] = ['2'];
this.onReady = function() {
var pathOne_ = '/foo/baz';
client.ext_openssh_statvfs(pathOne_, function(err, fsInfo) {
assert(++self.state.responses === 1,
makeMsg(what, 'Saw too many responses'));
assert(err && err.code === STATUS_CODE.OP_UNSUPPORTED,
makeMsg(what, 'Expected OP_UNSUPPORTED, got: ' + err));
var pathTwo_ = '/baz/foo';
client.ext_openssh_statvfs(pathTwo_, function(err, fsInfo) {
assert(++self.state.responses === 2,
makeMsg(what, 'Saw too many responses'));
assert(err && err.code === STATUS_CODE.OP_UNSUPPORTED,
makeMsg(what, 'Expected OP_UNSUPPORTED, got: ' + err));
next();
});
});
};
},
what: 'Multiple extended operations in sequence fail as expected'
},
];
function setup(self) {
var expectedRequests = (self.expected && self.expected.requests) || 1;
var expectedResponses = (self.expected && self.expected.responses) || 1;
var clientEnded = false;
var serverEnded = false;
self.state = {
clientReady: false,
serverReady: false,
requests: 0,
responses: 0
};
self.client = new SFTPStream();
self.server = new SFTPStream({ server: true });
self.server.on('error', onError)
.on('ready', onReady)
.on('end', onEnd);
self.client.on('error', onError)
.on('ready', onReady)
.on('end', onEnd);
function onError(err) {
var which = (this === self.server ? 'server' : 'client');
assert(false, makeMsg(self.what, 'Unexpected ' + which + ' error: ' + err));
}
function onReady() {
if (this === self.client) {
assert(!self.state.clientReady,
makeMsg(self.what, 'Received multiple ready events for client'));
self.state.clientReady = true;
} else {
assert(!self.state.serverReady,
makeMsg(self.what, 'Received multiple ready events for server'));
self.state.serverReady = true;
}
if (self.state.clientReady && self.state.serverReady)
self.onReady && self.onReady();
}
function onEnd() {
if (this === self.client) {
assert(!clientEnded,
makeMsg(self.what, 'Received multiple close events for client'));
clientEnded = true;
} else {
assert(!serverEnded,
makeMsg(self.what, 'Received multiple close events for server'));
serverEnded = true;
}
if (clientEnded && serverEnded) {
var msg;
if (expectedRequests > 0) {
msg = 'Expected ' + expectedRequests + ' request(s) but received '
+ self.state.requests;
assert(self.state.requests === expectedRequests,
makeMsg(self.what, msg));
}
if (expectedResponses > 0) {
msg = 'Expected ' + expectedResponses + ' response(s) but received '
+ self.state.responses;
assert(self.state.responses === expectedResponses,
makeMsg(self.what, msg));
}
next();
}
}
process.nextTick(function() {
self.client.pipe(self.server).pipe(self.client);
});
}
function flagsToHuman(flags) {
var ret = [];
for (var i = 0, keys = Object.keys(OPEN_MODE); i < keys.length; ++i)
if (flags & OPEN_MODE[keys[i]])
ret.push(keys[i]);
return ret.join(' | ');
}
function next() {
if (++t === tests.length)
return;
var v = tests[t];
v.run.call(v);
}
function makeMsg(what, msg) {
return '[' + group + what + ']: ' + msg;
}
process.once('exit', function() {
assert(t === tests.length,
makeMsg('_exit',
'Only finished ' + t + '/' + tests.length + ' tests'));
});
next();