90 lines
2.4 KiB
Go
90 lines
2.4 KiB
Go
/*
|
|
Copyright 2018 Google LLC
|
|
|
|
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 commands
|
|
|
|
import (
|
|
"github.com/containers/image/manifest"
|
|
"github.com/docker/docker/builder/dockerfile/instructions"
|
|
"github.com/sirupsen/logrus"
|
|
"os"
|
|
"os/exec"
|
|
"strconv"
|
|
"strings"
|
|
"syscall"
|
|
)
|
|
|
|
type RunCommand struct {
|
|
cmd *instructions.RunCommand
|
|
}
|
|
|
|
func (r *RunCommand) ExecuteCommand(config *manifest.Schema2Config) error {
|
|
var newCommand []string
|
|
if r.cmd.PrependShell {
|
|
// This is the default shell on Linux
|
|
// TODO: Support shell command here
|
|
shell := []string{"/bin/sh", "-c"}
|
|
newCommand = append(shell, strings.Join(r.cmd.CmdLine, " "))
|
|
} else {
|
|
newCommand = r.cmd.CmdLine
|
|
}
|
|
|
|
logrus.Infof("cmd: %s", newCommand[0])
|
|
logrus.Infof("args: %s", newCommand[1:])
|
|
|
|
cmd := exec.Command(newCommand[0], newCommand[1:]...)
|
|
cmd.Dir = config.WorkingDir
|
|
cmd.Stdout = os.Stdout
|
|
// If specified, run the command as a specific user
|
|
if config.User != "" {
|
|
userAndGroup := strings.Split(config.User, ":")
|
|
// uid and gid need to be uint32
|
|
uid64, err := strconv.ParseUint(userAndGroup[0], 10, 32)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
uid := uint32(uid64)
|
|
var gid uint32
|
|
if len(userAndGroup) > 1 {
|
|
gid64, err := strconv.ParseUint(userAndGroup[1], 10, 32)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
gid = uint32(gid64)
|
|
}
|
|
cmd.SysProcAttr = &syscall.SysProcAttr{}
|
|
cmd.SysProcAttr.Credential = &syscall.Credential{Uid: uid, Gid: gid}
|
|
}
|
|
return cmd.Run()
|
|
}
|
|
|
|
// FilesToSnapshot returns nil for this command because we don't know which files
|
|
// have changed, so we snapshot the entire system.
|
|
func (r *RunCommand) FilesToSnapshot() []string {
|
|
return nil
|
|
}
|
|
|
|
// Author returns some information about the command for the image config
|
|
func (r *RunCommand) CreatedBy() string {
|
|
cmdLine := strings.Join(r.cmd.CmdLine, " ")
|
|
if r.cmd.PrependShell {
|
|
// TODO: Support shell command here
|
|
shell := []string{"/bin/sh", "-c"}
|
|
return strings.Join(append(shell, cmdLine), " ")
|
|
}
|
|
return cmdLine
|
|
}
|