Files
Jonathan Boulle 9685f37575 *: adjust copyright header
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
2015-05-03 14:29:15 -07:00

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
}