2025-02-22 18:39:14 -08:00
|
|
|
// Copyright (C) 2025 klefki contributors
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Affero General Public License as
|
|
|
|
// published by the Free Software Foundation, either version 3 of the
|
|
|
|
// License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0
|
|
|
|
|
|
|
|
// Package server contains the gRPC server logic for the service.
|
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
2025-03-01 10:09:26 -08:00
|
|
|
"bytes"
|
2025-02-22 18:39:14 -08:00
|
|
|
"context"
|
|
|
|
"fmt"
|
2025-03-01 10:09:26 -08:00
|
|
|
"io"
|
2025-02-22 18:39:14 -08:00
|
|
|
"net"
|
2025-03-01 10:09:26 -08:00
|
|
|
"strings"
|
2025-02-22 18:39:14 -08:00
|
|
|
|
2025-03-01 10:09:26 -08:00
|
|
|
"git.rgst.io/homelab/klefki/internal/db"
|
|
|
|
"git.rgst.io/homelab/klefki/internal/db/ent"
|
|
|
|
"git.rgst.io/homelab/klefki/internal/machines"
|
2025-02-22 18:39:14 -08:00
|
|
|
pbgrpcv1 "git.rgst.io/homelab/klefki/internal/server/grpc/generated/go/rgst/klefki/v1"
|
2025-03-01 10:09:26 -08:00
|
|
|
"git.rgst.io/homelab/sigtool/v3/sign"
|
2025-02-22 18:39:14 -08:00
|
|
|
"google.golang.org/grpc"
|
|
|
|
"google.golang.org/grpc/reflection"
|
|
|
|
)
|
|
|
|
|
2025-03-01 10:09:26 -08:00
|
|
|
// nopWriteCloser is a no-op [io.WriteCloser]
|
|
|
|
type nopWriteCloser struct {
|
|
|
|
io.Writer
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close implements [io.Closer]
|
|
|
|
func (nwc nopWriteCloser) Close() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// newNopWriteCloser creates a new nopWriteCloser
|
|
|
|
func newNopWriteCloser(w io.Writer) *nopWriteCloser {
|
|
|
|
return &nopWriteCloser{w}
|
|
|
|
}
|
|
|
|
|
2025-02-22 18:39:14 -08:00
|
|
|
// Server is a Klefki gRPC server
|
|
|
|
type Server struct {
|
|
|
|
gs *grpc.Server
|
2025-03-01 10:09:26 -08:00
|
|
|
db *ent.Client
|
2025-02-22 18:39:14 -08:00
|
|
|
pbgrpcv1.UnimplementedKlefkiServiceServer
|
|
|
|
}
|
|
|
|
|
|
|
|
// Run starts the server
|
2025-03-01 10:09:26 -08:00
|
|
|
func (s *Server) Run(ctx context.Context) error {
|
|
|
|
var err error
|
|
|
|
s.db, err = db.New(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to open DB: %w", err)
|
|
|
|
}
|
|
|
|
|
2025-02-22 18:39:14 -08:00
|
|
|
s.gs = grpc.NewServer()
|
|
|
|
pbgrpcv1.RegisterKlefkiServiceServer(s.gs, s)
|
|
|
|
reflection.Register(s.gs)
|
|
|
|
|
|
|
|
lis, err := net.Listen("tcp", ":5300") //nolint:gosec // Why: This is fine.
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to create listener: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Printf("starting gRPC server on %s\n", lis.Addr())
|
|
|
|
return s.gs.Serve(lis)
|
|
|
|
}
|
|
|
|
|
2025-02-22 22:11:11 -08:00
|
|
|
// GetKey implements the GetKey request
|
2025-03-01 10:09:26 -08:00
|
|
|
func (s *Server) GetKey(ctx context.Context, req *pbgrpcv1.GetKeyRequest) (*pbgrpcv1.GetKeyResponse, error) {
|
2025-02-22 22:11:11 -08:00
|
|
|
resp := &pbgrpcv1.GetKeyResponse{}
|
2025-03-01 10:09:26 -08:00
|
|
|
|
|
|
|
nonce := req.GetNonce()
|
|
|
|
sig := req.GetSignature()
|
|
|
|
|
|
|
|
machine, err := s.db.Machine.Get(ctx, req.GetMachineId())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := machines.Verify(machine.PublicKey, sig, nonce); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
spubk, err := sign.PublicKeyFromBytes(machine.PublicKey)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to create pub key for encryption: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
enc, err := sign.NewEncryptor(nil, 1024)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to create encryptor instance: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := enc.AddRecipient(spubk); err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to add instance public key to encryptor: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(jaredallard): Wait for input here.
|
|
|
|
var buf bytes.Buffer
|
|
|
|
if err := enc.Encrypt(strings.NewReader("hello world"), newNopWriteCloser(&buf)); err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to encrypt passphrase: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp.SetKey(buf.Bytes())
|
2025-02-22 22:11:11 -08:00
|
|
|
return resp, nil
|
|
|
|
}
|
|
|
|
|
2025-02-22 18:39:14 -08:00
|
|
|
// Close closes the server
|
|
|
|
func (s *Server) Close(_ context.Context) error {
|
|
|
|
if s.gs == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Println("shutting down server")
|
|
|
|
s.gs.GracefulStop()
|
2025-03-01 10:09:26 -08:00
|
|
|
return s.db.Close()
|
2025-02-22 18:39:14 -08:00
|
|
|
}
|