Rectify kubectl error output
This commit is contained in:
		
				
					committed by
					
						
						Dr. Stefan Schimanski
					
				
			
			
				
	
			
			
			
						parent
						
							ce7f003f57
						
					
				
				
					commit
					6dcb0c9130
				
			@@ -46,8 +46,8 @@ import (
 | 
				
			|||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func initTestErrorHandler(t *testing.T) {
 | 
					func initTestErrorHandler(t *testing.T) {
 | 
				
			||||||
	cmdutil.BehaviorOnFatal(func(str string) {
 | 
						cmdutil.BehaviorOnFatal(func(str string, code int) {
 | 
				
			||||||
		t.Errorf("Error running command: %s", str)
 | 
							t.Errorf("Error running command (exit code %d): %s", code, str)
 | 
				
			||||||
	})
 | 
						})
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -177,7 +177,7 @@ func TestCordon(t *testing.T) {
 | 
				
			|||||||
				// Restore cmdutil behavior
 | 
									// Restore cmdutil behavior
 | 
				
			||||||
				cmdutil.DefaultBehaviorOnFatal()
 | 
									cmdutil.DefaultBehaviorOnFatal()
 | 
				
			||||||
			}()
 | 
								}()
 | 
				
			||||||
			cmdutil.BehaviorOnFatal(func(e string) { saw_fatal = true; panic(e) })
 | 
								cmdutil.BehaviorOnFatal(func(e string, code int) { saw_fatal = true; panic(e) })
 | 
				
			||||||
			cmd.SetArgs([]string{test.arg})
 | 
								cmd.SetArgs([]string{test.arg})
 | 
				
			||||||
			cmd.Execute()
 | 
								cmd.Execute()
 | 
				
			||||||
		}()
 | 
							}()
 | 
				
			||||||
@@ -521,7 +521,7 @@ func TestDrain(t *testing.T) {
 | 
				
			|||||||
				// Restore cmdutil behavior
 | 
									// Restore cmdutil behavior
 | 
				
			||||||
				cmdutil.DefaultBehaviorOnFatal()
 | 
									cmdutil.DefaultBehaviorOnFatal()
 | 
				
			||||||
			}()
 | 
								}()
 | 
				
			||||||
			cmdutil.BehaviorOnFatal(func(e string) { saw_fatal = true; panic(e) })
 | 
								cmdutil.BehaviorOnFatal(func(e string, code int) { saw_fatal = true; panic(e) })
 | 
				
			||||||
			cmd.SetArgs(test.args)
 | 
								cmd.SetArgs(test.args)
 | 
				
			||||||
			cmd.Execute()
 | 
								cmd.Execute()
 | 
				
			||||||
		}()
 | 
							}()
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -296,7 +296,7 @@ func TestTaint(t *testing.T) {
 | 
				
			|||||||
				// Restore cmdutil behavior
 | 
									// Restore cmdutil behavior
 | 
				
			||||||
				cmdutil.DefaultBehaviorOnFatal()
 | 
									cmdutil.DefaultBehaviorOnFatal()
 | 
				
			||||||
			}()
 | 
								}()
 | 
				
			||||||
			cmdutil.BehaviorOnFatal(func(e string) { saw_fatal = true; panic(e) })
 | 
								cmdutil.BehaviorOnFatal(func(e string, code int) { saw_fatal = true; panic(e) })
 | 
				
			||||||
			cmd.SetArgs(test.args)
 | 
								cmd.SetArgs(test.args)
 | 
				
			||||||
			cmd.Execute()
 | 
								cmd.Execute()
 | 
				
			||||||
		}()
 | 
							}()
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -50,6 +50,7 @@ import (
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
const (
 | 
					const (
 | 
				
			||||||
	ApplyAnnotationsFlag = "save-config"
 | 
						ApplyAnnotationsFlag = "save-config"
 | 
				
			||||||
 | 
						DefaultErrorExitCode = 1
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
type debugError interface {
 | 
					type debugError interface {
 | 
				
			||||||
@@ -74,9 +75,9 @@ func AddSourceToErr(verb string, source string, err error) error {
 | 
				
			|||||||
var fatalErrHandler = fatal
 | 
					var fatalErrHandler = fatal
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// BehaviorOnFatal allows you to override the default behavior when a fatal
 | 
					// BehaviorOnFatal allows you to override the default behavior when a fatal
 | 
				
			||||||
// error occurs, which is call os.Exit(1). You can pass 'panic' as a function
 | 
					// error occurs, which is to call os.Exit(code). You can pass 'panic' as a function
 | 
				
			||||||
// here if you prefer the panic() over os.Exit(1).
 | 
					// here if you prefer the panic() over os.Exit(1).
 | 
				
			||||||
func BehaviorOnFatal(f func(string)) {
 | 
					func BehaviorOnFatal(f func(string, int)) {
 | 
				
			||||||
	fatalErrHandler = f
 | 
						fatalErrHandler = f
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -86,9 +87,10 @@ func DefaultBehaviorOnFatal() {
 | 
				
			|||||||
	fatalErrHandler = fatal
 | 
						fatalErrHandler = fatal
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// fatal prints the message and then exits. If V(2) or greater, glog.Fatal
 | 
					// fatal prints the message if set and then exits. If V(2) or greater, glog.Fatal
 | 
				
			||||||
// is invoked for extended information.
 | 
					// is invoked for extended information.
 | 
				
			||||||
func fatal(msg string) {
 | 
					func fatal(msg string, code int) {
 | 
				
			||||||
 | 
						if len(msg) > 0 {
 | 
				
			||||||
		// add newline if needed
 | 
							// add newline if needed
 | 
				
			||||||
		if !strings.HasSuffix(msg, "\n") {
 | 
							if !strings.HasSuffix(msg, "\n") {
 | 
				
			||||||
			msg += "\n"
 | 
								msg += "\n"
 | 
				
			||||||
@@ -98,7 +100,8 @@ func fatal(msg string) {
 | 
				
			|||||||
			glog.FatalDepth(2, msg)
 | 
								glog.FatalDepth(2, msg)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		fmt.Fprint(os.Stderr, msg)
 | 
							fmt.Fprint(os.Stderr, msg)
 | 
				
			||||||
	os.Exit(1)
 | 
						}
 | 
				
			||||||
 | 
						os.Exit(code)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// CheckErr prints a user friendly error to STDERR and exits with a non-zero
 | 
					// CheckErr prints a user friendly error to STDERR and exits with a non-zero
 | 
				
			||||||
@@ -115,43 +118,39 @@ func checkErrWithPrefix(prefix string, err error) {
 | 
				
			|||||||
	checkErr(prefix, err, fatalErrHandler)
 | 
						checkErr(prefix, err, fatalErrHandler)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func checkErr(pref string, err error, handleErr func(string)) {
 | 
					// checkErr formats a given error as a string and calls the passed handleErr
 | 
				
			||||||
	if err == nil {
 | 
					// func with that string and an kubectl exit code.
 | 
				
			||||||
		return
 | 
					func checkErr(prefix string, err error, handleErr func(string, int)) {
 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if kerrors.IsInvalid(err) {
 | 
					 | 
				
			||||||
		details := err.(*kerrors.StatusError).Status().Details
 | 
					 | 
				
			||||||
		prefix := fmt.Sprintf("%sThe %s %q is invalid.\n", pref, details.Kind, details.Name)
 | 
					 | 
				
			||||||
		errs := statusCausesToAggrError(details.Causes)
 | 
					 | 
				
			||||||
		handleErr(MultilineError(prefix, errs))
 | 
					 | 
				
			||||||
		return
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if noMatch, ok := err.(*meta.NoResourceMatchError); ok {
 | 
					 | 
				
			||||||
	switch {
 | 
						switch {
 | 
				
			||||||
		case len(noMatch.PartialResource.Group) > 0 && len(noMatch.PartialResource.Version) > 0:
 | 
						case err == nil:
 | 
				
			||||||
			handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q in group %q and version %q", pref, noMatch.PartialResource.Resource, noMatch.PartialResource.Group, noMatch.PartialResource.Version))
 | 
							return
 | 
				
			||||||
		case len(noMatch.PartialResource.Group) > 0:
 | 
						case kerrors.IsInvalid(err):
 | 
				
			||||||
			handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q in group %q", pref, noMatch.PartialResource.Resource, noMatch.PartialResource.Group))
 | 
							details := err.(*kerrors.StatusError).Status().Details
 | 
				
			||||||
		case len(noMatch.PartialResource.Version) > 0:
 | 
							s := fmt.Sprintf("%sThe %s %q is invalid", prefix, details.Kind, details.Name)
 | 
				
			||||||
			handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q in version %q", pref, noMatch.PartialResource.Resource, noMatch.PartialResource.Version))
 | 
							if len(details.Causes) > 0 {
 | 
				
			||||||
 | 
								errs := statusCausesToAggrError(details.Causes)
 | 
				
			||||||
 | 
								handleErr(MultilineError(s+": ", errs), DefaultErrorExitCode)
 | 
				
			||||||
 | 
							} else {
 | 
				
			||||||
 | 
								handleErr(s, DefaultErrorExitCode)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						case clientcmd.IsConfigurationInvalid(err):
 | 
				
			||||||
 | 
							handleErr(MultilineError(fmt.Sprintf("%sError in configuration: ", prefix), err), DefaultErrorExitCode)
 | 
				
			||||||
	default:
 | 
						default:
 | 
				
			||||||
			handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q", pref, noMatch.PartialResource.Resource))
 | 
							switch err := err.(type) {
 | 
				
			||||||
 | 
							case *meta.NoResourceMatchError:
 | 
				
			||||||
 | 
								switch {
 | 
				
			||||||
 | 
								case len(err.PartialResource.Group) > 0 && len(err.PartialResource.Version) > 0:
 | 
				
			||||||
 | 
									handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q in group %q and version %q", prefix, err.PartialResource.Resource, err.PartialResource.Group, err.PartialResource.Version), DefaultErrorExitCode)
 | 
				
			||||||
 | 
								case len(err.PartialResource.Group) > 0:
 | 
				
			||||||
 | 
									handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q in group %q", prefix, err.PartialResource.Resource, err.PartialResource.Group), DefaultErrorExitCode)
 | 
				
			||||||
 | 
								case len(err.PartialResource.Version) > 0:
 | 
				
			||||||
 | 
									handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q in version %q", prefix, err.PartialResource.Resource, err.PartialResource.Version), DefaultErrorExitCode)
 | 
				
			||||||
 | 
								default:
 | 
				
			||||||
 | 
									handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q", prefix, err.PartialResource.Resource), DefaultErrorExitCode)
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
		return
 | 
							case utilerrors.Aggregate:
 | 
				
			||||||
	}
 | 
								handleErr(MultipleErrors(prefix, err.Errors()), DefaultErrorExitCode)
 | 
				
			||||||
 | 
							default: // for any other error type
 | 
				
			||||||
	// handle multiline errors
 | 
					 | 
				
			||||||
	if clientcmd.IsConfigurationInvalid(err) {
 | 
					 | 
				
			||||||
		handleErr(MultilineError(fmt.Sprintf("%sError in configuration: ", pref), err))
 | 
					 | 
				
			||||||
		return
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	if agg, ok := err.(utilerrors.Aggregate); ok && len(agg.Errors()) > 0 {
 | 
					 | 
				
			||||||
		handleErr(MultipleErrors(pref, agg.Errors()))
 | 
					 | 
				
			||||||
		return
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			msg, ok := StandardErrorMessage(err)
 | 
								msg, ok := StandardErrorMessage(err)
 | 
				
			||||||
			if !ok {
 | 
								if !ok {
 | 
				
			||||||
				msg = err.Error()
 | 
									msg = err.Error()
 | 
				
			||||||
@@ -159,7 +158,9 @@ func checkErr(pref string, err error, handleErr func(string)) {
 | 
				
			|||||||
					msg = fmt.Sprintf("error: %s", msg)
 | 
										msg = fmt.Sprintf("error: %s", msg)
 | 
				
			||||||
				}
 | 
									}
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
	handleErr(fmt.Sprintf("%s%s", pref, msg))
 | 
								handleErr(msg, DefaultErrorExitCode)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func statusCausesToAggrError(scs []unversioned.StatusCause) utilerrors.Aggregate {
 | 
					func statusCausesToAggrError(scs []unversioned.StatusCause) utilerrors.Aggregate {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -213,91 +213,78 @@ func (f *fileHandler) ServeHTTP(res http.ResponseWriter, req *http.Request) {
 | 
				
			|||||||
	res.Write(f.data)
 | 
						res.Write(f.data)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestCheckInvalidErr(t *testing.T) {
 | 
					type checkErrTestCase struct {
 | 
				
			||||||
	tests := []struct {
 | 
					 | 
				
			||||||
	err          error
 | 
						err          error
 | 
				
			||||||
		expected string
 | 
						expectedErr  string
 | 
				
			||||||
	}{
 | 
						expectedCode int
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestCheckInvalidErr(t *testing.T) {
 | 
				
			||||||
 | 
						testCheckError(t, []checkErrTestCase{
 | 
				
			||||||
		{
 | 
							{
 | 
				
			||||||
			errors.NewInvalid(api.Kind("Invalid1"), "invalidation", field.ErrorList{field.Invalid(field.NewPath("field"), "single", "details")}),
 | 
								errors.NewInvalid(api.Kind("Invalid1"), "invalidation", field.ErrorList{field.Invalid(field.NewPath("field"), "single", "details")}),
 | 
				
			||||||
			`error: The Invalid1 "invalidation" is invalid. field: Invalid value: "single": details`,
 | 
								"The Invalid1 \"invalidation\" is invalid: field: Invalid value: \"single\": details\n",
 | 
				
			||||||
 | 
								DefaultErrorExitCode,
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		{
 | 
							{
 | 
				
			||||||
			errors.NewInvalid(api.Kind("Invalid2"), "invalidation", field.ErrorList{field.Invalid(field.NewPath("field1"), "multi1", "details"), field.Invalid(field.NewPath("field2"), "multi2", "details")}),
 | 
								errors.NewInvalid(api.Kind("Invalid2"), "invalidation", field.ErrorList{field.Invalid(field.NewPath("field1"), "multi1", "details"), field.Invalid(field.NewPath("field2"), "multi2", "details")}),
 | 
				
			||||||
			`error: The Invalid2 "invalidation" is invalid. * field1: Invalid value: "multi1": details, * field2: Invalid value: "multi2": details`,
 | 
								"The Invalid2 \"invalidation\" is invalid: \n* field1: Invalid value: \"multi1\": details\n* field2: Invalid value: \"multi2\": details\n",
 | 
				
			||||||
 | 
								DefaultErrorExitCode,
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		{
 | 
							{
 | 
				
			||||||
			errors.NewInvalid(api.Kind("Invalid3"), "invalidation", field.ErrorList{}),
 | 
								errors.NewInvalid(api.Kind("Invalid3"), "invalidation", field.ErrorList{}),
 | 
				
			||||||
			`error: The Invalid3 "invalidation" is invalid. %!s(<nil>)`,
 | 
								"The Invalid3 \"invalidation\" is invalid",
 | 
				
			||||||
 | 
								DefaultErrorExitCode,
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		{
 | 
							{
 | 
				
			||||||
			errors.NewInvalid(api.Kind("Invalid4"), "invalidation", field.ErrorList{field.Invalid(field.NewPath("field4"), "multi4", "details"), field.Invalid(field.NewPath("field4"), "multi4", "details")}),
 | 
								errors.NewInvalid(api.Kind("Invalid4"), "invalidation", field.ErrorList{field.Invalid(field.NewPath("field4"), "multi4", "details"), field.Invalid(field.NewPath("field4"), "multi4", "details")}),
 | 
				
			||||||
			`error: The Invalid4 "invalidation" is invalid. field4: Invalid value: "multi4": details`,
 | 
								"The Invalid4 \"invalidation\" is invalid: field4: Invalid value: \"multi4\": details\n",
 | 
				
			||||||
 | 
								DefaultErrorExitCode,
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
	}
 | 
						})
 | 
				
			||||||
 | 
					 | 
				
			||||||
	var errReturned string
 | 
					 | 
				
			||||||
	errHandle := func(err string) {
 | 
					 | 
				
			||||||
		for _, v := range strings.Split(err, "\n") {
 | 
					 | 
				
			||||||
			separator := " "
 | 
					 | 
				
			||||||
			if errReturned == "" || v == "" {
 | 
					 | 
				
			||||||
				separator = ""
 | 
					 | 
				
			||||||
			} else if !strings.HasSuffix(errReturned, ".") {
 | 
					 | 
				
			||||||
				separator = ", "
 | 
					 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
			errReturned = fmt.Sprintf("%s%s%s", errReturned, separator, v)
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		if !strings.HasPrefix(errReturned, "error: ") {
 | 
					 | 
				
			||||||
			errReturned = fmt.Sprintf("error: %s", errReturned)
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		if strings.HasSuffix(errReturned, ", ") {
 | 
					 | 
				
			||||||
			errReturned = errReturned[:len(errReturned)-len(" ,")]
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	for _, test := range tests {
 | 
					 | 
				
			||||||
		checkErr("", test.err, errHandle)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		if errReturned != test.expected {
 | 
					 | 
				
			||||||
			t.Fatalf("Got: %s, expected: %s", errReturned, test.expected)
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
		errReturned = ""
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestCheckNoResourceMatchError(t *testing.T) {
 | 
					func TestCheckNoResourceMatchError(t *testing.T) {
 | 
				
			||||||
	tests := []struct {
 | 
						testCheckError(t, []checkErrTestCase{
 | 
				
			||||||
		err      error
 | 
					 | 
				
			||||||
		expected string
 | 
					 | 
				
			||||||
	}{
 | 
					 | 
				
			||||||
		{
 | 
							{
 | 
				
			||||||
			&meta.NoResourceMatchError{PartialResource: unversioned.GroupVersionResource{Resource: "foo"}},
 | 
								&meta.NoResourceMatchError{PartialResource: unversioned.GroupVersionResource{Resource: "foo"}},
 | 
				
			||||||
			`the server doesn't have a resource type "foo"`,
 | 
								`the server doesn't have a resource type "foo"`,
 | 
				
			||||||
 | 
								DefaultErrorExitCode,
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		{
 | 
							{
 | 
				
			||||||
			&meta.NoResourceMatchError{PartialResource: unversioned.GroupVersionResource{Version: "theversion", Resource: "foo"}},
 | 
								&meta.NoResourceMatchError{PartialResource: unversioned.GroupVersionResource{Version: "theversion", Resource: "foo"}},
 | 
				
			||||||
			`the server doesn't have a resource type "foo" in version "theversion"`,
 | 
								`the server doesn't have a resource type "foo" in version "theversion"`,
 | 
				
			||||||
 | 
								DefaultErrorExitCode,
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		{
 | 
							{
 | 
				
			||||||
			&meta.NoResourceMatchError{PartialResource: unversioned.GroupVersionResource{Group: "thegroup", Version: "theversion", Resource: "foo"}},
 | 
								&meta.NoResourceMatchError{PartialResource: unversioned.GroupVersionResource{Group: "thegroup", Version: "theversion", Resource: "foo"}},
 | 
				
			||||||
			`the server doesn't have a resource type "foo" in group "thegroup" and version "theversion"`,
 | 
								`the server doesn't have a resource type "foo" in group "thegroup" and version "theversion"`,
 | 
				
			||||||
 | 
								DefaultErrorExitCode,
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
		{
 | 
							{
 | 
				
			||||||
			&meta.NoResourceMatchError{PartialResource: unversioned.GroupVersionResource{Group: "thegroup", Resource: "foo"}},
 | 
								&meta.NoResourceMatchError{PartialResource: unversioned.GroupVersionResource{Group: "thegroup", Resource: "foo"}},
 | 
				
			||||||
			`the server doesn't have a resource type "foo" in group "thegroup"`,
 | 
								`the server doesn't have a resource type "foo" in group "thegroup"`,
 | 
				
			||||||
 | 
								DefaultErrorExitCode,
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func testCheckError(t *testing.T, tests []checkErrTestCase) {
 | 
				
			||||||
	var errReturned string
 | 
						var errReturned string
 | 
				
			||||||
	errHandle := func(err string) {
 | 
						var codeReturned int
 | 
				
			||||||
 | 
						errHandle := func(err string, code int) {
 | 
				
			||||||
		errReturned = err
 | 
							errReturned = err
 | 
				
			||||||
 | 
							codeReturned = code
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for _, test := range tests {
 | 
						for _, test := range tests {
 | 
				
			||||||
		checkErr("", test.err, errHandle)
 | 
							checkErr("", test.err, errHandle)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if errReturned != test.expected {
 | 
							if errReturned != test.expectedErr {
 | 
				
			||||||
			t.Fatalf("Got: %s, expected: %s", errReturned, test.expected)
 | 
								t.Fatalf("Got: %s, expected: %s", errReturned, test.expectedErr)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if codeReturned != test.expectedCode {
 | 
				
			||||||
 | 
								t.Fatalf("Got: %d, expected: %d", codeReturned, test.expectedCode)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -113,7 +113,7 @@ func (cmd *cmdWrapper) Output() ([]byte, error) {
 | 
				
			|||||||
func handleError(err error) error {
 | 
					func handleError(err error) error {
 | 
				
			||||||
	if ee, ok := err.(*osexec.ExitError); ok {
 | 
						if ee, ok := err.(*osexec.ExitError); ok {
 | 
				
			||||||
		// Force a compile fail if exitErrorWrapper can't convert to ExitError.
 | 
							// Force a compile fail if exitErrorWrapper can't convert to ExitError.
 | 
				
			||||||
		var x ExitError = &exitErrorWrapper{ee}
 | 
							var x ExitError = &ExitErrorWrapper{ee}
 | 
				
			||||||
		return x
 | 
							return x
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	if ee, ok := err.(*osexec.Error); ok {
 | 
						if ee, ok := err.(*osexec.Error); ok {
 | 
				
			||||||
@@ -124,14 +124,16 @@ func handleError(err error) error {
 | 
				
			|||||||
	return err
 | 
						return err
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// exitErrorWrapper is an implementation of ExitError in terms of os/exec ExitError.
 | 
					// ExitErrorWrapper is an implementation of ExitError in terms of os/exec ExitError.
 | 
				
			||||||
// Note: standard exec.ExitError is type *os.ProcessState, which already implements Exited().
 | 
					// Note: standard exec.ExitError is type *os.ProcessState, which already implements Exited().
 | 
				
			||||||
type exitErrorWrapper struct {
 | 
					type ExitErrorWrapper struct {
 | 
				
			||||||
	*osexec.ExitError
 | 
						*osexec.ExitError
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					var _ ExitError = ExitErrorWrapper{}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// ExitStatus is part of the ExitError interface.
 | 
					// ExitStatus is part of the ExitError interface.
 | 
				
			||||||
func (eew exitErrorWrapper) ExitStatus() int {
 | 
					func (eew ExitErrorWrapper) ExitStatus() int {
 | 
				
			||||||
	ws, ok := eew.Sys().(syscall.WaitStatus)
 | 
						ws, ok := eew.Sys().(syscall.WaitStatus)
 | 
				
			||||||
	if !ok {
 | 
						if !ok {
 | 
				
			||||||
		panic("can't call ExitStatus() on a non-WaitStatus exitErrorWrapper")
 | 
							panic("can't call ExitStatus() on a non-WaitStatus exitErrorWrapper")
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user