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