2010-08-13 18:04:21 +02:00
|
|
|
|
#include <string.h>
|
2010-08-14 01:40:48 +02:00
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
2010-07-05 09:09:36 +02:00
|
|
|
|
#include <gio/gio.h>
|
2010-08-13 18:04:21 +02:00
|
|
|
|
#include <gio/gfiledescriptorbased.h>
|
2013-03-17 23:33:59 +01:00
|
|
|
|
#ifdef G_OS_UNIX
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
#endif
|
2010-07-05 09:09:36 +02:00
|
|
|
|
|
|
|
|
|
static void
|
2017-11-04 19:50:38 +01:00
|
|
|
|
test_basic_for_file (GFile *file,
|
|
|
|
|
const gchar *suffix)
|
2010-07-05 09:09:36 +02:00
|
|
|
|
{
|
|
|
|
|
gchar *s;
|
|
|
|
|
|
|
|
|
|
s = g_file_get_basename (file);
|
|
|
|
|
g_assert_cmpstr (s, ==, "testfile");
|
|
|
|
|
g_free (s);
|
|
|
|
|
|
|
|
|
|
s = g_file_get_uri (file);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (g_str_has_prefix (s, "file://"));
|
|
|
|
|
g_assert_true (g_str_has_suffix (s, suffix));
|
2010-07-05 09:09:36 +02:00
|
|
|
|
g_free (s);
|
|
|
|
|
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (g_file_has_uri_scheme (file, "file"));
|
2010-07-05 09:09:36 +02:00
|
|
|
|
s = g_file_get_uri_scheme (file);
|
|
|
|
|
g_assert_cmpstr (s, ==, "file");
|
|
|
|
|
g_free (s);
|
2017-11-04 19:50:38 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
test_basic (void)
|
|
|
|
|
{
|
|
|
|
|
GFile *file;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_for_path ("./some/directory/testfile");
|
|
|
|
|
test_basic_for_file (file, "/some/directory/testfile");
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
test_build_filename (void)
|
|
|
|
|
{
|
|
|
|
|
GFile *file;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_build_filename (".", "some", "directory", "testfile", NULL);
|
|
|
|
|
test_basic_for_file (file, "/some/directory/testfile");
|
|
|
|
|
g_object_unref (file);
|
2010-07-05 09:09:36 +02:00
|
|
|
|
|
2017-11-04 19:50:38 +01:00
|
|
|
|
file = g_file_new_build_filename ("testfile", NULL);
|
|
|
|
|
test_basic_for_file (file, "/testfile");
|
2010-07-05 09:09:36 +02:00
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
test_parent (void)
|
|
|
|
|
{
|
|
|
|
|
GFile *file;
|
|
|
|
|
GFile *file2;
|
|
|
|
|
GFile *parent;
|
|
|
|
|
GFile *root;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_for_path ("./some/directory/testfile");
|
|
|
|
|
file2 = g_file_new_for_path ("./some/directory");
|
|
|
|
|
root = g_file_new_for_path ("/");
|
|
|
|
|
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (g_file_has_parent (file, file2));
|
2010-07-05 09:09:36 +02:00
|
|
|
|
|
|
|
|
|
parent = g_file_get_parent (file);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (g_file_equal (parent, file2));
|
2010-07-05 09:09:36 +02:00
|
|
|
|
g_object_unref (parent);
|
|
|
|
|
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_null (g_file_get_parent (root));
|
2010-07-05 09:09:36 +02:00
|
|
|
|
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
g_object_unref (file2);
|
|
|
|
|
g_object_unref (root);
|
|
|
|
|
}
|
|
|
|
|
|
2010-07-27 22:24:08 +02:00
|
|
|
|
static void
|
|
|
|
|
test_child (void)
|
|
|
|
|
{
|
|
|
|
|
GFile *file;
|
|
|
|
|
GFile *child;
|
|
|
|
|
GFile *child2;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_for_path ("./some/directory");
|
|
|
|
|
child = g_file_get_child (file, "child");
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (g_file_has_parent (child, file));
|
2010-07-27 22:24:08 +02:00
|
|
|
|
|
|
|
|
|
child2 = g_file_get_child_for_display_name (file, "child2", NULL);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (g_file_has_parent (child2, file));
|
2010-07-27 22:24:08 +02:00
|
|
|
|
|
|
|
|
|
g_object_unref (child);
|
|
|
|
|
g_object_unref (child2);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
test_type (void)
|
|
|
|
|
{
|
2013-05-20 22:33:00 +02:00
|
|
|
|
GFile *datapath_f;
|
2010-07-27 22:24:08 +02:00
|
|
|
|
GFile *file;
|
|
|
|
|
GFileType type;
|
2012-02-03 17:45:51 +01:00
|
|
|
|
GError *error = NULL;
|
2010-07-27 22:24:08 +02:00
|
|
|
|
|
2013-05-29 14:49:16 +02:00
|
|
|
|
datapath_f = g_file_new_for_path (g_test_get_dir (G_TEST_DIST));
|
2013-05-20 22:33:00 +02:00
|
|
|
|
|
|
|
|
|
file = g_file_get_child (datapath_f, "g-icon.c");
|
2010-07-27 22:24:08 +02:00
|
|
|
|
type = g_file_query_file_type (file, 0, NULL);
|
|
|
|
|
g_assert_cmpint (type, ==, G_FILE_TYPE_REGULAR);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
|
Rework the build system for a new tests approach
Perform a substantial cleanup of the build system with respect to
building and installing testcases.
First, Makefile.decl has been renamed glib.mk and substantially
expanded. We intend to add more stuff here in the future, like canned
rules for mkenums, marshallers, resources, etc.
By default, tests are no longer compiled as part of 'make'. They will
be built when 'make check' is run. The old behaviour can be obtained
with --enable-always-build-tests.
--disable-modular-tests is gone (because tests are no longer built by
default). There is no longer any way to cause 'make check' to be a
no-op, but that's not very useful anyway.
A new glibtests.m4 file is introduced. Along with glib.mk, this
provides for consistent handling of --enable-installed-tests and
--enable-always-build-tests (mentioned above).
Port our various test-installing Makefiles to the new framework.
This patch substantially improves the situation in the toplevel tests/
directory. Things are now somewhat under control there. There were
some tests being built that weren't even being run and we run those now.
The long-running GObject performance tests in this directory have been
removed from 'make check' because they take too long.
As an experiment, 'make check' now runs the testcases on win32 builds,
by default. We can't run them under gtester (since it uses a pipe to
communicate with the subprocess) so just toss them in TESTS. Most of
them are passing on win32.
Things are not quite done here, but this patch is already a substantial
improvement. More to come.
2013-05-30 06:07:32 +02:00
|
|
|
|
file = g_file_get_child (datapath_f, "cert-tests");
|
2010-07-27 22:24:08 +02:00
|
|
|
|
type = g_file_query_file_type (file, 0, NULL);
|
|
|
|
|
g_assert_cmpint (type, ==, G_FILE_TYPE_DIRECTORY);
|
2012-02-03 17:45:51 +01:00
|
|
|
|
|
|
|
|
|
g_file_read (file, NULL, &error);
|
|
|
|
|
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_IS_DIRECTORY);
|
|
|
|
|
g_error_free (error);
|
2010-07-27 22:24:08 +02:00
|
|
|
|
g_object_unref (file);
|
2013-05-20 22:33:00 +02:00
|
|
|
|
|
|
|
|
|
g_object_unref (datapath_f);
|
2010-07-27 22:24:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
2013-06-01 06:36:16 +02:00
|
|
|
|
static void
|
|
|
|
|
test_parse_name (void)
|
|
|
|
|
{
|
|
|
|
|
GFile *file;
|
|
|
|
|
gchar *name;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_for_uri ("file://somewhere");
|
|
|
|
|
name = g_file_get_parse_name (file);
|
|
|
|
|
g_assert_cmpstr (name, ==, "file://somewhere");
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
g_free (name);
|
|
|
|
|
|
|
|
|
|
file = g_file_parse_name ("~foo");
|
|
|
|
|
name = g_file_get_parse_name (file);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_nonnull (name);
|
2013-06-01 06:36:16 +02:00
|
|
|
|
g_object_unref (file);
|
|
|
|
|
g_free (name);
|
|
|
|
|
}
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
|
{
|
2020-05-22 12:05:27 +02:00
|
|
|
|
GMainContext *context;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
GFile *file;
|
|
|
|
|
GFileMonitor *monitor;
|
|
|
|
|
GOutputStream *ostream;
|
|
|
|
|
GInputStream *istream;
|
2010-08-13 23:21:22 +02:00
|
|
|
|
gint buffersize;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
gint monitor_created;
|
|
|
|
|
gint monitor_deleted;
|
|
|
|
|
gint monitor_changed;
|
2018-04-26 16:34:55 +02:00
|
|
|
|
gchar *monitor_path;
|
2010-08-13 23:21:22 +02:00
|
|
|
|
gint pos;
|
2018-04-26 16:16:45 +02:00
|
|
|
|
const gchar *data;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
gchar *buffer;
|
2010-08-13 23:21:22 +02:00
|
|
|
|
guint timeout;
|
2020-05-22 12:05:27 +02:00
|
|
|
|
gboolean file_deleted;
|
|
|
|
|
gboolean timed_out;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
} CreateDeleteData;
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
monitor_changed (GFileMonitor *monitor,
|
|
|
|
|
GFile *file,
|
|
|
|
|
GFile *other_file,
|
|
|
|
|
GFileMonitorEvent event_type,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
CreateDeleteData *data = user_data;
|
2012-08-25 02:43:54 +02:00
|
|
|
|
gchar *path;
|
2016-06-23 14:42:53 +02:00
|
|
|
|
const gchar *peeked_path;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
2012-08-25 02:43:54 +02:00
|
|
|
|
path = g_file_get_path (file);
|
2016-06-23 14:42:53 +02:00
|
|
|
|
peeked_path = g_file_peek_path (file);
|
2012-08-25 02:43:54 +02:00
|
|
|
|
g_assert_cmpstr (data->monitor_path, ==, path);
|
2016-06-23 14:42:53 +02:00
|
|
|
|
g_assert_cmpstr (path, ==, peeked_path);
|
2012-08-25 02:43:54 +02:00
|
|
|
|
g_free (path);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
|
|
|
|
if (event_type == G_FILE_MONITOR_EVENT_CREATED)
|
|
|
|
|
data->monitor_created++;
|
|
|
|
|
if (event_type == G_FILE_MONITOR_EVENT_DELETED)
|
|
|
|
|
data->monitor_deleted++;
|
|
|
|
|
if (event_type == G_FILE_MONITOR_EVENT_CHANGED)
|
|
|
|
|
data->monitor_changed++;
|
|
|
|
|
|
2020-05-22 12:05:27 +02:00
|
|
|
|
g_main_context_wakeup (data->context);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
iclosed_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
CreateDeleteData *data = user_data;
|
|
|
|
|
GError *error;
|
|
|
|
|
gboolean ret;
|
|
|
|
|
|
|
|
|
|
error = NULL;
|
|
|
|
|
ret = g_input_stream_close_finish (data->istream, res, &error);
|
|
|
|
|
g_assert_no_error (error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (ret);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (g_input_stream_is_closed (data->istream));
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
|
|
|
|
ret = g_file_delete (data->file, NULL, &error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (ret);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
2020-05-22 12:05:27 +02:00
|
|
|
|
data->file_deleted = TRUE;
|
|
|
|
|
g_main_context_wakeup (data->context);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
read_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
CreateDeleteData *data = user_data;
|
|
|
|
|
GError *error;
|
|
|
|
|
gssize size;
|
|
|
|
|
|
|
|
|
|
error = NULL;
|
|
|
|
|
size = g_input_stream_read_finish (data->istream, res, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
2010-08-13 23:21:22 +02:00
|
|
|
|
data->pos += size;
|
|
|
|
|
if (data->pos < strlen (data->data))
|
|
|
|
|
{
|
|
|
|
|
g_input_stream_read_async (data->istream,
|
|
|
|
|
data->buffer + data->pos,
|
|
|
|
|
strlen (data->data) - data->pos,
|
|
|
|
|
0,
|
|
|
|
|
NULL,
|
|
|
|
|
read_cb,
|
|
|
|
|
data);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
g_assert_cmpstr (data->buffer, ==, data->data);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_false (g_input_stream_is_closed (data->istream));
|
2010-08-13 23:21:22 +02:00
|
|
|
|
g_input_stream_close_async (data->istream, 0, NULL, iclosed_cb, data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
ipending_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
CreateDeleteData *data = user_data;
|
|
|
|
|
GError *error;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
2010-08-13 23:21:22 +02:00
|
|
|
|
error = NULL;
|
2011-04-08 21:44:25 +02:00
|
|
|
|
g_input_stream_read_finish (data->istream, res, &error);
|
2010-08-13 23:21:22 +02:00
|
|
|
|
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_PENDING);
|
|
|
|
|
g_error_free (error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
skipped_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
CreateDeleteData *data = user_data;
|
|
|
|
|
GError *error;
|
|
|
|
|
gssize size;
|
|
|
|
|
|
|
|
|
|
error = NULL;
|
|
|
|
|
size = g_input_stream_skip_finish (data->istream, res, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_cmpint (size, ==, data->pos);
|
|
|
|
|
|
|
|
|
|
g_input_stream_read_async (data->istream,
|
|
|
|
|
data->buffer + data->pos,
|
|
|
|
|
strlen (data->data) - data->pos,
|
|
|
|
|
0,
|
|
|
|
|
NULL,
|
|
|
|
|
read_cb,
|
|
|
|
|
data);
|
|
|
|
|
/* check that we get a pending error */
|
|
|
|
|
g_input_stream_read_async (data->istream,
|
|
|
|
|
data->buffer + data->pos,
|
|
|
|
|
strlen (data->data) - data->pos,
|
|
|
|
|
0,
|
|
|
|
|
NULL,
|
|
|
|
|
ipending_cb,
|
|
|
|
|
data);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
opened_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
2010-08-13 23:21:22 +02:00
|
|
|
|
GFileInputStream *base;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
CreateDeleteData *data = user_data;
|
|
|
|
|
GError *error;
|
|
|
|
|
|
|
|
|
|
error = NULL;
|
2010-08-13 23:21:22 +02:00
|
|
|
|
base = g_file_read_finish (data->file, res, &error);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
2010-08-13 23:21:22 +02:00
|
|
|
|
if (data->buffersize == 0)
|
|
|
|
|
data->istream = G_INPUT_STREAM (g_object_ref (base));
|
|
|
|
|
else
|
|
|
|
|
data->istream = g_buffered_input_stream_new_sized (G_INPUT_STREAM (base), data->buffersize);
|
|
|
|
|
g_object_unref (base);
|
|
|
|
|
|
2010-08-13 18:04:21 +02:00
|
|
|
|
data->buffer = g_new0 (gchar, strlen (data->data) + 1);
|
2010-08-13 23:21:22 +02:00
|
|
|
|
|
|
|
|
|
/* copy initial segment directly, then skip */
|
|
|
|
|
memcpy (data->buffer, data->data, 10);
|
|
|
|
|
data->pos = 10;
|
|
|
|
|
|
|
|
|
|
g_input_stream_skip_async (data->istream,
|
|
|
|
|
10,
|
2010-08-13 18:04:21 +02:00
|
|
|
|
0,
|
|
|
|
|
NULL,
|
2010-08-13 23:21:22 +02:00
|
|
|
|
skipped_cb,
|
2010-08-13 18:04:21 +02:00
|
|
|
|
data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
oclosed_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
CreateDeleteData *data = user_data;
|
|
|
|
|
GError *error;
|
|
|
|
|
gboolean ret;
|
|
|
|
|
|
|
|
|
|
error = NULL;
|
|
|
|
|
ret = g_output_stream_close_finish (data->ostream, res, &error);
|
|
|
|
|
g_assert_no_error (error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (ret);
|
|
|
|
|
g_assert_true (g_output_stream_is_closed (data->ostream));
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
|
|
|
|
g_file_read_async (data->file, 0, NULL, opened_cb, data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
written_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
CreateDeleteData *data = user_data;
|
|
|
|
|
gssize size;
|
|
|
|
|
GError *error;
|
|
|
|
|
|
|
|
|
|
error = NULL;
|
|
|
|
|
size = g_output_stream_write_finish (data->ostream, res, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
2010-08-13 23:21:22 +02:00
|
|
|
|
data->pos += size;
|
|
|
|
|
if (data->pos < strlen (data->data))
|
|
|
|
|
{
|
|
|
|
|
g_output_stream_write_async (data->ostream,
|
|
|
|
|
data->data + data->pos,
|
|
|
|
|
strlen (data->data) - data->pos,
|
|
|
|
|
0,
|
|
|
|
|
NULL,
|
|
|
|
|
written_cb,
|
|
|
|
|
data);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_false (g_output_stream_is_closed (data->ostream));
|
2010-08-13 23:21:22 +02:00
|
|
|
|
g_output_stream_close_async (data->ostream, 0, NULL, oclosed_cb, data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
opending_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
CreateDeleteData *data = user_data;
|
|
|
|
|
GError *error;
|
|
|
|
|
|
|
|
|
|
error = NULL;
|
2011-04-08 21:44:25 +02:00
|
|
|
|
g_output_stream_write_finish (data->ostream, res, &error);
|
2010-08-13 23:21:22 +02:00
|
|
|
|
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_PENDING);
|
|
|
|
|
g_error_free (error);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
created_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
2010-08-13 23:21:22 +02:00
|
|
|
|
GFileOutputStream *base;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
CreateDeleteData *data = user_data;
|
|
|
|
|
GError *error;
|
|
|
|
|
|
|
|
|
|
error = NULL;
|
2010-08-13 23:21:22 +02:00
|
|
|
|
base = g_file_create_finish (G_FILE (source), res, &error);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
g_assert_no_error (error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (g_file_query_exists (data->file, NULL));
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
2010-08-13 23:21:22 +02:00
|
|
|
|
if (data->buffersize == 0)
|
|
|
|
|
data->ostream = G_OUTPUT_STREAM (g_object_ref (base));
|
|
|
|
|
else
|
|
|
|
|
data->ostream = g_buffered_output_stream_new_sized (G_OUTPUT_STREAM (base), data->buffersize);
|
|
|
|
|
g_object_unref (base);
|
|
|
|
|
|
2010-08-13 18:04:21 +02:00
|
|
|
|
g_output_stream_write_async (data->ostream,
|
|
|
|
|
data->data,
|
|
|
|
|
strlen (data->data),
|
|
|
|
|
0,
|
|
|
|
|
NULL,
|
|
|
|
|
written_cb,
|
|
|
|
|
data);
|
2010-08-13 23:21:22 +02:00
|
|
|
|
/* check that we get a pending error */
|
|
|
|
|
g_output_stream_write_async (data->ostream,
|
|
|
|
|
data->data,
|
|
|
|
|
strlen (data->data),
|
|
|
|
|
0,
|
|
|
|
|
NULL,
|
|
|
|
|
opending_cb,
|
|
|
|
|
data);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static gboolean
|
2020-05-22 12:05:27 +02:00
|
|
|
|
stop_timeout (gpointer user_data)
|
2010-08-13 18:04:21 +02:00
|
|
|
|
{
|
2020-05-22 12:05:27 +02:00
|
|
|
|
CreateDeleteData *data = user_data;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
2020-05-22 12:05:27 +02:00
|
|
|
|
data->timed_out = TRUE;
|
|
|
|
|
g_main_context_wakeup (data->context);
|
|
|
|
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* This test does a fully async create-write-read-delete.
|
|
|
|
|
* Callbackistan.
|
|
|
|
|
*/
|
|
|
|
|
static void
|
2010-08-13 23:21:22 +02:00
|
|
|
|
test_create_delete (gconstpointer d)
|
2010-08-13 18:04:21 +02:00
|
|
|
|
{
|
|
|
|
|
GError *error;
|
|
|
|
|
CreateDeleteData *data;
|
2012-02-03 17:29:46 +01:00
|
|
|
|
GFileIOStream *iostream;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
|
|
|
|
data = g_new0 (CreateDeleteData, 1);
|
|
|
|
|
|
2010-08-13 23:21:22 +02:00
|
|
|
|
data->buffersize = GPOINTER_TO_INT (d);
|
|
|
|
|
data->data = "abcdefghijklmnopqrstuvxyzABCDEFGHIJKLMNOPQRSTUVXYZ0123456789";
|
|
|
|
|
data->pos = 0;
|
|
|
|
|
|
2012-02-03 17:29:46 +01:00
|
|
|
|
data->file = g_file_new_tmp ("g_file_create_delete_XXXXXX",
|
|
|
|
|
&iostream, NULL);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_nonnull (data->file);
|
2012-02-03 17:29:46 +01:00
|
|
|
|
g_object_unref (iostream);
|
|
|
|
|
|
|
|
|
|
data->monitor_path = g_file_get_path (data->file);
|
2010-08-20 00:24:53 +02:00
|
|
|
|
remove (data->monitor_path);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_false (g_file_query_exists (data->file, NULL));
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
2010-08-14 01:40:48 +02:00
|
|
|
|
error = NULL;
|
2010-08-13 18:04:21 +02:00
|
|
|
|
data->monitor = g_file_monitor_file (data->file, 0, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
2012-02-03 17:35:26 +01:00
|
|
|
|
|
|
|
|
|
/* This test doesn't work with GPollFileMonitor, because it assumes
|
|
|
|
|
* that the monitor will notice a create immediately followed by a
|
|
|
|
|
* delete, rather than coalescing them into nothing.
|
|
|
|
|
*/
|
2018-06-03 05:55:04 +02:00
|
|
|
|
/* This test also doesn't work with GKqueueFileMonitor because of
|
|
|
|
|
* the same reason. Kqueue is able to return a kevent when a file is
|
|
|
|
|
* created or deleted in a directory. However, the kernel doesn't tell
|
|
|
|
|
* the program file names, so GKqueueFileMonitor has to calculate the
|
|
|
|
|
* difference itself. This is usually too slow for rapid file creation
|
|
|
|
|
* and deletion tests.
|
|
|
|
|
*/
|
|
|
|
|
if (strcmp (G_OBJECT_TYPE_NAME (data->monitor), "GPollFileMonitor") == 0 ||
|
|
|
|
|
strcmp (G_OBJECT_TYPE_NAME (data->monitor), "GKqueueFileMonitor") == 0)
|
2012-02-03 17:35:26 +01:00
|
|
|
|
{
|
2015-05-11 17:50:00 +02:00
|
|
|
|
g_test_skip ("skipping test for this GFileMonitor implementation");
|
2012-02-03 17:35:26 +01:00
|
|
|
|
goto skip;
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-14 01:40:48 +02:00
|
|
|
|
g_file_monitor_set_rate_limit (data->monitor, 100);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
|
|
|
|
g_signal_connect (data->monitor, "changed", G_CALLBACK (monitor_changed), data);
|
|
|
|
|
|
2020-05-22 12:05:27 +02:00
|
|
|
|
/* Use the global default main context */
|
|
|
|
|
data->context = NULL;
|
|
|
|
|
data->timeout = g_timeout_add_seconds (10, stop_timeout, data);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
|
|
|
|
g_file_create_async (data->file, 0, 0, NULL, created_cb, data);
|
|
|
|
|
|
2020-05-22 12:05:27 +02:00
|
|
|
|
while (!data->timed_out &&
|
|
|
|
|
(data->monitor_created == 0 ||
|
|
|
|
|
data->monitor_deleted == 0 ||
|
|
|
|
|
data->monitor_changed == 0 ||
|
|
|
|
|
!data->file_deleted))
|
|
|
|
|
g_main_context_iteration (data->context, TRUE);
|
|
|
|
|
|
|
|
|
|
g_source_remove (data->timeout);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
|
2020-05-22 12:05:27 +02:00
|
|
|
|
g_assert_false (data->timed_out);
|
|
|
|
|
g_assert_true (data->file_deleted);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
g_assert_cmpint (data->monitor_created, ==, 1);
|
|
|
|
|
g_assert_cmpint (data->monitor_deleted, ==, 1);
|
|
|
|
|
g_assert_cmpint (data->monitor_changed, >, 0);
|
|
|
|
|
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_false (g_file_monitor_is_cancelled (data->monitor));
|
2010-08-13 18:04:21 +02:00
|
|
|
|
g_file_monitor_cancel (data->monitor);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (g_file_monitor_is_cancelled (data->monitor));
|
2010-08-13 23:21:22 +02:00
|
|
|
|
|
2020-05-22 12:05:27 +02:00
|
|
|
|
g_clear_pointer (&data->context, g_main_context_unref);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
g_object_unref (data->ostream);
|
|
|
|
|
g_object_unref (data->istream);
|
2012-02-03 17:35:26 +01:00
|
|
|
|
|
|
|
|
|
skip:
|
|
|
|
|
g_object_unref (data->monitor);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
g_object_unref (data->file);
|
2018-04-26 16:36:11 +02:00
|
|
|
|
g_free (data->monitor_path);
|
2010-08-13 18:04:21 +02:00
|
|
|
|
g_free (data->buffer);
|
|
|
|
|
g_free (data);
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-25 01:00:00 +01:00
|
|
|
|
static const gchar *original_data =
|
|
|
|
|
"/**\n"
|
|
|
|
|
" * g_file_replace_contents_async:\n"
|
|
|
|
|
"**/\n";
|
|
|
|
|
|
2012-11-07 16:36:05 +01:00
|
|
|
|
static const gchar *replace_data =
|
|
|
|
|
"/**\n"
|
|
|
|
|
" * g_file_replace_contents_async:\n"
|
|
|
|
|
" * @file: input #GFile.\n"
|
|
|
|
|
" * @contents: string of contents to replace the file with.\n"
|
|
|
|
|
" * @length: the length of @contents in bytes.\n"
|
2016-10-29 03:29:02 +02:00
|
|
|
|
" * @etag: (nullable): a new <link linkend=\"gfile-etag\">entity tag</link> for the @file, or %NULL\n"
|
2012-11-07 16:36:05 +01:00
|
|
|
|
" * @make_backup: %TRUE if a backup should be created.\n"
|
|
|
|
|
" * @flags: a set of #GFileCreateFlags.\n"
|
|
|
|
|
" * @cancellable: optional #GCancellable object, %NULL to ignore.\n"
|
|
|
|
|
" * @callback: a #GAsyncReadyCallback to call when the request is satisfied\n"
|
|
|
|
|
" * @user_data: the data to pass to callback function\n"
|
|
|
|
|
" * \n"
|
|
|
|
|
" * Starts an asynchronous replacement of @file with the given \n"
|
|
|
|
|
" * @contents of @length bytes. @etag will replace the document's\n"
|
|
|
|
|
" * current entity tag.\n"
|
|
|
|
|
" * \n"
|
|
|
|
|
" * When this operation has completed, @callback will be called with\n"
|
|
|
|
|
" * @user_user data, and the operation can be finalized with \n"
|
|
|
|
|
" * g_file_replace_contents_finish().\n"
|
|
|
|
|
" * \n"
|
|
|
|
|
" * If @cancellable is not %NULL, then the operation can be cancelled by\n"
|
|
|
|
|
" * triggering the cancellable object from another thread. If the operation\n"
|
|
|
|
|
" * was cancelled, the error %G_IO_ERROR_CANCELLED will be returned. \n"
|
|
|
|
|
" * \n"
|
|
|
|
|
" * If @make_backup is %TRUE, this function will attempt to \n"
|
|
|
|
|
" * make a backup of @file.\n"
|
|
|
|
|
" **/\n";
|
|
|
|
|
|
2010-08-14 01:40:48 +02:00
|
|
|
|
typedef struct
|
|
|
|
|
{
|
|
|
|
|
GFile *file;
|
|
|
|
|
const gchar *data;
|
|
|
|
|
GMainLoop *loop;
|
|
|
|
|
gboolean again;
|
|
|
|
|
} ReplaceLoadData;
|
|
|
|
|
|
|
|
|
|
static void replaced_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data);
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
loaded_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
ReplaceLoadData *data = user_data;
|
|
|
|
|
gboolean ret;
|
|
|
|
|
GError *error;
|
|
|
|
|
gchar *contents;
|
|
|
|
|
gsize length;
|
|
|
|
|
|
|
|
|
|
error = NULL;
|
|
|
|
|
ret = g_file_load_contents_finish (data->file, res, &contents, &length, NULL, &error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (ret);
|
2010-08-14 01:40:48 +02:00
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_cmpint (length, ==, strlen (data->data));
|
|
|
|
|
g_assert_cmpstr (contents, ==, data->data);
|
|
|
|
|
|
|
|
|
|
g_free (contents);
|
|
|
|
|
|
|
|
|
|
if (data->again)
|
|
|
|
|
{
|
|
|
|
|
data->again = FALSE;
|
|
|
|
|
data->data = "pi pa po";
|
|
|
|
|
|
|
|
|
|
g_file_replace_contents_async (data->file,
|
|
|
|
|
data->data,
|
|
|
|
|
strlen (data->data),
|
|
|
|
|
NULL,
|
|
|
|
|
FALSE,
|
|
|
|
|
0,
|
|
|
|
|
NULL,
|
|
|
|
|
replaced_cb,
|
|
|
|
|
data);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
error = NULL;
|
|
|
|
|
ret = g_file_delete (data->file, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (ret);
|
|
|
|
|
g_assert_false (g_file_query_exists (data->file, NULL));
|
2010-08-14 01:40:48 +02:00
|
|
|
|
|
|
|
|
|
g_main_loop_quit (data->loop);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
replaced_cb (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
ReplaceLoadData *data = user_data;
|
|
|
|
|
GError *error;
|
|
|
|
|
|
|
|
|
|
error = NULL;
|
2011-04-08 21:44:25 +02:00
|
|
|
|
g_file_replace_contents_finish (data->file, res, NULL, &error);
|
2010-08-14 01:40:48 +02:00
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
|
|
|
|
g_file_load_contents_async (data->file, NULL, loaded_cb, data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
test_replace_load (void)
|
|
|
|
|
{
|
|
|
|
|
ReplaceLoadData *data;
|
2016-06-23 14:42:53 +02:00
|
|
|
|
const gchar *path;
|
2012-02-03 17:29:46 +01:00
|
|
|
|
GFileIOStream *iostream;
|
2010-08-14 01:40:48 +02:00
|
|
|
|
|
|
|
|
|
data = g_new0 (ReplaceLoadData, 1);
|
|
|
|
|
data->again = TRUE;
|
2012-11-07 16:36:05 +01:00
|
|
|
|
data->data = replace_data;
|
2010-08-14 01:40:48 +02:00
|
|
|
|
|
2012-02-03 17:29:46 +01:00
|
|
|
|
data->file = g_file_new_tmp ("g_file_replace_load_XXXXXX",
|
|
|
|
|
&iostream, NULL);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_nonnull (data->file);
|
2012-02-03 17:29:46 +01:00
|
|
|
|
g_object_unref (iostream);
|
|
|
|
|
|
2016-06-23 14:42:53 +02:00
|
|
|
|
path = g_file_peek_path (data->file);
|
2010-08-20 00:24:53 +02:00
|
|
|
|
remove (path);
|
|
|
|
|
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_false (g_file_query_exists (data->file, NULL));
|
2010-08-14 01:40:48 +02:00
|
|
|
|
|
|
|
|
|
data->loop = g_main_loop_new (NULL, FALSE);
|
|
|
|
|
|
|
|
|
|
g_file_replace_contents_async (data->file,
|
|
|
|
|
data->data,
|
|
|
|
|
strlen (data->data),
|
|
|
|
|
NULL,
|
|
|
|
|
FALSE,
|
|
|
|
|
0,
|
|
|
|
|
NULL,
|
|
|
|
|
replaced_cb,
|
|
|
|
|
data);
|
|
|
|
|
|
|
|
|
|
g_main_loop_run (data->loop);
|
|
|
|
|
|
|
|
|
|
g_main_loop_unref (data->loop);
|
|
|
|
|
g_object_unref (data->file);
|
|
|
|
|
g_free (data);
|
|
|
|
|
}
|
|
|
|
|
|
2012-11-07 16:36:05 +01:00
|
|
|
|
static void
|
|
|
|
|
test_replace_cancel (void)
|
|
|
|
|
{
|
|
|
|
|
GFile *tmpdir, *file;
|
|
|
|
|
GFileOutputStream *ostream;
|
|
|
|
|
GFileEnumerator *fenum;
|
|
|
|
|
GFileInfo *info;
|
|
|
|
|
GCancellable *cancellable;
|
|
|
|
|
gchar *path;
|
2018-11-25 01:00:00 +01:00
|
|
|
|
gchar *contents;
|
|
|
|
|
gsize nwrote, length;
|
2015-02-13 00:20:14 +01:00
|
|
|
|
guint count;
|
2012-11-07 16:36:05 +01:00
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
2021-02-24 18:34:32 +01:00
|
|
|
|
g_test_bug ("https://bugzilla.gnome.org/629301");
|
2012-11-07 16:36:05 +01:00
|
|
|
|
|
|
|
|
|
path = g_dir_make_tmp ("g_file_replace_cancel_XXXXXX", &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
tmpdir = g_file_new_for_path (path);
|
|
|
|
|
g_free (path);
|
|
|
|
|
|
|
|
|
|
file = g_file_get_child (tmpdir, "file");
|
|
|
|
|
g_file_replace_contents (file,
|
2018-11-25 01:00:00 +01:00
|
|
|
|
original_data,
|
|
|
|
|
strlen (original_data),
|
2012-11-07 16:36:05 +01:00
|
|
|
|
NULL, FALSE, 0, NULL,
|
|
|
|
|
NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
|
|
|
|
ostream = g_file_replace (file, NULL, TRUE, 0, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
|
|
|
|
g_output_stream_write_all (G_OUTPUT_STREAM (ostream),
|
|
|
|
|
replace_data, strlen (replace_data),
|
|
|
|
|
&nwrote, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_cmpint (nwrote, ==, strlen (replace_data));
|
|
|
|
|
|
|
|
|
|
/* At this point there should be two files; the original and the
|
|
|
|
|
* temporary.
|
|
|
|
|
*/
|
|
|
|
|
fenum = g_file_enumerate_children (tmpdir, NULL, 0, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
|
|
|
|
info = g_file_enumerator_next_file (fenum, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_nonnull (info);
|
2012-11-07 16:36:05 +01:00
|
|
|
|
g_object_unref (info);
|
|
|
|
|
info = g_file_enumerator_next_file (fenum, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_nonnull (info);
|
2012-11-07 16:36:05 +01:00
|
|
|
|
g_object_unref (info);
|
|
|
|
|
|
|
|
|
|
g_file_enumerator_close (fenum, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_object_unref (fenum);
|
|
|
|
|
|
2015-02-13 00:20:14 +01:00
|
|
|
|
/* Also test the g_file_enumerator_iterate() API */
|
|
|
|
|
fenum = g_file_enumerate_children (tmpdir, NULL, 0, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
count = 0;
|
|
|
|
|
|
|
|
|
|
while (TRUE)
|
|
|
|
|
{
|
|
|
|
|
gboolean ret = g_file_enumerator_iterate (fenum, &info, NULL, NULL, &error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (ret);
|
2015-02-13 00:20:14 +01:00
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
if (!info)
|
|
|
|
|
break;
|
|
|
|
|
count++;
|
|
|
|
|
}
|
|
|
|
|
g_assert_cmpint (count, ==, 2);
|
|
|
|
|
|
|
|
|
|
g_file_enumerator_close (fenum, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_object_unref (fenum);
|
|
|
|
|
|
|
|
|
|
/* Now test just getting child from the g_file_enumerator_iterate() API */
|
|
|
|
|
fenum = g_file_enumerate_children (tmpdir, "standard::name", 0, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
count = 0;
|
|
|
|
|
|
|
|
|
|
while (TRUE)
|
|
|
|
|
{
|
|
|
|
|
GFile *child;
|
|
|
|
|
gboolean ret = g_file_enumerator_iterate (fenum, NULL, &child, NULL, &error);
|
|
|
|
|
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (ret);
|
2015-02-13 00:20:14 +01:00
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
|
|
|
|
if (!child)
|
|
|
|
|
break;
|
|
|
|
|
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (G_IS_FILE (child));
|
2015-02-13 00:20:14 +01:00
|
|
|
|
count++;
|
|
|
|
|
}
|
|
|
|
|
g_assert_cmpint (count, ==, 2);
|
|
|
|
|
|
|
|
|
|
g_file_enumerator_close (fenum, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_object_unref (fenum);
|
|
|
|
|
|
2012-11-07 16:36:05 +01:00
|
|
|
|
/* Make sure the temporary gets deleted even if we cancel. */
|
|
|
|
|
cancellable = g_cancellable_new ();
|
|
|
|
|
g_cancellable_cancel (cancellable);
|
|
|
|
|
g_output_stream_close (G_OUTPUT_STREAM (ostream), cancellable, &error);
|
|
|
|
|
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CANCELLED);
|
|
|
|
|
g_clear_error (&error);
|
|
|
|
|
|
|
|
|
|
g_object_unref (cancellable);
|
|
|
|
|
g_object_unref (ostream);
|
|
|
|
|
|
2018-11-25 01:00:00 +01:00
|
|
|
|
/* Make sure that file contents wasn't actually replaced. */
|
|
|
|
|
g_file_load_contents (file,
|
|
|
|
|
NULL,
|
|
|
|
|
&contents,
|
|
|
|
|
&length,
|
|
|
|
|
NULL,
|
|
|
|
|
&error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_cmpstr (contents, ==, original_data);
|
|
|
|
|
g_free (contents);
|
|
|
|
|
|
2012-11-07 16:36:05 +01:00
|
|
|
|
g_file_delete (file, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
|
|
|
|
|
/* This will only succeed if the temp file was deleted. */
|
|
|
|
|
g_file_delete (tmpdir, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_object_unref (tmpdir);
|
|
|
|
|
}
|
|
|
|
|
|
glocalfileoutputstream: Fix CREATE_REPLACE_DESTINATION with symlinks
The `G_FILE_CREATE_REPLACE_DESTINATION` flag is equivalent to unlinking
the destination file and re-creating it from scratch. That did
previously work, but in the process the code would call `open(O_CREAT)`
on the file. If the file was a dangling symlink, this would create the
destination file (empty). That’s not an intended side-effect, and has
security implications if the symlink is controlled by a lower-privileged
process.
Fix that by not opening the destination file if it’s a symlink, and
adjusting the rest of the code to cope with
- the fact that `fd == -1` is not an error iff `is_symlink` is true,
- and that `original_stat` will contain the `lstat()` results for the
symlink now, rather than the `stat()` results for its target (again,
iff `is_symlink` is true).
This means that the target of the dangling symlink is no longer created,
which was the bug. The symlink itself continues to be replaced (as
before) with the new file — this is the intended behaviour of
`g_file_replace()`.
The behaviour for non-symlink cases, or cases where the symlink was not
dangling, should be unchanged.
Includes a unit test.
Signed-off-by: Philip Withnall <pwithnall@endlessos.org>
Fixes: #2325
2021-02-24 18:36:07 +01:00
|
|
|
|
static void
|
|
|
|
|
test_replace_symlink (void)
|
|
|
|
|
{
|
|
|
|
|
#ifdef G_OS_UNIX
|
|
|
|
|
gchar *tmpdir_path = NULL;
|
|
|
|
|
GFile *tmpdir = NULL, *source_file = NULL, *target_file = NULL;
|
|
|
|
|
GFileOutputStream *stream = NULL;
|
|
|
|
|
const gchar *new_contents = "this is a test message which should be written to source and not target";
|
|
|
|
|
gsize n_written;
|
|
|
|
|
GFileEnumerator *enumerator = NULL;
|
|
|
|
|
GFileInfo *info = NULL;
|
|
|
|
|
gchar *contents = NULL;
|
|
|
|
|
gsize length = 0;
|
|
|
|
|
GError *local_error = NULL;
|
|
|
|
|
|
|
|
|
|
g_test_bug ("https://gitlab.gnome.org/GNOME/glib/-/issues/2325");
|
|
|
|
|
g_test_summary ("Test that G_FILE_CREATE_REPLACE_DESTINATION doesn’t follow symlinks");
|
|
|
|
|
|
|
|
|
|
/* Create a fresh, empty working directory. */
|
|
|
|
|
tmpdir_path = g_dir_make_tmp ("g_file_replace_symlink_XXXXXX", &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
tmpdir = g_file_new_for_path (tmpdir_path);
|
|
|
|
|
|
|
|
|
|
g_test_message ("Using temporary directory %s", tmpdir_path);
|
|
|
|
|
g_free (tmpdir_path);
|
|
|
|
|
|
|
|
|
|
/* Create symlink `source` which points to `target`. */
|
|
|
|
|
source_file = g_file_get_child (tmpdir, "source");
|
|
|
|
|
target_file = g_file_get_child (tmpdir, "target");
|
|
|
|
|
g_file_make_symbolic_link (source_file, "target", NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
|
|
|
|
|
/* Ensure that `target` doesn’t exist */
|
|
|
|
|
g_assert_false (g_file_query_exists (target_file, NULL));
|
|
|
|
|
|
|
|
|
|
/* Replace the `source` symlink with a regular file using
|
|
|
|
|
* %G_FILE_CREATE_REPLACE_DESTINATION, which should replace it *without*
|
|
|
|
|
* following the symlink */
|
|
|
|
|
stream = g_file_replace (source_file, NULL, FALSE /* no backup */,
|
|
|
|
|
G_FILE_CREATE_REPLACE_DESTINATION, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
|
|
|
|
|
g_output_stream_write_all (G_OUTPUT_STREAM (stream), new_contents, strlen (new_contents),
|
|
|
|
|
&n_written, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
g_assert_cmpint (n_written, ==, strlen (new_contents));
|
|
|
|
|
|
|
|
|
|
g_output_stream_close (G_OUTPUT_STREAM (stream), NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
|
|
|
|
|
g_clear_object (&stream);
|
|
|
|
|
|
|
|
|
|
/* At this point, there should still only be one file: `source`. It should
|
|
|
|
|
* now be a regular file. `target` should not exist. */
|
|
|
|
|
enumerator = g_file_enumerate_children (tmpdir,
|
|
|
|
|
G_FILE_ATTRIBUTE_STANDARD_NAME ","
|
|
|
|
|
G_FILE_ATTRIBUTE_STANDARD_TYPE,
|
|
|
|
|
G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
|
|
|
|
|
info = g_file_enumerator_next_file (enumerator, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
g_assert_nonnull (info);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpstr (g_file_info_get_name (info), ==, "source");
|
|
|
|
|
g_assert_cmpint (g_file_info_get_file_type (info), ==, G_FILE_TYPE_REGULAR);
|
|
|
|
|
|
|
|
|
|
g_clear_object (&info);
|
|
|
|
|
|
|
|
|
|
info = g_file_enumerator_next_file (enumerator, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
g_assert_null (info);
|
|
|
|
|
|
|
|
|
|
g_file_enumerator_close (enumerator, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
g_clear_object (&enumerator);
|
|
|
|
|
|
|
|
|
|
/* Double-check that `target` doesn’t exist */
|
|
|
|
|
g_assert_false (g_file_query_exists (target_file, NULL));
|
|
|
|
|
|
|
|
|
|
/* Check the content of `source`. */
|
|
|
|
|
g_file_load_contents (source_file,
|
|
|
|
|
NULL,
|
|
|
|
|
&contents,
|
|
|
|
|
&length,
|
|
|
|
|
NULL,
|
|
|
|
|
&local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
g_assert_cmpstr (contents, ==, new_contents);
|
|
|
|
|
g_assert_cmpuint (length, ==, strlen (new_contents));
|
|
|
|
|
g_free (contents);
|
|
|
|
|
|
|
|
|
|
/* Tidy up. */
|
|
|
|
|
g_file_delete (source_file, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
|
|
|
|
|
g_file_delete (tmpdir, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
|
|
|
|
|
g_clear_object (&target_file);
|
|
|
|
|
g_clear_object (&source_file);
|
|
|
|
|
g_clear_object (&tmpdir);
|
|
|
|
|
#else /* if !G_OS_UNIX */
|
|
|
|
|
g_test_skip ("Symlink replacement tests can only be run on Unix")
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2012-07-28 15:30:22 +02:00
|
|
|
|
static void
|
|
|
|
|
on_file_deleted (GObject *object,
|
|
|
|
|
GAsyncResult *result,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
GError *local_error = NULL;
|
|
|
|
|
GMainLoop *loop = user_data;
|
|
|
|
|
|
|
|
|
|
(void) g_file_delete_finish ((GFile*)object, result, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
|
|
|
|
|
g_main_loop_quit (loop);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
test_async_delete (void)
|
|
|
|
|
{
|
|
|
|
|
GFile *file;
|
|
|
|
|
GFileIOStream *iostream;
|
|
|
|
|
GError *local_error = NULL;
|
|
|
|
|
GError **error = &local_error;
|
|
|
|
|
GMainLoop *loop;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_tmp ("g_file_delete_XXXXXX",
|
|
|
|
|
&iostream, error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
g_object_unref (iostream);
|
|
|
|
|
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (g_file_query_exists (file, NULL));
|
2012-07-28 15:30:22 +02:00
|
|
|
|
|
|
|
|
|
loop = g_main_loop_new (NULL, TRUE);
|
|
|
|
|
|
|
|
|
|
g_file_delete_async (file, G_PRIORITY_DEFAULT, NULL, on_file_deleted, loop);
|
|
|
|
|
|
|
|
|
|
g_main_loop_run (loop);
|
|
|
|
|
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_false (g_file_query_exists (file, NULL));
|
2012-07-28 15:30:22 +02:00
|
|
|
|
|
|
|
|
|
g_main_loop_unref (loop);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
2013-03-17 23:33:59 +01:00
|
|
|
|
static void
|
|
|
|
|
test_copy_preserve_mode (void)
|
|
|
|
|
{
|
2019-09-27 18:43:29 +02:00
|
|
|
|
#ifdef G_OS_UNIX
|
2019-09-30 15:23:51 +02:00
|
|
|
|
mode_t current_umask = umask (0);
|
|
|
|
|
const struct
|
|
|
|
|
{
|
|
|
|
|
guint32 source_mode;
|
|
|
|
|
guint32 expected_destination_mode;
|
|
|
|
|
gboolean create_destination_before_copy;
|
|
|
|
|
GFileCopyFlags copy_flags;
|
|
|
|
|
}
|
|
|
|
|
vectors[] =
|
|
|
|
|
{
|
|
|
|
|
/* Overwriting the destination file should copy the permissions from the
|
|
|
|
|
* source file, even if %G_FILE_COPY_ALL_METADATA is set: */
|
|
|
|
|
{ 0600, 0600, TRUE, G_FILE_COPY_OVERWRITE | G_FILE_COPY_NOFOLLOW_SYMLINKS | G_FILE_COPY_ALL_METADATA },
|
|
|
|
|
{ 0600, 0600, TRUE, G_FILE_COPY_OVERWRITE | G_FILE_COPY_NOFOLLOW_SYMLINKS },
|
|
|
|
|
/* The same behaviour should hold if the destination file is not being
|
|
|
|
|
* overwritten because it doesn’t already exist: */
|
|
|
|
|
{ 0600, 0600, FALSE, G_FILE_COPY_NOFOLLOW_SYMLINKS | G_FILE_COPY_ALL_METADATA },
|
|
|
|
|
{ 0600, 0600, FALSE, G_FILE_COPY_NOFOLLOW_SYMLINKS },
|
|
|
|
|
/* Anything with %G_FILE_COPY_TARGET_DEFAULT_PERMS should use the current
|
|
|
|
|
* umask for the destination file: */
|
|
|
|
|
{ 0600, 0666 & ~current_umask, TRUE, G_FILE_COPY_TARGET_DEFAULT_PERMS | G_FILE_COPY_OVERWRITE | G_FILE_COPY_NOFOLLOW_SYMLINKS | G_FILE_COPY_ALL_METADATA },
|
|
|
|
|
{ 0600, 0666 & ~current_umask, TRUE, G_FILE_COPY_TARGET_DEFAULT_PERMS | G_FILE_COPY_OVERWRITE | G_FILE_COPY_NOFOLLOW_SYMLINKS },
|
|
|
|
|
{ 0600, 0666 & ~current_umask, FALSE, G_FILE_COPY_TARGET_DEFAULT_PERMS | G_FILE_COPY_NOFOLLOW_SYMLINKS | G_FILE_COPY_ALL_METADATA },
|
|
|
|
|
{ 0600, 0666 & ~current_umask, FALSE, G_FILE_COPY_TARGET_DEFAULT_PERMS | G_FILE_COPY_NOFOLLOW_SYMLINKS },
|
|
|
|
|
};
|
|
|
|
|
gsize i;
|
|
|
|
|
|
|
|
|
|
/* Reset the umask after querying it above. There’s no way to query it without
|
|
|
|
|
* changing it. */
|
|
|
|
|
umask (current_umask);
|
|
|
|
|
g_test_message ("Current umask: %u", current_umask);
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (vectors); i++)
|
|
|
|
|
{
|
|
|
|
|
GFile *tmpfile;
|
|
|
|
|
GFile *dest_tmpfile;
|
|
|
|
|
GFileInfo *dest_info;
|
|
|
|
|
GFileIOStream *iostream;
|
|
|
|
|
GError *local_error = NULL;
|
|
|
|
|
guint32 romode = vectors[i].source_mode;
|
|
|
|
|
guint32 dest_mode;
|
|
|
|
|
|
|
|
|
|
g_test_message ("Vector %" G_GSIZE_FORMAT, i);
|
|
|
|
|
|
|
|
|
|
tmpfile = g_file_new_tmp ("tmp-copy-preserve-modeXXXXXX",
|
|
|
|
|
&iostream, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
g_io_stream_close ((GIOStream*)iostream, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
g_clear_object (&iostream);
|
|
|
|
|
|
|
|
|
|
g_file_set_attribute (tmpfile, G_FILE_ATTRIBUTE_UNIX_MODE, G_FILE_ATTRIBUTE_TYPE_UINT32,
|
|
|
|
|
&romode, G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS,
|
|
|
|
|
NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
|
|
|
|
|
dest_tmpfile = g_file_new_tmp ("tmp-copy-preserve-modeXXXXXX",
|
|
|
|
|
&iostream, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
g_io_stream_close ((GIOStream*)iostream, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
g_clear_object (&iostream);
|
|
|
|
|
|
|
|
|
|
if (!vectors[i].create_destination_before_copy)
|
|
|
|
|
{
|
|
|
|
|
g_file_delete (dest_tmpfile, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_file_copy (tmpfile, dest_tmpfile, vectors[i].copy_flags,
|
|
|
|
|
NULL, NULL, NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
|
|
|
|
|
dest_info = g_file_query_info (dest_tmpfile, G_FILE_ATTRIBUTE_UNIX_MODE, G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS,
|
|
|
|
|
NULL, &local_error);
|
|
|
|
|
g_assert_no_error (local_error);
|
|
|
|
|
|
|
|
|
|
dest_mode = g_file_info_get_attribute_uint32 (dest_info, G_FILE_ATTRIBUTE_UNIX_MODE);
|
|
|
|
|
|
2019-11-21 11:33:43 +01:00
|
|
|
|
g_assert_cmpint (dest_mode & ~S_IFMT, ==, vectors[i].expected_destination_mode);
|
|
|
|
|
g_assert_cmpint (dest_mode & S_IFMT, ==, S_IFREG);
|
2019-09-30 15:23:51 +02:00
|
|
|
|
|
|
|
|
|
(void) g_file_delete (tmpfile, NULL, NULL);
|
|
|
|
|
(void) g_file_delete (dest_tmpfile, NULL, NULL);
|
|
|
|
|
|
|
|
|
|
g_clear_object (&tmpfile);
|
|
|
|
|
g_clear_object (&dest_tmpfile);
|
|
|
|
|
g_clear_object (&dest_info);
|
|
|
|
|
}
|
2019-09-27 18:43:29 +02:00
|
|
|
|
#else /* if !G_OS_UNIX */
|
|
|
|
|
g_test_skip ("File permissions tests can only be run on Unix")
|
2013-03-17 23:33:59 +01:00
|
|
|
|
#endif
|
2019-09-27 18:43:29 +02:00
|
|
|
|
}
|
2013-03-17 23:33:59 +01:00
|
|
|
|
|
2013-12-01 04:51:03 +01:00
|
|
|
|
static gchar *
|
|
|
|
|
splice_to_string (GInputStream *stream,
|
|
|
|
|
GError **error)
|
|
|
|
|
{
|
|
|
|
|
GMemoryOutputStream *buffer = NULL;
|
|
|
|
|
char *ret = NULL;
|
|
|
|
|
|
|
|
|
|
buffer = (GMemoryOutputStream*)g_memory_output_stream_new (NULL, 0, g_realloc, g_free);
|
|
|
|
|
if (g_output_stream_splice ((GOutputStream*)buffer, stream, 0, NULL, error) < 0)
|
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
|
|
if (!g_output_stream_write ((GOutputStream*)buffer, "\0", 1, NULL, error))
|
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
|
|
if (!g_output_stream_close ((GOutputStream*)buffer, NULL, error))
|
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
|
|
ret = g_memory_output_stream_steal_data (buffer);
|
|
|
|
|
out:
|
|
|
|
|
g_clear_object (&buffer);
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-26 09:52:16 +02:00
|
|
|
|
static gboolean
|
|
|
|
|
get_size_from_du (const gchar *path, guint64 *size)
|
2013-12-01 04:51:03 +01:00
|
|
|
|
{
|
|
|
|
|
GSubprocess *du;
|
2018-05-26 09:52:16 +02:00
|
|
|
|
gboolean ok;
|
2013-12-01 04:51:03 +01:00
|
|
|
|
gchar *result;
|
|
|
|
|
gchar *endptr;
|
|
|
|
|
GError *error = NULL;
|
2018-06-27 10:57:37 +02:00
|
|
|
|
gchar *du_path = NULL;
|
|
|
|
|
|
|
|
|
|
/* If we can’t find du, don’t try and run the test. */
|
|
|
|
|
du_path = g_find_program_in_path ("du");
|
|
|
|
|
if (du_path == NULL)
|
|
|
|
|
return FALSE;
|
|
|
|
|
g_free (du_path);
|
2013-12-01 04:51:03 +01:00
|
|
|
|
|
|
|
|
|
du = g_subprocess_new (G_SUBPROCESS_FLAGS_STDOUT_PIPE,
|
|
|
|
|
&error,
|
|
|
|
|
"du", "--bytes", "-s", path, NULL);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
|
|
|
|
result = splice_to_string (g_subprocess_get_stdout_pipe (du), &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
2018-05-26 09:52:16 +02:00
|
|
|
|
*size = g_ascii_strtoll (result, &endptr, 10);
|
|
|
|
|
|
|
|
|
|
g_subprocess_wait (du, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
|
|
|
|
ok = g_subprocess_get_successful (du);
|
2013-12-01 04:51:03 +01:00
|
|
|
|
|
|
|
|
|
g_object_unref (du);
|
|
|
|
|
g_free (result);
|
|
|
|
|
|
2018-05-26 09:52:16 +02:00
|
|
|
|
return ok;
|
2013-12-01 04:51:03 +01:00
|
|
|
|
}
|
|
|
|
|
|
2013-11-29 06:11:47 +01:00
|
|
|
|
static void
|
|
|
|
|
test_measure (void)
|
|
|
|
|
{
|
|
|
|
|
GFile *file;
|
|
|
|
|
guint64 size;
|
2013-12-01 04:51:03 +01:00
|
|
|
|
guint64 num_bytes;
|
2013-11-29 06:11:47 +01:00
|
|
|
|
guint64 num_dirs;
|
|
|
|
|
guint64 num_files;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gboolean ok;
|
|
|
|
|
gchar *path;
|
|
|
|
|
|
|
|
|
|
path = g_test_build_filename (G_TEST_DIST, "desktop-files", NULL);
|
|
|
|
|
file = g_file_new_for_path (path);
|
2013-12-01 04:51:03 +01:00
|
|
|
|
|
2018-06-27 10:57:37 +02:00
|
|
|
|
if (!get_size_from_du (path, &size))
|
2013-12-01 04:51:03 +01:00
|
|
|
|
{
|
2018-05-26 09:52:16 +02:00
|
|
|
|
g_test_message ("du not found or fail to run, skipping byte measurement");
|
2013-12-01 04:51:03 +01:00
|
|
|
|
size = 0;
|
|
|
|
|
}
|
2013-11-29 06:11:47 +01:00
|
|
|
|
|
|
|
|
|
ok = g_file_measure_disk_usage (file,
|
2013-12-01 04:51:03 +01:00
|
|
|
|
G_FILE_MEASURE_APPARENT_SIZE,
|
2013-11-29 06:11:47 +01:00
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
|
|
|
|
NULL,
|
2013-12-01 04:51:03 +01:00
|
|
|
|
&num_bytes,
|
2013-11-29 06:11:47 +01:00
|
|
|
|
&num_dirs,
|
|
|
|
|
&num_files,
|
|
|
|
|
&error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (ok);
|
2013-11-29 06:11:47 +01:00
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
2013-12-01 04:51:03 +01:00
|
|
|
|
if (size > 0)
|
|
|
|
|
g_assert_cmpuint (num_bytes, ==, size);
|
2013-11-29 06:11:47 +01:00
|
|
|
|
g_assert_cmpuint (num_dirs, ==, 6);
|
2021-01-09 16:00:46 +01:00
|
|
|
|
g_assert_cmpuint (num_files, ==, 32);
|
2013-11-29 06:11:47 +01:00
|
|
|
|
|
|
|
|
|
g_object_unref (file);
|
2013-12-01 04:51:03 +01:00
|
|
|
|
g_free (path);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
|
guint64 expected_bytes;
|
|
|
|
|
guint64 expected_dirs;
|
|
|
|
|
guint64 expected_files;
|
|
|
|
|
gint progress_count;
|
|
|
|
|
guint64 progress_bytes;
|
|
|
|
|
guint64 progress_dirs;
|
|
|
|
|
guint64 progress_files;
|
|
|
|
|
} MeasureData;
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
measure_progress (gboolean reporting,
|
|
|
|
|
guint64 current_size,
|
|
|
|
|
guint64 num_dirs,
|
|
|
|
|
guint64 num_files,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
MeasureData *data = user_data;
|
|
|
|
|
|
|
|
|
|
data->progress_count += 1;
|
|
|
|
|
|
|
|
|
|
g_assert_cmpuint (current_size, >=, data->progress_bytes);
|
|
|
|
|
g_assert_cmpuint (num_dirs, >=, data->progress_dirs);
|
|
|
|
|
g_assert_cmpuint (num_files, >=, data->progress_files);
|
|
|
|
|
|
|
|
|
|
data->progress_bytes = current_size;
|
|
|
|
|
data->progress_dirs = num_dirs;
|
|
|
|
|
data->progress_files = num_files;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
measure_done (GObject *source,
|
|
|
|
|
GAsyncResult *res,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
MeasureData *data = user_data;
|
|
|
|
|
guint64 num_bytes, num_dirs, num_files;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gboolean ok;
|
|
|
|
|
|
|
|
|
|
ok = g_file_measure_disk_usage_finish (G_FILE (source), res, &num_bytes, &num_dirs, &num_files, &error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_true (ok);
|
2013-12-01 04:51:03 +01:00
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
|
|
|
|
|
if (data->expected_bytes > 0)
|
|
|
|
|
g_assert_cmpuint (data->expected_bytes, ==, num_bytes);
|
|
|
|
|
g_assert_cmpuint (data->expected_dirs, ==, num_dirs);
|
|
|
|
|
g_assert_cmpuint (data->expected_files, ==, num_files);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpuint (data->progress_count, >, 0);
|
|
|
|
|
g_assert_cmpuint (num_bytes, >=, data->progress_bytes);
|
|
|
|
|
g_assert_cmpuint (num_dirs, >=, data->progress_dirs);
|
|
|
|
|
g_assert_cmpuint (num_files, >=, data->progress_files);
|
|
|
|
|
|
|
|
|
|
g_free (data);
|
|
|
|
|
g_object_unref (source);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
test_measure_async (void)
|
|
|
|
|
{
|
|
|
|
|
gchar *path;
|
|
|
|
|
GFile *file;
|
|
|
|
|
MeasureData *data;
|
|
|
|
|
|
|
|
|
|
data = g_new (MeasureData, 1);
|
|
|
|
|
|
|
|
|
|
data->progress_count = 0;
|
|
|
|
|
data->progress_bytes = 0;
|
|
|
|
|
data->progress_files = 0;
|
|
|
|
|
data->progress_dirs = 0;
|
|
|
|
|
|
|
|
|
|
path = g_test_build_filename (G_TEST_DIST, "desktop-files", NULL);
|
|
|
|
|
file = g_file_new_for_path (path);
|
|
|
|
|
|
2018-06-27 10:57:37 +02:00
|
|
|
|
if (!get_size_from_du (path, &data->expected_bytes))
|
2013-12-01 04:51:03 +01:00
|
|
|
|
{
|
2018-05-26 09:52:16 +02:00
|
|
|
|
g_test_message ("du not found or fail to run, skipping byte measurement");
|
2013-12-01 04:51:03 +01:00
|
|
|
|
data->expected_bytes = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
g_free (path);
|
|
|
|
|
|
|
|
|
|
data->expected_dirs = 6;
|
2021-01-09 16:00:46 +01:00
|
|
|
|
data->expected_files = 32;
|
2013-12-01 04:51:03 +01:00
|
|
|
|
|
|
|
|
|
g_file_measure_disk_usage_async (file,
|
|
|
|
|
G_FILE_MEASURE_APPARENT_SIZE,
|
|
|
|
|
0, NULL,
|
|
|
|
|
measure_progress, data,
|
|
|
|
|
measure_done, data);
|
2013-11-29 06:11:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
2017-11-14 10:40:44 +01:00
|
|
|
|
static void
|
|
|
|
|
test_load_bytes (void)
|
|
|
|
|
{
|
|
|
|
|
gchar filename[] = "g_file_load_bytes_XXXXXX";
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
GBytes *bytes;
|
|
|
|
|
GFile *file;
|
|
|
|
|
int len;
|
|
|
|
|
int fd;
|
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
|
|
fd = g_mkstemp (filename);
|
|
|
|
|
g_assert_cmpint (fd, !=, -1);
|
|
|
|
|
len = strlen ("test_load_bytes");
|
|
|
|
|
ret = write (fd, "test_load_bytes", len);
|
|
|
|
|
g_assert_cmpint (ret, ==, len);
|
|
|
|
|
close (fd);
|
|
|
|
|
|
|
|
|
|
file = g_file_new_for_path (filename);
|
|
|
|
|
bytes = g_file_load_bytes (file, NULL, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_nonnull (bytes);
|
2017-11-14 10:40:44 +01:00
|
|
|
|
g_assert_cmpint (len, ==, g_bytes_get_size (bytes));
|
|
|
|
|
g_assert_cmpstr ("test_load_bytes", ==, (gchar *)g_bytes_get_data (bytes, NULL));
|
|
|
|
|
|
|
|
|
|
g_file_delete (file, NULL, NULL);
|
|
|
|
|
|
|
|
|
|
g_bytes_unref (bytes);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
|
{
|
|
|
|
|
GMainLoop *main_loop;
|
|
|
|
|
GFile *file;
|
|
|
|
|
GBytes *bytes;
|
|
|
|
|
} LoadBytesAsyncData;
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
test_load_bytes_cb (GObject *object,
|
|
|
|
|
GAsyncResult *result,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
GFile *file = G_FILE (object);
|
|
|
|
|
LoadBytesAsyncData *data = user_data;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
|
|
data->bytes = g_file_load_bytes_finish (file, result, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
2020-05-22 12:04:34 +02:00
|
|
|
|
g_assert_nonnull (data->bytes);
|
2017-11-14 10:40:44 +01:00
|
|
|
|
|
|
|
|
|
g_main_loop_quit (data->main_loop);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
test_load_bytes_async (void)
|
|
|
|
|
{
|
|
|
|
|
LoadBytesAsyncData data = { 0 };
|
|
|
|
|
gchar filename[] = "g_file_load_bytes_XXXXXX";
|
|
|
|
|
int len;
|
|
|
|
|
int fd;
|
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
|
|
fd = g_mkstemp (filename);
|
|
|
|
|
g_assert_cmpint (fd, !=, -1);
|
|
|
|
|
len = strlen ("test_load_bytes_async");
|
|
|
|
|
ret = write (fd, "test_load_bytes_async", len);
|
|
|
|
|
g_assert_cmpint (ret, ==, len);
|
|
|
|
|
close (fd);
|
|
|
|
|
|
|
|
|
|
data.main_loop = g_main_loop_new (NULL, FALSE);
|
|
|
|
|
data.file = g_file_new_for_path (filename);
|
|
|
|
|
|
|
|
|
|
g_file_load_bytes_async (data.file, NULL, test_load_bytes_cb, &data);
|
|
|
|
|
g_main_loop_run (data.main_loop);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpint (len, ==, g_bytes_get_size (data.bytes));
|
|
|
|
|
g_assert_cmpstr ("test_load_bytes_async", ==, (gchar *)g_bytes_get_data (data.bytes, NULL));
|
|
|
|
|
|
|
|
|
|
g_file_delete (data.file, NULL, NULL);
|
|
|
|
|
g_object_unref (data.file);
|
|
|
|
|
g_bytes_unref (data.bytes);
|
|
|
|
|
g_main_loop_unref (data.main_loop);
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-14 11:54:00 +02:00
|
|
|
|
static void
|
|
|
|
|
test_writev_helper (GOutputVector *vectors,
|
|
|
|
|
gsize n_vectors,
|
|
|
|
|
gboolean use_bytes_written,
|
|
|
|
|
const guint8 *expected_contents,
|
|
|
|
|
gsize expected_length)
|
|
|
|
|
{
|
|
|
|
|
GFile *file;
|
|
|
|
|
GFileIOStream *iostream = NULL;
|
|
|
|
|
GOutputStream *ostream;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gsize bytes_written = 0;
|
|
|
|
|
gboolean res;
|
|
|
|
|
guint8 *contents;
|
|
|
|
|
gsize length;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_tmp ("g_file_writev_XXXXXX",
|
|
|
|
|
&iostream, NULL);
|
|
|
|
|
g_assert_nonnull (file);
|
|
|
|
|
g_assert_nonnull (iostream);
|
|
|
|
|
|
|
|
|
|
ostream = g_io_stream_get_output_stream (G_IO_STREAM (iostream));
|
|
|
|
|
|
|
|
|
|
res = g_output_stream_writev_all (ostream, vectors, n_vectors, use_bytes_written ? &bytes_written : NULL, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
if (use_bytes_written)
|
|
|
|
|
g_assert_cmpuint (bytes_written, ==, expected_length);
|
|
|
|
|
|
|
|
|
|
res = g_io_stream_close (G_IO_STREAM (iostream), NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_object_unref (iostream);
|
|
|
|
|
|
|
|
|
|
res = g_file_load_contents (file, NULL, (gchar **) &contents, &length, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpmem (contents, length, expected_contents, expected_length);
|
|
|
|
|
|
|
|
|
|
g_free (contents);
|
|
|
|
|
|
|
|
|
|
g_file_delete (file, NULL, NULL);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Test that writev() on local file output streams works on a non-empty vector */
|
|
|
|
|
static void
|
|
|
|
|
test_writev (void)
|
|
|
|
|
{
|
|
|
|
|
GOutputVector vectors[3];
|
|
|
|
|
const guint8 buffer[] = {1, 2, 3, 4, 5,
|
|
|
|
|
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12,
|
|
|
|
|
1, 2, 3};
|
|
|
|
|
|
|
|
|
|
vectors[0].buffer = buffer;
|
|
|
|
|
vectors[0].size = 5;
|
|
|
|
|
|
|
|
|
|
vectors[1].buffer = buffer + 5;
|
|
|
|
|
vectors[1].size = 12;
|
|
|
|
|
|
|
|
|
|
vectors[2].buffer = buffer + 5 + 12;
|
|
|
|
|
vectors[2].size = 3;
|
|
|
|
|
|
|
|
|
|
test_writev_helper (vectors, G_N_ELEMENTS (vectors), TRUE, buffer, sizeof buffer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Test that writev() on local file output streams works on a non-empty vector without returning bytes_written */
|
|
|
|
|
static void
|
|
|
|
|
test_writev_no_bytes_written (void)
|
|
|
|
|
{
|
|
|
|
|
GOutputVector vectors[3];
|
|
|
|
|
const guint8 buffer[] = {1, 2, 3, 4, 5,
|
|
|
|
|
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12,
|
|
|
|
|
1, 2, 3};
|
|
|
|
|
|
|
|
|
|
vectors[0].buffer = buffer;
|
|
|
|
|
vectors[0].size = 5;
|
|
|
|
|
|
|
|
|
|
vectors[1].buffer = buffer + 5;
|
|
|
|
|
vectors[1].size = 12;
|
|
|
|
|
|
|
|
|
|
vectors[2].buffer = buffer + 5 + 12;
|
|
|
|
|
vectors[2].size = 3;
|
|
|
|
|
|
|
|
|
|
test_writev_helper (vectors, G_N_ELEMENTS (vectors), FALSE, buffer, sizeof buffer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Test that writev() on local file output streams works on 0 vectors */
|
|
|
|
|
static void
|
|
|
|
|
test_writev_no_vectors (void)
|
|
|
|
|
{
|
|
|
|
|
test_writev_helper (NULL, 0, TRUE, NULL, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Test that writev() on local file output streams works on empty vectors */
|
|
|
|
|
static void
|
|
|
|
|
test_writev_empty_vectors (void)
|
|
|
|
|
{
|
|
|
|
|
GOutputVector vectors[3];
|
|
|
|
|
|
|
|
|
|
vectors[0].buffer = NULL;
|
|
|
|
|
vectors[0].size = 0;
|
|
|
|
|
vectors[1].buffer = NULL;
|
|
|
|
|
vectors[1].size = 0;
|
|
|
|
|
vectors[2].buffer = NULL;
|
|
|
|
|
vectors[2].size = 0;
|
|
|
|
|
|
|
|
|
|
test_writev_helper (vectors, G_N_ELEMENTS (vectors), TRUE, NULL, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Test that writev() fails if the sum of sizes in the vector is too big */
|
|
|
|
|
static void
|
|
|
|
|
test_writev_too_big_vectors (void)
|
|
|
|
|
{
|
|
|
|
|
GFile *file;
|
|
|
|
|
GFileIOStream *iostream = NULL;
|
|
|
|
|
GOutputStream *ostream;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gsize bytes_written = 0;
|
|
|
|
|
gboolean res;
|
|
|
|
|
guint8 *contents;
|
|
|
|
|
gsize length;
|
|
|
|
|
GOutputVector vectors[3];
|
|
|
|
|
|
|
|
|
|
vectors[0].buffer = (void*) 1;
|
|
|
|
|
vectors[0].size = G_MAXSIZE / 2;
|
|
|
|
|
|
|
|
|
|
vectors[1].buffer = (void*) 1;
|
|
|
|
|
vectors[1].size = G_MAXSIZE / 2;
|
|
|
|
|
|
|
|
|
|
vectors[2].buffer = (void*) 1;
|
|
|
|
|
vectors[2].size = G_MAXSIZE / 2;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_tmp ("g_file_writev_XXXXXX",
|
|
|
|
|
&iostream, NULL);
|
|
|
|
|
g_assert_nonnull (file);
|
|
|
|
|
g_assert_nonnull (iostream);
|
|
|
|
|
|
|
|
|
|
ostream = g_io_stream_get_output_stream (G_IO_STREAM (iostream));
|
|
|
|
|
|
|
|
|
|
res = g_output_stream_writev_all (ostream, vectors, G_N_ELEMENTS (vectors), &bytes_written, NULL, &error);
|
|
|
|
|
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT);
|
|
|
|
|
g_assert_cmpuint (bytes_written, ==, 0);
|
|
|
|
|
g_assert_false (res);
|
|
|
|
|
g_clear_error (&error);
|
|
|
|
|
|
|
|
|
|
res = g_io_stream_close (G_IO_STREAM (iostream), NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_object_unref (iostream);
|
|
|
|
|
|
|
|
|
|
res = g_file_load_contents (file, NULL, (gchar **) &contents, &length, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpmem (contents, length, NULL, 0);
|
|
|
|
|
|
|
|
|
|
g_free (contents);
|
|
|
|
|
|
|
|
|
|
g_file_delete (file, NULL, NULL);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
|
{
|
|
|
|
|
gsize bytes_written;
|
|
|
|
|
GOutputVector *vectors;
|
|
|
|
|
gsize n_vectors;
|
|
|
|
|
GError *error;
|
|
|
|
|
gboolean done;
|
|
|
|
|
} WritevAsyncData;
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
test_writev_async_cb (GObject *object,
|
|
|
|
|
GAsyncResult *result,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
GOutputStream *ostream = G_OUTPUT_STREAM (object);
|
|
|
|
|
WritevAsyncData *data = user_data;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gsize bytes_written;
|
|
|
|
|
gboolean res;
|
|
|
|
|
|
|
|
|
|
res = g_output_stream_writev_finish (ostream, result, &bytes_written, &error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
data->bytes_written += bytes_written;
|
|
|
|
|
|
|
|
|
|
/* skip vectors that have been written in full */
|
|
|
|
|
while (data->n_vectors > 0 && bytes_written >= data->vectors[0].size)
|
|
|
|
|
{
|
|
|
|
|
bytes_written -= data->vectors[0].size;
|
|
|
|
|
++data->vectors;
|
|
|
|
|
--data->n_vectors;
|
|
|
|
|
}
|
|
|
|
|
/* skip partially written vector data */
|
|
|
|
|
if (bytes_written > 0 && data->n_vectors > 0)
|
|
|
|
|
{
|
|
|
|
|
data->vectors[0].size -= bytes_written;
|
|
|
|
|
data->vectors[0].buffer = ((guint8 *) data->vectors[0].buffer) + bytes_written;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (data->n_vectors > 0)
|
|
|
|
|
g_output_stream_writev_async (ostream, data->vectors, data->n_vectors, 0, NULL, test_writev_async_cb, &data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Test that writev_async() on local file output streams works on a non-empty vector */
|
|
|
|
|
static void
|
|
|
|
|
test_writev_async (void)
|
|
|
|
|
{
|
|
|
|
|
WritevAsyncData data = { 0 };
|
|
|
|
|
GFile *file;
|
|
|
|
|
GFileIOStream *iostream = NULL;
|
|
|
|
|
GOutputVector vectors[3];
|
|
|
|
|
const guint8 buffer[] = {1, 2, 3, 4, 5,
|
|
|
|
|
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12,
|
|
|
|
|
1, 2, 3};
|
|
|
|
|
GOutputStream *ostream;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gboolean res;
|
|
|
|
|
guint8 *contents;
|
|
|
|
|
gsize length;
|
|
|
|
|
|
|
|
|
|
vectors[0].buffer = buffer;
|
|
|
|
|
vectors[0].size = 5;
|
|
|
|
|
|
|
|
|
|
vectors[1].buffer = buffer + 5;
|
|
|
|
|
vectors[1].size = 12;
|
|
|
|
|
|
|
|
|
|
vectors[2].buffer = buffer + 5 + 12;
|
|
|
|
|
vectors[2].size = 3;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_tmp ("g_file_writev_XXXXXX",
|
|
|
|
|
&iostream, NULL);
|
|
|
|
|
g_assert_nonnull (file);
|
|
|
|
|
g_assert_nonnull (iostream);
|
|
|
|
|
|
|
|
|
|
data.vectors = vectors;
|
|
|
|
|
data.n_vectors = G_N_ELEMENTS (vectors);
|
|
|
|
|
|
|
|
|
|
ostream = g_io_stream_get_output_stream (G_IO_STREAM (iostream));
|
|
|
|
|
|
|
|
|
|
g_output_stream_writev_async (ostream, data.vectors, data.n_vectors, 0, NULL, test_writev_async_cb, &data);
|
|
|
|
|
|
|
|
|
|
while (data.n_vectors > 0)
|
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpuint (data.bytes_written, ==, sizeof buffer);
|
|
|
|
|
|
|
|
|
|
res = g_io_stream_close (G_IO_STREAM (iostream), NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_object_unref (iostream);
|
|
|
|
|
|
|
|
|
|
res = g_file_load_contents (file, NULL, (gchar **) &contents, &length, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpmem (contents, length, buffer, sizeof buffer);
|
|
|
|
|
|
|
|
|
|
g_free (contents);
|
|
|
|
|
|
|
|
|
|
g_file_delete (file, NULL, NULL);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
test_writev_all_cb (GObject *object,
|
|
|
|
|
GAsyncResult *result,
|
|
|
|
|
gpointer user_data)
|
|
|
|
|
{
|
|
|
|
|
GOutputStream *ostream = G_OUTPUT_STREAM (object);
|
|
|
|
|
WritevAsyncData *data = user_data;
|
|
|
|
|
|
|
|
|
|
g_output_stream_writev_all_finish (ostream, result, &data->bytes_written, &data->error);
|
|
|
|
|
data->done = TRUE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Test that writev_async_all() on local file output streams works on a non-empty vector */
|
|
|
|
|
static void
|
|
|
|
|
test_writev_async_all (void)
|
|
|
|
|
{
|
|
|
|
|
WritevAsyncData data = { 0 };
|
|
|
|
|
GFile *file;
|
|
|
|
|
GFileIOStream *iostream = NULL;
|
|
|
|
|
GOutputStream *ostream;
|
|
|
|
|
GOutputVector vectors[3];
|
|
|
|
|
const guint8 buffer[] = {1, 2, 3, 4, 5,
|
|
|
|
|
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12,
|
|
|
|
|
1, 2, 3};
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gboolean res;
|
|
|
|
|
guint8 *contents;
|
|
|
|
|
gsize length;
|
|
|
|
|
|
|
|
|
|
vectors[0].buffer = buffer;
|
|
|
|
|
vectors[0].size = 5;
|
|
|
|
|
|
|
|
|
|
vectors[1].buffer = buffer + 5;
|
|
|
|
|
vectors[1].size = 12;
|
|
|
|
|
|
|
|
|
|
vectors[2].buffer = buffer + 5 + 12;
|
|
|
|
|
vectors[2].size = 3;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_tmp ("g_file_writev_XXXXXX",
|
|
|
|
|
&iostream, NULL);
|
|
|
|
|
g_assert_nonnull (file);
|
|
|
|
|
g_assert_nonnull (iostream);
|
|
|
|
|
|
|
|
|
|
ostream = g_io_stream_get_output_stream (G_IO_STREAM (iostream));
|
|
|
|
|
|
|
|
|
|
g_output_stream_writev_all_async (ostream, vectors, G_N_ELEMENTS (vectors), 0, NULL, test_writev_all_cb, &data);
|
|
|
|
|
|
|
|
|
|
while (!data.done)
|
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpuint (data.bytes_written, ==, sizeof buffer);
|
|
|
|
|
g_assert_no_error (data.error);
|
|
|
|
|
|
|
|
|
|
res = g_io_stream_close (G_IO_STREAM (iostream), NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_object_unref (iostream);
|
|
|
|
|
|
|
|
|
|
res = g_file_load_contents (file, NULL, (gchar **) &contents, &length, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpmem (contents, length, buffer, sizeof buffer);
|
|
|
|
|
|
|
|
|
|
g_free (contents);
|
|
|
|
|
|
|
|
|
|
g_file_delete (file, NULL, NULL);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Test that writev_async_all() on local file output streams handles cancellation correctly */
|
|
|
|
|
static void
|
|
|
|
|
test_writev_async_all_cancellation (void)
|
|
|
|
|
{
|
|
|
|
|
WritevAsyncData data = { 0 };
|
|
|
|
|
GFile *file;
|
|
|
|
|
GFileIOStream *iostream = NULL;
|
|
|
|
|
GOutputVector vectors[3];
|
|
|
|
|
const guint8 buffer[] = {1, 2, 3, 4, 5,
|
|
|
|
|
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12,
|
|
|
|
|
1, 2, 3};
|
|
|
|
|
GOutputStream *ostream;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gboolean res;
|
|
|
|
|
guint8 *contents;
|
|
|
|
|
gsize length;
|
|
|
|
|
GCancellable *cancellable;
|
|
|
|
|
|
|
|
|
|
vectors[0].buffer = buffer;
|
|
|
|
|
vectors[0].size = 5;
|
|
|
|
|
|
|
|
|
|
vectors[1].buffer = buffer + 5;
|
|
|
|
|
vectors[1].size = 12;
|
|
|
|
|
|
|
|
|
|
vectors[2].buffer = buffer + 5 + 12;
|
|
|
|
|
vectors[2].size = 3;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_tmp ("g_file_writev_XXXXXX",
|
|
|
|
|
&iostream, NULL);
|
|
|
|
|
g_assert_nonnull (file);
|
|
|
|
|
g_assert_nonnull (iostream);
|
|
|
|
|
|
|
|
|
|
ostream = g_io_stream_get_output_stream (G_IO_STREAM (iostream));
|
|
|
|
|
|
|
|
|
|
cancellable = g_cancellable_new ();
|
2019-01-26 12:38:20 +01:00
|
|
|
|
g_cancellable_cancel (cancellable);
|
2018-09-14 11:54:00 +02:00
|
|
|
|
|
|
|
|
|
g_output_stream_writev_all_async (ostream, vectors, G_N_ELEMENTS (vectors), 0, cancellable, test_writev_all_cb, &data);
|
|
|
|
|
|
|
|
|
|
while (!data.done)
|
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpuint (data.bytes_written, ==, 0);
|
|
|
|
|
g_assert_error (data.error, G_IO_ERROR, G_IO_ERROR_CANCELLED);
|
|
|
|
|
g_clear_error (&data.error);
|
|
|
|
|
|
|
|
|
|
res = g_io_stream_close (G_IO_STREAM (iostream), NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_object_unref (iostream);
|
|
|
|
|
|
|
|
|
|
res = g_file_load_contents (file, NULL, (gchar **) &contents, &length, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_assert_cmpuint (length, ==, 0);
|
|
|
|
|
|
|
|
|
|
g_free (contents);
|
|
|
|
|
|
|
|
|
|
g_file_delete (file, NULL, NULL);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
g_object_unref (cancellable);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Test that writev_async_all() with empty vectors is handled correctly */
|
|
|
|
|
static void
|
|
|
|
|
test_writev_async_all_empty_vectors (void)
|
|
|
|
|
{
|
|
|
|
|
WritevAsyncData data = { 0 };
|
|
|
|
|
GFile *file;
|
|
|
|
|
GFileIOStream *iostream = NULL;
|
|
|
|
|
GOutputVector vectors[3];
|
|
|
|
|
GOutputStream *ostream;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gboolean res;
|
|
|
|
|
guint8 *contents;
|
|
|
|
|
gsize length;
|
|
|
|
|
|
|
|
|
|
vectors[0].buffer = NULL;
|
|
|
|
|
vectors[0].size = 0;
|
|
|
|
|
|
|
|
|
|
vectors[1].buffer = NULL;
|
|
|
|
|
vectors[1].size = 0;
|
|
|
|
|
|
|
|
|
|
vectors[2].buffer = NULL;
|
|
|
|
|
vectors[2].size = 0;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_tmp ("g_file_writev_XXXXXX",
|
|
|
|
|
&iostream, NULL);
|
|
|
|
|
g_assert_nonnull (file);
|
|
|
|
|
g_assert_nonnull (iostream);
|
|
|
|
|
|
|
|
|
|
ostream = g_io_stream_get_output_stream (G_IO_STREAM (iostream));
|
|
|
|
|
|
|
|
|
|
g_output_stream_writev_all_async (ostream, vectors, G_N_ELEMENTS (vectors), 0, NULL, test_writev_all_cb, &data);
|
|
|
|
|
|
|
|
|
|
while (!data.done)
|
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpuint (data.bytes_written, ==, 0);
|
|
|
|
|
g_assert_no_error (data.error);
|
|
|
|
|
g_clear_error (&data.error);
|
|
|
|
|
|
|
|
|
|
res = g_io_stream_close (G_IO_STREAM (iostream), NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_object_unref (iostream);
|
|
|
|
|
|
|
|
|
|
res = g_file_load_contents (file, NULL, (gchar **) &contents, &length, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_assert_cmpuint (length, ==, 0);
|
|
|
|
|
|
|
|
|
|
g_free (contents);
|
|
|
|
|
|
|
|
|
|
g_file_delete (file, NULL, NULL);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Test that writev_async_all() with no vectors is handled correctly */
|
|
|
|
|
static void
|
|
|
|
|
test_writev_async_all_no_vectors (void)
|
|
|
|
|
{
|
|
|
|
|
WritevAsyncData data = { 0 };
|
|
|
|
|
GFile *file;
|
|
|
|
|
GFileIOStream *iostream = NULL;
|
|
|
|
|
GOutputStream *ostream;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gboolean res;
|
|
|
|
|
guint8 *contents;
|
|
|
|
|
gsize length;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_tmp ("g_file_writev_XXXXXX",
|
|
|
|
|
&iostream, NULL);
|
|
|
|
|
g_assert_nonnull (file);
|
|
|
|
|
g_assert_nonnull (iostream);
|
|
|
|
|
|
|
|
|
|
ostream = g_io_stream_get_output_stream (G_IO_STREAM (iostream));
|
|
|
|
|
|
|
|
|
|
g_output_stream_writev_all_async (ostream, NULL, 0, 0, NULL, test_writev_all_cb, &data);
|
|
|
|
|
|
|
|
|
|
while (!data.done)
|
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpuint (data.bytes_written, ==, 0);
|
|
|
|
|
g_assert_no_error (data.error);
|
|
|
|
|
g_clear_error (&data.error);
|
|
|
|
|
|
|
|
|
|
res = g_io_stream_close (G_IO_STREAM (iostream), NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_object_unref (iostream);
|
|
|
|
|
|
|
|
|
|
res = g_file_load_contents (file, NULL, (gchar **) &contents, &length, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_assert_cmpuint (length, ==, 0);
|
|
|
|
|
|
|
|
|
|
g_free (contents);
|
|
|
|
|
|
|
|
|
|
g_file_delete (file, NULL, NULL);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Test that writev_async_all() with too big vectors is handled correctly */
|
|
|
|
|
static void
|
|
|
|
|
test_writev_async_all_too_big_vectors (void)
|
|
|
|
|
{
|
|
|
|
|
WritevAsyncData data = { 0 };
|
|
|
|
|
GFile *file;
|
|
|
|
|
GFileIOStream *iostream = NULL;
|
|
|
|
|
GOutputVector vectors[3];
|
|
|
|
|
GOutputStream *ostream;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
gboolean res;
|
|
|
|
|
guint8 *contents;
|
|
|
|
|
gsize length;
|
|
|
|
|
|
|
|
|
|
vectors[0].buffer = (void*) 1;
|
|
|
|
|
vectors[0].size = G_MAXSIZE / 2;
|
|
|
|
|
|
|
|
|
|
vectors[1].buffer = (void*) 1;
|
|
|
|
|
vectors[1].size = G_MAXSIZE / 2;
|
|
|
|
|
|
|
|
|
|
vectors[2].buffer = (void*) 1;
|
|
|
|
|
vectors[2].size = G_MAXSIZE / 2;
|
|
|
|
|
|
|
|
|
|
file = g_file_new_tmp ("g_file_writev_XXXXXX",
|
|
|
|
|
&iostream, NULL);
|
|
|
|
|
g_assert_nonnull (file);
|
|
|
|
|
g_assert_nonnull (iostream);
|
|
|
|
|
|
|
|
|
|
ostream = g_io_stream_get_output_stream (G_IO_STREAM (iostream));
|
|
|
|
|
|
|
|
|
|
g_output_stream_writev_all_async (ostream, vectors, G_N_ELEMENTS (vectors), 0, NULL, test_writev_all_cb, &data);
|
|
|
|
|
|
|
|
|
|
while (!data.done)
|
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
|
|
g_assert_cmpuint (data.bytes_written, ==, 0);
|
|
|
|
|
g_assert_error (data.error, G_IO_ERROR, G_IO_ERROR_INVALID_ARGUMENT);
|
|
|
|
|
g_clear_error (&data.error);
|
|
|
|
|
|
|
|
|
|
res = g_io_stream_close (G_IO_STREAM (iostream), NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_object_unref (iostream);
|
|
|
|
|
|
|
|
|
|
res = g_file_load_contents (file, NULL, (gchar **) &contents, &length, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_true (res);
|
|
|
|
|
g_assert_cmpuint (length, ==, 0);
|
|
|
|
|
|
|
|
|
|
g_free (contents);
|
|
|
|
|
|
|
|
|
|
g_file_delete (file, NULL, NULL);
|
|
|
|
|
g_object_unref (file);
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-07 18:20:01 +02:00
|
|
|
|
static void
|
|
|
|
|
test_build_attribute_list_for_copy (void)
|
|
|
|
|
{
|
|
|
|
|
GFile *tmpfile;
|
|
|
|
|
GFileIOStream *iostream;
|
|
|
|
|
GError *error = NULL;
|
|
|
|
|
const GFileCopyFlags test_flags[] =
|
|
|
|
|
{
|
|
|
|
|
G_FILE_COPY_NONE,
|
|
|
|
|
G_FILE_COPY_TARGET_DEFAULT_PERMS,
|
|
|
|
|
G_FILE_COPY_ALL_METADATA,
|
|
|
|
|
G_FILE_COPY_ALL_METADATA | G_FILE_COPY_TARGET_DEFAULT_PERMS,
|
|
|
|
|
};
|
|
|
|
|
gsize i;
|
|
|
|
|
char *attrs;
|
|
|
|
|
gchar *attrs_with_commas;
|
|
|
|
|
|
|
|
|
|
tmpfile = g_file_new_tmp ("tmp-build-attribute-list-for-copyXXXXXX",
|
|
|
|
|
&iostream, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_io_stream_close ((GIOStream*)iostream, NULL, &error);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_clear_object (&iostream);
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (test_flags); i++)
|
|
|
|
|
{
|
|
|
|
|
GFileCopyFlags flags = test_flags[i];
|
|
|
|
|
|
|
|
|
|
attrs = g_file_build_attribute_list_for_copy (tmpfile, flags, NULL, &error);
|
|
|
|
|
g_test_message ("Attributes for copy: %s", attrs);
|
|
|
|
|
g_assert_no_error (error);
|
|
|
|
|
g_assert_nonnull (attrs);
|
|
|
|
|
attrs_with_commas = g_strconcat (",", attrs, ",", NULL);
|
|
|
|
|
g_free (attrs);
|
|
|
|
|
|
|
|
|
|
/* See g_local_file_class_init for reference. */
|
|
|
|
|
if (flags & G_FILE_COPY_TARGET_DEFAULT_PERMS)
|
|
|
|
|
g_assert_null (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_UNIX_MODE ","));
|
|
|
|
|
else
|
|
|
|
|
g_assert_nonnull (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_UNIX_MODE ","));
|
|
|
|
|
#ifdef G_OS_UNIX
|
|
|
|
|
if (flags & G_FILE_COPY_ALL_METADATA)
|
|
|
|
|
{
|
|
|
|
|
g_assert_nonnull (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_UNIX_UID ","));
|
|
|
|
|
g_assert_nonnull (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_UNIX_GID ","));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
g_assert_null (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_UNIX_UID ","));
|
|
|
|
|
g_assert_null (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_UNIX_GID ","));
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef HAVE_UTIMES
|
|
|
|
|
g_assert_nonnull (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_TIME_MODIFIED ","));
|
|
|
|
|
g_assert_nonnull (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_TIME_MODIFIED_USEC ","));
|
|
|
|
|
if (flags & G_FILE_COPY_ALL_METADATA)
|
|
|
|
|
{
|
|
|
|
|
g_assert_nonnull (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_TIME_ACCESS ","));
|
|
|
|
|
g_assert_nonnull (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_TIME_ACCESS_USEC ","));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
g_assert_null (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_TIME_ACCESS ","));
|
|
|
|
|
g_assert_null (g_strstr_len (attrs_with_commas, -1, "," G_FILE_ATTRIBUTE_TIME_ACCESS_USEC ","));
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
g_free (attrs_with_commas);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
(void) g_file_delete (tmpfile, NULL, NULL);
|
|
|
|
|
g_clear_object (&tmpfile);
|
|
|
|
|
}
|
|
|
|
|
|
2010-07-05 09:09:36 +02:00
|
|
|
|
int
|
|
|
|
|
main (int argc, char *argv[])
|
|
|
|
|
{
|
|
|
|
|
g_test_init (&argc, &argv, NULL);
|
|
|
|
|
|
|
|
|
|
g_test_add_func ("/file/basic", test_basic);
|
2017-11-04 19:50:38 +01:00
|
|
|
|
g_test_add_func ("/file/build-filename", test_build_filename);
|
2010-07-05 09:09:36 +02:00
|
|
|
|
g_test_add_func ("/file/parent", test_parent);
|
2010-07-27 22:24:08 +02:00
|
|
|
|
g_test_add_func ("/file/child", test_child);
|
|
|
|
|
g_test_add_func ("/file/type", test_type);
|
2013-06-01 06:36:16 +02:00
|
|
|
|
g_test_add_func ("/file/parse-name", test_parse_name);
|
2010-08-13 23:21:22 +02:00
|
|
|
|
g_test_add_data_func ("/file/async-create-delete/0", GINT_TO_POINTER (0), test_create_delete);
|
|
|
|
|
g_test_add_data_func ("/file/async-create-delete/1", GINT_TO_POINTER (1), test_create_delete);
|
|
|
|
|
g_test_add_data_func ("/file/async-create-delete/10", GINT_TO_POINTER (10), test_create_delete);
|
|
|
|
|
g_test_add_data_func ("/file/async-create-delete/25", GINT_TO_POINTER (25), test_create_delete);
|
|
|
|
|
g_test_add_data_func ("/file/async-create-delete/4096", GINT_TO_POINTER (4096), test_create_delete);
|
2010-08-14 01:40:48 +02:00
|
|
|
|
g_test_add_func ("/file/replace-load", test_replace_load);
|
2012-11-07 16:36:05 +01:00
|
|
|
|
g_test_add_func ("/file/replace-cancel", test_replace_cancel);
|
glocalfileoutputstream: Fix CREATE_REPLACE_DESTINATION with symlinks
The `G_FILE_CREATE_REPLACE_DESTINATION` flag is equivalent to unlinking
the destination file and re-creating it from scratch. That did
previously work, but in the process the code would call `open(O_CREAT)`
on the file. If the file was a dangling symlink, this would create the
destination file (empty). That’s not an intended side-effect, and has
security implications if the symlink is controlled by a lower-privileged
process.
Fix that by not opening the destination file if it’s a symlink, and
adjusting the rest of the code to cope with
- the fact that `fd == -1` is not an error iff `is_symlink` is true,
- and that `original_stat` will contain the `lstat()` results for the
symlink now, rather than the `stat()` results for its target (again,
iff `is_symlink` is true).
This means that the target of the dangling symlink is no longer created,
which was the bug. The symlink itself continues to be replaced (as
before) with the new file — this is the intended behaviour of
`g_file_replace()`.
The behaviour for non-symlink cases, or cases where the symlink was not
dangling, should be unchanged.
Includes a unit test.
Signed-off-by: Philip Withnall <pwithnall@endlessos.org>
Fixes: #2325
2021-02-24 18:36:07 +01:00
|
|
|
|
g_test_add_func ("/file/replace-symlink", test_replace_symlink);
|
2012-07-28 15:30:22 +02:00
|
|
|
|
g_test_add_func ("/file/async-delete", test_async_delete);
|
2013-03-17 23:33:59 +01:00
|
|
|
|
g_test_add_func ("/file/copy-preserve-mode", test_copy_preserve_mode);
|
2013-11-29 06:11:47 +01:00
|
|
|
|
g_test_add_func ("/file/measure", test_measure);
|
2013-12-01 04:51:03 +01:00
|
|
|
|
g_test_add_func ("/file/measure-async", test_measure_async);
|
2017-11-14 10:40:44 +01:00
|
|
|
|
g_test_add_func ("/file/load-bytes", test_load_bytes);
|
|
|
|
|
g_test_add_func ("/file/load-bytes-async", test_load_bytes_async);
|
2018-09-14 11:54:00 +02:00
|
|
|
|
g_test_add_func ("/file/writev", test_writev);
|
|
|
|
|
g_test_add_func ("/file/writev/no-bytes-written", test_writev_no_bytes_written);
|
|
|
|
|
g_test_add_func ("/file/writev/no-vectors", test_writev_no_vectors);
|
|
|
|
|
g_test_add_func ("/file/writev/empty-vectors", test_writev_empty_vectors);
|
|
|
|
|
g_test_add_func ("/file/writev/too-big-vectors", test_writev_too_big_vectors);
|
|
|
|
|
g_test_add_func ("/file/writev/async", test_writev_async);
|
|
|
|
|
g_test_add_func ("/file/writev/async_all", test_writev_async_all);
|
|
|
|
|
g_test_add_func ("/file/writev/async_all-empty-vectors", test_writev_async_all_empty_vectors);
|
|
|
|
|
g_test_add_func ("/file/writev/async_all-no-vectors", test_writev_async_all_no_vectors);
|
|
|
|
|
g_test_add_func ("/file/writev/async_all-to-big-vectors", test_writev_async_all_too_big_vectors);
|
|
|
|
|
g_test_add_func ("/file/writev/async_all-cancellation", test_writev_async_all_cancellation);
|
2020-09-07 18:20:01 +02:00
|
|
|
|
g_test_add_func ("/file/build-attribute-list-for-copy", test_build_attribute_list_for_copy);
|
2010-07-05 09:09:36 +02:00
|
|
|
|
|
|
|
|
|
return g_test_run ();
|
|
|
|
|
}
|