summaryrefslogtreecommitdiff
path: root/internal/server
diff options
context:
space:
mode:
authorPaul Buetow <paul@buetow.org>2026-03-03 10:10:19 +0200
committerPaul Buetow <paul@buetow.org>2026-03-03 10:10:19 +0200
commit7d3685a5ed4bfac85673793f8ae6d9c5a6cff962 (patch)
tree27bc845ef5758aa43662d0ce238436461d1893e7 /internal/server
parentf4898f746d03ff5dcf57d3967c594d98a9da7fe0 (diff)
feat(server): add AUTHKEY command handling
Diffstat (limited to 'internal/server')
-rw-r--r--internal/server/handlers/authkeycommand_test.go117
-rw-r--r--internal/server/handlers/serverhandler.go47
2 files changed, 159 insertions, 5 deletions
diff --git a/internal/server/handlers/authkeycommand_test.go b/internal/server/handlers/authkeycommand_test.go
new file mode 100644
index 0000000..bb9488b
--- /dev/null
+++ b/internal/server/handlers/authkeycommand_test.go
@@ -0,0 +1,117 @@
+package handlers
+
+import (
+ "context"
+ "crypto/ed25519"
+ "encoding/base64"
+ "testing"
+ "time"
+
+ "github.com/mimecast/dtail/internal"
+ "github.com/mimecast/dtail/internal/config"
+ "github.com/mimecast/dtail/internal/lcontext"
+ sshserver "github.com/mimecast/dtail/internal/ssh/server"
+ userserver "github.com/mimecast/dtail/internal/user/server"
+
+ gossh "golang.org/x/crypto/ssh"
+)
+
+func TestHandleAuthKeyCommandSuccess(t *testing.T) {
+ handler := newAuthKeyTestHandler("authkey-success-user", true)
+ key := handlerTestPublicKey(t, 31)
+ keyArg := base64.StdEncoding.EncodeToString(key.Marshal())
+
+ commandFinished := false
+ handler.handleAuthKeyCommand(context.Background(), lcontext.LContext{}, 2,
+ []string{"AUTHKEY", keyArg}, func() {
+ commandFinished = true
+ })
+
+ if !commandFinished {
+ t.Fatalf("Expected commandFinished callback to be called")
+ }
+ if message := readServerMessage(t, handler.serverMessages); message != "AUTHKEY OK\n" {
+ t.Fatalf("Unexpected response: %q", message)
+ }
+ if !sshserver.ServerAuthKeyStore().Has(handler.user.Name, key) {
+ t.Fatalf("Expected key to be stored for user")
+ }
+
+ sshserver.ServerAuthKeyStore().Remove(handler.user.Name, key)
+}
+
+func TestHandleAuthKeyCommandFeatureDisabled(t *testing.T) {
+ handler := newAuthKeyTestHandler("authkey-disabled-user", false)
+ key := handlerTestPublicKey(t, 32)
+ keyArg := base64.StdEncoding.EncodeToString(key.Marshal())
+
+ handler.handleAuthKeyCommand(context.Background(), lcontext.LContext{}, 2,
+ []string{"AUTHKEY", keyArg}, func() {})
+
+ if message := readServerMessage(t, handler.serverMessages); message != "AUTHKEY ERR feature disabled\n" {
+ t.Fatalf("Unexpected response: %q", message)
+ }
+ if sshserver.ServerAuthKeyStore().Has(handler.user.Name, key) {
+ t.Fatalf("Expected no key to be stored while feature is disabled")
+ }
+}
+
+func TestHandleAuthKeyCommandInvalidPayload(t *testing.T) {
+ handler := newAuthKeyTestHandler("authkey-invalid-user", true)
+
+ handler.handleAuthKeyCommand(context.Background(), lcontext.LContext{}, 2,
+ []string{"AUTHKEY", "not-base64"}, func() {})
+
+ if message := readServerMessage(t, handler.serverMessages); message != "AUTHKEY ERR invalid base64\n" {
+ t.Fatalf("Unexpected response for invalid base64: %q", message)
+ }
+
+ validButNonSSH := base64.StdEncoding.EncodeToString([]byte("not-an-ssh-key"))
+ handler.handleAuthKeyCommand(context.Background(), lcontext.LContext{}, 2,
+ []string{"AUTHKEY", validButNonSSH}, func() {})
+ if message := readServerMessage(t, handler.serverMessages); message != "AUTHKEY ERR invalid public key\n" {
+ t.Fatalf("Unexpected response for invalid key bytes: %q", message)
+ }
+}
+
+func newAuthKeyTestHandler(userName string, authKeyEnabled bool) *ServerHandler {
+ return &ServerHandler{
+ baseHandler: baseHandler{
+ done: internal.NewDone(),
+ serverMessages: make(chan string, 4),
+ user: &userserver.User{Name: userName},
+ },
+ serverCfg: &config.ServerConfig{
+ AuthKeyEnabled: authKeyEnabled,
+ },
+ }
+}
+
+func handlerTestPublicKey(t *testing.T, seedByte byte) gossh.PublicKey {
+ t.Helper()
+
+ seed := make([]byte, ed25519.SeedSize)
+ for i := range seed {
+ seed[i] = seedByte
+ }
+
+ privateKey := ed25519.NewKeyFromSeed(seed)
+ publicKey, err := gossh.NewPublicKey(privateKey.Public())
+ if err != nil {
+ t.Fatalf("Unable to build ssh public key: %s", err.Error())
+ }
+
+ return publicKey
+}
+
+func readServerMessage(t *testing.T, messages <-chan string) string {
+ t.Helper()
+
+ select {
+ case message := <-messages:
+ return message
+ case <-time.After(time.Second):
+ t.Fatalf("Timed out waiting for server message")
+ return ""
+ }
+}
diff --git a/internal/server/handlers/serverhandler.go b/internal/server/handlers/serverhandler.go
index f9aa499..53ab4e3 100644
--- a/internal/server/handlers/serverhandler.go
+++ b/internal/server/handlers/serverhandler.go
@@ -2,6 +2,7 @@ package handlers
import (
"context"
+ "encoding/base64"
"strings"
"sync/atomic"
@@ -11,7 +12,10 @@ import (
"github.com/mimecast/dtail/internal/io/line"
"github.com/mimecast/dtail/internal/lcontext"
"github.com/mimecast/dtail/internal/omode"
+ sshserver "github.com/mimecast/dtail/internal/ssh/server"
user "github.com/mimecast/dtail/internal/user/server"
+
+ gossh "golang.org/x/crypto/ssh"
)
// ServerHandler implements the Reader and Writer interfaces to handle
@@ -100,11 +104,13 @@ func (h *ServerHandler) handleUserCommand(ctx context.Context, ltx lcontext.LCon
func (h *ServerHandler) newCommandRegistry() map[string]commandHandler {
return map[string]commandHandler{
- "grep": h.makeReadCommandHandler(omode.GrepClient, 1),
- "cat": h.makeReadCommandHandler(omode.CatClient, 1),
- "tail": h.makeReadCommandHandler(omode.TailClient, 10),
- "map": h.handleMapCommand,
- ".ack": h.handleAckUserCommand,
+ "grep": h.makeReadCommandHandler(omode.GrepClient, 1),
+ "cat": h.makeReadCommandHandler(omode.CatClient, 1),
+ "tail": h.makeReadCommandHandler(omode.TailClient, 10),
+ "map": h.handleMapCommand,
+ ".ack": h.handleAckUserCommand,
+ "AUTHKEY": h.handleAuthKeyCommand,
+ "authkey": h.handleAuthKeyCommand,
}
}
@@ -139,3 +145,34 @@ func (h *ServerHandler) handleAckUserCommand(_ context.Context, _ lcontext.LCont
h.handleAckCommand(argc, args)
commandFinished()
}
+
+func (h *ServerHandler) handleAuthKeyCommand(_ context.Context, _ lcontext.LContext,
+ argc int, args []string, commandFinished func()) {
+
+ defer commandFinished()
+
+ if !h.serverCfg.AuthKeyEnabled {
+ h.sendln(h.serverMessages, "AUTHKEY ERR feature disabled")
+ return
+ }
+
+ if argc < 2 || strings.TrimSpace(args[1]) == "" {
+ h.sendln(h.serverMessages, "AUTHKEY ERR missing public key")
+ return
+ }
+
+ decodedPubKey, err := base64.StdEncoding.DecodeString(args[1])
+ if err != nil {
+ h.sendln(h.serverMessages, "AUTHKEY ERR invalid base64")
+ return
+ }
+
+ pubKey, err := gossh.ParsePublicKey(decodedPubKey)
+ if err != nil {
+ h.sendln(h.serverMessages, "AUTHKEY ERR invalid public key")
+ return
+ }
+
+ sshserver.ServerAuthKeyStore().Add(h.user.Name, pubKey)
+ h.sendln(h.serverMessages, "AUTHKEY OK")
+}