mirror of
https://github.com/sorenisanerd/gotty.git
synced 2024-11-09 23:34:26 +00:00
270 lines
7.6 KiB
Go
270 lines
7.6 KiB
Go
package server
|
|
|
|
import (
|
|
"context"
|
|
"crypto/tls"
|
|
"crypto/x509"
|
|
"html/template"
|
|
"io/fs"
|
|
"io/ioutil"
|
|
"log"
|
|
"net"
|
|
"net/http"
|
|
"regexp"
|
|
"strings"
|
|
noesctmpl "text/template"
|
|
"time"
|
|
|
|
"github.com/NYTimes/gziphandler"
|
|
"github.com/gorilla/websocket"
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/sorenisanerd/gotty/bindata"
|
|
"github.com/sorenisanerd/gotty/pkg/homedir"
|
|
"github.com/sorenisanerd/gotty/pkg/randomstring"
|
|
"github.com/sorenisanerd/gotty/webtty"
|
|
)
|
|
|
|
// Server provides a webtty HTTP endpoint.
|
|
type Server struct {
|
|
factory Factory
|
|
options *Options
|
|
|
|
upgrader *websocket.Upgrader
|
|
indexTemplate *template.Template
|
|
titleTemplate *noesctmpl.Template
|
|
manifestTemplate *template.Template
|
|
}
|
|
|
|
// New creates a new instance of Server.
|
|
// Server will use the New() of the factory provided to handle each request.
|
|
func New(factory Factory, options *Options) (*Server, error) {
|
|
indexData, err := bindata.Fs.ReadFile("static/index.html")
|
|
if err != nil {
|
|
panic("index not found") // must be in bindata
|
|
}
|
|
if options.IndexFile != "" {
|
|
path := homedir.Expand(options.IndexFile)
|
|
indexData, err = ioutil.ReadFile(path)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to read custom index file at `%s`", path)
|
|
}
|
|
}
|
|
indexTemplate, err := template.New("index").Parse(string(indexData))
|
|
if err != nil {
|
|
panic("index template parse failed") // must be valid
|
|
}
|
|
|
|
manifestData, err := bindata.Fs.ReadFile("static/manifest.json")
|
|
if err != nil {
|
|
panic("manifest not found") // must be in bindata
|
|
}
|
|
manifestTemplate, err := template.New("manifest").Parse(string(manifestData))
|
|
if err != nil {
|
|
panic("manifest template parse failed") // must be valid
|
|
}
|
|
|
|
titleTemplate, err := noesctmpl.New("title").Parse(options.TitleFormat)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to parse window title format `%s`", options.TitleFormat)
|
|
}
|
|
|
|
var originChekcer func(r *http.Request) bool
|
|
if options.WSOrigin != "" {
|
|
matcher, err := regexp.Compile(options.WSOrigin)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to compile regular expression of Websocket Origin: %s", options.WSOrigin)
|
|
}
|
|
originChekcer = func(r *http.Request) bool {
|
|
return matcher.MatchString(r.Header.Get("Origin"))
|
|
}
|
|
}
|
|
|
|
return &Server{
|
|
factory: factory,
|
|
options: options,
|
|
|
|
upgrader: &websocket.Upgrader{
|
|
ReadBufferSize: 1024,
|
|
WriteBufferSize: 1024,
|
|
Subprotocols: webtty.Protocols,
|
|
CheckOrigin: originChekcer,
|
|
},
|
|
indexTemplate: indexTemplate,
|
|
titleTemplate: titleTemplate,
|
|
manifestTemplate: manifestTemplate,
|
|
}, nil
|
|
}
|
|
|
|
// Run starts the main process of the Server.
|
|
// The cancelation of ctx will shutdown the server immediately with aborting
|
|
// existing connections. Use WithGracefullContext() to support gracefull shutdown.
|
|
func (server *Server) Run(ctx context.Context, options ...RunOption) error {
|
|
cctx, cancel := context.WithCancel(ctx)
|
|
opts := &RunOptions{gracefullCtx: context.Background()}
|
|
for _, opt := range options {
|
|
opt(opts)
|
|
}
|
|
|
|
counter := newCounter(time.Duration(server.options.Timeout) * time.Second)
|
|
|
|
path := server.options.Path
|
|
if server.options.EnableRandomUrl {
|
|
path = "/" + randomstring.Generate(server.options.RandomUrlLength) + "/"
|
|
}
|
|
if !strings.HasPrefix(path, "/") {
|
|
path = "/" + path
|
|
}
|
|
if !strings.HasSuffix(path, "/") {
|
|
path = path + "/"
|
|
}
|
|
handlers := server.setupHandlers(cctx, cancel, path, counter)
|
|
srv, err := server.setupHTTPServer(handlers)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "failed to setup an HTTP server")
|
|
}
|
|
|
|
if server.options.PermitWrite {
|
|
log.Printf("Permitting clients to write input to the PTY.")
|
|
}
|
|
if server.options.Once {
|
|
log.Printf("Once option is provided, accepting only one client")
|
|
}
|
|
|
|
if server.options.Port == "0" {
|
|
log.Printf("Port number configured to `0`, choosing a random port")
|
|
}
|
|
hostPort := net.JoinHostPort(server.options.Address, server.options.Port)
|
|
listener, err := net.Listen("tcp", hostPort)
|
|
if err != nil {
|
|
return errors.Wrapf(err, "failed to listen at `%s`", hostPort)
|
|
}
|
|
|
|
scheme := "http"
|
|
if server.options.EnableTLS {
|
|
scheme = "https"
|
|
}
|
|
host, port, _ := net.SplitHostPort(listener.Addr().String())
|
|
log.Printf("HTTP server is listening at: %s", scheme+"://"+net.JoinHostPort(host, port)+path)
|
|
if server.options.Address == "0.0.0.0" {
|
|
for _, address := range listAddresses() {
|
|
log.Printf("Alternative URL: %s", scheme+"://"+net.JoinHostPort(address, port)+path)
|
|
}
|
|
}
|
|
|
|
srvErr := make(chan error, 1)
|
|
go func() {
|
|
if server.options.EnableTLS {
|
|
crtFile := homedir.Expand(server.options.TLSCrtFile)
|
|
keyFile := homedir.Expand(server.options.TLSKeyFile)
|
|
log.Printf("TLS crt file: " + crtFile)
|
|
log.Printf("TLS key file: " + keyFile)
|
|
|
|
err = srv.ServeTLS(listener, crtFile, keyFile)
|
|
} else {
|
|
err = srv.Serve(listener)
|
|
}
|
|
if err != nil {
|
|
srvErr <- err
|
|
}
|
|
}()
|
|
|
|
go func() {
|
|
select {
|
|
case <-opts.gracefullCtx.Done():
|
|
srv.Shutdown(context.Background())
|
|
case <-cctx.Done():
|
|
}
|
|
}()
|
|
|
|
select {
|
|
case err = <-srvErr:
|
|
if err == http.ErrServerClosed { // by gracefull ctx
|
|
err = nil
|
|
} else {
|
|
cancel()
|
|
}
|
|
case <-cctx.Done():
|
|
srv.Close()
|
|
err = cctx.Err()
|
|
}
|
|
|
|
conn := counter.count()
|
|
if conn > 0 {
|
|
log.Printf("Waiting for %d connections to be closed", conn)
|
|
}
|
|
counter.wait()
|
|
|
|
return err
|
|
}
|
|
|
|
func (server *Server) setupHandlers(ctx context.Context, cancel context.CancelFunc, pathPrefix string, counter *counter) http.Handler {
|
|
fs, err := fs.Sub(bindata.Fs, "static")
|
|
if err != nil {
|
|
log.Fatalf("failed to open static/ subdirectory of embedded filesystem: %v", err)
|
|
}
|
|
staticFileHandler := http.FileServer(http.FS(fs))
|
|
|
|
var siteMux = http.NewServeMux()
|
|
siteMux.HandleFunc(pathPrefix, server.handleIndex)
|
|
siteMux.Handle(pathPrefix+"js/", http.StripPrefix(pathPrefix, staticFileHandler))
|
|
siteMux.Handle(pathPrefix+"favicon.ico", http.StripPrefix(pathPrefix, staticFileHandler))
|
|
siteMux.Handle(pathPrefix+"icon.svg", http.StripPrefix(pathPrefix, staticFileHandler))
|
|
siteMux.Handle(pathPrefix+"css/", http.StripPrefix(pathPrefix, staticFileHandler))
|
|
siteMux.Handle(pathPrefix+"icon_192.png", http.StripPrefix(pathPrefix, staticFileHandler))
|
|
|
|
siteMux.HandleFunc(pathPrefix+"manifest.json", server.handleManifest)
|
|
siteMux.HandleFunc(pathPrefix+"auth_token.js", server.handleAuthToken)
|
|
siteMux.HandleFunc(pathPrefix+"config.js", server.handleConfig)
|
|
|
|
siteHandler := http.Handler(siteMux)
|
|
|
|
if server.options.EnableBasicAuth {
|
|
log.Printf("Using Basic Authentication")
|
|
siteHandler = server.wrapBasicAuth(siteHandler, server.options.Credential)
|
|
}
|
|
|
|
withGz := gziphandler.GzipHandler(server.wrapHeaders(siteHandler))
|
|
siteHandler = server.wrapLogger(withGz)
|
|
|
|
wsMux := http.NewServeMux()
|
|
wsMux.Handle("/", siteHandler)
|
|
wsMux.HandleFunc(pathPrefix+"ws", server.generateHandleWS(ctx, cancel, counter))
|
|
siteHandler = http.Handler(wsMux)
|
|
|
|
return siteHandler
|
|
}
|
|
|
|
func (server *Server) setupHTTPServer(handler http.Handler) (*http.Server, error) {
|
|
srv := &http.Server{
|
|
Handler: handler,
|
|
}
|
|
|
|
if server.options.EnableTLSClientAuth {
|
|
tlsConfig, err := server.tlsConfig()
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to setup TLS configuration")
|
|
}
|
|
srv.TLSConfig = tlsConfig
|
|
}
|
|
|
|
return srv, nil
|
|
}
|
|
|
|
func (server *Server) tlsConfig() (*tls.Config, error) {
|
|
caFile := homedir.Expand(server.options.TLSCACrtFile)
|
|
caCert, err := ioutil.ReadFile(caFile)
|
|
if err != nil {
|
|
return nil, errors.New("could not open CA crt file " + caFile)
|
|
}
|
|
caCertPool := x509.NewCertPool()
|
|
if !caCertPool.AppendCertsFromPEM(caCert) {
|
|
return nil, errors.New("could not parse CA crt file data in " + caFile)
|
|
}
|
|
tlsConfig := &tls.Config{
|
|
ClientCAs: caCertPool,
|
|
ClientAuth: tls.RequireAndVerifyClientCert,
|
|
}
|
|
return tlsConfig, nil
|
|
}
|