mirror of
				https://github.com/optim-enterprises-bv/kubernetes.git
				synced 2025-11-04 12:18:16 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			339 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			339 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
Copyright 2014 The Kubernetes Authors.
 | 
						|
 | 
						|
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 main
 | 
						|
 | 
						|
import (
 | 
						|
	"bytes"
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"strings"
 | 
						|
	"testing"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"github.com/spf13/cobra"
 | 
						|
	"github.com/stretchr/testify/assert"
 | 
						|
 | 
						|
	"k8s.io/apimachinery/pkg/util/wait"
 | 
						|
)
 | 
						|
 | 
						|
type result struct {
 | 
						|
	err    error
 | 
						|
	output string
 | 
						|
}
 | 
						|
 | 
						|
func testServer(n string) *Server {
 | 
						|
	return &Server{
 | 
						|
		SimpleUsage: n,
 | 
						|
		Long:        fmt.Sprintf("A simple server named %s", n),
 | 
						|
		Run: func(s *Server, args []string, stopCh <-chan struct{}) error {
 | 
						|
			s.hk.Printf("%s Run\n", s.Name())
 | 
						|
			return nil
 | 
						|
		},
 | 
						|
	}
 | 
						|
}
 | 
						|
func testServerError(n string) *Server {
 | 
						|
	return &Server{
 | 
						|
		SimpleUsage: n,
 | 
						|
		Long:        fmt.Sprintf("A simple server named %s that returns an error", n),
 | 
						|
		Run: func(s *Server, args []string, stopCh <-chan struct{}) error {
 | 
						|
			s.hk.Printf("%s Run\n", s.Name())
 | 
						|
			return errors.New("server returning error")
 | 
						|
		},
 | 
						|
	}
 | 
						|
}
 | 
						|
func testStopChRespectingServer(n string) *Server {
 | 
						|
	return &Server{
 | 
						|
		SimpleUsage: n,
 | 
						|
		Long:        fmt.Sprintf("A simple server named %s", n),
 | 
						|
		Run: func(s *Server, args []string, stopCh <-chan struct{}) error {
 | 
						|
			s.hk.Printf("%s Run\n", s.Name())
 | 
						|
			<-stopCh
 | 
						|
			return nil
 | 
						|
		},
 | 
						|
		RespectsStopCh: true,
 | 
						|
	}
 | 
						|
}
 | 
						|
func testStopChIgnoringServer(n string) *Server {
 | 
						|
	return &Server{
 | 
						|
		SimpleUsage: n,
 | 
						|
		Long:        fmt.Sprintf("A simple server named %s", n),
 | 
						|
		Run: func(s *Server, args []string, stopCh <-chan struct{}) error {
 | 
						|
			<-wait.NeverStop // this leaks obviously, but we don't care about one go routine more or less in test
 | 
						|
			return nil
 | 
						|
		},
 | 
						|
		RespectsStopCh: false,
 | 
						|
	}
 | 
						|
}
 | 
						|
func testStopChRespectingServerWithError(n string) *Server {
 | 
						|
	return &Server{
 | 
						|
		SimpleUsage: n,
 | 
						|
		Long:        fmt.Sprintf("A simple server named %s", n),
 | 
						|
		Run: func(s *Server, args []string, stopCh <-chan struct{}) error {
 | 
						|
			s.hk.Printf("%s Run\n", s.Name())
 | 
						|
			<-stopCh
 | 
						|
			return errors.New("server returning error")
 | 
						|
		},
 | 
						|
		RespectsStopCh: true,
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
const defaultCobraMessage = "default message from cobra command"
 | 
						|
const defaultCobraSubMessage = "default sub-message from cobra command"
 | 
						|
const cobraMessageDesc = "message to print"
 | 
						|
const cobraSubMessageDesc = "sub-message to print"
 | 
						|
 | 
						|
func testCobraCommand(n string) *Server {
 | 
						|
 | 
						|
	var cobraServer *Server
 | 
						|
	var msg string
 | 
						|
	cmd := &cobra.Command{
 | 
						|
		Use:   n,
 | 
						|
		Long:  n,
 | 
						|
		Short: n,
 | 
						|
		Run: func(cmd *cobra.Command, args []string) {
 | 
						|
			cobraServer.hk.Printf("msg: %s\n", msg)
 | 
						|
		},
 | 
						|
	}
 | 
						|
	cmd.PersistentFlags().StringVar(&msg, "msg", defaultCobraMessage, cobraMessageDesc)
 | 
						|
 | 
						|
	var subMsg string
 | 
						|
	subCmdName := "subcommand"
 | 
						|
	subCmd := &cobra.Command{
 | 
						|
		Use:   subCmdName,
 | 
						|
		Long:  subCmdName,
 | 
						|
		Short: subCmdName,
 | 
						|
		Run: func(cmd *cobra.Command, args []string) {
 | 
						|
			cobraServer.hk.Printf("submsg: %s", subMsg)
 | 
						|
		},
 | 
						|
	}
 | 
						|
	subCmd.PersistentFlags().StringVar(&subMsg, "submsg", defaultCobraSubMessage, cobraSubMessageDesc)
 | 
						|
 | 
						|
	cmd.AddCommand(subCmd)
 | 
						|
 | 
						|
	localFlags := cmd.LocalFlags()
 | 
						|
	localFlags.SetInterspersed(false)
 | 
						|
	s := &Server{
 | 
						|
		SimpleUsage: n,
 | 
						|
		Long:        fmt.Sprintf("A server named %s which uses a cobra command", n),
 | 
						|
		Run: func(s *Server, args []string, stopCh <-chan struct{}) error {
 | 
						|
			cobraServer = s
 | 
						|
			cmd.SetOutput(s.hk.Out())
 | 
						|
			cmd.SetArgs(args)
 | 
						|
			return cmd.Execute()
 | 
						|
		},
 | 
						|
		flags: localFlags,
 | 
						|
	}
 | 
						|
 | 
						|
	return s
 | 
						|
}
 | 
						|
func runFull(t *testing.T, args string, stopCh <-chan struct{}) *result {
 | 
						|
	buf := new(bytes.Buffer)
 | 
						|
	hk := HyperKube{
 | 
						|
		Name: "hyperkube",
 | 
						|
		Long: "hyperkube is an all-in-one server binary.",
 | 
						|
	}
 | 
						|
	hk.SetOut(buf)
 | 
						|
 | 
						|
	hk.AddServer(testServer("test1"))
 | 
						|
	hk.AddServer(testServer("test2"))
 | 
						|
	hk.AddServer(testServer("test3"))
 | 
						|
	hk.AddAlphaServer(testServer("testAlpha1"))
 | 
						|
	hk.AddServer(testServerError("test-error"))
 | 
						|
	hk.AddServer(testStopChIgnoringServer("test-stop-ch-ignoring"))
 | 
						|
	hk.AddServer(testStopChRespectingServer("test-stop-ch-respecting"))
 | 
						|
	hk.AddServer(testStopChRespectingServerWithError("test-error-stop-ch-respecting"))
 | 
						|
	hk.AddServer(testCobraCommand("test-cobra-command"))
 | 
						|
 | 
						|
	a := strings.Split(args, " ")
 | 
						|
	t.Logf("Running full with args: %q", a)
 | 
						|
	err := hk.Run(a, stopCh)
 | 
						|
 | 
						|
	r := &result{err, buf.String()}
 | 
						|
	t.Logf("Result err: %v, output: %q", r.err, r.output)
 | 
						|
 | 
						|
	return r
 | 
						|
}
 | 
						|
 | 
						|
func TestRun(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube test1", wait.NeverStop)
 | 
						|
	assert.Contains(t, x.output, "test1 Run")
 | 
						|
	assert.NoError(t, x.err)
 | 
						|
}
 | 
						|
 | 
						|
func TestLinkRun(t *testing.T) {
 | 
						|
	x := runFull(t, "test1", wait.NeverStop)
 | 
						|
	assert.Contains(t, x.output, "test1 Run")
 | 
						|
	assert.NoError(t, x.err)
 | 
						|
}
 | 
						|
 | 
						|
func TestTopNoArgs(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube", wait.NeverStop)
 | 
						|
	assert.EqualError(t, x.err, "no server specified")
 | 
						|
}
 | 
						|
 | 
						|
func TestBadServer(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube bad-server", wait.NeverStop)
 | 
						|
	assert.EqualError(t, x.err, "Server not found: bad-server")
 | 
						|
	assert.Contains(t, x.output, "Usage")
 | 
						|
}
 | 
						|
 | 
						|
func TestTopHelp(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube --help", wait.NeverStop)
 | 
						|
	assert.NoError(t, x.err)
 | 
						|
	assert.Contains(t, x.output, "all-in-one")
 | 
						|
	assert.Contains(t, x.output, "A simple server named test1")
 | 
						|
}
 | 
						|
 | 
						|
func TestTopFlags(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube --help test1", wait.NeverStop)
 | 
						|
	assert.NoError(t, x.err)
 | 
						|
	assert.Contains(t, x.output, "all-in-one")
 | 
						|
	assert.Contains(t, x.output, "A simple server named test1")
 | 
						|
	assert.NotContains(t, x.output, "test1 Run")
 | 
						|
}
 | 
						|
 | 
						|
func TestTopFlagsBad(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube --bad-flag", wait.NeverStop)
 | 
						|
	assert.EqualError(t, x.err, "unknown flag: --bad-flag")
 | 
						|
	assert.Contains(t, x.output, "all-in-one")
 | 
						|
	assert.Contains(t, x.output, "A simple server named test1")
 | 
						|
}
 | 
						|
 | 
						|
func TestServerHelp(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube test1 --help", wait.NeverStop)
 | 
						|
	assert.NoError(t, x.err)
 | 
						|
	assert.Contains(t, x.output, "A simple server named test1")
 | 
						|
	assert.Contains(t, x.output, "-h, --help")
 | 
						|
	assert.Contains(t, x.output, "help for hyperkube")
 | 
						|
	assert.NotContains(t, x.output, "test1 Run")
 | 
						|
}
 | 
						|
 | 
						|
func TestServerFlagsBad(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube test1 --bad-flag", wait.NeverStop)
 | 
						|
	assert.EqualError(t, x.err, "unknown flag: --bad-flag")
 | 
						|
	assert.Contains(t, x.output, "A simple server named test1")
 | 
						|
	assert.Contains(t, x.output, "-h, --help")
 | 
						|
	assert.Contains(t, x.output, "help for hyperkube")
 | 
						|
	assert.NotContains(t, x.output, "test1 Run")
 | 
						|
}
 | 
						|
 | 
						|
func TestServerError(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube test-error", wait.NeverStop)
 | 
						|
	assert.Contains(t, x.output, "test-error Run")
 | 
						|
	assert.EqualError(t, x.err, "server returning error")
 | 
						|
}
 | 
						|
 | 
						|
func TestAlphaRun(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube alpha testAlpha1", wait.NeverStop)
 | 
						|
	assert.NoError(t, x.err)
 | 
						|
}
 | 
						|
 | 
						|
func TestAlphaNoArgs(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube alpha", wait.NeverStop)
 | 
						|
	assert.EqualError(t, x.err, "no alpha server specified")
 | 
						|
}
 | 
						|
 | 
						|
func TestAlphaBadServer(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube alpha bad-server", wait.NeverStop)
 | 
						|
	assert.EqualError(t, x.err, "Server not found: bad-server")
 | 
						|
	assert.Contains(t, x.output, "Usage")
 | 
						|
}
 | 
						|
 | 
						|
func TestStopChIgnoringServer(t *testing.T) {
 | 
						|
	stopCh := make(chan struct{})
 | 
						|
	returnedCh := make(chan struct{})
 | 
						|
	var x *result
 | 
						|
	go func() {
 | 
						|
		defer close(returnedCh)
 | 
						|
		x = runFull(t, "hyperkube test-stop-ch-ignoring", stopCh)
 | 
						|
	}()
 | 
						|
	close(stopCh)
 | 
						|
	select {
 | 
						|
	case <-time.After(wait.ForeverTestTimeout):
 | 
						|
		t.Fatalf("%q never returned after stopCh was closed", "hyperkube test-stop-ch-ignoring")
 | 
						|
	case <-returnedCh:
 | 
						|
	}
 | 
						|
	// we cannot be sure that the server had a chance to output anything
 | 
						|
	// assert.Contains(t, x.output, "test-error-stop-ch-ignoring Run")
 | 
						|
	assert.EqualError(t, x.err, "interrupted")
 | 
						|
}
 | 
						|
 | 
						|
func TestStopChRespectingServer(t *testing.T) {
 | 
						|
	stopCh := make(chan struct{})
 | 
						|
	returnedCh := make(chan struct{})
 | 
						|
	var x *result
 | 
						|
	go func() {
 | 
						|
		defer close(returnedCh)
 | 
						|
		x = runFull(t, "hyperkube test-stop-ch-respecting", stopCh)
 | 
						|
	}()
 | 
						|
	close(stopCh)
 | 
						|
	select {
 | 
						|
	case <-time.After(wait.ForeverTestTimeout):
 | 
						|
		t.Fatalf("%q never returned after stopCh was closed", "hyperkube test-stop-ch-respecting")
 | 
						|
	case <-returnedCh:
 | 
						|
	}
 | 
						|
	assert.Contains(t, x.output, "test-stop-ch-respecting Run")
 | 
						|
	assert.Nil(t, x.err)
 | 
						|
}
 | 
						|
 | 
						|
func TestStopChRespectingServerWithError(t *testing.T) {
 | 
						|
	stopCh := make(chan struct{})
 | 
						|
	returnedCh := make(chan struct{})
 | 
						|
	var x *result
 | 
						|
	go func() {
 | 
						|
		defer close(returnedCh)
 | 
						|
		x = runFull(t, "hyperkube test-error-stop-ch-respecting", stopCh)
 | 
						|
	}()
 | 
						|
	close(stopCh)
 | 
						|
	select {
 | 
						|
	case <-time.After(wait.ForeverTestTimeout):
 | 
						|
		t.Fatalf("%q never returned after stopCh was closed", "hyperkube test-error-stop-ch-respecting")
 | 
						|
	case <-returnedCh:
 | 
						|
	}
 | 
						|
	assert.Contains(t, x.output, "test-error-stop-ch-respecting Run")
 | 
						|
	assert.EqualError(t, x.err, "server returning error")
 | 
						|
}
 | 
						|
 | 
						|
func TestCobraCommandHelp(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube test-cobra-command --help", wait.NeverStop)
 | 
						|
	assert.NoError(t, x.err)
 | 
						|
	assert.Contains(t, x.output, "A server named test-cobra-command which uses a cobra command")
 | 
						|
	assert.Contains(t, x.output, cobraMessageDesc)
 | 
						|
}
 | 
						|
func TestCobraCommandDefaultMessage(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube test-cobra-command", wait.NeverStop)
 | 
						|
	assert.Contains(t, x.output, fmt.Sprintf("msg: %s", defaultCobraMessage))
 | 
						|
}
 | 
						|
func TestCobraCommandMessage(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube test-cobra-command --msg foobar", wait.NeverStop)
 | 
						|
	assert.Contains(t, x.output, "msg: foobar")
 | 
						|
}
 | 
						|
 | 
						|
func TestCobraSubCommandHelp(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube test-cobra-command subcommand --help", wait.NeverStop)
 | 
						|
	assert.NoError(t, x.err)
 | 
						|
	assert.Contains(t, x.output, cobraSubMessageDesc)
 | 
						|
}
 | 
						|
func TestCobraSubCommandDefaultMessage(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube test-cobra-command subcommand", wait.NeverStop)
 | 
						|
	assert.Contains(t, x.output, fmt.Sprintf("submsg: %s", defaultCobraSubMessage))
 | 
						|
}
 | 
						|
func TestCobraSubCommandMessage(t *testing.T) {
 | 
						|
	x := runFull(t, "hyperkube test-cobra-command subcommand --submsg foobar", wait.NeverStop)
 | 
						|
	assert.Contains(t, x.output, "submsg: foobar")
 | 
						|
}
 |