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

1063 lines
46 KiB
Java
Raw Normal View History

2018-08-02 13:33:06 +00:00
package eu.faircode.email;
/*
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
*/
import android.app.Notification;
import android.app.NotificationChannel;
import android.app.NotificationChannelGroup;
import android.app.NotificationManager;
2018-08-04 15:46:22 +00:00
import android.content.BroadcastReceiver;
2019-01-14 10:29:47 +00:00
import android.content.ContentResolver;
2018-08-04 15:46:22 +00:00
import android.content.Context;
2019-01-14 10:29:47 +00:00
import android.content.DialogInterface;
2018-08-04 15:46:22 +00:00
import android.content.Intent;
import android.content.IntentFilter;
2019-01-14 10:29:47 +00:00
import android.content.SharedPreferences;
import android.content.pm.PackageManager;
import android.content.res.AssetFileDescriptor;
import android.content.res.Configuration;
2019-04-29 09:39:24 +00:00
import android.graphics.Rect;
import android.graphics.drawable.Drawable;
import android.media.Ringtone;
import android.media.RingtoneManager;
2019-01-14 10:29:47 +00:00
import android.net.Uri;
2019-03-17 17:18:53 +00:00
import android.os.Build;
2018-08-02 13:33:06 +00:00
import android.os.Bundle;
2019-01-14 10:29:47 +00:00
import android.text.TextUtils;
import android.view.LayoutInflater;
2018-08-04 16:54:57 +00:00
import android.view.MenuItem;
2019-01-14 10:29:47 +00:00
import android.view.View;
2019-01-27 09:32:35 +00:00
import android.widget.TextView;
2018-08-02 13:33:06 +00:00
import androidx.annotation.RequiresApi;
import androidx.appcompat.app.ActionBarDrawerToggle;
2019-04-28 18:48:45 +00:00
import androidx.constraintlayout.widget.ConstraintLayout;
import androidx.documentfile.provider.DocumentFile;
import androidx.drawerlayout.widget.DrawerLayout;
import androidx.fragment.app.FragmentManager;
import androidx.fragment.app.FragmentTransaction;
import androidx.lifecycle.Lifecycle;
import androidx.lifecycle.Observer;
import androidx.localbroadcastmanager.content.LocalBroadcastManager;
import androidx.preference.PreferenceManager;
2019-04-29 09:39:24 +00:00
import androidx.recyclerview.widget.DividerItemDecoration;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
2019-01-14 10:29:47 +00:00
import com.google.android.material.snackbar.Snackbar;
import com.google.android.material.textfield.TextInputLayout;
import org.json.JSONArray;
2019-06-07 15:42:55 +00:00
import org.json.JSONException;
2019-01-14 10:29:47 +00:00
import org.json.JSONObject;
2019-01-27 09:32:35 +00:00
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
2019-01-14 10:29:47 +00:00
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.security.SecureRandom;
import java.security.spec.KeySpec;
import java.text.SimpleDateFormat;
2019-03-01 09:07:15 +00:00
import java.util.ArrayList;
2019-01-14 10:29:47 +00:00
import java.util.Date;
2019-06-07 15:42:55 +00:00
import java.util.HashMap;
2018-08-14 08:31:43 +00:00
import java.util.List;
2019-06-07 15:42:55 +00:00
import java.util.Map;
2019-02-26 10:05:21 +00:00
import java.util.Objects;
2018-08-14 08:31:43 +00:00
2019-01-14 10:29:47 +00:00
import javax.crypto.BadPaddingException;
import javax.crypto.Cipher;
import javax.crypto.CipherInputStream;
import javax.crypto.CipherOutputStream;
import javax.crypto.SecretKey;
import javax.crypto.SecretKeyFactory;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.PBEKeySpec;
2018-09-07 08:44:48 +00:00
public class ActivitySetup extends ActivityBilling implements FragmentManager.OnBackStackChangedListener {
2019-01-14 10:29:47 +00:00
private View view;
private DrawerLayout drawerLayout;
private ActionBarDrawerToggle drawerToggle;
2019-04-28 18:48:45 +00:00
private ConstraintLayout drawerContainer;
private RecyclerView rvMenu;
2019-01-14 10:29:47 +00:00
private boolean hasAccount;
2019-01-27 09:32:35 +00:00
private String password;
2018-08-14 08:31:43 +00:00
2019-01-14 10:29:47 +00:00
private static final int KEY_ITERATIONS = 65536;
private static final int KEY_LENGTH = 256;
2018-08-27 14:31:45 +00:00
static final int REQUEST_PERMISSION = 1;
static final int REQUEST_CHOOSE_ACCOUNT = 2;
static final int REQUEST_SOUND = 3;
2018-09-14 11:33:22 +00:00
static final int REQUEST_EXPORT = 4;
static final int REQUEST_IMPORT = 5;
2019-06-12 14:15:46 +00:00
static final String ACTION_QUICK_SETUP = BuildConfig.APPLICATION_ID + ".ACTION_QUICK_SETUP";
static final String ACTION_VIEW_ACCOUNTS = BuildConfig.APPLICATION_ID + ".ACTION_VIEW_ACCOUNTS";
static final String ACTION_VIEW_IDENTITIES = BuildConfig.APPLICATION_ID + ".ACTION_VIEW_IDENTITIES";
2018-08-04 15:46:22 +00:00
static final String ACTION_EDIT_ACCOUNT = BuildConfig.APPLICATION_ID + ".EDIT_ACCOUNT";
static final String ACTION_EDIT_IDENTITY = BuildConfig.APPLICATION_ID + ".EDIT_IDENTITY";
2019-05-06 08:48:14 +00:00
static final String ACTION_SHOW_PRO = BuildConfig.APPLICATION_ID + ".SHOW_PRO";
2018-08-04 15:46:22 +00:00
2018-08-02 13:33:06 +00:00
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
2019-01-14 10:29:47 +00:00
view = LayoutInflater.from(this).inflate(R.layout.activity_setup, null);
setContentView(view);
2018-08-02 13:33:06 +00:00
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
2018-08-04 16:54:57 +00:00
2019-01-14 10:29:47 +00:00
drawerLayout = findViewById(R.id.drawer_layout);
drawerLayout.setScrimColor(Helper.resolveColor(this, R.attr.colorDrawerScrim));
drawerToggle = new ActionBarDrawerToggle(this, drawerLayout, R.string.app_name, R.string.app_name) {
public void onDrawerClosed(View view) {
super.onDrawerClosed(view);
getSupportActionBar().setTitle(getString(R.string.app_name));
}
public void onDrawerOpened(View drawerView) {
super.onDrawerOpened(drawerView);
getSupportActionBar().setTitle(getString(R.string.app_name));
}
};
drawerLayout.addDrawerListener(drawerToggle);
2019-04-28 18:48:45 +00:00
drawerContainer = findViewById(R.id.drawer_container);
rvMenu = drawerContainer.findViewById(R.id.rvMenu);
2019-04-29 09:39:24 +00:00
LinearLayoutManager llm = new LinearLayoutManager(this);
rvMenu.setLayoutManager(llm);
final AdapterNavMenu adapter = new AdapterNavMenu(this, this);
rvMenu.setAdapter(adapter);
2019-02-26 17:44:15 +00:00
2019-04-29 09:39:24 +00:00
final Drawable d = getDrawable(R.drawable.divider);
DividerItemDecoration itemDecorator = new DividerItemDecoration(this, llm.getOrientation()) {
@Override
public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
int pos = parent.getChildAdapterPosition(view);
NavMenuItem menu = adapter.get(pos);
2019-04-29 10:04:45 +00:00
outRect.set(0, 0, 0, menu != null && menu.isSeparated() ? d.getIntrinsicHeight() : 0);
2019-04-29 09:39:24 +00:00
}
};
itemDecorator.setDrawable(d);
rvMenu.addItemDecoration(itemDecorator);
PackageManager pm = getPackageManager();
final List<NavMenuItem> menus = new ArrayList<>();
2019-02-26 17:44:15 +00:00
menus.add(new NavMenuItem(R.drawable.baseline_archive_24, R.string.title_setup_export, new Runnable() {
2019-01-14 10:29:47 +00:00
@Override
public void run() {
2019-04-28 18:48:45 +00:00
drawerLayout.closeDrawer(drawerContainer);
onMenuExport();
2019-01-14 10:29:47 +00:00
}
}));
2019-03-01 09:07:15 +00:00
menus.add(new NavMenuItem(R.drawable.baseline_unarchive_24, R.string.title_setup_import, new Runnable() {
@Override
public void run() {
drawerLayout.closeDrawer(drawerContainer);
onMenuImport();
}
2019-04-29 09:39:24 +00:00
}).setSeparated());
2019-01-14 10:29:47 +00:00
2019-05-01 05:58:45 +00:00
menus.add(new NavMenuItem(R.drawable.baseline_reorder_24, R.string.title_setup_reorder_accounts, new Runnable() {
@Override
public void run() {
drawerLayout.closeDrawer(drawerContainer);
2019-05-01 05:58:45 +00:00
onMenuOrder(R.string.title_setup_reorder_accounts, EntityAccount.class);
}
}));
menus.add(new NavMenuItem(R.drawable.baseline_reorder_24, R.string.title_setup_reorder_folders, new Runnable() {
@Override
public void run() {
drawerLayout.closeDrawer(drawerContainer);
onMenuOrder(R.string.title_setup_reorder_folders, TupleFolderSort.class);
}
}));
menus.add(new NavMenuItem(R.drawable.baseline_person_24, R.string.menu_contacts, new Runnable() {
@Override
public void run() {
drawerLayout.closeDrawer(drawerContainer);
onMenuContacts();
}
2019-04-29 09:39:24 +00:00
}).setSeparated());
2019-01-14 10:29:47 +00:00
menus.add(new NavMenuItem(R.drawable.baseline_help_24, R.string.menu_legend, new Runnable() {
@Override
public void run() {
drawerLayout.closeDrawer(drawerContainer);
onMenuLegend();
}
}));
if (Helper.getIntentFAQ().resolveActivity(pm) != null)
menus.add(new NavMenuItem(R.drawable.baseline_question_answer_24, R.string.menu_faq, new Runnable() {
@Override
public void run() {
drawerLayout.closeDrawer(drawerContainer);
onMenuFAQ();
}
}));
if (Helper.getIntentPrivacy().resolveActivity(pm) != null)
menus.add(new NavMenuItem(R.drawable.baseline_account_box_24, R.string.menu_privacy, new Runnable() {
@Override
public void run() {
drawerLayout.closeDrawer(drawerContainer);
onMenuPrivacy();
}
2019-05-01 20:01:58 +00:00
}, new Runnable() {
@Override
public void run() {
drawerLayout.closeDrawer(drawerContainer);
onCleanup();
}
}));
menus.add(new NavMenuItem(R.drawable.baseline_info_24, R.string.menu_about, new Runnable() {
@Override
public void run() {
drawerLayout.closeDrawer(drawerContainer);
onMenuAbout();
}
}));
2019-03-01 09:07:15 +00:00
adapter.set(menus);
2019-01-14 10:29:47 +00:00
2018-08-02 13:33:06 +00:00
getSupportFragmentManager().addOnBackStackChangedListener(this);
if (getSupportFragmentManager().getFragments().size() == 0) {
2019-05-30 07:19:54 +00:00
Intent intent = getIntent();
String target = intent.getStringExtra("target");
2018-08-02 13:33:06 +00:00
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
2019-05-30 07:19:54 +00:00
if ("accounts".equals(target))
fragmentTransaction.replace(R.id.content_frame, new FragmentAccounts()).addToBackStack("accounts");
else
2019-06-12 14:15:46 +00:00
fragmentTransaction.replace(R.id.content_frame, new FragmentOptions()).addToBackStack("options");
2018-08-02 13:33:06 +00:00
fragmentTransaction.commit();
2019-05-30 07:19:54 +00:00
if (intent.hasExtra("target")) {
intent.removeExtra("target");
setIntent(intent);
}
2018-08-02 13:33:06 +00:00
}
2018-08-14 08:31:43 +00:00
2019-01-14 10:29:47 +00:00
if (savedInstanceState != null)
2019-03-02 07:35:12 +00:00
drawerToggle.setDrawerIndicatorEnabled(savedInstanceState.getBoolean("fair:toggle"));
2019-01-14 10:29:47 +00:00
2019-02-17 18:29:00 +00:00
DB.getInstance(this).account().liveSynchronizingAccounts().observe(this, new Observer<List<EntityAccount>>() {
2018-08-14 08:31:43 +00:00
@Override
public void onChanged(List<EntityAccount> accounts) {
2018-08-14 08:31:43 +00:00
hasAccount = (accounts != null && accounts.size() > 0);
}
});
2018-08-02 13:33:06 +00:00
}
2019-01-14 10:29:47 +00:00
@Override
protected void onSaveInstanceState(Bundle outState) {
2019-03-02 07:35:12 +00:00
outState.putBoolean("fair:toggle", drawerToggle.isDrawerIndicatorEnabled());
2019-04-11 17:30:39 +00:00
super.onSaveInstanceState(outState);
2019-01-14 10:29:47 +00:00
}
@Override
protected void onPostCreate(Bundle savedInstanceState) {
super.onPostCreate(savedInstanceState);
drawerToggle.syncState();
}
2018-08-04 15:46:22 +00:00
@Override
protected void onResume() {
super.onResume();
LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(this);
IntentFilter iff = new IntentFilter();
2019-06-12 14:15:46 +00:00
iff.addAction(ACTION_QUICK_SETUP);
iff.addAction(ACTION_VIEW_ACCOUNTS);
iff.addAction(ACTION_VIEW_IDENTITIES);
2018-08-04 15:46:22 +00:00
iff.addAction(ACTION_EDIT_ACCOUNT);
iff.addAction(ACTION_EDIT_IDENTITY);
2019-05-06 08:48:14 +00:00
iff.addAction(ACTION_SHOW_PRO);
2018-08-04 15:46:22 +00:00
lbm.registerReceiver(receiver, iff);
}
@Override
protected void onPause() {
super.onPause();
LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(this);
lbm.unregisterReceiver(receiver);
}
2019-01-14 10:29:47 +00:00
@Override
public void onConfigurationChanged(Configuration newConfig) {
super.onConfigurationChanged(newConfig);
drawerToggle.onConfigurationChanged(newConfig);
}
@Override
public void onBackPressed() {
2019-04-28 18:48:45 +00:00
if (drawerLayout.isDrawerOpen(drawerContainer))
drawerLayout.closeDrawer(drawerContainer);
2019-01-14 10:29:47 +00:00
else
super.onBackPressed();
}
@Override
public void onBackStackChanged() {
int count = getSupportFragmentManager().getBackStackEntryCount();
if (count == 0) {
if (hasAccount)
startActivity(new Intent(this, ActivityView.class));
finish();
} else {
2019-04-28 18:48:45 +00:00
if (drawerLayout.isDrawerOpen(drawerContainer))
drawerLayout.closeDrawer(drawerContainer);
2019-01-14 10:29:47 +00:00
drawerToggle.setDrawerIndicatorEnabled(count == 1);
}
}
2018-08-04 16:54:57 +00:00
@Override
public boolean onOptionsItemSelected(MenuItem item) {
2019-01-14 10:29:47 +00:00
if (drawerToggle.onOptionsItemSelected(item))
return true;
2018-08-04 16:54:57 +00:00
switch (item.getItemId()) {
case android.R.id.home:
2018-08-17 05:34:46 +00:00
if (getLifecycle().getCurrentState().isAtLeast(Lifecycle.State.RESUMED))
getSupportFragmentManager().popBackStack();
2018-08-04 16:54:57 +00:00
return true;
}
return super.onOptionsItemSelected(item);
}
2018-08-02 13:33:06 +00:00
@Override
2019-01-14 11:11:03 +00:00
public void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
2019-01-27 09:32:35 +00:00
if (resultCode == RESULT_OK && data != null)
if (requestCode == REQUEST_EXPORT)
handleExport(data, this.password);
else if (requestCode == REQUEST_IMPORT)
handleImport(data, this.password);
2019-01-14 10:29:47 +00:00
}
private void onMenuExport() {
2019-05-27 10:15:25 +00:00
if (!Helper.isPro(this)) {
onShowPro(null);
return;
}
2019-05-14 19:47:07 +00:00
2019-05-27 10:15:25 +00:00
try {
askPassword(true);
} catch (Throwable ex) {
Helper.unexpectedError(this, this, ex);
2019-01-14 10:29:47 +00:00
}
}
private void onMenuImport() {
try {
2019-01-27 09:32:35 +00:00
askPassword(false);
2019-01-14 10:29:47 +00:00
} catch (Throwable ex) {
Helper.unexpectedError(this, this, ex);
}
}
2019-01-27 09:32:35 +00:00
private void askPassword(final boolean export) {
View dview = LayoutInflater.from(this).inflate(R.layout.dialog_password, null);
final TextInputLayout etPassword1 = dview.findViewById(R.id.tilPassword1);
final TextInputLayout etPassword2 = dview.findViewById(R.id.tilPassword2);
TextView tvImportHint = dview.findViewById(R.id.tvImporthint);
etPassword2.setVisibility(export ? View.VISIBLE : View.GONE);
tvImportHint.setVisibility(export ? View.GONE : View.VISIBLE);
new DialogBuilderLifecycle(this, this)
.setView(dview)
.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
String password1 = etPassword1.getEditText().getText().toString();
String password2 = etPassword2.getEditText().getText().toString();
if (!BuildConfig.DEBUG && TextUtils.isEmpty(password1))
Snackbar.make(view, R.string.title_setup_password_missing, Snackbar.LENGTH_LONG).show();
else {
if (!export || password1.equals(password2)) {
ActivitySetup.this.password = password1;
startActivityForResult(
Helper.getChooser(
ActivitySetup.this,
export ? getIntentExport() : getIntentImport()),
export ? REQUEST_EXPORT : REQUEST_IMPORT);
} else
Snackbar.make(view, R.string.title_setup_password_different, Snackbar.LENGTH_LONG).show();
}
}
})
.show();
}
2019-05-01 05:58:45 +00:00
private void onMenuOrder(int title, Class clazz) {
if (getLifecycle().getCurrentState().isAtLeast(Lifecycle.State.RESUMED))
getSupportFragmentManager().popBackStack("order", FragmentManager.POP_BACK_STACK_INCLUSIVE);
Bundle args = new Bundle();
args.putInt("title", title);
args.putString("class", clazz.getName());
FragmentOrder fragment = new FragmentOrder();
fragment.setArguments(args);
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.replace(R.id.content_frame, fragment).addToBackStack("order");
fragmentTransaction.commit();
}
private void onMenuContacts() {
2019-03-15 07:34:31 +00:00
if (getLifecycle().getCurrentState().isAtLeast(Lifecycle.State.RESUMED))
2019-05-01 05:58:45 +00:00
getSupportFragmentManager().popBackStack("contacts", FragmentManager.POP_BACK_STACK_INCLUSIVE);
2019-03-15 07:34:31 +00:00
2019-01-14 12:01:33 +00:00
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
2019-05-01 05:58:45 +00:00
fragmentTransaction.replace(R.id.content_frame, new FragmentContacts()).addToBackStack("contacts");
2019-01-14 12:01:33 +00:00
fragmentTransaction.commit();
}
2019-01-14 10:29:47 +00:00
private void onMenuLegend() {
2019-03-15 07:34:31 +00:00
if (getLifecycle().getCurrentState().isAtLeast(Lifecycle.State.RESUMED))
getSupportFragmentManager().popBackStack("legend", FragmentManager.POP_BACK_STACK_INCLUSIVE);
2019-01-14 10:29:47 +00:00
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.replace(R.id.content_frame, new FragmentLegend()).addToBackStack("legend");
fragmentTransaction.commit();
}
private void onMenuFAQ() {
Helper.view(this, this, Helper.getIntentFAQ());
}
private void onMenuPrivacy() {
Helper.view(this, this, Helper.getIntentPrivacy());
}
2019-05-01 20:01:58 +00:00
private void onCleanup() {
new SimpleTask<Void>() {
@Override
protected Void onExecute(Context context, Bundle args) {
WorkerCleanup.cleanup(context, true);
return null;
}
@Override
protected void onException(Bundle args, Throwable ex) {
Helper.unexpectedError(ActivitySetup.this, ActivitySetup.this, ex);
}
}.execute(this, new Bundle(), "cleanup:run");
}
2019-01-14 10:29:47 +00:00
private void onMenuAbout() {
2019-03-15 07:34:31 +00:00
if (getLifecycle().getCurrentState().isAtLeast(Lifecycle.State.RESUMED))
getSupportFragmentManager().popBackStack("about", FragmentManager.POP_BACK_STACK_INCLUSIVE);
2019-01-14 10:29:47 +00:00
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.replace(R.id.content_frame, new FragmentAbout()).addToBackStack("about");
fragmentTransaction.commit();
}
private static Intent getIntentExport() {
Intent intent = new Intent(Intent.ACTION_CREATE_DOCUMENT);
intent.addCategory(Intent.CATEGORY_OPENABLE);
intent.setType("*/*");
2019-01-27 09:32:35 +00:00
intent.putExtra(Intent.EXTRA_TITLE, "fairemail_" +
new SimpleDateFormat("yyyyMMdd").format(new Date().getTime()) + ".backup");
2019-01-14 10:29:47 +00:00
return intent;
}
private static Intent getIntentImport() {
Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT);
intent.addCategory(Intent.CATEGORY_OPENABLE);
intent.setType("*/*");
return intent;
}
private void handleExport(Intent data, String password) {
Bundle args = new Bundle();
args.putParcelable("uri", data.getData());
args.putString("password", password);
new SimpleTask<Void>() {
@Override
protected Void onExecute(Context context, Bundle args) throws Throwable {
Uri uri = args.getParcelable("uri");
String password = args.getString("password");
if ("file".equals(uri.getScheme())) {
Log.w("Export uri=" + uri);
throw new IllegalArgumentException(context.getString(R.string.title_no_stream));
}
2019-01-27 09:32:35 +00:00
Log.i("Collecting data");
DB db = DB.getInstance(context);
NotificationManager nm = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
2019-01-27 09:32:35 +00:00
// Accounts
JSONArray jaccounts = new JSONArray();
for (EntityAccount account : db.account().getAccounts()) {
// Account
JSONObject jaccount = account.toJSON();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
if (account.notify) {
NotificationChannel channel = nm.getNotificationChannel(
EntityAccount.getNotificationChannelId(account.id));
if (channel != null && channel.getImportance() != NotificationManager.IMPORTANCE_NONE) {
JSONObject jchannel = channelToJSON(channel);
jaccount.put("channel", jchannel);
Log.i("Exported account channel=" + jchannel);
}
}
}
2019-01-27 09:32:35 +00:00
// Identities
JSONArray jidentities = new JSONArray();
for (EntityIdentity identity : db.identity().getIdentities(account.id))
jidentities.put(identity.toJSON());
jaccount.put("identities", jidentities);
// Folders
JSONArray jfolders = new JSONArray();
for (EntityFolder folder : db.folder().getFolders(account.id)) {
JSONObject jfolder = folder.toJSON();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
NotificationChannel channel = nm.getNotificationChannel(
EntityFolder.getNotificationChannelId(folder.id));
if (channel != null && channel.getImportance() != NotificationManager.IMPORTANCE_NONE) {
JSONObject jchannel = channelToJSON(channel);
jfolder.put("channel", jchannel);
Log.i("Exported folder channel=" + jchannel);
}
}
2019-01-27 09:32:35 +00:00
JSONArray jrules = new JSONArray();
for (EntityRule rule : db.rule().getRules(folder.id))
jrules.put(rule.toJSON());
jfolder.put("rules", jrules);
2019-01-27 09:32:35 +00:00
jfolders.put(jfolder);
2019-01-14 10:29:47 +00:00
}
2019-01-27 09:32:35 +00:00
jaccount.put("folders", jfolders);
2019-01-14 10:29:47 +00:00
2019-03-17 08:26:17 +00:00
// Contacts
JSONArray jcontacts = new JSONArray();
for (EntityContact contact : db.contact().getContacts(account.id))
jcontacts.put(contact.toJSON());
jaccount.put("contacts", jcontacts);
2019-01-27 09:32:35 +00:00
jaccounts.put(jaccount);
}
2019-01-14 10:29:47 +00:00
2019-01-27 09:32:35 +00:00
// Answers
JSONArray janswers = new JSONArray();
2019-04-20 08:35:30 +00:00
for (EntityAnswer answer : db.answer().getAnswers(true))
2019-01-27 09:32:35 +00:00
janswers.put(answer.toJSON());
// Settings
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
JSONArray jsettings = new JSONArray();
for (String key : prefs.getAll().keySet())
if (!"pro".equals(key)) {
JSONObject jsetting = new JSONObject();
jsetting.put("key", key);
jsetting.put("value", prefs.getAll().get(key));
jsettings.put(jsetting);
}
JSONObject jexport = new JSONObject();
jexport.put("accounts", jaccounts);
jexport.put("answers", janswers);
jexport.put("settings", jsettings);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
JSONArray jchannels = new JSONArray();
for (NotificationChannel channel : nm.getNotificationChannels()) {
String id = channel.getId();
if (id.startsWith("notification.") && id.contains("@") &&
channel.getImportance() != NotificationManager.IMPORTANCE_NONE) {
JSONObject jchannel = channelToJSON(channel);
jchannels.put(jchannel);
Log.i("Exported contact channel=" + jchannel);
}
}
jexport.put("channels", jchannels);
}
2019-01-14 10:29:47 +00:00
2019-01-27 09:32:35 +00:00
ContentResolver resolver = context.getContentResolver();
DocumentFile file = DocumentFile.fromSingleUri(context, uri);
2019-02-22 15:59:23 +00:00
try (OutputStream raw = new BufferedOutputStream(resolver.openOutputStream(uri))) {
2019-01-27 09:32:35 +00:00
Log.i("Writing URI=" + uri + " name=" + file.getName() + " virtual=" + file.isVirtual());
if (TextUtils.isEmpty(password))
raw.write(jexport.toString(2).getBytes());
else {
byte[] salt = new byte[16];
SecureRandom random = new SecureRandom();
random.nextBytes(salt);
// https://docs.oracle.com/javase/7/docs/technotes/guides/security/StandardNames.html#Cipher
SecretKeyFactory keyFactory = SecretKeyFactory.getInstance("PBKDF2WithHmacSHA1");
KeySpec keySpec = new PBEKeySpec(password.toCharArray(), salt, KEY_ITERATIONS, KEY_LENGTH);
SecretKey secret = keyFactory.generateSecret(keySpec);
Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");
cipher.init(Cipher.ENCRYPT_MODE, secret);
raw.write(salt);
raw.write(cipher.getIV());
OutputStream cout = new CipherOutputStream(raw, cipher);
cout.write(jexport.toString(2).getBytes());
cout.flush();
2019-03-16 11:22:57 +00:00
raw.write(cipher.doFinal());
2019-01-27 09:32:35 +00:00
}
2019-01-14 10:29:47 +00:00
Log.i("Exported data");
}
return null;
}
@Override
protected void onExecuted(Bundle args, Void data) {
Snackbar.make(view, R.string.title_setup_exported, Snackbar.LENGTH_LONG).show();
}
@Override
protected void onException(Bundle args, Throwable ex) {
if (ex instanceof IllegalArgumentException)
Snackbar.make(view, ex.getMessage(), Snackbar.LENGTH_LONG).show();
else
Helper.unexpectedError(ActivitySetup.this, ActivitySetup.this, ex);
}
}.execute(this, args, "setup:export");
}
private void handleImport(Intent data, String password) {
Bundle args = new Bundle();
args.putParcelable("uri", data.getData());
args.putString("password", password);
new SimpleTask<Void>() {
@Override
protected Void onExecute(Context context, Bundle args) throws Throwable {
Uri uri = args.getParcelable("uri");
String password = args.getString("password");
if ("file".equals(uri.getScheme())) {
Log.w("Import uri=" + uri);
throw new IllegalArgumentException(context.getString(R.string.title_no_stream));
}
2019-01-27 09:32:35 +00:00
StringBuilder data = new StringBuilder();
2019-02-22 15:59:23 +00:00
Log.i("Reading URI=" + uri);
ContentResolver resolver = context.getContentResolver();
AssetFileDescriptor descriptor = resolver.openTypedAssetFileDescriptor(uri, "*/*", null);
try (InputStream raw = new BufferedInputStream(descriptor.createInputStream())) {
2019-01-27 09:32:35 +00:00
InputStream in;
if (TextUtils.isEmpty(password))
in = raw;
else {
byte[] salt = new byte[16];
byte[] prefix = new byte[16];
if (raw.read(salt) != salt.length)
throw new IOException("length");
if (raw.read(prefix) != prefix.length)
throw new IOException("length");
SecretKeyFactory keyFactory = SecretKeyFactory.getInstance("PBKDF2WithHmacSHA1");
KeySpec keySpec = new PBEKeySpec(password.toCharArray(), salt, KEY_ITERATIONS, KEY_LENGTH);
SecretKey secret = keyFactory.generateSecret(keySpec);
Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");
IvParameterSpec iv = new IvParameterSpec(prefix);
cipher.init(Cipher.DECRYPT_MODE, secret, iv);
in = new CipherInputStream(raw, cipher);
}
2019-01-14 10:29:47 +00:00
BufferedReader reader = new BufferedReader(new InputStreamReader(in));
String line;
while ((line = reader.readLine()) != null)
2019-01-27 09:32:35 +00:00
data.append(line);
}
2019-01-14 10:29:47 +00:00
2019-01-27 09:32:35 +00:00
Log.i("Importing data");
JSONObject jimport = new JSONObject(data.toString());
2019-01-14 10:29:47 +00:00
2019-01-27 09:32:35 +00:00
DB db = DB.getInstance(context);
NotificationManager nm = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
2019-01-27 09:32:35 +00:00
try {
db.beginTransaction();
2019-06-07 15:42:55 +00:00
// Answers
Map<Long, Long> xAnswer = new HashMap<>();
JSONArray janswers = jimport.getJSONArray("answers");
for (int a = 0; a < janswers.length(); a++) {
JSONObject janswer = (JSONObject) janswers.get(a);
EntityAnswer answer = EntityAnswer.fromJSON(janswer);
long id = answer.id;
answer.id = null;
answer.id = db.answer().insertAnswer(answer);
xAnswer.put(id, answer.id);
Log.i("Imported answer=" + answer.name);
}
2019-02-13 09:24:06 +00:00
// Accounts
2019-01-27 09:32:35 +00:00
JSONArray jaccounts = jimport.getJSONArray("accounts");
for (int a = 0; a < jaccounts.length(); a++) {
JSONObject jaccount = (JSONObject) jaccounts.get(a);
EntityAccount account = EntityAccount.fromJSON(jaccount);
// Forward referenced
Long swipe_left = account.swipe_left;
Long swipe_right = account.swipe_right;
account.swipe_left = null;
account.swipe_right = null;
2019-01-27 09:32:35 +00:00
account.created = new Date().getTime();
account.id = db.account().insertAccount(account);
Log.i("Imported account=" + account.name);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
account.deleteNotificationChannel(context);
if (account.notify)
if (jaccount.has("channel")) {
2019-06-08 07:51:00 +00:00
NotificationChannelGroup group = new NotificationChannelGroup("group." + account.id, account.name);
nm.createNotificationChannelGroup(group);
JSONObject jchannel = (JSONObject) jaccount.get("channel");
jchannel.put("id", EntityAccount.getNotificationChannelId(account.id));
2019-06-08 07:51:00 +00:00
jchannel.put("group", group.getId());
nm.createNotificationChannel(channelFromJSON(context, jchannel));
Log.i("Imported account channel=" + jchannel);
} else
account.createNotificationChannel(context);
}
2019-01-27 09:32:35 +00:00
2019-06-07 15:42:55 +00:00
Map<Long, Long> xIdentity = new HashMap<>();
2019-01-27 09:32:35 +00:00
JSONArray jidentities = (JSONArray) jaccount.get("identities");
for (int i = 0; i < jidentities.length(); i++) {
JSONObject jidentity = (JSONObject) jidentities.get(i);
EntityIdentity identity = EntityIdentity.fromJSON(jidentity);
2019-06-07 15:42:55 +00:00
long id = identity.id;
identity.id = null;
2019-01-27 09:32:35 +00:00
identity.account = account.id;
identity.id = db.identity().insertIdentity(identity);
2019-06-07 15:42:55 +00:00
xIdentity.put(id, identity.id);
2019-01-27 09:32:35 +00:00
Log.i("Imported identity=" + identity.email);
2019-01-14 10:29:47 +00:00
}
2019-06-07 15:42:55 +00:00
Map<Long, Long> xFolder = new HashMap<>();
List<EntityRule> rules = new ArrayList<>();
2019-01-27 09:32:35 +00:00
JSONArray jfolders = (JSONArray) jaccount.get("folders");
for (int f = 0; f < jfolders.length(); f++) {
JSONObject jfolder = (JSONObject) jfolders.get(f);
EntityFolder folder = EntityFolder.fromJSON(jfolder);
long id = folder.id;
folder.id = null;
2019-01-27 09:32:35 +00:00
folder.account = account.id;
folder.id = db.folder().insertFolder(folder);
2019-06-07 15:42:55 +00:00
xFolder.put(id, folder.id);
2019-02-26 10:05:21 +00:00
if (Objects.equals(swipe_left, id))
account.swipe_left = folder.id;
2019-02-26 10:05:21 +00:00
if (Objects.equals(swipe_right, id))
account.swipe_right = folder.id;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
if (jfolder.has("channel")) {
2019-06-08 07:51:00 +00:00
NotificationChannelGroup group = new NotificationChannelGroup("group." + account.id, account.name);
nm.createNotificationChannelGroup(group);
JSONObject jchannel = (JSONObject) jfolder.get("channel");
jchannel.put("id", EntityFolder.getNotificationChannelId(folder.id));
2019-06-08 07:51:00 +00:00
jchannel.put("group", group.getId());
nm.createNotificationChannel(channelFromJSON(context, jchannel));
Log.i("Imported folder channel=" + jchannel);
}
}
2019-01-27 09:32:35 +00:00
if (jfolder.has("rules")) {
JSONArray jrules = jfolder.getJSONArray("rules");
for (int r = 0; r < jrules.length(); r++) {
JSONObject jrule = (JSONObject) jrules.get(r);
EntityRule rule = EntityRule.fromJSON(jrule);
rule.folder = folder.id;
2019-06-07 15:42:55 +00:00
rules.add(rule);
2019-01-27 09:32:35 +00:00
}
2019-01-14 10:29:47 +00:00
}
2019-01-27 09:32:35 +00:00
Log.i("Imported folder=" + folder.name);
2019-01-14 10:29:47 +00:00
}
2019-06-07 15:42:55 +00:00
for (EntityRule rule : rules) {
try {
JSONObject jaction = new JSONObject(rule.action);
int type = jaction.getInt("type");
switch (type) {
case EntityRule.TYPE_MOVE:
case EntityRule.TYPE_COPY:
long target = jaction.getLong("target");
Log.i("XLAT target " + target + " > " + xFolder.get(target));
jaction.put("target", xFolder.get(target));
break;
case EntityRule.TYPE_ANSWER:
long iid = jaction.getLong("identity");
long aid = jaction.getLong("answer");
Log.i("XLAT identity " + iid + " > " + xIdentity.get(iid));
Log.i("XLAT target " + aid + " > " + xAnswer.get(aid));
jaction.put("identity", xIdentity.get(iid));
jaction.put("answer", xAnswer.get(aid));
break;
}
} catch (JSONException ex) {
Log.e(ex);
}
db.rule().insertRule(rule);
}
2019-03-17 08:26:17 +00:00
// Contacts
2019-03-26 06:52:46 +00:00
if (jaccount.has("contacts")) {
JSONArray jcontacts = jaccount.getJSONArray("contacts");
for (int c = 0; c < jcontacts.length(); c++) {
JSONObject jcontact = (JSONObject) jcontacts.get(c);
EntityContact contact = EntityContact.fromJSON(jcontact);
contact.account = account.id;
if (db.contact().getContact(contact.account, contact.type, contact.email) == null) {
contact.id = db.contact().insertContact(contact);
Log.i("Imported contact=" + contact);
}
2019-03-17 08:26:17 +00:00
}
}
// Update swipe left/right
db.account().updateAccount(account);
2019-01-27 09:32:35 +00:00
}
2019-01-14 10:29:47 +00:00
2019-02-13 09:24:06 +00:00
// Settings
2019-01-27 09:32:35 +00:00
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
SharedPreferences.Editor editor = prefs.edit();
JSONArray jsettings = jimport.getJSONArray("settings");
for (int s = 0; s < jsettings.length(); s++) {
JSONObject jsetting = (JSONObject) jsettings.get(s);
String key = jsetting.getString("key");
if (!"pro".equals(key)) {
Object value = jsetting.get("value");
if (value instanceof Boolean)
editor.putBoolean(key, (Boolean) value);
else if (value instanceof Integer)
editor.putInt(key, (Integer) value);
else if (value instanceof Long)
editor.putLong(key, (Long) value);
else if (value instanceof String)
editor.putString(key, (String) value);
else
throw new IllegalArgumentException("Unknown settings type key=" + key);
Log.i("Imported setting=" + key);
}
}
editor.apply();
ApplicationEx.upgrade(context);
2019-01-27 09:32:35 +00:00
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
2019-03-17 17:18:53 +00:00
if (jimport.has("channels")) {
JSONArray jchannels = jimport.getJSONArray("channels");
for (int i = 0; i < jchannels.length(); i++) {
JSONObject jchannel = (JSONObject) jchannels.get(i);
nm.createNotificationChannel(channelFromJSON(context, jchannel));
Log.i("Imported contact channel=" + jchannel);
}
2019-03-17 17:18:53 +00:00
}
}
2019-03-17 17:18:53 +00:00
2019-01-27 09:32:35 +00:00
db.setTransactionSuccessful();
2019-01-14 10:29:47 +00:00
} finally {
2019-01-27 09:32:35 +00:00
db.endTransaction();
2019-01-14 10:29:47 +00:00
}
2019-01-27 09:32:35 +00:00
Log.i("Imported data");
ServiceSynchronize.reload(context, "import");
2019-01-14 10:29:47 +00:00
return null;
}
@Override
protected void onExecuted(Bundle args, Void data) {
Snackbar.make(view, R.string.title_setup_imported, Snackbar.LENGTH_LONG).show();
}
@Override
protected void onException(Bundle args, Throwable ex) {
if (ex.getCause() instanceof BadPaddingException)
Snackbar.make(view, R.string.title_setup_password_invalid, Snackbar.LENGTH_LONG).show();
else if (ex instanceof IllegalArgumentException)
Snackbar.make(view, ex.getMessage(), Snackbar.LENGTH_LONG).show();
else
Helper.unexpectedError(ActivitySetup.this, ActivitySetup.this, ex);
}
}.execute(this, args, "setup:import");
}
@RequiresApi(api = Build.VERSION_CODES.O)
private JSONObject channelToJSON(NotificationChannel channel) throws JSONException {
JSONObject jchannel = new JSONObject();
jchannel.put("id", channel.getId());
jchannel.put("group", channel.getGroup());
jchannel.put("name", channel.getName());
jchannel.put("description", channel.getDescription());
jchannel.put("importance", channel.getImportance());
jchannel.put("dnd", channel.canBypassDnd());
jchannel.put("visibility", channel.getLockscreenVisibility());
jchannel.put("badge", channel.canShowBadge());
Uri sound = channel.getSound();
if (sound != null)
jchannel.put("sound", sound.toString());
// audio attributes
jchannel.put("light", channel.shouldShowLights());
// color
jchannel.put("vibrate", channel.shouldVibrate());
// pattern
return jchannel;
}
@RequiresApi(api = Build.VERSION_CODES.O)
static NotificationChannel channelFromJSON(Context context, JSONObject jchannel) throws JSONException {
NotificationChannel channel = new NotificationChannel(
jchannel.getString("id"),
jchannel.getString("name"),
jchannel.getInt("importance"));
channel.setGroup(jchannel.getString("group"));
if (jchannel.has("description") && !jchannel.isNull("description"))
channel.setDescription(jchannel.getString("description"));
channel.setBypassDnd(jchannel.getBoolean("dnd"));
channel.setLockscreenVisibility(jchannel.getInt("visibility"));
channel.setShowBadge(jchannel.getBoolean("badge"));
if (jchannel.has("sound") && !jchannel.isNull("sound")) {
Uri uri = Uri.parse(jchannel.getString("sound"));
Ringtone ringtone = RingtoneManager.getRingtone(context, uri);
if (ringtone != null)
channel.setSound(uri, Notification.AUDIO_ATTRIBUTES_DEFAULT);
}
channel.enableLights(jchannel.getBoolean("light"));
channel.enableVibration(jchannel.getBoolean("vibrate"));
return channel;
}
2019-06-12 14:15:46 +00:00
private void onViewQuickSetup(Intent intent) {
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.replace(R.id.content_frame, new FragmentQuickSetup()).addToBackStack("quick");
fragmentTransaction.commit();
}
private void onViewAccounts(Intent intent) {
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.replace(R.id.content_frame, new FragmentAccounts()).addToBackStack("accounts");
fragmentTransaction.commit();
}
private void onViewIdentities(Intent intent) {
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.replace(R.id.content_frame, new FragmentIdentities()).addToBackStack("identities");
fragmentTransaction.commit();
}
2019-01-14 12:58:33 +00:00
private void onEditAccount(Intent intent) {
FragmentAccount fragment = new FragmentAccount();
fragment.setArguments(intent.getExtras());
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.replace(R.id.content_frame, fragment).addToBackStack("account");
fragmentTransaction.commit();
}
private void onEditIdentity(Intent intent) {
FragmentIdentity fragment = new FragmentIdentity();
fragment.setArguments(intent.getExtras());
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.replace(R.id.content_frame, fragment).addToBackStack("identity");
fragmentTransaction.commit();
}
2019-05-06 08:48:14 +00:00
private void onShowPro(Intent intent) {
if (getLifecycle().getCurrentState().isAtLeast(Lifecycle.State.RESUMED))
getSupportFragmentManager().popBackStack("pro", FragmentManager.POP_BACK_STACK_INCLUSIVE);
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragmentTransaction.replace(R.id.content_frame, new FragmentPro()).addToBackStack("pro");
fragmentTransaction.commit();
}
2019-05-27 10:15:25 +00:00
private BroadcastReceiver receiver = new BroadcastReceiver() {
2018-08-04 15:46:22 +00:00
@Override
public void onReceive(Context context, Intent intent) {
2019-06-12 14:15:46 +00:00
String action = intent.getAction();
if (ACTION_QUICK_SETUP.equals(action))
onViewQuickSetup(intent);
else if (ACTION_VIEW_ACCOUNTS.equals(action))
onViewAccounts(intent);
else if (ACTION_VIEW_IDENTITIES.equals(action))
onViewIdentities(intent);
else if (ACTION_EDIT_ACCOUNT.equals(action))
2019-01-14 12:58:33 +00:00
onEditAccount(intent);
2019-06-12 14:15:46 +00:00
else if (ACTION_EDIT_IDENTITY.equals(action))
2019-01-14 12:58:33 +00:00
onEditIdentity(intent);
2019-06-12 14:15:46 +00:00
else if (ACTION_SHOW_PRO.equals(action))
2019-05-06 08:48:14 +00:00
onShowPro(intent);
2018-08-04 15:46:22 +00:00
}
};
2018-08-02 13:33:06 +00:00
}