mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-22 00:07:52 +01:00
feat(429): retry 429 request errors
This commit is contained in:
parent
73c21242b4
commit
70827727aa
@ -1,7 +1,7 @@
|
|||||||
const axios = require('axios')
|
const axios = require('axios')
|
||||||
const { levenshteinDistance } = require('../utils/index')
|
const { levenshteinDistance } = require('../utils/index')
|
||||||
const Logger = require('../Logger')
|
const Logger = require('../Logger')
|
||||||
const pThrottle = require('p-throttle')
|
const Throttle = require('p-throttle')
|
||||||
|
|
||||||
class Audnexus {
|
class Audnexus {
|
||||||
static _instance = null
|
static _instance = null
|
||||||
@ -16,13 +16,13 @@ class Audnexus {
|
|||||||
|
|
||||||
// Rate limit is 100 requests per minute.
|
// Rate limit is 100 requests per minute.
|
||||||
// @see https://github.com/laxamentumtech/audnexus#-deployment-
|
// @see https://github.com/laxamentumtech/audnexus#-deployment-
|
||||||
this.limiter = pThrottle({
|
this.limiter = Throttle({
|
||||||
// Setting the limit to 1 allows for a short pause between requests that is almost imperceptible to
|
// Setting the limit to 1 allows for a short pause between requests that is imperceptible to the end user.
|
||||||
// the end user. A larger limit will grab blocks faster and then wait for the alloted time(interval) before
|
// A larger limit will grab blocks faster and then wait for the alloted time(interval) before
|
||||||
// fetching another batch.
|
// fetching another batch, but with a discernable pause from the user perspective.
|
||||||
limit: 1,
|
limit: 1,
|
||||||
strict: true,
|
strict: true,
|
||||||
interval: 300
|
interval: 150
|
||||||
})
|
})
|
||||||
|
|
||||||
Audnexus._instance = this
|
Audnexus._instance = this
|
||||||
@ -35,8 +35,8 @@ class Audnexus {
|
|||||||
|
|
||||||
Logger.info(`[Audnexus] Searching for author "${authorRequestUrl}"`)
|
Logger.info(`[Audnexus] Searching for author "${authorRequestUrl}"`)
|
||||||
|
|
||||||
const throttle = this.limiter(() => axios.get(authorRequestUrl))
|
return this._processRequest(this.limiter(() => axios.get(authorRequestUrl)))
|
||||||
return throttle().then((res) => res.data || [])
|
.then((res) => res.data || [])
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
Logger.error(`[Audnexus] Author ASIN request failed for ${name}`, error)
|
Logger.error(`[Audnexus] Author ASIN request failed for ${name}`, error)
|
||||||
return []
|
return []
|
||||||
@ -50,8 +50,8 @@ class Audnexus {
|
|||||||
|
|
||||||
Logger.info(`[Audnexus] Searching for author "${authorRequestUrl}"`)
|
Logger.info(`[Audnexus] Searching for author "${authorRequestUrl}"`)
|
||||||
|
|
||||||
const throttle = this.limiter(() => axios.get(authorRequestUrl))
|
return this._processRequest(this.limiter(() => axios.get(authorRequestUrl)))
|
||||||
return throttle().then((res) => res.data)
|
.then((res) => res.data)
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
Logger.error(`[Audnexus] Author request failed for ${asin}`, error)
|
Logger.error(`[Audnexus] Author request failed for ${asin}`, error)
|
||||||
return null
|
return null
|
||||||
@ -93,13 +93,35 @@ class Audnexus {
|
|||||||
getChaptersByASIN(asin, region) {
|
getChaptersByASIN(asin, region) {
|
||||||
Logger.debug(`[Audnexus] Get chapters for ASIN ${asin}/${region}`)
|
Logger.debug(`[Audnexus] Get chapters for ASIN ${asin}/${region}`)
|
||||||
|
|
||||||
const throttle = this.limiter(() => axios.get(`${this.baseUrl}/books/${asin}/chapters?region=${region}`))
|
return this._processRequest(this.limiter(() => axios.get(`${this.baseUrl}/books/${asin}/chapters?region=${region}`)))
|
||||||
return throttle().then((res) => res.data)
|
.then((res) => res.data)
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
Logger.error(`[Audnexus] Chapter ASIN request failed for ${asin}/${region}`, error)
|
Logger.error(`[Audnexus] Chapter ASIN request failed for ${asin}/${region}`, error)
|
||||||
return null
|
return null
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = Audnexus
|
module.exports = Audnexus
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user