1 /* 2 * Copyright (C) 2017 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License 15 */ 16 17 package android.telephony.mbms; 18 19 import android.annotation.NonNull; 20 import android.annotation.Nullable; 21 import android.content.ContentProvider; 22 import android.content.ContentResolver; 23 import android.content.ContentValues; 24 import android.content.Context; 25 import android.content.SharedPreferences; 26 import android.content.pm.ProviderInfo; 27 import android.database.Cursor; 28 import android.net.Uri; 29 import android.os.ParcelFileDescriptor; 30 import android.telephony.MbmsDownloadSession; 31 32 import java.io.File; 33 import java.io.FileNotFoundException; 34 import java.io.IOException; 35 import java.util.Objects; 36 37 /** 38 * @hide 39 */ 40 public class MbmsTempFileProvider extends ContentProvider { 41 public static final String TEMP_FILE_ROOT_PREF_FILE_NAME = "MbmsTempFileRootPrefs"; 42 public static final String TEMP_FILE_ROOT_PREF_NAME = "mbms_temp_file_root"; 43 44 private String mAuthority; 45 private Context mContext; 46 47 @Override onCreate()48 public boolean onCreate() { 49 return true; 50 } 51 52 @Override query(@onNull Uri uri, @Nullable String[] projection, @Nullable String selection, @Nullable String[] selectionArgs, @Nullable String sortOrder)53 public Cursor query(@NonNull Uri uri, @Nullable String[] projection, 54 @Nullable String selection, @Nullable String[] selectionArgs, 55 @Nullable String sortOrder) { 56 throw new UnsupportedOperationException("No querying supported"); 57 } 58 59 @Override getType(@onNull Uri uri)60 public String getType(@NonNull Uri uri) { 61 // EMBMS temp files can contain arbitrary content. 62 return "application/octet-stream"; 63 } 64 65 @Override insert(@onNull Uri uri, @Nullable ContentValues values)66 public Uri insert(@NonNull Uri uri, @Nullable ContentValues values) { 67 throw new UnsupportedOperationException("No inserting supported"); 68 } 69 70 @Override delete(@onNull Uri uri, @Nullable String selection, @Nullable String[] selectionArgs)71 public int delete(@NonNull Uri uri, @Nullable String selection, 72 @Nullable String[] selectionArgs) { 73 throw new UnsupportedOperationException("No deleting supported"); 74 } 75 76 @Override update(@onNull Uri uri, @Nullable ContentValues values, @Nullable String selection, @Nullable String[] selectionArgs)77 public int update(@NonNull Uri uri, @Nullable ContentValues values, @Nullable String 78 selection, @Nullable String[] selectionArgs) { 79 throw new UnsupportedOperationException("No updating supported"); 80 } 81 82 @Override openFile(Uri uri, String mode)83 public ParcelFileDescriptor openFile(Uri uri, String mode) throws FileNotFoundException { 84 // ContentProvider has already checked granted permissions 85 final File file = getFileForUri(mContext, mAuthority, uri); 86 final int fileMode = ParcelFileDescriptor.parseMode(mode); 87 return ParcelFileDescriptor.open(file, fileMode); 88 } 89 90 @Override attachInfo(Context context, ProviderInfo info)91 public void attachInfo(Context context, ProviderInfo info) { 92 super.attachInfo(context, info); 93 94 // Sanity check our security 95 if (info.exported) { 96 throw new SecurityException("Provider must not be exported"); 97 } 98 if (!info.grantUriPermissions) { 99 throw new SecurityException("Provider must grant uri permissions"); 100 } 101 102 mAuthority = info.authority; 103 mContext = context; 104 } 105 getUriForFile(Context context, String authority, File file)106 public static Uri getUriForFile(Context context, String authority, File file) { 107 // Get the canonical path of the temp file 108 String filePath; 109 try { 110 filePath = file.getCanonicalPath(); 111 } catch (IOException e) { 112 throw new IllegalArgumentException("Could not get canonical path for file " + file); 113 } 114 115 // Make sure the temp file is contained in the temp file directory as configured in the 116 // manifest 117 File tempFileDir = getEmbmsTempFileDir(context); 118 if (!MbmsUtils.isContainedIn(tempFileDir, file)) { 119 throw new IllegalArgumentException("File " + file + " is not contained in the temp " + 120 "file directory, which is " + tempFileDir); 121 } 122 123 // Get the canonical path of the temp file directory 124 String tempFileDirPath; 125 try { 126 tempFileDirPath = tempFileDir.getCanonicalPath(); 127 } catch (IOException e) { 128 throw new RuntimeException( 129 "Could not get canonical path for temp file root dir " + tempFileDir); 130 } 131 132 // Start at first char of path under temp file directory 133 String pathFragment; 134 if (tempFileDirPath.endsWith("/")) { 135 pathFragment = filePath.substring(tempFileDirPath.length()); 136 } else { 137 pathFragment = filePath.substring(tempFileDirPath.length() + 1); 138 } 139 140 String encodedPath = Uri.encode(pathFragment); 141 return new Uri.Builder().scheme(ContentResolver.SCHEME_CONTENT) 142 .authority(authority).encodedPath(encodedPath).build(); 143 } 144 getFileForUri(Context context, String authority, Uri uri)145 public static File getFileForUri(Context context, String authority, Uri uri) 146 throws FileNotFoundException { 147 if (!ContentResolver.SCHEME_CONTENT.equals(uri.getScheme())) { 148 throw new IllegalArgumentException("Uri must have scheme content"); 149 } 150 if (!Objects.equals(authority, uri.getAuthority())) { 151 throw new IllegalArgumentException("Uri does not have a matching authority: " + 152 authority + ", " + uri.getAuthority()); 153 } 154 155 String relPath = Uri.decode(uri.getEncodedPath()); 156 File file; 157 File tempFileDir; 158 159 try { 160 tempFileDir = getEmbmsTempFileDir(context).getCanonicalFile(); 161 file = new File(tempFileDir, relPath).getCanonicalFile(); 162 } catch (IOException e) { 163 throw new FileNotFoundException("Could not resolve paths"); 164 } 165 166 if (!file.getPath().startsWith(tempFileDir.getPath())) { 167 throw new SecurityException("Resolved path jumped beyond configured root"); 168 } 169 170 return file; 171 } 172 173 /** 174 * Returns a File for the directory used to store temp files for this app 175 */ getEmbmsTempFileDir(Context context)176 public static File getEmbmsTempFileDir(Context context) { 177 SharedPreferences prefs = context.getSharedPreferences(TEMP_FILE_ROOT_PREF_FILE_NAME, 0); 178 String storedTempFileRoot = prefs.getString(TEMP_FILE_ROOT_PREF_NAME, null); 179 try { 180 if (storedTempFileRoot != null) { 181 return new File(storedTempFileRoot).getCanonicalFile(); 182 } else { 183 return new File(context.getFilesDir(), 184 MbmsDownloadSession.DEFAULT_TOP_LEVEL_TEMP_DIRECTORY).getCanonicalFile(); 185 } 186 } catch (IOException e) { 187 throw new RuntimeException("Unable to canonicalize temp file root path " + e); 188 } 189 } 190 } 191