provisioner/shell: start_retry_timeout for reboot handling [GH-260]

This commit is contained in:
Mitchell Hashimoto 2013-08-12 15:49:55 -07:00
parent 1ec2de97a6
commit 9a2dbd54bf
3 changed files with 53 additions and 6 deletions

View File

@ -1,6 +1,10 @@
## 0.3.1 (unreleased) ## 0.3.1 (unreleased)
IMPROVEMENTS:
* provisioner/shell: New setting `start_retry_timeout` which is the timeout
for the provisioner to attempt to _start_ the remote process. This allows
the shell provisioner to work properly with reboots. [GH-260]
## 0.3.0 (August 12, 2013) ## 0.3.0 (August 12, 2013)

View File

@ -12,6 +12,7 @@ import (
"log" "log"
"os" "os"
"strings" "strings"
"time"
) )
const DefaultRemotePath = "/tmp/script.sh" const DefaultRemotePath = "/tmp/script.sh"
@ -45,6 +46,12 @@ type config struct {
// can be used to inject the environment_vars into the environment. // can be used to inject the environment_vars into the environment.
ExecuteCommand string `mapstructure:"execute_command"` ExecuteCommand string `mapstructure:"execute_command"`
// The timeout for retrying to start the process. Until this timeout
// is reached, if the provisioner can't start a process, it retries.
// This can be set high to allow for reboots.
RawStartRetryTimeout string `mapstructure:"start_retry_timeout"`
startRetryTimeout time.Duration
tpl *common.Template tpl *common.Template
} }
@ -84,6 +91,10 @@ func (p *Provisioner) Prepare(raws ...interface{}) error {
p.config.InlineShebang = "/bin/sh" p.config.InlineShebang = "/bin/sh"
} }
if p.config.RawStartRetryTimeout == "" {
p.config.RawStartRetryTimeout = "5m"
}
if p.config.RemotePath == "" { if p.config.RemotePath == "" {
p.config.RemotePath = DefaultRemotePath p.config.RemotePath = DefaultRemotePath
} }
@ -108,6 +119,7 @@ func (p *Provisioner) Prepare(raws ...interface{}) error {
templates := map[string]*string{ templates := map[string]*string{
"inline_shebang": &p.config.InlineShebang, "inline_shebang": &p.config.InlineShebang,
"script": &p.config.Script, "script": &p.config.Script,
"start_retry_timeout": &p.config.RawStartRetryTimeout,
"remote_path": &p.config.RemotePath, "remote_path": &p.config.RemotePath,
} }
@ -161,6 +173,14 @@ func (p *Provisioner) Prepare(raws ...interface{}) error {
} }
} }
if p.config.RawStartRetryTimeout != "" {
p.config.startRetryTimeout, err = time.ParseDuration(p.config.RawStartRetryTimeout)
if err != nil {
errs = packer.MultiErrorAppend(
errs, fmt.Errorf("Failed parsing start_retry_timeout: %s", err))
}
}
if errs != nil && len(errs.Errors) > 0 { if errs != nil && len(errs.Errors) > 0 {
return errs return errs
} }
@ -238,9 +258,26 @@ func (p *Provisioner) Provision(ui packer.Ui, comm packer.Communicator) error {
} }
cmd := &packer.RemoteCmd{Command: command} cmd := &packer.RemoteCmd{Command: command}
startTimeout := time.After(p.config.startRetryTimeout)
log.Printf("Executing command: %s", cmd.Command) log.Printf("Executing command: %s", cmd.Command)
if err := cmd.StartWithUi(comm, ui); err != nil { for {
return fmt.Errorf("Failed executing command: %s", err) if err := cmd.StartWithUi(comm, ui); err == nil {
break
}
// Create an error and log it
err = fmt.Errorf("Error executing command: %s", err)
log.Printf(err.Error())
// Check if we timed out, otherwise we retry. It is safe to
// retry since the only error case above is if the command
// failed to START.
select {
case <-startTimeout:
return err
default:
time.Sleep(2 * time.Second)
}
} }
if cmd.ExitStatus != 0 { if cmd.ExitStatus != 0 {

View File

@ -67,6 +67,12 @@ Optional parameters:
in the machine. This defaults to "/tmp/script.sh". This value must be in the machine. This defaults to "/tmp/script.sh". This value must be
a writable location and any parent directories must already exist. a writable location and any parent directories must already exist.
* `start_retry_timeout` (string) - The amount of time to attempt to
_start_ the remote process. By default this is "5m" or 5 minutes. This
setting exists in order to deal with times when SSH may restart, such as
a system reboot. Set this to a higher value if reboots take a longer
amount of time.
## Execute Command Example ## Execute Command Example
To many new users, the `execute_command` is puzzling. However, it provides To many new users, the `execute_command` is puzzling. However, it provides