2021-11-18 02:19:24 +01:00
|
|
|
const axios = require('axios')
|
|
|
|
const { levenshteinDistance } = require('../utils/index')
|
|
|
|
const Logger = require('../Logger')
|
2024-01-23 07:19:05 +01:00
|
|
|
const Throttle = require('p-throttle')
|
2021-11-18 02:19:24 +01:00
|
|
|
|
|
|
|
class Audnexus {
|
2024-01-17 03:31:29 +01:00
|
|
|
static _instance = null
|
2023-10-05 22:48:55 +02:00
|
|
|
|
2021-11-18 02:19:24 +01:00
|
|
|
constructor() {
|
2023-10-05 22:48:55 +02:00
|
|
|
// ensures Audnexus class is singleton
|
|
|
|
if (Audnexus._instance) {
|
|
|
|
return Audnexus._instance
|
|
|
|
}
|
|
|
|
|
2021-11-18 02:19:24 +01:00
|
|
|
this.baseUrl = 'https://api.audnex.us'
|
2023-10-05 22:48:55 +02:00
|
|
|
|
2024-01-23 05:36:20 +01:00
|
|
|
// Rate limit is 100 requests per minute.
|
2023-10-05 22:48:55 +02:00
|
|
|
// @see https://github.com/laxamentumtech/audnexus#-deployment-
|
2024-01-23 07:19:05 +01:00
|
|
|
this.limiter = Throttle({
|
|
|
|
// Setting the limit to 1 allows for a short pause between requests that is imperceptible to the end user.
|
|
|
|
// A larger limit will grab blocks faster and then wait for the alloted time(interval) before
|
|
|
|
// fetching another batch, but with a discernable pause from the user perspective.
|
2024-01-23 05:36:20 +01:00
|
|
|
limit: 1,
|
|
|
|
strict: true,
|
2024-01-23 07:19:05 +01:00
|
|
|
interval: 150
|
2023-10-05 22:48:55 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
Audnexus._instance = this
|
2021-11-18 02:19:24 +01:00
|
|
|
}
|
|
|
|
|
2023-04-16 22:53:46 +02:00
|
|
|
authorASINsRequest(name, region) {
|
|
|
|
name = encodeURIComponent(name)
|
|
|
|
const regionQuery = region ? `®ion=${region}` : ''
|
|
|
|
const authorRequestUrl = `${this.baseUrl}/authors?name=${name}${regionQuery}`
|
2023-10-05 22:48:55 +02:00
|
|
|
|
2023-04-16 22:53:46 +02:00
|
|
|
Logger.info(`[Audnexus] Searching for author "${authorRequestUrl}"`)
|
2023-10-05 22:48:55 +02:00
|
|
|
|
2024-01-23 07:19:05 +01:00
|
|
|
return this._processRequest(this.limiter(() => axios.get(authorRequestUrl)))
|
|
|
|
.then((res) => res.data || [])
|
2023-10-05 22:48:55 +02:00
|
|
|
.catch((error) => {
|
|
|
|
Logger.error(`[Audnexus] Author ASIN request failed for ${name}`, error)
|
|
|
|
return []
|
|
|
|
})
|
2021-11-18 02:19:24 +01:00
|
|
|
}
|
|
|
|
|
2023-04-16 22:53:46 +02:00
|
|
|
authorRequest(asin, region) {
|
|
|
|
asin = encodeURIComponent(asin)
|
|
|
|
const regionQuery = region ? `?region=${region}` : ''
|
|
|
|
const authorRequestUrl = `${this.baseUrl}/authors/${asin}${regionQuery}`
|
2023-10-05 22:48:55 +02:00
|
|
|
|
2023-04-16 22:53:46 +02:00
|
|
|
Logger.info(`[Audnexus] Searching for author "${authorRequestUrl}"`)
|
2023-10-05 22:48:55 +02:00
|
|
|
|
2024-01-23 07:19:05 +01:00
|
|
|
return this._processRequest(this.limiter(() => axios.get(authorRequestUrl)))
|
|
|
|
.then((res) => res.data)
|
2023-10-05 22:48:55 +02:00
|
|
|
.catch((error) => {
|
|
|
|
Logger.error(`[Audnexus] Author request failed for ${asin}`, error)
|
|
|
|
return null
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-04-16 22:53:46 +02:00
|
|
|
async findAuthorByASIN(asin, region) {
|
|
|
|
const author = await this.authorRequest(asin, region)
|
2023-10-05 22:48:55 +02:00
|
|
|
|
|
|
|
return author ?
|
|
|
|
{
|
|
|
|
asin: author.asin,
|
|
|
|
description: author.description,
|
|
|
|
image: author.image || null,
|
|
|
|
name: author.name
|
|
|
|
} : null
|
2022-05-14 01:11:54 +02:00
|
|
|
}
|
|
|
|
|
2023-04-16 22:53:46 +02:00
|
|
|
async findAuthorByName(name, region, maxLevenshtein = 3) {
|
2021-11-18 02:19:24 +01:00
|
|
|
Logger.debug(`[Audnexus] Looking up author by name ${name}`)
|
2023-10-05 22:48:55 +02:00
|
|
|
|
2023-04-16 22:53:46 +02:00
|
|
|
const asins = await this.authorASINsRequest(name, region)
|
|
|
|
const matchingAsin = asins.find(obj => levenshteinDistance(obj.name, name) <= maxLevenshtein)
|
2023-10-05 22:48:55 +02:00
|
|
|
|
2021-11-18 02:19:24 +01:00
|
|
|
if (!matchingAsin) {
|
|
|
|
return null
|
|
|
|
}
|
2023-10-05 22:48:55 +02:00
|
|
|
|
2023-04-16 22:53:46 +02:00
|
|
|
const author = await this.authorRequest(matchingAsin.asin)
|
2023-10-05 22:48:55 +02:00
|
|
|
return author ?
|
|
|
|
{
|
|
|
|
description: author.description,
|
|
|
|
image: author.image || null,
|
|
|
|
asin: author.asin,
|
|
|
|
name: author.name
|
|
|
|
} : null
|
2021-11-18 02:19:24 +01:00
|
|
|
}
|
2022-05-11 00:03:41 +02:00
|
|
|
|
2022-10-15 22:31:07 +02:00
|
|
|
getChaptersByASIN(asin, region) {
|
|
|
|
Logger.debug(`[Audnexus] Get chapters for ASIN ${asin}/${region}`)
|
2023-10-05 22:48:55 +02:00
|
|
|
|
2024-01-23 07:19:05 +01:00
|
|
|
return this._processRequest(this.limiter(() => axios.get(`${this.baseUrl}/books/${asin}/chapters?region=${region}`)))
|
|
|
|
.then((res) => res.data)
|
2023-10-05 22:48:55 +02:00
|
|
|
.catch((error) => {
|
|
|
|
Logger.error(`[Audnexus] Chapter ASIN request failed for ${asin}/${region}`, error)
|
|
|
|
return null
|
|
|
|
})
|
2022-05-11 00:03:41 +02:00
|
|
|
}
|
2024-01-23 07:19:05 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Internal method to process requests and retry if rate limit is exceeded.
|
|
|
|
*/
|
|
|
|
async _processRequest(request) {
|
|
|
|
try {
|
|
|
|
const response = await request()
|
|
|
|
return response
|
|
|
|
} catch (error) {
|
|
|
|
if (error.response?.status === 429) {
|
|
|
|
const retryAfter = parseInt(error.response.headers?.['retry-after'], 10) || 5
|
|
|
|
|
|
|
|
Logger.warn(`[Audnexus] Rate limit exceeded. Retrying in ${retryAfter} seconds.`)
|
|
|
|
await new Promise(resolve => setTimeout(resolve, retryAfter * 1000))
|
|
|
|
|
|
|
|
return this._processRequest(request)
|
|
|
|
}
|
|
|
|
|
|
|
|
throw error
|
|
|
|
}
|
|
|
|
}
|
2021-11-18 02:19:24 +01:00
|
|
|
}
|
2023-10-05 22:48:55 +02:00
|
|
|
|
2024-01-23 07:19:05 +01:00
|
|
|
module.exports = Audnexus
|
|
|
|
|