[gnome-builder/gnome-builder-43] libide/foundry: land API for diagnostic tools to use run context
- From: Christian Hergert <chergert src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gnome-builder/gnome-builder-43] libide/foundry: land API for diagnostic tools to use run context
- Date: Thu, 22 Sep 2022 07:32:38 +0000 (UTC)
commit ef6d0507ba241b3c3b652185afe7d2055868fb15
Author: Christian Hergert <chergert redhat com>
Date: Thu Sep 22 00:00:35 2022 -0700
libide/foundry: land API for diagnostic tools to use run context
We aren't in a good place if we're relying on IdeSubprocessLauncher,
particularly from pipelines as we can get into really weird state with
how env FOO=BAR are processed, sh -c '', and more.
Give us some headway here by providing access to a run context so that
we can incrementally fix these without breaking ABI this cycle.
src/libide/foundry/ide-diagnostic-tool.c | 151 +++++++++++++++++++++++++++++--
src/libide/foundry/ide-diagnostic-tool.h | 26 ++++--
2 files changed, 159 insertions(+), 18 deletions(-)
---
diff --git a/src/libide/foundry/ide-diagnostic-tool.c b/src/libide/foundry/ide-diagnostic-tool.c
index c3c9ac74b..00dddbe3d 100644
--- a/src/libide/foundry/ide-diagnostic-tool.c
+++ b/src/libide/foundry/ide-diagnostic-tool.c
@@ -110,6 +110,123 @@ ide_diagnostic_tool_real_configure_launcher (IdeDiagnosticTool *self,
IDE_EXIT;
}
+static gboolean
+ide_diagnostic_tool_real_prepare_run_context (IdeDiagnosticTool *self,
+ IdeRunContext *run_context,
+ GFile *file,
+ GBytes *contents,
+ const char *language_id,
+ GError **error)
+{
+ g_autoptr(IdeSubprocessLauncher) launcher = NULL;
+ g_autoptr(IdeContext) context = NULL;
+ g_autoptr(GFile) workdir = NULL;
+ const char *srcdir = NULL;
+ const char *local_program_path;
+ const char *bundled_program_path;
+ const char *program_name;
+ g_autofree char *program_path = NULL;
+ IdeRuntimeManager *runtime_manager;
+ IdeRuntime *host = NULL;
+ IdePipeline *pipeline = NULL;
+
+ IDE_ENTRY;
+
+ g_assert (IDE_IS_DIAGNOSTIC_TOOL (self));
+
+ 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 (FALSE);
+ }
+
+ workdir = ide_context_ref_workdir (context);
+ srcdir = g_file_peek_path (workdir);
+
+ local_program_path = ide_diagnostic_tool_get_local_program_path (self);
+ bundled_program_path = ide_diagnostic_tool_get_bundled_program_path (self);
+ program_name = ide_diagnostic_tool_get_program_name (self);
+
+ if (program_name == NULL)
+ {
+ g_set_error (error,
+ G_IO_ERROR,
+ G_IO_ERROR_CANCELLED,
+ "No program-name provided for diagnostic tool");
+ IDE_RETURN (FALSE);
+ }
+
+ if (ide_context_has_project (context))
+ {
+ IdeBuildManager *build_manager = ide_build_manager_from_context (context);
+ pipeline = ide_build_manager_get_pipeline (build_manager);
+ runtime_manager = ide_runtime_manager_from_context (context);
+ host = ide_runtime_manager_get_runtime (runtime_manager, "host");
+ }
+
+ if (pipeline != NULL)
+ srcdir = ide_pipeline_get_srcdir (pipeline);
+
+ if (local_program_path != NULL)
+ {
+ g_autofree char *local_program = g_build_filename (srcdir, local_program_path, NULL);
+ if (g_file_test (local_program, G_FILE_TEST_IS_EXECUTABLE))
+ program_path = g_steal_pointer (&local_program);
+ }
+
+ if (pipeline != NULL &&
+ ((program_path != NULL && ide_pipeline_contains_program_in_path (pipeline, program_path, NULL)) ||
+ ide_pipeline_contains_program_in_path (pipeline, program_name, NULL)))
+ {
+ ide_pipeline_prepare_run_context (pipeline, run_context);
+ IDE_GOTO (setup_launcher);
+ }
+
+ if (host != NULL)
+ {
+ /* 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.
+ */
+ if (program_path != NULL ||
+ ide_runtime_contains_program_in_path (host, program_name, NULL))
+ {
+ ide_runtime_prepare_to_build (host, pipeline, run_context);
+ IDE_GOTO (setup_launcher);
+ }
+ }
+ else if (program_path != NULL)
+ {
+ ide_run_context_push_host (run_context);
+ IDE_GOTO (setup_launcher);
+ }
+
+ if (bundled_program_path != NULL && ide_is_flatpak ())
+ program_path = g_strdup (bundled_program_path);
+
+ /* See if Builder itself has bundled the program */
+ if (program_path || g_find_program_in_path (program_name))
+ IDE_GOTO (setup_launcher);
+
+ g_set_error (error,
+ G_IO_ERROR,
+ G_IO_ERROR_NOT_FOUND,
+ "Failed to locate program \"%s\"",
+ program_name);
+
+ IDE_RETURN (FALSE);
+
+setup_launcher:
+ ide_run_context_append_argv (run_context, program_path ? program_path : program_name);
+ ide_run_context_set_cwd (run_context, srcdir);
+
+ IDE_RETURN (TRUE);
+}
+
static IdeSubprocessLauncher *
ide_diagnostic_tool_real_create_launcher (IdeDiagnosticTool *self,
const char *program_name,
@@ -217,12 +334,6 @@ ide_diagnostic_tool_real_create_launcher (IdeDiagnosticTool *self,
setup_launcher:
ide_subprocess_launcher_push_argv (launcher, program_path ? program_path : 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, file, contents, language_id);
g_assert (IDE_IS_SUBPROCESS_LAUNCHER (launcher));
@@ -347,6 +458,7 @@ ide_diagnostic_tool_class_init (IdeDiagnosticToolClass *klass)
klass->configure_launcher = ide_diagnostic_tool_real_configure_launcher;
klass->get_stdin_bytes = ide_diagnostic_tool_real_get_stdin_bytes;
klass->can_diagnose = ide_diagnostic_tool_real_can_diagnose;
+ klass->prepare_run_context = ide_diagnostic_tool_real_prepare_run_context;
/**
* IdeDiagnosticTool:program-name:
@@ -507,12 +619,33 @@ ide_diagnostic_tool_diagnose_async (IdeDiagnosticProvider *provider,
IDE_EXIT;
}
- if (!(launcher = ide_diagnostic_tool_create_launcher (self, priv->program_name, file, contents, lang_id,
&error)))
+ if (IDE_DIAGNOSTIC_TOOL_GET_CLASS (self)->prepare_run_context)
{
- ide_task_return_error (task, g_steal_pointer (&error));
- IDE_EXIT;
+ g_autoptr(IdeRunContext) run_context = ide_run_context_new ();
+
+ if (!IDE_DIAGNOSTIC_TOOL_GET_CLASS (self)->prepare_run_context (self, run_context, file, contents,
lang_id, &error) ||
+ !(launcher = ide_run_context_end (run_context, &error)))
+ {
+ ide_task_return_error (task, g_steal_pointer (&error));
+ IDE_EXIT;
+ }
+ }
+ else
+ {
+ if (!(launcher = ide_diagnostic_tool_create_launcher (self, priv->program_name, file, contents,
lang_id, &error)))
+ {
+ ide_task_return_error (task, g_steal_pointer (&error));
+ IDE_EXIT;
+ }
}
+ 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, file, contents, lang_id);
+
if (!(subprocess = ide_subprocess_launcher_spawn (launcher, cancellable, &error)))
{
ide_task_return_error (task, g_steal_pointer (&error));
diff --git a/src/libide/foundry/ide-diagnostic-tool.h b/src/libide/foundry/ide-diagnostic-tool.h
index cedb50927..befcde21b 100644
--- a/src/libide/foundry/ide-diagnostic-tool.h
+++ b/src/libide/foundry/ide-diagnostic-tool.h
@@ -24,6 +24,8 @@
#include <libide-core.h>
#include <libide-threading.h>
+#include "ide-run-context.h"
+
G_BEGIN_DECLS
#define IDE_TYPE_DIAGNOSTIC_TOOL (ide_diagnostic_tool_get_type())
@@ -35,14 +37,7 @@ struct _IdeDiagnosticToolClass
{
IdeObjectClass parent_class;
- IdeSubprocessLauncher *(*create_launcher) (IdeDiagnosticTool *self,
- const char *program_name,
- GFile *file,
- GBytes *contents,
- const char *language_id,
- GError **error);
- void (*configure_launcher) (IdeDiagnosticTool *self,
- IdeSubprocessLauncher *launcher,
+ gboolean (*can_diagnose) (IdeDiagnosticTool *self,
GFile *file,
GBytes *contents,
const char *language_id);
@@ -55,7 +50,20 @@ struct _IdeDiagnosticToolClass
GFile *file,
const char *stdout_buf,
const char *stderr_buf);
- gboolean (*can_diagnose) (IdeDiagnosticTool *self,
+ gboolean (*prepare_run_context) (IdeDiagnosticTool *self,
+ IdeRunContext *run_context,
+ GFile *file,
+ GBytes *contents,
+ const char *language_id,
+ GError **error);
+ IdeSubprocessLauncher *(*create_launcher) (IdeDiagnosticTool *self,
+ const char *program_name,
+ GFile *file,
+ GBytes *contents,
+ const char *language_id,
+ GError **error);
+ void (*configure_launcher) (IdeDiagnosticTool *self,
+ IdeSubprocessLauncher *launcher,
GFile *file,
GBytes *contents,
const char *language_id);
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]