From dff39c9819f767e098305aabc50f5b2c7530cb1a Mon Sep 17 00:00:00 2001 From: carabasdaniel Date: Wed, 29 Nov 2023 12:34:53 +0200 Subject: [PATCH] Add policy CLI errors package --- cmd/policy/build.go | 6 ++++-- cmd/policy/images.go | 4 ++-- cmd/policy/inspect.go | 4 ++-- cmd/policy/login.go | 20 +++++++++--------- cmd/policy/logout.go | 4 +++- cmd/policy/save.go | 4 ++-- cmd/policy/tag.go | 4 ++-- cmd/policy/templates.go | 6 +++--- pkg/app/push.go | 12 +++++------ pkg/app/repl.go | 12 +++++------ pkg/app/rm.go | 4 ++-- pkg/app/save.go | 14 ++++++------ pkg/errors/errors.go | 47 +++++++++++++++++++++++++++++++++++++++++ 13 files changed, 96 insertions(+), 45 deletions(-) create mode 100644 pkg/errors/errors.go diff --git a/cmd/policy/build.go b/cmd/policy/build.go index a10899d..8fe812a 100644 --- a/cmd/policy/build.go +++ b/cmd/policy/build.go @@ -1,6 +1,8 @@ package main -import "github.com/pkg/errors" +import ( + perr "github.com/opcr-io/policy/pkg/errors" +) type BuildCmd struct { Tag string `name:"tag" short:"t" help:"Name and optionally a tag in the 'name:tag' format, if not provided it will be 'default:latest'"` @@ -44,7 +46,7 @@ func (c *BuildCmd) Run(g *Globals) error { c.ClaimsFile, ) if err != nil { - return errors.Wrap(err, "build failed") + return perr.BuildFailed.WithError(err) } <-g.App.Context.Done() diff --git a/cmd/policy/images.go b/cmd/policy/images.go index 1c8511c..30bc77d 100644 --- a/cmd/policy/images.go +++ b/cmd/policy/images.go @@ -1,7 +1,7 @@ package main import ( - "github.com/pkg/errors" + "github.com/opcr-io/policy/pkg/errors" ) type ImagesCmd struct { @@ -14,7 +14,7 @@ func (c *ImagesCmd) Run(g *Globals) error { err := g.App.Images() if err != nil { - return errors.Wrap(err, "failed to list local policies") + return errors.ImagesFailed.WithError(err) } <-g.App.Context.Done() diff --git a/cmd/policy/inspect.go b/cmd/policy/inspect.go index f2e6f16..d3a17a3 100644 --- a/cmd/policy/inspect.go +++ b/cmd/policy/inspect.go @@ -1,6 +1,6 @@ package main -import "github.com/pkg/errors" +import "github.com/opcr-io/policy/pkg/errors" type InspectCmd struct { Policy string `arg:"" name:"policy" help:"Policy to inspect."` @@ -9,7 +9,7 @@ type InspectCmd struct { func (c *InspectCmd) Run(g *Globals) error { err := g.App.Inspect(c.Policy) if err != nil { - return errors.Wrap(err, "failed to inspect policy") + return errors.InspectFailed.WithError(err) } <-g.App.Context.Done() diff --git a/cmd/policy/login.go b/cmd/policy/login.go index 099b833..36f848a 100644 --- a/cmd/policy/login.go +++ b/cmd/policy/login.go @@ -8,7 +8,7 @@ import ( "github.com/docker/cli/cli/config/types" - "github.com/pkg/errors" + perr "github.com/opcr-io/policy/pkg/errors" "golang.org/x/term" ) @@ -25,18 +25,18 @@ func (c *LoginCmd) Run(g *Globals) error { g.App.UI.Exclamation().Msg("Using --password via the CLI is insecure. Use --password-stdin.") if c.PasswordStdin { - return errors.New("--password and --password-stdin are mutually exclusive") + return perr.LoginFailed.WithMessage("--password and --password-stdin are mutually exclusive") } } if c.PasswordStdin { if c.Username == "" { - return errors.New("Must provide --username with --password-stdin") + return perr.LoginFailed.WithMessage("Must provide --username with --password-stdin") } contents, err := io.ReadAll(g.App.UI.Input()) if err != nil { - return err + return perr.LoginFailed.WithError(err) } c.Password = strings.TrimSuffix(string(contents), "\n") @@ -44,7 +44,7 @@ func (c *LoginCmd) Run(g *Globals) error { } if c.Server == "" { - return errors.New("Must provide --server") + return perr.LoginFailed.WithMessage("Must provide --server") } password := c.Password @@ -52,7 +52,7 @@ func (c *LoginCmd) Run(g *Globals) error { g.App.UI.Normal().NoNewline().Msg("Password: ") bytePassword, err := term.ReadPassword(int(syscall.Stdin)) // nolint:unconvert // needed for windows if err != nil { - return errors.Wrap(err, "failed to read password from stdin") + return perr.LoginFailed.WithError(err) } password = string(bytePassword) @@ -64,12 +64,12 @@ func (c *LoginCmd) Run(g *Globals) error { Msg("Logging in.") err := g.App.Ping(c.Server, c.Username, password) if err != nil { - return err + return perr.LoginFailed.WithError(err) } var setDefault bool stat, err := os.Stdin.Stat() if err != nil { - return err + return perr.LoginFailed.WithError(err) } if (stat.Mode() & os.ModeCharDevice) == 0 { @@ -85,7 +85,7 @@ func (c *LoginCmd) Run(g *Globals) error { Password: password, }) if err != nil { - return err + return perr.LoginFailed.WithError(err) } if setDefault { @@ -94,7 +94,7 @@ func (c *LoginCmd) Run(g *Globals) error { err = g.App.Configuration.SaveDefaultDomain() if err != nil { - return err + return perr.LoginFailed.WithError(err) } g.App.UI.Normal().Msg("OK.") diff --git a/cmd/policy/logout.go b/cmd/policy/logout.go index bac1448..011d0d2 100644 --- a/cmd/policy/logout.go +++ b/cmd/policy/logout.go @@ -1,5 +1,7 @@ package main +import "github.com/opcr-io/policy/pkg/errors" + type LogoutCmd struct { Server string `name:"server" short:"s" help:"Server to logout from." default:"{{ .DefaultDomain }}"` } @@ -11,7 +13,7 @@ func (c *LogoutCmd) Run(g *Globals) error { err := g.App.RemoveServerCreds(c.Server) if err != nil { - return err + return errors.LogoutFailed.WithError(err) } g.App.UI.Normal().Msg("OK.") diff --git a/cmd/policy/save.go b/cmd/policy/save.go index 9b29938..c64f919 100644 --- a/cmd/policy/save.go +++ b/cmd/policy/save.go @@ -1,6 +1,6 @@ package main -import "github.com/pkg/errors" +import "github.com/opcr-io/policy/pkg/errors" type SaveCmd struct { Policy string `arg:"" name:"policy" help:"Policy to save."` @@ -10,7 +10,7 @@ type SaveCmd struct { func (c *SaveCmd) Run(g *Globals) error { err := g.App.Save(c.Policy, c.File) if err != nil { - return errors.Wrap(err, "failed to save local bundle tarball") + return errors.SaveFailed.WithError(err) } <-g.App.Context.Done() diff --git a/cmd/policy/tag.go b/cmd/policy/tag.go index 4695159..303a17a 100644 --- a/cmd/policy/tag.go +++ b/cmd/policy/tag.go @@ -1,6 +1,6 @@ package main -import "github.com/pkg/errors" +import "github.com/opcr-io/policy/pkg/errors" type TagCmd struct { Policy string `arg:"" name:"policy" help:"Source policy name." type:"string"` @@ -10,7 +10,7 @@ type TagCmd struct { func (c *TagCmd) Run(g *Globals) error { err := g.App.Tag(c.Policy, c.Tag) if err != nil { - return errors.Wrap(err, "tagging failed") + return errors.TagFailed.WithError(err) } <-g.App.Context.Done() diff --git a/cmd/policy/templates.go b/cmd/policy/templates.go index 9f1ddd0..4de8f6f 100644 --- a/cmd/policy/templates.go +++ b/cmd/policy/templates.go @@ -1,6 +1,6 @@ package main -import "github.com/pkg/errors" +import "github.com/opcr-io/policy/pkg/errors" type TemplatesCmd struct { Apply ApplyCmd `cmd:"" name:"apply" help:"Create or update a policy or related artifacts from a template."` @@ -19,7 +19,7 @@ type ListCmd struct { func (c *ApplyCmd) Run(g *Globals) error { err := g.App.TemplateApply(c.Template, c.Output, c.Overwrite) if err != nil { - return errors.Wrapf(err, "failed to apply template '%s'", c.Template) + return errors.TemplateFailed.WithError(err) } <-g.App.Context.Done() @@ -29,7 +29,7 @@ func (c *ApplyCmd) Run(g *Globals) error { func (c *ListCmd) Run(g *Globals) error { err := g.App.TemplatesList() if err != nil { - return errors.Wrap(err, "failed to list templates") + return errors.TemplateFailed.WithError(err) } <-g.App.Context.Done() diff --git a/pkg/app/push.go b/pkg/app/push.go index 637c1df..eac6ef9 100644 --- a/pkg/app/push.go +++ b/pkg/app/push.go @@ -3,7 +3,7 @@ package app import ( "github.com/opcr-io/policy/oci" "github.com/opcr-io/policy/parser" - "github.com/pkg/errors" + "github.com/opcr-io/policy/pkg/errors" ) func (c *PolicyApp) Push(userRef string) error { @@ -11,22 +11,22 @@ func (c *PolicyApp) Push(userRef string) error { ref, err := parser.CalculatePolicyRef(userRef, c.Configuration.DefaultDomain) if err != nil { - return err + return errors.PushFailed.WithError(err) } ociClient, err := oci.NewOCI(c.Context, c.Logger, c.getHosts, c.Configuration.PoliciesRoot()) if err != nil { - return err + return errors.PushFailed.WithError(err) } refs, err := ociClient.ListReferences() if err != nil { - return err + return errors.PushFailed.WithError(err) } refDescriptor, ok := refs[ref] if !ok { - return errors.Errorf("policy [%s] not found in the local store", ref) + return errors.NotFound.WithMessage("policy [%s] not in the local store", ref) } c.UI.Normal(). @@ -36,7 +36,7 @@ func (c *PolicyApp) Push(userRef string) error { digest, err := ociClient.Push(ref) if err != nil { - return err + return errors.PushFailed.WithError(err) } c.UI.Normal(). diff --git a/pkg/app/repl.go b/pkg/app/repl.go index 8769b50..cc24bf0 100644 --- a/pkg/app/repl.go +++ b/pkg/app/repl.go @@ -9,9 +9,9 @@ import ( "github.com/aserto-dev/runtime" "github.com/opcr-io/policy/oci" "github.com/opcr-io/policy/parser" + "github.com/opcr-io/policy/pkg/errors" "github.com/open-policy-agent/opa/repl" ocispec "github.com/opencontainers/image-spec/specs-go/v1" - "github.com/pkg/errors" ) func (c *PolicyApp) Repl(ref string, maxErrors int) error { @@ -49,7 +49,7 @@ func (c *PolicyApp) Repl(ref string, maxErrors int) error { } descriptor, ok = existingRefs[existingRefParsed] if !ok { - return errors.Errorf("ref [%s] not found in the local store", ref) + return errors.NotFound.WithMessage("policy [%s] not in the local store", ref) } } @@ -68,18 +68,18 @@ func (c *PolicyApp) Repl(ref string, maxErrors int) error { }, }) if err != nil { - return errors.Wrap(err, "failed to setup the OPA runtime") + return errors.ReplFailed.WithError(err) } defer cleanup() err = opaRuntime.Start(c.Context) if err != nil { - return errors.Wrap(err, "OPA runtime failed to start") + return errors.ReplFailed.WithError(err) } err = opaRuntime.WaitForPlugins(c.Context, time.Minute*1) if err != nil { - return errors.Wrap(err, "plugins didn't start on time") + return errors.ReplFailed.WithError(err) } loop := repl.New(opaRuntime.GetPluginsManager().Store, c.Configuration.ReplHistoryFile(), c.UI.Output(), "", maxErrors, fmt.Sprintf("running policy [%s]", ref)) @@ -98,7 +98,7 @@ func (c *PolicyApp) getBundleHex(ociClient *oci.Oci, descriptor *ocispec.Descrip } bundleHex = bundleDescriptor.Digest.Hex() if bundleHex == "" { - return "", errors.New("current manifest does not contain a MediaTypeImageLayerGzip") + return "", errors.ReplFailed.WithMessage("current manifest does not contain a MediaTypeImageLayerGzip") } } else { bundleHex = descriptor.Digest.Hex() diff --git a/pkg/app/rm.go b/pkg/app/rm.go index 9a95344..d574991 100644 --- a/pkg/app/rm.go +++ b/pkg/app/rm.go @@ -7,8 +7,8 @@ import ( "github.com/opcr-io/policy/oci" "github.com/opcr-io/policy/parser" + "github.com/opcr-io/policy/pkg/errors" ocispec "github.com/opencontainers/image-spec/specs-go/v1" - "github.com/pkg/errors" ) func (c *PolicyApp) Rm(existingRef string, force bool) error { @@ -47,7 +47,7 @@ func (c *PolicyApp) Rm(existingRef string, force bool) error { ref, ok := existingRefs[existingRefParsed] if !ok { - return errors.Errorf("ref [%s] not found in the local store", existingRef) + return errors.NotFound.WithMessage("policy [%s] not in the local store", existingRef) } // attach ref name annotation for comparison. if len(ref.Annotations) == 0 || ref.Annotations[ocispec.AnnotationRefName] == "" { diff --git a/pkg/app/save.go b/pkg/app/save.go index d52e49e..f6d3ada 100644 --- a/pkg/app/save.go +++ b/pkg/app/save.go @@ -7,8 +7,8 @@ import ( "github.com/opcr-io/policy/oci" "github.com/opcr-io/policy/parser" + "github.com/opcr-io/policy/pkg/errors" ocispec "github.com/opencontainers/image-spec/specs-go/v1" - "github.com/pkg/errors" ) func (c *PolicyApp) Save(userRef, outputFilePath string) error { @@ -16,18 +16,18 @@ func (c *PolicyApp) Save(userRef, outputFilePath string) error { var outputFile *os.File ref, err := parser.CalculatePolicyRef(userRef, c.Configuration.DefaultDomain) if err != nil { - return err + return errors.SaveFailed.WithError(err) } ociClient, err := oci.NewOCI(c.Context, c.Logger, c.getHosts, c.Configuration.PoliciesRoot()) if err != nil { - return err + return errors.SaveFailed.WithError(err) } // if the reference descriptor is the manifest get the tarball descriptor information from the manifest layers. refDescriptor, err := c.getRefDescriptor(ociClient, ref) if err != nil { - return err + return errors.SaveFailed.WithError(err) } if outputFilePath == "-" { @@ -39,7 +39,7 @@ func (c *PolicyApp) Save(userRef, outputFilePath string) error { outputFile, err = os.Create(outputFilePath) if err != nil { - return errors.Wrapf(err, "failed to create output file [%s]", outputFilePath) + return errors.SaveFailed.WithError(err).WithMessage("failed to create output file [%s]", outputFilePath) } defer func() { @@ -52,7 +52,7 @@ func (c *PolicyApp) Save(userRef, outputFilePath string) error { err = c.writePolicy(ociClient, refDescriptor, outputFile) if err != nil { - return err + return errors.SaveFailed.WithError(err) } return nil @@ -66,7 +66,7 @@ func (c *PolicyApp) getRefDescriptor(ociClient *oci.Oci, ref string) (*ocispec.D refDescriptor, ok := refs[ref] if !ok { - return nil, errors.Errorf("Image %s not found", ref) + return nil, errors.NotFound.WithMessage("policy [%s] not in the local store", ref) } if refDescriptor.MediaType == ocispec.MediaTypeImageManifest { diff --git a/pkg/errors/errors.go b/pkg/errors/errors.go new file mode 100644 index 0000000..2f37031 --- /dev/null +++ b/pkg/errors/errors.go @@ -0,0 +1,47 @@ +package errors + +import "fmt" + +var ( + NotFound = NewPolicyError("policy not found") + BuildFailed = NewPolicyError("build failed") + LoginFailed = NewPolicyError("login failed") + LogoutFailed = NewPolicyError("logout failed") + ImagesFailed = NewPolicyError("list images failed") + InspectFailed = NewPolicyError("inspect failed") + PullFailed = NewPolicyError("pull failed") + PushFailed = NewPolicyError("push failed") + SaveFailed = NewPolicyError("save failed") + ReplFailed = NewPolicyError("repl failed") + TagFailed = NewPolicyError("tag failed") + TemplateFailed = NewPolicyError("template failed") +) + +type PolicyCLIError struct { + Message string + Err error +} + +func NewPolicyError(message string) *PolicyCLIError { + return &PolicyCLIError{ + Message: message, + } +} + +func (e *PolicyCLIError) Error() string { + response := e.Message + if e.Err != nil { + response += " -> " + e.Err.Error() + } + return response +} + +func (e *PolicyCLIError) WithMessage(message string, args ...interface{}) *PolicyCLIError { + e.Message += " -> " + fmt.Sprintf(message, args...) + return e +} + +func (e *PolicyCLIError) WithError(base error) *PolicyCLIError { + e.Err = base + return e +}