mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-24 23:38:56 -04:00 
			
		
		
		
	Initial commit for server side approach
This is the first commit for bringing this over to the server side. It works! Right now it fails if the autoscanner or or the manual individual book scanner try to do it's thing. I'll need to update those
This commit is contained in:
		
							parent
							
								
									effc63755b
								
							
						
					
					
						commit
						b3d9323f66
					
				| @ -103,7 +103,7 @@ class PodcastController { | |||||||
|         Logger.error('Invalid podcast feed request response') |         Logger.error('Invalid podcast feed request response') | ||||||
|         return res.status(500).send('Bad response from feed request') |         return res.status(500).send('Bad response from feed request') | ||||||
|       } |       } | ||||||
|       Logger.debug(`[PdocastController] Podcast feed size ${(data.data.length / 1024 / 1024).toFixed(2)}MB`) |       Logger.debug(`[PodcastController] Podcast feed size ${(data.data.length / 1024 / 1024).toFixed(2)}MB`) | ||||||
|       var payload = await parsePodcastRssFeedXml(data.data, false, includeRaw) |       var payload = await parsePodcastRssFeedXml(data.data, false, includeRaw) | ||||||
|       if (!payload) { |       if (!payload) { | ||||||
|         return res.status(500).send('Invalid podcast RSS feed') |         return res.status(500).send('Invalid podcast RSS feed') | ||||||
|  | |||||||
| @ -360,10 +360,12 @@ class Book { | |||||||
|     this.rebuildTracks() |     this.rebuildTracks() | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   rebuildTracks() { |   rebuildTracks(preferOverdriveMediaMarker = false) { | ||||||
|  |     Logger.debug(`[Book] we are rebuilding the tracks!`) | ||||||
|  |     Logger.debug(`[Book] preferOverdriveMediaMarker: ${preferOverdriveMediaMarker}`) | ||||||
|     this.audioFiles.sort((a, b) => a.index - b.index) |     this.audioFiles.sort((a, b) => a.index - b.index) | ||||||
|     this.missingParts = [] |     this.missingParts = [] | ||||||
|     this.setChapters() |     this.setChapters(preferOverdriveMediaMarker) | ||||||
|     this.checkUpdateMissingTracks() |     this.checkUpdateMissingTracks() | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| @ -395,7 +397,103 @@ class Book { | |||||||
|     return wasUpdated |     return wasUpdated | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   setChapters() { |   generateChaptersFromOverdriveMediaMarkers(overdriveMediaMarkers, includedAudioFiles) { | ||||||
|  |     var parseString = require('xml2js').parseString; // function to convert xml to JSON
 | ||||||
|  |      | ||||||
|  |     var parsedOverdriveMediaMarkers = [] // an array of objects. each object being a chapter with a name and time key. the values are arrays of strings
 | ||||||
|  | 
 | ||||||
|  |     overdriveMediaMarkers.forEach(function (item, index) {      | ||||||
|  |       var parsed_result | ||||||
|  |       parseString(item, function (err, result) { | ||||||
|  |         // result.Markers.Marker is the result of parsing the XML for the MediaMarker tags for the MP3 file (Part##.mp3)
 | ||||||
|  |         // it is shaped like this:
 | ||||||
|  |         // [
 | ||||||
|  |         //   {
 | ||||||
|  |         //     "Name": [
 | ||||||
|  |         //       "Chapter 1:  "
 | ||||||
|  |         //     ],
 | ||||||
|  |         //     "Time": [
 | ||||||
|  |         //       "0:00.000"
 | ||||||
|  |         //     ]
 | ||||||
|  |         //   },
 | ||||||
|  |         //   {
 | ||||||
|  |         //     "Name": [
 | ||||||
|  |         //       "Chapter 2: "
 | ||||||
|  |         //     ],
 | ||||||
|  |         //     "Time": [
 | ||||||
|  |         //       "15:51.000"
 | ||||||
|  |         //     ]
 | ||||||
|  |         //   }
 | ||||||
|  |         // ]
 | ||||||
|  | 
 | ||||||
|  |         parsed_result = result.Markers.Marker | ||||||
|  |          | ||||||
|  |         // The values for Name and Time in parsed_results are returned as Arrays from parseString
 | ||||||
|  |         // update them to be strings
 | ||||||
|  |         parsed_result.forEach((item, index) => { | ||||||
|  |           Object.keys(item).forEach(key => { | ||||||
|  |             item[key] = item[key].toString() | ||||||
|  |           }) | ||||||
|  |         }) | ||||||
|  |       }) | ||||||
|  | 
 | ||||||
|  |       parsedOverdriveMediaMarkers.push(parsed_result) | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     // go from an array of arrays of objects to an array of objects
 | ||||||
|  |     // end result looks like:
 | ||||||
|  |     // [
 | ||||||
|  |     //   {
 | ||||||
|  |     //     "Name": "Chapter 1:  The Worst Birthday",
 | ||||||
|  |     //     "Time": "0:00.000"
 | ||||||
|  |     //   },
 | ||||||
|  |     //   {
 | ||||||
|  |     //     "Name": "Chapter 2:  Dobby's Warning",
 | ||||||
|  |     //     "Time": "15:51.000"
 | ||||||
|  |     //   },
 | ||||||
|  |     //   { redacted }
 | ||||||
|  |     // ]
 | ||||||
|  |     parsedOverdriveMediaMarkers = parsedOverdriveMediaMarkers | ||||||
|  | 
 | ||||||
|  |     var index = 0 | ||||||
|  |      | ||||||
|  |     var time = 0.0 | ||||||
|  |      | ||||||
|  | 
 | ||||||
|  |     // actually generate the chapter object
 | ||||||
|  |     // logic ported over from benonymity's OverdriveChapterizer:
 | ||||||
|  |     //    https://github.com/benonymity/OverdriveChapterizer/blob/main/chapters.py
 | ||||||
|  |     var length = 0.0 | ||||||
|  |     var newOChapters = [] | ||||||
|  |     const weirdChapterFilterRegex = /([(]\d|[cC]ontinued)/ | ||||||
|  |     includedAudioFiles.forEach((track, track_index) => { | ||||||
|  |       parsedOverdriveMediaMarkers[track_index].forEach((chapter) => { | ||||||
|  |         Logger.debug(`[Book] Attempting regex check for ${chapter.Name}!`) | ||||||
|  |         if (weirdChapterFilterRegex.test(chapter.Name)) { | ||||||
|  |           Logger.debug(`[Book] That shit weird yo`) | ||||||
|  |           return | ||||||
|  |         } | ||||||
|  |         time = chapter.Time.split(":") | ||||||
|  |         time = length + parseFloat(time[0]) * 60 + parseFloat(time[1]) | ||||||
|  |         newOChapters.push( | ||||||
|  |           { | ||||||
|  |             id: index++, | ||||||
|  |             start: time, | ||||||
|  |             end: length, | ||||||
|  |             title: chapter.Name | ||||||
|  |           } | ||||||
|  |         ) | ||||||
|  |       }) | ||||||
|  |       length += track.duration | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     Logger.debug(`[Book] newOChapters: ${JSON.stringify(newOChapters)}`) | ||||||
|  |     return newOChapters | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   setChapters(preferOverdriveMediaMarker = false) { | ||||||
|  |     Logger.debug('[Book] inside setChapters!') | ||||||
|     // If 1 audio file without chapters, then no chapters will be set
 |     // If 1 audio file without chapters, then no chapters will be set
 | ||||||
|     var includedAudioFiles = this.audioFiles.filter(af => !af.exclude) |     var includedAudioFiles = this.audioFiles.filter(af => !af.exclude) | ||||||
|     if (includedAudioFiles.length === 1) { |     if (includedAudioFiles.length === 1) { | ||||||
| @ -407,7 +505,17 @@ class Book { | |||||||
|       this.chapters = [] |       this.chapters = [] | ||||||
|       var currChapterId = 0 |       var currChapterId = 0 | ||||||
|       var currStartTime = 0 |       var currStartTime = 0 | ||||||
|  |       var overdriveMediaMarkers = includedAudioFiles.map((af) => af.metaTags.tagOverdriveMediaMarker).filter(notUndefined => notUndefined !== undefined) || [] | ||||||
|  |       Logger.debug(`[setChapters] overdriveMediaMarkers: ${JSON.stringify(overdriveMediaMarkers)}`) | ||||||
|  | 
 | ||||||
|  |       // If preferOverdriveMediaMarker is set, try and use that first
 | ||||||
|  |       if (preferOverdriveMediaMarker) { | ||||||
|  |         Logger.debug(`[Book] preferring overdrive media markers! Lets generate em.`) | ||||||
|  |         this.chapters = this.generateChaptersFromOverdriveMediaMarkers(overdriveMediaMarkers, includedAudioFiles) | ||||||
|  | 
 | ||||||
|  |       } else { | ||||||
|         includedAudioFiles.forEach((file) => { |         includedAudioFiles.forEach((file) => { | ||||||
|  |           //console.log(`audiofile MetaTags Overdrive: ${JSON.stringify(file.metaTags.tagOverdriveMediaMarker)}}`)
 | ||||||
|           // If audio file has chapters use chapters
 |           // If audio file has chapters use chapters
 | ||||||
|             if (file.chapters && file.chapters.length) { |             if (file.chapters && file.chapters.length) { | ||||||
|               file.chapters.forEach((chapter) => { |               file.chapters.forEach((chapter) => { | ||||||
| @ -447,6 +555,7 @@ class Book { | |||||||
|         }) |         }) | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|   // Only checks container format
 |   // Only checks container format
 | ||||||
|   checkCanDirectPlay(payload) { |   checkCanDirectPlay(payload) { | ||||||
|  | |||||||
| @ -13,6 +13,7 @@ class ServerSettings { | |||||||
|     this.scannerPreferOpfMetadata = false |     this.scannerPreferOpfMetadata = false | ||||||
|     this.scannerPreferMatchedMetadata = false |     this.scannerPreferMatchedMetadata = false | ||||||
|     this.scannerDisableWatcher = false  |     this.scannerDisableWatcher = false  | ||||||
|  |     this.scannerPreferOverdriveMediaMarker = false | ||||||
| 
 | 
 | ||||||
|     // Metadata - choose to store inside users library item folder
 |     // Metadata - choose to store inside users library item folder
 | ||||||
|     this.storeCoverWithItem = false |     this.storeCoverWithItem = false | ||||||
| @ -65,6 +66,7 @@ class ServerSettings { | |||||||
|     this.scannerPreferOpfMetadata = !!settings.scannerPreferOpfMetadata |     this.scannerPreferOpfMetadata = !!settings.scannerPreferOpfMetadata | ||||||
|     this.scannerPreferMatchedMetadata = !!settings.scannerPreferMatchedMetadata |     this.scannerPreferMatchedMetadata = !!settings.scannerPreferMatchedMetadata | ||||||
|     this.scannerDisableWatcher = !!settings.scannerDisableWatcher |     this.scannerDisableWatcher = !!settings.scannerDisableWatcher | ||||||
|  |     this.scannerPreferOverdriveMediaMarker = !!settings.scannerPreferOverdriveMediaMarker | ||||||
| 
 | 
 | ||||||
|     this.storeCoverWithItem = !!settings.storeCoverWithItem |     this.storeCoverWithItem = !!settings.storeCoverWithItem | ||||||
|     if (settings.storeCoverWithBook != undefined) { // storeCoverWithBook was old name of setting < v2
 |     if (settings.storeCoverWithBook != undefined) { // storeCoverWithBook was old name of setting < v2
 | ||||||
| @ -111,6 +113,7 @@ class ServerSettings { | |||||||
|       scannerPreferOpfMetadata: this.scannerPreferOpfMetadata, |       scannerPreferOpfMetadata: this.scannerPreferOpfMetadata, | ||||||
|       scannerPreferMatchedMetadata: this.scannerPreferMatchedMetadata, |       scannerPreferMatchedMetadata: this.scannerPreferMatchedMetadata, | ||||||
|       scannerDisableWatcher: this.scannerDisableWatcher, |       scannerDisableWatcher: this.scannerDisableWatcher, | ||||||
|  |       scannerPreferOverdriveMediaMarker: this.scannerPreferOverdriveMediaMarker, | ||||||
|       storeCoverWithItem: this.storeCoverWithItem, |       storeCoverWithItem: this.storeCoverWithItem, | ||||||
|       storeMetadataWithItem: this.storeMetadataWithItem, |       storeMetadataWithItem: this.storeMetadataWithItem, | ||||||
|       rateLimitLoginRequests: this.rateLimitLoginRequests, |       rateLimitLoginRequests: this.rateLimitLoginRequests, | ||||||
|  | |||||||
| @ -196,6 +196,9 @@ class MediaFileScanner { | |||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   async scanMediaFiles(mediaLibraryFiles, scanData, libraryItem, preferAudioMetadata, libraryScan = null) { |   async scanMediaFiles(mediaLibraryFiles, scanData, libraryItem, preferAudioMetadata, libraryScan = null) { | ||||||
|  |     Logger.debug('[scanMediaFiles] inside scan media files!') | ||||||
|  |     Logger.debug(`[scanMediaFiles] libraryScan: ${JSON.stringify(libraryScan)}`) | ||||||
|  | 
 | ||||||
|     var hasUpdated = false |     var hasUpdated = false | ||||||
| 
 | 
 | ||||||
|     var mediaScanResult = await this.executeMediaFileScans(libraryItem.mediaType, mediaLibraryFiles, scanData) |     var mediaScanResult = await this.executeMediaFileScans(libraryItem.mediaType, mediaLibraryFiles, scanData) | ||||||
| @ -208,6 +211,7 @@ class MediaFileScanner { | |||||||
|     } else if (mediaScanResult.audioFiles.length) { |     } else if (mediaScanResult.audioFiles.length) { | ||||||
|       if (libraryScan) { |       if (libraryScan) { | ||||||
|         libraryScan.addLog(LogLevel.DEBUG, `Library Item "${scanData.path}" Audio file scan took ${mediaScanResult.elapsed}ms for ${mediaScanResult.audioFiles.length} with average time of ${mediaScanResult.averageScanDuration}ms`) |         libraryScan.addLog(LogLevel.DEBUG, `Library Item "${scanData.path}" Audio file scan took ${mediaScanResult.elapsed}ms for ${mediaScanResult.audioFiles.length} with average time of ${mediaScanResult.averageScanDuration}ms`) | ||||||
|  |         Logger.debug(`Library Item "${scanData.path}" Audio file scan took ${mediaScanResult.elapsed}ms for ${mediaScanResult.audioFiles.length} with average time of ${mediaScanResult.averageScanDuration}ms`) | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       var totalAudioFilesToInclude = mediaScanResult.audioFiles.length |       var totalAudioFilesToInclude = mediaScanResult.audioFiles.length | ||||||
| @ -217,18 +221,23 @@ class MediaFileScanner { | |||||||
| 
 | 
 | ||||||
|       // Book: Adding audio files to book media
 |       // Book: Adding audio files to book media
 | ||||||
|       if (libraryItem.mediaType === 'book') { |       if (libraryItem.mediaType === 'book') { | ||||||
|  |         Logger.debug('Its a book!') | ||||||
|         if (newAudioFiles.length) { |         if (newAudioFiles.length) { | ||||||
|  |           Logger.debug('[MediaFileScanner] newAudioFiles.length was true?') | ||||||
|           // Single Track Audiobooks
 |           // Single Track Audiobooks
 | ||||||
|           if (totalAudioFilesToInclude === 1) { |           if (totalAudioFilesToInclude === 1) { | ||||||
|  |             Logger.debug('[MediaFileScanner] totalAudioFilesToInclude === 1') | ||||||
|             var af = mediaScanResult.audioFiles[0] |             var af = mediaScanResult.audioFiles[0] | ||||||
|             af.index = 1 |             af.index = 1 | ||||||
|             libraryItem.media.addAudioFile(af) |             libraryItem.media.addAudioFile(af) | ||||||
|             hasUpdated = true |             hasUpdated = true | ||||||
|           } else { |           } else { | ||||||
|  |             Logger.debug('[MediaFileScanner] totalAudioFilesToInclude === 1 WAS FALSE') | ||||||
|             this.runSmartTrackOrder(libraryItem, mediaScanResult.audioFiles) |             this.runSmartTrackOrder(libraryItem, mediaScanResult.audioFiles) | ||||||
|             hasUpdated = true |             hasUpdated = true | ||||||
|           } |           } | ||||||
|         } else { |         } else { | ||||||
|  |           Logger.debug('[MediaFileScanner] Only updating metadata?') | ||||||
|           // Only update metadata not index
 |           // Only update metadata not index
 | ||||||
|           mediaScanResult.audioFiles.forEach((af) => { |           mediaScanResult.audioFiles.forEach((af) => { | ||||||
|             var existingAF = libraryItem.media.findFileWithInode(af.ino) |             var existingAF = libraryItem.media.findFileWithInode(af.ino) | ||||||
| @ -247,7 +256,9 @@ class MediaFileScanner { | |||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if (hasUpdated) { |         if (hasUpdated) { | ||||||
|           libraryItem.media.rebuildTracks() |           Logger.debug('[MediaFileScanner] hasUpdated is true! Going to rebuild tracks now...') | ||||||
|  |           Logger.debug(`[MediaFileScanner] libraryScan: ${JSON.stringify(libraryScan)}`) | ||||||
|  |           libraryItem.media.rebuildTracks(libraryScan.scanOptions.preferOverdriveMediaMarker) | ||||||
|         } |         } | ||||||
|       } else { // Podcast Media Type
 |       } else { // Podcast Media Type
 | ||||||
|         var existingAudioFiles = mediaScanResult.audioFiles.filter(af => libraryItem.media.findFileWithInode(af.ino)) |         var existingAudioFiles = mediaScanResult.audioFiles.filter(af => libraryItem.media.findFileWithInode(af.ino)) | ||||||
| @ -264,6 +275,7 @@ class MediaFileScanner { | |||||||
|         // Update audio file metadata for audio files already there
 |         // Update audio file metadata for audio files already there
 | ||||||
|         existingAudioFiles.forEach((af) => { |         existingAudioFiles.forEach((af) => { | ||||||
|           var peAudioFile = libraryItem.media.findFileWithInode(af.ino) |           var peAudioFile = libraryItem.media.findFileWithInode(af.ino) | ||||||
|  |           Logger.debug(`[MediaFileScanner] peAudioFile: ${JSON.stringify(peAudioFile)}`) | ||||||
|           if (peAudioFile.updateFromScan && peAudioFile.updateFromScan(af)) { |           if (peAudioFile.updateFromScan && peAudioFile.updateFromScan(af)) { | ||||||
|             hasUpdated = true |             hasUpdated = true | ||||||
|           } |           } | ||||||
|  | |||||||
| @ -9,6 +9,7 @@ class ScanOptions { | |||||||
|     this.preferAudioMetadata = false |     this.preferAudioMetadata = false | ||||||
|     this.preferOpfMetadata = false |     this.preferOpfMetadata = false | ||||||
|     this.preferMatchedMetadata = false |     this.preferMatchedMetadata = false | ||||||
|  |     this.preferOverdriveMediaMarker = false | ||||||
| 
 | 
 | ||||||
|     if (options) { |     if (options) { | ||||||
|       this.construct(options) |       this.construct(options) | ||||||
| @ -34,7 +35,8 @@ class ScanOptions { | |||||||
|       storeCoverWithItem: this.storeCoverWithItem, |       storeCoverWithItem: this.storeCoverWithItem, | ||||||
|       preferAudioMetadata: this.preferAudioMetadata, |       preferAudioMetadata: this.preferAudioMetadata, | ||||||
|       preferOpfMetadata: this.preferOpfMetadata, |       preferOpfMetadata: this.preferOpfMetadata, | ||||||
|       preferMatchedMetadata: this.preferMatchedMetadata |       preferMatchedMetadata: this.preferMatchedMetadata, | ||||||
|  |       preferOverdriveMediaMarker: this.preferOverdriveMediaMarker | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| @ -47,6 +49,7 @@ class ScanOptions { | |||||||
|     this.preferAudioMetadata = serverSettings.scannerPreferAudioMetadata |     this.preferAudioMetadata = serverSettings.scannerPreferAudioMetadata | ||||||
|     this.preferOpfMetadata = serverSettings.scannerPreferOpfMetadata |     this.preferOpfMetadata = serverSettings.scannerPreferOpfMetadata | ||||||
|     this.scannerPreferMatchedMetadata = serverSettings.scannerPreferMatchedMetadata |     this.scannerPreferMatchedMetadata = serverSettings.scannerPreferMatchedMetadata | ||||||
|  |     this.preferOverdriveMediaMarker = serverSettings.scannerPreferOverdriveMediaMarker | ||||||
|   } |   } | ||||||
| } | } | ||||||
| module.exports = ScanOptions | module.exports = ScanOptions | ||||||
| @ -62,6 +62,7 @@ class Scanner { | |||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   async scanLibraryItem(libraryMediaType, folder, libraryItem) { |   async scanLibraryItem(libraryMediaType, folder, libraryItem) { | ||||||
|  |     Logger.debug(`[Scanner] SCANNING ITEMS JOE`) | ||||||
|     // TODO: Support for single media item
 |     // TODO: Support for single media item
 | ||||||
|     var libraryItemData = await getLibraryItemFileData(libraryMediaType, folder, libraryItem.path, false, this.db.serverSettings) |     var libraryItemData = await getLibraryItemFileData(libraryMediaType, folder, libraryItem.path, false, this.db.serverSettings) | ||||||
|     if (!libraryItemData) { |     if (!libraryItemData) { | ||||||
| @ -114,6 +115,7 @@ class Scanner { | |||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   async scan(library, options = {}) { |   async scan(library, options = {}) { | ||||||
|  |     Logger.debug('[scan] inside of scan') | ||||||
|     if (this.isLibraryScanning(library.id)) { |     if (this.isLibraryScanning(library.id)) { | ||||||
|       Logger.error(`[Scanner] Already scanning ${library.id}`) |       Logger.error(`[Scanner] Already scanning ${library.id}`) | ||||||
|       return |       return | ||||||
| @ -126,6 +128,7 @@ class Scanner { | |||||||
| 
 | 
 | ||||||
|     var scanOptions = new ScanOptions() |     var scanOptions = new ScanOptions() | ||||||
|     scanOptions.setData(options, this.db.serverSettings) |     scanOptions.setData(options, this.db.serverSettings) | ||||||
|  |     Logger.debug(`[Scanner] scanOptions: ${JSON.stringify(scanOptions)}`) | ||||||
| 
 | 
 | ||||||
|     var libraryScan = new LibraryScan() |     var libraryScan = new LibraryScan() | ||||||
|     libraryScan.setData(library, scanOptions) |     libraryScan.setData(library, scanOptions) | ||||||
| @ -165,6 +168,8 @@ class Scanner { | |||||||
|   async scanLibrary(libraryScan) { |   async scanLibrary(libraryScan) { | ||||||
|     var libraryItemDataFound = [] |     var libraryItemDataFound = [] | ||||||
| 
 | 
 | ||||||
|  |     Logger.debug(`[scanLibrary] libraryScan: ${JSON.stringify(libraryScan)}`) | ||||||
|  | 
 | ||||||
|     // Scan each library
 |     // Scan each library
 | ||||||
|     for (let i = 0; i < libraryScan.folders.length; i++) { |     for (let i = 0; i < libraryScan.folders.length; i++) { | ||||||
|       var folder = libraryScan.folders[i] |       var folder = libraryScan.folders[i] | ||||||
| @ -750,6 +755,7 @@ class Scanner { | |||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   async matchLibraryItems(library) { |   async matchLibraryItems(library) { | ||||||
|  |     Logger.debug("SCANNING!") | ||||||
|     if (library.mediaType === 'podcast') { |     if (library.mediaType === 'podcast') { | ||||||
|       Logger.error(`[Scanner] matchLibraryItems: Match all not supported for podcasts yet`) |       Logger.error(`[Scanner] matchLibraryItems: Match all not supported for podcasts yet`) | ||||||
|       return |       return | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user