mirror of
https://github.com/clearlinux/rkt.git
synced 2026-06-16 02:05:48 +00:00
9685f37575
Inspired by a similar change in the Kubernetes project [1], this changes the copyright header to the more generic "The rkt Authors" rather than "CoreOS, Inc.", which is only sometimes correct. https://github.com/GoogleCloudPlatform/kubernetes/commit/6b3a6e6b983f967c88d14d26542ec6e30c49ebd3
135 lines
3.6 KiB
Go
135 lines
3.6 KiB
Go
// Copyright 2015 The rkt 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 networking
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/coreos/rkt/Godeps/_workspace/src/github.com/appc/cni/pkg/plugin"
|
|
|
|
"github.com/coreos/rkt/common"
|
|
)
|
|
|
|
// TODO(eyakubovich): make this configurable in rkt.conf
|
|
const UserNetPluginsPath = "/usr/lib/rkt/plugins/net"
|
|
const BuiltinNetPluginsPath = "usr/lib/rkt/plugins/net"
|
|
|
|
func (e *podEnv) netPluginAdd(n *activeNet, netns string) (ip, hostIP net.IP, err error) {
|
|
output, err := e.execNetPlugin("ADD", n, netns)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
pr := plugin.Result{}
|
|
if err = json.Unmarshal(output, &pr); err != nil {
|
|
return nil, nil, fmt.Errorf("error parsing %q result: %v", n.conf.Name, err)
|
|
}
|
|
|
|
if pr.IP4 == nil {
|
|
return nil, nil, fmt.Errorf("net-plugin returned no IPv4 configuration")
|
|
}
|
|
|
|
return pr.IP4.IP.IP, pr.IP4.Gateway, nil
|
|
}
|
|
|
|
func (e *podEnv) netPluginDel(n *activeNet, netns string) error {
|
|
_, err := e.execNetPlugin("DEL", n, netns)
|
|
return err
|
|
}
|
|
|
|
func (e *podEnv) pluginPaths() []string {
|
|
// try 3rd-party path first
|
|
return []string{
|
|
UserNetPluginsPath,
|
|
filepath.Join(common.Stage1RootfsPath(e.podRoot), BuiltinNetPluginsPath),
|
|
}
|
|
}
|
|
|
|
func (e *podEnv) findNetPlugin(plugin string) string {
|
|
for _, p := range e.pluginPaths() {
|
|
fullname := filepath.Join(p, plugin)
|
|
if fi, err := os.Stat(fullname); err == nil && fi.Mode().IsRegular() {
|
|
return fullname
|
|
}
|
|
}
|
|
|
|
return ""
|
|
}
|
|
|
|
func envVars(vars [][2]string) []string {
|
|
env := os.Environ()
|
|
|
|
for _, kv := range vars {
|
|
env = append(env, strings.Join(kv[:], "="))
|
|
}
|
|
|
|
return env
|
|
}
|
|
|
|
func (e *podEnv) execNetPlugin(cmd string, n *activeNet, netns string) ([]byte, error) {
|
|
pluginPath := e.findNetPlugin(n.conf.Type)
|
|
if pluginPath == "" {
|
|
return nil, fmt.Errorf("Could not find plugin %q", n.conf.Type)
|
|
}
|
|
|
|
// TODO(jonboulle): This is a temporary workaround for an upstream
|
|
// issue in CNI: various plugins expect CNI_NETNS to be unique (e.g.
|
|
// veth uses it as a source of entropy, host-local uses it as an
|
|
// identifier), but rkt was passing a relative path which is identical
|
|
// for all pods. For now, let's use an absolute path, until the
|
|
// upstream issue is sorted. **This will break host-local removals**, but
|
|
// allow --private-net to work with multiple pods.
|
|
// (In future we will probably pass CNI_CONTAINERID and use that for
|
|
// uniqueness-requiring operations instead.)
|
|
// https://github.com/appc/cni/issues/5
|
|
netns, err := filepath.Abs(netns)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
vars := [][2]string{
|
|
{"CNI_COMMAND", cmd},
|
|
{"CNI_PODID", e.podID.String()},
|
|
{"CNI_NETNS", netns},
|
|
{"CNI_ARGS", n.runtime.Args},
|
|
{"CNI_IFNAME", n.runtime.IfName},
|
|
{"CNI_PATH", strings.Join(e.pluginPaths(), ":")},
|
|
}
|
|
|
|
stdin := bytes.NewBuffer(n.confBytes)
|
|
stdout := &bytes.Buffer{}
|
|
|
|
c := exec.Cmd{
|
|
Path: pluginPath,
|
|
Args: []string{pluginPath},
|
|
Env: envVars(vars),
|
|
Stdin: stdin,
|
|
Stdout: stdout,
|
|
Stderr: os.Stderr,
|
|
}
|
|
|
|
if err := c.Run(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return stdout.Bytes(), nil
|
|
}
|