Allow to search for custom field values

This commit is contained in:
Eike Kettner 2020-11-22 18:25:27 +01:00
parent c5ab663091
commit 066c856981
4 changed files with 58 additions and 4 deletions

View File

@ -53,6 +53,9 @@ trait OItemSearch[F[_]] {
object OItemSearch { object OItemSearch {
type CustomValue = QItem.CustomValue
val CustomValue = QItem.CustomValue
type Query = QItem.Query type Query = QItem.Query
val Query = QItem.Query val Query = QItem.Query

View File

@ -4952,6 +4952,7 @@ components:
- inbox - inbox
- offset - offset
- limit - limit
- customValues
properties: properties:
tagsInclude: tagsInclude:
type: array type: array
@ -5031,6 +5032,10 @@ components:
format: date-time format: date-time
itemSubset: itemSubset:
$ref: "#/components/schemas/IdList" $ref: "#/components/schemas/IdList"
customValues:
type: array
items:
$ref: "#/components/schemas/CustomFieldValue"
ItemLight: ItemLight:
description: | description: |
An item with only a few important properties. An item with only a few important properties.

View File

@ -143,9 +143,13 @@ trait Conversions {
m.itemSubset m.itemSubset
.map(_.ids.flatMap(i => Ident.fromString(i).toOption).toSet) .map(_.ids.flatMap(i => Ident.fromString(i).toOption).toSet)
.filter(_.nonEmpty), .filter(_.nonEmpty),
m.customValues.map(mkCustomValue),
None None
) )
def mkCustomValue(v: CustomFieldValue): OItemSearch.CustomValue =
OItemSearch.CustomValue(v.field, v.value)
def mkItemList(v: Vector[OItemSearch.ListItem]): ItemLightList = { def mkItemList(v: Vector[OItemSearch.ListItem]): ItemLightList = {
val groups = v.groupBy(item => item.date.toUtcDate.toString.substring(0, 7)) val groups = v.groupBy(item => item.date.toUtcDate.toString.substring(0, 7))

View File

@ -139,7 +139,7 @@ object QItem {
val sources = RAttachmentSource.findByItemWithMeta(id) val sources = RAttachmentSource.findByItemWithMeta(id)
val archives = RAttachmentArchive.findByItemWithMeta(id) val archives = RAttachmentArchive.findByItemWithMeta(id)
val tags = RTag.findByItem(id) val tags = RTag.findByItem(id)
val customfields = findCustomFieldValues(id) val customfields = findCustomFieldValuesForItem(id)
for { for {
data <- q data <- q
@ -153,7 +153,9 @@ object QItem {
) )
} }
def findCustomFieldValues(itemId: Ident): ConnectionIO[Vector[ItemFieldValue]] = { def findCustomFieldValuesForItem(
itemId: Ident
): ConnectionIO[Vector[ItemFieldValue]] = {
val cfId = RCustomField.Columns.id.prefix("cf") val cfId = RCustomField.Columns.id.prefix("cf")
val cfName = RCustomField.Columns.name.prefix("cf") val cfName = RCustomField.Columns.name.prefix("cf")
val cfLabel = RCustomField.Columns.label.prefix("cf") val cfLabel = RCustomField.Columns.label.prefix("cf")
@ -191,6 +193,8 @@ object QItem {
notes: Option[String] notes: Option[String]
) )
case class CustomValue(field: Ident, value: String)
case class Query( case class Query(
account: AccountId, account: AccountId,
name: Option[String], name: Option[String],
@ -211,6 +215,7 @@ object QItem {
dueDateTo: Option[Timestamp], dueDateTo: Option[Timestamp],
allNames: Option[String], allNames: Option[String],
itemIds: Option[Set[Ident]], itemIds: Option[Set[Ident]],
customValues: Seq[CustomValue],
orderAsc: Option[RItem.Columns.type => Column] orderAsc: Option[RItem.Columns.type => Column]
) )
@ -236,6 +241,7 @@ object QItem {
None, None,
None, None,
None, None,
Seq.empty,
None None
) )
} }
@ -261,6 +267,35 @@ object QItem {
Batch(0, c) Batch(0, c)
} }
private def findCustomFieldValuesForColl(
coll: Ident,
cv: Seq[CustomValue]
): Seq[(String, Fragment)] = {
val cfId = RCustomField.Columns.id.prefix("cf")
val cfName = RCustomField.Columns.name.prefix("cf")
val cfColl = RCustomField.Columns.cid.prefix("cf")
val cvValue = RCustomFieldValue.Columns.value.prefix("cvf")
val cvField = RCustomFieldValue.Columns.field.prefix("cvf")
val cvItem = RCustomFieldValue.Columns.itemId.prefix("cvf")
val cfFrom =
RCustomFieldValue.table ++ fr"cvf INNER JOIN" ++ RCustomField.table ++ fr"cf ON" ++ cvField
.is(cfId)
def singleSelect(v: CustomValue) =
selectSimple(
Seq(cvItem),
cfFrom,
and(
cfColl.is(coll),
or(cfName.is(v.field), cfId.is(v.field)),
cvValue.is(v.value)
)
)
if (cv.isEmpty) Seq.empty
else Seq("customvalues" -> cv.map(singleSelect).reduce(_ ++ fr"INTERSECT" ++ _))
}
private def findItemsBase( private def findItemsBase(
q: Query, q: Query,
distinct: Boolean, distinct: Boolean,
@ -279,6 +314,7 @@ object QItem {
val orgCols = List(OC.oid, OC.name) val orgCols = List(OC.oid, OC.name)
val equipCols = List(EC.eid, EC.name) val equipCols = List(EC.eid, EC.name)
val folderCols = List(FC.id, FC.name) val folderCols = List(FC.id, FC.name)
val cvItem = RCustomFieldValue.Columns.itemId.prefix("cv")
val finalCols = commas( val finalCols = commas(
Seq( Seq(
@ -325,6 +361,9 @@ object QItem {
val withAttach = fr"SELECT COUNT(" ++ AC.id.f ++ fr") as num, " ++ AC.itemId.f ++ val withAttach = fr"SELECT COUNT(" ++ AC.id.f ++ fr") as num, " ++ AC.itemId.f ++
fr"from" ++ RAttachment.table ++ fr"GROUP BY (" ++ AC.itemId.f ++ fr")" fr"from" ++ RAttachment.table ++ fr"GROUP BY (" ++ AC.itemId.f ++ fr")"
val withCustomValues =
findCustomFieldValuesForColl(q.account.collective, q.customValues)
val selectKW = if (distinct) fr"SELECT DISTINCT" else fr"SELECT" val selectKW = if (distinct) fr"SELECT DISTINCT" else fr"SELECT"
withCTE( withCTE(
(Seq( (Seq(
@ -334,7 +373,7 @@ object QItem {
"equips" -> withEquips, "equips" -> withEquips,
"attachs" -> withAttach, "attachs" -> withAttach,
"folders" -> withFolder "folders" -> withFolder
) ++ ctes): _* ) ++ withCustomValues ++ ctes): _*
) ++ ) ++
selectKW ++ finalCols ++ fr" FROM items i" ++ selectKW ++ finalCols ++ fr" FROM items i" ++
fr"LEFT JOIN attachs a ON" ++ IC.id.prefix("i").is(AC.itemId.prefix("a")) ++ fr"LEFT JOIN attachs a ON" ++ IC.id.prefix("i").is(AC.itemId.prefix("a")) ++
@ -344,7 +383,10 @@ object QItem {
fr"LEFT JOIN equips e1 ON" ++ IC.concEquipment fr"LEFT JOIN equips e1 ON" ++ IC.concEquipment
.prefix("i") .prefix("i")
.is(EC.eid.prefix("e1")) ++ .is(EC.eid.prefix("e1")) ++
fr"LEFT JOIN folders f1 ON" ++ IC.folder.prefix("i").is(FC.id.prefix("f1")) fr"LEFT JOIN folders f1 ON" ++ IC.folder.prefix("i").is(FC.id.prefix("f1")) ++
(if (q.customValues.isEmpty) Fragment.empty
else
fr"INNER JOIN customvalues cv ON" ++ cvItem.is(IC.id.prefix("i")))
} }
def findItems( def findItems(