basic bitch

master
Bel LaPointe 2020-10-07 21:41:03 -06:00
commit efad96aa36
3 changed files with 136 additions and 0 deletions

31
config.go Normal file
View File

@ -0,0 +1,31 @@
package main
import (
"fmt"
"local/args"
"time"
)
type Config struct {
Listen string
Timeout time.Duration
TLSInsecure bool
}
func NewConfig() *Config {
as := args.NewArgSet()
as.Append(args.INT, "p", "port to listen on", 61113)
as.Append(args.BOOL, "tls-insecure", "permit tls insecure", false)
as.Append(args.DURATION, "t", "timeout", time.Minute)
if err := as.Parse(); err != nil {
panic(err)
}
return &Config{
Listen: fmt.Sprintf(":%v", as.GetInt("p")),
Timeout: as.GetDuration("t"),
TLSInsecure: as.GetBool("tls-insecure"),
}
}

16
main.go Normal file
View File

@ -0,0 +1,16 @@
package main
import (
"log"
"net/http"
)
func main() {
config := NewConfig()
server := NewServer(config)
log.Printf("config: %+v", *config)
if err := http.ListenAndServe(config.Listen, server); err != nil {
panic(err)
}
}

89
server.go Normal file
View File

@ -0,0 +1,89 @@
package main
import (
"crypto/tls"
"errors"
"io"
"log"
"net"
"net/http"
"time"
)
type Server struct {
Transport http.RoundTripper
Timeout time.Duration
}
func NewServer(c *Config) *Server {
transport := &http.Transport{}
transport.TLSClientConfig = &tls.Config{
InsecureSkipVerify: c.TLSInsecure,
}
return &Server{
Transport: transport,
Timeout: c.Timeout,
}
}
func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
switch r.Method {
case http.MethodConnect:
s.Connect(w, r)
default:
s.Serve(w, r)
}
}
func (s *Server) Error(w http.ResponseWriter, err error) {
log.Println(err)
http.Error(w, err.Error(), http.StatusServiceUnavailable)
return
}
func (s *Server) Connect(w http.ResponseWriter, r *http.Request) {
dest, err := net.DialTimeout("tcp", r.Host, 30*time.Second)
if err != nil {
s.Error(w, err)
return
}
w.WriteHeader(http.StatusOK)
hijacker, ok := w.(http.Hijacker)
if !ok {
s.Error(w, errors.New("hijack not available"))
return
}
client, _, err := hijacker.Hijack()
if err != nil {
s.Error(w, err)
return
}
xfer := func(dst io.WriteCloser, src io.ReadCloser) {
defer dst.Close()
defer src.Close()
io.Copy(dst, src)
}
go xfer(dest, client)
go xfer(client, dest)
}
func (s *Server) Serve(w http.ResponseWriter, r *http.Request) {
resp, err := s.Transport.RoundTrip(r)
if err != nil {
s.Error(w, err)
return
}
defer resp.Body.Close()
w.WriteHeader(resp.StatusCode)
for k, v := range resp.Header {
for _, s := range v {
w.Header().Add(k, s)
}
}
io.Copy(w, resp.Body)
}