0
0
mirror of https://github.com/nodejs/node.git synced 2024-12-01 07:53:06 +01:00
nodejs/lib/fs.js

885 lines
21 KiB
JavaScript
Raw Normal View History

var sys = require('sys'),
events = require('events'),
Buffer = require('buffer').Buffer;
2010-04-08 19:22:55 +02:00
var binding = process.binding('fs');
var fs = exports;
2010-03-11 23:32:10 +01:00
2010-04-08 19:22:55 +02:00
fs.Stats = binding.Stats;
2010-03-15 21:48:03 +01:00
fs.Stats.prototype._checkModeProperty = function (property) {
return ((this.mode & property) === property);
};
2010-03-15 21:48:03 +01:00
fs.Stats.prototype.isDirectory = function () {
return this._checkModeProperty(process.S_IFDIR);
};
2010-03-15 21:48:03 +01:00
fs.Stats.prototype.isFile = function () {
return this._checkModeProperty(process.S_IFREG);
};
2010-03-15 21:48:03 +01:00
fs.Stats.prototype.isBlockDevice = function () {
return this._checkModeProperty(process.S_IFBLK);
};
2010-03-15 21:48:03 +01:00
fs.Stats.prototype.isCharacterDevice = function () {
return this._checkModeProperty(process.S_IFCHR);
};
2010-03-15 21:48:03 +01:00
fs.Stats.prototype.isSymbolicLink = function () {
return this._checkModeProperty(process.S_IFLNK);
};
2010-03-15 21:48:03 +01:00
fs.Stats.prototype.isFIFO = function () {
return this._checkModeProperty(process.S_IFIFO);
};
2010-03-15 21:48:03 +01:00
fs.Stats.prototype.isSocket = function () {
return this._checkModeProperty(process.S_IFSOCK);
};
2010-04-08 19:22:55 +02:00
fs.readFile = function (path, encoding_, callback) {
var encoding = typeof(encoding_) == 'string' ? encoding_ : null;
2010-03-15 18:41:58 +01:00
var callback_ = arguments[arguments.length - 1];
var callback = (typeof(callback_) == 'function' ? callback_ : noop);
binding.stat(path, function (err, stat) {
if (err) { callback(err); return; }
binding.open(path, process.O_RDONLY, 0666, function (err, fd) {
if (err) { callback(err); return; }
var size = stat.size;
var buffer = new Buffer(size);
var offset = 0;
function doRead() {
2010-05-13 21:42:17 +02:00
if (size < 1) {
callback(null, buffer);
return;
}
// position is offset or null so we can read files on unseekable mediums
binding.read(fd, buffer, offset, size - offset, offset || null, function (err, amount) {
2010-05-07 21:50:36 +02:00
if (err) {
callback(err);
binding.close(fd);
return;
}
if (amount + offset < size) {
offset += amount;
doRead();
return;
}
2010-05-07 21:50:36 +02:00
binding.close(fd);
if (encoding) {
try {
callback(null, buffer.toString(encoding));
} catch (err) {
callback(err);
}
} else {
callback(null, buffer);
}
});
}
doRead();
});
2010-03-15 18:41:58 +01:00
});
};
2010-04-08 19:22:55 +02:00
fs.readFileSync = function (path, encoding) {
2010-03-15 18:41:58 +01:00
encoding = encoding || "utf8"; // default to utf8
2010-04-08 19:22:55 +02:00
var fd = binding.open(path, process.O_RDONLY, 0666);
2010-03-15 18:41:58 +01:00
var content = '';
var pos = null; // leave null to allow reads on unseekable devices
2010-03-15 18:41:58 +01:00
var r;
while ((r = fs.read(fd, 4*1024, pos, encoding)) && r[0]) {
2010-03-15 18:41:58 +01:00
content += r[0];
pos += r[1]
}
2010-04-08 19:22:55 +02:00
binding.close(fd);
2010-03-15 18:41:58 +01:00
return content;
};
2010-04-08 19:22:55 +02:00
// Used by binding.open and friends
function stringToFlags(flag) {
// Only mess with strings
if (typeof flag !== 'string') {
return flag;
}
switch (flag) {
case "r": return process.O_RDONLY;
case "r+": return process.O_RDWR;
case "w": return process.O_CREAT | process.O_TRUNC | process.O_WRONLY;
case "w+": return process.O_CREAT | process.O_TRUNC | process.O_RDWR;
2010-04-11 22:46:24 +02:00
case "a": return process.O_APPEND | process.O_CREAT | process.O_WRONLY;
case "a+": return process.O_APPEND | process.O_CREAT | process.O_RDWR;
default: throw new Error("Unknown file open flag: " + flag);
}
}
function noop () {}
// Yes, the follow could be easily DRYed up but I provide the explicit
// list to make the arguments clear.
2010-04-08 19:22:55 +02:00
fs.close = function (fd, callback) {
binding.close(fd, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.closeSync = function (fd) {
return binding.close(fd);
};
2010-04-08 19:22:55 +02:00
fs.open = function (path, flags, mode, callback) {
if (mode === undefined) { mode = 0666; }
2010-04-08 19:22:55 +02:00
binding.open(path, stringToFlags(flags), mode, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.openSync = function (path, flags, mode) {
if (mode === undefined) { mode = 0666; }
2010-04-08 19:22:55 +02:00
return binding.open(path, stringToFlags(flags), mode);
};
fs.read = function (fd, buffer, offset, length, position, callback) {
if (!(buffer instanceof Buffer)) {
// legacy string interface (fd, length, position, encoding, callback)
var cb = arguments[4];
encoding = arguments[3];
position = arguments[2];
length = arguments[1];
buffer = new Buffer(length);
offset = 0;
callback = function(err, bytesRead) {
if (!cb) {
return;
}
var str = (bytesRead > 0)
? buffer.toString(encoding, 0, bytesRead)
: '';
(cb)(err, str, bytesRead);
};
}
binding.read(fd, buffer, offset, length, position, callback || noop);
};
fs.readSync = function (fd, buffer, offset, length, position) {
var legacy = false;
if (!(buffer instanceof Buffer)) {
// legacy string interface (fd, length, position, encoding, callback)
legacy = true;
encoding = arguments[3];
position = arguments[2];
length = arguments[1];
buffer = new Buffer(length);
offset = 0;
}
var r = binding.read(fd, buffer, offset, length, position);
if (!legacy) {
return r;
}
var str = (r > 0)
? buffer.toString(encoding, 0, r)
: '';
return [str, r];
};
fs.write = function (fd, buffer, offset, length, position, callback) {
if (!(buffer instanceof Buffer)) {
// legacy string interface (fd, data, position, encoding, callback)
callback = arguments[4];
position = arguments[2];
buffer = new Buffer(''+arguments[1], arguments[3]);
offset = 0;
length = buffer.length;
}
binding.write(fd, buffer, offset, length, position, callback || noop);
};
fs.writeSync = function (fd, buffer, offset, length, position) {
if (!(buffer instanceof Buffer)) {
// legacy string interface (fd, data, position, encoding)
position = arguments[2];
buffer = new Buffer(''+arguments[1], arguments[3]);
offset = 0;
length = buffer.length;
}
binding.write(fd, buffer, offset, length, position);
};
2010-04-08 19:22:55 +02:00
fs.rename = function (oldPath, newPath, callback) {
binding.rename(oldPath, newPath, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.renameSync = function (oldPath, newPath) {
return binding.rename(oldPath, newPath);
};
2010-04-08 19:22:55 +02:00
fs.truncate = function (fd, len, callback) {
binding.truncate(fd, len, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.truncateSync = function (fd, len) {
return binding.truncate(fd, len);
};
2010-04-08 19:22:55 +02:00
fs.rmdir = function (path, callback) {
binding.rmdir(path, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.rmdirSync = function (path) {
return binding.rmdir(path);
};
fs.fdatasync = function (fd, callback) {
binding.fdatasync(fd, callback || noop);
};
fs.fdatasyncSync = function (fd) {
return binding.fdatasync(fd);
};
fs.fsync = function (fd, callback) {
binding.fsync(fd, callback || noop);
};
fs.fsyncSync = function (fd) {
return binding.fsync(fd);
};
2010-04-08 19:22:55 +02:00
fs.mkdir = function (path, mode, callback) {
binding.mkdir(path, mode, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.mkdirSync = function (path, mode) {
return binding.mkdir(path, mode);
};
2010-04-08 19:22:55 +02:00
fs.sendfile = function (outFd, inFd, inOffset, length, callback) {
binding.sendfile(outFd, inFd, inOffset, length, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.sendfileSync = function (outFd, inFd, inOffset, length) {
return binding.sendfile(outFd, inFd, inOffset, length);
};
2010-04-08 19:22:55 +02:00
fs.readdir = function (path, callback) {
binding.readdir(path, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.readdirSync = function (path) {
return binding.readdir(path);
};
fs.fstat = function (fd, callback) {
binding.fstat(fd, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.lstat = function (path, callback) {
binding.lstat(path, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.stat = function (path, callback) {
binding.stat(path, callback || noop);
};
fs.fstatSync = function (fd) {
return binding.fstat(fd);
};
2010-04-08 19:22:55 +02:00
fs.lstatSync = function (path) {
return binding.lstat(path);
};
2010-04-08 19:22:55 +02:00
fs.statSync = function (path) {
return binding.stat(path);
};
2010-04-08 19:22:55 +02:00
fs.readlink = function (path, callback) {
binding.readlink(path, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.readlinkSync = function (path) {
return binding.readlink(path);
};
2010-04-08 19:22:55 +02:00
fs.symlink = function (destination, path, callback) {
binding.symlink(destination, path, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.symlinkSync = function (destination, path) {
return binding.symlink(destination, path);
};
2010-04-08 19:22:55 +02:00
fs.link = function (srcpath, dstpath, callback) {
binding.link(srcpath, dstpath, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.linkSync = function (srcpath, dstpath) {
return binding.link(srcpath, dstpath);
};
2010-04-08 19:22:55 +02:00
fs.unlink = function (path, callback) {
binding.unlink(path, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.unlinkSync = function (path) {
return binding.unlink(path);
};
2010-04-08 19:22:55 +02:00
fs.chmod = function (path, mode, callback) {
binding.chmod(path, mode, callback || noop);
};
2010-04-08 19:22:55 +02:00
fs.chmodSync = function (path, mode) {
return binding.chmod(path, mode);
};
function writeAll (fd, data, encoding, callback) {
2010-04-08 19:22:55 +02:00
fs.write(fd, data, 0, encoding, function (writeErr, written) {
if (writeErr) {
2010-04-08 19:22:55 +02:00
fs.close(fd, function () {
if (callback) callback(writeErr);
});
} else {
if (written === data.length) {
2010-04-08 19:22:55 +02:00
fs.close(fd, callback);
} else {
writeAll(fd, data.slice(written), encoding, callback);
}
}
});
}
2010-04-08 19:22:55 +02:00
fs.writeFile = function (path, data, encoding_, callback) {
var encoding = (typeof(encoding_) == 'string' ? encoding_ : 'utf8');
var callback_ = arguments[arguments.length - 1];
var callback = (typeof(callback_) == 'function' ? callback_ : null);
2010-04-08 19:22:55 +02:00
fs.open(path, 'w', 0666, function (openErr, fd) {
if (openErr) {
if (callback) callback(openErr);
} else {
writeAll(fd, data, encoding, callback);
}
});
};
2010-04-08 19:22:55 +02:00
fs.writeFileSync = function (path, data, encoding) {
encoding = encoding || "utf8"; // default to utf8
2010-04-08 19:22:55 +02:00
var fd = fs.openSync(path, "w");
var written = 0;
while (written < data.length) {
2010-04-08 19:22:55 +02:00
written += fs.writeSync(fd, data, 0, encoding);
data = data.slice(written);
}
2010-04-08 19:22:55 +02:00
fs.closeSync(fd);
};
2010-04-08 19:22:55 +02:00
fs.cat = function () {
throw new Error("fs.cat is deprecated. Please use fs.readFile instead.");
};
2010-04-08 19:22:55 +02:00
fs.catSync = function () {
throw new Error("fs.catSync is deprecated. Please use fs.readFileSync instead.");
};
2010-03-01 19:42:37 +01:00
// Stat Change Watchers
var statWatchers = {};
2010-04-08 19:22:55 +02:00
fs.watchFile = function (filename) {
2010-03-01 19:42:37 +01:00
var stat;
var options;
var listener;
if ("object" == typeof arguments[1]) {
options = arguments[1];
listener = arguments[2];
} else {
options = {};
listener = arguments[1];
}
if (options.persistent === undefined) options.persistent = true;
if (options.interval === undefined) options.interval = 0;
if (statWatchers[filename]) {
2010-03-01 19:42:37 +01:00
stat = statWatchers[filename];
} else {
2010-04-08 19:22:55 +02:00
statWatchers[filename] = new binding.StatWatcher();
2010-03-01 19:42:37 +01:00
stat = statWatchers[filename];
stat.start(filename, options.persistent, options.interval);
}
stat.addListener("change", listener);
return stat;
};
2010-04-08 19:22:55 +02:00
fs.unwatchFile = function (filename) {
if (statWatchers[filename]) {
2010-03-01 19:42:37 +01:00
stat = statWatchers[filename];
stat.stop();
statWatchers[filename] = undefined;
}
};
// Realpath
var path = require('path');
2010-05-19 08:04:24 +02:00
var normalize = path.normalize;
var normalizeArray = path.normalizeArray;
2010-04-08 19:22:55 +02:00
fs.realpathSync = function (path) {
var seen_links = {}, knownHards = {}, buf, i = 0, part, x, stats;
if (path.charAt(0) !== '/') {
var cwd = process.cwd().split('/');
path = cwd.concat(path.split('/'));
path = normalizeArray(path);
i = cwd.length;
buf = [].concat(cwd);
} else {
path = normalizeArray(path.split('/'));
buf = [''];
}
for (; i<path.length; i++) {
part = path.slice(0, i+1).join('/');
if (part.length !== 0) {
if (part in knownHards) {
buf.push(path[i]);
} else {
2010-04-08 19:22:55 +02:00
stats = fs.lstatSync(part);
if (stats.isSymbolicLink()) {
x = stats.dev.toString(32)+":"+stats.ino.toString(32);
if (x in seen_links)
throw new Error("cyclic link at "+part);
seen_links[x] = true;
2010-04-08 19:22:55 +02:00
part = fs.readlinkSync(part);
if (part.charAt(0) === '/') {
// absolute
path = normalizeArray(part.split('/'));
buf = [''];
i = 0;
} else {
// relative
Array.prototype.splice.apply(path, [i, 1].concat(part.split('/')));
part = normalizeArray(path);
var y = 0, L = Math.max(path.length, part.length), delta;
for (; y<L && path[y] === part[y]; y++);
if (y !== L) {
path = part;
delta = i-y;
i = y-1;
if (delta > 0) buf.splice(y, delta);
} else {
i--;
}
}
} else {
buf.push(path[i]);
knownHards[buf.join('/')] = true;
}
}
}
}
return buf.join('/');
}
2010-04-08 19:22:55 +02:00
fs.realpath = function (path, callback) {
var seen_links = {}, knownHards = {}, buf = [''], i = 0, part, x;
if (path.charAt(0) !== '/') {
// assumes cwd is canonical
var cwd = process.cwd().split('/');
path = cwd.concat(path.split('/'));
path = normalizeArray(path);
i = cwd.length-1;
buf = [].concat(cwd);
} else {
path = normalizeArray(path.split('/'));
}
function done(err) {
if (callback) {
if (!err) callback(err, buf.join('/'));
else callback(err);
}
}
function next() {
if (++i === path.length) return done();
part = path.slice(0, i+1).join('/');
if (part.length === 0) return next();
if (part in knownHards) {
buf.push(path[i]);
next();
} else {
2010-04-08 19:22:55 +02:00
fs.lstat(part, function(err, stats){
if (err) return done(err);
if (stats.isSymbolicLink()) {
x = stats.dev.toString(32)+":"+stats.ino.toString(32);
if (x in seen_links)
return done(new Error("cyclic link at "+part));
seen_links[x] = true;
2010-04-08 19:22:55 +02:00
fs.readlink(part, function(err, npart){
if (err) return done(err);
part = npart;
if (part.charAt(0) === '/') {
// absolute
path = normalizeArray(part.split('/'));
buf = [''];
i = 0;
} else {
// relative
Array.prototype.splice.apply(path, [i, 1].concat(part.split('/')));
part = normalizeArray(path);
var y = 0, L = Math.max(path.length, part.length), delta;
for (; y<L && path[y] === part[y]; y++);
if (y !== L) {
path = part;
delta = i-y;
i = y-1; // resolve new node if needed
if (delta > 0) buf.splice(y, delta);
}
else {
i--; // resolve new node if needed
}
}
next();
2010-04-08 19:22:55 +02:00
}); // binding.readlink
}
else {
buf.push(path[i]);
knownHards[buf.join('/')] = true;
next();
}
2010-04-08 19:22:55 +02:00
}); // binding.lstat
}
}
next();
}
2010-03-02 23:12:52 +01:00
2010-04-08 19:22:55 +02:00
fs.createReadStream = function(path, options) {
return new ReadStream(path, options);
2010-03-03 12:39:41 +01:00
};
var ReadStream = fs.ReadStream = function(path, options) {
events.EventEmitter.call(this);
2010-03-03 12:39:41 +01:00
this.path = path;
this.fd = null;
this.readable = true;
this.paused = false;
this.flags = 'r';
this.encoding = 'binary';
this.mode = 0666;
this.bufferSize = 4 * 1024;
options = options || {};
// Mixin options into this
var keys = Object.keys(options);
for (var index = 0, length = keys.length; index < length; index++) {
var key = keys[index];
this[key] = options[key];
}
2010-03-03 12:39:41 +01:00
2010-04-28 01:17:28 +02:00
if (this.fd !== null) {
return;
}
2010-03-03 12:39:41 +01:00
2010-04-28 01:17:28 +02:00
var self = this;
fs.open(this.path, this.flags, this.mode, function(err, fd) {
if (err) {
self.emit('error', err);
self.readable = false;
2010-03-03 12:39:41 +01:00
return;
}
self.fd = fd;
self.emit('open', fd);
self._read();
});
};
sys.inherits(ReadStream, events.EventEmitter);
2010-03-03 12:39:41 +01:00
fs.FileReadStream = fs.ReadStream; // support the legacy name
ReadStream.prototype.setEncoding = function(encoding) {
2010-04-28 01:16:05 +02:00
this.encoding = encoding;
};
ReadStream.prototype._read = function () {
var self = this;
if (!self.readable || self.paused) return;
2010-03-03 12:39:41 +01:00
fs.read(self.fd,
self.bufferSize,
undefined,
self.encoding,
function(err, data, bytesRead) {
2010-03-03 12:39:41 +01:00
if (err) {
self.emit('error', err);
self.readable = false;
2010-03-03 12:39:41 +01:00
return;
}
if (bytesRead === 0) {
self.emit('end');
self.destroy();
return;
}
// do not emit events if the stream is paused
if (self.paused) {
self.buffer = data;
return;
}
// do not emit events anymore after we declared the stream unreadable
if (!self.readable) {
return;
}
self.emit('data', data);
self._read();
2010-03-03 12:39:41 +01:00
});
};
2010-03-03 12:39:41 +01:00
var readStreamForceCloseWarning;
ReadStream.prototype.forceClose = function (cb) {
if (!readStreamForceCloseWarning) {
readStreamForceCloseWarning = "ReadStream.prototype.forceClose renamed to destroy()";
sys.error(readStreamForceCloseWarning);
}
return this.destroy(cb);
}
ReadStream.prototype.destroy = function (cb) {
var self = this;
this.readable = false;
function close() {
fs.close(self.fd, function(err) {
if (err) {
if (cb) {
cb(err);
}
self.emit('error', err);
return;
}
if (cb) {
cb(null);
}
self.emit('close');
});
}
if (this.fd) {
close();
} else {
this.addListener('open', close);
}
};
2010-03-03 12:39:41 +01:00
ReadStream.prototype.pause = function() {
this.paused = true;
};
2010-03-03 12:39:41 +01:00
ReadStream.prototype.resume = function() {
this.paused = false;
if (this.buffer) {
this.emit('data', this.buffer);
this.buffer = null;
}
this._read();
2010-03-03 12:39:41 +01:00
};
2010-03-03 12:39:41 +01:00
2010-04-08 19:22:55 +02:00
fs.createWriteStream = function(path, options) {
return new WriteStream(path, options);
2010-03-02 23:12:52 +01:00
};
var WriteStream = fs.WriteStream = function(path, options) {
events.EventEmitter.call(this);
2010-03-02 23:12:52 +01:00
this.path = path;
this.fd = null;
this.writeable = true;
2010-03-02 23:12:52 +01:00
this.flags = 'w';
this.encoding = 'binary';
this.mode = 0666;
options = options || {};
// Mixin options into this
var keys = Object.keys(options);
for (var index = 0, length = keys.length; index < length; index++) {
var key = keys[index];
this[key] = options[key];
}
2010-03-02 23:12:52 +01:00
this.busy = false;
this._queue = [];
2010-03-02 23:12:52 +01:00
2010-04-20 00:22:59 +02:00
if (this.fd === null) {
this._queue.push([fs.open, this.path, this.flags, this.mode, undefined]);
this.flush();
}
};
sys.inherits(WriteStream, events.EventEmitter);
2010-03-02 23:12:52 +01:00
fs.FileWriteStream = fs.WriteStream; // support the legacy name
2010-03-02 23:12:52 +01:00
WriteStream.prototype.flush = function () {
if (this.busy) return;
var self = this;
2010-03-02 23:12:52 +01:00
var args = this._queue.shift();
if (!args) return self.emit('drain');
2010-03-02 23:12:52 +01:00
this.busy = true;
2010-03-02 23:12:52 +01:00
var method = args.shift(),
cb = args.pop();
2010-03-02 23:12:52 +01:00
var self = this;
2010-03-02 23:12:52 +01:00
args.push(function(err) {
self.busy = false;
if (err) {
self.writeable = false;
if (cb) {
cb(err);
}
self.emit('error', err);
return;
}
// stop flushing after close
if (method === fs.close) {
if (cb) {
cb(null);
}
self.emit('close');
return;
}
2010-03-02 23:12:52 +01:00
// save reference for file pointer
if (method === fs.open) {
self.fd = arguments[1];
self.emit('open', self.fd);
} else if (cb) {
// write callback
cb(null, arguments[1]);
2010-03-02 23:12:52 +01:00
}
self.flush();
});
2010-03-02 23:12:52 +01:00
// Inject the file pointer
if (method !== fs.open) {
args.unshift(self.fd);
}
2010-03-02 23:12:52 +01:00
method.apply(this, args);
};
2010-03-02 23:12:52 +01:00
WriteStream.prototype.write = function (data) {
if (!this.writeable) {
throw new Error('stream not writeable');
}
var cb;
if (typeof(arguments[arguments.length-1]) == 'function') {
cb = arguments[arguments.length-1];
}
if (data instanceof Buffer) {
this._queue.push([fs.write, data, 0, data.length, null, cb]);
} else {
var encoding = 'utf8';
if (typeof(arguments[1]) == 'string') encoding = arguments[1];
this._queue.push([fs.write, data, undefined, encoding, cb]);
}
this.flush();
return false;
};
var writeStreamCloseWarning;
WriteStream.prototype.close = function (cb) {
if (!writeStreamCloseWarning) {
writeStreamCloseWarning = "WriteStream.prototype.close renamed to end()";
2010-04-11 22:46:24 +02:00
sys.error(writeStreamCloseWarning);
}
return this.end(cb);
}
WriteStream.prototype.end = function (cb) {
this.writeable = false;
this._queue.push([fs.close, cb]);
this.flush();
};
var writeStreamForceCloseWarning;
WriteStream.prototype.forceClose = function (cb) {
if (!writeStreamForceCloseWarning) {
writeStreamForceCloseWarning = "WriteStream.prototype.forceClose renamed to destroy()";
sys.error(writeStreamForceCloseWarning);
}
return this.destroy(cb);
}
WriteStream.prototype.forceClose = function (cb) {
this.writeable = false;
fs.close(self.fd, function(err) {
if (err) {
if (cb) {
cb(err);
}
self.emit('error', err);
return;
}
if (cb) {
cb(null);
}
self.emit('close');
});
2010-03-02 23:12:52 +01:00
};