com.androidrocks.bex.provider.BooksUpdater.java Source code

Java tutorial

Introduction

Here is the source code for com.androidrocks.bex.provider.BooksUpdater.java

Source

/*
 * Copyright (C) 2009 Muthu Ramadoss. All rights reserved.
 *
 * Modified from Romain Guy Shelves project to suit Books-Exchange requirements.
 * Original source from Shelves - http://code.google.com/p/shelves/
 */

/*
 * Copyright (C) 2008 Romain Guy
 *
 * 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 com.androidrocks.bex.provider;

import android.content.Context;
import android.content.ContentResolver;
import android.content.ContentValues;
import android.graphics.Bitmap;
import android.os.Process;

import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.Calendar;
import java.util.GregorianCalendar;
import java.util.HashMap;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.text.ParseException;

import org.apache.http.client.methods.HttpGet;
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.HttpStatus;
import org.apache.http.Header;

import com.androidrocks.bex.util.HttpManager;
import com.androidrocks.bex.util.ImageUtilities;
import com.androidrocks.bex.util.ImportUtilities;

public class BooksUpdater implements Runnable {
    private static final String LOG_TAG = "BooksUpdater";

    private static final long ONE_DAY = 24 * 60 * 60 * 1000;

    private static final HashMap<String, Long> sLastChecks = new HashMap<String, Long>();

    private final BlockingQueue<String> mQueue = new ArrayBlockingQueue<String>(12);
    private final ContentResolver mResolver;
    private final SimpleDateFormat mLastModifiedFormat;
    private final String mSelection;
    private final String[] mArguments = new String[1];
    private final ContentValues mValues = new ContentValues();

    private Thread mThread;
    private volatile boolean mStopped;

    public BooksUpdater(Context context) {
        mResolver = context.getContentResolver();
        mLastModifiedFormat = new SimpleDateFormat("EEE, dd MMM yyyy HH:mm:ss z");
        mSelection = BooksStore.Book._ID + "=?";
    }

    public void start() {
        if (mThread == null) {
            mStopped = false;
            mThread = new Thread(this, "BooksUpdater");
            mThread.start();
        }
    }

    public void stop() {
        if (mThread != null) {
            mStopped = true;
            mThread.interrupt();
            mThread = null;
        }
    }

    public void offer(String... books) {
        for (String bookId : books) {
            if (bookId != null)
                mQueue.offer(bookId);
        }
    }

    public void clear() {
        mQueue.clear();
    }

    public void run() {
        Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND);
        final ImageUtilities.ExpiringBitmap expiring = new ImageUtilities.ExpiringBitmap();

        while (!mStopped) {
            try {
                final String bookId = mQueue.take();

                final Long lastCheck = sLastChecks.get(bookId);
                if (lastCheck != null && (lastCheck + ONE_DAY) >= System.currentTimeMillis()) {
                    continue;
                }
                sLastChecks.put(bookId, System.currentTimeMillis());

                final BooksStore.Book book = BooksManager.findBook(mResolver, bookId);
                if (book.getLastModified() == null || book.getImageUrl(BooksStore.ImageSize.TINY) == null) {
                    continue;
                }

                if (bookCoverUpdated(book, expiring) && expiring.lastModified != null) {
                    ImageUtilities.deleteCachedCover(bookId);
                    final Bitmap bitmap = book.loadCover(BooksStore.ImageSize.TINY);
                    ImportUtilities.addBookCoverToCache(book, bitmap);

                    mValues.put(BooksStore.Book.LAST_MODIFIED, expiring.lastModified.getTimeInMillis());
                    mArguments[0] = bookId;
                    mResolver.update(BooksStore.Book.CONTENT_URI, mValues, mSelection, mArguments);
                }

                Thread.sleep(1000);
            } catch (InterruptedException e) {
                // Ignore
            }
        }
    }

    private boolean bookCoverUpdated(BooksStore.Book book, ImageUtilities.ExpiringBitmap expiring) {
        expiring.lastModified = null;
        final HttpGet get = new HttpGet(book.getImageUrl(BooksStore.ImageSize.TINY));

        HttpEntity entity = null;
        try {
            final HttpResponse response = HttpManager.execute(get);
            if (response.getStatusLine().getStatusCode() == HttpStatus.SC_OK) {
                entity = response.getEntity();

                final Header header = response.getFirstHeader("Last-Modified");
                if (header != null) {
                    final Calendar calendar = GregorianCalendar.getInstance();
                    try {
                        calendar.setTime(mLastModifiedFormat.parse(header.getValue()));
                        expiring.lastModified = calendar;
                        return calendar.after(book.getLastModified());
                    } catch (ParseException e) {
                        return false;
                    }
                }
            }
        } catch (IOException e) {
            android.util.Log.e(LOG_TAG, "Could not check modification date for " + book, e);
        } finally {
            if (entity != null) {
                try {
                    entity.consumeContent();
                } catch (IOException e) {
                    android.util.Log.e(LOG_TAG, "Could not check modification date for " + book, e);
                }
            }
        }

        return false;
    }
}