mirror of
https://github.com/TheAnachronism/docspell.git
synced 2025-04-05 19:09:32 +00:00
Merge pull request #140 from eikek/itemlist-with-tags
Itemlist with tags
This commit is contained in:
commit
c595f3b737
@ -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))
|
||||||
|
@ -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.
|
||||||
|
@ -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 = {
|
||||||
|
@ -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)
|
||||||
|
@ -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))
|
||||||
|
@ -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]
|
||||||
}
|
}
|
||||||
|
@ -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
|
||||||
|
@ -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,9 +173,33 @@ viewItem item =
|
|||||||
[ i [ class "exclamation icon" ] []
|
[ i [ class "exclamation icon" ] []
|
||||||
, text " New"
|
, text " New"
|
||||||
]
|
]
|
||||||
|
, span
|
||||||
|
[ classList
|
||||||
|
[ ( "right floated", not isConfirmed )
|
||||||
|
]
|
||||||
|
]
|
||||||
|
[ Util.Time.formatDate item.date |> text
|
||||||
|
]
|
||||||
]
|
]
|
||||||
, span [ class "right floated meta" ]
|
, div [ class "meta description" ]
|
||||||
[ Util.Time.formatDate item.date |> text
|
[ 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" ]
|
||||||
|
@ -36,7 +36,7 @@ type alias UiSettings =
|
|||||||
|
|
||||||
defaults : UiSettings
|
defaults : UiSettings
|
||||||
defaults =
|
defaults =
|
||||||
{ itemSearchPageSize = 90
|
{ itemSearchPageSize = 60
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user