mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-03 00:06:46 +01:00
Add:iTunes search api metadata provider #381
This commit is contained in:
parent
43f48b65f8
commit
a907c88f66
@ -47,9 +47,9 @@
|
|||||||
<ui-dropdown v-model="provider" :items="providers" label="Provider" small />
|
<ui-dropdown v-model="provider" :items="providers" label="Provider" small />
|
||||||
</div>
|
</div>
|
||||||
<div class="w-72 px-1">
|
<div class="w-72 px-1">
|
||||||
<ui-text-input-with-label v-model="searchTitle" :label="provider == 'audible' ? 'Search Title or ASIN' : 'Search Title'" placeholder="Search" />
|
<ui-text-input-with-label v-model="searchTitle" :label="searchTitleLabel" placeholder="Search" />
|
||||||
</div>
|
</div>
|
||||||
<div class="w-72 px-1">
|
<div v-show="provider != 'itunes'" class="w-72 px-1">
|
||||||
<ui-text-input-with-label v-model="searchAuthor" label="Author" />
|
<ui-text-input-with-label v-model="searchAuthor" label="Author" />
|
||||||
</div>
|
</div>
|
||||||
<ui-btn class="mt-5 ml-1" type="submit">Search</ui-btn>
|
<ui-btn class="mt-5 ml-1" type="submit">Search</ui-btn>
|
||||||
@ -98,20 +98,6 @@ export default {
|
|||||||
showLocalCovers: false,
|
showLocalCovers: false,
|
||||||
previewUpload: null,
|
previewUpload: null,
|
||||||
selectedFile: null,
|
selectedFile: null,
|
||||||
providers: [
|
|
||||||
{
|
|
||||||
text: 'Google Books',
|
|
||||||
value: 'google'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: 'Open Library',
|
|
||||||
value: 'openlibrary'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
text: 'Audible',
|
|
||||||
value: 'audible'
|
|
||||||
}
|
|
||||||
],
|
|
||||||
provider: 'google'
|
provider: 'google'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -134,6 +120,14 @@ export default {
|
|||||||
this.$emit('update:processing', val)
|
this.$emit('update:processing', val)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
providers() {
|
||||||
|
return this.$store.state.scanners.providers
|
||||||
|
},
|
||||||
|
searchTitleLabel() {
|
||||||
|
if (this.provider == 'audible') return 'Search Title or ASIN'
|
||||||
|
else if (this.provider == 'itunes') return 'Search Term'
|
||||||
|
return 'Search Title'
|
||||||
|
},
|
||||||
coverAspectRatio() {
|
coverAspectRatio() {
|
||||||
return this.$store.getters['getServerSetting']('coverAspectRatio')
|
return this.$store.getters['getServerSetting']('coverAspectRatio')
|
||||||
},
|
},
|
||||||
|
@ -6,9 +6,9 @@
|
|||||||
<ui-dropdown v-model="provider" :items="providers" label="Provider" small />
|
<ui-dropdown v-model="provider" :items="providers" label="Provider" small />
|
||||||
</div>
|
</div>
|
||||||
<div class="w-72 px-1">
|
<div class="w-72 px-1">
|
||||||
<ui-text-input-with-label v-model="searchTitle" :label="provider == 'audible' ? 'Search Title or ASIN' : 'Search Title'" placeholder="Search" />
|
<ui-text-input-with-label v-model="searchTitle" :label="searchTitleLabel" placeholder="Search" />
|
||||||
</div>
|
</div>
|
||||||
<div class="w-72 px-1">
|
<div v-show="provider != 'itunes'" class="w-72 px-1">
|
||||||
<ui-text-input-with-label v-model="searchAuthor" label="Author" />
|
<ui-text-input-with-label v-model="searchAuthor" label="Author" />
|
||||||
</div>
|
</div>
|
||||||
<ui-btn class="mt-5 ml-1" type="submit">Search</ui-btn>
|
<ui-btn class="mt-5 ml-1" type="submit">Search</ui-btn>
|
||||||
@ -147,6 +147,11 @@ export default {
|
|||||||
},
|
},
|
||||||
providers() {
|
providers() {
|
||||||
return this.$store.state.scanners.providers
|
return this.$store.state.scanners.providers
|
||||||
|
},
|
||||||
|
searchTitleLabel() {
|
||||||
|
if (this.provider == 'audible') return 'Search Title or ASIN'
|
||||||
|
else if (this.provider == 'itunes') return 'Search Term'
|
||||||
|
return 'Search Title'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
@ -12,6 +12,10 @@ export const state = () => ({
|
|||||||
{
|
{
|
||||||
text: 'Audible',
|
text: 'Audible',
|
||||||
value: 'audible'
|
value: 'audible'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: 'iTunes',
|
||||||
|
value: 'itunes'
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
})
|
})
|
||||||
|
@ -2,6 +2,7 @@ const OpenLibrary = require('../providers/OpenLibrary')
|
|||||||
const LibGen = require('../providers/LibGen')
|
const LibGen = require('../providers/LibGen')
|
||||||
const GoogleBooks = require('../providers/GoogleBooks')
|
const GoogleBooks = require('../providers/GoogleBooks')
|
||||||
const Audible = require('../providers/Audible')
|
const Audible = require('../providers/Audible')
|
||||||
|
const iTunes = require('../providers/iTunes')
|
||||||
const Logger = require('../Logger')
|
const Logger = require('../Logger')
|
||||||
const { levenshteinDistance } = require('../utils/index')
|
const { levenshteinDistance } = require('../utils/index')
|
||||||
|
|
||||||
@ -11,6 +12,7 @@ class BookFinder {
|
|||||||
this.libGen = new LibGen()
|
this.libGen = new LibGen()
|
||||||
this.googleBooks = new GoogleBooks()
|
this.googleBooks = new GoogleBooks()
|
||||||
this.audible = new Audible()
|
this.audible = new Audible()
|
||||||
|
this.iTunesApi = new iTunes()
|
||||||
|
|
||||||
this.verbose = false
|
this.verbose = false
|
||||||
}
|
}
|
||||||
@ -158,6 +160,10 @@ class BookFinder {
|
|||||||
return books
|
return books
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getiTunesAudiobooksResults(title, author) {
|
||||||
|
return this.iTunesApi.searchAudiobooks(title)
|
||||||
|
}
|
||||||
|
|
||||||
async getAudibleResults(title, author) {
|
async getAudibleResults(title, author) {
|
||||||
var books = await this.audible.search(title, author);
|
var books = await this.audible.search(title, author);
|
||||||
if (this.verbose) Logger.debug(`Audible Book Search Results: ${books.length || 0}`)
|
if (this.verbose) Logger.debug(`Audible Book Search Results: ${books.length || 0}`)
|
||||||
@ -175,6 +181,8 @@ class BookFinder {
|
|||||||
return this.getGoogleBooksResults(title, author)
|
return this.getGoogleBooksResults(title, author)
|
||||||
} else if (provider === 'audible') {
|
} else if (provider === 'audible') {
|
||||||
return this.getAudibleResults(title, author)
|
return this.getAudibleResults(title, author)
|
||||||
|
} else if (provider === 'itunes') {
|
||||||
|
return this.getiTunesAudiobooksResults(title, author)
|
||||||
} else if (provider === 'libgen') {
|
} else if (provider === 'libgen') {
|
||||||
books = await this.getLibGenResults(title, author, maxTitleDistance, maxAuthorDistance)
|
books = await this.getLibGenResults(title, author, maxTitleDistance, maxAuthorDistance)
|
||||||
} else if (provider === 'openlibrary') {
|
} else if (provider === 'openlibrary') {
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
const axios = require('axios')
|
const axios = require('axios')
|
||||||
const Logger = require('../Logger')
|
const Logger = require('../Logger')
|
||||||
|
const { stripHtml } = require('string-strip-html')
|
||||||
class iTunes {
|
class iTunes {
|
||||||
constructor() { }
|
constructor() { }
|
||||||
|
|
||||||
@ -18,16 +18,35 @@ class iTunes {
|
|||||||
limit: options.limit,
|
limit: options.limit,
|
||||||
country: options.country
|
country: options.country
|
||||||
}
|
}
|
||||||
console.log('Query', query)
|
|
||||||
|
|
||||||
return axios.get('https://itunes.apple.com/search', { params: query }).then((response) => {
|
return axios.get('https://itunes.apple.com/search', { params: query }).then((response) => {
|
||||||
var data = response.data
|
return response.data.results || []
|
||||||
console.log('data', data)
|
|
||||||
return data.results || []
|
|
||||||
}).catch((error) => {
|
}).catch((error) => {
|
||||||
Logger.error(`[iTunes] search request error`, error)
|
Logger.error(`[iTunes] search request error`, error)
|
||||||
return []
|
return []
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cleanAudiobook(data) {
|
||||||
|
// Example cover art: https://is1-ssl.mzstatic.com/image/thumb/Music118/v4/cb/ea/73/cbea739b-ff3b-11c4-fb93-7889fbec7390/9781598874983_cover.jpg/100x100bb.jpg
|
||||||
|
// 100x100bb can be replaced by other values https://github.com/bendodson/itunes-artwork-finder
|
||||||
|
var cover = data.artworkUrl100 || data.artworkUrl60 || ''
|
||||||
|
cover = cover.replace('100x100bb', '600x600bb').replace('60x60bb', '600x600bb')
|
||||||
|
return {
|
||||||
|
id: data.collectionId,
|
||||||
|
artistId: data.artistId,
|
||||||
|
title: data.collectionName,
|
||||||
|
author: data.artistName,
|
||||||
|
description: stripHtml(data.description || '').result,
|
||||||
|
publishYear: data.releaseDate ? data.releaseDate.split('-')[0] : null,
|
||||||
|
genres: data.primaryGenreName ? [data.primaryGenreName] : [],
|
||||||
|
cover
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
searchAudiobooks(term) {
|
||||||
|
return this.search({ term, entity: 'audiobook', media: 'audiobook' }).then((results) => {
|
||||||
|
return results.map(this.cleanAudiobook)
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
module.exports = iTunes
|
module.exports = iTunes
|
Loading…
Reference in New Issue
Block a user