Skip to content

Commit

Permalink
Merge pull request #16090 from mheon/backport_16084_411rhel
Browse files Browse the repository at this point in the history
[v4.1.1-rhel] Backported #16084 to v4.1.1-rhel
  • Loading branch information
openshift-merge-robot authored Oct 10, 2022
2 parents 735c155 + 64da44b commit 46e1da6
Show file tree
Hide file tree
Showing 3 changed files with 32 additions and 25 deletions.
47 changes: 26 additions & 21 deletions libpod/healthcheck.go
Original file line number Diff line number Diff line change
Expand Up @@ -33,31 +33,32 @@ func (r *Runtime) HealthCheck(name string) (define.HealthCheckStatus, error) {
}

hcStatus, err := checkHealthCheckCanBeRun(container)
if err == nil {
hcStatus, err := container.runHealthCheck()
if err := container.processHealthCheckStatus(hcStatus); err != nil {
return hcStatus, err
}
if err != nil {
return hcStatus, err
}

hcStatus, logStatus, err := container.runHealthCheck()
if err := container.processHealthCheckStatus(logStatus); err != nil {
return hcStatus, err
}
return hcStatus, err
}

// runHealthCheck runs the health check as defined by the container
func (c *Container) runHealthCheck() (define.HealthCheckStatus, error) {
func (c *Container) runHealthCheck() (define.HealthCheckStatus, string, error) {
var (
newCommand []string
returnCode int
inStartPeriod bool
)
hcCommand := c.HealthCheckConfig().Test
if len(hcCommand) < 1 {
return define.HealthCheckNotDefined, errors.Errorf("container %s has no defined healthcheck", c.ID())
return define.HealthCheckNotDefined, "", errors.Errorf("container %s has no defined healthcheck", c.ID())
}
switch hcCommand[0] {
case "", "NONE":
return define.HealthCheckNotDefined, errors.Errorf("container %s has no defined healthcheck", c.ID())
case "CMD":
case "", define.HealthConfigTestNone:
return define.HealthCheckNotDefined, "", errors.Errorf("container %s has no defined healthcheck", c.ID())
case define.HealthConfigTestCmd:
newCommand = hcCommand[1:]
case "CMD-SHELL":
// TODO: SHELL command from image not available in Container - use Docker default
Expand All @@ -67,11 +68,11 @@ func (c *Container) runHealthCheck() (define.HealthCheckStatus, error) {
newCommand = hcCommand
}
if len(newCommand) < 1 || newCommand[0] == "" {
return define.HealthCheckNotDefined, errors.Errorf("container %s has no defined healthcheck", c.ID())
return define.HealthCheckNotDefined, "", errors.Errorf("container %s has no defined healthcheck", c.ID())
}
rPipe, wPipe, err := os.Pipe()
if err != nil {
return define.HealthCheckInternalError, errors.Wrapf(err, "unable to create pipe for healthcheck session")
return define.HealthCheckInternalError, "", errors.Wrapf(err, "unable to create pipe for healthcheck session")
}
defer wPipe.Close()
defer rPipe.Close()
Expand Down Expand Up @@ -137,15 +138,16 @@ func (c *Container) runHealthCheck() (define.HealthCheckStatus, error) {
}

hcl := newHealthCheckLog(timeStart, timeEnd, returnCode, eventLog)
if err := c.updateHealthCheckLog(hcl, inStartPeriod); err != nil {
return hcResult, errors.Wrapf(err, "unable to update health check log %s for %s", c.healthCheckLogPath(), c.ID())
logStatus, err := c.updateHealthCheckLog(hcl, inStartPeriod)
if err != nil {
return hcResult, "", errors.Wrapf(err, "unable to update health check log %s for %s", c.healthCheckLogPath(), c.ID())
}

return hcResult, hcErr
return hcResult, logStatus, hcErr
}

func (c *Container) processHealthCheckStatus(status define.HealthCheckStatus) error {
if status == define.HealthCheckSuccess {
func (c *Container) processHealthCheckStatus(status string) error {
if status != define.HealthCheckUnhealthy {
return nil
}

Expand Down Expand Up @@ -213,10 +215,13 @@ func (c *Container) updateHealthStatus(status string) error {
}

// UpdateHealthCheckLog parses the health check results and writes the log
func (c *Container) updateHealthCheckLog(hcl define.HealthCheckLog, inStartPeriod bool) error {
func (c *Container) updateHealthCheckLog(hcl define.HealthCheckLog, inStartPeriod bool) (string, error) {
c.lock.Lock()
defer c.lock.Unlock()

healthCheck, err := c.getHealthCheckLog()
if err != nil {
return err
return "", err
}
if hcl.ExitCode == 0 {
// set status to healthy, reset failing state to 0
Expand All @@ -241,9 +246,9 @@ func (c *Container) updateHealthCheckLog(hcl define.HealthCheckLog, inStartPerio
}
newResults, err := json.Marshal(healthCheck)
if err != nil {
return errors.Wrapf(err, "unable to marshall healthchecks for writing")
return "", errors.Wrapf(err, "unable to marshall healthchecks for writing")
}
return ioutil.WriteFile(c.healthCheckLogPath(), newResults, 0700)
return healthCheck.Status, os.WriteFile(c.healthCheckLogPath(), newResults, 0700)
}

// HealthCheckLogPath returns the path for where the health check log is
Expand Down
9 changes: 5 additions & 4 deletions test/system/220-healthcheck.bats
Original file line number Diff line number Diff line change
Expand Up @@ -28,10 +28,11 @@ function _check_health {
--health-cmd /healthcheck \
--health-interval 1s \
--health-retries 3 \
--health-on-failure=kill \
healthcheck_i

run_podman inspect healthcheck_c --format "{{.Config.HealthcheckOnFailureAction}}"
is "$output" "none" "default on-failure action is none"
is "$output" "kill" "on-failure action is set to kill"

# We can't check for 'starting' because a 1-second interval is too
# short; it could run healthcheck before we get to our first check.
Expand Down Expand Up @@ -67,9 +68,8 @@ Log[-1].ExitCode | 1
Log[-1].Output | \"Uh-oh on stdout!\\\nUh-oh on stderr!\"
"

# healthcheck should now fail, with exit status 1 and 'unhealthy' output
run_podman 1 healthcheck run healthcheck_c
is "$output" "unhealthy" "output from 'podman healthcheck run'"
# now the on-failure should kick in and kill the container
podman wait healthcheck_c

# Clean up
run_podman rm -t 0 -f healthcheck_c
Expand All @@ -95,6 +95,7 @@ Log[-1].Output | \"Uh-oh on stdout!\\\nUh-oh on stderr!\"
# Run that healthcheck image.
run_podman run -d --name $ctr \
--health-cmd /healthcheck \
--health-retries=1 \
--health-on-failure=$policy \
$img

Expand Down
1 change: 1 addition & 0 deletions test/system/250-systemd.bats
Original file line number Diff line number Diff line change
Expand Up @@ -300,6 +300,7 @@ LISTEN_FDNAMES=listen_fdnames" | sort)
run_podman create --name $cname \
--health-cmd /healthcheck \
--health-on-failure=kill \
--health-retries=1 \
--restart=on-failure \
$img

Expand Down

0 comments on commit 46e1da6

Please sign in to comment.