All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.coolerfall.download.DownloadDispatcher Maven / Gradle / Ivy

package com.coolerfall.download;

import android.os.Process;
import android.util.Log;

import com.coolerfall.download.DownloadRequest.DownloadState;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.concurrent.BlockingQueue;

import static com.coolerfall.download.DownloadManager.HTTP_ERROR_NETWORK;
import static com.coolerfall.download.DownloadManager.HTTP_ERROR_SIZE;
import static com.coolerfall.download.DownloadManager.HTTP_INVALID;
import static java.net.HttpURLConnection.HTTP_INTERNAL_ERROR;
import static java.net.HttpURLConnection.HTTP_MOVED_PERM;
import static java.net.HttpURLConnection.HTTP_MOVED_TEMP;
import static java.net.HttpURLConnection.HTTP_OK;
import static java.net.HttpURLConnection.HTTP_PARTIAL;
import static java.net.HttpURLConnection.HTTP_SEE_OTHER;
import static java.net.HttpURLConnection.HTTP_UNAVAILABLE;

/**
 * Download dispatcher: used to dispatch downloader, this is desinged according to
 * NetworkDispatcher in Android-Volley.
 * 
 * @author Vincent Cheung
 * @since  Nov. 24, 2014
 */
public class DownloadDispatcher extends Thread {
	private static final String TAG = DownloadDispatcher.class.getSimpleName();
	
	/** sleep time before download */
	private static final int SLEEP_BEFORE_DOWNLOAD = 1500;
	
	/** sleep time before restrying download */
	private static final int SLEEP_BEFORE_RETRY = 3500;
	
	/** http connection time out */
	private static final int DEFAUL_TIME_OUT = 20 * 1000;
	
	/** buffer size used in data tranfering */
	private static final int BUFFER_SIZE = 4096;
	
	/** maxmum of redirections, should not more than 7 */
	private static final int MAX_REDIRECTION = 5;
	
	/** range not satisfiable */
	private static final int HTTP_REQUESTED_RANGE_NOT_SATISFIABLE = 416;
	
	/** http temp redirect */
	private static final int HTTP_TEMP_REDIRECT = 307;

	/** accept encoding in request header */
	private static final String ACCPET_ENCODING = "Accept-Encoding";
	
	/** transfer encoding in header */
	private static final String TRANSFER_ENCODING = "Transfer-Encoding";
	
	/** content length in header */
	private static final String CONTENT_LENGTH = "Content-Length";
	
	/** redirect location */
	private static final String LOCATION = "Location";
	
	/** end of input stream */
	private static final String END_OF_STREAM = "unexpected end of stream";
	
	/** default thread name */
	private static final String DEFAULT_THREAD_NAME = "DownloadDispatcher";
	
	/** idle thread name */
	private static final String IDLE_THREAD_NAME = "DownloadDispatcher-Idle";
	
	/** the queue of download reuqests */
	private BlockingQueue mQueue;
	
	/** to deliver callback on main thread */
	private DownloadDelivery mDelivery;
	
	/** redirection time happens in this request */
	private int mRedirectionCount = 0;
	
	/* save total bytes in case */
	private long mTotalBytes = 0;
	
	/** used to tell us this dispatcher has dead. */
	private volatile boolean mQuit = false;
	
	/**
	 * Default constructor, with queue and delivery.
	 */
	public DownloadDispatcher(BlockingQueue queue, DownloadDelivery delivery) {
		mQueue = queue;
		mDelivery = delivery;
		
		/* set thread name to idle */
		setName(IDLE_THREAD_NAME);
	}
	
	@Override
	public void run() {
		Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND);
		DownloadRequest request = null;
		
		while (true) {
			try {
				setName(IDLE_THREAD_NAME);
				request = mQueue.take();
				sleep(SLEEP_BEFORE_DOWNLOAD);
				setName(DEFAULT_THREAD_NAME);
				
				mTotalBytes = 0;
				mRedirectionCount = 0;
				
				/* start download */
				executeDownload(request);
			} catch (InterruptedException e) {
				/* we may have been interrupted because it was time to quit */
				if (mQuit) {
					if (request != null) {
						request.finish();
					}

					return;
				}
			}
		}
	}
	
	/** update download state */
	private void updateState(DownloadRequest request, DownloadState state) {
		request.setDownloadState(state);
	}
	
	/** update download start state */
	private void updateStart(DownloadRequest request, long totalBytes) {
		/* if the request has failed before, donnot deliver callback */
		if (request.getDownloadState() == DownloadState.FAILURE) {
			updateState(request, DownloadState.RUNNING);
			return;
		}
		
		/* set the download state of this request as running */
		updateState(request, DownloadState.RUNNING);
		mDelivery.postStart(request, totalBytes);
	}
	
	/** update download retrying */
	private void updateRetry(DownloadRequest request) {
		mDelivery.postRetry(request);
	}
	
	/** update download progress */
	private void updateProgress(DownloadRequest request, long bytesWritten, long totalBytes) {
		if (!request.isCanceled()) {
			mDelivery.postProgress(request, bytesWritten, totalBytes);
		}
	}

	/** update download success */
	@SuppressWarnings("ResultOfMethodCallIgnored")
	private void updateSuccess(DownloadRequest request) {
		updateState(request, DownloadState.SUCCESSFUL);
		
		/* notify the request download finish */
		request.finish();

		File file = new File(request.getTmpDestinationPath());
		if (file.exists()) {
			file.renameTo(new File(request.getDestFilePath()));
		}

		/* deliver success message */
		mDelivery.postSuccess(request);
	}
	
	/** update download failure */
	private void updateFailure(DownloadRequest request, int statusCode, String errMsg) {
		updateState(request, DownloadState.FAILURE);

		/* if the status code is 0, may be cause by the net error */
		if ((statusCode == HTTP_INVALID || statusCode == HTTP_ERROR_SIZE) && 
				request.getRetryTime() >= 0) {
			try {
				/* update progress in case */
				long bytesWritten = new File(request.getTmpDestinationPath()).length();
				updateProgress(request, bytesWritten, mTotalBytes);
				
				/* sleep a while before retrying */
				sleep(SLEEP_BEFORE_RETRY);
			} catch (InterruptedException e) {
				/* we may have been interrupted because it was time to quit */
				if (mQuit) {
					request.finish();

					return;
				}
			}
			
			/* retry downloading */
			if (!request.isCanceled()) {
				updateRetry(request);
				executeDownload(request);
			}
			
			return;
		}
		
		/* notify the request that downloading has finished */
		request.finish();
		
		/* deliver failure message */
		mDelivery.postFailure(request, statusCode, errMsg);
	}
	
	/** execute downloading */
	private void executeDownload(DownloadRequest request) {
		if (Thread.currentThread().isInterrupted()) {
			return;
		}

		//TODO: handle https url
		/* use http url connection to download file */
		HttpURLConnection conn = null;
		
		try {
			URL url = new URL(request.getUrl());
			conn = (HttpURLConnection) url.openConnection();
			
			/* config http url connection */
			conn.setInstanceFollowRedirects(false);
			conn.setUseCaches(false);
			conn.setRequestProperty(ACCPET_ENCODING, "identity");
			conn.setConnectTimeout(DEFAUL_TIME_OUT);
			conn.setReadTimeout(DEFAUL_TIME_OUT);

			/* if the file existed, restart from breakpoint */
			File file = new File(request.getTmpDestinationPath());
			if (file.exists()) {
				long breakpoint = file.length();
				/* set the range to continue the downloading */
				conn.setRequestProperty("Range", "bytes=" + breakpoint + "-");
			}

			/* status code */
			int statusCode = conn.getResponseCode();
			switch (statusCode) {
			case HTTP_OK:
			case HTTP_PARTIAL:
				transferData(conn, request);
				break;
				
			case HTTP_MOVED_PERM:
			case HTTP_MOVED_TEMP:
			case HTTP_TEMP_REDIRECT:
			case HTTP_SEE_OTHER:
				if (mRedirectionCount ++ < MAX_REDIRECTION) {
					Log.i(TAG, "redirect for download id: " + request.getDownloadId());
					/* take redirect url and call executeDownload recursively */
					String redirectUrl = conn.getHeaderField(LOCATION);
					request.setUrl(redirectUrl);
					executeDownload(request);
				} else {
					/* redirect to many times */
					updateFailure(request, statusCode, "redirect too many times");
				}
				break;

			case HTTP_REQUESTED_RANGE_NOT_SATISFIABLE:
			case HTTP_UNAVAILABLE:
			case HTTP_INTERNAL_ERROR:
			default:
				updateFailure(request, statusCode, conn.getResponseMessage());
				break;
			}
		} catch (IOException e) {
			updateFailure(request, HTTP_INVALID, e.getMessage());
		} finally {
			if (conn != null) {
				conn.disconnect();
			}
		}
	}
	
	/** transfer data from server to local file */
	private void transferData(HttpURLConnection conn, DownloadRequest request) throws IOException {
		long contentLength = getContentLength(conn);
		if (contentLength == -1) {
			return;
		}
		
		InputStream is = null;
		File file = new File(request.getTmpDestinationPath());
		RandomAccessFile raf = new RandomAccessFile(file, "rw");
		long bytesWritten = file.length();
		contentLength += bytesWritten;
		/* if the file has existed, seek to breakpoint */
		if (bytesWritten > 0) {
			raf.seek(bytesWritten);
		}

		mTotalBytes = contentLength;
		/* deliver start callback */
		updateStart(request, contentLength);
		
		try {
			/* get input stream from connection */
			is = conn.getInputStream();
			if (is != null) {
				byte[] buffer = new byte[BUFFER_SIZE];
				int length;
				
				while (true) {
					/* if the request has canceld, stop the downloading */
					if (Thread.currentThread().isInterrupted() || request.isCanceled()) {
						Log.i(TAG, "download has canceled, download id: " + request.getDownloadId());
						request.finish();
						return;
					}

					/* if current is not wifi and mobile network is not allowed, stop */
					if (request.getAllowedNetworkTypes() != 0 &&
						!DownloadUtils.isWifi(request.getContext()) &&
						(request.getAllowedNetworkTypes() &
									DownloadRequest.NETWORK_MOBILE) == 0) {
						updateFailure(request, HTTP_ERROR_NETWORK, "network error");
						return;
					}
					
					/* read data into buffer from input stream */
					length = readFromInputStream(buffer, is);
					if (length == -1) {
						long fileSize = new File(request.getTmpDestinationPath()).length();
						
						/* deliver progress callback before deliver success */
						updateProgress(request, fileSize, contentLength);
						
						/* if file size equals total bytes, then download successfully */
						if (fileSize == contentLength) {
							updateSuccess(request);
						} else {
							updateFailure(request, HTTP_INVALID, "file size error");
						}
						
						return;
					} else if (length == Integer.MIN_VALUE) {
						updateFailure(request, HTTP_ERROR_SIZE, "transfer data error");
						return;
					}
					
					bytesWritten += length;
					/* write buffer into local file */
					raf.write(buffer, 0, length);
					
					/* deliver progress callback */
					updateProgress(request, bytesWritten, contentLength);
				}
			}
		} finally {
			raf.close();
			silentCloseInputStream(is);
		}
	}
	
	/** read data from input stream */
	private int readFromInputStream(byte[] buffer, InputStream is) {
		try {
			return is.read(buffer);
		} catch (IOException e) {
			if (END_OF_STREAM.equals(e.getMessage())) {
				return -1;
			}
			
			return Integer.MIN_VALUE;
		}
	}
	
	/** a utility function to close an input stream without raising an exception. */
	private void silentCloseInputStream(InputStream is) {
		try {
			if (is != null) {
				is.close();
			}
		} catch (IOException e) {
			Log.w(TAG, "cannot close input stream", e);
		}
	}
	
	/** read response header from server */
	private long getContentLength(HttpURLConnection conn) {
		String transferEncoding = conn.getHeaderField(TRANSFER_ENCODING);
		if (transferEncoding == null || transferEncoding.equalsIgnoreCase("chunked")) {
			return conn.getHeaderFieldInt(CONTENT_LENGTH, -1);
		} else {
			return -1;
		}
	}
	
	/**
	 * Forces this dispatcher to quit immediately. If any download requests are still in
	 * the queue, they are not guaranteed to be processed.
	 */
	protected void quit() {
		mQuit = true;
		
		/* interrupt current thread */
		interrupt();
		
		Log.i(TAG, "download dispatcher has quit");
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy