Merge pull request #755 from eikek/fix-category-count

Fix category count
This commit is contained in:
eikek 2021-04-11 14:59:49 +02:00 committed by GitHub
commit cdbd65388d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 122 additions and 67 deletions

View File

@ -1,4 +1,12 @@
pull_request_rules: pull_request_rules:
- name: assign and label scala-steward's PRs
conditions:
- author=scala-steward
actions:
assign:
users: [eikek]
label:
add: ["type: dependencies"]
- name: automatically merge Scala Steward PRs on CI success - name: automatically merge Scala Steward PRs on CI success
conditions: conditions:
- author=scala-steward - author=scala-steward

View File

@ -71,6 +71,9 @@ object OCollective {
type TagCount = docspell.store.queries.TagCount type TagCount = docspell.store.queries.TagCount
val TagCount = docspell.store.queries.TagCount val TagCount = docspell.store.queries.TagCount
type CategoryCount = docspell.store.queries.CategoryCount
val CategoryCount = docspell.store.queries.CategoryCount
type InsightData = QCollective.InsightData type InsightData = QCollective.InsightData
val insightData = QCollective.InsightData val insightData = QCollective.InsightData

View File

@ -4242,6 +4242,7 @@ components:
required: required:
- count - count
- tagCloud - tagCloud
- tagCategoryCloud
- fieldStats - fieldStats
- folderStats - folderStats
properties: properties:
@ -4250,6 +4251,8 @@ components:
format: int32 format: int32
tagCloud: tagCloud:
$ref: "#/components/schemas/TagCloud" $ref: "#/components/schemas/TagCloud"
tagCategoryCloud:
$ref: "#/components/schemas/NameCloud"
fieldStats: fieldStats:
type: array type: array
items: items:
@ -4354,7 +4357,7 @@ components:
$ref: "#/components/schemas/TagCount" $ref: "#/components/schemas/TagCount"
TagCount: TagCount:
description: | description: |
Generic structure for counting something. Structure for counting tags.
required: required:
- tag - tag
- count - count
@ -4364,6 +4367,30 @@ components:
count: count:
type: integer type: integer
format: int32 format: int32
NameCloud:
description: |
A set of counters.
required:
- items
properties:
items:
type: array
items:
$ref: "#/components/schemas/NameCount"
NameCount:
description: |
Generic structure for counting something.
required:
- name
- count
properties:
name:
type: string
count:
type: integer
format: int32
AttachmentMeta: AttachmentMeta:
description: | description: |
Extracted meta data of an attachment. Extracted meta data of an attachment.

View File

@ -31,6 +31,7 @@ trait Conversions {
SearchStats( SearchStats(
sum.count, sum.count,
mkTagCloud(sum.tags), mkTagCloud(sum.tags),
mkTagCategoryCloud(sum.cats),
sum.fields.map(mkFieldStats), sum.fields.map(mkFieldStats),
sum.folders.map(mkFolderStats) sum.folders.map(mkFolderStats)
) )
@ -63,6 +64,9 @@ trait Conversions {
def mkTagCloud(tags: List[OCollective.TagCount]) = def mkTagCloud(tags: List[OCollective.TagCount]) =
TagCloud(tags.map(tc => TagCount(mkTag(tc.tag), tc.count))) TagCloud(tags.map(tc => TagCount(mkTag(tc.tag), tc.count)))
def mkTagCategoryCloud(tags: List[OCollective.CategoryCount]) =
NameCloud(tags.map(tc => NameCount(tc.category, tc.count)))
// attachment meta // attachment meta
def mkAttachmentMeta(rm: RAttachmentMeta): AttachmentMeta = def mkAttachmentMeta(rm: RAttachmentMeta): AttachmentMeta =
AttachmentMeta( AttachmentMeta(

View File

@ -9,11 +9,11 @@ object DBFunction {
val countAll: DBFunction = CountAll val countAll: DBFunction = CountAll
def countAs[A](column: Column[A]): DBFunction = def countAs[A](column: Column[A]): DBFunction =
Count(column) Count(column, false)
case object CountAll extends DBFunction case object CountAll extends DBFunction
case class Count(column: Column[_]) extends DBFunction case class Count(column: Column[_], distinct: Boolean) extends DBFunction
case class Max(expr: SelectExpr) extends DBFunction case class Max(expr: SelectExpr) extends DBFunction

View File

@ -63,7 +63,10 @@ trait DSL extends DoobieMeta {
FromExpr.From(sel, alias) FromExpr.From(sel, alias)
def count(c: Column[_]): DBFunction = def count(c: Column[_]): DBFunction =
DBFunction.Count(c) DBFunction.Count(c, false)
def countDistinct(c: Column[_]): DBFunction =
DBFunction.Count(c, true)
def countAll: DBFunction = def countAll: DBFunction =
DBFunction.CountAll DBFunction.CountAll

View File

@ -13,8 +13,9 @@ object DBFunctionBuilder extends CommonBuilder {
case DBFunction.CountAll => case DBFunction.CountAll =>
sql"COUNT(*)" sql"COUNT(*)"
case DBFunction.Count(col) => case DBFunction.Count(col, distinct) =>
sql"COUNT(" ++ column(col) ++ fr")" if (distinct) sql"COUNT(DISTINCT " ++ column(col) ++ fr")"
else sql"COUNT(" ++ column(col) ++ fr")"
case DBFunction.Max(expr) => case DBFunction.Max(expr) =>
sql"MAX(" ++ SelectExprBuilder.build(expr) ++ fr")" sql"MAX(" ++ SelectExprBuilder.build(expr) ++ fr")"

View File

@ -0,0 +1,3 @@
package docspell.store.queries
final case class CategoryCount(category: String, count: Int)

View File

@ -190,9 +190,38 @@ object QItem {
for { for {
count <- searchCountSummary(today)(q) count <- searchCountSummary(today)(q)
tags <- searchTagSummary(today)(q) tags <- searchTagSummary(today)(q)
cats <- searchTagCategorySummary(today)(q)
fields <- searchFieldSummary(today)(q) fields <- searchFieldSummary(today)(q)
folders <- searchFolderSummary(today)(q) folders <- searchFolderSummary(today)(q)
} yield SearchSummary(count, tags, fields, folders) } yield SearchSummary(count, tags, cats, fields, folders)
def searchTagCategorySummary(
today: LocalDate
)(q: Query): ConnectionIO[List[CategoryCount]] = {
val tagFrom =
from(ti)
.innerJoin(tag, tag.tid === ti.tagId)
.innerJoin(i, i.id === ti.itemId)
val tagCloud =
findItemsBase(q.fix, today, 0).unwrap
.withSelect(select(tag.category).append(countDistinct(i.id).as("num")))
.changeFrom(_.prepend(tagFrom))
.changeWhere(c => c && queryCondition(today, q.fix.account.collective, q.cond))
.groupBy(tag.category)
.build
.query[CategoryCount]
.to[List]
// the previous query starts from tags, so items with tag-count=0
// are not included they are fetched separately
for {
existing <- tagCloud
allCats <- RTag.listCategories(q.fix.account.collective)
other = allCats.diff(existing.map(_.category))
} yield existing ++ other.map(CategoryCount(_, 0))
}
def searchTagSummary(today: LocalDate)(q: Query): ConnectionIO[List[TagCount]] = { def searchTagSummary(today: LocalDate)(q: Query): ConnectionIO[List[TagCount]] = {
val tagFrom = val tagFrom =

View File

@ -3,6 +3,7 @@ package docspell.store.queries
case class SearchSummary( case class SearchSummary(
count: Int, count: Int,
tags: List[TagCount], tags: List[TagCount],
cats: List[CategoryCount],
fields: List[FieldStats], fields: List[FieldStats],
folders: List[FolderCount] folders: List[FolderCount]
) )

View File

@ -2,4 +2,4 @@ package docspell.store.queries
import docspell.store.records.RTag import docspell.store.records.RTag
case class TagCount(tag: RTag, count: Int) final case class TagCount(tag: RTag, count: Int)

View File

@ -92,7 +92,7 @@ type TextSearchModel
init : Flags -> Model init : Flags -> Model
init flags = init flags =
{ tagSelectModel = Comp.TagSelect.init [] [] { tagSelectModel = Comp.TagSelect.init [] [] [] []
, tagSelection = Comp.TagSelect.emptySelection , tagSelection = Comp.TagSelect.emptySelection
, directionModel = , directionModel =
Comp.Dropdown.makeSingleList Comp.Dropdown.makeSingleList
@ -483,7 +483,9 @@ updateDrop ddm flags settings msg model =
GetAllTagsResp (Ok stats) -> GetAllTagsResp (Ok stats) ->
let let
tagSel = tagSel =
Comp.TagSelect.modifyAll stats.tagCloud.items model.tagSelectModel Comp.TagSelect.modifyAll stats.tagCloud.items
stats.tagCategoryCloud.items
model.tagSelectModel
in in
{ model = { model | tagSelectModel = tagSel } { model = { model | tagSelectModel = tagSel }
, cmd = Cmd.none , cmd = Cmd.none
@ -500,9 +502,14 @@ updateDrop ddm flags settings msg model =
GetStatsResp (Ok stats) -> GetStatsResp (Ok stats) ->
let let
selectModel = tagCount =
List.sortBy .count stats.tagCloud.items List.sortBy .count stats.tagCloud.items
|> Comp.TagSelect.modifyCount model.tagSelectModel
catCount =
List.sortBy .count stats.tagCategoryCloud.items
selectModel =
Comp.TagSelect.modifyCount model.tagSelectModel tagCount catCount
model_ = model_ =
{ model { model

View File

@ -1,6 +1,5 @@
module Comp.TagSelect exposing module Comp.TagSelect exposing
( Category ( Model
, Model
, Msg , Msg
, Selection , Selection
, WorkModel , WorkModel
@ -18,6 +17,7 @@ module Comp.TagSelect exposing
, viewTagsDrop2 , viewTagsDrop2
) )
import Api.Model.NameCount exposing (NameCount)
import Api.Model.Tag exposing (Tag) import Api.Model.Tag exposing (Tag)
import Api.Model.TagCount exposing (TagCount) import Api.Model.TagCount exposing (TagCount)
import Data.Icons as I import Data.Icons as I
@ -38,9 +38,9 @@ import Util.Maybe
type alias Model = type alias Model =
{ availableTags : Dict String TagCount { availableTags : Dict String TagCount
, availableCats : Dict String Category , availableCats : Dict String NameCount
, tagCounts : List TagCount , tagCounts : List TagCount
, categoryCounts : List Category , categoryCounts : List NameCount
, filterTerm : Maybe String , filterTerm : Maybe String
, expandedTags : Bool , expandedTags : Bool
, expandedCats : Bool , expandedCats : Bool
@ -48,23 +48,16 @@ type alias Model =
} }
type alias Category = init : List TagCount -> List NameCount -> List TagCount -> List NameCount -> Model
{ name : String init allTags allCats tags cats =
, count : Int
}
init : List TagCount -> List TagCount -> Model
init allTags tags =
{ availableTags = { availableTags =
List.map (\e -> ( e.tag.id, e )) allTags List.map (\e -> ( e.tag.id, e )) allTags
|> Dict.fromList |> Dict.fromList
, availableCats = sumCategories allTags , availableCats =
List.map (\e -> ( e.name, e )) allCats
|> Dict.fromList
, tagCounts = tags , tagCounts = tags
, categoryCounts = , categoryCounts = cats
sumCategories tags
|> Dict.toList
|> List.map Tuple.second
, filterTerm = Nothing , filterTerm = Nothing
, expandedTags = False , expandedTags = False
, expandedCats = False , expandedCats = False
@ -72,24 +65,23 @@ init allTags tags =
} }
modifyAll : List TagCount -> Model -> Model modifyAll : List TagCount -> List NameCount -> Model -> Model
modifyAll allTags model = modifyAll allTags allCats model =
{ model { model
| availableTags = | availableTags =
List.map (\e -> ( e.tag.id, e )) allTags List.map (\e -> ( e.tag.id, e )) allTags
|> Dict.fromList |> Dict.fromList
, availableCats = sumCategories allTags , availableCats =
List.map (\e -> ( e.name, e )) allCats
|> Dict.fromList
} }
modifyCount : Model -> List TagCount -> Model modifyCount : Model -> List TagCount -> List NameCount -> Model
modifyCount model tags = modifyCount model tags cats =
{ model { model
| tagCounts = tags | tagCounts = tags
, categoryCounts = , categoryCounts = cats
sumCategories tags
|> Dict.toList
|> List.map Tuple.second
} }
@ -108,34 +100,11 @@ toggleTag id =
ToggleTag id ToggleTag id
sumCategories : List TagCount -> Dict String Category
sumCategories tags =
let
filterCat tc =
Maybe.map (\cat -> Category cat tc.count) tc.tag.category
withCats =
List.filterMap filterCat tags
sum cat mc =
Maybe.map ((+) cat.count) mc
|> Maybe.withDefault cat.count
|> Just
sumCounts cat dict =
Dict.update cat.name (sum cat) dict
cats =
List.foldl sumCounts Dict.empty withCats
in
Dict.map (\name -> \count -> Category name count) cats
type alias Selection = type alias Selection =
{ includeTags : List TagCount { includeTags : List TagCount
, excludeTags : List TagCount , excludeTags : List TagCount
, includeCats : List Category , includeCats : List NameCount
, excludeCats : List Category , excludeCats : List NameCount
} }
@ -145,7 +114,7 @@ emptySelection =
type alias WorkModel = type alias WorkModel =
{ filteredCats : List Category { filteredCats : List NameCount
, filteredTags : List TagCount , filteredTags : List TagCount
, selectedTags : Dict String Bool , selectedTags : Dict String Bool
, selectedCats : Dict String Bool , selectedCats : Dict String Bool
@ -166,7 +135,7 @@ orderTagCountStable model tagCounts =
List.sortBy order tagCounts List.sortBy order tagCounts
orderCatCountStable : Model -> List Category -> List Category orderCatCountStable : Model -> List NameCount -> List NameCount
orderCatCountStable model catCounts = orderCatCountStable model catCounts =
let let
order cat = order cat =
@ -193,7 +162,7 @@ removeEmptyTagCounts sel tagCounts =
List.filter (\tc -> isSelected tc || tc.count > 0) tagCounts List.filter (\tc -> isSelected tc || tc.count > 0) tagCounts
removeEmptyCatCounts : Selection -> List Category -> List Category removeEmptyCatCounts : Selection -> List NameCount -> List NameCount
removeEmptyCatCounts sel catCounts = removeEmptyCatCounts sel catCounts =
let let
selected = selected =
@ -548,7 +517,7 @@ viewTagItem2 ddm settings model tag =
] ]
viewCategoryItem2 : UiSettings -> WorkModel -> Category -> Html Msg viewCategoryItem2 : UiSettings -> WorkModel -> NameCount -> Html Msg
viewCategoryItem2 settings model cat = viewCategoryItem2 settings model cat =
let let
state = state =