http3-ytproxy/main.go

348 lines
7.9 KiB
Go
Raw Normal View History

2020-10-24 12:47:41 -03:00
package main
import (
2024-09-17 03:23:49 -03:00
"flag"
2020-10-24 12:47:41 -03:00
"fmt"
"image/jpeg"
2020-10-25 09:41:17 -03:00
"io"
2020-10-24 12:47:41 -03:00
"log"
2020-10-25 09:41:17 -03:00
"net"
2020-10-24 12:47:41 -03:00
"net/http"
2020-10-25 09:41:17 -03:00
"net/url"
2020-10-25 11:01:23 -03:00
"os"
2021-11-07 15:23:39 -03:00
"regexp"
2020-10-25 09:41:17 -03:00
"strings"
"syscall"
2021-03-12 03:59:53 -03:00
"time"
2020-10-24 12:47:41 -03:00
"github.com/kolesa-team/go-webp/encoder"
"github.com/kolesa-team/go-webp/webp"
2023-11-07 11:05:53 -03:00
"github.com/quic-go/quic-go/http3"
2020-10-24 12:47:41 -03:00
)
2020-10-25 09:41:17 -03:00
// http/3 client
var h3client = &http.Client{
2020-10-24 12:47:41 -03:00
Transport: &http3.RoundTripper{},
2024-09-17 03:23:49 -03:00
Timeout: 10 * time.Second,
2020-10-24 12:47:41 -03:00
}
2022-06-27 08:25:31 -04:00
var dialer = &net.Dialer{
Timeout: 30 * time.Second,
KeepAlive: 30 * time.Second,
}
2020-10-25 09:41:17 -03:00
// http/2 client
2021-03-12 03:59:53 -03:00
var h2client = &http.Client{
Transport: &http.Transport{
2022-06-27 08:25:31 -04:00
Dial: func(network, addr string) (net.Conn, error) {
if disable_ipv6 {
network = "tcp4"
}
return dialer.Dial(network, addr)
},
2021-03-12 03:59:53 -03:00
TLSHandshakeTimeout: 10 * time.Second,
2021-06-20 06:37:39 -04:00
ResponseHeaderTimeout: 20 * time.Second,
2021-03-12 03:59:53 -03:00
ExpectContinueTimeout: 1 * time.Second,
2021-04-09 04:50:14 -04:00
IdleConnTimeout: 30 * time.Second,
ReadBufferSize: 16 * 1024,
ForceAttemptHTTP2: true,
MaxConnsPerHost: 0,
MaxIdleConnsPerHost: 10,
MaxIdleConns: 0,
2021-03-12 03:59:53 -03:00
},
}
2020-10-25 09:41:17 -03:00
// user agent to use
2024-09-17 03:36:24 -03:00
var ua = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/128.0.0.0 Safari/537.36"
2020-10-25 09:41:17 -03:00
2021-07-21 14:23:27 -04:00
var allowed_hosts = []string{
"youtube.com",
"googlevideo.com",
"ytimg.com",
"ggpht.com",
"googleusercontent.com",
2021-09-02 09:44:01 -04:00
"lbryplayer.xyz",
"odycdn.com",
2021-07-21 14:23:27 -04:00
}
2022-05-17 06:19:43 -04:00
var strip_headers = []string{
"Accept-Encoding",
"Authorization",
"Origin",
2022-05-18 08:50:31 -04:00
"Referer",
2022-05-17 06:19:43 -04:00
"Cookie",
"Set-Cookie",
"Etag",
2022-05-17 06:19:43 -04:00
}
var path_prefix = ""
2021-11-07 15:23:39 -03:00
var manifest_re = regexp.MustCompile(`(?m)URI="([^"]+)"`)
2022-06-27 08:25:31 -04:00
var disable_ipv6 = false
var disable_webp = false
2022-06-27 08:25:31 -04:00
2021-03-04 05:57:42 -03:00
type requesthandler struct{}
func (*requesthandler) ServeHTTP(w http.ResponseWriter, req *http.Request) {
2020-10-25 09:41:17 -03:00
2021-11-07 15:23:39 -03:00
w.Header().Set("Access-Control-Allow-Origin", "*")
w.Header().Set("Access-Control-Allow-Headers", "*")
w.Header().Set("Access-Control-Max-Age", "1728000")
if req.Method == "OPTIONS" {
w.WriteHeader(200)
return
}
2020-10-25 09:41:17 -03:00
q := req.URL.Query()
host := q.Get("host")
q.Del("host")
2020-12-10 10:37:10 -03:00
if len(host) <= 0 {
host = q.Get("hls_chunk_host")
}
2020-10-25 09:41:17 -03:00
if len(host) <= 0 {
2021-06-09 17:11:57 -04:00
host = getHost(req.URL.EscapedPath())
2020-10-25 09:41:17 -03:00
}
if len(host) <= 0 {
io.WriteString(w, "No host in query parameters.")
return
2020-10-25 09:41:17 -03:00
}
2020-10-24 12:47:41 -03:00
2021-07-21 14:23:27 -04:00
parts := strings.Split(strings.ToLower(host), ".")
if len(parts) < 2 {
io.WriteString(w, "Invalid hostname.")
return
}
domain := parts[len(parts)-2] + "." + parts[len(parts)-1]
disallowed := true
for _, value := range allowed_hosts {
if domain == value {
disallowed = false
break
}
}
if disallowed {
io.WriteString(w, "Non YouTube domains are not supported.")
return
}
if req.Method != "GET" && req.Method != "HEAD" {
io.WriteString(w, "Only GET and HEAD requests are allowed.")
return
}
2021-06-09 17:11:57 -04:00
path := req.URL.EscapedPath()
path = strings.Replace(path, "/ggpht", "", 1)
path = strings.Replace(path, "/i/", "/", 1)
proxyURL, err := url.Parse("https://" + host + path)
2020-10-24 12:47:41 -03:00
if err != nil {
2020-10-25 09:41:17 -03:00
log.Panic(err)
2020-10-24 12:47:41 -03:00
}
2020-10-25 09:41:17 -03:00
proxyURL.RawQuery = q.Encode()
2021-06-09 17:11:57 -04:00
if strings.HasSuffix(proxyURL.EscapedPath(), "maxres.jpg") {
proxyURL.Path = getBestThumbnail(proxyURL.EscapedPath())
}
request, err := http.NewRequest(req.Method, proxyURL.String(), nil)
2020-10-25 09:41:17 -03:00
copyHeaders(req.Header, request.Header, false)
2020-10-25 09:41:17 -03:00
request.Header.Set("User-Agent", ua)
2020-10-24 12:47:41 -03:00
if err != nil {
2020-10-25 09:41:17 -03:00
log.Panic(err)
2020-10-24 12:47:41 -03:00
}
2020-10-25 09:41:17 -03:00
var client *http.Client
2020-10-25 12:57:05 -03:00
// https://github.com/lucas-clemente/quic-go/issues/2836
2024-09-17 03:23:49 -03:00
client = h3client
2020-10-25 09:41:17 -03:00
resp, err := client.Do(request)
if err != nil {
log.Panic(err)
}
2021-03-04 05:57:42 -03:00
defer resp.Body.Close()
NoRewrite := strings.HasPrefix(resp.Header.Get("Content-Type"), "audio") || strings.HasPrefix(resp.Header.Get("Content-Type"), "video") || strings.HasPrefix(resp.Header.Get("Content-Type"), "webp")
copyHeaders(resp.Header, w.Header(), NoRewrite)
2020-10-25 09:41:17 -03:00
w.WriteHeader(resp.StatusCode)
2021-11-07 15:23:39 -03:00
if req.Method == "GET" && (resp.Header.Get("Content-Type") == "application/x-mpegurl" || resp.Header.Get("Content-Type") == "application/vnd.apple.mpegurl") {
bytes, err := io.ReadAll(resp.Body)
if err != nil {
log.Panic(err)
}
lines := strings.Split(string(bytes), "\n")
reqUrl := resp.Request.URL
for i := 0; i < len(lines); i++ {
line := lines[i]
if !strings.HasPrefix(line, "https://") && (strings.HasSuffix(line, ".m3u8") || strings.HasSuffix(line, ".ts")) {
path := reqUrl.EscapedPath()
path = path[0 : strings.LastIndex(path, "/")+1]
2021-11-07 15:23:39 -03:00
line = "https://" + reqUrl.Hostname() + path + line
}
if strings.HasPrefix(line, "https://") {
lines[i] = RelativeUrl(line)
}
if manifest_re.MatchString(line) {
url := manifest_re.FindStringSubmatch(line)[1]
lines[i] = strings.Replace(line, url, RelativeUrl(url), 1)
}
}
io.WriteString(w, strings.Join(lines, "\n"))
} else if !disable_webp && resp.Header.Get("Content-Type") == "image/jpeg" {
img, err := jpeg.Decode(resp.Body)
if err != nil {
log.Panic(err)
}
options, _ := encoder.NewLossyEncoderOptions(encoder.PresetDefault, 85)
w.Header().Set("Content-Type", "image/webp")
webp.Encode(w, img, options)
} else {
io.Copy(w, resp.Body)
}
2020-10-25 09:41:17 -03:00
}
func copyHeaders(from http.Header, to http.Header, length bool) {
2020-10-25 09:41:17 -03:00
// Loop over header names
2022-05-17 06:19:43 -04:00
outer:
2020-10-25 09:41:17 -03:00
for name, values := range from {
2022-05-17 06:19:43 -04:00
for _, header := range strip_headers {
if name == header {
continue outer
}
}
2022-05-18 08:32:39 -04:00
if (name != "Content-Length" || length) && !strings.HasPrefix(name, "Access-Control") {
// Loop over all values for the name.
for _, value := range values {
if strings.Contains(value, "jpeg") {
continue
}
to.Set(name, value)
}
2020-10-25 09:41:17 -03:00
}
}
}
func getHost(path string) (host string) {
host = ""
2020-11-13 04:45:51 -03:00
if strings.HasPrefix(path, "/vi/") || strings.HasPrefix(path, "/vi_webp/") || strings.HasPrefix(path, "/sb/") {
2020-10-25 09:41:17 -03:00
host = "i.ytimg.com"
}
if strings.HasPrefix(path, "/ggpht/") {
host = "yt3.ggpht.com"
}
if strings.HasPrefix(path, "/a/") || strings.HasPrefix(path, "/ytc/") {
2020-10-25 09:41:17 -03:00
host = "yt3.ggpht.com"
}
2021-06-20 03:19:07 -04:00
if strings.Contains(path, "/host/") {
path = path[(strings.Index(path, "/host/") + 6):]
host = path[0:strings.Index(path, "/")]
}
2020-10-25 09:41:17 -03:00
return host
}
func getBestThumbnail(path string) (newpath string) {
formats := [4]string{"maxresdefault.jpg", "sddefault.jpg", "hqdefault.jpg", "mqdefault.jpg"}
for _, format := range formats {
newpath = strings.Replace(path, "maxres.jpg", format, 1)
url := "https://i.ytimg.com" + newpath
resp, _ := h2client.Head(url)
if resp.StatusCode == 200 {
return newpath
}
}
return strings.Replace(path, "maxres.jpg", "mqdefault.jpg", 1)
}
2021-11-07 15:23:39 -03:00
func RelativeUrl(in string) (newurl string) {
segment_url, err := url.Parse(in)
if err != nil {
log.Panic(err)
}
segment_query := segment_url.Query()
segment_query.Set("host", segment_url.Hostname())
segment_url.RawQuery = segment_query.Encode()
segment_url.Path = path_prefix + segment_url.Path
return segment_url.RequestURI()
2021-11-07 15:23:39 -03:00
}
2020-10-25 09:41:17 -03:00
func main() {
var sock string
2023-11-07 12:18:20 -03:00
var port string
2023-11-07 11:33:20 -03:00
path_prefix = os.Getenv("PREFIX_PATH")
2022-06-27 08:25:31 -04:00
disable_ipv6 = os.Getenv("DISABLE_IPV6") == "1"
disable_webp = os.Getenv("DISABLE_WEBP") == "1"
2022-06-27 08:25:31 -04:00
2023-11-07 11:33:20 -03:00
if _, err := os.Stat("socket"); os.IsNotExist(err) {
fmt.Println("socket folder doesn't exists, creating one now.")
2024-09-17 03:23:49 -03:00
err = os.Mkdir("socket", 0755)
if err != nil {
2023-11-07 12:18:20 -03:00
fmt.Println("Failed to create folder, error: ")
2024-09-17 03:23:49 -03:00
log.Fatal(err)
}
2023-11-07 11:33:20 -03:00
}
flag.StringVar(&sock, "s", "http-proxy.sock", "Specify a socket name")
2023-11-07 12:18:20 -03:00
flag.StringVar(&port, "p", "8080", "Specify a port number")
2024-09-17 03:23:49 -03:00
flag.Parse()
2023-11-07 11:33:20 -03:00
socket := "socket" + string(os.PathSeparator) + string(sock)
2020-10-25 11:01:23 -03:00
syscall.Unlink(socket)
listener, err := net.Listen("unix", socket)
2021-03-12 03:59:53 -03:00
srv := &http.Server{
ReadTimeout: 5 * time.Second,
WriteTimeout: 1 * time.Hour,
2023-11-07 12:18:20 -03:00
Addr: ":" + string(port),
2021-03-12 03:59:53 -03:00
Handler: &requesthandler{},
}
2020-10-25 09:41:17 -03:00
if err != nil {
2023-11-07 12:18:20 -03:00
fmt.Println("Failed to bind to UDS, please check the socket name, falling back to TCP/IP")
2020-10-25 09:41:17 -03:00
fmt.Println(err.Error())
2023-11-07 12:18:20 -03:00
err := srv.ListenAndServe()
if err != nil {
fmt.Println("Cannot bind to port", string(port), "Error:", err)
fmt.Println("Please try changing the port number")
}
2020-10-25 09:41:17 -03:00
} else {
2021-03-04 05:57:42 -03:00
defer listener.Close()
srv.ListenAndServe()
2020-10-25 09:41:17 -03:00
}
2020-10-24 12:47:41 -03:00
}