package cli import ( "testing" "github.com/anuvu/zot/errors" "github.com/anuvu/zot/pkg/api" "github.com/anuvu/zot/pkg/compliance" "github.com/anuvu/zot/pkg/storage" "github.com/mitchellh/mapstructure" dspec "github.com/opencontainers/distribution-spec" "github.com/rs/zerolog/log" "github.com/spf13/cobra" "github.com/spf13/viper" "github.com/anuvu/zot/pkg/compliance/v1_0_0" ) // metadataConfig reports metadata after parsing, which we use to track // errors func metadataConfig(md *mapstructure.Metadata) viper.DecoderConfigOption { return func(c *mapstructure.DecoderConfig) { c.Metadata = md } } func NewRootCmd() *cobra.Command { showVersion := false config := api.NewConfig() // "serve" serveCmd := &cobra.Command{ Use: "serve ", Aliases: []string{"serve"}, Short: "`serve` stores and distributes OCI images", Long: "`serve` stores and distributes OCI images", Run: func(cmd *cobra.Command, args []string) { if len(args) > 0 { viper.SetConfigFile(args[0]) if err := viper.ReadInConfig(); err != nil { panic(err) } md := &mapstructure.Metadata{} if err := viper.Unmarshal(&config, metadataConfig(md)); err != nil { panic(err) } // if haven't found a single key or there were unused keys, report it as // a error if len(md.Keys) == 0 || len(md.Unused) > 0 { panic(errors.ErrBadConfig) } } c := api.NewController(config) if err := c.Run(); err != nil { panic(err) } }, } // "garbage-collect" gcDelUntagged := false gcDryRun := false gcCmd := &cobra.Command{ Use: "garbage-collect ", Aliases: []string{"gc"}, Short: "`garbage-collect` deletes layers not referenced by any manifests", Long: "`garbage-collect` deletes layers not referenced by any manifests", Run: func(cmd *cobra.Command, args []string) { log.Info().Interface("values", config).Msg("configuration settings") if config.Storage.RootDirectory != "" { if err := storage.Scrub(config.Storage.RootDirectory, gcDryRun); err != nil { panic(err) } } }, } gcCmd.Flags().StringVarP(&config.Storage.RootDirectory, "storage-root-dir", "r", "", "Use specified directory for filestore backing image data") _ = gcCmd.MarkFlagRequired("storage-root-dir") gcCmd.Flags().BoolVarP(&gcDelUntagged, "delete-untagged", "m", false, "delete manifests that are not currently referenced via tag") gcCmd.Flags().BoolVarP(&gcDryRun, "dry-run", "d", false, "do everything except remove the blobs") // "compliance" complianceConfig := compliance.NewConfig() complianceCmd := &cobra.Command{ Use: "compliance", Aliases: []string{"co"}, Short: "`compliance` checks compliance with respect to OCI distribution-spec", Long: "`compliance` checks compliance with respect to OCI distribution-spec", Run: func(cmd *cobra.Command, args []string) { t := &testing.T{} switch complianceConfig.Version { case "all": fallthrough default: v1_0_0.CheckWorkflows(t, complianceConfig) } }, } complianceCmd.Flags().StringVarP(&complianceConfig.Address, "address", "H", "", "Registry server address") if err := complianceCmd.MarkFlagRequired("address"); err != nil { panic(err) } complianceCmd.Flags().StringVarP(&complianceConfig.Port, "port", "P", "", "Registry server port") if err := complianceCmd.MarkFlagRequired("port"); err != nil { panic(err) } complianceCmd.Flags().StringVarP(&complianceConfig.Version, "version", "V", "all", "OCI dist-spec version to check") rootCmd := &cobra.Command{ Use: "zot", Short: "`zot`", Long: "`zot`", Run: func(cmd *cobra.Command, args []string) { if showVersion { log.Info().Str("distribution-spec", dspec.Version).Str("commit", api.Commit).Msg("version") } _ = cmd.Usage() }, } rootCmd.AddCommand(serveCmd) rootCmd.AddCommand(gcCmd) rootCmd.AddCommand(complianceCmd) rootCmd.Flags().BoolVarP(&showVersion, "version", "v", false, "show the version and exit") return rootCmd }