Java tutorial
/* * Copyright 2013 Karsten Patzwaldt * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package org.muckebox.android.net; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.nio.ByteBuffer; import java.nio.channels.ClosedByInterruptException; import org.apache.http.HttpEntity; import org.apache.http.HttpResponse; import org.apache.http.StatusLine; import org.apache.http.client.methods.HttpGet; import org.muckebox.android.Muckebox; import org.muckebox.android.utils.BufferUtils; import org.muckebox.android.utils.Preferences; import android.content.Context; import android.os.Handler; import android.util.Log; public class DownloadRunnable implements Runnable { private static final String LOG_TAG = "DownloadRunnable"; private final static int BUFFER_SIZE = 32 * 1024; private final static int RETRY_COUNT = 5; private final static int RETRY_INTERVAL = 5; private boolean mTranscodingEnabled; private String mTranscodingType; private String mTranscodingQuality; private long mTrackId; private Handler mHandler; private String mOutputPath; private int mBytesTotal = 0; private FileOutputStream mOutputStream = null; public static final int MESSAGE_DOWNLOAD_STARTED = 1; public static final int MESSAGE_DATA_RECEIVED = 2; public static final int MESSAGE_DOWNLOAD_FINISHED = 3; public static final int MESSAGE_DOWNLOAD_FAILED = 4; public static final int MESSAGE_DOWNLOAD_INTERRUPTED = 5; public static class FileInfo { public String mimeType; public String path; } public static class Result { public long trackId; public String path; public String mimeType; public int size; public boolean transcodingEnabled; public String transcodingType; public String transcodingQuality; } public static class Chunk { public long bytesTotal; public ByteBuffer buffer; } DownloadRunnable(long trackId, Handler resultHandler) { init(trackId, resultHandler, null); } public DownloadRunnable(long trackId, Handler resultHandler, String outputPath) { init(trackId, resultHandler, outputPath); } private void init(long trackId, Handler resultHandler, String outputPath) { mTranscodingEnabled = Preferences.getTranscodingEnabled(); mTranscodingType = Preferences.getTranscodingType(); mTranscodingQuality = Preferences.getTranscodingQuality(); mTrackId = trackId; mHandler = resultHandler; mOutputPath = outputPath; } private String getDownloadUrl() throws IOException { String url; String idString = Long.toString(mTrackId); if (mTranscodingEnabled) { url = ApiHelper.getApiUrl("stream", idString, new String[] { "format", "quality" }, new String[] { mTranscodingType, mTranscodingQuality }); } else { url = ApiHelper.getApiUrl("stream", idString, null, null); } return url; } private void handleReceivedData(ByteBuffer data) throws IOException { if (!BufferUtils.isEmpty(data)) { mBytesTotal += data.position(); if (mOutputStream != null) { mOutputStream.write(data.array(), 0, data.position()); } Chunk chunk = new Chunk(); chunk.bytesTotal = mBytesTotal; chunk.buffer = data; if (mHandler != null) mHandler.sendMessage(mHandler.obtainMessage(MESSAGE_DATA_RECEIVED, (int) mTrackId, 0, chunk)); } } private void ensureOutputStream(String mimeType) throws IOException { if (mOutputStream == null) { if (mOutputPath != null) { mOutputStream = Muckebox.getAppContext().openFileOutput(mOutputPath, Context.MODE_PRIVATE); mOutputStream.flush(); Log.d(LOG_TAG, "Saving to " + mOutputPath); } FileInfo fileInfo = makeFileInfo(mimeType, mOutputPath); if (mHandler != null) mHandler.sendMessage(mHandler.obtainMessage(MESSAGE_DOWNLOAD_STARTED, (int) mTrackId, 0, fileInfo)); } } public void closeOutputStream() { if (mOutputStream != null) { try { mOutputStream.close(); mOutputStream = null; } catch (IOException eInner) { Log.e(LOG_TAG, "Yo dawg, i heard ya like exceptions!"); eInner.printStackTrace(); } } } private Result makeResult(String mimeType, int bytesTotal) { Result res = new Result(); res.trackId = mTrackId; res.path = mOutputPath; res.mimeType = mimeType; res.size = bytesTotal; res.transcodingEnabled = mTranscodingEnabled; res.transcodingType = mTranscodingType; res.transcodingQuality = mTranscodingQuality; return res; } private FileInfo makeFileInfo(String mimeType, String path) { FileInfo ret = new FileInfo(); ret.mimeType = mimeType; ret.path = path; return ret; } public boolean downloadFile() throws IOException { MuckeboxHttpClient httpClient = new MuckeboxHttpClient(); HttpGet httpGet = null; try { httpGet = new HttpGet(getDownloadUrl()); httpGet.addHeader("Range", String.format("bytes=%d-", mBytesTotal)); HttpResponse httpResponse = httpClient.execute(httpGet); StatusLine statusLine = httpResponse.getStatusLine(); if (statusLine.getStatusCode() != 200) { Log.e(LOG_TAG, "HTTP request failed, response: '" + statusLine.getReasonPhrase() + "'"); return false; } HttpEntity httpEntity = httpResponse.getEntity(); String mimeType = httpEntity.getContentType().getValue(); InputStream is = httpEntity.getContent(); Log.i(LOG_TAG, "Downloading from " + httpGet.getURI() + " (offset " + mBytesTotal + ")"); ensureOutputStream(mimeType); while (true) { ByteBuffer buf = ByteBuffer.allocate(BUFFER_SIZE); boolean eosReached = BufferUtils.readIntoBuffer(is, buf); handleReceivedData(buf); if (eosReached) { Log.v(LOG_TAG, "Download finished!"); Result res = makeResult(mimeType, mBytesTotal); if (mHandler != null) mHandler.sendMessage( mHandler.obtainMessage(MESSAGE_DOWNLOAD_FINISHED, (int) mTrackId, 0, res)); return true; } if (Thread.interrupted()) { throw new ClosedByInterruptException(); } } } finally { if (httpGet != null) httpGet.abort(); if (httpClient != null) httpClient.destroy(); } } public void run() { int retriesLeft = RETRY_COUNT; int lastProgress = mBytesTotal; try { while (true) { try { if (downloadFile()) { break; } else { throw new IOException(); } } catch (ClosedByInterruptException e) { throw e; } catch (IOException e) { if (retriesLeft == 0) throw e; if (mBytesTotal > lastProgress) { lastProgress = mBytesTotal; retriesLeft = RETRY_COUNT; } else { retriesLeft--; } Log.w(LOG_TAG, "Download failed, retrying after " + RETRY_INTERVAL + " seconds"); Thread.sleep(RETRY_INTERVAL * 1000); } } } catch (InterruptedException e) { Log.w(LOG_TAG, "Download interrupted"); handleFailure(MESSAGE_DOWNLOAD_INTERRUPTED); } catch (ClosedByInterruptException e) { Log.w(LOG_TAG, "Download interrupted"); handleFailure(MESSAGE_DOWNLOAD_INTERRUPTED); } catch (IOException e) { Log.e(LOG_TAG, "Error downloading"); e.printStackTrace(); handleFailure(MESSAGE_DOWNLOAD_FAILED); } finally { closeOutputStream(); } } void handleFailure(int messageType) { closeOutputStream(); Muckebox.getAppContext().deleteFile(mOutputPath); if (mHandler != null) mHandler.sendMessage(mHandler.obtainMessage(messageType, (int) mTrackId, 0)); } }