Skip to content

Commit

Permalink
feat: [sc-114709] [troubleshoot] gracefully handle inability to load …
Browse files Browse the repository at this point in the history
…specs if multiple specs provided (#1686)

gracefully handle unable to load URL spec
  • Loading branch information
nvanthao authored Nov 12, 2024
1 parent 0213fbb commit 9970fb1
Show file tree
Hide file tree
Showing 2 changed files with 65 additions and 23 deletions.
45 changes: 22 additions & 23 deletions internal/specs/specs.go
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import (
"strings"
"time"

"github.com/fatih/color"
"github.com/pkg/errors"
"github.com/replicatedhq/troubleshoot/internal/util"
"github.com/replicatedhq/troubleshoot/pkg/constants"
Expand Down Expand Up @@ -78,7 +79,7 @@ func LoadFromCLIArgs(ctx context.Context, client kubernetes.Interface, args []st
ctx = context.Background()
}

var kindsFromURL *loader.TroubleshootKinds
allURLSpecs := loader.NewTroubleshootKinds()
rawSpecs := []string{}

for _, v := range args {
Expand Down Expand Up @@ -176,34 +177,33 @@ func LoadFromCLIArgs(ctx context.Context, client kubernetes.Interface, args []st
return nil, types.NewExitCodeError(constants.EXIT_CODE_SPEC_ISSUES, err)
}

var specFromURL string
var rawURLSpec string
headers := map[string]string{}

if parsedURL.Host == "kots.io" {
// To download specs from kots.io, we need to set the User-Agent header
specFromURL, err = downloadFromHttpURL(ctx, v, map[string]string{
"User-Agent": "Replicated_Troubleshoot/v1beta1",
})
if err != nil {
return nil, err
}
} else {
specFromURL, err = downloadFromHttpURL(ctx, v, nil)
if err != nil {
return nil, err
}
headers["User-Agent"] = "Replicated_Troubleshoot/v1beta1"
}
rawURLSpec, err = downloadFromHttpURL(ctx, v, headers)
if err != nil {
fmt.Println(color.YellowString("failed to download spec from URI %q: %v\n", v, err))
continue
}

// load URL spec first to remove URI key from the spec
kindsFromURL, err = loader.LoadSpecs(ctx, loader.LoadOptions{
RawSpec: specFromURL,
urlSpec, err := loader.LoadSpecs(ctx, loader.LoadOptions{
RawSpec: rawURLSpec,
})
if err != nil {
return nil, err
fmt.Println(color.YellowString("failed to load spec from URI %q: %v\n", v, err))
continue
}
// remove URI key from the spec if any
for i := range kindsFromURL.SupportBundlesV1Beta2 {
kindsFromURL.SupportBundlesV1Beta2[i].Spec.Uri = ""
for i := range urlSpec.SupportBundlesV1Beta2 {
urlSpec.SupportBundlesV1Beta2[i].Spec.Uri = ""
}

allURLSpecs.Add(urlSpec)

}
}
}
Expand All @@ -214,8 +214,8 @@ func LoadFromCLIArgs(ctx context.Context, client kubernetes.Interface, args []st
if err != nil {
return nil, err
}
if kindsFromURL != nil {
kinds.Add(kindsFromURL)
if allURLSpecs.Len() > 0 {
kinds.Add(allURLSpecs)
}

if vp.GetBool("load-cluster-specs") {
Expand All @@ -224,8 +224,7 @@ func LoadFromCLIArgs(ctx context.Context, client kubernetes.Interface, args []st
if kinds.IsEmpty() {
return nil, types.NewExitCodeError(constants.EXIT_CODE_SPEC_ISSUES, err)
}
// TODO: Consider colour coding and graceful failures when loading specs
fmt.Printf("failed to load specs from the cluster: %v\n", err)
fmt.Println(color.YellowString("failed to load specs from cluster: %v\n", err))
} else {
kinds.Add(clusterKinds)
}
Expand Down
43 changes: 43 additions & 0 deletions internal/specs/specs_test.go
Original file line number Diff line number Diff line change
@@ -1,10 +1,13 @@
package specs

import (
"bytes"
"context"
"fmt"
"io"
"net/http"
"net/http/httptest"
"os"
"reflect"
"testing"

Expand Down Expand Up @@ -225,6 +228,46 @@ spec:
require.Len(t, specs.HostCollectorsV1Beta2, 1)
}

func TestLoadFromMultipleURIs(t *testing.T) {
server1 := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
w.Write([]byte(`apiVersion: troubleshoot.sh/v1beta2
kind: HostCollector
metadata:
name: cpu
spec:
collectors:
- cpu: {}
`))
}))
defer server1.Close()

server2 := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
http.Error(w, "Internal Server Error", http.StatusInternalServerError)
}))
// immediately close the server to force the error
server2.Close()

// Capture stdout
oldStdout := os.Stdout
r, w, _ := os.Pipe()
os.Stdout = w

client := testclient.NewSimpleClientset()
specs, err := LoadFromCLIArgs(context.Background(), client, []string{server1.URL, server2.URL}, viper.New())

// Restore stdout
w.Close()
os.Stdout = oldStdout

var buf bytes.Buffer
io.Copy(&buf, r)
output := buf.String()

require.NoError(t, err)
require.Len(t, specs.HostCollectorsV1Beta2, 1)
assert.Contains(t, output, "failed to download spec from URI")
}

func TestLoadAdditionalSpecFromURIs(t *testing.T) {
m := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
w.Write([]byte(`apiVersion: troubleshoot.sh/v1beta2
Expand Down

0 comments on commit 9970fb1

Please sign in to comment.