Skip to content

Commit

Permalink
Optimize filtering with label = expression
Browse files Browse the repository at this point in the history
  • Loading branch information
grzesiek2010 committed Jan 7, 2025
1 parent 184e878 commit 3e87ac0
Show file tree
Hide file tree
Showing 6 changed files with 130 additions and 11 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -222,6 +222,20 @@ class DatabaseEntitiesRepository(context: Context, dbPath: String) : EntitiesRep
}
}

override fun getByLabel(list: String, label: String?): List<Entity.Saved> {
if (!listExists(list)) {
return emptyList()
}

return queryWithAttachedRowId(
list,
selectionColumn = EntitiesTable.COLUMN_LABEL,
selectionArg = label
).foldAndClose {
mapCursorRowToEntity(it, it.getInt(ROW_ID))
}
}

override fun getAllByProperty(
list: String,
property: String,
Expand Down Expand Up @@ -290,18 +304,30 @@ class DatabaseEntitiesRepository(context: Context, dbPath: String) : EntitiesRep
private fun queryWithAttachedRowId(
list: String,
selectionColumn: String,
selectionArg: String
selectionArg: String?
): Cursor {
return databaseConnection.withConnection {
readableDatabase.rawQuery(
"""
SELECT *, i.$ROW_ID
FROM "$list" e, "${getRowIdTableName(list)}" i
WHERE e._id = i._id AND $selectionColumn = ?
ORDER BY i.$ROW_ID
""".trimIndent(),
arrayOf(selectionArg)
)
if (selectionArg == null) {
readableDatabase.rawQuery(
"""
SELECT *, i.$ROW_ID
FROM "$list" e, "${getRowIdTableName(list)}" i
WHERE e._id = i._id AND $selectionColumn IS NULL
ORDER BY i.$ROW_ID
""".trimIndent(),
null
)
} else {
readableDatabase.rawQuery(
"""
SELECT *, i.$ROW_ID
FROM "$list" e, "${getRowIdTableName(list)}" i
WHERE e._id = i._id AND $selectionColumn = ?
ORDER BY i.$ROW_ID
""".trimIndent(),
arrayOf(selectionArg)
)
}
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -515,6 +515,86 @@ abstract class EntitiesRepositoryTest {
assertThat(queriedCanet, equalTo(otherFavouriteWines.first { it.id == "2" }))
}

@Test
fun `#getByLabel returns entities with matching label`() {
val repository = buildSubject()

val leoville1 = Entity.New("1", "Léoville Barton 2008")
val canet = Entity.New("2", "Pontet-Canet 2014")
val leoville2 = Entity.New("3", "Léoville Barton 2008")

repository.save("wines", leoville1, canet, leoville2)

val wines = repository.getEntities("wines")

val queriedLeoville = repository.getByLabel("wines", "Léoville Barton 2008")
assertThat(queriedLeoville, containsInAnyOrder(wines.first { it.id == "1" }, wines.first { it.id == "3" }))

val queriedCanet = repository.getByLabel("wines", "Pontet-Canet 2014")
assertThat(queriedCanet, contains(wines.first { it.id == "2" }))
}

@Test
fun `#getByLabel returns entities with matching label when label is null`() {
val repository = buildSubject()

val leoville = Entity.New("1", null)
val canet = Entity.New("2", "Pontet-Canet 2014")
repository.save("wines", leoville, canet)

val queriedNull = repository.getByLabel("wines", null)
val wines = repository.getEntities("wines")
assertThat(queriedNull, contains(wines.first { it.id == "1" }))
}

@Test
fun `#getByLabel returns empty list when there are no matches`() {
val repository = buildSubject()

val leoville = Entity.New("1", "Léoville Barton 2008")
val canet = Entity.New("2", "Pontet-Canet 2014")
repository.save("wines", leoville, canet)

assertThat(repository.getByLabel("wines", "Ardbeg 10"), equalTo(emptyList()))
}

@Test
fun `#getByLabel returns empty list when there is a match in a different list`() {
val repository = buildSubject()

val leoville = Entity.New("1", "Léoville Barton 2008")
val ardbeg = Entity.New("2", "Ardbeg 10")
repository.save("wines", leoville)
repository.save("whisky", ardbeg)

assertThat(repository.getByLabel("whisky", "Léoville Barton 2008"), equalTo(emptyList()))
}

@Test
fun `#getByLabel returns empty list where there are no entities in the list`() {
val repository = buildSubject()
assertThat(repository.getByLabel("wines", "Léoville Barton 2008"), equalTo(emptyList()))
}

@Test
fun `#getByLabel supports list names with dots and dashes`() {
val repository = buildSubject()

val leoville = Entity.New("1", "Léoville Barton 2008")
val canet = Entity.New("2", "Pontet-Canet 2014")
repository.save("favourite-wines", leoville)
repository.save("other.favourite.wines", canet)

val favouriteWines = repository.getEntities("favourite-wines")
val otherFavouriteWines = repository.getEntities("other.favourite.wines")

val queriedLeoville = repository.getByLabel("favourite-wines", "Léoville Barton 2008")
assertThat(queriedLeoville, contains(favouriteWines.first { it.id == "1" }))

val queriedCanet = repository.getByLabel("other.favourite.wines", "Pontet-Canet 2014")
assertThat(queriedCanet, contains(otherFavouriteWines.first { it.id == "2" }))
}

@Test
fun `#getByAllByProperty returns entities with matching property value`() {
val repository = buildSubject()
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,10 @@ class LocalEntitiesInstanceAdapter(private val entitiesRepository: EntitiesRepos
}

EntityItemElement.LABEL -> {
filterAndConvertEntities(instanceId) { it.label == value }
entitiesRepository.getByLabel(
instanceId,
value
).map { convertToElement(it) }
}

EntityItemElement.VERSION -> {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ interface EntitiesRepository {
fun addList(list: String)
fun delete(id: String)
fun getById(list: String, id: String): Entity.Saved?
fun getByLabel(list: String, label: String?): List<Entity.Saved>
fun getAllByProperty(list: String, property: String, value: String): List<Entity.Saved>
fun getByIndex(list: String, index: Int): Entity.Saved?
fun updateListHash(list: String, hash: String)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,10 @@ class InMemEntitiesRepository : EntitiesRepository {
return getEntities(list).firstOrNull { it.id == id }
}

override fun getByLabel(list: String, label: String?): List<Entity.Saved> {
return getEntities(list).filter { it.label == label }
}

override fun getAllByProperty(
list: String,
property: String,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -546,6 +546,11 @@ private class MeasurableEntitiesRepository(private val wrapped: EntitiesReposito
return wrapped.getById(list, id)
}

override fun getByLabel(list: String, label: String?): List<Entity.Saved> {
accesses += 1
return wrapped.getByLabel(list, label)
}

override fun getAllByProperty(
list: String,
property: String,
Expand Down

0 comments on commit 3e87ac0

Please sign in to comment.