Home | History | Annotate | Download | only in content
      1 /*
      2  * Copyright (C) 2009 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.content;
     18 
     19 import android.content.ContentProvider;
     20 import android.content.ContentValues;
     21 import android.content.Context;
     22 import android.content.UriMatcher;
     23 import android.content.res.AssetFileDescriptor;
     24 import android.database.Cursor;
     25 import android.database.sqlite.SQLiteDatabase;
     26 import android.database.sqlite.SQLiteOpenHelper;
     27 import android.net.Uri;
     28 import android.os.MemoryFile;
     29 import android.os.ParcelFileDescriptor;
     30 import android.util.Log;
     31 
     32 import java.io.File;
     33 import java.io.FileNotFoundException;
     34 import java.io.IOException;
     35 import java.io.OutputStream;
     36 
     37 /** Simple test provider that runs in the local process. */
     38 public class MemoryFileProvider extends ContentProvider {
     39     private static final String TAG = "MemoryFileProvider";
     40 
     41     private static final String DATA_FILE = "data.bin";
     42 
     43     // some random data
     44     public static final byte[] TEST_BLOB = new byte[] {
     45         -12,  127, 0, 3, 1, 2, 3, 4, 5, 6, 1, -128, -1, -54, -65, 35,
     46         -53, -96, -74, -74, -55, -43, -69, 3, 52, -58,
     47         -121, 127, 87, -73, 16, -13, -103, -65, -128, -36,
     48         107, 24, 118, -17, 97, 97, -88, 19, -94, -54,
     49         53, 43, 44, -27, -124, 28, -74, 26, 35, -36,
     50         16, -124, -31, -31, -128, -79, 108, 116, 43, -17 };
     51 
     52     private SQLiteOpenHelper mOpenHelper;
     53 
     54     private static final int DATA_ID_BLOB = 1;
     55     private static final int HUGE = 2;
     56     private static final int FILE = 3;
     57 
     58     private static final UriMatcher sURLMatcher = new UriMatcher(
     59             UriMatcher.NO_MATCH);
     60 
     61     static {
     62         sURLMatcher.addURI("*", "data/#/blob", DATA_ID_BLOB);
     63         sURLMatcher.addURI("*", "huge", HUGE);
     64         sURLMatcher.addURI("*", "file", FILE);
     65     }
     66 
     67     private static class DatabaseHelper extends SQLiteOpenHelper {
     68         private static final String DATABASE_NAME = "local.db";
     69         private static final int DATABASE_VERSION = 1;
     70 
     71         public DatabaseHelper(Context context) {
     72             super(context, DATABASE_NAME, null, DATABASE_VERSION);
     73         }
     74 
     75         @Override
     76         public void onCreate(SQLiteDatabase db) {
     77             db.execSQL("CREATE TABLE data (" +
     78                        "_id INTEGER PRIMARY KEY," +
     79                        "_blob TEXT, " +
     80                        "integer INTEGER);");
     81 
     82             // insert alarms
     83             ContentValues values = new ContentValues();
     84             values.put("_id", 1);
     85             values.put("_blob", TEST_BLOB);
     86             values.put("integer", 100);
     87             db.insert("data", null, values);
     88         }
     89 
     90         @Override
     91         public void onUpgrade(SQLiteDatabase db, int oldVersion, int currentVersion) {
     92             Log.w(TAG, "Upgrading test database from version " +
     93                   oldVersion + " to " + currentVersion +
     94                   ", which will destroy all old data");
     95             db.execSQL("DROP TABLE IF EXISTS data");
     96             onCreate(db);
     97         }
     98     }
     99 
    100 
    101     public MemoryFileProvider() {
    102     }
    103 
    104     @Override
    105     public boolean onCreate() {
    106         mOpenHelper = new DatabaseHelper(getContext());
    107         try {
    108             OutputStream out = getContext().openFileOutput(DATA_FILE, Context.MODE_PRIVATE);
    109             out.write(TEST_BLOB);
    110             out.close();
    111         } catch (IOException ex) {
    112             ex.printStackTrace();
    113         }
    114         return true;
    115     }
    116 
    117     @Override
    118     public Cursor query(Uri url, String[] projectionIn, String selection,
    119             String[] selectionArgs, String sort) {
    120         throw new UnsupportedOperationException("query not supported");
    121     }
    122 
    123     @Override
    124     public String getType(Uri url) {
    125         int match = sURLMatcher.match(url);
    126         switch (match) {
    127             case DATA_ID_BLOB:
    128                 return "application/octet-stream";
    129             case FILE:
    130                 return "application/octet-stream";
    131             default:
    132                 throw new IllegalArgumentException("Unknown URL");
    133         }
    134     }
    135 
    136     @Override
    137     public AssetFileDescriptor openAssetFile(Uri url, String mode) throws FileNotFoundException {
    138         int match = sURLMatcher.match(url);
    139         switch (match) {
    140             case DATA_ID_BLOB:
    141                 String sql = "SELECT _blob FROM data WHERE _id=" + url.getPathSegments().get(1);
    142                 return getBlobColumnAsAssetFile(url, mode, sql);
    143             case HUGE:
    144                 try {
    145                     MemoryFile memoryFile = new MemoryFile(null, 5000000);
    146                     memoryFile.writeBytes(TEST_BLOB, 0, 1000000, TEST_BLOB.length);
    147                     memoryFile.deactivate();
    148                     return AssetFileDescriptor.fromMemoryFile(memoryFile);
    149                 } catch (IOException ex) {
    150                     throw new FileNotFoundException("Error reading " + url + ":" + ex.toString());
    151                 }
    152             case FILE:
    153                 File file = getContext().getFileStreamPath(DATA_FILE);
    154                 ParcelFileDescriptor fd =
    155                         ParcelFileDescriptor.open(file, ParcelFileDescriptor.MODE_READ_ONLY);
    156                 return new AssetFileDescriptor(fd, 0, AssetFileDescriptor.UNKNOWN_LENGTH);
    157             default:
    158                 throw new FileNotFoundException("No files supported by provider at " + url);
    159         }
    160     }
    161 
    162     private AssetFileDescriptor getBlobColumnAsAssetFile(Uri url, String mode, String sql)
    163             throws FileNotFoundException {
    164         if (!"r".equals(mode)) {
    165             throw new FileNotFoundException("Mode " + mode + " not supported for " + url);
    166         }
    167         try {
    168             SQLiteDatabase db = mOpenHelper.getReadableDatabase();
    169             MemoryFile file = simpleQueryForBlobMemoryFile(db, sql);
    170             if (file == null) throw new FileNotFoundException("No such entry: " + url);
    171             AssetFileDescriptor afd = AssetFileDescriptor.fromMemoryFile(file);
    172             file.deactivate();
    173             // need to dup and then close? openFileHelper() doesn't do that though
    174             return afd;
    175         } catch (IOException ex) {
    176             throw new FileNotFoundException("Error reading " + url + ":" + ex.toString());
    177         }
    178     }
    179 
    180     private MemoryFile simpleQueryForBlobMemoryFile(SQLiteDatabase db, String sql) throws IOException {
    181         Cursor cursor = db.rawQuery(sql, null);
    182         try {
    183             if (!cursor.moveToFirst()) {
    184                 return null;
    185             }
    186             byte[] bytes = cursor.getBlob(0);
    187             MemoryFile file = new MemoryFile(null, bytes.length);
    188             file.writeBytes(bytes, 0, 0, bytes.length);
    189             return file;
    190         } finally {
    191             if (cursor != null) {
    192                 cursor.close();
    193             }
    194         }
    195     }
    196 
    197     @Override
    198     public int update(Uri url, ContentValues values, String where, String[] whereArgs) {
    199         throw new UnsupportedOperationException("update not supported");
    200     }
    201 
    202     @Override
    203     public Uri insert(Uri url, ContentValues initialValues) {
    204         throw new UnsupportedOperationException("insert not supported");
    205     }
    206 
    207     @Override
    208     public int delete(Uri url, String where, String[] whereArgs) {
    209         throw new UnsupportedOperationException("delete not supported");
    210     }
    211 }
    212