2020-10-21 14:02:59 +00:00
|
|
|
package meme
|
|
|
|
|
|
|
|
import (
|
2021-07-29 16:33:03 +00:00
|
|
|
"embed"
|
2020-10-21 14:02:59 +00:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"sort"
|
|
|
|
|
2021-07-21 18:52:45 +00:00
|
|
|
"github.com/go-chi/chi/v5"
|
2020-10-21 14:02:59 +00:00
|
|
|
"github.com/rs/zerolog/log"
|
|
|
|
|
|
|
|
"github.com/velour/catbase/bot"
|
|
|
|
)
|
|
|
|
|
2021-07-29 16:33:03 +00:00
|
|
|
//go:embed *.html
|
|
|
|
var embeddedFS embed.FS
|
|
|
|
|
2020-10-21 19:55:28 +00:00
|
|
|
func (p *MemePlugin) registerWeb(c bot.Connector) {
|
2021-07-21 18:52:45 +00:00
|
|
|
r := chi.NewRouter()
|
|
|
|
r.HandleFunc("/slash", p.slashMeme(c))
|
|
|
|
r.HandleFunc("/img", p.img)
|
|
|
|
r.HandleFunc("/all", p.all)
|
|
|
|
r.HandleFunc("/add", p.addMeme)
|
|
|
|
r.HandleFunc("/rm", p.rmMeme)
|
|
|
|
r.HandleFunc("/", p.webRoot)
|
2021-07-28 15:32:59 +00:00
|
|
|
p.bot.RegisterWebName(r, "/meme", "Memes")
|
2020-10-21 19:55:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type webResp struct {
|
|
|
|
Name string `json:"name"`
|
|
|
|
URL string `json:"url"`
|
|
|
|
Config string `json:"config"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type webResps []webResp
|
|
|
|
|
|
|
|
func (w webResps) Len() int { return len(w) }
|
|
|
|
func (w webResps) Swap(i, j int) { w[i], w[j] = w[j], w[i] }
|
|
|
|
|
|
|
|
type ByName struct{ webResps }
|
|
|
|
|
|
|
|
func (s ByName) Less(i, j int) bool { return s.webResps[i].Name < s.webResps[j].Name }
|
|
|
|
|
2020-10-21 14:02:59 +00:00
|
|
|
func (p *MemePlugin) all(w http.ResponseWriter, r *http.Request) {
|
|
|
|
memes := p.c.GetMap("meme.memes", defaultFormats)
|
2020-10-21 19:55:28 +00:00
|
|
|
configs := p.c.GetMap("meme.memeconfigs", map[string]string{})
|
|
|
|
|
2020-10-21 14:02:59 +00:00
|
|
|
values := webResps{}
|
|
|
|
for n, u := range memes {
|
2020-10-21 19:55:28 +00:00
|
|
|
config, ok := configs[n]
|
|
|
|
if !ok {
|
2022-05-25 19:13:44 +00:00
|
|
|
b, _ := json.Marshal(p.defaultFormatConfig())
|
2020-10-21 19:55:28 +00:00
|
|
|
config = string(b)
|
|
|
|
}
|
2020-10-21 14:02:59 +00:00
|
|
|
realURL, err := url.Parse(u)
|
|
|
|
if err != nil || realURL.Scheme == "" {
|
2021-02-18 22:37:24 +00:00
|
|
|
values = append(values, webResp{n, "404.png", config})
|
|
|
|
log.Error().Err(err).Msgf("invalid URL")
|
|
|
|
continue
|
2020-10-21 14:02:59 +00:00
|
|
|
}
|
2020-10-21 19:55:28 +00:00
|
|
|
values = append(values, webResp{n, realURL.String(), config})
|
2020-10-21 14:02:59 +00:00
|
|
|
}
|
|
|
|
sort.Sort(ByName{values})
|
|
|
|
|
|
|
|
out, err := json.Marshal(values)
|
|
|
|
if err != nil {
|
|
|
|
w.WriteHeader(500)
|
|
|
|
log.Error().Err(err).Msgf("could not serve all memes route")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
w.Write(out)
|
|
|
|
}
|
|
|
|
|
|
|
|
func mkCheckError(w http.ResponseWriter) func(error) bool {
|
|
|
|
return func(err error) bool {
|
|
|
|
if err != nil {
|
|
|
|
log.Error().Err(err).Msgf("meme failed")
|
|
|
|
w.WriteHeader(500)
|
|
|
|
e, _ := json.Marshal(err)
|
|
|
|
w.Write(e)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *MemePlugin) rmMeme(w http.ResponseWriter, r *http.Request) {
|
|
|
|
if r.Method != http.MethodDelete {
|
|
|
|
w.WriteHeader(405)
|
|
|
|
fmt.Fprintf(w, "Incorrect HTTP method")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
checkError := mkCheckError(w)
|
|
|
|
decoder := json.NewDecoder(r.Body)
|
|
|
|
values := webResp{}
|
|
|
|
err := decoder.Decode(&values)
|
|
|
|
if checkError(err) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
formats := p.c.GetMap("meme.memes", defaultFormats)
|
|
|
|
delete(formats, values.Name)
|
|
|
|
err = p.c.SetMap("meme.memes", formats)
|
|
|
|
checkError(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *MemePlugin) addMeme(w http.ResponseWriter, r *http.Request) {
|
|
|
|
if r.Method != http.MethodPost {
|
|
|
|
w.WriteHeader(405)
|
|
|
|
fmt.Fprintf(w, "Incorrect HTTP method")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
checkError := mkCheckError(w)
|
|
|
|
decoder := json.NewDecoder(r.Body)
|
|
|
|
values := webResp{}
|
|
|
|
err := decoder.Decode(&values)
|
|
|
|
if checkError(err) {
|
2020-10-21 19:55:28 +00:00
|
|
|
log.Error().Err(err).Msgf("could not decode body")
|
2020-10-21 14:02:59 +00:00
|
|
|
return
|
|
|
|
}
|
2020-10-21 19:55:28 +00:00
|
|
|
log.Debug().Msgf("POSTed values: %+v", values)
|
2020-10-21 14:02:59 +00:00
|
|
|
formats := p.c.GetMap("meme.memes", defaultFormats)
|
|
|
|
formats[values.Name] = values.URL
|
|
|
|
err = p.c.SetMap("meme.memes", formats)
|
|
|
|
checkError(err)
|
2021-01-11 21:23:55 +00:00
|
|
|
|
|
|
|
if values.Config == "" {
|
2022-05-25 19:13:44 +00:00
|
|
|
values.Config = p.defaultFormatConfigJSON()
|
2021-01-11 21:23:55 +00:00
|
|
|
}
|
2020-10-21 19:55:28 +00:00
|
|
|
configs := p.c.GetMap("meme.memeconfigs", map[string]string{})
|
|
|
|
configs[values.Name] = values.Config
|
|
|
|
err = p.c.SetMap("meme.memeconfigs", configs)
|
|
|
|
checkError(err)
|
2020-10-21 14:02:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (p *MemePlugin) webRoot(w http.ResponseWriter, r *http.Request) {
|
2021-07-29 16:33:03 +00:00
|
|
|
index, _ := embeddedFS.ReadFile("index.html")
|
|
|
|
w.Write(index)
|
2020-10-21 14:02:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (p *MemePlugin) img(w http.ResponseWriter, r *http.Request) {
|
2021-02-18 22:37:24 +00:00
|
|
|
q := r.URL.Query()
|
|
|
|
spec := q.Get("spec")
|
|
|
|
if spec == "" {
|
|
|
|
log.Debug().Msgf("No spec found for img")
|
|
|
|
w.WriteHeader(404)
|
|
|
|
w.Write([]byte{})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
s, err := SpecFromJSON([]byte(spec))
|
|
|
|
if err != nil {
|
|
|
|
w.WriteHeader(400)
|
|
|
|
w.Write([]byte(err.Error()))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
img, err := p.genMeme(s)
|
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
w.Write(img)
|
2020-10-21 14:02:59 +00:00
|
|
|
} else {
|
2021-02-18 22:37:24 +00:00
|
|
|
log.Error().
|
|
|
|
Err(err).
|
|
|
|
Interface("spec", s).
|
|
|
|
Msg("Unable to generate meme image")
|
2020-10-21 14:02:59 +00:00
|
|
|
w.WriteHeader(404)
|
|
|
|
w.Write([]byte("not found"))
|
|
|
|
}
|
|
|
|
p.images.cleanup()
|
|
|
|
}
|