Skip to content

Commit

Permalink
Merge pull request #4 from lightstep/steps
Browse files Browse the repository at this point in the history
Refactor Checker to be more CI like
  • Loading branch information
jaronoff97 authored Jul 12, 2023
2 parents 872a6b1 + ac9c722 commit eb587e7
Show file tree
Hide file tree
Showing 54 changed files with 1,989 additions and 1,681 deletions.
204 changes: 137 additions & 67 deletions cmd/check.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@ package cmd

import (
"fmt"
"github.com/lightstep/collector-cluster-check/pkg/checks/runcrd"
"os"
"path/filepath"
"strings"
Expand All @@ -27,59 +26,109 @@ import (
"github.com/spf13/cobra"
"k8s.io/client-go/util/homedir"

"github.com/lightstep/collector-cluster-check/pkg/checks"
"github.com/lightstep/collector-cluster-check/pkg/checks/certmanager"
"github.com/lightstep/collector-cluster-check/pkg/checks/dns"
"github.com/lightstep/collector-cluster-check/pkg/checks/kubernetes"
"github.com/lightstep/collector-cluster-check/pkg/checks/lightstep"
"github.com/lightstep/collector-cluster-check/pkg/checks/oteloperator"
"github.com/lightstep/collector-cluster-check/pkg/checks/prometheus"
"github.com/lightstep/collector-cluster-check/pkg/dependencies"
"github.com/lightstep/collector-cluster-check/pkg/steps"
"github.com/lightstep/collector-cluster-check/pkg/steps/dns"
"github.com/lightstep/collector-cluster-check/pkg/steps/kubernetes"
"github.com/lightstep/collector-cluster-check/pkg/steps/metrics"
"github.com/lightstep/collector-cluster-check/pkg/steps/otel"
"github.com/lightstep/collector-cluster-check/pkg/steps/traces"
)

type checkGroup struct {
dependencies []dependencies.Initializer
checkers []checks.NewChecker
}

var (
kubeConfig string
accessToken string
endpoint string
http bool
insecure bool
availableChecks = map[string]checkGroup{
"metrics": {
dependencies: []dependencies.Initializer{dependencies.MetricInitializer},
checkers: []checks.NewChecker{lightstep.NewMetricCheck},
},
"tracing": {
dependencies: []dependencies.Initializer{dependencies.TraceInitializer},
checkers: []checks.NewChecker{lightstep.NewTraceCheck},
},
"preflight": {
dependencies: []dependencies.Initializer{dependencies.KubernetesClientInitializer, dependencies.CustomResourceClientInitializer},
checkers: []checks.NewChecker{kubernetes.NewVersionCheck, prometheus.NewCheck, certmanager.NewCheck, oteloperator.NewCheck},
},
"dns": {
dependencies: []dependencies.Initializer{},
checkers: []checks.NewChecker{dns.NewLookupCheck, dns.NewDialCheck},
},
"inflight": {
dependencies: []dependencies.Initializer{dependencies.DynamicClientInitializer, dependencies.KubernetesClientInitializer, dependencies.KubeConfigInitializer, dependencies.OtelCollectorConfigInitializer, dependencies.OtelColMetricInitializer, dependencies.OtelColTraceInitializer},
checkers: []checks.NewChecker{runcrd.NewRunCollectorCheck},
},
"all": {
dependencies: []dependencies.Initializer{dependencies.KubernetesClientInitializer, dependencies.CustomResourceClientInitializer, dependencies.MetricInitializer, dependencies.TraceInitializer},
checkers: []checks.NewChecker{dns.NewLookupCheck, dns.NewDialCheck, kubernetes.NewVersionCheck, prometheus.NewCheck, certmanager.NewCheck, oteloperator.NewCheck, lightstep.NewMetricCheck, lightstep.NewTraceCheck},
},
availableChecks = map[string]*steps.Check{
"metrics": steps.NewCheck(
"metrics",
"Initializes a meter, creates a counter, flushes metrics",
[]steps.Step{
metrics.CreateCounter{},
metrics.ShutdownMeter{},
}),
"tracing": steps.NewCheck(
"tracing",
"Initializes a trace provider, starts and finishes a trace, flushes the trace",
[]steps.Step{
traces.StartTrace{},
traces.ShutdownTracer{},
}),
"preflight": steps.NewCheck(
"preflight",
"Runs preflight checks to ensure that a collector CRD can be created",
[]steps.Step{
kubernetes.Version{},
kubernetes.NewCrdExists(steps.CertManagerCrdName),
kubernetes.NewCrdExists(steps.OtelCrdName),
kubernetes.NewCrdExists(steps.ServiceMonitorCrdName),
kubernetes.NewPodRunning(steps.OtelOperatorSelector),
kubernetes.NewPodRunning(steps.CertManagerSelector),
}),
"dns": steps.NewCheck(
"dns",
"Runs basic DNS checks to verify a connection to Lightstep from your local machine",
[]steps.Step{
dns.IPLookup{},
dns.Ping{},
dns.Dial{},
}),
"inflight": steps.NewCheck(
"inflight",
"Creates a collector, sends telemetry, queries that the telemetry was sent successfully to Lightstep",
[]steps.Step{
kubernetes.NewCrdExists(steps.OtelCrdName),
otel.CreateCollector{},
otel.PodWatcher{},
kubernetes.StartPortForward{Port: 4317, LabelSelector: steps.LabelSelector},
metrics.NewCreateCounter("localhost:4317", true),
metrics.ShutdownMeter{},
traces.NewStartTrace("localhost:4317", true),
traces.ShutdownTracer{},
kubernetes.FinishPortForward{Port: 4317, LabelSelector: steps.LabelSelector},
kubernetes.StartPortForward{Port: 8888, LabelSelector: steps.LabelSelector},
otel.QueryCollector{},
kubernetes.FinishPortForward{Port: 8888, LabelSelector: steps.LabelSelector},
otel.DeleteCollector{},
}),
"all": steps.NewCheck(
"all",
"Runs every available step",
[]steps.Step{
kubernetes.Version{},
kubernetes.NewCrdExists(steps.CertManagerCrdName),
kubernetes.NewCrdExists(steps.OtelCrdName),
kubernetes.NewCrdExists(steps.ServiceMonitorCrdName),
kubernetes.NewPodRunning(steps.OtelOperatorSelector),
kubernetes.NewPodRunning(steps.CertManagerSelector),
metrics.CreateCounter{},
metrics.ShutdownMeter{},
traces.StartTrace{},
traces.ShutdownTracer{},
dns.IPLookup{},
dns.Ping{},
dns.Dial{},
otel.CreateCollector{},
otel.PodWatcher{},
kubernetes.StartPortForward{Port: 4317, LabelSelector: steps.LabelSelector},
metrics.NewCreateCounter("localhost:4317", true),
metrics.ShutdownMeter{},
traces.NewStartTrace("localhost:4317", true),
traces.ShutdownTracer{},
kubernetes.FinishPortForward{Port: 4317, LabelSelector: steps.LabelSelector},
kubernetes.StartPortForward{Port: 8888, LabelSelector: steps.LabelSelector},
otel.QueryCollector{},
kubernetes.FinishPortForward{Port: 8888, LabelSelector: steps.LabelSelector},
otel.DeleteCollector{},
}),
}
)

// checkCmd represents the check command
var checkCmd = &cobra.Command{
Use: "check [metrics|tracing|preflight|all]",
Short: "runs one of multiple checks, use -h for more",
Use: "check [metrics|tracing|preflight|inflight|all]",
Short: "Check can run one of multiple checks, use -h for more",
Args: func(cmd *cobra.Command, args []string) error {
if len(args) == 0 {
return fmt.Errorf("must specify at least one check to run")
Expand Down Expand Up @@ -111,59 +160,51 @@ var checkCmd = &cobra.Command{
Run: func(cmd *cobra.Command, args []string) {
for _, c := range args {
group := availableChecks[c]
var depResults []*checks.Check
var results map[string]checks.CheckerResult
var opts []checks.RunnerOption
for _, d := range group.dependencies {
runnerOption, checkResult := d.Apply(cmd.Context(), endpoint, insecure, http, accessToken, kubeConfig)
depResults = append(depResults, checkResult)
if checkResult.IsFailure() {
return
}
opts = append(opts, runnerOption)
}
runner := checks.NewRunner(group.checkers, opts...)
results = runner.Run(cmd.Context())
conf := GetConfig()
deps := steps.NewDependencies()
depResults, checkResults := group.Run(cmd.Context(), deps, conf)
prettyPrintDependenciesResults(depResults)
prettyPrint(results)
prettyPrint(checkResults)
}
},
}

func prettyPrintDependenciesResults(results []*checks.Check) {
func prettyPrintDependenciesResults(checkResults []steps.Results) {
t := table.NewWriter()
rowConfigAutoMerge := table.RowConfig{AutoMerge: true}
t.AppendHeader(table.Row{"dependency", "Result", "Message", "Error"})
t.SetColumnConfigs([]table.ColumnConfig{
{Number: 1, AutoMerge: true},
})
for _, check := range results {
prettyResult := "🟩"
if !check.IsSuccess() {
prettyResult = "🟥"
for _, results := range checkResults {
for _, result := range results.Steps() {
prettyResult := "🟩"
if !result.Successful() {
prettyResult = "🟥"
}
t.AppendRow(table.Row{results.StepName(), prettyResult, result.Message(), result.Err()}, rowConfigAutoMerge)
}
t.AppendRow(table.Row{check.Name, prettyResult, check.Message, check.Error}, rowConfigAutoMerge)
}
t.SetOutputMirror(os.Stdout)
t.SetStyle(table.StyleLight)
t.Style().Options.SeparateRows = true
t.Render()
}

func prettyPrint(results map[string]checks.CheckerResult) {
func prettyPrint(checkResults []steps.Results) {
t := table.NewWriter()
rowConfigAutoMerge := table.RowConfig{AutoMerge: true}
t.AppendHeader(table.Row{"Checker", "Result", "Check Name", "Message", "Error"})
t.AppendHeader(table.Row{"Checker", "Result", "Message", "Error"})
t.SetColumnConfigs([]table.ColumnConfig{
{Number: 1, AutoMerge: true},
})
for checker, result := range results {
for _, check := range result {
for _, results := range checkResults {
for _, result := range results.Steps() {
prettyResult := "🟩"
if !check.IsSuccess() {
if !result.Successful() {
prettyResult = "🟥"
}
t.AppendRow(table.Row{checker, prettyResult, check.Name, check.Message, check.Error}, rowConfigAutoMerge)
t.AppendRow(table.Row{results.StepName(), prettyResult, result.Message(), result.Err()}, rowConfigAutoMerge)
}
}
t.SetOutputMirror(os.Stdout)
Expand All @@ -172,6 +213,16 @@ func prettyPrint(results map[string]checks.CheckerResult) {
t.Render()
}

func GetConfig() *steps.Config {
return &steps.Config{
Endpoint: endpoint,
Insecure: insecure,
Http: http,
Token: accessToken,
KubeConfig: kubeConfig,
}
}

func init() {
rootCmd.AddCommand(checkCmd)

Expand All @@ -184,6 +235,25 @@ func init() {
checkCmd.PersistentFlags().StringVarP(&endpoint, "endpoint", "", "ingest.lightstep.com:443", "destination for OTLP data")
checkCmd.PersistentFlags().BoolVarP(&http, "http", "", false, "should telemetry be sent over http")
checkCmd.PersistentFlags().BoolVarP(&insecure, "insecure", "", false, "should telemetry be sent insecurely")
checkCmd.SetHelpFunc(func(command *cobra.Command, i []string) {
// If help was called only on the base command
command.Println(checkCmd.UsageString())
if len(i) == 2 {
command.Println(command.Short)
return
} else if len(i) == 3 {
// We expect the check name to be at the first index
name := i[1]
check, ok := availableChecks[name]
if !ok {
command.Println(fmt.Sprintf("command %s not found", name))
command.Println(command.Short)
return
}
command.Println(fmt.Sprintf("----%s check description----", check.Name()))
command.Println(check.Description())
}
})

// Here you will define your flags and configuration settings.

Expand Down
66 changes: 0 additions & 66 deletions pkg/checks/certmanager/check.go

This file was deleted.

Loading

0 comments on commit eb587e7

Please sign in to comment.