Merge pull request #70 from ispedals/truncate
Good code, good tests. Merging.
This commit is contained in:
commit
170af0baa8
|
@ -328,3 +328,11 @@ Asynchronous readlink(2). Callback gets `(error, linkContents)`, where `linkCont
|
||||||
#### fs.lstat(path, callback)
|
#### fs.lstat(path, callback)
|
||||||
|
|
||||||
Asynchronous lstat(2). Callback gets `(error, stats)`, See `fs.stat`.
|
Asynchronous lstat(2). Callback gets `(error, stats)`, See `fs.stat`.
|
||||||
|
|
||||||
|
#### fs.truncate(path, length, callback)
|
||||||
|
|
||||||
|
Asynchronous truncate(2). Callback gets no additional arguments.
|
||||||
|
|
||||||
|
#### fs.ftruncate(fd, length, callback)
|
||||||
|
|
||||||
|
Asynchronous ftruncate(2). Callback gets no additional arguments.
|
||||||
|
|
148
src/fs.js
148
src/fs.js
|
@ -943,6 +943,96 @@ define(function(require) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function truncate_file(context, path, length, callback) {
|
||||||
|
path = normalize(path);
|
||||||
|
|
||||||
|
var fileNode;
|
||||||
|
|
||||||
|
function read_file_data (error, node) {
|
||||||
|
if (error) {
|
||||||
|
callback(error);
|
||||||
|
} else if(node.mode == MODE_DIRECTORY ) {
|
||||||
|
callback(new EIsDirectory('the named file is a directory'));
|
||||||
|
} else{
|
||||||
|
fileNode = node;
|
||||||
|
context.get(fileNode.data, truncate_file_data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function truncate_file_data(error, fileData) {
|
||||||
|
if (error) {
|
||||||
|
callback(error);
|
||||||
|
} else {
|
||||||
|
var data = new Uint8Array(length);
|
||||||
|
if(fileData) {
|
||||||
|
data.set(fileData.subarray(0, length));
|
||||||
|
}
|
||||||
|
context.put(fileNode.data, data, update_file_node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function update_file_node (error) {
|
||||||
|
if(error) {
|
||||||
|
callback(error);
|
||||||
|
} else {
|
||||||
|
fileNode.size = length;
|
||||||
|
fileNode.mtime = Date.now();
|
||||||
|
fileNode.version += 1;
|
||||||
|
context.put(fileNode.id, fileNode, callback);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(length < 0) {
|
||||||
|
callback(new EInvalid('length cannot be negative'));
|
||||||
|
} else {
|
||||||
|
find_node(context, path, read_file_data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function ftruncate_file(context, ofd, length, callback) {
|
||||||
|
var fileNode;
|
||||||
|
|
||||||
|
function read_file_data (error, node) {
|
||||||
|
if (error) {
|
||||||
|
callback(error);
|
||||||
|
} else if(node.mode == MODE_DIRECTORY ) {
|
||||||
|
callback(new EIsDirectory('the named file is a directory'));
|
||||||
|
} else{
|
||||||
|
fileNode = node;
|
||||||
|
context.get(fileNode.data, truncate_file_data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function truncate_file_data(error, fileData) {
|
||||||
|
if (error) {
|
||||||
|
callback(error);
|
||||||
|
} else {
|
||||||
|
var data = new Uint8Array(length);
|
||||||
|
if(fileData) {
|
||||||
|
data.set(fileData.subarray(0, length));
|
||||||
|
}
|
||||||
|
context.put(fileNode.data, data, update_file_node);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function update_file_node (error) {
|
||||||
|
if(error) {
|
||||||
|
callback(error);
|
||||||
|
} else {
|
||||||
|
fileNode.size = length;
|
||||||
|
fileNode.mtime = Date.now();
|
||||||
|
fileNode.version += 1;
|
||||||
|
context.put(fileNode.id, fileNode, callback);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(length < 0) {
|
||||||
|
callback(new EInvalid('length cannot be negative'));
|
||||||
|
} else {
|
||||||
|
context.get(ofd.id, read_file_data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function validate_flags(flags) {
|
function validate_flags(flags) {
|
||||||
if(!_(O_FLAGS).has(flags)) {
|
if(!_(O_FLAGS).has(flags)) {
|
||||||
return null;
|
return null;
|
||||||
|
@ -1505,14 +1595,38 @@ define(function(require) {
|
||||||
lstat_file(context, path, check_result);
|
lstat_file(context, path, check_result);
|
||||||
}
|
}
|
||||||
|
|
||||||
function _truncate(path, length, callback) {
|
function _truncate(context, path, length, callback) {
|
||||||
// TODO
|
if(!nullCheck(path, callback)) return;
|
||||||
// if(!nullCheck(path, callback)) return;
|
|
||||||
|
function check_result(error) {
|
||||||
|
if(error) {
|
||||||
|
callback(error);
|
||||||
|
} else {
|
||||||
|
callback(null);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function _ftruncate(fd, length, callback) {
|
truncate_file(context, path, length, check_result);
|
||||||
// TODO
|
}
|
||||||
// if(!nullCheck(path, callback)) return;
|
|
||||||
|
function _ftruncate(fs, context, fd, length, callback) {
|
||||||
|
function check_result(error) {
|
||||||
|
if(error) {
|
||||||
|
callback(error);
|
||||||
|
} else {
|
||||||
|
callback(null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var ofd = fs.openFiles[fd];
|
||||||
|
|
||||||
|
if(!ofd) {
|
||||||
|
callback(new EBadFileDescriptor('invalid file descriptor'));
|
||||||
|
} else if(!_(ofd.flags).contains(O_WRITE)) {
|
||||||
|
callback(new EBadFileDescriptor('descriptor does not permit writing'));
|
||||||
|
} else {
|
||||||
|
ftruncate_file(context, ofd, length, check_result);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1723,6 +1837,28 @@ define(function(require) {
|
||||||
);
|
);
|
||||||
if(error) callback(error);
|
if(error) callback(error);
|
||||||
};
|
};
|
||||||
|
FileSystem.prototype.truncate = function(path, length, callback) {
|
||||||
|
callback = maybeCallback(callback);
|
||||||
|
var fs = this;
|
||||||
|
var error = fs.queueOrRun(
|
||||||
|
function() {
|
||||||
|
var context = fs.provider.getReadWriteContext();
|
||||||
|
_truncate(context, path, length, callback);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
if(error) callback(error);
|
||||||
|
};
|
||||||
|
FileSystem.prototype.ftruncate = function(fd, length, callback) {
|
||||||
|
callback = maybeCallback(callback);
|
||||||
|
var fs = this;
|
||||||
|
var error = fs.queueOrRun(
|
||||||
|
function() {
|
||||||
|
var context = fs.provider.getReadWriteContext();
|
||||||
|
_ftruncate(fs, context, fd, length, callback);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
if(error) callback(error);
|
||||||
|
};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
FileSystem: FileSystem,
|
FileSystem: FileSystem,
|
||||||
|
|
|
@ -0,0 +1,278 @@
|
||||||
|
define(["IDBFS"], function(IDBFS) {
|
||||||
|
|
||||||
|
describe('fs.truncate', function() {
|
||||||
|
beforeEach(function() {
|
||||||
|
this.db_name = mk_db_name();
|
||||||
|
this.fs = new IDBFS.FileSystem({
|
||||||
|
name: this.db_name,
|
||||||
|
flags: 'FORMAT'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(function() {
|
||||||
|
indexedDB.deleteDatabase(this.db_name);
|
||||||
|
delete this.fs;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should be a function', function() {
|
||||||
|
expect(typeof this.fs.truncate).toEqual('function');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should error when length is negative', function() {
|
||||||
|
var complete = false;
|
||||||
|
var _error, _result;
|
||||||
|
var that = this;
|
||||||
|
|
||||||
|
var contents = "This is a file.";
|
||||||
|
|
||||||
|
that.fs.writeFile('/myfile', contents, function(error) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
that.fs.truncate('/myfile', -1, function(error) {
|
||||||
|
_error = error;
|
||||||
|
complete = true;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
waitsFor(function() {
|
||||||
|
return complete;
|
||||||
|
}, 'test to complete', DEFAULT_TIMEOUT);
|
||||||
|
|
||||||
|
runs(function() {
|
||||||
|
expect(_error).toBeDefined();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should error when path is not a file', function() {
|
||||||
|
var complete = false;
|
||||||
|
var _error, _result;
|
||||||
|
var that = this;
|
||||||
|
|
||||||
|
that.fs.truncate('/', 0, function(error) {
|
||||||
|
_error = error;
|
||||||
|
complete = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
waitsFor(function() {
|
||||||
|
return complete;
|
||||||
|
}, 'test to complete', DEFAULT_TIMEOUT);
|
||||||
|
|
||||||
|
runs(function() {
|
||||||
|
expect(_error).toBeDefined();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should truncate a file', function() {
|
||||||
|
var complete = false;
|
||||||
|
var _error, _result;
|
||||||
|
var that = this;
|
||||||
|
|
||||||
|
var buffer = new Uint8Array([1, 2, 3, 4, 5, 6, 7, 8]);
|
||||||
|
var truncated = new Uint8Array([1]);
|
||||||
|
|
||||||
|
that.fs.open('/myfile', 'w', function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
var fd = result;
|
||||||
|
that.fs.write(fd, buffer, 0, buffer.length, 0, function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
that.fs.close(fd, function(error) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
that.fs.truncate('/myfile', 1, function(error) {
|
||||||
|
_error = error;
|
||||||
|
|
||||||
|
that.fs.readFile('/myfile', function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
_result = result;
|
||||||
|
complete = true;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
waitsFor(function() {
|
||||||
|
return complete;
|
||||||
|
}, 'test to complete', DEFAULT_TIMEOUT);
|
||||||
|
|
||||||
|
runs(function() {
|
||||||
|
expect(_error).toEqual(null);
|
||||||
|
expect(_result).toEqual(truncated);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should pad a file with zeros when the length is greater than the file size', function() {
|
||||||
|
var complete = false;
|
||||||
|
var _error, _result;
|
||||||
|
var that = this;
|
||||||
|
|
||||||
|
var buffer = new Uint8Array([1, 2, 3, 4, 5, 6, 7, 8]);
|
||||||
|
var truncated = new Uint8Array([1, 2, 3, 4, 5, 6, 7, 8, 0]);
|
||||||
|
|
||||||
|
that.fs.open('/myfile', 'w', function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
var fd = result;
|
||||||
|
that.fs.write(fd, buffer, 0, buffer.length, 0, function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
that.fs.close(fd, function(error) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
that.fs.truncate('/myfile', 9, function(error) {
|
||||||
|
_error = error;
|
||||||
|
|
||||||
|
that.fs.readFile('/myfile', function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
_result = result;
|
||||||
|
complete = true;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
waitsFor(function() {
|
||||||
|
return complete;
|
||||||
|
}, 'test to complete', DEFAULT_TIMEOUT);
|
||||||
|
|
||||||
|
runs(function() {
|
||||||
|
expect(_error).toEqual(null);
|
||||||
|
expect(_result).toEqual(truncated);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should update the file size', function() {
|
||||||
|
var complete = false;
|
||||||
|
var _error, _result;
|
||||||
|
var that = this;
|
||||||
|
|
||||||
|
var buffer = new Uint8Array([1, 2, 3, 4, 5, 6, 7, 8]);
|
||||||
|
|
||||||
|
that.fs.open('/myfile', 'w', function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
var fd = result;
|
||||||
|
that.fs.write(fd, buffer, 0, buffer.length, 0, function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
that.fs.close(fd, function(error) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
that.fs.truncate('/myfile', 0, function(error) {
|
||||||
|
_error = error;
|
||||||
|
|
||||||
|
that.fs.stat('/myfile', function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
_result = result;
|
||||||
|
complete = true;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
waitsFor(function() {
|
||||||
|
return complete;
|
||||||
|
}, 'test to complete', DEFAULT_TIMEOUT);
|
||||||
|
|
||||||
|
runs(function() {
|
||||||
|
expect(_error).toEqual(null);
|
||||||
|
expect(_result.size).toEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should truncate a valid descriptor', function() {
|
||||||
|
var complete = false;
|
||||||
|
var _error, _result;
|
||||||
|
var that = this;
|
||||||
|
|
||||||
|
var buffer = new Uint8Array([1, 2, 3, 4, 5, 6, 7, 8]);
|
||||||
|
|
||||||
|
that.fs.open('/myfile', 'w', function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
var fd = result;
|
||||||
|
that.fs.write(fd, buffer, 0, buffer.length, 0, function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
that.fs.ftruncate(fd, 0, function(error) {
|
||||||
|
_error = error;
|
||||||
|
|
||||||
|
that.fs.fstat(fd, function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
_result = result;
|
||||||
|
complete=true;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
waitsFor(function() {
|
||||||
|
return complete;
|
||||||
|
}, 'test to complete', DEFAULT_TIMEOUT);
|
||||||
|
|
||||||
|
runs(function() {
|
||||||
|
expect(_error).toEqual(null);
|
||||||
|
expect(_result.size).toEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should follow symbolic links', function() {
|
||||||
|
var complete = false;
|
||||||
|
var _error, _result, _result2;
|
||||||
|
var that = this;
|
||||||
|
|
||||||
|
var buffer = new Uint8Array([1, 2, 3, 4, 5, 6, 7, 8]);
|
||||||
|
|
||||||
|
that.fs.open('/myfile', 'w', function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
var fd = result;
|
||||||
|
that.fs.write(fd, buffer, 0, buffer.length, 0, function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
that.fs.close(fd, function(error) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
that.fs.symlink('/myfile', '/mylink', function(error) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
that.fs.truncate('/mylink', 0, function(error) {
|
||||||
|
_error = error;
|
||||||
|
|
||||||
|
that.fs.stat('/myfile', function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
_result = result;
|
||||||
|
that.fs.lstat('/mylink', function(error, result) {
|
||||||
|
if(error) throw error;
|
||||||
|
|
||||||
|
_result2 = result;
|
||||||
|
complete=true;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
waitsFor(function() {
|
||||||
|
return complete;
|
||||||
|
}, 'test to complete', DEFAULT_TIMEOUT);
|
||||||
|
|
||||||
|
runs(function() {
|
||||||
|
expect(_error).toEqual(null);
|
||||||
|
expect(_result.size).toEqual(0);
|
||||||
|
expect(_result2.size).not.toEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -26,6 +26,7 @@ define([
|
||||||
"spec/fs.lseek.spec",
|
"spec/fs.lseek.spec",
|
||||||
"spec/fs.symlink.spec",
|
"spec/fs.symlink.spec",
|
||||||
"spec/fs.readlink.spec",
|
"spec/fs.readlink.spec",
|
||||||
|
"spec/fs.truncate.spec",
|
||||||
"spec/path-resolution.spec",
|
"spec/path-resolution.spec",
|
||||||
|
|
||||||
// IDBFS.FileSystem.providers.*
|
// IDBFS.FileSystem.providers.*
|
||||||
|
|
Loading…
Reference in New Issue