Working date rolling file stream
This commit is contained in:
parent
be1272cd7c
commit
185f343e68
@ -2,18 +2,20 @@ var fs = require('fs'),
|
|||||||
util = require('util');
|
util = require('util');
|
||||||
|
|
||||||
function debug(message) {
|
function debug(message) {
|
||||||
// util.debug(message);
|
|
||||||
// console.log(message);
|
// console.log(message);
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = BaseRollingFileStream;
|
module.exports = BaseRollingFileStream;
|
||||||
|
|
||||||
function BaseRollingFileStream(filename, options) {
|
function BaseRollingFileStream(filename, options) {
|
||||||
|
|
||||||
|
debug("In BaseRollingFileStream");
|
||||||
this.filename = filename;
|
this.filename = filename;
|
||||||
this.options = options || { encoding: 'utf8', mode: 0644, flags: 'a' };
|
this.options = options || { encoding: 'utf8', mode: 0644, flags: 'a' };
|
||||||
this.rolling = false;
|
this.rolling = false;
|
||||||
this.writesWhileRolling = [];
|
this.writesWhileRolling = [];
|
||||||
this.currentSize = 0;
|
this.currentSize = 0;
|
||||||
|
this.rollBeforeWrite = false;
|
||||||
|
|
||||||
function currentFileSize(file) {
|
function currentFileSize(file) {
|
||||||
var fileSize = 0;
|
var fileSize = 0;
|
||||||
@ -32,7 +34,7 @@ function BaseRollingFileStream(filename, options) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
throwErrorIfArgumentsAreNotValid();
|
throwErrorIfArgumentsAreNotValid();
|
||||||
|
debug("Calling BaseRollingFileStream.super");
|
||||||
BaseRollingFileStream.super_.call(this, this.filename, this.options);
|
BaseRollingFileStream.super_.call(this, this.filename, this.options);
|
||||||
this.currentSize = currentFileSize(this.filename);
|
this.currentSize = currentFileSize(this.filename);
|
||||||
}
|
}
|
||||||
@ -67,18 +69,24 @@ BaseRollingFileStream.prototype.initRolling = function() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
BaseRollingFileStream.prototype.write = function(data, encoding) {
|
BaseRollingFileStream.prototype.write = function(data, encoding) {
|
||||||
|
var canWrite = false;
|
||||||
if (this.rolling) {
|
if (this.rolling) {
|
||||||
this.writesWhileRolling.push({ data: data, encoding: encoding });
|
this.writesWhileRolling.push({ data: data, encoding: encoding });
|
||||||
return false;
|
|
||||||
} else {
|
} else {
|
||||||
var canWrite = BaseRollingFileStream.super_.prototype.write.call(this, data, encoding);
|
if (this.rollBeforeWrite && this.shouldRoll()) {
|
||||||
this.currentSize += data.length;
|
this.writesWhileRolling.push({ data: data, encoding: encoding });
|
||||||
debug('current size = ' + this.currentSize);
|
|
||||||
if (this.shouldRoll()) {
|
|
||||||
this.initRolling();
|
this.initRolling();
|
||||||
|
} else {
|
||||||
|
canWrite = BaseRollingFileStream.super_.prototype.write.call(this, data, encoding);
|
||||||
|
this.currentSize += data.length;
|
||||||
|
debug('current size = ' + this.currentSize);
|
||||||
|
|
||||||
|
if (!this.rollBeforeWrite && this.shouldRoll()) {
|
||||||
|
this.initRolling();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return canWrite;
|
|
||||||
}
|
}
|
||||||
|
return canWrite;
|
||||||
};
|
};
|
||||||
|
|
||||||
BaseRollingFileStream.prototype.shouldRoll = function() {
|
BaseRollingFileStream.prototype.shouldRoll = function() {
|
||||||
|
@ -1,16 +1,89 @@
|
|||||||
var BaseRollingFileStream = require('./BaseRollingFileStream'),
|
var BaseRollingFileStream = require('./BaseRollingFileStream'),
|
||||||
|
format = require('../date_format'),
|
||||||
|
async = require('async'),
|
||||||
|
fs = require('fs'),
|
||||||
util = require('util');
|
util = require('util');
|
||||||
|
|
||||||
module.exports = DateRollingFileStream;
|
module.exports = DateRollingFileStream;
|
||||||
|
|
||||||
function DateRollingFileStream(filename, pattern, options) {
|
function debug(message) {
|
||||||
if (typeof(pattern) === 'object') {
|
// console.log(message);
|
||||||
|
}
|
||||||
|
|
||||||
|
function DateRollingFileStream(filename, pattern, options, now) {
|
||||||
|
debug("Now is " + now);
|
||||||
|
if (pattern && typeof(pattern) === 'object') {
|
||||||
|
now = options;
|
||||||
options = pattern;
|
options = pattern;
|
||||||
pattern = null;
|
pattern = null;
|
||||||
}
|
}
|
||||||
this.pattern = pattern || 'yyyy-mm-dd';
|
this.pattern = pattern || '.yyyy-MM-dd';
|
||||||
|
this.now = now || Date.now;
|
||||||
|
this.lastTimeWeWroteSomething = format.asString(this.pattern, new Date(this.now()));
|
||||||
|
debug("this.now is " + this.now + ", now is " + now);
|
||||||
|
|
||||||
DateRollingFileStream.super_.call(this, filename, options);
|
DateRollingFileStream.super_.call(this, filename, options);
|
||||||
|
this.rollBeforeWrite = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
util.inherits(DateRollingFileStream, BaseRollingFileStream);
|
util.inherits(DateRollingFileStream, BaseRollingFileStream);
|
||||||
|
|
||||||
|
DateRollingFileStream.prototype.shouldRoll = function() {
|
||||||
|
var lastTime = this.lastTimeWeWroteSomething,
|
||||||
|
thisTime = format.asString(this.pattern, new Date(this.now()));
|
||||||
|
|
||||||
|
debug("DateRollingFileStream.shouldRoll with now = " + this.now() + ", thisTime = " + thisTime + ", lastTime = " + lastTime);
|
||||||
|
|
||||||
|
this.lastTimeWeWroteSomething = thisTime;
|
||||||
|
this.previousTime = lastTime;
|
||||||
|
|
||||||
|
return thisTime !== lastTime;
|
||||||
|
};
|
||||||
|
|
||||||
|
DateRollingFileStream.prototype.roll = function(filename, callback) {
|
||||||
|
var that = this,
|
||||||
|
newFilename = filename + this.previousTime;
|
||||||
|
|
||||||
|
debug("Starting roll");
|
||||||
|
debug("Queueing up data until we've finished rolling");
|
||||||
|
debug("Flushing underlying stream");
|
||||||
|
this.flush();
|
||||||
|
|
||||||
|
async.series([
|
||||||
|
deleteAnyExistingFile,
|
||||||
|
renameTheCurrentFile,
|
||||||
|
openANewFile
|
||||||
|
], callback);
|
||||||
|
|
||||||
|
function deleteAnyExistingFile(cb) {
|
||||||
|
//on windows, you can get a EEXIST error if you rename a file to an existing file
|
||||||
|
//so, we'll try to delete the file we're renaming to first
|
||||||
|
fs.unlink(newFilename, function (err) {
|
||||||
|
//ignore err: if we could not delete, it's most likely that it doesn't exist
|
||||||
|
cb();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function renameTheCurrentFile(cb) {
|
||||||
|
debug("Renaming the " + filename + " -> " + newFilename);
|
||||||
|
fs.rename(filename, newFilename, cb);
|
||||||
|
}
|
||||||
|
|
||||||
|
function openANewFile(cb) {
|
||||||
|
debug("Opening a new file");
|
||||||
|
fs.open(
|
||||||
|
filename,
|
||||||
|
that.options.flags,
|
||||||
|
that.options.mode,
|
||||||
|
function (err, fd) {
|
||||||
|
debug("opened new file");
|
||||||
|
var oldLogFileFD = that.fd;
|
||||||
|
that.fd = fd;
|
||||||
|
that.writable = true;
|
||||||
|
fs.close(oldLogFileFD, cb);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
};
|
||||||
|
@ -1,14 +1,26 @@
|
|||||||
var vows = require('vows'),
|
var vows = require('vows'),
|
||||||
assert = require('assert'),
|
assert = require('assert'),
|
||||||
fs = require('fs'),
|
fs = require('fs'),
|
||||||
DateRollingFileStream = require('../../lib/streams').DateRollingFileStream;
|
DateRollingFileStream = require('../../lib/streams').DateRollingFileStream,
|
||||||
|
testTime = new Date(2012, 8, 12, 10, 37, 11);
|
||||||
|
|
||||||
|
function cleanUp(filename) {
|
||||||
|
return function() {
|
||||||
|
fs.unlink(filename);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function now() {
|
||||||
|
return testTime.getTime();
|
||||||
|
}
|
||||||
|
|
||||||
vows.describe('DateRollingFileStream').addBatch({
|
vows.describe('DateRollingFileStream').addBatch({
|
||||||
'arguments': {
|
'arguments': {
|
||||||
topic: new DateRollingFileStream('test-date-rolling-file-stream', 'yyyy-mm-dd.hh'),
|
topic: new DateRollingFileStream(__dirname + '/test-date-rolling-file-stream-1', 'yyyy-mm-dd.hh'),
|
||||||
|
teardown: cleanUp(__dirname + '/test-date-rolling-file-stream-1'),
|
||||||
|
|
||||||
'should take a filename and a pattern and return a FileWriteStream': function(stream) {
|
'should take a filename and a pattern and return a FileWriteStream': function(stream) {
|
||||||
assert.equal(stream.filename, 'test-date-rolling-file-stream');
|
assert.equal(stream.filename, __dirname + '/test-date-rolling-file-stream-1');
|
||||||
assert.equal(stream.pattern, 'yyyy-mm-dd.hh');
|
assert.equal(stream.pattern, 'yyyy-mm-dd.hh');
|
||||||
assert.instanceOf(stream, fs.FileWriteStream);
|
assert.instanceOf(stream, fs.FileWriteStream);
|
||||||
},
|
},
|
||||||
@ -20,15 +32,17 @@ vows.describe('DateRollingFileStream').addBatch({
|
|||||||
},
|
},
|
||||||
|
|
||||||
'default arguments': {
|
'default arguments': {
|
||||||
topic: new DateRollingFileStream('test-date-rolling-file-stream'),
|
topic: new DateRollingFileStream(__dirname + '/test-date-rolling-file-stream-2'),
|
||||||
|
teardown: cleanUp(__dirname + '/test-date-rolling-file-stream-2'),
|
||||||
|
|
||||||
'pattern should be yyyy-mm-dd': function(stream) {
|
'pattern should be .yyyy-MM-dd': function(stream) {
|
||||||
assert.equal(stream.pattern, 'yyyy-mm-dd');
|
assert.equal(stream.pattern, '.yyyy-MM-dd');
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
'with stream arguments': {
|
'with stream arguments': {
|
||||||
topic: new DateRollingFileStream('test-rolling-file-stream', 'yyyy-mm-dd', { mode: 0666 }),
|
topic: new DateRollingFileStream(__dirname + '/test-date-rolling-file-stream-3', 'yyyy-MM-dd', { mode: 0666 }),
|
||||||
|
teardown: cleanUp(__dirname + '/test-date-rolling-file-stream-3'),
|
||||||
|
|
||||||
'should pass them to the underlying stream': function(stream) {
|
'should pass them to the underlying stream': function(stream) {
|
||||||
assert.equal(stream.mode, 0666);
|
assert.equal(stream.mode, 0666);
|
||||||
@ -36,13 +50,73 @@ vows.describe('DateRollingFileStream').addBatch({
|
|||||||
},
|
},
|
||||||
|
|
||||||
'with stream arguments but no pattern': {
|
'with stream arguments but no pattern': {
|
||||||
topic: new DateRollingFileStream('test-rolling-file-stream', { mode: 0666 }),
|
topic: new DateRollingFileStream(__dirname + '/test-date-rolling-file-stream-4', { mode: 0666 }),
|
||||||
|
teardown: cleanUp(__dirname + '/test-date-rolling-file-stream-4'),
|
||||||
|
|
||||||
'should pass them to the underlying stream': function(stream) {
|
'should pass them to the underlying stream': function(stream) {
|
||||||
assert.equal(stream.mode, 0666);
|
assert.equal(stream.mode, 0666);
|
||||||
},
|
},
|
||||||
'should use default pattern': function(stream) {
|
'should use default pattern': function(stream) {
|
||||||
assert.equal(stream.pattern, 'yyyy-mm-dd');
|
assert.equal(stream.pattern, '.yyyy-MM-dd');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
'with a pattern of .yyyy-MM-dd': {
|
||||||
|
topic: function() {
|
||||||
|
var that = this,
|
||||||
|
stream = new DateRollingFileStream(__dirname + '/test-date-rolling-file-stream-5', '.yyyy-MM-dd', null, now);
|
||||||
|
stream.on("open", function() {
|
||||||
|
stream.write("First message\n");
|
||||||
|
//wait for the file system to catch up with us
|
||||||
|
that.callback(null, stream);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
teardown: cleanUp(__dirname + '/test-date-rolling-file-stream-5'),
|
||||||
|
|
||||||
|
'should create a file with the base name': {
|
||||||
|
topic: function(stream) {
|
||||||
|
fs.readFile(__dirname + '/test-date-rolling-file-stream-5', this.callback);
|
||||||
|
},
|
||||||
|
'file should contain first message': function(result) {
|
||||||
|
assert.equal(result.toString(), "First message\n");
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
'when the day changes': {
|
||||||
|
topic: function(stream) {
|
||||||
|
testTime = new Date(2012, 8, 13, 0, 10, 12);
|
||||||
|
stream.write("Second message\n");
|
||||||
|
setTimeout(this.callback, 100);
|
||||||
|
},
|
||||||
|
teardown: cleanUp(__dirname + '/test-date-rolling-file-stream-5.2012-09-12'),
|
||||||
|
|
||||||
|
|
||||||
|
'the number of files': {
|
||||||
|
topic: function() {
|
||||||
|
fs.readdir(__dirname, this.callback);
|
||||||
|
},
|
||||||
|
'should be two': function(files) {
|
||||||
|
assert.equal(files.filter(function(file) { return file.indexOf('test-date-rolling-file-stream-5') > -1; }).length, 2);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
'the file without a date': {
|
||||||
|
topic: function() {
|
||||||
|
fs.readFile(__dirname + '/test-date-rolling-file-stream-5', this.callback);
|
||||||
|
},
|
||||||
|
'should contain the second message': function(contents) {
|
||||||
|
assert.equal(contents.toString(), "Second message\n");
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
'the file with the date': {
|
||||||
|
topic: function() {
|
||||||
|
fs.readFile(__dirname + '/test-date-rolling-file-stream-5.2012-09-12', this.callback);
|
||||||
|
},
|
||||||
|
'should contain the first message': function(contents) {
|
||||||
|
assert.equal(contents.toString(), "First message\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user