mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-20 13:30:34 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			438 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			438 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| const child_process = require('child_process')
 | |
| const { promisify } = require('util')
 | |
| const exec = promisify(child_process.exec)
 | |
| const os = require('os')
 | |
| const axios = require('axios')
 | |
| const path = require('path')
 | |
| const which = require('../libs/which')
 | |
| const fs = require('../libs/fsExtra')
 | |
| const Logger = require('../Logger')
 | |
| const fileUtils = require('../utils/fileUtils')
 | |
| const StreamZip = require('../libs/nodeStreamZip')
 | |
| 
 | |
| class ZippedAssetDownloader {
 | |
|   constructor() {
 | |
|     this.assetCache = {}
 | |
|   }
 | |
| 
 | |
|   getReleaseUrl(releaseTag) {
 | |
|     throw new Error('Not implemented')
 | |
|   }
 | |
| 
 | |
|   extractAssetUrl(assets, assetName) {
 | |
|     throw new Error('Not implemented')
 | |
|   }
 | |
| 
 | |
|   getAssetName(binaryName, releaseTag) {
 | |
|     throw new Error('Not implemented')
 | |
|   }
 | |
| 
 | |
|   getAssetFileName(binaryName) {
 | |
|     throw new Error('Not implemented')
 | |
|   }
 | |
| 
 | |
|   async getAssetUrl(releaseTag, assetName) {
 | |
|     // Check if the assets information is already cached for the release tag
 | |
|     if (this.assetCache[releaseTag]) {
 | |
|       Logger.debug(`[ZippedAssetDownloader] release ${releaseTag}: assets found in cache.`)
 | |
|     } else {
 | |
|       // Get the release information
 | |
|       const releaseUrl = this.getReleaseUrl(releaseTag)
 | |
|       const releaseResponse = await axios.get(releaseUrl, { headers: { 'User-Agent': 'axios' } })
 | |
| 
 | |
|       // Cache the assets information for the release tag
 | |
|       this.assetCache[releaseTag] = releaseResponse.data
 | |
|       Logger.debug(`[ZippedAssetDownloader] release ${releaseTag}: assets fetched from API.`)
 | |
|     }
 | |
| 
 | |
|     const assets = this.assetCache[releaseTag]
 | |
|     const assetUrl = this.extractAssetUrl(assets, assetName)
 | |
| 
 | |
|     return assetUrl
 | |
|   }
 | |
| 
 | |
|   async downloadAsset(assetUrl, destDir) {
 | |
|     const zipPath = path.join(destDir, 'temp.zip')
 | |
|     const writer = fs.createWriteStream(zipPath)
 | |
| 
 | |
|     const assetResponse = await axios({ url: assetUrl, responseType: 'stream' })
 | |
| 
 | |
|     assetResponse.data.pipe(writer)
 | |
| 
 | |
|     await new Promise((resolve, reject) => {
 | |
|       writer.on('finish', () => {
 | |
|         Logger.debug(`[ZippedAssetDownloader] Downloaded asset ${assetUrl} to ${zipPath}`)
 | |
|         resolve()
 | |
|       })
 | |
|       writer.on('error', (err) => {
 | |
|         Logger.error(`[ZippedAssetDownloader] Error downloading asset ${assetUrl}: ${err.message}`)
 | |
|         reject(err)
 | |
|       })
 | |
|     })
 | |
| 
 | |
|     return zipPath
 | |
|   }
 | |
| 
 | |
|   async extractFiles(zipPath, filesToExtract, destDir) {
 | |
|     const zip = new StreamZip.async({ file: zipPath })
 | |
| 
 | |
|     try {
 | |
|       for (const file of filesToExtract) {
 | |
|         const outputPath = path.join(destDir, file.outputFileName)
 | |
|         if (!(await zip.entry(file.pathInsideZip))) {
 | |
|           Logger.error(`[ZippedAssetDownloader] File ${file.pathInsideZip} not found in zip file ${zipPath}`)
 | |
|           continue
 | |
|         }
 | |
|         await zip.extract(file.pathInsideZip, outputPath)
 | |
|         Logger.debug(`[ZippedAssetDownloader] Extracted file ${file.pathInsideZip} to ${outputPath}`)
 | |
| 
 | |
|         // Set executable permission for Linux
 | |
|         if (process.platform !== 'win32') {
 | |
|           await fs.chmod(outputPath, 0o755)
 | |
|         }
 | |
|       }
 | |
|     } catch (error) {
 | |
|       Logger.error('[ZippedAssetDownloader] Error extracting files:', error)
 | |
|       throw error
 | |
|     } finally {
 | |
|       await zip.close()
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   async downloadAndExtractFiles(releaseTag, assetName, filesToExtract, destDir) {
 | |
|     let zipPath
 | |
|     try {
 | |
|       await fs.ensureDir(destDir)
 | |
|       const assetUrl = await this.getAssetUrl(releaseTag, assetName)
 | |
|       zipPath = await this.downloadAsset(assetUrl, destDir)
 | |
|       await this.extractFiles(zipPath, filesToExtract, destDir)
 | |
|     } catch (error) {
 | |
|       Logger.error(`[ZippedAssetDownloader] Error downloading or extracting files: ${error.message}`)
 | |
|     } finally {
 | |
|       if (zipPath) await fs.remove(zipPath)
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   async downloadBinary(binaryName, releaseTag, destDir) {
 | |
|     const assetName = this.getAssetName(binaryName, releaseTag)
 | |
|     const fileName = this.getAssetFileName(binaryName)
 | |
|     const filesToExtract = [{ pathInsideZip: fileName, outputFileName: fileName }]
 | |
| 
 | |
|     await this.downloadAndExtractFiles(releaseTag, assetName, filesToExtract, destDir)
 | |
|   }
 | |
| }
 | |
| 
 | |
| class FFBinariesDownloader extends ZippedAssetDownloader {
 | |
|   constructor() {
 | |
|     super()
 | |
|     this.platformSuffix = this.getPlatformSuffix()
 | |
|   }
 | |
| 
 | |
|   getPlatformSuffix() {
 | |
|     var type = os.type().toLowerCase()
 | |
|     var arch = os.arch().toLowerCase()
 | |
| 
 | |
|     if (type === 'darwin') {
 | |
|       return 'osx-64'
 | |
|     }
 | |
| 
 | |
|     if (type === 'windows_nt') {
 | |
|       return arch === 'x64' ? 'windows-64' : 'windows-32'
 | |
|     }
 | |
| 
 | |
|     if (type === 'linux') {
 | |
|       if (arch === 'arm') return 'linux-armel'
 | |
|       if (arch === 'arm64') return 'linux-arm64'
 | |
|       return arch === 'x64' ? 'linux-64' : 'linux-32'
 | |
|     }
 | |
| 
 | |
|     return null
 | |
|   }
 | |
| 
 | |
|   getReleaseUrl(releaseTag) {
 | |
|     return `https://ffbinaries.com/api/v1/version/${releaseTag}`
 | |
|   }
 | |
| 
 | |
|   extractAssetUrl(assets, assetName) {
 | |
|     const assetUrl = assets?.bin?.[this.platformSuffix]?.[assetName]
 | |
| 
 | |
|     if (!assetUrl) {
 | |
|       throw new Error(`[FFBinariesDownloader] Asset ${assetName} not found for platform ${this.platformSuffix}`)
 | |
|     }
 | |
| 
 | |
|     return assetUrl
 | |
|   }
 | |
| 
 | |
|   getAssetName(binaryName, releaseTag) {
 | |
|     return binaryName
 | |
|   }
 | |
| 
 | |
|   getAssetFileName(binaryName) {
 | |
|     return process.platform === 'win32' ? `${binaryName}.exe` : binaryName
 | |
|   }
 | |
| }
 | |
| 
 | |
| class NunicodeDownloader extends ZippedAssetDownloader {
 | |
|   constructor() {
 | |
|     super()
 | |
|     this.platformSuffix = this.getPlatformSuffix()
 | |
|   }
 | |
| 
 | |
|   getPlatformSuffix() {
 | |
|     const platform = process.platform
 | |
|     const arch = process.arch
 | |
| 
 | |
|     if (platform === 'win32' && arch === 'x64') {
 | |
|       return 'win-x64'
 | |
|     } else if (platform === 'darwin' && (arch === 'x64' || arch === 'arm64')) {
 | |
|       return 'osx-arm64'
 | |
|     } else if (platform === 'linux' && arch === 'x64') {
 | |
|       return 'linux-x64'
 | |
|     } else if (platform === 'linux' && arch === 'arm64') {
 | |
|       return 'linux-arm64'
 | |
|     }
 | |
| 
 | |
|     return null
 | |
|   }
 | |
| 
 | |
|   async getAssetUrl(releaseTag, assetName) {
 | |
|     return `https://github.com/mikiher/nunicode-sqlite/releases/download/v${releaseTag}/${assetName}`
 | |
|   }
 | |
| 
 | |
|   getAssetName(binaryName, releaseTag) {
 | |
|     if (!this.platformSuffix) {
 | |
|       throw new Error(`[NunicodeDownloader] Platform ${process.platform}-${process.arch} not supported`)
 | |
|     }
 | |
|     return `${binaryName}-${this.platformSuffix}.zip`
 | |
|   }
 | |
| 
 | |
|   getAssetFileName(binaryName) {
 | |
|     if (process.platform === 'win32') {
 | |
|       return `${binaryName}.dll`
 | |
|     } else if (process.platform === 'darwin') {
 | |
|       return `${binaryName}.dylib`
 | |
|     } else if (process.platform === 'linux') {
 | |
|       return `${binaryName}.so`
 | |
|     }
 | |
| 
 | |
|     throw new Error(`[NunicodeDownloader] Platform ${process.platform} not supported`)
 | |
|   }
 | |
| }
 | |
| 
 | |
| class Binary {
 | |
|   constructor(name, type, envVariable, validVersions, source, required = true) {
 | |
|     if (!name) throw new Error('Binary name is required')
 | |
|     this.name = name
 | |
|     if (!type) throw new Error('Binary type is required')
 | |
|     this.type = type
 | |
|     if (!envVariable) throw new Error('Binary environment variable name is required')
 | |
|     this.envVariable = envVariable
 | |
|     if (!validVersions || !validVersions.length) throw new Error(`No valid versions specified for ${type} ${name}. At least one version is required.`)
 | |
|     this.validVersions = validVersions
 | |
|     if (!source || !(source instanceof ZippedAssetDownloader)) throw new Error('Binary source is required, and must be an instance of ZippedAssetDownloader')
 | |
|     this.source = source
 | |
|     this.fileName = this.getFileName()
 | |
|     this.required = required
 | |
|     this.exec = exec
 | |
|   }
 | |
| 
 | |
|   async find(mainInstallDir, altInstallDir) {
 | |
|     // 1. check path specified in environment variable
 | |
|     const defaultPath = process.env[this.envVariable]
 | |
|     if (await this.isGood(defaultPath)) return defaultPath
 | |
|     // 2. find the first instance of the binary in the PATH environment variable
 | |
|     if (this.type === 'executable') {
 | |
|       const whichPath = which.sync(this.fileName, { nothrow: true })
 | |
|       if (await this.isGood(whichPath)) return whichPath
 | |
|     }
 | |
|     // 3. check main install path (binary root dir)
 | |
|     const mainInstallPath = path.join(mainInstallDir, this.fileName)
 | |
|     if (await this.isGood(mainInstallPath)) return mainInstallPath
 | |
|     // 4. check alt install path (/config)
 | |
|     const altInstallPath = path.join(altInstallDir, this.fileName)
 | |
|     if (await this.isGood(altInstallPath)) return altInstallPath
 | |
|     return null
 | |
|   }
 | |
| 
 | |
|   getFileName() {
 | |
|     const platform = process.platform
 | |
| 
 | |
|     if (this.type === 'executable') {
 | |
|       return this.name + (platform == 'win32' ? '.exe' : '')
 | |
|     } else if (this.type === 'library') {
 | |
|       return this.name + (platform == 'win32' ? '.dll' : platform == 'darwin' ? '.dylib' : '.so')
 | |
|     } else {
 | |
|       return this.name
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   async isLibraryVersionValid(libraryPath) {
 | |
|     try {
 | |
|       const versionFilePath = libraryPath + '.ver'
 | |
|       if (!(await fs.pathExists(versionFilePath))) return false
 | |
|       const version = (await fs.readFile(versionFilePath, 'utf8')).trim()
 | |
|       return this.validVersions.some((validVersion) => version.startsWith(validVersion))
 | |
|     } catch (err) {
 | |
|       Logger.error(`[Binary] Failed to check version of ${libraryPath}`, err)
 | |
|       return false
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   async isExecutableVersionValid(executablePath) {
 | |
|     try {
 | |
|       const { stdout } = await this.exec('"' + executablePath + '"' + ' -version')
 | |
|       const version = stdout.match(/version\s([\d\.]+)/)?.[1]
 | |
|       if (!version) return false
 | |
|       return this.validVersions.some((validVersion) => version.startsWith(validVersion))
 | |
|     } catch (err) {
 | |
|       Logger.error(`[Binary] Failed to check version of ${executablePath}`, err)
 | |
|       return false
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   async isGood(binaryPath) {
 | |
|     try {
 | |
|       if (!binaryPath || !(await fs.pathExists(binaryPath))) return false
 | |
|       if (this.type === 'library') return await this.isLibraryVersionValid(binaryPath)
 | |
|       else if (this.type === 'executable') return await this.isExecutableVersionValid(binaryPath)
 | |
|       else return true
 | |
|     } catch (err) {
 | |
|       Logger.error(`[Binary] Failed to check ${this.type} ${this.name} at ${binaryPath}`, err)
 | |
|       return false
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   async download(destination) {
 | |
|     const version = this.validVersions[0]
 | |
|     try {
 | |
|       await this.source.downloadBinary(this.name, version, destination)
 | |
|       // if it's a library, write the version string to a file
 | |
|       if (this.type === 'library') {
 | |
|         const libraryPath = path.join(destination, this.fileName)
 | |
|         await fs.writeFile(libraryPath + '.ver', version)
 | |
|       }
 | |
|     } catch (err) {
 | |
|       Logger.error(`[Binary] Failed to download ${this.type} ${this.name} version ${version} to ${destination}`, err)
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| const ffbinaries = new FFBinariesDownloader()
 | |
| const nunicode = new NunicodeDownloader()
 | |
| 
 | |
| class BinaryManager {
 | |
|   defaultRequiredBinaries = [
 | |
|     new Binary('ffmpeg', 'executable', 'FFMPEG_PATH', ['5.1'], ffbinaries), // ffmpeg executable
 | |
|     new Binary('ffprobe', 'executable', 'FFPROBE_PATH', ['5.1'], ffbinaries), // ffprobe executable
 | |
|     new Binary('libnusqlite3', 'library', 'NUSQLITE3_PATH', ['1.2'], nunicode, false) // nunicode sqlite3 extension
 | |
|   ]
 | |
| 
 | |
|   constructor(requiredBinaries = this.defaultRequiredBinaries) {
 | |
|     this.requiredBinaries = requiredBinaries
 | |
|     this.mainInstallDir = process.pkg ? path.dirname(process.execPath) : global.appRoot
 | |
|     this.altInstallDir = global.ConfigPath
 | |
|     this.initialized = false
 | |
|   }
 | |
| 
 | |
|   async init() {
 | |
|     // Optional skip binaries check
 | |
|     if (process.env.SKIP_BINARIES_CHECK === '1') {
 | |
|       for (const binary of this.requiredBinaries) {
 | |
|         if (!process.env[binary.envVariable] && binary.required) {
 | |
|           await Logger.fatal(`[BinaryManager] Environment variable ${binary.envVariable} must be set`)
 | |
|           process.exit(1)
 | |
|         }
 | |
|       }
 | |
|       Logger.info('[BinaryManager] Skipping check for binaries')
 | |
|       return
 | |
|     }
 | |
| 
 | |
|     if (this.initialized) return
 | |
| 
 | |
|     const missingBinaries = await this.findRequiredBinaries()
 | |
|     if (missingBinaries.length == 0) return
 | |
|     await this.removeOldBinaries(missingBinaries)
 | |
|     await this.install(missingBinaries)
 | |
|     const missingBinariesAfterInstall = await this.findRequiredBinaries()
 | |
|     const missingRequiredBinryNames = missingBinariesAfterInstall.filter((binary) => binary.required).map((binary) => binary.name)
 | |
|     if (missingRequiredBinryNames.length) {
 | |
|       Logger.error(`[BinaryManager] Failed to find or install required binaries: ${missingRequiredBinryNames.join(', ')}`)
 | |
|       process.exit(1)
 | |
|     }
 | |
|     this.initialized = true
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Remove binary
 | |
|    *
 | |
|    * @param {string} destination
 | |
|    * @param {Binary} binary
 | |
|    */
 | |
|   async removeBinary(destination, binary) {
 | |
|     try {
 | |
|       const binaryPath = path.join(destination, binary.fileName)
 | |
|       if (await fs.pathExists(binaryPath)) {
 | |
|         Logger.debug(`[BinaryManager] Removing binary: ${binaryPath}`)
 | |
|         await fs.remove(binaryPath)
 | |
|       }
 | |
|     } catch (err) {
 | |
|       Logger.error(`[BinaryManager] Error removing binary: ${binaryPath}`)
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Remove old binaries
 | |
|    *
 | |
|    * @param {Binary[]} binaries
 | |
|    */
 | |
|   async removeOldBinaries(binaries) {
 | |
|     for (const binary of binaries) {
 | |
|       await this.removeBinary(this.mainInstallDir, binary)
 | |
|       await this.removeBinary(this.altInstallDir, binary)
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Find required binaries and return array of binary names that are missing
 | |
|    *
 | |
|    * @returns {Promise<Binary[]>} Array of missing binaries
 | |
|    */
 | |
|   async findRequiredBinaries() {
 | |
|     const missingBinaries = []
 | |
|     for (const binary of this.requiredBinaries) {
 | |
|       const binaryPath = await binary.find(this.mainInstallDir, this.altInstallDir)
 | |
|       if (binaryPath) {
 | |
|         Logger.info(`[BinaryManager] Found valid ${binary.type} ${binary.name} at ${binaryPath}`)
 | |
|         if (process.env[binary.envVariable] !== binaryPath) {
 | |
|           Logger.info(`[BinaryManager] Updating process.env.${binary.envVariable}`)
 | |
|           process.env[binary.envVariable] = binaryPath
 | |
|         }
 | |
|       } else {
 | |
|         Logger.info(`[BinaryManager] ${binary.name} not found or not a valid version`)
 | |
|         missingBinaries.push(binary)
 | |
|       }
 | |
|     }
 | |
|     return missingBinaries
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Install missing binaries
 | |
|    *
 | |
|    * @param {Binary[]} binaries
 | |
|    */
 | |
|   async install(binaries) {
 | |
|     if (!binaries.length) return
 | |
|     Logger.info(`[BinaryManager] Installing binaries: ${binaries.map((binary) => binary.name).join(', ')}`)
 | |
|     let destination = (await fileUtils.isWritable(this.mainInstallDir)) ? this.mainInstallDir : this.altInstallDir
 | |
|     for (const binary of binaries) {
 | |
|       await binary.download(destination)
 | |
|     }
 | |
|     Logger.info(`[BinaryManager] Binaries installed to ${destination}`)
 | |
|   }
 | |
| }
 | |
| 
 | |
| module.exports = BinaryManager
 | |
| module.exports.Binary = Binary // for testing
 | |
| module.exports.ffbinaries = ffbinaries // for testing
 | |
| module.exports.nunicode = nunicode // for testing
 |