mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-11-04 03:17:00 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			145 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			145 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
const axios = require('axios')
 | 
						|
const htmlSanitizer = require('../utils/htmlSanitizer')
 | 
						|
const Logger = require('../Logger')
 | 
						|
 | 
						|
class Audible {
 | 
						|
    constructor() {
 | 
						|
        this.regionMap = {
 | 
						|
            'us': '.com',
 | 
						|
            'ca': '.ca',
 | 
						|
            'uk': '.co.uk',
 | 
						|
            'au': '.com.au',
 | 
						|
            'fr': '.fr',
 | 
						|
            'de': '.de',
 | 
						|
            'jp': '.co.jp',
 | 
						|
            'it': '.it',
 | 
						|
            'in': '.in',
 | 
						|
            'es': '.es'
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Audible will sometimes send sequences with "Book 1" or "2, Dramatized Adaptation"
 | 
						|
     * @see https://github.com/advplyr/audiobookshelf/issues/2380
 | 
						|
     * @see https://github.com/advplyr/audiobookshelf/issues/1339
 | 
						|
     * 
 | 
						|
     * @param {string} seriesName
 | 
						|
     * @param {string} sequence 
 | 
						|
     * @returns {string}
 | 
						|
     */
 | 
						|
    cleanSeriesSequence(seriesName, sequence) {
 | 
						|
        if (!sequence) return ''
 | 
						|
        // match any number with optional decimal (e.g, 1 or 1.5 or .5)
 | 
						|
        let numberFound = sequence.match(/\.\d+|\d+(?:\.\d+)?/)
 | 
						|
        let updatedSequence = numberFound ? numberFound[0] : sequence
 | 
						|
        if (sequence !== updatedSequence) {
 | 
						|
            Logger.debug(`[Audible] Series "${seriesName}" sequence was cleaned from "${sequence}" to "${updatedSequence}"`)
 | 
						|
        }
 | 
						|
        return updatedSequence
 | 
						|
    }
 | 
						|
 | 
						|
    cleanResult(item) {
 | 
						|
        const { title, subtitle, asin, authors, narrators, publisherName, summary, releaseDate, image, genres, seriesPrimary, seriesSecondary, language, runtimeLengthMin, formatType } = item
 | 
						|
 | 
						|
        const series = []
 | 
						|
        if (seriesPrimary) {
 | 
						|
            series.push({
 | 
						|
                series: seriesPrimary.name,
 | 
						|
                sequence: this.cleanSeriesSequence(seriesPrimary.name, seriesPrimary.position || '')
 | 
						|
            })
 | 
						|
        }
 | 
						|
        if (seriesSecondary) {
 | 
						|
            series.push({
 | 
						|
                series: seriesSecondary.name,
 | 
						|
                sequence: this.cleanSeriesSequence(seriesSecondary.name, seriesSecondary.position || '')
 | 
						|
            })
 | 
						|
        }
 | 
						|
 | 
						|
        const genresFiltered = genres ? genres.filter(g => g.type == "genre").map(g => g.name) : []
 | 
						|
        const tagsFiltered = genres ? genres.filter(g => g.type == "tag").map(g => g.name) : []
 | 
						|
 | 
						|
        return {
 | 
						|
            title,
 | 
						|
            subtitle: subtitle || null,
 | 
						|
            author: authors ? authors.map(({ name }) => name).join(', ') : null,
 | 
						|
            narrator: narrators ? narrators.map(({ name }) => name).join(', ') : null,
 | 
						|
            publisher: publisherName,
 | 
						|
            publishedYear: releaseDate ? releaseDate.split('-')[0] : null,
 | 
						|
            description: summary ? htmlSanitizer.stripAllTags(summary) : null,
 | 
						|
            cover: image,
 | 
						|
            asin,
 | 
						|
            genres: genresFiltered.length ? genresFiltered : null,
 | 
						|
            tags: tagsFiltered.length ? tagsFiltered.join(', ') : null,
 | 
						|
            series: series.length ? series : null,
 | 
						|
            language: language ? language.charAt(0).toUpperCase() + language.slice(1) : null,
 | 
						|
            duration: runtimeLengthMin && !isNaN(runtimeLengthMin) ? Number(runtimeLengthMin) : 0,
 | 
						|
            region: item.region || null,
 | 
						|
            rating: item.rating || null,
 | 
						|
            abridged: formatType === 'abridged'
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * Test if a search title matches an ASIN. Supports lowercase letters
 | 
						|
     * 
 | 
						|
     * @param {string} title 
 | 
						|
     * @returns {boolean}
 | 
						|
     */
 | 
						|
    isProbablyAsin(title) {
 | 
						|
        return /^[0-9A-Za-z]{10}$/.test(title)
 | 
						|
    }
 | 
						|
 | 
						|
    asinSearch(asin, region) {
 | 
						|
        if (!asin) return []
 | 
						|
        asin = encodeURIComponent(asin.toUpperCase())
 | 
						|
        var regionQuery = region ? `?region=${region}` : ''
 | 
						|
        var url = `https://api.audnex.us/books/${asin}${regionQuery}`
 | 
						|
        Logger.debug(`[Audible] ASIN url: ${url}`)
 | 
						|
        return axios.get(url).then((res) => {
 | 
						|
            if (!res || !res.data || !res.data.asin) return null
 | 
						|
            return res.data
 | 
						|
        }).catch(error => {
 | 
						|
            Logger.error('[Audible] ASIN search error', error)
 | 
						|
            return []
 | 
						|
        })
 | 
						|
    }
 | 
						|
 | 
						|
    async search(title, author, asin, region) {
 | 
						|
        if (region && !this.regionMap[region]) {
 | 
						|
            Logger.error(`[Audible] search: Invalid region ${region}`)
 | 
						|
            region = ''
 | 
						|
        }
 | 
						|
 | 
						|
        let items
 | 
						|
        if (asin) {
 | 
						|
            items = [await this.asinSearch(asin, region)]
 | 
						|
        }
 | 
						|
 | 
						|
        if (!items && this.isProbablyAsin(title)) {
 | 
						|
            items = [await this.asinSearch(title, region)]
 | 
						|
        }
 | 
						|
 | 
						|
        if (!items) {
 | 
						|
            const queryObj = {
 | 
						|
                num_results: '10',
 | 
						|
                products_sort_by: 'Relevance',
 | 
						|
                title: title
 | 
						|
            }
 | 
						|
            if (author) queryObj.author = author
 | 
						|
            const queryString = (new URLSearchParams(queryObj)).toString()
 | 
						|
            const tld = region ? this.regionMap[region] : '.com'
 | 
						|
            const url = `https://api.audible${tld}/1.0/catalog/products?${queryString}`
 | 
						|
            Logger.debug(`[Audible] Search url: ${url}`)
 | 
						|
            items = await axios.get(url).then((res) => {
 | 
						|
                if (!res?.data?.products) return null
 | 
						|
                return Promise.all(res.data.products.map(result => this.asinSearch(result.asin, region)))
 | 
						|
            }).catch(error => {
 | 
						|
                Logger.error('[Audible] query search error', error)
 | 
						|
                return []
 | 
						|
            })
 | 
						|
        }
 | 
						|
        return items ? items.map(item => this.cleanResult(item)) : []
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
module.exports = Audible |