2023-12-22 01:23:18 +00:00
|
|
|
package legal
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/jackc/pgx/v4"
|
|
|
|
|
|
|
|
"dev.tandem.ws/tandem/camper/pkg/auth"
|
|
|
|
"dev.tandem.ws/tandem/camper/pkg/database"
|
|
|
|
"dev.tandem.ws/tandem/camper/pkg/form"
|
|
|
|
httplib "dev.tandem.ws/tandem/camper/pkg/http"
|
|
|
|
"dev.tandem.ws/tandem/camper/pkg/locale"
|
|
|
|
"dev.tandem.ws/tandem/camper/pkg/template"
|
|
|
|
)
|
|
|
|
|
|
|
|
type AdminHandler struct {
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewAdminHandler() *AdminHandler {
|
|
|
|
return &AdminHandler{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *AdminHandler) Handler(user *auth.User, company *auth.Company, conn *database.Conn) http.Handler {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
var head string
|
|
|
|
head, r.URL.Path = httplib.ShiftPath(r.URL.Path)
|
|
|
|
|
|
|
|
switch head {
|
|
|
|
case "":
|
|
|
|
switch r.Method {
|
|
|
|
case http.MethodGet:
|
|
|
|
serveLegalIndex(w, r, user, company, conn)
|
|
|
|
case http.MethodPost:
|
|
|
|
addLegal(w, r, user, company, conn)
|
|
|
|
default:
|
|
|
|
httplib.MethodNotAllowed(w, r, http.MethodGet, http.MethodPost)
|
|
|
|
}
|
|
|
|
case "new":
|
|
|
|
switch r.Method {
|
|
|
|
case http.MethodGet:
|
|
|
|
f := newLegalForm(company)
|
|
|
|
f.MustRender(w, r, user, company)
|
|
|
|
default:
|
|
|
|
httplib.MethodNotAllowed(w, r, http.MethodGet)
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
f := newLegalForm(company)
|
2024-01-12 18:51:12 +00:00
|
|
|
if err := f.FillFromDatabase(r.Context(), conn, company, head); err != nil {
|
2023-12-22 01:23:18 +00:00
|
|
|
if database.ErrorIsNotFound(err) {
|
|
|
|
http.NotFound(w, r)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
h.legalHandler(user, company, conn, f).ServeHTTP(w, r)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *AdminHandler) legalHandler(user *auth.User, company *auth.Company, conn *database.Conn, f *legalForm) http.Handler {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
var head string
|
|
|
|
head, r.URL.Path = httplib.ShiftPath(r.URL.Path)
|
|
|
|
|
|
|
|
switch head {
|
|
|
|
case "":
|
|
|
|
switch r.Method {
|
|
|
|
case http.MethodGet:
|
|
|
|
f.MustRender(w, r, user, company)
|
|
|
|
case http.MethodPut:
|
|
|
|
editLegal(w, r, user, company, conn, f)
|
|
|
|
default:
|
|
|
|
httplib.MethodNotAllowed(w, r, http.MethodGet, http.MethodPut)
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
http.NotFound(w, r)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func serveLegalIndex(w http.ResponseWriter, r *http.Request, user *auth.User, company *auth.Company, conn *database.Conn) {
|
|
|
|
legals, err := collectLegalEntries(r.Context(), conn, company)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
page := &legalIndex{
|
|
|
|
Texts: legals,
|
|
|
|
}
|
|
|
|
page.MustRender(w, r, user, company)
|
|
|
|
}
|
|
|
|
|
|
|
|
func collectLegalEntries(ctx context.Context, conn *database.Conn, company *auth.Company) ([]*legalEntry, error) {
|
|
|
|
rows, err := conn.Query(ctx, `
|
|
|
|
select '/admin/legal/' || slug
|
|
|
|
, name
|
|
|
|
from legal_text
|
|
|
|
where company_id = $1
|
|
|
|
order by name
|
|
|
|
`, company.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer rows.Close()
|
|
|
|
|
|
|
|
var legals []*legalEntry
|
|
|
|
for rows.Next() {
|
|
|
|
legal := &legalEntry{}
|
|
|
|
if err = rows.Scan(&legal.URL, &legal.Name); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
legals = append(legals, legal)
|
|
|
|
}
|
|
|
|
|
|
|
|
return legals, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type legalEntry struct {
|
|
|
|
URL string
|
|
|
|
Name string
|
|
|
|
}
|
|
|
|
|
|
|
|
type legalIndex struct {
|
|
|
|
Texts []*legalEntry
|
|
|
|
}
|
|
|
|
|
|
|
|
func (page *legalIndex) MustRender(w http.ResponseWriter, r *http.Request, user *auth.User, company *auth.Company) {
|
|
|
|
template.MustRenderAdmin(w, r, user, company, "legal/index.gohtml", page)
|
|
|
|
}
|
|
|
|
|
|
|
|
func addLegal(w http.ResponseWriter, r *http.Request, user *auth.User, company *auth.Company, conn *database.Conn) {
|
|
|
|
f := newLegalForm(company)
|
|
|
|
processLegalForm(w, r, user, company, conn, f, func(ctx context.Context, tx *database.Tx) error {
|
|
|
|
_, err := tx.Exec(ctx, `
|
|
|
|
insert into legal_text (company_id, slug, name, content)
|
|
|
|
values ($1, $2, $3, xmlparse(content $4))
|
2024-01-12 18:51:12 +00:00
|
|
|
`, company.ID, f.Slug.Val, f.Name[f.DefaultLang].Val, f.Content[f.DefaultLang])
|
2023-12-22 01:23:18 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-01-12 18:57:11 +00:00
|
|
|
return translateLegal(ctx, tx, company, f)
|
2023-12-22 01:23:18 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-01-12 18:57:11 +00:00
|
|
|
func translateLegal(ctx context.Context, tx *database.Tx, company *auth.Company, f *legalForm) error {
|
|
|
|
for lang := range company.Locales {
|
|
|
|
l := lang.String()
|
|
|
|
if l == f.DefaultLang {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err := tx.TranslateLegalText(ctx, company.ID, f.Slug.Val, lang, f.Name[l].Val, f.Content[l].Val); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-12-22 01:23:18 +00:00
|
|
|
func editLegal(w http.ResponseWriter, r *http.Request, user *auth.User, company *auth.Company, conn *database.Conn, f *legalForm) {
|
|
|
|
processLegalForm(w, r, user, company, conn, f, func(ctx context.Context, tx *database.Tx) error {
|
|
|
|
_, err := tx.Exec(ctx, `
|
|
|
|
update legal_text
|
|
|
|
set name = $3
|
|
|
|
, content = xmlparse(content $4)
|
|
|
|
where company_id = $1
|
|
|
|
and slug = $2
|
2024-01-12 18:51:12 +00:00
|
|
|
`, company.ID, f.Slug.Val, f.Name[f.DefaultLang].Val, f.Content[f.DefaultLang])
|
2023-12-22 01:23:18 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-01-12 18:57:11 +00:00
|
|
|
return translateLegal(ctx, tx, company, f)
|
2023-12-22 01:23:18 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func processLegalForm(w http.ResponseWriter, r *http.Request, user *auth.User, company *auth.Company, conn *database.Conn, f *legalForm, act func(ctx context.Context, tx *database.Tx) error) {
|
|
|
|
if ok, err := form.Handle(f, w, r, user); err != nil {
|
|
|
|
return
|
|
|
|
} else if !ok {
|
|
|
|
f.MustRender(w, r, user, company)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
tx := conn.MustBegin(r.Context())
|
|
|
|
if err := act(r.Context(), tx); err == nil {
|
|
|
|
if err := tx.Commit(r.Context()); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if err := tx.Rollback(r.Context()); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
httplib.Redirect(w, r, "/admin/legal/"+f.Slug.Val, http.StatusSeeOther)
|
|
|
|
}
|
|
|
|
|
|
|
|
type legalForm struct {
|
2024-01-12 18:51:12 +00:00
|
|
|
DefaultLang string
|
|
|
|
URL string
|
|
|
|
Slug *form.Input
|
|
|
|
Name form.I18nInput
|
|
|
|
Content form.I18nInput
|
2023-12-22 01:23:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func newLegalForm(company *auth.Company) *legalForm {
|
|
|
|
f := &legalForm{
|
2024-01-12 18:51:12 +00:00
|
|
|
DefaultLang: company.DefaultLanguage.String(),
|
2023-12-22 01:23:18 +00:00
|
|
|
Slug: &form.Input{
|
|
|
|
Name: "slug",
|
|
|
|
},
|
|
|
|
Name: form.NewI18nInput(company.Locales, "name"),
|
|
|
|
Content: form.NewI18nInput(company.Locales, "content"),
|
|
|
|
}
|
|
|
|
return f
|
|
|
|
}
|
|
|
|
|
2024-01-12 18:51:12 +00:00
|
|
|
func (f *legalForm) FillFromDatabase(ctx context.Context, conn *database.Conn, company *auth.Company, slug string) error {
|
2023-12-22 01:23:18 +00:00
|
|
|
var name database.RecordArray
|
|
|
|
var content database.RecordArray
|
|
|
|
row := conn.QueryRow(ctx, `
|
|
|
|
select '/admin/legal/' || text.slug
|
|
|
|
, text.slug
|
|
|
|
, text.name
|
|
|
|
, text.content::text
|
|
|
|
, array_agg((lang_tag, i18n.name))
|
|
|
|
, array_agg((lang_tag, i18n.content::text))
|
|
|
|
from legal_text as text
|
|
|
|
left join legal_text_i18n as i18n using (company_id, slug)
|
|
|
|
where text.company_id = $1
|
|
|
|
and text.slug = $2
|
|
|
|
group by text.slug
|
|
|
|
, text.name
|
|
|
|
, text.content::text
|
2024-01-12 18:51:12 +00:00
|
|
|
`, pgx.QueryResultFormats{pgx.BinaryFormatCode}, company.ID, slug)
|
|
|
|
if err := row.Scan(&f.URL, &f.Slug.Val, &f.Name[f.DefaultLang].Val, &f.Content[f.DefaultLang].Val, &name, &content); err != nil {
|
2023-12-22 01:23:18 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := f.Name.FillArray(name); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := f.Content.FillArray(content); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *legalForm) Parse(r *http.Request) error {
|
|
|
|
if err := r.ParseForm(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
f.Slug.FillValue(r)
|
|
|
|
f.Name.FillValue(r)
|
|
|
|
f.Content.FillValue(r)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *legalForm) Valid(l *locale.Locale) bool {
|
|
|
|
v := form.NewValidator(l)
|
2024-01-12 18:51:12 +00:00
|
|
|
if v.CheckRequired(f.Name[f.DefaultLang], l.GettextNoop("Name can not be empty.")) {
|
|
|
|
v.CheckMinLength(f.Name[f.DefaultLang], 1, l.GettextNoop("Name must have at least one letter."))
|
2023-12-22 01:23:18 +00:00
|
|
|
}
|
|
|
|
return v.AllOK
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *legalForm) MustRender(w http.ResponseWriter, r *http.Request, user *auth.User, company *auth.Company) {
|
|
|
|
template.MustRenderAdmin(w, r, user, company, "legal/form.gohtml", f)
|
|
|
|
}
|