เปิดเผยนอกเหนือจากแอปผ่าน ClipDataItem.getUri


109

ฉันกำลังพยายามแก้ไขปัญหาหลังจากเพิ่มคุณสมบัติใหม่ในระบบไฟล์ Android แต่ได้รับข้อผิดพลาดนี้:

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

ดังนั้นฉันหวังว่าจะมีใครช่วยฉันแก้ไขได้ :)

ขอบคุณ

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;
}

คำตอบ:


180

สำหรับ sdk 24 ขึ้นไปหากคุณต้องการรับ Uri ของไฟล์ที่อยู่นอกพื้นที่จัดเก็บแอปคุณจะพบข้อผิดพลาดนี้
โซลูชัน @ eranda.del ช่วยให้คุณเปลี่ยนนโยบายเพื่ออนุญาตสิ่งนี้และทำงานได้ดี

อย่างไรก็ตามหากคุณต้องการทำตามหลักเกณฑ์ของ Google โดยไม่ต้องเปลี่ยนนโยบาย API ของแอปคุณต้องใช้ FileProvider

ในตอนแรกเพื่อรับ URI ของไฟล์คุณต้องใช้เมธอด FileProvider.getUriForFile ():

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

จากนั้นคุณต้องกำหนดค่าผู้ให้บริการของคุณในรายการ Android ของคุณ:

<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>

(ใน "เจ้าหน้าที่" ใช้ค่าเดียวกันกับอาร์กิวเมนต์ที่สองของเมธอด getUriForFile () (ลายเซ็นแอป + ".provider"))

และสุดท้ายคุณต้องสร้างไฟล์ ressources: "file_paths" ไฟล์นี้ต้องสร้างภายใต้ไดเร็กทอรี res / xml (คุณอาจต้องสร้างไดเร็กทอรีนี้ด้วย):

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path name="external_files" path="." />
</paths>

88
"ทำให้ชีวิตนักพัฒนาง่ายขึ้น" นี่คือแท็กไลน์สำหรับ API ใหม่ทุกตัวใช่ไหม Google
Kirill Karmazin

14
โซลูชันที่ Brendon จัดเตรียมไว้เป็นวิธีที่ถูกต้อง แต่ยังไม่พร้อมคัดลอกวางคุณต้องตรวจสอบว่าเหมาะสมกับความต้องการของคุณหรือไม่เช่น <external-path ...ถูกใช้เมื่อคุณต้องการ Environment.getExternalStorageDirectory () และใช้<files-path ...หากคุณต้องการ Context.getFilesDir () คุณควรตรวจสอบเอกสารอย่างเป็นทางการก่อน: developer.android.com/reference/android/support/v4/content/…
Kirill Karmazin

10
การแจ้งเตือนหากคุณใช้ androidx (เนมสเปซการสนับสนุนใหม่) คุณสามารถดูเอกสารที่เหมาะสมได้ที่นี่ ( developer.android.com/reference/androidx/core/content/… )
Rodrirokr

23
โปรดทราบว่าในandroidxนั้นคือ:android:name="androidx.core.content.FileProvider"
smörkex

151

เพิ่มบล็อกรหัสต่อไปนี้ก่อนเริ่มกล้องหรือเรียกดูไฟล์

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

โปรดดูโหมดที่เข้มงวดของลิงก์อ้างอิงและอธิบายการใช้งานและรายละเอียดทางเทคนิคทั้งหมด


25
ไม่ใช่วิธีที่ถูกต้องในการแก้ปัญหานี้โดยพื้นฐานแล้วจะเป็นการลบนโยบายโหมดเข้มงวด และจะเพิกเฉยต่อคำเตือนด้านความปลอดภัย
Ahsan Zaheer

มีข้อดีข้อเสียบางประการเกี่ยวกับเรื่องนี้ โปรดอ้างอิงข้อมูลต่อไปนี้เพื่อรับข้อมูลเพิ่มเติม stackoverflow.com/questions/18100161/…
eranda.del

ได้ผลหากมีการดึงกลับโปรดแจ้งให้เราทราบ :)
Hanan

รหัสแก้ไขข้อยกเว้นให้ฉัน อย่างไรก็ตามความคิดเห็นด้านบนบอกว่าโค้ดไม่ใช่วิธีที่ถูกต้องในการแก้ปัญหานี้อย่างไร ไม่มีใครรู้ว่าวิธีการที่เหมาะสมจะเป็นอย่างไร?
Shn_Android_Dev

1
Google เกลียดโปรแกรมเมอร์
Joubert Vasconcelos

0

ไม่จำเป็นต้องกำหนดค่าผู้ให้บริการใน AndroidManifest อนุญาตให้ใช้เฉพาะกล้องถ่ายรูปและที่เก็บข้อมูลเท่านั้น ใช้รหัสต่อไปนี้เพื่อเริ่มการทำงานของกล้อง:

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);
}

หลังจากจับภาพแล้วคุณสามารถค้นหาภาพที่คุณถ่ายได้ใน:

"your_file_name_with_dir"

อธิบายการกำหนดค่าของผู้ให้บริการใน AndroidManifest ได้ดีกว่าการให้รหัสอื่น ๆ
fury.slay

0

เพิ่มรหัสใน

onCreate(Budle savedInstancesState){
    if (Build.VERSION.SDK_INT >= 24) {
        try {
            Method m = StrictMode.class.getMethod("disableDeathOnFileUriExposure");
            m.invoke(null);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.