offline-twitter/internal/webserver/response_helpers.go
2023-08-29 12:27:53 -03:00

236 lines
6.0 KiB
Go

package webserver
import (
"bytes"
"fmt"
"html/template"
"io"
"io/fs"
"net/http"
"path"
"path/filepath"
"regexp"
"runtime"
"runtime/debug"
"github.com/Masterminds/sprig/v3"
"gitlab.com/offline-twitter/twitter_offline_engine/pkg/scraper"
)
func panic_if(err error) {
if err != nil {
panic(err)
}
}
var this_dir string
func init() {
_, this_file, _, _ := runtime.Caller(0) // `this_file` is absolute path to this source file
this_dir = path.Dir(this_file)
}
func get_filepath(s string) string {
if use_embedded == "true" {
return s
}
return path.Join(this_dir, s)
}
func glob(path string) []string {
var ret []string
var err error
if use_embedded == "true" {
ret, err = fs.Glob(embedded_files, get_filepath(path))
} else {
ret, err = filepath.Glob(get_filepath(path))
}
panic_if(err)
return ret
}
// func (app *Application) error_400(w http.ResponseWriter) {
// http.Error(w, "Bad Request", 400)
// }
func (app *Application) error_400_with_message(w http.ResponseWriter, msg string) {
http.Error(w, fmt.Sprintf("Bad Request\n\n%s", msg), 400)
}
func (app *Application) error_404(w http.ResponseWriter) {
http.Error(w, "Not Found", 404)
}
func (app *Application) error_500(w http.ResponseWriter, err error) {
trace := fmt.Sprintf("%s\n%s", err.Error(), debug.Stack())
err2 := app.ErrorLog.Output(2, trace) // Magic
if err2 != nil {
panic(err2)
}
http.Error(w, "Server error :(", 500)
}
type TweetCollection interface {
Tweet(id scraper.TweetID) scraper.Tweet
User(id scraper.UserID) scraper.User
Retweet(id scraper.TweetID) scraper.Retweet
Space(id scraper.SpaceID) scraper.Space
FocusedTweetID() scraper.TweetID
}
// Creates a template from the given template file using all the available partials.
// Calls `app.buffered_render` to render the created template.
func (app *Application) buffered_render_tweet_page(w http.ResponseWriter, tpl_file string, data TweetCollection) {
partials := append(glob("tpl/includes/*.tpl"), glob("tpl/tweet_page_includes/*.tpl")...)
r := renderer{
Funcs: func_map(template.FuncMap{
"tweet": data.Tweet,
"user": data.User,
"retweet": data.Retweet,
"space": data.Space,
"active_user": app.get_active_user,
"focused_tweet_id": data.FocusedTweetID,
"get_entities": get_entities,
"get_tombstone_text": get_tombstone_text,
}),
Filenames: append(partials, get_filepath(tpl_file)),
TplName: "base",
Data: data,
}
r.BufferedRender(w)
}
// Creates a template from the given template file using all the available partials.
// Calls `app.buffered_render` to render the created template.
func (app *Application) buffered_render_basic_page(w http.ResponseWriter, tpl_file string, data interface{}) {
partials := glob("tpl/includes/*.tpl")
r := renderer{
Funcs: func_map(template.FuncMap{"active_user": app.get_active_user}),
Filenames: append(partials, get_filepath(tpl_file)),
TplName: "base",
Data: data,
}
r.BufferedRender(w)
}
func (app *Application) buffered_render_tweet_htmx(w http.ResponseWriter, tpl_name string, data TweetCollection) {
partials := append(glob("tpl/includes/*.tpl"), glob("tpl/tweet_page_includes/*.tpl")...)
r := renderer{
Funcs: func_map(template.FuncMap{
"tweet": data.Tweet,
"user": data.User,
"retweet": data.Retweet,
"space": data.Space,
"active_user": app.get_active_user,
"focused_tweet_id": data.FocusedTweetID,
"get_entities": get_entities,
"get_tombstone_text": get_tombstone_text,
}),
Filenames: partials,
TplName: tpl_name,
Data: data,
}
r.BufferedRender(w)
}
func (app *Application) buffered_render_basic_htmx(w http.ResponseWriter, tpl_name string, data interface{}) {
partials := glob("tpl/includes/*.tpl")
r := renderer{
Funcs: func_map(template.FuncMap{"active_user": app.get_active_user}),
Filenames: partials,
TplName: tpl_name,
Data: data,
}
r.BufferedRender(w)
}
func (app *Application) get_active_user() scraper.User {
return app.ActiveUser
}
type EntityType int
const (
ENTITY_TYPE_TEXT EntityType = iota
ENTITY_TYPE_MENTION
ENTITY_TYPE_HASHTAG
)
type Entity struct {
EntityType
Contents string
}
func get_entities(text string) []Entity {
ret := []Entity{}
start := 0
for _, idxs := range regexp.MustCompile(`(\s|^)[@#]\w+`).FindAllStringIndex(text, -1) {
// Handle leading whitespace. Only match start-of-string or leading whitespace to avoid matching, e.g., emails
if text[idxs[0]] == ' ' || text[idxs[0]] == '\n' {
idxs[0] += 1
}
if start != idxs[0] {
ret = append(ret, Entity{ENTITY_TYPE_TEXT, text[start:idxs[0]]})
}
piece := text[idxs[0]+1 : idxs[1]] // Chop off the "#" or "@"
if text[idxs[0]] == '@' {
ret = append(ret, Entity{ENTITY_TYPE_MENTION, piece})
} else {
ret = append(ret, Entity{ENTITY_TYPE_HASHTAG, piece})
}
start = idxs[1]
}
if start < len(text) {
ret = append(ret, Entity{ENTITY_TYPE_TEXT, text[start:]})
}
return ret
}
func get_tombstone_text(t scraper.Tweet) string {
if t.TombstoneText != "" {
return t.TombstoneText
}
return t.TombstoneType
}
func func_map(extras template.FuncMap) template.FuncMap {
ret := sprig.FuncMap()
for i := range extras {
ret[i] = extras[i]
}
return ret
}
type renderer struct {
Funcs template.FuncMap
Filenames []string
TplName string
Data interface{}
}
// Render the given template using a bytes.Buffer. This avoids the possibility of failing partway
// through the rendering, and sending an imcomplete response with "Bad Request" or "Server Error" at the end.
func (r renderer) BufferedRender(w io.Writer) {
var tpl *template.Template
var err error
if use_embedded == "true" {
tpl, err = template.New("").Funcs(r.Funcs).ParseFS(embedded_files, r.Filenames...)
} else {
tpl, err = template.New("").Funcs(r.Funcs).ParseFiles(r.Filenames...)
}
panic_if(err)
buf := new(bytes.Buffer)
err = tpl.ExecuteTemplate(buf, r.TplName, r.Data)
panic_if(err)
_, err = buf.WriteTo(w)
panic_if(err)
}