216 lines
5.4 KiB
Go
216 lines
5.4 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"crypto/tls"
|
|
"database/sql"
|
|
"embed"
|
|
"encoding/json"
|
|
"io"
|
|
"net/http"
|
|
"time"
|
|
|
|
"tailscale.com/client/tailscale"
|
|
"tailscale.com/client/tailscale/apitype"
|
|
"tailscale.com/tsnet"
|
|
"within.website/ln"
|
|
"within.website/ln/ex"
|
|
)
|
|
|
|
//go:embed static/*
|
|
var staticFS embed.FS
|
|
|
|
func withWhois(next func(whois *apitype.WhoIsResponse, w http.ResponseWriter, r *http.Request)) http.Handler {
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
userInfo, err := tailscale.WhoIs(r.Context(), r.RemoteAddr)
|
|
if err != nil {
|
|
http.Error(w, "can't get whois response: "+err.Error(), http.StatusInternalServerError)
|
|
ln.Error(r.Context(), err)
|
|
return
|
|
}
|
|
|
|
next(userInfo, w, r)
|
|
})
|
|
}
|
|
|
|
type Server struct {
|
|
db *sql.DB
|
|
mux *http.ServeMux
|
|
srv *tsnet.Server
|
|
}
|
|
|
|
func NewServer(db *sql.DB, srv *tsnet.Server) *Server {
|
|
mux := http.NewServeMux()
|
|
|
|
s := &Server{
|
|
db: db,
|
|
mux: mux,
|
|
srv: srv,
|
|
}
|
|
|
|
mux.HandleFunc("/schema.sql", func(w http.ResponseWriter, r *http.Request) {
|
|
w.Header().Set("Content-Type", "text/plain")
|
|
fin, err := schemaFS.Open("schema/schema.sql")
|
|
if err != nil {
|
|
http.Error(w, "not found??", http.StatusNotFound)
|
|
return
|
|
}
|
|
defer fin.Close()
|
|
io.Copy(w, fin)
|
|
})
|
|
|
|
mux.HandleFunc("/api/whois", s.whois)
|
|
mux.HandleFunc("/api/importcsv", s.importCSV)
|
|
mux.HandleFunc("/api/queries/get", s.getQueries)
|
|
mux.Handle("/static/", http.FileServer(http.FS(staticFS)))
|
|
|
|
return s
|
|
}
|
|
|
|
func (s *Server) ListenAndServe() error {
|
|
l, err := s.srv.Listen("tcp", ":443")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
l = tls.NewListener(l, &tls.Config{
|
|
GetCertificate: func(chi *tls.ClientHelloInfo) (*tls.Certificate, error) {
|
|
c, err := tailscale.GetCertificate(chi)
|
|
if err != nil {
|
|
ln.Error(context.Background(), err, ln.F{"remote_addr": chi.Conn.RemoteAddr().String()})
|
|
}
|
|
return c, err
|
|
},
|
|
})
|
|
|
|
hs := &http.Server{
|
|
IdleTimeout: 5 * time.Minute,
|
|
Handler: ex.HTTPLog(s),
|
|
}
|
|
|
|
ln.Log(context.Background(), ln.Info("listening on https://twitchalitics.shark-harmonic.ts.net"))
|
|
|
|
return hs.Serve(l)
|
|
}
|
|
|
|
func (s *Server) whois(w http.ResponseWriter, r *http.Request) {
|
|
userInfo, err := tailscale.WhoIs(r.Context(), r.RemoteAddr)
|
|
if err != nil {
|
|
http.Error(w, "can't get whois response: "+err.Error(), http.StatusInternalServerError)
|
|
ln.Error(r.Context(), err)
|
|
return
|
|
}
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
|
w.WriteHeader(http.StatusOK)
|
|
json.NewEncoder(w).Encode(userInfo)
|
|
}
|
|
|
|
func (s *Server) addQuery(whois *apitype.WhoIsResponse, w http.ResponseWriter, r *http.Request) {
|
|
type AddQueryRequest struct {
|
|
Name string `json:"name"`
|
|
Query string `json:"query"`
|
|
}
|
|
|
|
var aqr AddQueryRequest
|
|
err := json.NewDecoder(r.Body).Decode(&aqr)
|
|
if err != nil {
|
|
ln.Error(r.Context(), err)
|
|
http.Error(w, "bad json", http.StatusBadRequest)
|
|
return
|
|
}
|
|
|
|
who := whois.UserProfile.LoginName
|
|
|
|
_, err = s.db.Exec("INSERT INTO sql_queries(name, query, who) VALUES (?, ?, ?)", aqr.Name, aqr.Query, who)
|
|
if err != nil {
|
|
ln.Error(r.Context(), err)
|
|
http.Error(w, "can't insert query", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
}
|
|
|
|
func (s *Server) getQueries(w http.ResponseWriter, r *http.Request) {
|
|
type SQLQuery struct {
|
|
Name string `json:"name"`
|
|
Query string `json:"query"`
|
|
Who string `json:"who"`
|
|
}
|
|
|
|
var queries []SQLQuery
|
|
rows, err := s.db.QueryContext(r.Context(), "SELECT name, query, who FROM sql_queries")
|
|
if err != nil {
|
|
http.Error(w, "can't get query rows", http.StatusInternalServerError)
|
|
ln.Error(r.Context(), err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
|
|
for rows.Next() {
|
|
var sq SQLQuery
|
|
err := rows.Scan(&sq.Name, &sq.Query, &sq.Who)
|
|
if err != nil {
|
|
http.Error(w, "can't get query row", http.StatusInternalServerError)
|
|
ln.Error(r.Context(), err)
|
|
}
|
|
|
|
queries = append(queries, sq)
|
|
}
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
|
w.WriteHeader(http.StatusOK)
|
|
json.NewEncoder(w).Encode(queries)
|
|
}
|
|
|
|
// curl https://twitchalitics.shark-harmonic.ts.net/api/importcsv -v -F csv=@"Channel Analytics and Revenue by day from Dec_19_2021 to Jan_17_2022.csv"
|
|
func (s *Server) importCSV(w http.ResponseWriter, r *http.Request) {
|
|
tx, err := s.db.BeginTx(r.Context(), nil)
|
|
if err != nil {
|
|
ln.Error(r.Context(), err)
|
|
http.Error(w, "can't hit db", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
defer tx.Rollback()
|
|
|
|
r.ParseMultipartForm(10 << 20)
|
|
file, handler, err := r.FormFile("csv")
|
|
if err != nil {
|
|
ln.Error(r.Context(), err)
|
|
http.Error(w, "can't read file", http.StatusInternalServerError)
|
|
}
|
|
|
|
ln.WithF(r.Context(), ln.F{"filename": handler.Filename, "size": handler.Size})
|
|
rows, err := ReadRows(file)
|
|
|
|
for _, row := range rows {
|
|
err := row.Upsert(tx)
|
|
if err != nil {
|
|
ln.Error(r.Context(), err)
|
|
http.Error(w, "can't upsert row", http.StatusInternalServerError)
|
|
return
|
|
}
|
|
}
|
|
|
|
err = tx.Commit()
|
|
if err != nil {
|
|
ln.Error(r.Context(), err)
|
|
http.Error(w, "can't commit data", http.StatusInternalServerError)
|
|
}
|
|
|
|
http.Redirect(w, r, "/", http.StatusTemporaryRedirect)
|
|
}
|
|
|
|
func (s Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
userInfo, err := tailscale.WhoIs(r.Context(), r.RemoteAddr)
|
|
if err != nil {
|
|
http.Error(w, "can't get whois response: "+err.Error(), http.StatusInternalServerError)
|
|
ln.Error(r.Context(), err)
|
|
return
|
|
}
|
|
|
|
ctx := ln.WithF(r.Context(), ln.F{"username": userInfo.UserProfile.LoginName, "hostname": userInfo.Node.ComputedName, "os": userInfo.Node.Hostinfo.OS})
|
|
r = r.WithContext(ctx)
|
|
|
|
s.mux.ServeHTTP(w, r)
|
|
}
|