main.go 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261
  1. package main
  2. import (
  3. "fmt"
  4. "os"
  5. "path/filepath"
  6. "slices"
  7. "strings"
  8. "github.com/fatih/color"
  9. cc "github.com/ivanpirog/coloredcobra"
  10. log "github.com/sirupsen/logrus"
  11. "github.com/spf13/cobra"
  12. "github.com/spf13/cobra/doc"
  13. "github.com/crowdsecurity/crowdsec/pkg/csconfig"
  14. "github.com/crowdsecurity/crowdsec/pkg/cwhub"
  15. "github.com/crowdsecurity/crowdsec/pkg/cwversion"
  16. "github.com/crowdsecurity/crowdsec/pkg/database"
  17. "github.com/crowdsecurity/crowdsec/pkg/fflag"
  18. )
  19. var trace_lvl, dbg_lvl, nfo_lvl, wrn_lvl, err_lvl bool
  20. var ConfigFilePath string
  21. var csConfig *csconfig.Config
  22. var dbClient *database.Client
  23. var OutputFormat string
  24. var OutputColor string
  25. var mergedConfig string
  26. func initConfig() {
  27. var err error
  28. if trace_lvl {
  29. log.SetLevel(log.TraceLevel)
  30. } else if dbg_lvl {
  31. log.SetLevel(log.DebugLevel)
  32. } else if nfo_lvl {
  33. log.SetLevel(log.InfoLevel)
  34. } else if wrn_lvl {
  35. log.SetLevel(log.WarnLevel)
  36. } else if err_lvl {
  37. log.SetLevel(log.ErrorLevel)
  38. }
  39. if !slices.Contains(NoNeedConfig, os.Args[1]) {
  40. log.Debugf("Using %s as configuration file", ConfigFilePath)
  41. csConfig, mergedConfig, err = csconfig.NewConfig(ConfigFilePath, false, false, true)
  42. if err != nil {
  43. log.Fatal(err)
  44. }
  45. } else {
  46. // XXX: check all the defaults
  47. csConfig = csconfig.NewDefaultConfig()
  48. }
  49. // recap of the enabled feature flags, because logging
  50. // was not enabled when we set them from envvars
  51. if fflist := csconfig.ListFeatureFlags(); fflist != "" {
  52. log.Debugf("Enabled feature flags: %s", fflist)
  53. }
  54. if csConfig.Cscli == nil {
  55. log.Fatalf("missing 'cscli' configuration in '%s', exiting", ConfigFilePath)
  56. }
  57. if cwhub.HubBranch == "" && csConfig.Cscli.HubBranch != "" {
  58. cwhub.HubBranch = csConfig.Cscli.HubBranch
  59. }
  60. if OutputFormat != "" {
  61. csConfig.Cscli.Output = OutputFormat
  62. if OutputFormat != "json" && OutputFormat != "raw" && OutputFormat != "human" {
  63. log.Fatalf("output format %s unknown", OutputFormat)
  64. }
  65. }
  66. if csConfig.Cscli.Output == "" {
  67. csConfig.Cscli.Output = "human"
  68. }
  69. if csConfig.Cscli.Output == "json" {
  70. log.SetFormatter(&log.JSONFormatter{})
  71. log.SetLevel(log.ErrorLevel)
  72. } else if csConfig.Cscli.Output == "raw" {
  73. log.SetLevel(log.ErrorLevel)
  74. }
  75. if OutputColor != "" {
  76. csConfig.Cscli.Color = OutputColor
  77. if OutputColor != "yes" && OutputColor != "no" && OutputColor != "auto" {
  78. log.Fatalf("output color %s unknown", OutputColor)
  79. }
  80. }
  81. }
  82. var validArgs = []string{
  83. "scenarios", "parsers", "collections", "capi", "lapi", "postoverflows", "machines",
  84. "metrics", "bouncers", "alerts", "decisions", "simulation", "hub", "dashboard",
  85. "config", "completion", "version", "console", "notifications", "support",
  86. }
  87. func prepender(filename string) string {
  88. const header = `---
  89. id: %s
  90. title: %s
  91. ---
  92. `
  93. name := filepath.Base(filename)
  94. base := strings.TrimSuffix(name, filepath.Ext(name))
  95. return fmt.Sprintf(header, base, strings.ReplaceAll(base, "_", " "))
  96. }
  97. func linkHandler(name string) string {
  98. return fmt.Sprintf("/cscli/%s", name)
  99. }
  100. var (
  101. NoNeedConfig = []string{
  102. "help",
  103. "completion",
  104. "version",
  105. "hubtest",
  106. }
  107. )
  108. func main() {
  109. // set the formatter asap and worry about level later
  110. logFormatter := &log.TextFormatter{TimestampFormat: "02-01-2006 15:04:05", FullTimestamp: true}
  111. log.SetFormatter(logFormatter)
  112. if err := fflag.RegisterAllFeatures(); err != nil {
  113. log.Fatalf("failed to register features: %s", err)
  114. }
  115. if err := csconfig.LoadFeatureFlagsEnv(log.StandardLogger()); err != nil {
  116. log.Fatalf("failed to set feature flags from env: %s", err)
  117. }
  118. var rootCmd = &cobra.Command{
  119. Use: "cscli",
  120. Short: "cscli allows you to manage crowdsec",
  121. Long: `cscli is the main command to interact with your crowdsec service, scenarios & db.
  122. It is meant to allow you to manage bans, parsers/scenarios/etc, api and generally manage you crowdsec setup.`,
  123. ValidArgs: validArgs,
  124. DisableAutoGenTag: true,
  125. SilenceErrors: true,
  126. SilenceUsage: true,
  127. /*TBD examples*/
  128. }
  129. cc.Init(&cc.Config{
  130. RootCmd: rootCmd,
  131. Headings: cc.Yellow,
  132. Commands: cc.Green + cc.Bold,
  133. CmdShortDescr: cc.Cyan,
  134. Example: cc.Italic,
  135. ExecName: cc.Bold,
  136. Aliases: cc.Bold + cc.Italic,
  137. FlagsDataType: cc.White,
  138. Flags: cc.Green,
  139. FlagsDescr: cc.Cyan,
  140. })
  141. rootCmd.SetOut(color.Output)
  142. var cmdDocGen = &cobra.Command{
  143. Use: "doc",
  144. Short: "Generate the documentation in `./doc/`. Directory must exist.",
  145. Args: cobra.ExactArgs(0),
  146. Hidden: true,
  147. DisableAutoGenTag: true,
  148. RunE: func(cmd *cobra.Command, args []string) error {
  149. if err := doc.GenMarkdownTreeCustom(rootCmd, "./doc/", prepender, linkHandler); err != nil {
  150. return fmt.Errorf("Failed to generate cobra doc: %s", err)
  151. }
  152. return nil
  153. },
  154. }
  155. rootCmd.AddCommand(cmdDocGen)
  156. /*usage*/
  157. var cmdVersion = &cobra.Command{
  158. Use: "version",
  159. Short: "Display version",
  160. Args: cobra.ExactArgs(0),
  161. DisableAutoGenTag: true,
  162. Run: func(cmd *cobra.Command, args []string) {
  163. cwversion.Show()
  164. },
  165. }
  166. rootCmd.AddCommand(cmdVersion)
  167. rootCmd.PersistentFlags().StringVarP(&ConfigFilePath, "config", "c", csconfig.DefaultConfigPath("config.yaml"), "path to crowdsec config file")
  168. rootCmd.PersistentFlags().StringVarP(&OutputFormat, "output", "o", "", "Output format: human, json, raw")
  169. rootCmd.PersistentFlags().StringVarP(&OutputColor, "color", "", "auto", "Output color: yes, no, auto")
  170. rootCmd.PersistentFlags().BoolVar(&dbg_lvl, "debug", false, "Set logging to debug")
  171. rootCmd.PersistentFlags().BoolVar(&nfo_lvl, "info", false, "Set logging to info")
  172. rootCmd.PersistentFlags().BoolVar(&wrn_lvl, "warning", false, "Set logging to warning")
  173. rootCmd.PersistentFlags().BoolVar(&err_lvl, "error", false, "Set logging to error")
  174. rootCmd.PersistentFlags().BoolVar(&trace_lvl, "trace", false, "Set logging to trace")
  175. rootCmd.PersistentFlags().StringVar(&cwhub.HubBranch, "branch", "", "Override hub branch on github")
  176. if err := rootCmd.PersistentFlags().MarkHidden("branch"); err != nil {
  177. log.Fatalf("failed to hide flag: %s", err)
  178. }
  179. // Look for "-c /path/to/config.yaml"
  180. // This duplicates the logic in cobra, but we need to do it before
  181. // because feature flags can change which subcommands are available.
  182. for i, arg := range os.Args {
  183. if arg == "-c" || arg == "--config" {
  184. if len(os.Args) > i+1 {
  185. ConfigFilePath = os.Args[i+1]
  186. }
  187. }
  188. }
  189. if err := csconfig.LoadFeatureFlagsFile(ConfigFilePath, log.StandardLogger()); err != nil {
  190. log.Fatal(err)
  191. }
  192. if len(os.Args) > 1 {
  193. cobra.OnInitialize(initConfig)
  194. }
  195. /*don't sort flags so we can enforce order*/
  196. rootCmd.Flags().SortFlags = false
  197. rootCmd.PersistentFlags().SortFlags = false
  198. rootCmd.AddCommand(NewConfigCmd())
  199. rootCmd.AddCommand(NewHubCmd())
  200. rootCmd.AddCommand(NewMetricsCmd())
  201. rootCmd.AddCommand(NewDashboardCmd())
  202. rootCmd.AddCommand(NewDecisionsCmd())
  203. rootCmd.AddCommand(NewAlertsCmd())
  204. rootCmd.AddCommand(NewSimulationCmds())
  205. rootCmd.AddCommand(NewBouncersCmd())
  206. rootCmd.AddCommand(NewMachinesCmd())
  207. rootCmd.AddCommand(NewCapiCmd())
  208. rootCmd.AddCommand(NewLapiCmd())
  209. rootCmd.AddCommand(NewCompletionCmd())
  210. rootCmd.AddCommand(NewConsoleCmd())
  211. rootCmd.AddCommand(NewExplainCmd())
  212. rootCmd.AddCommand(NewHubTestCmd())
  213. rootCmd.AddCommand(NewNotificationsCmd())
  214. rootCmd.AddCommand(NewSupportCmd())
  215. rootCmd.AddCommand(NewItemsCmd("collections"))
  216. rootCmd.AddCommand(NewItemsCmd("parsers"))
  217. rootCmd.AddCommand(NewItemsCmd("scenarios"))
  218. rootCmd.AddCommand(NewItemsCmd("postoverflows"))
  219. if fflag.CscliSetup.IsEnabled() {
  220. rootCmd.AddCommand(NewSetupCmd())
  221. }
  222. if fflag.PapiClient.IsEnabled() {
  223. rootCmd.AddCommand(NewPapiCmd())
  224. }
  225. if err := rootCmd.Execute(); err != nil {
  226. log.Fatal(err)
  227. }
  228. }