zrok/cmd/zrok/testEndpoint.go
Cam Otts 8ea8ec93ed
Updated ziti sdk package to better handle proxy requests.
Extended test endpoint to be able to handle ziti network.
Added a test websocket client to hit the test endpoint's /echo path
2023-03-06 09:53:28 -06:00

221 lines
5.4 KiB
Go

package main
import (
"context"
"encoding/json"
"flag"
"fmt"
"html/template"
"io"
"net"
"net/http"
"os"
"time"
"github.com/openziti/sdk-golang/ziti"
"github.com/openziti/sdk-golang/ziti/config"
"github.com/openziti/zrok/cmd/zrok/endpointUi"
"github.com/openziti/zrok/tui"
"github.com/pkg/errors"
"github.com/sirupsen/logrus"
"github.com/spf13/cobra"
"golang.org/x/time/rate"
"nhooyr.io/websocket"
)
func init() {
testCmd.AddCommand(newTestEndpointCommand().cmd)
}
type testEndpointCommand struct {
address string
port uint16
t *template.Template
cmd *cobra.Command
enableZiti bool
serviceName string
identityJsonFile string
}
func newTestEndpointCommand() *testEndpointCommand {
cmd := &cobra.Command{
Use: "endpoint",
Short: "Start a simple HTTP endpoint",
Args: cobra.ExactArgs(0),
}
command := &testEndpointCommand{cmd: cmd}
var err error
if command.t, err = template.ParseFS(endpointUi.FS, "index.gohtml"); err != nil {
if !panicInstead {
tui.Error("unable to parse index template", err)
}
panic(err)
}
cmd.Flags().StringVarP(&command.address, "address", "a", "127.0.0.1", "The address for the HTTP listener")
cmd.Flags().Uint16VarP(&command.port, "port", "P", 9090, "The port for the HTTP listener")
cmd.Flags().BoolVar(&command.enableZiti, "ziti", false, "Enable the usage of a ziti network")
cmd.Flags().StringVar(&command.identityJsonFile, "ziti-identity", "", "Path to Ziti Identity json file")
cmd.Flags().StringVar(&command.serviceName, "ziti-name", "", "Name of the Ziti Service")
cmd.Run = command.run
return command
}
func (cmd *testEndpointCommand) run(_ *cobra.Command, _ []string) {
var listener net.Listener
var err error
if cmd.enableZiti {
identityJsonBytes, err := os.ReadFile(cmd.identityJsonFile)
if err != nil {
fmt.Fprintf(os.Stderr, "Error: failed to read identity config JSON from file %s: %s\n", cmd.identityJsonFile, err)
os.Exit(1)
}
if len(identityJsonBytes) == 0 {
fmt.Fprintf(os.Stderr, "Error: When running a ziti enabled service must have ziti identity provided\n\n")
flag.Usage()
os.Exit(1)
}
config := config.Config{}
err = json.Unmarshal(identityJsonBytes, &config)
if err != nil {
fmt.Fprintf(os.Stderr, "failed to load ziti configuration JSON: %v", err)
os.Exit(1)
}
zitiContext := ziti.NewContextWithConfig(&config)
if err := zitiContext.Authenticate(); err != nil {
fmt.Fprintf(os.Stderr, "Error: Unable to authenticate ziti: %v\n\n", err)
os.Exit(1)
}
listener, err = zitiContext.Listen(cmd.serviceName)
if err != nil {
fmt.Fprintf(os.Stderr, "Error: Unable to listen on ziti network: %v\n\n", err)
os.Exit(1)
}
} else {
listener, err = net.Listen("tcp", fmt.Sprintf("%v:%d", cmd.address, cmd.port))
if err != nil {
fmt.Fprintf(os.Stderr, "Error: Unable to listen on %s: %v\n\n", fmt.Sprintf("%v:%d", cmd.address, cmd.port), err)
os.Exit(1)
}
}
server := &http.Server{}
http.HandleFunc("/", cmd.serveIndex)
http.HandleFunc("/echo", cmd.websocketEcho)
if err := server.Serve(listener); err != nil {
if !panicInstead {
tui.Error("unable to start http listener", err)
}
panic(err)
}
}
func (cmd *testEndpointCommand) serveIndex(w http.ResponseWriter, r *http.Request) {
logrus.Infof("%v {%v} | %v -> /index.gohtml", r.RemoteAddr, r.Host, r.RequestURI)
if err := cmd.t.Execute(w, newEndpointData(r)); err != nil {
logrus.Error(err)
}
}
func (cmd *testEndpointCommand) websocketEcho(w http.ResponseWriter, r *http.Request) {
c, err := websocket.Accept(w, r, nil)
if err != nil {
logrus.Error(err)
return
}
defer func() { _ = c.Close(websocket.StatusInternalError, "connection terminated") }()
l := rate.NewLimiter(rate.Every(time.Millisecond*100), 10)
for {
err = cmd.doEcho(r.Context(), c, l)
if websocket.CloseStatus(err) == websocket.StatusNormalClosure {
return
}
if err != nil {
logrus.Errorf("failed to echo for '%v': %v", r.RemoteAddr, err)
return
}
}
}
func (cmd *testEndpointCommand) doEcho(ctx context.Context, c *websocket.Conn, l *rate.Limiter) error {
ctx, cancel := context.WithTimeout(ctx, time.Second*10)
defer cancel()
err := l.Wait(ctx)
if err != nil {
return err
}
typ, r, err := c.Reader(ctx)
if err != nil {
return err
}
w, err := c.Writer(ctx, typ)
if err != nil {
return err
}
_, err = w.Write([]byte("i received: "))
_, err = io.Copy(w, r)
if err != nil {
return errors.Wrap(err, "failed to copy")
}
err = w.Close()
return err
}
type endpointData struct {
RequestedPath string
Now time.Time
RemoteAddr string
Host string
HostDetail string
Ips string
HostHeader string
Headers map[string][]string
}
func newEndpointData(r *http.Request) *endpointData {
ed := &endpointData{
RequestedPath: r.RequestURI,
Now: time.Now(),
HostHeader: r.Host,
Headers: r.Header,
RemoteAddr: r.RemoteAddr,
}
ed.getHostInfo()
ed.getIps()
return ed
}
func (ed *endpointData) getHostInfo() {
host, hostDetail, err := getHost()
if err != nil {
logrus.Errorf("error getting host detail: %v", err)
}
ed.Host = host
ed.HostDetail = hostDetail
}
func (ed *endpointData) getIps() {
addrs, err := net.InterfaceAddrs()
if err == nil {
for _, address := range addrs {
if ipnet, ok := address.(*net.IPNet); ok && !ipnet.IP.IsLoopback() {
if len(ed.Ips) != 0 {
ed.Ips += ", "
}
ed.Ips += ipnet.IP.String()
}
}
}
}