/** # Copyright (c) 2021, 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 toolkit import ( "fmt" "io" "os" "path/filepath" "sort" "strings" log "github.com/sirupsen/logrus" ) type executableTarget struct { dotfileName string wrapperName string } type executable struct { source string target executableTarget env map[string]string preLines []string argLines []string } // install installs an executable component of the NVIDIA container toolkit. The source executable // is copied to a `.real` file and a wapper is created to set up the environment as required. func (e executable) install(destFolder string) (string, error) { log.Infof("Installing executable '%v' to %v", e.source, destFolder) dotfileName := e.dotfileName() installedDotfileName, err := installFileToFolderWithName(destFolder, dotfileName, e.source) if err != nil { return "", fmt.Errorf("error installing file '%v' as '%v': %v", e.source, dotfileName, err) } log.Infof("Installed '%v'", installedDotfileName) wrapperFilename, err := e.installWrapper(destFolder, installedDotfileName) if err != nil { return "", fmt.Errorf("error wrapping '%v': %v", installedDotfileName, err) } log.Infof("Installed wrapper '%v'", wrapperFilename) return wrapperFilename, nil } func (e executable) dotfileName() string { return e.target.dotfileName } func (e executable) wrapperName() string { return e.target.wrapperName } func (e executable) installWrapper(destFolder string, dotfileName string) (string, error) { wrapperPath := filepath.Join(destFolder, e.wrapperName()) wrapper, err := os.Create(wrapperPath) if err != nil { return "", fmt.Errorf("error creating executable wrapper: %v", err) } defer wrapper.Close() err = e.writeWrapperTo(wrapper, destFolder, dotfileName) if err != nil { return "", fmt.Errorf("error writing wrapper contents: %v", err) } err = ensureExecutable(wrapperPath) if err != nil { return "", fmt.Errorf("error making wrapper executable: %v", err) } return wrapperPath, nil } func (e executable) writeWrapperTo(wrapper io.Writer, destFolder string, dotfileName string) error { r := newReplacements(destDirPattern, destFolder) // Add the shebang fmt.Fprintln(wrapper, "#! /bin/sh") // Add the preceding lines if any for _, line := range e.preLines { fmt.Fprintf(wrapper, "%s\n", r.apply(line)) } // Update the path to include the destination folder var env map[string]string if e.env == nil { env = make(map[string]string) } else { env = e.env } path, specified := env["PATH"] if !specified { path = "$PATH" } env["PATH"] = strings.Join([]string{destFolder, path}, ":") var sortedEnvvars []string for e := range env { sortedEnvvars = append(sortedEnvvars, e) } sort.Strings(sortedEnvvars) for _, e := range sortedEnvvars { v := env[e] fmt.Fprintf(wrapper, "%s=%s \\\n", e, r.apply(v)) } // Add the call to the target executable fmt.Fprintf(wrapper, "%s \\\n", dotfileName) // Insert additional lines in the `arg` list for _, line := range e.argLines { fmt.Fprintf(wrapper, "\t%s \\\n", r.apply(line)) } // Add the script arguments "$@" fmt.Fprintln(wrapper, "\t\"$@\"") return nil } // ensureExecutable is equivalent to running chmod +x on the specified file func ensureExecutable(path string) error { info, err := os.Stat(path) if err != nil { return fmt.Errorf("error getting file info for '%v': %v", path, err) } executableMode := info.Mode() | 0111 err = os.Chmod(path, executableMode) if err != nil { return fmt.Errorf("error setting executable mode for '%v': %v", path, err) } return nil }