From e88f6a9e707267524583f6efceae34f877ca246a Mon Sep 17 00:00:00 2001 From: Mark Hammond Date: Wed, 4 Oct 2023 11:41:13 -0400 Subject: [PATCH] Add a Places API to recursively count the number of descendants in a set of folders --- .../mozilla/appservices/places/Bookmarks.kt | 13 ++ .../appservices/places/PlacesConnection.kt | 6 + .../places/PlacesConnectionTest.kt | 34 +++++ components/places/src/ffi.rs | 5 + components/places/src/places.udl | 7 + components/places/src/storage/bookmarks.rs | 127 ++++++++++++++++++ 6 files changed, 192 insertions(+) diff --git a/components/places/android/src/main/java/mozilla/appservices/places/Bookmarks.kt b/components/places/android/src/main/java/mozilla/appservices/places/Bookmarks.kt index da21b494c4..59b1e2cb9d 100644 --- a/components/places/android/src/main/java/mozilla/appservices/places/Bookmarks.kt +++ b/components/places/android/src/main/java/mozilla/appservices/places/Bookmarks.kt @@ -119,6 +119,19 @@ interface ReadableBookmarksConnection : InterruptibleConnection { * has its `interrupt()` method called on another thread. */ fun getRecentBookmarks(limit: Int): List + + /** + * Counts the number of bookmark items in the bookmark trees under the specified GUIDs. + + * @param guids The guids of folders to query. + * @return Count of all bookmark items (ie, not folders or separators) in all specified folders recursively. + * Empty folders, non-existing GUIDs and non-existing items will return zero. + * The result is implementation dependant if the trees overlap. + * + * @throws OperationInterrupted if this database implements [InterruptibleConnection] and + * has its `interrupt()` method called on another thread. + */ + fun countBookmarksInTrees(guids: List): UInt } /** diff --git a/components/places/android/src/main/java/mozilla/appservices/places/PlacesConnection.kt b/components/places/android/src/main/java/mozilla/appservices/places/PlacesConnection.kt index 023373a3be..489cb2b332 100644 --- a/components/places/android/src/main/java/mozilla/appservices/places/PlacesConnection.kt +++ b/components/places/android/src/main/java/mozilla/appservices/places/PlacesConnection.kt @@ -262,6 +262,12 @@ open class PlacesReaderConnection internal constructor(conn: UniffiPlacesConnect } } + override fun countBookmarksInTrees(guids: List): UInt { + return readQueryCounters.measure { + this.conn.bookmarksCountBookmarksInTrees(guids) + } + } + private val readQueryCounters: PlacesManagerCounterMetrics by lazy { PlacesManagerCounterMetrics( PlacesManagerMetrics.readQueryCount, diff --git a/components/places/android/src/test/java/mozilla/appservices/places/PlacesConnectionTest.kt b/components/places/android/src/test/java/mozilla/appservices/places/PlacesConnectionTest.kt index 893272235a..10767e60fd 100644 --- a/components/places/android/src/test/java/mozilla/appservices/places/PlacesConnectionTest.kt +++ b/components/places/android/src/test/java/mozilla/appservices/places/PlacesConnectionTest.kt @@ -384,6 +384,40 @@ class PlacesConnectionTest { assertEquals(folder.f.parentGuid, BookmarkRoot.Unfiled.id) } + @Test + fun testCountBookmarks() { + assertEquals(db.countBookmarksInTrees(listOf(BookmarkRoot.Unfiled.id)), 0U) + + db.createBookmarkItem( + parentGUID = BookmarkRoot.Unfiled.id, + url = "https://www.example.com/", + title = "example", + ) + assertEquals(db.countBookmarksInTrees(listOf(BookmarkRoot.Unfiled.id)), 1U) + + val folderGUID = db.createFolder( + parentGUID = BookmarkRoot.Unfiled.id, + title = "example folder", + ) + // Folders don't get counted. + assertEquals(db.countBookmarksInTrees(listOf(BookmarkRoot.Unfiled.id)), 1U) + + // new item in the child folder does. + db.createBookmarkItem( + parentGUID = folderGUID, + url = "https://www.example.com/", + title = "example", + ) + assertEquals(db.countBookmarksInTrees(listOf(BookmarkRoot.Unfiled.id)), 2U) + + // A separator is not counted. + db.createSeparator( + parentGUID = BookmarkRoot.Unfiled.id, + position = 0u, + ) + assertEquals(db.countBookmarksInTrees(listOf(BookmarkRoot.Unfiled.id)), 2U) + } + @Test fun testHistoryMetricsGathering() { assertNull(PlacesManagerMetrics.writeQueryCount.testGetValue()) diff --git a/components/places/src/ffi.rs b/components/places/src/ffi.rs index 398572ecc4..adc4855028 100644 --- a/components/places/src/ffi.rs +++ b/components/places/src/ffi.rs @@ -573,6 +573,11 @@ impl PlacesConnection { self.with_conn(|conn| bookmarks::update_bookmark_from_info(conn, item)) } + #[handle_error(crate::Error)] + pub fn bookmarks_count_bookmarks_in_trees(&self, guids: &[Guid]) -> ApiResult { + self.with_conn(|conn| bookmarks::count_bookmarks_in_trees(conn, guids)) + } + #[handle_error(crate::Error)] pub fn places_history_import_from_ios( &self, diff --git a/components/places/src/places.udl b/components/places/src/places.udl index bd4dda9bc8..2221a6f0d8 100644 --- a/components/places/src/places.udl +++ b/components/places/src/places.udl @@ -204,6 +204,13 @@ interface PlacesConnection { [Throws=PlacesApiError] Guid bookmarks_insert(InsertableBookmarkItem bookmark); + // Counts the number of bookmarks in the bookmark tree under the specified GUID. Does not count + // the passed item, so an empty folder will return zero, as will a non-existing GUID or the + // guid of a non-folder item. + // Counts only bookmark items - ie, sub-folders and separators are not counted. + [Throws=PlacesApiError] + u32 bookmarks_count_bookmarks_in_trees([ByRef] sequence folder_guids); + [Throws=PlacesApiError] HistoryMigrationResult places_history_import_from_ios(string db_path, i64 last_sync_timestamp); }; diff --git a/components/places/src/storage/bookmarks.rs b/components/places/src/storage/bookmarks.rs index c67908bb2c..3ad450852c 100644 --- a/components/places/src/storage/bookmarks.rs +++ b/components/places/src/storage/bookmarks.rs @@ -786,6 +786,40 @@ pub fn bookmarks_get_url_for_keyword(db: &PlacesDb, keyword: &str) -> Result String { + assert_ne!(count, 0); + let mut s = "?,".repeat(count); + // Remove trailing comma + s.pop(); + s +} + +// Counts the number of bookmark items in the bookmark trees under the specified GUIDs. +// Does not count folder items, separators. A set of empty folders will return zero, as will +// a set of non-existing GUIDs or guids of a non-folder item. +// The result is implementation dependant if the trees overlap. +pub fn count_bookmarks_in_trees(db: &PlacesDb, item_guids: &[SyncGuid]) -> Result { + if item_guids.is_empty() { + return Ok(0); + } + let vars = repeat_vars(item_guids.len()); + let sql = format!( + r#" + WITH RECURSIVE bookmark_tree(id, parent, type) + AS ( + SELECT id, parent, type FROM moz_bookmarks + WHERE parent IN (SELECT id from moz_bookmarks WHERE guid IN ({vars})) + UNION ALL + SELECT bm.id, bm.parent, bm.type FROM moz_bookmarks bm, bookmark_tree bt WHERE bm.parent = bt.id + ) + SELECT COUNT(*) from bookmark_tree WHERE type = {}; + "#, + BookmarkType::Bookmark as u8 + ); + let params = rusqlite::params_from_iter(item_guids); + Ok(db.query_row_and_then(&sql, params, |row| row.get(0))?) +} + /// Erases all bookmarks and resets all Sync metadata. pub fn delete_everything(db: &PlacesDb) -> Result<()> { let tx = db.begin_transaction()?; @@ -1919,4 +1953,97 @@ mod tests { Ok(()) } + + #[test] + fn test_count_tree() -> Result<()> { + let conn = new_mem_connection(); + let unfiled = BookmarkRootGuid::Unfiled.as_guid(); + + insert_json_tree( + &conn, + json!({ + "guid": &unfiled, + "children": [ + { + "guid": "folder1_____", + "title": "A folder", + "children": [ + { + "guid": "bookmark1___", + "title": "bookmark in A folder", + "url": "https://www.example2.com/" + }, + { + "guid": "separator1__", + "type": BookmarkType::Separator, + }, + { + "guid": "bookmark2___", + "title": "next bookmark in A folder", + "url": "https://www.example3.com/" + }, + ] + }, + { + "guid": "folder2_____", + "title": "folder 2", + }, + { + "guid": "folder3_____", + "title": "Another folder", + "children": [ + { + "guid": "bookmark3___", + "title": "bookmark in folder 3", + "url": "https://www.example2.com/" + }, + { + "guid": "separator2__", + "type": BookmarkType::Separator, + }, + { + "guid": "bookmark4___", + "title": "next bookmark in folder 3", + "url": "https://www.example3.com/" + }, + ] + }, + ] + }), + ); + assert_eq!(count_bookmarks_in_trees(&conn, &[])?, 0); + // A folder with sub-folders + assert_eq!(count_bookmarks_in_trees(&conn, &[unfiled])?, 4); + // A folder with items but no folders. + assert_eq!( + count_bookmarks_in_trees(&conn, &[SyncGuid::from("folder1_____")])?, + 2 + ); + // Asking for a bookmark (ie, not a folder) or an invalid guid gives zero. + assert_eq!( + count_bookmarks_in_trees(&conn, &[SyncGuid::from("bookmark1___")])?, + 0 + ); + assert_eq!( + count_bookmarks_in_trees(&conn, &[SyncGuid::from("no_such_guid")])?, + 0 + ); + // empty folder also zero. + assert_eq!( + count_bookmarks_in_trees(&conn, &[SyncGuid::from("folder2_____")])?, + 0 + ); + // multiple folders + assert_eq!( + count_bookmarks_in_trees( + &conn, + &[ + SyncGuid::from("folder1_____"), + SyncGuid::from("folder3_____") + ] + )?, + 4 + ); + Ok(()) + } }