|
@@ -1,303 +0,0 @@
|
|
|
-/*
|
|
|
- * Nextcloud Android client application
|
|
|
- *
|
|
|
- * @author Tobias Kaminsky
|
|
|
- * Copyright (C) 2017 Tobias Kaminsky
|
|
|
- * Copyright (C) 2017 Nextcloud GmbH.
|
|
|
- *
|
|
|
- * This program is free software: you can redistribute it and/or modify
|
|
|
- * it under the terms of the GNU Affero General Public License as published by
|
|
|
- * the Free Software Foundation, either version 3 of the License, or
|
|
|
- * at your option) any later version.
|
|
|
- *
|
|
|
- * This program is distributed in the hope that it will be useful,
|
|
|
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
- * GNU Affero General Public License for more details.
|
|
|
- *
|
|
|
- * You should have received a copy of the GNU Affero General Public License
|
|
|
- * along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
- */
|
|
|
-
|
|
|
-package com.owncloud.android.jobs;
|
|
|
-
|
|
|
-import android.content.ComponentName;
|
|
|
-import android.content.Intent;
|
|
|
-import android.content.ServiceConnection;
|
|
|
-import android.database.Cursor;
|
|
|
-import android.net.Uri;
|
|
|
-import android.os.IBinder;
|
|
|
-import android.provider.ContactsContract;
|
|
|
-import android.text.TextUtils;
|
|
|
-import android.text.format.DateFormat;
|
|
|
-
|
|
|
-import com.evernote.android.job.Job;
|
|
|
-import com.evernote.android.job.util.support.PersistableBundleCompat;
|
|
|
-import com.nextcloud.client.account.User;
|
|
|
-import com.nextcloud.client.account.UserAccountManager;
|
|
|
-import com.nextcloud.java.util.Optional;
|
|
|
-import com.owncloud.android.R;
|
|
|
-import com.owncloud.android.datamodel.ArbitraryDataProvider;
|
|
|
-import com.owncloud.android.datamodel.FileDataStorageManager;
|
|
|
-import com.owncloud.android.datamodel.OCFile;
|
|
|
-import com.owncloud.android.files.services.FileUploader;
|
|
|
-import com.owncloud.android.lib.common.utils.Log_OC;
|
|
|
-import com.owncloud.android.operations.UploadFileOperation;
|
|
|
-import com.owncloud.android.services.OperationsService;
|
|
|
-
|
|
|
-import java.io.File;
|
|
|
-import java.io.FileWriter;
|
|
|
-import java.io.IOException;
|
|
|
-import java.io.InputStream;
|
|
|
-import java.io.InputStreamReader;
|
|
|
-import java.util.ArrayList;
|
|
|
-import java.util.Calendar;
|
|
|
-import java.util.List;
|
|
|
-
|
|
|
-import androidx.annotation.NonNull;
|
|
|
-import ezvcard.Ezvcard;
|
|
|
-import ezvcard.VCardVersion;
|
|
|
-
|
|
|
-import static com.owncloud.android.ui.activity.ContactsPreferenceActivity.PREFERENCE_CONTACTS_LAST_BACKUP;
|
|
|
-
|
|
|
-/**
|
|
|
- * Job that backup contacts to /Contacts-Backup and deletes files older than x days
|
|
|
- */
|
|
|
-
|
|
|
-public class ContactsBackupJob extends Job {
|
|
|
- public static final String TAG = "ContactsBackupJob";
|
|
|
- public static final String ACCOUNT = "account";
|
|
|
- public static final String FORCE = "force";
|
|
|
-
|
|
|
- private OperationsServiceConnection operationsServiceConnection;
|
|
|
- private OperationsService.OperationsServiceBinder operationsServiceBinder;
|
|
|
- private UserAccountManager accountManager;
|
|
|
-
|
|
|
- public ContactsBackupJob(UserAccountManager accountManager) {
|
|
|
- this.accountManager = accountManager;
|
|
|
- }
|
|
|
-
|
|
|
- @NonNull
|
|
|
- @Override
|
|
|
- protected Result onRunJob(@NonNull Params params) {
|
|
|
- PersistableBundleCompat bundle = params.getExtras();
|
|
|
-
|
|
|
- final String accountName = bundle.getString(ACCOUNT, "");
|
|
|
- if (TextUtils.isEmpty(accountName)) {
|
|
|
- // no account provided
|
|
|
- return Result.FAILURE;
|
|
|
- }
|
|
|
- final Optional<User> optionalUser = accountManager.getUser(accountName);
|
|
|
- if (!optionalUser.isPresent()) {
|
|
|
- return Result.FAILURE;
|
|
|
- }
|
|
|
- User user = optionalUser.get();
|
|
|
-
|
|
|
- ArbitraryDataProvider arbitraryDataProvider = new ArbitraryDataProvider(getContext().getContentResolver());
|
|
|
- Long lastExecution = arbitraryDataProvider.getLongValue(user.toPlatformAccount(),
|
|
|
- PREFERENCE_CONTACTS_LAST_BACKUP);
|
|
|
-
|
|
|
- boolean force = bundle.getBoolean(FORCE, false);
|
|
|
- if (force || (lastExecution + 24 * 60 * 60 * 1000) < Calendar.getInstance().getTimeInMillis()) {
|
|
|
- Log_OC.d(TAG, "start contacts backup job");
|
|
|
-
|
|
|
- String backupFolder = getContext().getResources().getString(R.string.contacts_backup_folder) +
|
|
|
- OCFile.PATH_SEPARATOR;
|
|
|
- Integer daysToExpire = getContext().getResources().getInteger(R.integer.contacts_backup_expire);
|
|
|
-
|
|
|
- backupContact(user, backupFolder);
|
|
|
-
|
|
|
- // bind to Operations Service
|
|
|
- operationsServiceConnection = new OperationsServiceConnection(daysToExpire,
|
|
|
- backupFolder,
|
|
|
- user);
|
|
|
-
|
|
|
- getContext().bindService(new Intent(getContext(), OperationsService.class), operationsServiceConnection,
|
|
|
- OperationsService.BIND_AUTO_CREATE);
|
|
|
-
|
|
|
- // store execution date
|
|
|
- arbitraryDataProvider.storeOrUpdateKeyValue(user.getAccountName(),
|
|
|
- PREFERENCE_CONTACTS_LAST_BACKUP,
|
|
|
- Calendar.getInstance().getTimeInMillis());
|
|
|
- } else {
|
|
|
- Log_OC.d(TAG, "last execution less than 24h ago");
|
|
|
- }
|
|
|
-
|
|
|
- return Result.SUCCESS;
|
|
|
- }
|
|
|
-
|
|
|
- private void backupContact(User user, String backupFolder) {
|
|
|
- ArrayList<String> vCard = new ArrayList<>();
|
|
|
-
|
|
|
- Cursor cursor = getContext().getContentResolver().query(ContactsContract.Contacts.CONTENT_URI, null,
|
|
|
- null, null, null);
|
|
|
-
|
|
|
- if (cursor != null && cursor.getCount() > 0) {
|
|
|
- cursor.moveToFirst();
|
|
|
- for (int i = 0; i < cursor.getCount(); i++) {
|
|
|
-
|
|
|
- vCard.add(getContactFromCursor(cursor));
|
|
|
- cursor.moveToNext();
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- String filename = DateFormat.format("yyyy-MM-dd_HH-mm-ss", Calendar.getInstance()).toString() + ".vcf";
|
|
|
- Log_OC.d(TAG, "Storing: " + filename);
|
|
|
- File file = new File(getContext().getCacheDir(), filename);
|
|
|
-
|
|
|
- FileWriter fw = null;
|
|
|
- try {
|
|
|
- fw = new FileWriter(file);
|
|
|
-
|
|
|
- for (String card : vCard) {
|
|
|
- fw.write(card);
|
|
|
- }
|
|
|
-
|
|
|
- } catch (IOException e) {
|
|
|
- Log_OC.d(TAG, "Error ", e);
|
|
|
- } finally {
|
|
|
- if (cursor != null) {
|
|
|
- cursor.close();
|
|
|
- }
|
|
|
-
|
|
|
- if (fw != null) {
|
|
|
- try {
|
|
|
- fw.close();
|
|
|
- } catch (IOException e) {
|
|
|
- Log_OC.d(TAG, "Error closing file writer ", e);
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- FileUploader.uploadNewFile(
|
|
|
- getContext(),
|
|
|
- user.toPlatformAccount(),
|
|
|
- file.getAbsolutePath(),
|
|
|
- backupFolder + filename,
|
|
|
- FileUploader.LOCAL_BEHAVIOUR_MOVE,
|
|
|
- null,
|
|
|
- true,
|
|
|
- UploadFileOperation.CREATED_BY_USER,
|
|
|
- false,
|
|
|
- false,
|
|
|
- FileUploader.NameCollisionPolicy.ASK_USER
|
|
|
- );
|
|
|
- }
|
|
|
-
|
|
|
- private void expireFiles(Integer daysToExpire, String backupFolderString, User account) {
|
|
|
- // -1 disables expiration
|
|
|
- if (daysToExpire > -1) {
|
|
|
- FileDataStorageManager storageManager = new FileDataStorageManager(account.toPlatformAccount(),
|
|
|
- getContext().getContentResolver());
|
|
|
- OCFile backupFolder = storageManager.getFileByPath(backupFolderString);
|
|
|
- Calendar cal = Calendar.getInstance();
|
|
|
- cal.add(Calendar.DAY_OF_YEAR, -daysToExpire);
|
|
|
- Long timestampToExpire = cal.getTimeInMillis();
|
|
|
-
|
|
|
- if (backupFolder != null) {
|
|
|
- Log_OC.d(TAG, "expire: " + daysToExpire + " " + backupFolder.getFileName());
|
|
|
- }
|
|
|
-
|
|
|
- List<OCFile> backups = storageManager.getFolderContent(backupFolder, false);
|
|
|
-
|
|
|
- for (OCFile backup : backups) {
|
|
|
- if (timestampToExpire > backup.getModificationTimestamp()) {
|
|
|
- Log_OC.d(TAG, "delete " + backup.getRemotePath());
|
|
|
-
|
|
|
- // delete backups
|
|
|
- Intent service = new Intent(getContext(), OperationsService.class);
|
|
|
- service.setAction(OperationsService.ACTION_REMOVE);
|
|
|
- service.putExtra(OperationsService.EXTRA_ACCOUNT, account);
|
|
|
- service.putExtra(OperationsService.EXTRA_REMOTE_PATH, backup.getRemotePath());
|
|
|
- service.putExtra(OperationsService.EXTRA_REMOVE_ONLY_LOCAL, false);
|
|
|
- operationsServiceBinder.queueNewOperation(service);
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- getContext().unbindService(operationsServiceConnection);
|
|
|
- }
|
|
|
-
|
|
|
- private String getContactFromCursor(Cursor cursor) {
|
|
|
- String lookupKey = cursor.getString(cursor.getColumnIndex(ContactsContract.Contacts.LOOKUP_KEY));
|
|
|
- Uri uri = Uri.withAppendedPath(ContactsContract.Contacts.CONTENT_VCARD_URI, lookupKey);
|
|
|
-
|
|
|
- String vCard = "";
|
|
|
- InputStream inputStream = null;
|
|
|
- InputStreamReader inputStreamReader = null;
|
|
|
- try {
|
|
|
- inputStream = getContext().getContentResolver().openInputStream(uri);
|
|
|
- char[] buffer = new char[1024];
|
|
|
- StringBuilder stringBuilder = new StringBuilder();
|
|
|
-
|
|
|
- if (inputStream != null) {
|
|
|
- inputStreamReader = new InputStreamReader(inputStream);
|
|
|
-
|
|
|
- while (true) {
|
|
|
- int byteCount = inputStreamReader.read(buffer, 0, buffer.length);
|
|
|
-
|
|
|
- if (byteCount > 0) {
|
|
|
- stringBuilder.append(buffer, 0, byteCount);
|
|
|
- } else {
|
|
|
- break;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- vCard = stringBuilder.toString();
|
|
|
-
|
|
|
- // bump to vCard 3.0 format (min version supported by server) since Android OS exports to 2.1
|
|
|
- return Ezvcard.write(Ezvcard.parse(vCard).all()).version(VCardVersion.V3_0).go();
|
|
|
-
|
|
|
- } catch (IOException e) {
|
|
|
- Log_OC.d(TAG, e.getMessage());
|
|
|
- } finally {
|
|
|
- try {
|
|
|
- if (inputStream != null) {
|
|
|
- inputStream.close();
|
|
|
- }
|
|
|
- if (inputStreamReader != null) {
|
|
|
- inputStreamReader.close();
|
|
|
- }
|
|
|
- } catch (IOException e) {
|
|
|
- Log_OC.e(TAG, "failed to close stream");
|
|
|
- }
|
|
|
- }
|
|
|
- return vCard;
|
|
|
- }
|
|
|
-
|
|
|
- /**
|
|
|
- * Implements callback methods for service binding.
|
|
|
- */
|
|
|
- private class OperationsServiceConnection implements ServiceConnection {
|
|
|
- private Integer daysToExpire;
|
|
|
- private String backupFolder;
|
|
|
- private User user;
|
|
|
-
|
|
|
- OperationsServiceConnection(Integer daysToExpire, String backupFolder, User user) {
|
|
|
- this.daysToExpire = daysToExpire;
|
|
|
- this.backupFolder = backupFolder;
|
|
|
- this.user = user;
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- public void onServiceConnected(ComponentName component, IBinder service) {
|
|
|
- Log_OC.d(TAG, "service connected");
|
|
|
-
|
|
|
-
|
|
|
- if (component.equals(new ComponentName(getContext(), OperationsService.class))) {
|
|
|
- operationsServiceBinder = (OperationsService.OperationsServiceBinder) service;
|
|
|
- expireFiles(daysToExpire, backupFolder, user);
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- @Override
|
|
|
- public void onServiceDisconnected(ComponentName component) {
|
|
|
- Log_OC.d(TAG, "service disconnected");
|
|
|
-
|
|
|
- if (component.equals(new ComponentName(getContext(), OperationsService.class))) {
|
|
|
- operationsServiceBinder = null;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-}
|