caddy/middleware/websockets/websockets.go

135 lines
3.1 KiB
Go
Raw Normal View History

2015-03-03 19:49:45 +03:00
// Package websockets implements a WebSocket server by executing
// a command and piping its input and output through the WebSocket
// connection.
package websockets
import (
"net/http"
"github.com/mholt/caddy/middleware"
"golang.org/x/net/websocket"
)
2015-03-04 03:36:18 +03:00
type (
// WebSockets is a type that holds configuration for the
// websocket middleware generally, like a list of all the
// websocket endpoints.
WebSockets struct {
2015-03-20 08:52:56 +03:00
// Next is the next HTTP handler in the chain for when the path doesn't match
Next middleware.Handler
2015-03-20 08:52:56 +03:00
// Sockets holds all the web socket endpoint configurations
2015-03-04 03:36:18 +03:00
Sockets []WSConfig
}
// WSConfig holds the configuration for a single websocket
2015-03-20 08:52:56 +03:00
// endpoint which may serve multiple websocket connections.
2015-03-04 03:36:18 +03:00
WSConfig struct {
Path string
Command string
Arguments []string
2015-03-20 08:52:56 +03:00
Respawn bool // TODO: Not used, but parser supports it until we decide on it
2015-03-04 03:36:18 +03:00
}
)
2015-03-03 19:49:45 +03:00
2015-03-04 03:36:18 +03:00
// ServeHTTP converts the HTTP request to a WebSocket connection and serves it up.
func (ws WebSockets) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) {
2015-03-04 03:36:18 +03:00
for _, sockconfig := range ws.Sockets {
if middleware.Path(r.URL.Path).Matches(sockconfig.Path) {
socket := WebSocket{
WSConfig: sockconfig,
Request: r,
}
2015-03-03 19:49:45 +03:00
websocket.Handler(socket.Handle).ServeHTTP(w, r)
return 0, nil
2015-03-03 19:49:45 +03:00
}
}
2015-03-20 08:52:56 +03:00
// Didn't match a websocket path, so pass-thru
return ws.Next.ServeHTTP(w, r)
2015-03-03 19:49:45 +03:00
}
// New constructs and configures a new websockets middleware instance.
func New(c middleware.Controller) (middleware.Middleware, error) {
2015-03-04 03:36:18 +03:00
var websocks []WSConfig
2015-03-20 08:52:56 +03:00
var respawn bool
optionalBlock := func() (hadBlock bool, err error) {
for c.NextBlock() {
hadBlock = true
if c.Val() == "respawn" {
respawn = true
} else {
return true, c.Err("Expected websocket configuration parameter in block")
}
}
return
}
2015-03-03 19:49:45 +03:00
for c.Next() {
var val, path, command string
2015-03-03 19:49:45 +03:00
// Path or command; not sure which yet
if !c.NextArg() {
return nil, c.ArgErr()
}
val = c.Val()
2015-03-20 08:52:56 +03:00
// Extra configuration may be in a block
hadBlock, err := optionalBlock()
if err != nil {
return nil, err
2015-03-03 19:49:45 +03:00
}
2015-03-20 08:52:56 +03:00
if !hadBlock {
// The next argument on this line will be the command or an open curly brace
if c.NextArg() {
path = val
command = c.Val()
} else {
path = "/"
command = val
}
2015-03-03 19:49:45 +03:00
2015-03-20 08:52:56 +03:00
// Okay, check again for optional block
hadBlock, err = optionalBlock()
if err != nil {
return nil, err
}
2015-03-03 19:49:45 +03:00
}
2015-03-20 08:52:56 +03:00
// Split command into the actual command and its arguments
2015-03-26 18:52:03 +03:00
cmd, args, err := middleware.SplitCommandAndArgs(command)
2015-03-20 08:52:56 +03:00
if err != nil {
return nil, err
2015-03-03 19:49:45 +03:00
}
2015-03-04 03:36:18 +03:00
websocks = append(websocks, WSConfig{
2015-03-03 19:49:45 +03:00
Path: path,
Command: cmd,
Arguments: args,
2015-04-12 02:24:47 +03:00
Respawn: respawn, // TODO: This isn't used currently
2015-03-03 19:49:45 +03:00
})
}
GatewayInterface = envGatewayInterface
ServerSoftware = envServerSoftware
return func(next middleware.Handler) middleware.Handler {
return WebSockets{Next: next, Sockets: websocks}
2015-03-03 19:49:45 +03:00
}, nil
}
var (
// See CGI spec, 4.1.4
GatewayInterface string
// See CGI spec, 4.1.17
ServerSoftware string
)
const (
envGatewayInterface = "caddy-CGI/1.1"
2015-03-20 08:52:56 +03:00
envServerSoftware = "caddy/?.?.?" // TODO
)