File export / import
This commit is contained in:
parent
0b5fa43f47
commit
4dee17b7de
7 changed files with 315 additions and 5 deletions
|
@ -1,7 +1,11 @@
|
||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
<manifest
|
||||||
|
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:tools="http://schemas.android.com/tools">
|
xmlns:tools="http://schemas.android.com/tools">
|
||||||
|
|
||||||
|
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
|
||||||
|
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
|
||||||
|
|
||||||
<application
|
<application
|
||||||
android:allowBackup="true"
|
android:allowBackup="true"
|
||||||
android:dataExtractionRules="@xml/data_extraction_rules"
|
android:dataExtractionRules="@xml/data_extraction_rules"
|
||||||
|
|
|
@ -34,7 +34,7 @@ public class DBWrapper extends Wrapper {
|
||||||
public static final boolean DEBUG = false;
|
public static final boolean DEBUG = false;
|
||||||
|
|
||||||
public DBWrapper(Context context) {
|
public DBWrapper(Context context) {
|
||||||
this.db = openDB(context);
|
this.db = openDB(context, DB_PATH);
|
||||||
this.queueMaker = new QueueMakerImpl(this.db);
|
this.queueMaker = new QueueMakerImpl(this.db);
|
||||||
this.eventMaker = new EventMakerImpl(this.db);
|
this.eventMaker = new EventMakerImpl(this.db);
|
||||||
this.tagMaker = new TagMakerImpl(this.db);
|
this.tagMaker = new TagMakerImpl(this.db);
|
||||||
|
@ -127,19 +127,33 @@ public class DBWrapper extends Wrapper {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get internal database path
|
||||||
|
*/
|
||||||
|
static File getDBPath(Context context) {
|
||||||
|
return new File(context.getFilesDir(), DB_PATH);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return opened and initialized database
|
* @return opened and initialized database
|
||||||
*/
|
*/
|
||||||
private SQLiteDatabase openDB(Context context) {
|
private static SQLiteDatabase openDB(Context context, String dbPath) {
|
||||||
if (DBWrapper.DEBUG) {
|
if (DBWrapper.DEBUG) {
|
||||||
try {
|
try {
|
||||||
new File(context.getFilesDir(), DB_PATH).delete();
|
new File(context.getFilesDir(), dbPath).delete();
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
Log.wtf(TAG, e);
|
Log.wtf(TAG, e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
SQLiteDatabase db = SQLiteDatabase.openDatabase(new File(context.getFilesDir(), DB_PATH).getPath(), null, SQLiteDatabase.OPEN_READWRITE | SQLiteDatabase.CREATE_IF_NECESSARY);
|
File path = getDBPath(context);
|
||||||
|
SQLiteDatabase db;
|
||||||
|
if (!path.exists()) {
|
||||||
|
db = SQLiteDatabase.openDatabase(path.getPath(), null, SQLiteDatabase.OPEN_READWRITE | SQLiteDatabase.CREATE_IF_NECESSARY);
|
||||||
|
} else {
|
||||||
|
db = SQLiteDatabase.openDatabase(path.getPath(), null, SQLiteDatabase.OPEN_READWRITE);
|
||||||
|
}
|
||||||
|
|
||||||
initDB(db);
|
initDB(db);
|
||||||
return db;
|
return db;
|
||||||
}
|
}
|
||||||
|
|
121
Yeeemp/app/src/main/java/art/pegasko/yeeemp/impl/DataUtils.java
Normal file
121
Yeeemp/app/src/main/java/art/pegasko/yeeemp/impl/DataUtils.java
Normal file
|
@ -0,0 +1,121 @@
|
||||||
|
/**
|
||||||
|
* Copyright 2024 pegasko
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package art.pegasko.yeeemp.impl;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.database.sqlite.SQLiteDatabase;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.os.FileUtils;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.FileOutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.text.SimpleDateFormat;
|
||||||
|
|
||||||
|
import art.pegasko.yeeemp.ui.activity.Utils;
|
||||||
|
|
||||||
|
public class DataUtils {
|
||||||
|
public static final String DATE_FORMAT = "yyyy-MM-dd-HH-mm-ss";
|
||||||
|
|
||||||
|
public static String formatTs(long timestamp) {
|
||||||
|
SimpleDateFormat sdf = new SimpleDateFormat(DataUtils.DATE_FORMAT);
|
||||||
|
return sdf.format(timestamp);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void copyStream(InputStream fis, OutputStream fos) throws IOException {
|
||||||
|
byte[] buffer = new byte[1024];
|
||||||
|
int lengthRead;
|
||||||
|
while ((lengthRead = fis.read(buffer)) > 0) {
|
||||||
|
fos.write(buffer, 0, lengthRead);
|
||||||
|
fos.flush();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* just copy internal db to external storage */
|
||||||
|
public static void exportDatabase(Context context, File directory) throws Exception {
|
||||||
|
directory.mkdirs();
|
||||||
|
|
||||||
|
File internalFile = DBWrapper.getDBPath(context);
|
||||||
|
File externalFile = new File(directory, "export_" + formatTs(System.currentTimeMillis()) + ".db");
|
||||||
|
FileInputStream fis = new FileInputStream(internalFile);
|
||||||
|
FileOutputStream fos = new FileOutputStream(externalFile);
|
||||||
|
|
||||||
|
copyStream(fis, fos);
|
||||||
|
|
||||||
|
fis.close();
|
||||||
|
fos.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* just copy internal db to external storage */
|
||||||
|
public static void exportDatabase(Context context, Uri uri) throws Exception {
|
||||||
|
File internalFile = DBWrapper.getDBPath(context);
|
||||||
|
InputStream fis = new FileInputStream(internalFile);
|
||||||
|
OutputStream fos = context.getContentResolver().openOutputStream(uri);
|
||||||
|
|
||||||
|
copyStream(fis, fos);
|
||||||
|
|
||||||
|
fis.close();
|
||||||
|
fos.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* just copy external db to internal storage */
|
||||||
|
public static void importDatabase(Context context, Uri uri) throws Exception {
|
||||||
|
File internalFile = DBWrapper.getDBPath(context);
|
||||||
|
InputStream fis = context.getContentResolver().openInputStream(uri);
|
||||||
|
OutputStream fos = new FileOutputStream(internalFile);
|
||||||
|
|
||||||
|
copyStream(fis, fos);
|
||||||
|
|
||||||
|
fis.close();
|
||||||
|
fos.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void backupDatabase(Context context) throws IOException {
|
||||||
|
File internalFile = DBWrapper.getDBPath(context);
|
||||||
|
File backupFile = new File(context.getFilesDir(), "backup.db");
|
||||||
|
|
||||||
|
FileInputStream fis = new FileInputStream(internalFile);
|
||||||
|
FileOutputStream fos = new FileOutputStream(backupFile);
|
||||||
|
|
||||||
|
copyStream(fis, fos);
|
||||||
|
|
||||||
|
fis.close();
|
||||||
|
fos.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void restoreDatabase(Context context) throws IOException {
|
||||||
|
File internalFile = DBWrapper.getDBPath(context);
|
||||||
|
File backupFile = new File(context.getFilesDir(), "backup.db");
|
||||||
|
|
||||||
|
FileInputStream fis = new FileInputStream(backupFile);
|
||||||
|
FileOutputStream fos = new FileOutputStream(internalFile);
|
||||||
|
|
||||||
|
copyStream(fis, fos);
|
||||||
|
|
||||||
|
fis.close();
|
||||||
|
fos.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void deleteBackupDatabase(Context context) throws IOException {
|
||||||
|
File backupFile = new File(context.getFilesDir(), "backup.db");
|
||||||
|
backupFile.delete();
|
||||||
|
}
|
||||||
|
}
|
|
@ -25,4 +25,8 @@ public class Init {
|
||||||
if (Wrapper.instance() == null)
|
if (Wrapper.instance() == null)
|
||||||
Wrapper.setInstance(new DBWrapper(context));
|
Wrapper.setInstance(new DBWrapper(context));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static void reinitDB(Context context) {
|
||||||
|
Wrapper.setInstance(new DBWrapper(context));
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -16,29 +16,53 @@
|
||||||
|
|
||||||
package art.pegasko.yeeemp.ui.activity;
|
package art.pegasko.yeeemp.ui.activity;
|
||||||
|
|
||||||
|
import android.Manifest;
|
||||||
|
import android.app.Activity;
|
||||||
|
import android.app.AlertDialog;
|
||||||
|
import android.content.DialogInterface;
|
||||||
|
import android.content.Intent;
|
||||||
|
import android.content.pm.PackageManager;
|
||||||
|
import android.net.Uri;
|
||||||
|
import android.os.Build;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
|
|
||||||
import com.google.android.material.snackbar.Snackbar;
|
import com.google.android.material.snackbar.Snackbar;
|
||||||
|
|
||||||
|
import androidx.activity.result.ActivityResultLauncher;
|
||||||
import androidx.appcompat.app.AppCompatActivity;
|
import androidx.appcompat.app.AppCompatActivity;
|
||||||
|
|
||||||
|
import android.os.Environment;
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
|
import android.view.Menu;
|
||||||
|
import android.view.MenuItem;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
|
import android.widget.Toast;
|
||||||
|
|
||||||
|
import androidx.core.app.ActivityCompat;
|
||||||
|
import androidx.core.content.ContextCompat;
|
||||||
|
import androidx.core.content.PermissionChecker;
|
||||||
import androidx.navigation.ui.AppBarConfiguration;
|
import androidx.navigation.ui.AppBarConfiguration;
|
||||||
import androidx.recyclerview.widget.LinearLayoutManager;
|
import androidx.recyclerview.widget.LinearLayoutManager;
|
||||||
import androidx.recyclerview.widget.RecyclerView;
|
import androidx.recyclerview.widget.RecyclerView;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.security.Permission;
|
||||||
|
|
||||||
import art.pegasko.yeeemp.base.Queue;
|
import art.pegasko.yeeemp.base.Queue;
|
||||||
import art.pegasko.yeeemp.base.Wrapper;
|
import art.pegasko.yeeemp.base.Wrapper;
|
||||||
import art.pegasko.yeeemp.databinding.ActivityQueueListBinding;
|
import art.pegasko.yeeemp.databinding.ActivityQueueListBinding;
|
||||||
|
|
||||||
import art.pegasko.yeeemp.R;
|
import art.pegasko.yeeemp.R;
|
||||||
|
import art.pegasko.yeeemp.impl.DBWrapper;
|
||||||
|
import art.pegasko.yeeemp.impl.DataUtils;
|
||||||
import art.pegasko.yeeemp.impl.Init;
|
import art.pegasko.yeeemp.impl.Init;
|
||||||
|
|
||||||
public class QueueListActivity extends AppCompatActivity {
|
public class QueueListActivity extends AppCompatActivity {
|
||||||
public static final String TAG = QueueListActivity.class.getSimpleName();
|
public static final String TAG = QueueListActivity.class.getSimpleName();
|
||||||
|
|
||||||
|
private static final int REQUEST_CODE_CREATE_FILE = 37;
|
||||||
|
private static final int REQUEST_CODE_OPEN_FILE = 19;
|
||||||
|
|
||||||
private AppBarConfiguration appBarConfiguration;
|
private AppBarConfiguration appBarConfiguration;
|
||||||
private ActivityQueueListBinding binding;
|
private ActivityQueueListBinding binding;
|
||||||
private RecyclerView queueList;
|
private RecyclerView queueList;
|
||||||
|
@ -60,6 +84,26 @@ public class QueueListActivity extends AppCompatActivity {
|
||||||
setContentView(binding.getRoot());
|
setContentView(binding.getRoot());
|
||||||
setSupportActionBar(binding.toolbar);
|
setSupportActionBar(binding.toolbar);
|
||||||
|
|
||||||
|
/* Toolbar menu */
|
||||||
|
binding.toolbar.inflateMenu(R.menu.queue_list_toolbar_menu);
|
||||||
|
binding.toolbar.setOnMenuItemClickListener((MenuItem item) -> {
|
||||||
|
if (item.getItemId() == R.id.queue_list_toolbar_menu_export) {
|
||||||
|
Intent intent = new Intent(Intent.ACTION_CREATE_DOCUMENT);
|
||||||
|
intent.setType("*/*");
|
||||||
|
intent.putExtra(Intent.EXTRA_TITLE, "export_" + DataUtils.formatTs(System.currentTimeMillis()) + ".db");
|
||||||
|
startActivityForResult(intent, REQUEST_CODE_CREATE_FILE);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
} else if (item.getItemId() == R.id.queue_list_toolbar_menu_import) {
|
||||||
|
Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT);
|
||||||
|
intent.setType("*/*");
|
||||||
|
startActivityForResult(intent, REQUEST_CODE_OPEN_FILE);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
/* Queue list */
|
/* Queue list */
|
||||||
queueListAdapter = new QueueRecyclerViewAdapter();
|
queueListAdapter = new QueueRecyclerViewAdapter();
|
||||||
queueList = findViewById(R.id.content_queue_list__list);
|
queueList = findViewById(R.id.content_queue_list__list);
|
||||||
|
@ -86,6 +130,113 @@ public class QueueListActivity extends AppCompatActivity {
|
||||||
updateList();
|
updateList();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onActivityResult(int requestCode, int resultCode, Intent resultData) {
|
||||||
|
super.onActivityResult(requestCode, resultCode, resultData);
|
||||||
|
|
||||||
|
if (requestCode == REQUEST_CODE_CREATE_FILE && resultCode == Activity.RESULT_OK) {
|
||||||
|
if (resultData != null) {
|
||||||
|
Uri uri = resultData.getData();
|
||||||
|
Log.i(TAG, "Exporting file to " + uri.toString());
|
||||||
|
|
||||||
|
try {
|
||||||
|
DataUtils.exportDatabase(getApplicationContext(), uri);
|
||||||
|
Toast.makeText(QueueListActivity.this, "Exported database", Toast.LENGTH_SHORT).show();
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.wtf(TAG, e);
|
||||||
|
|
||||||
|
new AlertDialog
|
||||||
|
.Builder(QueueListActivity.this)
|
||||||
|
.setTitle("Export failed")
|
||||||
|
.setMessage(e.getMessage())
|
||||||
|
.setCancelable(true)
|
||||||
|
.setNegativeButton("OK", (DialogInterface dialog, int id) -> { dialog.cancel(); })
|
||||||
|
.show();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (requestCode == REQUEST_CODE_OPEN_FILE && resultCode == Activity.RESULT_OK) {
|
||||||
|
if (resultData != null) {
|
||||||
|
final Uri uri = resultData.getData();
|
||||||
|
|
||||||
|
new AlertDialog
|
||||||
|
.Builder(QueueListActivity.this)
|
||||||
|
.setTitle("Confirm action")
|
||||||
|
.setMessage("Import file as database? This will overwrite local database with external file without checking")
|
||||||
|
.setCancelable(true)
|
||||||
|
.setPositiveButton("Yes", (DialogInterface dialog, int id) -> {
|
||||||
|
Log.i(TAG, "Importing file from " + uri.toString());
|
||||||
|
|
||||||
|
try {
|
||||||
|
Log.i(TAG, "Backup before restore");
|
||||||
|
DataUtils.backupDatabase(getApplicationContext());
|
||||||
|
|
||||||
|
Log.i(TAG, "Importing database");
|
||||||
|
DataUtils.importDatabase(getApplicationContext(), uri);
|
||||||
|
Toast.makeText(QueueListActivity.this, "Imported database", Toast.LENGTH_SHORT).show();
|
||||||
|
|
||||||
|
Log.i(TAG, "Reloading database");
|
||||||
|
Init.reinitDB(getApplicationContext());
|
||||||
|
updateList();
|
||||||
|
|
||||||
|
Log.i(TAG, "Delete backup");
|
||||||
|
DataUtils.deleteBackupDatabase(getApplicationContext());
|
||||||
|
} catch (Exception e) {
|
||||||
|
Log.e(TAG, "Import failed");
|
||||||
|
Log.wtf(TAG, e);
|
||||||
|
|
||||||
|
new AlertDialog
|
||||||
|
.Builder(QueueListActivity.this)
|
||||||
|
.setTitle("Import failed")
|
||||||
|
.setMessage(e.getMessage())
|
||||||
|
.setCancelable(true)
|
||||||
|
.setNegativeButton("OK", (DialogInterface dialog2, int id2) -> { dialog2.cancel(); })
|
||||||
|
.show();
|
||||||
|
|
||||||
|
try {
|
||||||
|
Log.i(TAG, "Trying to restore backup");
|
||||||
|
DataUtils.restoreDatabase(getApplicationContext());
|
||||||
|
|
||||||
|
Log.i(TAG, "Reloading database");
|
||||||
|
Init.reinitDB(getApplicationContext());
|
||||||
|
updateList();
|
||||||
|
|
||||||
|
Log.i(TAG, "Delete backup");
|
||||||
|
DataUtils.deleteBackupDatabase(getApplicationContext());
|
||||||
|
|
||||||
|
new AlertDialog
|
||||||
|
.Builder(QueueListActivity.this)
|
||||||
|
.setTitle("Info")
|
||||||
|
.setMessage("Restored backup")
|
||||||
|
.setCancelable(true)
|
||||||
|
.setNegativeButton("OK", (DialogInterface dialog2, int id2) -> { dialog2.cancel(); })
|
||||||
|
.show();
|
||||||
|
|
||||||
|
} catch (Exception e2) {
|
||||||
|
Log.e(TAG, "Restore backup failed");
|
||||||
|
Log.wtf(TAG, e2);
|
||||||
|
|
||||||
|
new AlertDialog
|
||||||
|
.Builder(QueueListActivity.this)
|
||||||
|
.setTitle("Restore backup failed")
|
||||||
|
.setMessage(e2.getMessage())
|
||||||
|
.setCancelable(true)
|
||||||
|
.setNegativeButton("OK", (DialogInterface dialog2, int id2) -> { dialog2.cancel(); })
|
||||||
|
.show();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.setNegativeButton("Cancel", (DialogInterface dialog, int id) -> { dialog.cancel(); })
|
||||||
|
.show();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean onCreateOptionsMenu(Menu menu) {
|
||||||
|
getMenuInflater().inflate(R.menu.queue_list_toolbar_menu, menu);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onResume() {
|
protected void onResume() {
|
||||||
super.onResume();
|
super.onResume();
|
||||||
|
|
|
@ -49,4 +49,10 @@ public class Utils {
|
||||||
|
|
||||||
return finalItems;
|
return finalItems;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static int positiveHashCode16(Object o) {
|
||||||
|
if (o == null)
|
||||||
|
return 0;
|
||||||
|
return Math.abs(o.hashCode()) & ((1 << 16) - 1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
10
Yeeemp/app/src/main/res/menu/queue_list_toolbar_menu.xml
Normal file
10
Yeeemp/app/src/main/res/menu/queue_list_toolbar_menu.xml
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<menu xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
<item
|
||||||
|
android:id="@+id/queue_list_toolbar_menu_export"
|
||||||
|
android:title="Export" />
|
||||||
|
|
||||||
|
<item
|
||||||
|
android:id="@+id/queue_list_toolbar_menu_import"
|
||||||
|
android:title="Import" />
|
||||||
|
</menu>
|
Loading…
Reference in a new issue