// Copyright 2020 The Kubernetes Authors. // SPDX-License-Identifier: Apache-2.0 package preview import ( "context" "github.com/spf13/cobra" "k8s.io/cli-runtime/pkg/genericclioptions" cmdutil "k8s.io/kubectl/pkg/cmd/util" "k8s.io/kubectl/pkg/util/i18n" "sigs.k8s.io/cli-utils/cmd/printers" "sigs.k8s.io/cli-utils/pkg/apply" "sigs.k8s.io/cli-utils/pkg/apply/event" "sigs.k8s.io/cli-utils/pkg/common" "sigs.k8s.io/cli-utils/pkg/inventory" "sigs.k8s.io/cli-utils/pkg/provider" "sigs.k8s.io/kustomize/kyaml/setters2" ) var ( noPrune = false serverDryRun = false previewDestroy = false ) // GetPreviewRunner creates and returns the PreviewRunner which stores the cobra command. func GetPreviewRunner(provider provider.Provider, ioStreams genericclioptions.IOStreams) *PreviewRunner { r := &PreviewRunner{ Applier: apply.NewApplier(provider), Destroyer: apply.NewDestroyer(provider), ioStreams: ioStreams, provider: provider, } cmd := &cobra.Command{ Use: "preview (DIRECTORY | STDIN)", DisableFlagsInUseLine: true, Short: i18n.T("Preview the apply of a configuration"), Args: cobra.MaximumNArgs(1), RunE: r.RunE, } cmd.Flags().BoolVar(&noPrune, "no-prune", noPrune, "If true, do not prune previously applied objects.") cmd.Flags().BoolVar(&serverDryRun, "server-side", serverDryRun, "If true, preview runs in the server instead of the client.") cmd.Flags().BoolVar(&previewDestroy, "destroy", previewDestroy, "If true, preview of destroy operations will be displayed.") r.Command = cmd return r } // PreviewCommand creates the PreviewRunner, returning the cobra command associated with it. func PreviewCommand(f cmdutil.Factory, ioStreams genericclioptions.IOStreams) *cobra.Command { provider := provider.NewProvider(f) return GetPreviewRunner(provider, ioStreams).Command } // PreviewRunner encapsulates data necessary to run the preview command. type PreviewRunner struct { Command *cobra.Command ioStreams genericclioptions.IOStreams Applier *apply.Applier Destroyer *apply.Destroyer provider provider.Provider } // RunE is the function run from the cobra command. func (r *PreviewRunner) RunE(cmd *cobra.Command, args []string) error { err := setters2.CheckRequiredSettersSet() if err != nil { return err } var ch <-chan event.Event err = r.Destroyer.Initialize() if err != nil { return err } drs := common.DryRunClient if serverDryRun { drs = common.DryRunServer } if previewDestroy { r.Destroyer.DryRunStrategy = drs } reader, err := r.provider.ManifestReader(cmd.InOrStdin(), args) if err != nil { return err } objs, err := reader.Read() if err != nil { return err } // if destroy flag is set in preview, transmit it to destroyer DryRunStrategy flag // and pivot execution to destroy with dry-run if !r.Destroyer.DryRunStrategy.ClientOrServerDryRun() { err = r.Applier.Initialize() if err != nil { return err } // Create a context ctx := context.Background() _, err := common.DemandOneDirectory(args) if err != nil { return err } // Run the applier. It will return a channel where we can receive updates // to keep track of progress and any issues. serverSideOptions := common.ServerSideOptions{ ServerSideApply: false, ForceConflicts: false, FieldManager: common.DefaultFieldManager, } ch = r.Applier.Run(ctx, objs, apply.Options{ EmitStatusEvents: false, NoPrune: noPrune, DryRunStrategy: drs, ServerSideOptions: serverSideOptions, }) } else { inv, _, err := inventory.SplitUnstructureds(objs) if err != nil { return err } ch = r.Destroyer.Run(inv) } // The printer will print updates from the channel. It will block // until the channel is closed. printer := printers.GetPrinter(printers.EventsPrinter, r.ioStreams) return printer.Print(ch, drs) }