asteris-llc/converge

View on GitHub
cmd/healthcheck.go

Summary

Maintainability
A
0 mins
Test Coverage
// Copyright © 2016 Asteris, LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package cmd

import (
    "errors"
    "fmt"

    log "github.com/Sirupsen/logrus"
    "github.com/asteris-llc/converge/graph"
    "github.com/asteris-llc/converge/graph/node"
    "github.com/asteris-llc/converge/helpers/logging"
    "github.com/asteris-llc/converge/rpc/pb"
    "github.com/spf13/cobra"
    "github.com/spf13/viper"
    "golang.org/x/net/context"
)

// healthcheckCmd represents the 'healthcheck' command
var healthcheckCmd = &cobra.Command{
    Use:   "healthcheck",
    Short: "display a system health check",
    Long: `Health checks determine the health status of your system.  Health
checks are similar to 'plan' but will not calculate potential deltas, and will
not display healthy checks.`,
    PreRunE: func(cmd *cobra.Command, args []string) error {
        if len(args) == 0 {
            return errors.New("Need at least one module filename as argument, got 0")
        }
        return nil
    },
    Run: func(cmd *cobra.Command, args []string) {
        // set up execution context
        ctx, cancel := context.WithCancel(context.Background())
        defer cancel()
        GracefulExit(cancel)

        // logging
        clog := log.WithField("component", "client")
        ctx = logging.WithLogger(ctx, clog)

        maybeSetToken()

        if err := maybeStartSelfHostedRPC(ctx); err != nil {
            clog.WithError(err).Fatal("could not start RPC")
        }

        client, err := getRPCExecutorClient(ctx, getSecurityConfig())
        if err != nil {
            clog.WithError(err).Fatal("could not get client")
        }

        rpcParams := getParamsRPC(cmd)

        verifyModules := viper.GetBool("verify-modules")
        if !verifyModules {
            clog.Warn("skipping module verification")
        }

        // execute files
        for _, fname := range args {
            flog := clog.WithField("file", fname)

            flog.Debug("running healthcheck")

            stream, err := client.HealthCheck(
                ctx,
                &pb.LoadRequest{
                    Location:   fname,
                    Parameters: rpcParams,
                    Verify:     verifyModules,
                },
            )
            if err != nil {
                flog.WithError(err).Fatal("error getting RPC stream")
            }

            g := graph.New()

            // get edges
            edges, err := getMeta(stream)
            if err != nil {
                flog.WithError(err).Fatal("error getting RPC metadata")
            }
            for _, edge := range edges {
                g.Connect(edge.Source, edge.Dest)
            }

            // get vertices
            err = iterateOverStream(
                stream,
                func(resp *pb.StatusResponse) {
                    slog := flog.WithFields(log.Fields{
                        "stage": resp.Stage,
                        "run":   resp.Run,
                        "id":    resp.Meta.Id,
                    })
                    if resp.Run == pb.StatusResponse_STARTED {
                        slog.Info("got status")
                    } else {
                        slog.Debug("got status")
                    }

                    if resp.Run == pb.StatusResponse_FINISHED {
                        details := resp.GetDetails()
                        if details != nil {
                            g.Add(node.New(resp.Id, details.ToPrintable()))
                        }
                    }
                },
            )
            if err != nil {
                flog.WithError(err).Fatal("could not get responses")
            }

            // validate resulting graph
            if err := g.Validate(); err != nil {
                flog.WithError(err).Warning("graph is not valid")
            }

            // print results
            out, err := healthPrinter().Show(ctx, g)
            if err != nil {
                flog.WithError(err).Fatal("failed to print results")
            }

            fmt.Print("\n")
            fmt.Print(out)
        }
    },
}

func init() {
    healthcheckCmd.Flags().Bool("quiet", false, "show only a short summary of the status")
    healthcheckCmd.Flags().Bool("verify-modules", false, "verify module signatures")
    registerRPCFlags(healthcheckCmd.Flags())
    registerLocalRPCFlags(healthcheckCmd.Flags())
    registerSSLFlags(healthcheckCmd.Flags())
    registerParamsFlags(healthcheckCmd.Flags())

    RootCmd.AddCommand(healthcheckCmd)
}