forked from p4c9ryihk/yss__miui__notes
				
			
			You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							801 lines
						
					
					
						
							29 KiB
						
					
					
				
			
		
		
	
	
							801 lines
						
					
					
						
							29 KiB
						
					
					
				/*
 | 
						|
 * Copyright (c) 2010-2011, The MiCode Open Source Community (www.micode.net)
 | 
						|
 *
 | 
						|
 * 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 net.micode.notes.gtask.remote;
 | 
						|
 | 
						|
import android.app.Activity;
 | 
						|
import android.content.ContentResolver;
 | 
						|
import android.content.ContentUris;
 | 
						|
import android.content.ContentValues;
 | 
						|
import android.content.Context;
 | 
						|
import android.database.Cursor;
 | 
						|
import android.util.Log;
 | 
						|
 | 
						|
import net.micode.notes.R;
 | 
						|
import net.micode.notes.data.Notes;
 | 
						|
import net.micode.notes.data.Notes.DataColumns;
 | 
						|
import net.micode.notes.data.Notes.NoteColumns;
 | 
						|
import net.micode.notes.gtask.data.MetaData;
 | 
						|
import net.micode.notes.gtask.data.Node;
 | 
						|
import net.micode.notes.gtask.data.SqlNote;
 | 
						|
import net.micode.notes.gtask.data.Task;
 | 
						|
import net.micode.notes.gtask.data.TaskList;
 | 
						|
import net.micode.notes.gtask.exception.ActionFailureException;
 | 
						|
import net.micode.notes.gtask.exception.NetworkFailureException;
 | 
						|
import net.micode.notes.tool.DataUtils;
 | 
						|
import net.micode.notes.tool.GTaskStringUtils;
 | 
						|
 | 
						|
import org.json.JSONArray;
 | 
						|
import org.json.JSONException;
 | 
						|
import org.json.JSONObject;
 | 
						|
 | 
						|
import java.util.HashMap;
 | 
						|
import java.util.HashSet;
 | 
						|
import java.util.Iterator;
 | 
						|
import java.util.Map;
 | 
						|
 | 
						|
 | 
						|
public class GTaskManager {
 | 
						|
    private static final String TAG = GTaskManager.class.getSimpleName();
 | 
						|
 | 
						|
    public static final int STATE_SUCCESS = 0;
 | 
						|
 | 
						|
    public static final int STATE_NETWORK_ERROR = 1;
 | 
						|
 | 
						|
    public static final int STATE_INTERNAL_ERROR = 2;
 | 
						|
 | 
						|
    public static final int STATE_SYNC_IN_PROGRESS = 3;
 | 
						|
 | 
						|
    public static final int STATE_SYNC_CANCELLED = 4;
 | 
						|
 | 
						|
    private static GTaskManager mInstance = null;
 | 
						|
 | 
						|
    private Activity mActivity;
 | 
						|
 | 
						|
    private Context mContext;
 | 
						|
 | 
						|
    private ContentResolver mContentResolver;
 | 
						|
 | 
						|
    private boolean mSyncing;
 | 
						|
 | 
						|
    private boolean mCancelled;
 | 
						|
 | 
						|
    private HashMap<String, TaskList> mGTaskListHashMap;
 | 
						|
 | 
						|
    private HashMap<String, Node> mGTaskHashMap;
 | 
						|
 | 
						|
    private HashMap<String, MetaData> mMetaHashMap;
 | 
						|
 | 
						|
    private TaskList mMetaList;
 | 
						|
 | 
						|
    private HashSet<Long> mLocalDeleteIdMap;
 | 
						|
 | 
						|
    private HashMap<String, Long> mGidToNid;
 | 
						|
 | 
						|
    private HashMap<Long, String> mNidToGid;
 | 
						|
 | 
						|
    private GTaskManager() {
 | 
						|
        mSyncing = false;
 | 
						|
        mCancelled = false;
 | 
						|
        mGTaskListHashMap = new HashMap<String, TaskList>();
 | 
						|
        mGTaskHashMap = new HashMap<String, Node>();
 | 
						|
        mMetaHashMap = new HashMap<String, MetaData>();
 | 
						|
        mMetaList = null;
 | 
						|
        mLocalDeleteIdMap = new HashSet<Long>();
 | 
						|
        mGidToNid = new HashMap<String, Long>();
 | 
						|
        mNidToGid = new HashMap<Long, String>();
 | 
						|
    }
 | 
						|
 | 
						|
    public static synchronized GTaskManager getInstance() {
 | 
						|
        if (mInstance == null) {
 | 
						|
            mInstance = new GTaskManager();
 | 
						|
        }
 | 
						|
        return mInstance;
 | 
						|
    }
 | 
						|
 | 
						|
    public synchronized void setActivityContext(Activity activity) {
 | 
						|
        // used for getting authtoken
 | 
						|
        mActivity = activity;
 | 
						|
    }
 | 
						|
 | 
						|
    public int sync(Context context, GTaskASyncTask asyncTask) {
 | 
						|
        if (mSyncing) {
 | 
						|
            Log.d(TAG, "Sync is in progress");
 | 
						|
            return STATE_SYNC_IN_PROGRESS;
 | 
						|
        }
 | 
						|
        mContext = context;
 | 
						|
        mContentResolver = mContext.getContentResolver();
 | 
						|
        mSyncing = true;
 | 
						|
        mCancelled = false;
 | 
						|
        mGTaskListHashMap.clear();
 | 
						|
        mGTaskHashMap.clear();
 | 
						|
        mMetaHashMap.clear();
 | 
						|
        mLocalDeleteIdMap.clear();
 | 
						|
        mGidToNid.clear();
 | 
						|
        mNidToGid.clear();
 | 
						|
 | 
						|
        try {
 | 
						|
            GTaskClient client = GTaskClient.getInstance();
 | 
						|
            client.resetUpdateArray();
 | 
						|
 | 
						|
            // login google task
 | 
						|
            if (!mCancelled) {
 | 
						|
                if (!client.login(mActivity)) {
 | 
						|
                    throw new NetworkFailureException("login google task failed");
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            // get the task list from google
 | 
						|
            asyncTask.publishProgess(mContext.getString(R.string.sync_progress_init_list));
 | 
						|
            initGTaskList();
 | 
						|
 | 
						|
            // do content sync work
 | 
						|
            asyncTask.publishProgess(mContext.getString(R.string.sync_progress_syncing));
 | 
						|
            syncContent();
 | 
						|
        } catch (NetworkFailureException e) {
 | 
						|
            Log.e(TAG, e.toString());
 | 
						|
            return STATE_NETWORK_ERROR;
 | 
						|
        } catch (ActionFailureException e) {
 | 
						|
            Log.e(TAG, e.toString());
 | 
						|
            return STATE_INTERNAL_ERROR;
 | 
						|
        } catch (Exception e) {
 | 
						|
            Log.e(TAG, e.toString());
 | 
						|
            e.printStackTrace();
 | 
						|
            return STATE_INTERNAL_ERROR;
 | 
						|
        } finally {
 | 
						|
            mGTaskListHashMap.clear();
 | 
						|
            mGTaskHashMap.clear();
 | 
						|
            mMetaHashMap.clear();
 | 
						|
            mLocalDeleteIdMap.clear();
 | 
						|
            mGidToNid.clear();
 | 
						|
            mNidToGid.clear();
 | 
						|
            mSyncing = false;
 | 
						|
        }
 | 
						|
 | 
						|
        return mCancelled ? STATE_SYNC_CANCELLED : STATE_SUCCESS;
 | 
						|
    }
 | 
						|
 | 
						|
    private void initGTaskList() throws NetworkFailureException {
 | 
						|
        if (mCancelled)
 | 
						|
            return;
 | 
						|
        GTaskClient client = GTaskClient.getInstance();
 | 
						|
        try {
 | 
						|
            JSONArray jsTaskLists = client.getTaskLists();
 | 
						|
 | 
						|
            // init meta list first
 | 
						|
            mMetaList = null;
 | 
						|
            for (int i = 0; i < jsTaskLists.length(); i++) {
 | 
						|
                JSONObject object = jsTaskLists.getJSONObject(i);
 | 
						|
                String gid = object.getString(GTaskStringUtils.GTASK_JSON_ID);
 | 
						|
                String name = object.getString(GTaskStringUtils.GTASK_JSON_NAME);
 | 
						|
 | 
						|
                if (name
 | 
						|
                        .equals(GTaskStringUtils.MIUI_FOLDER_PREFFIX + GTaskStringUtils.FOLDER_META)) {
 | 
						|
                    mMetaList = new TaskList();
 | 
						|
                    mMetaList.setContentByRemoteJSON(object);
 | 
						|
 | 
						|
                    // load meta data
 | 
						|
                    JSONArray jsMetas = client.getTaskList(gid);
 | 
						|
                    for (int j = 0; j < jsMetas.length(); j++) {
 | 
						|
                        object = (JSONObject) jsMetas.getJSONObject(j);
 | 
						|
                        MetaData metaData = new MetaData();
 | 
						|
                        metaData.setContentByRemoteJSON(object);
 | 
						|
                        if (metaData.isWorthSaving()) {
 | 
						|
                            mMetaList.addChildTask(metaData);
 | 
						|
                            if (metaData.getGid() != null) {
 | 
						|
                                mMetaHashMap.put(metaData.getRelatedGid(), metaData);
 | 
						|
                            }
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            // create meta list if not existed
 | 
						|
            if (mMetaList == null) {
 | 
						|
                mMetaList = new TaskList();
 | 
						|
                mMetaList.setName(GTaskStringUtils.MIUI_FOLDER_PREFFIX
 | 
						|
                        + GTaskStringUtils.FOLDER_META);
 | 
						|
                GTaskClient.getInstance().createTaskList(mMetaList);
 | 
						|
            }
 | 
						|
 | 
						|
            // init task list
 | 
						|
            for (int i = 0; i < jsTaskLists.length(); i++) {
 | 
						|
                JSONObject object = jsTaskLists.getJSONObject(i);
 | 
						|
                String gid = object.getString(GTaskStringUtils.GTASK_JSON_ID);
 | 
						|
                String name = object.getString(GTaskStringUtils.GTASK_JSON_NAME);
 | 
						|
 | 
						|
                if (name.startsWith(GTaskStringUtils.MIUI_FOLDER_PREFFIX)
 | 
						|
                        && !name.equals(GTaskStringUtils.MIUI_FOLDER_PREFFIX
 | 
						|
                                + GTaskStringUtils.FOLDER_META)) {
 | 
						|
                    TaskList tasklist = new TaskList();
 | 
						|
                    tasklist.setContentByRemoteJSON(object);
 | 
						|
                    mGTaskListHashMap.put(gid, tasklist);
 | 
						|
                    mGTaskHashMap.put(gid, tasklist);
 | 
						|
 | 
						|
                    // load tasks
 | 
						|
                    JSONArray jsTasks = client.getTaskList(gid);
 | 
						|
                    for (int j = 0; j < jsTasks.length(); j++) {
 | 
						|
                        object = (JSONObject) jsTasks.getJSONObject(j);
 | 
						|
                        gid = object.getString(GTaskStringUtils.GTASK_JSON_ID);
 | 
						|
                        Task task = new Task();
 | 
						|
                        task.setContentByRemoteJSON(object);
 | 
						|
                        if (task.isWorthSaving()) {
 | 
						|
                            task.setMetaInfo(mMetaHashMap.get(gid));
 | 
						|
                            tasklist.addChildTask(task);
 | 
						|
                            mGTaskHashMap.put(gid, task);
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
        } catch (JSONException e) {
 | 
						|
            Log.e(TAG, e.toString());
 | 
						|
            e.printStackTrace();
 | 
						|
            throw new ActionFailureException("initGTaskList: handing JSONObject failed");
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    private void syncContent() throws NetworkFailureException {
 | 
						|
        int syncType;
 | 
						|
        Cursor c = null;
 | 
						|
        String gid;
 | 
						|
        Node node;
 | 
						|
 | 
						|
        mLocalDeleteIdMap.clear();
 | 
						|
 | 
						|
        if (mCancelled) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        // for local deleted note
 | 
						|
        try {
 | 
						|
            c = mContentResolver.query(Notes.CONTENT_NOTE_URI, SqlNote.PROJECTION_NOTE,
 | 
						|
                    "(type<>? AND parent_id=?)", new String[] {
 | 
						|
                            String.valueOf(Notes.TYPE_SYSTEM), String.valueOf(Notes.ID_TRASH_FOLER)
 | 
						|
                    }, null);
 | 
						|
            if (c != null) {
 | 
						|
                while (c.moveToNext()) {
 | 
						|
                    gid = c.getString(SqlNote.GTASK_ID_COLUMN);
 | 
						|
                    node = mGTaskHashMap.get(gid);
 | 
						|
                    if (node != null) {
 | 
						|
                        mGTaskHashMap.remove(gid);
 | 
						|
                        doContentSync(Node.SYNC_ACTION_DEL_REMOTE, node, c);
 | 
						|
                    }
 | 
						|
 | 
						|
                    mLocalDeleteIdMap.add(c.getLong(SqlNote.ID_COLUMN));
 | 
						|
                }
 | 
						|
            } else {
 | 
						|
                Log.w(TAG, "failed to query trash folder");
 | 
						|
            }
 | 
						|
        } finally {
 | 
						|
            if (c != null) {
 | 
						|
                c.close();
 | 
						|
                c = null;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // sync folder first
 | 
						|
        syncFolder();
 | 
						|
 | 
						|
        // for note existing in database
 | 
						|
        try {
 | 
						|
            c = mContentResolver.query(Notes.CONTENT_NOTE_URI, SqlNote.PROJECTION_NOTE,
 | 
						|
                    "(type=? AND parent_id<>?)", new String[] {
 | 
						|
                            String.valueOf(Notes.TYPE_NOTE), String.valueOf(Notes.ID_TRASH_FOLER)
 | 
						|
                    }, NoteColumns.TYPE + " DESC");
 | 
						|
            if (c != null) {
 | 
						|
                while (c.moveToNext()) {
 | 
						|
                    gid = c.getString(SqlNote.GTASK_ID_COLUMN);
 | 
						|
                    node = mGTaskHashMap.get(gid);
 | 
						|
                    if (node != null) {
 | 
						|
                        mGTaskHashMap.remove(gid);
 | 
						|
                        mGidToNid.put(gid, c.getLong(SqlNote.ID_COLUMN));
 | 
						|
                        mNidToGid.put(c.getLong(SqlNote.ID_COLUMN), gid);
 | 
						|
                        syncType = node.getSyncAction(c);
 | 
						|
                    } else {
 | 
						|
                        if (c.getString(SqlNote.GTASK_ID_COLUMN).trim().length() == 0) {
 | 
						|
                            // local add
 | 
						|
                            syncType = Node.SYNC_ACTION_ADD_REMOTE;
 | 
						|
                        } else {
 | 
						|
                            // remote delete
 | 
						|
                            syncType = Node.SYNC_ACTION_DEL_LOCAL;
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
                    doContentSync(syncType, node, c);
 | 
						|
                }
 | 
						|
            } else {
 | 
						|
                Log.w(TAG, "failed to query existing note in database");
 | 
						|
            }
 | 
						|
 | 
						|
        } finally {
 | 
						|
            if (c != null) {
 | 
						|
                c.close();
 | 
						|
                c = null;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // go through remaining items
 | 
						|
        Iterator<Map.Entry<String, Node>> iter = mGTaskHashMap.entrySet().iterator();
 | 
						|
        while (iter.hasNext()) {
 | 
						|
            Map.Entry<String, Node> entry = iter.next();
 | 
						|
            node = entry.getValue();
 | 
						|
            doContentSync(Node.SYNC_ACTION_ADD_LOCAL, node, null);
 | 
						|
        }
 | 
						|
 | 
						|
        // mCancelled can be set by another thread, so we neet to check one by
 | 
						|
        // one
 | 
						|
        // clear local delete table
 | 
						|
        if (!mCancelled) {
 | 
						|
            if (!DataUtils.batchDeleteNotes(mContentResolver, mLocalDeleteIdMap)) {
 | 
						|
                throw new ActionFailureException("failed to batch-delete local deleted notes");
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // refresh local sync id
 | 
						|
        if (!mCancelled) {
 | 
						|
            GTaskClient.getInstance().commitUpdate();
 | 
						|
            refreshLocalSyncId();
 | 
						|
        }
 | 
						|
 | 
						|
    }
 | 
						|
 | 
						|
    private void syncFolder() throws NetworkFailureException {
 | 
						|
        Cursor c = null;
 | 
						|
        String gid;
 | 
						|
        Node node;
 | 
						|
        int syncType;
 | 
						|
 | 
						|
        if (mCancelled) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        // for root folder
 | 
						|
        try {
 | 
						|
            c = mContentResolver.query(ContentUris.withAppendedId(Notes.CONTENT_NOTE_URI,
 | 
						|
                    Notes.ID_ROOT_FOLDER), SqlNote.PROJECTION_NOTE, null, null, null);
 | 
						|
            if (c != null) {
 | 
						|
                c.moveToNext();
 | 
						|
                gid = c.getString(SqlNote.GTASK_ID_COLUMN);
 | 
						|
                node = mGTaskHashMap.get(gid);
 | 
						|
                if (node != null) {
 | 
						|
                    mGTaskHashMap.remove(gid);
 | 
						|
                    mGidToNid.put(gid, (long) Notes.ID_ROOT_FOLDER);
 | 
						|
                    mNidToGid.put((long) Notes.ID_ROOT_FOLDER, gid);
 | 
						|
                    // for system folder, only update remote name if necessary
 | 
						|
                    if (!node.getName().equals(
 | 
						|
                            GTaskStringUtils.MIUI_FOLDER_PREFFIX + GTaskStringUtils.FOLDER_DEFAULT))
 | 
						|
                        doContentSync(Node.SYNC_ACTION_UPDATE_REMOTE, node, c);
 | 
						|
                } else {
 | 
						|
                    doContentSync(Node.SYNC_ACTION_ADD_REMOTE, node, c);
 | 
						|
                }
 | 
						|
            } else {
 | 
						|
                Log.w(TAG, "failed to query root folder");
 | 
						|
            }
 | 
						|
        } finally {
 | 
						|
            if (c != null) {
 | 
						|
                c.close();
 | 
						|
                c = null;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // for call-note folder
 | 
						|
        try {
 | 
						|
            c = mContentResolver.query(Notes.CONTENT_NOTE_URI, SqlNote.PROJECTION_NOTE, "(_id=?)",
 | 
						|
                    new String[] {
 | 
						|
                        String.valueOf(Notes.ID_CALL_RECORD_FOLDER)
 | 
						|
                    }, null);
 | 
						|
            if (c != null) {
 | 
						|
                if (c.moveToNext()) {
 | 
						|
                    gid = c.getString(SqlNote.GTASK_ID_COLUMN);
 | 
						|
                    node = mGTaskHashMap.get(gid);
 | 
						|
                    if (node != null) {
 | 
						|
                        mGTaskHashMap.remove(gid);
 | 
						|
                        mGidToNid.put(gid, (long) Notes.ID_CALL_RECORD_FOLDER);
 | 
						|
                        mNidToGid.put((long) Notes.ID_CALL_RECORD_FOLDER, gid);
 | 
						|
                        // for system folder, only update remote name if
 | 
						|
                        // necessary
 | 
						|
                        if (!node.getName().equals(
 | 
						|
                                GTaskStringUtils.MIUI_FOLDER_PREFFIX
 | 
						|
                                        + GTaskStringUtils.FOLDER_CALL_NOTE))
 | 
						|
                            doContentSync(Node.SYNC_ACTION_UPDATE_REMOTE, node, c);
 | 
						|
                    } else {
 | 
						|
                        doContentSync(Node.SYNC_ACTION_ADD_REMOTE, node, c);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            } else {
 | 
						|
                Log.w(TAG, "failed to query call note folder");
 | 
						|
            }
 | 
						|
        } finally {
 | 
						|
            if (c != null) {
 | 
						|
                c.close();
 | 
						|
                c = null;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // for local existing folders
 | 
						|
        try {
 | 
						|
            c = mContentResolver.query(Notes.CONTENT_NOTE_URI, SqlNote.PROJECTION_NOTE,
 | 
						|
                    "(type=? AND parent_id<>?)", new String[] {
 | 
						|
                            String.valueOf(Notes.TYPE_FOLDER), String.valueOf(Notes.ID_TRASH_FOLER)
 | 
						|
                    }, NoteColumns.TYPE + " DESC");
 | 
						|
            if (c != null) {
 | 
						|
                while (c.moveToNext()) {
 | 
						|
                    gid = c.getString(SqlNote.GTASK_ID_COLUMN);
 | 
						|
                    node = mGTaskHashMap.get(gid);
 | 
						|
                    if (node != null) {
 | 
						|
                        mGTaskHashMap.remove(gid);
 | 
						|
                        mGidToNid.put(gid, c.getLong(SqlNote.ID_COLUMN));
 | 
						|
                        mNidToGid.put(c.getLong(SqlNote.ID_COLUMN), gid);
 | 
						|
                        syncType = node.getSyncAction(c);
 | 
						|
                    } else {
 | 
						|
                        if (c.getString(SqlNote.GTASK_ID_COLUMN).trim().length() == 0) {
 | 
						|
                            // local add
 | 
						|
                            syncType = Node.SYNC_ACTION_ADD_REMOTE;
 | 
						|
                        } else {
 | 
						|
                            // remote delete
 | 
						|
                            syncType = Node.SYNC_ACTION_DEL_LOCAL;
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
                    doContentSync(syncType, node, c);
 | 
						|
                }
 | 
						|
            } else {
 | 
						|
                Log.w(TAG, "failed to query existing folder");
 | 
						|
            }
 | 
						|
        } finally {
 | 
						|
            if (c != null) {
 | 
						|
                c.close();
 | 
						|
                c = null;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // for remote add folders
 | 
						|
        Iterator<Map.Entry<String, TaskList>> iter = mGTaskListHashMap.entrySet().iterator();
 | 
						|
        while (iter.hasNext()) {
 | 
						|
            Map.Entry<String, TaskList> entry = iter.next();
 | 
						|
            gid = entry.getKey();
 | 
						|
            node = entry.getValue();
 | 
						|
            if (mGTaskHashMap.containsKey(gid)) {
 | 
						|
                mGTaskHashMap.remove(gid);
 | 
						|
                doContentSync(Node.SYNC_ACTION_ADD_LOCAL, node, null);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        if (!mCancelled)
 | 
						|
            GTaskClient.getInstance().commitUpdate();
 | 
						|
    }
 | 
						|
 | 
						|
    private void doContentSync(int syncType, Node node, Cursor c) throws NetworkFailureException {
 | 
						|
        if (mCancelled) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        MetaData meta;
 | 
						|
        switch (syncType) {
 | 
						|
            case Node.SYNC_ACTION_ADD_LOCAL:
 | 
						|
                addLocalNode(node);
 | 
						|
                break;
 | 
						|
            case Node.SYNC_ACTION_ADD_REMOTE:
 | 
						|
                addRemoteNode(node, c);
 | 
						|
                break;
 | 
						|
            case Node.SYNC_ACTION_DEL_LOCAL:
 | 
						|
                meta = mMetaHashMap.get(c.getString(SqlNote.GTASK_ID_COLUMN));
 | 
						|
                if (meta != null) {
 | 
						|
                    GTaskClient.getInstance().deleteNode(meta);
 | 
						|
                }
 | 
						|
                mLocalDeleteIdMap.add(c.getLong(SqlNote.ID_COLUMN));
 | 
						|
                break;
 | 
						|
            case Node.SYNC_ACTION_DEL_REMOTE:
 | 
						|
                meta = mMetaHashMap.get(node.getGid());
 | 
						|
                if (meta != null) {
 | 
						|
                    GTaskClient.getInstance().deleteNode(meta);
 | 
						|
                }
 | 
						|
                GTaskClient.getInstance().deleteNode(node);
 | 
						|
                break;
 | 
						|
            case Node.SYNC_ACTION_UPDATE_LOCAL:
 | 
						|
                updateLocalNode(node, c);
 | 
						|
                break;
 | 
						|
            case Node.SYNC_ACTION_UPDATE_REMOTE:
 | 
						|
                updateRemoteNode(node, c);
 | 
						|
                break;
 | 
						|
            case Node.SYNC_ACTION_UPDATE_CONFLICT:
 | 
						|
                // merging both modifications maybe a good idea
 | 
						|
                // right now just use local update simply
 | 
						|
                updateRemoteNode(node, c);
 | 
						|
                break;
 | 
						|
            case Node.SYNC_ACTION_NONE:
 | 
						|
                break;
 | 
						|
            case Node.SYNC_ACTION_ERROR:
 | 
						|
            default:
 | 
						|
                throw new ActionFailureException("unkown sync action type");
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    private void addLocalNode(Node node) throws NetworkFailureException {
 | 
						|
        if (mCancelled) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        SqlNote sqlNote;
 | 
						|
        if (node instanceof TaskList) {
 | 
						|
            if (node.getName().equals(
 | 
						|
                    GTaskStringUtils.MIUI_FOLDER_PREFFIX + GTaskStringUtils.FOLDER_DEFAULT)) {
 | 
						|
                sqlNote = new SqlNote(mContext, Notes.ID_ROOT_FOLDER);
 | 
						|
            } else if (node.getName().equals(
 | 
						|
                    GTaskStringUtils.MIUI_FOLDER_PREFFIX + GTaskStringUtils.FOLDER_CALL_NOTE)) {
 | 
						|
                sqlNote = new SqlNote(mContext, Notes.ID_CALL_RECORD_FOLDER);
 | 
						|
            } else {
 | 
						|
                sqlNote = new SqlNote(mContext);
 | 
						|
                sqlNote.setContent(node.getLocalJSONFromContent());
 | 
						|
                sqlNote.setParentId(Notes.ID_ROOT_FOLDER);
 | 
						|
            }
 | 
						|
        } else {
 | 
						|
            sqlNote = new SqlNote(mContext);
 | 
						|
            JSONObject js = node.getLocalJSONFromContent();
 | 
						|
            try {
 | 
						|
                if (js.has(GTaskStringUtils.META_HEAD_NOTE)) {
 | 
						|
                    JSONObject note = js.getJSONObject(GTaskStringUtils.META_HEAD_NOTE);
 | 
						|
                    if (note.has(NoteColumns.ID)) {
 | 
						|
                        long id = note.getLong(NoteColumns.ID);
 | 
						|
                        if (DataUtils.existInNoteDatabase(mContentResolver, id)) {
 | 
						|
                            // the id is not available, have to create a new one
 | 
						|
                            note.remove(NoteColumns.ID);
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                if (js.has(GTaskStringUtils.META_HEAD_DATA)) {
 | 
						|
                    JSONArray dataArray = js.getJSONArray(GTaskStringUtils.META_HEAD_DATA);
 | 
						|
                    for (int i = 0; i < dataArray.length(); i++) {
 | 
						|
                        JSONObject data = dataArray.getJSONObject(i);
 | 
						|
                        if (data.has(DataColumns.ID)) {
 | 
						|
                            long dataId = data.getLong(DataColumns.ID);
 | 
						|
                            if (DataUtils.existInDataDatabase(mContentResolver, dataId)) {
 | 
						|
                                // the data id is not available, have to create
 | 
						|
                                // a new one
 | 
						|
                                data.remove(DataColumns.ID);
 | 
						|
                            }
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
 | 
						|
                }
 | 
						|
            } catch (JSONException e) {
 | 
						|
                Log.w(TAG, e.toString());
 | 
						|
                e.printStackTrace();
 | 
						|
            }
 | 
						|
            sqlNote.setContent(js);
 | 
						|
 | 
						|
            Long parentId = mGidToNid.get(((Task) node).getParent().getGid());
 | 
						|
            if (parentId == null) {
 | 
						|
                Log.e(TAG, "cannot find task's parent id locally");
 | 
						|
                throw new ActionFailureException("cannot add local node");
 | 
						|
            }
 | 
						|
            sqlNote.setParentId(parentId.longValue());
 | 
						|
        }
 | 
						|
 | 
						|
        // create the local node
 | 
						|
        sqlNote.setGtaskId(node.getGid());
 | 
						|
        sqlNote.commit(false);
 | 
						|
 | 
						|
        // update gid-nid mapping
 | 
						|
        mGidToNid.put(node.getGid(), sqlNote.getId());
 | 
						|
        mNidToGid.put(sqlNote.getId(), node.getGid());
 | 
						|
 | 
						|
        // update meta
 | 
						|
        updateRemoteMeta(node.getGid(), sqlNote);
 | 
						|
    }
 | 
						|
 | 
						|
    private void updateLocalNode(Node node, Cursor c) throws NetworkFailureException {
 | 
						|
        if (mCancelled) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        SqlNote sqlNote;
 | 
						|
        // update the note locally
 | 
						|
        sqlNote = new SqlNote(mContext, c);
 | 
						|
        sqlNote.setContent(node.getLocalJSONFromContent());
 | 
						|
 | 
						|
        Long parentId = (node instanceof Task) ? mGidToNid.get(((Task) node).getParent().getGid())
 | 
						|
                : new Long(Notes.ID_ROOT_FOLDER);
 | 
						|
        if (parentId == null) {
 | 
						|
            Log.e(TAG, "cannot find task's parent id locally");
 | 
						|
            throw new ActionFailureException("cannot update local node");
 | 
						|
        }
 | 
						|
        sqlNote.setParentId(parentId.longValue());
 | 
						|
        sqlNote.commit(true);
 | 
						|
 | 
						|
        // update meta info
 | 
						|
        updateRemoteMeta(node.getGid(), sqlNote);
 | 
						|
    }
 | 
						|
 | 
						|
    private void addRemoteNode(Node node, Cursor c) throws NetworkFailureException {
 | 
						|
        if (mCancelled) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        SqlNote sqlNote = new SqlNote(mContext, c);
 | 
						|
        Node n;
 | 
						|
 | 
						|
        // update remotely
 | 
						|
        if (sqlNote.isNoteType()) {
 | 
						|
            Task task = new Task();
 | 
						|
            task.setContentByLocalJSON(sqlNote.getContent());
 | 
						|
 | 
						|
            String parentGid = mNidToGid.get(sqlNote.getParentId());
 | 
						|
            if (parentGid == null) {
 | 
						|
                Log.e(TAG, "cannot find task's parent tasklist");
 | 
						|
                throw new ActionFailureException("cannot add remote task");
 | 
						|
            }
 | 
						|
            mGTaskListHashMap.get(parentGid).addChildTask(task);
 | 
						|
 | 
						|
            GTaskClient.getInstance().createTask(task);
 | 
						|
            n = (Node) task;
 | 
						|
 | 
						|
            // add meta
 | 
						|
            updateRemoteMeta(task.getGid(), sqlNote);
 | 
						|
        } else {
 | 
						|
            TaskList tasklist = null;
 | 
						|
 | 
						|
            // we need to skip folder if it has already existed
 | 
						|
            String folderName = GTaskStringUtils.MIUI_FOLDER_PREFFIX;
 | 
						|
            if (sqlNote.getId() == Notes.ID_ROOT_FOLDER)
 | 
						|
                folderName += GTaskStringUtils.FOLDER_DEFAULT;
 | 
						|
            else if (sqlNote.getId() == Notes.ID_CALL_RECORD_FOLDER)
 | 
						|
                folderName += GTaskStringUtils.FOLDER_CALL_NOTE;
 | 
						|
            else
 | 
						|
                folderName += sqlNote.getSnippet();
 | 
						|
 | 
						|
            Iterator<Map.Entry<String, TaskList>> iter = mGTaskListHashMap.entrySet().iterator();
 | 
						|
            while (iter.hasNext()) {
 | 
						|
                Map.Entry<String, TaskList> entry = iter.next();
 | 
						|
                String gid = entry.getKey();
 | 
						|
                TaskList list = entry.getValue();
 | 
						|
 | 
						|
                if (list.getName().equals(folderName)) {
 | 
						|
                    tasklist = list;
 | 
						|
                    if (mGTaskHashMap.containsKey(gid)) {
 | 
						|
                        mGTaskHashMap.remove(gid);
 | 
						|
                    }
 | 
						|
                    break;
 | 
						|
                }
 | 
						|
            }
 | 
						|
 | 
						|
            // no match we can add now
 | 
						|
            if (tasklist == null) {
 | 
						|
                tasklist = new TaskList();
 | 
						|
                tasklist.setContentByLocalJSON(sqlNote.getContent());
 | 
						|
                GTaskClient.getInstance().createTaskList(tasklist);
 | 
						|
                mGTaskListHashMap.put(tasklist.getGid(), tasklist);
 | 
						|
            }
 | 
						|
            n = (Node) tasklist;
 | 
						|
        }
 | 
						|
 | 
						|
        // update local note
 | 
						|
        sqlNote.setGtaskId(n.getGid());
 | 
						|
        sqlNote.commit(false);
 | 
						|
        sqlNote.resetLocalModified();
 | 
						|
        sqlNote.commit(true);
 | 
						|
 | 
						|
        // gid-id mapping
 | 
						|
        mGidToNid.put(n.getGid(), sqlNote.getId());
 | 
						|
        mNidToGid.put(sqlNote.getId(), n.getGid());
 | 
						|
    }
 | 
						|
 | 
						|
    private void updateRemoteNode(Node node, Cursor c) throws NetworkFailureException {
 | 
						|
        if (mCancelled) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        SqlNote sqlNote = new SqlNote(mContext, c);
 | 
						|
 | 
						|
        // update remotely
 | 
						|
        node.setContentByLocalJSON(sqlNote.getContent());
 | 
						|
        GTaskClient.getInstance().addUpdateNode(node);
 | 
						|
 | 
						|
        // update meta
 | 
						|
        updateRemoteMeta(node.getGid(), sqlNote);
 | 
						|
 | 
						|
        // move task if necessary
 | 
						|
        if (sqlNote.isNoteType()) {
 | 
						|
            Task task = (Task) node;
 | 
						|
            TaskList preParentList = task.getParent();
 | 
						|
 | 
						|
            String curParentGid = mNidToGid.get(sqlNote.getParentId());
 | 
						|
            if (curParentGid == null) {
 | 
						|
                Log.e(TAG, "cannot find task's parent tasklist");
 | 
						|
                throw new ActionFailureException("cannot update remote task");
 | 
						|
            }
 | 
						|
            TaskList curParentList = mGTaskListHashMap.get(curParentGid);
 | 
						|
 | 
						|
            if (preParentList != curParentList) {
 | 
						|
                preParentList.removeChildTask(task);
 | 
						|
                curParentList.addChildTask(task);
 | 
						|
                GTaskClient.getInstance().moveTask(task, preParentList, curParentList);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // clear local modified flag
 | 
						|
        sqlNote.resetLocalModified();
 | 
						|
        sqlNote.commit(true);
 | 
						|
    }
 | 
						|
 | 
						|
    private void updateRemoteMeta(String gid, SqlNote sqlNote) throws NetworkFailureException {
 | 
						|
        if (sqlNote != null && sqlNote.isNoteType()) {
 | 
						|
            MetaData metaData = mMetaHashMap.get(gid);
 | 
						|
            if (metaData != null) {
 | 
						|
                metaData.setMeta(gid, sqlNote.getContent());
 | 
						|
                GTaskClient.getInstance().addUpdateNode(metaData);
 | 
						|
            } else {
 | 
						|
                metaData = new MetaData();
 | 
						|
                metaData.setMeta(gid, sqlNote.getContent());
 | 
						|
                mMetaList.addChildTask(metaData);
 | 
						|
                mMetaHashMap.put(gid, metaData);
 | 
						|
                GTaskClient.getInstance().createTask(metaData);
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    private void refreshLocalSyncId() throws NetworkFailureException {
 | 
						|
        if (mCancelled) {
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        // get the latest gtask list
 | 
						|
        mGTaskHashMap.clear();
 | 
						|
        mGTaskListHashMap.clear();
 | 
						|
        mMetaHashMap.clear();
 | 
						|
        initGTaskList();
 | 
						|
 | 
						|
        Cursor c = null;
 | 
						|
        try {
 | 
						|
            c = mContentResolver.query(Notes.CONTENT_NOTE_URI, SqlNote.PROJECTION_NOTE,
 | 
						|
                    "(type<>? AND parent_id<>?)", new String[] {
 | 
						|
                            String.valueOf(Notes.TYPE_SYSTEM), String.valueOf(Notes.ID_TRASH_FOLER)
 | 
						|
                    }, NoteColumns.TYPE + " DESC");
 | 
						|
            if (c != null) {
 | 
						|
                while (c.moveToNext()) {
 | 
						|
                    String gid = c.getString(SqlNote.GTASK_ID_COLUMN);
 | 
						|
                    Node node = mGTaskHashMap.get(gid);
 | 
						|
                    if (node != null) {
 | 
						|
                        mGTaskHashMap.remove(gid);
 | 
						|
                        ContentValues values = new ContentValues();
 | 
						|
                        values.put(NoteColumns.SYNC_ID, node.getLastModified());
 | 
						|
                        mContentResolver.update(ContentUris.withAppendedId(Notes.CONTENT_NOTE_URI,
 | 
						|
                                c.getLong(SqlNote.ID_COLUMN)), values, null, null);
 | 
						|
                    } else {
 | 
						|
                        Log.e(TAG, "something is missed");
 | 
						|
                        throw new ActionFailureException(
 | 
						|
                                "some local items don't have gid after sync");
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            } else {
 | 
						|
                Log.w(TAG, "failed to query local note to refresh sync id");
 | 
						|
            }
 | 
						|
        } finally {
 | 
						|
            if (c != null) {
 | 
						|
                c.close();
 | 
						|
                c = null;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public String getSyncAccount() {
 | 
						|
        return GTaskClient.getInstance().getSyncAccount().name;
 | 
						|
    }
 | 
						|
 | 
						|
    public void cancelSync() {
 | 
						|
        mCancelled = true;
 | 
						|
    }
 | 
						|
}
 |