1 /* ownCloud Android client application
2 * Copyright (C) 2015 ownCloud Inc.
4 * This program is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2,
6 * as published by the Free Software Foundation.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
13 * You should have received a copy of the GNU General Public License
14 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 package com
.owncloud
.android
.services
;
20 import android
.accounts
.Account
;
21 import android
.accounts
.AccountsException
;
22 import android
.content
.Intent
;
23 import android
.os
.Handler
;
24 import android
.os
.Looper
;
25 import android
.os
.Message
;
26 import android
.util
.Pair
;
28 import com
.owncloud
.android
.datamodel
.FileDataStorageManager
;
29 import com
.owncloud
.android
.datamodel
.OCFile
;
30 import com
.owncloud
.android
.files
.services
.FileDownloader
;
31 import com
.owncloud
.android
.files
.services
.IndexedForest
;
32 import com
.owncloud
.android
.lib
.common
.OwnCloudAccount
;
33 import com
.owncloud
.android
.lib
.common
.OwnCloudClient
;
34 import com
.owncloud
.android
.lib
.common
.OwnCloudClientManagerFactory
;
35 import com
.owncloud
.android
.lib
.common
.operations
.RemoteOperationResult
;
36 import com
.owncloud
.android
.lib
.common
.utils
.Log_OC
;
37 import com
.owncloud
.android
.operations
.SynchronizeFolderOperation
;
38 import com
.owncloud
.android
.utils
.FileStorageUtils
;
40 import java
.io
.IOException
;
43 * SyncFolder worker. Performs the pending operations in the order they were requested.
45 * Created with the Looper of a new thread, started in
46 * {@link com.owncloud.android.services.OperationsService#onCreate()}.
48 class SyncFolderHandler
extends Handler
{
50 private static final String TAG
= SyncFolderHandler
.class.getSimpleName();
53 OperationsService mService
;
55 private IndexedForest
<SynchronizeFolderOperation
> mPendingOperations
=
56 new IndexedForest
<SynchronizeFolderOperation
>();
58 private OwnCloudClient mOwnCloudClient
= null
;
59 private Account mCurrentAccount
= null
;
60 private FileDataStorageManager mStorageManager
;
61 private SynchronizeFolderOperation mCurrentSyncOperation
;
64 public SyncFolderHandler(Looper looper
, OperationsService service
) {
66 if (service
== null
) {
67 throw new IllegalArgumentException("Received invalid NULL in parameter 'service'");
74 * Returns True when the folder located in 'remotePath' in the ownCloud account 'account', or any of its
75 * descendants, is being synchronized (or waiting for it).
77 * @param account ownCloud account where the remote folder is stored.
78 * @param remotePath The path to a folder that could be in the queue of synchronizations.
80 public boolean isSynchronizing(Account account
, String remotePath
) {
81 if (account
== null
|| remotePath
== null
) return false
;
82 return (mPendingOperations
.contains(account
, remotePath
));
87 public void handleMessage(Message msg
) {
88 Pair
<Account
, String
> itemSyncKey
= (Pair
<Account
, String
>) msg
.obj
;
89 /*Log_OC.v( "NOW " + TAG + ", thread " + Thread.currentThread().getName(),
90 "Handling sync folder " + itemSyncKey.second);*/
91 doOperation(itemSyncKey
.first
, itemSyncKey
.second
);
92 mService
.stopSelf(msg
.arg1
);
97 * Performs the next operation in the queue
99 private void doOperation(Account account
, String remotePath
) {
101 /*Log_OC.v( "NOW " + TAG + ", thread " + Thread.currentThread().getName(),
102 "Getting sync folder " + remotePath);*/
103 mCurrentSyncOperation
= mPendingOperations
.get(account
, remotePath
);
105 if (mCurrentSyncOperation
!= null
) {
106 RemoteOperationResult result
= null
;
110 if (mCurrentAccount
== null
|| !mCurrentAccount
.equals(account
)) {
111 mCurrentAccount
= account
;
112 mStorageManager
= new FileDataStorageManager(
114 mService
.getContentResolver()
116 } // else, reuse storage manager from previous operation
118 // always get client from client manager, to get fresh credentials in case of update
119 OwnCloudAccount ocAccount
= new OwnCloudAccount(account
, mService
);
120 mOwnCloudClient
= OwnCloudClientManagerFactory
.getDefaultSingleton().
121 getClientFor(ocAccount
, mService
);
123 /*Log_OC.v( "NOW " + TAG + ", thread " + Thread.currentThread().getName(),
124 "Executing sync folder " + remotePath);*/
125 result
= mCurrentSyncOperation
.execute(mOwnCloudClient
, mStorageManager
);
127 } catch (AccountsException e
) {
128 Log_OC
.e(TAG
, "Error while trying to get authorization", e
);
129 } catch (IOException e
) {
130 Log_OC
.e(TAG
, "Error while trying to get authorization", e
);
132 /*Log_OC.v( "NOW " + TAG + ", thread " + Thread.currentThread().getName(),
133 "Removing payload " + remotePath);*/
135 mPendingOperations
.removePayload(account
, remotePath
);
137 mService
.dispatchResultToOperationListeners(null
, mCurrentSyncOperation
, result
);
139 sendBroadcastFinishedSyncFolder(account
, remotePath
, result
.isSuccess());
144 public void add(Account account
, String remotePath
, SynchronizeFolderOperation syncFolderOperation
){
145 mPendingOperations
.putIfAbsent(account
, remotePath
, syncFolderOperation
);
146 sendBroadcastNewSyncFolder(account
, remotePath
); // TODO upgrade!
151 * Cancels a pending or current sync' operation.
153 * @param account ownCloud account where the remote file is stored.
154 * @param file A file in the queue of pending synchronizations
156 public void cancel(Account account
, OCFile file
){
157 if (account
== null
|| file
== null
) {
158 Log_OC
.e(TAG
, "Cannot cancel with NULL parameters");
161 /*Log_OC.v( "NOW " + TAG + ", thread " + Thread.currentThread().getName(),
162 "Removing sync folder " + file.getRemotePath());*/
163 Pair
<SynchronizeFolderOperation
, String
> removeResult
=
164 mPendingOperations
.remove(account
, file
.getRemotePath());
165 SynchronizeFolderOperation synchronization
= removeResult
.first
;
166 if (synchronization
!= null
) {
167 /*Log_OC.v( "NOW " + TAG + ", thread " + Thread.currentThread().getName(),
168 "Canceling returned sync of " + file.getRemotePath());*/
169 synchronization
.cancel();
172 if (mCurrentSyncOperation
!= null
&& mCurrentAccount
!= null
&&
173 mCurrentSyncOperation
.getRemotePath().startsWith(file
.getRemotePath()) &&
174 account
.name
.equals(mCurrentAccount
.name
)) {
175 /*Log_OC.v( "NOW " + TAG + ", thread " + Thread.currentThread().getName(),
176 "Canceling current sync as descendant: " + mCurrentSyncOperation.getRemotePath());*/
177 mCurrentSyncOperation
.cancel();
179 /*Log_OC.v( "NOW " + TAG + ", thread " + Thread.currentThread().getName(),
180 "Nothing else in cancelation of " + file.getRemotePath());*/
184 //sendBroadcastFinishedSyncFolder(account, file.getRemotePath());
188 * TODO review this method when "folder synchronization" replaces "folder download"; this is a fast and ugly
191 private void sendBroadcastNewSyncFolder(Account account
, String remotePath
) {
192 Intent added
= new Intent(FileDownloader
.getDownloadAddedMessage());
193 added
.putExtra(FileDownloader
.ACCOUNT_NAME
, account
.name
);
194 added
.putExtra(FileDownloader
.EXTRA_REMOTE_PATH
, remotePath
);
195 added
.putExtra(FileDownloader
.EXTRA_FILE_PATH
, FileStorageUtils
.getSavePath(account
.name
) + remotePath
);
196 mService
.sendStickyBroadcast(added
);
200 * TODO review this method when "folder synchronization" replaces "folder download"; this is a fast and ugly
203 private void sendBroadcastFinishedSyncFolder(Account account
, String remotePath
, boolean success
) {
204 Intent finished
= new Intent(FileDownloader
.getDownloadFinishMessage());
205 finished
.putExtra(FileDownloader
.ACCOUNT_NAME
, account
.name
);
206 finished
.putExtra(FileDownloader
.EXTRA_REMOTE_PATH
, remotePath
);
207 finished
.putExtra(FileDownloader
.EXTRA_FILE_PATH
, FileStorageUtils
.getSavePath(account
.name
) + remotePath
);
208 finished
.putExtra(FileDownloader
.EXTRA_DOWNLOAD_RESULT
, success
);
209 mService
.sendStickyBroadcast(finished
);