diff --git a/man/systemd-run.xml b/man/systemd-run.xml
index 998e934911..a9ccc167b6 100644
--- a/man/systemd-run.xml
+++ b/man/systemd-run.xml
@@ -394,6 +394,17 @@
+
+
+
+
+
+ Display unit log output while running.
+
+
+
+
+
diff --git a/src/run/run.c b/src/run/run.c
index 44949d0bcf..a5073d18ec 100644
--- a/src/run/run.c
+++ b/src/run/run.c
@@ -30,6 +30,7 @@
#include "exec-util.h"
#include "exit-status.h"
#include "fd-util.h"
+#include "fork-journal.h"
#include "format-util.h"
#include "fs-util.h"
#include "hostname-setup.h"
@@ -92,6 +93,7 @@ static char **arg_socket_property = NULL;
static char **arg_timer_property = NULL;
static bool arg_with_timer = false;
static bool arg_quiet = false;
+static bool arg_verbose = false;
static bool arg_aggressive_gc = false;
static char *arg_working_directory = NULL;
static bool arg_shell = false;
@@ -159,6 +161,7 @@ static int help(void) {
" agents until unit is started up\n"
" -P --pipe Pass STDIN/STDOUT/STDERR directly to service\n"
" -q --quiet Suppress information messages during runtime\n"
+ " -v --verbose Show unit logs while executing operation\n"
" --json=pretty|short|off Print unit name and invocation id as JSON\n"
" -G --collect Unload unit after it ran, even when failed\n"
" -S --shell Invoke a $SHELL interactively\n"
@@ -338,6 +341,7 @@ static int parse_argv(int argc, char *argv[]) {
{ "pty-late", no_argument, NULL, 'T' },
{ "pipe", no_argument, NULL, 'P' },
{ "quiet", no_argument, NULL, 'q' },
+ { "verbose", no_argument, NULL, 'v' },
{ "on-active", required_argument, NULL, ARG_ON_ACTIVE },
{ "on-boot", required_argument, NULL, ARG_ON_BOOT },
{ "on-startup", required_argument, NULL, ARG_ON_STARTUP },
@@ -371,7 +375,7 @@ static int parse_argv(int argc, char *argv[]) {
/* Resetting to 0 forces the invocation of an internal initialization routine of getopt_long()
* that checks for GNU extensions in optstring ('-' or '+' at the beginning). */
optind = 0;
- while ((c = getopt_long(argc, argv, "+hrC:H:M:E:p:tTPqGdSu:", options, NULL)) >= 0)
+ while ((c = getopt_long(argc, argv, "+hrC:H:M:E:p:tTPqvGdSu:", options, NULL)) >= 0)
switch (c) {
@@ -498,6 +502,10 @@ static int parse_argv(int argc, char *argv[]) {
arg_quiet = true;
break;
+ case 'v':
+ arg_verbose = true;
+ break;
+
case ARG_ON_ACTIVE:
r = add_timer_property("OnActiveSec", optarg);
if (r < 0)
@@ -2291,6 +2299,10 @@ static int start_transient_service(sd_bus *bus) {
return r;
peer_fd = safe_close(peer_fd);
+ _cleanup_(journal_terminate) PidRef journal_pid = PIDREF_NULL;
+ if (arg_verbose)
+ (void) journal_fork(arg_runtime_scope, (const char**) STRV_MAKE(c.unit), &journal_pid);
+
r = bus_call_with_hint(bus, m, "service", &reply);
if (r < 0)
return r;
@@ -2365,6 +2377,9 @@ static int start_transient_service(sd_bus *bus) {
if (r < 0)
return log_error_errno(r, "Failed to run event loop: %m");
+ /* Close the journal watch logic before we output the exit summary */
+ journal_terminate(&journal_pid);
+
if (arg_wait && !arg_quiet) {
if (!isempty(c.result))