lil refactor

main
Bel LaPointe 2024-11-12 07:48:14 -07:00
parent fc5b26d74d
commit 9525032c24
2 changed files with 82 additions and 59 deletions

BIN
json-adapter Executable file

Binary file not shown.

89
main.go
View File

@ -15,6 +15,40 @@ import (
"time" "time"
) )
var httpc = &http.Client{
Timeout: time.Minute,
Transport: &http.Transport{
DisableKeepAlives: true,
},
}
type Handler struct {
tmpl *template.Template
target *url.URL
}
func (h Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
if err := h.serveHTTP(w, r); err != nil {
log.Println("!", err)
http.Error(w, err.Error(), http.StatusBadRequest)
}
}
func (h Handler) serveHTTP(w http.ResponseWriter, r *http.Request) error {
b, err := adapt(r.Body, h.tmpl)
if err != nil {
return err
}
resp, err := proxy(h.target, r, io.NopCloser(bytes.NewReader(b)))
if err != nil {
return err
}
defer resp.Body.Close()
return forward(w, resp)
}
func main() { func main() {
fs := flag.NewFlagSet(os.Args[0], flag.ContinueOnError) fs := flag.NewFlagSet(os.Args[0], flag.ContinueOnError)
p := fs.Int("p", 28080, "port") p := fs.Int("p", 28080, "port")
@ -34,31 +68,32 @@ func main() {
panic(err) panic(err)
} }
c := &http.Client{ h := Handler{tmpl: tmpl, target: u}
Timeout: time.Minute, log.Println("listening on", *p)
Transport: &http.Transport{ if err := http.ListenAndServe(":"+strconv.Itoa(*p), h); err != nil {
DisableKeepAlives: true, panic(err)
},
} }
}
handle := func(w http.ResponseWriter, r *http.Request) error { func adapt(r io.Reader, tmpl *template.Template) ([]byte, error) {
b, _ := io.ReadAll(io.LimitReader(r.Body, 1024*1024)) b, _ := io.ReadAll(io.LimitReader(r, 1024*1024))
var v interface{} var v interface{}
if err := json.Unmarshal(b, &v); err != nil {
return err
}
buff := bytes.NewBuffer(nil) buff := bytes.NewBuffer(nil)
if err := tmpl.Execute(buff, v); err != nil { if err := json.Unmarshal(b, &v); err != nil {
return err return nil, err
} else if err := tmpl.Execute(buff, v); err != nil {
return nil, err
} }
log.Printf("%s => %s", b, buff.Bytes())
return buff.Bytes(), nil
}
log.Printf("%s => %s => %s", b, buff.Bytes(), u.String()) func proxy(u *url.URL, r *http.Request, rc io.ReadCloser) (*http.Response, error) {
req, err := http.NewRequest(r.Method, u.String(), rc)
req, err := http.NewRequest(r.Method, u.String(), io.NopCloser(buff))
if err != nil { if err != nil {
return err return nil, err
} }
req = req.WithContext(r.Context()) req = req.WithContext(r.Context())
for k, v := range r.Header { for k, v := range r.Header {
switch strings.ToLower(k) { switch strings.ToLower(k) {
@ -70,12 +105,11 @@ func main() {
} }
} }
} }
resp, err := c.Do(req)
if err != nil {
return err
}
defer resp.Body.Close()
return httpc.Do(req)
}
func forward(w http.ResponseWriter, resp *http.Response) error {
for k, v := range resp.Header { for k, v := range resp.Header {
w.Header().Set(k, v[0]) w.Header().Set(k, v[0])
for _, v := range v[1:] { for _, v := range v[1:] {
@ -84,16 +118,5 @@ func main() {
} }
w.WriteHeader(resp.StatusCode) w.WriteHeader(resp.StatusCode)
io.Copy(w, resp.Body) io.Copy(w, resp.Body)
return nil return nil
}
log.Println("listening on", *p)
if err := http.ListenAndServe(":"+strconv.Itoa(*p), http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if err := handle(w, r); err != nil {
log.Println("!", err)
http.Error(w, err.Error(), http.StatusBadRequest)
}
})); err != nil {
panic(err)
}
} }