kubernetes visitor_test 源码

  • 2022-09-18
  • 浏览 (234)

kubernetes visitor_test 代码

文件路径:/staging/src/k8s.io/cli-runtime/pkg/resource/visitor_test.go

/*
Copyright 2016 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 resource

import (
	"bytes"
	"errors"
	"fmt"
	"io"
	"io/fs"
	"io/ioutil"
	"os"
	"path/filepath"
	"strings"
	"testing"
	"time"

	"github.com/davecgh/go-spew/spew"
	"github.com/stretchr/testify/assert"
)

func TestVisitorHttpGet(t *testing.T) {
	type httpArgs struct {
		duration time.Duration
		u        string
		attempts int
	}

	i := 0
	tests := []struct {
		name        string
		httpRetries httpget
		args        httpArgs
		expectedErr error
		actualBytes io.ReadCloser
		actualErr   error
		count       int
		isNotNil    bool
	}{
		{
			name: "Test retries on errors",
			httpRetries: func(url string) (int, string, io.ReadCloser, error) {
				assert.Equal(t, "hello", url)
				i++
				if i > 2 {
					return 0, "", nil, fmt.Errorf("Failed to get http")
				}
				return 0, "", nil, fmt.Errorf("Unexpected error")

			},
			expectedErr: fmt.Errorf("Failed to get http"),
			args: httpArgs{
				duration: 0,
				u:        "hello",
				attempts: 3,
			},
			count: 3,
		},
		{
			name: "Test that 500s are retried",
			httpRetries: func(url string) (int, string, io.ReadCloser, error) {
				assert.Equal(t, "hello", url)
				i++
				return 501, "Status", ioutil.NopCloser(new(bytes.Buffer)), nil
			},
			args: httpArgs{
				duration: 0,
				u:        "hello",
				attempts: 3,
			},
			count: 3,
		},
		{
			name: "Test that 300s are not retried",
			httpRetries: func(url string) (int, string, io.ReadCloser, error) {
				assert.Equal(t, "hello", url)
				i++
				return 300, "Status", ioutil.NopCloser(new(bytes.Buffer)), nil

			},
			args: httpArgs{
				duration: 0,
				u:        "hello",
				attempts: 3,
			},
			count: 1,
		},
		{
			name: "Test attempt count is respected",
			httpRetries: func(url string) (int, string, io.ReadCloser, error) {
				assert.Equal(t, "hello", url)
				i++
				return 501, "Status", ioutil.NopCloser(new(bytes.Buffer)), nil

			},
			args: httpArgs{
				duration: 0,
				u:        "hello",
				attempts: 1,
			},
			count: 1,
		},
		{
			name: "Test attempts less than 1 results in an error",
			httpRetries: func(url string) (int, string, io.ReadCloser, error) {
				return 200, "Status", ioutil.NopCloser(new(bytes.Buffer)), nil

			},
			args: httpArgs{
				duration: 0,
				u:        "hello",
				attempts: 0,
			},
			count: 0,
		},
		{
			name: "Test Success",
			httpRetries: func(url string) (int, string, io.ReadCloser, error) {
				assert.Equal(t, "hello", url)
				i++
				if i > 1 {
					return 200, "Status", ioutil.NopCloser(new(bytes.Buffer)), nil
				}
				return 501, "Status", ioutil.NopCloser(new(bytes.Buffer)), nil

			},
			args: httpArgs{
				duration: 0,
				u:        "hello",
				attempts: 3,
			},
			count:    2,
			isNotNil: true,
		},
	}

	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			i = 0
			actualBytes, actualErr := readHttpWithRetries(tt.httpRetries, tt.args.duration, tt.args.u, tt.args.attempts)

			if tt.isNotNil {
				assert.Nil(t, actualErr)
				assert.NotNil(t, actualBytes)
			} else {
				if tt.expectedErr != nil {
					assert.Equal(t, tt.expectedErr, actualErr)
				} else {
					assert.Error(t, actualErr)
				}
				assert.Nil(t, actualBytes)
			}

			assert.Equal(t, tt.count, i)
		})
	}
}

func TestFlattenListVisitor(t *testing.T) {
	b := newDefaultBuilder().
		FilenameParam(false, &FilenameOptions{Recursive: false, Filenames: []string{"../../artifacts/deeply-nested.yaml"}}).
		Flatten()

	test := &testVisitor{}

	err := b.Do().Visit(test.Handle)
	if err != nil {
		t.Fatal(err)
	}
	if len(test.Infos) != 6 {
		t.Fatal(spew.Sdump(test.Infos))
	}
}

func TestFlattenListVisitorWithVisitorError(t *testing.T) {
	b := newDefaultBuilder().
		FilenameParam(false, &FilenameOptions{Recursive: false, Filenames: []string{"../../artifacts/deeply-nested.yaml"}}).
		Flatten()

	test := &testVisitor{InjectErr: errors.New("visitor error")}
	err := b.Do().Visit(test.Handle)
	if err == nil || !strings.Contains(err.Error(), "visitor error") {
		t.Fatal(err)
	}
	if len(test.Infos) != 6 {
		t.Fatal(spew.Sdump(test.Infos))
	}
}

func TestExpandPathsToFileVisitors(t *testing.T) {
	// Define a directory structure that will be used for testing and create empty files
	testDir := t.TempDir()
	filePaths := []string{
		filepath.Join(testDir, "0", "10.yaml"),
		filepath.Join(testDir, "0", "a", "10.yaml"),
		filepath.Join(testDir, "02.yaml"),
		filepath.Join(testDir, "10.yaml"),
		filepath.Join(testDir, "2.yaml"),
		filepath.Join(testDir, "AB.yaml"),
		filepath.Join(testDir, "a", "a.yaml"),
		filepath.Join(testDir, "a", "b.json"),
		filepath.Join(testDir, "a.yaml"),
		filepath.Join(testDir, "aa.yaml"),
		filepath.Join(testDir, "b.yml"),
	}
	for _, fp := range filePaths {
		if err := os.MkdirAll(filepath.Dir(fp), 0700); err != nil {
			t.Fatal(err)
		}
		func() {
			f, err := os.Create(fp)
			if err != nil {
				t.Fatal(err)
			}
			defer f.Close()
		}()
	}

	// Define and execute test cases
	tests := []struct {
		name            string
		path            string
		recursive       bool
		fileExtensions  []string
		expectedPaths   []string
		expectPathError bool
	}{
		{
			name:           "Recursive with default file extensions",
			path:           testDir,
			recursive:      true,
			fileExtensions: FileExtensions,
			expectedPaths: []string{
				filepath.Join(testDir, "0", "10.yaml"),
				filepath.Join(testDir, "0", "a", "10.yaml"),
				filepath.Join(testDir, "02.yaml"),
				filepath.Join(testDir, "10.yaml"),
				filepath.Join(testDir, "2.yaml"),
				filepath.Join(testDir, "AB.yaml"),
				filepath.Join(testDir, "a", "a.yaml"),
				filepath.Join(testDir, "a", "b.json"),
				filepath.Join(testDir, "a.yaml"),
				filepath.Join(testDir, "aa.yaml"),
				filepath.Join(testDir, "b.yml"),
			},
		},
		{
			name:           "Non-recursive with default file extensions",
			path:           testDir,
			fileExtensions: FileExtensions,
			expectedPaths: []string{
				filepath.Join(testDir, "02.yaml"),
				filepath.Join(testDir, "10.yaml"),
				filepath.Join(testDir, "2.yaml"),
				filepath.Join(testDir, "AB.yaml"),
				filepath.Join(testDir, "a.yaml"),
				filepath.Join(testDir, "aa.yaml"),
				filepath.Join(testDir, "b.yml"),
			},
		},
		{
			name:           "Recursive with yaml file extension",
			path:           testDir,
			recursive:      true,
			fileExtensions: []string{".yaml"},
			expectedPaths: []string{
				filepath.Join(testDir, "0", "10.yaml"),
				filepath.Join(testDir, "0", "a", "10.yaml"),
				filepath.Join(testDir, "02.yaml"),
				filepath.Join(testDir, "10.yaml"),
				filepath.Join(testDir, "2.yaml"),
				filepath.Join(testDir, "AB.yaml"),
				filepath.Join(testDir, "a", "a.yaml"),
				filepath.Join(testDir, "a.yaml"),
				filepath.Join(testDir, "aa.yaml"),
			},
		},
		{
			name:           "Recursive with json and yml file extensions",
			path:           testDir,
			recursive:      true,
			fileExtensions: []string{".json", ".yml"},
			expectedPaths: []string{
				filepath.Join(testDir, "a", "b.json"),
				filepath.Join(testDir, "b.yml"),
			},
		},
		{
			name:           "Non-recursive with json and yml file extensions",
			path:           testDir,
			fileExtensions: []string{".json", ".yml"},
			expectedPaths: []string{
				filepath.Join(testDir, "b.yml"),
			},
		},
		{
			name:           "Non-existent file extensions should return nothing",
			path:           testDir,
			recursive:      true,
			fileExtensions: []string{".foo"},
			expectedPaths:  []string{},
		},
		{
			name:            "Non-existent path should return file not found error",
			path:            filepath.Join(testDir, "does", "not", "exist"),
			recursive:       true,
			fileExtensions:  []string{".foo"},
			expectedPaths:   []string{},
			expectPathError: true,
		},
		{
			name:           "Visitor for single file is returned even if extension does not match",
			path:           filepath.Join(testDir, "a.yaml"),
			recursive:      true,
			fileExtensions: []string{"foo"},
			expectedPaths: []string{
				filepath.Join(testDir, "a.yaml"),
			},
		},
	}

	for _, tt := range tests {
		t.Run(tt.name, func(t *testing.T) {
			visitors, err := ExpandPathsToFileVisitors(nil, tt.path, tt.recursive, tt.fileExtensions, nil)
			if err != nil {
				switch e := err.(type) {
				case *fs.PathError:
					if tt.expectPathError {
						// The other details of PathError are os-specific, so only assert that the error has the path
						assert.Equal(t, tt.path, e.Path)
						return
					}
				}
				t.Fatal(err)
			}

			actualPaths := []string{}
			for _, v := range visitors {
				actualPaths = append(actualPaths, v.(*FileVisitor).Path)
			}
			assert.Equal(t, tt.expectedPaths, actualPaths)
		})
	}
}

相关信息

kubernetes 源码目录

相关文章

kubernetes builder 源码

kubernetes builder_example_test 源码

kubernetes builder_test 源码

kubernetes client 源码

kubernetes crd_finder 源码

kubernetes crd_finder_test 源码

kubernetes doc 源码

kubernetes fake 源码

kubernetes helper 源码

kubernetes helper_test 源码

0  赞