From 0b860e0d40aef065f5dd2168072049c6430de896 Mon Sep 17 00:00:00 2001 From: Chance Zibolski Date: Tue, 20 Aug 2024 19:00:29 -0700 Subject: [PATCH] Fix Content-Type header when browser user-agent is from an Apple mobile device Fixes #3310 Signed-off-by: Chance Zibolski --- server/controllers/LibraryItemController.js | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/server/controllers/LibraryItemController.js b/server/controllers/LibraryItemController.js index 0ec2b49e73..9a87f7a798 100644 --- a/server/controllers/LibraryItemController.js +++ b/server/controllers/LibraryItemController.js @@ -1,13 +1,14 @@ const { Request, Response, NextFunction } = require('express') const Path = require('path') const fs = require('../libs/fsExtra') +const uaParserJs = require('../libs/uaParser') const Logger = require('../Logger') const SocketAuthority = require('../SocketAuthority') const Database = require('../Database') const zipHelpers = require('../utils/zipHelpers') const { reqSupportsWebp } = require('../utils/index') -const { ScanResult } = require('../utils/constants') +const { ScanResult, AudioMimeType } = require('../utils/constants') const { getAudioMimeTypeFromExtname, encodeUriPath } = require('../utils/fileUtils') const LibraryItemScanner = require('../scanner/LibraryItemScanner') const AudioFileScanner = require('../scanner/AudioFileScanner') @@ -799,6 +800,7 @@ class LibraryItemController { */ async downloadLibraryFile(req, res) { const libraryFile = req.libraryFile + const ua = uaParserJs(req.headers['user-agent']) if (!req.user.canDownload) { Logger.error(`[LibraryItemController] User "${req.user.username}" without download permission attempted to download file "${libraryFile.metadata.path}"`) @@ -814,8 +816,15 @@ class LibraryItemController { } // Express does not set the correct mimetype for m4b files so use our defined mimetypes if available - const audioMimeType = getAudioMimeTypeFromExtname(Path.extname(libraryFile.metadata.path)) + let audioMimeType = getAudioMimeTypeFromExtname(Path.extname(libraryFile.metadata.path)) if (audioMimeType) { + // Work-around for Apple devices mishandling Content-Type on mobile browsers: + // https://github.com/advplyr/audiobookshelf/issues/3310 + // We actually need to check for Webkit on Apple mobile devices because this issue impacts all browsers on iOS/iPadOS/etc, not just Safari. + const isAppleMobileBrowser = ua.device.vendor === 'Apple' && ua.device.type === 'mobile' && ua.engine.name === 'WebKit' + if (isAppleMobileBrowser && audioMimeType === AudioMimeType.M4B) { + audioMimeType = 'audio/m4b' + } res.setHeader('Content-Type', audioMimeType) }