exposed beyond app through ClipData.Item.getUri
Asked Answered
S

4

147

I am trying to fix a problem after the new feature added in Android file system but I get this error:

android.os.FileUriExposedException: file:///storage/emulated/0/MyApp/Camera_20180105_172234.jpg exposed beyond app through ClipData.Item.getUri()

So I hope some one can help me fix this :)

Thanks

private Uri getTempUri() {
    // Create an image file name
    SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMdd_HHmmss");
    String dt = sdf.format(new Date());
    imageFile = null;
    imageFile = new File(Environment.getExternalStorageDirectory()
            + "/MyApp/", "Camera_" + dt + ".jpg");
    AppLog.Log(
            TAG,
            "New Camera Image Path:- "
                    + Environment.getExternalStorageDirectory()
                    + "/MyApp/" + "Camera_" + dt + ".jpg");
    File file = new File(Environment.getExternalStorageDirectory() + "/MyApp");
    if (!file.exists()) {
        file.mkdir();
    }
    imagePath = Environment.getExternalStorageDirectory() + "/MyApp/"
            + "Camera_" + dt + ".jpg";
    imageUri = Uri.fromFile(imageFile);
    return imageUri;
}
Sukin answered 5/1, 2018 at 16:24 Comment(1)
#38200782Lou
R
266

For sdk 24 and up, if you need to get the Uri of a file outside your app storage you have this error.
@eranda.del solutions let you change the policy to allow this and it works fine.

However if you want to follow the google guideline without having to change the API policy of your app you have to use a FileProvider.

At first to get the URI of a file you need to use FileProvider.getUriForFile() method:

Uri imageUri = FileProvider.getUriForFile(
            MainActivity.this,
            "com.example.homefolder.example.provider", //(use your app signature + ".provider" )
            imageFile);

Then you need to configure your provider in your android manifest :

<application>
  ...
     <provider
        android:name="android.support.v4.content.FileProvider"
        android:authorities="com.example.homefolder.example.provider"
        android:exported="false"
        android:grantUriPermissions="true">
        <!-- ressource file to create -->
        <meta-data
            android:name="android.support.FILE_PROVIDER_PATHS"
            android:resource="@xml/file_paths">  
        </meta-data>
    </provider>
</application>

(In "authorities" use the same value than the second argument of the getUriForFile() method (app signature + ".provider"))

And finally you need to create the ressources file: "file_paths". This file need to be created under the res/xml directory (you probably need to create this directory too) :

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path name="external_files" path="." />
</paths>
Resurrectionism answered 10/5, 2018 at 4:9 Comment(4)
"Make developers life easier" - this is the tagline for every new API, right Google?Selfdrive
The solution provided by Brendon is the correct one but it's not copy-paste ready, you need to check if it suits your needs, for ex. <external-path ... is used when you want Environment.getExternalStorageDirectory(). and use <files-path ... if you need Context.getFilesDir(). You better check official docs first: developer.android.com/reference/android/support/v4/content/…Selfdrive
A reminder if you're using androidx (new support namespace), you can find here (developer.android.com/reference/androidx/core/content/…) the proper documentation.Zoogeography
Note that in androidx it is: android:name="androidx.core.content.FileProvider"Pedrick
T
176

Add following code block before start camera or file browsing

    StrictMode.VmPolicy.Builder builder = new StrictMode.VmPolicy.Builder();
    StrictMode.setVmPolicy(builder.build());

Please refer the referral link strict mode and its explained all the usage and technical details.

Thoroughpaced answered 18/2, 2018 at 12:16 Comment(6)
Its not the right way to solve this, this will basically remove the strictmode policies. and will ignore the security warningLayby
There are some pros and cons on this. Please refer the following to get more information #18100661Thoroughpaced
It works, if there is any draw back then please let me know :)Osteotome
The code solved the exception for me. However, the top comment here says how the code is not the correct way to solve this. Does anyone know what the proper method would be then?Lunarian
Google hates programmers.Tautology
Thanks!!! It worked for me. But I will still investigate why the provider setting in the manifest is not working for meButtercup
T
4

No required provider configuration into AndroidManifest only Camera and Storage permission should be permitted. Use the following code to start the camera:

final int REQUEST_ACTION_CAMERA = 9;
void openCamra() {
Intent cameraImgIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

cameraImgIntent.putExtra(MediaStore.EXTRA_OUTPUT,
                FileProvider.getUriForFile(this, BuildConfig.APPLICATION_ID +".provider",
                new File("your_file_name_with_dir")));
startActivityForResult(cameraImgIntent, REQUEST_ACTION_CAMERA);
}

After capturing the image, you can find your captured image in:

"your_file_name_with_dir"
Traction answered 11/5, 2020 at 13:25 Comment(2)
It is better to explain the configuration of provider in AndroidManifest, than giving other code.Aeroneurosis
This worked for me with AndroidManifest <provider tagCaddis
S
0

add a code in

onCreate(Budle savedInstancesState){
    if (Build.VERSION.SDK_INT >= 24) {
        try {
            Method m = StrictMode.class.getMethod("disableDeathOnFileUriExposure");
            m.invoke(null);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
Sheelah answered 17/9, 2020 at 4:57 Comment(2)
getting file format not supported error for image.jpgCarolinacaroline
This trick no longer works on Android 11 :)Utilitarianism

© 2022 - 2024 — McMap. All rights reserved.