diff options
| author | Franck Bui <fbui@suse.com> | 2016-11-02 13:51:02 +0100 | 
|---|---|---|
| committer | Franck Bui <fbui@suse.com> | 2016-11-17 18:16:49 +0100 | 
| commit | 3b20f877ade5599dab124474055d2e56c3dcdb15 (patch) | |
| tree | 4b6f001ff5d9c783932c49ae00dbbd9ff992e4e5 | |
| parent | 7d5ceb641659b29204598fde9110913765c2aa9e (diff) | |
core: rework ask_for_confirmation()
Now the reponses are handled by ask_for_confirmation() as well as the report of
any errors occuring during the process of retrieving the confirmation response.
One benefit of this is that there's no need to open/close the console one more
time when reporting error/status messages.
The caller now just needs to care about the return values whose meanings are:
 - don't execute and pretend that the command failed
 - don't execute and pretend that the command succeeed
 - positive answer, execute the command
Also some slight code reorganization and introduce write_confirm_error() and
write_confirm_error_fd(). write_confim_message becomes unneeded.
| -rw-r--r-- | src/core/execute.c | 97 | 
1 files changed, 65 insertions, 32 deletions
| diff --git a/src/core/execute.c b/src/core/execute.c index 43a0a5cafd..8b09f71717 100644 --- a/src/core/execute.c +++ b/src/core/execute.c @@ -669,21 +669,27 @@ static int setup_confirm_stdio(const char *vc, int *_saved_stdin, int *_saved_st          return 0;  } -_printf_(2, 3) static int write_confirm_message(const char *vc, const char *format, ...) { +static void write_confirm_error_fd(int err, int fd) { +        assert(err < 0); + +        if (err == -ETIMEDOUT) +                dprintf(fd, "Confirmation question timed out, assuming positive response.\n"); +        else { +                errno = -err; +                dprintf(fd, "Couldn't ask confirmation: %m, assuming positive response.\n"); +        } +} + +static void write_confirm_error(int err, const char *vc) {          _cleanup_close_ int fd = -1; -        va_list ap; -        assert(format); +        assert(vc);          fd = open_terminal(vc, O_WRONLY|O_NOCTTY|O_CLOEXEC);          if (fd < 0) -                return fd; - -        va_start(ap, format); -        vdprintf(fd, format, ap); -        va_end(ap); +                return; -        return 0; +        write_confirm_error_fd(err, fd);  }  static int restore_confirm_stdio(int *saved_stdin, int *saved_stdout) { @@ -708,22 +714,48 @@ static int restore_confirm_stdio(int *saved_stdin, int *saved_stdout) {          return r;  } -static int ask_for_confirmation(const char *vc, char *response, char **argv) { +enum { +        CONFIRM_PRETEND_FAILURE = -1, +        CONFIRM_PRETEND_SUCCESS =  0, +        CONFIRM_EXECUTE = 1, +}; + +static int ask_for_confirmation(const char *vc, const char *cmdline) {          int saved_stdout = -1, saved_stdin = -1, r; -        _cleanup_free_ char *line = NULL; +        char c; +        /* For any internal errors, assume a positive response. */          r = setup_confirm_stdio(vc, &saved_stdin, &saved_stdout); -        if (r < 0) -                return r; - -        line = exec_command_line(argv); -        if (!line) -                return -ENOMEM; +        if (r < 0) { +                write_confirm_error(r, vc); +                return CONFIRM_EXECUTE; +        } -        r = ask_char(response, "yns", "Execute %s? [Yes, No, Skip] ", line); +        r = ask_char(&c, "yns", "Execute %s? [Yes, No, Skip] ", cmdline); +        if (r < 0) { +                write_confirm_error_fd(r, STDOUT_FILENO); +                r = CONFIRM_EXECUTE; +                goto restore_stdio; +        } + +        switch (c) { +        case 'n': +                printf("Failing execution.\n"); +                r = CONFIRM_PRETEND_SUCCESS; +                break; +        case 's': +                printf("Skipping execution.\n"); +                r = CONFIRM_PRETEND_FAILURE; +                break; +        case 'y': +                r = CONFIRM_EXECUTE; +                break; +        default: +                assert_not_reached("Unhandled choice"); +        } +restore_stdio:          restore_confirm_stdio(&saved_stdin, &saved_stdout); -          return r;  } @@ -2311,21 +2343,22 @@ static int exec_child(          if (params->confirm_spawn) {                  const char *vc = params->confirm_spawn; -                char response; - -                r = ask_for_confirmation(vc, &response, argv); -                if (r == -ETIMEDOUT) -                        write_confirm_message(vc, "Confirmation question timed out, assuming positive response.\n"); -                else if (r < 0) -                        write_confirm_message(vc, "Couldn't ask confirmation question, assuming positive response: %s\n", strerror(-r)); -                else if (response == 's') { -                        write_confirm_message(vc, "Skipping execution.\n"); +                _cleanup_free_ char *cmdline = NULL; + +                cmdline = exec_command_line(argv); +                if (!cmdline) { +                        *exit_status = EXIT_CONFIRM; +                        return -ENOMEM; +                } + +                r = ask_for_confirmation(vc, cmdline); +                if (r != CONFIRM_EXECUTE) { +                        if (r == CONFIRM_PRETEND_SUCCESS) { +                                *exit_status = EXIT_SUCCESS; +                                return 0; +                        }                          *exit_status = EXIT_CONFIRM;                          return -ECANCELED; -                } else if (response == 'n') { -                        write_confirm_message(vc, "Failing execution.\n"); -                        *exit_status = 0; -                        return 0;                  }          } | 
