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

209 lines
7.8 KiB
Java
Raw Normal View History

2018-08-02 13:33:06 +00:00
package eu.faircode.email;
import android.content.Context;
2018-08-06 16:22:01 +00:00
import android.text.TextUtils;
2018-08-02 13:33:06 +00:00
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;
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-08-02 13:33:06 +00:00
/*
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 <http://www.gnu.org/licenses/>.
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,
2018-08-03 12:07:51 +00:00
EntityAttachment.class,
2018-08-02 13:33:06 +00:00
EntityOperation.class
},
2018-08-07 18:31:14 +00:00
version = 6,
2018-08-02 13:33:06 +00:00
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();
2018-08-03 12:07:51 +00:00
public abstract DaoAttachment attachment();
2018-08-02 13:33:06 +00:00
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;
}
2018-08-03 18:07:12 +00:00
public static DB getBlockingInstance(Context context) {
return migrate(Room.databaseBuilder(context.getApplicationContext(), DB.class, DB_NAME).allowMainThreadQueries());
}
2018-08-02 13:33:06 +00:00
private static DB migrate(RoomDatabase.Builder<DB> builder) {
return builder
2018-08-03 12:07:51 +00:00
.addMigrations(MIGRATION_1_2)
2018-08-04 10:32:34 +00:00
.addMigrations(MIGRATION_2_3)
2018-08-05 11:44:46 +00:00
.addMigrations(MIGRATION_3_4)
2018-08-07 08:00:17 +00:00
.addMigrations(MIGRATION_4_5)
2018-08-07 18:31:14 +00:00
.addMigrations(MIGRATION_5_6)
2018-08-02 13:33:06 +00:00
.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);
2018-08-03 12:07:51 +00:00
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`)");
2018-08-02 13:33:06 +00:00
}
};
2018-08-04 10:32:34 +00:00
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");
}
};
2018-08-05 11:44:46 +00:00
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);
2018-08-07 08:00:17 +00:00
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`)");
2018-08-05 11:44:46 +00:00
}
};
2018-08-07 18:31:14 +00:00
private static final Migration MIGRATION_5_6 = new Migration(5, 6) {
@Override
public void migrate(SupportSQLiteDatabase db) {
Log.i(Helper.TAG, "DB migration from version " + startVersion + " to " + endVersion);
db.execSQL("ALTER TABLE `account` ADD COLUMN `seen_until` INTEGER");
}
};
2018-08-02 13:33:06 +00:00
public static class Converters {
@TypeConverter
2018-08-06 16:22:01 +00:00
public static String[] fromStringArray(String value) {
return value.split(",");
2018-08-02 13:33:06 +00:00
}
@TypeConverter
2018-08-06 16:22:01 +00:00
public static String toStringArray(String[] value) {
2018-08-08 06:55:47 +00:00
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();
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)
2018-08-08 06:55:47 +00:00
jaddress.put("address", a);
if (p != null)
2018-08-08 06:55:47 +00:00
jaddress.put("personal", p);
jaddresses.put(jaddress);
} else {
JSONObject jaddress = new JSONObject();
2018-08-08 06:55:47 +00:00
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<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) {
Log.e(Helper.TAG, ex + "\n" + Log.getStackTraceString(ex));
}
return result.toArray(new Address[0]);
}
2018-08-02 13:33:06 +00:00
}
}