Add:mtime,ctime,birthtime to audiobook folder and files

This commit is contained in:
advplyr 2022-02-27 18:07:36 -06:00
parent c81b12f459
commit 428a515c6a
7 changed files with 115 additions and 31 deletions

View File

@ -11,6 +11,9 @@ class AudioFile {
this.ext = null this.ext = null
this.path = null this.path = null
this.fullPath = null this.fullPath = null
this.mtimeMs = null
this.ctimeMs = null
this.birthtimeMs = null
this.addedAt = null this.addedAt = null
this.trackNumFromMeta = null this.trackNumFromMeta = null
@ -51,6 +54,9 @@ class AudioFile {
ext: this.ext, ext: this.ext,
path: this.path, path: this.path,
fullPath: this.fullPath, fullPath: this.fullPath,
mtimeMs: this.mtimeMs,
ctimeMs: this.ctimeMs,
birthtimeMs: this.birthtimeMs,
addedAt: this.addedAt, addedAt: this.addedAt,
trackNumFromMeta: this.trackNumFromMeta, trackNumFromMeta: this.trackNumFromMeta,
discNumFromMeta: this.discNumFromMeta, discNumFromMeta: this.discNumFromMeta,
@ -82,6 +88,9 @@ class AudioFile {
this.ext = data.ext this.ext = data.ext
this.path = data.path this.path = data.path
this.fullPath = data.fullPath this.fullPath = data.fullPath
this.mtimeMs = data.mtimeMs || 0
this.ctimeMs = data.ctimeMs || 0
this.birthtimeMs = data.birthtimeMs || 0
this.addedAt = data.addedAt this.addedAt = data.addedAt
this.manuallyVerified = !!data.manuallyVerified this.manuallyVerified = !!data.manuallyVerified
this.invalid = !!data.invalid this.invalid = !!data.invalid
@ -124,6 +133,9 @@ class AudioFile {
this.ext = fileData.ext this.ext = fileData.ext
this.path = fileData.path this.path = fileData.path
this.fullPath = fileData.fullPath this.fullPath = fileData.fullPath
this.mtimeMs = fileData.mtimeMs || 0
this.ctimeMs = fileData.ctimeMs || 0
this.birthtimeMs = fileData.birthtimeMs || 0
this.addedAt = Date.now() this.addedAt = Date.now()
this.trackNumFromMeta = fileData.trackNumFromMeta this.trackNumFromMeta = fileData.trackNumFromMeta

View File

@ -1,7 +1,7 @@
const Path = require('path') const Path = require('path')
const fs = require('fs-extra') const fs = require('fs-extra')
const { bytesPretty, readTextFile } = require('../utils/fileUtils') const { bytesPretty, readTextFile, getIno } = require('../utils/fileUtils')
const { comparePaths, getIno, getId, elapsedPretty } = require('../utils/index') const { comparePaths, getId, elapsedPretty } = require('../utils/index')
const { parseOpfMetadataXML } = require('../utils/parseOpfMetadata') const { parseOpfMetadataXML } = require('../utils/parseOpfMetadata')
const { extractCoverArt } = require('../utils/ffmpegHelpers') const { extractCoverArt } = require('../utils/ffmpegHelpers')
const nfoGenerator = require('../utils/nfoGenerator') const nfoGenerator = require('../utils/nfoGenerator')
@ -22,6 +22,9 @@ class Audiobook {
this.path = null this.path = null
this.fullPath = null this.fullPath = null
this.mtimeMs = null
this.ctimeMs = null
this.birthtimeMs = null
this.addedAt = null this.addedAt = null
this.lastUpdate = null this.lastUpdate = null
this.lastScan = null this.lastScan = null
@ -57,6 +60,9 @@ class Audiobook {
this.folderId = audiobook.folderId || 'audiobooks' this.folderId = audiobook.folderId || 'audiobooks'
this.path = audiobook.path this.path = audiobook.path
this.fullPath = audiobook.fullPath this.fullPath = audiobook.fullPath
this.mtimeMs = audiobook.mtimeMs || 0
this.ctimeMs = audiobook.ctimeMs || 0
this.birthtimeMs = audiobook.birthtimeMs || 0
this.addedAt = audiobook.addedAt this.addedAt = audiobook.addedAt
this.lastUpdate = audiobook.lastUpdate || this.addedAt this.lastUpdate = audiobook.lastUpdate || this.addedAt
this.lastScan = audiobook.lastScan || null this.lastScan = audiobook.lastScan || null
@ -179,6 +185,9 @@ class Audiobook {
folderId: this.folderId, folderId: this.folderId,
path: this.path, path: this.path,
fullPath: this.fullPath, fullPath: this.fullPath,
mtimeMs: this.mtimeMs,
ctimeMs: this.ctimeMs,
birthtimeMs: this.birthtimeMs,
addedAt: this.addedAt, addedAt: this.addedAt,
lastUpdate: this.lastUpdate, lastUpdate: this.lastUpdate,
lastScan: this.lastScan, lastScan: this.lastScan,
@ -205,6 +214,9 @@ class Audiobook {
tags: this.tags, tags: this.tags,
path: this.path, path: this.path,
fullPath: this.fullPath, fullPath: this.fullPath,
mtimeMs: this.mtimeMs,
ctimeMs: this.ctimeMs,
birthtimeMs: this.birthtimeMs,
addedAt: this.addedAt, addedAt: this.addedAt,
lastUpdate: this.lastUpdate, lastUpdate: this.lastUpdate,
duration: this.duration, duration: this.duration,
@ -228,6 +240,9 @@ class Audiobook {
folderId: this.folderId, folderId: this.folderId,
path: this.path, path: this.path,
fullPath: this.fullPath, fullPath: this.fullPath,
mtimeMs: this.mtimeMs,
ctimeMs: this.ctimeMs,
birthtimeMs: this.birthtimeMs,
addedAt: this.addedAt, addedAt: this.addedAt,
lastUpdate: this.lastUpdate, lastUpdate: this.lastUpdate,
duration: this.duration, duration: this.duration,
@ -335,6 +350,9 @@ class Audiobook {
this.path = data.path this.path = data.path
this.fullPath = data.fullPath this.fullPath = data.fullPath
this.mtimeMs = data.mtimeMs || 0
this.ctimeMs = data.ctimeMs || 0
this.birthtimeMs = data.birthtimeMs || 0
this.addedAt = Date.now() this.addedAt = Date.now()
this.lastUpdate = this.addedAt this.lastUpdate = this.addedAt
@ -903,12 +921,6 @@ class Audiobook {
} }
} }
if (existingFile.filename !== fileFound.filename) {
existingFile.filename = fileFound.filename
existingFile.ext = fileFound.ext
hasUpdated = true
}
if (existingFile.path !== fileFound.path) { if (existingFile.path !== fileFound.path) {
existingFile.path = fileFound.path existingFile.path = fileFound.path
existingFile.fullPath = fileFound.fullPath existingFile.fullPath = fileFound.fullPath
@ -918,6 +930,14 @@ class Audiobook {
hasUpdated = true hasUpdated = true
} }
var keysToCheck = ['filename', 'ext', 'mtimeMs', 'ctimeMs', 'birthtimeMs', 'size']
keysToCheck.forEach((key) => {
if (existingFile[key] !== fileFound[key]) {
existingFile[key] = fileFound[key]
hasUpdated = true
}
})
if (!isAudioFile && existingFile.filetype !== fileFound.filetype) { if (!isAudioFile && existingFile.filetype !== fileFound.filetype) {
existingFile.filetype = fileFound.filetype existingFile.filetype = fileFound.filetype
hasUpdated = true hasUpdated = true
@ -957,6 +977,14 @@ class Audiobook {
hasUpdated = true hasUpdated = true
} }
var keysToCheck = ['mtimeMs', 'ctimeMs', 'birthtimeMs']
keysToCheck.forEach((key) => {
if (dataFound[key] != this[key]) {
this[key] = dataFound[key] || 0
hasUpdated = true
}
})
var newAudioFileData = [] var newAudioFileData = []
var newOtherFileData = [] var newOtherFileData = []
var existingAudioFileData = [] var existingAudioFileData = []

View File

@ -6,6 +6,11 @@ class AudiobookFile {
this.ext = null this.ext = null
this.path = null this.path = null
this.fullPath = null this.fullPath = null
this.size = null
this.mtimeMs = null
this.ctimeMs = null
this.birthtimeMs = null
this.addedAt = null this.addedAt = null
if (data) { if (data) {
@ -25,6 +30,10 @@ class AudiobookFile {
ext: this.ext, ext: this.ext,
path: this.path, path: this.path,
fullPath: this.fullPath, fullPath: this.fullPath,
size: this.size,
mtimeMs: this.mtimeMs,
ctimeMs: this.ctimeMs,
birthtimeMs: this.birthtimeMs,
addedAt: this.addedAt addedAt: this.addedAt
} }
} }
@ -36,6 +45,10 @@ class AudiobookFile {
this.ext = data.ext this.ext = data.ext
this.path = data.path this.path = data.path
this.fullPath = data.fullPath this.fullPath = data.fullPath
this.size = data.size || 0
this.mtimeMs = data.mtimeMs || 0
this.ctimeMs = data.ctimeMs || 0
this.birthtimeMs = data.birthtimeMs || 0
this.addedAt = data.addedAt this.addedAt = data.addedAt
} }
@ -46,6 +59,10 @@ class AudiobookFile {
this.ext = data.ext this.ext = data.ext
this.path = data.path this.path = data.path
this.fullPath = data.fullPath this.fullPath = data.fullPath
this.size = data.size || 0
this.mtimeMs = data.mtimeMs || 0
this.ctimeMs = data.ctimeMs || 0
this.birthtimeMs = data.birthtimeMs || 0
this.addedAt = Date.now() this.addedAt = Date.now()
} }
} }

View File

@ -205,6 +205,7 @@ class Scanner {
// Check for existing & removed audiobooks // Check for existing & removed audiobooks
for (let i = 0; i < audiobooksInLibrary.length; i++) { for (let i = 0; i < audiobooksInLibrary.length; i++) {
var audiobook = audiobooksInLibrary[i] var audiobook = audiobooksInLibrary[i]
// Find audiobook folder with matching inode or matching path
var dataFound = audiobookDataFound.find(abd => abd.ino === audiobook.ino || comparePaths(abd.path, audiobook.path)) var dataFound = audiobookDataFound.find(abd => abd.ino === audiobook.ino || comparePaths(abd.path, audiobook.path))
if (!dataFound) { if (!dataFound) {
libraryScan.addLog(LogLevel.WARN, `Audiobook "${audiobook.title}" is missing`) libraryScan.addLog(LogLevel.WARN, `Audiobook "${audiobook.title}" is missing`)

View File

@ -20,6 +20,23 @@ async function getFileStat(path) {
} }
module.exports.getFileStat = getFileStat module.exports.getFileStat = getFileStat
async function getFileTimestampsWithIno(path) {
try {
var stat = await fs.stat(path, { bigint: true })
return {
size: Number(stat.size),
mtimeMs: Number(stat.mtimeMs),
ctimeMs: Number(stat.ctimeMs),
birthtimeMs: Number(stat.birthtimeMs),
ino: String(stat.ino)
}
} catch (err) {
console.error('Failed to getFileTimestampsWithIno', err)
return false
}
}
module.exports.getFileTimestampsWithIno = getFileTimestampsWithIno
async function getFileSize(path) { async function getFileSize(path) {
var stat = await getFileStat(path) var stat = await getFileStat(path)
if (!stat) return 0 if (!stat) return 0
@ -27,6 +44,15 @@ async function getFileSize(path) {
} }
module.exports.getFileSize = getFileSize module.exports.getFileSize = getFileSize
function getIno(path) {
return fs.stat(path, { bigint: true }).then((data => String(data.ino))).catch((err) => {
Logger.error('[Utils] Failed to get ino for path', path, err)
return null
})
}
module.exports.getIno = getIno
async function readTextFile(path) { async function readTextFile(path) {
try { try {
var data = await fs.readFile(path) var data = await fs.readFile(path)

View File

@ -38,13 +38,6 @@ module.exports.comparePaths = (path1, path2) => {
return path1 === path2 || Path.normalize(path1) === Path.normalize(path2) return path1 === path2 || Path.normalize(path1) === Path.normalize(path2)
} }
module.exports.getIno = (path) => {
return fs.promises.stat(path, { bigint: true }).then((data => String(data.ino))).catch((err) => {
Logger.error('[Utils] Failed to get ino for path', path, err)
return null
})
}
module.exports.isNullOrNaN = (num) => { module.exports.isNullOrNaN = (num) => {
return num === null || isNaN(num) return num === null || isNaN(num)
} }

View File

@ -1,8 +1,7 @@
const Path = require('path') const Path = require('path')
const fs = require('fs-extra') const fs = require('fs-extra')
const Logger = require('../Logger') const Logger = require('../Logger')
const { getIno } = require('./index') const { recurseFiles, getFileTimestampsWithIno } = require('./fileUtils')
const { recurseFiles } = require('./fileUtils')
const globals = require('./globals') const globals = require('./globals')
function isBookFile(path) { function isBookFile(path) {
@ -114,16 +113,20 @@ function groupFileItemsIntoBooks(fileItems) {
} }
function cleanFileObjects(basepath, abrelpath, files) { function cleanFileObjects(basepath, abrelpath, files) {
return files.map((file) => { return Promise.all(files.map(async (file) => {
var fullPath = Path.posix.join(basepath, file)
var fileTsData = await getFileTimestampsWithIno(fullPath)
var ext = Path.extname(file) var ext = Path.extname(file)
return { return {
filetype: getFileType(ext), filetype: getFileType(ext),
filename: Path.basename(file), filename: Path.basename(file),
path: Path.posix.join(abrelpath, file), // /AUDIOBOOK/PATH/filename.mp3 path: Path.posix.join(abrelpath, file), // /AUDIOBOOK/PATH/filename.mp3
fullPath: Path.posix.join(basepath, file), // /audiobooks/AUDIOBOOK/PATH/filename.mp3 fullPath, // /audiobooks/AUDIOBOOK/PATH/filename.mp3
ext: ext ext: ext,
...fileTsData
} }
}) }))
} }
function getFileType(ext) { function getFileType(ext) {
@ -162,15 +165,15 @@ async function scanRootDir(folder, serverSettings = {}) {
for (const audiobookPath in audiobookGrouping) { for (const audiobookPath in audiobookGrouping) {
var audiobookData = getAudiobookDataFromDir(folderPath, audiobookPath, parseSubtitle) var audiobookData = getAudiobookDataFromDir(folderPath, audiobookPath, parseSubtitle)
var fileObjs = cleanFileObjects(audiobookData.fullPath, audiobookPath, audiobookGrouping[audiobookPath]) var fileObjs = await cleanFileObjects(audiobookData.fullPath, audiobookPath, audiobookGrouping[audiobookPath])
for (let i = 0; i < fileObjs.length; i++) { var audiobookFolderStats = await getFileTimestampsWithIno(audiobookData.fullPath)
fileObjs[i].ino = await getIno(fileObjs[i].fullPath)
}
var audiobookIno = await getIno(audiobookData.fullPath)
audiobooks.push({ audiobooks.push({
folderId: folder.id, folderId: folder.id,
libraryId: folder.libraryId, libraryId: folder.libraryId,
ino: audiobookIno, ino: audiobookFolderStats.ino,
mtimeMs: audiobookFolderStats.mtimeMs || 0,
ctimeMs: audiobookFolderStats.ctimeMs || 0,
birthtimeMs: audiobookFolderStats.birthtimeMs || 0,
...audiobookData, ...audiobookData,
audioFiles: fileObjs.filter(f => f.filetype === 'audio'), audioFiles: fileObjs.filter(f => f.filetype === 'audio'),
otherFiles: fileObjs.filter(f => f.filetype !== 'audio') otherFiles: fileObjs.filter(f => f.filetype !== 'audio')
@ -282,8 +285,12 @@ async function getAudiobookFileData(folder, audiobookPath, serverSettings = {})
var audiobookDir = audiobookPath.replace(folderFullPath, '').slice(1) var audiobookDir = audiobookPath.replace(folderFullPath, '').slice(1)
var audiobookData = getAudiobookDataFromDir(folderFullPath, audiobookDir, parseSubtitle) var audiobookData = getAudiobookDataFromDir(folderFullPath, audiobookDir, parseSubtitle)
var audiobookFolderStats = await getFileTimestampsWithIno(audiobookData.fullPath)
var audiobook = { var audiobook = {
ino: await getIno(audiobookData.fullPath), ino: audiobookFolderStats.ino,
mtimeMs: audiobookFolderStats.mtimeMs || 0,
ctimeMs: audiobookFolderStats.ctimeMs || 0,
birthtimeMs: audiobookFolderStats.birthtimeMs || 0,
folderId: folder.id, folderId: folder.id,
libraryId: folder.libraryId, libraryId: folder.libraryId,
...audiobookData, ...audiobookData,
@ -294,14 +301,14 @@ async function getAudiobookFileData(folder, audiobookPath, serverSettings = {})
for (let i = 0; i < fileItems.length; i++) { for (let i = 0; i < fileItems.length; i++) {
var fileItem = fileItems[i] var fileItem = fileItems[i]
var ino = await getIno(fileItem.fullpath) var fileStatData = await getFileTimestampsWithIno(fileItem.fullpath)
var fileObj = { var fileObj = {
ino,
filetype: getFileType(fileItem.extension), filetype: getFileType(fileItem.extension),
filename: fileItem.name, filename: fileItem.name,
path: fileItem.path, path: fileItem.path,
fullPath: fileItem.fullpath, fullPath: fileItem.fullpath,
ext: fileItem.extension ext: fileItem.extension,
...fileStatData
} }
if (fileObj.filetype === 'audio') { if (fileObj.filetype === 'audio') {
audiobook.audioFiles.push(fileObj) audiobook.audioFiles.push(fileObj)