2023-08-04 15:56:15 +00:00
|
|
|
/**
|
|
|
|
# Copyright (c) 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 config
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"regexp"
|
|
|
|
|
|
|
|
"github.com/pelletier/go-toml"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Toml is a type for the TOML representation of a config.
|
|
|
|
type Toml toml.Tree
|
|
|
|
|
|
|
|
type options struct {
|
|
|
|
configFile string
|
2023-09-06 15:24:36 +00:00
|
|
|
required bool
|
2023-08-04 15:56:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Option is a functional option for loading TOML config files.
|
|
|
|
type Option func(*options)
|
|
|
|
|
|
|
|
// WithConfigFile sets the config file option.
|
|
|
|
func WithConfigFile(configFile string) Option {
|
|
|
|
return func(o *options) {
|
|
|
|
o.configFile = configFile
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-06 15:24:36 +00:00
|
|
|
// WithRequired sets the required option.
|
|
|
|
// If this is set to true, a failure to open the specified file is treated as an error
|
|
|
|
func WithRequired(required bool) Option {
|
|
|
|
return func(o *options) {
|
|
|
|
o.required = required
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-04 15:56:15 +00:00
|
|
|
// New creates a new toml tree based on the provided options
|
|
|
|
func New(opts ...Option) (*Toml, error) {
|
|
|
|
o := &options{}
|
|
|
|
for _, opt := range opts {
|
|
|
|
opt(o)
|
|
|
|
}
|
|
|
|
|
2023-09-06 15:24:36 +00:00
|
|
|
return o.loadConfigToml()
|
2023-08-04 15:56:15 +00:00
|
|
|
}
|
|
|
|
|
2023-09-06 15:24:36 +00:00
|
|
|
func (o options) loadConfigToml() (*Toml, error) {
|
|
|
|
filename := o.configFile
|
2023-08-04 15:56:15 +00:00
|
|
|
if filename == "" {
|
|
|
|
return defaultToml()
|
|
|
|
}
|
|
|
|
|
2023-09-06 15:24:36 +00:00
|
|
|
_, err := os.Stat(filename)
|
|
|
|
if os.IsNotExist(err) && o.required {
|
|
|
|
return nil, os.ErrNotExist
|
|
|
|
}
|
|
|
|
|
2023-08-04 15:56:15 +00:00
|
|
|
tomlFile, err := os.Open(filename)
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
return defaultToml()
|
|
|
|
} else if err != nil {
|
2023-09-06 15:24:36 +00:00
|
|
|
return nil, fmt.Errorf("failed to load specified config file: %w", err)
|
2023-08-04 15:56:15 +00:00
|
|
|
}
|
|
|
|
defer tomlFile.Close()
|
|
|
|
|
|
|
|
return loadConfigTomlFrom(tomlFile)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func defaultToml() (*Toml, error) {
|
|
|
|
cfg, err := GetDefault()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
contents, err := toml.Marshal(cfg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return loadConfigTomlFrom(bytes.NewReader(contents))
|
|
|
|
}
|
|
|
|
|
|
|
|
func loadConfigTomlFrom(reader io.Reader) (*Toml, error) {
|
|
|
|
tree, err := toml.LoadReader(reader)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return (*Toml)(tree), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Config returns the typed config associated with the toml tree.
|
|
|
|
func (t *Toml) Config() (*Config, error) {
|
2024-09-11 14:47:16 +00:00
|
|
|
cfg, err := t.configNoOverrides()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if err := cfg.assertValid(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return cfg, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// configNoOverrides returns the typed config associated with the toml tree.
|
|
|
|
// This config does not include feature-specific overrides.
|
|
|
|
func (t *Toml) configNoOverrides() (*Config, error) {
|
2023-08-04 15:56:15 +00:00
|
|
|
cfg, err := GetDefault()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if t == nil {
|
|
|
|
return cfg, nil
|
|
|
|
}
|
|
|
|
if err := t.Unmarshal(cfg); err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to unmarshal config: %v", err)
|
|
|
|
}
|
|
|
|
return cfg, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unmarshal wraps the toml.Tree Unmarshal function.
|
|
|
|
func (t *Toml) Unmarshal(v interface{}) error {
|
|
|
|
return (*toml.Tree)(t).Unmarshal(v)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save saves the config to the specified Writer.
|
|
|
|
func (t *Toml) Save(w io.Writer) (int64, error) {
|
|
|
|
contents, err := t.contents()
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
n, err := w.Write(contents)
|
|
|
|
return int64(n), err
|
|
|
|
}
|
|
|
|
|
|
|
|
// contents returns the config TOML as a byte slice.
|
|
|
|
// Any required formatting is applied.
|
|
|
|
func (t Toml) contents() ([]byte, error) {
|
|
|
|
commented := t.commentDefaults()
|
|
|
|
|
|
|
|
buffer := bytes.NewBuffer(nil)
|
|
|
|
|
|
|
|
enc := toml.NewEncoder(buffer).Indentation("")
|
|
|
|
if err := enc.Encode((*toml.Tree)(commented)); err != nil {
|
|
|
|
return nil, fmt.Errorf("invalid config: %v", err)
|
|
|
|
}
|
|
|
|
return t.format(buffer.Bytes())
|
|
|
|
}
|
|
|
|
|
|
|
|
// format fixes the comments for the config to ensure that they start in column
|
|
|
|
// 1 and are not followed by a space.
|
|
|
|
func (t Toml) format(contents []byte) ([]byte, error) {
|
2023-08-25 14:49:31 +00:00
|
|
|
r := regexp.MustCompile(`(\n*)\s*?#\s*(\S.*)`)
|
2023-08-04 15:56:15 +00:00
|
|
|
replaced := r.ReplaceAll(contents, []byte("$1#$2"))
|
|
|
|
|
|
|
|
return replaced, nil
|
|
|
|
}
|
|
|
|
|
2023-08-08 11:43:45 +00:00
|
|
|
// Delete deletes the specified key from the TOML config.
|
|
|
|
func (t *Toml) Delete(key string) error {
|
|
|
|
return (*toml.Tree)(t).Delete(key)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get returns the value for the specified key.
|
|
|
|
func (t *Toml) Get(key string) interface{} {
|
|
|
|
return (*toml.Tree)(t).Get(key)
|
|
|
|
}
|
|
|
|
|
2024-08-08 22:40:00 +00:00
|
|
|
// GetDefault returns the value for the specified key and falls back to the default value if the Get call fails
|
|
|
|
func (t *Toml) GetDefault(key string, def interface{}) interface{} {
|
|
|
|
return (*toml.Tree)(t).GetDefault(key, def)
|
|
|
|
}
|
|
|
|
|
2023-08-08 11:43:45 +00:00
|
|
|
// Set sets the specified key to the specified value in the TOML config.
|
|
|
|
func (t *Toml) Set(key string, value interface{}) {
|
|
|
|
(*toml.Tree)(t).Set(key, value)
|
|
|
|
}
|
|
|
|
|
2024-08-08 22:40:00 +00:00
|
|
|
// WriteTo encode the Tree as Toml and writes it to the writer w.
|
|
|
|
// Returns the number of bytes written in case of success, or an error if anything happened.
|
|
|
|
func (t *Toml) WriteTo(w io.Writer) (int64, error) {
|
|
|
|
return (*toml.Tree)(t).WriteTo(w)
|
|
|
|
}
|
|
|
|
|
2023-08-04 15:56:15 +00:00
|
|
|
// commentDefaults applies the required comments for default values to the Toml.
|
|
|
|
func (t *Toml) commentDefaults() *Toml {
|
|
|
|
asToml := (*toml.Tree)(t)
|
|
|
|
commentedDefaults := map[string]interface{}{
|
|
|
|
"swarm-resource": "DOCKER_RESOURCE_GPU",
|
|
|
|
"accept-nvidia-visible-devices-envvar-when-unprivileged": true,
|
|
|
|
"accept-nvidia-visible-devices-as-volume-mounts": false,
|
|
|
|
"nvidia-container-cli.root": "/run/nvidia/driver",
|
|
|
|
"nvidia-container-cli.path": "/usr/bin/nvidia-container-cli",
|
|
|
|
"nvidia-container-cli.debug": "/var/log/nvidia-container-toolkit.log",
|
|
|
|
"nvidia-container-cli.ldcache": "/etc/ld.so.cache",
|
|
|
|
"nvidia-container-cli.no-cgroups": false,
|
|
|
|
"nvidia-container-cli.user": "root:video",
|
|
|
|
"nvidia-container-runtime.debug": "/var/log/nvidia-container-runtime.log",
|
|
|
|
}
|
|
|
|
for k, v := range commentedDefaults {
|
|
|
|
set := asToml.Get(k)
|
|
|
|
if !shouldComment(k, v, set) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
asToml.SetWithComment(k, "", true, v)
|
|
|
|
}
|
|
|
|
return (*Toml)(asToml)
|
|
|
|
}
|
|
|
|
|
|
|
|
func shouldComment(key string, defaultValue interface{}, setTo interface{}) bool {
|
2024-01-09 09:51:12 +00:00
|
|
|
if key == "nvidia-container-cli.user" && defaultValue == setTo && isSuse() {
|
2023-08-04 15:56:15 +00:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
if key == "nvidia-container-runtime.debug" && setTo == "/dev/null" {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if setTo == nil || defaultValue == setTo || setTo == "" {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|