all repos — honk @ 9d02c2482cd2755169c8d7e3ca7c7113a71d0fff

my fork of honk

masto.go (view raw)

  1package main
  2
  3import (
  4	"database/sql"
  5	"fmt"
  6	"log"
  7	"net/http"
  8
  9	"humungus.tedunangst.com/r/webs/junk"
 10	"humungus.tedunangst.com/r/webs/login"
 11)
 12
 13func showoauthlogin(rw http.ResponseWriter, r *http.Request) {
 14	templinfo := make(map[string]interface{})
 15	templinfo = getInfo(r)
 16	templinfo["ClientID"] = r.URL.Query().Get("client_id")
 17	templinfo["RedirectURI"] = r.URL.Query().Get("redirect_uri")
 18
 19	if err := readviews.Execute(rw, "oauthlogin.html", templinfo); err != nil {
 20		elog.Println(err)
 21	}
 22}
 23
 24// https://docs.joinmastodon.org/methods/apps/#create
 25func apiapps(rw http.ResponseWriter, r *http.Request) {
 26	if err := r.ParseForm(); err != nil {
 27		http.Error(rw, "invalid input", http.StatusUnprocessableEntity)
 28		elog.Println(err)
 29		return
 30	}
 31	clientName := r.Form.Get("client_name")
 32	redirectURI := r.Form.Get("redirect_uris")
 33	scopes := r.Form.Get("scopes")
 34	website := r.Form.Get("website")
 35	clientID := tokengen()
 36	clientSecret := tokengen()
 37	vapidKey := tokengen()
 38
 39	_, err := stmtSaveMastoApp.Exec(clientName, redirectURI, scopes, clientID, clientSecret, vapidKey, "")
 40	if err != nil {
 41		elog.Printf("error saving masto app: %v", err)
 42		http.Error(rw, "error saving masto app", http.StatusUnprocessableEntity)
 43		return
 44	}
 45
 46	j := junk.New()
 47	j["id"] = fmt.Sprintf("%d", snowflake())
 48	j["website"] = website
 49	j["name"] = clientName
 50	j["redirect_uri"] = redirectURI
 51	j["client_id"] = clientID
 52	j["client_secret"] = clientSecret
 53	j["vapid_key"] = vapidKey
 54
 55	fmt.Println(j.ToString())
 56	goodjunk(rw, j)
 57}
 58
 59// https://docs.joinmastodon.org/methods/oauth/#authorize
 60func oauthorize(rw http.ResponseWriter, r *http.Request) {
 61	clientID := r.FormValue("client_id")
 62	redirectURI := r.FormValue("redirect_uri")
 63
 64	err := stmtCheckClientId.QueryRow(clientID).Scan()
 65	if err == sql.ErrNoRows {
 66		elog.Println("oauth: no such client:", clientID)
 67		rw.WriteHeader(http.StatusUnauthorized)
 68		return
 69	}
 70
 71	var nrw NotResponseWriter
 72	login.LoginFunc(&nrw, r)
 73
 74	_, err = stmtSaveMastoAppToken.Exec(nrw.auth)
 75	if err != nil {
 76		elog.Println("oauth: failed to save masto app token", err)
 77		rw.WriteHeader(http.StatusInternalServerError)
 78		return
 79	}
 80
 81	log.Println("redirecting to", redirectURI+"?code="+nrw.auth)
 82	rw.Header().Set("Content-Type", "")
 83	http.Redirect(rw, r, redirectURI+"?code="+nrw.auth, 302)
 84}
 85
 86// https://docs.joinmastodon.org/methods/instance/#v2
 87func instance(rw http.ResponseWriter, r *http.Request) {
 88	j := junk.New()
 89
 90	var servername string
 91	if err := getconfig("servername", &servername); err != nil {
 92		http.Error(rw, "getting servername", http.StatusInternalServerError)
 93		return
 94	}
 95
 96	j["uri"] = servername
 97	j["title"] = "honk"
 98	j["description"] = "federated honk conveyance"
 99	j["version"] = "develop"
100
101	thumbnail := junk.New()
102	thumbnail["url"] = fmt.Sprintf("https://%s/icon.png", servername)
103	j["thumbnail"] = thumbnail
104	j["languages"] = []string{"en"}
105
106	config := junk.New()
107
108	a := junk.New()
109	a["max_featured_tags"] = 10
110	config["accounts"] = a
111
112	s := junk.New()
113	s["max_characters"] = 5000
114	s["max_media_attachments"] = 1
115	s["characters_reserved_per_url"] = 23
116	config["statuses"] = s
117
118	m := junk.New()
119	m["supported_mime_types"] = []string{
120		"image/jpeg",
121		"image/png",
122		"image/gif",
123		"image/heic",
124		"image/heif",
125		"image/webp",
126		"image/avif",
127		"video/webm",
128		"video/mp4",
129		"video/quicktime",
130		"video/ogg",
131		"audio/wave",
132		"audio/wav",
133		"audio/x-wav",
134		"audio/x-pn-wave",
135		"audio/vnd.wave",
136		"audio/ogg",
137		"audio/vorbis",
138		"audio/mpeg",
139		"audio/mp3",
140		"audio/webm",
141		"audio/flac",
142		"audio/aac",
143		"audio/m4a",
144		"audio/x-m4a",
145		"audio/mp4",
146		"audio/3gpp",
147		"video/x-ms-asf",
148	}
149
150	m["image_size_limit"] = 10485760
151	m["image_matrix_limit"] = 16777216
152	m["video_size_limit"] = 41943040
153	m["video_frame_rate_limit"] = 60
154	m["video_matrix_limit"] = 2304000
155	j["media_attachments"] = m
156
157	goodjunk(rw, j)
158
159	return
160}