(root)/
glib-2.79.0/
gio/
tests/
filter-streams.c
       1  /*
       2   * Copyright © 2009 Codethink Limited
       3   *
       4   * SPDX-License-Identifier: LGPL-2.1-or-later
       5   *
       6   * This library is free software; you can redistribute it and/or
       7   * modify it under the terms of the GNU Lesser General Public
       8   * License as published by the Free Software Foundation; either
       9   * version 2.1 of the License, or (at your option) any later version.
      10   *
      11   * See the included COPYING file for more information.
      12   *
      13   * Author: Ryan Lortie <desrt@desrt.ca>
      14   */
      15  
      16  #include <string.h>
      17  #include <glib/glib.h>
      18  #include <gio/gio.h>
      19  
      20  /* GFilterInputStream and GFilterOutputStream are abstract, so define
      21   * minimal subclasses for testing. (This used to use
      22   * GBufferedInputStream and GBufferedOutputStream, but those have
      23   * their own test program, and they override some methods, meaning the
      24   * core filter stream functionality wasn't getting fully tested.)
      25   */
      26  
      27  GType test_filter_input_stream_get_type (void);
      28  GType test_filter_output_stream_get_type (void);
      29  
      30  #define TEST_TYPE_FILTER_INPUT_STREAM  (test_filter_input_stream_get_type ())
      31  #define TEST_FILTER_INPUT_STREAM(o)    (G_TYPE_CHECK_INSTANCE_CAST ((o), TEST_TYPE_FILTER_INPUT_STREAM, TestFilterInputStream))
      32  #define TEST_TYPE_FILTER_OUTPUT_STREAM (test_filter_output_stream_get_type ())
      33  #define TEST_FILTER_OUTPUT_STREAM(o)   (G_TYPE_CHECK_INSTANCE_CAST ((o), TEST_TYPE_FILTER_OUTPUT_STREAM, TestFilterOutputStream))
      34  
      35  typedef GFilterInputStream       TestFilterInputStream;
      36  typedef GFilterInputStreamClass  TestFilterInputStreamClass;
      37  typedef GFilterOutputStream      TestFilterOutputStream;
      38  typedef GFilterOutputStreamClass TestFilterOutputStreamClass;
      39  
      40  G_DEFINE_TYPE (TestFilterInputStream, test_filter_input_stream, G_TYPE_FILTER_INPUT_STREAM)
      41  G_DEFINE_TYPE (TestFilterOutputStream, test_filter_output_stream, G_TYPE_FILTER_OUTPUT_STREAM)
      42  
      43  static void
      44  test_filter_input_stream_init (TestFilterInputStream *stream)
      45  {
      46  }
      47  
      48  static void
      49  test_filter_input_stream_class_init (TestFilterInputStreamClass *klass)
      50  {
      51  }
      52  
      53  static void
      54  test_filter_output_stream_init (TestFilterOutputStream *stream)
      55  {
      56  }
      57  
      58  static void
      59  test_filter_output_stream_class_init (TestFilterOutputStreamClass *klass)
      60  {
      61  }
      62  
      63  /* Now the tests */
      64  
      65  static void
      66  test_input_filter (void)
      67  {
      68    GInputStream *base, *f1, *f2, *s;
      69    gboolean close_base;
      70    gchar buf[1024];
      71    GError *error = NULL;
      72  
      73    g_test_bug ("https://bugzilla.gnome.org/show_bug.cgi?id=568394");
      74    base = g_memory_input_stream_new_from_data ("abcdefghijk", -1, NULL);
      75    f1 = g_object_new (TEST_TYPE_FILTER_INPUT_STREAM,
      76                       "base-stream", base,
      77                       "close-base-stream", FALSE,
      78                       NULL);
      79    f2 = g_object_new (TEST_TYPE_FILTER_INPUT_STREAM,
      80                       "base-stream", base,
      81                       NULL);
      82  
      83    g_assert (g_filter_input_stream_get_base_stream (G_FILTER_INPUT_STREAM (f1)) == base);
      84    g_assert (g_filter_input_stream_get_base_stream (G_FILTER_INPUT_STREAM (f2)) == base);
      85  
      86    g_assert (!g_input_stream_is_closed (base));
      87    g_assert (!g_input_stream_is_closed (f1));
      88    g_assert (!g_input_stream_is_closed (f2));
      89  
      90    g_object_get (f1,
      91                  "close-base-stream", &close_base,
      92                  "base-stream", &s,
      93                  NULL);
      94    g_assert (!close_base);
      95    g_assert (s == base);
      96    g_object_unref (s);
      97  
      98    g_object_unref (f1);
      99  
     100    g_assert (!g_input_stream_is_closed (base));
     101    g_assert (!g_input_stream_is_closed (f2));
     102  
     103    g_input_stream_skip (f2, 3, NULL, &error);
     104    g_assert_no_error (error);
     105  
     106    memset (buf, 0, 1024);
     107    g_input_stream_read_all (f2, buf, 1024, NULL, NULL, &error);
     108    g_assert_no_error (error);
     109    g_assert_cmpstr (buf, ==, "defghijk");
     110  
     111    g_object_unref (f2);
     112  
     113    g_assert (g_input_stream_is_closed (base));
     114  
     115    g_object_unref (base);
     116  }
     117  
     118  static void
     119  test_output_filter (void)
     120  {
     121    GOutputStream *base, *f1, *f2;
     122  
     123    base = g_memory_output_stream_new (NULL, 0, g_realloc, g_free);
     124    f1 = g_object_new (TEST_TYPE_FILTER_OUTPUT_STREAM,
     125                       "base-stream", base,
     126                       "close-base-stream", FALSE,
     127                       NULL);
     128    f2 = g_object_new (TEST_TYPE_FILTER_OUTPUT_STREAM,
     129                       "base-stream", base,
     130                       NULL);
     131  
     132    g_assert (g_filter_output_stream_get_base_stream (G_FILTER_OUTPUT_STREAM (f1)) == base);
     133    g_assert (g_filter_output_stream_get_base_stream (G_FILTER_OUTPUT_STREAM (f2)) == base);
     134  
     135    g_assert (!g_output_stream_is_closed (base));
     136    g_assert (!g_output_stream_is_closed (f1));
     137    g_assert (!g_output_stream_is_closed (f2));
     138  
     139    g_object_unref (f1);
     140  
     141    g_assert (!g_output_stream_is_closed (base));
     142    g_assert (!g_output_stream_is_closed (f2));
     143  
     144    g_object_unref (f2);
     145  
     146    g_assert (g_output_stream_is_closed (base));
     147  
     148    g_object_unref (base);
     149  }
     150  
     151  gpointer expected_obj;
     152  gpointer expected_data;
     153  gboolean callback_happened;
     154  GMainLoop *loop;
     155  
     156  static void
     157  return_result_cb (GObject      *object,
     158                    GAsyncResult *result,
     159                    gpointer      user_data)
     160  {
     161    GAsyncResult **ret = user_data;
     162  
     163    *ret = g_object_ref (result);
     164    g_main_loop_quit (loop);
     165  }
     166  
     167  static void
     168  in_cb (GObject      *object,
     169         GAsyncResult *result,
     170         gpointer      user_data)
     171  {
     172    GError *error = NULL;
     173  
     174    g_assert (object == expected_obj);
     175    g_assert (user_data == expected_data);
     176    g_assert (callback_happened == FALSE);
     177  
     178    g_input_stream_close_finish (expected_obj, result, &error);
     179    g_assert (error == NULL);
     180  
     181    callback_happened = TRUE;
     182    g_main_loop_quit (loop);
     183  }
     184  
     185  static void
     186  test_input_async (void)
     187  {
     188    GInputStream *base, *f1, *f2;
     189    char buf[20];
     190    GAsyncResult *result = NULL;
     191    GError *error = NULL;
     192  
     193    loop = g_main_loop_new (NULL, FALSE);
     194  
     195    base = g_memory_input_stream_new_from_data ("abcdefghijklmnopqrstuvwxyz", -1, NULL);
     196    f1 = g_object_new (TEST_TYPE_FILTER_INPUT_STREAM,
     197                       "base-stream", base,
     198                       "close-base-stream", FALSE,
     199                       NULL);
     200    f2 = g_object_new (TEST_TYPE_FILTER_INPUT_STREAM,
     201                       "base-stream", base,
     202                       NULL);
     203  
     204    g_assert (g_filter_input_stream_get_base_stream (G_FILTER_INPUT_STREAM (f1)) == base);
     205    g_assert (g_filter_input_stream_get_base_stream (G_FILTER_INPUT_STREAM (f2)) == base);
     206  
     207  
     208    memset (buf, 0, sizeof (buf));
     209    g_input_stream_read_async (f1, buf, 10, G_PRIORITY_DEFAULT,
     210                               NULL, return_result_cb, &result);
     211    g_main_loop_run (loop);
     212    g_assert_cmpint (g_input_stream_read_finish (f1, result, &error), ==, 10);
     213    g_assert_cmpstr (buf, ==, "abcdefghij");
     214    g_assert_no_error (error);
     215    g_clear_object (&result);
     216  
     217    g_assert_cmpint (g_seekable_tell (G_SEEKABLE (base)), ==, 10);
     218  
     219    g_input_stream_skip_async (f2, 10, G_PRIORITY_DEFAULT,
     220                               NULL, return_result_cb, &result);
     221    g_main_loop_run (loop);
     222    g_assert_cmpint (g_input_stream_skip_finish (f2, result, &error), ==, 10);
     223    g_assert_no_error (error);
     224    g_clear_object (&result);
     225  
     226    g_assert_cmpint (g_seekable_tell (G_SEEKABLE (base)), ==, 20);
     227  
     228    memset (buf, 0, sizeof (buf));
     229    g_input_stream_read_async (f1, buf, 10, G_PRIORITY_DEFAULT,
     230                               NULL, return_result_cb, &result);
     231    g_main_loop_run (loop);
     232    g_assert_cmpint (g_input_stream_read_finish (f1, result, &error), ==, 6);
     233    g_assert_cmpstr (buf, ==, "uvwxyz");
     234    g_assert_no_error (error);
     235    g_clear_object (&result);
     236  
     237    g_assert_cmpint (g_seekable_tell (G_SEEKABLE (base)), ==, 26);
     238  
     239  
     240    g_assert (!g_input_stream_is_closed (base));
     241    g_assert (!g_input_stream_is_closed (f1));
     242    g_assert (!g_input_stream_is_closed (f2));
     243  
     244    expected_obj = f1;
     245    expected_data = g_malloc (20);
     246    callback_happened = FALSE;
     247    g_input_stream_close_async (f1, 0, NULL, in_cb, expected_data);
     248  
     249    g_assert (callback_happened == FALSE);
     250    g_main_loop_run (loop);
     251    g_assert (callback_happened == TRUE);
     252  
     253    g_assert (!g_input_stream_is_closed (base));
     254    g_assert (!g_input_stream_is_closed (f2));
     255    g_free (expected_data);
     256    g_object_unref (f1);
     257    g_assert (!g_input_stream_is_closed (base));
     258    g_assert (!g_input_stream_is_closed (f2));
     259  
     260    expected_obj = f2;
     261    expected_data = g_malloc (20);
     262    callback_happened = FALSE;
     263    g_input_stream_close_async (f2, 0, NULL, in_cb, expected_data);
     264  
     265    g_assert (callback_happened == FALSE);
     266    g_main_loop_run (loop);
     267    g_assert (callback_happened == TRUE);
     268  
     269    g_assert (g_input_stream_is_closed (base));
     270    g_assert (g_input_stream_is_closed (f2));
     271    g_free (expected_data);
     272    g_object_unref (f2);
     273  
     274    g_assert (g_input_stream_is_closed (base));
     275    g_object_unref (base);
     276    g_main_loop_unref (loop);
     277  }
     278  
     279  static void
     280  out_cb (GObject      *object,
     281          GAsyncResult *result,
     282          gpointer      user_data)
     283  {
     284    GError *error = NULL;
     285  
     286    g_assert (object == expected_obj);
     287    g_assert (user_data == expected_data);
     288    g_assert (callback_happened == FALSE);
     289  
     290    g_output_stream_close_finish (expected_obj, result, &error);
     291    g_assert (error == NULL);
     292  
     293    callback_happened = TRUE;
     294    g_main_loop_quit (loop);
     295  }
     296  
     297  static void
     298  test_output_async (void)
     299  {
     300    GOutputStream *base, *f1, *f2;
     301    GAsyncResult *result = NULL;
     302    GError *error = NULL;
     303  
     304    loop = g_main_loop_new (NULL, FALSE);
     305  
     306    base = g_memory_output_stream_new (NULL, 0, g_realloc, g_free);
     307    f1 = g_object_new (TEST_TYPE_FILTER_OUTPUT_STREAM,
     308                       "base-stream", base,
     309                       "close-base-stream", FALSE,
     310                       NULL);
     311    f2 = g_object_new (TEST_TYPE_FILTER_OUTPUT_STREAM,
     312                       "base-stream", base,
     313                       NULL);
     314  
     315    g_assert (g_filter_output_stream_get_base_stream (G_FILTER_OUTPUT_STREAM (f1)) == base);
     316    g_assert (g_filter_output_stream_get_base_stream (G_FILTER_OUTPUT_STREAM (f2)) == base);
     317  
     318  
     319    g_output_stream_write_async (f1, "abcdefghijklm", 13, G_PRIORITY_DEFAULT,
     320                                 NULL, return_result_cb, &result);
     321    g_main_loop_run (loop);
     322    g_assert_cmpint (g_output_stream_write_finish (f1, result, &error), ==, 13);
     323    g_assert_no_error (error);
     324    g_clear_object (&result);
     325  
     326    g_assert_cmpint (g_seekable_tell (G_SEEKABLE (base)), ==, 13);
     327  
     328    g_output_stream_write_async (f2, "nopqrstuvwxyz", 13, G_PRIORITY_DEFAULT,
     329                                 NULL, return_result_cb, &result);
     330    g_main_loop_run (loop);
     331    g_assert_cmpint (g_output_stream_write_finish (f2, result, &error), ==, 13);
     332    g_assert_no_error (error);
     333    g_clear_object (&result);
     334  
     335    g_assert_cmpint (g_seekable_tell (G_SEEKABLE (base)), ==, 26);
     336  
     337    g_assert_cmpint (g_memory_output_stream_get_data_size (G_MEMORY_OUTPUT_STREAM (base)), ==, 26);
     338    g_output_stream_write (base, "\0", 1, NULL, &error);
     339    g_assert_no_error (error);
     340    g_assert_cmpstr (g_memory_output_stream_get_data (G_MEMORY_OUTPUT_STREAM (base)), ==, "abcdefghijklmnopqrstuvwxyz");
     341  
     342  
     343    g_assert (!g_output_stream_is_closed (base));
     344    g_assert (!g_output_stream_is_closed (f1));
     345    g_assert (!g_output_stream_is_closed (f2));
     346  
     347    expected_obj = f1;
     348    expected_data = g_malloc (20);
     349    callback_happened = FALSE;
     350    g_output_stream_close_async (f1, 0, NULL, out_cb, expected_data);
     351  
     352    g_assert (callback_happened == FALSE);
     353    g_main_loop_run (loop);
     354    g_assert (callback_happened == TRUE);
     355  
     356    g_assert (!g_output_stream_is_closed (base));
     357    g_assert (!g_output_stream_is_closed (f2));
     358    g_free (expected_data);
     359    g_object_unref (f1);
     360    g_assert (!g_output_stream_is_closed (base));
     361    g_assert (!g_output_stream_is_closed (f2));
     362  
     363    expected_obj = f2;
     364    expected_data = g_malloc (20);
     365    callback_happened = FALSE;
     366    g_output_stream_close_async (f2, 0, NULL, out_cb, expected_data);
     367  
     368    g_assert (callback_happened == FALSE);
     369    g_main_loop_run (loop);
     370    g_assert (callback_happened == TRUE);
     371  
     372    g_assert (g_output_stream_is_closed (base));
     373    g_assert (g_output_stream_is_closed (f2));
     374    g_free (expected_data);
     375    g_object_unref (f2);
     376  
     377    g_assert (g_output_stream_is_closed (base));
     378    g_object_unref (base);
     379    g_main_loop_unref (loop);
     380  }
     381  
     382  int
     383  main (int argc, char **argv)
     384  {
     385    g_test_init (&argc, &argv, NULL);
     386  
     387    g_test_add_func ("/filter-stream/input", test_input_filter);
     388    g_test_add_func ("/filter-stream/output", test_output_filter);
     389    g_test_add_func ("/filter-stream/async-input", test_input_async);
     390    g_test_add_func ("/filter-stream/async-output", test_output_async);
     391  
     392    return g_test_run();
     393  }