mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-03 20:49:04 +01:00
477 lines
17 KiB
JavaScript
477 lines
17 KiB
JavaScript
const Path = require('path')
|
|
const fs = require('fs-extra')
|
|
const njodb = require("njodb")
|
|
|
|
const { SupportedEbookTypes } = require('./globals')
|
|
const { PlayMethod } = require('./constants')
|
|
const { getId } = require('./index')
|
|
const Logger = require('../Logger')
|
|
|
|
const LegacyAudiobook = require('../objects/legacy/Audiobook')
|
|
const UserAudiobookData = require('../objects/legacy/UserAudiobookData')
|
|
|
|
const Library = require('../objects/Library')
|
|
const LibraryItem = require('../objects/LibraryItem')
|
|
const Book = require('../objects/mediaTypes/Book')
|
|
|
|
const BookMetadata = require('../objects/metadata/BookMetadata')
|
|
const FileMetadata = require('../objects/metadata/FileMetadata')
|
|
|
|
const AudioFile = require('../objects/files/AudioFile')
|
|
const EBookFile = require('../objects/files/EBookFile')
|
|
const LibraryFile = require('../objects/files/LibraryFile')
|
|
const AudioMetaTags = require('../objects/metadata/AudioMetaTags')
|
|
|
|
const Author = require('../objects/entities/Author')
|
|
const Series = require('../objects/entities/Series')
|
|
const Audiobook = require('../objects/entities/Audiobook')
|
|
const EBook = require('../objects/entities/EBook')
|
|
|
|
const LibraryItemProgress = require('../objects/user/LibraryItemProgress')
|
|
const PlaybackSession = require('../objects/PlaybackSession')
|
|
|
|
const { isObject } = require('.')
|
|
const User = require('../objects/user/User')
|
|
|
|
var authorsToAdd = []
|
|
var existingDbAuthors = []
|
|
var seriesToAdd = []
|
|
var existingDbSeries = []
|
|
|
|
// Load old audiobooks
|
|
async function loadAudiobooks() {
|
|
var audiobookPath = Path.join(global.ConfigPath, 'audiobooks')
|
|
|
|
var pathExists = await fs.pathExists(audiobookPath)
|
|
if (!pathExists) {
|
|
return []
|
|
}
|
|
|
|
var audiobooksDb = new njodb.Database(audiobookPath)
|
|
return audiobooksDb.select(() => true).then((results) => {
|
|
return results.data.map(a => new LegacyAudiobook(a))
|
|
})
|
|
}
|
|
|
|
function makeAuthorsFromOldAb(authorsList) {
|
|
return authorsList.filter(a => !!a).map(authorName => {
|
|
var existingAuthor = authorsToAdd.find(a => a.name.toLowerCase() === authorName.toLowerCase())
|
|
if (existingAuthor) {
|
|
return existingAuthor.toJSONMinimal()
|
|
}
|
|
var existingDbAuthor = existingDbAuthors.find(a => a.name.toLowerCase() === authorName.toLowerCase())
|
|
if (existingDbAuthor) {
|
|
return existingDbAuthor.toJSONMinimal()
|
|
}
|
|
|
|
var newAuthor = new Author()
|
|
newAuthor.setData({ name: authorName })
|
|
authorsToAdd.push(newAuthor)
|
|
Logger.debug(`>>> Created new author named "${authorName}"`)
|
|
return newAuthor.toJSONMinimal()
|
|
})
|
|
}
|
|
|
|
function makeSeriesFromOldAb({ series, volumeNumber }) {
|
|
var existingSeries = seriesToAdd.find(s => s.name.toLowerCase() === series.toLowerCase())
|
|
if (existingSeries) {
|
|
return [existingSeries.toJSONMinimal(volumeNumber)]
|
|
}
|
|
var existingDbSeriesItem = existingDbSeries.find(s => s.name.toLowerCase() === series.toLowerCase())
|
|
if (existingDbSeriesItem) {
|
|
return [existingDbSeriesItem.toJSONMinimal(volumeNumber)]
|
|
}
|
|
var newSeries = new Series()
|
|
newSeries.setData({ name: series })
|
|
seriesToAdd.push(newSeries)
|
|
Logger.info(`>>> Created new series named "${series}"`)
|
|
return [newSeries.toJSONMinimal(volumeNumber)]
|
|
}
|
|
|
|
function getRelativePath(srcPath, basePath) {
|
|
srcPath = srcPath.replace(/\\/g, '/')
|
|
basePath = basePath.replace(/\\/g, '/')
|
|
return srcPath.replace(basePath, '')
|
|
}
|
|
|
|
function makeFilesFromOldAb(audiobook) {
|
|
var libraryFiles = []
|
|
var ebookFiles = []
|
|
|
|
var audioFiles = audiobook._audioFiles.map((af) => {
|
|
var fileMetadata = new FileMetadata(af)
|
|
fileMetadata.path = af.fullPath
|
|
fileMetadata.relPath = getRelativePath(af.fullPath, audiobook.fullPath)
|
|
|
|
var newLibraryFile = new LibraryFile()
|
|
newLibraryFile.ino = af.ino
|
|
newLibraryFile.metadata = fileMetadata.clone()
|
|
newLibraryFile.addedAt = af.addedAt
|
|
newLibraryFile.updatedAt = Date.now()
|
|
libraryFiles.push(newLibraryFile)
|
|
|
|
var audioMetaTags = new AudioMetaTags(af.metadata || {}) // Old metaTags was named metadata
|
|
delete af.metadata
|
|
|
|
var newAudioFile = new AudioFile(af)
|
|
newAudioFile.metadata = fileMetadata
|
|
newAudioFile.metaTags = audioMetaTags
|
|
newAudioFile.updatedAt = Date.now()
|
|
return newAudioFile
|
|
})
|
|
|
|
audiobook._otherFiles.forEach((file) => {
|
|
var fileMetadata = new FileMetadata(file)
|
|
fileMetadata.path = file.fullPath
|
|
fileMetadata.relPath = getRelativePath(file.fullPath, audiobook.fullPath)
|
|
|
|
var newLibraryFile = new LibraryFile()
|
|
newLibraryFile.ino = file.ino
|
|
newLibraryFile.metadata = fileMetadata.clone()
|
|
newLibraryFile.addedAt = file.addedAt
|
|
newLibraryFile.updatedAt = Date.now()
|
|
libraryFiles.push(newLibraryFile)
|
|
|
|
var formatExt = (file.ext || '').slice(1)
|
|
if (SupportedEbookTypes.includes(formatExt)) {
|
|
var newEBookFile = new EBookFile()
|
|
newEBookFile.ino = file.ino
|
|
newEBookFile.metadata = fileMetadata
|
|
newEBookFile.ebookFormat = formatExt
|
|
newEBookFile.addedAt = file.addedAt
|
|
newEBookFile.updatedAt = Date.now()
|
|
ebookFiles.push(newEBookFile)
|
|
}
|
|
})
|
|
|
|
return {
|
|
libraryFiles,
|
|
ebookFiles,
|
|
audioFiles
|
|
}
|
|
}
|
|
|
|
// Metadata path was changed to /metadata/items make sure cover is using new path
|
|
function cleanOldCoverPath(coverPath) {
|
|
if (!coverPath) return null
|
|
var oldMetadataPath = Path.posix.join(global.MetadataPath, 'books')
|
|
if (coverPath.startsWith(oldMetadataPath)) {
|
|
const newMetadataPath = Path.posix.join(global.MetadataPath, 'items')
|
|
return coverPath.replace(oldMetadataPath, newMetadataPath)
|
|
}
|
|
return coverPath
|
|
}
|
|
|
|
function makeLibraryItemFromOldAb(audiobook) {
|
|
var libraryItem = new LibraryItem()
|
|
libraryItem.id = getId('li')
|
|
libraryItem.ino = audiobook.ino
|
|
libraryItem.libraryId = audiobook.libraryId
|
|
libraryItem.folderId = audiobook.folderId
|
|
libraryItem.path = audiobook.fullPath
|
|
libraryItem.relPath = audiobook.path
|
|
libraryItem.mtimeMs = audiobook.mtimeMs || 0
|
|
libraryItem.ctimeMs = audiobook.ctimeMs || 0
|
|
libraryItem.birthtimeMs = audiobook.birthtimeMs || 0
|
|
libraryItem.addedAt = audiobook.addedAt
|
|
libraryItem.updatedAt = audiobook.lastUpdate
|
|
libraryItem.lastScan = audiobook.lastScan
|
|
libraryItem.scanVersion = audiobook.scanVersion
|
|
libraryItem.isMissing = audiobook.isMissing
|
|
libraryItem.mediaType = 'book'
|
|
|
|
var bookEntity = new Book()
|
|
var bookMetadata = new BookMetadata(audiobook.book)
|
|
bookMetadata.publishedYear = audiobook.book.publishYear || null
|
|
if (audiobook.book.narrator) {
|
|
bookMetadata.narrators = audiobook.book._narratorsList
|
|
}
|
|
// Returns array of json minimal authors
|
|
bookMetadata.authors = makeAuthorsFromOldAb(audiobook.book._authorsList)
|
|
|
|
// Returns array of json minimal series
|
|
if (audiobook.book.series) {
|
|
bookMetadata.series = makeSeriesFromOldAb(audiobook.book)
|
|
}
|
|
|
|
bookEntity.metadata = bookMetadata
|
|
bookEntity.coverPath = cleanOldCoverPath(audiobook.book.coverFullPath)
|
|
bookEntity.tags = [...audiobook.tags]
|
|
|
|
var payload = makeFilesFromOldAb(audiobook)
|
|
if (payload.audioFiles.length) {
|
|
var newAudiobook = new Audiobook()
|
|
newAudiobook.id = audiobook.id
|
|
newAudiobook.index = 1
|
|
newAudiobook.name = 'default'
|
|
newAudiobook.audioFiles = payload.audioFiles
|
|
if (audiobook.chapters && audiobook.chapters.length) {
|
|
newAudiobook.chapters = audiobook.chapters.map(c => ({ ...c }))
|
|
}
|
|
newAudiobook.missingParts = audiobook.missingParts || []
|
|
newAudiobook.addedAt = audiobook.addedAt
|
|
newAudiobook.updatedAt = audiobook.lastUpdate
|
|
|
|
bookEntity.audiobooks.push(newAudiobook)
|
|
}
|
|
|
|
var ebookIndex = 1
|
|
payload.ebookFiles.forEach(ebookFile => {
|
|
var newEBook = new EBook()
|
|
newEBook.id = getId('eb')
|
|
newEBook.index = ebookIndex++
|
|
newEBook.name = ebookFile.metadata.filenameNoExt
|
|
newEBook.ebookFile = ebookFile
|
|
newEBook.addedAt = audiobook.addedAt
|
|
newEBook.updatedAt = audiobook.lastUpdate
|
|
|
|
bookEntity.ebooks.push(newEBook)
|
|
})
|
|
|
|
libraryItem.media = bookEntity
|
|
libraryItem.libraryFiles = payload.libraryFiles
|
|
return libraryItem
|
|
}
|
|
|
|
async function migrateLibraryItems(db) {
|
|
Logger.info(`==== Starting Library Item migration ====`)
|
|
|
|
var audiobooks = await loadAudiobooks()
|
|
if (!audiobooks.length) {
|
|
Logger.info(`>>> No audiobooks in db, no migration necessary`)
|
|
return
|
|
}
|
|
|
|
Logger.info(`>>> Loaded old audiobook data with ${audiobooks.length} records`)
|
|
|
|
if (db.libraryItems.length) {
|
|
Logger.info(`>>> Some library items already loaded ${db.libraryItems.length} items | ${db.series.length} series | ${db.authors.length} authors`)
|
|
return
|
|
}
|
|
|
|
if (db.authors && db.authors.length) {
|
|
existingDbAuthors = db.authors
|
|
}
|
|
if (db.series && db.series.length) {
|
|
existingDbSeries = db.series
|
|
}
|
|
|
|
var libraryItems = audiobooks.map((ab) => makeLibraryItemFromOldAb(ab))
|
|
|
|
// User library item progress was using the auidobook ID when migrated
|
|
// now that library items are created the LibraryItemProgress objects
|
|
// need the library item id to be set
|
|
for (const user of db.users) {
|
|
if (user.libraryItemProgress.length) {
|
|
user.libraryItemProgress = user.libraryItemProgress.map(lip => {
|
|
var audiobookId = lip.id
|
|
var libraryItemWithAudiobook = libraryItems.find(li => li.media.getAudiobookById && !!li.media.getAudiobookById(audiobookId))
|
|
if (!libraryItemWithAudiobook) {
|
|
Logger.error('[dbMigration] Failed to find library item with audiobook id', audiobookId)
|
|
return null
|
|
}
|
|
lip.id = libraryItemWithAudiobook.id
|
|
lip.libraryItemId = libraryItemWithAudiobook.id
|
|
return lip
|
|
}).filter(lip => !!lip)
|
|
}
|
|
if (user.bookmarks.length) {
|
|
user.bookmarks = user.bookmarks.map((bookmark) => {
|
|
var audiobookId = bookmark.libraryItemId
|
|
var libraryItemWithAudiobook = libraryItems.find(li => li.media.getAudiobookById && !!li.media.getAudiobookById(audiobookId))
|
|
if (!libraryItemWithAudiobook) {
|
|
Logger.error('[dbMigration] Failed to find library item with audiobook id', audiobookId)
|
|
return null
|
|
}
|
|
bookmark.libraryItemId = libraryItemWithAudiobook.id
|
|
return bookmark
|
|
}).filter(bm => !!bm)
|
|
}
|
|
if (user.libraryItemProgress.length || user.bookmarks.length) {
|
|
await db.updateEntity('user', user)
|
|
}
|
|
}
|
|
|
|
// Update session LibraryItemId's
|
|
var sessions = await db.sessionsDb.select(() => true).then((results) => results.data)
|
|
if (sessions.length) {
|
|
sessions = sessions.map(se => {
|
|
var libraryItemWithAudiobook = libraryItems.find(li => li.media.getAudiobookById && !!li.media.getAudiobookById(se.mediaEntityId))
|
|
if (!libraryItemWithAudiobook) {
|
|
Logger.error('[dbMigration] Failed to find library item with audiobook id', se.mediaEntityId)
|
|
return null
|
|
}
|
|
se.libraryItemId = libraryItemWithAudiobook.id
|
|
return se
|
|
}).filter(se => !!se)
|
|
await db.updateEntities('session', sessions)
|
|
}
|
|
|
|
Logger.info(`>>> ${libraryItems.length} Library Items made`)
|
|
await db.insertEntities('libraryItem', libraryItems)
|
|
if (authorsToAdd.length) {
|
|
Logger.info(`>>> ${authorsToAdd.length} Authors made`)
|
|
await db.insertEntities('author', authorsToAdd)
|
|
}
|
|
if (seriesToAdd.length) {
|
|
Logger.info(`>>> ${seriesToAdd.length} Series made`)
|
|
await db.insertEntities('series', seriesToAdd)
|
|
}
|
|
existingDbSeries = []
|
|
existingDbAuthors = []
|
|
authorsToAdd = []
|
|
seriesToAdd = []
|
|
Logger.info(`==== Library Item migration complete ====`)
|
|
}
|
|
|
|
function cleanUserObject(db, userObj) {
|
|
var cleanedUserPayload = {
|
|
...userObj,
|
|
libraryItemProgress: [],
|
|
bookmarks: []
|
|
}
|
|
|
|
// UserAudiobookData is now LibraryItemProgress and AudioBookmarks separated
|
|
if (userObj.audiobooks) {
|
|
for (const audiobookId in userObj.audiobooks) {
|
|
if (isObject(userObj.audiobooks[audiobookId])) {
|
|
// Bookmarks now live on User.js object instead of inside UserAudiobookData
|
|
if (userObj.audiobooks[audiobookId].bookmarks) {
|
|
const cleanedBookmarks = userObj.audiobooks[audiobookId].bookmarks.map((bm) => {
|
|
bm.libraryItemId = audiobookId // Temp placeholder replace with libraryItemId when created
|
|
return bm
|
|
})
|
|
cleanedUserPayload.bookmarks = cleanedUserPayload.bookmarks.concat(cleanedBookmarks)
|
|
}
|
|
|
|
var userAudiobookData = new UserAudiobookData(userObj.audiobooks[audiobookId]) // Legacy object
|
|
var liProgress = new LibraryItemProgress() // New Progress Object
|
|
liProgress.id = userAudiobookData.audiobookId // This ID will be updated when library item is created
|
|
liProgress.libraryItemId = userAudiobookData.audiobookId
|
|
liProgress.mediaEntityId = userAudiobookData.audiobookId
|
|
liProgress.duration = userAudiobookData.totalDuration
|
|
liProgress.isFinished = !!userAudiobookData.isRead
|
|
Object.keys(liProgress.toJSON()).forEach((key) => {
|
|
if (userAudiobookData[key] !== undefined) {
|
|
liProgress[key] = userAudiobookData[key]
|
|
}
|
|
})
|
|
cleanedUserPayload.libraryItemProgress.push(liProgress.toJSON())
|
|
}
|
|
}
|
|
}
|
|
|
|
const user = new User(cleanedUserPayload)
|
|
return db.usersDb.update((record) => record.id === user.id, () => user).then((results) => {
|
|
Logger.debug(`[dbMigration] Updated User: ${results.updated} | Selected: ${results.selected}`)
|
|
return true
|
|
}).catch((error) => {
|
|
Logger.error(`[dbMigration] Update User Failed: ${error}`)
|
|
return false
|
|
})
|
|
}
|
|
|
|
function cleanSessionObj(db, userListeningSession) {
|
|
var newPlaybackSession = new PlaybackSession(userListeningSession)
|
|
newPlaybackSession.id = getId('play')
|
|
newPlaybackSession.mediaType = 'book'
|
|
newPlaybackSession.updatedAt = userListeningSession.lastUpdate
|
|
newPlaybackSession.libraryItemId = userListeningSession.audiobookId // Temp
|
|
newPlaybackSession.mediaEntityId = userListeningSession.audiobookId
|
|
newPlaybackSession.playMethod = PlayMethod.TRANSCODE
|
|
|
|
// We only have title to transfer over nicely
|
|
var bookMetadata = new BookMetadata()
|
|
bookMetadata.title = userListeningSession.audiobookTitle || ''
|
|
newPlaybackSession.mediaMetadata = bookMetadata
|
|
|
|
return db.sessionsDb.update((record) => record.id === userListeningSession.id, () => newPlaybackSession).then((results) => true).catch((error) => {
|
|
Logger.error(`[dbMigration] Update Session Failed: ${error}`)
|
|
return false
|
|
})
|
|
}
|
|
|
|
async function migrateUserData(db) {
|
|
Logger.info(`==== Starting User migration ====`)
|
|
|
|
// Libraries with previous mediaType of "podcast" moved to "book"
|
|
// because migrating those items to podcast objects will be a nightmare
|
|
// users will need to create a new library for podcasts
|
|
var availableIcons = ['database', 'audiobook', 'book', 'comic', 'podcast']
|
|
const libraries = await db.librariesDb.select((result) => (result.mediaType != 'book' || !availableIcons.includes(result.icon)))
|
|
.then((results) => results.data.map(lib => new Library(lib)))
|
|
if (!libraries.length) {
|
|
Logger.info('[dbMigration] No libraries found needing migration')
|
|
} else {
|
|
for (const library of libraries) {
|
|
Logger.info(`>> Migrating library "${library.name}" with media type "${library.mediaType}"`)
|
|
await db.librariesDb.update((record) => record.id === library.id, () => library).then(() => true).catch((error) => {
|
|
Logger.error(`[dbMigration] Update library failed: ${error}`)
|
|
return false
|
|
})
|
|
}
|
|
}
|
|
|
|
|
|
const userObjects = await db.usersDb.select((result) => result.audiobooks != undefined).then((results) => results.data)
|
|
if (!userObjects.length) {
|
|
Logger.warn('[dbMigration] No users found needing migration')
|
|
return
|
|
}
|
|
|
|
var userCount = 0
|
|
for (const userObj of userObjects) {
|
|
Logger.info(`[dbMigration] Migrating User "${userObj.username}"`)
|
|
var success = await cleanUserObject(db, userObj)
|
|
if (!success) {
|
|
await new Promise((resolve) => setTimeout(resolve, 500))
|
|
Logger.warn(`[dbMigration] Second attempt Migrating User "${userObj.username}"`)
|
|
success = await cleanUserObject(db, userObj)
|
|
if (!success) {
|
|
throw new Error('Db migration failed migrating users')
|
|
}
|
|
}
|
|
userCount++
|
|
}
|
|
|
|
var sessionCount = 0
|
|
const userListeningSessions = await db.sessionsDb.select((result) => result.audiobookId != undefined).then((results) => results.data)
|
|
if (userListeningSessions.length) {
|
|
|
|
for (const session of userListeningSessions) {
|
|
var success = await cleanSessionObj(db, session)
|
|
if (!success) {
|
|
await new Promise((resolve) => setTimeout(resolve, 500))
|
|
Logger.warn(`[dbMigration] Second attempt Migrating Session "${session.id}"`)
|
|
success = await cleanSessionObj(db, session)
|
|
if (!success) {
|
|
Logger.error(`[dbMigration] Failed to migrate session "${session.id}"`)
|
|
}
|
|
}
|
|
if (success) sessionCount++
|
|
}
|
|
}
|
|
|
|
Logger.info(`==== User migration complete (${userCount} Users, ${sessionCount} Sessions) ====`)
|
|
}
|
|
|
|
async function checkUpdateMetadataPath() {
|
|
var bookMetadataPath = Path.posix.join(global.MetadataPath, 'books') // OLD
|
|
if (!(await fs.pathExists(bookMetadataPath))) {
|
|
Logger.debug(`[dbMigration] No need to update books metadata path`)
|
|
return
|
|
}
|
|
var itemsMetadataPath = Path.posix.join(global.MetadataPath, 'items')
|
|
await fs.rename(bookMetadataPath, itemsMetadataPath)
|
|
Logger.info(`>>> Renamed metadata dir from /metadata/books to /metadata/items`)
|
|
}
|
|
|
|
module.exports.migrate = async (db) => {
|
|
await checkUpdateMetadataPath()
|
|
// Before DB Load clean data
|
|
await migrateUserData(db)
|
|
await db.init()
|
|
// After DB Load
|
|
await migrateLibraryItems(db)
|
|
// TODO: Eventually remove audiobooks db when stable
|
|
} |