org.muckebox.android.net.DownloadRunnable.java Source code

Java tutorial

Introduction

Here is the source code for org.muckebox.android.net.DownloadRunnable.java

Source

/*   
 * 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));
    }
}