1 /*
2  * Copyright (C) 2020 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 com.android.browser.provider;
18 
19 import android.content.ContentProvider;
20 import android.content.ContentProviderOperation;
21 import android.content.ContentProviderResult;
22 import android.content.ContentResolver;
23 import android.content.ContentValues;
24 import android.content.Context;
25 import android.content.OperationApplicationException;
26 import android.database.sqlite.SQLiteDatabase;
27 import android.database.sqlite.SQLiteOpenHelper;
28 import android.net.Uri;
29 
30 import java.util.ArrayList;
31 import java.util.HashSet;
32 import java.util.Set;
33 
34 /**
35  * General purpose {@link ContentProvider} base class that uses SQLiteDatabase for storage.
36  */
37 public abstract class SQLiteContentProvider extends ContentProvider {
38 
39     private static final String TAG = "SQLiteContentProvider";
40 
41     private SQLiteOpenHelper mOpenHelper;
42     private Set<Uri> mChangedUris;
43     protected SQLiteDatabase mDb;
44 
45     private final ThreadLocal<Boolean> mApplyingBatch = new ThreadLocal<Boolean>();
46     private static final int SLEEP_AFTER_YIELD_DELAY = 4000;
47 
48     /**
49      * Maximum number of operations allowed in a batch between yield points.
50      */
51     private static final int MAX_OPERATIONS_PER_YIELD_POINT = 500;
52 
53     @Override
onCreate()54     public boolean onCreate() {
55         Context context = getContext();
56         mOpenHelper = getDatabaseHelper(context);
57         mChangedUris = new HashSet<Uri>();
58         return true;
59     }
60 
61     /**
62      * Returns a {@link SQLiteOpenHelper} that can open the database.
63      */
getDatabaseHelper(Context context)64     public abstract SQLiteOpenHelper getDatabaseHelper(Context context);
65 
66     /**
67      * The equivalent of the {@link #insert} method, but invoked within a transaction.
68      */
insertInTransaction(Uri uri, ContentValues values, boolean callerIsSyncAdapter)69     public abstract Uri insertInTransaction(Uri uri, ContentValues values,
70             boolean callerIsSyncAdapter);
71 
72     /**
73      * The equivalent of the {@link #update} method, but invoked within a transaction.
74      */
updateInTransaction(Uri uri, ContentValues values, String selection, String[] selectionArgs, boolean callerIsSyncAdapter)75     public abstract int updateInTransaction(Uri uri, ContentValues values, String selection,
76             String[] selectionArgs, boolean callerIsSyncAdapter);
77 
78     /**
79      * The equivalent of the {@link #delete} method, but invoked within a transaction.
80      */
deleteInTransaction(Uri uri, String selection, String[] selectionArgs, boolean callerIsSyncAdapter)81     public abstract int deleteInTransaction(Uri uri, String selection, String[] selectionArgs,
82             boolean callerIsSyncAdapter);
83 
84     /**
85      * Call this to add a URI to the list of URIs to be notified when the transaction
86      * is committed.
87      */
postNotifyUri(Uri uri)88     protected void postNotifyUri(Uri uri) {
89         synchronized (mChangedUris) {
90             mChangedUris.add(uri);
91         }
92     }
93 
isCallerSyncAdapter(Uri uri)94     public boolean isCallerSyncAdapter(Uri uri) {
95         return false;
96     }
97 
getDatabaseHelper()98     public SQLiteOpenHelper getDatabaseHelper() {
99         return mOpenHelper;
100     }
101 
applyingBatch()102     private boolean applyingBatch() {
103         return mApplyingBatch.get() != null && mApplyingBatch.get();
104     }
105 
106     @Override
insert(Uri uri, ContentValues values)107     public Uri insert(Uri uri, ContentValues values) {
108         Uri result = null;
109         boolean callerIsSyncAdapter = isCallerSyncAdapter(uri);
110         boolean applyingBatch = applyingBatch();
111         if (!applyingBatch) {
112             mDb = mOpenHelper.getWritableDatabase();
113             mDb.beginTransaction();
114             try {
115                 result = insertInTransaction(uri, values, callerIsSyncAdapter);
116                 mDb.setTransactionSuccessful();
117             } finally {
118                 mDb.endTransaction();
119             }
120 
121             onEndTransaction(callerIsSyncAdapter);
122         } else {
123             result = insertInTransaction(uri, values, callerIsSyncAdapter);
124         }
125         return result;
126     }
127 
128     @Override
bulkInsert(Uri uri, ContentValues[] values)129     public int bulkInsert(Uri uri, ContentValues[] values) {
130         int numValues = values.length;
131         boolean callerIsSyncAdapter = isCallerSyncAdapter(uri);
132         mDb = mOpenHelper.getWritableDatabase();
133         mDb.beginTransaction();
134         try {
135             for (int i = 0; i < numValues; i++) {
136                 Uri result = insertInTransaction(uri, values[i], callerIsSyncAdapter);
137                 mDb.yieldIfContendedSafely();
138             }
139             mDb.setTransactionSuccessful();
140         } finally {
141             mDb.endTransaction();
142         }
143 
144         onEndTransaction(callerIsSyncAdapter);
145         return numValues;
146     }
147 
148     @Override
update(Uri uri, ContentValues values, String selection, String[] selectionArgs)149     public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
150         int count = 0;
151         boolean callerIsSyncAdapter = isCallerSyncAdapter(uri);
152         boolean applyingBatch = applyingBatch();
153         if (!applyingBatch) {
154             mDb = mOpenHelper.getWritableDatabase();
155             mDb.beginTransaction();
156             try {
157                 count = updateInTransaction(uri, values, selection, selectionArgs,
158                         callerIsSyncAdapter);
159                 mDb.setTransactionSuccessful();
160             } finally {
161                 mDb.endTransaction();
162             }
163 
164             onEndTransaction(callerIsSyncAdapter);
165         } else {
166             count = updateInTransaction(uri, values, selection, selectionArgs, callerIsSyncAdapter);
167         }
168 
169         return count;
170     }
171 
172     @Override
delete(Uri uri, String selection, String[] selectionArgs)173     public int delete(Uri uri, String selection, String[] selectionArgs) {
174         int count = 0;
175         boolean callerIsSyncAdapter = isCallerSyncAdapter(uri);
176         boolean applyingBatch = applyingBatch();
177         if (!applyingBatch) {
178             mDb = mOpenHelper.getWritableDatabase();
179             mDb.beginTransaction();
180             try {
181                 count = deleteInTransaction(uri, selection, selectionArgs, callerIsSyncAdapter);
182                 mDb.setTransactionSuccessful();
183             } finally {
184                 mDb.endTransaction();
185             }
186 
187             onEndTransaction(callerIsSyncAdapter);
188         } else {
189             count = deleteInTransaction(uri, selection, selectionArgs, callerIsSyncAdapter);
190         }
191         return count;
192     }
193 
194     @Override
applyBatch(ArrayList<ContentProviderOperation> operations)195     public ContentProviderResult[] applyBatch(ArrayList<ContentProviderOperation> operations)
196             throws OperationApplicationException {
197         int ypCount = 0;
198         int opCount = 0;
199         boolean callerIsSyncAdapter = false;
200         mDb = mOpenHelper.getWritableDatabase();
201         mDb.beginTransaction();
202         try {
203             mApplyingBatch.set(true);
204             final int numOperations = operations.size();
205             final ContentProviderResult[] results = new ContentProviderResult[numOperations];
206             for (int i = 0; i < numOperations; i++) {
207                 if (++opCount >= MAX_OPERATIONS_PER_YIELD_POINT) {
208                     throw new OperationApplicationException(
209                             "Too many content provider operations between yield points. "
210                                     + "The maximum number of operations per yield point is "
211                                     + MAX_OPERATIONS_PER_YIELD_POINT, ypCount);
212                 }
213                 final ContentProviderOperation operation = operations.get(i);
214                 if (!callerIsSyncAdapter && isCallerSyncAdapter(operation.getUri())) {
215                     callerIsSyncAdapter = true;
216                 }
217                 if (i > 0 && operation.isYieldAllowed()) {
218                     opCount = 0;
219                     if (mDb.yieldIfContendedSafely(SLEEP_AFTER_YIELD_DELAY)) {
220                         ypCount++;
221                     }
222                 }
223                 results[i] = operation.apply(this, results, i);
224             }
225             mDb.setTransactionSuccessful();
226             return results;
227         } finally {
228             mApplyingBatch.set(false);
229             mDb.endTransaction();
230             onEndTransaction(callerIsSyncAdapter);
231         }
232     }
233 
onEndTransaction(boolean callerIsSyncAdapter)234     protected void onEndTransaction(boolean callerIsSyncAdapter) {
235         Set<Uri> changed;
236         synchronized (mChangedUris) {
237             changed = new HashSet<Uri>(mChangedUris);
238             mChangedUris.clear();
239         }
240         ContentResolver resolver = getContext().getContentResolver();
241         for (Uri uri : changed) {
242             boolean syncToNetwork = !callerIsSyncAdapter && syncToNetwork(uri);
243             resolver.notifyChange(uri, null, syncToNetwork);
244         }
245     }
246 
syncToNetwork(Uri uri)247     protected boolean syncToNetwork(Uri uri) {
248         return false;
249     }
250 }
251