mirror of
https://github.com/TheAnachronism/docspell.git
synced 2025-04-05 10:59:33 +00:00
Improve on basic search summary
This commit is contained in:
parent
f3855628d5
commit
77627534bc
@ -145,10 +145,7 @@ object OItemSearch {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def findItemsSummary(q: Query): F[SearchSummary] =
|
def findItemsSummary(q: Query): F[SearchSummary] =
|
||||||
for {
|
store.transact(QItem.searchStats(q))
|
||||||
tags <- store.transact(QItem.searchTagSummary(q))
|
|
||||||
count <- store.transact(QItem.searchCountSummary(q))
|
|
||||||
} yield SearchSummary(count, tags)
|
|
||||||
|
|
||||||
def findAttachment(id: Ident, collective: Ident): F[Option[AttachmentData[F]]] =
|
def findAttachment(id: Ident, collective: Ident): F[Option[AttachmentData[F]]] =
|
||||||
store
|
store
|
||||||
|
@ -2,6 +2,7 @@ package docspell.common
|
|||||||
|
|
||||||
import java.time.LocalDate
|
import java.time.LocalDate
|
||||||
|
|
||||||
|
import cats.data.NonEmptyList
|
||||||
import cats.implicits._
|
import cats.implicits._
|
||||||
|
|
||||||
import io.circe._
|
import io.circe._
|
||||||
@ -92,7 +93,8 @@ object CustomFieldType {
|
|||||||
def bool: CustomFieldType = Bool
|
def bool: CustomFieldType = Bool
|
||||||
def money: CustomFieldType = Money
|
def money: CustomFieldType = Money
|
||||||
|
|
||||||
val all: List[CustomFieldType] = List(Text, Numeric, Date, Bool, Money)
|
val all: NonEmptyList[CustomFieldType] =
|
||||||
|
NonEmptyList.of(Text, Numeric, Date, Bool, Money)
|
||||||
|
|
||||||
def fromString(str: String): Either[String, CustomFieldType] =
|
def fromString(str: String): Either[String, CustomFieldType] =
|
||||||
str.toLowerCase match {
|
str.toLowerCase match {
|
||||||
|
@ -0,0 +1,17 @@
|
|||||||
|
package docspell.store.queries
|
||||||
|
|
||||||
|
import docspell.store.records.RCustomField
|
||||||
|
|
||||||
|
case class FieldStats(
|
||||||
|
field: RCustomField,
|
||||||
|
count: Int,
|
||||||
|
avg: BigDecimal,
|
||||||
|
sum: BigDecimal,
|
||||||
|
max: BigDecimal,
|
||||||
|
min: BigDecimal
|
||||||
|
)
|
||||||
|
|
||||||
|
object FieldStats {
|
||||||
|
def apply(field: RCustomField, count: Int): FieldStats =
|
||||||
|
FieldStats(field, count, BigDecimal(0), BigDecimal(0), BigDecimal(0), BigDecimal(0))
|
||||||
|
}
|
@ -15,7 +15,7 @@ import docspell.store.records._
|
|||||||
|
|
||||||
import doobie.implicits._
|
import doobie.implicits._
|
||||||
import doobie.{Query => _, _}
|
import doobie.{Query => _, _}
|
||||||
import org.log4s._
|
import org.log4s.getLogger
|
||||||
|
|
||||||
object QItem {
|
object QItem {
|
||||||
private[this] val logger = getLogger
|
private[this] val logger = getLogger
|
||||||
@ -234,13 +234,20 @@ object QItem {
|
|||||||
sql.query[ListItem].stream
|
sql.query[ListItem].stream
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def searchStats(q: Query): ConnectionIO[SearchSummary] =
|
||||||
|
for {
|
||||||
|
count <- searchCountSummary(q)
|
||||||
|
tags <- searchTagSummary(q)
|
||||||
|
fields <- searchFieldSummary(q)
|
||||||
|
} yield SearchSummary(count, tags, fields)
|
||||||
|
|
||||||
def searchTagSummary(q: Query): ConnectionIO[List[TagCount]] = {
|
def searchTagSummary(q: Query): ConnectionIO[List[TagCount]] = {
|
||||||
val tagFrom =
|
val tagFrom =
|
||||||
from(ti)
|
from(ti)
|
||||||
.innerJoin(tag, tag.tid === ti.tagId)
|
.innerJoin(tag, tag.tid === ti.tagId)
|
||||||
.innerJoin(i, i.id === ti.itemId)
|
.innerJoin(i, i.id === ti.itemId)
|
||||||
|
|
||||||
findItemsBase(q, 0)
|
findItemsBase(q, 0).unwrap
|
||||||
.withSelect(select(tag.all).append(count(i.id).as("num")))
|
.withSelect(select(tag.all).append(count(i.id).as("num")))
|
||||||
.changeFrom(_.prepend(tagFrom))
|
.changeFrom(_.prepend(tagFrom))
|
||||||
.changeWhere(c => c && queryCondition(q))
|
.changeWhere(c => c && queryCondition(q))
|
||||||
@ -251,13 +258,70 @@ object QItem {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def searchCountSummary(q: Query): ConnectionIO[Int] =
|
def searchCountSummary(q: Query): ConnectionIO[Int] =
|
||||||
findItemsBase(q, 0)
|
findItemsBase(q, 0).unwrap
|
||||||
.withSelect(Nel.of(count(i.id).as("num")))
|
.withSelect(Nel.of(count(i.id).as("num")))
|
||||||
.changeWhere(c => c && queryCondition(q))
|
.changeWhere(c => c && queryCondition(q))
|
||||||
.build
|
.build
|
||||||
.query[Int]
|
.query[Int]
|
||||||
.unique
|
.unique
|
||||||
|
|
||||||
|
def searchFieldSummary(q: Query): ConnectionIO[List[FieldStats]] = {
|
||||||
|
val fieldJoin =
|
||||||
|
from(cv)
|
||||||
|
.innerJoin(cf, cf.id === cv.field)
|
||||||
|
.innerJoin(i, i.id === cv.itemId)
|
||||||
|
|
||||||
|
val base =
|
||||||
|
findItemsBase(q, 0).unwrap
|
||||||
|
.changeFrom(_.prepend(fieldJoin))
|
||||||
|
.changeWhere(c => c && queryCondition(q))
|
||||||
|
.groupBy(GroupBy(cf.all))
|
||||||
|
|
||||||
|
val basicFields = Nel.of(
|
||||||
|
count(i.id).as("fc"),
|
||||||
|
lit(0).as("favg"),
|
||||||
|
lit(0).as("fsum"),
|
||||||
|
lit(0).as("fmax"),
|
||||||
|
lit(0).as("fmin")
|
||||||
|
)
|
||||||
|
val valueNum = cast(cv.value.s, "decimal").s
|
||||||
|
val numericFields = Nel.of(
|
||||||
|
count(i.id).as("fc"),
|
||||||
|
avg(valueNum).as("favg"),
|
||||||
|
sum(valueNum).as("fsum"),
|
||||||
|
max(valueNum).as("fmax"),
|
||||||
|
min(valueNum).as("fmin")
|
||||||
|
)
|
||||||
|
|
||||||
|
val numTypes = Nel.of(CustomFieldType.money, CustomFieldType.numeric)
|
||||||
|
val query =
|
||||||
|
union(
|
||||||
|
base
|
||||||
|
.withSelect(select(cf.all).concatNel(basicFields))
|
||||||
|
.changeWhere(c => c && cf.ftype.notIn(numTypes)),
|
||||||
|
base
|
||||||
|
.withSelect(select(cf.all).concatNel(numericFields))
|
||||||
|
.changeWhere(c => c && cf.ftype.in(numTypes))
|
||||||
|
).build.query[FieldStats].to[List]
|
||||||
|
|
||||||
|
val fallback = base
|
||||||
|
.withSelect(select(cf.all).concatNel(basicFields))
|
||||||
|
.build
|
||||||
|
.query[FieldStats]
|
||||||
|
.to[List]
|
||||||
|
|
||||||
|
query.attemptSql.flatMap {
|
||||||
|
case Right(res) => res.pure[ConnectionIO]
|
||||||
|
case Left(ex) =>
|
||||||
|
Logger
|
||||||
|
.log4s[ConnectionIO](logger)
|
||||||
|
.error(ex)(
|
||||||
|
s"Calculating custom field summary failed. You may have invalid custom field values according to their type."
|
||||||
|
) *>
|
||||||
|
fallback
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
def findSelectedItems(
|
def findSelectedItems(
|
||||||
q: Query,
|
q: Query,
|
||||||
maxNoteLen: Int,
|
maxNoteLen: Int,
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
package docspell.store.queries
|
package docspell.store.queries
|
||||||
|
|
||||||
case class SearchSummary(count: Int, tags: List[TagCount])
|
case class SearchSummary(count: Int, tags: List[TagCount], fields: List[FieldStats])
|
||||||
|
Loading…
x
Reference in New Issue
Block a user