Zbigniew Jędrzejewski-Szmek 930e1b
From e94957b964380c0c9b0f3264ba5e35166d543ca7 Mon Sep 17 00:00:00 2001
Zbigniew Jędrzejewski-Szmek 930e1b
From: =?UTF-8?q?Zbigniew=20J=C4=99drzejewski-Szmek?= <zbyszek@in.waw.pl>
Zbigniew Jędrzejewski-Szmek 930e1b
Date: Mon, 28 Oct 2013 23:43:57 -0400
Zbigniew Jędrzejewski-Szmek 930e1b
Subject: [PATCH] journalctl: add --list-boots to show boot IDs and times
Zbigniew Jędrzejewski-Szmek 930e1b
Zbigniew Jędrzejewski-Szmek 930e1b
Suggested by David Wilkins <dwilkins@maths.tcd.ie> in
Zbigniew Jędrzejewski-Szmek 930e1b
https://bugzilla.redhat.com/show_bug.cgi?id=967521:
Zbigniew Jędrzejewski-Szmek 930e1b
Zbigniew Jędrzejewski-Szmek 930e1b
> [Specific boot ID is a] bit of a palaver to obtain.  I consulted the
Zbigniew Jędrzejewski-Szmek 930e1b
> verbose dump of the journal to discover the _BOOT_ID for the
Zbigniew Jędrzejewski-Szmek 930e1b
> timestamp, and then generated the journal dump for that boot using
Zbigniew Jędrzejewski-Szmek 930e1b
> journalctl _BOOT_ID=foo -o short-monotonic.
Zbigniew Jędrzejewski-Szmek 930e1b
---
Zbigniew Jędrzejewski-Szmek 930e1b
 man/journalctl.xml               |  11 ++++
Zbigniew Jędrzejewski-Szmek 930e1b
 shell-completion/bash/journalctl |   2 +-
Zbigniew Jędrzejewski-Szmek 930e1b
 shell-completion/zsh/_journalctl |   1 +
Zbigniew Jędrzejewski-Szmek 930e1b
 src/journal/journalctl.c         | 105 +++++++++++++++++++++++++++++++++++++--
Zbigniew Jędrzejewski-Szmek 930e1b
 4 files changed, 113 insertions(+), 6 deletions(-)
Zbigniew Jędrzejewski-Szmek 930e1b
Zbigniew Jędrzejewski-Szmek 930e1b
diff --git a/man/journalctl.xml b/man/journalctl.xml
Zbigniew Jędrzejewski-Szmek 930e1b
index b5a0c53..c0cc96d 100644
Zbigniew Jędrzejewski-Szmek 930e1b
--- a/man/journalctl.xml
Zbigniew Jędrzejewski-Szmek 930e1b
+++ b/man/journalctl.xml
Zbigniew Jędrzejewski-Szmek 930e1b
@@ -478,6 +478,17 @@
Zbigniew Jędrzejewski-Szmek 930e1b
                         </varlistentry>
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
                         <varlistentry>
Zbigniew Jędrzejewski-Szmek 930e1b
+                                <term><option>--list-boots</option></term>
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                                <listitem><para>Show a tabular list of
Zbigniew Jędrzejewski-Szmek 930e1b
+                                boot numbers (relative to current
Zbigniew Jędrzejewski-Szmek 930e1b
+                                boot), their IDs, and the timestamps
Zbigniew Jędrzejewski-Szmek 930e1b
+                                of the first and last message
Zbigniew Jędrzejewski-Szmek 930e1b
+                                pertaining to the boot.
Zbigniew Jędrzejewski-Szmek 930e1b
+                                </para></listitem>
Zbigniew Jędrzejewski-Szmek 930e1b
+                        </varlistentry>
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                        <varlistentry>
Zbigniew Jędrzejewski-Szmek 930e1b
                                 <term><option>-k</option></term>
Zbigniew Jędrzejewski-Szmek 930e1b
                                 <term><option>--dmesg</option></term>
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
diff --git a/shell-completion/bash/journalctl b/shell-completion/bash/journalctl
Zbigniew Jędrzejewski-Szmek 930e1b
index 3c40d57..942a253 100644
Zbigniew Jędrzejewski-Szmek 930e1b
--- a/shell-completion/bash/journalctl
Zbigniew Jędrzejewski-Szmek 930e1b
+++ b/shell-completion/bash/journalctl
Zbigniew Jędrzejewski-Szmek 930e1b
@@ -42,7 +42,7 @@ _journalctl() {
Zbigniew Jędrzejewski-Szmek 930e1b
                               --disk-usage -f --follow --header
Zbigniew Jędrzejewski-Szmek 930e1b
                               -h --help -l --local --new-id128 -m --merge --no-pager
Zbigniew Jędrzejewski-Szmek 930e1b
                               --no-tail -q --quiet --setup-keys --this-boot --verify
Zbigniew Jędrzejewski-Szmek 930e1b
-                              --version --list-catalog --update-catalog'
Zbigniew Jędrzejewski-Szmek 930e1b
+                              --version --list-catalog --update-catalog --list-boots'
Zbigniew Jędrzejewski-Szmek 930e1b
                        [ARG]='-b --boot --this-boot -D --directory -F --field
Zbigniew Jędrzejewski-Szmek 930e1b
                               -o --output -u --unit --user-unit'
Zbigniew Jędrzejewski-Szmek 930e1b
                 [ARGUNKNOWN]='-c --cursor --interval -n --lines -p --priority --since --until
Zbigniew Jędrzejewski-Szmek 930e1b
diff --git a/shell-completion/zsh/_journalctl b/shell-completion/zsh/_journalctl
Zbigniew Jędrzejewski-Szmek 930e1b
index 73646b5..29ff3e3 100644
Zbigniew Jędrzejewski-Szmek 930e1b
--- a/shell-completion/zsh/_journalctl
Zbigniew Jędrzejewski-Szmek 930e1b
+++ b/shell-completion/zsh/_journalctl
Zbigniew Jędrzejewski-Szmek 930e1b
@@ -70,6 +70,7 @@ _arguments -s \
Zbigniew Jędrzejewski-Szmek 930e1b
     {-q,--quiet}"[Don't show privilege warning]" \
Zbigniew Jędrzejewski-Szmek 930e1b
     {-m,--merge}'[Show entries from all available journals]' \
Zbigniew Jędrzejewski-Szmek 930e1b
     {-b+,--boot=}'[Show data only from the specified boot or offset]:boot id or offset:_journal_boots' \
Zbigniew Jędrzejewski-Szmek 930e1b
+    '--list-boots[List boots ordered by time]' \
Zbigniew Jędrzejewski-Szmek 930e1b
     {-k,--dmesg}'[Show only kernel messages, Implies -b]' \
Zbigniew Jędrzejewski-Szmek 930e1b
     {-u+,--unit=}'[Show data only from the specified unit]:units:_journal_fields _SYSTEMD_UNIT' \
Zbigniew Jędrzejewski-Szmek 930e1b
     '--user-unit=[Show data only from the specified user session unit]:units:_journal_fields USER_UNIT' \
Zbigniew Jędrzejewski-Szmek 930e1b
diff --git a/src/journal/journalctl.c b/src/journal/journalctl.c
Zbigniew Jędrzejewski-Szmek 930e1b
index 0876ee6..a5c4779 100644
Zbigniew Jędrzejewski-Szmek 930e1b
--- a/src/journal/journalctl.c
Zbigniew Jędrzejewski-Szmek 930e1b
+++ b/src/journal/journalctl.c
Zbigniew Jędrzejewski-Szmek 930e1b
@@ -104,12 +104,14 @@ static enum {
Zbigniew Jędrzejewski-Szmek 930e1b
         ACTION_DISK_USAGE,
Zbigniew Jędrzejewski-Szmek 930e1b
         ACTION_LIST_CATALOG,
Zbigniew Jędrzejewski-Szmek 930e1b
         ACTION_DUMP_CATALOG,
Zbigniew Jędrzejewski-Szmek 930e1b
-        ACTION_UPDATE_CATALOG
Zbigniew Jędrzejewski-Szmek 930e1b
+        ACTION_UPDATE_CATALOG,
Zbigniew Jędrzejewski-Szmek 930e1b
+        ACTION_LIST_BOOTS,
Zbigniew Jędrzejewski-Szmek 930e1b
 } arg_action = ACTION_SHOW;
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
 typedef struct boot_id_t {
Zbigniew Jędrzejewski-Szmek 930e1b
         sd_id128_t id;
Zbigniew Jędrzejewski-Szmek 930e1b
-        uint64_t timestamp;
Zbigniew Jędrzejewski-Szmek 930e1b
+        uint64_t first;
Zbigniew Jędrzejewski-Szmek 930e1b
+        uint64_t last;
Zbigniew Jędrzejewski-Szmek 930e1b
 } boot_id_t;
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
 static int help(void) {
Zbigniew Jędrzejewski-Szmek 930e1b
@@ -125,6 +127,7 @@ static int help(void) {
Zbigniew Jędrzejewski-Szmek 930e1b
                "     --after-cursor=CURSOR Start showing entries from specified cursor\n"
Zbigniew Jędrzejewski-Szmek 930e1b
                "     --show-cursor         Print the cursor after all the entries\n"
Zbigniew Jędrzejewski-Szmek 930e1b
                "  -b --boot[=ID]           Show data only from ID or current boot if unspecified\n"
Zbigniew Jędrzejewski-Szmek 930e1b
+               "     --list-boots          Show terse information about recorded boots\n"
Zbigniew Jędrzejewski-Szmek 930e1b
                "  -k --dmesg               Show kernel message log from current boot\n"
Zbigniew Jędrzejewski-Szmek 930e1b
                "  -u --unit=UNIT           Show data only from the specified unit\n"
Zbigniew Jędrzejewski-Szmek 930e1b
                "     --user-unit=UNIT      Show data only from the specified user session unit\n"
Zbigniew Jędrzejewski-Szmek 930e1b
@@ -177,6 +180,7 @@ static int parse_argv(int argc, char *argv[]) {
Zbigniew Jędrzejewski-Szmek 930e1b
                 ARG_NO_PAGER,
Zbigniew Jędrzejewski-Szmek 930e1b
                 ARG_NO_TAIL,
Zbigniew Jędrzejewski-Szmek 930e1b
                 ARG_NEW_ID128,
Zbigniew Jędrzejewski-Szmek 930e1b
+                ARG_LIST_BOOTS,
Zbigniew Jędrzejewski-Szmek 930e1b
                 ARG_USER,
Zbigniew Jędrzejewski-Szmek 930e1b
                 ARG_SYSTEM,
Zbigniew Jędrzejewski-Szmek 930e1b
                 ARG_ROOT,
Zbigniew Jędrzejewski-Szmek 930e1b
@@ -214,6 +218,7 @@ static int parse_argv(int argc, char *argv[]) {
Zbigniew Jędrzejewski-Szmek 930e1b
                 { "quiet",          no_argument,       NULL, 'q'                },
Zbigniew Jędrzejewski-Szmek 930e1b
                 { "merge",          no_argument,       NULL, 'm'                },
Zbigniew Jędrzejewski-Szmek 930e1b
                 { "boot",           optional_argument, NULL, 'b'                },
Zbigniew Jędrzejewski-Szmek 930e1b
+                { "list-boots",     no_argument,       NULL, ARG_LIST_BOOTS     },
Zbigniew Jędrzejewski-Szmek 930e1b
                 { "this-boot",      optional_argument, NULL, 'b'                }, /* deprecated */
Zbigniew Jędrzejewski-Szmek 930e1b
                 { "dmesg",          no_argument,       NULL, 'k'                },
Zbigniew Jędrzejewski-Szmek 930e1b
                 { "system",         no_argument,       NULL, ARG_SYSTEM         },
Zbigniew Jędrzejewski-Szmek 930e1b
@@ -364,6 +369,10 @@ static int parse_argv(int argc, char *argv[]) {
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
                         break;
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
+                case ARG_LIST_BOOTS:
Zbigniew Jędrzejewski-Szmek 930e1b
+                        arg_action = ACTION_LIST_BOOTS;
Zbigniew Jędrzejewski-Szmek 930e1b
+                        break;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
                 case 'k':
Zbigniew Jędrzejewski-Szmek 930e1b
                         arg_boot = arg_dmesg = true;
Zbigniew Jędrzejewski-Szmek 930e1b
                         break;
Zbigniew Jędrzejewski-Szmek 930e1b
@@ -692,12 +701,93 @@ static int add_matches(sd_journal *j, char **args) {
Zbigniew Jędrzejewski-Szmek 930e1b
 static int boot_id_cmp(const void *a, const void *b) {
Zbigniew Jędrzejewski-Szmek 930e1b
         uint64_t _a, _b;
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
-        _a = ((const boot_id_t *)a)->timestamp;
Zbigniew Jędrzejewski-Szmek 930e1b
-        _b = ((const boot_id_t *)b)->timestamp;
Zbigniew Jędrzejewski-Szmek 930e1b
+        _a = ((const boot_id_t *)a)->first;
Zbigniew Jędrzejewski-Szmek 930e1b
+        _b = ((const boot_id_t *)b)->first;
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
         return _a < _b ? -1 : (_a > _b ? 1 : 0);
Zbigniew Jędrzejewski-Szmek 930e1b
 }
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
+static int list_boots(sd_journal *j) {
Zbigniew Jędrzejewski-Szmek 930e1b
+        int r;
Zbigniew Jędrzejewski-Szmek 930e1b
+        const void *data;
Zbigniew Jędrzejewski-Szmek 930e1b
+        unsigned int count = 0;
Zbigniew Jędrzejewski-Szmek 930e1b
+        int w, i;
Zbigniew Jędrzejewski-Szmek 930e1b
+        size_t length, allocated = 0;
Zbigniew Jędrzejewski-Szmek 930e1b
+        boot_id_t *id;
Zbigniew Jędrzejewski-Szmek 930e1b
+        _cleanup_free_ boot_id_t *all_ids = NULL;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+        r = sd_journal_query_unique(j, "_BOOT_ID");
Zbigniew Jędrzejewski-Szmek 930e1b
+        if (r < 0)
Zbigniew Jędrzejewski-Szmek 930e1b
+                return r;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+        SD_JOURNAL_FOREACH_UNIQUE(j, data, length) {
Zbigniew Jędrzejewski-Szmek 930e1b
+                if (length < strlen("_BOOT_ID="))
Zbigniew Jędrzejewski-Szmek 930e1b
+                        continue;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                if (!GREEDY_REALLOC(all_ids, allocated, count + 1))
Zbigniew Jędrzejewski-Szmek 930e1b
+                        return log_oom();
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                id = &all_ids[count];
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                r = sd_id128_from_string(((const char *)data) + strlen("_BOOT_ID="), &id->id);
Zbigniew Jędrzejewski-Szmek 930e1b
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 930e1b
+                        continue;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                r = sd_journal_add_match(j, data, length);
Zbigniew Jędrzejewski-Szmek 930e1b
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 930e1b
+                        return r;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                r = sd_journal_seek_head(j);
Zbigniew Jędrzejewski-Szmek 930e1b
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 930e1b
+                        return r;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                r = sd_journal_next(j);
Zbigniew Jędrzejewski-Szmek 930e1b
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 930e1b
+                        return r;
Zbigniew Jędrzejewski-Szmek 930e1b
+                else if (r == 0)
Zbigniew Jędrzejewski-Szmek 930e1b
+                        goto flush;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                r = sd_journal_get_realtime_usec(j, &id->first);
Zbigniew Jędrzejewski-Szmek 930e1b
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 930e1b
+                        return r;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                r = sd_journal_seek_tail(j);
Zbigniew Jędrzejewski-Szmek 930e1b
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 930e1b
+                        return r;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                r = sd_journal_previous(j);
Zbigniew Jędrzejewski-Szmek 930e1b
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 930e1b
+                        return r;
Zbigniew Jędrzejewski-Szmek 930e1b
+                else if (r == 0)
Zbigniew Jędrzejewski-Szmek 930e1b
+                        goto flush;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                r = sd_journal_get_realtime_usec(j, &id->last);
Zbigniew Jędrzejewski-Szmek 930e1b
+                if (r < 0)
Zbigniew Jędrzejewski-Szmek 930e1b
+                        return r;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                count++;
Zbigniew Jędrzejewski-Szmek 930e1b
+        flush:
Zbigniew Jędrzejewski-Szmek 930e1b
+                sd_journal_flush_matches(j);
Zbigniew Jędrzejewski-Szmek 930e1b
+        }
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+        qsort_safe(all_ids, count, sizeof(boot_id_t), boot_id_cmp);
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+        /* numbers are one less, but we need an extra char for the sign */
Zbigniew Jędrzejewski-Szmek 930e1b
+        w = DECIMAL_STR_WIDTH(count - 1) + 1;
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+        for (id = all_ids, i = 0; id < all_ids + count; id++, i++) {
Zbigniew Jędrzejewski-Szmek 930e1b
+                char a[FORMAT_TIMESTAMP_MAX], b[FORMAT_TIMESTAMP_MAX];
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+                printf("% *i " SD_ID128_FORMAT_STR " %s—%s\n",
Zbigniew Jędrzejewski-Szmek 930e1b
+                       w, i - count + 1,
Zbigniew Jędrzejewski-Szmek 930e1b
+                       SD_ID128_FORMAT_VAL(id->id),
Zbigniew Jędrzejewski-Szmek 930e1b
+                       format_timestamp(a, sizeof(a), id->first),
Zbigniew Jędrzejewski-Szmek 930e1b
+                       format_timestamp(b, sizeof(b), id->last));
Zbigniew Jędrzejewski-Szmek 930e1b
+        }
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
+        return 0;
Zbigniew Jędrzejewski-Szmek 930e1b
+}
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
 static int get_relative_boot_id(sd_journal *j, sd_id128_t *boot_id, int relative) {
Zbigniew Jędrzejewski-Szmek 930e1b
         int r;
Zbigniew Jędrzejewski-Szmek 930e1b
         const void *data;
Zbigniew Jędrzejewski-Szmek 930e1b
@@ -743,7 +833,7 @@ static int get_relative_boot_id(sd_journal *j, sd_id128_t *boot_id, int relative
Zbigniew Jędrzejewski-Szmek 930e1b
                 else if (r == 0)
Zbigniew Jędrzejewski-Szmek 930e1b
                         goto flush;
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
-                r = sd_journal_get_realtime_usec(j, &id->timestamp);
Zbigniew Jędrzejewski-Szmek 930e1b
+                r = sd_journal_get_realtime_usec(j, &id->first);
Zbigniew Jędrzejewski-Szmek 930e1b
                 if (r < 0)
Zbigniew Jędrzejewski-Szmek 930e1b
                         return r;
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
@@ -1403,6 +1493,11 @@ int main(int argc, char *argv[]) {
Zbigniew Jędrzejewski-Szmek 930e1b
                 return EXIT_SUCCESS;
Zbigniew Jędrzejewski-Szmek 930e1b
         }
Zbigniew Jędrzejewski-Szmek 930e1b
 
Zbigniew Jędrzejewski-Szmek 930e1b
+        if (arg_action == ACTION_LIST_BOOTS) {
Zbigniew Jędrzejewski-Szmek 930e1b
+                r = list_boots(j);
Zbigniew Jędrzejewski-Szmek 930e1b
+                goto finish;
Zbigniew Jędrzejewski-Szmek 930e1b
+        }
Zbigniew Jędrzejewski-Szmek 930e1b
+
Zbigniew Jędrzejewski-Szmek 930e1b
         /* add_boot() must be called first!
Zbigniew Jędrzejewski-Szmek 930e1b
          * It may need to seek the journal to find parent boot IDs. */
Zbigniew Jędrzejewski-Szmek 930e1b
         r = add_boot(j);