Index functions

This commit is contained in:
Simon Martens
2025-03-26 16:18:53 +01:00
parent dd24aea0a3
commit 052f21e87a
10 changed files with 240 additions and 23 deletions

View File

@@ -8,6 +8,9 @@ import (
const DEFAULT_YEAR = 1765
func GetIndex(fiber *fiber.Ctx) error {
_ = xmlmodels.Get()
return fiber.Render("/", nil)
lib := xmlmodels.Get()
// Years
years, yearmap := lib.Years()
return fiber.Render("/", map[string]any{"years": years, "yearmap": yearmap})
}

View File

@@ -43,12 +43,14 @@ func main() {
}
// INFO: the lib, engine and storage objects passed to the server should never be recreated.
err = xmlmodels.New(dir, commit.Hash)
lib, err := xmlmodels.Parse(dir, commit.Hash)
if err != nil {
panic(err)
}
engine := templating.New(&views.LayoutFS, &views.RoutesFS)
engine.AddFuncs(lib.FuncMap())
storage := memory.New(memory.Config{
GCInterval: 24 * time.Hour,
})

View File

@@ -11,7 +11,6 @@ import (
"sync"
"github.com/Theodor-Springmann-Stiftung/lenz-web/helpers/functions"
"github.com/gofiber/fiber/v2"
"golang.org/x/net/websocket"
)
@@ -180,9 +179,17 @@ func (e *Engine) AddFunc(name string, fn any) {
e.FuncMap[name] = fn
}
func (e *Engine) AddFuncs(funcs template.FuncMap) {
e.mu.Lock()
defer e.mu.Unlock()
for k, v := range funcs {
e.FuncMap[k] = v
}
}
func (e *Engine) Render(out io.Writer, path string, data any, layout ...string) error {
e.mu.RLock()
ld := data.(fiber.Map)
ld := data.(map[string]any)
if e.GlobalData != nil {
maps.Copy(ld, e.GlobalData)
}

View File

@@ -1 +1,15 @@
Hello from body! What!
{{ $model := . }}
{{- if .years -}}
{{- range $y := .years -}}
<div>
{{- printf "%d " $y -}}
{{- $letters := index $model.yearmap $y -}}
{{- len $letters }}
{{ range $l := $letters -}}
<div>{{ $l.Letter }}</div>
<div>{{ $l.Earliest.Text -}}</div>
{{- end -}}
</div>
{{- end -}}
{{- end -}}

View File

@@ -89,13 +89,13 @@ func (p *XMLParser[T]) Serialize(dataholder XMLRootElement[T], path string, late
// It deletes all items that have not been parsed in the last commit,
// and whose filepath has not been marked as failed.
func (p *XMLParser[T]) Cleanup(latest ParseMeta) {
todelete := make([]string, 0)
todelete := make([]any, 0)
toappend := make([]*T, 0)
p.Infos.Range(func(key, value interface{}) bool {
info := value.(ItemInfo)
if !info.Parse.Equals(latest) {
if !latest.Failed(info.Source) {
todelete = append(todelete, key.(string))
todelete = append(todelete, key)
} else {
item, ok := p.Items.Load(key)
if ok {
@@ -165,7 +165,7 @@ func (p *XMLParser[T]) Info(id string) ItemInfo {
return info.(ItemInfo)
}
func (p *XMLParser[T]) Item(id string) *T {
func (p *XMLParser[T]) Item(id any) *T {
item, ok := p.Items.Load(id)
if !ok {
return nil

View File

@@ -231,17 +231,17 @@ func (xsdd XSDDate) Type() XSDDatetype {
}
func (xsdd *XSDDate) Validate() bool {
if xsdd.error {
if xsdd.error || len(xsdd.base) == 0 {
xsdd.state = Invalid
return false
}
xsdd.state = xsdd.inferState()
if xsdd.state != Invalid {
return true
if xsdd.state == Invalid {
return false
}
return false
return true
}
func (xsdd *XSDDate) parseError(s string) error {
@@ -309,6 +309,56 @@ func (xsdd XSDDate) inferState() XSDDatetype {
return Invalid
}
func (xsdd XSDDate) Before(other XSDDate) bool {
if xsdd.Year < other.Year {
return true
} else if xsdd.Year > other.Year {
return false
}
if xsdd.Month < other.Month {
return true
} else if xsdd.Month > other.Month {
return false
}
if xsdd.Day < other.Day {
return true
}
return false
}
func (xsddate *XSDDate) Compare(other *XSDDate) int {
if !xsddate.Validate() {
return -1
}
if !other.Validate() {
return 1
}
if xsddate.Year < other.Year {
return -1
} else if xsddate.Year > other.Year {
return 1
}
if xsddate.Month < other.Month {
return -1
} else if xsddate.Month > other.Month {
return 1
}
if xsddate.Day < other.Day {
return -1
} else if xsddate.Day > other.Day {
return 1
}
return 0
}
func validDay(i int) bool {
if i < 1 || i > 31 {
return false

View File

@@ -17,3 +17,23 @@ type Date struct {
Cert string `xml:"cert,attr"`
Text string `xml:",chardata"`
}
func (d *Date) Sort() *xmlparsing.XSDDate {
if d.NotBefore.Validate() {
return &d.NotBefore
}
if d.From.Validate() {
return &d.From
}
if d.When.Validate() {
return &d.When
}
if d.To.Validate() {
return &d.To
}
if d.NotAfter.Validate() {
return &d.NotAfter
}
return nil
}

View File

@@ -2,8 +2,11 @@ package xmlmodels
import (
"fmt"
"html/template"
"log/slog"
"maps"
"path/filepath"
"slices"
"strconv"
"strings"
"sync"
@@ -30,6 +33,8 @@ type Library struct {
Letters *xmlparsing.XMLParser[Letter]
Traditions *xmlparsing.XMLParser[Tradition]
Metas *xmlparsing.XMLParser[Meta]
cache sync.Map
}
func (l *Library) String() string {
@@ -89,6 +94,7 @@ func (l *Library) Parse(source xmlparsing.ParseSource, baseDir, commit string) e
// INFO: this lock prevents multiple parses from happening at the same time.
l.mu.Lock()
defer l.mu.Unlock()
l.cache.Clear()
wg := sync.WaitGroup{}
meta := xmlparsing.ParseMeta{
@@ -233,3 +239,73 @@ func (l *Library) cleanup(meta xmlparsing.ParseMeta) {
wg.Wait()
}
func (l *Library) Years() ([]int, map[int][]Meta) {
if years, ok := l.cache.Load("years"); ok {
if yearmap, ok := l.cache.Load("yearmap"); ok {
return years.([]int), yearmap.(map[int][]Meta)
}
}
mapYears := make(map[int][]Meta)
for item := range l.Metas.Iterate() {
earliest := item.Earliest()
if earliest != nil {
mapYears[earliest.Sort().Year] = append(mapYears[earliest.Sort().Year], item)
}
}
ret := slices.Collect(maps.Keys(mapYears))
slices.Sort(ret)
for _, items := range mapYears {
slices.SortFunc(items, func(a, b Meta) int {
return a.Earliest().Sort().Compare(b.Earliest().Sort())
})
}
l.cache.Store("years", ret)
l.cache.Store("yearmap", mapYears)
return ret, mapYears
}
func (l *Library) LettersForYear(year int) (ret []Meta) {
for l := range l.Metas.Filter(func(item Meta) bool {
return item.Earliest().Sort().Year == year
}) {
ret = append(ret, l)
}
return
}
func (l *Library) Person(id int) (ret *PersonDef) {
ret = l.Persons.Item(id)
return
}
func (l *Library) Place(id int) (ret *LocationDef) {
ret = l.Places.Item(id)
return
}
func (l *Library) GetPersons(id []int) (ret []*PersonDef) {
for _, i := range id {
ret = append(ret, l.Person(i))
}
return
}
func (l *Library) GetPlaces(id []int) (ret []*LocationDef) {
for _, i := range id {
ret = append(ret, l.Place(i))
}
return
}
func (l *Library) FuncMap() template.FuncMap {
return template.FuncMap{
"Person": l.Person,
"Place": l.Place,
"Persons": l.GetPersons,
"Places": l.GetPlaces,
}
}

View File

@@ -3,6 +3,7 @@ package xmlmodels
import (
"encoding/json"
"encoding/xml"
"iter"
xmlparsing "github.com/Theodor-Springmann-Stiftung/lenz-web/xml"
)
@@ -17,6 +18,28 @@ type Meta struct {
Recieved []Action `xml:"recieved"`
}
func (m *Meta) Earliest() *Date {
var earliest *Date
for _, action := range m.Sent {
if earliest == nil || action.Earliest().Sort().Before(*earliest.Sort()) {
earliest = action.Earliest()
}
}
if earliest != nil {
return earliest
}
for _, action := range m.Recieved {
if earliest == nil || action.Earliest().Sort().Before(*earliest.Sort()) {
earliest = action.Earliest()
}
}
return earliest
}
func (m Meta) Keys() []any {
return []any{m.Letter}
}
@@ -33,8 +56,30 @@ func (m Meta) String() string {
return string(json)
}
func (m Meta) SendRecieved() iter.Seq2[*Action, *Action] {
return func(yield func(*Action, *Action) bool) {
for i, sent := range m.Sent {
var rec *Action
if i < len(m.Recieved) {
rec = &m.Recieved[i]
}
if !yield(&sent, rec) {
return
}
}
}
}
type Action struct {
Dates []Date `xml:"date,attr"`
Dates []Date `xml:"date"`
Places []RefElement `xml:"place"`
Persons []RefElement `xml:"person"`
}
func (a *Action) Earliest() *Date {
if len(a.Dates) == 0 {
return nil
}
return &a.Dates[0]
}

View File

@@ -27,14 +27,14 @@ func Get() *Library {
return lib
}
func New(dir, hash string) error {
func Parse(dir, hash string) (*Library, error) {
if lib == nil {
Set(NewLibrary())
return Parse(dir, hash)
}
func Parse(dir, hash string) error {
if hash == "" {
return lib.Parse(xmlparsing.Path, dir, hash)
}
return lib.Parse(xmlparsing.Commit, dir, hash)
if hash == "" {
return Get(), lib.Parse(xmlparsing.Path, dir, hash)
}
return Get(), lib.Parse(xmlparsing.Commit, dir, hash)
}