Получение доступа к кешу медиаплеера
Я хочу переместить постепенно потоковый mp3 файл на SD-карту после его полной загрузки. Есть ли способ достичь этого.
Я видел, что MediaPlayer
полностью загружает весь файл во время прогрессивной потоковой передачи, а затем мы можем искать любую часть файла. Я хочу переместить полностью потоковый файл на внешнее хранилище, чтобы в будущем воспроизведение не теряло данных и батареи.
Ответы
Ответ 1
Комментарий к исходному сообщению указывает вам в правильном направлении, но я подумал, что может быть полезно изложить немного...
Что я сделал - это создать легкий прокси-сервер, используя Naga и Apache HTTP-библиотеки. Там должно быть много примеров, чтобы получить основы этой части. Предоставьте MediaPlayer соответствующий URL-адрес локального хоста, чтобы он открыл сокет для вашего прокси-сервера. Когда MediaPlayer делает запрос, используйте прокси-сервер для отправки эквивалентного запроса на реальный медиа-хост. Вы получите данные byte [] в прокси-пакетеReceived, который я использую для создания HttpGet и отправьте его на свой путь с AndroidHttpClient.
Вы получите HttpResponse, и вы можете использовать HttpEntity внутри для доступа к данным потокового байта. Я использую ReadableByteChannel, например:
HttpEntityWrapper entity = (HttpEntityWrapper)response.getEntity();
ReadableByteChannel src = Channels.newChannel(entity.getContent());
Сделайте все, что захотите, с данными при их чтении (например, кешируйте его в файл на SD-карте). Чтобы передать правильный материал MediaPlayer, получите SocketChannel из Socket клиента, сначала напишите заголовки ответов непосредственно на этот канал, а затем приступите к записи данных байта сущности. Я использую NIO ByteBuffer в цикле while (клиент - Socket, а buffer - ByteBuffer).
int read, written;
SocketChannel dst = client.getChannel();
while (dst.isConnected() &&
dst.isOpen() &&
src.isOpen() &&
(read = src.read(buffer)) >= 0) {
try {
buffer.flip();
// This is one point where you can access the stream data.
// Just remember to reset the buffer position before trying
// to write to the destination.
if (buffer.hasRemaining()) {
written = dst.write(buffer);
// If the player isn't reading, wait a bit.
if (written == 0) Thread.sleep(15);
buffer.compact();
}
}
catch (IOException ex) {
// handle error
}
}
Возможно, вам придется изменить заголовок узла в ответе, прежде чем передавать его вместе с игроком, чтобы он выглядел так, как будто ваш прокси-сервер является отправителем, но я имею дело с проприетарной реализацией MediaPlayer, поэтому поведение может быть немного другой. Надеюсь, что это поможет.
Ответ 2
Идея заключается в создании прокси-сервера, с которого может считываться медиаплеер, вместо чтения данных непосредственно из Интернета.
Я использовал danikula/AndroidVideoCache, который очень просто построить/использовать.
Я использовал его для Audio not Video, но его все равно.
Ответ 3
Поздно, но я обнаружил, что большинство людей по-прежнему нуждаются в решении. Мое решение основано на JakeWharton DiskLruCache.
Нам нужны две вещи
Шаг 1:
import android.content.Context;
import android.os.AsyncTask;
import org.apache.commons.io.IOUtils;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.URL;
// you can use FileDescriptor as
// extends AsyncTask<String, Void, FileDescriptor>
public class AudioStreamWorkerTask extends AsyncTask<String, Void, FileInputStream> {
private OnCacheCallback callback = null;
private Context context = null;
public AudioStreamWorkerTask(Context context, OnCacheCallback callback) {
this.context = context;
this.callback = callback;
}
@Override
protected FileInputStream doInBackground(String... params) {
String data = params[0];
// Application class where i did open DiskLruCache
DiskLruCache cache = MyApplication.getDiskCache(context);
if (cache == null)
return null;
String key = hashKeyForDisk(data);
final int DISK_CACHE_INDEX = 0;
long currentMaxSize = cache.getMaxSize();
float percentageSize = Math.round((cache.size() * 100.0f) / currentMaxSize);
if (percentageSize >= 90) // cache size reaches 90%
cache.setMaxSize(currentMaxSize + (10 * 1024 * 1024)); // increase size to 10MB
try {
DiskLruCache.Snapshot snapshot = cache.get(key);
if (snapshot == null) {
Log.i(getTag(), "Snapshot is not available downloading...");
DiskLruCache.Editor editor = cache.edit(key);
if (editor != null) {
if (downloadUrlToStream(data, editor.newOutputStream(DISK_CACHE_INDEX)))
editor.commit();
else
editor.abort();
}
snapshot = cache.get(key);
} else
Log.i(getTag(), "Snapshot found sending");
if (snapshot != null)
return (FileInputStream) snapshot.getInputStream(DISK_CACHE_INDEX);
} catch (IOException e) {
e.printStackTrace();
}
Log.i(getTag(), "File stream is null");
return null;
}
@Override
protected void onPostExecute(FileInputStream fileInputStream) {
super.onPostExecute(fileInputStream);
if (callback != null) {
if (fileInputStream != null)
callback.onSuccess(fileInputStream);
else
callback.onError();
}
callback = null;
context = null;
}
public boolean downloadUrlToStream(String urlString, OutputStream outputStream) {
HttpURLConnection urlConnection = null;
try {
final URL url = new URL(urlString);
urlConnection = (HttpURLConnection) url.openConnection();
InputStream stream = urlConnection.getInputStream();
// you can use BufferedInputStream and BufferOuInputStream
IOUtils.copy(stream, outputStream);
IOUtils.closeQuietly(outputStream);
IOUtils.closeQuietly(stream);
Log.i(getTag(), "Stream closed all done");
return true;
} catch (final IOException e) {
e.printStackTrace();
} finally {
if (urlConnection != null)
IOUtils.close(urlConnection);
}
return false;
}
private String getTag() {
return getClass().getSimpleName();
}
private String hashKeyForDisk(String key) {
String cacheKey;
try {
final MessageDigest mDigest = MessageDigest.getInstance("MD5");
mDigest.update(key.getBytes());
cacheKey = bytesToHexString(mDigest.digest());
} catch (NoSuchAlgorithmException e) {
cacheKey = String.valueOf(key.hashCode());
}
return cacheKey;
}
private String bytesToHexString(byte[] bytes) {
// http://stackoverflow.com/info/332079
StringBuilder sb = new StringBuilder();
for (byte aByte : bytes) {
String hex = Integer.toHexString(0xFF & aByte);
if (hex.length() == 1)
sb.append('0');
sb.append(hex);
}
return sb.toString();
}
}
Шаг 2:
public interface OnCacheCallback {
void onSuccess(FileInputStream stream);
void onError();
}
Пример
final String path = "http://www.example.com/test.mp3";
new AudioStreamWorkerTask (TestActivity.this, new OnCacheCallback() {
@Override
public void onSuccess(FileInputStream fileInputStream) {
Log.i(getClass().getSimpleName() + ".MediaPlayer", "now playing...");
if (fileInputStream != null) {
// reset media player here if necessary
mediaPlayer = new MediaPlayer();
try {
mediaPlayer.setDataSource(fileInputStream.getFD());
mediaPlayer.prepare();
mediaPlayer.setVolume(1f, 1f);
mediaPlayer.setLooping(false);
mediaPlayer.start();
fileInputStream.close();
} catch (IOException | IllegalStateException e) {
e.printStackTrace();
}
} else {
Log.e(getClass().getSimpleName() + ".MediaPlayer", "fileDescriptor is not valid");
}
}
@Override
public void onError() {
Log.e(getClass().getSimpleName() + ".MediaPlayer", "Can't play audio file");
}
}).execute(path);
Примечание:
Это проверенный, но грубый образец для кеширования аудиофайлов, могут возникнуть проблемы, если вы найдете что-нибудь, пожалуйста, сообщите мне:)