main.go 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. package main
  2. import (
  3. "os"
  4. "slices"
  5. "time"
  6. "github.com/fatih/color"
  7. cc "github.com/ivanpirog/coloredcobra"
  8. log "github.com/sirupsen/logrus"
  9. "github.com/spf13/cobra"
  10. "github.com/crowdsecurity/crowdsec/pkg/csconfig"
  11. "github.com/crowdsecurity/crowdsec/pkg/database"
  12. "github.com/crowdsecurity/crowdsec/pkg/fflag"
  13. )
  14. var trace_lvl, dbg_lvl, nfo_lvl, wrn_lvl, err_lvl bool
  15. var ConfigFilePath string
  16. var csConfig *csconfig.Config
  17. var dbClient *database.Client
  18. var outputFormat string
  19. var OutputColor string
  20. var mergedConfig string
  21. // flagBranch overrides the value in csConfig.Cscli.HubBranch
  22. var flagBranch = ""
  23. type configGetter func() *csconfig.Config
  24. func initConfig() {
  25. var err error
  26. if trace_lvl {
  27. log.SetLevel(log.TraceLevel)
  28. } else if dbg_lvl {
  29. log.SetLevel(log.DebugLevel)
  30. } else if nfo_lvl {
  31. log.SetLevel(log.InfoLevel)
  32. } else if wrn_lvl {
  33. log.SetLevel(log.WarnLevel)
  34. } else if err_lvl {
  35. log.SetLevel(log.ErrorLevel)
  36. }
  37. if !slices.Contains(NoNeedConfig, os.Args[1]) {
  38. log.Debugf("Using %s as configuration file", ConfigFilePath)
  39. csConfig, mergedConfig, err = csconfig.NewConfig(ConfigFilePath, false, false, true)
  40. if err != nil {
  41. log.Fatal(err)
  42. }
  43. } else {
  44. csConfig = csconfig.NewDefaultConfig()
  45. }
  46. // recap of the enabled feature flags, because logging
  47. // was not enabled when we set them from envvars
  48. if fflist := csconfig.ListFeatureFlags(); fflist != "" {
  49. log.Debugf("Enabled feature flags: %s", fflist)
  50. }
  51. if flagBranch != "" {
  52. csConfig.Cscli.HubBranch = flagBranch
  53. }
  54. if outputFormat != "" {
  55. csConfig.Cscli.Output = outputFormat
  56. }
  57. if csConfig.Cscli.Output == "" {
  58. csConfig.Cscli.Output = "human"
  59. }
  60. if csConfig.Cscli.Output != "human" && csConfig.Cscli.Output != "json" && csConfig.Cscli.Output != "raw" {
  61. log.Fatalf("output format '%s' not supported: must be one of human, json, raw", csConfig.Cscli.Output)
  62. }
  63. if csConfig.Cscli.Output == "json" {
  64. log.SetFormatter(&log.JSONFormatter{})
  65. log.SetLevel(log.ErrorLevel)
  66. } else if csConfig.Cscli.Output == "raw" {
  67. log.SetLevel(log.ErrorLevel)
  68. }
  69. if OutputColor != "" {
  70. csConfig.Cscli.Color = OutputColor
  71. if OutputColor != "yes" && OutputColor != "no" && OutputColor != "auto" {
  72. log.Fatalf("output color %s unknown", OutputColor)
  73. }
  74. }
  75. }
  76. // list of valid subcommands for the shell completion
  77. var validArgs = []string{
  78. "alerts", "appsec-configs", "appsec-rules", "bouncers", "capi", "collections",
  79. "completion", "config", "console", "contexts", "dashboard", "decisions", "explain",
  80. "hub", "hubtest", "lapi", "machines", "metrics", "notifications", "parsers",
  81. "postoverflows", "scenarios", "simulation", "support", "version",
  82. }
  83. var NoNeedConfig = []string{
  84. "doc",
  85. "help",
  86. "completion",
  87. "version",
  88. "hubtest",
  89. }
  90. func main() {
  91. // set the formatter asap and worry about level later
  92. logFormatter := &log.TextFormatter{TimestampFormat: time.RFC3339, FullTimestamp: true}
  93. log.SetFormatter(logFormatter)
  94. if err := fflag.RegisterAllFeatures(); err != nil {
  95. log.Fatalf("failed to register features: %s", err)
  96. }
  97. if err := csconfig.LoadFeatureFlagsEnv(log.StandardLogger()); err != nil {
  98. log.Fatalf("failed to set feature flags from env: %s", err)
  99. }
  100. cmd := &cobra.Command{
  101. Use: "cscli",
  102. Short: "cscli allows you to manage crowdsec",
  103. Long: `cscli is the main command to interact with your crowdsec service, scenarios & db.
  104. It is meant to allow you to manage bans, parsers/scenarios/etc, api and generally manage you crowdsec setup.`,
  105. ValidArgs: validArgs,
  106. DisableAutoGenTag: true,
  107. SilenceErrors: true,
  108. SilenceUsage: true,
  109. /*TBD examples*/
  110. }
  111. cc.Init(&cc.Config{
  112. RootCmd: cmd,
  113. Headings: cc.Yellow,
  114. Commands: cc.Green + cc.Bold,
  115. CmdShortDescr: cc.Cyan,
  116. Example: cc.Italic,
  117. ExecName: cc.Bold,
  118. Aliases: cc.Bold + cc.Italic,
  119. FlagsDataType: cc.White,
  120. Flags: cc.Green,
  121. FlagsDescr: cc.Cyan,
  122. })
  123. cmd.SetOut(color.Output)
  124. cmd.PersistentFlags().StringVarP(&ConfigFilePath, "config", "c", csconfig.DefaultConfigPath("config.yaml"), "path to crowdsec config file")
  125. cmd.PersistentFlags().StringVarP(&outputFormat, "output", "o", "", "Output format: human, json, raw")
  126. cmd.PersistentFlags().StringVarP(&OutputColor, "color", "", "auto", "Output color: yes, no, auto")
  127. cmd.PersistentFlags().BoolVar(&dbg_lvl, "debug", false, "Set logging to debug")
  128. cmd.PersistentFlags().BoolVar(&nfo_lvl, "info", false, "Set logging to info")
  129. cmd.PersistentFlags().BoolVar(&wrn_lvl, "warning", false, "Set logging to warning")
  130. cmd.PersistentFlags().BoolVar(&err_lvl, "error", false, "Set logging to error")
  131. cmd.PersistentFlags().BoolVar(&trace_lvl, "trace", false, "Set logging to trace")
  132. cmd.PersistentFlags().StringVar(&flagBranch, "branch", "", "Override hub branch on github")
  133. if err := cmd.PersistentFlags().MarkHidden("branch"); err != nil {
  134. log.Fatalf("failed to hide flag: %s", err)
  135. }
  136. // Look for "-c /path/to/config.yaml"
  137. // This duplicates the logic in cobra, but we need to do it before
  138. // because feature flags can change which subcommands are available.
  139. for i, arg := range os.Args {
  140. if arg == "-c" || arg == "--config" {
  141. if len(os.Args) > i+1 {
  142. ConfigFilePath = os.Args[i+1]
  143. }
  144. }
  145. }
  146. if err := csconfig.LoadFeatureFlagsFile(ConfigFilePath, log.StandardLogger()); err != nil {
  147. log.Fatal(err)
  148. }
  149. if len(os.Args) > 1 {
  150. cobra.OnInitialize(initConfig)
  151. }
  152. /*don't sort flags so we can enforce order*/
  153. cmd.Flags().SortFlags = false
  154. cmd.PersistentFlags().SortFlags = false
  155. // we use a getter because the config is not initialized until the Execute() call
  156. getconfig := func() *csconfig.Config {
  157. return csConfig
  158. }
  159. cmd.AddCommand(NewCLIDoc().NewCommand(cmd))
  160. cmd.AddCommand(NewCLIVersion().NewCommand())
  161. cmd.AddCommand(NewConfigCmd())
  162. cmd.AddCommand(NewCLIHub(getconfig).NewCommand())
  163. cmd.AddCommand(NewMetricsCmd())
  164. cmd.AddCommand(NewCLIDashboard(getconfig).NewCommand())
  165. cmd.AddCommand(NewCLIDecisions().NewCommand())
  166. cmd.AddCommand(NewCLIAlerts().NewCommand())
  167. cmd.AddCommand(NewCLISimulation(getconfig).NewCommand())
  168. cmd.AddCommand(NewCLIBouncers(getconfig).NewCommand())
  169. cmd.AddCommand(NewCLIMachines(getconfig).NewCommand())
  170. cmd.AddCommand(NewCLICapi().NewCommand())
  171. cmd.AddCommand(NewLapiCmd())
  172. cmd.AddCommand(NewCompletionCmd())
  173. cmd.AddCommand(NewConsoleCmd())
  174. cmd.AddCommand(NewCLIExplain().NewCommand())
  175. cmd.AddCommand(NewCLIHubTest().NewCommand())
  176. cmd.AddCommand(NewCLINotifications().NewCommand())
  177. cmd.AddCommand(NewCLISupport().NewCommand())
  178. cmd.AddCommand(NewCLIPapi(getconfig).NewCommand())
  179. cmd.AddCommand(NewCLICollection().NewCommand())
  180. cmd.AddCommand(NewCLIParser().NewCommand())
  181. cmd.AddCommand(NewCLIScenario().NewCommand())
  182. cmd.AddCommand(NewCLIPostOverflow().NewCommand())
  183. cmd.AddCommand(NewCLIContext().NewCommand())
  184. cmd.AddCommand(NewCLIAppsecConfig().NewCommand())
  185. cmd.AddCommand(NewCLIAppsecRule().NewCommand())
  186. if fflag.CscliSetup.IsEnabled() {
  187. cmd.AddCommand(NewSetupCmd())
  188. }
  189. if err := cmd.Execute(); err != nil {
  190. log.Fatal(err)
  191. }
  192. }