DocumentsStorageProvider.java 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629
  1. /*
  2. * Nextcloud Android client application
  3. *
  4. * @author Bartosz Przybylski
  5. * @author Chris Narkiewicz
  6. * Copyright (C) 2016 Bartosz Przybylski <bart.p.pl@gmail.com>
  7. * Copyright (C) 2019 Chris Narkiewicz <hello@ezaquarii.com>
  8. *
  9. * This program is free software: you can redistribute it and/or modify
  10. * it under the terms of the GNU General Public License version 2,
  11. * as published by the Free Software Foundation.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  20. *
  21. */
  22. package com.owncloud.android.providers;
  23. import android.accounts.Account;
  24. import android.accounts.AuthenticatorException;
  25. import android.accounts.OperationCanceledException;
  26. import android.annotation.SuppressLint;
  27. import android.annotation.TargetApi;
  28. import android.content.ContentResolver;
  29. import android.content.Context;
  30. import android.content.Intent;
  31. import android.content.res.AssetFileDescriptor;
  32. import android.database.Cursor;
  33. import android.graphics.Point;
  34. import android.net.Uri;
  35. import android.os.Build;
  36. import android.os.CancellationSignal;
  37. import android.os.Handler;
  38. import android.os.Looper;
  39. import android.os.ParcelFileDescriptor;
  40. import android.provider.DocumentsProvider;
  41. import android.util.Log;
  42. import android.widget.Toast;
  43. import com.evernote.android.job.JobRequest;
  44. import com.evernote.android.job.util.Device;
  45. import com.nextcloud.client.account.UserAccountManager;
  46. import com.nextcloud.client.preferences.AppPreferences;
  47. import com.nextcloud.client.preferences.AppPreferencesImpl;
  48. import com.owncloud.android.MainApp;
  49. import com.owncloud.android.R;
  50. import com.owncloud.android.datamodel.FileDataStorageManager;
  51. import com.owncloud.android.datamodel.OCFile;
  52. import com.owncloud.android.datamodel.ThumbnailsCacheManager;
  53. import com.owncloud.android.files.services.FileDownloader;
  54. import com.owncloud.android.files.services.FileUploader;
  55. import com.owncloud.android.lib.common.OwnCloudAccount;
  56. import com.owncloud.android.lib.common.OwnCloudClient;
  57. import com.owncloud.android.lib.common.OwnCloudClientManagerFactory;
  58. import com.owncloud.android.lib.common.operations.RemoteOperationResult;
  59. import com.owncloud.android.lib.common.utils.Log_OC;
  60. import com.owncloud.android.operations.CopyFileOperation;
  61. import com.owncloud.android.operations.CreateFolderOperation;
  62. import com.owncloud.android.operations.MoveFileOperation;
  63. import com.owncloud.android.operations.RefreshFolderOperation;
  64. import com.owncloud.android.operations.RemoveFileOperation;
  65. import com.owncloud.android.operations.RenameFileOperation;
  66. import com.owncloud.android.operations.SynchronizeFileOperation;
  67. import com.owncloud.android.operations.UploadFileOperation;
  68. import com.owncloud.android.ui.activity.ConflictsResolveActivity;
  69. import com.owncloud.android.ui.activity.SettingsActivity;
  70. import com.owncloud.android.utils.FileStorageUtils;
  71. import com.owncloud.android.utils.UriUtils;
  72. import org.nextcloud.providers.cursors.FileCursor;
  73. import org.nextcloud.providers.cursors.RootCursor;
  74. import java.io.File;
  75. import java.io.FileNotFoundException;
  76. import java.io.IOException;
  77. import java.util.ArrayList;
  78. import java.util.HashMap;
  79. import java.util.List;
  80. import java.util.Map;
  81. import javax.inject.Inject;
  82. import dagger.android.AndroidInjection;
  83. @TargetApi(Build.VERSION_CODES.KITKAT)
  84. public class DocumentsStorageProvider extends DocumentsProvider {
  85. private static final String TAG = "DocumentsStorageProvider";
  86. private FileDataStorageManager currentStorageManager;
  87. private Map<Long, FileDataStorageManager> rootIdToStorageManager;
  88. private OwnCloudClient client;
  89. @Inject UserAccountManager accountManager;
  90. @Override
  91. public Cursor queryRoots(String[] projection) throws FileNotFoundException {
  92. Context context = MainApp.getAppContext();
  93. AppPreferences preferences = AppPreferencesImpl.fromContext(context);
  94. if (SettingsActivity.LOCK_PASSCODE.equals(preferences.getLockPreference()) ||
  95. SettingsActivity.LOCK_DEVICE_CREDENTIALS.equals(preferences.getLockPreference())) {
  96. return new FileCursor();
  97. }
  98. initiateStorageMap();
  99. final RootCursor result = new RootCursor(projection);
  100. for (Account account : accountManager.getAccounts()) {
  101. result.addRoot(account, getContext());
  102. }
  103. return result;
  104. }
  105. @Override
  106. public Cursor queryDocument(String documentId, String[] projection) throws FileNotFoundException {
  107. final long docId = Long.parseLong(documentId);
  108. updateCurrentStorageManagerIfNeeded(docId);
  109. if (currentStorageManager == null) {
  110. for (Map.Entry<Long, FileDataStorageManager> entry : rootIdToStorageManager.entrySet()) {
  111. if (entry.getValue().getFileById(docId) != null) {
  112. currentStorageManager = entry.getValue();
  113. break;
  114. }
  115. }
  116. }
  117. if (currentStorageManager == null) {
  118. throw new FileNotFoundException("File with id " + documentId + " not found");
  119. }
  120. final FileCursor result = new FileCursor(projection);
  121. OCFile file = currentStorageManager.getFileById(docId);
  122. if (file != null) {
  123. result.addFile(file);
  124. }
  125. return result;
  126. }
  127. @SuppressLint("LongLogTag")
  128. @Override
  129. public Cursor queryChildDocuments(String parentDocumentId, String[] projection, String sortOrder)
  130. throws FileNotFoundException {
  131. final long folderId = Long.parseLong(parentDocumentId);
  132. updateCurrentStorageManagerIfNeeded(folderId);
  133. Context context = getContext();
  134. if (context == null) {
  135. throw new FileNotFoundException("Context may not be null");
  136. }
  137. Account account = currentStorageManager.getAccount();
  138. final OCFile browsedDir = currentStorageManager.getFileById(folderId);
  139. if (Device.getNetworkType(context).equals(JobRequest.NetworkType.UNMETERED)) {
  140. RemoteOperationResult result = new RefreshFolderOperation(browsedDir, System.currentTimeMillis(), false,
  141. false, true, currentStorageManager, account,
  142. getContext()).execute(client);
  143. if (!result.isSuccess()) {
  144. throw new FileNotFoundException("Failed to update document " + parentDocumentId);
  145. }
  146. }
  147. final FileCursor resultCursor = new FileCursor(projection);
  148. for (OCFile file : currentStorageManager.getFolderContent(browsedDir, false)) {
  149. resultCursor.addFile(file);
  150. }
  151. return resultCursor;
  152. }
  153. @SuppressLint("LongLogTag")
  154. @Override
  155. public ParcelFileDescriptor openDocument(String documentId, String mode, CancellationSignal cancellationSignal)
  156. throws FileNotFoundException {
  157. final long docId = Long.parseLong(documentId);
  158. updateCurrentStorageManagerIfNeeded(docId);
  159. OCFile ocFile = currentStorageManager.getFileById(docId);
  160. if (ocFile == null) {
  161. throw new FileNotFoundException("File not found: " + documentId);
  162. }
  163. Context context = getContext();
  164. if (context == null) {
  165. throw new FileNotFoundException("Context may not be null!");
  166. }
  167. Account account = currentStorageManager.getAccount();
  168. if (!ocFile.isDown()) {
  169. Intent i = new Intent(getContext(), FileDownloader.class);
  170. i.putExtra(FileDownloader.EXTRA_ACCOUNT, account);
  171. i.putExtra(FileDownloader.EXTRA_FILE, ocFile);
  172. if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) {
  173. context.startForegroundService(i);
  174. } else {
  175. context.startService(i);
  176. }
  177. do {
  178. if (!waitOrGetCancelled(cancellationSignal)) {
  179. throw new FileNotFoundException("File with id " + documentId + " not found!");
  180. }
  181. ocFile = currentStorageManager.getFileById(docId);
  182. if (ocFile == null) {
  183. throw new FileNotFoundException("File with id " + documentId + " not found!");
  184. }
  185. } while (!ocFile.isDown());
  186. } else {
  187. OCFile finalFile = ocFile;
  188. Thread syncThread = new Thread(() -> {
  189. try {
  190. FileDataStorageManager storageManager =
  191. new FileDataStorageManager(account, context.getContentResolver());
  192. SynchronizeFileOperation sfo =
  193. new SynchronizeFileOperation(finalFile, null, account, true, context);
  194. RemoteOperationResult result = sfo.execute(storageManager, context);
  195. if (result.getCode() == RemoteOperationResult.ResultCode.SYNC_CONFLICT) {
  196. // ISSUE 5: if the user is not running the app (this is a service!),
  197. // this can be very intrusive; a notification should be preferred
  198. Intent i = new Intent(context, ConflictsResolveActivity.class);
  199. i.setFlags(i.getFlags() | Intent.FLAG_ACTIVITY_NEW_TASK);
  200. i.putExtra(ConflictsResolveActivity.EXTRA_FILE, finalFile);
  201. i.putExtra(ConflictsResolveActivity.EXTRA_ACCOUNT, account);
  202. context.startActivity(i);
  203. } else {
  204. FileStorageUtils.checkIfFileFinishedSaving(finalFile);
  205. if (!result.isSuccess()) {
  206. showToast();
  207. }
  208. }
  209. } catch (Exception exception) {
  210. showToast();
  211. }
  212. });
  213. syncThread.start();
  214. try {
  215. syncThread.join();
  216. } catch (InterruptedException e) {
  217. Log.e(TAG, "Failed to wait for thread to finish");
  218. }
  219. }
  220. File file = new File(ocFile.getStoragePath());
  221. int accessMode = ParcelFileDescriptor.parseMode(mode);
  222. boolean isWrite = mode.indexOf('w') != -1;
  223. final OCFile oldFile = ocFile;
  224. final OCFile newFile = ocFile;
  225. if (isWrite) {
  226. try {
  227. Handler handler = new Handler(context.getMainLooper());
  228. return ParcelFileDescriptor.open(file, accessMode, handler, l -> {
  229. RemoteOperationResult result = new SynchronizeFileOperation(newFile, oldFile, account, true,
  230. context)
  231. .execute(client, currentStorageManager);
  232. boolean success = result.isSuccess();
  233. if (!success) {
  234. Log_OC.e(TAG, "Failed to update document with id " + documentId);
  235. }
  236. });
  237. } catch (IOException e) {
  238. throw new FileNotFoundException("Failed to open/edit document with id " + documentId);
  239. }
  240. } else {
  241. return ParcelFileDescriptor.open(file, accessMode);
  242. }
  243. }
  244. private void showToast() {
  245. Handler handler = new Handler(Looper.getMainLooper());
  246. handler.post(() -> Toast.makeText(MainApp.getAppContext(),
  247. R.string.file_not_synced,
  248. Toast.LENGTH_SHORT).show());
  249. }
  250. @Override
  251. public boolean onCreate() {
  252. AndroidInjection.inject(this);
  253. return true;
  254. }
  255. @Override
  256. public AssetFileDescriptor openDocumentThumbnail(String documentId,
  257. Point sizeHint,
  258. CancellationSignal signal)
  259. throws FileNotFoundException {
  260. long docId = Long.parseLong(documentId);
  261. updateCurrentStorageManagerIfNeeded(docId);
  262. OCFile file = currentStorageManager.getFileById(docId);
  263. if (file == null) {
  264. throw new FileNotFoundException("File with id " + documentId + " not found!");
  265. }
  266. Context context = getContext();
  267. if (context == null) {
  268. throw new FileNotFoundException("Context may not be null!");
  269. }
  270. boolean exists = ThumbnailsCacheManager.containsBitmap(ThumbnailsCacheManager.PREFIX_THUMBNAIL
  271. + file.getRemoteId());
  272. if (!exists) {
  273. ThumbnailsCacheManager.generateThumbnailFromOCFile(file);
  274. }
  275. Uri uri = Uri.parse(UriUtils.URI_CONTENT_SCHEME + context.getResources().getString(
  276. R.string.image_cache_provider_authority) + file.getRemotePath());
  277. return context.getContentResolver().openAssetFileDescriptor(uri, "r");
  278. }
  279. @Override
  280. public String renameDocument(String documentId, String displayName) throws FileNotFoundException {
  281. long docId = Long.parseLong(documentId);
  282. updateCurrentStorageManagerIfNeeded(docId);
  283. OCFile file = currentStorageManager.getFileById(docId);
  284. if (file == null) {
  285. throw new FileNotFoundException("File " + documentId + " not found!");
  286. }
  287. RemoteOperationResult result = new RenameFileOperation(file.getRemotePath(), displayName)
  288. .execute(client, currentStorageManager);
  289. if (!result.isSuccess()) {
  290. throw new FileNotFoundException("Failed to rename document with documentId " + documentId + ": " +
  291. result.getException());
  292. }
  293. return null;
  294. }
  295. @Override
  296. public String copyDocument(String sourceDocumentId, String targetParentDocumentId) throws FileNotFoundException {
  297. long sourceId = Long.parseLong(sourceDocumentId);
  298. updateCurrentStorageManagerIfNeeded(sourceId);
  299. OCFile file = currentStorageManager.getFileById(sourceId);
  300. if (file == null) {
  301. throw new FileNotFoundException("File " + sourceDocumentId + " not found!");
  302. }
  303. long targetId = Long.parseLong(targetParentDocumentId);
  304. OCFile targetFolder = currentStorageManager.getFileById(targetId);
  305. if (targetFolder == null) {
  306. throw new FileNotFoundException("File " + targetParentDocumentId + " not found!");
  307. }
  308. RemoteOperationResult result = new CopyFileOperation(file.getRemotePath(), targetFolder.getRemotePath())
  309. .execute(client, currentStorageManager);
  310. if (!result.isSuccess()) {
  311. throw new FileNotFoundException("Failed to copy document with documentId " + sourceDocumentId
  312. + " to " + targetParentDocumentId);
  313. }
  314. Account account = currentStorageManager.getAccount();
  315. RemoteOperationResult updateParent = new RefreshFolderOperation(targetFolder, System.currentTimeMillis(),
  316. false, false, true, currentStorageManager,
  317. account, getContext()).execute(client);
  318. if (!updateParent.isSuccess()) {
  319. throw new FileNotFoundException("Failed to copy document with documentId " + sourceDocumentId
  320. + " to " + targetParentDocumentId);
  321. }
  322. String newPath = targetFolder.getRemotePath() + file.getFileName();
  323. if (file.isFolder()) {
  324. newPath = newPath + "/";
  325. }
  326. OCFile newFile = currentStorageManager.getFileByPath(newPath);
  327. return String.valueOf(newFile.getFileId());
  328. }
  329. @Override
  330. public String moveDocument(String sourceDocumentId, String sourceParentDocumentId, String targetParentDocumentId)
  331. throws FileNotFoundException {
  332. long sourceId = Long.parseLong(sourceDocumentId);
  333. updateCurrentStorageManagerIfNeeded(sourceId);
  334. OCFile file = currentStorageManager.getFileById(sourceId);
  335. if (file == null) {
  336. throw new FileNotFoundException("File " + sourceDocumentId + " not found!");
  337. }
  338. long targetId = Long.parseLong(targetParentDocumentId);
  339. OCFile targetFolder = currentStorageManager.getFileById(targetId);
  340. if (targetFolder == null) {
  341. throw new FileNotFoundException("File " + targetParentDocumentId + " not found!");
  342. }
  343. RemoteOperationResult result = new MoveFileOperation(file.getRemotePath(), targetFolder.getRemotePath())
  344. .execute(client, currentStorageManager);
  345. if (!result.isSuccess()) {
  346. throw new FileNotFoundException("Failed to move document with documentId " + sourceDocumentId
  347. + " to " + targetParentDocumentId);
  348. }
  349. return String.valueOf(file.getFileId());
  350. }
  351. @Override
  352. public Cursor querySearchDocuments(String rootId, String query, String[] projection) {
  353. updateCurrentStorageManagerIfNeeded(rootId);
  354. OCFile root = currentStorageManager.getFileByPath("/");
  355. FileCursor result = new FileCursor(projection);
  356. for (OCFile f : findFiles(root, query)) {
  357. result.addFile(f);
  358. }
  359. return result;
  360. }
  361. @Override
  362. public String createDocument(String documentId, String mimeType, String displayName) throws FileNotFoundException {
  363. long docId = Long.parseLong(documentId);
  364. updateCurrentStorageManagerIfNeeded(docId);
  365. OCFile parent = currentStorageManager.getFileById(docId);
  366. if (parent == null) {
  367. throw new FileNotFoundException("Parent file not found");
  368. }
  369. if ("vnd.android.document/directory".equalsIgnoreCase(mimeType)) {
  370. return createFolder(parent, displayName, documentId);
  371. } else {
  372. return createFile(parent, displayName, documentId);
  373. }
  374. }
  375. private String createFolder(OCFile parent, String displayName, String documentId) throws FileNotFoundException {
  376. CreateFolderOperation createFolderOperation = new CreateFolderOperation(parent.getRemotePath() + displayName
  377. + "/", true);
  378. RemoteOperationResult result = createFolderOperation.execute(client, currentStorageManager);
  379. if (!result.isSuccess()) {
  380. throw new FileNotFoundException("Failed to create document with name " +
  381. displayName + " and documentId " + documentId);
  382. }
  383. String newDirPath = parent.getRemotePath() + displayName + "/";
  384. OCFile newFolder = currentStorageManager.getFileByPath(newDirPath);
  385. return String.valueOf(newFolder.getFileId());
  386. }
  387. private String createFile(OCFile parent, String displayName, String documentId) throws FileNotFoundException {
  388. Context context = getContext();
  389. if (context == null) {
  390. throw new FileNotFoundException("Context may not be null!");
  391. }
  392. Account account = currentStorageManager.getAccount();
  393. // create dummy file
  394. File tempDir = new File(FileStorageUtils.getTemporalPath(account.name));
  395. File emptyFile = new File(tempDir, displayName);
  396. try {
  397. if (!emptyFile.createNewFile()) {
  398. throw new FileNotFoundException("File could not be created");
  399. }
  400. } catch (IOException e) {
  401. throw new FileNotFoundException("File could not be created");
  402. }
  403. FileUploader.UploadRequester requester = new FileUploader.UploadRequester();
  404. requester.uploadNewFile(getContext(), account, new String[]{emptyFile.getAbsolutePath()},
  405. new String[]{parent.getRemotePath() + displayName}, null,
  406. FileUploader.LOCAL_BEHAVIOUR_MOVE, true, UploadFileOperation.CREATED_BY_USER, false,
  407. false);
  408. try {
  409. Thread.sleep(2000);
  410. } catch (InterruptedException e) {
  411. Log_OC.e(TAG, "Thread interruption error");
  412. }
  413. RemoteOperationResult updateParent = new RefreshFolderOperation(parent, System.currentTimeMillis(),
  414. false, false, true, currentStorageManager,
  415. account, getContext()).execute(client);
  416. if (!updateParent.isSuccess()) {
  417. throw new FileNotFoundException("Failed to create document with documentId " + documentId);
  418. }
  419. String newFilePath = parent.getRemotePath() + displayName;
  420. OCFile newFile = currentStorageManager.getFileByPath(newFilePath);
  421. return String.valueOf(newFile.getFileId());
  422. }
  423. @Override
  424. public void deleteDocument(String documentId) throws FileNotFoundException {
  425. long docId = Long.parseLong(documentId);
  426. updateCurrentStorageManagerIfNeeded(docId);
  427. OCFile file = currentStorageManager.getFileById(docId);
  428. if (file == null) {
  429. throw new FileNotFoundException("File " + documentId + " not found!");
  430. }
  431. Account account = currentStorageManager.getAccount();
  432. RemoveFileOperation removeFileOperation = new RemoveFileOperation(file.getRemotePath(), false, account, true,
  433. getContext());
  434. RemoteOperationResult result = removeFileOperation.execute(client, currentStorageManager);
  435. if (!result.isSuccess()) {
  436. throw new FileNotFoundException("Failed to delete document with documentId " + documentId);
  437. }
  438. }
  439. @SuppressLint("LongLogTag")
  440. private void updateCurrentStorageManagerIfNeeded(long docId) {
  441. if (rootIdToStorageManager == null) {
  442. try {
  443. queryRoots(FileCursor.DEFAULT_DOCUMENT_PROJECTION);
  444. } catch (FileNotFoundException e) {
  445. Log.e(TAG, "Failed to query roots");
  446. }
  447. }
  448. if (currentStorageManager == null ||
  449. rootIdToStorageManager.containsKey(docId) && currentStorageManager != rootIdToStorageManager.get(docId)) {
  450. currentStorageManager = rootIdToStorageManager.get(docId);
  451. }
  452. try {
  453. Account account = currentStorageManager.getAccount();
  454. OwnCloudAccount ocAccount = new OwnCloudAccount(account, MainApp.getAppContext());
  455. client = OwnCloudClientManagerFactory.getDefaultSingleton().getClientFor(ocAccount, getContext());
  456. } catch (OperationCanceledException | IOException | AuthenticatorException |
  457. com.owncloud.android.lib.common.accounts.AccountUtils.AccountNotFoundException e) {
  458. Log_OC.e(TAG, "Failed to set client", e);
  459. }
  460. }
  461. private void updateCurrentStorageManagerIfNeeded(String rootId) {
  462. for (FileDataStorageManager data : rootIdToStorageManager.values()) {
  463. if (data.getAccount().name.equals(rootId)) {
  464. currentStorageManager = data;
  465. }
  466. }
  467. }
  468. @SuppressLint("UseSparseArrays")
  469. private void initiateStorageMap() throws FileNotFoundException {
  470. rootIdToStorageManager = new HashMap<>();
  471. Context context = getContext();
  472. if (context == null) {
  473. throw new FileNotFoundException("Context may not be null!");
  474. }
  475. ContentResolver contentResolver = context.getContentResolver();
  476. for (Account account : accountManager.getAccounts()) {
  477. final FileDataStorageManager storageManager = new FileDataStorageManager(account, contentResolver);
  478. final OCFile rootDir = storageManager.getFileByPath("/");
  479. rootIdToStorageManager.put(rootDir.getFileId(), storageManager);
  480. }
  481. }
  482. private boolean waitOrGetCancelled(CancellationSignal cancellationSignal) {
  483. try {
  484. Thread.sleep(1000);
  485. } catch (InterruptedException e) {
  486. return false;
  487. }
  488. return !(cancellationSignal != null && cancellationSignal.isCanceled());
  489. }
  490. List<OCFile> findFiles(OCFile root, String query) {
  491. List<OCFile> result = new ArrayList<>();
  492. for (OCFile f : currentStorageManager.getFolderContent(root, false)) {
  493. if (f.isFolder()) {
  494. result.addAll(findFiles(f, query));
  495. } else if (f.getFileName().contains(query)) {
  496. result.add(f);
  497. }
  498. }
  499. return result;
  500. }
  501. }