mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-11-03 23:32:24 -06:00 
			
		
		
		
	* [chore] Refactor HTML templates and CSS * eslint * ignore "Local" * rss tests * fiddle with OG just a tiny bit * dick around with polls a bit more so SR stops saying "clickable" * remove break * oh lord * don't lazy load avatar * fix ogmeta tests * clean up some cruft * catch remaining calls to c.HTML * fix error rendering + stack overflow in tag * allow templating attributes * fix indent * set aria-hidden on status complementary content, since it's already present in the label anyway * tidy up templating calls a little * try to make styling a bit more consistent + readable * fix up some remaining CSS issues * fix up reports
		
			
				
	
	
		
			369 lines
		
	
	
	
		
			9.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			369 lines
		
	
	
	
		
			9.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// GoToSocial
 | 
						|
// Copyright (C) GoToSocial Authors admin@gotosocial.org
 | 
						|
// SPDX-License-Identifier: AGPL-3.0-or-later
 | 
						|
//
 | 
						|
// This program is free software: you can redistribute it and/or modify
 | 
						|
// it under the terms of the GNU Affero General Public License as published by
 | 
						|
// the Free Software Foundation, either version 3 of the License, or
 | 
						|
// (at your option) any later version.
 | 
						|
//
 | 
						|
// This program is distributed in the hope that it will be useful,
 | 
						|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
// GNU Affero General Public License for more details.
 | 
						|
//
 | 
						|
// You should have received a copy of the GNU Affero General Public License
 | 
						|
// along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | 
						|
 | 
						|
package router
 | 
						|
 | 
						|
import (
 | 
						|
	"bytes"
 | 
						|
	"fmt"
 | 
						|
	"html/template"
 | 
						|
	"os"
 | 
						|
	"path/filepath"
 | 
						|
	"reflect"
 | 
						|
	"regexp"
 | 
						|
	"strings"
 | 
						|
	"time"
 | 
						|
	"unsafe"
 | 
						|
 | 
						|
	"github.com/gin-gonic/gin"
 | 
						|
	"github.com/gin-gonic/gin/render"
 | 
						|
	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model"
 | 
						|
	"github.com/superseriousbusiness/gotosocial/internal/config"
 | 
						|
	"github.com/superseriousbusiness/gotosocial/internal/gtserror"
 | 
						|
	"github.com/superseriousbusiness/gotosocial/internal/log"
 | 
						|
	"github.com/superseriousbusiness/gotosocial/internal/regexes"
 | 
						|
	"github.com/superseriousbusiness/gotosocial/internal/text"
 | 
						|
	"github.com/superseriousbusiness/gotosocial/internal/util"
 | 
						|
)
 | 
						|
 | 
						|
// LoadTemplates loads templates found at `web-template-base-dir`
 | 
						|
// into the Gin engine, or errors if templates cannot be loaded.
 | 
						|
//
 | 
						|
// The special functions "include" and "includeAttr" will be added
 | 
						|
// to the template funcMap for use in any template. Use these "include"
 | 
						|
// functions when you need to pass a template through a pipeline.
 | 
						|
// Otherwise, prefer the built-in "template" function.
 | 
						|
func LoadTemplates(engine *gin.Engine) error {
 | 
						|
	templateBaseDir := config.GetWebTemplateBaseDir()
 | 
						|
	if templateBaseDir == "" {
 | 
						|
		return gtserror.Newf(
 | 
						|
			"%s cannot be empty and must be a relative or absolute path",
 | 
						|
			config.WebTemplateBaseDirFlag(),
 | 
						|
		)
 | 
						|
	}
 | 
						|
 | 
						|
	templateDirAbs, err := filepath.Abs(templateBaseDir)
 | 
						|
	if err != nil {
 | 
						|
		return gtserror.Newf(
 | 
						|
			"error getting absolute path of web-template-base-dir %s: %w",
 | 
						|
			templateBaseDir, err,
 | 
						|
		)
 | 
						|
	}
 | 
						|
 | 
						|
	indexTmplPath := filepath.Join(templateDirAbs, "index.tmpl")
 | 
						|
	if _, err := os.Stat(indexTmplPath); err != nil {
 | 
						|
		return gtserror.Newf(
 | 
						|
			"cannot find index.tmpl in web template directory %s: %w",
 | 
						|
			templateDirAbs, err,
 | 
						|
		)
 | 
						|
	}
 | 
						|
 | 
						|
	// Bring base template into scope.
 | 
						|
	tmpl := template.New("base")
 | 
						|
 | 
						|
	// Set additional "include" functions to render
 | 
						|
	// provided template name using the base template.
 | 
						|
	funcMap["include"] = func(name string, data any) (template.HTML, error) {
 | 
						|
		var buf strings.Builder
 | 
						|
		err := tmpl.ExecuteTemplate(&buf, name, data)
 | 
						|
 | 
						|
		// Template was already escaped by
 | 
						|
		// ExecuteTemplate so we can trust it.
 | 
						|
		return noescape(buf.String()), err
 | 
						|
	}
 | 
						|
 | 
						|
	funcMap["includeAttr"] = func(name string, data any) (template.HTMLAttr, error) {
 | 
						|
		var buf strings.Builder
 | 
						|
		err := tmpl.ExecuteTemplate(&buf, name, data)
 | 
						|
 | 
						|
		// Template was already escaped by
 | 
						|
		// ExecuteTemplate so we can trust it.
 | 
						|
		return noescapeAttr(buf.String()), err
 | 
						|
	}
 | 
						|
 | 
						|
	// Load functions into the base template, and
 | 
						|
	// associate other templates with base template.
 | 
						|
	templateGlob := filepath.Join(templateDirAbs, "*")
 | 
						|
	tmpl, err = tmpl.Funcs(funcMap).ParseGlob(templateGlob)
 | 
						|
	if err != nil {
 | 
						|
		return gtserror.Newf("error loading templates: %w", err)
 | 
						|
	}
 | 
						|
 | 
						|
	// Almost done; teach the
 | 
						|
	// engine how to render.
 | 
						|
	engine.SetFuncMap(funcMap)
 | 
						|
	engine.HTMLRender = render.HTMLProduction{Template: tmpl}
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
var funcMap = template.FuncMap{
 | 
						|
	"add":              add,
 | 
						|
	"acctInstance":     acctInstance,
 | 
						|
	"demojify":         demojify,
 | 
						|
	"deref":            deref,
 | 
						|
	"emojify":          emojify,
 | 
						|
	"escape":           escape,
 | 
						|
	"increment":        increment,
 | 
						|
	"indent":           indent,
 | 
						|
	"indentAttr":       indentAttr,
 | 
						|
	"isNil":            isNil,
 | 
						|
	"outdentPre":       outdentPre,
 | 
						|
	"noescapeAttr":     noescapeAttr,
 | 
						|
	"noescape":         noescape,
 | 
						|
	"oddOrEven":        oddOrEven,
 | 
						|
	"subtract":         subtract,
 | 
						|
	"timestampPrecise": timestampPrecise,
 | 
						|
	"timestamp":        timestamp,
 | 
						|
	"timestampVague":   timestampVague,
 | 
						|
	"visibilityIcon":   visibilityIcon,
 | 
						|
}
 | 
						|
 | 
						|
func oddOrEven(n int) string {
 | 
						|
	if n%2 == 0 {
 | 
						|
		return "even"
 | 
						|
	}
 | 
						|
	return "odd"
 | 
						|
}
 | 
						|
 | 
						|
// escape HTML escapes the given string,
 | 
						|
// returning a trusted template.
 | 
						|
func escape(str string) template.HTML {
 | 
						|
	/* #nosec G203 */
 | 
						|
	return template.HTML(template.HTMLEscapeString(str))
 | 
						|
}
 | 
						|
 | 
						|
// noescape marks the given string as a
 | 
						|
// trusted template. The provided string
 | 
						|
// MUST have already passed through a
 | 
						|
// template or escaping function.
 | 
						|
func noescape(str string) template.HTML {
 | 
						|
	/* #nosec G203 */
 | 
						|
	return template.HTML(str)
 | 
						|
}
 | 
						|
 | 
						|
// noescapeAttr marks the given string as a
 | 
						|
// trusted HTML attribute. The provided string
 | 
						|
// MUST have already passed through a template
 | 
						|
// or escaping function.
 | 
						|
func noescapeAttr(str string) template.HTMLAttr {
 | 
						|
	/* #nosec G203 */
 | 
						|
	return template.HTMLAttr(str)
 | 
						|
}
 | 
						|
 | 
						|
const (
 | 
						|
	justTime     = "15:04"
 | 
						|
	dateYear     = "Jan 02, 2006"
 | 
						|
	dateTime     = "Jan 02, 15:04"
 | 
						|
	dateYearTime = "Jan 02, 2006, 15:04"
 | 
						|
	monthYear    = "Jan, 2006"
 | 
						|
	badTimestamp = "bad timestamp"
 | 
						|
)
 | 
						|
 | 
						|
func timestamp(stamp string) string {
 | 
						|
	t, err := util.ParseISO8601(stamp)
 | 
						|
	if err != nil {
 | 
						|
		log.Errorf(nil, "error parsing timestamp %s: %s", stamp, err)
 | 
						|
		return badTimestamp
 | 
						|
	}
 | 
						|
 | 
						|
	t = t.Local()
 | 
						|
 | 
						|
	tYear, tMonth, tDay := t.Date()
 | 
						|
	now := time.Now()
 | 
						|
	currentYear, currentMonth, currentDay := now.Date()
 | 
						|
 | 
						|
	switch {
 | 
						|
	case tYear == currentYear && tMonth == currentMonth && tDay == currentDay:
 | 
						|
		return "Today, " + t.Format(justTime)
 | 
						|
	case tYear == currentYear:
 | 
						|
		return t.Format(dateTime)
 | 
						|
	default:
 | 
						|
		return t.Format(dateYear)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func timestampPrecise(stamp string) string {
 | 
						|
	t, err := util.ParseISO8601(stamp)
 | 
						|
	if err != nil {
 | 
						|
		log.Errorf(nil, "error parsing timestamp %s: %s", stamp, err)
 | 
						|
		return badTimestamp
 | 
						|
	}
 | 
						|
	return t.Local().Format(dateYearTime)
 | 
						|
}
 | 
						|
 | 
						|
func timestampVague(stamp string) string {
 | 
						|
	t, err := util.ParseISO8601(stamp)
 | 
						|
	if err != nil {
 | 
						|
		log.Errorf(nil, "error parsing timestamp %s: %s", stamp, err)
 | 
						|
		return badTimestamp
 | 
						|
	}
 | 
						|
	return t.Format(monthYear)
 | 
						|
}
 | 
						|
 | 
						|
func visibilityIcon(visibility apimodel.Visibility) template.HTML {
 | 
						|
	var (
 | 
						|
		label string
 | 
						|
		icon  string
 | 
						|
	)
 | 
						|
 | 
						|
	switch visibility {
 | 
						|
	case apimodel.VisibilityPublic:
 | 
						|
		label = "public"
 | 
						|
		icon = "globe"
 | 
						|
	case apimodel.VisibilityUnlisted:
 | 
						|
		label = "unlisted"
 | 
						|
		icon = "unlock"
 | 
						|
	case apimodel.VisibilityPrivate:
 | 
						|
		label = "private"
 | 
						|
		icon = "lock"
 | 
						|
	case apimodel.VisibilityMutualsOnly:
 | 
						|
		label = "mutuals-only"
 | 
						|
		icon = "handshake-o"
 | 
						|
	case apimodel.VisibilityDirect:
 | 
						|
		label = "direct"
 | 
						|
		icon = "envelope"
 | 
						|
	}
 | 
						|
 | 
						|
	/* #nosec G203 */
 | 
						|
	return template.HTML(fmt.Sprintf(
 | 
						|
		`<i aria-label="Visibility: %s" class="fa fa-%s"></i>`,
 | 
						|
		label, icon,
 | 
						|
	))
 | 
						|
}
 | 
						|
 | 
						|
// emojify replaces emojis in the given
 | 
						|
// html fragment with suitable <img> tags.
 | 
						|
//
 | 
						|
// The provided input must have been
 | 
						|
// escaped / templated already!
 | 
						|
func emojify(
 | 
						|
	emojis []apimodel.Emoji,
 | 
						|
	html template.HTML,
 | 
						|
) template.HTML {
 | 
						|
	return text.EmojifyWeb(emojis, html)
 | 
						|
}
 | 
						|
 | 
						|
// demojify replaces emoji shortcodes in
 | 
						|
// the given fragment with empty strings.
 | 
						|
//
 | 
						|
// Output must then be escaped as appropriate.
 | 
						|
func demojify(input string) string {
 | 
						|
	return text.Demojify(input)
 | 
						|
}
 | 
						|
 | 
						|
func acctInstance(acct string) string {
 | 
						|
	parts := strings.Split(acct, "@")
 | 
						|
	if len(parts) > 1 {
 | 
						|
		return "@" + parts[1]
 | 
						|
	}
 | 
						|
 | 
						|
	return ""
 | 
						|
}
 | 
						|
 | 
						|
// increment adds 1
 | 
						|
// to the given int.
 | 
						|
func increment(i int) int {
 | 
						|
	return i + 1
 | 
						|
}
 | 
						|
 | 
						|
// add adds n2 to n1.
 | 
						|
func add(n1 int, n2 int) int {
 | 
						|
	return n1 + n2
 | 
						|
}
 | 
						|
 | 
						|
// subtract subtracts n2 from n1.
 | 
						|
func subtract(n1 int, n2 int) int {
 | 
						|
	return n1 - n2
 | 
						|
}
 | 
						|
 | 
						|
var (
 | 
						|
	indentRegex  = regexp.MustCompile(`(?m)^`)
 | 
						|
	indentStr    = "    "
 | 
						|
	indentStrLen = len(indentStr)
 | 
						|
	indents      = strings.Repeat(indentStr, 12)
 | 
						|
	indentPre    = regexp.MustCompile(fmt.Sprintf(`(?Ums)^((?:%s)+)<pre>.*</pre>`, indentStr))
 | 
						|
)
 | 
						|
 | 
						|
// indent appropriately indents the given html
 | 
						|
// by prepending each line with the indentStr.
 | 
						|
func indent(n int, html template.HTML) template.HTML {
 | 
						|
	out := indentRegex.ReplaceAllString(
 | 
						|
		string(html),
 | 
						|
		indents[:n*indentStrLen],
 | 
						|
	)
 | 
						|
	return noescape(out)
 | 
						|
}
 | 
						|
 | 
						|
// indentAttr appropriately indents the given html
 | 
						|
// attribute by prepending each line with the indentStr.
 | 
						|
func indentAttr(n int, html template.HTMLAttr) template.HTMLAttr {
 | 
						|
	out := indentRegex.ReplaceAllString(
 | 
						|
		string(html),
 | 
						|
		indents[:n*indentStrLen],
 | 
						|
	)
 | 
						|
	return noescapeAttr(out)
 | 
						|
}
 | 
						|
 | 
						|
// outdentPre outdents all `<pre></pre>` tags in the
 | 
						|
// given HTML so that they render correctly in code
 | 
						|
// blocks, even if they were indented before.
 | 
						|
func outdentPre(html template.HTML) template.HTML {
 | 
						|
	input := string(html)
 | 
						|
	output := regexes.ReplaceAllStringFunc(indentPre, input,
 | 
						|
		func(match string, buf *bytes.Buffer) string {
 | 
						|
			// Reuse the regex to pull out submatches.
 | 
						|
			matches := indentPre.FindAllStringSubmatch(match, -1)
 | 
						|
			if len(matches) != 1 {
 | 
						|
				return match
 | 
						|
			}
 | 
						|
 | 
						|
			var (
 | 
						|
				indented = matches[0][0]
 | 
						|
				indent   = matches[0][1]
 | 
						|
			)
 | 
						|
 | 
						|
			// Outdent everything in the inner match, add
 | 
						|
			// a newline at the end to make it a bit neater.
 | 
						|
			outdented := strings.ReplaceAll(indented, indent, "")
 | 
						|
 | 
						|
			// Replace original match with the outdented version.
 | 
						|
			return strings.ReplaceAll(match, indented, outdented)
 | 
						|
		},
 | 
						|
	)
 | 
						|
	return noescape(output)
 | 
						|
}
 | 
						|
 | 
						|
// isNil will safely check if 'v' is nil without
 | 
						|
// dealing with weird Go interface nil bullshit.
 | 
						|
func isNil(i interface{}) bool {
 | 
						|
	type eface struct{ _, data unsafe.Pointer }
 | 
						|
	return (*eface)(unsafe.Pointer(&i)).data == nil
 | 
						|
}
 | 
						|
 | 
						|
// deref returns the dereferenced value of
 | 
						|
// its input. To ensure you don't pass nil
 | 
						|
// pointers into this func, use isNil first.
 | 
						|
func deref(i any) any {
 | 
						|
	vOf := reflect.ValueOf(i)
 | 
						|
	if vOf.Kind() != reflect.Pointer {
 | 
						|
		// Not a pointer.
 | 
						|
		return i
 | 
						|
	}
 | 
						|
 | 
						|
	return vOf.Elem()
 | 
						|
}
 |