2021-12-07 22:31:32 +00:00
|
|
|
package secretsencrypt
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2022-02-28 17:14:32 +00:00
|
|
|
"strings"
|
2021-12-07 22:31:32 +00:00
|
|
|
"text/tabwriter"
|
2024-02-09 19:37:37 +00:00
|
|
|
"time"
|
2021-12-07 22:31:32 +00:00
|
|
|
|
|
|
|
"github.com/erikdubbelboer/gspt"
|
2022-03-02 23:47:27 +00:00
|
|
|
"github.com/k3s-io/k3s/pkg/cli/cmds"
|
|
|
|
"github.com/k3s-io/k3s/pkg/clientaccess"
|
|
|
|
"github.com/k3s-io/k3s/pkg/secretsencrypt"
|
|
|
|
"github.com/k3s-io/k3s/pkg/server"
|
|
|
|
"github.com/k3s-io/k3s/pkg/version"
|
2022-03-24 19:23:59 +00:00
|
|
|
"github.com/pkg/errors"
|
2021-12-07 22:31:32 +00:00
|
|
|
"github.com/urfave/cli"
|
|
|
|
"k8s.io/utils/pointer"
|
|
|
|
)
|
|
|
|
|
2023-08-02 22:17:41 +00:00
|
|
|
func commandPrep(cfg *cmds.Server) (*clientaccess.Info, error) {
|
2021-12-07 22:31:32 +00:00
|
|
|
// hide process arguments from ps output, since they may contain
|
|
|
|
// database credentials or other secrets.
|
2022-01-06 18:38:22 +00:00
|
|
|
gspt.SetProcTitle(os.Args[0] + " secrets-encrypt")
|
2021-12-07 22:31:32 +00:00
|
|
|
|
2022-02-28 17:14:32 +00:00
|
|
|
dataDir, err := server.ResolveDataDir(cfg.DataDir)
|
2021-12-07 22:31:32 +00:00
|
|
|
if err != nil {
|
2022-02-28 17:14:32 +00:00
|
|
|
return nil, err
|
2021-12-07 22:31:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if cfg.Token == "" {
|
2022-02-28 17:14:32 +00:00
|
|
|
fp := filepath.Join(dataDir, "token")
|
2022-10-08 00:36:57 +00:00
|
|
|
tokenByte, err := os.ReadFile(fp)
|
2021-12-07 22:31:32 +00:00
|
|
|
if err != nil {
|
2022-02-28 17:14:32 +00:00
|
|
|
return nil, err
|
2021-12-07 22:31:32 +00:00
|
|
|
}
|
2022-02-28 17:14:32 +00:00
|
|
|
cfg.Token = string(bytes.TrimRight(tokenByte, "\n"))
|
2021-12-07 22:31:32 +00:00
|
|
|
}
|
2022-12-08 23:59:21 +00:00
|
|
|
return clientaccess.ParseAndValidateToken(cmds.ServerConfig.ServerURL, cfg.Token, clientaccess.WithUser("server"))
|
2021-12-07 22:31:32 +00:00
|
|
|
}
|
|
|
|
|
2022-03-24 19:23:59 +00:00
|
|
|
func wrapServerError(err error) error {
|
|
|
|
return errors.Wrap(err, "see server log for details")
|
|
|
|
}
|
|
|
|
|
2021-12-07 22:31:32 +00:00
|
|
|
func Enable(app *cli.Context) error {
|
2023-08-02 22:17:41 +00:00
|
|
|
if err := cmds.InitLogging(); err != nil {
|
2021-12-07 22:31:32 +00:00
|
|
|
return err
|
|
|
|
}
|
2023-08-02 22:17:41 +00:00
|
|
|
info, err := commandPrep(&cmds.ServerConfig)
|
2021-12-07 22:31:32 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b, err := json.Marshal(server.EncryptionRequest{Enable: pointer.Bool(true)})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err = info.Put("/v1-"+version.Program+"/encrypt/config", b); err != nil {
|
2022-03-24 19:23:59 +00:00
|
|
|
return wrapServerError(err)
|
2021-12-07 22:31:32 +00:00
|
|
|
}
|
|
|
|
fmt.Println("secrets-encryption enabled")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func Disable(app *cli.Context) error {
|
|
|
|
|
|
|
|
if err := cmds.InitLogging(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-08-02 22:17:41 +00:00
|
|
|
info, err := commandPrep(&cmds.ServerConfig)
|
2021-12-07 22:31:32 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b, err := json.Marshal(server.EncryptionRequest{Enable: pointer.Bool(false)})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err = info.Put("/v1-"+version.Program+"/encrypt/config", b); err != nil {
|
2022-03-24 19:23:59 +00:00
|
|
|
return wrapServerError(err)
|
2021-12-07 22:31:32 +00:00
|
|
|
}
|
|
|
|
fmt.Println("secrets-encryption disabled")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func Status(app *cli.Context) error {
|
|
|
|
if err := cmds.InitLogging(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-08-02 22:17:41 +00:00
|
|
|
info, err := commandPrep(&cmds.ServerConfig)
|
2021-12-07 22:31:32 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
data, err := info.Get("/v1-" + version.Program + "/encrypt/status")
|
|
|
|
if err != nil {
|
2022-03-24 19:23:59 +00:00
|
|
|
return wrapServerError(err)
|
2021-12-07 22:31:32 +00:00
|
|
|
}
|
|
|
|
status := server.EncryptionState{}
|
|
|
|
if err := json.Unmarshal(data, &status); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-02-28 17:14:32 +00:00
|
|
|
if strings.ToLower(cmds.ServerConfig.EncryptOutput) == "json" {
|
|
|
|
json, err := json.MarshalIndent(status, "", "\t")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fmt.Println(string(json))
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-12-07 22:31:32 +00:00
|
|
|
if status.Enable == nil {
|
|
|
|
fmt.Println("Encryption Status: Disabled, no configuration file found")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var statusOutput string
|
|
|
|
if *status.Enable {
|
|
|
|
statusOutput += "Encryption Status: Enabled\n"
|
|
|
|
} else {
|
|
|
|
statusOutput += "Encryption Status: Disabled\n"
|
|
|
|
}
|
|
|
|
statusOutput += fmt.Sprintln("Current Rotation Stage:", status.Stage)
|
|
|
|
|
|
|
|
if status.HashMatch {
|
|
|
|
statusOutput += fmt.Sprintln("Server Encryption Hashes: All hashes match")
|
|
|
|
} else {
|
|
|
|
statusOutput += fmt.Sprintf("Server Encryption Hashes: %s\n", status.HashError)
|
|
|
|
}
|
|
|
|
|
|
|
|
var tabBuffer bytes.Buffer
|
|
|
|
w := tabwriter.NewWriter(&tabBuffer, 0, 0, 2, ' ', 0)
|
|
|
|
fmt.Fprintf(w, "\n")
|
|
|
|
fmt.Fprintf(w, "Active\tKey Type\tName\n")
|
|
|
|
fmt.Fprintf(w, "------\t--------\t----\n")
|
|
|
|
if status.ActiveKey != "" {
|
|
|
|
fmt.Fprintf(w, " *\t%s\t%s\n", "AES-CBC", status.ActiveKey)
|
|
|
|
}
|
|
|
|
for _, k := range status.InactiveKeys {
|
|
|
|
fmt.Fprintf(w, "\t%s\t%s\n", "AES-CBC", k)
|
|
|
|
}
|
|
|
|
w.Flush()
|
|
|
|
fmt.Println(statusOutput + tabBuffer.String())
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func Prepare(app *cli.Context) error {
|
2023-08-02 22:17:41 +00:00
|
|
|
if err := cmds.InitLogging(); err != nil {
|
2021-12-07 22:31:32 +00:00
|
|
|
return err
|
|
|
|
}
|
2023-08-02 22:17:41 +00:00
|
|
|
info, err := commandPrep(&cmds.ServerConfig)
|
2021-12-07 22:31:32 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b, err := json.Marshal(server.EncryptionRequest{
|
2023-08-01 17:20:21 +00:00
|
|
|
Stage: pointer.String(secretsencrypt.EncryptionPrepare),
|
2022-02-28 17:14:32 +00:00
|
|
|
Force: cmds.ServerConfig.EncryptForce,
|
2021-12-07 22:31:32 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err = info.Put("/v1-"+version.Program+"/encrypt/config", b); err != nil {
|
2022-03-24 19:23:59 +00:00
|
|
|
return wrapServerError(err)
|
2021-12-07 22:31:32 +00:00
|
|
|
}
|
|
|
|
fmt.Println("prepare completed successfully")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func Rotate(app *cli.Context) error {
|
|
|
|
if err := cmds.InitLogging(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-08-02 22:17:41 +00:00
|
|
|
info, err := commandPrep(&cmds.ServerConfig)
|
2021-12-07 22:31:32 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b, err := json.Marshal(server.EncryptionRequest{
|
2023-08-01 17:20:21 +00:00
|
|
|
Stage: pointer.String(secretsencrypt.EncryptionRotate),
|
2022-02-28 17:14:32 +00:00
|
|
|
Force: cmds.ServerConfig.EncryptForce,
|
2021-12-07 22:31:32 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err = info.Put("/v1-"+version.Program+"/encrypt/config", b); err != nil {
|
2022-03-24 19:23:59 +00:00
|
|
|
return wrapServerError(err)
|
2021-12-07 22:31:32 +00:00
|
|
|
}
|
|
|
|
fmt.Println("rotate completed successfully")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func Reencrypt(app *cli.Context) error {
|
2023-08-02 22:17:41 +00:00
|
|
|
if err := cmds.InitLogging(); err != nil {
|
2021-12-07 22:31:32 +00:00
|
|
|
return err
|
|
|
|
}
|
2023-08-02 22:17:41 +00:00
|
|
|
info, err := commandPrep(&cmds.ServerConfig)
|
2021-12-07 22:31:32 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b, err := json.Marshal(server.EncryptionRequest{
|
2023-08-01 17:20:21 +00:00
|
|
|
Stage: pointer.String(secretsencrypt.EncryptionReencryptActive),
|
2022-02-28 17:14:32 +00:00
|
|
|
Force: cmds.ServerConfig.EncryptForce,
|
|
|
|
Skip: cmds.ServerConfig.EncryptSkip,
|
2021-12-07 22:31:32 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err = info.Put("/v1-"+version.Program+"/encrypt/config", b); err != nil {
|
2022-03-24 19:23:59 +00:00
|
|
|
return wrapServerError(err)
|
2021-12-07 22:31:32 +00:00
|
|
|
}
|
|
|
|
fmt.Println("reencryption started")
|
|
|
|
return nil
|
|
|
|
}
|
2023-08-01 17:20:21 +00:00
|
|
|
|
|
|
|
func RotateKeys(app *cli.Context) error {
|
2023-08-02 22:17:41 +00:00
|
|
|
if err := cmds.InitLogging(); err != nil {
|
2023-08-01 17:20:21 +00:00
|
|
|
return err
|
|
|
|
}
|
2023-08-02 22:17:41 +00:00
|
|
|
info, err := commandPrep(&cmds.ServerConfig)
|
2023-08-01 17:20:21 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b, err := json.Marshal(server.EncryptionRequest{
|
|
|
|
Stage: pointer.String(secretsencrypt.EncryptionRotateKeys),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-02-09 19:37:37 +00:00
|
|
|
timeout := 70 * time.Second
|
|
|
|
if err = info.Put("/v1-"+version.Program+"/encrypt/config", b, clientaccess.WithTimeout(timeout)); err != nil {
|
2023-08-01 17:20:21 +00:00
|
|
|
return wrapServerError(err)
|
|
|
|
}
|
2023-08-02 22:17:41 +00:00
|
|
|
fmt.Println("keys rotated, reencryption started")
|
2023-08-01 17:20:21 +00:00
|
|
|
return nil
|
|
|
|
}
|