/* 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 cni import ( "sort" "strings" cnilibrary "github.com/containernetworking/cni/libcni" "github.com/pkg/errors" ) // Opt sets options for a CNI instance type Opt func(c *libcni) error // WithInterfacePrefix sets the prefix for network interfaces // e.g. eth or wlan func WithInterfacePrefix(prefix string) Opt { return func(c *libcni) error { c.prefix = prefix return nil } } // WithPluginDir can be used to set the locations of // the cni plugin binaries func WithPluginDir(dirs []string) Opt { return func(c *libcni) error { c.pluginDirs = dirs c.cniConfig = &cnilibrary.CNIConfig{Path: dirs} return nil } } // WithPluginConfDir can be used to configure the // cni configuration directory. func WithPluginConfDir(dir string) Opt { return func(c *libcni) error { c.pluginConfDir = dir return nil } } // WithPluginMaxConfNum can be used to configure the // max cni plugin config file num. func WithPluginMaxConfNum(max int) Opt { return func(c *libcni) error { c.pluginMaxConfNum = max return nil } } // WithMinNetworkCount can be used to configure the // minimum networks to be configured and initialized // for the status to report success. By default its 1. func WithMinNetworkCount(count int) Opt { return func(c *libcni) error { c.networkCount = count return nil } } // WithLoNetwork can be used to load the loopback // network config. func WithLoNetwork(c *libcni) error { loConfig, _ := cnilibrary.ConfListFromBytes([]byte(`{ "cniVersion": "0.3.1", "name": "cni-loopback", "plugins": [{ "type": "loopback" }] }`)) c.networks = append(c.networks, &Network{ cni: c.cniConfig, config: loConfig, ifName: "lo", }) return nil } // WithConf can be used to load config directly // from byte. func WithConf(bytes []byte) Opt { return WithConfIndex(bytes, 0) } // WithConfIndex can be used to load config directly // from byte and set the interface name's index. func WithConfIndex(bytes []byte, index int) Opt { return func(c *libcni) error { conf, err := cnilibrary.ConfFromBytes(bytes) if err != nil { return err } confList, err := cnilibrary.ConfListFromConf(conf) if err != nil { return err } c.networks = append(c.networks, &Network{ cni: c.cniConfig, config: confList, ifName: getIfName(c.prefix, index), }) return nil } } // WithConfFile can be used to load network config // from an .conf file. Supported with absolute fileName // with path only. func WithConfFile(fileName string) Opt { return func(c *libcni) error { conf, err := cnilibrary.ConfFromFile(fileName) if err != nil { return err } // upconvert to conf list confList, err := cnilibrary.ConfListFromConf(conf) if err != nil { return err } c.networks = append(c.networks, &Network{ cni: c.cniConfig, config: confList, ifName: getIfName(c.prefix, 0), }) return nil } } // WithConfListBytes can be used to load network config list directly // from byte func WithConfListBytes(bytes []byte) Opt { return func(c *libcni) error { confList, err := cnilibrary.ConfListFromBytes(bytes) if err != nil { return err } i := len(c.networks) c.networks = append(c.networks, &Network{ cni: c.cniConfig, config: confList, ifName: getIfName(c.prefix, i), }) return nil } } // WithConfListFile can be used to load network config // from an .conflist file. Supported with absolute fileName // with path only. func WithConfListFile(fileName string) Opt { return func(c *libcni) error { confList, err := cnilibrary.ConfListFromFile(fileName) if err != nil { return err } i := len(c.networks) c.networks = append(c.networks, &Network{ cni: c.cniConfig, config: confList, ifName: getIfName(c.prefix, i), }) return nil } } // WithDefaultConf can be used to detect the default network // config file from the configured cni config directory and load // it. // Since the CNI spec does not specify a way to detect default networks, // the convention chosen is - the first network configuration in the sorted // list of network conf files as the default network. func WithDefaultConf(c *libcni) error { return loadFromConfDir(c, c.pluginMaxConfNum) } // WithAllConf can be used to detect all network config // files from the configured cni config directory and load // them. func WithAllConf(c *libcni) error { return loadFromConfDir(c, 0) } // loadFromConfDir detects network config files from the // configured cni config directory and load them. max is // the maximum network config to load (max i<= 0 means no limit). func loadFromConfDir(c *libcni, max int) error { files, err := cnilibrary.ConfFiles(c.pluginConfDir, []string{".conf", ".conflist", ".json"}) switch { case err != nil: return errors.Wrapf(ErrRead, "failed to read config file: %v", err) case len(files) == 0: return errors.Wrapf(ErrCNINotInitialized, "no network config found in %s", c.pluginConfDir) } // files contains the network config files associated with cni network. // Use lexicographical way as a defined order for network config files. sort.Strings(files) // Since the CNI spec does not specify a way to detect default networks, // the convention chosen is - the first network configuration in the sorted // list of network conf files as the default network and choose the default // interface provided during init as the network interface for this default // network. For every other network use a generated interface id. i := 0 var networks []*Network for _, confFile := range files { var confList *cnilibrary.NetworkConfigList if strings.HasSuffix(confFile, ".conflist") { confList, err = cnilibrary.ConfListFromFile(confFile) if err != nil { return errors.Wrapf(ErrInvalidConfig, "failed to load CNI config list file %s: %v", confFile, err) } } else { conf, err := cnilibrary.ConfFromFile(confFile) if err != nil { return errors.Wrapf(ErrInvalidConfig, "failed to load CNI config file %s: %v", confFile, err) } // Ensure the config has a "type" so we know what plugin to run. // Also catches the case where somebody put a conflist into a conf file. if conf.Network.Type == "" { return errors.Wrapf(ErrInvalidConfig, "network type not found in %s", confFile) } confList, err = cnilibrary.ConfListFromConf(conf) if err != nil { return errors.Wrapf(ErrInvalidConfig, "failed to convert CNI config file %s to CNI config list: %v", confFile, err) } } if len(confList.Plugins) == 0 { return errors.Wrapf(ErrInvalidConfig, "CNI config list in config file %s has no networks, skipping", confFile) } networks = append(networks, &Network{ cni: c.cniConfig, config: confList, ifName: getIfName(c.prefix, i), }) i++ if i == max { break } } if len(networks) == 0 { return errors.Wrapf(ErrCNINotInitialized, "no valid networks found in %s", c.pluginDirs) } c.networks = append(c.networks, networks...) return nil }