| 
									
										
										
										
											2023-03-12 16:00:57 +01:00
										 |  |  | // 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/>. | 
					
						
							| 
									
										
										
										
											2021-09-01 22:12:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-07 15:46:42 +02:00
										 |  |  | package router | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	"bytes" | 
					
						
							| 
									
										
										
										
											2021-07-07 15:46:42 +02:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2021-07-13 16:05:03 +02:00
										 |  |  | 	"html/template" | 
					
						
							| 
									
										
										
										
											2021-07-07 15:46:42 +02:00
										 |  |  | 	"os" | 
					
						
							|  |  |  | 	"path/filepath" | 
					
						
							| 
									
										
										
										
											2023-12-12 15:44:54 +01:00
										 |  |  | 	"reflect" | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	"regexp" | 
					
						
							| 
									
										
										
										
											2023-05-11 17:46:32 +02:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2023-12-12 13:47:07 +00:00
										 |  |  | 	"unsafe" | 
					
						
							| 
									
										
										
										
											2021-07-07 15:46:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/gin-gonic/gin" | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	"github.com/gin-gonic/gin/render" | 
					
						
							| 
									
										
										
										
											2023-01-02 13:10:50 +01:00
										 |  |  | 	apimodel "github.com/superseriousbusiness/gotosocial/internal/api/model" | 
					
						
							| 
									
										
										
										
											2021-07-07 15:46:42 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/config" | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/gtserror" | 
					
						
							| 
									
										
										
										
											2022-10-02 15:54:42 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/log" | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/regexes" | 
					
						
							| 
									
										
										
										
											2022-10-08 14:00:39 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/text" | 
					
						
							| 
									
										
										
										
											2022-10-02 15:54:42 +02:00
										 |  |  | 	"github.com/superseriousbusiness/gotosocial/internal/util" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | // 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. | 
					
						
							| 
									
										
										
										
											2022-07-12 08:32:20 +01:00
										 |  |  | func LoadTemplates(engine *gin.Engine) error { | 
					
						
							| 
									
										
										
										
											2022-05-30 13:41:24 +01:00
										 |  |  | 	templateBaseDir := config.GetWebTemplateBaseDir() | 
					
						
							| 
									
										
										
										
											2022-05-02 16:06:03 +02:00
										 |  |  | 	if templateBaseDir == "" { | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 		return gtserror.Newf( | 
					
						
							|  |  |  | 			"%s cannot be empty and must be a relative or absolute path", | 
					
						
							|  |  |  | 			config.WebTemplateBaseDirFlag(), | 
					
						
							|  |  |  | 		) | 
					
						
							| 
									
										
										
										
											2022-05-02 16:06:03 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-07-07 15:46:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	templateDirAbs, err := filepath.Abs(templateBaseDir) | 
					
						
							| 
									
										
										
										
											2022-05-02 16:06:03 +02:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 		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. | 
					
						
							| 
									
										
										
										
											2025-03-26 16:59:39 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Include renders the given template with the given data. | 
					
						
							|  |  |  | 	// Unlike `template`, `include` can be chained with `indent` | 
					
						
							|  |  |  | 	// to produce nicely-indented HTML. | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	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 | 
					
						
							| 
									
										
										
										
											2022-04-29 02:00:25 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-26 16:59:39 +01:00
										 |  |  | 	// includeIndex is like `include` but an index can be specified at | 
					
						
							|  |  |  | 	// `.Index` and data will be nested at `.Item`. Useful when ranging. | 
					
						
							|  |  |  | 	funcMap["includeIndex"] = func(name string, data any, index int) (template.HTML, error) { | 
					
						
							|  |  |  | 		var buf strings.Builder | 
					
						
							|  |  |  | 		withIndex := struct { | 
					
						
							|  |  |  | 			Item  any | 
					
						
							|  |  |  | 			Index int | 
					
						
							|  |  |  | 		}{ | 
					
						
							|  |  |  | 			Item:  data, | 
					
						
							|  |  |  | 			Index: index, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		err := tmpl.ExecuteTemplate(&buf, name, withIndex) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Template was already escaped by | 
					
						
							|  |  |  | 		// ExecuteTemplate so we can trust it. | 
					
						
							|  |  |  | 		return noescape(buf.String()), err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// includeAttr is like `include` but for element attributes. | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	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 | 
					
						
							| 
									
										
										
										
											2022-02-07 11:04:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	// 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} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-07 15:46:42 +02:00
										 |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-07-13 16:05:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 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, | 
					
						
							|  |  |  | 	"timestampVague":   timestampVague, | 
					
						
							|  |  |  | 	"visibilityIcon":   visibilityIcon, | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | func oddOrEven(n int) string { | 
					
						
							|  |  |  | 	if n%2 == 0 { | 
					
						
							|  |  |  | 		return "even" | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-09-30 11:16:23 +02:00
										 |  |  | 	return "odd" | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | // escape HTML escapes the given string, | 
					
						
							|  |  |  | // returning a trusted template. | 
					
						
							| 
									
										
										
										
											2022-09-02 05:54:32 -04:00
										 |  |  | func escape(str string) template.HTML { | 
					
						
							|  |  |  | 	/* #nosec G203 */ | 
					
						
							|  |  |  | 	return template.HTML(template.HTMLEscapeString(str)) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | // noescape marks the given string as a | 
					
						
							|  |  |  | // trusted template. The provided string | 
					
						
							|  |  |  | // MUST have already passed through a | 
					
						
							|  |  |  | // template or escaping function. | 
					
						
							| 
									
										
										
										
											2021-07-13 16:05:03 +02:00
										 |  |  | func noescape(str string) template.HTML { | 
					
						
							| 
									
										
										
										
											2021-11-22 08:46:19 +01:00
										 |  |  | 	/* #nosec G203 */ | 
					
						
							| 
									
										
										
										
											2021-07-13 16:05:03 +02:00
										 |  |  | 	return template.HTML(str) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | // noescapeAttr marks the given string as a | 
					
						
							|  |  |  | // trusted HTML attribute. The provided string | 
					
						
							|  |  |  | // MUST have already passed through a template | 
					
						
							|  |  |  | // or escaping function. | 
					
						
							| 
									
										
										
										
											2022-09-07 16:53:12 +02:00
										 |  |  | func noescapeAttr(str string) template.HTMLAttr { | 
					
						
							|  |  |  | 	/* #nosec G203 */ | 
					
						
							|  |  |  | 	return template.HTMLAttr(str) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 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" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-02 15:54:42 +02:00
										 |  |  | func timestampPrecise(stamp string) string { | 
					
						
							|  |  |  | 	t, err := util.ParseISO8601(stamp) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 		log.Errorf(nil, "error parsing timestamp %s: %s", stamp, err) | 
					
						
							| 
									
										
										
										
											2022-10-02 15:54:42 +02:00
										 |  |  | 		return badTimestamp | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return t.Local().Format(dateYearTime) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func timestampVague(stamp string) string { | 
					
						
							|  |  |  | 	t, err := util.ParseISO8601(stamp) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2023-02-17 12:02:29 +01:00
										 |  |  | 		log.Errorf(nil, "error parsing timestamp %s: %s", stamp, err) | 
					
						
							| 
									
										
										
										
											2022-10-02 15:54:42 +02:00
										 |  |  | 		return badTimestamp | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return t.Format(monthYear) | 
					
						
							| 
									
										
										
										
											2022-04-15 14:33:01 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-02 13:10:50 +01:00
										 |  |  | func visibilityIcon(visibility apimodel.Visibility) template.HTML { | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	var ( | 
					
						
							|  |  |  | 		label string | 
					
						
							|  |  |  | 		icon  string | 
					
						
							|  |  |  | 	) | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-22 08:46:19 +01:00
										 |  |  | 	switch visibility { | 
					
						
							| 
									
										
										
										
											2023-01-02 13:10:50 +01:00
										 |  |  | 	case apimodel.VisibilityPublic: | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 		label = "public" | 
					
						
							|  |  |  | 		icon = "globe" | 
					
						
							| 
									
										
										
										
											2023-01-02 13:10:50 +01:00
										 |  |  | 	case apimodel.VisibilityUnlisted: | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 		label = "unlisted" | 
					
						
							|  |  |  | 		icon = "unlock" | 
					
						
							| 
									
										
										
										
											2023-01-02 13:10:50 +01:00
										 |  |  | 	case apimodel.VisibilityPrivate: | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 		label = "private" | 
					
						
							|  |  |  | 		icon = "lock" | 
					
						
							| 
									
										
										
										
											2023-01-02 13:10:50 +01:00
										 |  |  | 	case apimodel.VisibilityMutualsOnly: | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 		label = "mutuals-only" | 
					
						
							|  |  |  | 		icon = "handshake-o" | 
					
						
							| 
									
										
										
										
											2023-01-02 13:10:50 +01:00
										 |  |  | 	case apimodel.VisibilityDirect: | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 		label = "direct" | 
					
						
							|  |  |  | 		icon = "envelope" | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-22 08:46:19 +01:00
										 |  |  | 	/* #nosec G203 */ | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | 	return template.HTML(fmt.Sprintf( | 
					
						
							|  |  |  | 		`<i aria-label="Visibility: %s" class="fa fa-%s"></i>`, | 
					
						
							|  |  |  | 		label, icon, | 
					
						
							|  |  |  | 	)) | 
					
						
							| 
									
										
										
										
											2021-09-13 14:45:33 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | // 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) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-09-02 05:54:32 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | // 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) | 
					
						
							| 
									
										
										
										
											2022-09-02 05:54:32 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-11 17:46:32 +02:00
										 |  |  | func acctInstance(acct string) string { | 
					
						
							|  |  |  | 	parts := strings.Split(acct, "@") | 
					
						
							|  |  |  | 	if len(parts) > 1 { | 
					
						
							|  |  |  | 		return "@" + parts[1] | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return "" | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | // increment adds 1 | 
					
						
							|  |  |  | // to the given int. | 
					
						
							| 
									
										
										
										
											2023-11-22 12:17:42 +01:00
										 |  |  | func increment(i int) int { | 
					
						
							|  |  |  | 	return i + 1 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-27 11:23:52 +01:00
										 |  |  | // 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) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-12 13:47:07 +00:00
										 |  |  | // 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 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-12 15:44:54 +01:00
										 |  |  | // 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() | 
					
						
							|  |  |  | } |