这是我的问题:我正在编写一个laravel后端,必须提供一个必须使用android标准媒体播放器重现的mp3文件。
对于laravel后端,我需要使用JWT来处理身份验证,因此在每个请求标头上我必须将“授权”字段设置为“ Bearer {token} ”。
laravel路由是“ / songs / {id} “并以这种方式处理:
public function getSong(Song $song) {
$file = new File(storage_path()."/songs/".$song->path.".mp3");
$headers = array();
$headers['Content-Type'] = 'audio/mpeg, audio/x-mpeg, audio/x-mpeg-3, audio/mpeg3';
$headers['Content-Length'] = $file->getSize();
$headers['Content-Transfer-Encoding'] = 'binary';
$headers['Accept-Range'] = 'bytes';
$headers['Cache-Control'] = 'must-revalidate, post-check=0, pre-check=0';
$headers['Connection'] = 'Keep-Alive';
$headers['Content-Disposition'] = 'attachment; filename="'.$song->path.'.mp3"';
$user = \Auth::user();
if($user->activated_at) {
return Response::download($file, $song->path, $headers);
}
\App::abort(400);
}
在android方面,我正在使用MediaPlayer以这种方式传输mp3文件:
media_player = new MediaPlayer();
try {
media_player.setAudioStreamType(AudioManager.STREAM_MUSIC);
String token = getSharedPreferences("p_shared", MODE_PRIVATE).getString("token", null);
Map<String, String> headers = new HashMap<>();
headers.put("Authorization", "Bearer " + token);
media_player.setDataSource(
getApplicationContext(),
Uri.parse(ConnectionHelper.SERVER + "/songs/" + song.getId()),
headers
);
} catch (IOException e) {
finish();
Toast.makeText(
Round.this,
"Some error occurred. Retry in some minutes.",
Toast.LENGTH_SHORT
).show();
}
media_player.setOnCompletionListener(this);
media_player.setOnErrorListener(this);
media_player.setOnPreparedListener(this);
但每次执行代码时,我都会在错误监听器上获得额外代码 -1005 ,这意味着 ERROR_CONNECTION_LOST 。
答案 0 :(得分:5)
问题: Response :: download(...)不会产生流,因此我无法提供.mp3文件。
解决方案: 正如Symfony HttpFoundation doc.在服务文件段落中所说:
"if you are serving a static file, you can use a BinaryFileResponse"
我需要提供的.mp3文件是服务器中的静态文件并存储在“ / storage / songs / ”中,所以我决定使用 BinaryFileResponse ,并且服务.mp3的方法成为:
use Symfony\Component\HttpFoundation\BinaryFileResponse;
[...]
public function getSong(Song $song) {
$path = storage_path().DIRECTORY_SEPARATOR."songs".DIRECTORY_SEPARATOR.$song->path.".mp3");
$user = \Auth::user();
if($user->activated_at) {
$response = new BinaryFileResponse($path);
BinaryFileResponse::trustXSendfileTypeHeader();
return $response;
}
\App::abort(400);
}
BinaryFileResponse自动处理请求并允许您完全提供文件(通过只使用Http 200代码发出一个请求)或分割为较慢的连接(使用Http 206代码的更多请求和使用200代码的最终请求)。
如果您有 mod_xsendfile ,您可以通过添加以下内容来更快地使用流式传输:
BinaryFileResponse::trustXSendfileTypeHeader();
安卓代码无需更改即可流式传输文件。