com.google.samples.apps.iosched.sync.userdata.gms.GMSUserDataSyncHelper.java Source code

Java tutorial

Introduction

Here is the source code for com.google.samples.apps.iosched.sync.userdata.gms.GMSUserDataSyncHelper.java

Source

/*
 * Copyright 2014 Google Inc. All rights reserved.
 *
 * 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.google.samples.apps.iosched.sync.userdata.gms;

import android.app.PendingIntent;
import android.content.Context;
import android.support.v4.app.NotificationCompat;
import android.support.v4.app.NotificationManagerCompat;
import android.text.TextUtils;
import android.util.Log;

import com.google.android.gms.common.ConnectionResult;
import com.google.android.gms.common.GooglePlayServicesUtil;
import com.google.android.gms.common.api.GoogleApiClient;
import com.google.android.gms.common.api.Status;
import com.google.android.gms.drive.Drive;
import com.google.android.gms.drive.DriveId;
import com.google.samples.apps.iosched.R;
import com.google.samples.apps.iosched.sync.userdata.AbstractUserDataSyncHelper;
import com.google.samples.apps.iosched.sync.userdata.UserAction;
import com.google.samples.apps.iosched.sync.userdata.util.UserDataHelper;
import com.google.samples.apps.iosched.util.AccountUtils;

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

import static com.google.samples.apps.iosched.util.LogUtils.LOGD;
import static com.google.samples.apps.iosched.util.LogUtils.LOGE;
import static com.google.samples.apps.iosched.util.LogUtils.makeLogTag;

/**
 * Helper class that syncs user data with Google Drive AppData folder using the Google Play services
 * Drive API.
 */
public class GMSUserDataSyncHelper extends AbstractUserDataSyncHelper {
    private static final String TAG = makeLogTag(GMSUserDataSyncHelper.class);
    private static final String APPDATA_FILE_NAME = "user_data.json";
    private static final String FILE_MIME_TYPE = "application/json";

    private static final int ERROR_NOTIFICATION_ID = 101;

    private GoogleApiClient mGoogleApiClient;

    public GMSUserDataSyncHelper(Context context, String accountName) {
        super(context, accountName);
    }

    /**
     * Synchronization steps:
     * 1. If there are local changes, sync the latest local version with remote
     *    and ignore merge conflicts. The last write wins.
     * 2. If there are no local changes, fetch the latest remote version. If
     *    it includes changes, notify that preferences have changed.
     */
    @Override
    protected boolean syncImpl(List<UserAction> actions, boolean hasPendingLocalData) {
        try {
            setupApiClient();
            DriveHelper helper = new DriveHelper(mGoogleApiClient);
            ConnectionResult result = helper.connectIfNecessary();
            if (result != null && !result.isSuccess()) {
                handleConnectionResult(result);
                return false;
            }

            // Call and await a sync using DriveApi#requestSync to ensure that the caches
            // are up-to-date.
            Status status = helper.requestSync();
            LOGD(TAG, "Drive Request Sync status " + status.isSuccess());
            if (!status.isSuccess() && status.hasResolution()) {
                postNotification(status.getResolution());
                return false;
            }

            DriveId driveId = helper.getOrCreateFile(APPDATA_FILE_NAME, FILE_MIME_TYPE);
            // Get the remote and local user data.
            UserDataHelper.UserData remote = UserDataHelper.fromString(fetchRemoteData(helper, driveId));
            UserDataHelper.UserData local = UserDataHelper.getUserData(actions);

            String remoteGcmKey = remote.getGcmKey();
            String localGcmKey = AccountUtils.getGcmKey(mContext, mAccountName);
            LOGD(TAG, "Local GCM key: " + AccountUtils.sanitizeGcmKey(localGcmKey));
            LOGD(TAG, "Remote GCM key: "
                    + (remoteGcmKey == null ? "(null)" : AccountUtils.sanitizeGcmKey(remoteGcmKey)));

            // if the remote data came with a GCM key, it should override ours
            if (!TextUtils.isEmpty(remoteGcmKey)) {
                if (remoteGcmKey.equals(localGcmKey)) {
                    LOGD(TAG, "Remote GCM key is the same as local, so no action necessary.");
                } else {
                    LOGD(TAG, "Remote GCM key is different from local. OVERRIDING local.");
                    localGcmKey = remoteGcmKey;
                    AccountUtils.setGcmKey(mContext, mAccountName, localGcmKey);
                }
            }

            // If remote data is the same as local, and the remote end already has a GCM key,
            // there is nothing we need to do.
            if (remote.equals(local) && !TextUtils.isEmpty(remoteGcmKey)) {
                LOGD(TAG, "Update is not needed (local is same as remote, and remote has key)");
                return false;
            }

            UserDataHelper.UserData merged;
            if (hasPendingLocalData || TextUtils.isEmpty(remoteGcmKey)) {
                // merge local dirty actions into remote content
                if (hasPendingLocalData) {
                    LOGD(TAG, "Has pending local data, merging.");
                    merged = mergeDirtyActions(actions, remote);
                } else {
                    LOGD(TAG, "No pending local data, just updating remote GCM key.");
                    merged = remote;
                }
                // add the GCM key special item
                merged.setGcmKey(localGcmKey);
                // save to remote
                LOGD(TAG, "Sending user data to Drive, gcm key " + AccountUtils.sanitizeGcmKey(localGcmKey));

                String content = UserDataHelper.toJsonString(merged);
                helper.saveDriveFile(driveId, content);
            } else {
                merged = remote;
            }

            UserDataHelper.setLocalUserData(mContext, merged, mAccountName);
            return true;

        } catch (IOException ex) {
            LOGE(TAG, "Could not sync myschedule", ex);
        }
        return false;
    }

    private void postNotification(PendingIntent resolution) {
        NotificationCompat.Builder builder = new NotificationCompat.Builder(mContext)
                .setContentTitle(mContext.getString(R.string.app_data_notification_resolution_title))
                .setContentText(mContext.getString(R.string.app_data_notification_resolution_content))
                .setSmallIcon(R.drawable.ic_stat_notification).setContentIntent(resolution).setAutoCancel(true);
        NotificationManagerCompat.from(mContext).notify(ERROR_NOTIFICATION_ID, builder.build());
    }

    private void setupApiClient() {
        mGoogleApiClient = new GoogleApiClient.Builder(mContext).addApi(Drive.API).addScope(Drive.SCOPE_APPFOLDER)
                .setAccountName(mAccountName).build();
    }

    private String fetchRemoteData(DriveHelper helper, DriveId driveId) throws IOException {
        return helper.getContentsFromDrive(driveId);
    }

    private void handleConnectionResult(ConnectionResult result) {
        if (result.hasResolution()) {
            postNotification(result.getResolution());
        } else {
            // No resolution exists so silently fail. This can happen when a user cancels the OAuth2
            // request. In this case we will rely on the OAuth2 consent prompts the next time the
            // app is started.
        }
    }

    /**
     * Updates the remote preferences file with the given JSON content.
     */
    private UserDataHelper.UserData mergeDirtyActions(List<UserAction> actions, UserDataHelper.UserData userData)
            throws IOException {
        // apply "dirty" actions:
        for (UserAction action : actions) {
            if (action.requiresSync) {
                if (UserAction.TYPE.ADD_STAR.equals(action.type)) {
                    if (userData.getStarredSessionIds() == null) {
                        userData.setStarredSessionIds(new HashSet<String>());
                    }
                    userData.getStarredSessionIds().add(action.sessionId);
                } else if (UserAction.TYPE.SUBMIT_FEEDBACK.equals(action.type)) {
                    if (userData.getFeedbackSubmittedSessionIds() == null) {
                        userData.setFeedbackSubmittedSessionIds(new HashSet<String>());
                    }
                    userData.getFeedbackSubmittedSessionIds().add(action.sessionId);
                } else if (UserAction.TYPE.VIEW_VIDEO.equals(action.type)) {
                    if (userData.getViewedVideoIds() == null) {
                        userData.setViewedVideoIds(new HashSet<String>());
                    }
                    userData.getViewedVideoIds().add(action.videoId);
                } else {
                    if (userData.getStarredSessionIds() == null) {
                        userData.setStarredSessionIds(new HashSet<String>());
                    }
                    userData.getStarredSessionIds().remove(action.sessionId);
                }
            }
        }
        return userData;
    }
}