mirror of
https://github.com/ceph/ceph-csi.git
synced 2025-01-08 12:59:30 +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")
|
|
}
|