gtk/testsuite/testutils.c
Benjamin Otte dd93aa9f50 testsuite: Rewrite diff_with_file()
use the modern version using GSubprocess that already exists in
node-parser.

Also change from one function to two - so tests can diff GBytes and
strings, depending on which they prefer.
2024-11-05 01:07:22 +01:00

147 lines
4.0 KiB
C

/*
* Copyright © 2021 Red Hat, Inc.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
*
* Authors: Matthias Clasen <mclasen@redhat.com>
*/
#include <glib.h>
#include <glib/gstdio.h>
#include <gio/gio.h>
#ifdef G_OS_WIN32
#include <io.h>
#else
#include <unistd.h>
#endif
#include "testsuite/testutils.h"
/*<private>
* diff_bytes_with_file:
* @file1: The filename of the original. This is assumed
* to be the reference to compare against.
* @input: some text contained in a `GBytes` that is
* supposed to match the file's contents
* @error: A return location for an error if diffing could
* not happen
*
* Diffs generated text with a reference file.
*
* If the diffing runs into any error, NULL is returned and
* `error` is set. If diffing succeeds, the error is not set
* and NULL is returned if the file was identical to the
* contents of the file or the actual diff is returned if
* they were not.
*
* Returns: NULL on success or failure, the diff on failure
*/
char *
diff_bytes_with_file (const char *file1,
GBytes *input,
GError **error)
{
char *diff_cmd, *diff;
diff = NULL;
diff_cmd = g_find_program_in_path ("diff");
if (diff_cmd)
{
GSubprocess *process;
GBytes *output;
process = g_subprocess_new (G_SUBPROCESS_FLAGS_STDIN_PIPE
| G_SUBPROCESS_FLAGS_STDOUT_PIPE,
error,
diff_cmd, "--strip-trailing-cr", "-u", file1, "-", NULL);
if (process == NULL)
return NULL;
if (!g_subprocess_communicate (process,
input,
NULL,
&output,
NULL,
error))
{
g_object_unref (process);
return NULL;
}
if (g_subprocess_get_successful (process))
{
g_clear_pointer (&output, g_bytes_unref);
}
else if (g_subprocess_get_if_exited (process) && g_subprocess_get_exit_status (process) == 1)
{
gsize size;
/* this is the condition when the files differ */
diff = g_bytes_unref_to_data (output, &size);
output = NULL;
}
else
{
g_clear_pointer (&output, g_bytes_unref);
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
"The `diff' process exited with error status %d",
g_subprocess_get_exit_status (process));
}
g_object_unref (process);
}
else
{
char *buf1;
const char *buf2;
gsize len1, len2;
buf2 = g_bytes_get_data (input, &len2);
if (!g_file_get_contents (file1, &buf1, &len1, error))
return NULL;
if ((len2 != len1) ||
strncmp (buf2, buf1, len1) != 0)
diff = g_strdup ("Files differ.\n");
g_free (buf1);
}
return diff;
}
char *
diff_string_with_file (const char *file1,
const char *text,
gssize len,
GError **error)
{
GBytes *bytes;
char *result;
if (len < 0)
len = strlen (text);
bytes = g_bytes_new_static (text, len);
result = diff_bytes_with_file (file1, bytes, error);
g_bytes_unref (bytes);
return result;
}