/* * 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.Nullable; import androidx.annotation.RestrictTo; import androidx.paging.PositionalDataSource; 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; /** * 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. * * @param Data type returned by the data source. * * @hide */ @RestrictTo(RestrictTo.Scope.LIBRARY_GROUP) public abstract class LimitOffsetDataSource extends 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; protected LimitOffsetDataSource(RoomDatabase db, SupportSQLiteQuery query, boolean inTransaction, String... tables) { this(db, RoomSQLiteQuery.copyFrom(query), inTransaction, tables); } protected LimitOffsetDataSource(RoomDatabase db, RoomSQLiteQuery query, boolean inTransaction, 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(); } }; db.getInvalidationTracker().addWeakObserver(mObserver); } /** * Count number of rows query can return */ @SuppressWarnings("WeakerAccess") public int countItems() { 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() { mDb.getInvalidationTracker().refreshVersionsSync(); return super.isInvalid(); } @SuppressWarnings("WeakerAccess") protected abstract List convertRows(Cursor cursor); @Override public void loadInitial(@NonNull LoadInitialParams params, @NonNull LoadInitialCallback callback) { int totalCount = countItems(); if (totalCount == 0) { callback.onResult(Collections.emptyList(), 0, 0); return; } // bound the size requested, based on known count final int firstLoadPosition = computeInitialLoadPosition(params, totalCount); final int firstLoadSize = computeInitialLoadSize(params, firstLoadPosition, totalCount); List list = loadRange(firstLoadPosition, firstLoadSize); if (list != null && list.size() == firstLoadSize) { callback.onResult(list, firstLoadPosition, totalCount); } else { // null list, or size doesn't match request - DB modified between count and load invalidate(); } } @Override public void loadRange(@NonNull LoadRangeParams params, @NonNull LoadRangeCallback callback) { List list = loadRange(params.startPosition, params.loadSize); if (list != null) { callback.onResult(list); } else { invalidate(); } } /** * Return the rows from startPos to startPos + loadCount */ @Nullable public List loadRange(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); if (mInTransaction) { mDb.beginTransaction(); Cursor cursor = null; 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(); } } } }