mirror of
				https://github.com/advplyr/audiobookshelf.git
				synced 2025-10-25 07:48:56 -04:00 
			
		
		
		
	Add:Cover image cache, resize & use webp image #223
This commit is contained in:
		
							parent
							
								
									d04f3450ec
								
							
						
					
					
						commit
						ddf0fa72e8
					
				| @ -10,7 +10,7 @@ | |||||||
|         <p :style="{ fontSize: sizeMultiplier * 0.8 + 'rem' }" class="font-book text-gray-300 text-center">{{ title }}</p> |         <p :style="{ fontSize: sizeMultiplier * 0.8 + 'rem' }" class="font-book text-gray-300 text-center">{{ title }}</p> | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|       <img v-show="audiobook" ref="cover" :src="bookCoverSrc" class="w-full h-full object-contain transition-opacity duration-300" @load="imageLoaded" :style="{ opacity: imageReady ? 1 : 0 }" /> |       <img v-show="audiobook" ref="cover" :src="bookCoverSrc" class="w-full h-full object-contain transition-opacity duration-300" :class="showCoverBg ? 'object-contain' : 'object-fill'" @load="imageLoaded" :style="{ opacity: imageReady ? 1 : 0 }" /> | ||||||
| 
 | 
 | ||||||
|       <!-- Placeholder Cover Title & Author --> |       <!-- Placeholder Cover Title & Author --> | ||||||
|       <div v-if="!hasCover" class="absolute top-0 left-0 right-0 bottom-0 w-full h-full flex items-center justify-center" :style="{ padding: placeholderCoverPadding + 'rem' }"> |       <div v-if="!hasCover" class="absolute top-0 left-0 right-0 bottom-0 w-full h-full flex items-center justify-center" :style="{ padding: placeholderCoverPadding + 'rem' }"> | ||||||
|  | |||||||
| @ -53,7 +53,8 @@ | |||||||
|     <div class="h-0.5 bg-primary bg-opacity-30 w-full" /> |     <div class="h-0.5 bg-primary bg-opacity-30 w-full" /> | ||||||
| 
 | 
 | ||||||
|     <div class="flex items-center py-4"> |     <div class="flex items-center py-4"> | ||||||
|       <ui-btn color="bg" small :padding-x="4" class="hidden lg:block" :loading="isResettingAudiobooks" @click="resetAudiobooks">Reset All Audiobooks</ui-btn> |       <ui-btn color="bg" small :padding-x="4" class="hidden lg:block mr-2" :loading="isPurgingCache" @click="purgeCache">Purge Cache</ui-btn> | ||||||
|  |       <ui-btn color="bg" small :padding-x="4" class="hidden lg:block" :loading="isResettingAudiobooks" @click="resetAudiobooks">Remove All Audiobooks</ui-btn> | ||||||
|       <div class="flex-grow" /> |       <div class="flex-grow" /> | ||||||
|       <p class="pr-2 text-sm font-book text-yellow-400">Report bugs, request features, provide feedback, and contribute on <a class="underline" href="https://github.com/advplyr/audiobookshelf" target="_blank">github</a>.</p> |       <p class="pr-2 text-sm font-book text-yellow-400">Report bugs, request features, provide feedback, and contribute on <a class="underline" href="https://github.com/advplyr/audiobookshelf" target="_blank">github</a>.</p> | ||||||
|       <a href="https://github.com/advplyr/audiobookshelf" target="_blank" class="text-white hover:text-gray-200 hover:scale-150 hover:rotate-6 transform duration-500"> |       <a href="https://github.com/advplyr/audiobookshelf" target="_blank" class="text-white hover:text-gray-200 hover:scale-150 hover:rotate-6 transform duration-500"> | ||||||
| @ -93,6 +94,7 @@ export default { | |||||||
|       storeCoversInAudiobookDir: false, |       storeCoversInAudiobookDir: false, | ||||||
|       updatingServerSettings: false, |       updatingServerSettings: false, | ||||||
|       useSquareBookCovers: false, |       useSquareBookCovers: false, | ||||||
|  |       isPurgingCache: false, | ||||||
|       newServerSettings: {} |       newServerSettings: {} | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
| @ -209,6 +211,19 @@ export default { | |||||||
|             this.$toast.error('Failed to reset audiobooks - manually remove the /config/audiobooks folder') |             this.$toast.error('Failed to reset audiobooks - manually remove the /config/audiobooks folder') | ||||||
|           }) |           }) | ||||||
|       } |       } | ||||||
|  |     }, | ||||||
|  |     async purgeCache() { | ||||||
|  |       this.isPurgingCache = true | ||||||
|  |       await this.$axios | ||||||
|  |         .$post('/api/purgecache') | ||||||
|  |         .then(() => { | ||||||
|  |           this.$toast.success('Cache Purged!') | ||||||
|  |         }) | ||||||
|  |         .catch((error) => { | ||||||
|  |           console.error('Failed to purge cache', error) | ||||||
|  |           this.$toast.error('Failed to purge cache') | ||||||
|  |         }) | ||||||
|  |       this.isPurgingCache = false | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   mounted() { |   mounted() { | ||||||
|  | |||||||
| @ -6,8 +6,6 @@ Vue.directive('click-outside', vClickOutside.directive) | |||||||
| 
 | 
 | ||||||
| Vue.prototype.$eventBus = new Vue() | Vue.prototype.$eventBus = new Vue() | ||||||
| 
 | 
 | ||||||
| Vue.prototype.$isDev = process.env.NODE_ENV !== 'production' |  | ||||||
| 
 |  | ||||||
| Vue.prototype.$dateDistanceFromNow = (unixms) => { | Vue.prototype.$dateDistanceFromNow = (unixms) => { | ||||||
|   if (!unixms) return '' |   if (!unixms) return '' | ||||||
|   return formatDistance(unixms, Date.now(), { addSuffix: true }) |   return formatDistance(unixms, Date.now(), { addSuffix: true }) | ||||||
| @ -145,4 +143,5 @@ export { | |||||||
| export default ({ app }, inject) => { | export default ({ app }, inject) => { | ||||||
|   app.$decode = decode |   app.$decode = decode | ||||||
|   app.$encode = encode |   app.$encode = encode | ||||||
|  |   app.$isDev = process.env.NODE_ENV !== 'production' | ||||||
| } | } | ||||||
| @ -16,36 +16,17 @@ export const getters = { | |||||||
|     if (!bookItem) return placeholder |     if (!bookItem) return placeholder | ||||||
|     var book = bookItem.book |     var book = bookItem.book | ||||||
|     if (!book || !book.cover || book.cover === placeholder) return placeholder |     if (!book || !book.cover || book.cover === placeholder) return placeholder | ||||||
|     var cover = book.cover |  | ||||||
| 
 | 
 | ||||||
|     // Absolute URL covers (should no longer be used)
 |     // Absolute URL covers (should no longer be used)
 | ||||||
|     if (cover.startsWith('http:') || cover.startsWith('https:')) return cover |     if (book.cover.startsWith('http:') || book.cover.startsWith('https:')) return book.cover | ||||||
| 
 | 
 | ||||||
|     // Server hosted covers
 |     var userToken = rootGetters['user/getToken'] | ||||||
|     try { |     var bookLastUpdate = book.lastUpdate || Date.now() | ||||||
|       // Ensure cover is refreshed if cached
 |  | ||||||
|       var bookLastUpdate = book.lastUpdate || Date.now() |  | ||||||
|       var userToken = rootGetters['user/getToken'] |  | ||||||
| 
 | 
 | ||||||
|       cover = cover.replace(/\\/g, '/') |     if (process.env.NODE_ENV !== 'production') { // Testing
 | ||||||
| 
 |       return `http://localhost:3333/api/books/${bookItem.id}/cover?token=${userToken}&ts=${bookLastUpdate}` | ||||||
|       // Map old covers to new format /s/book/{bookid}/*
 |  | ||||||
|       if (cover.startsWith('/local')) { |  | ||||||
|         cover = cover.replace('local', `s/book/${bookItem.id}`) |  | ||||||
|         if (cover.includes(bookItem.path + '/')) { // Remove book path
 |  | ||||||
|           cover = cover.replace(bookItem.path + '/', '') |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       // Easier to replace these special characters then to encodeUriComponent of the filename
 |  | ||||||
|       var encodedCover = cover.replace(/%/g, '%25').replace(/#/g, '%23') |  | ||||||
| 
 |  | ||||||
|       var url = new URL(encodedCover, document.baseURI) |  | ||||||
|       return url.href + `?token=${userToken}&ts=${bookLastUpdate}` |  | ||||||
|     } catch (err) { |  | ||||||
|       console.error(err) |  | ||||||
|       return placeholder |  | ||||||
|     } |     } | ||||||
|  |     return `/api/books/${bookItem.id}/cover?token=${userToken}` | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										234
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										234
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							| @ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "audiobookshelf", |   "name": "audiobookshelf", | ||||||
|   "version": "1.6.30", |   "version": "1.6.39", | ||||||
|   "lockfileVersion": 1, |   "lockfileVersion": 1, | ||||||
|   "requires": true, |   "requires": true, | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
| @ -105,14 +105,12 @@ | |||||||
|     "ansi-regex": { |     "ansi-regex": { | ||||||
|       "version": "2.1.1", |       "version": "2.1.1", | ||||||
|       "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", |       "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", | ||||||
|       "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", |       "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "aproba": { |     "aproba": { | ||||||
|       "version": "1.2.0", |       "version": "1.2.0", | ||||||
|       "resolved": "https://registry.npmjs.org/aproba/-/aproba-1.2.0.tgz", |       "resolved": "https://registry.npmjs.org/aproba/-/aproba-1.2.0.tgz", | ||||||
|       "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw==", |       "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw==" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "archiver": { |     "archiver": { | ||||||
|       "version": "5.3.0", |       "version": "5.3.0", | ||||||
| @ -173,7 +171,6 @@ | |||||||
|       "version": "1.1.7", |       "version": "1.1.7", | ||||||
|       "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.7.tgz", |       "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.7.tgz", | ||||||
|       "integrity": "sha512-nxwy40TuMiUGqMyRHgCSWZ9FM4VAoRP4xUYSTv5ImRog+h9yISPbVH7H8fASCIzYn9wlEv4zvFL7uKDMCFQm3g==", |       "integrity": "sha512-nxwy40TuMiUGqMyRHgCSWZ9FM4VAoRP4xUYSTv5ImRog+h9yISPbVH7H8fASCIzYn9wlEv4zvFL7uKDMCFQm3g==", | ||||||
|       "optional": true, |  | ||||||
|       "requires": { |       "requires": { | ||||||
|         "delegates": "^1.0.0", |         "delegates": "^1.0.0", | ||||||
|         "readable-stream": "^2.0.6" |         "readable-stream": "^2.0.6" | ||||||
| @ -183,7 +180,6 @@ | |||||||
|           "version": "2.3.7", |           "version": "2.3.7", | ||||||
|           "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", |           "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", | ||||||
|           "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", |           "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", | ||||||
|           "optional": true, |  | ||||||
|           "requires": { |           "requires": { | ||||||
|             "core-util-is": "~1.0.0", |             "core-util-is": "~1.0.0", | ||||||
|             "inherits": "~2.0.3", |             "inherits": "~2.0.3", | ||||||
| @ -346,8 +342,7 @@ | |||||||
|     "chownr": { |     "chownr": { | ||||||
|       "version": "1.1.4", |       "version": "1.1.4", | ||||||
|       "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", |       "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", | ||||||
|       "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", |       "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "clone-response": { |     "clone-response": { | ||||||
|       "version": "1.0.2", |       "version": "1.0.2", | ||||||
| @ -360,8 +355,38 @@ | |||||||
|     "code-point-at": { |     "code-point-at": { | ||||||
|       "version": "1.1.0", |       "version": "1.1.0", | ||||||
|       "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", |       "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", | ||||||
|       "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=", |       "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=" | ||||||
|       "optional": true |     }, | ||||||
|  |     "color": { | ||||||
|  |       "version": "4.1.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/color/-/color-4.1.0.tgz", | ||||||
|  |       "integrity": "sha512-o2rkkxyLGgYoeUy1OodXpbPAQNmlNBrirQ8ODO8QutzDiDMNdezSOZLNnusQ6pUpCQJUsaJIo9DZJKqa2HgH7A==", | ||||||
|  |       "requires": { | ||||||
|  |         "color-convert": "^2.0.1", | ||||||
|  |         "color-string": "^1.9.0" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "color-convert": { | ||||||
|  |       "version": "2.0.1", | ||||||
|  |       "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", | ||||||
|  |       "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", | ||||||
|  |       "requires": { | ||||||
|  |         "color-name": "~1.1.4" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "color-name": { | ||||||
|  |       "version": "1.1.4", | ||||||
|  |       "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", | ||||||
|  |       "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" | ||||||
|  |     }, | ||||||
|  |     "color-string": { | ||||||
|  |       "version": "1.9.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/color-string/-/color-string-1.9.0.tgz", | ||||||
|  |       "integrity": "sha512-9Mrz2AQLefkH1UvASKj6v6hj/7eWgjnT/cVsR8CumieLoT+g900exWeNogqtweI8dxloXN9BDQTYro1oWu/5CQ==", | ||||||
|  |       "requires": { | ||||||
|  |         "color-name": "^1.0.0", | ||||||
|  |         "simple-swizzle": "^0.2.2" | ||||||
|  |       } | ||||||
|     }, |     }, | ||||||
|     "command-line-args": { |     "command-line-args": { | ||||||
|       "version": "5.2.0", |       "version": "5.2.0", | ||||||
| @ -398,8 +423,7 @@ | |||||||
|     "console-control-strings": { |     "console-control-strings": { | ||||||
|       "version": "1.1.0", |       "version": "1.1.0", | ||||||
|       "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz", |       "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz", | ||||||
|       "integrity": "sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4=", |       "integrity": "sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4=" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "content-disposition": { |     "content-disposition": { | ||||||
|       "version": "0.5.3", |       "version": "0.5.3", | ||||||
| @ -492,8 +516,7 @@ | |||||||
|     "deep-extend": { |     "deep-extend": { | ||||||
|       "version": "0.6.0", |       "version": "0.6.0", | ||||||
|       "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", |       "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", | ||||||
|       "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", |       "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "defer-to-connect": { |     "defer-to-connect": { | ||||||
|       "version": "2.0.1", |       "version": "2.0.1", | ||||||
| @ -503,8 +526,7 @@ | |||||||
|     "delegates": { |     "delegates": { | ||||||
|       "version": "1.0.0", |       "version": "1.0.0", | ||||||
|       "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz", |       "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz", | ||||||
|       "integrity": "sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o=", |       "integrity": "sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o=" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "depd": { |     "depd": { | ||||||
|       "version": "1.1.2", |       "version": "1.1.2", | ||||||
| @ -519,8 +541,7 @@ | |||||||
|     "detect-libc": { |     "detect-libc": { | ||||||
|       "version": "1.0.3", |       "version": "1.0.3", | ||||||
|       "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-1.0.3.tgz", |       "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-1.0.3.tgz", | ||||||
|       "integrity": "sha1-+hN8S9aY7fVc1c0CrFWfkaTEups=", |       "integrity": "sha1-+hN8S9aY7fVc1c0CrFWfkaTEups=" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "dicer": { |     "dicer": { | ||||||
|       "version": "0.3.0", |       "version": "0.3.0", | ||||||
| @ -623,6 +644,11 @@ | |||||||
|       "resolved": "https://registry.npmjs.org/exit-on-epipe/-/exit-on-epipe-1.0.1.tgz", |       "resolved": "https://registry.npmjs.org/exit-on-epipe/-/exit-on-epipe-1.0.1.tgz", | ||||||
|       "integrity": "sha512-h2z5mrROTxce56S+pnvAV890uu7ls7f1kEvVGJbw1OlFH3/mlJ5bkXu0KRyW94v37zzHPiUd55iLn3DA7TjWpw==" |       "integrity": "sha512-h2z5mrROTxce56S+pnvAV890uu7ls7f1kEvVGJbw1OlFH3/mlJ5bkXu0KRyW94v37zzHPiUd55iLn3DA7TjWpw==" | ||||||
|     }, |     }, | ||||||
|  |     "expand-template": { | ||||||
|  |       "version": "2.0.3", | ||||||
|  |       "resolved": "https://registry.npmjs.org/expand-template/-/expand-template-2.0.3.tgz", | ||||||
|  |       "integrity": "sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==" | ||||||
|  |     }, | ||||||
|     "express": { |     "express": { | ||||||
|       "version": "4.17.1", |       "version": "4.17.1", | ||||||
|       "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", |       "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", | ||||||
| @ -762,7 +788,6 @@ | |||||||
|       "version": "2.7.4", |       "version": "2.7.4", | ||||||
|       "resolved": "https://registry.npmjs.org/gauge/-/gauge-2.7.4.tgz", |       "resolved": "https://registry.npmjs.org/gauge/-/gauge-2.7.4.tgz", | ||||||
|       "integrity": "sha1-LANAXHU4w51+s3sxcCLjJfsBi/c=", |       "integrity": "sha1-LANAXHU4w51+s3sxcCLjJfsBi/c=", | ||||||
|       "optional": true, |  | ||||||
|       "requires": { |       "requires": { | ||||||
|         "aproba": "^1.0.3", |         "aproba": "^1.0.3", | ||||||
|         "console-control-strings": "^1.0.0", |         "console-control-strings": "^1.0.0", | ||||||
| @ -782,6 +807,11 @@ | |||||||
|         "pump": "^3.0.0" |         "pump": "^3.0.0" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "github-from-package": { | ||||||
|  |       "version": "0.0.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/github-from-package/-/github-from-package-0.0.0.tgz", | ||||||
|  |       "integrity": "sha1-l/tdlr/eiXMxPyDoKI75oWf6ZM4=" | ||||||
|  |     }, | ||||||
|     "glob": { |     "glob": { | ||||||
|       "version": "7.1.7", |       "version": "7.1.7", | ||||||
|       "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz", |       "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.7.tgz", | ||||||
| @ -822,8 +852,7 @@ | |||||||
|     "has-unicode": { |     "has-unicode": { | ||||||
|       "version": "2.0.1", |       "version": "2.0.1", | ||||||
|       "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz", |       "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz", | ||||||
|       "integrity": "sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk=", |       "integrity": "sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk=" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "html-entities": { |     "html-entities": { | ||||||
|       "version": "2.3.2", |       "version": "2.3.2", | ||||||
| @ -903,8 +932,7 @@ | |||||||
|     "ini": { |     "ini": { | ||||||
|       "version": "1.3.8", |       "version": "1.3.8", | ||||||
|       "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", |       "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", | ||||||
|       "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", |       "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "ip": { |     "ip": { | ||||||
|       "version": "1.1.5", |       "version": "1.1.5", | ||||||
| @ -916,11 +944,15 @@ | |||||||
|       "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", |       "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", | ||||||
|       "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" |       "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" | ||||||
|     }, |     }, | ||||||
|  |     "is-arrayish": { | ||||||
|  |       "version": "0.3.2", | ||||||
|  |       "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.2.tgz", | ||||||
|  |       "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==" | ||||||
|  |     }, | ||||||
|     "is-fullwidth-code-point": { |     "is-fullwidth-code-point": { | ||||||
|       "version": "1.0.0", |       "version": "1.0.0", | ||||||
|       "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", |       "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", | ||||||
|       "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", |       "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", | ||||||
|       "optional": true, |  | ||||||
|       "requires": { |       "requires": { | ||||||
|         "number-is-nan": "^1.0.0" |         "number-is-nan": "^1.0.0" | ||||||
|       } |       } | ||||||
| @ -1117,6 +1149,21 @@ | |||||||
|       "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", |       "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", | ||||||
|       "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==" |       "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==" | ||||||
|     }, |     }, | ||||||
|  |     "lru-cache": { | ||||||
|  |       "version": "6.0.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", | ||||||
|  |       "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", | ||||||
|  |       "requires": { | ||||||
|  |         "yallist": "^4.0.0" | ||||||
|  |       }, | ||||||
|  |       "dependencies": { | ||||||
|  |         "yallist": { | ||||||
|  |           "version": "4.0.0", | ||||||
|  |           "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", | ||||||
|  |           "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==" | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "media-typer": { |     "media-typer": { | ||||||
|       "version": "0.3.0", |       "version": "0.3.0", | ||||||
|       "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", |       "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", | ||||||
| @ -1166,8 +1213,7 @@ | |||||||
|     "minimist": { |     "minimist": { | ||||||
|       "version": "1.2.5", |       "version": "1.2.5", | ||||||
|       "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", |       "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", | ||||||
|       "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", |       "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "minipass": { |     "minipass": { | ||||||
|       "version": "2.9.0", |       "version": "2.9.0", | ||||||
| @ -1197,6 +1243,11 @@ | |||||||
|         "minimist": "^1.2.5" |         "minimist": "^1.2.5" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "mkdirp-classic": { | ||||||
|  |       "version": "0.5.3", | ||||||
|  |       "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz", | ||||||
|  |       "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==" | ||||||
|  |     }, | ||||||
|     "moment": { |     "moment": { | ||||||
|       "version": "2.29.1", |       "version": "2.29.1", | ||||||
|       "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz", |       "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.1.tgz", | ||||||
| @ -1221,6 +1272,11 @@ | |||||||
|       "integrity": "sha512-bAdJv7fBLhWC+/Bls0Oza+mvTaNQtP+1RyhhhvD95pgUJz6XM5IzgmxOkItJ9tkoCiplvAnXI1tNmmUD/eScyA==", |       "integrity": "sha512-bAdJv7fBLhWC+/Bls0Oza+mvTaNQtP+1RyhhhvD95pgUJz6XM5IzgmxOkItJ9tkoCiplvAnXI1tNmmUD/eScyA==", | ||||||
|       "optional": true |       "optional": true | ||||||
|     }, |     }, | ||||||
|  |     "napi-build-utils": { | ||||||
|  |       "version": "1.0.2", | ||||||
|  |       "resolved": "https://registry.npmjs.org/napi-build-utils/-/napi-build-utils-1.0.2.tgz", | ||||||
|  |       "integrity": "sha512-ONmRUqK7zj7DWX0D9ADe03wbwOBZxNAfF20PlGfCWQcD3+/MakShIHrMqx9YwPTfxDdF1zLeL+RGZiR9kGMLdg==" | ||||||
|  |     }, | ||||||
|     "needle": { |     "needle": { | ||||||
|       "version": "2.9.1", |       "version": "2.9.1", | ||||||
|       "resolved": "https://registry.npmjs.org/needle/-/needle-2.9.1.tgz", |       "resolved": "https://registry.npmjs.org/needle/-/needle-2.9.1.tgz", | ||||||
| @ -1262,6 +1318,29 @@ | |||||||
|         "proper-lockfile": "^4.1.2" |         "proper-lockfile": "^4.1.2" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "node-abi": { | ||||||
|  |       "version": "3.5.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.5.0.tgz", | ||||||
|  |       "integrity": "sha512-LtHvNIBgOy5mO8mPEUtkCW/YCRWYEKshIvqhe1GHHyXEHEB5mgICyYnAcl4qan3uFeRROErKGzatFHPf6kDxWw==", | ||||||
|  |       "requires": { | ||||||
|  |         "semver": "^7.3.5" | ||||||
|  |       }, | ||||||
|  |       "dependencies": { | ||||||
|  |         "semver": { | ||||||
|  |           "version": "7.3.5", | ||||||
|  |           "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", | ||||||
|  |           "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", | ||||||
|  |           "requires": { | ||||||
|  |             "lru-cache": "^6.0.0" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "node-addon-api": { | ||||||
|  |       "version": "4.2.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-4.2.0.tgz", | ||||||
|  |       "integrity": "sha512-eazsqzwG2lskuzBqCGPi7Ac2UgOoMz8JVOXVhTvvPDYhthvNpefx8jWD8Np7Gv+2Sz0FlPWZk0nJV0z598Wn8Q==" | ||||||
|  |     }, | ||||||
|     "node-cron": { |     "node-cron": { | ||||||
|       "version": "3.0.0", |       "version": "3.0.0", | ||||||
|       "resolved": "https://registry.npmjs.org/node-cron/-/node-cron-3.0.0.tgz", |       "resolved": "https://registry.npmjs.org/node-cron/-/node-cron-3.0.0.tgz", | ||||||
| @ -1343,7 +1422,6 @@ | |||||||
|       "version": "4.1.2", |       "version": "4.1.2", | ||||||
|       "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-4.1.2.tgz", |       "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-4.1.2.tgz", | ||||||
|       "integrity": "sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg==", |       "integrity": "sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg==", | ||||||
|       "optional": true, |  | ||||||
|       "requires": { |       "requires": { | ||||||
|         "are-we-there-yet": "~1.1.2", |         "are-we-there-yet": "~1.1.2", | ||||||
|         "console-control-strings": "~1.1.0", |         "console-control-strings": "~1.1.0", | ||||||
| @ -1354,8 +1432,7 @@ | |||||||
|     "number-is-nan": { |     "number-is-nan": { | ||||||
|       "version": "1.0.1", |       "version": "1.0.1", | ||||||
|       "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", |       "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", | ||||||
|       "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=", |       "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "object-assign": { |     "object-assign": { | ||||||
|       "version": "4.1.1", |       "version": "4.1.1", | ||||||
| @ -1443,6 +1520,26 @@ | |||||||
|         "rss": "^1.2.2" |         "rss": "^1.2.2" | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "prebuild-install": { | ||||||
|  |       "version": "7.0.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-7.0.0.tgz", | ||||||
|  |       "integrity": "sha512-IvSenf33K7JcgddNz2D5w521EgO+4aMMjFt73Uk9FRzQ7P+QZPKrp7qPsDydsSwjGt3T5xRNnM1bj1zMTD5fTA==", | ||||||
|  |       "requires": { | ||||||
|  |         "detect-libc": "^1.0.3", | ||||||
|  |         "expand-template": "^2.0.3", | ||||||
|  |         "github-from-package": "0.0.0", | ||||||
|  |         "minimist": "^1.2.3", | ||||||
|  |         "mkdirp-classic": "^0.5.3", | ||||||
|  |         "napi-build-utils": "^1.0.1", | ||||||
|  |         "node-abi": "^3.3.0", | ||||||
|  |         "npmlog": "^4.0.1", | ||||||
|  |         "pump": "^3.0.0", | ||||||
|  |         "rc": "^1.2.7", | ||||||
|  |         "simple-get": "^4.0.0", | ||||||
|  |         "tar-fs": "^2.0.0", | ||||||
|  |         "tunnel-agent": "^0.6.0" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "printj": { |     "printj": { | ||||||
|       "version": "1.1.2", |       "version": "1.1.2", | ||||||
|       "resolved": "https://registry.npmjs.org/printj/-/printj-1.1.2.tgz", |       "resolved": "https://registry.npmjs.org/printj/-/printj-1.1.2.tgz", | ||||||
| @ -1554,7 +1651,6 @@ | |||||||
|       "version": "1.2.8", |       "version": "1.2.8", | ||||||
|       "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", |       "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", | ||||||
|       "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", |       "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", | ||||||
|       "optional": true, |  | ||||||
|       "requires": { |       "requires": { | ||||||
|         "deep-extend": "^0.6.0", |         "deep-extend": "^0.6.0", | ||||||
|         "ini": "~1.3.0", |         "ini": "~1.3.0", | ||||||
| @ -1719,19 +1815,66 @@ | |||||||
|     "set-blocking": { |     "set-blocking": { | ||||||
|       "version": "2.0.0", |       "version": "2.0.0", | ||||||
|       "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", |       "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", | ||||||
|       "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", |       "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "setprototypeof": { |     "setprototypeof": { | ||||||
|       "version": "1.1.1", |       "version": "1.1.1", | ||||||
|       "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", |       "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", | ||||||
|       "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" |       "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" | ||||||
|     }, |     }, | ||||||
|  |     "sharp": { | ||||||
|  |       "version": "0.29.3", | ||||||
|  |       "resolved": "https://registry.npmjs.org/sharp/-/sharp-0.29.3.tgz", | ||||||
|  |       "integrity": "sha512-fKWUuOw77E4nhpyzCCJR1ayrttHoFHBT2U/kR/qEMRhvPEcluG4BKj324+SCO1e84+knXHwhJ1HHJGnUt4ElGA==", | ||||||
|  |       "requires": { | ||||||
|  |         "color": "^4.0.1", | ||||||
|  |         "detect-libc": "^1.0.3", | ||||||
|  |         "node-addon-api": "^4.2.0", | ||||||
|  |         "prebuild-install": "^7.0.0", | ||||||
|  |         "semver": "^7.3.5", | ||||||
|  |         "simple-get": "^4.0.0", | ||||||
|  |         "tar-fs": "^2.1.1", | ||||||
|  |         "tunnel-agent": "^0.6.0" | ||||||
|  |       }, | ||||||
|  |       "dependencies": { | ||||||
|  |         "semver": { | ||||||
|  |           "version": "7.3.5", | ||||||
|  |           "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", | ||||||
|  |           "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", | ||||||
|  |           "requires": { | ||||||
|  |             "lru-cache": "^6.0.0" | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "signal-exit": { |     "signal-exit": { | ||||||
|       "version": "3.0.3", |       "version": "3.0.3", | ||||||
|       "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", |       "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", | ||||||
|       "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==" |       "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==" | ||||||
|     }, |     }, | ||||||
|  |     "simple-concat": { | ||||||
|  |       "version": "1.0.1", | ||||||
|  |       "resolved": "https://registry.npmjs.org/simple-concat/-/simple-concat-1.0.1.tgz", | ||||||
|  |       "integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==" | ||||||
|  |     }, | ||||||
|  |     "simple-get": { | ||||||
|  |       "version": "4.0.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-4.0.0.tgz", | ||||||
|  |       "integrity": "sha512-ZalZGexYr3TA0SwySsr5HlgOOinS4Jsa8YB2GJ6lUNAazyAu4KG/VmzMTwAt2YVXzzVj8QmefmAonZIK2BSGcQ==", | ||||||
|  |       "requires": { | ||||||
|  |         "decompress-response": "^6.0.0", | ||||||
|  |         "once": "^1.3.1", | ||||||
|  |         "simple-concat": "^1.0.0" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     "simple-swizzle": { | ||||||
|  |       "version": "0.2.2", | ||||||
|  |       "resolved": "https://registry.npmjs.org/simple-swizzle/-/simple-swizzle-0.2.2.tgz", | ||||||
|  |       "integrity": "sha1-pNprY1/8zMoz9w0Xy5JZLeleVXo=", | ||||||
|  |       "requires": { | ||||||
|  |         "is-arrayish": "^0.3.1" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "socket.io": { |     "socket.io": { | ||||||
|       "version": "4.1.3", |       "version": "4.1.3", | ||||||
|       "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-4.1.3.tgz", |       "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-4.1.3.tgz", | ||||||
| @ -1853,7 +1996,6 @@ | |||||||
|       "version": "1.0.2", |       "version": "1.0.2", | ||||||
|       "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", |       "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", | ||||||
|       "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", |       "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", | ||||||
|       "optional": true, |  | ||||||
|       "requires": { |       "requires": { | ||||||
|         "code-point-at": "^1.0.0", |         "code-point-at": "^1.0.0", | ||||||
|         "is-fullwidth-code-point": "^1.0.0", |         "is-fullwidth-code-point": "^1.0.0", | ||||||
| @ -1872,7 +2014,6 @@ | |||||||
|       "version": "3.0.1", |       "version": "3.0.1", | ||||||
|       "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", |       "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", | ||||||
|       "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", |       "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", | ||||||
|       "optional": true, |  | ||||||
|       "requires": { |       "requires": { | ||||||
|         "ansi-regex": "^2.0.0" |         "ansi-regex": "^2.0.0" | ||||||
|       } |       } | ||||||
| @ -1880,8 +2021,7 @@ | |||||||
|     "strip-json-comments": { |     "strip-json-comments": { | ||||||
|       "version": "2.0.1", |       "version": "2.0.1", | ||||||
|       "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", |       "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", | ||||||
|       "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", |       "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=" | ||||||
|       "optional": true |  | ||||||
|     }, |     }, | ||||||
|     "tar": { |     "tar": { | ||||||
|       "version": "4.4.19", |       "version": "4.4.19", | ||||||
| @ -1906,6 +2046,17 @@ | |||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|  |     "tar-fs": { | ||||||
|  |       "version": "2.1.1", | ||||||
|  |       "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.1.tgz", | ||||||
|  |       "integrity": "sha512-V0r2Y9scmbDRLCNex/+hYzvp/zyYjvFbHPNgVTKfQvVrb6guiE/fxP+XblDNR011utopbkex2nM4dHNV6GDsng==", | ||||||
|  |       "requires": { | ||||||
|  |         "chownr": "^1.1.1", | ||||||
|  |         "mkdirp-classic": "^0.5.2", | ||||||
|  |         "pump": "^3.0.0", | ||||||
|  |         "tar-stream": "^2.1.4" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "tar-stream": { |     "tar-stream": { | ||||||
|       "version": "2.2.0", |       "version": "2.2.0", | ||||||
|       "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", |       "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", | ||||||
| @ -1931,6 +2082,14 @@ | |||||||
|       "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", |       "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", | ||||||
|       "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" |       "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" | ||||||
|     }, |     }, | ||||||
|  |     "tunnel-agent": { | ||||||
|  |       "version": "0.6.0", | ||||||
|  |       "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", | ||||||
|  |       "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", | ||||||
|  |       "requires": { | ||||||
|  |         "safe-buffer": "^5.0.1" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "type-is": { |     "type-is": { | ||||||
|       "version": "1.6.18", |       "version": "1.6.18", | ||||||
|       "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", |       "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", | ||||||
| @ -1995,7 +2154,6 @@ | |||||||
|       "version": "1.1.3", |       "version": "1.1.3", | ||||||
|       "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", |       "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", | ||||||
|       "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", |       "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", | ||||||
|       "optional": true, |  | ||||||
|       "requires": { |       "requires": { | ||||||
|         "string-width": "^1.0.2 || 2" |         "string-width": "^1.0.2 || 2" | ||||||
|       } |       } | ||||||
|  | |||||||
| @ -52,4 +52,4 @@ | |||||||
|     "xml2js": "^0.4.23" |     "xml2js": "^0.4.23" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": {} |   "devDependencies": {} | ||||||
| } | } | ||||||
|  | |||||||
| @ -5,7 +5,6 @@ const date = require('date-and-time') | |||||||
| 
 | 
 | ||||||
| const Logger = require('./Logger') | const Logger = require('./Logger') | ||||||
| const { isObject } = require('./utils/index') | const { isObject } = require('./utils/index') | ||||||
| const resize = require('./utils/resizeImage') |  | ||||||
| const audioFileScanner = require('./utils/audioFileScanner') | const audioFileScanner = require('./utils/audioFileScanner') | ||||||
| 
 | 
 | ||||||
| const BookController = require('./controllers/BookController') | const BookController = require('./controllers/BookController') | ||||||
| @ -19,7 +18,7 @@ const BookFinder = require('./BookFinder') | |||||||
| const AuthorFinder = require('./AuthorFinder') | const AuthorFinder = require('./AuthorFinder') | ||||||
| 
 | 
 | ||||||
| class ApiController { | class ApiController { | ||||||
|   constructor(MetadataPath, db, scanner, auth, streamManager, rssFeeds, downloadManager, coverController, backupManager, watcher, emitter, clientEmitter) { |   constructor(MetadataPath, db, scanner, auth, streamManager, rssFeeds, downloadManager, coverController, backupManager, watcher, cacheManager, emitter, clientEmitter) { | ||||||
|     this.db = db |     this.db = db | ||||||
|     this.scanner = scanner |     this.scanner = scanner | ||||||
|     this.auth = auth |     this.auth = auth | ||||||
| @ -29,6 +28,7 @@ class ApiController { | |||||||
|     this.backupManager = backupManager |     this.backupManager = backupManager | ||||||
|     this.coverController = coverController |     this.coverController = coverController | ||||||
|     this.watcher = watcher |     this.watcher = watcher | ||||||
|  |     this.cacheManager = cacheManager | ||||||
|     this.emitter = emitter |     this.emitter = emitter | ||||||
|     this.clientEmitter = clientEmitter |     this.clientEmitter = clientEmitter | ||||||
|     this.MetadataPath = MetadataPath |     this.MetadataPath = MetadataPath | ||||||
| @ -62,12 +62,10 @@ class ApiController { | |||||||
|     this.router.get('/libraries/:id/authors', LibraryController.middleware.bind(this), LibraryController.getAuthors.bind(this)) |     this.router.get('/libraries/:id/authors', LibraryController.middleware.bind(this), LibraryController.getAuthors.bind(this)) | ||||||
|     this.router.post('/libraries/order', LibraryController.reorder.bind(this)) |     this.router.post('/libraries/order', LibraryController.reorder.bind(this)) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     // TEMP: Support old syntax for mobile app
 |     // TEMP: Support old syntax for mobile app
 | ||||||
|     this.router.get('/library/:id/audiobooks', LibraryController.middleware.bind(this), LibraryController.getBooksForLibrary.bind(this)) |     this.router.get('/library/:id/audiobooks', LibraryController.middleware.bind(this), LibraryController.getBooksForLibrary.bind(this)) | ||||||
|     this.router.get('/library/:id/search', LibraryController.middleware.bind(this), LibraryController.search.bind(this)) |     this.router.get('/library/:id/search', LibraryController.middleware.bind(this), LibraryController.search.bind(this)) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     //
 |     //
 | ||||||
|     // Book Routes
 |     // Book Routes
 | ||||||
|     //
 |     //
 | ||||||
| @ -83,7 +81,7 @@ class ApiController { | |||||||
|     this.router.patch('/books/:id/tracks', BookController.updateTracks.bind(this)) |     this.router.patch('/books/:id/tracks', BookController.updateTracks.bind(this)) | ||||||
|     this.router.get('/books/:id/stream', BookController.openStream.bind(this)) |     this.router.get('/books/:id/stream', BookController.openStream.bind(this)) | ||||||
|     this.router.post('/books/:id/cover', BookController.uploadCover.bind(this)) |     this.router.post('/books/:id/cover', BookController.uploadCover.bind(this)) | ||||||
|     this.router.get('/books/:id/cover', this.resizeCover.bind(this)) |     this.router.get('/books/:id/cover', BookController.getCover.bind(this)) | ||||||
|     this.router.patch('/books/:id/coverfile', BookController.updateCoverFromFile.bind(this)) |     this.router.patch('/books/:id/coverfile', BookController.updateCoverFromFile.bind(this)) | ||||||
| 
 | 
 | ||||||
|     // TEMP: Support old syntax for mobile app
 |     // TEMP: Support old syntax for mobile app
 | ||||||
| @ -91,7 +89,6 @@ class ApiController { | |||||||
|     this.router.get('/audiobook/:id', BookController.findOne.bind(this)) |     this.router.get('/audiobook/:id', BookController.findOne.bind(this)) | ||||||
|     this.router.get('/audiobook/:id/stream', BookController.openStream.bind(this)) |     this.router.get('/audiobook/:id/stream', BookController.openStream.bind(this)) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     //
 |     //
 | ||||||
|     // User Routes
 |     // User Routes
 | ||||||
|     //
 |     //
 | ||||||
| @ -104,7 +101,6 @@ class ApiController { | |||||||
|     this.router.get('/users/:id/listening-sessions', UserController.getListeningStats.bind(this)) |     this.router.get('/users/:id/listening-sessions', UserController.getListeningStats.bind(this)) | ||||||
|     this.router.get('/users/:id/listening-stats', UserController.getListeningStats.bind(this)) |     this.router.get('/users/:id/listening-stats', UserController.getListeningStats.bind(this)) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     //
 |     //
 | ||||||
|     // Collection Routes
 |     // Collection Routes
 | ||||||
|     //
 |     //
 | ||||||
| @ -123,7 +119,6 @@ class ApiController { | |||||||
|     this.router.get('/collection/:id', CollectionController.findOne.bind(this)) |     this.router.get('/collection/:id', CollectionController.findOne.bind(this)) | ||||||
|     this.router.delete('/collection/:id/book/:bookId', CollectionController.removeBook.bind(this)) |     this.router.delete('/collection/:id/book/:bookId', CollectionController.removeBook.bind(this)) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     //
 |     //
 | ||||||
|     // Current User Routes (Me)
 |     // Current User Routes (Me)
 | ||||||
|     //
 |     //
 | ||||||
| @ -140,7 +135,6 @@ class ApiController { | |||||||
|     this.router.patch('/user/audiobook/:id', MeController.updateAudiobookData.bind(this)) |     this.router.patch('/user/audiobook/:id', MeController.updateAudiobookData.bind(this)) | ||||||
|     this.router.patch('/user/settings', MeController.updateSettings.bind(this)) |     this.router.patch('/user/settings', MeController.updateSettings.bind(this)) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     //
 |     //
 | ||||||
|     // Backup Routes
 |     // Backup Routes
 | ||||||
|     //
 |     //
 | ||||||
| @ -176,31 +170,10 @@ class ApiController { | |||||||
|     this.router.get('/scantracks/:id', this.scanAudioTrackNums.bind(this)) |     this.router.get('/scantracks/:id', this.scanAudioTrackNums.bind(this)) | ||||||
| 
 | 
 | ||||||
|     this.router.post('/syncUserAudiobookData', this.syncUserAudiobookData.bind(this)) |     this.router.post('/syncUserAudiobookData', this.syncUserAudiobookData.bind(this)) | ||||||
|  | 
 | ||||||
|  |     this.router.post('/purgecache', this.purgeCache.bind(this)) | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   async resizeCover(req, res) { |  | ||||||
|     let { query: { width, height }, params: { id }, user } = req;  |  | ||||||
|     if (!user) { |  | ||||||
|       return res.sendStatus(403) |  | ||||||
|     } |  | ||||||
|     var audiobook = this.db.audiobooks.find(a => a.id === id) |  | ||||||
|     if (!audiobook) return res.sendStatus(404) |  | ||||||
| 
 |  | ||||||
|     // Check user can access this audiobooks library
 |  | ||||||
|     if (!req.user.checkCanAccessLibrary(audiobook.libraryId)) { |  | ||||||
|       return res.sendStatus(403) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|      |  | ||||||
|     res.type('image/jpeg'); |  | ||||||
| 
 |  | ||||||
|     if (width) width = parseInt(width) |  | ||||||
|     if (height) height = parseInt(height) |  | ||||||
| 
 |  | ||||||
|     return resize(audiobook.book.coverFullPath, width, height).pipe(res) |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|   async findBooks(req, res) { |   async findBooks(req, res) { | ||||||
|     var provider = req.query.provider || 'google' |     var provider = req.query.provider || 'google' | ||||||
|     var title = req.query.title || '' |     var title = req.query.title || '' | ||||||
| @ -485,6 +458,11 @@ class ApiController { | |||||||
|       this.clientEmitter(collection.userId, 'collection_updated', collection.toJSONExpanded(this.db.audiobooks)) |       this.clientEmitter(collection.userId, 'collection_updated', collection.toJSONExpanded(this.db.audiobooks)) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     // purge cover cache
 | ||||||
|  |     if (audiobook.cover) { | ||||||
|  |       await this.cacheManager.purgeCoverCache(audiobook.id) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     var audiobookJSON = audiobook.toJSONMinified() |     var audiobookJSON = audiobook.toJSONMinified() | ||||||
|     await this.db.removeEntity('audiobook', audiobook.id) |     await this.db.removeEntity('audiobook', audiobook.id) | ||||||
|     this.emitter('audiobook_removed', audiobookJSON) |     this.emitter('audiobook_removed', audiobookJSON) | ||||||
| @ -527,5 +505,14 @@ class ApiController { | |||||||
|     }) |     }) | ||||||
|     return listeningStats |     return listeningStats | ||||||
|   } |   } | ||||||
|  | 
 | ||||||
|  |   async purgeCache(req, res) { | ||||||
|  |     if (!req.user.isRoot) { | ||||||
|  |       return res.sendStatus(403) | ||||||
|  |     } | ||||||
|  |     Logger.info(`[ApiController] Purging all cache`) | ||||||
|  |     await this.cacheManager.purgeAll() | ||||||
|  |     res.sendStatus(200) | ||||||
|  |   } | ||||||
| } | } | ||||||
| module.exports = ApiController | module.exports = ApiController | ||||||
							
								
								
									
										75
									
								
								server/CacheManager.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								server/CacheManager.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,75 @@ | |||||||
|  | const Path = require('path') | ||||||
|  | const fs = require('fs-extra') | ||||||
|  | const stream = require('stream') | ||||||
|  | const resize = require('./utils/resizeImage') | ||||||
|  | const Logger = require('./Logger') | ||||||
|  | 
 | ||||||
|  | class CacheManager { | ||||||
|  |   constructor(MetadataPath) { | ||||||
|  |     this.MetadataPath = MetadataPath | ||||||
|  |     this.CachePath = Path.join(this.MetadataPath, 'cache') | ||||||
|  |     this.CoverCachePath = Path.join(this.CachePath, 'covers') | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   async handleCoverCache(res, audiobook, options = {}) { | ||||||
|  |     const format = options.format || 'webp' | ||||||
|  |     const width = options.width || 400 | ||||||
|  |     const height = options.height || null | ||||||
|  | 
 | ||||||
|  |     res.type(`image/${format}`) | ||||||
|  | 
 | ||||||
|  |     var path = Path.join(this.CoverCachePath, audiobook.id) + '.' + format | ||||||
|  | 
 | ||||||
|  |     // Cache exists
 | ||||||
|  |     if (await fs.pathExists(path)) { | ||||||
|  |       const r = fs.createReadStream(path) | ||||||
|  |       const ps = new stream.PassThrough() | ||||||
|  |       stream.pipeline(r, ps, (err) => { | ||||||
|  |         if (err) { | ||||||
|  |           console.log(err) | ||||||
|  |           return res.sendStatus(400) | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |       return ps.pipe(res) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Write cache
 | ||||||
|  |     await fs.ensureDir(this.CoverCachePath) | ||||||
|  |     var readStream = resize(audiobook.book.coverFullPath, width, height, format) | ||||||
|  |     var writeStream = fs.createWriteStream(path) | ||||||
|  |     writeStream.on('error', (e) => { | ||||||
|  |       Logger.error(`[CacheManager] Cache write error ${e.message}`) | ||||||
|  |     }) | ||||||
|  |     readStream.pipe(writeStream) | ||||||
|  | 
 | ||||||
|  |     readStream.pipe(res) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   purgeCoverCache(audiobookId) { | ||||||
|  |     var basepath = Path.join(this.CoverCachePath, audiobookId) | ||||||
|  |     // Remove both webp and jpg caches if exist
 | ||||||
|  |     var webpPath = basepath + '.webp' | ||||||
|  |     var jpgPath = basepath + '.jpg' | ||||||
|  |     return Promise.all([this.removeCache(webpPath), this.removeCache(jpgPath)]) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   removeCache(path) { | ||||||
|  |     if (!path) return false | ||||||
|  |     return fs.pathExists(path).then((exists) => { | ||||||
|  |       if (!exists) return false | ||||||
|  |       return fs.unlink(path).then(() => true).catch((err) => { | ||||||
|  |         Logger.error(`[CacheManager] Failed to remove cache "${path}"`, err) | ||||||
|  |         return false | ||||||
|  |       }) | ||||||
|  |     }) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   async purgeAll() { | ||||||
|  |     if (await fs.pathExists(this.CachePath)) { | ||||||
|  |       await fs.remove(this.CachePath).catch((error) => { | ||||||
|  |         Logger.error(`[CacheManager] Failed to remove cache dir "${this.CachePath}"`, error) | ||||||
|  |       }) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | module.exports = CacheManager | ||||||
| @ -10,8 +10,10 @@ const { CoverDestination } = require('./utils/constants') | |||||||
| const { downloadFile } = require('./utils/fileUtils') | const { downloadFile } = require('./utils/fileUtils') | ||||||
| 
 | 
 | ||||||
| class CoverController { | class CoverController { | ||||||
|   constructor(db, MetadataPath, AudiobookPath) { |   constructor(db, cacheManager, MetadataPath, AudiobookPath) { | ||||||
|     this.db = db |     this.db = db | ||||||
|  |     this.cacheManager = cacheManager | ||||||
|  | 
 | ||||||
|     this.MetadataPath = MetadataPath.replace(/\\/g, '/') |     this.MetadataPath = MetadataPath.replace(/\\/g, '/') | ||||||
|     this.BookMetadataPath = Path.posix.join(this.MetadataPath, 'books') |     this.BookMetadataPath = Path.posix.join(this.MetadataPath, 'books') | ||||||
|     this.AudiobookPath = AudiobookPath |     this.AudiobookPath = AudiobookPath | ||||||
| @ -115,6 +117,7 @@ class CoverController { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     await this.removeOldCovers(fullPath, extname) |     await this.removeOldCovers(fullPath, extname) | ||||||
|  |     await this.cacheManager.purgeCoverCache(audiobook.id) | ||||||
| 
 | 
 | ||||||
|     Logger.info(`[CoverController] Uploaded audiobook cover "${coverPath}" for "${audiobook.title}"`) |     Logger.info(`[CoverController] Uploaded audiobook cover "${coverPath}" for "${audiobook.title}"`) | ||||||
| 
 | 
 | ||||||
| @ -152,6 +155,7 @@ class CoverController { | |||||||
|       await fs.rename(temppath, coverFullPath) |       await fs.rename(temppath, coverFullPath) | ||||||
| 
 | 
 | ||||||
|       await this.removeOldCovers(fullPath, '.' + imgtype.ext) |       await this.removeOldCovers(fullPath, '.' + imgtype.ext) | ||||||
|  |       await this.cacheManager.purgeCoverCache(audiobook.id) | ||||||
| 
 | 
 | ||||||
|       Logger.info(`[CoverController] Downloaded audiobook cover "${coverPath}" from url "${url}" for "${audiobook.title}"`) |       Logger.info(`[CoverController] Downloaded audiobook cover "${coverPath}" from url "${url}" for "${audiobook.title}"`) | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -28,6 +28,7 @@ const StreamManager = require('./StreamManager') | |||||||
| const RssFeeds = require('./RssFeeds') | const RssFeeds = require('./RssFeeds') | ||||||
| const DownloadManager = require('./DownloadManager') | const DownloadManager = require('./DownloadManager') | ||||||
| const CoverController = require('./CoverController') | const CoverController = require('./CoverController') | ||||||
|  | const CacheManager = require('./CacheManager') | ||||||
| 
 | 
 | ||||||
| class Server { | class Server { | ||||||
|   constructor(PORT, UID, GID, CONFIG_PATH, METADATA_PATH, AUDIOBOOK_PATH) { |   constructor(PORT, UID, GID, CONFIG_PATH, METADATA_PATH, AUDIOBOOK_PATH) { | ||||||
| @ -47,15 +48,16 @@ class Server { | |||||||
|     this.auth = new Auth(this.db) |     this.auth = new Auth(this.db) | ||||||
|     this.backupManager = new BackupManager(this.MetadataPath, this.Uid, this.Gid, this.db) |     this.backupManager = new BackupManager(this.MetadataPath, this.Uid, this.Gid, this.db) | ||||||
|     this.logManager = new LogManager(this.MetadataPath, this.db) |     this.logManager = new LogManager(this.MetadataPath, this.db) | ||||||
|  |     this.cacheManager = new CacheManager(this.MetadataPath) | ||||||
|     this.watcher = new Watcher(this.AudiobookPath) |     this.watcher = new Watcher(this.AudiobookPath) | ||||||
|     this.coverController = new CoverController(this.db, this.MetadataPath, this.AudiobookPath) |     this.coverController = new CoverController(this.db, this.cacheManager, this.MetadataPath, this.AudiobookPath) | ||||||
|     this.scanner = new Scanner(this.AudiobookPath, this.MetadataPath, this.db, this.coverController, this.emitter.bind(this)) |     this.scanner = new Scanner(this.AudiobookPath, this.MetadataPath, this.db, this.coverController, this.emitter.bind(this)) | ||||||
|     this.scanner2 = new Scanner2(this.AudiobookPath, this.MetadataPath, this.db, this.coverController, this.emitter.bind(this)) |     this.scanner2 = new Scanner2(this.AudiobookPath, this.MetadataPath, this.db, this.coverController, this.emitter.bind(this)) | ||||||
| 
 | 
 | ||||||
|     this.streamManager = new StreamManager(this.db, this.MetadataPath, this.emitter.bind(this), this.clientEmitter.bind(this)) |     this.streamManager = new StreamManager(this.db, this.MetadataPath, this.emitter.bind(this), this.clientEmitter.bind(this)) | ||||||
|     this.rssFeeds = new RssFeeds(this.Port, this.db) |     this.rssFeeds = new RssFeeds(this.Port, this.db) | ||||||
|     this.downloadManager = new DownloadManager(this.db, this.MetadataPath, this.AudiobookPath, this.emitter.bind(this)) |     this.downloadManager = new DownloadManager(this.db, this.MetadataPath, this.AudiobookPath, this.emitter.bind(this)) | ||||||
|     this.apiController = new ApiController(this.MetadataPath, this.db, this.scanner, this.auth, this.streamManager, this.rssFeeds, this.downloadManager, this.coverController, this.backupManager, this.watcher, this.emitter.bind(this), this.clientEmitter.bind(this)) |     this.apiController = new ApiController(this.MetadataPath, this.db, this.scanner, this.auth, this.streamManager, this.rssFeeds, this.downloadManager, this.coverController, this.backupManager, this.watcher, this.cacheManager, this.emitter.bind(this), this.clientEmitter.bind(this)) | ||||||
|     this.hlsController = new HlsController(this.db, this.scanner, this.auth, this.streamManager, this.emitter.bind(this), this.streamManager.StreamsPath) |     this.hlsController = new HlsController(this.db, this.scanner, this.auth, this.streamManager, this.emitter.bind(this), this.streamManager.StreamsPath) | ||||||
| 
 | 
 | ||||||
|     Logger.logManager = this.logManager |     Logger.logManager = this.logManager | ||||||
|  | |||||||
| @ -72,7 +72,7 @@ class StreamManager { | |||||||
|       if (!dirs || !dirs.length) return true |       if (!dirs || !dirs.length) return true | ||||||
| 
 | 
 | ||||||
|       await Promise.all(dirs.map(async (dirname) => { |       await Promise.all(dirs.map(async (dirname) => { | ||||||
|         if (dirname !== 'streams' && dirname !== 'books' && dirname !== 'downloads' && dirname !== 'backups' && dirname !== 'logs') { |         if (dirname !== 'streams' && dirname !== 'books' && dirname !== 'downloads' && dirname !== 'backups' && dirname !== 'logs' && dirname !== 'cache') { | ||||||
|           var fullPath = Path.join(this.MetadataPath, dirname) |           var fullPath = Path.join(this.MetadataPath, dirname) | ||||||
|           Logger.warn(`Removing OLD Orphan Stream ${dirname}`) |           Logger.warn(`Removing OLD Orphan Stream ${dirname}`) | ||||||
|           return fs.remove(fullPath) |           return fs.remove(fullPath) | ||||||
|  | |||||||
| @ -1,4 +1,5 @@ | |||||||
| const Logger = require('../Logger') | const Logger = require('../Logger') | ||||||
|  | const { reqSupportsWebp } = require('../utils/index') | ||||||
| 
 | 
 | ||||||
| class BookController { | class BookController { | ||||||
|   constructor() { } |   constructor() { } | ||||||
| @ -38,6 +39,12 @@ class BookController { | |||||||
|     } |     } | ||||||
|     var audiobook = this.db.audiobooks.find(a => a.id === req.params.id) |     var audiobook = this.db.audiobooks.find(a => a.id === req.params.id) | ||||||
|     if (!audiobook) return res.sendStatus(404) |     if (!audiobook) return res.sendStatus(404) | ||||||
|  | 
 | ||||||
|  |     // Book has cover and update is removing cover then purge cache
 | ||||||
|  |     if (audiobook.cover && req.body.book && (req.body.book.cover === '' || req.body.book.cover === null)) { | ||||||
|  |       await this.cacheManager.purgeCoverCache(audiobook.id) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     var hasUpdates = audiobook.update(req.body) |     var hasUpdates = audiobook.update(req.body) | ||||||
|     if (hasUpdates) { |     if (hasUpdates) { | ||||||
|       await this.db.updateAudiobook(audiobook) |       await this.db.updateAudiobook(audiobook) | ||||||
| @ -222,5 +229,24 @@ class BookController { | |||||||
|     if (updated) res.status(200).send('Cover updated successfully') |     if (updated) res.status(200).send('Cover updated successfully') | ||||||
|     else res.status(200).send('No update was made to cover') |     else res.status(200).send('No update was made to cover') | ||||||
|   } |   } | ||||||
|  | 
 | ||||||
|  |   // GET api/books/:id/cover
 | ||||||
|  |   async getCover(req, res) { | ||||||
|  |     let { query: { width, height, format }, params: { id } } = req | ||||||
|  |     var audiobook = this.db.audiobooks.find(a => a.id === id) | ||||||
|  |     if (!audiobook || !audiobook.book.coverFullPath) return res.sendStatus(404) | ||||||
|  | 
 | ||||||
|  |     // Check user can access this audiobooks library
 | ||||||
|  |     if (!req.user.checkCanAccessLibrary(audiobook.libraryId)) { | ||||||
|  |       return res.sendStatus(403) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const options = { | ||||||
|  |       format: format || (reqSupportsWebp(req) ? 'webp' : 'jpg'), | ||||||
|  |       height: height ? parseInt(height) : null, | ||||||
|  |       width: width ? parseInt(width) : null | ||||||
|  |     } | ||||||
|  |     return this.cacheManager.handleCoverCache(res, audiobook, options) | ||||||
|  |   } | ||||||
| } | } | ||||||
| module.exports = new BookController() | module.exports = new BookController() | ||||||
| @ -101,4 +101,9 @@ function secondsToTimestamp(seconds, includeMs = false) { | |||||||
| } | } | ||||||
| module.exports.secondsToTimestamp = secondsToTimestamp | module.exports.secondsToTimestamp = secondsToTimestamp | ||||||
| 
 | 
 | ||||||
| module.exports.msToTimestamp = (ms, includeMs) => secondsToTimestamp(ms / 1000, includeMs) | module.exports.msToTimestamp = (ms, includeMs) => secondsToTimestamp(ms / 1000, includeMs) | ||||||
|  | 
 | ||||||
|  | module.exports.reqSupportsWebp = (req) => { | ||||||
|  |   if (!req || !req.headers || !req.headers.accept) return false | ||||||
|  |   return req.headers.accept.includes('image/webp') || req.headers.accept === '*/*' | ||||||
|  | } | ||||||
| @ -1,13 +1,13 @@ | |||||||
| const sharp = require('sharp') | const sharp = require('sharp') | ||||||
| const fs = require('fs') | const fs = require('fs') | ||||||
| 
 | 
 | ||||||
| function resize(filePath, width, height) { | function resize(filePath, width, height, format = 'webp') { | ||||||
|   const readStream = fs.createReadStream(filePath); |   const readStream = fs.createReadStream(filePath); | ||||||
|   let sharpie = sharp() |   let sharpie = sharp() | ||||||
|   sharpie.toFormat('jpeg') |   sharpie.toFormat(format) | ||||||
| 
 | 
 | ||||||
|   if (width || height) { |   if (width || height) { | ||||||
|     sharpie.resize(width, height) |     sharpie.resize(width, height, { withoutEnlargement: true }) | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   return readStream.pipe(sharpie) |   return readStream.pipe(sharpie) | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user