2  *   ownCloud Android client application 
   4  *   @author Bartosz Przybylski 
   5  *   Copyright (C) 2015 ownCloud Inc. 
   6  *   Copyright (C) 2015 Bartosz Przybylski 
   8  *   This program is free software: you can redistribute it and/or modify 
   9  *   it under the terms of the GNU General Public License version 2, 
  10  *   as published by the Free Software Foundation. 
  12  *   This program is distributed in the hope that it will be useful, 
  13  *   but WITHOUT ANY WARRANTY; without even the implied warranty of 
  14  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the 
  15  *   GNU General Public License for more details. 
  17  *   You should have received a copy of the GNU General Public License 
  18  *   along with this program.  If not, see <http://www.gnu.org/licenses/>. 
  21 package com
.owncloud
.android
.ui
.activity
; 
  23 import android
.accounts
.Account
; 
  24 import android
.accounts
.AccountManager
; 
  25 import android
.content
.ContentResolver
; 
  26 import android
.content
.Context
; 
  27 import android
.content
.Intent
; 
  28 import android
.os
.AsyncTask
; 
  29 import android
.os
.Bundle
; 
  30 import android
.support
.v7
.app
.AppCompatActivity
; 
  31 import android
.view
.View
; 
  32 import android
.widget
.Button
; 
  33 import android
.widget
.ProgressBar
; 
  34 import android
.widget
.TextView
; 
  36 import com
.owncloud
.android
.MainApp
; 
  37 import com
.owncloud
.android
.R
; 
  38 import com
.owncloud
.android
.datamodel
.FileDataStorageManager
; 
  39 import com
.owncloud
.android
.lib
.common
.utils
.Log_OC
; 
  40 import com
.owncloud
.android
.utils
.FileStorageUtils
; 
  45  * Created by Bartosz Przybylski on 07.11.2015. 
  47 public class StorageMigrationActivity 
extends AppCompatActivity 
{ 
  48         private static final String TAG 
= StorageMigrationActivity
.class.getName(); 
  49         public static final String KEY_MIGRATION_TARGET_DIR 
= "MIGRATION_TARGET"; 
  50         public static final String KEY_MIGRATION_SOURCE_DIR 
= "MIGRATION_SOURCE"; 
  52         private ProgressBar mProgressBar
; 
  53         private Button mFinishButton
; 
  54         private TextView mFeedbackText
; 
  57         public void onCreate(Bundle savedInstanceState
) { 
  58                 super.onCreate(savedInstanceState
); 
  60                 setContentView(R
.layout
.migration_layout
); 
  61                 mProgressBar 
= (ProgressBar
)findViewById(R
.id
.migrationProgress
); 
  62                 mFinishButton 
= (Button
)findViewById(R
.id
.finishButton
); 
  63                 mFeedbackText 
= (TextView
)findViewById(R
.id
.migrationText
); 
  65                 mProgressBar
.setProgress(0); 
  66                 mFinishButton
.setVisibility(View
.INVISIBLE
); 
  67                 mFeedbackText
.setText(R
.string
.file_migration_preparing
); 
  69                 mFinishButton
.setOnClickListener(new View
.OnClickListener() { 
  71                         public void onClick(View view
) { 
  72                                 setResult(RESULT_CANCELED
); 
  77                 String source 
= getIntent().getStringExtra(KEY_MIGRATION_SOURCE_DIR
); 
  78                 String destination 
= getIntent().getStringExtra(KEY_MIGRATION_TARGET_DIR
); 
  80                 if (source 
== null 
|| destination 
== null
) { 
  81                         Log_OC
.e(TAG
, "source or destination is null"); 
  85                 new FileMigrationTask().execute(source
, destination
); 
  88         private class FileMigrationTask 
extends AsyncTask
<String
, Integer
, Integer
> { 
  90                 private String mStorageTarget
; 
  91                 private String mStorageSource
; 
  92                 private int mProgress
; 
  94                 private static final int mProgressCopyUpperBound 
= 98; 
  96                 private class MigrationException 
extends Exception 
{ 
  99                          * @param resId resource identifier to use for displaying error 
 101                         MigrationException(int resId
) { 
 106                         int getResId() { return mResId
; } 
 109                 private class MigrationCleanupException 
extends Exception 
{ 
 110                         MigrationCleanupException() {} 
 114                 protected Integer 
doInBackground(String
... args
) { 
 116                         mStorageSource 
= args
[0]; 
 117                         mStorageTarget 
= args
[1]; 
 120                         publishProgress(mProgress
++, R
.string
.file_migration_preparing
); 
 122                         Context context 
= StorageMigrationActivity
.this; 
 123                         String ocAuthority 
= context
.getString(R
.string
.authority
); 
 125                         Account
[] ocAccounts 
= AccountManager
.get(context
).getAccountsByType(MainApp
.getAccountType()); 
 126                         boolean[] oldAutoSync 
= new boolean[ocAccounts
.length
]; 
 129                                 publishProgress(mProgress
++, R
.string
.file_migration_checking_destination
); 
 131                                 checkDestinationAvailability(); 
 133                                 publishProgress(mProgress
++, R
.string
.file_migration_saving_accounts_configuration
); 
 134                                 saveAccountsSyncStatus(ocAuthority
, ocAccounts
, oldAutoSync
); 
 136                                 publishProgress(mProgress
++, R
.string
.file_migration_waiting_for_unfinished_sync
); 
 137                                 stopAccountsSyncing(ocAuthority
, ocAccounts
); 
 138                                 waitForUnfinishedSynchronizations(ocAuthority
, ocAccounts
); 
 140                                 publishProgress(mProgress
++, R
.string
.file_migration_migrating
); 
 143                                 publishProgress(mProgress
++, R
.string
.file_migration_updating_index
); 
 144                                 updateIndex(context
); 
 146                                 publishProgress(mProgress
++, R
.string
.file_migration_cleaning
); 
 149                         } catch (MigrationException e
) { 
 152                         } catch (MigrationCleanupException e
) { 
 153                                 Log_OC
.w(TAG
, "Migration cleanup step failed"); 
 156                                 publishProgress(mProgress
++, R
.string
.file_migration_restoring_accounts_configuration
); 
 157                                 restoreAccountsSyncStatus(ocAuthority
, ocAccounts
, oldAutoSync
); 
 160                         publishProgress(mProgress
++, R
.string
.file_migration_ok_finished
); 
 166                 protected void onProgressUpdate(Integer
... progress
) { 
 167                         mProgressBar
.setProgress(progress
[0]); 
 168                         if (progress
.length 
> 1) 
 169                                 mFeedbackText
.setText(progress
[1]); 
 173                 protected void onPostExecute(Integer code
) { 
 174                         mFinishButton
.setVisibility(View
.VISIBLE
); 
 176                                 mFeedbackText
.setText(code
); 
 178                                 mFeedbackText
.setText(R
.string
.file_migration_ok_finished
); 
 179                                 mFinishButton
.setOnClickListener(new View
.OnClickListener() { 
 181                                         public void onClick(View view
) { 
 182                                                 Intent resultIntent 
= new Intent(); 
 183                                                 resultIntent
.putExtra(KEY_MIGRATION_TARGET_DIR
, mStorageTarget
); 
 184                                                 setResult(RESULT_OK
, resultIntent
); 
 191                 void checkDestinationAvailability() throws MigrationException 
{ 
 192                         File srcFile 
= new File(mStorageSource
); 
 193                         File dstFile 
= new File(mStorageTarget
); 
 195                         if (!dstFile
.canRead() || !srcFile
.canRead()) 
 196                                 throw new MigrationException(R
.string
.file_migration_failed_not_readable
); 
 198                         if (!dstFile
.canWrite() || !srcFile
.canWrite()) 
 199                                 throw new MigrationException(R
.string
.file_migration_failed_not_writable
); 
 201                         if (new File(dstFile
, MainApp
.getDataFolder()).exists()) 
 202                                 throw new MigrationException(R
.string
.file_migration_failed_dir_already_exists
); 
 204                         if (dstFile
.getFreeSpace() < FileStorageUtils
.getFolderSize(new File(srcFile
, MainApp
.getDataFolder()))) 
 205                                 throw new MigrationException(R
.string
.file_migration_failed_not_enough_space
); 
 208                 void copyFiles() throws MigrationException 
{ 
 209                         File srcFile 
= new File(mStorageSource 
+ File
.separator 
+ MainApp
.getDataFolder()); 
 210                         File dstFile 
= new File(mStorageTarget 
+ File
.separator 
+ MainApp
.getDataFolder()); 
 212                         copyDirs(srcFile
, dstFile
); 
 213                         mProgress 
= Math
.max(mProgress
, mProgressCopyUpperBound
); 
 214                         publishProgress(mProgress
); 
 217                 void copyDirs(File src
, File dst
) throws MigrationException 
{ 
 219                                 throw new MigrationException(R
.string
.file_migration_failed_while_coping
); 
 221                         for (File f 
: src
.listFiles()) { 
 223                                 mProgress 
= Math
.min(mProgress
+1, mProgressCopyUpperBound
); 
 224                                 publishProgress(mProgress
); 
 227                                         copyDirs(f
, new File(dst
, f
.getName())); 
 228                                 else if (!FileStorageUtils
.copyFile(f
, new File(dst
, f
.getName()))) 
 229                                         throw new MigrationException(R
.string
.file_migration_failed_while_coping
); 
 234                 void updateIndex(Context context
) throws MigrationException 
{ 
 235                         FileDataStorageManager manager 
= new FileDataStorageManager(null
, context
.getContentResolver()); 
 238                                 manager
.migrateStoredFiles(mStorageSource
, mStorageTarget
); 
 239                         } catch (Exception e
) { 
 240                                 throw new MigrationException(R
.string
.file_migration_failed_while_updating_index
); 
 244                 void cleanup() throws MigrationCleanupException 
{ 
 245                         File srcFile 
= new File(mStorageSource 
+ File
.separator 
+ MainApp
.getDataFolder()); 
 246                         if (!srcFile
.delete()) 
 247                                 throw new MigrationCleanupException(); 
 251                         File dstFile 
= new File(mStorageTarget 
+ File
.separator 
+ MainApp
.getDataFolder()); 
 252                         if (dstFile
.exists()) 
 253                                 if (!dstFile
.delete()) 
 254                                         Log_OC
.w(TAG
, "Rollback step failed"); 
 257                 void saveAccountsSyncStatus(String authority
, Account accounts
[], boolean syncs
[]) { 
 258                         for (int i 
= 0; i 
< accounts
.length
; ++i
) 
 259                                 syncs
[i
] = ContentResolver
.getSyncAutomatically(accounts
[i
], authority
); 
 262                 void stopAccountsSyncing(String authority
, Account accounts
[]) { 
 263                         for (int i 
= 0; i 
< accounts
.length
; ++i
) 
 264                                 ContentResolver
.setSyncAutomatically(accounts
[i
], authority
, false
); 
 267                 void waitForUnfinishedSynchronizations(String authority
, Account accounts
[]) { 
 268                         for (int i 
= 0; i 
< accounts
.length
; ++i
) 
 269                                 while (ContentResolver
.isSyncActive(accounts
[i
], authority
)) 
 272                                         } catch (InterruptedException e
) { 
 273                                                 Log_OC
.w(TAG
, "Thread interrupted while waiting for account to end syncing"); 
 274                                                 Thread
.currentThread().interrupt(); 
 278                 void restoreAccountsSyncStatus(String authority
, Account accounts
[], boolean oldSync
[]) { 
 279                         for (int i 
= 0; i 
< accounts
.length
; ++i
) 
 280                                 ContentResolver
.setSyncAutomatically(accounts
[i
], authority
, oldSync
[i
]);