diff --git a/lib/preprocessor.js b/lib/preprocessor.js index 860bcdbdc..1c47849f4 100644 --- a/lib/preprocessor.js +++ b/lib/preprocessor.js @@ -75,10 +75,19 @@ var createPreprocessor = function (config, basePath, injector) { return function preprocess (file, done) { patterns = Object.keys(config) - - return fs.readFile(file.originalPath, function (err, buffer) { + var retryCount = 0 + var maxRetries = 3 + function readFileCallback (err, buffer) { if (err) { - throw err + log.warn(err) + if (retryCount < maxRetries) { + retryCount++ + log.warn('retrying ' + retryCount) + fs.readFile(file.originalPath, readFileCallback) + return + } else { + throw err + } } isBinaryFile(buffer, buffer.length, function (err, thisFileIsBinary) { @@ -121,7 +130,8 @@ var createPreprocessor = function (config, basePath, injector) { nextPreprocessor(null, thisFileIsBinary ? buffer : buffer.toString()) }) - }) + } + return fs.readFile(file.originalPath, readFileCallback) } } createPreprocessor.$inject = ['config.preprocessors', 'config.basePath', 'injector'] diff --git a/test/unit/preprocessor.spec.js b/test/unit/preprocessor.spec.js index 2de625023..ccd488f3a 100644 --- a/test/unit/preprocessor.spec.js +++ b/test/unit/preprocessor.spec.js @@ -229,6 +229,54 @@ describe('preprocessor', () => { }) }) + describe('when fs.readFile fails', () => { + var file = {originalPath: '/some/a.js', path: 'path'} + var getReadFileCallback = (nthCall) => { + return mockFs.readFile.args[nthCall][1] + } + + beforeEach(() => { + sinon.stub(mockFs, 'readFile') + }) + + it('should retry up to 3 times', (done) => { + var fakePreprocessor = sinon.spy((content, file, done) => { + done(null, content) + }) + + var injector = new di.Injector([{ + 'preprocessor:fake': ['factory', () => fakePreprocessor] + }, emitterSetting]) + + var pp = m.createPreprocessor({'**/*.js': ['fake']}, null, injector) + + pp(file, () => { + expect(fakePreprocessor).to.have.been.called + done() + }) + getReadFileCallback(0)('error') + getReadFileCallback(1)('error') + var thirdCallback = getReadFileCallback(2) + mockFs.readFile.restore() + thirdCallback('error') + }) + + it('should tbrow after 3 retries', (done) => { + var injector = new di.Injector([{}, emitterSetting]) + + var pp = m.createPreprocessor({'**/*.js': []}, null, injector) + + pp(file, () => { }) + + getReadFileCallback(0)('error') + getReadFileCallback(1)('error') + getReadFileCallback(2)('error') + + expect(() => getReadFileCallback(0)('error')).to.throw('error') + done() + }) + }) + it('should not preprocess binary files', (done) => { var fakePreprocessor = sinon.spy((content, file, done) => { done(null, content)