[gnome-builder] foundry: add IdeDiagnosticTool



commit 65c6b1ada4cbfee041071e742ba562e388376326
Author: Christian Hergert <chergert redhat com>
Date:   Sat Jan 22 14:09:08 2022 -0800

    foundry: add IdeDiagnosticTool
    
    This is meant to be a base-class helper for writing diagnostic providers
    that allows you to focus more on how to parse the output from the tool
    and less about where that tool is run.

 src/libide/foundry/ide-diagnostic-tool.c | 406 +++++++++++++++++++++++++++++++
 src/libide/foundry/ide-diagnostic-tool.h |  59 +++++
 src/libide/foundry/libide-foundry.h      |   1 +
 src/libide/foundry/meson.build           |   2 +
 4 files changed, 468 insertions(+)
---
diff --git a/src/libide/foundry/ide-diagnostic-tool.c b/src/libide/foundry/ide-diagnostic-tool.c
new file mode 100644
index 000000000..0b003c566
--- /dev/null
+++ b/src/libide/foundry/ide-diagnostic-tool.c
@@ -0,0 +1,406 @@
+/* ide-diagnostic-tool.c
+ *
+ * Copyright 2022 Christian Hergert <chergert redhat com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ *
+ * SPDX-License-Identifier: GPL-3.0-or-later
+ */
+
+#define G_LOG_DOMAIN "ide-diagnostic-tool"
+
+#include "config.h"
+
+#include <libide-threading.h>
+
+#include "ide-build-manager.h"
+#include "ide-diagnostic-tool.h"
+#include "ide-pipeline.h"
+#include "ide-runtime.h"
+#include "ide-runtime-manager.h"
+
+typedef struct
+{
+  char *program_name;
+} IdeDiagnosticToolPrivate;
+
+static void diagnostic_provider_iface_init (IdeDiagnosticProviderInterface *iface);
+
+G_DEFINE_ABSTRACT_TYPE_WITH_CODE (IdeDiagnosticTool, ide_diagnostic_tool, IDE_TYPE_OBJECT,
+                                  G_ADD_PRIVATE (IdeDiagnosticTool)
+                                  G_IMPLEMENT_INTERFACE (IDE_TYPE_DIAGNOSTIC_PROVIDER,
+                                                         diagnostic_provider_iface_init))
+
+enum {
+  PROP_0,
+  PROP_PROGRAM_NAME,
+  N_PROPS
+};
+
+static GParamSpec *properties [N_PROPS];
+
+static GBytes *
+ide_diagnostic_tool_real_get_stdin_bytes (IdeDiagnosticTool *self,
+                                          GFile             *file,
+                                          GBytes            *contents,
+                                          const char        *language_id)
+{
+  if (contents != NULL)
+    return g_bytes_ref (contents);
+  return NULL;
+}
+
+static void
+ide_diagnostic_tool_real_configure_launcher (IdeDiagnosticTool     *self,
+                                             IdeSubprocessLauncher *launcher)
+{
+  g_assert (IDE_IS_DIAGNOSTIC_TOOL (self));
+  g_assert (IDE_IS_SUBPROCESS_LAUNCHER (launcher));
+}
+
+static IdeSubprocessLauncher *
+ide_diagnostic_tool_real_create_launcher (IdeDiagnosticTool  *self,
+                                          const char         *program_name,
+                                          GError            **error)
+{
+  g_autoptr(IdeSubprocessLauncher) launcher = NULL;
+  g_autoptr(IdeContext) context = NULL;
+  g_autoptr(GFile) workdir = NULL;
+  const char *srcdir = NULL;
+  IdeRuntimeManager *runtime_manager;
+  IdeRuntime *host;
+
+  IDE_ENTRY;
+
+  g_assert (IDE_IS_DIAGNOSTIC_TOOL (self));
+  g_assert (program_name != NULL);
+
+  if (!(context = ide_object_ref_context (IDE_OBJECT (self))))
+    {
+      g_set_error (error,
+                   G_IO_ERROR,
+                   G_IO_ERROR_CANCELLED,
+                   "Context lost, cancelling request");
+      IDE_RETURN (NULL);
+    }
+
+  workdir = ide_context_ref_workdir (context);
+  srcdir = g_file_peek_path (workdir);
+
+  if (ide_context_has_project (context))
+    {
+      IdeBuildManager *build_manager = ide_build_manager_from_context (context);
+      IdePipeline *pipeline = ide_build_manager_get_pipeline (build_manager);
+
+      if (pipeline != NULL)
+        {
+          srcdir = ide_pipeline_get_srcdir (pipeline);
+
+          if (ide_pipeline_contains_program_in_path (pipeline, program_name, NULL))
+            {
+              if ((launcher = ide_pipeline_create_launcher (pipeline, NULL)))
+                goto setup_launcher;
+            }
+        }
+
+      /* Now try on the host using the "host" runtime which can do
+       * a better job of discovering the program on the host and
+       * take into account if the user has something modifying the
+       * shell like .bashrc.
+       */
+      runtime_manager = ide_runtime_manager_from_context (context);
+      host = ide_runtime_manager_get_runtime (runtime_manager, "host");
+      if (ide_runtime_contains_program_in_path (host, program_name, NULL))
+        {
+          launcher = ide_runtime_create_launcher (host, NULL);
+          goto setup_launcher;
+        }
+    }
+
+  /* See if Builder itself has bundled the program */
+  if (g_find_program_in_path (program_name))
+    {
+      launcher = ide_subprocess_launcher_new (0);
+      goto setup_launcher;
+    }
+
+  g_set_error (error,
+               G_IO_ERROR,
+               G_IO_ERROR_NOT_FOUND,
+               "Failed to locate program \"%s\"",
+               program_name);
+
+  IDE_RETURN (NULL);
+
+setup_launcher:
+  ide_subprocess_launcher_push_argv (launcher, program_name);
+  ide_subprocess_launcher_set_cwd (launcher, srcdir);
+  ide_subprocess_launcher_set_flags (launcher,
+                                     (G_SUBPROCESS_FLAGS_STDIN_PIPE |
+                                      G_SUBPROCESS_FLAGS_STDOUT_PIPE |
+                                      G_SUBPROCESS_FLAGS_STDERR_PIPE));
+
+  IDE_DIAGNOSTIC_TOOL_GET_CLASS (self)->configure_launcher (self, launcher);
+
+  IDE_RETURN (g_steal_pointer (&launcher));
+}
+
+static void
+ide_diagnostic_tool_constructed (GObject *object)
+{
+  IdeDiagnosticTool *self = (IdeDiagnosticTool *)object;
+
+  G_OBJECT_CLASS (ide_diagnostic_tool_parent_class)->constructed (object);
+
+  if (IDE_DIAGNOSTIC_TOOL_GET_CLASS (self)->populate_diagnostics)
+    g_critical ("%s inherits from IdeDiagnosticTool but does not implement populate_diagnostics(). This will 
not work.",
+                G_OBJECT_TYPE_NAME (self));
+}
+
+static void
+ide_diagnostic_tool_finalize (GObject *object)
+{
+  IdeDiagnosticTool *self = (IdeDiagnosticTool *)object;
+  IdeDiagnosticToolPrivate *priv = ide_diagnostic_tool_get_instance_private (self);
+
+  g_clear_pointer (&priv->program_name, g_free);
+
+  G_OBJECT_CLASS (ide_diagnostic_tool_parent_class)->finalize (object);
+}
+
+static void
+ide_diagnostic_tool_get_property (GObject    *object,
+                                  guint       prop_id,
+                                  GValue     *value,
+                                  GParamSpec *pspec)
+{
+  IdeDiagnosticTool *self = IDE_DIAGNOSTIC_TOOL (object);
+
+  switch (prop_id)
+    {
+    case PROP_PROGRAM_NAME:
+      g_value_set_string (value, ide_diagnostic_tool_get_program_name (self));
+      break;
+
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+    }
+}
+
+static void
+ide_diagnostic_tool_set_property (GObject      *object,
+                                  guint         prop_id,
+                                  const GValue *value,
+                                  GParamSpec   *pspec)
+{
+  IdeDiagnosticTool *self = IDE_DIAGNOSTIC_TOOL (object);
+
+  switch (prop_id)
+    {
+    case PROP_PROGRAM_NAME:
+      ide_diagnostic_tool_set_program_name (self, g_value_get_string (value));
+      break;
+
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+    }
+}
+
+static void
+ide_diagnostic_tool_class_init (IdeDiagnosticToolClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+  object_class->constructed = ide_diagnostic_tool_constructed;
+  object_class->finalize = ide_diagnostic_tool_finalize;
+  object_class->get_property = ide_diagnostic_tool_get_property;
+  object_class->set_property = ide_diagnostic_tool_set_property;
+
+  klass->create_launcher = ide_diagnostic_tool_real_create_launcher;
+  klass->configure_launcher = ide_diagnostic_tool_real_configure_launcher;
+  klass->get_stdin_bytes = ide_diagnostic_tool_real_get_stdin_bytes;
+
+  /**
+   * IdeDiagnosticTool:program-name:
+   *
+   * The "program-name" property contains the name of the executable to
+   * locate within the build container, host system, or within Builder's
+   * own runtime container.
+   */
+  properties [PROP_PROGRAM_NAME] =
+    g_param_spec_string ("program-name",
+                         "Program Name",
+                         "The name of the program executable to locate",
+                         NULL,
+                         (G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
+
+  g_object_class_install_properties (object_class, N_PROPS, properties);
+}
+
+static void
+ide_diagnostic_tool_init (IdeDiagnosticTool *self)
+{
+}
+
+static IdeSubprocessLauncher *
+ide_diagnostic_tool_create_launcher (IdeDiagnosticTool  *self,
+                                     const char         *program_name,
+                                     GError            **error)
+{
+  g_assert (IDE_IS_DIAGNOSTIC_TOOL (self));
+  g_assert (program_name != NULL);
+
+  return IDE_DIAGNOSTIC_TOOL_GET_CLASS (self)->create_launcher (self, program_name, error);
+}
+
+static void
+ide_diagnostic_tool_communicate_cb (GObject      *object,
+                                    GAsyncResult *result,
+                                    gpointer      user_data)
+{
+  IdeSubprocess *subprocess = (IdeSubprocess *)object;
+  g_autoptr(IdeDiagnostics) diagnostics = NULL;
+  g_autoptr(IdeTask) task = user_data;
+  g_autoptr(GError) error = NULL;
+  g_autofree char *stdout_buf = NULL;
+  g_autofree char *stderr_buf = NULL;
+  IdeDiagnosticTool *self;
+
+  IDE_ENTRY;
+
+  g_assert (IDE_IS_SUBPROCESS (subprocess));
+  g_assert (G_IS_ASYNC_RESULT (result));
+  g_assert (IDE_IS_TASK (task));
+
+  if (!ide_subprocess_communicate_utf8_finish (subprocess, result, &stdout_buf, &stderr_buf, &error))
+    {
+      ide_task_return_error (task, g_steal_pointer (&error));
+      IDE_EXIT;
+    }
+
+  self = ide_task_get_source_object (task);
+  diagnostics = ide_diagnostics_new ();
+
+  if (IDE_DIAGNOSTIC_TOOL_GET_CLASS (self)->populate_diagnostics != NULL)
+    IDE_DIAGNOSTIC_TOOL_GET_CLASS (self)->populate_diagnostics (self, diagnostics, stdout_buf, stderr_buf);
+
+  ide_task_return_object (task, g_steal_pointer (&diagnostics));
+
+  IDE_EXIT;
+}
+
+static void
+ide_diagnostic_tool_diagnose_async (IdeDiagnosticProvider *provider,
+                                    GFile                 *file,
+                                    GBytes                *contents,
+                                    const gchar           *lang_id,
+                                    GCancellable          *cancellable,
+                                    GAsyncReadyCallback    callback,
+                                    gpointer               user_data)
+{
+  IdeDiagnosticTool *self = (IdeDiagnosticTool *)provider;
+  IdeDiagnosticToolPrivate *priv = ide_diagnostic_tool_get_instance_private (self);
+  g_autoptr(IdeSubprocessLauncher) launcher = NULL;
+  g_autoptr(IdeSubprocess) subprocess = NULL;
+  g_autoptr(IdeTask) task = NULL;
+  g_autoptr(GError) error = NULL;
+  g_autoptr(GBytes) stdin_bytes = NULL;
+  const char *stdin_data = NULL;
+
+  IDE_ENTRY;
+
+  g_assert (IDE_IS_DIAGNOSTIC_TOOL (self));
+  g_assert (G_IS_FILE (file) || contents != NULL);
+  g_assert (!file || G_IS_FILE (file));
+  g_assert (!cancellable || G_IS_CANCELLABLE (cancellable));
+
+  task = ide_task_new (self, cancellable, callback, user_data);
+  ide_task_set_source_tag (task, ide_diagnostic_tool_diagnose_async);
+
+  if (priv->program_name == NULL)
+    {
+      ide_task_return_new_error (task,
+                                 G_IO_ERROR,
+                                 G_IO_ERROR_NOT_SUPPORTED,
+                                 "Program name must be set before diagnosing");
+      IDE_EXIT;
+    }
+
+  if (!(launcher = ide_diagnostic_tool_create_launcher (self, priv->program_name, &error)))
+    {
+      ide_task_return_error (task, g_steal_pointer (&error));
+      IDE_EXIT;
+    }
+
+  if (!(subprocess = ide_subprocess_launcher_spawn (launcher, cancellable, &error)))
+    {
+      ide_task_return_error (task, g_steal_pointer (&error));
+      IDE_EXIT;
+    }
+
+  if ((stdin_bytes = IDE_DIAGNOSTIC_TOOL_GET_CLASS (self)->get_stdin_bytes (self, file, contents, lang_id)))
+    stdin_data = g_bytes_get_data (stdin_bytes, NULL);
+
+  ide_subprocess_communicate_utf8_async (subprocess,
+                                         stdin_data,
+                                         cancellable,
+                                         ide_diagnostic_tool_communicate_cb,
+                                         g_object_ref (task));
+
+  IDE_EXIT;
+}
+
+static IdeDiagnostics *
+ide_diagnostic_tool_diagnose_finish (IdeDiagnosticProvider  *provider,
+                                     GAsyncResult           *result,
+                                     GError                **error)
+{
+  g_assert (IDE_IS_DIAGNOSTIC_PROVIDER (provider));
+  g_assert (IDE_IS_TASK (result));
+
+  return ide_task_propagate_object (IDE_TASK (result), error);
+}
+
+static void
+diagnostic_provider_iface_init (IdeDiagnosticProviderInterface *iface)
+{
+  iface->diagnose_async = ide_diagnostic_tool_diagnose_async;
+  iface->diagnose_finish = ide_diagnostic_tool_diagnose_finish;
+}
+
+const char *
+ide_diagnostic_tool_get_program_name (IdeDiagnosticTool *self)
+{
+  IdeDiagnosticToolPrivate *priv = ide_diagnostic_tool_get_instance_private (self);
+
+  g_return_val_if_fail (IDE_IS_DIAGNOSTIC_TOOL (self), NULL);
+
+  return priv->program_name;
+}
+
+void
+ide_diagnostic_tool_set_program_name (IdeDiagnosticTool *self,
+                                      const char        *program_name)
+{
+  IdeDiagnosticToolPrivate *priv = ide_diagnostic_tool_get_instance_private (self);
+
+  g_return_if_fail (IDE_IS_DIAGNOSTIC_TOOL (self));
+
+  if (g_strcmp0 (program_name, priv->program_name) != 0)
+    {
+      g_free (priv->program_name);
+      priv->program_name = g_strdup (program_name);
+      g_object_notify_by_pspec (G_OBJECT (self), properties [PROP_PROGRAM_NAME]);
+    }
+}
diff --git a/src/libide/foundry/ide-diagnostic-tool.h b/src/libide/foundry/ide-diagnostic-tool.h
new file mode 100644
index 000000000..f498f084d
--- /dev/null
+++ b/src/libide/foundry/ide-diagnostic-tool.h
@@ -0,0 +1,59 @@
+/* ide-diagnostic-tool.h
+ *
+ * Copyright 2022 Christian Hergert <chergert redhat com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
+ *
+ * SPDX-License-Identifier: GPL-3.0-or-later
+ */
+
+#pragma once
+
+#include <libide-code.h>
+#include <libide-core.h>
+#include <libide-threading.h>
+
+G_BEGIN_DECLS
+
+#define IDE_TYPE_DIAGNOSTIC_TOOL (ide_diagnostic_tool_get_type())
+
+IDE_AVAILABLE_IN_42
+G_DECLARE_DERIVABLE_TYPE (IdeDiagnosticTool, ide_diagnostic_tool, IDE, DIAGNOSTIC_TOOL, IdeObject)
+
+struct _IdeDiagnosticToolClass
+{
+  IdeObjectClass parent_class;
+
+  IdeSubprocessLauncher *(*create_launcher)      (IdeDiagnosticTool      *self,
+                                                  const char             *program_name,
+                                                  GError                **error);
+  void                   (*configure_launcher)   (IdeDiagnosticTool      *self,
+                                                  IdeSubprocessLauncher  *launcher);
+  GBytes                *(*get_stdin_bytes)      (IdeDiagnosticTool      *self,
+                                                  GFile                  *file,
+                                                  GBytes                 *contents,
+                                                  const char             *language_id);
+  void                   (*populate_diagnostics) (IdeDiagnosticTool      *self,
+                                                  IdeDiagnostics         *diagnostics,
+                                                  const char             *stdout_buf,
+                                                  const char             *stderr_buf);
+};
+
+IDE_AVAILABLE_IN_42
+const char *ide_diagnostic_tool_get_program_name (IdeDiagnosticTool *self);
+IDE_AVAILABLE_IN_42
+void        ide_diagnostic_tool_set_program_name (IdeDiagnosticTool *self,
+                                                  const char        *program_name);
+
+G_END_DECLS
diff --git a/src/libide/foundry/libide-foundry.h b/src/libide/foundry/libide-foundry.h
index bbbff8a6f..506a4e827 100644
--- a/src/libide/foundry/libide-foundry.h
+++ b/src/libide/foundry/libide-foundry.h
@@ -44,6 +44,7 @@ G_BEGIN_DECLS
 #include "ide-device-manager.h"
 #include "ide-device-provider.h"
 #include "ide-device.h"
+#include "ide-diagnostic-tool.h"
 #include "ide-fallback-build-system.h"
 #include "ide-foundry-compat.h"
 #include "ide-local-device.h"
diff --git a/src/libide/foundry/meson.build b/src/libide/foundry/meson.build
index 95f325043..9faaf1f39 100644
--- a/src/libide/foundry/meson.build
+++ b/src/libide/foundry/meson.build
@@ -27,6 +27,7 @@ libide_foundry_public_headers = [
   'ide-device-manager.h',
   'ide-device-provider.h',
   'ide-device.h',
+  'ide-diagnostic-tool.h',
   'ide-fallback-build-system.h',
   'ide-foundry-compat.h',
   'ide-foundry-types.h',
@@ -101,6 +102,7 @@ libide_foundry_public_sources = [
   'ide-device-manager.c',
   'ide-device-provider.c',
   'ide-device.c',
+  'ide-diagnostic-tool.c',
   'ide-fallback-build-system.c',
   'ide-foundry-compat.c',
   'ide-local-device.c',


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]