mirror of https://github.com/k3s-io/k3s
176 lines
4.9 KiB
Go
176 lines
4.9 KiB
Go
/*
|
|
Copyright 2019 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 operationexecutor
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"strconv"
|
|
"testing"
|
|
"time"
|
|
|
|
"k8s.io/kubernetes/pkg/kubelet/pluginmanager/cache"
|
|
)
|
|
|
|
const (
|
|
numPluginsToRegister = 2
|
|
numPluginsToUnregister = 2
|
|
)
|
|
|
|
var _ OperationGenerator = &fakeOperationGenerator{}
|
|
var socketDir string
|
|
|
|
func init() {
|
|
d, err := ioutil.TempDir("", "operation_executor_test")
|
|
if err != nil {
|
|
panic(fmt.Sprintf("Could not create a temp directory: %s", d))
|
|
}
|
|
socketDir = d
|
|
}
|
|
|
|
func TestOperationExecutor_RegisterPlugin_ConcurrentRegisterPlugin(t *testing.T) {
|
|
ch, quit, oe := setup()
|
|
for i := 0; i < numPluginsToRegister; i++ {
|
|
socketPath := fmt.Sprintf("%s/plugin-%d.sock", socketDir, i)
|
|
oe.RegisterPlugin(socketPath, false /* foundInDeprecatedDir */, time.Now(), nil /* plugin handlers */, nil /* actual state of the world updator */)
|
|
}
|
|
if !isOperationRunConcurrently(ch, quit, numPluginsToRegister) {
|
|
t.Fatalf("Unable to start register operations in Concurrent for plugins")
|
|
}
|
|
}
|
|
|
|
func TestOperationExecutor_RegisterPlugin_SerialRegisterPlugin(t *testing.T) {
|
|
ch, quit, oe := setup()
|
|
socketPath := fmt.Sprintf("%s/plugin-serial.sock", socketDir)
|
|
for i := 0; i < numPluginsToRegister; i++ {
|
|
oe.RegisterPlugin(socketPath, false /* foundInDeprecatedDir */, time.Now(), nil /* plugin handlers */, nil /* actual state of the world updator */)
|
|
|
|
}
|
|
if !isOperationRunSerially(ch, quit) {
|
|
t.Fatalf("Unable to start register operations serially for plugins")
|
|
}
|
|
}
|
|
|
|
func TestOperationExecutor_UnregisterPlugin_ConcurrentUnregisterPlugin(t *testing.T) {
|
|
ch, quit, oe := setup()
|
|
for i := 0; i < numPluginsToUnregister; i++ {
|
|
socketPath := "socket-path" + strconv.Itoa(i)
|
|
oe.UnregisterPlugin(socketPath, nil /* plugin handlers */, nil /* actual state of the world updator */)
|
|
|
|
}
|
|
if !isOperationRunConcurrently(ch, quit, numPluginsToUnregister) {
|
|
t.Fatalf("Unable to start unregister operations in Concurrent for plugins")
|
|
}
|
|
}
|
|
|
|
func TestOperationExecutor_UnregisterPlugin_SerialUnregisterPlugin(t *testing.T) {
|
|
ch, quit, oe := setup()
|
|
socketPath := fmt.Sprintf("%s/plugin-serial.sock", socketDir)
|
|
for i := 0; i < numPluginsToUnregister; i++ {
|
|
oe.UnregisterPlugin(socketPath, nil /* plugin handlers */, nil /* actual state of the world updator */)
|
|
|
|
}
|
|
if !isOperationRunSerially(ch, quit) {
|
|
t.Fatalf("Unable to start unregister operations serially for plugins")
|
|
}
|
|
}
|
|
|
|
type fakeOperationGenerator struct {
|
|
ch chan interface{}
|
|
quit chan interface{}
|
|
}
|
|
|
|
func newFakeOperationGenerator(ch chan interface{}, quit chan interface{}) OperationGenerator {
|
|
return &fakeOperationGenerator{
|
|
ch: ch,
|
|
quit: quit,
|
|
}
|
|
}
|
|
|
|
func (fopg *fakeOperationGenerator) GenerateRegisterPluginFunc(
|
|
socketPath string,
|
|
foundInDeprecatedDir bool,
|
|
timestamp time.Time,
|
|
pluginHandlers map[string]cache.PluginHandler,
|
|
actualStateOfWorldUpdater ActualStateOfWorldUpdater) func() error {
|
|
|
|
opFunc := func() error {
|
|
startOperationAndBlock(fopg.ch, fopg.quit)
|
|
return nil
|
|
}
|
|
return opFunc
|
|
}
|
|
|
|
func (fopg *fakeOperationGenerator) GenerateUnregisterPluginFunc(
|
|
socketPath string,
|
|
pluginHandlers map[string]cache.PluginHandler,
|
|
actualStateOfWorldUpdater ActualStateOfWorldUpdater) func() error {
|
|
opFunc := func() error {
|
|
startOperationAndBlock(fopg.ch, fopg.quit)
|
|
return nil
|
|
}
|
|
return opFunc
|
|
}
|
|
|
|
func isOperationRunSerially(ch <-chan interface{}, quit chan<- interface{}) bool {
|
|
defer close(quit)
|
|
numOperationsStarted := 0
|
|
loop:
|
|
for {
|
|
select {
|
|
case <-ch:
|
|
numOperationsStarted++
|
|
if numOperationsStarted > 1 {
|
|
return false
|
|
}
|
|
case <-time.After(5 * time.Second):
|
|
break loop
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func isOperationRunConcurrently(ch <-chan interface{}, quit chan<- interface{}, numOperationsToRun int) bool {
|
|
defer close(quit)
|
|
numOperationsStarted := 0
|
|
loop:
|
|
for {
|
|
select {
|
|
case <-ch:
|
|
numOperationsStarted++
|
|
if numOperationsStarted == numOperationsToRun {
|
|
return true
|
|
}
|
|
case <-time.After(5 * time.Second):
|
|
break loop
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func setup() (chan interface{}, chan interface{}, OperationExecutor) {
|
|
ch, quit := make(chan interface{}), make(chan interface{})
|
|
return ch, quit, NewOperationExecutor(newFakeOperationGenerator(ch, quit))
|
|
}
|
|
|
|
// This function starts by writing to ch and blocks on the quit channel
|
|
// until it is closed by the currently running test
|
|
func startOperationAndBlock(ch chan<- interface{}, quit <-chan interface{}) {
|
|
ch <- nil
|
|
<-quit
|
|
}
|