Documentation ¶
Index ¶
Constants ¶
This section is empty.
Variables ¶
View Source
var AddCanary = &cobra.Command{ Use: "canary <system.yaml>", Short: "Add a new canary spec", Run: func(cmd *cobra.Command, configFiles []string) { if ctx, err := InitContext(); err != nil { logger.Fatalf("error connecting with postgres %v", err) } else { if err := configSync.SyncCanary(ctx, dataFile, configFiles...); err != nil { logger.Fatalf("Could not sync canaries: %v", err) } } }, }
View Source
var AddTopology = &cobra.Command{ Use: "topology <system.yaml>", Short: "Add a new topology spec", Run: func(cmd *cobra.Command, configFiles []string) { if err := configSync.SyncTopology(apicontext.DefaultContext, dataFile, configFiles...); err != nil { logger.Fatalf("Could not sync topology: %v", err) } }, }
View Source
var Docs = &cobra.Command{ Use: "docs", Short: "Generate docs ", Hidden: true, Args: cobra.MinimumNArgs(0), Run: func(cmd *cobra.Command, args []string) { printAPIDocs(args) }, }
View Source
var GoOffline = &cobra.Command{ Use: "go-offline", Long: "Download all dependencies.", Run: func(cmd *cobra.Command, args []string) { if err := postgrest.GoOffline(); err != nil { logger.Fatalf("Failed to go offline: %+v", err) } databaseDir := "temp-database-dir" if err := os.Mkdir(databaseDir, 0755); err != nil { logger.Fatalf("Failed to create database directory[%s]: %+v", err) } defer os.RemoveAll(databaseDir) api.DefaultConfig.ConnectionString = "embedded://" + databaseDir _, closer, err := duty.Start("embedded-temp") runner.AddShutdownHook(closer) if err != nil { logger.Fatalf("Failed to run in embedded mode: %+v", err) } runner.ShutdownAndExit(0, "Finished downloading dependencies") }, }
View Source
var (
Operator = &cobra.Command{
Use: "operator",
Short: "Start the kubernetes operator",
Run: run,
}
)
View Source
var QueryTopology = &cobra.Command{ Use: "query", Short: "Query the topology", Args: cobra.MinimumNArgs(0), Run: func(cmd *cobra.Command, args []string) { if apicontext.DefaultContext.DB() == nil { logger.Fatalf("Database object not found, might have to specify --db or DB_URL env") } results, err := topology.Query(apicontext.DefaultContext, queryParams) if err != nil { logger.Fatalf("Failed to query topology: %v", err) } data, _ := json.MarshalIndent(results, "", " ") fmt.Println(string(data)) }, }
View Source
var Root = &cobra.Command{ Use: "canary-checker", PersistentPostRun: func(cmd *cobra.Command, args []string) { runner.Shutdown() }, PersistentPreRun: func(cmd *cobra.Command, args []string) { logger.UseSlog() canary.LogFail = logFail || logger.IsLevelEnabled(3) canary.LogPass = logPass || logger.IsLevelEnabled(4) if canary.UpstreamConf.Valid() { canary.UpstreamConf.Options = append(canary.UpstreamConf.Options, func(c *http.Client) { c.UserAgent(fmt.Sprintf("canary-checker %s", runner.Version)) }) logger.Infof("Pushing checks %s", canary.UpstreamConf) } else if partial, err := canary.UpstreamConf.IsPartiallyFilled(); partial && err != nil { logger.Warnf("Upstream not fully configured: %s", canary.UpstreamConf) } if otelcollectorURL != "" { logger.Infof("Sending traces to %s", otelcollectorURL) runner.AddShutdownHook(telemetry.InitTracer(otelServiceName, otelcollectorURL, true)) } if prometheus.PrometheusURL != "" { logger.Infof("Setting default prometheus: %s", prometheus.PrometheusURL) runner.Prometheus, _ = prometheus.NewPrometheusAPI( context.New(), connection.HTTPConnection{URL: prometheus.PrometheusURL}, ) } go func() { quit := make(chan os.Signal, 1) signal.Notify(quit, os.Interrupt) <-quit logger.Infof("Caught Ctrl+C") runner.Shutdown() }() }, }
View Source
var Run = &cobra.Command{ Use: "run <canary.yaml>", Short: "Execute checks and return", Run: func(cmd *cobra.Command, configFiles []string) { timer := timer.NewTimer() if len(configFiles) == 0 { log.Fatalln("Must specify at least one canary") } ctx, closer, err := duty.Start("canary-checker", duty.ClientOnly, duty.SkipMigrationByDefaultMode) if err != nil { logger.Fatalf("Failed to initialize db: %v", err.Error()) } runner.AddShutdownHook(closer) apicontext.DefaultContext = ctx var results = []*pkg.CheckResult{} wg := sync.WaitGroup{} queue := make(chan []*pkg.CheckResult, 1) for _, configfile := range configFiles { configs, err := pkg.ParseConfig(configfile, dataFile) if err != nil { logger.Errorf("Could not parse %s: %v", configfile, err) continue } logger.Infof("Checking %s, %d checks found", configfile, len(configs)) for _, config := range configs { if runNamespace != "" { config.Namespace = runNamespace } if config.Name == "" { config.Name = CleanupFilename(configfile) } log := logger.StandardLogger().Named(config.Name) wg.Add(1) _config := config go func() { defer wg.Done() res, err := checks.RunChecks(apicontext.New(apicontext.DefaultContext.WithName(_config.Name), _config)) if err != nil { log.Errorf("error running checks: %v", err) return } queue <- res }() } } failed := 0 passed := 0 go func() { wg.Wait() close(queue) }() for item := range queue { for _, result := range item { if !result.Pass { failed++ } else { passed++ } if result.Pass || result.ErrorObject == nil { logger.GetLogger(result.LoggerName()).Infof("%s", result.String()) } else { logger.GetLogger(result.LoggerName()).Infof("%s %+v", result.String(), result.ErrorObject) } results = append(results, result) } } if junit { report := output.GetJunitReport(results) if err := output.HandleOutput(report, outputFile); err != nil { logger.Errorf("error writing output file: %v", err) os.Exit(1) } } if csv { report, err := output.GetCSVReport(results) if err != nil { logger.Errorf("error generating CSV file: %v", err) os.Exit(1) } if err := output.HandleOutput(report, outputFile); err != nil { logger.Errorf("error writing output file: %v", err) os.Exit(1) } } if jsonExport { for _, result := range results { result.Name = def(result.Name, result.Check.GetName(), result.Canary.Name) result.Description = def(result.Description, result.Check.GetDescription()) result.Labels = merge(result.Check.GetLabels(), result.Labels) } data, err := json.Marshal(results) if err != nil { logger.Errorf("Failed to marshall json: %s", err) os.Exit(1) } _ = output.HandleOutput(string(data), outputFile) } logger.Infof("%d passed, %d failed in %s", passed, failed, timer) if failed > 0 { os.Exit(1) } }, }
View Source
var RunTopology = &cobra.Command{ Use: "run <system.yaml>", Short: "Execute topology and return", Run: func(cmd *cobra.Command, configFiles []string) { timer := timer.NewTimer() if len(configFiles) == 0 { logger.Errorf("Must specify at least one topology definition") os.Exit(1) } defer runner.Shutdown() var err error apicontext.DefaultContext, _, err = duty.Start("canary-checker", duty.ClientOnly, duty.SkipMigrationByDefaultMode) if err != nil { logger.Errorf(err.Error()) return } var results = []*pkg.Component{} wg := sync.WaitGroup{} for _, configfile := range configFiles { _configfile := configfile configs, err := pkg.ParseTopology(configfile, dataFile) if err != nil { logger.Errorf("Could not parse %s: %v", configfile, err) continue } logger.Infof("Checking %s, %d topologies found", configfile, len(configs)) for _, config := range configs { _config := config if _config.ID == uuid.Nil { _config.ID = uuid.MustParse(StaticTemplatedID) if _, err := db.PersistTopology(apicontext.DefaultContext, _config); err != nil { logger.Errorf(err.Error()) continue } } wg.Add(1) go func() { ctx, span := apicontext.DefaultContext.StartSpan("Topology") defer span.End() components, _, err := topology.Run(ctx, *_config) if err != nil { logger.Errorf("[%s] error running %v", _configfile, err) } if ctx.DB() != nil { if err := db.PersistComponents(ctx, components); err != nil { logger.Errorf("error persisting results: %v", err) } } results = append(results, components...) wg.Done() }() } } wg.Wait() logger.Infof("Checked %d systems in %v", len(results), timer) if topologyOutput != "" { data, _ := json.Marshal(results) logger.Infof("Writing results to %s", topologyOutput) if err := os.WriteFile(topologyOutput, data, 0644); err != nil { log.Fatalln(err) } } }, }
View Source
var Serve = &cobra.Command{ Use: "serve config.yaml", Short: "Start a server to execute checks", Run: func(cmd *cobra.Command, configFiles []string) { defer runner.Shutdown() canaryJobs.StartScanCanaryConfigs(setup(), dataFile, configFiles) if executor { jobs.Start() } serve() }, }
View Source
var StaticTemplatedID string = "cf38821d-434f-4496-9bbd-c0c633bb2699"
StaticTemplatedID for topologies created by CLI, to ensure that components are updated rather than duplicated
View Source
var Sync = &cobra.Command{
Use: "sync",
}
View Source
var Topology = &cobra.Command{
Use: "topology",
}
Functions ¶
func CleanupFilename ¶ added in v0.33.0
func InitContext ¶ added in v1.0.152
func ParseDocumentationFrom ¶
ParseDocumentationFrom gets all types' documentation and returns them as an array. Each type is again represented as an array (we have to use arrays as we need to be sure for the order of the fields). This function returns fields and struct definitions that have no documentation as {name, ""}.
func ServerFlags ¶ added in v0.38.4
Types ¶
type Struct ¶
type Struct struct {
Name, Doc string
StructType *ast.StructType
DocType *doc.Type
Fields StructFields
}
type StructField ¶
func NewStructField ¶
func NewStructField(field *ast.Field) StructField
func (StructField) MarkdownRow ¶ added in v0.38.32
func (f StructField) MarkdownRow(structs map[string]*Struct) string
type StructFields ¶ added in v0.38.32
type StructFields []StructField
func (StructFields) GetAllFields ¶ added in v0.38.32
func (s StructFields) GetAllFields(structs map[string]*Struct) StructFields
func (StructFields) Len ¶ added in v0.38.32
func (s StructFields) Len() int
func (StructFields) Less ¶ added in v0.38.32
func (s StructFields) Less(i, j int) bool
func (StructFields) Swap ¶ added in v0.38.32
func (s StructFields) Swap(i, j int)
Source Files ¶
Click to show internal directories.
Click to hide internal directories.