about summary refs log tree commit diff
path: root/cmd/k9p/main.go
diff options
context:
space:
mode:
Diffstat (limited to 'cmd/k9p/main.go')
-rw-r--r--cmd/k9p/main.go102
1 files changed, 102 insertions, 0 deletions
diff --git a/cmd/k9p/main.go b/cmd/k9p/main.go
new file mode 100644
index 0000000..dd0ed80
--- /dev/null
+++ b/cmd/k9p/main.go
@@ -0,0 +1,102 @@
+package main
+
+import (
+	"context"
+	"flag"
+	"net"
+	"os"
+
+	p9p "github.com/docker/go-p9p"
+	"github.com/oklog/run"
+	"github.com/rs/zerolog"
+	"go.terinstock.com/k9p/pkg/k9p"
+	"go.terinstock.com/k9p/pkg/k9p/logger"
+	"k8s.io/client-go/kubernetes"
+	_ "k8s.io/client-go/plugin/pkg/client/auth"
+	"k8s.io/client-go/tools/clientcmd"
+	"k8s.io/klog"
+)
+
+func main() {
+	fs := flag.NewFlagSet("k9p", flag.ExitOnError)
+	klog.InitFlags(fs)
+
+	fs.Set("logtostderr", "false")
+	fs.Set("alsologtostderr", "false")
+
+	var (
+		prettyLog  = fs.Bool("pretty-log", false, "output human-friendly logs")
+		master     = fs.String("master", "", "The address of the Kubernetes API server (overrides any value in kubeconfig).")
+		kubeconfig = fs.String("kubeconfig", "", "Path to kubeconfig file with authorization and master location information.")
+		bind9p     = fs.String("bind-9p", ":564", "The address the 9P server should bind and listen on")
+	)
+	fs.Parse(os.Args[1:])
+
+	ctx := context.Background()
+
+	var log zerolog.Logger
+	if *prettyLog {
+		log = zerolog.New(zerolog.ConsoleWriter{Out: os.Stderr}).With().Timestamp().Logger()
+	} else {
+		log = zerolog.New(os.Stderr)
+	}
+
+	client, err := createClient(*master, *kubeconfig)
+	if err != nil {
+		log.Fatal().Err(err).Send()
+	}
+
+	klog.SetOutput(log.With().Str("component", "klog").Logger())
+
+	var g run.Group
+	{
+		ln, err := net.Listen("tcp", *bind9p)
+		if err != nil {
+			log.Fatal().Err(err).Msg("error listening")
+		}
+
+		g.Add(func() error {
+			for {
+				c, err := ln.Accept()
+				if err != nil {
+					log.Warn().Err(err).Msg("error accepting")
+					continue
+				}
+
+				go func(conn net.Conn) {
+					defer conn.Close()
+
+					ctx, cancel := context.WithCancel(context.WithValue(ctx, "conn", conn))
+					defer cancel()
+
+					log.Info().Str("remote", conn.RemoteAddr().String()).Msg("connected")
+
+					var session p9p.Session
+					{
+						ksession := k9p.New(ctx, client)
+						ksession.WaitForCacheSync(ctx.Done())
+						session = logger.New(
+							log.With().Str("component", "9p").Logger(),
+							ksession,
+						)
+					}
+					if err := p9p.ServeConn(ctx, conn, p9p.Dispatch(session)); err != nil {
+						log.Warn().Err(err).Msg("ServeConn")
+					}
+				}(c)
+			}
+		}, func(error) {
+			ln.Close()
+		})
+	}
+
+	log.Info().Err(g.Run())
+}
+
+func createClient(master string, kubeconfig string) (kubernetes.Interface, error) {
+	config, err := clientcmd.BuildConfigFromFlags(master, kubeconfig)
+	if err != nil {
+		return nil, err
+	}
+	return kubernetes.NewForConfig(config)
+}