/** # Copyright (c) 2022, NVIDIA CORPORATION. All rights reserved. # # 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 containerd import ( "fmt" "github.com/NVIDIA/nvidia-container-toolkit/internal/logger" "github.com/NVIDIA/nvidia-container-toolkit/pkg/config/engine" "github.com/NVIDIA/nvidia-container-toolkit/pkg/config/toml" ) // Config represents the containerd config type Config struct { *toml.Tree Logger logger.Interface RuntimeType string UseDefaultRuntimeName bool ContainerAnnotations []string } var _ engine.Interface = (*Config)(nil) type containerdCfgRuntime struct { tree *toml.Tree } var _ engine.RuntimeConfig = (*containerdCfgRuntime)(nil) // GetBinaryPath retrieves the path to the low-level runtime binary for a runtime. // If no path is available, the empty string is returned. func (c *containerdCfgRuntime) GetBinaryPath() string { if c == nil || c.tree == nil { return "" } binPath, _ := c.tree.GetPath([]string{"options", "BinaryName"}).(string) return binPath } // New creates a containerd config with the specified options func New(opts ...Option) (engine.Interface, error) { b := &builder{ runtimeType: defaultRuntimeType, } for _, opt := range opts { opt(b) } if b.logger == nil { b.logger = logger.New() } if b.configSource == nil { b.configSource = toml.FromFile(b.path) } tomlConfig, err := b.configSource.Load() if err != nil { return nil, fmt.Errorf("failed to load config: %v", err) } cfg := &Config{ Tree: tomlConfig, Logger: b.logger, RuntimeType: b.runtimeType, UseDefaultRuntimeName: b.useLegacyConfig, ContainerAnnotations: b.containerAnnotations, } version, err := cfg.parseVersion(b.useLegacyConfig) if err != nil { return nil, fmt.Errorf("failed to parse config version: %v", err) } switch version { case 1: return (*ConfigV1)(cfg), nil case 2: return cfg, nil } return nil, fmt.Errorf("unsupported config version: %v", version) } // parseVersion returns the version of the config func (c *Config) parseVersion(useLegacyConfig bool) (int, error) { defaultVersion := 2 if useLegacyConfig { defaultVersion = 1 } switch v := c.Get("version").(type) { case nil: switch len(c.Keys()) { case 0: // No config exists, or the config file is empty, use version inferred from containerd return defaultVersion, nil default: // A config file exists, has content, and no version is set return 1, nil } case int64: return int(v), nil default: return -1, fmt.Errorf("unsupported type for version field: %v", v) } } func (c *Config) GetRuntimeConfig(name string) (engine.RuntimeConfig, error) { if c == nil || c.Tree == nil { return nil, fmt.Errorf("config is nil") } runtimeData := c.GetSubtreeByPath([]string{"plugins", "io.containerd.grpc.v1.cri", "containerd", "runtimes", name}) return &containerdCfgRuntime{ tree: runtimeData, }, nil } // CommandLineSource returns the CLI-based containerd config loader func CommandLineSource(hostRoot string) toml.Loader { commandLine := chrootIfRequired(hostRoot, "containerd", "config", "dump") return toml.FromCommandLine(commandLine[0], commandLine[1:]...) } func chrootIfRequired(hostRoot string, commandLine ...string) []string { if hostRoot == "" || hostRoot == "/" { return commandLine } return append([]string{"chroot", hostRoot}, commandLine...) }