// Copyright 2015 The Project Buendia Authors // // 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 distrib- // uted 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 // specific language governing permissions and limitations under the License. package org.projectbuendia.client.models.tasks; import android.content.ContentResolver; import android.database.Cursor; import android.net.Uri; import android.os.AsyncTask; import com.android.volley.toolbox.RequestFuture; import org.projectbuendia.client.events.CrudEventBus; import org.projectbuendia.client.events.data.ItemFetchFailedEvent; import org.projectbuendia.client.events.data.ItemFetchedEvent; import org.projectbuendia.client.filter.db.patient.UuidFilter; import org.projectbuendia.client.models.Patient; import org.projectbuendia.client.models.LoaderSet; import org.projectbuendia.client.net.Server; import org.projectbuendia.client.json.JsonPatient; import org.projectbuendia.client.providers.Contracts.Patients; import org.projectbuendia.client.utils.Logger; import java.util.concurrent.ExecutionException; /** * An {@link AsyncTask} that downloads one specific patient from the server and * stores it locally (unlike sync, which ensures all patients are stored locally). * <p/> * <p>Posts an {@link ItemFetchedEvent} or an {@link ItemFetchFailedEvent} on * the given {@link CrudEventBus} to indicate success or failure. */ public class DownloadSinglePatientTask extends AsyncTask<Void, Void, ItemFetchFailedEvent> { private static final Logger LOG = Logger.create(); private final TaskFactory mTaskFactory; private final LoaderSet mLoaderSet; private final Server mServer; private final ContentResolver mContentResolver; private final String mPatientId; private final CrudEventBus mBus; private String mUuid; /** Creates a new {@link DownloadSinglePatientTask}. */ public DownloadSinglePatientTask( TaskFactory taskFactory, LoaderSet loaderSet, Server server, ContentResolver contentResolver, String patientId, CrudEventBus bus) { mTaskFactory = taskFactory; mLoaderSet = loaderSet; mServer = server; mContentResolver = contentResolver; mPatientId = patientId; mBus = bus; } @Override protected ItemFetchFailedEvent doInBackground(Void... params) { RequestFuture<JsonPatient> future = RequestFuture.newFuture(); // Try to download the specified patient from the server. LOG.i("Downloading single patient %s from server...", mPatientId); mServer.getPatient(mPatientId, future, future); JsonPatient json; try { json = future.get(); } catch (InterruptedException e) { return new ItemFetchFailedEvent("interrupted", mPatientId, e); } catch (ExecutionException e) { return new ItemFetchFailedEvent("network error", mPatientId, e); } if (json == null) { LOG.i("Patient ID %s not found on server", mPatientId); return new ItemFetchFailedEvent("not found", mPatientId); } // Update the patient in the local database. Patient patient = Patient.fromJson(json); Uri uri = null; try (Cursor c = mContentResolver.query(Patients.CONTENT_URI, null, Patients.ID + " = ?", new String[] {mPatientId}, null)) { if (c.moveToNext()) { LOG.i("Updating existing local patient."); uri = Patients.CONTENT_URI.buildUpon().appendPath(mPatientId).build(); mContentResolver.update(uri, patient.toContentValues(), Patients.ID + " = ?", new String[] {mPatientId}); } else { LOG.i("Adding new local copy of patient."); uri = mContentResolver.insert( Patients.CONTENT_URI, patient.toContentValues()); } } // Record the UUID to use for fetching the patient back from the local database. if (uri == null || uri.equals(Uri.EMPTY)) { LOG.i("Patient ID %s not found on server", mPatientId); return new ItemFetchFailedEvent("not found", mPatientId); } if (json.uuid == null) { return new ItemFetchFailedEvent("server error", mPatientId); } mUuid = json.uuid; return null; } @Override protected void onPostExecute(ItemFetchFailedEvent event) { // If an error occurred, post the error event. if (event != null) { mBus.post(event); return; } // After updating a patient, we fetch the patient from the database. The // result of the fetch determines if adding a patient was truly successful // and propagates a new event to report success/failure. mTaskFactory.newFetchItemTask( Patients.CONTENT_URI, null, new UuidFilter(), mUuid, mLoaderSet.patientLoader, mBus).execute(); } }