2021-07-30 03:40:47 +00:00
|
|
|
package models
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
|
|
|
"git.handmade.network/hmn/hmn/src/db"
|
|
|
|
"git.handmade.network/hmn/hmn/src/oops"
|
|
|
|
"github.com/jackc/pgx/v4/pgxpool"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Subforum struct {
|
|
|
|
ID int `db:"id"`
|
|
|
|
|
|
|
|
ParentID *int `db:"parent_id"`
|
|
|
|
ProjectID int `db:"project_id"`
|
|
|
|
|
|
|
|
Slug string `db:"slug"`
|
|
|
|
Name string `db:"name"`
|
|
|
|
Blurb string `db:"blurb"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type SubforumTree map[int]*SubforumTreeNode
|
|
|
|
|
|
|
|
type SubforumTreeNode struct {
|
|
|
|
Subforum
|
|
|
|
Parent *SubforumTreeNode
|
|
|
|
Children []*SubforumTreeNode
|
|
|
|
}
|
|
|
|
|
|
|
|
func (node *SubforumTreeNode) GetLineage() []*Subforum {
|
|
|
|
current := node
|
|
|
|
length := 0
|
|
|
|
for current != nil {
|
|
|
|
current = current.Parent
|
|
|
|
length += 1
|
|
|
|
}
|
|
|
|
result := make([]*Subforum, length)
|
|
|
|
current = node
|
|
|
|
for i := length - 1; i >= 0; i -= 1 {
|
|
|
|
result[i] = ¤t.Subforum
|
|
|
|
current = current.Parent
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetFullSubforumTree(ctx context.Context, conn *pgxpool.Pool) SubforumTree {
|
2022-04-16 17:49:29 +00:00
|
|
|
subforums, err := db.Query[Subforum](ctx, conn,
|
2021-07-30 03:40:47 +00:00
|
|
|
`
|
|
|
|
SELECT $columns
|
2022-05-07 13:11:05 +00:00
|
|
|
FROM subforum
|
2021-09-23 06:18:45 +00:00
|
|
|
ORDER BY sort, id ASC
|
2021-07-30 03:40:47 +00:00
|
|
|
`,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
panic(oops.New(err, "failed to fetch subforum tree"))
|
|
|
|
}
|
|
|
|
|
2022-04-16 17:49:29 +00:00
|
|
|
sfTreeMap := make(map[int]*SubforumTreeNode, len(subforums))
|
|
|
|
for _, sf := range subforums {
|
|
|
|
sfTreeMap[sf.ID] = &SubforumTreeNode{Subforum: *sf}
|
2021-07-30 03:40:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, node := range sfTreeMap {
|
|
|
|
if node.ParentID != nil {
|
|
|
|
node.Parent = sfTreeMap[*node.ParentID]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-16 17:49:29 +00:00
|
|
|
for _, cat := range subforums {
|
2022-06-14 19:52:50 +00:00
|
|
|
// NOTE(asaf): Doing this in a separate loop over subforums to ensure that Children are in db order.
|
2021-07-30 03:40:47 +00:00
|
|
|
node := sfTreeMap[cat.ID]
|
|
|
|
if node.Parent != nil {
|
|
|
|
node.Parent.Children = append(node.Parent.Children, node)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return sfTreeMap
|
|
|
|
}
|
|
|
|
|
|
|
|
type SubforumLineageBuilder struct {
|
|
|
|
Tree SubforumTree
|
|
|
|
SubforumCache map[int][]*Subforum
|
|
|
|
SlugCache map[int][]string
|
|
|
|
}
|
|
|
|
|
|
|
|
func MakeSubforumLineageBuilder(fullSubforumTree SubforumTree) *SubforumLineageBuilder {
|
|
|
|
return &SubforumLineageBuilder{
|
|
|
|
Tree: fullSubforumTree,
|
|
|
|
SubforumCache: make(map[int][]*Subforum),
|
|
|
|
SlugCache: make(map[int][]string),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cl *SubforumLineageBuilder) GetLineage(sfId int) []*Subforum {
|
|
|
|
_, ok := cl.SubforumCache[sfId]
|
|
|
|
if !ok {
|
|
|
|
cl.SubforumCache[sfId] = cl.Tree[sfId].GetLineage()
|
|
|
|
}
|
|
|
|
return cl.SubforumCache[sfId]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cl *SubforumLineageBuilder) GetSubforumLineage(sfId int) []*Subforum {
|
|
|
|
return cl.GetLineage(sfId)[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cl *SubforumLineageBuilder) GetLineageSlugs(sfId int) []string {
|
|
|
|
_, ok := cl.SlugCache[sfId]
|
|
|
|
if !ok {
|
|
|
|
lineage := cl.GetLineage(sfId)
|
|
|
|
result := make([]string, 0, len(lineage))
|
|
|
|
for _, cat := range lineage {
|
|
|
|
result = append(result, cat.Slug)
|
|
|
|
}
|
|
|
|
cl.SlugCache[sfId] = result
|
|
|
|
}
|
|
|
|
return cl.SlugCache[sfId]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cl *SubforumLineageBuilder) GetSubforumLineageSlugs(sfId int) []string {
|
|
|
|
return cl.GetLineageSlugs(sfId)[1:]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (cl *SubforumLineageBuilder) FindIdBySlug(projectId int, slug string) int {
|
|
|
|
for _, node := range cl.Tree {
|
|
|
|
if node.Slug == slug && node.ProjectID == projectId {
|
|
|
|
return node.ID
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1
|
|
|
|
}
|