Android Open Source - lightbox-android-webservices Cached Operation






From Project

Back to project page lightbox-android-webservices.

License

The source code is released under:

Apache License

If you think the Android project lightbox-android-webservices listed in this page is inappropriate, such as containing malicious code/tools or violating the copyright, please email info at java2s dot com, thanks.

Java Source Code

/**
 * Copyright (c) 2012 Lightbox/*from   w  w  w .j a v a  2  s. com*/
 *
 * 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.lightbox.android.operations;

import java.io.IOException;
import java.io.InputStream;
import java.util.List;

import org.apache.commons.io.IOUtils;

import android.util.Log;

import com.lightbox.android.cache.ApiCache;
import com.lightbox.android.cache.Cache;
import com.lightbox.android.utils.debug.DebugLog;
import com.lightbox.android.webservices.processors.ParsingException;
import com.lightbox.android.webservices.requests.ApiRequest;
import com.lightbox.android.webservices.responses.ApiException;
import com.lightbox.android.webservices.responses.ApiResponse;

/** 
 * CachedOperation 
 * @author Fabien Devos
 */
public class CachedOperation<T> extends AbstractOperation<T> {
  /** Used to tag logs */
  //@SuppressWarnings("unused")
  private static final String TAG = "CachedOperation";
  
  private boolean mIsDataTooOld = false;
  private Object mContext = null;
  private boolean mRetrieveFromLocalDataOnly = false;

  /**
   * Constructor.
   * @param dataClass
   * @param apiRequest
   */
  public CachedOperation(Class<T> dataClass, ApiRequest apiRequest) {
    this(dataClass, apiRequest, false);
  }
  public CachedOperation(Class<T> dataClass, ApiRequest apiRequest, boolean retrieveFromLocalDataOnly) {
    super(dataClass, apiRequest);
    mRetrieveFromLocalDataOnly = retrieveFromLocalDataOnly;
  }
  
  //----------------------------------------------
  // Operation
  
  @Override
  public Object getContext() {
    return mContext;
  }

  // Override the default executeAsync to add support for memory caching on the main thread.
  @Override
  public final void executeAsync(OperationListener<T> listener) {
    List<T> resultList = null;
    try {
      resultList = executeLocalOperationSync();
    } catch (Throwable t) {
      Log.w(TAG, "Failed to retrieve data from cache", t);
    }
    
    if (resultList != null && ! resultList.isEmpty()) {
      DebugLog.d(TAG, "Getting result from cache.");
      
      // Call back on main thread
      if (listener != null) {
        listener.onSuccess(this, resultList);
      }
    }
    
    if ( ! mRetrieveFromLocalDataOnly && mIsDataTooOld) {
      // Proceed with normal operation execution
      super.executeAsync(listener);
    }
  }

  @Override
  public final List<T> executeSync() throws Exception {
    List<T> resultList = null;

    // If (there is no result OR the data are too old) AND we are not asking for local data only
    if ((resultList == null || mIsDataTooOld) && ! mRetrieveFromLocalDataOnly) {
      DebugLog.d(TAG, "No result from cache OR data is too old");
      
      // Try to retrieve data from web services
      resultList = executeServerOperationSync();

      // No need to send back result from cache because it's already done
    }
    
    return resultList;
  }

  @Override
  public List<T> executeLocalOperationSync() throws Exception {
    // Try to retrieve from disk
    Cache.Result<String> cacheResult = ApiCache.getInstance().getFromDisk(getId());
    String stringResult = cacheResult.getData();
    mIsDataTooOld = isDataTooOld(cacheResult.getUpdatedTime());
    // Data is treated as "too old" if empty
    if (stringResult == null) { return null; }
    
    // Parse the cached response
    ApiResponse<?> apiResponse = parseString(stringResult);
    Object result = apiResponse.getContent();
    mContext = apiResponse.getContext();
    if (result == null) { return null; }
    
    DebugLog.d(TAG, "Getting result from disk cache. Data is too old:" + mIsDataTooOld);
  
    return wrapInList(result);
  }

  @Override
  public List<T> executeServerOperationSync() throws Exception {
    //Log.e(TAG, "CALLING SERVER id: %s" + getId());
    
    // Query server
    InputStream inputStream = getApiRequest().callApi().getEntity().getContent();
    String stringResultFromServer = IOUtils.toString(inputStream);
    ApiResponse<?> apiResponse = getApiRequest().parseInputStream(IOUtils.toInputStream(stringResultFromServer));  
    Object result = apiResponse.getContent();
    mContext = apiResponse.getContext();
    if (result == null) { return null; }
    List<T> resultList = wrapInList(result);
    
    DebugLog.d(TAG, "Getting result from server.");
    
    // Save in disk cache
    ApiCache.getInstance().putOnDisk(getId(), stringResultFromServer);

    // Hook
    if (resultList != null) { onRetrieveDataFromServer(resultList); }
    
    return resultList;
  }
  
  //----------------------------------------------
  // Utility methods
  
  private boolean isDataTooOld(long updatedTime) {
    return (System.currentTimeMillis() - updatedTime) > RetrieveOperation.CACHE_DURATION;
  }
  
  private ApiResponse<?> parseString(String stringResult) throws ParsingException, IOException, ApiException {
    if (stringResult == null) { return null; }
    return getApiRequest().parseInputStream(IOUtils.toInputStream(stringResult));    
  }
  
  //----------------------------------------------
  // Hooks for subclasses

  protected void onRetrieveDataFromServer(List<T> result) throws Exception {
    // Nothing by default
  }
}




Java Source Code List

com.lightbox.android.bitmap.BitmapFileCleanerTask.java
com.lightbox.android.bitmap.BitmapLoaderListener.java
com.lightbox.android.bitmap.BitmapLoaderTask.java
com.lightbox.android.bitmap.BitmapLoader.java
com.lightbox.android.bitmap.BitmapSize.java
com.lightbox.android.bitmap.BitmapSource.java
com.lightbox.android.bitmap.BitmapUtils.java
com.lightbox.android.cache.AbstractCache.java
com.lightbox.android.cache.ApiCache.java
com.lightbox.android.cache.BitmapCache.java
com.lightbox.android.cache.Cache.java
com.lightbox.android.data.ClearAndSaveBatchTask.java
com.lightbox.android.data.Data.java
com.lightbox.android.data.DatabaseCleanerTask.java
com.lightbox.android.data.DatabaseHelper.java
com.lightbox.android.data.DeleteBatchTask.java
com.lightbox.android.data.SaveBatchTask.java
com.lightbox.android.io.RandomAccessFileOutputStream.java
com.lightbox.android.lifecycle.LifeCycleListener.java
com.lightbox.android.lifecycle.ManagedLifeCycleActivity.java
com.lightbox.android.location.LocationHelper.java
com.lightbox.android.location.LocationListener.java
com.lightbox.android.network.HttpHelper.java
com.lightbox.android.network.NetworkUtils.java
com.lightbox.android.operations.AbstractOperation.java
com.lightbox.android.operations.CachedOperation.java
com.lightbox.android.operations.DeleteOperation.java
com.lightbox.android.operations.FailureOperation.java
com.lightbox.android.operations.ModificationNetworkOnlyOperation.java
com.lightbox.android.operations.NetworkOnlyOperation.java
com.lightbox.android.operations.OperationListener.java
com.lightbox.android.operations.OperationTask.java
com.lightbox.android.operations.Operation.java
com.lightbox.android.operations.Retrievable.java
com.lightbox.android.operations.RetrieveOperation.java
com.lightbox.android.operations.SaveOperation.java
com.lightbox.android.operations.Updatable.java
com.lightbox.android.tasks.BackgroundTaskWeak.java
com.lightbox.android.tasks.BackgroundTask.java
com.lightbox.android.tasks.DefaultExecutor.java
com.lightbox.android.utils.AndroidUtils.java
com.lightbox.android.utils.Base64.java
com.lightbox.android.utils.IntentUtils.java
com.lightbox.android.utils.MediaUtils.java
com.lightbox.android.utils.ResUtils.java
com.lightbox.android.utils.debug.DebugLifeCycleListener.java
com.lightbox.android.utils.debug.DebugLog.java
com.lightbox.android.views.RemoteImageView.java
com.lightbox.android.views.RemoteThumbImageView.java
com.lightbox.android.webservices.processors.GenerationException.java
com.lightbox.android.webservices.processors.JacksonProcessor.java
com.lightbox.android.webservices.processors.ParsingException.java
com.lightbox.android.webservices.processors.Processor.java
com.lightbox.android.webservices.requests.ApiRequestFactory.java
com.lightbox.android.webservices.requests.ApiRequestListener.java
com.lightbox.android.webservices.requests.ApiRequestTask.java
com.lightbox.android.webservices.requests.ApiRequest.java
com.lightbox.android.webservices.requests.ApiRequests.java
com.lightbox.android.webservices.requests.JacksonApiRequestFactory.java
com.lightbox.android.webservices.responses.ApiException.java
com.lightbox.android.webservices.responses.ApiResponse.java
com.lightbox.tweetsnearby.TweetsNearbyApplication.java
com.lightbox.tweetsnearby.activities.MainActivity.java
com.lightbox.tweetsnearby.activities.PickPlaceActivity.java
com.lightbox.tweetsnearby.model.Tweet.java
com.lightbox.tweetsnearby.model.Venue.java
com.lightbox.tweetsnearby.responses.foursquare.FoursquareApiResponse.java
com.lightbox.tweetsnearby.responses.foursquare.VenueListResponse.java
com.lightbox.tweetsnearby.responses.twitter.TweetListResponse.java
com.lightbox.tweetsnearby.responses.twitter.TwitterApiResponse.java