2017-04-23 20:02:26 +02:00
|
|
|
'use strict';
|
|
|
|
|
2016-05-01 10:02:01 +01:00
|
|
|
/**
|
|
|
|
* file-locking.js - file system locking (replaces fs-ext)
|
|
|
|
*/
|
|
|
|
|
2017-05-20 12:33:23 +02:00
|
|
|
const async = require('async');
|
|
|
|
const locker = require('lockfile');
|
|
|
|
const fs = require('fs');
|
|
|
|
const path = require('path');
|
2016-05-01 10:02:01 +01:00
|
|
|
|
|
|
|
// locks a file by creating a lock file
|
2017-05-20 12:33:23 +02:00
|
|
|
const lockFile = function(name, next) {
|
|
|
|
const lockFileName = `${name}.lock`;
|
|
|
|
const lockOpts = {
|
2016-05-01 10:02:01 +01:00
|
|
|
wait: 1000, // time (ms) to wait when checking for stale locks
|
|
|
|
pollPeriod: 100, // how often (ms) to re-check stale locks
|
|
|
|
|
|
|
|
stale: 5 * 60 * 1000, // locks are considered stale after 5 minutes
|
|
|
|
|
|
|
|
retries: 100, // number of times to attempt to create a lock
|
2017-04-23 20:02:26 +02:00
|
|
|
retryWait: 100, // time (ms) between tries
|
2017-05-20 12:33:23 +02:00
|
|
|
};
|
2016-05-01 10:02:01 +01:00
|
|
|
|
|
|
|
async.series({
|
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
statdir: function(callback) {
|
2016-05-01 10:02:01 +01:00
|
|
|
// test to see if the directory exists
|
2017-04-23 20:02:26 +02:00
|
|
|
fs.stat(path.dirname(name), function(err, stats) {
|
2016-05-01 10:02:01 +01:00
|
|
|
if (err) {
|
2017-04-23 20:02:26 +02:00
|
|
|
callback(err);
|
2016-05-01 10:02:01 +01:00
|
|
|
} else if (!stats.isDirectory()) {
|
2017-04-23 20:02:26 +02:00
|
|
|
callback(new Error(path.dirname(name) + ' is not a directory'));
|
2016-05-01 10:02:01 +01:00
|
|
|
} else {
|
2017-04-23 20:02:26 +02:00
|
|
|
callback(null);
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
2017-04-23 20:02:26 +02:00
|
|
|
});
|
2016-05-01 10:02:01 +01:00
|
|
|
},
|
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
statfile: function(callback) {
|
2016-05-01 10:02:01 +01:00
|
|
|
// test to see if the file to lock exists
|
2017-04-23 20:02:26 +02:00
|
|
|
fs.stat(name, function(err, stats) {
|
2016-05-01 10:02:01 +01:00
|
|
|
if (err) {
|
2017-04-23 20:02:26 +02:00
|
|
|
callback(err);
|
2016-05-01 10:02:01 +01:00
|
|
|
} else if (!stats.isFile()) {
|
2017-04-23 20:02:26 +02:00
|
|
|
callback(new Error(path.dirname(name) + ' is not a file'));
|
2016-05-01 10:02:01 +01:00
|
|
|
} else {
|
2017-04-23 20:02:26 +02:00
|
|
|
callback(null);
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
lockfile: function(callback) {
|
2016-05-01 10:02:01 +01:00
|
|
|
// try to lock the file
|
2017-04-23 20:02:26 +02:00
|
|
|
locker.lock(lockFileName, lockOpts, callback);
|
|
|
|
},
|
2016-05-01 10:02:01 +01:00
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
}, function(err) {
|
2016-05-01 10:02:01 +01:00
|
|
|
if (err) {
|
|
|
|
// lock failed
|
2017-04-23 20:02:26 +02:00
|
|
|
return next(err);
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// lock succeeded
|
|
|
|
return next(null);
|
2017-04-23 20:02:26 +02:00
|
|
|
});
|
2017-05-20 12:33:23 +02:00
|
|
|
};
|
2016-05-01 10:02:01 +01:00
|
|
|
|
|
|
|
// unlocks file by removing existing lock file
|
2017-05-20 12:33:23 +02:00
|
|
|
const unlockFile= function(name, next) {
|
|
|
|
const lockFileName = `${name}.lock`;
|
2017-04-23 20:02:26 +02:00
|
|
|
locker.unlock(lockFileName, function(err) {
|
2016-05-01 10:02:01 +01:00
|
|
|
if (err) {
|
2017-04-23 20:02:26 +02:00
|
|
|
return next(err);
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
2017-04-23 20:02:26 +02:00
|
|
|
return next(null);
|
|
|
|
});
|
2017-05-20 12:33:23 +02:00
|
|
|
};
|
2016-05-01 10:02:01 +01:00
|
|
|
|
|
|
|
/**
|
2017-05-20 12:33:23 +02:00
|
|
|
* Reads a local file, which involves
|
2016-05-01 10:02:01 +01:00
|
|
|
* optionally taking a lock
|
|
|
|
* reading the file contents
|
|
|
|
* optionally parsing JSON contents
|
2017-05-20 12:33:23 +02:00
|
|
|
* @param {*} name
|
|
|
|
* @param {*} options
|
|
|
|
* @param {*} next
|
2016-05-01 10:02:01 +01:00
|
|
|
*/
|
|
|
|
function readFile(name, options, next) {
|
|
|
|
if (typeof options === 'function' && next === null) {
|
|
|
|
next = options;
|
2017-04-23 20:02:26 +02:00
|
|
|
options = {};
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
options = options || {};
|
|
|
|
options.lock = options.lock || false;
|
|
|
|
options.parse = options.parse || false;
|
2016-05-01 10:02:01 +01:00
|
|
|
|
2017-05-20 12:33:23 +02:00
|
|
|
const lock = function(callback) {
|
2016-05-01 10:02:01 +01:00
|
|
|
if (!options.lock) {
|
2017-04-23 20:02:26 +02:00
|
|
|
return callback(null);
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
lockFile(name, function(err) {
|
2016-05-01 10:02:01 +01:00
|
|
|
if (err) {
|
2017-04-23 20:02:26 +02:00
|
|
|
return callback(err);
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
2017-04-23 20:02:26 +02:00
|
|
|
return callback(null);
|
|
|
|
});
|
2017-05-20 12:33:23 +02:00
|
|
|
};
|
2016-05-01 10:02:01 +01:00
|
|
|
|
2017-05-20 12:33:23 +02:00
|
|
|
const read = function(callback) {
|
2017-04-23 20:02:26 +02:00
|
|
|
fs.readFile(name, 'utf8', function(err, contents) {
|
2016-05-01 10:02:01 +01:00
|
|
|
if (err) {
|
2017-04-23 20:02:26 +02:00
|
|
|
return callback(err);
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
callback(null, contents);
|
|
|
|
});
|
2017-05-20 12:33:23 +02:00
|
|
|
};
|
2016-05-01 10:02:01 +01:00
|
|
|
|
2017-05-20 12:33:23 +02:00
|
|
|
const parseJSON = function(contents, callback) {
|
2016-05-01 10:02:01 +01:00
|
|
|
if (!options.parse) {
|
2017-04-23 20:02:26 +02:00
|
|
|
return callback(null, contents);
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
2017-04-23 20:02:26 +02:00
|
|
|
contents = JSON.parse(contents);
|
|
|
|
return callback(null, contents);
|
2016-05-01 10:02:01 +01:00
|
|
|
} catch (err) {
|
2017-04-23 20:02:26 +02:00
|
|
|
return callback(err);
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
2017-05-20 12:33:23 +02:00
|
|
|
};
|
2016-05-01 10:02:01 +01:00
|
|
|
|
|
|
|
async.waterfall([
|
|
|
|
lock,
|
|
|
|
read,
|
2017-04-23 20:02:26 +02:00
|
|
|
parseJSON,
|
2016-05-01 10:02:01 +01:00
|
|
|
],
|
|
|
|
|
2017-04-23 20:02:26 +02:00
|
|
|
function(err, result) {
|
2016-05-01 10:02:01 +01:00
|
|
|
if (err) {
|
2017-04-23 20:02:26 +02:00
|
|
|
return next(err);
|
2016-05-01 10:02:01 +01:00
|
|
|
} else {
|
2017-04-23 20:02:26 +02:00
|
|
|
return next(null, result);
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
2017-04-23 20:02:26 +02:00
|
|
|
});
|
2016-05-01 10:02:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
exports.lockFile = lockFile;
|
|
|
|
exports.unlockFile = unlockFile;
|
|
|
|
exports.readFile = readFile;
|