mirror of
https://github.com/caddyserver/caddy.git
synced 2025-03-21 04:58:58 +01:00
removed panics, cleaned up leaking ticker routine
This commit is contained in:
parent
222781abca
commit
24893bf740
1 changed files with 24 additions and 15 deletions
|
@ -83,35 +83,35 @@ func serveWS(w http.ResponseWriter, r *http.Request, config *Config) (int, error
|
||||||
}
|
}
|
||||||
conn, err := upgrader.Upgrade(w, r, nil)
|
conn, err := upgrader.Upgrade(w, r, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, err
|
return http.StatusBadRequest, err
|
||||||
}
|
}
|
||||||
defer conn.Close()
|
defer conn.Close()
|
||||||
|
|
||||||
cmd := exec.Command(config.Command, config.Arguments...)
|
cmd := exec.Command(config.Command, config.Arguments...)
|
||||||
stdout, err := cmd.StdoutPipe()
|
stdout, err := cmd.StdoutPipe()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err) // TODO
|
return http.StatusBadGateway, err
|
||||||
}
|
}
|
||||||
|
|
||||||
stdin, err := cmd.StdinPipe()
|
stdin, err := cmd.StdinPipe()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err) // TODO
|
return http.StatusBadGateway, err
|
||||||
}
|
}
|
||||||
|
|
||||||
metavars, err := buildEnv(cmd.Path, r)
|
metavars, err := buildEnv(cmd.Path, r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err) // TODO
|
return http.StatusBadGateway, err
|
||||||
}
|
}
|
||||||
|
|
||||||
cmd.Env = metavars
|
cmd.Env = metavars
|
||||||
|
|
||||||
if err := cmd.Start(); err != nil {
|
if err := cmd.Start(); err != nil {
|
||||||
panic(err)
|
return http.StatusBadGateway, err
|
||||||
}
|
}
|
||||||
|
|
||||||
reader(conn, stdout, stdin)
|
reader(conn, stdout, stdin)
|
||||||
|
|
||||||
return 0, nil // we shouldn't get here.
|
return 0, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// buildEnv creates the meta-variables for the child process according
|
// buildEnv creates the meta-variables for the child process according
|
||||||
|
@ -171,32 +171,39 @@ func reader(conn *websocket.Conn, stdout io.ReadCloser, stdin io.WriteCloser) {
|
||||||
conn.SetReadLimit(maxMessageSize)
|
conn.SetReadLimit(maxMessageSize)
|
||||||
conn.SetReadDeadline(time.Now().Add(pongWait))
|
conn.SetReadDeadline(time.Now().Add(pongWait))
|
||||||
conn.SetPongHandler(func(string) error { conn.SetReadDeadline(time.Now().Add(pongWait)); return nil })
|
conn.SetPongHandler(func(string) error { conn.SetReadDeadline(time.Now().Add(pongWait)); return nil })
|
||||||
go ticker(conn)
|
tickerChan := make(chan bool)
|
||||||
|
defer func() { tickerChan <- true }() // make sure to close the ticker when we are done.
|
||||||
|
go ticker(conn, tickerChan)
|
||||||
|
|
||||||
for {
|
for {
|
||||||
msgType, r, err := conn.NextReader()
|
msgType, r, err := conn.NextReader()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if msgType == -1 {
|
if msgType == -1 {
|
||||||
return // we are done, as we got a close method.
|
return // we got a disconnect from the client. We are good to close.
|
||||||
}
|
}
|
||||||
panic(err) // TODO do something else here.
|
conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, ""), time.Time{})
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
w, err := conn.NextWriter(msgType)
|
w, err := conn.NextWriter(msgType)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err) // TODO do something else here.
|
conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, ""), time.Time{})
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, err := io.Copy(stdin, r); err != nil {
|
if _, err := io.Copy(stdin, r); err != nil {
|
||||||
panic(err) // TODO do something else here.
|
conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, ""), time.Time{})
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
if _, err := io.Copy(w, stdout); err != nil {
|
if _, err := io.Copy(w, stdout); err != nil {
|
||||||
panic(err) // TODO do something else here.
|
conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, ""), time.Time{})
|
||||||
|
return
|
||||||
}
|
}
|
||||||
if err := w.Close(); err != nil {
|
if err := w.Close(); err != nil {
|
||||||
panic(err) // TODO do something else here.
|
conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, ""), time.Time{})
|
||||||
|
return
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
@ -204,17 +211,19 @@ func reader(conn *websocket.Conn, stdout io.ReadCloser, stdin io.WriteCloser) {
|
||||||
|
|
||||||
// ticker is start by the reader. Basically it is the method that simulates the websocket
|
// ticker is start by the reader. Basically it is the method that simulates the websocket
|
||||||
// between the server and client to keep it alive with ping messages.
|
// between the server and client to keep it alive with ping messages.
|
||||||
func ticker(conn *websocket.Conn) {
|
func ticker(conn *websocket.Conn, c chan bool) {
|
||||||
ticker := time.NewTicker(pingPeriod)
|
ticker := time.NewTicker(pingPeriod)
|
||||||
defer func() {
|
defer func() {
|
||||||
ticker.Stop()
|
ticker.Stop()
|
||||||
conn.WriteMessage(websocket.CloseMessage, nil)
|
close(c)
|
||||||
}()
|
}()
|
||||||
|
|
||||||
for { // blocking loop with select to wait for stimulation.
|
for { // blocking loop with select to wait for stimulation.
|
||||||
select {
|
select {
|
||||||
case <-ticker.C:
|
case <-ticker.C:
|
||||||
conn.WriteMessage(websocket.PingMessage, nil)
|
conn.WriteMessage(websocket.PingMessage, nil)
|
||||||
|
case <-c:
|
||||||
|
return // clean up this routine.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue