Blame gio/tests/sleepy-stream.c

Packit ae235b
/*
Packit ae235b
 * Copyright © 2009 Codethink Limited
Packit ae235b
 *
Packit ae235b
 * This library is free software; you can redistribute it and/or
Packit ae235b
 * modify it under the terms of the GNU Lesser General Public
Packit ae235b
 * License as published by the Free Software Foundation; either
Packit ae235b
 * version 2.1 of the License, or (at your option) any later version.
Packit ae235b
 *
Packit ae235b
 * See the included COPYING file for more information.
Packit ae235b
 *
Packit ae235b
 * Author: Ryan Lortie <desrt@desrt.ca>
Packit ae235b
 */
Packit ae235b
Packit ae235b
#include <gio/gio.h>
Packit ae235b
#include <string.h>
Packit ae235b
Packit ae235b
#define MAX_PIECE_SIZE  100
Packit ae235b
#define MAX_PIECES       60
Packit ae235b
Packit ae235b
static gchar *
Packit ae235b
cook_piece (void)
Packit ae235b
{
Packit ae235b
  char buffer[MAX_PIECE_SIZE * 2];
Packit ae235b
  gint symbols, i = 0;
Packit ae235b
Packit ae235b
  symbols = g_test_rand_int_range (1, MAX_PIECE_SIZE + 1);
Packit ae235b
Packit ae235b
  while (symbols--)
Packit ae235b
    {
Packit ae235b
      gint c = g_test_rand_int_range (0, 30);
Packit ae235b
Packit ae235b
      switch (c)
Packit ae235b
        {
Packit ae235b
         case 26:
Packit ae235b
          buffer[i++] = '\n';
Packit ae235b
         case 27:
Packit ae235b
          buffer[i++] = '\r';
Packit ae235b
          break;
Packit ae235b
Packit ae235b
         case 28:
Packit ae235b
          buffer[i++] = '\r';
Packit ae235b
         case 29:
Packit ae235b
          buffer[i++] = '\n';
Packit ae235b
          break;
Packit ae235b
Packit ae235b
         default:
Packit ae235b
          buffer[i++] = c + 'a';
Packit ae235b
          break;
Packit ae235b
        }
Packit ae235b
Packit ae235b
      g_assert_cmpint (i, <=, sizeof buffer);
Packit ae235b
    }
Packit ae235b
Packit ae235b
  return g_strndup (buffer, i);
Packit ae235b
}
Packit ae235b
Packit ae235b
static gchar **
Packit ae235b
cook_pieces (void)
Packit ae235b
{
Packit ae235b
  gchar **array;
Packit ae235b
  gint pieces;
Packit ae235b
Packit ae235b
  pieces = g_test_rand_int_range (0, MAX_PIECES + 1);
Packit ae235b
  array = g_new (char *, pieces + 1);
Packit ae235b
  array[pieces] = NULL;
Packit ae235b
Packit ae235b
  while (pieces--)
Packit ae235b
    array[pieces] = cook_piece ();
Packit ae235b
Packit ae235b
  return array;
Packit ae235b
}
Packit ae235b
Packit ae235b
typedef struct
Packit ae235b
{
Packit ae235b
  GInputStream parent_instance;
Packit ae235b
Packit ae235b
  gboolean built_to_fail;
Packit ae235b
  gchar **pieces;
Packit ae235b
  gint index;
Packit ae235b
Packit ae235b
  const gchar *current;
Packit ae235b
} SleepyStream;
Packit ae235b
Packit ae235b
typedef GInputStreamClass SleepyStreamClass;
Packit ae235b
Packit ae235b
GType sleepy_stream_get_type (void);
Packit ae235b
Packit ae235b
G_DEFINE_TYPE (SleepyStream, sleepy_stream, G_TYPE_INPUT_STREAM)
Packit ae235b
Packit ae235b
static gssize
Packit ae235b
sleepy_stream_read (GInputStream  *stream,
Packit ae235b
                    void          *buffer,
Packit ae235b
                    gsize          length,
Packit ae235b
                    GCancellable  *cancellable,
Packit ae235b
                    GError       **error)
Packit ae235b
{
Packit ae235b
  SleepyStream *sleepy = (SleepyStream *) stream;
Packit ae235b
Packit ae235b
  if (sleepy->pieces[sleepy->index] == NULL)
Packit ae235b
    {
Packit ae235b
      if (sleepy->built_to_fail)
Packit ae235b
        {
Packit ae235b
          g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED, "fail");
Packit ae235b
          return -1;
Packit ae235b
        }
Packit ae235b
      else
Packit ae235b
        return 0;
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      if (!sleepy->current)
Packit ae235b
        sleepy->current = sleepy->pieces[sleepy->index++];
Packit ae235b
Packit ae235b
      length = MIN (strlen (sleepy->current), length);
Packit ae235b
      memcpy (buffer, sleepy->current, length);
Packit ae235b
Packit ae235b
      sleepy->current += length;
Packit ae235b
      if (*sleepy->current == '\0')
Packit ae235b
        sleepy->current = NULL;
Packit ae235b
Packit ae235b
      return length;
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
sleepy_stream_init (SleepyStream *sleepy)
Packit ae235b
{
Packit ae235b
  sleepy->pieces = cook_pieces ();
Packit ae235b
  sleepy->built_to_fail = FALSE;
Packit ae235b
  sleepy->index = 0;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
sleepy_stream_finalize (GObject *object)
Packit ae235b
{
Packit ae235b
  SleepyStream *sleepy = (SleepyStream *) object;
Packit ae235b
Packit ae235b
  g_strfreev (sleepy->pieces);
Packit ae235b
  G_OBJECT_CLASS (sleepy_stream_parent_class)
Packit ae235b
    ->finalize (object);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
sleepy_stream_class_init (SleepyStreamClass *class)
Packit ae235b
{
Packit ae235b
  G_OBJECT_CLASS (class)->finalize = sleepy_stream_finalize;
Packit ae235b
  class->read_fn = sleepy_stream_read;
Packit ae235b
Packit ae235b
  /* no read_async implementation.
Packit ae235b
   * main thread will sleep while read runs in a worker.
Packit ae235b
   */
Packit ae235b
}
Packit ae235b
Packit ae235b
static SleepyStream *
Packit ae235b
sleepy_stream_new (void)
Packit ae235b
{
Packit ae235b
  return g_object_new (sleepy_stream_get_type (), NULL);
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
read_line (GDataInputStream  *stream,
Packit ae235b
           GString           *string,
Packit ae235b
           const gchar       *eol,
Packit ae235b
           GError           **error)
Packit ae235b
{
Packit ae235b
  gsize length;
Packit ae235b
  char *str;
Packit ae235b
Packit ae235b
  str = g_data_input_stream_read_line (stream, &length, NULL, error);
Packit ae235b
Packit ae235b
  if (str == NULL)
Packit ae235b
    return FALSE;
Packit ae235b
Packit ae235b
  g_assert (strstr (str, eol) == NULL);
Packit ae235b
  g_assert (strlen (str) == length);
Packit ae235b
Packit ae235b
  g_string_append (string, str);
Packit ae235b
  g_string_append (string, eol);
Packit ae235b
  g_free (str);
Packit ae235b
Packit ae235b
  return TRUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
build_comparison (GString      *str,
Packit ae235b
                  SleepyStream *stream)
Packit ae235b
{
Packit ae235b
  /* build this for comparison */
Packit ae235b
  gint i;
Packit ae235b
Packit ae235b
  for (i = 0; stream->pieces[i]; i++)
Packit ae235b
    g_string_append (str, stream->pieces[i]);
Packit ae235b
Packit ae235b
  if (str->len && str->str[str->len - 1] != '\n')
Packit ae235b
    g_string_append_c (str, '\n');
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
static void
Packit ae235b
test (void)
Packit ae235b
{
Packit ae235b
  SleepyStream *stream = sleepy_stream_new ();
Packit ae235b
  GDataInputStream *data;
Packit ae235b
  GError *error = NULL;
Packit ae235b
  GString *one;
Packit ae235b
  GString *two;
Packit ae235b
Packit ae235b
  one = g_string_new (NULL);
Packit ae235b
  two = g_string_new (NULL);
Packit ae235b
Packit ae235b
  data = g_data_input_stream_new (G_INPUT_STREAM (stream));
Packit ae235b
  g_data_input_stream_set_newline_type (data, G_DATA_STREAM_NEWLINE_TYPE_LF);
Packit ae235b
  build_comparison (one, stream);
Packit ae235b
Packit ae235b
  while (read_line (data, two, "\n", &error));
Packit ae235b
Packit ae235b
  g_assert_cmpstr (one->str, ==, two->str);
Packit ae235b
  g_string_free (one, TRUE);
Packit ae235b
  g_string_free (two, TRUE);
Packit ae235b
  g_object_unref (stream);
Packit ae235b
  g_object_unref (data);
Packit ae235b
}
Packit ae235b
Packit ae235b
static GDataInputStream *data;
Packit ae235b
static GString *one, *two;
Packit ae235b
static GMainLoop *loop;
Packit ae235b
static const gchar *eol;
Packit ae235b
Packit ae235b
static void
Packit ae235b
asynch_ready (GObject      *object,
Packit ae235b
              GAsyncResult *result,
Packit ae235b
              gpointer      user_data)
Packit ae235b
{
Packit ae235b
  GError *error = NULL;
Packit ae235b
  gsize length;
Packit ae235b
  gchar *str;
Packit ae235b
Packit ae235b
  g_assert (data == G_DATA_INPUT_STREAM (object));
Packit ae235b
Packit ae235b
  str = g_data_input_stream_read_line_finish (data, result, &length, &error);
Packit ae235b
Packit ae235b
  if (str == NULL)
Packit ae235b
    {
Packit ae235b
      g_main_loop_quit (loop);
Packit ae235b
      if (error)
Packit ae235b
        g_error_free (error);
Packit ae235b
    }
Packit ae235b
  else
Packit ae235b
    {
Packit ae235b
      g_assert (length == strlen (str));
Packit ae235b
      g_string_append (two, str);
Packit ae235b
      g_string_append (two, eol);
Packit ae235b
      g_free (str);
Packit ae235b
Packit ae235b
      /* MOAR!! */
Packit ae235b
      g_data_input_stream_read_line_async (data, 0, NULL, asynch_ready, NULL);
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
static void
Packit ae235b
asynch (void)
Packit ae235b
{
Packit ae235b
  SleepyStream *sleepy = sleepy_stream_new ();
Packit ae235b
Packit ae235b
  data = g_data_input_stream_new (G_INPUT_STREAM (sleepy));
Packit ae235b
  one = g_string_new (NULL);
Packit ae235b
  two = g_string_new (NULL);
Packit ae235b
  eol = "\n";
Packit ae235b
Packit ae235b
  build_comparison (one, sleepy);
Packit ae235b
  g_data_input_stream_read_line_async (data, 0, NULL, asynch_ready, NULL);
Packit ae235b
  g_main_loop_run (loop = g_main_loop_new (NULL, FALSE));
Packit ae235b
Packit ae235b
  g_assert_cmpstr (one->str, ==, two->str);
Packit ae235b
  g_string_free (one, TRUE);
Packit ae235b
  g_string_free (two, TRUE);
Packit ae235b
  g_object_unref (sleepy);
Packit ae235b
  g_object_unref (data);
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
  g_test_bug_base ("http://bugzilla.gnome.org/");
Packit ae235b
Packit ae235b
  g_test_add_func ("/filter-stream/input", test);
Packit ae235b
  g_test_add_func ("/filter-stream/async", asynch);
Packit ae235b
Packit ae235b
  return g_test_run();
Packit ae235b
}