viddl/main.go

203 lines
5.0 KiB
Go
Raw Normal View History

2023-09-10 22:18:30 +02:00
package main
import (
"encoding/json"
2023-09-11 01:46:14 +02:00
"html/template"
2023-09-10 22:18:30 +02:00
"log"
"net/http"
"os"
2023-09-10 22:18:30 +02:00
"time"
)
const (
DEFAULT_HTTP_PORT = "8080"
DEFAULT_HTTPS_PORT = "4433"
)
2023-09-11 01:46:14 +02:00
type DownloadFormats struct {
VideoRes string
videoOnly bool
audioOnly bool
}
2023-09-10 22:18:30 +02:00
type apiMessageResponse struct {
Message string
}
type Logger struct {
Handler http.Handler
}
func (l *Logger) ServeHTTP(w http.ResponseWriter, r *http.Request) {
tNow := time.Now().UTC()
l.Handler.ServeHTTP(w, r)
methodString := l.getMethodLogString(r.Method)
remote := r.RemoteAddr
realIP := r.Header.Get("X-Real-IP")
if realIP != "" {
remote = realIP
}
log.Printf(" %s %s %s %v", remote, methodString, r.URL, time.Since(tNow))
2023-09-10 22:18:30 +02:00
}
func (l *Logger) getMethodLogString(method string) string {
colorBlue := "\033[34m"
colorReset := "\033[0m"
2023-09-10 22:33:06 +02:00
colorRed := "\033[31m"
colorGreen := "\033[32m"
colorYellow := "\033[33m"
2023-09-10 22:18:30 +02:00
// colorPurple := "\033[35m"
// colorCyan := "\033[36m"
// colorWhite := "\033[37m"
2023-09-10 22:33:06 +02:00
switch method {
case "GET": return colorBlue + "GET" + colorReset
case "POST": return colorGreen + "POST" + colorReset
case "DELETE": return colorRed + "DELETE" + colorReset
case "PUT": return colorYellow + "PUT" + colorReset
default: return method
}
2023-09-10 22:18:30 +02:00
}
func NewLogger(handler http.Handler) *Logger {
return &Logger{Handler: handler}
}
func writeJSONResponse(w http.ResponseWriter, s string) http.ResponseWriter {
w.WriteHeader(http.StatusBadRequest)
2023-09-11 01:46:14 +02:00
w.Header().Set("Content-Type", "application/json")
2023-09-10 22:18:30 +02:00
jsonResp, err := json.Marshal(apiMessageResponse{Message: s})
if err != nil {
w.WriteHeader(http.StatusInternalServerError)
return w
}
_, err = w.Write(jsonResp)
if err != nil {
w.WriteHeader(http.StatusInternalServerError)
return w
}
return w
}
var templates *template.Template
// TODO: Change all this to have a unified context
2023-09-11 22:27:18 +02:00
type Context struct {
request *http.Request
Formats []DownloadFormats
AppURL string
StatusCode int
Err *error
IsTLS bool
DownloadURL string
2023-09-11 22:27:18 +02:00
}
func NewContext(r *http.Request) *Context {
isTLS := false
tls := os.Getenv("TLS")
if tls != "" {
isTLS = true
}
2023-09-11 22:27:18 +02:00
return &Context{
request: r,
StatusCode: 200,
Formats: []DownloadFormats{
{
VideoRes: "720p",
videoOnly: false,
audioOnly: false,
},
},
AppURL: r.Host,
IsTLS: isTLS,
2023-09-11 22:27:18 +02:00
}
}
func init() {
log.Println("[ init ] Starting...")
templates = template.Must(template.ParseFS(TemplatesFS , "templates/*.html"))
log.Println("[ init ] Templates Loaded")
}
2023-09-10 22:18:30 +02:00
func main() {
handler := http.NewServeMux()
handler.Handle("/static/", http.StripPrefix("/static/", http.FileServer(http.FS(PublicFS))))
handler.Handle("/assets/", http.FileServer(http.FS(AssetsFS)))
2023-09-10 22:18:30 +02:00
handler.HandleFunc(
"/download",
func(w http.ResponseWriter, r *http.Request) {
2023-09-11 22:27:18 +02:00
ctx := NewContext(r)
2023-09-11 01:46:14 +02:00
if r.Method != "POST" {
w.WriteHeader(400)
return
}
err := r.ParseForm()
if err != nil {
w.WriteHeader(400)
return
}
userURL := r.FormValue("URL")
2023-09-10 22:18:30 +02:00
if userURL == "" {
w = writeJSONResponse(w, "Provide URL as query")
return
}
2023-09-11 01:46:14 +02:00
downloadURL, err := getYoutubeDownloadURL(userURL)
2023-09-11 01:46:14 +02:00
if err != nil {
log.Println(err.Error())
2023-09-11 22:27:18 +02:00
ctx.StatusCode = 500
ctx.Err = &err
err = templates.ExecuteTemplate(w,"download-result.html", ctx)
2023-09-11 01:46:14 +02:00
return
}
ctx.DownloadURL = downloadURL
2023-09-11 22:27:18 +02:00
err = templates.ExecuteTemplate(w,"download-result.html", ctx)
2023-09-11 01:46:14 +02:00
if err != nil {
log.Println(err.Error())
2023-09-11 22:27:18 +02:00
ctx.StatusCode = 500
ctx.Err = &err
err = templates.ExecuteTemplate(w,"download-result.html", ctx)
2023-09-11 01:46:14 +02:00
return
}
},
2023-09-10 22:18:30 +02:00
)
2023-09-11 01:46:14 +02:00
handler.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
2023-09-11 22:27:18 +02:00
ctx := NewContext(r)
2023-09-11 01:46:14 +02:00
formats := []DownloadFormats{}
formats = append(formats, DownloadFormats{
VideoRes: "720p",
audioOnly: false,
videoOnly: false,
})
2023-09-11 22:27:18 +02:00
err := templates.ExecuteTemplate(w, "download.html", ctx)
2023-09-11 01:46:14 +02:00
if err != nil {
log.Println(err.Error())
w.WriteHeader(500)
return
}
})
2023-09-10 22:18:30 +02:00
wrappedHandler := NewLogger(handler)
srv := http.Server{
2023-09-11 01:46:14 +02:00
ReadTimeout: 10 * time.Second,
2023-09-10 22:18:30 +02:00
WriteTimeout: 10 * time.Second,
Addr: ":" + DEFAULT_HTTP_PORT,
Handler: wrappedHandler,
}
log.Printf("Starting HTTP on %s", DEFAULT_HTTP_PORT)
log.Fatalln(srv.ListenAndServe())
}