/* * Copyright (C) 2017 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package androidx.room.paging; import android.database.Cursor; import androidx.annotation.NonNull; import androidx.annotation.RestrictTo; import androidx.room.InvalidationTracker; import androidx.room.RoomDatabase; import androidx.room.RoomSQLiteQuery; import androidx.sqlite.db.SupportSQLiteQuery; import java.util.Collections; import java.util.List; import java.util.Set; import java.util.concurrent.atomic.AtomicBoolean; /** * A simple data source implementation that uses Limit & Offset to page the query. *

* This is NOT the most efficient way to do paging on SQLite. It is * recommended to use an indexed * ORDER BY statement but that requires a more complex API. This solution is technically equal to * receiving a {@link Cursor} from a large query but avoids the need to manually manage it, and * never returns inconsistent data if it is invalidated. * * This class is used for both Paging2 and Pagin3 (via its compat API). When used with Paging3, * it does lazy registration for observers to be suitable for initialization on the main thread * whereas in Paging2, it will register observer eagerly to obey Paging2's strict Data Source * rules. (Paging2 does not let data source to possibly return invalidated data). * * @param Data type returned by the data source. * * @hide */ @SuppressWarnings("deprecation") @RestrictTo(RestrictTo.Scope.LIBRARY_GROUP_PREFIX) public abstract class LimitOffsetDataSource extends androidx.paging.PositionalDataSource { private final RoomSQLiteQuery mSourceQuery; private final String mCountQuery; private final String mLimitOffsetQuery; private final RoomDatabase mDb; @SuppressWarnings("FieldCanBeLocal") private final InvalidationTracker.Observer mObserver; private final boolean mInTransaction; private final AtomicBoolean mRegisteredObserver = new AtomicBoolean(false); protected LimitOffsetDataSource(@NonNull RoomDatabase db, @NonNull SupportSQLiteQuery query, boolean inTransaction, @NonNull String... tables) { this(db, RoomSQLiteQuery.copyFrom(query), inTransaction, tables); } protected LimitOffsetDataSource( @NonNull RoomDatabase db, @NonNull SupportSQLiteQuery query, boolean inTransaction, boolean registerObserverImmediately, @NonNull String... tables) { this(db, RoomSQLiteQuery.copyFrom(query), inTransaction, registerObserverImmediately, tables); } protected LimitOffsetDataSource( @NonNull RoomDatabase db, @NonNull RoomSQLiteQuery query, boolean inTransaction, @NonNull String... tables) { this(db, query, inTransaction, true /*register registerObserverImmediately*/, tables); } protected LimitOffsetDataSource( @NonNull RoomDatabase db, @NonNull RoomSQLiteQuery query, boolean inTransaction, boolean registerObserverImmediately, @NonNull String... tables) { mDb = db; mSourceQuery = query; mInTransaction = inTransaction; mCountQuery = "SELECT COUNT(*) FROM ( " + mSourceQuery.getSql() + " )"; mLimitOffsetQuery = "SELECT * FROM ( " + mSourceQuery.getSql() + " ) LIMIT ? OFFSET ?"; mObserver = new InvalidationTracker.Observer(tables) { @Override public void onInvalidated(@NonNull Set tables) { invalidate(); } }; if (registerObserverImmediately) { registerObserverIfNecessary(); } } private void registerObserverIfNecessary() { if (mRegisteredObserver.compareAndSet(false, true)) { mDb.getInvalidationTracker().addWeakObserver(mObserver); } } /** * Count number of rows query can return * * @hide */ @SuppressWarnings("WeakerAccess") public int countItems() { registerObserverIfNecessary(); final RoomSQLiteQuery sqLiteQuery = RoomSQLiteQuery.acquire(mCountQuery, mSourceQuery.getArgCount()); sqLiteQuery.copyArgumentsFrom(mSourceQuery); Cursor cursor = mDb.query(sqLiteQuery); try { if (cursor.moveToFirst()) { return cursor.getInt(0); } return 0; } finally { cursor.close(); sqLiteQuery.release(); } } @Override public boolean isInvalid() { registerObserverIfNecessary(); mDb.getInvalidationTracker().refreshVersionsSync(); return super.isInvalid(); } @NonNull @SuppressWarnings("WeakerAccess") protected abstract List convertRows(@NonNull Cursor cursor); @SuppressWarnings("deprecation") @Override public void loadInitial(@NonNull LoadInitialParams params, @NonNull LoadInitialCallback callback) { registerObserverIfNecessary(); List list = Collections.emptyList(); int totalCount; int firstLoadPosition = 0; RoomSQLiteQuery sqLiteQuery = null; Cursor cursor = null; mDb.beginTransaction(); try { totalCount = countItems(); if (totalCount != 0) { // bound the size requested, based on known count firstLoadPosition = computeInitialLoadPosition(params, totalCount); int firstLoadSize = computeInitialLoadSize(params, firstLoadPosition, totalCount); sqLiteQuery = getSQLiteQuery(firstLoadPosition, firstLoadSize); cursor = mDb.query(sqLiteQuery); List rows = convertRows(cursor); mDb.setTransactionSuccessful(); list = rows; } } finally { if (cursor != null) { cursor.close(); } mDb.endTransaction(); if (sqLiteQuery != null) { sqLiteQuery.release(); } } callback.onResult(list, firstLoadPosition, totalCount); } @Override public void loadRange(@NonNull LoadRangeParams params, @NonNull LoadRangeCallback callback) { callback.onResult(loadRange(params.startPosition, params.loadSize)); } /** * Return the rows from startPos to startPos + loadCount * * @hide */ @SuppressWarnings("deprecation") @NonNull public List loadRange(int startPosition, int loadCount) { final RoomSQLiteQuery sqLiteQuery = getSQLiteQuery(startPosition, loadCount); if (mInTransaction) { mDb.beginTransaction(); Cursor cursor = null; //noinspection TryFinallyCanBeTryWithResources try { cursor = mDb.query(sqLiteQuery); List rows = convertRows(cursor); mDb.setTransactionSuccessful(); return rows; } finally { if (cursor != null) { cursor.close(); } mDb.endTransaction(); sqLiteQuery.release(); } } else { Cursor cursor = mDb.query(sqLiteQuery); //noinspection TryFinallyCanBeTryWithResources try { return convertRows(cursor); } finally { cursor.close(); sqLiteQuery.release(); } } } private RoomSQLiteQuery getSQLiteQuery(int startPosition, int loadCount) { final RoomSQLiteQuery sqLiteQuery = RoomSQLiteQuery.acquire(mLimitOffsetQuery, mSourceQuery.getArgCount() + 2); sqLiteQuery.copyArgumentsFrom(mSourceQuery); sqLiteQuery.bindLong(sqLiteQuery.getArgCount() - 1, loadCount); sqLiteQuery.bindLong(sqLiteQuery.getArgCount(), startPosition); return sqLiteQuery; } }