FairEmail/app/src/main/java/eu/faircode/email/DB.java

536 lines
26 KiB
Java
Raw Normal View History

2018-08-02 13:33:06 +00:00
package eu.faircode.email;
import android.content.Context;
import android.content.SharedPreferences;
2018-08-24 09:20:14 +00:00
import android.database.Cursor;
import android.preference.PreferenceManager;
2018-08-06 16:22:01 +00:00
import android.text.TextUtils;
2018-08-02 13:33:06 +00:00
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import javax.mail.Address;
import javax.mail.internet.InternetAddress;
2018-08-08 06:55:47 +00:00
import androidx.room.Database;
import androidx.room.Room;
import androidx.room.RoomDatabase;
import androidx.room.TypeConverter;
import androidx.room.TypeConverters;
import androidx.room.migration.Migration;
import androidx.sqlite.db.SupportSQLiteDatabase;
2018-11-09 07:38:07 +00:00
import io.requery.android.database.sqlite.RequerySQLiteOpenHelperFactory;
2018-08-08 06:55:47 +00:00
2018-08-02 13:33:06 +00:00
/*
2018-08-14 05:53:24 +00:00
This file is part of FairEmail.
2018-08-02 13:33:06 +00:00
2018-08-14 05:53:24 +00:00
FairEmail is free software: you can redistribute it and/or modify
2018-08-02 13:33:06 +00:00
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
2018-10-29 10:46:49 +00:00
FairEmail is distributed in the hope that it will be useful,
2018-08-02 13:33:06 +00:00
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
2018-10-29 10:46:49 +00:00
along with FairEmail. If not, see <http://www.gnu.org/licenses/>.
2018-08-02 13:33:06 +00:00
2018-12-31 08:04:33 +00:00
Copyright 2018-2019 by Marcel Bokhorst (M66B)
2018-08-02 13:33:06 +00:00
*/
// https://developer.android.com/topic/libraries/architecture/room.html
@Database(
2019-01-18 10:58:55 +00:00
version = 38,
2018-08-02 13:33:06 +00:00
entities = {
EntityIdentity.class,
EntityAccount.class,
EntityFolder.class,
EntityMessage.class,
2018-08-03 12:07:51 +00:00
EntityAttachment.class,
EntityOperation.class,
EntityAnswer.class,
2019-01-17 13:29:35 +00:00
EntityRule.class,
2018-09-03 19:11:16 +00:00
EntityLog.class
2018-08-08 10:22:12 +00:00
}
2018-08-02 13:33:06 +00:00
)
@TypeConverters({DB.Converters.class})
public abstract class DB extends RoomDatabase {
public abstract DaoIdentity identity();
public abstract DaoAccount account();
public abstract DaoFolder folder();
public abstract DaoMessage message();
2018-08-03 12:07:51 +00:00
public abstract DaoAttachment attachment();
2018-08-02 13:33:06 +00:00
public abstract DaoOperation operation();
public abstract DaoAnswer answer();
2019-01-17 13:29:35 +00:00
public abstract DaoRule rule();
2018-09-03 19:11:16 +00:00
public abstract DaoLog log();
2018-08-02 13:33:06 +00:00
private static DB sInstance;
2019-01-10 13:56:53 +00:00
private static ExecutorService executor = Executors.newSingleThreadExecutor(Helper.backgroundThreadFactory);
2018-08-02 13:33:06 +00:00
2018-11-13 09:05:33 +00:00
private static final String DB_NAME = "fairemail";
2018-08-02 13:33:06 +00:00
public static synchronized DB getInstance(Context context) {
if (sInstance == null) {
2019-01-06 16:31:53 +00:00
sInstance = migrate(context, getBuilder(context));
2018-08-24 10:15:54 +00:00
2018-12-24 12:27:45 +00:00
Log.i("sqlite version=" + exec(sInstance, "SELECT sqlite_version() AS sqlite_version"));
Log.i("sqlite sync=" + exec(sInstance, "PRAGMA synchronous"));
Log.i("sqlite journal=" + exec(sInstance, "PRAGMA journal_mode"));
}
2018-08-02 13:33:06 +00:00
return sInstance;
}
2019-01-06 16:31:53 +00:00
private static RoomDatabase.Builder getBuilder(Context context) {
return Room
.databaseBuilder(context.getApplicationContext(), DB.class, DB_NAME)
.openHelperFactory(new RequerySQLiteOpenHelperFactory())
.setQueryExecutor(executor)
.setJournalMode(JournalMode.WRITE_AHEAD_LOGGING);
}
2018-12-09 14:49:43 +00:00
private static String exec(DB db, String command) {
Cursor cursor = null;
try {
cursor = db.query(command, new Object[0]);
2018-12-24 10:51:32 +00:00
if (cursor != null && cursor.moveToNext())
return cursor.getString(0);
else
return null;
} finally {
if (cursor != null)
cursor.close();
}
2018-08-03 18:07:12 +00:00
}
private static DB migrate(final Context context, RoomDatabase.Builder<DB> builder) {
2018-11-13 09:05:33 +00:00
// https://www.sqlite.org/lang_altertable.html
2018-08-02 13:33:06 +00:00
return builder
2018-08-08 10:22:12 +00:00
.addCallback(new Callback() {
@Override
public void onOpen(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("Database version=" + db.getVersion());
2018-08-08 10:22:12 +00:00
super.onOpen(db);
}
})
2018-08-23 09:48:27 +00:00
.addMigrations(new Migration(1, 2) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-11-14 09:49:59 +00:00
db.execSQL("ALTER TABLE `folder` RENAME COLUMN `after` TO `sync_days`");
db.execSQL("ALTER TABLE `folder` ADD COLUMN `keep_days` INTEGER NOT NULL DEFAULT 30");
db.execSQL("UPDATE `folder` SET keep_days = sync_days");
2018-11-12 13:45:02 +00:00
}
})
2018-11-14 13:51:50 +00:00
.addMigrations(new Migration(2, 3) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-11-14 13:51:50 +00:00
db.execSQL("ALTER TABLE `identity` ADD COLUMN `signature` TEXT");
db.execSQL("UPDATE `identity` SET signature =" +
" (SELECT account.signature FROM account WHERE account.id = identity.account)");
}
})
.addMigrations(new Migration(3, 4) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `message` ADD COLUMN `forwarding` INTEGER" +
" REFERENCES `message`(`id`) ON UPDATE NO ACTION ON DELETE SET NULL");
2018-12-06 10:59:57 +00:00
db.execSQL("CREATE INDEX `index_message_forwarding` ON `message` (`forwarding`)");
}
})
2018-11-19 13:14:02 +00:00
.addMigrations(new Migration(4, 5) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-11-19 13:14:02 +00:00
db.execSQL("ALTER TABLE `account` ADD COLUMN `last_connected` INTEGER");
db.execSQL("ALTER TABLE `message` ADD COLUMN `last_attempt` INTEGER");
}
})
.addMigrations(new Migration(5, 6) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `account` ADD COLUMN `notify` INTEGER NOT NULL DEFAULT 0");
}
})
2018-11-24 18:14:28 +00:00
.addMigrations(new Migration(6, 7) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-11-24 18:14:28 +00:00
db.execSQL("ALTER TABLE `message` ADD COLUMN `answered` INTEGER NOT NULL DEFAULT 0");
db.execSQL("ALTER TABLE `message` ADD COLUMN `ui_answered` INTEGER NOT NULL DEFAULT 0");
}
})
2018-11-25 12:34:08 +00:00
.addMigrations(new Migration(7, 8) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-11-25 12:34:08 +00:00
db.execSQL("ALTER TABLE `message` ADD COLUMN `keywords` TEXT");
}
})
2018-11-26 08:12:44 +00:00
.addMigrations(new Migration(8, 9) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-11-26 08:12:44 +00:00
db.execSQL("ALTER TABLE `folder` ADD COLUMN `keywords` TEXT");
}
})
2018-11-26 15:57:00 +00:00
.addMigrations(new Migration(9, 10) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-11-26 15:57:00 +00:00
db.execSQL("ALTER TABLE `message` ADD COLUMN `ui_browsed` INTEGER NOT NULL DEFAULT 0");
2018-12-06 10:59:57 +00:00
db.execSQL("CREATE INDEX `index_message_ui_browsed` ON `message` (`ui_browsed`)");
2018-11-26 15:57:00 +00:00
}
})
2018-12-01 14:13:57 +00:00
.addMigrations(new Migration(10, 11) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-12-01 14:13:57 +00:00
db.execSQL("ALTER TABLE `operation` ADD COLUMN `error` TEXT");
}
})
2018-12-02 13:19:54 +00:00
.addMigrations(new Migration(11, 12) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-12-02 13:19:54 +00:00
db.execSQL("DROP INDEX `index_operation_folder`");
db.execSQL("DROP INDEX `index_operation_message`");
db.execSQL("DROP TABLE `operation`");
db.execSQL("CREATE TABLE `operation`" +
" (`id` INTEGER PRIMARY KEY AUTOINCREMENT" +
", `folder` INTEGER NOT NULL" +
", `message` INTEGER" +
", `name` TEXT NOT NULL" +
", `args` TEXT NOT NULL" +
", `created` INTEGER NOT NULL" +
", `error` TEXT" +
", FOREIGN KEY(`folder`) REFERENCES `folder`(`id`) ON UPDATE NO ACTION ON DELETE CASCADE" +
", FOREIGN KEY(`message`) REFERENCES `message`(`id`) ON UPDATE NO ACTION ON DELETE CASCADE)");
db.execSQL("CREATE INDEX `index_operation_folder` ON `operation` (`folder`)");
db.execSQL("CREATE INDEX `index_operation_message` ON `operation` (`message`)");
}
})
.addMigrations(new Migration(12, 13) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("CREATE INDEX `index_message_ui_flagged` ON `message` (`ui_flagged`)");
}
})
2018-12-03 09:39:44 +00:00
.addMigrations(new Migration(13, 14) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-12-03 09:39:44 +00:00
db.execSQL("ALTER TABLE `folder` ADD COLUMN `level` INTEGER NOT NULL DEFAULT 0");
}
})
2018-12-03 12:41:36 +00:00
.addMigrations(new Migration(14, 15) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-12-03 12:41:36 +00:00
db.execSQL("ALTER TABLE `folder` ADD COLUMN `sync_state` TEXT");
}
})
.addMigrations(new Migration(15, 16) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `folder` ADD COLUMN `poll` INTEGER NOT NULL DEFAULT 0");
}
})
2018-12-06 10:59:57 +00:00
.addMigrations(new Migration(16, 17) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-12-06 10:59:57 +00:00
db.execSQL("DELETE FROM `message` WHERE ui_found");
db.execSQL("DROP INDEX `index_message_folder_uid_ui_found`");
db.execSQL("DROP INDEX `index_message_msgid_folder_ui_found`");
db.execSQL("CREATE UNIQUE INDEX `index_message_folder_uid` ON `message` (`folder`, `uid`)");
db.execSQL("CREATE UNIQUE INDEX `index_message_msgid_folder` ON `message` (`msgid`, `folder`)");
}
})
.addMigrations(new Migration(17, 18) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `account` ADD COLUMN `tbd` INTEGER");
db.execSQL("ALTER TABLE `identity` ADD COLUMN `tbd` INTEGER");
db.execSQL("ALTER TABLE `folder` ADD COLUMN `tbd` INTEGER");
}
})
.addMigrations(new Migration(18, 19) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `identity` ADD COLUMN `delivery_receipt` INTEGER NOT NULL DEFAULT 0");
db.execSQL("ALTER TABLE `identity` ADD COLUMN `read_receipt` INTEGER NOT NULL DEFAULT 0");
}
})
.addMigrations(new Migration(19, 20) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `folder` ADD COLUMN `notify` INTEGER NOT NULL DEFAULT 0");
db.execSQL("UPDATE `folder` SET notify = unified");
}
})
.addMigrations(new Migration(20, 21) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `identity` ADD COLUMN `display` TEXT");
db.execSQL("ALTER TABLE `identity` ADD COLUMN `bcc` TEXT");
}
})
.addMigrations(new Migration(21, 22) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `folder` ADD COLUMN `initialize` INTEGER NOT NULL DEFAULT 1");
db.execSQL("UPDATE `folder` SET sync_days = 1");
}
})
.addMigrations(new Migration(22, 23) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `folder` ADD COLUMN `download` INTEGER NOT NULL DEFAULT 1");
}
})
2018-12-22 11:40:47 +00:00
.addMigrations(new Migration(23, 24) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-12-22 11:40:47 +00:00
db.execSQL("ALTER TABLE `folder` ADD COLUMN `tbc` INTEGER");
}
})
2018-12-22 17:39:16 +00:00
.addMigrations(new Migration(24, 25) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-12-22 17:39:16 +00:00
db.execSQL("ALTER TABLE `account` ADD COLUMN `prefix` TEXT");
}
})
.addMigrations(new Migration(25, 26) {
@Override
public void migrate(SupportSQLiteDatabase db) {
2018-12-24 12:27:45 +00:00
Log.i("DB migration from version " + startVersion + " to " + endVersion);
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
int browse = (prefs.getBoolean("browse", true) ? 1 : 0);
db.execSQL("ALTER TABLE `account` ADD COLUMN `browse` INTEGER NOT NULL DEFAULT " + browse);
}
})
2018-12-27 11:32:20 +00:00
.addMigrations(new Migration(26, 27) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `message` ADD COLUMN `sender` TEXT");
2018-12-28 08:04:56 +00:00
db.execSQL("CREATE INDEX `index_message_sender` ON `message` (`sender`)");
}
})
.addMigrations(new Migration(27, 28) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
2018-12-27 11:32:20 +00:00
Cursor cursor = null;
try {
cursor = db.query("SELECT `id`, `from` FROM message");
2018-12-28 08:04:56 +00:00
while (cursor.moveToNext())
try {
long id = cursor.getLong(0);
String json = cursor.getString(1);
Address[] from = Converters.decodeAddresses(json);
String sender = MessageHelper.getSortKey(from);
2018-12-27 11:32:20 +00:00
db.execSQL(
"UPDATE message SET sender = ? WHERE id = ?",
new Object[]{sender, id});
2018-12-28 08:04:56 +00:00
} catch (Throwable ex) {
Log.e(ex);
}
2018-12-27 11:32:20 +00:00
} finally {
if (cursor != null)
cursor.close();
}
}
})
2019-01-01 18:49:21 +00:00
.addMigrations(new Migration(28, 29) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `folder` ADD COLUMN `last_sync` INTEGER");
}
})
2019-01-05 14:09:47 +00:00
.addMigrations(new Migration(29, 30) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `attachment` ADD COLUMN `encryption` INTEGER");
db.execSQL("UPDATE attachment SET encryption = " + EntityAttachment.PGP_MESSAGE + " where name = 'encrypted.asc'");
}
})
.addMigrations(new Migration(30, 31) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `attachment` ADD COLUMN `disposition` TEXT");
}
})
2019-01-07 15:05:24 +00:00
.addMigrations(new Migration(31, 32) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `message` ADD COLUMN `ui_snoozed` INTEGER");
db.execSQL("CREATE INDEX `index_message_ui_snoozed` ON `message` (`ui_snoozed`)");
}
})
2019-01-10 18:24:20 +00:00
.addMigrations(new Migration(32, 33) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `account` ADD COLUMN `realm` TEXT");
db.execSQL("ALTER TABLE `identity` ADD COLUMN `realm` TEXT");
}
})
.addMigrations(new Migration(33, 34) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `message` ADD COLUMN `raw` INTEGER");
}
})
2019-01-16 18:27:03 +00:00
.addMigrations(new Migration(34, 35) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `attachment` ADD COLUMN `error` TEXT");
}
})
2019-01-17 10:49:18 +00:00
.addMigrations(new Migration(35, 36) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `message` ADD COLUMN `warning` TEXT");
}
})
2019-01-17 13:29:35 +00:00
.addMigrations(new Migration(36, 37) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("CREATE TABLE `rule`" +
" (`id` INTEGER PRIMARY KEY AUTOINCREMENT," +
" `folder` INTEGER NOT NULL," +
" `name` TEXT NOT NULL," +
" `order` INTEGER NOT NULL," +
2019-01-17 21:25:22 +00:00
" `enabled` INTEGER NOT NULL," +
2019-01-17 13:29:35 +00:00
" `condition` TEXT NOT NULL," +
" `action` TEXT NOT NULL," +
" FOREIGN KEY(`folder`) REFERENCES `folder`(`id`) ON UPDATE NO ACTION ON DELETE CASCADE)");
db.execSQL("CREATE INDEX `index_rule_folder` ON `rule` (`folder`)");
db.execSQL("CREATE INDEX `index_rule_order` ON `rule` (`order`)");
}
})
2019-01-18 10:58:55 +00:00
.addMigrations(new Migration(37, 38) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i("DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `rule` ADD COLUMN `stop` INTEGER NOT NULL DEFAULT 0");
}
})
2018-08-02 13:33:06 +00:00
.build();
}
public static class Converters {
@TypeConverter
2018-11-25 12:34:08 +00:00
public static String[] toStringArray(String value) {
if (value == null)
return new String[0];
else
2018-11-25 17:29:11 +00:00
return TextUtils.split(value, " ");
2018-08-02 13:33:06 +00:00
}
@TypeConverter
2018-11-25 12:34:08 +00:00
public static String fromStringArray(String[] value) {
if (value == null || value.length == 0)
return null;
else
return TextUtils.join(" ", value);
2018-08-02 13:33:06 +00:00
}
@TypeConverter
public static String encodeAddresses(Address[] addresses) {
if (addresses == null)
return null;
JSONArray jaddresses = new JSONArray();
2018-12-09 14:49:43 +00:00
for (Address address : addresses)
try {
if (address instanceof InternetAddress) {
String a = ((InternetAddress) address).getAddress();
String p = ((InternetAddress) address).getPersonal();
JSONObject jaddress = new JSONObject();
if (a != null)
jaddress.put("address", a);
if (p != null)
jaddress.put("personal", p);
jaddresses.put(jaddress);
} else {
JSONObject jaddress = new JSONObject();
jaddress.put("address", address.toString());
jaddresses.put(jaddress);
}
2018-12-09 14:49:43 +00:00
} catch (JSONException ex) {
2018-12-24 12:27:45 +00:00
Log.e(ex);
2018-12-09 14:49:43 +00:00
}
return jaddresses.toString();
}
@TypeConverter
public static Address[] decodeAddresses(String json) {
if (json == null)
return null;
List<Address> result = new ArrayList<>();
try {
JSONArray jaddresses = new JSONArray(json);
for (int i = 0; i < jaddresses.length(); i++) {
JSONObject jaddress = (JSONObject) jaddresses.get(i);
if (jaddress.has("personal"))
result.add(new InternetAddress(
jaddress.getString("address"),
jaddress.getString("personal")));
else
result.add(new InternetAddress(
jaddress.getString("address")));
}
} catch (Throwable ex) {
2018-08-29 05:31:00 +00:00
// Compose can store invalid addresses
2018-12-24 12:27:45 +00:00
Log.w(ex);
}
return result.toArray(new Address[0]);
}
2018-08-02 13:33:06 +00:00
}
}