mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-11 16:38:17 +01:00
Add BinaryManager
This commit is contained in:
parent
b1b325d00b
commit
2e989fbe83
2
.gitignore
vendored
2
.gitignore
vendored
@ -13,6 +13,8 @@
|
|||||||
/deploy/
|
/deploy/
|
||||||
/coverage/
|
/coverage/
|
||||||
/.nyc_output/
|
/.nyc_output/
|
||||||
|
/ffmpeg*
|
||||||
|
/ffprobe*
|
||||||
|
|
||||||
sw.*
|
sw.*
|
||||||
.DS_STORE
|
.DS_STORE
|
||||||
|
@ -33,6 +33,7 @@ const AudioMetadataMangaer = require('./managers/AudioMetadataManager')
|
|||||||
const RssFeedManager = require('./managers/RssFeedManager')
|
const RssFeedManager = require('./managers/RssFeedManager')
|
||||||
const CronManager = require('./managers/CronManager')
|
const CronManager = require('./managers/CronManager')
|
||||||
const ApiCacheManager = require('./managers/ApiCacheManager')
|
const ApiCacheManager = require('./managers/ApiCacheManager')
|
||||||
|
const BinaryManager = require('./managers/BinaryManager')
|
||||||
const LibraryScanner = require('./scanner/LibraryScanner')
|
const LibraryScanner = require('./scanner/LibraryScanner')
|
||||||
|
|
||||||
//Import the main Passport and Express-Session library
|
//Import the main Passport and Express-Session library
|
||||||
@ -74,6 +75,7 @@ class Server {
|
|||||||
this.rssFeedManager = new RssFeedManager()
|
this.rssFeedManager = new RssFeedManager()
|
||||||
this.cronManager = new CronManager(this.podcastManager)
|
this.cronManager = new CronManager(this.podcastManager)
|
||||||
this.apiCacheManager = new ApiCacheManager()
|
this.apiCacheManager = new ApiCacheManager()
|
||||||
|
this.binaryManager = new BinaryManager()
|
||||||
|
|
||||||
// Routers
|
// Routers
|
||||||
this.apiRouter = new ApiRouter(this)
|
this.apiRouter = new ApiRouter(this)
|
||||||
@ -119,6 +121,7 @@ class Server {
|
|||||||
const libraries = await Database.libraryModel.getAllOldLibraries()
|
const libraries = await Database.libraryModel.getAllOldLibraries()
|
||||||
await this.cronManager.init(libraries)
|
await this.cronManager.init(libraries)
|
||||||
this.apiCacheManager.init()
|
this.apiCacheManager.init()
|
||||||
|
await this.binaryManager.init()
|
||||||
|
|
||||||
if (Database.serverSettings.scannerDisableWatcher) {
|
if (Database.serverSettings.scannerDisableWatcher) {
|
||||||
Logger.info(`[Server] Watcher is disabled`)
|
Logger.info(`[Server] Watcher is disabled`)
|
||||||
|
79
server/managers/BinaryManager.js
Normal file
79
server/managers/BinaryManager.js
Normal file
@ -0,0 +1,79 @@
|
|||||||
|
const path = require('path')
|
||||||
|
const which = require('../libs/which')
|
||||||
|
const fs = require('../libs/fsExtra')
|
||||||
|
const Logger = require('../Logger')
|
||||||
|
const ffbinaries = require('ffbinaries')
|
||||||
|
const { promisify } = require('util')
|
||||||
|
|
||||||
|
class BinaryManager {
|
||||||
|
downloadBinaries = promisify(ffbinaries.downloadBinaries)
|
||||||
|
|
||||||
|
defaultRequiredBinaries = [
|
||||||
|
{ name: 'ffmpeg', envVariable: 'FFMPEG_PATH' },
|
||||||
|
{ name: 'ffprobe', envVariable: 'FFPROBE_PATH' }
|
||||||
|
]
|
||||||
|
|
||||||
|
constructor(requiredBinaries = this.defaultRequiredBinaries) {
|
||||||
|
this.requiredBinaries = requiredBinaries
|
||||||
|
this.mainInstallPath = process.pkg ? path.dirname(process.execPath) : global.appRoot
|
||||||
|
this.altInstallPath = global.ConfigPath
|
||||||
|
}
|
||||||
|
|
||||||
|
async init() {
|
||||||
|
if (this.initialized) return
|
||||||
|
const missingBinaries = await this.findRequiredBinaries()
|
||||||
|
if (missingBinaries.length == 0) return
|
||||||
|
await this.install(missingBinaries)
|
||||||
|
const missingBinariesAfterInstall = await this.findRequiredBinaries()
|
||||||
|
if (missingBinariesAfterInstall.length != 0) {
|
||||||
|
Logger.error(`[BinaryManager] Failed to find or install required binaries: ${missingBinariesAfterInstall.join(', ')}`)
|
||||||
|
process.exit(1)
|
||||||
|
}
|
||||||
|
this.initialized = true
|
||||||
|
}
|
||||||
|
|
||||||
|
async findRequiredBinaries() {
|
||||||
|
const missingBinaries = []
|
||||||
|
for (const binary of this.requiredBinaries) {
|
||||||
|
const binaryPath = await this.findBinary(binary.name, binary.envVariable)
|
||||||
|
if (binaryPath) {
|
||||||
|
Logger.info(`[BinaryManager] Found ${binary.name} at ${binaryPath}`)
|
||||||
|
Logger.info(`[BinaryManager] Updating process.env.${binary.envVariable}`)
|
||||||
|
process.env[binary.envVariable] = binaryPath
|
||||||
|
} else {
|
||||||
|
Logger.info(`[BinaryManager] ${binary.name} not found`)
|
||||||
|
missingBinaries.push(binary.name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return missingBinaries
|
||||||
|
}
|
||||||
|
|
||||||
|
async findBinary(name, envVariable) {
|
||||||
|
const executable = name + (process.platform == 'win32' ? '.exe' : '')
|
||||||
|
const defaultPath = process.env[envVariable]
|
||||||
|
if (defaultPath && await fs.pathExists(defaultPath)) return defaultPath
|
||||||
|
const whichPath = which.sync(executable, { nothrow: true })
|
||||||
|
if (whichPath) return whichPath
|
||||||
|
const mainInstallPath = path.join(this.mainInstallPath, executable)
|
||||||
|
if (await fs.pathExists(mainInstallPath)) return mainInstallPath
|
||||||
|
const altInstallPath = path.join(this.altInstallPath, executable)
|
||||||
|
if (await fs.pathExists(altInstallPath)) return altInstallPath
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
async install(binaries) {
|
||||||
|
if (binaries.length == 0) return
|
||||||
|
Logger.info(`[BinaryManager] Installing binaries: ${binaries.join(', ')}`)
|
||||||
|
let destination = this.mainInstallPath
|
||||||
|
try {
|
||||||
|
await fs.access(destination, fs.constants.W_OK)
|
||||||
|
} catch (err) {
|
||||||
|
destination = this.altInstallPath
|
||||||
|
}
|
||||||
|
await this.downloadBinaries(binaries, { destination })
|
||||||
|
Logger.info(`[BinaryManager] Binaries installed to ${destination}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = BinaryManager
|
262
test/server/managers/BinaryManager.test.js
Normal file
262
test/server/managers/BinaryManager.test.js
Normal file
@ -0,0 +1,262 @@
|
|||||||
|
const chai = require('chai');
|
||||||
|
const sinon = require('sinon');
|
||||||
|
const fs = require('../../../server/libs/fsExtra');
|
||||||
|
const which = require('../../../server/libs/which');
|
||||||
|
const path = require('path');
|
||||||
|
const BinaryManager = require('../../../server/managers/BinaryManager');
|
||||||
|
|
||||||
|
const expect = chai.expect;
|
||||||
|
|
||||||
|
describe('BinaryManager', () => {
|
||||||
|
let binaryManager;
|
||||||
|
|
||||||
|
describe('init', () => {
|
||||||
|
let findStub;
|
||||||
|
let installStub;
|
||||||
|
let errorStub;
|
||||||
|
let exitStub;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
binaryManager = new BinaryManager();
|
||||||
|
findStub = sinon.stub(binaryManager, 'findRequiredBinaries');
|
||||||
|
installStub = sinon.stub(binaryManager, 'install');
|
||||||
|
errorStub = sinon.stub(console, 'error');
|
||||||
|
exitStub = sinon.stub(process, 'exit');
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
findStub.restore();
|
||||||
|
installStub.restore();
|
||||||
|
errorStub.restore();
|
||||||
|
exitStub.restore();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should not install binaries if they are already found', async () => {
|
||||||
|
findStub.resolves([]);
|
||||||
|
|
||||||
|
await binaryManager.init();
|
||||||
|
|
||||||
|
expect(installStub.called).to.be.false;
|
||||||
|
expect(findStub.calledOnce).to.be.true;
|
||||||
|
expect(errorStub.called).to.be.false;
|
||||||
|
expect(exitStub.called).to.be.false;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should install missing binaries', async () => {
|
||||||
|
const missingBinaries = ['ffmpeg', 'ffprobe'];
|
||||||
|
const missingBinariesAfterInstall = [];
|
||||||
|
findStub.onFirstCall().resolves(missingBinaries);
|
||||||
|
findStub.onSecondCall().resolves(missingBinariesAfterInstall);
|
||||||
|
|
||||||
|
await binaryManager.init();
|
||||||
|
|
||||||
|
expect(findStub.calledTwice).to.be.true;
|
||||||
|
expect(installStub.calledOnce).to.be.true;
|
||||||
|
expect(errorStub.called).to.be.false;
|
||||||
|
expect(exitStub.called).to.be.false;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('exit if binaries are not found after installation', async () => {
|
||||||
|
const missingBinaries = ['ffmpeg', 'ffprobe'];
|
||||||
|
const missingBinariesAfterInstall = ['ffmpeg', 'ffprobe'];
|
||||||
|
findStub.onFirstCall().resolves(missingBinaries);
|
||||||
|
findStub.onSecondCall().resolves(missingBinariesAfterInstall);
|
||||||
|
|
||||||
|
await binaryManager.init();
|
||||||
|
|
||||||
|
expect(findStub.calledTwice).to.be.true;
|
||||||
|
expect(installStub.calledOnce).to.be.true;
|
||||||
|
expect(errorStub.calledOnce).to.be.true;
|
||||||
|
expect(exitStub.calledOnce).to.be.true;
|
||||||
|
expect(exitStub.calledWith(1)).to.be.true;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
describe('findRequiredBinaries', () => {
|
||||||
|
let findBinaryStub;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
const requiredBinaries = [{ name: 'ffmpeg', envVariable: 'FFMPEG_PATH' }];
|
||||||
|
binaryManager = new BinaryManager(requiredBinaries);
|
||||||
|
findBinaryStub = sinon.stub(binaryManager, 'findBinary');
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
findBinaryStub.restore();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should put found paths in the correct environment variables', async () => {
|
||||||
|
const pathToFFmpeg = '/path/to/ffmpeg';
|
||||||
|
const missingBinaries = [];
|
||||||
|
delete process.env.FFMPEG_PATH;
|
||||||
|
findBinaryStub.resolves(pathToFFmpeg);
|
||||||
|
|
||||||
|
const result = await binaryManager.findRequiredBinaries();
|
||||||
|
|
||||||
|
expect(result).to.deep.equal(missingBinaries);
|
||||||
|
expect(findBinaryStub.calledOnce).to.be.true;
|
||||||
|
expect(process.env.FFMPEG_PATH).to.equal(pathToFFmpeg);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should add missing binaries to result', async () => {
|
||||||
|
const missingBinaries = ['ffmpeg'];
|
||||||
|
delete process.env.FFMPEG_PATH;
|
||||||
|
findBinaryStub.resolves(null);
|
||||||
|
|
||||||
|
const result = await binaryManager.findRequiredBinaries();
|
||||||
|
|
||||||
|
expect(result).to.deep.equal(missingBinaries);
|
||||||
|
expect(findBinaryStub.calledOnce).to.be.true;
|
||||||
|
expect(process.env.FFMPEG_PATH).to.be.undefined;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('install', () => {
|
||||||
|
let accessStub;
|
||||||
|
let downloadBinariesStub;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
binaryManager = new BinaryManager();
|
||||||
|
accessStub = sinon.stub(fs, 'access');
|
||||||
|
downloadBinariesStub = sinon.stub(binaryManager, 'downloadBinaries');
|
||||||
|
binaryManager.mainInstallPath = '/path/to/main/install'
|
||||||
|
binaryManager.altInstallPath = '/path/to/alt/install'
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
accessStub.restore();
|
||||||
|
downloadBinariesStub.restore();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should not install binaries if no binaries are passed', async () => {
|
||||||
|
const binaries = [];
|
||||||
|
|
||||||
|
await binaryManager.install(binaries);
|
||||||
|
|
||||||
|
expect(accessStub.called).to.be.false;
|
||||||
|
expect(downloadBinariesStub.called).to.be.false;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should install binaries in main install path if has access', async () => {
|
||||||
|
const binaries = ['ffmpeg'];
|
||||||
|
const destination = binaryManager.mainInstallPath;
|
||||||
|
accessStub.withArgs(destination, fs.constants.W_OK).resolves();
|
||||||
|
downloadBinariesStub.resolves();
|
||||||
|
|
||||||
|
await binaryManager.install(binaries);
|
||||||
|
|
||||||
|
expect(accessStub.calledOnce).to.be.true;
|
||||||
|
expect(downloadBinariesStub.calledOnce).to.be.true;
|
||||||
|
expect(downloadBinariesStub.calledWith(binaries, sinon.match({ destination: destination }))).to.be.true;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should install binaries in alt install path if has no access to main', async () => {
|
||||||
|
const binaries = ['ffmpeg'];
|
||||||
|
const mainDestination = binaryManager.mainInstallPath;
|
||||||
|
const destination = binaryManager.altInstallPath;
|
||||||
|
accessStub.withArgs(mainDestination, fs.constants.W_OK).rejects();
|
||||||
|
downloadBinariesStub.resolves();
|
||||||
|
|
||||||
|
await binaryManager.install(binaries);
|
||||||
|
|
||||||
|
expect(accessStub.calledOnce).to.be.true;
|
||||||
|
expect(downloadBinariesStub.calledOnce).to.be.true;
|
||||||
|
expect(downloadBinariesStub.calledWith(binaries, sinon.match({ destination: destination }))).to.be.true;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('findBinary', () => {
|
||||||
|
let binaryManager;
|
||||||
|
let fsPathExistsStub;
|
||||||
|
let whichSyncStub;
|
||||||
|
let mainInstallPath;
|
||||||
|
let altInstallPath;
|
||||||
|
|
||||||
|
const name = 'ffmpeg';
|
||||||
|
const envVariable = 'FFMPEG_PATH';
|
||||||
|
const defaultPath = '/path/to/ffmpeg';
|
||||||
|
const executable = name + (process.platform == 'win32' ? '.exe' : '');
|
||||||
|
const whichPath = '/usr/bin/ffmpeg';
|
||||||
|
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
binaryManager = new BinaryManager();
|
||||||
|
fsPathExistsStub = sinon.stub(fs, 'pathExists');
|
||||||
|
whichSyncStub = sinon.stub(which, 'sync');
|
||||||
|
binaryManager.mainInstallPath = '/path/to/main/install'
|
||||||
|
mainInstallPath = path.join(binaryManager.mainInstallPath, executable);
|
||||||
|
binaryManager.altInstallPath = '/path/to/alt/install'
|
||||||
|
altInstallPath = path.join(binaryManager.altInstallPath, executable);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
fsPathExistsStub.restore();
|
||||||
|
whichSyncStub.restore();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return defaultPath if it exists', async () => {
|
||||||
|
process.env[envVariable] = defaultPath;
|
||||||
|
fsPathExistsStub.withArgs(defaultPath).resolves(true);
|
||||||
|
|
||||||
|
const result = await binaryManager.findBinary(name, envVariable);
|
||||||
|
|
||||||
|
expect(result).to.equal(defaultPath);
|
||||||
|
expect(fsPathExistsStub.calledOnceWith(defaultPath)).to.be.true;
|
||||||
|
expect(whichSyncStub.notCalled).to.be.true;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return whichPath if it exists', async () => {
|
||||||
|
delete process.env[envVariable];
|
||||||
|
whichSyncStub.returns(whichPath);
|
||||||
|
|
||||||
|
const result = await binaryManager.findBinary(name, envVariable);
|
||||||
|
|
||||||
|
expect(result).to.equal(whichPath);
|
||||||
|
expect(fsPathExistsStub.notCalled).to.be.true;
|
||||||
|
expect(whichSyncStub.calledOnce).to.be.true;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return mainInstallPath if it exists', async () => {
|
||||||
|
delete process.env[envVariable];
|
||||||
|
whichSyncStub.returns(null);
|
||||||
|
fsPathExistsStub.withArgs(mainInstallPath).resolves(true);
|
||||||
|
|
||||||
|
const result = await binaryManager.findBinary(name, envVariable);
|
||||||
|
|
||||||
|
expect(result).to.equal(mainInstallPath);
|
||||||
|
expect(whichSyncStub.calledOnce).to.be.true;
|
||||||
|
expect(fsPathExistsStub.calledOnceWith(mainInstallPath)).to.be.true;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return altInstallPath if it exists', async () => {
|
||||||
|
delete process.env[envVariable];
|
||||||
|
whichSyncStub.returns(null);
|
||||||
|
fsPathExistsStub.withArgs(mainInstallPath).resolves(false);
|
||||||
|
fsPathExistsStub.withArgs(altInstallPath).resolves(true);
|
||||||
|
|
||||||
|
const result = await binaryManager.findBinary(name, envVariable);
|
||||||
|
|
||||||
|
expect(result).to.equal(altInstallPath);
|
||||||
|
expect(whichSyncStub.calledOnce).to.be.true;
|
||||||
|
expect(fsPathExistsStub.calledTwice).to.be.true;
|
||||||
|
expect(fsPathExistsStub.calledWith(mainInstallPath)).to.be.true;
|
||||||
|
expect(fsPathExistsStub.calledWith(altInstallPath)).to.be.true;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return null if binary is not found', async () => {
|
||||||
|
delete process.env[envVariable];
|
||||||
|
whichSyncStub.returns(null);
|
||||||
|
fsPathExistsStub.withArgs(mainInstallPath).resolves(false);
|
||||||
|
fsPathExistsStub.withArgs(altInstallPath).resolves(false);
|
||||||
|
|
||||||
|
const result = await binaryManager.findBinary(name, envVariable);
|
||||||
|
|
||||||
|
expect(result).to.be.null;
|
||||||
|
expect(whichSyncStub.calledOnce).to.be.true;
|
||||||
|
expect(fsPathExistsStub.calledTwice).to.be.true;
|
||||||
|
expect(fsPathExistsStub.calledWith(mainInstallPath)).to.be.true;
|
||||||
|
expect(fsPathExistsStub.calledWith(altInstallPath)).to.be.true;
|
||||||
|
});
|
||||||
|
});
|
Loading…
Reference in New Issue
Block a user