mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-26 08:12:43 -04:00 
			
		
		
		
	fixed data races in websockets
This commit is contained in:
		
							parent
							
								
									b6078eded1
								
							
						
					
					
						commit
						abc7c6a148
					
				| @ -4,9 +4,11 @@ | |||||||
| package websocket | package websocket | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"bufio" | ||||||
| 	"io" | 	"io" | ||||||
| 	"net" | 	"net" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  | 	"os" | ||||||
| 	"os/exec" | 	"os/exec" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| @ -88,15 +90,18 @@ func serveWS(w http.ResponseWriter, r *http.Request, config *Config) (int, error | |||||||
| 	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 { | ||||||
| 		return http.StatusBadGateway, err | 		return http.StatusBadGateway, err | ||||||
| 	} | 	} | ||||||
|  | 	defer stdout.Close() | ||||||
| 
 | 
 | ||||||
| 	stdin, err := cmd.StdinPipe() | 	stdin, err := cmd.StdinPipe() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return http.StatusBadGateway, err | 		return http.StatusBadGateway, err | ||||||
| 	} | 	} | ||||||
|  | 	defer stdin.Close() | ||||||
| 
 | 
 | ||||||
| 	metavars, err := buildEnv(cmd.Path, r) | 	metavars, err := buildEnv(cmd.Path, r) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @ -109,7 +114,31 @@ func serveWS(w http.ResponseWriter, r *http.Request, config *Config) (int, error | |||||||
| 		return http.StatusBadGateway, err | 		return http.StatusBadGateway, err | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	reader(conn, stdout, stdin) | 	done := make(chan struct{}) | ||||||
|  | 	go pumpStdout(conn, stdout, done) | ||||||
|  | 	pumpStdin(conn, stdin) | ||||||
|  | 
 | ||||||
|  | 	stdin.Close() // close stdin to end the process | ||||||
|  | 
 | ||||||
|  | 	if err := cmd.Process.Signal(os.Interrupt); err != nil { // signal an interrupt to kill the process | ||||||
|  | 		return http.StatusInternalServerError, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	select { | ||||||
|  | 	case <-done: | ||||||
|  | 	case <-time.After(time.Second): | ||||||
|  | 		// terminate with extreme prejudice. | ||||||
|  | 		if err := cmd.Process.Signal(os.Kill); err != nil { | ||||||
|  | 			return http.StatusInternalServerError, err | ||||||
|  | 		} | ||||||
|  | 		<-done | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// not sure what we want to do here. | ||||||
|  | 	// status for an "exited" process is greater | ||||||
|  | 	// than 0, but isn't really an error per se. | ||||||
|  | 	// just going to ignore it for now. | ||||||
|  | 	cmd.Wait() | ||||||
| 
 | 
 | ||||||
| 	return 0, nil | 	return 0, nil | ||||||
| } | } | ||||||
| @ -163,63 +192,60 @@ func buildEnv(cmdPath string, r *http.Request) (metavars []string, err error) { | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // reader is the guts of this package. It takes the stdin and stdout pipes | // pumpStdin handles reading data from the websocket connection and writing | ||||||
| // of the cmd we created in ServeWS and pipes them between the client and server | // it to stdin of the process. | ||||||
| // over websockets. | func pumpStdin(conn *websocket.Conn, stdin io.WriteCloser) { | ||||||
| func reader(conn *websocket.Conn, stdout io.ReadCloser, stdin io.WriteCloser) { |  | ||||||
| 	// Setup our connection's websocket ping/pong handlers from our const values. | 	// Setup our connection's websocket ping/pong handlers from our const values. | ||||||
|  | 	defer conn.Close() | ||||||
| 	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 }) | ||||||
| 	tickerChan := make(chan bool) |  | ||||||
| 	defer close(tickerChan) // make sure to close the ticker when we are done. |  | ||||||
| 	go ticker(conn, tickerChan) |  | ||||||
| 
 |  | ||||||
| 	for { | 	for { | ||||||
| 		msgType, r, err := conn.NextReader() | 		_, message, err := conn.ReadMessage() | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			if msgType == -1 { | 			break | ||||||
| 				return // we got a disconnect from the client. We are good to close. | 		} | ||||||
|  | 		message = append(message, '\n') | ||||||
|  | 		if _, err := stdin.Write(message); err != nil { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 			conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, ""), time.Time{}) |  | ||||||
| 			return |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 		w, err := conn.NextWriter(msgType) | // pumpStdout handles reading data from stdout of the process and writing | ||||||
| 		if err != nil { | // it to websocket connection. | ||||||
| 			conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, ""), time.Time{}) | func pumpStdout(conn *websocket.Conn, stdout io.Reader, done chan struct{}) { | ||||||
| 			return | 	go pinger(conn, done) | ||||||
| 		} | 	defer func() { | ||||||
| 
 | 		conn.Close() | ||||||
| 		if _, err := io.Copy(stdin, r); err != nil { | 		close(done) // make sure to close the pinger when we are done. | ||||||
| 			conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, ""), time.Time{}) |  | ||||||
| 			return |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		go func() { |  | ||||||
| 			if _, err := io.Copy(w, stdout); err != nil { |  | ||||||
| 				conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, ""), time.Time{}) |  | ||||||
| 				return |  | ||||||
| 			} |  | ||||||
| 			if err := w.Close(); err != nil { |  | ||||||
| 				conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, ""), time.Time{}) |  | ||||||
| 				return |  | ||||||
| 			} |  | ||||||
| 	}() | 	}() | ||||||
|  | 
 | ||||||
|  | 	s := bufio.NewScanner(stdout) | ||||||
|  | 	for s.Scan() { | ||||||
|  | 		conn.SetWriteDeadline(time.Now().Add(writeWait)) | ||||||
|  | 		if err := conn.WriteMessage(websocket.TextMessage, s.Bytes()); err != nil { | ||||||
|  | 			break | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if s.Err() != nil { | ||||||
|  | 		conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, s.Err().Error()), time.Time{}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // ticker is start by the reader. Basically it is the method that simulates the websocket | // pinger simulates the websocket to keep it alive with ping messages. | ||||||
| // between the server and client to keep it alive with ping messages. | func pinger(conn *websocket.Conn, done chan struct{}) { | ||||||
| func ticker(conn *websocket.Conn, c chan bool) { |  | ||||||
| 	ticker := time.NewTicker(pingPeriod) | 	ticker := time.NewTicker(pingPeriod) | ||||||
| 	defer ticker.Stop() | 	defer ticker.Stop() | ||||||
| 
 | 
 | ||||||
| 	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) | 			if err := conn.WriteControl(websocket.PingMessage, []byte{}, time.Now().Add(writeWait)); err != nil { | ||||||
| 		case <-c: | 				conn.WriteControl(websocket.CloseMessage, websocket.FormatCloseMessage(websocket.CloseGoingAway, err.Error()), time.Time{}) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 		case <-done: | ||||||
| 			return // clean up this routine. | 			return // clean up this routine. | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user