Save button

This commit is contained in:
Simon Martens
2026-01-08 17:46:50 +01:00
parent 1656f60ac4
commit 93ea651c79
5 changed files with 1503 additions and 382 deletions

View File

@@ -1,18 +1,27 @@
package controllers
import (
"fmt"
"net/http"
"slices"
"strings"
"time"
"github.com/Theodor-Springmann-Stiftung/musenalm/app"
"github.com/Theodor-Springmann-Stiftung/musenalm/dbmodels"
"github.com/Theodor-Springmann-Stiftung/musenalm/helpers/functions"
"github.com/Theodor-Springmann-Stiftung/musenalm/middleware"
"github.com/Theodor-Springmann-Stiftung/musenalm/pagemodels"
"github.com/Theodor-Springmann-Stiftung/musenalm/templating"
"github.com/pocketbase/pocketbase/core"
"github.com/pocketbase/pocketbase/tools/router"
"github.com/pocketbase/pocketbase/tools/types"
)
const (
URL_ALMANACH_EDIT = "edit/"
TEMPLATE_ALMANACH_EDIT = "/almanach/edit/"
URL_ALMANACH_EDIT = "edit/"
TEMPLATE_ALMANACH_EDIT = "/almanach/edit/"
preferredSeriesRelationType = "Bevorzugter Reihentitel"
)
func init() {
@@ -35,6 +44,7 @@ func (p *AlmanachEditPage) Setup(router *router.Router[*core.RequestEvent], app
rg := router.Group(URL_ALMANACH)
rg.BindFunc(middleware.IsAdminOrEditor())
rg.GET(URL_ALMANACH_EDIT, p.GET(engine, app))
rg.POST(URL_ALMANACH_EDIT+"save", p.POSTSave(engine, app))
return nil
}
@@ -60,6 +70,10 @@ func (p *AlmanachEditPage) GET(engine *templating.Engine, app core.App) HandleFu
data["abbrs"] = abbrs
}
if msg := e.Request.URL.Query().Get("saved_message"); msg != "" {
data["success"] = msg
}
return engine.Response200(e, p.Template, data, p.Layout)
}
}
@@ -97,3 +111,481 @@ func NewAlmanachEditResult(app core.App, id string, filters BeitraegeFilterParam
Prev: prev,
}, nil
}
func (p *AlmanachEditPage) POSTSave(engine *templating.Engine, app core.App) HandleFunc {
return func(e *core.RequestEvent) error {
id := e.Request.PathValue("id")
req := templating.NewRequest(e)
payload := almanachEditPayload{}
if err := e.BindBody(&payload); err != nil {
return e.JSON(http.StatusBadRequest, map[string]any{
"error": "Ungültige Formulardaten.",
})
}
if err := req.CheckCSRF(payload.CSRFToken); err != nil {
return e.JSON(http.StatusBadRequest, map[string]any{
"error": err.Error(),
})
}
entry, err := dbmodels.Entries_MusenalmID(app, id)
if err != nil {
return e.JSON(http.StatusNotFound, map[string]any{
"error": "Band wurde nicht gefunden.",
})
}
if err := payload.Validate(); err != nil {
return e.JSON(http.StatusBadRequest, map[string]any{
"error": err.Error(),
})
}
if payload.LastEdited != "" {
lastEdited, err := types.ParseDateTime(payload.LastEdited)
if err != nil {
return e.JSON(http.StatusBadRequest, map[string]any{
"error": "Ungültiger Bearbeitungszeitstempel.",
})
}
if !entry.Updated().Time().Equal(lastEdited.Time()) {
return e.JSON(http.StatusConflict, map[string]any{
"error": "Der Eintrag wurde inzwischen geändert. Bitte Seite neu laden.",
})
}
}
user := req.User()
if err := app.RunInTransaction(func(tx core.App) error {
if err := applyEntryChanges(tx, entry, &payload, user); err != nil {
return err
}
if err := applyItemsChanges(tx, entry, &payload); err != nil {
return err
}
if err := applySeriesRelations(tx, entry, &payload); err != nil {
return err
}
if err := applyAgentRelations(tx, entry, &payload); err != nil {
return err
}
return nil
}); err != nil {
app.Logger().Error("Failed to save almanach entry", "entry_id", entry.Id, "error", err)
return e.JSON(http.StatusInternalServerError, map[string]any{
"error": "Speichern fehlgeschlagen.",
})
}
freshEntry, err := dbmodels.Entries_MusenalmID(app, id)
if err == nil {
entry = freshEntry
}
updatedInfo := map[string]string{
"raw": entry.Updated().Time().Format(time.RFC3339Nano),
"date": functions.GermanDate(entry.Updated()),
"time": functions.GermanTime(entry.Updated()),
}
if user != nil && strings.TrimSpace(user.Name) != "" {
updatedInfo["user"] = user.Name
}
return e.JSON(http.StatusOK, map[string]any{
"success": true,
"message": "Änderungen gespeichert.",
"updated": updatedInfo,
})
}
}
type almanachEditPayload struct {
CSRFToken string `json:"csrf_token"`
LastEdited string `json:"last_edited"`
Entry almanachEntryPayload `json:"entry"`
Languages []string `json:"languages"`
Places []string `json:"places"`
Items []almanachItemPayload `json:"items"`
DeletedItemIDs []string `json:"deleted_item_ids"`
SeriesRelations []almanachRelationPayload `json:"series_relations"`
NewSeriesRelations []almanachNewRelationPayload `json:"new_series_relations"`
DeletedSeriesRelationIDs []string `json:"deleted_series_relation_ids"`
AgentRelations []almanachRelationPayload `json:"agent_relations"`
NewAgentRelations []almanachNewRelationPayload `json:"new_agent_relations"`
DeletedAgentRelationIDs []string `json:"deleted_agent_relation_ids"`
}
type almanachEntryPayload struct {
PreferredTitle string `json:"preferred_title"`
Title string `json:"title"`
ParallelTitle string `json:"parallel_title"`
Subtitle string `json:"subtitle"`
VariantTitle string `json:"variant_title"`
Incipit string `json:"incipit"`
ResponsibilityStatement string `json:"responsibility_statement"`
PublicationStatement string `json:"publication_statement"`
PlaceStatement string `json:"place_statement"`
Edition string `json:"edition"`
Annotation string `json:"annotation"`
EditComment string `json:"edit_comment"`
Extent string `json:"extent"`
Dimensions string `json:"dimensions"`
References string `json:"references"`
Status string `json:"status"`
Year *int `json:"year"`
}
type almanachItemPayload struct {
ID string `json:"id"`
Owner string `json:"owner"`
Identifier string `json:"identifier"`
Location string `json:"location"`
Media []string `json:"media"`
Annotation string `json:"annotation"`
URI string `json:"uri"`
}
type almanachRelationPayload struct {
ID string `json:"id"`
TargetID string `json:"target_id"`
Type string `json:"type"`
Uncertain bool `json:"uncertain"`
}
type almanachNewRelationPayload struct {
TargetID string `json:"target_id"`
Type string `json:"type"`
Uncertain bool `json:"uncertain"`
}
func (payload *almanachEditPayload) Validate() error {
payload.Entry.Status = strings.TrimSpace(payload.Entry.Status)
if strings.TrimSpace(payload.Entry.PreferredTitle) == "" {
return fmt.Errorf("Kurztitel ist erforderlich.")
}
if payload.Entry.Year == nil {
return fmt.Errorf("Jahr muss angegeben werden.")
}
if payload.Entry.Status == "" || !slices.Contains(dbmodels.EDITORSTATE_VALUES, payload.Entry.Status) {
return fmt.Errorf("Ungültiger Status.")
}
for _, relation := range payload.SeriesRelations {
if err := validateRelationType(relation.Type, dbmodels.SERIES_RELATIONS); err != nil {
return err
}
}
for _, relation := range payload.NewSeriesRelations {
if err := validateRelationType(relation.Type, dbmodels.SERIES_RELATIONS); err != nil {
return err
}
}
for _, relation := range payload.AgentRelations {
if err := validateRelationType(relation.Type, dbmodels.AGENT_RELATIONS); err != nil {
return err
}
}
for _, relation := range payload.NewAgentRelations {
if err := validateRelationType(relation.Type, dbmodels.AGENT_RELATIONS); err != nil {
return err
}
}
hasPreferred := false
for _, relation := range payload.SeriesRelations {
if strings.TrimSpace(relation.Type) == preferredSeriesRelationType {
hasPreferred = true
break
}
}
if !hasPreferred {
for _, relation := range payload.NewSeriesRelations {
if strings.TrimSpace(relation.Type) == preferredSeriesRelationType {
hasPreferred = true
break
}
}
}
if !hasPreferred {
return fmt.Errorf("Mindestens ein bevorzugter Reihentitel muss verknüpft sein.")
}
return nil
}
func validateRelationType(value string, allowed []string) error {
value = strings.TrimSpace(value)
if value == "" {
return fmt.Errorf("Ungültiger Beziehungstyp.")
}
if !slices.Contains(allowed, value) {
return fmt.Errorf("Ungültiger Beziehungstyp.")
}
return nil
}
func sanitizeStrings(values []string) []string {
seen := map[string]struct{}{}
cleaned := make([]string, 0, len(values))
for _, value := range values {
value = strings.TrimSpace(value)
if value == "" {
continue
}
if _, ok := seen[value]; ok {
continue
}
seen[value] = struct{}{}
cleaned = append(cleaned, value)
}
return cleaned
}
func applyEntryChanges(tx core.App, entry *dbmodels.Entry, payload *almanachEditPayload, user *dbmodels.FixedUser) error {
entry.SetPreferredTitle(strings.TrimSpace(payload.Entry.PreferredTitle))
entry.SetTitleStmt(strings.TrimSpace(payload.Entry.Title))
entry.SetParallelTitle(strings.TrimSpace(payload.Entry.ParallelTitle))
entry.SetSubtitleStmt(strings.TrimSpace(payload.Entry.Subtitle))
entry.SetVariantTitle(strings.TrimSpace(payload.Entry.VariantTitle))
entry.SetIncipitStmt(strings.TrimSpace(payload.Entry.Incipit))
entry.SetResponsibilityStmt(strings.TrimSpace(payload.Entry.ResponsibilityStatement))
entry.SetPublicationStmt(strings.TrimSpace(payload.Entry.PublicationStatement))
entry.SetPlaceStmt(strings.TrimSpace(payload.Entry.PlaceStatement))
entry.SetEdition(strings.TrimSpace(payload.Entry.Edition))
entry.SetAnnotation(strings.TrimSpace(payload.Entry.Annotation))
entry.SetComment(strings.TrimSpace(payload.Entry.EditComment))
entry.SetExtent(strings.TrimSpace(payload.Entry.Extent))
entry.SetDimensions(strings.TrimSpace(payload.Entry.Dimensions))
entry.SetReferences(strings.TrimSpace(payload.Entry.References))
entry.SetYear(*payload.Entry.Year)
entry.SetEditState(payload.Entry.Status)
entry.SetLanguage(sanitizeStrings(payload.Languages))
entry.SetPlaces(sanitizeStrings(payload.Places))
if user != nil {
entry.SetEditor(user.Id)
}
return tx.Save(entry)
}
func applyItemsChanges(tx core.App, entry *dbmodels.Entry, payload *almanachEditPayload) error {
var itemsCollection *core.Collection
getItemsCollection := func() (*core.Collection, error) {
if itemsCollection != nil {
return itemsCollection, nil
}
collection, err := tx.FindCollectionByNameOrId(dbmodels.ITEMS_TABLE)
if err != nil {
return nil, err
}
itemsCollection = collection
return itemsCollection, nil
}
for _, itemPayload := range payload.Items {
itemID := strings.TrimSpace(itemPayload.ID)
var item *dbmodels.Item
if itemID != "" {
record, err := tx.FindRecordById(dbmodels.ITEMS_TABLE, itemID)
if err != nil {
return err
}
item = dbmodels.NewItem(record)
if item.Entry() != entry.Id {
return fmt.Errorf("Exemplar %s gehört zu einem anderen Eintrag.", itemID)
}
} else {
collection, err := getItemsCollection()
if err != nil {
return err
}
item = dbmodels.NewItem(core.NewRecord(collection))
}
item.SetEntry(entry.Id)
item.SetOwner(strings.TrimSpace(itemPayload.Owner))
item.SetIdentifier(strings.TrimSpace(itemPayload.Identifier))
item.SetLocation(strings.TrimSpace(itemPayload.Location))
item.SetAnnotation(strings.TrimSpace(itemPayload.Annotation))
item.SetUri(strings.TrimSpace(itemPayload.URI))
item.SetMedia(sanitizeStrings(itemPayload.Media))
if err := tx.Save(item); err != nil {
return err
}
}
for _, id := range payload.DeletedItemIDs {
itemID := strings.TrimSpace(id)
if itemID == "" {
continue
}
record, err := tx.FindRecordById(dbmodels.ITEMS_TABLE, itemID)
if err != nil {
continue
}
item := dbmodels.NewItem(record)
if item.Entry() != entry.Id {
continue
}
if err := tx.Delete(record); err != nil {
return err
}
}
return nil
}
func applySeriesRelations(tx core.App, entry *dbmodels.Entry, payload *almanachEditPayload) error {
tableName := dbmodels.RelationTableName(dbmodels.ENTRIES_TABLE, dbmodels.SERIES_TABLE)
var collection *core.Collection
getCollection := func() (*core.Collection, error) {
if collection != nil {
return collection, nil
}
col, err := tx.FindCollectionByNameOrId(tableName)
if err != nil {
return nil, err
}
collection = col
return collection, nil
}
for _, relation := range payload.SeriesRelations {
relationID := strings.TrimSpace(relation.ID)
if relationID == "" {
continue
}
record, err := tx.FindRecordById(tableName, relationID)
if err != nil {
return err
}
proxy := dbmodels.NewREntriesSeries(record)
if proxy.Entry() != entry.Id {
return fmt.Errorf("Relation %s gehört zu einem anderen Eintrag.", relationID)
}
proxy.SetEntry(entry.Id)
proxy.SetSeries(strings.TrimSpace(relation.TargetID))
proxy.SetType(strings.TrimSpace(relation.Type))
proxy.SetUncertain(relation.Uncertain)
if err := tx.Save(proxy); err != nil {
return err
}
}
for _, relationID := range payload.DeletedSeriesRelationIDs {
relationID = strings.TrimSpace(relationID)
if relationID == "" {
continue
}
record, err := tx.FindRecordById(tableName, relationID)
if err != nil {
continue
}
proxy := dbmodels.NewREntriesSeries(record)
if proxy.Entry() != entry.Id {
continue
}
if err := tx.Delete(record); err != nil {
return err
}
}
for _, relation := range payload.NewSeriesRelations {
targetID := strings.TrimSpace(relation.TargetID)
if targetID == "" {
continue
}
col, err := getCollection()
if err != nil {
return err
}
proxy := dbmodels.NewREntriesSeries(core.NewRecord(col))
proxy.SetEntry(entry.Id)
proxy.SetSeries(targetID)
proxy.SetType(strings.TrimSpace(relation.Type))
proxy.SetUncertain(relation.Uncertain)
if err := tx.Save(proxy); err != nil {
return err
}
}
return nil
}
func applyAgentRelations(tx core.App, entry *dbmodels.Entry, payload *almanachEditPayload) error {
tableName := dbmodels.RelationTableName(dbmodels.ENTRIES_TABLE, dbmodels.AGENTS_TABLE)
var collection *core.Collection
getCollection := func() (*core.Collection, error) {
if collection != nil {
return collection, nil
}
col, err := tx.FindCollectionByNameOrId(tableName)
if err != nil {
return nil, err
}
collection = col
return collection, nil
}
for _, relation := range payload.AgentRelations {
relationID := strings.TrimSpace(relation.ID)
if relationID == "" {
continue
}
record, err := tx.FindRecordById(tableName, relationID)
if err != nil {
return err
}
proxy := dbmodels.NewREntriesAgents(record)
if proxy.Entry() != entry.Id {
return fmt.Errorf("Relation %s gehört zu einem anderen Eintrag.", relationID)
}
proxy.SetEntry(entry.Id)
proxy.SetAgent(strings.TrimSpace(relation.TargetID))
proxy.SetType(strings.TrimSpace(relation.Type))
proxy.SetUncertain(relation.Uncertain)
if err := tx.Save(proxy); err != nil {
return err
}
}
for _, relationID := range payload.DeletedAgentRelationIDs {
relationID = strings.TrimSpace(relationID)
if relationID == "" {
continue
}
record, err := tx.FindRecordById(tableName, relationID)
if err != nil {
continue
}
proxy := dbmodels.NewREntriesAgents(record)
if proxy.Entry() != entry.Id {
continue
}
if err := tx.Delete(record); err != nil {
return err
}
}
for _, relation := range payload.NewAgentRelations {
targetID := strings.TrimSpace(relation.TargetID)
if targetID == "" {
continue
}
col, err := getCollection()
if err != nil {
return err
}
proxy := dbmodels.NewREntriesAgents(core.NewRecord(col))
proxy.SetEntry(entry.Id)
proxy.SetAgent(targetID)
proxy.SetType(strings.TrimSpace(relation.Type))
proxy.SetUncertain(relation.Uncertain)
if err := tx.Save(proxy); err != nil {
return err
}
}
return nil
}