Cloud sync: insert/update accounts/identities

pull/212/head
M66B 2 years ago
parent e53f65c90d
commit 9b57abd880

@ -75,6 +75,7 @@ public class CloudSync {
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context); SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
String user = prefs.getString("cloud_user", null); String user = prefs.getString("cloud_user", null);
String password = prefs.getString("cloud_password", null); String password = prefs.getString("cloud_password", null);
if (TextUtils.isEmpty(user) || TextUtils.isEmpty(password)) if (TextUtils.isEmpty(user) || TextUtils.isEmpty(password))
return; return;
if (!ActivityBilling.isPro(context)) if (!ActivityBilling.isPro(context))
@ -83,7 +84,7 @@ public class CloudSync {
JSONObject jrequest = new JSONObject(); JSONObject jrequest = new JSONObject();
if ("sync".equals(command)) { if ("sync".equals(command)) {
long lrevision = prefs.getLong("cloud_revision", new Date().getTime()); long lrevision = prefs.getLong("cloud_lrevision", 0);
Log.i("Cloud local revision=" + lrevision + " (" + new Date(lrevision) + ")"); Log.i("Cloud local revision=" + lrevision + " (" + new Date(lrevision) + ")");
Long lastUpdate = updateSyncdata(context); Long lastUpdate = updateSyncdata(context);
@ -91,7 +92,8 @@ public class CloudSync {
if (lastUpdate != null && lrevision > lastUpdate) { if (lastUpdate != null && lrevision > lastUpdate) {
Log.w("Cloud invalid local revision" + Log.w("Cloud invalid local revision" +
" lrevision=" + lrevision + " last=" + lastUpdate); " lrevision=" + lrevision + " last=" + lastUpdate);
prefs.edit().putLong("cloud_revision", lastUpdate).apply(); lrevision = lastUpdate;
prefs.edit().putLong("cloud_lrevision", lrevision).apply();
} }
JSONObject jsyncstatus = new JSONObject(); JSONObject jsyncstatus = new JSONObject();
@ -108,7 +110,9 @@ public class CloudSync {
if (jitems.length() == 0) { if (jitems.length() == 0) {
Log.i("Cloud server is empty"); Log.i("Cloud server is empty");
sendLocalData(context, user, password, lrevision); sendLocalData(context, user, password, lrevision == 0
? (lastUpdate == null ? new Date().getTime() : lastUpdate)
: lrevision);
} else if (jitems.length() == 1) { } else if (jitems.length() == 1) {
Log.i("Cloud sync check"); Log.i("Cloud sync check");
jsyncstatus = jitems.getJSONObject(0); jsyncstatus = jitems.getJSONObject(0);
@ -134,11 +138,11 @@ public class CloudSync {
if (lastUpdate >= rrevision) if (lastUpdate >= rrevision)
sendLocalData(context, user, password, lastUpdate); sendLocalData(context, user, password, lastUpdate);
else else
receiveRemoteData(context, user, password, lrevision, jstatus); receiveRemoteData(context, user, password, lrevision, rrevision, jstatus);
} else } else
receiveRemoteData(context, user, password, lrevision, jstatus); receiveRemoteData(context, user, password, lrevision, rrevision, jstatus);
else if (BuildConfig.DEBUG && false) else if (BuildConfig.DEBUG && false)
receiveRemoteData(context, user, password, lrevision - 1, jstatus); receiveRemoteData(context, user, password, lrevision - 1, rrevision, jstatus);
} else } else
throw new IllegalArgumentException("Expected one status item"); throw new IllegalArgumentException("Expected one status item");
} else { } else {
@ -156,10 +160,10 @@ public class CloudSync {
Long last = null; Long last = null;
List<EntityAccount> accounts = db.account().getSynchronizingAccounts(null); List<EntityAccount> accounts = db.account().getAccounts();
if (accounts != null) if (accounts != null)
for (EntityAccount account : accounts) for (EntityAccount account : accounts)
if (!TextUtils.isEmpty(account.uuid)) { if (account.synchronize && !TextUtils.isEmpty(account.uuid)) {
EntityAccount aexisting = null; EntityAccount aexisting = null;
File afile = new File(dir, "account." + account.uuid + ".json"); File afile = new File(dir, "account." + account.uuid + ".json");
if (afile.exists()) if (afile.exists())
@ -182,7 +186,7 @@ public class CloudSync {
List<EntityIdentity> identities = db.identity().getIdentities(account.id); List<EntityIdentity> identities = db.identity().getIdentities(account.id);
if (identities != null) if (identities != null)
for (EntityIdentity identity : identities) for (EntityIdentity identity : identities)
if (!TextUtils.isEmpty(identity.uuid)) { if (identity.synchronize && !TextUtils.isEmpty(identity.uuid)) {
EntityIdentity iexisting = null; EntityIdentity iexisting = null;
File ifile = new File(dir, "identity." + identity.uuid + ".json"); File ifile = new File(dir, "identity." + identity.uuid + ".json");
if (ifile.exists()) if (ifile.exists())
@ -233,9 +237,12 @@ public class CloudSync {
if (!TextUtils.isEmpty(identity.uuid)) { if (!TextUtils.isEmpty(identity.uuid)) {
jidentitieuuids.put(identity.uuid); jidentitieuuids.put(identity.uuid);
JSONObject jidentity = identity.toJSON();
jidentity.put("account_uuid", account.uuid);
JSONObject jidentitykv = new JSONObject(); JSONObject jidentitykv = new JSONObject();
jidentitykv.put("key", "identity." + identity.uuid); jidentitykv.put("key", "identity." + identity.uuid);
jidentitykv.put("val", identity.toJSON().toString()); jidentitykv.put("val", jidentity.toString());
jidentitykv.put("rev", lrevision); jidentitykv.put("rev", lrevision);
jupload.put(jidentitykv); jupload.put(jidentitykv);
} }
@ -285,10 +292,10 @@ public class CloudSync {
jrequest.put("items", jupload); jrequest.put("items", jupload);
call(context, user, password, "write", jrequest); call(context, user, password, "write", jrequest);
prefs.edit().putLong("cloud_revision", lrevision).apply(); prefs.edit().putLong("cloud_lrevision", lrevision).apply();
} }
private static void receiveRemoteData(Context context, String user, String password, long lrevision, JSONObject jstatus) private static void receiveRemoteData(Context context, String user, String password, long lrevision, long rrevision, JSONObject jstatus)
throws JSONException, GeneralSecurityException, IOException { throws JSONException, GeneralSecurityException, IOException {
DB db = DB.getInstance(context); DB db = DB.getInstance(context);
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context); SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
@ -333,23 +340,15 @@ public class CloudSync {
EntityFolder left = null; EntityFolder left = null;
if (jaccount.has("swipe_left_name") && !jaccount.isNull("swipe_left_name")) { if (jaccount.has("swipe_left_name") && !jaccount.isNull("swipe_left_name")) {
left = new EntityFolder(); left = new EntityFolder();
left.account = null;
left.name = jaccount.getString("swipe_left_name"); left.name = jaccount.getString("swipe_left_name");
left.type = jaccount.getString("swipe_left_type"); left.type = jaccount.getString("swipe_left_type");
left.setProperties();
//left.setSpecials(account);
//folder.id = db.folder().insertFolder(folder);
} }
EntityFolder right = null; EntityFolder right = null;
if (jaccount.has("swipe_right_name") && !jaccount.isNull("swipe_right_name")) { if (jaccount.has("swipe_right_name") && !jaccount.isNull("swipe_right_name")) {
right = new EntityFolder(); right = new EntityFolder();
right.account = null;
right.name = jaccount.getString("swipe_right_name"); right.name = jaccount.getString("swipe_right_name");
right.type = jaccount.getString("swipe_right_type"); right.type = jaccount.getString("swipe_right_type");
right.setProperties();
//right.setSpecials(account);
//folder.id = db.folder().insertFolder(folder);
} }
Log.i("Cloud account " + raccount.uuid + "=" + Log.i("Cloud account " + raccount.uuid + "=" +
@ -362,6 +361,69 @@ public class CloudSync {
" identities=" + jidentities + " identities=" + jidentities +
" size=" + value.length()); " size=" + value.length());
raccount.id = null;
try {
db.beginTransaction();
if (laccount == null) {
raccount.notify = false; // TODO
if (raccount.swipe_left != null && raccount.swipe_left > 0)
raccount.swipe_left = null;
if (raccount.swipe_right != null && raccount.swipe_right > 0)
raccount.swipe_right = null;
raccount.move_to = null; // TODO
raccount.id = db.account().insertAccount(raccount);
if (raccount.protocol == EntityAccount.TYPE_POP) {
for (EntityFolder folder : EntityFolder.getPopFolders(context)) {
EntityFolder existing = db.folder().getFolderByType(raccount.id, folder.type);
if (existing == null) {
folder.account = raccount.id;
folder.id = db.folder().insertFolder(folder);
}
}
} else {
if (left != null) {
left.account = raccount.id;
left.setProperties();
left.setSpecials(raccount);
left.id = db.folder().insertFolder(left);
}
if (right != null) {
right.account = raccount.id;
right.setProperties();
right.setSpecials(raccount);
right.id = db.folder().insertFolder(right);
}
db.account().setAccountSwipes(raccount.id,
left == null ? null : left.id,
right == null ? null : right.id);
}
} else {
raccount.id = laccount.id;
raccount.notify = laccount.notify; // TODO
raccount.swipe_left = laccount.swipe_left; // TODO
raccount.swipe_right = laccount.swipe_right; // TODO
raccount.move_to = laccount.move_to; // TODO
db.account().updateAccount(raccount);
}
if (raccount.id != null) {
if (raccount.primary) {
db.account().resetPrimary();
db.account().setAccountPrimary(raccount.id, true);
}
db.account().setLastModified(raccount.id, rrevision);
}
db.setTransactionSuccessful();
} finally {
db.endTransaction();
}
for (int j = 0; j < jidentities.length(); j++) { for (int j = 0; j < jidentities.length(); j++) {
JSONObject jidentitykv = new JSONObject(); JSONObject jidentitykv = new JSONObject();
jidentitykv.put("key", "identity." + jidentities.getString(j)); jidentitykv.put("key", "identity." + jidentities.getString(j));
@ -393,15 +455,40 @@ public class CloudSync {
? "equal" : "update")) + ? "equal" : "update")) +
" rev=" + revision + " rev=" + revision +
" size=" + value.length()); " size=" + value.length());
ridentity.id = null;
ridentity.primary = false;
ridentity.last_modified = rrevision;
try {
db.beginTransaction();
if (lidentity == null) {
EntityAccount account = db.account().getAccountByUUID(jidentity.getString("account_uuid"));
if (account != null) {
ridentity.account = account.id;
ridentity.id = db.identity().insertIdentity(ridentity);
}
} else {
ridentity.id = lidentity.id;
db.identity().updateIdentity(ridentity);
}
db.setTransactionSuccessful();
} finally {
db.endTransaction();
}
} }
} }
} }
prefs.edit().putLong("cloud_revision", lrevision).apply(); Log.i("Cloud set lrevision=" + rrevision);
prefs.edit().putLong("cloud_lrevision", rrevision).apply();
if (updates) if (updates)
ServiceSynchronize.reload(context, null, true, "sync"); ServiceSynchronize.reload(context, null, true, "sync");
} }
// Lower level // Lower level
public static JSONObject call(Context context, String user, String password, String command, JSONObject jrequest) public static JSONObject call(Context context, String user, String password, String command, JSONObject jrequest)

@ -562,6 +562,7 @@ public abstract class DB extends RoomDatabase {
" BEGIN" + " BEGIN" +
" UPDATE account SET last_modified = strftime('%s') * 1000" + " UPDATE account SET last_modified = strftime('%s') * 1000" +
" WHERE id = NEW.id" + " WHERE id = NEW.id" +
" AND OLD.last_modified = NEW.last_modified" +
" AND (NEW.auth_type = " + AUTH_TYPE_PASSWORD + " OR OLD.password = NEW.password)" + " AND (NEW.auth_type = " + AUTH_TYPE_PASSWORD + " OR OLD.password = NEW.password)" +
" AND OLD.keep_alive_ok IS NEW.keep_alive_ok" + " AND OLD.keep_alive_ok IS NEW.keep_alive_ok" +
" AND OLD.keep_alive_failed IS NEW.keep_alive_failed" + " AND OLD.keep_alive_failed IS NEW.keep_alive_failed" +
@ -580,6 +581,7 @@ public abstract class DB extends RoomDatabase {
" BEGIN" + " BEGIN" +
" UPDATE identity SET last_modified = strftime('%s') * 1000" + " UPDATE identity SET last_modified = strftime('%s') * 1000" +
" WHERE id = NEW.id" + " WHERE id = NEW.id" +
" AND OLD.last_modified = NEW.last_modified" +
" AND OLD.state IS NEW.state" + " AND OLD.state IS NEW.state" +
" AND OLD.error IS NEW.error" + " AND OLD.error IS NEW.error" +
" AND OLD.last_connected IS NEW.last_connected" + " AND OLD.last_connected IS NEW.last_connected" +

@ -288,6 +288,11 @@ public interface DaoAccount {
" AND (NOT (swipe_left IS :left) OR NOT (swipe_right IS :right))") " AND (NOT (swipe_left IS :left) OR NOT (swipe_right IS :right))")
int setAccountSwipes(long id, Long left, Long right); int setAccountSwipes(long id, Long left, Long right);
@Query("UPDATE account" +
" SET last_modified = :last_modified" +
" WHERE id = :id")
int setLastModified(long id, Long last_modified);
@Query("UPDATE account SET `primary` = 0 WHERE NOT (`primary` IS 0)") @Query("UPDATE account SET `primary` = 0 WHERE NOT (`primary` IS 0)")
void resetPrimary(); void resetPrimary();

@ -303,13 +303,25 @@
app:layout_constraintStart_toStartOf="parent" app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/tvRegister" /> app:layout_constraintTop_toBottomOf="@id/tvRegister" />
<TextView
android:id="@+id/tvAccountRemark"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginTop="12dp"
android:text="@string/title_advanced_cloud_account_remark"
android:textAppearance="@style/TextAppearance.AppCompat.Small"
android:textStyle="italic"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/tvLogin" />
<ImageButton <ImageButton
android:id="@+id/ibSync" android:id="@+id/ibSync"
android:layout_width="wrap_content" android:layout_width="wrap_content"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:layout_marginTop="12dp" android:layout_marginTop="12dp"
app:layout_constraintStart_toStartOf="parent" app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@id/tvLogin" app:layout_constraintTop_toBottomOf="@id/tvAccountRemark"
app:srcCompat="@drawable/twotone_compare_arrows_24" /> app:srcCompat="@drawable/twotone_compare_arrows_24" />
<TextView <TextView
@ -358,7 +370,7 @@
android:layout_width="0dp" android:layout_width="0dp"
android:layout_height="0dp" android:layout_height="0dp"
app:constraint_referenced_ids=" app:constraint_referenced_ids="
tvLogin, tvLogin,tvAccountRemark,
ibSync,tvLastSync, ibSync,tvLastSync,
btnLogout,cbDelete" /> btnLogout,cbDelete" />
</androidx.constraintlayout.widget.ConstraintLayout> </androidx.constraintlayout.widget.ConstraintLayout>

@ -983,6 +983,7 @@
<string name="title_advanced_cloud_login" translatable="false">Login</string> <string name="title_advanced_cloud_login" translatable="false">Login</string>
<string name="title_advanced_cloud_register" translatable="false">Logging in for the first time will automatically create an account</string> <string name="title_advanced_cloud_register" translatable="false">Logging in for the first time will automatically create an account</string>
<string name="title_advanced_cloud_invalid" translatable="false">Invalid username or password</string> <string name="title_advanced_cloud_invalid" translatable="false">Invalid username or password</string>
<string name="title_advanced_cloud_account_remark" translatable="false">Only the data of enabled accounts will be synced and existing accounts on the device will never be deleted.</string>
<string name="title_advanced_cloud_last_sync" translatable="false">Last sync: %1$s</string> <string name="title_advanced_cloud_last_sync" translatable="false">Last sync: %1$s</string>
<string name="title_advanced_cloud_logout" translatable="false">Logout</string> <string name="title_advanced_cloud_logout" translatable="false">Logout</string>
<string name="title_advanced_cloud_wipe" translatable="false">Wipe cloud data on logout</string> <string name="title_advanced_cloud_wipe" translatable="false">Wipe cloud data on logout</string>

Loading…
Cancel
Save