Skip to content
This repository has been archived by the owner on Jan 14, 2021. It is now read-only.

Are permissions really needed on Android? #112

Closed
ChristianKleineidam opened this issue Nov 19, 2018 · 15 comments · Fixed by #114
Closed

Are permissions really needed on Android? #112

ChristianKleineidam opened this issue Nov 19, 2018 · 15 comments · Fixed by #114

Comments

@ChristianKleineidam
Copy link

I created a pull request that requested the permission on run-time but I just found a StackOverflow answer that says:

"Does ACTION_GET_CONTENT with setType("image/*) not need a permission to read external storage?" -- correct. On Android 4.3 and older, you do not need a permission to read external storage in general. On Android 4.4+, ACTION_GET_CONTENT should return a Uri with a content scheme, not a file scheme, and you have temporary rights to access the content identified by that Uri.

If that's the case, maybe the FilePicker Plugin can work completely without permissions on Android?

@vividos
Copy link
Collaborator

vividos commented Nov 20, 2018

Well, you may not need a permission to read external storage, but often the picked files come as a content:// link, e.g. the Google Drive or OneDrive apps return that. For content:// Uris you need to use Android's ContentResolver to access the stream, as there are no actual physical files involved.

Technically, picking a file also doesn't need the permission, but the developer will immediately access the file using FileData.DataBytes or FileData.GetStream(), and then the permission is needed. So it's a user experience decision if you ask for permission first and then start picking, or if you start picking and then ask for the permission. With the current code the plugin user has the choice, and the sample just asks for permission first.

Your PR #109 puts asking for permission first, which I think is a good thing, since we got some issue tickets complaining about why picking throws an exception that permissions are not set. Do you now reconsider? I was going to merge the PR today...

@ChristianKleineidam
Copy link
Author

I think my PR is a clear improvement over the status quo. I however get the feeling that the expection that exists is unnecessary:

It gets currently thrown in Plugin.FilePicker.Android/FilePickerImplementation.cs via:

        if (this.context.PackageManager.CheckPermission(
            Android.Manifest.Permission.ReadExternalStorage,
            this.context.PackageName) == Android.Content.PM.Permission.Denied)
        {
            throw new InvalidOperationException("Android permission READ_EXTERNAL_STORAGE is missing or was denied by user");
        }

It might be that we could simply remove the portion of the code that throws the expection and be good.

1 similar comment
@ChristianKleineidam
Copy link
Author

I think my PR is a clear improvement over the status quo. I however get the feeling that the expection that exists is unnecessary:

It gets currently thrown in Plugin.FilePicker.Android/FilePickerImplementation.cs via:

        if (this.context.PackageManager.CheckPermission(
            Android.Manifest.Permission.ReadExternalStorage,
            this.context.PackageName) == Android.Content.PM.Permission.Denied)
        {
            throw new InvalidOperationException("Android permission READ_EXTERNAL_STORAGE is missing or was denied by user");
        }

It might be that we could simply remove the portion of the code that throws the expection and be good.

@vividos
Copy link
Collaborator

vividos commented Nov 20, 2018

Hmm but how do you communicate to the developer that the permission has to be added to the Android manifest? I agree on the case when the user refused granting permission; then we could just return null. Error handling is always need when calling PickFile() - who knows what Google does require in Android 10 :-D

@ChristianKleineidam
Copy link
Author

I don't think that's something that's needed to be communicated to the developer. If the thing I quoted above is correct it seems like the permission doesn't have to be added on Android 4.3 and older to the manifest at all (only 3% of the devices that are out there run something older).

I would expect that Android manages to throw it's own error message when running on 4.2 and the user actually needs the permission.

For many people who use that plugin this error message is the only reason they currently have to have the permission and they will be able to remove the permission from their manifest when that code is gone (or otherwise only gets displayed below 4.3).

@vividos
Copy link
Collaborator

vividos commented Nov 20, 2018

I put in the check for the permission so that the user gets a clear message with a defined exception text that is also mentioned in the README.md, where it can be found in the Troubleshooting sections.

I'll merge the PR #109 anyway and we'll see how users accept it. What do you think?

@vividos
Copy link
Collaborator

vividos commented Nov 20, 2018

I'm not entirely satisfied with the exception thrown on API <= 23 and user forgot to add READ_EXTERNAL_STORAGE permission. What do you think about that: We add the permission automatically for the user, just like the MediaPicker plugin from James Montemagno does it:
https://github.com/jamesmontemagno/MediaPlugin/blob/master/src/Media.Plugin/Android/ManifestInfo.cs

@ChristianKleineidam
Copy link
Author

I think a sizeable portion of the developers who use the plugin won't release versions for below API level 18 (Android 4.3) and it seems that above that point the permission is not required.

When a user downloads an App there a benefit on asking the user for less permissions and thus I think the developers who don't develop for <18 would prefer the permission not to be there. It's analog to how the Windows UWP plugin of the FilePicker doesn't need any permission.

Is it possible to do different assembly based on the API level?

@vividos
Copy link
Collaborator

vividos commented Nov 21, 2018

Well, as soon as your use picks a file that is on external storage, you need the permission, and you can't add it dynamically, it must be present in the manifest. Starting from Android 6.0 you have to ask the user at runtime to grant the permission, too. So I don't know how you would handle the case where you didn't add the permission but the user picked a file on external storage.
The question is if the plugin NuGet adds the permission or the adds it.

@ChristianKleineidam
Copy link
Author

ChristianKleineidam commented Nov 22, 2018

It seems to me that you need the permission for reading files with the file://-scheme but not files with the content://-scheme.
Starting Android 4.3 (API level 18) the picked file gets passed with the content://-scheme and thus no permission is needed. If you think there still a permisssion needed, where do you think that need comes from?

@vividos
Copy link
Collaborator

vividos commented Nov 22, 2018

Ok maybe it's the other way around. As soon as the user picks a file on an actual storage, e.g. external SD card, the app needs to have the permission. How do you handle this?

@ChristianKleineidam
Copy link
Author

Let's say this plugin is installed on AppA. When the intent with Intent.ACTION_GET_CONTENT is send another app will listen to the intent AppB.

AppB will usually a system provided app but it's also plausible that another app listens to the intent. AppB needs the permission to access storage. AppB implements a FileProvider that allows it to pass a reference to the file with the content://-scheme as an answer to the intent (and a flag that provides the permission to access the file).

I don't think whether or not the file is located on an external SD or on internal storage has any bearing for the ability of AppB to pass the reference to the file via the content://-scheme.

Only on devices under Android 4.4 (API level 19) the file that gets passed back might be under the file://-scheme and thus it's only necessary for those to have the permission.

@vividos
Copy link
Collaborator

vividos commented Nov 22, 2018

I just checked with the project's sample app and I picked a file from my download folder. I got back a part to storage, not a content:// Uri. Now what? In order for the file to be read by the sample app, the app must have the permission to access storage.

@ChristianKleineidam
Copy link
Author

Okay, then what's written in the linked post feels strange. =It seems to make sense to have the permission. Having it via in the way of the MediaPlugin might make it more user-friendly.

@vividos
Copy link
Collaborator

vividos commented Nov 22, 2018

I just added some changes as we discussed. Available in NuGet package 2.0.133-beta. Closing this issue then.

@vividos vividos closed this as completed Nov 22, 2018
jfversluis pushed a commit that referenced this issue Nov 26, 2018
* Added support for file types across platforms (#26)

* Added support for file types across platforms

- Added per platform file type support, so you can specify what kind of files you want the picker to support
- Updated iOS's picker to use the non-deprecated `UIDocumentPickerViewController`
- Minor tweaks, such as package updates

* Fixed an issue with the iOS implementation, where the returned path wasn't properly formatted/absolute, which in turn meant that the path wasn't valid as is.

* A few iOS specific improvements

* Get latest changes to master (#51)

* pull master into develop before merging back (#47)

* Resolved issues with referencing from NuGet package (#29)

Simplified the nuspec file to work with wildcards and fixed macOS assembly name.
Also added development NuGet feed to readme along with some other tweaks

* Project fixes (#35)

* fixed paths to NuGet packages folder; always relative to the .sln file

* converted samples solution file to VS 2017 and fixed "Deploy" flag for Android app

* ignore Android Resource designer generated files

# Conflicts:
#	.gitignore

* changes automatically applied by VS 2017 when opening iOS project file

* Added issue template

* fixes crash in Xamarin.Forms sample when user doesn't select file; also added example code in README file (#50)

* fixed picking files from the download folder; on newer devices the document ID may not be a number, but the real filename prefixed with "raw:" (#49)

* Updated dependencies

* Revert "Fixed Path" (#65)



* Revert "Fixed Path (#54)"

* when opening files with the sample app, ignore if filename extension is uppercase or lowercase

* removed unused method DidPickDocumentPicker() in iOS implementation

* added troubleshooting chapter to readme with common errors and their solutions

* updated sample projects to use PackageReference to reference NuGet packages

* removed obsolete project property AndroidUseLatestPlatformSdk and set target framework to Android 9.0 (API 28)

* updated Plugin.FilePicker.UWP to use PackageReference

* removed obsolete property AndroidUseLatestPlatformSdk and target Android 9.0, in Plugin.FilePicker.Android project

* updated sample projects to use Xamarin.Forms 3.3.0 NuGet package

* updated UWP project to latest stable UWP NuGet package

* fixed Android sample project by re-adding Xamarin.Forms NuGet package

* added UWP sample project

* find correct view controller to present document picker, by finding currently presented view controller

* Android: return from task with Exception that was thrown during file picking

before that, the caller only got a null FileData object without knowing what happened

* Android: also pass exception to task when setting up picking

* iOS: pass exceptions in DocumentPicker handler to picker task

* renamed Forms project folder to a shorter name and renamed the only page to MainPage

* added try-catch to call to PickFile() in order to display exceptions from picking

* delete Resource.designer.cs file from repository that is being regenerated by the compiler

* converted Xamarin.Forms sample project to use Project Sdk style .csproj proect file

* added second button to samples to pick image files

* added check if Android permission was granted

* fix android mime types implementation (#85)

(cherry picked from commit 04a5292)

* added explanation for requesting permissions on Android (#89)

* fixed exception when trying to pick a downloaded file; the Android DownloadManager won't return an actual file path; downloaded files can only be resolved using ContentResolver (#86)

* added solution folders to organize projects

* added UTType.Image file types value for picking on iOS

* Forms sample project can use .NET Standard 1.0; no need for version 2.0, which might not be installed

* added README.md for samples folder

* added Plugin.Permissions NuGet package to sample project, in order to check for permissions

* added checking Storage permission when on Android device

* iOS: simplified getting filename from pathname

* iOS: fixed getting pathname from FileUrl (fix of PR #54 that was reverted in PR #65); tested on iOS 11 device

* added documentation about FileData class to README

* Fix FileName for Mac

FileName on Mac returned the path instead.

* Fix OpenFile for Mac

* fixed getting path from OneDrive (#79)

the content provider only returned a relative filename in the data column, so it's better to use the content:// uri to access the file

* fixed indentation of release notes text in .nuspec - it should look nice on the nuget.org page now

* Android: file bytes are not read directly after picking, but only when FileData.DataArray is accessed or FileData.GetStream() is called; this fixes accessing large files that wouldn't fit in the device's memory, e.g. videos (#38)

* iOS: also removed getting data bytes when it's not used in FilePickerEventArgs

* added a troubleshooting section to explain why picked files on iOS may be gone after some time (#87, #74)

* added SourceLink integration, see https://github.com/dotnet/sourcelink

* added documentation for method PickFile(), in order to document how to use allowedTypes (#59)

* Revert "added SourceLink integration, see https://github.com/dotnet/sourcelink"

This reverts commit 1ca5796.

* documented classes in Plugin.FilePicker.Abstractions and fixed line endings

* added WPF sample project (#3)

* implemented FilePicker for WPF (#64)

* added detailed platforms where FilePicker can be used, in description (appears in NuGet package manager) and releaseNotes (appears on NuGet homepage)

* added more designed generated files to ignore

* forgot to add FilePicker WPF implementation to main solution

* fixed OpenFile() call with relative filename that uses wrong parent directory (#71)

* marked ReadFully() with Obsolete attribute; will be made internal in 2.1

* samples: added specifying file types for UWP platform

* added or corrected documentation

* Android: fixed stack overflow when calling OpenFile() with FileData argument

* marked FileData as sealed; no use deriving from FileData

* reformatting C# files; no real code changes

- formatted files with Visual Studio editor to fix spaces and line endings
- added xml documentation
- added this. for accessing fields and properties and for calling methods
- sorted and removed unused usings

* changed sample solution configuration so that Any CPU also compiles on UWP x86 and targets iOS iPhone

* updated Resource.designer.cs file for Forms sample

* removed property FileByte from internal class FilePickerEventArgs, as it is never used

* using Intent Extra constant for passing allowed types to FilePickerActivity, instead of using nameof()

* added pull request template, adapted from the Xamarin.Forms github project

* When Android doesn't have Permission.ReadExternalStorage the App will now ask the user to grant the permission on API level >=23

* Added necessary imports

* Added requested fix of inverted if's to get the right action when permission is granted

* Added comment for REQUEST_STORAGE

* Changed variable to PascalCase

* cleaning up code from PR #109

* Revert "added checking Storage permission when on Android device"

This reverts commit 779c5a1.

* Revert "added Plugin.Permissions NuGet package to sample project, in order to check for permissions"

This reverts commit e81ed29.

* removed compiler warnings in Mac implementation

* permission READ_EXTERNAL_STORAGE is automatically added to the Android app that uses this plugin; no need to add it to AndroidManifest.xml now (#112)

* removed READ_EXTERNAL_STORE permission from sample project
jfversluis pushed a commit that referenced this issue Nov 28, 2018
* Added support for file types across platforms (#26)

* Added support for file types across platforms

- Added per platform file type support, so you can specify what kind of files you want the picker to support
- Updated iOS's picker to use the non-deprecated `UIDocumentPickerViewController`
- Minor tweaks, such as package updates

* Fixed an issue with the iOS implementation, where the returned path wasn't properly formatted/absolute, which in turn meant that the path wasn't valid as is.

* A few iOS specific improvements

* Get latest changes to master (#51)

* pull master into develop before merging back (#47)

* Resolved issues with referencing from NuGet package (#29)

Simplified the nuspec file to work with wildcards and fixed macOS assembly name.
Also added development NuGet feed to readme along with some other tweaks

* Project fixes (#35)

* fixed paths to NuGet packages folder; always relative to the .sln file

* converted samples solution file to VS 2017 and fixed "Deploy" flag for Android app

* ignore Android Resource designer generated files

# Conflicts:
#	.gitignore

* changes automatically applied by VS 2017 when opening iOS project file

* Added issue template

* fixes crash in Xamarin.Forms sample when user doesn't select file; also added example code in README file (#50)

* fixed picking files from the download folder; on newer devices the document ID may not be a number, but the real filename prefixed with "raw:" (#49)

* Updated dependencies

* Revert "Fixed Path" (#65)



* Revert "Fixed Path (#54)"

* when opening files with the sample app, ignore if filename extension is uppercase or lowercase

* removed unused method DidPickDocumentPicker() in iOS implementation

* added troubleshooting chapter to readme with common errors and their solutions

* updated sample projects to use PackageReference to reference NuGet packages

* removed obsolete project property AndroidUseLatestPlatformSdk and set target framework to Android 9.0 (API 28)

* updated Plugin.FilePicker.UWP to use PackageReference

* removed obsolete property AndroidUseLatestPlatformSdk and target Android 9.0, in Plugin.FilePicker.Android project

* updated sample projects to use Xamarin.Forms 3.3.0 NuGet package

* updated UWP project to latest stable UWP NuGet package

* fixed Android sample project by re-adding Xamarin.Forms NuGet package

* added UWP sample project

* find correct view controller to present document picker, by finding currently presented view controller

* Android: return from task with Exception that was thrown during file picking

before that, the caller only got a null FileData object without knowing what happened

* Android: also pass exception to task when setting up picking

* iOS: pass exceptions in DocumentPicker handler to picker task

* renamed Forms project folder to a shorter name and renamed the only page to MainPage

* added try-catch to call to PickFile() in order to display exceptions from picking

* delete Resource.designer.cs file from repository that is being regenerated by the compiler

* converted Xamarin.Forms sample project to use Project Sdk style .csproj proect file

* added second button to samples to pick image files

* added check if Android permission was granted

* fix android mime types implementation (#85)

(cherry picked from commit 04a5292)

* added explanation for requesting permissions on Android (#89)

* fixed exception when trying to pick a downloaded file; the Android DownloadManager won't return an actual file path; downloaded files can only be resolved using ContentResolver (#86)

* added solution folders to organize projects

* added UTType.Image file types value for picking on iOS

* Forms sample project can use .NET Standard 1.0; no need for version 2.0, which might not be installed

* added README.md for samples folder

* added Plugin.Permissions NuGet package to sample project, in order to check for permissions

* added checking Storage permission when on Android device

* iOS: simplified getting filename from pathname

* iOS: fixed getting pathname from FileUrl (fix of PR #54 that was reverted in PR #65); tested on iOS 11 device

* added documentation about FileData class to README

* Fix FileName for Mac

FileName on Mac returned the path instead.

* Fix OpenFile for Mac

* fixed getting path from OneDrive (#79)

the content provider only returned a relative filename in the data column, so it's better to use the content:// uri to access the file

* fixed indentation of release notes text in .nuspec - it should look nice on the nuget.org page now

* Android: file bytes are not read directly after picking, but only when FileData.DataArray is accessed or FileData.GetStream() is called; this fixes accessing large files that wouldn't fit in the device's memory, e.g. videos (#38)

* iOS: also removed getting data bytes when it's not used in FilePickerEventArgs

* added a troubleshooting section to explain why picked files on iOS may be gone after some time (#87, #74)

* added SourceLink integration, see https://github.com/dotnet/sourcelink

* added documentation for method PickFile(), in order to document how to use allowedTypes (#59)

* Revert "added SourceLink integration, see https://github.com/dotnet/sourcelink"

This reverts commit 1ca5796.

* documented classes in Plugin.FilePicker.Abstractions and fixed line endings

* added WPF sample project (#3)

* implemented FilePicker for WPF (#64)

* added detailed platforms where FilePicker can be used, in description (appears in NuGet package manager) and releaseNotes (appears on NuGet homepage)

* added more designed generated files to ignore

* forgot to add FilePicker WPF implementation to main solution

* fixed OpenFile() call with relative filename that uses wrong parent directory (#71)

* marked ReadFully() with Obsolete attribute; will be made internal in 2.1

* samples: added specifying file types for UWP platform

* added or corrected documentation

* Android: fixed stack overflow when calling OpenFile() with FileData argument

* marked FileData as sealed; no use deriving from FileData

* reformatting C# files; no real code changes

- formatted files with Visual Studio editor to fix spaces and line endings
- added xml documentation
- added this. for accessing fields and properties and for calling methods
- sorted and removed unused usings

* changed sample solution configuration so that Any CPU also compiles on UWP x86 and targets iOS iPhone

* updated Resource.designer.cs file for Forms sample

* removed property FileByte from internal class FilePickerEventArgs, as it is never used

* using Intent Extra constant for passing allowed types to FilePickerActivity, instead of using nameof()

* added pull request template, adapted from the Xamarin.Forms github project

* When Android doesn't have Permission.ReadExternalStorage the App will now ask the user to grant the permission on API level >=23

* Added necessary imports

* Added requested fix of inverted if's to get the right action when permission is granted

* Added comment for REQUEST_STORAGE

* Changed variable to PascalCase

* cleaning up code from PR #109

* Revert "added checking Storage permission when on Android device"

This reverts commit 779c5a1.

* Revert "added Plugin.Permissions NuGet package to sample project, in order to check for permissions"

This reverts commit e81ed29.

* removed compiler warnings in Mac implementation

* permission READ_EXTERNAL_STORAGE is automatically added to the Android app that uses this plugin; no need to add it to AndroidManifest.xml now (#112)

* removed READ_EXTERNAL_STORE permission from sample project

* fixed file mode for several files; no actual code changes

old mode: 100755, new mode 100644
changed using the commands:
git config core.filemode true
chmod -R -x .
see also: https://stackoverflow.com/questions/1257592/
git also reports some files have all their lines changed

* Update README.md

* Multi targeting (#116)

* added multi-targeting project Plugin.FilePicker

changed solution to only include one project
renamed all shared, abstraction and implementation files according to their platforms
removed platform specific .csproj files
removed AssemblyInfo.cs and resources; infos now also in .csproj
new project generated by jamesmontemagno's "Cross-Platform .NET Standard Plugin Templates" project template
see: https://marketplace.visualstudio.com/items?itemName=vs-publisher-473885.PluginForXamarinTemplates

* updated sample projects to use multi-targeting project

updated solution files with single new csproj file
replaced direct references from sample platform project to plugin platform project with reference to multi-targeting project

* made FilePickerEventArgs and FilePickerCancelledEventArgs internal

* made ReadFully() internal and removed Obsolete attribute

* Revert "Merge branch 'master' into develop"

This reverts commit ea8feba, reversing
changes made to 752f6b6.
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

Successfully merging a pull request may close this issue.

2 participants