Question

My development phone is a Nexus 5, running Android 4.4.2.

In my application, I am attempting to create a folder on external storage that will store debug information for my application. Basically it will contain all the commands executed by the application, so that when a user encounters a problem, I have the option of having them send me the information from the debug folder to analyse.

I started off by trying to write a file to the folder, but found there was an error creating the folder. At first I was using mkdir(), then I moved onto mkdirs() which also didn't work.

I have <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> in my manifest.

Here is the code for creating the folder:

    File folder = new File(Environment.getExternalStorageDirectory() + "/DebugData");

    String path = folder.getPath();

    if(!folder.mkdirs() || !folder.exists()){        
            Log.e(LOG_TAG, path + " failed");
        } else {
            Log.d(LOG_TAG, path + " succeeded");
        } 

Here is what I have also tried:

    //Check SD card state
    String state = Environment.getExternalStorageState();
    if (Environment.MEDIA_MOUNTED_READ_ONLY.equals(state) || !Environment.MEDIA_MOUNTED.equals(state)) {
        Log.e(LOG_TAG, "Error: external storage is read only or unavailable");
    } else {
        Log.d(LOG_TAG, "External storage is not read only or unavailable");
    }

This returns that the external storage is not read only or unavailable.

I have also tried different paths, such as File folder = new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES), "Folder1");

This is where it became really confusing.

I tried development on different phones. Firstly, I grabbed a Galaxy S4 GT-i9505 running Android 4.2.2 and it worked. I was able to create the folders and write to them. This showed me that the code was working. Also the path returned by running the code on the S4 and Nexus 5 was the same.

Then I thought it may be android version specific. So I grabbed a Nexus 4 with Android 4.4.2 and the code worked on it as well. Created the folders and allowed me to write to them.

None of the phones are rooted and are all stock standard. There's no special applications or anything I can think of settings wise on the Nexus 5 that would cause permissions problems. The connection is set to Media Device (MTP).

EDIT:

I should add that I have tried the follow which also did not work:

  • Writing a file to the root directory of the external storage
  • Creating the file in the external storage root directory and writing to it
  • Creating a folder in a path outlined and writing a file to it
  • Creating the file in the path outlined and writing to it

I am confused as to what is causing this, is there anything else I can test or change to fix the issue?

EDIT 2:

Turns out the issue was due to, I think, indexing.

Basically all of the other devices I tested on, allowed me to reconnect the USB connection and view the created files and folders.

For some reason my Nexus 5 doesn't index the folders/files, even though they exist.

I downloaded a different 3rd party file explorer application and noticed all the folders and files were there.

So to view these folders and files via USB debugging, I have to restart the phone in order to re-index them, which seems quite annoying but it is better than it not working at all.

Thanks.

Was it helpful?

Solution

In terms of this being an indexing issue with the Nexus, this worked for me:

MediaScannerConnection.scanFile(this, new String[] { file.toString() }, null,
        new MediaScannerConnection.OnScanCompletedListener() {
            public void onScanCompleted(String path, Uri uri) {
                Log.i("ExternalStorage", "Scanned " + path + ":");
                Log.i("ExternalStorage", "-> uri=" + uri);
            }
});

You should call it straight after creating and saving the file. By using the scanner, I was able to see newly created files and directories simply by replugging the device in.

According to the docs:

MediaScannerConnection provides a way for applications to pass a newly created or downloaded media file to the media scanner service. The media scanner service will read metadata from the file and add the file to the media content provider.

Hope this helps someone else.

OTHER TIPS

Turns out the issue was due to, I think, indexing.

Basically all of the other devices I tested on, allowed me to reconnect the USB connection and view the created files and folders.

For some reason my Nexus 5 doesn't index the folders/files, even though they exist.

I downloaded a different 3rd party file explorer application and noticed all the folders and files were there.

So to view these folders and files via USB debugging, I have to restart the phone in order to re-index them, which seems quite annoying but it is better than it not working at all.

For android sdk version 23 and above you should check if the user has granted permission of external storage.

private void createFolder() {
    if (isStoragePermissionGranted()) {
        File folder = new File(Environment.getExternalStorageDirectory()+ File.separator + "DebugData");

        if(!folder.exists()){
            folder.mkdir();
        }
    }

public  boolean isStoragePermissionGranted() {
    if (Build.VERSION.SDK_INT >= 23) {
        if (checkSelfPermission(android.Manifest.permission.WRITE_EXTERNAL_STORAGE)
                == PackageManager.PERMISSION_GRANTED) {
            return true;
        } else {
            ActivityCompat.requestPermissions(this,
                    new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1);
            return false;
        }
    }
    else { //permission is automatically granted on sdk<23 upon installation
        return true;
    }
}

The code above worked for me and I hope it will work for you.

Had the same problem. Turned out I was testing on Android 6 and did not check for runtime WRITE_EXTERNAL_STORAGE permission.

First, new File() is just create object for file connect.
you want use this file, file is exists or createNewFile().
If you want use this file to create directory, check file is exists, isDirectory() after then mkDir().
Second, check Environment.getExternalStorageDirectory is variable path.
You can use DDMS or ADB for work.
Additionaly, I think you add permission for read external storage for something error.

If you are running an Android device with api >= 23, you need to request permission from user before you call mkdir() function. My previous answer with example code can be found https://stackoverflow.com/a/38694026/5089713

For Android 10+ add following into manifest:

<application
    android:requestLegacyExternalStorage="true"
    ...

as asking for Manifest.permission.WRITE_EXTERNAL_STORAGE is no more enough.

Typically, I do this:

/**
 * Return a dedicated directory named "MyApp" at the top of external storage
 */
public static file getDataDir() {
    File sdcard = Environment.getExternalStorageDirectory();
    if( sdcard == null || !sdcard.isDirectory() ) {
        // TODO: warning popup
        Log.w(TAG, "Storage card not found " + sdcard);
        return null;
    }
    File dataDir = new File(sdcard, "MyApp");
    if( !confirmDir(dataDir) ) {
        // TODO: warning popup
        Log.e(TAG, "Unable to create " + dataDir);
        return null;
    }
    return dataDir;
}   

private static boolean confirmDir(File dir) {
    if (dir.isDirectory()) return true;  // already exists
    if (dir.exists()) return false;      // already exists, but is not a directory
    return dir.mkdirs();                 // create it
}       

Also, add this permission to your manifest:

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

If you want private storage (typically under /data/) for your app (not on sdcard), then look into getDir(), getFilesDir(), fileList(), openFileInput(), openFileOutput(), etc.

There are also helper functions to get private directories within the sdcard for API 8 and above: getExternalFilesDir(), getExternalCacheDir(), etc.

Licensed under: CC-BY-SA with attribution
Not affiliated with StackOverflow
scroll top