more knecht

This commit is contained in:
2026-04-04 15:17:33 +02:00
parent b4fddbb5b6
commit 397cbea7fb
6 changed files with 194 additions and 12 deletions

45
knecht/cmd/restart.go Normal file
View File

@@ -0,0 +1,45 @@
package cmd
import (
"fmt"
"github.com/spf13/cobra"
)
var restartCmd = &cobra.Command{
Use: "restart <stack>",
Short: "Stop and start a stack",
Args: cobra.ExactArgs(1),
RunE: func(cmd *cobra.Command, args []string) error {
_, client, _, err := setup()
if err != nil {
return err
}
name := args[0]
remote, err := client.GetStackByName(name)
if err != nil {
return err
}
if remote == nil {
return fmt.Errorf("stack %q not found", name)
}
fmt.Printf("Stopping %q...\n", name)
if err := client.StopStack(remote.ID); err != nil {
return fmt.Errorf("stop failed: %w", err)
}
fmt.Printf("Starting %q...\n", name)
if err := client.StartStack(remote.ID); err != nil {
return fmt.Errorf("start failed: %w", err)
}
fmt.Printf("Restarted %q\n", name)
return nil
},
}
func init() {
rootCmd.AddCommand(restartCmd)
}

View File

@@ -9,6 +9,8 @@ import (
"github.com/spf13/cobra"
)
var pruneEnvFlag bool
var updateCmd = &cobra.Command{
Use: "update <stack>",
Short: "Update an existing stack, preserving env vars",
@@ -43,7 +45,7 @@ var updateCmd = &cobra.Command{
return err
}
env, err := mergeEnvVars(name, remote.Env, exampleKeys)
env, err := mergeEnvVars(name, remote.Env, exampleKeys, pruneEnvFlag)
if err != nil {
return err
}
@@ -64,12 +66,26 @@ var updateCmd = &cobra.Command{
// mergeEnvVars preserves existing Portainer env vars and prompts via TUI for
// any keys present in .env.example but missing from Portainer.
// Returns nil if the user cancelled the form.
func mergeEnvVars(stackName string, existing []portainer.EnvVar, exampleKeys []string) ([]portainer.EnvVar, error) {
func mergeEnvVars(stackName string, existing []portainer.EnvVar, exampleKeys []string, prune bool) ([]portainer.EnvVar, error) {
envMap := make(map[string]string, len(existing))
for _, e := range existing {
envMap[e.Name] = e.Value
}
// Remove keys not in .env.example when pruning
if prune && exampleKeys != nil {
allowed := make(map[string]bool, len(exampleKeys))
for _, k := range exampleKeys {
allowed[k] = true
}
for k := range envMap {
if !allowed[k] {
fmt.Printf("Removing unknown env var %q\n", k)
delete(envMap, k)
}
}
}
var missing []string
for _, key := range exampleKeys {
if _, ok := envMap[key]; !ok {
@@ -98,5 +114,6 @@ func mergeEnvVars(stackName string, existing []portainer.EnvVar, exampleKeys []s
}
func init() {
updateCmd.Flags().BoolVar(&pruneEnvFlag, "prune-env", false, "remove env vars not defined in .env.example")
rootCmd.AddCommand(updateCmd)
}