Blame gio/tests/testfilemonitor.c

Packit ae235b
#include <stdlib.h>
Packit ae235b
#include <gio/gio.h>
Packit ae235b
Packit ae235b
/* These tests were written for the inotify implementation.
Packit ae235b
 * Other implementations may require slight adjustments in
Packit ae235b
 * the tests, e.g. the length of timeouts
Packit ae235b
 */
Packit ae235b
Packit ae235b
typedef struct
Packit ae235b
{
Packit ae235b
  gint event_type;
Packit ae235b
  gchar *file;
Packit ae235b
  gchar *other_file;
Packit ae235b
  gint step;
Packit ae235b
} RecordedEvent;
Packit ae235b
Packit ae235b
static void
Packit ae235b
free_recorded_event (RecordedEvent *event)
Packit ae235b
{
Packit ae235b
  g_free (event->file);
Packit ae235b
  g_free (event->other_file);
Packit ae235b
  g_free (event);
Packit ae235b
}
Packit ae235b
Packit ae235b
typedef struct
Packit ae235b
{
Packit ae235b
  GFile *file;
Packit ae235b
  GFileMonitor *monitor;
Packit ae235b
  GMainLoop *loop;
Packit ae235b
  gint step;
Packit ae235b
  GList *events;
Packit ae235b
} TestData;
Packit ae235b
Packit ae235b
#if 0
Packit ae235b
static void
Packit ae235b
output_event (RecordedEvent *event)
Packit ae235b
{
Packit ae235b
  if (event->step >= 0)
Packit ae235b
    g_print (">>>> step %d\n", event->step);
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      GTypeClass *class;
Packit ae235b
Packit ae235b
      class = g_type_class_ref (g_type_from_name ("GFileMonitorEvent"));
Packit ae235b
      g_print ("%s file=%s other_file=%s\n",
Packit ae235b
               g_enum_get_value (G_ENUM_CLASS (class), event->event_type)->value_nick,
Packit ae235b
               event->file,
Packit ae235b
               event->other_file);
Packit ae235b
      g_type_class_unref (class);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
output_events (GList *list)
Packit ae235b
{
Packit ae235b
  GList *l;
Packit ae235b
Packit ae235b
  g_print (">>>output events\n");
Packit ae235b
  for (l = list; l; l = l->next)
Packit ae235b
    output_event ((RecordedEvent *)l->data);
Packit ae235b
}
Packit ae235b
#endif
Packit ae235b
Packit ae235b
/* a placeholder for temp file names we don't want to compare */
Packit ae235b
static const gchar DONT_CARE[] = "";
Packit ae235b
Packit ae235b
static void
Packit ae235b
check_expected_event (gint           i,
Packit ae235b
                      RecordedEvent *e1,
Packit ae235b
                      RecordedEvent *e2)
Packit ae235b
{
Packit ae235b
  g_assert_cmpint (e1->step, ==, e2->step);
Packit ae235b
  if (e1->step < 0)
Packit ae235b
    return;
Packit ae235b
Packit ae235b
  g_assert_cmpint (e1->event_type, ==, e2->event_type);
Packit ae235b
Packit ae235b
  if (e1->file != DONT_CARE)
Packit ae235b
    g_assert_cmpstr (e1->file, ==, e2->file);
Packit ae235b
Packit ae235b
  if (e1->other_file != DONT_CARE)
Packit ae235b
    g_assert_cmpstr (e1->other_file, ==, e2->other_file);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
check_expected_events (RecordedEvent *expected,
Packit ae235b
                       gsize          n_expected,
Packit ae235b
                       GList         *recorded)
Packit ae235b
{
Packit ae235b
  gint i;
Packit ae235b
  GList *l;
Packit ae235b
Packit ae235b
  g_assert_cmpint (n_expected, ==, g_list_length (recorded));
Packit ae235b
Packit ae235b
  for (i = 0, l = recorded; i < n_expected; i++, l = l->next)
Packit ae235b
    {
Packit ae235b
      RecordedEvent *e1 = &expected[i];
Packit ae235b
      RecordedEvent *e2 = (RecordedEvent *)l->data;
Packit ae235b
Packit ae235b
      check_expected_event (i, e1, e2);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
record_event (TestData    *data,
Packit ae235b
              gint         event_type,
Packit ae235b
              const gchar *file,
Packit ae235b
              const gchar *other_file,
Packit ae235b
              gint         step)
Packit ae235b
{
Packit ae235b
  RecordedEvent *event;
Packit ae235b
Packit ae235b
  event = g_new0 (RecordedEvent, 1);
Packit ae235b
  event->event_type = event_type;
Packit ae235b
  event->file = g_strdup (file);
Packit ae235b
  event->other_file = g_strdup (other_file);
Packit ae235b
  event->step = step;
Packit ae235b
Packit ae235b
  data->events = g_list_append (data->events, event);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
monitor_changed (GFileMonitor      *monitor,
Packit ae235b
                 GFile             *file,
Packit ae235b
                 GFile             *other_file,
Packit ae235b
                 GFileMonitorEvent  event_type,
Packit ae235b
                 gpointer           user_data)
Packit ae235b
{
Packit ae235b
  TestData *data = user_data;
Packit ae235b
  gchar *basename, *other_base;
Packit ae235b
Packit ae235b
  basename = g_file_get_basename (file);
Packit ae235b
  if (other_file)
Packit ae235b
    other_base = g_file_get_basename (other_file);
Packit ae235b
  else
Packit ae235b
    other_base = NULL;
Packit ae235b
Packit ae235b
  record_event (data, event_type, basename, other_base, -1);
Packit ae235b
Packit ae235b
  g_free (basename);
Packit ae235b
  g_free (other_base);
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
atomic_replace_step (gpointer user_data)
Packit ae235b
{
Packit ae235b
  TestData *data = user_data;
Packit ae235b
  GError *error = NULL;
Packit ae235b
Packit ae235b
  switch (data->step)
Packit ae235b
    {
Packit ae235b
    case 0:
Packit ae235b
      record_event (data, -1, NULL, NULL, 0);
Packit ae235b
      g_file_replace_contents (data->file, "step 0", 6, NULL, FALSE, G_FILE_CREATE_NONE, NULL, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      break;
Packit ae235b
    case 1:
Packit ae235b
      record_event (data, -1, NULL, NULL, 1);
Packit ae235b
      g_file_replace_contents (data->file, "step 1", 6, NULL, FALSE, G_FILE_CREATE_NONE, NULL, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      break;
Packit ae235b
    case 2:
Packit ae235b
      record_event (data, -1, NULL, NULL, 2);
Packit ae235b
      g_file_delete (data->file, NULL, NULL);
Packit ae235b
      break;
Packit ae235b
    case 3:
Packit ae235b
      record_event (data, -1, NULL, NULL, 3);
Packit ae235b
      g_main_loop_quit (data->loop);
Packit ae235b
      return G_SOURCE_REMOVE;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->step++;
Packit ae235b
Packit ae235b
  return G_SOURCE_CONTINUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
/* this is the output we expect from the above steps */
Packit ae235b
static RecordedEvent atomic_replace_output[] = {
Packit ae235b
  { -1, NULL, NULL, 0 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CREATED, "atomic_replace_file", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGED, "atomic_replace_file", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT, "atomic_replace_file", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_RENAMED, (gchar*)DONT_CARE, "atomic_replace_file", -1 },
Packit ae235b
  { -1, NULL, NULL, 2 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_DELETED, "atomic_replace_file", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 3 }
Packit ae235b
};
Packit ae235b
Packit ae235b
static void
Packit ae235b
test_atomic_replace (void)
Packit ae235b
{
Packit ae235b
  GError *error = NULL;
Packit ae235b
  TestData data;
Packit ae235b
Packit ae235b
  data.step = 0;
Packit ae235b
  data.events = NULL;
Packit ae235b
Packit ae235b
  data.file = g_file_new_for_path ("atomic_replace_file");
Packit ae235b
  g_file_delete (data.file, NULL, NULL);
Packit ae235b
Packit ae235b
  data.monitor = g_file_monitor_file (data.file, G_FILE_MONITOR_WATCH_MOVES, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
Packit ae235b
  g_file_monitor_set_rate_limit (data.monitor, 200);
Packit ae235b
  g_signal_connect (data.monitor, "changed", G_CALLBACK (monitor_changed), &data);
Packit ae235b
Packit ae235b
  data.loop = g_main_loop_new (NULL, TRUE);
Packit ae235b
Packit ae235b
  g_timeout_add (500, atomic_replace_step, &data);
Packit ae235b
Packit ae235b
  g_main_loop_run (data.loop);
Packit ae235b
Packit ae235b
  /*output_events (data.events);*/
Packit ae235b
  check_expected_events (atomic_replace_output, G_N_ELEMENTS (atomic_replace_output), data.events);
Packit ae235b
Packit ae235b
  g_list_free_full (data.events, (GDestroyNotify)free_recorded_event);
Packit ae235b
  g_main_loop_unref (data.loop);
Packit ae235b
  g_object_unref (data.monitor);
Packit ae235b
  g_object_unref (data.file);
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
change_step (gpointer user_data)
Packit ae235b
{
Packit ae235b
  TestData *data = user_data;
Packit ae235b
  GOutputStream *stream;
Packit ae235b
  GError *error = NULL;
Packit ae235b
  guint32 mode = 0660;
Packit ae235b
Packit ae235b
  switch (data->step)
Packit ae235b
    {
Packit ae235b
    case 0:
Packit ae235b
      record_event (data, -1, NULL, NULL, 0);
Packit ae235b
      g_file_replace_contents (data->file, "step 0", 6, NULL, FALSE, G_FILE_CREATE_NONE, NULL, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      break;
Packit ae235b
    case 1:
Packit ae235b
      record_event (data, -1, NULL, NULL, 1);
Packit ae235b
      stream = (GOutputStream *)g_file_append_to (data->file, G_FILE_CREATE_NONE, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      g_output_stream_write_all (stream, " step 1", 7, NULL, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      g_output_stream_close (stream, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      break;
Packit ae235b
    case 2:
Packit ae235b
      record_event (data, -1, NULL, NULL, 2);
Packit ae235b
      g_file_set_attribute (data->file,
Packit ae235b
                            G_FILE_ATTRIBUTE_UNIX_MODE,
Packit ae235b
                            G_FILE_ATTRIBUTE_TYPE_UINT32,
Packit ae235b
                            &mode,
Packit ae235b
                            G_FILE_QUERY_INFO_NONE,
Packit ae235b
                            NULL,
Packit ae235b
                            &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      break;
Packit ae235b
    case 3:
Packit ae235b
      record_event (data, -1, NULL, NULL, 3);
Packit ae235b
      g_file_delete (data->file, NULL, NULL);
Packit ae235b
      break;
Packit ae235b
    case 4:
Packit ae235b
      record_event (data, -1, NULL, NULL, 4);
Packit ae235b
      g_main_loop_quit (data->loop);
Packit ae235b
      return G_SOURCE_REMOVE;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->step++;
Packit ae235b
Packit ae235b
  return G_SOURCE_CONTINUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
/* this is the output we expect from the above steps */
Packit ae235b
static RecordedEvent change_output[] = {
Packit ae235b
  { -1, NULL, NULL, 0 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CREATED, "change_file", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGED, "change_file", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT, "change_file", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGED, "change_file", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT, "change_file", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 2 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_ATTRIBUTE_CHANGED, "change_file", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 3 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_DELETED, "change_file", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 4 }
Packit ae235b
};
Packit ae235b
Packit ae235b
static void
Packit ae235b
test_file_changes (void)
Packit ae235b
{
Packit ae235b
  GError *error = NULL;
Packit ae235b
  TestData data;
Packit ae235b
Packit ae235b
  data.step = 0;
Packit ae235b
  data.events = NULL;
Packit ae235b
Packit ae235b
  data.file = g_file_new_for_path ("change_file");
Packit ae235b
  g_file_delete (data.file, NULL, NULL);
Packit ae235b
Packit ae235b
  data.monitor = g_file_monitor_file (data.file, G_FILE_MONITOR_WATCH_MOVES, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
Packit ae235b
  g_file_monitor_set_rate_limit (data.monitor, 200);
Packit ae235b
  g_signal_connect (data.monitor, "changed", G_CALLBACK (monitor_changed), &data);
Packit ae235b
Packit ae235b
  data.loop = g_main_loop_new (NULL, TRUE);
Packit ae235b
Packit ae235b
  g_timeout_add (500, change_step, &data);
Packit ae235b
Packit ae235b
  g_main_loop_run (data.loop);
Packit ae235b
Packit ae235b
  /*output_events (data.events);*/
Packit ae235b
  check_expected_events (change_output, G_N_ELEMENTS (change_output), data.events);
Packit ae235b
Packit ae235b
  g_list_free_full (data.events, (GDestroyNotify)free_recorded_event);
Packit ae235b
  g_main_loop_unref (data.loop);
Packit ae235b
  g_object_unref (data.monitor);
Packit ae235b
  g_object_unref (data.file);
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
dir_step (gpointer user_data)
Packit ae235b
{
Packit ae235b
  TestData *data = user_data;
Packit ae235b
  GFile *parent, *file, *file2;
Packit ae235b
  GError *error = NULL;
Packit ae235b
Packit ae235b
  switch (data->step)
Packit ae235b
    {
Packit ae235b
    case 1:
Packit ae235b
      record_event (data, -1, NULL, NULL, 1);
Packit ae235b
      parent = g_file_get_parent (data->file);
Packit ae235b
      file = g_file_get_child (parent, "dir_test_file");
Packit ae235b
      g_file_replace_contents (file, "step 1", 6, NULL, FALSE, G_FILE_CREATE_NONE, NULL, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      g_object_unref (file);
Packit ae235b
      g_object_unref (parent);
Packit ae235b
      break;
Packit ae235b
    case 2:
Packit ae235b
      record_event (data, -1, NULL, NULL, 2);
Packit ae235b
      parent = g_file_get_parent (data->file);
Packit ae235b
      file = g_file_get_child (parent, "dir_test_file");
Packit ae235b
      file2 = g_file_get_child (data->file, "dir_test_file");
Packit ae235b
      g_file_move (file, file2, G_FILE_COPY_NONE, NULL, NULL, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      g_object_unref (file);
Packit ae235b
      g_object_unref (file2);
Packit ae235b
      g_object_unref (parent);
Packit ae235b
      break;
Packit ae235b
    case 3:
Packit ae235b
      record_event (data, -1, NULL, NULL, 3);
Packit ae235b
      file = g_file_get_child (data->file, "dir_test_file");
Packit ae235b
      file2 = g_file_get_child (data->file, "dir_test_file2");
Packit ae235b
      g_file_move (file, file2, G_FILE_COPY_NONE, NULL, NULL, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      g_object_unref (file);
Packit ae235b
      g_object_unref (file2);
Packit ae235b
      break;
Packit ae235b
    case 4:
Packit ae235b
      record_event (data, -1, NULL, NULL, 4);
Packit ae235b
      parent = g_file_get_parent (data->file);
Packit ae235b
      file = g_file_get_child (data->file, "dir_test_file2");
Packit ae235b
      file2 = g_file_get_child (parent, "dir_test_file2");
Packit ae235b
      g_file_move (file, file2, G_FILE_COPY_NONE, NULL, NULL, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      g_file_delete (file2, NULL, NULL);
Packit ae235b
      g_object_unref (file);
Packit ae235b
      g_object_unref (file2);
Packit ae235b
      g_object_unref (parent);
Packit ae235b
      break;
Packit ae235b
    case 5:
Packit ae235b
      record_event (data, -1, NULL, NULL, 5);
Packit ae235b
      g_file_delete (data->file, NULL, NULL);
Packit ae235b
      break;
Packit ae235b
    case 6:
Packit ae235b
      record_event (data, -1, NULL, NULL, 6);
Packit ae235b
      g_main_loop_quit (data->loop);
Packit ae235b
      return G_SOURCE_REMOVE;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->step++;
Packit ae235b
Packit ae235b
  return G_SOURCE_CONTINUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
/* this is the output we expect from the above steps */
Packit ae235b
static RecordedEvent dir_output[] = {
Packit ae235b
  { -1, NULL, NULL, 1 },
Packit ae235b
  { -1, NULL, NULL, 2 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_MOVED_IN, "dir_test_file", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 3 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_RENAMED, "dir_test_file", "dir_test_file2", -1 },
Packit ae235b
  { -1, NULL, NULL, 4 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_MOVED_OUT, "dir_test_file2", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 5 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_DELETED, "dir_monitor_test", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 6 }
Packit ae235b
};
Packit ae235b
Packit ae235b
static void
Packit ae235b
test_dir_monitor (void)
Packit ae235b
{
Packit ae235b
  GError *error = NULL;
Packit ae235b
  TestData data;
Packit ae235b
Packit ae235b
  data.step = 0;
Packit ae235b
  data.events = NULL;
Packit ae235b
Packit ae235b
  data.file = g_file_new_for_path ("dir_monitor_test");
Packit ae235b
  g_file_delete (data.file, NULL, NULL);
Packit ae235b
  g_file_make_directory (data.file, NULL, &error);
Packit ae235b
Packit ae235b
  data.monitor = g_file_monitor_directory (data.file, G_FILE_MONITOR_WATCH_MOVES, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
Packit ae235b
  g_file_monitor_set_rate_limit (data.monitor, 200);
Packit ae235b
  g_signal_connect (data.monitor, "changed", G_CALLBACK (monitor_changed), &data);
Packit ae235b
Packit ae235b
  data.loop = g_main_loop_new (NULL, TRUE);
Packit ae235b
Packit ae235b
  g_timeout_add (500, dir_step, &data);
Packit ae235b
Packit ae235b
  g_main_loop_run (data.loop);
Packit ae235b
Packit ae235b
  /*output_events (data.events);*/
Packit ae235b
  check_expected_events (dir_output, G_N_ELEMENTS (dir_output), data.events);
Packit ae235b
Packit ae235b
  g_list_free_full (data.events, (GDestroyNotify)free_recorded_event);
Packit ae235b
  g_main_loop_unref (data.loop);
Packit ae235b
  g_object_unref (data.monitor);
Packit ae235b
  g_object_unref (data.file);
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
nodir_step (gpointer user_data)
Packit ae235b
{
Packit ae235b
  TestData *data = user_data;
Packit ae235b
  GFile *parent;
Packit ae235b
  GError *error = NULL;
Packit ae235b
Packit ae235b
  switch (data->step)
Packit ae235b
    {
Packit ae235b
    case 0:
Packit ae235b
      record_event (data, -1, NULL, NULL, 0);
Packit ae235b
      parent = g_file_get_parent (data->file);
Packit ae235b
      g_file_make_directory (parent, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      g_object_unref (parent);
Packit ae235b
      break;
Packit ae235b
    case 1:
Packit ae235b
      record_event (data, -1, NULL, NULL, 1);
Packit ae235b
      g_file_replace_contents (data->file, "step 1", 6, NULL, FALSE, G_FILE_CREATE_NONE, NULL, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      break;
Packit ae235b
    case 2:
Packit ae235b
      record_event (data, -1, NULL, NULL, 2);
Packit ae235b
      g_file_delete (data->file, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      break;
Packit ae235b
    case 3:
Packit ae235b
      record_event (data, -1, NULL, NULL, 3);
Packit ae235b
      parent = g_file_get_parent (data->file);
Packit ae235b
      g_file_delete (parent, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      g_object_unref (parent);
Packit ae235b
      break;
Packit ae235b
    case 4:
Packit ae235b
      record_event (data, -1, NULL, NULL, 4);
Packit ae235b
      g_main_loop_quit (data->loop);
Packit ae235b
      return G_SOURCE_REMOVE;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->step++;
Packit ae235b
Packit ae235b
  return G_SOURCE_CONTINUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static RecordedEvent nodir_output[] = {
Packit ae235b
  { -1, NULL, NULL, 0 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CREATED, "nosuchfile", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT, "nosuchfile", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CREATED, "nosuchfile", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGED, "nosuchfile", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT, "nosuchfile", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 2 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_DELETED, "nosuchfile", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 3 },
Packit ae235b
  { -1, NULL, NULL, 4 }
Packit ae235b
};
Packit ae235b
Packit ae235b
static void
Packit ae235b
test_dir_non_existent (void)
Packit ae235b
{
Packit ae235b
  TestData data;
Packit ae235b
  GError *error = NULL;
Packit ae235b
Packit ae235b
  data.step = 0;
Packit ae235b
  data.events = NULL;
Packit ae235b
Packit ae235b
  data.file = g_file_new_for_path ("nosuchdir/nosuchfile");
Packit ae235b
  data.monitor = g_file_monitor_file (data.file, G_FILE_MONITOR_WATCH_MOVES, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
Packit ae235b
  g_file_monitor_set_rate_limit (data.monitor, 200);
Packit ae235b
  g_signal_connect (data.monitor, "changed", G_CALLBACK (monitor_changed), &data);
Packit ae235b
Packit ae235b
  data.loop = g_main_loop_new (NULL, TRUE);
Packit ae235b
Packit ae235b
  /* we need a long timeout here, since the inotify implementation only scans
Packit ae235b
   * for missing files every 4 seconds.
Packit ae235b
   */
Packit ae235b
  g_timeout_add (5000, nodir_step, &data);
Packit ae235b
Packit ae235b
  g_main_loop_run (data.loop);
Packit ae235b
Packit ae235b
  /*output_events (data.events);*/
Packit ae235b
  check_expected_events (nodir_output, G_N_ELEMENTS (nodir_output), data.events);
Packit ae235b
Packit ae235b
  g_list_free_full (data.events, (GDestroyNotify)free_recorded_event);
Packit ae235b
  g_main_loop_unref (data.loop);
Packit ae235b
  g_object_unref (data.monitor);
Packit ae235b
  g_object_unref (data.file);
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
cross_dir_step (gpointer user_data)
Packit ae235b
{
Packit ae235b
  TestData *data = user_data;
Packit ae235b
  GFile *file, *file2;
Packit ae235b
  GError *error = NULL;
Packit ae235b
Packit ae235b
  switch (data[0].step)
Packit ae235b
    {
Packit ae235b
    case 0:
Packit ae235b
      record_event (&data[0], -1, NULL, NULL, 0);
Packit ae235b
      record_event (&data[1], -1, NULL, NULL, 0);
Packit ae235b
      file = g_file_get_child (data[1].file, "a");
Packit ae235b
      g_file_replace_contents (file, "step 0", 6, NULL, FALSE, G_FILE_CREATE_NONE, NULL, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      g_object_unref (file);
Packit ae235b
      break;
Packit ae235b
    case 1:
Packit ae235b
      record_event (&data[0], -1, NULL, NULL, 1);
Packit ae235b
      record_event (&data[1], -1, NULL, NULL, 1);
Packit ae235b
      file = g_file_get_child (data[1].file, "a");
Packit ae235b
      file2 = g_file_get_child (data[0].file, "a");
Packit ae235b
      g_file_move (file, file2, 0, NULL, NULL, NULL, &error);
Packit ae235b
      g_assert_no_error (error);
Packit ae235b
      g_object_unref (file);
Packit ae235b
      g_object_unref (file2);
Packit ae235b
      break;
Packit ae235b
    case 2:
Packit ae235b
      record_event (&data[0], -1, NULL, NULL, 2);
Packit ae235b
      record_event (&data[1], -1, NULL, NULL, 2);
Packit ae235b
      file2 = g_file_get_child (data[0].file, "a");
Packit ae235b
      g_file_delete (file2, NULL, NULL);
Packit ae235b
      g_file_delete (data[0].file, NULL, NULL);
Packit ae235b
      g_file_delete (data[1].file, NULL, NULL);
Packit ae235b
      g_object_unref (file2);
Packit ae235b
      break;
Packit ae235b
    case 3:
Packit ae235b
      record_event (&data[0], -1, NULL, NULL, 3);
Packit ae235b
      record_event (&data[1], -1, NULL, NULL, 3);
Packit ae235b
      g_main_loop_quit (data->loop);
Packit ae235b
      return G_SOURCE_REMOVE;
Packit ae235b
    }
Packit ae235b
Packit ae235b
  data->step++;
Packit ae235b
Packit ae235b
  return G_SOURCE_CONTINUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static RecordedEvent cross_dir_a_output[] = {
Packit ae235b
  { -1, NULL, NULL, 0 },
Packit ae235b
  { -1, NULL, NULL, 1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CREATED, "a", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT, "a", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 2 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_DELETED, "a", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_DELETED, "cross_dir_a", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 3 },
Packit ae235b
};
Packit ae235b
Packit ae235b
static RecordedEvent cross_dir_b_output[] = {
Packit ae235b
  { -1, NULL, NULL, 0 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CREATED, "a", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGED, "a", NULL, -1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT, "a", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 1 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_MOVED_OUT, "a", "a", -1 },
Packit ae235b
  { -1, NULL, NULL, 2 },
Packit ae235b
  { G_FILE_MONITOR_EVENT_DELETED, "cross_dir_b", NULL, -1 },
Packit ae235b
  { -1, NULL, NULL, 3 },
Packit ae235b
};
Packit ae235b
static void
Packit ae235b
test_cross_dir_moves (void)
Packit ae235b
{
Packit ae235b
  GError *error = NULL;
Packit ae235b
  TestData data[2];
Packit ae235b
Packit ae235b
  data[0].step = 0;
Packit ae235b
  data[0].events = NULL;
Packit ae235b
Packit ae235b
  data[0].file = g_file_new_for_path ("cross_dir_a");
Packit ae235b
  g_file_delete (data[0].file, NULL, NULL);
Packit ae235b
  g_file_make_directory (data[0].file, NULL, &error);
Packit ae235b
Packit ae235b
  data[0].monitor = g_file_monitor_directory (data[0].file, 0, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
Packit ae235b
  g_file_monitor_set_rate_limit (data[0].monitor, 200);
Packit ae235b
  g_signal_connect (data[0].monitor, "changed", G_CALLBACK (monitor_changed), &data[0]);
Packit ae235b
Packit ae235b
  data[1].step = 0;
Packit ae235b
  data[1].events = NULL;
Packit ae235b
Packit ae235b
  data[1].file = g_file_new_for_path ("cross_dir_b");
Packit ae235b
  g_file_delete (data[1].file, NULL, NULL);
Packit ae235b
  g_file_make_directory (data[1].file, NULL, &error);
Packit ae235b
Packit ae235b
  data[1].monitor = g_file_monitor_directory (data[1].file, G_FILE_MONITOR_WATCH_MOVES, NULL, &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
Packit ae235b
  g_file_monitor_set_rate_limit (data[1].monitor, 200);
Packit ae235b
  g_signal_connect (data[1].monitor, "changed", G_CALLBACK (monitor_changed), &data[1]);
Packit ae235b
Packit ae235b
  data[0].loop = g_main_loop_new (NULL, TRUE);
Packit ae235b
Packit ae235b
  g_timeout_add (500, cross_dir_step, data);
Packit ae235b
Packit ae235b
  g_main_loop_run (data[0].loop);
Packit ae235b
Packit ae235b
#if 0
Packit ae235b
  g_print ("monitor a:\n");
Packit ae235b
  output_events (data[0].events);
Packit ae235b
  g_print ("monitor b:\n");
Packit ae235b
  output_events (data[1].events);
Packit ae235b
#endif
Packit ae235b
Packit ae235b
  check_expected_events (cross_dir_a_output, G_N_ELEMENTS (cross_dir_a_output), data[0].events);
Packit ae235b
  check_expected_events (cross_dir_b_output, G_N_ELEMENTS (cross_dir_b_output), data[1].events);
Packit ae235b
Packit ae235b
  g_list_free_full (data[0].events, (GDestroyNotify)free_recorded_event);
Packit ae235b
  g_main_loop_unref (data[0].loop);
Packit ae235b
  g_object_unref (data[0].monitor);
Packit ae235b
  g_object_unref (data[0].file);
Packit ae235b
Packit ae235b
  g_list_free_full (data[1].events, (GDestroyNotify)free_recorded_event);
Packit ae235b
  g_object_unref (data[1].monitor);
Packit ae235b
  g_object_unref (data[1].file);
Packit ae235b
}
Packit ae235b
Packit ae235b
int
Packit ae235b
main (int argc, char *argv[])
Packit ae235b
{
Packit ae235b
  g_test_init (&argc, &argv, NULL);
Packit ae235b
Packit ae235b
  g_test_add_func ("/monitor/atomic-replace", test_atomic_replace);
Packit ae235b
  g_test_add_func ("/monitor/file-changes", test_file_changes);
Packit ae235b
  g_test_add_func ("/monitor/dir-monitor", test_dir_monitor);
Packit ae235b
  g_test_add_func ("/monitor/dir-not-existent", test_dir_non_existent);
Packit ae235b
  g_test_add_func ("/monitor/cross-dir-moves", test_cross_dir_moves);
Packit ae235b
Packit ae235b
  return g_test_run ();
Packit ae235b
}