diff --git a/WordPress/src/main/java/org/wordpress/android/networking/GlideRequestFactory.kt b/WordPress/src/main/java/org/wordpress/android/networking/GlideRequestFactory.kt index cb5498869106..15e5db1dda99 100644 --- a/WordPress/src/main/java/org/wordpress/android/networking/GlideRequestFactory.kt +++ b/WordPress/src/main/java/org/wordpress/android/networking/GlideRequestFactory.kt @@ -1,14 +1,11 @@ package org.wordpress.android.networking -import android.util.Base64 import com.android.volley.Request import com.android.volley.Request.Priority import com.bumptech.glide.integration.volley.VolleyRequestFactory import com.bumptech.glide.integration.volley.VolleyStreamFetcher import com.bumptech.glide.load.data.DataFetcher.DataCallback -import org.wordpress.android.fluxc.network.HTTPAuthManager -import org.wordpress.android.fluxc.network.UserAgent -import org.wordpress.android.fluxc.network.rest.wpcom.auth.AccessToken +import org.wordpress.android.ui.utils.AuthenticationUtils import org.wordpress.android.util.UrlUtils import org.wordpress.android.util.WPUrlUtils import java.io.InputStream @@ -21,9 +18,7 @@ import javax.inject.Singleton */ @Singleton class GlideRequestFactory @Inject constructor( - private val accessToken: AccessToken, - private val httpAuthManager: HTTPAuthManager, - private val userAgent: UserAgent + private val authenticationUtils: AuthenticationUtils ) : VolleyRequestFactory { override fun create( url: String, @@ -40,20 +35,10 @@ class GlideRequestFactory @Inject constructor( } private fun addAuthHeaders(url: String, currentHeaders: Map): MutableMap { + val authenticationHeaders = authenticationUtils.getAuthHeaders(url) val headers = currentHeaders.toMutableMap() - headers["User-Agent"] = userAgent.userAgent - if (WPUrlUtils.safeToAddWordPressComAuthToken(url)) { - if (accessToken.exists()) { - headers["Authorization"] = "Bearer " + accessToken.get() - } - } else { - // Check if we had HTTP Auth credentials for the root url - val httpAuthModel = httpAuthManager.getHTTPAuthModel(url) - if (httpAuthModel != null) { - val creds = String.format("%s:%s", httpAuthModel.username, httpAuthModel.password) - val auth = "Basic " + Base64.encodeToString(creds.toByteArray(), Base64.NO_WRAP) - headers["Authorization"] = auth - } + authenticationHeaders.entries.forEach { (key, value) -> + headers[key] = value } return headers } diff --git a/WordPress/src/main/java/org/wordpress/android/ui/reader/ReaderFileDownloadManager.kt b/WordPress/src/main/java/org/wordpress/android/ui/reader/ReaderFileDownloadManager.kt new file mode 100644 index 000000000000..0d3674a12010 --- /dev/null +++ b/WordPress/src/main/java/org/wordpress/android/ui/reader/ReaderFileDownloadManager.kt @@ -0,0 +1,57 @@ +package org.wordpress.android.ui.reader + +import android.app.DownloadManager +import android.content.BroadcastReceiver +import android.content.Context +import android.content.Intent +import android.os.Environment +import org.wordpress.android.ui.utils.AuthenticationUtils +import org.wordpress.android.ui.utils.DownloadManagerWrapper +import javax.inject.Inject + +class ReaderFileDownloadManager +@Inject constructor( + private val authenticationUtils: AuthenticationUtils, + private val downloadManager: DownloadManagerWrapper +) : BroadcastReceiver() { + override fun onReceive(context: Context, intent: Intent) { + if (DownloadManager.ACTION_DOWNLOAD_COMPLETE == intent.action) { + val downloadId = intent.getLongExtra( + DownloadManager.EXTRA_DOWNLOAD_ID, 0 + ) + openDownloadedAttachment(context, downloadId) + } + } + + fun downloadFile(fileUrl: String) { + val request = downloadManager.buildRequest(fileUrl) + + for (entry in authenticationUtils.getAuthHeaders(fileUrl).entries) { + request.addRequestHeader(entry.key, entry.value) + } + + val fileName = downloadManager.guessUrl(fileUrl) + request.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, fileName) + request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED) + request.setMimeType(downloadManager.getMimeType(fileUrl)) + request.setTitle(fileName) + request.allowScanningByMediaScanner() + + downloadManager.enqueue(request) + } + + private fun openDownloadedAttachment(context: Context, downloadId: Long) { + val query = downloadManager.buildQuery() + query.setFilterById(downloadId) + val cursor = downloadManager.query(query) + if (cursor.moveToFirst()) { + val downloadStatus = cursor.getInt(cursor.getColumnIndex(DownloadManager.COLUMN_STATUS)) + val downloadLocalUri = cursor.getString(cursor.getColumnIndex(DownloadManager.COLUMN_LOCAL_URI)) + val downloadMimeType = cursor.getString(cursor.getColumnIndex(DownloadManager.COLUMN_MEDIA_TYPE)) + if (downloadStatus == DownloadManager.STATUS_SUCCESSFUL && downloadLocalUri != null) { + downloadManager.openDownloadedAttachment(context, downloadLocalUri, downloadMimeType) + } + } + cursor.close() + } +} diff --git a/WordPress/src/main/java/org/wordpress/android/ui/reader/ReaderPostDetailFragment.java b/WordPress/src/main/java/org/wordpress/android/ui/reader/ReaderPostDetailFragment.java index c0d7b3e16246..3d67802ee2eb 100644 --- a/WordPress/src/main/java/org/wordpress/android/ui/reader/ReaderPostDetailFragment.java +++ b/WordPress/src/main/java/org/wordpress/android/ui/reader/ReaderPostDetailFragment.java @@ -3,12 +3,11 @@ import android.app.Activity; import android.app.DownloadManager; import android.content.Intent; +import android.content.IntentFilter; import android.content.pm.PackageManager; import android.graphics.Rect; -import android.net.Uri; import android.os.AsyncTask; import android.os.Bundle; -import android.os.Environment; import android.text.Html; import android.text.TextUtils; import android.view.LayoutInflater; @@ -17,7 +16,6 @@ import android.view.MenuItem; import android.view.View; import android.view.ViewGroup; -import android.webkit.URLUtil; import android.webkit.WebView; import android.widget.ProgressBar; import android.widget.TextView; @@ -76,6 +74,7 @@ import org.wordpress.android.ui.reader.views.ReaderWebView.ReaderCustomViewListener; import org.wordpress.android.ui.reader.views.ReaderWebView.ReaderWebViewPageFinishedListener; import org.wordpress.android.ui.reader.views.ReaderWebView.ReaderWebViewUrlClickListener; +import org.wordpress.android.ui.utils.AuthenticationUtils; import org.wordpress.android.util.AniUtils; import org.wordpress.android.util.AppLog; import org.wordpress.android.util.AppLog.T; @@ -98,7 +97,6 @@ import javax.inject.Inject; -import static android.content.Context.DOWNLOAD_SERVICE; import static org.wordpress.android.fluxc.generated.AccountActionBuilder.newUpdateSubscriptionNotificationPostAction; import static org.wordpress.android.util.WPPermissionUtils.READER_FILE_DOWNLOAD_PERMISSION_REQUEST_CODE; import static org.wordpress.android.util.WPSwipeToRefreshHelper.buildSwipeToRefreshHelper; @@ -160,6 +158,8 @@ public class ReaderPostDetailFragment extends Fragment @Inject AccountStore mAccountStore; @Inject SiteStore mSiteStore; @Inject Dispatcher mDispatcher; + @Inject AuthenticationUtils mAuthenticationUtils; + @Inject ReaderFileDownloadManager mReaderFileDownloadManager; public static ReaderPostDetailFragment newInstance(long blogId, long postId) { return newInstance(false, blogId, postId, null, 0, false, null, null, false); @@ -431,6 +431,11 @@ public void onStart() { super.onStart(); mDispatcher.register(this); EventBus.getDefault().register(this); + FragmentActivity activity = getActivity(); + if (activity != null) { + activity.registerReceiver(mReaderFileDownloadManager, + new IntentFilter(DownloadManager.ACTION_DOWNLOAD_COMPLETE)); + } } @Override @@ -438,6 +443,10 @@ public void onStop() { super.onStop(); mDispatcher.unregister(this); EventBus.getDefault().unregister(this); + FragmentActivity activity = getActivity(); + if (activity != null) { + activity.unregisterReceiver(mReaderFileDownloadManager); + } } /* @@ -1351,8 +1360,12 @@ public boolean onUrlClick(String url) { return true; } - OpenUrlType openUrlType = shouldOpenExternal(url) ? OpenUrlType.EXTERNAL : OpenUrlType.INTERNAL; - ReaderActivityLauncher.openUrl(getActivity(), url, openUrlType); + if (isFile(url)) { + onFileDownloadClick(url); + } else { + OpenUrlType openUrlType = shouldOpenExternal(url) ? OpenUrlType.EXTERNAL : OpenUrlType.INTERNAL; + ReaderActivityLauncher.openUrl(getActivity(), url, openUrlType); + } return true; } @@ -1368,12 +1381,17 @@ private boolean shouldOpenExternal(String url) { // if the mime type starts with "application" open it externally - this will either // open it in the associated app or the default browser (which will enable the user // to download it) + if (isFile(url)) return true; + + // open all other urls using an AuthenticatedWebViewActivity + return false; + } + + private boolean isFile(String url) { String mimeType = UrlUtils.getUrlMimeType(url); if (mimeType != null && mimeType.startsWith("application")) { return true; } - - // open all other urls using an AuthenticatedWebViewActivity return false; } @@ -1388,7 +1406,7 @@ public boolean onFileDownloadClick(String fileUrl) { if (activity != null && fileUrl != null && PermissionUtils.checkAndRequestStoragePermission(this, READER_FILE_DOWNLOAD_PERMISSION_REQUEST_CODE)) { - downloadFile(fileUrl, activity); + mReaderFileDownloadManager.downloadFile(fileUrl); return true; } else { mFileForDownload = fileUrl; @@ -1403,23 +1421,13 @@ public void onRequestPermissionsResult(int requestCode, @NonNull String[] permis if (activity != null && requestCode == READER_FILE_DOWNLOAD_PERMISSION_REQUEST_CODE && (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED)) { - downloadFile(mFileForDownload, activity); + mReaderFileDownloadManager.downloadFile(mFileForDownload); mFileForDownload = null; } else { mFileForDownload = null; } } - private void downloadFile(String fileUrl, FragmentActivity activity) { - DownloadManager.Request r = new DownloadManager.Request(Uri.parse(fileUrl)); - String fileName = URLUtil.guessUrl(fileUrl); - r.setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, fileName); - r.allowScanningByMediaScanner(); - r.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED); - DownloadManager dm = (DownloadManager) activity.getSystemService(DOWNLOAD_SERVICE); - dm.enqueue(r); - } - private ActionBar getActionBar() { if (isAdded() && getActivity() instanceof AppCompatActivity) { return ((AppCompatActivity) getActivity()).getSupportActionBar(); diff --git a/WordPress/src/main/java/org/wordpress/android/ui/utils/AuthenticationUtils.kt b/WordPress/src/main/java/org/wordpress/android/ui/utils/AuthenticationUtils.kt new file mode 100644 index 000000000000..d9c2b95135d0 --- /dev/null +++ b/WordPress/src/main/java/org/wordpress/android/ui/utils/AuthenticationUtils.kt @@ -0,0 +1,36 @@ +package org.wordpress.android.ui.utils + +import android.util.Base64 +import org.wordpress.android.fluxc.network.HTTPAuthManager +import org.wordpress.android.fluxc.network.UserAgent +import org.wordpress.android.fluxc.network.rest.wpcom.auth.AccessToken +import org.wordpress.android.util.WPUrlUtils +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class AuthenticationUtils +@Inject constructor( + private val accessToken: AccessToken, + private val httpAuthManager: HTTPAuthManager, + private val userAgent: UserAgent +) { + fun getAuthHeaders(url: String): Map { + val headers = mutableMapOf() + headers["User-Agent"] = userAgent.userAgent + if (WPUrlUtils.safeToAddWordPressComAuthToken(url)) { + if (accessToken.exists()) { + headers["Authorization"] = "Bearer " + accessToken.get() + } + } else { + // Check if we had HTTP Auth credentials for the root url + val httpAuthModel = httpAuthManager.getHTTPAuthModel(url) + if (httpAuthModel != null) { + val creds = String.format("%s:%s", httpAuthModel.username, httpAuthModel.password) + val auth = "Basic " + Base64.encodeToString(creds.toByteArray(), Base64.NO_WRAP) + headers["Authorization"] = auth + } + } + return headers + } +} diff --git a/WordPress/src/main/java/org/wordpress/android/ui/utils/DownloadManagerWrapper.kt b/WordPress/src/main/java/org/wordpress/android/ui/utils/DownloadManagerWrapper.kt new file mode 100644 index 000000000000..1a9ff09979cf --- /dev/null +++ b/WordPress/src/main/java/org/wordpress/android/ui/utils/DownloadManagerWrapper.kt @@ -0,0 +1,78 @@ +package org.wordpress.android.ui.utils + +import android.app.DownloadManager +import android.app.DownloadManager.Query +import android.app.DownloadManager.Request +import android.content.ActivityNotFoundException +import android.content.ContentResolver +import android.content.Context +import android.content.Intent +import android.database.Cursor +import android.net.Uri +import android.webkit.MimeTypeMap +import android.webkit.URLUtil +import androidx.core.content.FileProvider +import org.wordpress.android.BuildConfig +import org.wordpress.android.util.AppLog +import org.wordpress.android.util.AppLog.T +import java.io.File +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class DownloadManagerWrapper +@Inject constructor(private val context: Context) { + fun enqueue(request: Request): Long = downloadManager().enqueue(request) + + fun buildRequest(fileUrl: String) = Request(Uri.parse(fileUrl)) + + fun query(query: Query): Cursor = downloadManager().query(query) + + fun buildQuery() = Query() + + fun guessUrl(fileUrl: String): String = URLUtil.guessUrl(fileUrl) + + fun getMimeType(url: String): String? { + var type: String? = null + val extension = MimeTypeMap.getFileExtensionFromUrl(url) + if (extension != null) { + val mime = MimeTypeMap.getSingleton() + type = mime.getMimeTypeFromExtension(extension) + } + return type + } + + private fun toPublicUri(fileUrl: String): Uri { + val fileUri = Uri.parse(fileUrl) + return if (ContentResolver.SCHEME_FILE == fileUri.scheme) { + val file = File(fileUri.path) + FileProvider.getUriForFile( + context, + "${BuildConfig.APPLICATION_ID}.provider", + file + ) + } else { + fileUri + } + } + + fun openDownloadedAttachment( + context: Context, + fileUrl: String, + attachmentMimeType: String + ) { + val attachmentUri = toPublicUri(fileUrl) + + val openAttachmentIntent = Intent(Intent.ACTION_VIEW) + openAttachmentIntent.setDataAndType(attachmentUri, attachmentMimeType) + openAttachmentIntent.flags = Intent.FLAG_GRANT_READ_URI_PERMISSION or Intent.FLAG_ACTIVITY_NEW_TASK + try { + context.startActivity(openAttachmentIntent) + } catch (e: ActivityNotFoundException) { + AppLog.e(T.READER, "No browser found on the device: ${e.message}") + } + } + + private fun downloadManager() = + (context.getSystemService(Context.DOWNLOAD_SERVICE) as DownloadManager) +} diff --git a/WordPress/src/test/java/org/wordpress/android/ui/reader/ReaderFileDownloadManagerTest.kt b/WordPress/src/test/java/org/wordpress/android/ui/reader/ReaderFileDownloadManagerTest.kt new file mode 100644 index 000000000000..c47833e66c36 --- /dev/null +++ b/WordPress/src/test/java/org/wordpress/android/ui/reader/ReaderFileDownloadManagerTest.kt @@ -0,0 +1,88 @@ +package org.wordpress.android.ui.reader + +import android.app.DownloadManager +import android.content.Context +import android.content.Intent +import android.database.Cursor +import android.os.Environment +import com.nhaarman.mockitokotlin2.KArgumentCaptor +import com.nhaarman.mockitokotlin2.any +import com.nhaarman.mockitokotlin2.argumentCaptor +import com.nhaarman.mockitokotlin2.verify +import com.nhaarman.mockitokotlin2.whenever +import org.junit.Before +import org.junit.Test +import org.junit.runner.RunWith +import org.mockito.Mock +import org.mockito.junit.MockitoJUnitRunner +import org.wordpress.android.ui.utils.AuthenticationUtils +import org.wordpress.android.ui.utils.DownloadManagerWrapper + +@RunWith(MockitoJUnitRunner::class) +class ReaderFileDownloadManagerTest { + @Mock lateinit var authenticationUtils: AuthenticationUtils + @Mock lateinit var downloadManager: DownloadManagerWrapper + @Mock lateinit var request: DownloadManager.Request + @Mock lateinit var query: DownloadManager.Query + @Mock lateinit var context: Context + @Mock lateinit var intent: Intent + @Mock lateinit var cursor: Cursor + private lateinit var readerFileDownloadManager: ReaderFileDownloadManager + private lateinit var intentCaptor: KArgumentCaptor + @Before + fun setUp() { + readerFileDownloadManager = ReaderFileDownloadManager(authenticationUtils, downloadManager) + intentCaptor = argumentCaptor() + } + + @Test + fun `enqueues file for download`() { + val url = "http://wordpress.com/file_name.pdf" + val header = "Authentication" + val headerValue = "token123" + val fileName = "file_name.pdf" + val mimeType = "application/pdf" + whenever(authenticationUtils.getAuthHeaders(url)).thenReturn(mapOf(header to headerValue)) + whenever(downloadManager.buildRequest(url)).thenReturn(request) + whenever(downloadManager.guessUrl(url)).thenReturn(fileName) + whenever(downloadManager.getMimeType(url)).thenReturn(mimeType) + + readerFileDownloadManager.downloadFile(url) + + verify(downloadManager).enqueue(request) + + verify(request).addRequestHeader(header, headerValue) + verify(request).setDestinationInExternalPublicDir(Environment.DIRECTORY_DOWNLOADS, fileName) + verify(request).allowScanningByMediaScanner() + verify(request).setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED) + verify(request).setMimeType(mimeType) + verify(request).setTitle(fileName) + } + + @Test + fun `opens file on download complete`() { + val downloadId = 1L + whenever(intent.action).thenReturn(DownloadManager.ACTION_DOWNLOAD_COMPLETE) + whenever(intent.getLongExtra(DownloadManager.EXTRA_DOWNLOAD_ID, 0)).thenReturn(downloadId) + whenever(downloadManager.buildQuery()).thenReturn(query) + whenever(downloadManager.query(any())).thenReturn(cursor) + whenever(cursor.moveToFirst()).thenReturn(true) + val statusColumnIndex = 1 + val localUriColumnIndex = 2 + val mediaTypeColumnIndex = 3 + whenever(cursor.getColumnIndex(DownloadManager.COLUMN_STATUS)).thenReturn(statusColumnIndex) + whenever(cursor.getColumnIndex(DownloadManager.COLUMN_LOCAL_URI)).thenReturn(localUriColumnIndex) + whenever(cursor.getColumnIndex(DownloadManager.COLUMN_MEDIA_TYPE)).thenReturn(mediaTypeColumnIndex) + whenever(cursor.getInt(statusColumnIndex)).thenReturn(DownloadManager.STATUS_SUCCESSFUL) + val localUri = "/sdcard/file_name.pdf" + val mediaType = "pdf" + whenever(cursor.getString(localUriColumnIndex)).thenReturn(localUri) + whenever(cursor.getString(mediaTypeColumnIndex)).thenReturn(mediaType) + + readerFileDownloadManager.onReceive(context, intent) + + verify(downloadManager).query(any()) + verify(cursor).close() + verify(downloadManager).openDownloadedAttachment(context, localUri, mediaType) + } +}