package eu.faircode.email; import android.arch.persistence.db.SupportSQLiteDatabase; import android.arch.persistence.room.Database; import android.arch.persistence.room.Room; import android.arch.persistence.room.RoomDatabase; import android.arch.persistence.room.TypeConverter; import android.arch.persistence.room.TypeConverters; import android.arch.persistence.room.migration.Migration; import android.content.Context; import android.text.TextUtils; import android.util.Log; import org.json.JSONArray; import org.json.JSONException; import org.json.JSONObject; import java.util.ArrayList; import java.util.List; import javax.mail.Address; import javax.mail.internet.InternetAddress; /* This file is part of Safe email. Safe email is free software: you can redistribute it and/or modify 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. NetGuard is distributed in the hope that it will be useful, 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 along with NetGuard. If not, see . Copyright 2018 by Marcel Bokhorst (M66B) */ // https://developer.android.com/topic/libraries/architecture/room.html @Database( entities = { EntityIdentity.class, EntityAccount.class, EntityFolder.class, EntityMessage.class, EntityAttachment.class, EntityOperation.class }, version = 5, exportSchema = true ) @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(); public abstract DaoAttachment attachment(); public abstract DaoOperation operation(); private static DB sInstance; private static final String DB_NAME = "email.db"; public static synchronized DB getInstance(Context context) { if (sInstance == null) sInstance = migrate(Room.databaseBuilder(context.getApplicationContext(), DB.class, DB_NAME)); return sInstance; } public static DB getBlockingInstance(Context context) { return migrate(Room.databaseBuilder(context.getApplicationContext(), DB.class, DB_NAME).allowMainThreadQueries()); } private static DB migrate(RoomDatabase.Builder builder) { return builder .addMigrations(MIGRATION_1_2) .addMigrations(MIGRATION_2_3) .addMigrations(MIGRATION_3_4) .addMigrations(MIGRATION_4_5) .build(); } private static final Migration MIGRATION_1_2 = new Migration(1, 2) { @Override public void migrate(SupportSQLiteDatabase db) { Log.i(Helper.TAG, "DB migration from version " + startVersion + " to " + endVersion); db.execSQL("CREATE TABLE IF NOT EXISTS `attachment`" + " (`id` INTEGER PRIMARY KEY AUTOINCREMENT" + ", `message` INTEGER NOT NULL" + ", `sequence` INTEGER NOT NULL" + ", `type` TEXT NOT NULL, `name` TEXT" + ", `content` BLOB, FOREIGN KEY(`message`) REFERENCES `message`(`id`) ON UPDATE NO ACTION ON DELETE CASCADE )"); db.execSQL("CREATE INDEX `index_attachment_message` ON `attachment` (`message`)"); db.execSQL("CREATE UNIQUE INDEX `index_attachment_message_sequence` ON `attachment` (`message`, `sequence`)"); } }; private static final Migration MIGRATION_2_3 = new Migration(2, 3) { @Override public void migrate(SupportSQLiteDatabase db) { Log.i(Helper.TAG, "DB migration from version " + startVersion + " to " + endVersion); db.execSQL("ALTER TABLE `attachment` ADD COLUMN `size` INTEGER"); db.execSQL("ALTER TABLE `attachment` ADD COLUMN `progress` INTEGER"); } }; private static final Migration MIGRATION_3_4 = new Migration(3, 4) { @Override public void migrate(SupportSQLiteDatabase db) { Log.i(Helper.TAG, "DB migration from version " + startVersion + " to " + endVersion); db.execSQL("CREATE INDEX `index_message_ui_seen` ON `message` (`ui_seen`)"); } }; private static final Migration MIGRATION_4_5 = new Migration(4, 5) { @Override public void migrate(SupportSQLiteDatabase db) { Log.i(Helper.TAG, "DB migration from version " + startVersion + " to " + endVersion); db.execSQL("CREATE INDEX `index_message_ui_hide` ON `message` (`ui_hide`)"); } }; public static class Converters { @TypeConverter public static String[] fromStringArray(String value) { return value.split(","); } @TypeConverter public static String toStringArray(String[] value) { return TextUtils.join("," , value); } @TypeConverter public static String encodeAddresses(Address[] addresses) { if (addresses == null) return null; JSONArray jaddresses = new JSONArray(); if (addresses != null) 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); } } catch (JSONException ex) { Log.e(Helper.TAG, ex + "\n" + Log.getStackTraceString(ex)); } return jaddresses.toString(); } @TypeConverter public static Address[] decodeAddresses(String json) { if (json == null) return null; List
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) { Log.e(Helper.TAG, ex + "\n" + Log.getStackTraceString(ex)); } return result.toArray(new Address[0]); } } }