Xamarin を使用して Android Q (API 29) を対象とする Android プロジェクトで、sdcard のルートにある "Download" フォルダーに (apk) ファイルをダウンロードしようとしています。
まず、sdcard/Android/data/com.my.app/files にあるアプリのプライベート ストレージ スペースにファイルをダウンロードします。
次に、MediaStore API を使用して、ファイルをパブリックの "Download" フォルダーにコピーします。これは私のコードです:
private void SaveTest(string appPath, string fileName)
{
var contentValues = new ContentValues();
contentValues.Put(MediaStore.MediaColumns.DisplayName, fileName);
contentValues.Put(MediaStore.MediaColumns.MimeType, "application/vnd.android.package-archive");
contentValues.Put(MediaStore.MediaColumns.RelativePath, Android.OS.Environment.DirectoryDownloads);
var context = (Context)this;
var resolver = context.ContentResolver;
Stream stream = null;
Android.Net.Uri uri = null;
try
{
var contentUri = MediaStore.Downloads.ExternalContentUri;
var cursor = resolver.Query(
contentUri,
new[] { MediaStore.MediaColumns.Id, MediaStore.MediaColumns.DisplayName, MediaStore.MediaColumns.MimeType, MediaStore.MediaColumns.RelativePath },
$"{MediaStore.MediaColumns.DisplayName} = ? AND {MediaStore.MediaColumns.MimeType} = ? AND {MediaStore.MediaColumns.RelativePath} = ?",
new[] { fileName, "application/vnd.android.package-archive", Android.OS.Environment.DirectoryDownloads },
null
);
if(cursor != null && cursor.Count >= 1)
{
cursor.MoveToFirst();
var id = cursor.GetLong(cursor.GetColumnIndexOrThrow(MediaStore.MediaColumns.Id));
var displayName = cursor.GetString(cursor.GetColumnIndexOrThrow(MediaStore.MediaColumns.DisplayName));
var relativePath = cursor.GetString(cursor.GetColumnIndexOrThrow(MediaStore.MediaColumns.MimeType));
var lastModifiedDate = cursor.GetString(cursor.GetColumnIndexOrThrow(MediaStore.MediaColumns.RelativePath));
uri = ContentUris.WithAppendedId(contentUri, id);
}
else
{
uri = resolver.Insert(contentUri, contentValues);
}
cursor?.Close();
if (uri == null)
{
throw new Exception("Retrieving of creating MediaStore record failed.");
}
byte[] buffer = new byte[1024];
stream = resolver.OpenOutputStream(uri);
var fullPathAndFileName = Path.Combine(appPath, fileName);
using (var inputstream = new FileStream(fullPathAndFileName, FileMode.Open))
{
CopyFile(inputstream, stream);
}
}
catch
{
if (uri != null)
{
// Don't leave an orphan entry in the MediaStore
resolver.Delete(uri, null, null);
}
throw;
}
finally
{
stream?.Close();
}
}
MediaStore への挿入は初めて機能しますが、2 回目に挿入しようとすると、返された Uri が null になります。したがって、既存のレコードを検出できるように ContentResolver クエリを追加しましたが、このクエリは結果を返しません。別のファイル名を使用した場合にのみ、再び機能します。
デバイス ログを見ると、次のエラーが表示されます。
時間 デバイス名 タイプ PID タグ メッセージ 05-14 14:14:27.442 エラー 14571 SQLiteDatabase android.database.sqlite.SQLiteConstraintException: UNIQUE 制約が失敗しました: files._data (コード 2067 SQLITE_CONSTRAINT_UNIQUE) at android.database.sqlite.SQLiteConnection.nativeExecuteForLastInsertedRowId(Nativeメソッド) で android.database.sqlite.SQLiteConnection.executeForLastInsertedRowId(SQLiteConnection.java:879) で android.database.sqlite.SQLiteSession.executeForLastInsertedRowId(SQLiteSession.java:790) で android.database.sqlite.SQLiteStatement.executeInsert(SQLiteStatement.java :88) com.android.providers.media.MediaProvider.com.android.providers.media.MediaProvider.insertInternal(MediaProvider.java:3317) の insertFile(MediaProvider.java:2776) com.android.providers.media.MediaProvider.insert(MediaProvider.java:2966) の android.content .ContentProvider$Transport.insert(ContentProvider.java:309) で android.content.ContentProviderNative.onTransact(ContentProviderNative.java:154) で android.os.Binder.execTransactInternal(Binder.java:1021) で android.os.Binder. execTransact(Binder.java:994) 05-14 14:14:27.442 エラー 14571 SQLiteDatabase バケットの挿入中にエラーが発生しました_display_name=ダウンロード owner_package_name=com.my.app 親=8 ボリューム名=external_primary date_modified=1589458467 _display_name=XXX.apk mime_type=application/vnd .android.package-archive _data=/storage/emulated/0/Download/XXX.apk title=XXX.apk group_id=102105 is_download=true date_added=1589458467 primary_directory=ダウンロード bucket_id=540528482 media_type=0 relative_path=Download/
フィルターなしでメディアストア クエリを実行しても、結果カウントは 0 です。正しい Uri を取得するにはどうすればよいですか?