[gnome-logs/wip/gl-journal: 3/17] Add simple journal query method
- From: David King <davidk src gnome org>
- To: commits-list gnome org
- Cc: 
- Subject: [gnome-logs/wip/gl-journal: 3/17] Add simple journal query method
- Date: Mon,  7 Oct 2013 09:55:44 +0000 (UTC)
commit 09689b30631b0758c8e2ec339cff52e30cc72c18
Author: David King <davidk gnome org>
Date:   Wed Oct 2 17:20:38 2013 +0100
    Add simple journal query method
 src/gl-journal.c |  184 ++++++++++++++++++++++++++++++++++++++++++++++++++++++
 src/gl-journal.h |   19 ++++++
 2 files changed, 203 insertions(+), 0 deletions(-)
---
diff --git a/src/gl-journal.c b/src/gl-journal.c
index 26bf9f4..ac6383c 100644
--- a/src/gl-journal.c
+++ b/src/gl-journal.c
@@ -19,6 +19,7 @@
 #include "gl-journal.h"
 
 #include <glib-unix.h>
+#include <stdlib.h>
 
 typedef struct
 {
@@ -132,6 +133,189 @@ gl_journal_init (GlJournal *self)
 
 }
 
+GList *
+gl_journal_query (GlJournal *self, const GlJournalQuery *query)
+{
+    GlJournalPrivate *priv;
+    sd_journal *journal;
+    gint ret;
+    gsize i;
+    GList *results = NULL;
+
+    g_return_val_if_fail (GL_JOURNAL (self), NULL);
+    g_return_val_if_fail (query != NULL, NULL);
+
+    priv = gl_journal_get_instance_private (self);
+    journal = priv->journal;
+
+    ret = sd_journal_seek_tail (journal);
+
+    if (ret < 0)
+    {
+        g_warning ("Error seeking to end of systemd journal: %s",
+                   g_strerror (-ret));
+    }
+
+    for (i = 0; i < query->n_results; i++)
+    {
+        /* TODO: Handle matches. */
+        GlJournalResult *result;
+
+        const gchar *message;
+        const gchar *comm;
+        const gchar *kernel_device;
+        const gchar *audit_session;
+        const gchar *priority;
+        gsize length;
+
+        ret = sd_journal_previous (journal);
+
+        if (ret < 0)
+        {
+            g_warning ("Error setting cursor to end of systemd journal: %s",
+                       g_strerror (-ret));
+            break;
+        }
+        else if (ret == 0)
+        {
+            g_debug ("End of systemd journal reached");
+            break;
+        }
+
+        result = g_slice_new (GlJournalResult);
+
+        ret = sd_journal_get_realtime_usec (journal, &result->timestamp);
+
+        if (ret < 0)
+        {
+            g_warning ("Error getting timestamp from systemd journal: %s",
+                       g_strerror (-ret));
+            goto out;
+        }
+
+        ret = sd_journal_get_cursor (journal, &result->cursor);
+
+        if (ret < 0)
+        {
+            g_warning ("Error getting cursor for current journal entry: %s",
+                       g_strerror (-ret));
+            goto out;
+        }
+
+        ret = sd_journal_test_cursor (journal, result->cursor);
+
+        if (ret < 0)
+        {
+            g_warning ("Error testing cursor string: %s", g_strerror (-ret));
+            free (result->cursor);
+            result->cursor = NULL;
+            goto out;
+        }
+        else if (ret == 0)
+        {
+            g_warning ("Cursor string does not match journal entry");
+            /* Not a problem at this point, but would be when seeking to the
+             * cursor later on. */
+        }
+
+        ret = sd_journal_get_catalog (journal, &result->catalog);
+
+        if (ret == -ENOENT)
+        {
+            g_debug ("No message for this log entry was found in the catalog");
+            result->catalog = NULL;
+        }
+        else if (ret < 0)
+        {
+            g_warning ("Error while getting message from catalog: %s",
+                       g_strerror (-ret));
+            free (result->cursor);
+            goto out;
+        }
+
+        ret = sd_journal_get_data (journal, "_COMM", (const void **)&comm,
+                                   &length);
+
+        if (ret < 0)
+        {
+            g_debug ("Unable to get commandline from systemd journal: %s",
+                     g_strerror (-ret));
+            comm = "_COMM=";
+        }
+
+        result->comm = strchr (comm, '=') + 1;
+
+        ret = sd_journal_get_data (journal, "_KERNEL_DEVICE",
+                                   (const void **)&kernel_device, &length);
+
+        if (ret < 0)
+        {
+            g_debug ("Unable to get kernel device from systemd journal: %s",
+                     g_strerror (-ret));
+            kernel_device = "_KERNEL_DEVICE=";
+        }
+
+        result->kernel_device = strchr (kernel_device, '=') + 1;
+
+        ret = sd_journal_get_data (journal, "_AUDIT_SESSION",
+                                   (const void **)&audit_session, &length);
+
+        if (ret < 0)
+        {
+            g_debug ("Unable to get audit session from systemd journal: %s",
+                     g_strerror (-ret));
+            audit_session = "_AUDIT_SESSION=";
+        }
+
+        result->audit_session = strchr (audit_session, '=') + 1;
+
+        ret = sd_journal_get_data (journal, "MESSAGE", (const void **)&message,
+                                   &length);
+
+        if (ret < 0)
+        {
+            g_warning ("Error getting message from systemd journal: %s",
+                       g_strerror (-ret));
+            free (result->cursor);
+            free (result->catalog);
+            goto out;
+        }
+
+        result->message = strchr (message, '=') + 1;
+
+        ret = sd_journal_get_data (journal, "PRIORITY",
+                                   (const void **)&priority, &length);
+
+        if (ret == -ENOENT)
+        {
+            g_warning ("No priority was set for this message");
+            free (result->cursor);
+            free (result->catalog);
+            goto out;
+        }
+        else if (ret < 0)
+        {
+            g_warning ("Error getting priority from systemd journal: %s",
+                       g_strerror (-ret));
+            free (result->cursor);
+            free (result->catalog);
+            goto out;
+        }
+
+        result->priority = atoi (strchr (priority, '=') + 1);
+
+        results = g_list_prepend (results, result);
+        continue;
+
+out:
+        g_slice_free (GlJournalResult, result);
+    }
+
+    sd_journal_flush_matches (journal);
+
+    return g_list_reverse (results);
+}
+
 sd_journal *
 gl_journal_get_journal (GlJournal *self)
 {
diff --git a/src/gl-journal.h b/src/gl-journal.h
index 3c54706..3728a87 100644
--- a/src/gl-journal.h
+++ b/src/gl-journal.h
@@ -26,6 +26,24 @@ G_BEGIN_DECLS
 
 typedef struct
 {
+    gsize n_results;
+    gchar **matches;
+} GlJournalQuery;
+
+typedef struct
+{
+    guint64 timestamp;
+    gchar *cursor;
+    const gchar *message;
+    const gchar *comm;
+    const gchar *kernel_device;
+    const gchar *audit_session;
+    gchar *catalog;
+    guint priority;
+} GlJournalResult;
+
+typedef struct
+{
     /*< private >*/
     GObject parent_instance;
 } GlJournal;
@@ -40,6 +58,7 @@ typedef struct
 #define GL_JOURNAL(object) (G_TYPE_CHECK_INSTANCE_CAST ((object), GL_TYPE_JOURNAL, GlJournal))
 
 GType gl_journal_get_type (void);
+GList * gl_journal_query (GlJournal *self, const GlJournalQuery *query);
 sd_journal * gl_journal_get_journal (GlJournal *self);
 GlJournal * gl_journal_new (void);
 
[
Date Prev][
Date Next]   [
Thread Prev][
Thread Next]   
[
Thread Index]
[
Date Index]
[
Author Index]