integration: simplify CNI-fp and add README.md
* Use delegated plugin call to simplify cni-bridge-cni * Add README.md for cni-bridge-cni Signed-off-by: Wei Fu <fuweid89@gmail.com>
This commit is contained in:
		
							
								
								
									
										2
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								go.mod
									
									
									
									
									
								
							| @@ -20,6 +20,7 @@ require ( | ||||
| 	github.com/containerd/ttrpc v1.1.1-0.20220420014843-944ef4a40df3 | ||||
| 	github.com/containerd/typeurl v1.0.3-0.20220422153119-7f6e6d160d67 | ||||
| 	github.com/containerd/zfs v1.0.0 | ||||
| 	github.com/containernetworking/cni v1.1.1 | ||||
| 	github.com/containernetworking/plugins v1.1.1 | ||||
| 	github.com/coreos/go-systemd/v22 v22.3.2 | ||||
| 	github.com/davecgh/go-spew v1.1.1 | ||||
| @@ -82,7 +83,6 @@ require ( | ||||
| 	github.com/cenkalti/backoff/v4 v4.1.3 // indirect | ||||
| 	github.com/cespare/xxhash/v2 v2.1.2 // indirect | ||||
| 	github.com/cilium/ebpf v0.7.0 // indirect | ||||
| 	github.com/containernetworking/cni v1.1.1 // indirect | ||||
| 	github.com/containers/ocicrypt v1.1.3 // indirect | ||||
| 	github.com/cpuguy83/go-md2man/v2 v2.0.1 // indirect | ||||
| 	github.com/cyphar/filepath-securejoin v0.2.3 // indirect | ||||
|   | ||||
							
								
								
									
										159
									
								
								integration/failpoint/cmd/cni-bridge-fp/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										159
									
								
								integration/failpoint/cmd/cni-bridge-fp/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,159 @@ | ||||
| ## cni-bridge-f(ail)p(oint) | ||||
|  | ||||
| ### Overview | ||||
|  | ||||
| The `cni-bridge-fp` is a CNI plugin which delegates interface-creating function | ||||
| to [CNI bridge plugin][1] and allows user to inject failpoint before delegation. | ||||
|  | ||||
| Since the CNI plugin is invoked by binary call from CRI and it is short-lived, | ||||
| the failpoint need to be configured by a JSON file, which can be persisted. | ||||
| There is an example about failpoint description. | ||||
|  | ||||
| ```json | ||||
| { | ||||
| 	"cmdAdd":   "1*error(you-shall-not-pass!)->1*panic(again)", | ||||
| 	"cmdDel":   "1*error(try-again)", | ||||
| 	"cmdCheck": "10*off" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| * `cmdAdd` (string, optional): The failpoint for `ADD` command. | ||||
| * `cmdDel` (string, optional): The failpoint for `DEL` command. | ||||
| * `cmdCheck` (string, optional): The failpoint for `CHECK` command. | ||||
|  | ||||
| Since the `cmdXXX` can be multiple failpoints, each CNI binary call will update | ||||
| the current state to make sure the order of execution is expected. | ||||
|  | ||||
| And the failpoint injection is enabled by pod's annotation. Currently, the key | ||||
| of customized CNI capabilities in containerd can only be `io.kubernetes.cri.pod-annotations` | ||||
| and containerd will pass pod's annotations to CNI under the that object. The | ||||
| user can use the `failpoint.cni.containerd.io/confpath` annotation to enable | ||||
| failpoint for the pod. | ||||
|  | ||||
| ```yaml | ||||
| apiVersion: v1 | ||||
| kind: Pod | ||||
| metadata: | ||||
|   name: nginx | ||||
|   annotations: | ||||
|     failpoint.cni.containerd.io/confpath: "/tmp/pod-failpoints.json" | ||||
| spec: | ||||
|   containers: | ||||
|   - name: nginx | ||||
|     image: nginx:1.14.2 | ||||
|     ports: | ||||
|     - containerPort: 80 | ||||
| ``` | ||||
|  | ||||
| ### Example | ||||
|  | ||||
| Let's use the following json as failpoint description. | ||||
|  | ||||
| ```bash | ||||
| $ cat <<EOF | tee /tmp/cni-failpoint.json | ||||
| { | ||||
| 	"cmdAdd":   "1*error(try-again)", | ||||
| 	"cmdDel":   "2*error(oops)", | ||||
| 	"cmdCheck": "1*off->1*panic(sorry)" | ||||
| } | ||||
| EOF | ||||
| ``` | ||||
|  | ||||
| And use `ip netns` to create persisted net namespace named by `failpoint`. | ||||
|  | ||||
| ```bash | ||||
| $ sudo ip netns add failpoint | ||||
| ``` | ||||
|  | ||||
| And then setup the following bash script for demo. | ||||
|  | ||||
| ```bash | ||||
| $ cat <<EOFDEMO | tee /tmp/cni-failpoint-demo-helper.sh | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| export CNI_CONTAINERID=failpoint-testing | ||||
| export CNI_NETNS=/run/netns/failpoint | ||||
| export CNI_IFNAME=fpeni0 | ||||
| export CNI_PATH=/opt/cni/bin/ | ||||
|  | ||||
| cat <<EOF | /opt/cni/bin/cni-bridge-fp | ||||
| { | ||||
|   "cniVersion": "0.3.0", | ||||
|   "name": "containerd-net-fp", | ||||
|   "type": "cni-bridge-fp", | ||||
|   "bridge": "fp-cni0", | ||||
|   "isGateway": true, | ||||
|   "ipMasq": true, | ||||
|   "promiscMode": true, | ||||
|   "ipam": { | ||||
|     "type": "host-local", | ||||
|     "ranges": [ | ||||
|       [{ | ||||
|         "subnet": "10.88.0.0/16" | ||||
|       }], | ||||
|       [{ | ||||
|         "subnet": "2001:4860:4860::/64" | ||||
|       }] | ||||
|     ], | ||||
|     "routes": [ | ||||
|       { "dst": "0.0.0.0/0" }, | ||||
|       { "dst": "::/0" } | ||||
|     ] | ||||
|   }, | ||||
|   "runtimeConfig": { | ||||
|     "io.kubernetes.cri.pod-annotations": { | ||||
|       "failpoint.cni.containerd.io/confpath": "/tmp/cni-failpoint.json" | ||||
|     } | ||||
|   } | ||||
| } | ||||
| EOF | ||||
|  | ||||
| EOFDEMO | ||||
| ``` | ||||
|  | ||||
| Let's try to setup CNI and we should get a error `try-again`. | ||||
|  | ||||
| ```bash | ||||
| $ sudo CNI_COMMAND=ADD bash /tmp/cni-failpoint-demo-helper.sh | ||||
| { | ||||
|     "code": 999, | ||||
|     "msg": "try-again" | ||||
| } | ||||
|  | ||||
| # there is no failpoint for ADD command. | ||||
| $ cat /tmp/cni-failpoint.json | jq . | ||||
| { | ||||
|   "cmdAdd": "0*error(try-again)", | ||||
|   "cmdDel": "2*error(oops)", | ||||
|   "cmdCheck": "1*off->1*panic(sorry)" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| We should setup CNI successfully after retry. When we teardown the interface, | ||||
| there should be two failpoints. | ||||
|  | ||||
| ```bash | ||||
| $ sudo CNI_COMMAND=ADD bash /tmp/cni-failpoint-demo-helper.sh | ||||
| ... | ||||
|  | ||||
| $ sudo CNI_COMMAND=DEL bash /tmp/cni-failpoint-demo-helper.sh | ||||
| { | ||||
|     "code": 999, | ||||
|     "msg": "oops" | ||||
| } | ||||
|  | ||||
| $ sudo CNI_COMMAND=DEL bash /tmp/cni-failpoint-demo-helper.sh | ||||
| { | ||||
|     "code": 999, | ||||
|     "msg": "oops" | ||||
| } | ||||
|  | ||||
| $ cat /tmp/cni-failpoint.json | jq . | ||||
| { | ||||
|   "cmdAdd": "0*error(try-again)", | ||||
|   "cmdDel": "0*error(oops)", | ||||
|   "cmdCheck": "1*off->1*panic(sorry)" | ||||
| } | ||||
| ``` | ||||
|  | ||||
| [1]: <https://www.cni.dev/plugins/current/main/bridge/> | ||||
| @@ -1,209 +0,0 @@ | ||||
| //go:build linux | ||||
| // +build linux | ||||
|  | ||||
| /* | ||||
|    Copyright The containerd 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" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"os" | ||||
| 	"os/exec" | ||||
| 	"path/filepath" | ||||
| 	"strings" | ||||
| 	"syscall" | ||||
|  | ||||
| 	"github.com/containerd/containerd/pkg/failpoint" | ||||
| 	"github.com/containerd/continuity" | ||||
| 	"github.com/sirupsen/logrus" | ||||
| ) | ||||
|  | ||||
| type inheritedPodAnnotations struct { | ||||
| 	// CNIFailpointControlStateDir is used to specify the location of | ||||
| 	// failpoint control setting. In that such stateDir, the failpoint | ||||
| 	// setting is stored in the json file named by | ||||
| 	// `${K8S_POD_NAMESPACE}-${K8S_POD_NAME}.json`. The detail of json file | ||||
| 	// is described by FailpointConf. | ||||
| 	CNIFailpointControlStateDir string `json:"cniFailpointControlStateDir,omitempty"` | ||||
| } | ||||
|  | ||||
| // FailpointConf is used to describe cmdAdd/cmdDel/cmdCheck command's failpoint. | ||||
| type FailpointConf struct { | ||||
| 	Add   string `json:"cmdAdd"` | ||||
| 	Del   string `json:"cmdDel"` | ||||
| 	Check string `json:"cmdCheck"` | ||||
| } | ||||
|  | ||||
| type netConf struct { | ||||
| 	RuntimeConfig struct { | ||||
| 		PodAnnotations inheritedPodAnnotations `json:"io.kubernetes.cri.pod-annotations"` | ||||
| 	} `json:"runtimeConfig,omitempty"` | ||||
| } | ||||
|  | ||||
| func main() { | ||||
| 	stdinData, err := ioutil.ReadAll(os.Stdin) | ||||
| 	if err != nil { | ||||
| 		logrus.Fatalf("failed to read stdin: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	var conf netConf | ||||
| 	if err := json.Unmarshal(stdinData, &conf); err != nil { | ||||
| 		logrus.Fatalf("failed to parse network configuration: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	cniCmd, ok := os.LookupEnv("CNI_COMMAND") | ||||
| 	if !ok { | ||||
| 		logrus.Fatal("required env CNI_COMMAND") | ||||
| 	} | ||||
|  | ||||
| 	cniPath, ok := os.LookupEnv("CNI_PATH") | ||||
| 	if !ok { | ||||
| 		logrus.Fatal("required env CNI_PATH") | ||||
| 	} | ||||
|  | ||||
| 	evalFn, err := buildFailpointEval(conf.RuntimeConfig.PodAnnotations.CNIFailpointControlStateDir, cniCmd) | ||||
| 	if err != nil { | ||||
| 		logrus.Fatalf("failed to build failpoint evaluate function: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	if err := evalFn(); err != nil { | ||||
| 		logrus.Fatalf("failpoint: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	cmd := exec.Command(filepath.Join(cniPath, "bridge")) | ||||
| 	cmd.Stdin = bytes.NewReader(stdinData) | ||||
| 	cmd.Stdout = os.Stdout | ||||
| 	cmd.Stderr = os.Stderr | ||||
|  | ||||
| 	if err := cmd.Start(); err != nil { | ||||
| 		logrus.Fatalf("failed to start bridge cni plugin: %v", err) | ||||
| 	} | ||||
|  | ||||
| 	if err := cmd.Wait(); err != nil { | ||||
| 		logrus.Fatalf("failed to wait for bridge cni plugin: %v", err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // buildFailpointEval will read and update the failpoint setting and then | ||||
| // return delegated failpoint evaluate function | ||||
| func buildFailpointEval(stateDir string, cniCmd string) (failpoint.EvalFn, error) { | ||||
| 	cniArgs, ok := os.LookupEnv("CNI_ARGS") | ||||
| 	if !ok { | ||||
| 		return nopEvalFn, nil | ||||
| 	} | ||||
|  | ||||
| 	target := buildPodFailpointFilepath(stateDir, cniArgs) | ||||
| 	if target == "" { | ||||
| 		return nopEvalFn, nil | ||||
| 	} | ||||
|  | ||||
| 	f, err := os.OpenFile(target, os.O_RDWR, 0666) | ||||
| 	if err != nil { | ||||
| 		if os.IsNotExist(err) { | ||||
| 			return nopEvalFn, nil | ||||
| 		} | ||||
| 		return nil, fmt.Errorf("failed to open file %s: %w", target, err) | ||||
| 	} | ||||
| 	defer f.Close() | ||||
|  | ||||
| 	if err := syscall.Flock(int(f.Fd()), syscall.LOCK_EX); err != nil { | ||||
| 		return nil, fmt.Errorf("failed to lock failpoint setting %s: %w", target, err) | ||||
| 	} | ||||
| 	defer syscall.Flock(int(f.Fd()), syscall.LOCK_UN) | ||||
|  | ||||
| 	data, err := ioutil.ReadAll(f) | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("failed to read failpoint setting %s: %w", target, err) | ||||
| 	} | ||||
|  | ||||
| 	var conf FailpointConf | ||||
| 	if err := json.Unmarshal(data, &conf); err != nil { | ||||
| 		return nil, fmt.Errorf("failed to unmarshal failpoint conf %s: %w", string(data), err) | ||||
| 	} | ||||
|  | ||||
| 	var fpStr *string | ||||
| 	switch cniCmd { | ||||
| 	case "ADD": | ||||
| 		fpStr = &conf.Add | ||||
| 	case "DEL": | ||||
| 		fpStr = &conf.Del | ||||
| 	case "CHECK": | ||||
| 		fpStr = &conf.Check | ||||
| 	} | ||||
|  | ||||
| 	if fpStr == nil || *fpStr == "" { | ||||
| 		return nopEvalFn, nil | ||||
| 	} | ||||
|  | ||||
| 	fp, err := failpoint.NewFailpoint(cniCmd, *fpStr) | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("failed to parse failpoint %s: %w", *fpStr, err) | ||||
| 	} | ||||
|  | ||||
| 	evalFn := fp.DelegatedEval() | ||||
|  | ||||
| 	*fpStr = fp.Marshal() | ||||
|  | ||||
| 	data, err = json.Marshal(conf) | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("failed to marshal failpoint conf: %w", err) | ||||
| 	} | ||||
| 	return evalFn, continuity.AtomicWriteFile(target, data, 0666) | ||||
| } | ||||
|  | ||||
| // buildPodFailpointFilepath returns the expected failpoint setting filepath | ||||
| // by Pod metadata. | ||||
| func buildPodFailpointFilepath(stateDir, cniArgs string) string { | ||||
| 	args := cniArgsIntoKeyValue(cniArgs) | ||||
|  | ||||
| 	res := make([]string, 0, 2) | ||||
| 	for _, key := range []string{"K8S_POD_NAMESPACE", "K8S_POD_NAME"} { | ||||
| 		v, ok := args[key] | ||||
| 		if !ok { | ||||
| 			break | ||||
| 		} | ||||
| 		res = append(res, v) | ||||
| 	} | ||||
| 	if len(res) != 2 { | ||||
| 		return "" | ||||
| 	} | ||||
| 	return filepath.Join(stateDir, strings.Join(res, "-")+".json") | ||||
| } | ||||
|  | ||||
| // cniArgsIntoKeyValue converts the CNI ARGS from `key1=value1;key2=value2...` | ||||
| // into key/value hashmap. | ||||
| func cniArgsIntoKeyValue(envStr string) map[string]string { | ||||
| 	parts := strings.Split(envStr, ";") | ||||
| 	res := make(map[string]string, len(parts)) | ||||
|  | ||||
| 	for _, part := range parts { | ||||
| 		keyValue := strings.SplitN(part, "=", 2) | ||||
| 		if len(keyValue) != 2 { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		res[keyValue[0]] = keyValue[1] | ||||
| 	} | ||||
| 	return res | ||||
| } | ||||
|  | ||||
| func nopEvalFn() error { | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										202
									
								
								integration/failpoint/cmd/cni-bridge-fp/main_linux.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										202
									
								
								integration/failpoint/cmd/cni-bridge-fp/main_linux.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,202 @@ | ||||
| /* | ||||
|    Copyright The containerd 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 ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"syscall" | ||||
|  | ||||
| 	"github.com/containerd/containerd/pkg/failpoint" | ||||
| 	"github.com/containerd/continuity" | ||||
| 	"github.com/containernetworking/cni/pkg/invoke" | ||||
| 	"github.com/containernetworking/cni/pkg/skel" | ||||
| 	"github.com/containernetworking/cni/pkg/version" | ||||
| ) | ||||
|  | ||||
| const delegatedPlugin = "bridge" | ||||
|  | ||||
| type netConf struct { | ||||
| 	RuntimeConfig struct { | ||||
| 		PodAnnotations inheritedPodAnnotations `json:"io.kubernetes.cri.pod-annotations"` | ||||
| 	} `json:"runtimeConfig,omitempty"` | ||||
| } | ||||
|  | ||||
| type inheritedPodAnnotations struct { | ||||
| 	// FailpointConfPath represents filepath of failpoint settings. | ||||
| 	FailpointConfPath string `json:"failpoint.cni.containerd.io/confpath,omitempty"` | ||||
| } | ||||
|  | ||||
| // failpointConf is used to describe cmdAdd/cmdDel/cmdCheck command's failpoint. | ||||
| type failpointConf struct { | ||||
| 	Add   string `json:"cmdAdd,omitempty"` | ||||
| 	Del   string `json:"cmdDel,omitempty"` | ||||
| 	Check string `json:"cmdCheck,omitempty"` | ||||
| } | ||||
|  | ||||
| func main() { | ||||
| 	skel.PluginMain(cmdAdd, cmdCheck, cmdDel, version.All, "bridge with failpoint support") | ||||
| } | ||||
|  | ||||
| func cmdAdd(args *skel.CmdArgs) error { | ||||
| 	if err := handleFailpoint(args, "ADD"); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	result, err := invoke.DelegateAdd(context.TODO(), delegatedPlugin, args.StdinData, nil) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return result.Print() | ||||
| } | ||||
|  | ||||
| func cmdCheck(args *skel.CmdArgs) error { | ||||
| 	if err := handleFailpoint(args, "CHECK"); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	return invoke.DelegateCheck(context.TODO(), delegatedPlugin, args.StdinData, nil) | ||||
| } | ||||
|  | ||||
| func cmdDel(args *skel.CmdArgs) error { | ||||
| 	if err := handleFailpoint(args, "DEL"); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	return invoke.DelegateDel(context.TODO(), delegatedPlugin, args.StdinData, nil) | ||||
| } | ||||
|  | ||||
| func handleFailpoint(args *skel.CmdArgs, cmdKind string) error { | ||||
| 	var conf netConf | ||||
| 	if err := json.Unmarshal(args.StdinData, &conf); err != nil { | ||||
| 		return fmt.Errorf("failed to parse network configuration: %w", err) | ||||
| 	} | ||||
|  | ||||
| 	confPath := conf.RuntimeConfig.PodAnnotations.FailpointConfPath | ||||
| 	if len(confPath) == 0 { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	control, err := newFailpointControl(confPath) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	evalFn, err := control.delegatedEvalFn(cmdKind) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return evalFn() | ||||
| } | ||||
|  | ||||
| type failpointControl struct { | ||||
| 	confPath string | ||||
| } | ||||
|  | ||||
| func newFailpointControl(confPath string) (*failpointControl, error) { | ||||
| 	if !filepath.IsAbs(confPath) { | ||||
| 		return nil, fmt.Errorf("failpoint confPath(%s) is required to be absolute", confPath) | ||||
| 	} | ||||
|  | ||||
| 	return &failpointControl{ | ||||
| 		confPath: confPath, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| func (c *failpointControl) delegatedEvalFn(cmdKind string) (failpoint.EvalFn, error) { | ||||
| 	var resFn failpoint.EvalFn = nopEvalFn | ||||
|  | ||||
| 	if err := c.updateTx(func(conf *failpointConf) error { | ||||
| 		var fpStr *string | ||||
|  | ||||
| 		switch cmdKind { | ||||
| 		case "ADD": | ||||
| 			fpStr = &conf.Add | ||||
| 		case "DEL": | ||||
| 			fpStr = &conf.Del | ||||
| 		case "CHECK": | ||||
| 			fpStr = &conf.Check | ||||
| 		} | ||||
|  | ||||
| 		if fpStr == nil || *fpStr == "" { | ||||
| 			return nil | ||||
| 		} | ||||
|  | ||||
| 		fp, err := failpoint.NewFailpoint(cmdKind, *fpStr) | ||||
| 		if err != nil { | ||||
| 			return fmt.Errorf("failed to parse failpoint %s: %w", *fpStr, err) | ||||
| 		} | ||||
|  | ||||
| 		resFn = fp.DelegatedEval() | ||||
|  | ||||
| 		*fpStr = fp.Marshal() | ||||
| 		return nil | ||||
|  | ||||
| 	}); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return resFn, nil | ||||
| } | ||||
|  | ||||
| func (c *failpointControl) updateTx(updateFn func(conf *failpointConf) error) error { | ||||
| 	f, err := os.OpenFile(c.confPath, os.O_RDWR, 0666) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("failed to open confPath %s: %w", c.confPath, err) | ||||
| 	} | ||||
| 	defer f.Close() | ||||
|  | ||||
| 	if err := flock(f.Fd()); err != nil { | ||||
| 		return fmt.Errorf("failed to lock failpoint setting %s: %w", c.confPath, err) | ||||
| 	} | ||||
| 	defer unflock(f.Fd()) | ||||
|  | ||||
| 	data, err := ioutil.ReadAll(f) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("failed to read failpoint setting %s: %w", c.confPath, err) | ||||
| 	} | ||||
|  | ||||
| 	var conf failpointConf | ||||
| 	if err := json.Unmarshal(data, &conf); err != nil { | ||||
| 		return fmt.Errorf("failed to unmarshal failpoint conf %s: %w", string(data), err) | ||||
| 	} | ||||
|  | ||||
| 	if err := updateFn(&conf); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	data, err = json.Marshal(conf) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("failed to marshal failpoint conf: %w", err) | ||||
| 	} | ||||
| 	return continuity.AtomicWriteFile(c.confPath, data, 0666) | ||||
| } | ||||
|  | ||||
| func nopEvalFn() error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func flock(fd uintptr) error { | ||||
| 	return syscall.Flock(int(fd), syscall.LOCK_EX) | ||||
| } | ||||
|  | ||||
| func unflock(fd uintptr) error { | ||||
| 	return syscall.Flock(int(fd), syscall.LOCK_UN) | ||||
| } | ||||
| @@ -1,6 +1,3 @@ | ||||
| //go:build linux | ||||
| // +build linux | ||||
| 
 | ||||
| /* | ||||
|    Copyright The containerd Authors. | ||||
| 
 | ||||
| @@ -1,6 +1,3 @@ | ||||
| //go:build linux | ||||
| // +build linux | ||||
| 
 | ||||
| /* | ||||
|    Copyright The containerd Authors. | ||||
| 
 | ||||
| @@ -39,7 +39,7 @@ const ( | ||||
|  | ||||
| 	failpointShimPrefixKey = "io.containerd.runtime.v2.shim.failpoint." | ||||
|  | ||||
| 	failpointCNIStateDirKey = "cniFailpointControlStateDir" | ||||
| 	failpointCNIConfPathKey = "failpoint.cni.containerd.io/confpath" | ||||
| ) | ||||
|  | ||||
| func TestRunPodSandboxWithSetupCNIFailure(t *testing.T) { | ||||
| @@ -109,7 +109,7 @@ func injectCNIFailpoint(t *testing.T, sbConfig *criapiv1.PodSandboxConfig, conf | ||||
| 	err = os.WriteFile(fpFilename, data, 0666) | ||||
| 	require.NoError(t, err) | ||||
|  | ||||
| 	sbConfig.Annotations[failpointCNIStateDirKey] = stateDir | ||||
| 	sbConfig.Annotations[failpointCNIConfPathKey] = fpFilename | ||||
| } | ||||
|  | ||||
| func injectShimFailpoint(t *testing.T, sbConfig *criapiv1.PodSandboxConfig, methodFps map[string]string) { | ||||
|   | ||||
							
								
								
									
										294
									
								
								vendor/github.com/containernetworking/cni/pkg/skel/skel.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										294
									
								
								vendor/github.com/containernetworking/cni/pkg/skel/skel.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,294 @@ | ||||
| // Copyright 2014-2016 CNI 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 skel provides skeleton code for a CNI plugin. | ||||
| // In particular, it implements argument parsing and validation. | ||||
| package skel | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/containernetworking/cni/pkg/types" | ||||
| 	"github.com/containernetworking/cni/pkg/utils" | ||||
| 	"github.com/containernetworking/cni/pkg/version" | ||||
| ) | ||||
|  | ||||
| // CmdArgs captures all the arguments passed in to the plugin | ||||
| // via both env vars and stdin | ||||
| type CmdArgs struct { | ||||
| 	ContainerID string | ||||
| 	Netns       string | ||||
| 	IfName      string | ||||
| 	Args        string | ||||
| 	Path        string | ||||
| 	StdinData   []byte | ||||
| } | ||||
|  | ||||
| type dispatcher struct { | ||||
| 	Getenv func(string) string | ||||
| 	Stdin  io.Reader | ||||
| 	Stdout io.Writer | ||||
| 	Stderr io.Writer | ||||
|  | ||||
| 	ConfVersionDecoder version.ConfigDecoder | ||||
| 	VersionReconciler  version.Reconciler | ||||
| } | ||||
|  | ||||
| type reqForCmdEntry map[string]bool | ||||
|  | ||||
| func (t *dispatcher) getCmdArgsFromEnv() (string, *CmdArgs, *types.Error) { | ||||
| 	var cmd, contID, netns, ifName, args, path string | ||||
|  | ||||
| 	vars := []struct { | ||||
| 		name      string | ||||
| 		val       *string | ||||
| 		reqForCmd reqForCmdEntry | ||||
| 	}{ | ||||
| 		{ | ||||
| 			"CNI_COMMAND", | ||||
| 			&cmd, | ||||
| 			reqForCmdEntry{ | ||||
| 				"ADD":   true, | ||||
| 				"CHECK": true, | ||||
| 				"DEL":   true, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"CNI_CONTAINERID", | ||||
| 			&contID, | ||||
| 			reqForCmdEntry{ | ||||
| 				"ADD":   true, | ||||
| 				"CHECK": true, | ||||
| 				"DEL":   true, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"CNI_NETNS", | ||||
| 			&netns, | ||||
| 			reqForCmdEntry{ | ||||
| 				"ADD":   true, | ||||
| 				"CHECK": true, | ||||
| 				"DEL":   false, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"CNI_IFNAME", | ||||
| 			&ifName, | ||||
| 			reqForCmdEntry{ | ||||
| 				"ADD":   true, | ||||
| 				"CHECK": true, | ||||
| 				"DEL":   true, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"CNI_ARGS", | ||||
| 			&args, | ||||
| 			reqForCmdEntry{ | ||||
| 				"ADD":   false, | ||||
| 				"CHECK": false, | ||||
| 				"DEL":   false, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			"CNI_PATH", | ||||
| 			&path, | ||||
| 			reqForCmdEntry{ | ||||
| 				"ADD":   true, | ||||
| 				"CHECK": true, | ||||
| 				"DEL":   true, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	argsMissing := make([]string, 0) | ||||
| 	for _, v := range vars { | ||||
| 		*v.val = t.Getenv(v.name) | ||||
| 		if *v.val == "" { | ||||
| 			if v.reqForCmd[cmd] || v.name == "CNI_COMMAND" { | ||||
| 				argsMissing = append(argsMissing, v.name) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if len(argsMissing) > 0 { | ||||
| 		joined := strings.Join(argsMissing, ",") | ||||
| 		return "", nil, types.NewError(types.ErrInvalidEnvironmentVariables, fmt.Sprintf("required env variables [%s] missing", joined), "") | ||||
| 	} | ||||
|  | ||||
| 	if cmd == "VERSION" { | ||||
| 		t.Stdin = bytes.NewReader(nil) | ||||
| 	} | ||||
|  | ||||
| 	stdinData, err := ioutil.ReadAll(t.Stdin) | ||||
| 	if err != nil { | ||||
| 		return "", nil, types.NewError(types.ErrIOFailure, fmt.Sprintf("error reading from stdin: %v", err), "") | ||||
| 	} | ||||
|  | ||||
| 	cmdArgs := &CmdArgs{ | ||||
| 		ContainerID: contID, | ||||
| 		Netns:       netns, | ||||
| 		IfName:      ifName, | ||||
| 		Args:        args, | ||||
| 		Path:        path, | ||||
| 		StdinData:   stdinData, | ||||
| 	} | ||||
| 	return cmd, cmdArgs, nil | ||||
| } | ||||
|  | ||||
| func (t *dispatcher) checkVersionAndCall(cmdArgs *CmdArgs, pluginVersionInfo version.PluginInfo, toCall func(*CmdArgs) error) *types.Error { | ||||
| 	configVersion, err := t.ConfVersionDecoder.Decode(cmdArgs.StdinData) | ||||
| 	if err != nil { | ||||
| 		return types.NewError(types.ErrDecodingFailure, err.Error(), "") | ||||
| 	} | ||||
| 	verErr := t.VersionReconciler.Check(configVersion, pluginVersionInfo) | ||||
| 	if verErr != nil { | ||||
| 		return types.NewError(types.ErrIncompatibleCNIVersion, "incompatible CNI versions", verErr.Details()) | ||||
| 	} | ||||
|  | ||||
| 	if err = toCall(cmdArgs); err != nil { | ||||
| 		if e, ok := err.(*types.Error); ok { | ||||
| 			// don't wrap Error in Error | ||||
| 			return e | ||||
| 		} | ||||
| 		return types.NewError(types.ErrInternal, err.Error(), "") | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func validateConfig(jsonBytes []byte) *types.Error { | ||||
| 	var conf struct { | ||||
| 		Name string `json:"name"` | ||||
| 	} | ||||
| 	if err := json.Unmarshal(jsonBytes, &conf); err != nil { | ||||
| 		return types.NewError(types.ErrDecodingFailure, fmt.Sprintf("error unmarshall network config: %v", err), "") | ||||
| 	} | ||||
| 	if conf.Name == "" { | ||||
| 		return types.NewError(types.ErrInvalidNetworkConfig, "missing network name", "") | ||||
| 	} | ||||
| 	if err := utils.ValidateNetworkName(conf.Name); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (t *dispatcher) pluginMain(cmdAdd, cmdCheck, cmdDel func(_ *CmdArgs) error, versionInfo version.PluginInfo, about string) *types.Error { | ||||
| 	cmd, cmdArgs, err := t.getCmdArgsFromEnv() | ||||
| 	if err != nil { | ||||
| 		// Print the about string to stderr when no command is set | ||||
| 		if err.Code == types.ErrInvalidEnvironmentVariables && t.Getenv("CNI_COMMAND") == "" && about != "" { | ||||
| 			_, _ = fmt.Fprintln(t.Stderr, about) | ||||
| 			_, _ = fmt.Fprintf(t.Stderr, "CNI protocol versions supported: %s\n", strings.Join(versionInfo.SupportedVersions(), ", ")) | ||||
| 			return nil | ||||
| 		} | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if cmd != "VERSION" { | ||||
| 		if err = validateConfig(cmdArgs.StdinData); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		if err = utils.ValidateContainerID(cmdArgs.ContainerID); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		if err = utils.ValidateInterfaceName(cmdArgs.IfName); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	switch cmd { | ||||
| 	case "ADD": | ||||
| 		err = t.checkVersionAndCall(cmdArgs, versionInfo, cmdAdd) | ||||
| 	case "CHECK": | ||||
| 		configVersion, err := t.ConfVersionDecoder.Decode(cmdArgs.StdinData) | ||||
| 		if err != nil { | ||||
| 			return types.NewError(types.ErrDecodingFailure, err.Error(), "") | ||||
| 		} | ||||
| 		if gtet, err := version.GreaterThanOrEqualTo(configVersion, "0.4.0"); err != nil { | ||||
| 			return types.NewError(types.ErrDecodingFailure, err.Error(), "") | ||||
| 		} else if !gtet { | ||||
| 			return types.NewError(types.ErrIncompatibleCNIVersion, "config version does not allow CHECK", "") | ||||
| 		} | ||||
| 		for _, pluginVersion := range versionInfo.SupportedVersions() { | ||||
| 			gtet, err := version.GreaterThanOrEqualTo(pluginVersion, configVersion) | ||||
| 			if err != nil { | ||||
| 				return types.NewError(types.ErrDecodingFailure, err.Error(), "") | ||||
| 			} else if gtet { | ||||
| 				if err := t.checkVersionAndCall(cmdArgs, versionInfo, cmdCheck); err != nil { | ||||
| 					return err | ||||
| 				} | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 		return types.NewError(types.ErrIncompatibleCNIVersion, "plugin version does not allow CHECK", "") | ||||
| 	case "DEL": | ||||
| 		err = t.checkVersionAndCall(cmdArgs, versionInfo, cmdDel) | ||||
| 	case "VERSION": | ||||
| 		if err := versionInfo.Encode(t.Stdout); err != nil { | ||||
| 			return types.NewError(types.ErrIOFailure, err.Error(), "") | ||||
| 		} | ||||
| 	default: | ||||
| 		return types.NewError(types.ErrInvalidEnvironmentVariables, fmt.Sprintf("unknown CNI_COMMAND: %v", cmd), "") | ||||
| 	} | ||||
|  | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| // PluginMainWithError is the core "main" for a plugin. It accepts | ||||
| // callback functions for add, check, and del CNI commands and returns an error. | ||||
| // | ||||
| // The caller must also specify what CNI spec versions the plugin supports. | ||||
| // | ||||
| // It is the responsibility of the caller to check for non-nil error return. | ||||
| // | ||||
| // For a plugin to comply with the CNI spec, it must print any error to stdout | ||||
| // as JSON and then exit with nonzero status code. | ||||
| // | ||||
| // To let this package automatically handle errors and call os.Exit(1) for you, | ||||
| // use PluginMain() instead. | ||||
| func PluginMainWithError(cmdAdd, cmdCheck, cmdDel func(_ *CmdArgs) error, versionInfo version.PluginInfo, about string) *types.Error { | ||||
| 	return (&dispatcher{ | ||||
| 		Getenv: os.Getenv, | ||||
| 		Stdin:  os.Stdin, | ||||
| 		Stdout: os.Stdout, | ||||
| 		Stderr: os.Stderr, | ||||
| 	}).pluginMain(cmdAdd, cmdCheck, cmdDel, versionInfo, about) | ||||
| } | ||||
|  | ||||
| // PluginMain is the core "main" for a plugin which includes automatic error handling. | ||||
| // | ||||
| // The caller must also specify what CNI spec versions the plugin supports. | ||||
| // | ||||
| // The caller can specify an "about" string, which is printed on stderr | ||||
| // when no CNI_COMMAND is specified. The recommended output is "CNI plugin <foo> v<version>" | ||||
| // | ||||
| // When an error occurs in either cmdAdd, cmdCheck, or cmdDel, PluginMain will print the error | ||||
| // as JSON to stdout and call os.Exit(1). | ||||
| // | ||||
| // To have more control over error handling, use PluginMainWithError() instead. | ||||
| func PluginMain(cmdAdd, cmdCheck, cmdDel func(_ *CmdArgs) error, versionInfo version.PluginInfo, about string) { | ||||
| 	if e := PluginMainWithError(cmdAdd, cmdCheck, cmdDel, versionInfo, about); e != nil { | ||||
| 		if err := e.Print(); err != nil { | ||||
| 			log.Print("Error writing error JSON to stdout: ", err) | ||||
| 		} | ||||
| 		os.Exit(1) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										1
									
								
								vendor/modules.txt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								vendor/modules.txt
									
									
									
									
										vendored
									
									
								
							| @@ -132,6 +132,7 @@ github.com/containerd/zfs/plugin | ||||
| ## explicit; go 1.14 | ||||
| github.com/containernetworking/cni/libcni | ||||
| github.com/containernetworking/cni/pkg/invoke | ||||
| github.com/containernetworking/cni/pkg/skel | ||||
| github.com/containernetworking/cni/pkg/types | ||||
| github.com/containernetworking/cni/pkg/types/020 | ||||
| github.com/containernetworking/cni/pkg/types/040 | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Wei Fu
					Wei Fu