add backup import and export
This commit is contained in:
		
							parent
							
								
									5f673eadcb
								
							
						
					
					
						commit
						c4b63e5086
					
				|  | @ -1,20 +1,49 @@ | ||||||
| package eu.siacs.conversations.ui; | package eu.siacs.conversations.ui; | ||||||
| 
 | 
 | ||||||
|  | import android.Manifest; | ||||||
|  | import android.app.ProgressDialog; | ||||||
|  | import android.content.DialogInterface; | ||||||
| import android.content.Intent; | import android.content.Intent; | ||||||
| import android.content.pm.ActivityInfo; | import android.content.pm.ActivityInfo; | ||||||
|  | import android.content.pm.PackageManager; | ||||||
|  | import android.database.sqlite.SQLiteDatabase; | ||||||
|  | import android.database.sqlite.SQLiteException; | ||||||
|  | import android.os.Build; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
| import android.support.v7.app.ActionBar; | import android.support.v7.app.ActionBar; | ||||||
|  | import android.support.v7.app.AlertDialog; | ||||||
| import android.support.v7.app.AppCompatActivity; | import android.support.v7.app.AppCompatActivity; | ||||||
|  | import android.util.Log; | ||||||
|  | import android.view.LayoutInflater; | ||||||
|  | import android.view.View; | ||||||
| import android.widget.Button; | import android.widget.Button; | ||||||
|  | import android.widget.EditText; | ||||||
|  | import android.widget.TextView; | ||||||
|  | import android.widget.Toast; | ||||||
| 
 | 
 | ||||||
|  | import java.io.File; | ||||||
|  | import java.io.FileInputStream; | ||||||
|  | import java.io.FileOutputStream; | ||||||
|  | import java.io.IOException; | ||||||
|  | import java.io.InputStream; | ||||||
|  | import java.io.OutputStream; | ||||||
|  | import java.security.InvalidKeyException; | ||||||
|  | import java.security.NoSuchAlgorithmException; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| 
 | 
 | ||||||
|  | import javax.crypto.NoSuchPaddingException; | ||||||
|  | 
 | ||||||
|  | import eu.siacs.conversations.Config; | ||||||
| import eu.siacs.conversations.R; | import eu.siacs.conversations.R; | ||||||
| import eu.siacs.conversations.entities.Account; | import eu.siacs.conversations.entities.Account; | ||||||
| import eu.siacs.conversations.utils.XmppUri; | import eu.siacs.conversations.persistance.DatabaseBackend; | ||||||
|  | import eu.siacs.conversations.persistance.FileBackend; | ||||||
|  | import eu.siacs.conversations.utils.EncryptDecryptFile; | ||||||
| 
 | 
 | ||||||
| public class WelcomeActivity extends XmppActivity { | public class WelcomeActivity extends XmppActivity { | ||||||
| 
 | 
 | ||||||
|  | 	boolean importSuccessful = false; | ||||||
|  | 
 | ||||||
| 	@Override | 	@Override | ||||||
| 	protected void refreshUiReal() { | 	protected void refreshUiReal() { | ||||||
| 
 | 
 | ||||||
|  | @ -25,6 +54,8 @@ public class WelcomeActivity extends XmppActivity { | ||||||
| 
 | 
 | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	private static final int REQUEST_READ_EXTERNAL_STORAGE = 0XD737; | ||||||
|  | 
 | ||||||
| 	@Override | 	@Override | ||||||
| 	public void onStart() { | 	public void onStart() { | ||||||
| 		super.onStart(); | 		super.onStart(); | ||||||
|  | @ -54,7 +85,19 @@ public class WelcomeActivity extends XmppActivity { | ||||||
| 			ab.setDisplayShowHomeEnabled(false); | 			ab.setDisplayShowHomeEnabled(false); | ||||||
| 			ab.setDisplayHomeAsUpEnabled(false); | 			ab.setDisplayHomeAsUpEnabled(false); | ||||||
| 		} | 		} | ||||||
|  | 		//check if there is a backed up database -- | ||||||
|  | 		if (hasStoragePermission(REQUEST_READ_EXTERNAL_STORAGE)) { | ||||||
|  | 			backupAvailable(); | ||||||
|  | 		} | ||||||
|  | 		final Button importDatabase = findViewById(R.id.import_database); | ||||||
|  | 		final TextView importText = findViewById(R.id.import_text); | ||||||
| 		final Button createAccount = findViewById(R.id.create_account); | 		final Button createAccount = findViewById(R.id.create_account); | ||||||
|  | 		if (backupAvailable()) { | ||||||
|  | 			importDatabase.setVisibility(View.VISIBLE); | ||||||
|  | 			importText.setVisibility(View.VISIBLE); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		importDatabase.setOnClickListener(v -> enterPasswordDialog()); | ||||||
| 		createAccount.setOnClickListener(v -> { | 		createAccount.setOnClickListener(v -> { | ||||||
| 			final Intent intent = new Intent(WelcomeActivity.this, MagicCreateActivity.class); | 			final Intent intent = new Intent(WelcomeActivity.this, MagicCreateActivity.class); | ||||||
| 			intent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION); | 			intent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION); | ||||||
|  | @ -77,6 +120,205 @@ public class WelcomeActivity extends XmppActivity { | ||||||
| 
 | 
 | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	public void enterPasswordDialog() { | ||||||
|  | 		LayoutInflater li = LayoutInflater.from(WelcomeActivity.this); | ||||||
|  | 		View promptsView = li.inflate(R.layout.backup_password, null); | ||||||
|  | 		final AlertDialog.Builder alertDialogBuilder = new AlertDialog.Builder(WelcomeActivity.this); | ||||||
|  | 		alertDialogBuilder.setView(promptsView); | ||||||
|  | 		final EditText userInput = promptsView | ||||||
|  | 				.findViewById(R.id.password); | ||||||
|  | 		alertDialogBuilder.setTitle(R.string.enter_password); | ||||||
|  | 		alertDialogBuilder.setMessage(R.string.enter_backup_password); | ||||||
|  | 		alertDialogBuilder | ||||||
|  | 				.setCancelable(false) | ||||||
|  | 				.setPositiveButton(R.string.ok, | ||||||
|  | 						new DialogInterface.OnClickListener() { | ||||||
|  | 							public void onClick(DialogInterface dialog, int id) { | ||||||
|  | 								final String password = userInput.getText().toString(); | ||||||
|  | 								final ProgressDialog pd = ProgressDialog.show(WelcomeActivity.this, getString(R.string.please_wait), getString(R.string.import_started), true); | ||||||
|  | 								if (!password.isEmpty()) { | ||||||
|  | 									new Thread(new Runnable() { | ||||||
|  | 										@Override | ||||||
|  | 										public void run() { | ||||||
|  | 											try { | ||||||
|  | 												checkDatabase(password); | ||||||
|  | 											} catch (IOException e) { | ||||||
|  | 												e.printStackTrace(); | ||||||
|  | 											} catch (Exception e) { | ||||||
|  | 												e.printStackTrace(); | ||||||
|  | 											} | ||||||
|  | 											pd.dismiss(); | ||||||
|  | 										} | ||||||
|  | 									}).start(); | ||||||
|  | 								} else { | ||||||
|  | 									AlertDialog.Builder builder = new AlertDialog.Builder(WelcomeActivity.this); | ||||||
|  | 									builder.setTitle(R.string.error); | ||||||
|  | 									builder.setMessage(R.string.password_should_not_be_empty); | ||||||
|  | 									builder.setNegativeButton(R.string.cancel, null); | ||||||
|  | 									builder.setPositiveButton(R.string.try_again, new DialogInterface.OnClickListener() { | ||||||
|  | 										@Override | ||||||
|  | 										public void onClick(DialogInterface dialog, int id) { | ||||||
|  | 											enterPasswordDialog(); | ||||||
|  | 										} | ||||||
|  | 									}); | ||||||
|  | 									builder.create().show(); | ||||||
|  | 								} | ||||||
|  | 							} | ||||||
|  | 						}) | ||||||
|  | 				.setNegativeButton(R.string.cancel, | ||||||
|  | 						new DialogInterface.OnClickListener() { | ||||||
|  | 							public void onClick(DialogInterface dialog, int id) { | ||||||
|  | 								Toast.makeText(WelcomeActivity.this, R.string.import_canceled, Toast.LENGTH_LONG).show(); | ||||||
|  | 								dialog.dismiss(); | ||||||
|  | 							} | ||||||
|  | 						} | ||||||
|  | 				); | ||||||
|  | 		WelcomeActivity.this.runOnUiThread(new Runnable() { | ||||||
|  | 			public void run() { | ||||||
|  | 				// create alert dialog | ||||||
|  | 				AlertDialog alertDialog = alertDialogBuilder.create(); | ||||||
|  | 				// show it | ||||||
|  | 				alertDialog.show(); | ||||||
|  | 			} | ||||||
|  | 		}); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	private boolean backupAvailable() { | ||||||
|  | 		// Set the folder on the SDcard | ||||||
|  | 		File filePath = new File(FileBackend.getBackupDirectory() + "/database.db.crypt"); | ||||||
|  | 		Log.d(Config.LOGTAG, "DB Path: " + filePath.toString()); | ||||||
|  | 		if (filePath.exists()) { | ||||||
|  | 			Log.d(Config.LOGTAG, "DB Path existing"); | ||||||
|  | 			return true; | ||||||
|  | 		} else { | ||||||
|  | 			Log.d(Config.LOGTAG, "DB Path not existing"); | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	private void checkDatabase(String DecryptionKey) throws IOException { | ||||||
|  | 		// Set the folder on the SDcard | ||||||
|  | 		File directory = new File(FileBackend.getBackupDirectory()); | ||||||
|  | 		// Set the input file stream up: | ||||||
|  | 		FileInputStream InputFile = new FileInputStream(directory.getPath() + "/database.db.crypt"); | ||||||
|  | 		// Temp output for DB checks | ||||||
|  | 		File TempFile = new File(directory.getPath() + "database.bak"); | ||||||
|  | 		FileOutputStream OutputTemp = new FileOutputStream(TempFile); | ||||||
|  | 
 | ||||||
|  | 		try { | ||||||
|  | 			EncryptDecryptFile.decrypt(InputFile, OutputTemp, DecryptionKey); | ||||||
|  | 		} catch (NoSuchAlgorithmException e) { | ||||||
|  | 			Log.d(Config.LOGTAG, "Database importer: decryption failed with " + e); | ||||||
|  | 			e.printStackTrace(); | ||||||
|  | 		} catch (NoSuchPaddingException e) { | ||||||
|  | 			Log.d(Config.LOGTAG, "Database importer: decryption failed with " + e); | ||||||
|  | 			e.printStackTrace(); | ||||||
|  | 		} catch (InvalidKeyException e) { | ||||||
|  | 			Log.d(Config.LOGTAG, "Database importer: decryption failed (invalid key) with " + e); | ||||||
|  | 			e.printStackTrace(); | ||||||
|  | 		} catch (IOException e) { | ||||||
|  | 			Log.d(Config.LOGTAG, "Database importer: decryption failed (IO) with " + e); | ||||||
|  | 			e.printStackTrace(); | ||||||
|  | 		} catch (Exception e) { | ||||||
|  | 			Log.d(Config.LOGTAG, "Database importer: Error " + e); | ||||||
|  | 			e.printStackTrace(); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		SQLiteDatabase checkDB = null; | ||||||
|  | 		int DB_Version = DatabaseBackend.DATABASE_VERSION; | ||||||
|  | 		int Backup_DB_Version = 0; | ||||||
|  | 
 | ||||||
|  | 		try { | ||||||
|  | 			String dbPath = TempFile.toString(); | ||||||
|  | 			checkDB = SQLiteDatabase.openDatabase(dbPath, null, SQLiteDatabase.OPEN_READONLY); | ||||||
|  | 			Backup_DB_Version = checkDB.getVersion(); | ||||||
|  | 			Log.d(Config.LOGTAG, "Backup found: " + checkDB + " Version: " + checkDB.getVersion()); | ||||||
|  | 		} catch (SQLiteException e) { | ||||||
|  | 			//database does't exist yet. | ||||||
|  | 			Log.d(Config.LOGTAG, "No backup found: " + checkDB); | ||||||
|  | 		} catch (Exception e) { | ||||||
|  | 			Log.d(Config.LOGTAG, "Error importing backup: " + e); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (checkDB != null) { | ||||||
|  | 			checkDB.close(); | ||||||
|  | 		} | ||||||
|  | 		if (checkDB != null) { | ||||||
|  | 			Log.d(Config.LOGTAG, "checkDB = " + checkDB.toString() + ", Backup DB = " + Backup_DB_Version + ", DB = " + DB_Version); | ||||||
|  | 		} | ||||||
|  | 		if (checkDB != null && Backup_DB_Version != 0 && Backup_DB_Version <= DB_Version) { | ||||||
|  | 			try { | ||||||
|  | 				importDatabase(); | ||||||
|  | 				importSuccessful = true; | ||||||
|  | 			} catch (Exception e) { | ||||||
|  | 				importSuccessful = false; | ||||||
|  | 				e.printStackTrace(); | ||||||
|  | 			} finally { | ||||||
|  | 				if (importSuccessful) { | ||||||
|  | 					restart(); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} else if (checkDB != null && Backup_DB_Version == 0) { | ||||||
|  | 			WelcomeActivity.this.runOnUiThread(new Runnable() { | ||||||
|  | 				public void run() { | ||||||
|  | 					Toast.makeText(WelcomeActivity.this, R.string.password_wrong, Toast.LENGTH_LONG).show(); | ||||||
|  | 					enterPasswordDialog(); | ||||||
|  | 				} | ||||||
|  | 			}); | ||||||
|  | 		} else { | ||||||
|  | 			WelcomeActivity.this.runOnUiThread(new Runnable() { | ||||||
|  | 				public void run() { | ||||||
|  | 					Toast.makeText(WelcomeActivity.this, R.string.import_failed, Toast.LENGTH_LONG).show(); | ||||||
|  | 				} | ||||||
|  | 			}); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	private void importDatabase() throws Exception { | ||||||
|  | 		// Set location for the db: | ||||||
|  | 		final OutputStream OutputFile = new FileOutputStream(this.getDatabasePath(DatabaseBackend.DATABASE_NAME)); | ||||||
|  | 		// Set the folder on the SDcard | ||||||
|  | 		File directory = new File(FileBackend.getBackupDirectory()); | ||||||
|  | 		// Set the input file stream up: | ||||||
|  | 		final InputStream InputFile = new FileInputStream(directory.getPath() + "database.bak"); | ||||||
|  | 		//set temp file | ||||||
|  | 		File TempFile = new File(directory.getPath() + "database.bak"); | ||||||
|  | 
 | ||||||
|  | 		// Transfer bytes from the input file to the output file | ||||||
|  | 		byte[] buffer = new byte[1024]; | ||||||
|  | 		int length; | ||||||
|  | 		while ((length = InputFile.read(buffer)) > 0) { | ||||||
|  | 			OutputFile.write(buffer, 0, length); | ||||||
|  | 		} | ||||||
|  | 		if (TempFile.exists()) { | ||||||
|  | 			Log.d(Config.LOGTAG, "Delete temp file from " + TempFile.toString()); | ||||||
|  | 			TempFile.delete(); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	private void restart() { | ||||||
|  | 		//restart app | ||||||
|  | 		Log.d(Config.LOGTAG, "Restarting " + getBaseContext().getPackageManager().getLaunchIntentForPackage(getBaseContext().getPackageName())); | ||||||
|  | 		Intent intent = getBaseContext().getPackageManager().getLaunchIntentForPackage(getBaseContext().getPackageName()); | ||||||
|  | 		intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); | ||||||
|  | 		intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); | ||||||
|  | 		startActivity(intent); | ||||||
|  | 		System.exit(0); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	public boolean hasStoragePermission(int requestCode) { | ||||||
|  | 		if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { | ||||||
|  | 			if (checkSelfPermission(Manifest.permission.READ_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) { | ||||||
|  | 				requestPermissions(new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, requestCode); | ||||||
|  | 				return false; | ||||||
|  | 			} else { | ||||||
|  | 				return true; | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			return true; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	public void addInviteUri(Intent intent) { | 	public void addInviteUri(Intent intent) { | ||||||
| 		StartConversationActivity.addInviteUri(intent, getIntent()); | 		StartConversationActivity.addInviteUri(intent, getIntent()); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | @ -42,6 +42,23 @@ | ||||||
|                     android:layout_marginTop="8dp" |                     android:layout_marginTop="8dp" | ||||||
|                     android:text="@string/welcome_text" |                     android:text="@string/welcome_text" | ||||||
|                     android:textAppearance="@style/TextAppearance.Conversations.Body1"/> |                     android:textAppearance="@style/TextAppearance.Conversations.Body1"/> | ||||||
|  |                 <TextView | ||||||
|  |                     android:id="@+id/import_text" | ||||||
|  |                     android:layout_width="wrap_content" | ||||||
|  |                     android:layout_height="wrap_content" | ||||||
|  |                     android:layout_marginTop="8dp" | ||||||
|  |                     android:text="@string/import_text" | ||||||
|  |                     android:textAppearance="@style/TextAppearance.Conversations.Caption.Highlight" | ||||||
|  |                     android:visibility="gone" /> | ||||||
|  |                 <Button | ||||||
|  |                     android:id="@+id/import_database" | ||||||
|  |                     style="@style/Widget.Conversations.Button.Borderless" | ||||||
|  |                     android:layout_width="wrap_content" | ||||||
|  |                     android:layout_height="wrap_content" | ||||||
|  |                     android:layout_gravity="right" | ||||||
|  |                     android:text="@string/import_database" | ||||||
|  |                     android:textColor="?colorAccent" | ||||||
|  |                     android:visibility="gone" /> | ||||||
|                 <Button |                 <Button | ||||||
|                     android:id="@+id/create_account" |                     android:id="@+id/create_account" | ||||||
|                     style="@style/Widget.Conversations.Button.Borderless" |                     style="@style/Widget.Conversations.Button.Borderless" | ||||||
|  | @ -57,7 +74,7 @@ | ||||||
|                     android:layout_height="wrap_content" |                     android:layout_height="wrap_content" | ||||||
|                     android:layout_gravity="right" |                     android:layout_gravity="right" | ||||||
|                     android:text="@string/use_own_provider" |                     android:text="@string/use_own_provider" | ||||||
|                     android:textColor="?android:textColorSecondary"/> |                     android:textColor="?colorAccent"/> | ||||||
|             </LinearLayout> |             </LinearLayout> | ||||||
|             <RelativeLayout |             <RelativeLayout | ||||||
|                 android:layout_width="match_parent" |                 android:layout_width="match_parent" | ||||||
|  |  | ||||||
|  | @ -226,6 +226,7 @@ | ||||||
|             android:label="@string/media_browser"/> |             android:label="@string/media_browser"/> | ||||||
| 
 | 
 | ||||||
|         <service android:name=".services.ExportLogsService"/> |         <service android:name=".services.ExportLogsService"/> | ||||||
|  |         <service android:name=".services.BackupService"/> | ||||||
|         <service |         <service | ||||||
|             android:name=".services.ContactChooserTargetService" |             android:name=".services.ContactChooserTargetService" | ||||||
|             android:permission="android.permission.BIND_CHOOSER_TARGET_SERVICE"> |             android:permission="android.permission.BIND_CHOOSER_TARGET_SERVICE"> | ||||||
|  |  | ||||||
|  | @ -61,8 +61,8 @@ import rocks.xmpp.addr.Jid; | ||||||
| 
 | 
 | ||||||
| public class DatabaseBackend extends SQLiteOpenHelper { | public class DatabaseBackend extends SQLiteOpenHelper { | ||||||
| 
 | 
 | ||||||
| 	private static final String DATABASE_NAME = "history"; | 	public static final String DATABASE_NAME = "history"; | ||||||
| 	private static final int DATABASE_VERSION = 42; | 	public static final int DATABASE_VERSION = 42; | ||||||
| 	private static DatabaseBackend instance = null; | 	private static DatabaseBackend instance = null; | ||||||
| 	private static String CREATE_CONTATCS_STATEMENT = "create table " | 	private static String CREATE_CONTATCS_STATEMENT = "create table " | ||||||
| 			+ Contact.TABLENAME + "(" + Contact.ACCOUNT + " TEXT, " | 			+ Contact.TABLENAME + "(" + Contact.ACCOUNT + " TEXT, " | ||||||
|  |  | ||||||
|  | @ -148,6 +148,10 @@ public class FileBackend { | ||||||
|         return Environment.getExternalStorageDirectory().getAbsolutePath() + "/" + context.getString(R.string.app_name) + "/Media/"; |         return Environment.getExternalStorageDirectory().getAbsolutePath() + "/" + context.getString(R.string.app_name) + "/Media/"; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public static String getBackupDirectory() { | ||||||
|  |         return Environment.getExternalStorageDirectory().getAbsolutePath() + "/Conversations/Database"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     public static String getConversationsLogsDirectory() { |     public static String getConversationsLogsDirectory() { | ||||||
|         return Environment.getExternalStorageDirectory().getAbsolutePath() + "/Conversations/"; |         return Environment.getExternalStorageDirectory().getAbsolutePath() + "/Conversations/"; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -0,0 +1,138 @@ | ||||||
|  | package eu.siacs.conversations.services; | ||||||
|  | 
 | ||||||
|  | import android.app.AlarmManager; | ||||||
|  | import android.app.NotificationManager; | ||||||
|  | import android.app.PendingIntent; | ||||||
|  | import android.app.Service; | ||||||
|  | import android.content.Context; | ||||||
|  | import android.content.Intent; | ||||||
|  | import android.content.SharedPreferences; | ||||||
|  | import android.os.IBinder; | ||||||
|  | import android.support.v4.app.NotificationCompat; | ||||||
|  | import android.os.PowerManager; | ||||||
|  | import android.os.PowerManager.WakeLock; | ||||||
|  | import android.preference.PreferenceManager; | ||||||
|  | import android.util.Log; | ||||||
|  | 
 | ||||||
|  | import java.io.File; | ||||||
|  | import java.io.FileInputStream; | ||||||
|  | import java.io.FileOutputStream; | ||||||
|  | import java.io.IOException; | ||||||
|  | import java.security.InvalidKeyException; | ||||||
|  | import java.security.NoSuchAlgorithmException; | ||||||
|  | import java.util.Calendar; | ||||||
|  | import java.util.concurrent.atomic.AtomicBoolean; | ||||||
|  | 
 | ||||||
|  | import javax.crypto.NoSuchPaddingException; | ||||||
|  | 
 | ||||||
|  | import eu.siacs.conversations.Config; | ||||||
|  | import eu.siacs.conversations.R; | ||||||
|  | import eu.siacs.conversations.persistance.DatabaseBackend; | ||||||
|  | import eu.siacs.conversations.persistance.FileBackend; | ||||||
|  | import eu.siacs.conversations.utils.EncryptDecryptFile; | ||||||
|  | import eu.siacs.conversations.utils.WakeLockHelper; | ||||||
|  | 
 | ||||||
|  | public class BackupService extends Service { | ||||||
|  | 
 | ||||||
|  |     private static final int NOTIFICATION_ID = 1; | ||||||
|  |     private static AtomicBoolean running = new AtomicBoolean(false); | ||||||
|  |     private WakeLock wakeLock; | ||||||
|  |     private PowerManager pm; | ||||||
|  |     private AlarmManager alarmMgr; | ||||||
|  |     private PendingIntent alarmIntent; | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public void onCreate() { | ||||||
|  |         pm = (PowerManager) getSystemService(Context.POWER_SERVICE); | ||||||
|  |         wakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, Config.LOGTAG + ": BackupService"); | ||||||
|  | 
 | ||||||
|  |         alarmIntent = PendingIntent.getService(this, 0, new Intent(this, BackupService.class), 0); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         Calendar calendar = Calendar.getInstance(); | ||||||
|  |         calendar.setTimeInMillis(System.currentTimeMillis()); | ||||||
|  |         calendar.set(Calendar.HOUR_OF_DAY, 4); | ||||||
|  | 
 | ||||||
|  |         // With setInexactRepeating(), you have to use one of the AlarmManager interval | ||||||
|  |         // constants--in this case, AlarmManager.INTERVAL_DAY. | ||||||
|  |         alarmMgr.setInexactRepeating(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(), | ||||||
|  |         AlarmManager.INTERVAL_DAY, alarmIntent); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public int onStartCommand(Intent intent, int flags, int startId) { | ||||||
|  |         if (running.compareAndSet(false, true)) { | ||||||
|  |             new Thread(() -> { | ||||||
|  |                 export(); | ||||||
|  |                 stopForeground(true); | ||||||
|  |                 WakeLockHelper.release(wakeLock); | ||||||
|  |                 running.set(false); | ||||||
|  |                 stopSelf(); | ||||||
|  |             }).start(); | ||||||
|  |         } | ||||||
|  |         return START_NOT_STICKY; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void export() { | ||||||
|  |         try{ | ||||||
|  |             final SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(this); | ||||||
|  |             String encryptionKey = prefs.getString("backup_password", null); | ||||||
|  |             if (encryptionKey == null || encryptionKey.length() < 3) { | ||||||
|  |                 Log.d(Config.LOGTAG, "BackupService: failed to write encryted backup to sdcard because of missing password"); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             wakeLock.acquire(); | ||||||
|  | 
 | ||||||
|  |             NotificationManager mNotifyManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE); | ||||||
|  |             NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(getBaseContext(), "backup"); | ||||||
|  |             mBuilder.setContentTitle(getString(R.string.notification_backup_create)) | ||||||
|  |                     .setSmallIcon(R.drawable.ic_import_export_white_24dp); | ||||||
|  |             startForeground(NOTIFICATION_ID, mBuilder.build()); | ||||||
|  | 
 | ||||||
|  |             Log.d(Config.LOGTAG, "BackupService: start creating backup"); | ||||||
|  |             // Get hold of the db: | ||||||
|  |             FileInputStream inputFile = new FileInputStream(this.getDatabasePath(DatabaseBackend.DATABASE_NAME)); | ||||||
|  |             // Set the output folder on the SDcard | ||||||
|  |             File directory = new File(FileBackend.getBackupDirectory()); | ||||||
|  |             // Create the folder if it doesn't exist: | ||||||
|  |             if (!directory.exists()) { | ||||||
|  |                 boolean directory_created = directory.mkdirs(); | ||||||
|  |                 Log.d(Config.LOGTAG, "BackupService: backup directory created " + directory_created); | ||||||
|  |             } | ||||||
|  |             //Delete old database export file | ||||||
|  |             File tempDBFile = new File(directory + "/database.bak"); | ||||||
|  |             if (tempDBFile.exists()) { | ||||||
|  |                 Log.d(Config.LOGTAG, "BackupService: Delete temp database backup file from " + tempDBFile.toString()); | ||||||
|  |                 boolean temp_db_file_deleted = tempDBFile.delete(); | ||||||
|  |                 Log.d(Config.LOGTAG, "BackupService: old backup file deleted " + temp_db_file_deleted); | ||||||
|  |             } | ||||||
|  |             // Set the output file stream up: | ||||||
|  |             FileOutputStream outputFile = new FileOutputStream(directory.getPath() + "/database.db.crypt"); | ||||||
|  | 
 | ||||||
|  |             // encrypt database from the input file to the output file | ||||||
|  |             try { | ||||||
|  |                 EncryptDecryptFile.encrypt(inputFile, outputFile, encryptionKey); | ||||||
|  |                 Log.d(Config.LOGTAG, "BackupService: starting encrypted output to " + outputFile.toString()); | ||||||
|  |             } catch (NoSuchAlgorithmException | NoSuchPaddingException e) { | ||||||
|  |                 Log.d(Config.LOGTAG, "BackupService: Database exporter: encryption failed with " + e); | ||||||
|  |                 e.printStackTrace(); | ||||||
|  |             } catch (InvalidKeyException e) { | ||||||
|  |                 Log.d(Config.LOGTAG, "BackupService: Database exporter: encryption failed (invalid key) with " + e); | ||||||
|  |                 e.printStackTrace(); | ||||||
|  |             } catch (IOException e) { | ||||||
|  |                 Log.d(Config.LOGTAG, "BackupService: Database exporter: encryption failed (IO) with " + e); | ||||||
|  |                 e.printStackTrace(); | ||||||
|  |             } finally { | ||||||
|  |                 Log.d(Config.LOGTAG, "BackupService: backup job finished"); | ||||||
|  |             } | ||||||
|  |             mNotifyManager.notify(NOTIFICATION_ID, mBuilder.build()); | ||||||
|  |         } catch (IOException e) { | ||||||
|  |             e.printStackTrace(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @Override | ||||||
|  |     public IBinder onBind(Intent intent) { | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -139,6 +139,13 @@ public class NotificationService { | ||||||
|         exportChannel.setGroup("status"); |         exportChannel.setGroup("status"); | ||||||
|         notificationManager.createNotificationChannel(exportChannel); |         notificationManager.createNotificationChannel(exportChannel); | ||||||
| 
 | 
 | ||||||
|  |         final NotificationChannel backupChannel = new NotificationChannel("backup", | ||||||
|  |                 c.getString(R.string.backup_channel_name), | ||||||
|  |                 NotificationManager.IMPORTANCE_LOW); | ||||||
|  |         backupChannel.setShowBadge(false); | ||||||
|  |         backupChannel.setGroup("status"); | ||||||
|  |         notificationManager.createNotificationChannel(backupChannel); | ||||||
|  | 
 | ||||||
|         final NotificationChannel messagesChannel = new NotificationChannel("messages", |         final NotificationChannel messagesChannel = new NotificationChannel("messages", | ||||||
|                 c.getString(R.string.messages_channel_name), |                 c.getString(R.string.messages_channel_name), | ||||||
|                 NotificationManager.IMPORTANCE_HIGH); |                 NotificationManager.IMPORTANCE_HIGH); | ||||||
|  |  | ||||||
|  | @ -21,8 +21,9 @@ import android.preference.PreferenceManager; | ||||||
| import android.preference.PreferenceScreen; | import android.preference.PreferenceScreen; | ||||||
| import android.provider.MediaStore; | import android.provider.MediaStore; | ||||||
| import android.util.Log; | import android.util.Log; | ||||||
|  | import android.view.LayoutInflater; | ||||||
| import android.view.View; | import android.view.View; | ||||||
| import android.view.ViewGroup; | import android.widget.EditText; | ||||||
| import android.widget.Toast; | import android.widget.Toast; | ||||||
| 
 | 
 | ||||||
| import java.io.File; | import java.io.File; | ||||||
|  | @ -36,6 +37,7 @@ import eu.siacs.conversations.Config; | ||||||
| import eu.siacs.conversations.R; | import eu.siacs.conversations.R; | ||||||
| import eu.siacs.conversations.crypto.OmemoSetting; | import eu.siacs.conversations.crypto.OmemoSetting; | ||||||
| import eu.siacs.conversations.entities.Account; | import eu.siacs.conversations.entities.Account; | ||||||
|  | import eu.siacs.conversations.services.BackupService; | ||||||
| import eu.siacs.conversations.services.ExportLogsService; | import eu.siacs.conversations.services.ExportLogsService; | ||||||
| import eu.siacs.conversations.services.MemorizingTrustManager; | import eu.siacs.conversations.services.MemorizingTrustManager; | ||||||
| import eu.siacs.conversations.services.QuickConversationsService; | import eu.siacs.conversations.services.QuickConversationsService; | ||||||
|  | @ -60,6 +62,7 @@ public class SettingsActivity extends XmppActivity implements | ||||||
| 	public static final String OMEMO_SETTING = "omemo"; | 	public static final String OMEMO_SETTING = "omemo"; | ||||||
| 
 | 
 | ||||||
| 	public static final int REQUEST_WRITE_LOGS = 0xbf8701; | 	public static final int REQUEST_WRITE_LOGS = 0xbf8701; | ||||||
|  | 	public static final int REQUEST_WRITE_BACKUP = 0xbf8702; | ||||||
| 	private SettingsFragment mSettingsFragment; | 	private SettingsFragment mSettingsFragment; | ||||||
| 
 | 
 | ||||||
| 	@Override | 	@Override | ||||||
|  | @ -219,6 +222,19 @@ public class SettingsActivity extends XmppActivity implements | ||||||
| 			}); | 			}); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | 		final Preference backupPassword = mSettingsFragment.findPreference("backup_password"); | ||||||
|  | 		backupPassword.setOnPreferenceClickListener(preference -> { | ||||||
|  | 			if (preference.isEnabled()) { | ||||||
|  | 				enterPasswordDialog(); | ||||||
|  | 			} else { | ||||||
|  | 				return getPreferences().edit().putString("backup_password", null).commit(); | ||||||
|  | 			} | ||||||
|  | 			if (hasStoragePermission(REQUEST_WRITE_BACKUP)) { | ||||||
|  | 				startBackup(); | ||||||
|  | 			} | ||||||
|  | 			return true; | ||||||
|  | 		}); | ||||||
|  | 
 | ||||||
| 		final Preference exportLogsPreference = mSettingsFragment.findPreference("export_logs"); | 		final Preference exportLogsPreference = mSettingsFragment.findPreference("export_logs"); | ||||||
| 		if (exportLogsPreference != null) { | 		if (exportLogsPreference != null) { | ||||||
| 			exportLogsPreference.setOnPreferenceClickListener(preference -> { | 			exportLogsPreference.setOnPreferenceClickListener(preference -> { | ||||||
|  | @ -350,6 +366,56 @@ public class SettingsActivity extends XmppActivity implements | ||||||
| 		return true; | 		return true; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	public void enterPasswordDialog() { | ||||||
|  | 		LayoutInflater li = LayoutInflater.from(this); | ||||||
|  | 		View promptsView = li.inflate(R.layout.backup_password, null); | ||||||
|  | 
 | ||||||
|  | 		final Preference preference = mSettingsFragment.findPreference("enable_multi_accounts"); | ||||||
|  | 
 | ||||||
|  | 		final AlertDialog.Builder alertDialogBuilder = new AlertDialog.Builder(this); | ||||||
|  | 		alertDialogBuilder.setView(promptsView); | ||||||
|  | 		final EditText password = promptsView.findViewById(R.id.password); | ||||||
|  | 		final EditText confirm_password = promptsView.findViewById(R.id.confirm_password); | ||||||
|  | 		confirm_password.setVisibility(View.VISIBLE); | ||||||
|  | 		alertDialogBuilder.setTitle(R.string.enter_password); | ||||||
|  | 		alertDialogBuilder.setMessage(R.string.enter_password); | ||||||
|  | 		alertDialogBuilder | ||||||
|  | 				.setCancelable(false) | ||||||
|  | 				.setPositiveButton(R.string.ok, | ||||||
|  | 						(dialog, id) -> { | ||||||
|  | 							final String pw1 = password.getText().toString(); | ||||||
|  | 							final String pw2 = confirm_password.getText().toString(); | ||||||
|  | 							if (!pw1.equals(pw2)) { | ||||||
|  | 								((CheckBoxPreference) preference).setChecked(false); | ||||||
|  | 								AlertDialog.Builder builder = new AlertDialog.Builder(this); | ||||||
|  | 								builder.setTitle(R.string.error); | ||||||
|  | 								builder.setMessage(R.string.passwords_do_not_match); | ||||||
|  | 								builder.setNegativeButton(R.string.cancel, null); | ||||||
|  | 								builder.setPositiveButton(R.string.try_again, (dialog12, id12) -> enterPasswordDialog()); | ||||||
|  | 								builder.create().show(); | ||||||
|  | 							} else if (pw1.trim().isEmpty()) { | ||||||
|  | 								((CheckBoxPreference) preference).setChecked(false); | ||||||
|  | 								AlertDialog.Builder builder = new AlertDialog.Builder(this); | ||||||
|  | 								builder.setTitle(R.string.error); | ||||||
|  | 								builder.setMessage(R.string.password_should_not_be_empty); | ||||||
|  | 								builder.setNegativeButton(R.string.cancel, null); | ||||||
|  | 								builder.setPositiveButton(R.string.try_again, (dialog1, id1) -> enterPasswordDialog()); | ||||||
|  | 								builder.create().show(); | ||||||
|  | 							} else { | ||||||
|  | 								((CheckBoxPreference) preference).setChecked(true); | ||||||
|  | 								boolean passwordstored = getPreferences().edit().putString("backup_password", pw1).commit();; | ||||||
|  | 								Log.d(Config.LOGTAG, "saving multiaccount password " + passwordstored); | ||||||
|  | 								if (passwordstored) { | ||||||
|  | 									recreate(); | ||||||
|  | 								} else { | ||||||
|  | 									//handleMultiAccountChanges(); | ||||||
|  | 								} | ||||||
|  | 							} | ||||||
|  | 						}) | ||||||
|  | 				.setNegativeButton(R.string.cancel, null); | ||||||
|  | 		alertDialogBuilder.create().show(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	@Override | 	@Override | ||||||
| 	public void onStop() { | 	public void onStop() { | ||||||
| 		super.onStop(); | 		super.onStop(); | ||||||
|  | @ -399,6 +465,9 @@ public class SettingsActivity extends XmppActivity implements | ||||||
| 	public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { | 	public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { | ||||||
| 		if (grantResults.length > 0) | 		if (grantResults.length > 0) | ||||||
| 			if (grantResults[0] == PackageManager.PERMISSION_GRANTED) { | 			if (grantResults[0] == PackageManager.PERMISSION_GRANTED) { | ||||||
|  | 				if (requestCode == REQUEST_WRITE_BACKUP) { | ||||||
|  | 					startBackup(); | ||||||
|  | 				} | ||||||
| 				if (requestCode == REQUEST_WRITE_LOGS) { | 				if (requestCode == REQUEST_WRITE_LOGS) { | ||||||
| 					startExport(); | 					startExport(); | ||||||
| 				} | 				} | ||||||
|  | @ -407,6 +476,10 @@ public class SettingsActivity extends XmppActivity implements | ||||||
| 			} | 			} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	private void startBackup() { | ||||||
|  | 		ContextCompat.startForegroundService(this, new Intent(this, BackupService.class)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	private void startExport() { | 	private void startExport() { | ||||||
| 		ContextCompat.startForegroundService(this, new Intent(this, ExportLogsService.class)); | 		ContextCompat.startForegroundService(this, new Intent(this, ExportLogsService.class)); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | @ -0,0 +1,73 @@ | ||||||
|  | package eu.siacs.conversations.utils; | ||||||
|  | 
 | ||||||
|  | import android.util.Log; | ||||||
|  | 
 | ||||||
|  | import java.io.FileInputStream; | ||||||
|  | import java.io.FileOutputStream; | ||||||
|  | import java.io.IOException; | ||||||
|  | import java.security.InvalidKeyException; | ||||||
|  | import java.security.MessageDigest; | ||||||
|  | import java.security.NoSuchAlgorithmException; | ||||||
|  | import java.util.Arrays; | ||||||
|  | 
 | ||||||
|  | import javax.crypto.Cipher; | ||||||
|  | import javax.crypto.CipherInputStream; | ||||||
|  | import javax.crypto.CipherOutputStream; | ||||||
|  | import javax.crypto.NoSuchPaddingException; | ||||||
|  | import javax.crypto.spec.SecretKeySpec; | ||||||
|  | 
 | ||||||
|  | import eu.siacs.conversations.Config; | ||||||
|  | 
 | ||||||
|  | public class EncryptDecryptFile { | ||||||
|  |     private static String cipher_mode = "AES/ECB/PKCS5Padding"; | ||||||
|  | 
 | ||||||
|  |     public static void encrypt(FileInputStream iFile, FileOutputStream oFile, String iKey) throws IOException, NoSuchAlgorithmException, NoSuchPaddingException, InvalidKeyException { | ||||||
|  |         byte[] key = iKey.getBytes("UTF-8"); | ||||||
|  |         Log.d(Config.LOGTAG, "Cipher key: " + Arrays.toString(key)); | ||||||
|  |         MessageDigest sha = MessageDigest.getInstance("SHA-1"); | ||||||
|  |         Log.d(Config.LOGTAG, "Cipher sha: " + sha.toString()); | ||||||
|  |         key = sha.digest(key); | ||||||
|  |         Log.d(Config.LOGTAG, "Cipher sha key: " + Arrays.toString(key)); | ||||||
|  |         key = Arrays.copyOf(key, 16); // use only first 128 bit | ||||||
|  |         Log.d(Config.LOGTAG, "Cipher sha key 16 bytes: " + Arrays.toString(key)); | ||||||
|  |         SecretKeySpec sks = new SecretKeySpec(key, "AES"); | ||||||
|  |         Cipher cipher = Cipher.getInstance(cipher_mode); | ||||||
|  |         cipher.init(Cipher.ENCRYPT_MODE, sks); | ||||||
|  |         Log.d(Config.LOGTAG, "Cipher IV: " + Arrays.toString(cipher.getIV())); | ||||||
|  |         CipherOutputStream cos = new CipherOutputStream(oFile, cipher); | ||||||
|  |         Log.d(Config.LOGTAG, "Encryption with: " + cos.toString()); | ||||||
|  |         int b; | ||||||
|  |         byte[] d = new byte[8]; | ||||||
|  |         while ((b = iFile.read(d)) != -1) { | ||||||
|  |             cos.write(d, 0, b); | ||||||
|  |         } | ||||||
|  |         cos.flush(); | ||||||
|  |         cos.close(); | ||||||
|  |         iFile.close(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static void decrypt(FileInputStream iFile, FileOutputStream oFile, String iKey) throws IOException, NoSuchAlgorithmException, NoSuchPaddingException, InvalidKeyException { | ||||||
|  |         byte[] key = iKey.getBytes("UTF-8"); | ||||||
|  |         Log.d(Config.LOGTAG, "Cipher key: " + Arrays.toString(key)); | ||||||
|  |         MessageDigest sha = MessageDigest.getInstance("SHA-1"); | ||||||
|  |         Log.d(Config.LOGTAG, "Cipher sha: " + sha.toString()); | ||||||
|  |         key = sha.digest(key); | ||||||
|  |         Log.d(Config.LOGTAG, "Cipher sha key: " + Arrays.toString(key)); | ||||||
|  |         key = Arrays.copyOf(key, 16); // use only first 128 bit | ||||||
|  |         Log.d(Config.LOGTAG, "Cipher sha key 16 bytes: " + Arrays.toString(key)); | ||||||
|  |         SecretKeySpec sks = new SecretKeySpec(key, "AES"); | ||||||
|  |         Cipher cipher = Cipher.getInstance(cipher_mode); | ||||||
|  |         cipher.init(Cipher.DECRYPT_MODE, sks); | ||||||
|  |         Log.d(Config.LOGTAG, "Cipher IV: " + Arrays.toString(cipher.getIV())); | ||||||
|  |         CipherInputStream cis = new CipherInputStream(iFile, cipher); | ||||||
|  |         Log.d(Config.LOGTAG, "Decryption with: " + cis.toString()); | ||||||
|  |         int b; | ||||||
|  |         byte[] d = new byte[8]; | ||||||
|  |         while ((b = cis.read(d)) != -1) { | ||||||
|  |             oFile.write(d, 0, b); | ||||||
|  |         } | ||||||
|  |         oFile.flush(); | ||||||
|  |         oFile.close(); | ||||||
|  |         cis.close(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,21 @@ | ||||||
|  | <?xml version="1.0" encoding="utf-8"?> | ||||||
|  | <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" | ||||||
|  |     android:orientation="vertical" | ||||||
|  |     android:layout_width="match_parent" | ||||||
|  |     android:layout_height="match_parent"> | ||||||
|  | 
 | ||||||
|  |     <EditText | ||||||
|  |         android:id="@+id/password" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="wrap_content" | ||||||
|  |         android:hint="@string/password" | ||||||
|  |         android:inputType="textPassword" /> | ||||||
|  | 
 | ||||||
|  |     <EditText | ||||||
|  |         android:id="@+id/confirm_password" | ||||||
|  |         android:layout_width="match_parent" | ||||||
|  |         android:layout_height="wrap_content" | ||||||
|  |         android:hint="@string/confirm_password" | ||||||
|  |         android:inputType="textPassword" | ||||||
|  |         android:visibility="gone"/> | ||||||
|  | </LinearLayout> | ||||||
|  | @ -172,6 +172,9 @@ | ||||||
|   <string name="block_jabber_id">Sperre Jabber-ID</string> |   <string name="block_jabber_id">Sperre Jabber-ID</string> | ||||||
|   <string name="account_settings_example_jabber_id">benutzer@domain.de</string> |   <string name="account_settings_example_jabber_id">benutzer@domain.de</string> | ||||||
|   <string name="password">Passwort</string> |   <string name="password">Passwort</string> | ||||||
|  |   <string name="password_wrong">Falsches Passwort, erneut versuchen</string> | ||||||
|  |   <string name="confirm_password">Passwort bestätigen</string> | ||||||
|  |   <string name="passwords_do_not_match">Passwörter stimmen nicht überein</string> | ||||||
|   <string name="invalid_jid">Ungültige Jabber-ID</string> |   <string name="invalid_jid">Ungültige Jabber-ID</string> | ||||||
|   <string name="error_out_of_memory">Zu wenig Speicher vorhanden. Das Bild ist zu groß</string> |   <string name="error_out_of_memory">Zu wenig Speicher vorhanden. Das Bild ist zu groß</string> | ||||||
|   <string name="add_phone_book_text">%s zum Telefonbuch hinzufügen</string> |   <string name="add_phone_book_text">%s zum Telefonbuch hinzufügen</string> | ||||||
|  | @ -531,6 +534,12 @@ | ||||||
|   <string name="dialog_title_create_conference">Gruppenchat erstellen</string> |   <string name="dialog_title_create_conference">Gruppenchat erstellen</string> | ||||||
|   <string name="choose_participants">Mitglieder wählen</string> |   <string name="choose_participants">Mitglieder wählen</string> | ||||||
|   <string name="creating_conference">Erstelle Gruppenchat…</string> |   <string name="creating_conference">Erstelle Gruppenchat…</string> | ||||||
|  |   <string name="import_text">Es wurde ein Backup gefunden, welches importiert werden kann.\nDein Messenger startet während des Importvorgangs neu. Soll das Backup importiert werden?</string> | ||||||
|  |   <string name="import_database">Backup importieren</string> | ||||||
|  |   <string name="import_started">Backup wird importiert, dies wird eine Weile dauern.</string> | ||||||
|  |   <string name="import_canceled">Import abgebrochen</string> | ||||||
|  |   <string name="import_failed">Backup import ist fehlgeschlagen und nicht möglich.</string> | ||||||
|  |   <string name="enter_backup_password">Bitte gib das Passwort ein, um das Backup zu importieren.</string> | ||||||
|   <string name="invite_again">Erneut einladen</string> |   <string name="invite_again">Erneut einladen</string> | ||||||
|   <string name="gp_disable">Deaktivieren</string> |   <string name="gp_disable">Deaktivieren</string> | ||||||
|   <string name="gp_short">kurz</string> |   <string name="gp_short">kurz</string> | ||||||
|  |  | ||||||
|  | @ -172,6 +172,9 @@ | ||||||
|   <string name="block_jabber_id">Bloquear Identificador Jabber</string> |   <string name="block_jabber_id">Bloquear Identificador Jabber</string> | ||||||
|   <string name="account_settings_example_jabber_id">usuario@ejemplo.com</string> |   <string name="account_settings_example_jabber_id">usuario@ejemplo.com</string> | ||||||
|   <string name="password">Contraseña</string> |   <string name="password">Contraseña</string> | ||||||
|  |   <string name="password_wrong">Contraseña incorrecta, inténtalo de nuevo</string> | ||||||
|  |   <string name="confirm_password">Confirmar contraseña</string> | ||||||
|  |   <string name="passwords_do_not_match">Las contraseñas no coinciden</string> | ||||||
|   <string name="invalid_jid">El identificador no es un identificador Jabber válido</string> |   <string name="invalid_jid">El identificador no es un identificador Jabber válido</string> | ||||||
|   <string name="error_out_of_memory">Sin memoria. La imagen es demasiado grande</string> |   <string name="error_out_of_memory">Sin memoria. La imagen es demasiado grande</string> | ||||||
|   <string name="add_phone_book_text">¿Quieres añadir a %s a tus contactos?</string> |   <string name="add_phone_book_text">¿Quieres añadir a %s a tus contactos?</string> | ||||||
|  | @ -531,6 +534,11 @@ | ||||||
|   <string name="dialog_title_create_conference">Crear Conversación en Grupo</string> |   <string name="dialog_title_create_conference">Crear Conversación en Grupo</string> | ||||||
|   <string name="choose_participants">Elige a los participantes</string> |   <string name="choose_participants">Elige a los participantes</string> | ||||||
|   <string name="creating_conference">Creando conversación en grupo...</string> |   <string name="creating_conference">Creando conversación en grupo...</string> | ||||||
|  |   <string name="import_text">Hay una copia de seguridad en tu dispositivo que se puede importar.\nTu aplicación se reiniciará durante el proceso. ¿Quieres importar la copia de seguridad?</string> | ||||||
|  |   <string name="import_database">Importar copia de seguridad</string> | ||||||
|  |   <string name="import_started">Se importará la copia de seguridad, esto puede llevar un tiempo.</string> | ||||||
|  |   <string name="import_canceled">Importación cancelada</string> | ||||||
|  |   <string name="import_failed">La importación de la base de datos falló, una importación no es posible</string> | ||||||
|   <string name="invite_again">Invitar de nuevo</string> |   <string name="invite_again">Invitar de nuevo</string> | ||||||
|   <string name="gp_disable">Deshabilitar</string> |   <string name="gp_disable">Deshabilitar</string> | ||||||
|   <string name="gp_short">Corto</string> |   <string name="gp_short">Corto</string> | ||||||
|  |  | ||||||
|  | @ -172,6 +172,7 @@ | ||||||
|   <string name="block_jabber_id">Bloquer l\'ID Jabber</string> |   <string name="block_jabber_id">Bloquer l\'ID Jabber</string> | ||||||
|   <string name="account_settings_example_jabber_id">nom@exemple.com</string> |   <string name="account_settings_example_jabber_id">nom@exemple.com</string> | ||||||
|   <string name="password">Mot de passe</string> |   <string name="password">Mot de passe</string> | ||||||
|  |   <string name="passwords_do_not_match">Les deux mots de passe ne correspondent pas.</string> | ||||||
|   <string name="invalid_jid">Cet identifiant n\'est pas valide</string> |   <string name="invalid_jid">Cet identifiant n\'est pas valide</string> | ||||||
|   <string name="error_out_of_memory">Plus de mémoire disponible. L\'image est trop volumineuse.</string> |   <string name="error_out_of_memory">Plus de mémoire disponible. L\'image est trop volumineuse.</string> | ||||||
|   <string name="add_phone_book_text">Voulez-vous ajouter %s à votre carnet d\'adresses ?</string> |   <string name="add_phone_book_text">Voulez-vous ajouter %s à votre carnet d\'adresses ?</string> | ||||||
|  |  | ||||||
|  | @ -139,6 +139,9 @@ | ||||||
|   <string name="account_settings_jabber_id">Jabber ID</string> |   <string name="account_settings_jabber_id">Jabber ID</string> | ||||||
|   <string name="account_settings_example_jabber_id">username@example.com</string> |   <string name="account_settings_example_jabber_id">username@example.com</string> | ||||||
|   <string name="password">Password</string> |   <string name="password">Password</string> | ||||||
|  |   <string name="password_wrong">Kata sandi salah, coba lagi</string> | ||||||
|  |   <string name="confirm_password">Konfirmasi sandi</string> | ||||||
|  |   <string name="passwords_do_not_match">Kata sandi tidak cocok</string> | ||||||
|   <string name="invalid_jid">Jabber ID tidak valid</string> |   <string name="invalid_jid">Jabber ID tidak valid</string> | ||||||
|   <string name="error_out_of_memory">Memori habis. Gambar terlalu besar</string> |   <string name="error_out_of_memory">Memori habis. Gambar terlalu besar</string> | ||||||
|   <string name="server_info_show_more">Info Server</string> |   <string name="server_info_show_more">Info Server</string> | ||||||
|  | @ -357,6 +360,10 @@ | ||||||
|   <string name="secure_password_generated">Password aman berhasil dibuat</string> |   <string name="secure_password_generated">Password aman berhasil dibuat</string> | ||||||
|   <string name="registration_please_wait">Registrasi gagal: Coba lagi nanti</string> |   <string name="registration_please_wait">Registrasi gagal: Coba lagi nanti</string> | ||||||
|   <string name="choose_participants">Undang user</string> |   <string name="choose_participants">Undang user</string> | ||||||
|  |   <string name="import_database">Impor cadangan</string> | ||||||
|  |   <string name="import_started">Cadangan akan diimpor, ini mungkin memakan waktu sebentar.</string> | ||||||
|  |   <string name="import_canceled">Impor dibatalkan</string> | ||||||
|  |   <string name="import_failed">Gagal memasukkan penyimpanan data, masukan tidak mungkin dilakukan</string> | ||||||
|   <string name="invite_again">Undang lagi</string> |   <string name="invite_again">Undang lagi</string> | ||||||
|   <string name="gp_short">Pendek</string> |   <string name="gp_short">Pendek</string> | ||||||
|   <string name="gp_medium">Sedang</string> |   <string name="gp_medium">Sedang</string> | ||||||
|  |  | ||||||
|  | @ -171,6 +171,7 @@ | ||||||
|   <string name="block_jabber_id">Заблокировать Jabber ID</string> |   <string name="block_jabber_id">Заблокировать Jabber ID</string> | ||||||
|   <string name="account_settings_example_jabber_id">username@example.com</string> |   <string name="account_settings_example_jabber_id">username@example.com</string> | ||||||
|   <string name="password">Пароль</string> |   <string name="password">Пароль</string> | ||||||
|  |   <string name="passwords_do_not_match">Пароли не совпадают</string> | ||||||
|   <string name="invalid_jid">Недопустимый Jabber ID</string> |   <string name="invalid_jid">Недопустимый Jabber ID</string> | ||||||
|   <string name="error_out_of_memory">Недостаточно памяти. Изображение слишком большое</string> |   <string name="error_out_of_memory">Недостаточно памяти. Изображение слишком большое</string> | ||||||
|   <string name="add_phone_book_text">Вы хотите добавить %s в вашу адресную книгу?</string> |   <string name="add_phone_book_text">Вы хотите добавить %s в вашу адресную книгу?</string> | ||||||
|  |  | ||||||
|  | @ -175,6 +175,9 @@ | ||||||
|     <string name="block_jabber_id">Block Jabber ID</string> |     <string name="block_jabber_id">Block Jabber ID</string> | ||||||
|     <string name="account_settings_example_jabber_id">username@example.com</string> |     <string name="account_settings_example_jabber_id">username@example.com</string> | ||||||
|     <string name="password">Password</string> |     <string name="password">Password</string> | ||||||
|  |     <string name="password_wrong">Wrong password, try again</string> | ||||||
|  |     <string name="confirm_password">Confirm password</string> | ||||||
|  |     <string name="passwords_do_not_match">Passwords do not match</string> | ||||||
|     <string name="invalid_jid">This is not a valid Jabber ID</string> |     <string name="invalid_jid">This is not a valid Jabber ID</string> | ||||||
|     <string name="error_out_of_memory">Out of memory. Image is too large</string> |     <string name="error_out_of_memory">Out of memory. Image is too large</string> | ||||||
|     <string name="add_phone_book_text">Do you want to add %s to your address book?</string> |     <string name="add_phone_book_text">Do you want to add %s to your address book?</string> | ||||||
|  | @ -538,6 +541,14 @@ | ||||||
|     <string name="dialog_title_create_conference">Create Group Chat</string> |     <string name="dialog_title_create_conference">Create Group Chat</string> | ||||||
|     <string name="choose_participants">Choose participants</string> |     <string name="choose_participants">Choose participants</string> | ||||||
|     <string name="creating_conference">Creating group chat…</string> |     <string name="creating_conference">Creating group chat…</string> | ||||||
|  |     <string name="backup_channel_name">Connectivity Problems</string> | ||||||
|  |     <string name="notification_backup_create">Creating backup</string> | ||||||
|  |     <string name="import_text">There is a backup on your device which can be imported.\nYour Messenger will be restarted during backup process. Shall the backup be imported?</string> | ||||||
|  |     <string name="import_database">Import backup</string> | ||||||
|  |     <string name="import_started">Backup will be imported, this may take awhile.</string> | ||||||
|  |     <string name="import_canceled">Import canceled</string> | ||||||
|  |     <string name="import_failed">Database import failed, an import is not possible</string> | ||||||
|  |     <string name="enter_backup_password">Please enter your password to import your backup.</string> | ||||||
|     <string name="invite_again">Invite again</string> |     <string name="invite_again">Invite again</string> | ||||||
|     <string name="gp_disable">Disable</string> |     <string name="gp_disable">Disable</string> | ||||||
|     <string name="gp_short">Short</string> |     <string name="gp_short">Short</string> | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue