Cloud sync improvements

pull/212/head
M66B 2 years ago
parent 630164979a
commit 632e045c7c

@ -55,8 +55,9 @@ public class CloudSync {
private static final Map<String, Pair<byte[], byte[]>> keyCache = new HashMap<>(); private static final Map<String, Pair<byte[], byte[]>> keyCache = new HashMap<>();
public static JSONObject perform(Context context, String user, String password, JSONObject jrequest) public static JSONObject perform(Context context, String user, String password, String command, JSONObject jrequest)
throws GeneralSecurityException, JSONException, IOException { throws GeneralSecurityException, JSONException, IOException {
jrequest.put("command", command);
List<JSONObject> responses = new ArrayList<>(); List<JSONObject> responses = new ArrayList<>();
for (JSONArray batch : partition(jrequest.getJSONArray("items"))) { for (JSONArray batch : partition(jrequest.getJSONArray("items"))) {
jrequest.put("items", batch); jrequest.put("items", batch);

@ -1542,157 +1542,161 @@ public class FragmentOptionsBackup extends FragmentBase implements SharedPrefere
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context); SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
int sync_status = prefs.getInt("sync_status", 0); int sync_status = prefs.getInt("sync_status", 0);
JSONObject jstatus = new JSONObject(); JSONObject jsyncstatus = new JSONObject();
jstatus.put("key", "sync.status"); jsyncstatus.put("key", "sync.status");
jstatus.put("rev", sync_status * 0); jsyncstatus.put("rev", sync_status);
JSONArray jitems = new JSONArray(); JSONArray jitems = new JSONArray();
jitems.put(jstatus); jitems.put(jsyncstatus);
jrequest.put("items", jitems); jrequest.put("items", jitems);
JSONObject jresponse = CloudSync.perform(context, user, password, jrequest); JSONObject jresponse = CloudSync.perform(context, user, password, "read", jrequest);
jitems = jresponse.getJSONArray("items"); jitems = jresponse.getJSONArray("items");
int count = jresponse.getInt("count");
if (count == 0) { if (jitems.length() == 0) {
Log.i("Cloud server is empty"); Log.i("Cloud server is empty");
List<EntityAccount> accounts = db.account().getSynchronizingAccounts(null); List<EntityAccount> accounts = db.account().getSynchronizingAccounts(null);
Log.i("Cloud accounts=" + (accounts == null ? null : accounts.size())); Log.i("Cloud accounts=" + (accounts == null ? null : accounts.size()));
if (accounts != null && accounts.size() != 0) { if (accounts == null || accounts.size() == 0)
JSONArray jupload = new JSONArray(); return null; // nothing to offer
JSONArray juuids = new JSONArray(); JSONArray jupload = new JSONArray();
for (EntityAccount account : accounts)
if (!TextUtils.isEmpty(account.uuid)) { JSONArray juuids = new JSONArray();
juuids.put(account.uuid); for (EntityAccount account : accounts)
if (!TextUtils.isEmpty(account.uuid)) {
JSONArray jidentities = new JSONArray(); juuids.put(account.uuid);
List<EntityIdentity> identities = db.identity().getIdentities(account.id);
if (identities != null) JSONArray jidentities = new JSONArray();
for (EntityIdentity identity : identities) List<EntityIdentity> identities = db.identity().getIdentities(account.id);
if (!TextUtils.isEmpty(identity.uuid)) { if (identities != null)
jidentities.put(identity.uuid); for (EntityIdentity identity : identities)
if (!TextUtils.isEmpty(identity.uuid)) {
JSONObject jitem = new JSONObject(); jidentities.put(identity.uuid);
jitem.put("key", "identity." + identity.uuid);
jitem.put("val", identity.toJSON().toString()); JSONObject jitem = new JSONObject();
jitem.put("rev", 1); jitem.put("key", "identity." + identity.uuid);
jupload.put(jitem); jitem.put("val", identity.toJSON().toString());
} jitem.put("rev", 1);
jupload.put(jitem);
}
JSONObject jaccountdata = new JSONObject(); JSONObject jaccountdata = new JSONObject();
jaccountdata.put("account", account.toJSON()); jaccountdata.put("account", account.toJSON());
jaccountdata.put("identities", jidentities); jaccountdata.put("identities", jidentities);
JSONObject jitem = new JSONObject(); JSONObject jitem = new JSONObject();
jitem.put("key", "account." + account.uuid); jitem.put("key", "account." + account.uuid);
jitem.put("val", jaccountdata.toString()); jitem.put("val", jaccountdata.toString());
jitem.put("rev", 1); jitem.put("rev", 1);
jupload.put(jitem); jupload.put(jitem);
} }
JSONObject jaccounts = new JSONObject(); JSONObject jaccounts = new JSONObject();
jaccounts.put("uuids", juuids); jaccounts.put("uuids", juuids);
JSONObject jstatusdata = new JSONObject(); JSONObject jstatus = new JSONObject();
jstatusdata.put("accounts", jaccounts); jstatus.put("accounts", jaccounts);
jstatus.put("key", "sync.status"); jsyncstatus.put("key", "sync.status");
jstatus.put("val", jstatusdata.toString()); jsyncstatus.put("val", jstatus.toString());
jstatus.put("rev", 1); jsyncstatus.put("rev", 1);
jupload.put(jstatus); jupload.put(jsyncstatus);
jrequest.put("command", "write"); jrequest.put("items", jupload);
jrequest.put("items", jupload); CloudSync.perform(context, user, password, "write", jrequest);
CloudSync.perform(context, user, password, jrequest);
prefs.edit().putInt("sync_status", 1).apply(); prefs.edit().putInt("sync_status", 1).apply();
}
return null; return null;
} else if (count == 1) { } else if (jitems.length() == 1) {
if (jitems.length() == 1) { JSONObject jitem = jitems.getJSONObject(0);
// New revision int rev = jitem.getInt("rev");
JSONObject jitem = jitems.getJSONObject(0); Log.i("Cloud status revision=" + rev + "/" + sync_status);
Log.i("Cloud status revision=" + jitem.getInt("rev") + "/" + sync_status);
if (BuildConfig.DEBUG)
JSONArray jdownload = new JSONArray(); sync_status--;
// Get accounts if (rev <= sync_status)
JSONObject jstatusdata = new JSONObject(jitem.getString("val")); return null; // no changes
JSONObject jaccountinfo = jstatusdata.getJSONObject("accounts");
JSONArray juuids = jaccountinfo.getJSONArray("uuids"); // New revision
for (int i = 0; i < juuids.length(); i++) { JSONArray jdownload = new JSONArray();
String uuid = juuids.getString(i);
JSONObject jaccount = new JSONObject(); // Get accounts
jaccount.put("key", "account." + uuid); JSONObject jstatus = new JSONObject(jitem.getString("val"));
jaccount.put("rev", sync_status * 0); JSONObject jaccounts = jstatus.getJSONObject("accounts");
jdownload.put(jaccount); JSONArray juuids = jaccounts.getJSONArray("uuids");
Log.i("Cloud account " + uuid); for (int i = 0; i < juuids.length(); i++) {
String uuid = juuids.getString(i);
JSONObject jaccount = new JSONObject();
jaccount.put("key", "account." + uuid);
jaccount.put("rev", sync_status);
jdownload.put(jaccount);
Log.i("Cloud account " + uuid);
}
if (jdownload.length() > 0) {
Log.i("Cloud getting accounts");
jrequest.put("items", jdownload);
jresponse = CloudSync.perform(context, user, password, "sync", jrequest);
// Process accounts
Log.i("Cloud processing accounts");
jitems = jresponse.getJSONArray("items");
jdownload = new JSONArray();
for (int i = 0; i < jitems.length(); i++) {
JSONObject jaccount = jitems.getJSONObject(i);
String value = jaccount.getString("val");
int revision = jaccount.getInt("rev");
JSONObject jaccountdata = new JSONObject(value);
EntityAccount raccount = EntityAccount.fromJSON(jaccountdata.getJSONObject("account"));
EntityAccount laccount = db.account().getAccountByUUID(raccount.uuid);
JSONArray jidentities = jaccountdata.getJSONArray("identities");
Log.i("Cloud account " + raccount.uuid + "=" + (laccount == null ? "insert" : "update") +
" rev=" + revision +
" identities=" + jidentities +
" size=" + value.length());
for (int j = 0; j < jidentities.length(); j++) {
JSONObject jidentity = new JSONObject();
jidentity.put("key", "identity." + jidentities.getString(j));
jidentity.put("rev", sync_status);
jdownload.put(jidentity);
}
} }
if (jdownload.length() > 0) { if (jdownload.length() > 0) {
Log.i("Cloud getting accounts"); // Get identities
Log.i("Cloud getting identities");
jrequest.put("items", jdownload); jrequest.put("items", jdownload);
jresponse = CloudSync.perform(context, user, password, jrequest); jresponse = CloudSync.perform(context, user, password, "sync", jrequest);
// Process accounts // Process identities
Log.i("Cloud processing accounts"); Log.i("Cloud processing identities");
jitems = jresponse.getJSONArray("items"); jitems = jresponse.getJSONArray("items");
jdownload = new JSONArray();
for (int i = 0; i < jitems.length(); i++) { for (int i = 0; i < jitems.length(); i++) {
JSONObject jaccount = jitems.getJSONObject(i); JSONObject jaccount = jitems.getJSONObject(i);
String key = jaccount.getString("key");
String value = jaccount.getString("val"); String value = jaccount.getString("val");
int revision = jaccount.getInt("rev"); int revision = jaccount.getInt("rev");
String uuid = key.split("\\.")[1]; EntityIdentity ridentity = EntityIdentity.fromJSON(new JSONObject(value));
EntityAccount account = db.account().getAccountByUUID(uuid); EntityIdentity lidentity = db.identity().getIdentityByUUID(ridentity.uuid);
JSONObject jaccountdata = new JSONObject(value); Log.i("Cloud identity " + ridentity.uuid + "=" + (lidentity == null ? "insert" : "update") +
JSONArray jidentities = jaccountdata.getJSONArray("identities");
Log.i("Cloud account " + uuid + "=" + (account != null) +
" rev=" + revision + " rev=" + revision +
" identities=" + jidentities +
" size=" + value.length()); " size=" + value.length());
for (int j = 0; j < jidentities.length(); j++) {
JSONObject jidentity = new JSONObject();
jidentity.put("key", "identity." + jidentities.getString(j));
jidentity.put("rev", sync_status * 0);
jdownload.put(jidentity);
}
}
if (jdownload.length() > 0) {
// Get identities
Log.i("Cloud getting identities");
jrequest.put("items", jdownload);
jresponse = CloudSync.perform(context, user, password, jrequest);
// Process identities
Log.i("Cloud processing identities");
jitems = jresponse.getJSONArray("items");
for (int i = 0; i < jitems.length(); i++) {
JSONObject jaccount = jitems.getJSONObject(i);
String key = jaccount.getString("key");
String value = jaccount.getString("val");
int revision = jaccount.getInt("rev");
String uuid = key.split("\\.")[1];
EntityIdentity identity = db.identity().getIdentityByUUID(uuid);
Log.i("Cloud identity " + uuid + "=" + (identity != null) +
" rev=" + revision +
" size=" + value.length());
}
} }
} }
} else {
// No changes
} }
prefs.edit().putInt("sync_status", rev).apply();
} else } else
throw new IllegalArgumentException("Expected one status item"); throw new IllegalArgumentException("Expected one status item");
} else } else
CloudSync.perform(context, user, password, jrequest); CloudSync.perform(context, user, password, command, jrequest);
return null; return null;
} }

Loading…
Cancel
Save