basic bitch
commit
efad96aa36
|
|
@ -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"),
|
||||
}
|
||||
}
|
||||
|
|
@ -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)
|
||||
}
|
||||
}
|
||||
|
|
@ -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)
|
||||
}
|
||||
Loading…
Reference in New Issue