Merge pull request #140 from eikek/itemlist-with-tags

Itemlist with tags
This commit is contained in:
eikek 2020-06-07 18:12:49 +02:00 committed by GitHub
commit c595f3b737
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 146 additions and 19 deletions

View File

@ -15,6 +15,7 @@ import OItem.{
Batch, Batch,
ItemData, ItemData,
ListItem, ListItem,
ListItemWithTags,
Query Query
} }
import bitpeace.{FileMeta, RangeDef} import bitpeace.{FileMeta, RangeDef}
@ -27,6 +28,9 @@ trait OItem[F[_]] {
def findItems(q: Query, batch: Batch): F[Vector[ListItem]] def findItems(q: Query, batch: Batch): F[Vector[ListItem]]
/** Same as `findItems` but does more queries per item to find all tags. */
def findItemsWithTags(q: Query, batch: Batch): F[Vector[ListItemWithTags]]
def findAttachment(id: Ident, collective: Ident): F[Option[AttachmentData[F]]] def findAttachment(id: Ident, collective: Ident): F[Option[AttachmentData[F]]]
def findAttachmentSource( def findAttachmentSource(
@ -91,6 +95,9 @@ object OItem {
type ListItem = QItem.ListItem type ListItem = QItem.ListItem
val ListItem = QItem.ListItem val ListItem = QItem.ListItem
type ListItemWithTags = QItem.ListItemWithTags
val ListItemWithTags = QItem.ListItemWithTags
type ItemData = QItem.ItemData type ItemData = QItem.ItemData
val ItemData = QItem.ItemData val ItemData = QItem.ItemData
@ -148,6 +155,12 @@ object OItem {
.compile .compile
.toVector .toVector
def findItemsWithTags(q: Query, batch: Batch): F[Vector[ListItemWithTags]] =
store
.transact(QItem.findItemsWithTags(q, batch).take(batch.limit.toLong))
.compile
.toVector
def findAttachment(id: Ident, collective: Ident): F[Option[AttachmentData[F]]] = def findAttachment(id: Ident, collective: Ident): F[Option[AttachmentData[F]]] =
store store
.transact(RAttachment.findByIdAndCollective(id, collective)) .transact(RAttachment.findByIdAndCollective(id, collective))

View File

@ -987,7 +987,31 @@ paths:
summary: Search for items. summary: Search for items.
description: | description: |
Search for items given a search form. The results are grouped Search for items given a search form. The results are grouped
by month by default. by month by default. Tags are *not* resolved! The results will
always contain an empty list for item tags. Use
`/searchWithTags` to also retrieve all tags of an item.
security:
- authTokenHeader: []
requestBody:
content:
application/json:
schema:
$ref: "#/components/schemas/ItemSearch"
responses:
200:
description: Ok
content:
application/json:
schema:
$ref: "#/components/schemas/ItemLightList"
/sec/item/searchWithTags:
post:
tags: [ Item ]
summary: Search for items.
description: |
Search for items given a search form. The results are grouped
by month by default. For each item, its tags are also
returned. This uses more queries and is therefore slower.
security: security:
- authTokenHeader: [] - authTokenHeader: []
requestBody: requestBody:
@ -3188,6 +3212,7 @@ components:
- date - date
- source - source
- fileCount - fileCount
- tags
properties: properties:
id: id:
type: string type: string
@ -3221,6 +3246,10 @@ components:
fileCount: fileCount:
type: integer type: integer
format: int32 format: int32
tags:
type: array
items:
$ref: "#/components/schemas/Tag"
IdName: IdName:
description: | description: |
The identifier and a human readable name of some entity. The identifier and a human readable name of some entity.

View File

@ -135,6 +135,17 @@ trait Conversions {
ItemLightList(gs) ItemLightList(gs)
} }
def mkItemListWithTags(v: Vector[OItem.ListItemWithTags]): ItemLightList = {
val groups = v.groupBy(ti => ti.item.date.toUtcDate.toString.substring(0, 7))
def mkGroup(g: (String, Vector[OItem.ListItemWithTags])): ItemLightGroup =
ItemLightGroup(g._1, g._2.map(mkItemLightWithTags).toList)
val gs =
groups.map(mkGroup _).toList.sortWith((g1, g2) => g1.name.compareTo(g2.name) >= 0)
ItemLightList(gs)
}
def mkItemLight(i: OItem.ListItem): ItemLight = def mkItemLight(i: OItem.ListItem): ItemLight =
ItemLight( ItemLight(
i.id, i.id,
@ -148,9 +159,13 @@ trait Conversions {
i.corrPerson.map(mkIdName), i.corrPerson.map(mkIdName),
i.concPerson.map(mkIdName), i.concPerson.map(mkIdName),
i.concEquip.map(mkIdName), i.concEquip.map(mkIdName),
i.fileCount i.fileCount,
Nil
) )
def mkItemLightWithTags(i: OItem.ListItemWithTags): ItemLight =
mkItemLight(i.item).copy(tags = i.tags.map(mkTag))
// job // job
def mkJobQueueState(state: OJob.CollectiveQueueState): JobQueueState = { def mkJobQueueState(state: OJob.CollectiveQueueState): JobQueueState = {
def desc(f: JobDetail => Option[Timestamp])(j1: JobDetail, j2: JobDetail): Boolean = { def desc(f: JobDetail => Option[Timestamp])(j1: JobDetail, j2: JobDetail): Boolean = {

View File

@ -36,6 +36,19 @@ object ItemRoutes {
resp <- Ok(Conversions.mkItemList(items)) resp <- Ok(Conversions.mkItemList(items))
} yield resp } yield resp
case req @ POST -> Root / "searchWithTags" =>
for {
mask <- req.as[ItemSearch]
_ <- logger.ftrace(s"Got search mask: $mask")
query = Conversions.mkQuery(mask, user.account.collective)
_ <- logger.ftrace(s"Running query: $query")
items <- backend.item.findItemsWithTags(
query,
Batch(mask.offset, mask.limit).restrictLimitTo(500)
)
resp <- Ok(Conversions.mkItemListWithTags(items))
} yield resp
case GET -> Root / Ident(id) => case GET -> Root / Ident(id) =>
for { for {
item <- backend.item.findItem(id, user.account.collective) item <- backend.item.findItem(id, user.account.collective)

View File

@ -4,6 +4,7 @@ import bitpeace.FileMeta
import cats.effect.Sync import cats.effect.Sync
import cats.data.OptionT import cats.data.OptionT
import cats.implicits._ import cats.implicits._
import cats.effect.concurrent.Ref
import fs2.Stream import fs2.Stream
import doobie._ import doobie._
import doobie.implicits._ import doobie.implicits._
@ -255,18 +256,10 @@ object QItem {
) ++ ) ++
fr"SELECT DISTINCT" ++ finalCols ++ fr" FROM items i" ++ fr"SELECT DISTINCT" ++ 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")) ++
fr"LEFT JOIN persons p0 ON" ++ IC.corrPerson fr"LEFT JOIN persons p0 ON" ++ IC.corrPerson.prefix("i").is(PC.pid.prefix("p0")) ++
.prefix("i") fr"LEFT JOIN orgs o0 ON" ++ IC.corrOrg.prefix("i").is(OC.oid.prefix("o0")) ++
.is(PC.pid.prefix("p0")) ++ // i.corrperson = p0.pid" ++ fr"LEFT JOIN persons p1 ON" ++ IC.concPerson.prefix("i").is(PC.pid.prefix("p1")) ++
fr"LEFT JOIN orgs o0 ON" ++ IC.corrOrg fr"LEFT JOIN equips e1 ON" ++ IC.concEquipment.prefix("i").is(EC.eid.prefix("e1"))
.prefix("i")
.is(OC.oid.prefix("o0")) ++ // i.corrorg = o0.oid" ++
fr"LEFT JOIN persons p1 ON" ++ IC.concPerson
.prefix("i")
.is(PC.pid.prefix("p1")) ++ // i.concperson = p1.pid" ++
fr"LEFT JOIN equips e1 ON" ++ IC.concEquipment
.prefix("i")
.is(EC.eid.prefix("e1")) // i.concequipment = e1.eid"
// inclusive tags are AND-ed // inclusive tags are AND-ed
val tagSelectsIncl = q.tagsInclude val tagSelectsIncl = q.tagsInclude
@ -339,6 +332,43 @@ object QItem {
frag.query[ListItem].stream frag.query[ListItem].stream
} }
case class ListItemWithTags(item: ListItem, tags: List[RTag])
/** Same as `findItems` but resolves the tags for each item. Note that
* this is implemented by running an additional query per item.
*/
def findItemsWithTags(
q: Query,
batch: Batch
): Stream[ConnectionIO, ListItemWithTags] = {
def findTag(
cache: Ref[ConnectionIO, Map[Ident, RTag]],
tagItem: RTagItem
): ConnectionIO[Option[RTag]] =
for {
cc <- cache.get
fromCache = cc.get(tagItem.tagId)
orFromDB <-
if (fromCache.isDefined) fromCache.pure[ConnectionIO]
else RTag.findById(tagItem.tagId)
_ <-
if (fromCache.isDefined) ().pure[ConnectionIO]
else
orFromDB match {
case Some(t) => cache.update(tmap => tmap.updated(t.tagId, t))
case None => ().pure[ConnectionIO]
}
} yield orFromDB
for {
resolvedTags <- Stream.eval(Ref.of[ConnectionIO, Map[Ident, RTag]](Map.empty))
item <- findItems(q, batch)
tagItems <- Stream.eval(RTagItem.findByItem(item.id))
tags <- Stream.eval(tagItems.traverse(ti => findTag(resolvedTags, ti)))
ftags = tags.flatten.filter(t => t.collective == q.collective)
} yield ListItemWithTags(item, ftags.toList.sortBy(_.name))
}
def delete[F[_]: Sync](store: Store[F])(itemId: Ident, collective: Ident): F[Int] = def delete[F[_]: Sync](store: Store[F])(itemId: Ident, collective: Ident): F[Int] =
for { for {
tn <- store.transact(RTagItem.deleteItemTags(itemId)) tn <- store.transact(RTagItem.deleteItemTags(itemId))

View File

@ -38,4 +38,7 @@ object RTagItem {
tagFrag = tagValues.map(v => fr"${v.tagItemId},${v.itemId},${v.tagId}") tagFrag = tagValues.map(v => fr"${v.tagItemId},${v.itemId},${v.tagId}")
ins <- insertRows(table, all, tagFrag).update.run ins <- insertRows(table, all, tagFrag).update.run
} yield ins } yield ins
def findByItem(item: Ident): ConnectionIO[Vector[RTagItem]] =
selectSimple(all, table, itemId.is(item)).query[RTagItem].to[Vector]
} }

View File

@ -1029,7 +1029,7 @@ moveAttachmentBefore flags itemId data receive =
itemSearch : Flags -> ItemSearch -> (Result Http.Error ItemLightList -> msg) -> Cmd msg itemSearch : Flags -> ItemSearch -> (Result Http.Error ItemLightList -> msg) -> Cmd msg
itemSearch flags search receive = itemSearch flags search receive =
Http2.authPost Http2.authPost
{ url = flags.config.baseUrl ++ "/api/v1/sec/item/search" { url = flags.config.baseUrl ++ "/api/v1/sec/item/searchWithTags"
, account = getAccount flags , account = getAccount flags
, body = Http.jsonBody (Api.Model.ItemSearch.encode search) , body = Http.jsonBody (Api.Model.ItemSearch.encode search)
, expect = Http.expectJson receive Api.Model.ItemLightList.decoder , expect = Http.expectJson receive Api.Model.ItemLightList.decoder

View File

@ -162,7 +162,7 @@ viewItem item =
, Util.String.underscoreToSpace item.name , Util.String.underscoreToSpace item.name
|> text |> text
] ]
, span [ class "meta" ] , div [ class "meta" ]
[ div [ div
[ classList [ classList
[ ( "ui ribbon label", True ) [ ( "ui ribbon label", True )
@ -173,11 +173,35 @@ viewItem item =
[ i [ class "exclamation icon" ] [] [ i [ class "exclamation icon" ] []
, text " New" , text " New"
] ]
, span
[ classList
[ ( "right floated", not isConfirmed )
]
] ]
, span [ class "right floated meta" ]
[ Util.Time.formatDate item.date |> text [ Util.Time.formatDate item.date |> text
] ]
] ]
, div [ class "meta description" ]
[ div
[ classList
[ ( "ui right floated tiny labels", True )
, ( "invisible hidden", item.tags == [] )
]
]
(List.map
(\tag ->
div
[ classList
[ ( "ui basic label", True )
, ( "blue", tag.category /= Nothing )
]
]
[ text tag.name ]
)
item.tags
)
]
]
, div [ class "content" ] , div [ class "content" ]
[ div [ class "ui horizontal list" ] [ div [ class "ui horizontal list" ]
[ div [ div

View File

@ -36,7 +36,7 @@ type alias UiSettings =
defaults : UiSettings defaults : UiSettings
defaults = defaults =
{ itemSearchPageSize = 90 { itemSearchPageSize = 60
} }