/* GLib testing framework examples and tests * Copyright (C) 2008 Red Hat, Inc. * Authors: Tomas Bzatek * * SPDX-License-Identifier: LicenseRef-old-glib-tests * * This work is provided "as is"; redistribution and modification * in whole or in part, in any medium, physical or electronic is * permitted without restriction. * * This work is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. * * In no event shall the authors or contributors be liable for any * direct, indirect, incidental, special, exemplary, or consequential * damages (including, but not limited to, procurement of substitute * goods or services; loss of use, data, or profits; or business * interruption) however caused and on any theory of liability, whether * in contract, strict liability, or tort (including negligence or * otherwise) arising in any way out of the use of this software, even * if advised of the possibility of such damage. */ #include #include #include #include #define MAX_LINES 0xFFF #define MAX_BYTES 0x10000 static void test_basic (void) { GInputStream *stream; GInputStream *base_stream; gint val; base_stream = g_memory_input_stream_new (); stream = G_INPUT_STREAM (g_data_input_stream_new (base_stream)); g_object_get (stream, "byte-order", &val, NULL); g_assert_cmpint (val, ==, G_DATA_STREAM_BYTE_ORDER_BIG_ENDIAN); g_object_set (stream, "byte-order", G_DATA_STREAM_BYTE_ORDER_LITTLE_ENDIAN, NULL); g_assert_cmpint (g_data_input_stream_get_byte_order (G_DATA_INPUT_STREAM (stream)), ==, G_DATA_STREAM_BYTE_ORDER_LITTLE_ENDIAN); g_object_get (stream, "newline-type", &val, NULL); g_assert_cmpint (val, ==, G_DATA_STREAM_NEWLINE_TYPE_LF); g_object_set (stream, "newline-type", G_DATA_STREAM_NEWLINE_TYPE_CR_LF, NULL); g_assert_cmpint (g_data_input_stream_get_newline_type (G_DATA_INPUT_STREAM (stream)), ==, G_DATA_STREAM_NEWLINE_TYPE_CR_LF); g_object_unref (stream); g_object_unref (base_stream); } static void test_seek_to_start (GInputStream *stream) { GError *error = NULL; gboolean res = g_seekable_seek (G_SEEKABLE (stream), 0, G_SEEK_SET, NULL, &error); g_assert_cmpint (res, ==, TRUE); g_assert_no_error (error); } static void test_read_lines (GDataStreamNewlineType newline_type) { GInputStream *stream; GInputStream *base_stream; GError *error = NULL; char *data; int line; const char* lines[MAX_LINES]; const char* endl[4] = {"\n", "\r", "\r\n", "\n"}; /* prepare data */ int i; for (i = 0; i < MAX_LINES; i++) lines[i] = "some_text"; base_stream = g_memory_input_stream_new (); g_assert_nonnull (base_stream); stream = G_INPUT_STREAM (g_data_input_stream_new (base_stream)); g_assert_nonnull (stream); /* Byte order testing */ g_data_input_stream_set_byte_order (G_DATA_INPUT_STREAM (stream), G_DATA_STREAM_BYTE_ORDER_BIG_ENDIAN); g_assert_cmpint (g_data_input_stream_get_byte_order (G_DATA_INPUT_STREAM (stream)), ==, G_DATA_STREAM_BYTE_ORDER_BIG_ENDIAN); g_data_input_stream_set_byte_order (G_DATA_INPUT_STREAM (stream), G_DATA_STREAM_BYTE_ORDER_LITTLE_ENDIAN); g_assert_cmpint (g_data_input_stream_get_byte_order (G_DATA_INPUT_STREAM (stream)), ==, G_DATA_STREAM_BYTE_ORDER_LITTLE_ENDIAN); /* Line ends testing */ g_data_input_stream_set_newline_type (G_DATA_INPUT_STREAM (stream), newline_type); g_assert_cmpint (g_data_input_stream_get_newline_type (G_DATA_INPUT_STREAM (stream)), ==, newline_type); /* Add sample data */ for (i = 0; i < MAX_LINES; i++) g_memory_input_stream_add_data (G_MEMORY_INPUT_STREAM (base_stream), g_strconcat (lines[i], endl[newline_type], NULL), -1, g_free); /* Seek to the start */ test_seek_to_start (base_stream); /* Test read line */ error = NULL; data = (char*)1; line = 0; while (data) { gsize length = -1; data = g_data_input_stream_read_line (G_DATA_INPUT_STREAM (stream), &length, NULL, &error); if (data) { g_assert_cmpstr (data, ==, lines[line]); g_free (data); g_assert_no_error (error); line++; } if (error) g_error_free (error); } g_assert_cmpint (line, ==, MAX_LINES); g_object_unref (base_stream); g_object_unref (stream); } static void test_read_lines_LF (void) { test_read_lines (G_DATA_STREAM_NEWLINE_TYPE_LF); } static void test_read_lines_CR (void) { test_read_lines (G_DATA_STREAM_NEWLINE_TYPE_CR); } static void test_read_lines_CR_LF (void) { test_read_lines (G_DATA_STREAM_NEWLINE_TYPE_CR_LF); } static void test_read_lines_any (void) { test_read_lines (G_DATA_STREAM_NEWLINE_TYPE_ANY); } static void test_read_lines_LF_valid_utf8 (void) { GInputStream *stream; GInputStream *base_stream; GError *error = NULL; char *line; guint n_lines = 0; base_stream = g_memory_input_stream_new (); stream = G_INPUT_STREAM (g_data_input_stream_new (base_stream)); g_memory_input_stream_add_data (G_MEMORY_INPUT_STREAM (base_stream), "foo\nthis is valid UTF-8 ☺!\nbar\n", -1, NULL); /* Test read line */ error = NULL; while (TRUE) { gsize length = -1; line = g_data_input_stream_read_line_utf8 (G_DATA_INPUT_STREAM (stream), &length, NULL, &error); g_assert_no_error (error); if (line == NULL) { g_assert_cmpuint (length, ==, 0); break; } else { g_assert_cmpuint (length, >, 0); } n_lines++; g_free (line); } g_assert_cmpint (n_lines, ==, 3); g_object_unref (base_stream); g_object_unref (stream); } static void test_read_lines_LF_invalid_utf8 (void) { GInputStream *stream; GInputStream *base_stream; GError *error = NULL; char *line; guint n_lines = 0; base_stream = g_memory_input_stream_new (); stream = G_INPUT_STREAM (g_data_input_stream_new (base_stream)); g_memory_input_stream_add_data (G_MEMORY_INPUT_STREAM (base_stream), "foo\nthis is not valid UTF-8 \xE5 =(\nbar\n", -1, NULL); /* Test read line */ error = NULL; while (TRUE) { gsize length = -1; line = g_data_input_stream_read_line_utf8 (G_DATA_INPUT_STREAM (stream), &length, NULL, &error); if (n_lines == 0) { /* First line is valid UTF-8 */ g_assert_no_error (error); g_assert_cmpuint (length, ==, 3); } else { g_assert_error (error, G_CONVERT_ERROR, G_CONVERT_ERROR_ILLEGAL_SEQUENCE); g_clear_error (&error); g_assert_cmpuint (length, ==, 0); g_free (line); break; } n_lines++; g_free (line); } g_assert_cmpint (n_lines, ==, 1); g_object_unref (base_stream); g_object_unref (stream); } G_GNUC_BEGIN_IGNORE_DEPRECATIONS static void test_read_until (void) { GInputStream *stream; GInputStream *base_stream; GError *error = NULL; char *data; int line; int i; #define REPEATS 10 /* number of rounds */ #define DATA_STRING " part1 # part2 $ part3 % part4 ^" #define DATA_PART_LEN 7 /* number of characters between separators */ #define DATA_SEP "#$%^" #define DATA_SEP_LEN 4 const int DATA_PARTS_NUM = DATA_SEP_LEN * REPEATS; base_stream = g_memory_input_stream_new (); stream = G_INPUT_STREAM (g_data_input_stream_new (base_stream)); for (i = 0; i < REPEATS; i++) g_memory_input_stream_add_data (G_MEMORY_INPUT_STREAM (base_stream), DATA_STRING, -1, NULL); /* Test stop characters */ error = NULL; data = (char*)1; line = 0; while (data) { gsize length = -1; data = g_data_input_stream_read_until (G_DATA_INPUT_STREAM (stream), DATA_SEP, &length, NULL, &error); if (data) { g_assert_cmpint (strlen (data), ==, DATA_PART_LEN); g_free (data); g_assert_no_error (error); line++; } } g_assert_no_error (error); g_assert_cmpint (line, ==, DATA_PARTS_NUM); g_object_unref (base_stream); g_object_unref (stream); } G_GNUC_END_IGNORE_DEPRECATIONS static char * escape_data_string (const char *str, size_t len) { char *escaped = g_memdup2 (str, len + 1); for (size_t i = 0; i < len; i++) { if (escaped[i] == '\0') escaped[i] = '?'; } return escaped; } static void test_read_upto (void) { const struct { int n_repeats; const char *data_string; size_t data_string_len; size_t data_part_len; const char *data_sep; size_t data_sep_len; } vectors[] = { { 10, " part1 # part2 $ part3 \0 part4 ", 32, 7, "#$\0^", 4 }, { 20, "{\"key\": \"value\"}\0", 17, 16, "\0", 1 }, }; #undef REPEATS #undef DATA_STRING #undef DATA_PART_LEN #undef DATA_SEP #undef DATA_SEP_LEN #define REPEATS vectors[n].n_repeats #define DATA_STRING vectors[n].data_string #define DATA_STRING_LEN vectors[n].data_string_len #define DATA_PART_LEN vectors[n].data_part_len #define DATA_SEP vectors[n].data_sep #define DATA_SEP_LEN vectors[n].data_sep_len for (guint n = 0; n < G_N_ELEMENTS (vectors); n++) { const int DATA_PARTS_NUM = DATA_SEP_LEN * REPEATS; GInputStream *stream; GInputStream *base_stream; GError *error = NULL; char *data; int line; int i; guchar stop_char; char *escaped_data = escape_data_string (DATA_STRING, DATA_STRING_LEN); char *escaped_sep = escape_data_string (DATA_SEP, DATA_SEP_LEN); g_test_message ("Test vector %u: %s and %s", n, escaped_data, escaped_sep); g_free (escaped_data); g_free (escaped_sep); base_stream = g_memory_input_stream_new (); stream = G_INPUT_STREAM (g_data_input_stream_new (base_stream)); for (i = 0; i < REPEATS; i++) g_memory_input_stream_add_data (G_MEMORY_INPUT_STREAM (base_stream), DATA_STRING, DATA_STRING_LEN, NULL); /* Test stop characters */ error = NULL; data = (char*)1; line = 0; while (data) { gsize length = -1; data = g_data_input_stream_read_upto (G_DATA_INPUT_STREAM (stream), DATA_SEP, DATA_SEP_LEN, &length, NULL, &error); if (data) { g_assert_cmpint (strlen (data), ==, DATA_PART_LEN); g_assert_no_error (error); line++; stop_char = g_data_input_stream_read_byte (G_DATA_INPUT_STREAM (stream), NULL, &error); g_assert_nonnull (memchr (DATA_SEP, stop_char, DATA_SEP_LEN)); g_assert_no_error (error); } g_free (data); } g_assert_no_error (error); g_assert_cmpint (line, ==, DATA_PARTS_NUM); g_object_unref (base_stream); g_object_unref (stream); } } enum TestDataType { TEST_DATA_BYTE = 0, TEST_DATA_INT16, TEST_DATA_UINT16, TEST_DATA_INT32, TEST_DATA_UINT32, TEST_DATA_INT64, TEST_DATA_UINT64 }; /* The order is reversed to avoid -Wduplicated-branches. */ #define TEST_DATA_RETYPE_BUFF(a, t, v) \ (a == TEST_DATA_UINT64 ? (t) *(guint64*)v : \ (a == TEST_DATA_INT64 ? (t) *(gint64*)v : \ (a == TEST_DATA_UINT32 ? (t) *(guint32*)v : \ (a == TEST_DATA_INT32 ? (t) *(gint32*)v : \ (a == TEST_DATA_UINT16 ? (t) *(guint16*)v : \ (a == TEST_DATA_INT16 ? (t) *(gint16*)v : \ (t) *(guchar*)v )))))) static void test_data_array (GInputStream *stream, GInputStream *base_stream, gpointer buffer, int len, enum TestDataType data_type, GDataStreamByteOrder byte_order) { GError *error = NULL; int pos = 0; int data_size = 1; gint64 data; GDataStreamByteOrder native; gboolean swap; /* Seek to start */ test_seek_to_start (base_stream); /* Set correct data size */ switch (data_type) { case TEST_DATA_BYTE: data_size = 1; break; case TEST_DATA_INT16: case TEST_DATA_UINT16: data_size = 2; break; case TEST_DATA_INT32: case TEST_DATA_UINT32: data_size = 4; break; case TEST_DATA_INT64: case TEST_DATA_UINT64: data_size = 8; break; default: g_assert_not_reached (); break; } /* Set flag to swap bytes if needed */ native = (G_BYTE_ORDER == G_BIG_ENDIAN) ? G_DATA_STREAM_BYTE_ORDER_BIG_ENDIAN : G_DATA_STREAM_BYTE_ORDER_LITTLE_ENDIAN; swap = (byte_order != G_DATA_STREAM_BYTE_ORDER_HOST_ENDIAN) && (byte_order != native); data = 1; while (data != 0) { switch (data_type) { case TEST_DATA_BYTE: data = g_data_input_stream_read_byte (G_DATA_INPUT_STREAM (stream), NULL, &error); break; case TEST_DATA_INT16: data = g_data_input_stream_read_int16 (G_DATA_INPUT_STREAM (stream), NULL, &error); if (swap) data = (gint16)GUINT16_SWAP_LE_BE((gint16)data); break; case TEST_DATA_UINT16: data = g_data_input_stream_read_uint16 (G_DATA_INPUT_STREAM (stream), NULL, &error); if (swap) data = (guint16)GUINT16_SWAP_LE_BE((guint16)data); break; case TEST_DATA_INT32: data = g_data_input_stream_read_int32 (G_DATA_INPUT_STREAM (stream), NULL, &error); if (swap) data = (gint32)GUINT32_SWAP_LE_BE((gint32)data); break; case TEST_DATA_UINT32: data = g_data_input_stream_read_uint32 (G_DATA_INPUT_STREAM (stream), NULL, &error); if (swap) data = (guint32)GUINT32_SWAP_LE_BE((guint32)data); break; case TEST_DATA_INT64: data = g_data_input_stream_read_int64 (G_DATA_INPUT_STREAM (stream), NULL, &error); if (swap) data = (gint64)GUINT64_SWAP_LE_BE((gint64)data); break; case TEST_DATA_UINT64: data = g_data_input_stream_read_uint64 (G_DATA_INPUT_STREAM (stream), NULL, &error); if (swap) data = (guint64)GUINT64_SWAP_LE_BE((guint64)data); break; default: g_assert_not_reached (); break; } if (!error) g_assert_cmpint (data, ==, TEST_DATA_RETYPE_BUFF(data_type, gint64, ((guchar*)buffer + pos))); pos += data_size; } if (pos < len + 1) g_assert_no_error (error); if (error) g_error_free (error); g_assert_cmpint (pos - data_size, ==, len); } static void test_read_int (void) { GInputStream *stream; GInputStream *base_stream; GRand *randomizer; int i; gpointer buffer; randomizer = g_rand_new (); buffer = g_malloc0 (MAX_BYTES); /* Fill in some random data */ for (i = 0; i < MAX_BYTES; i++) { guchar x = 0; while (! x) x = (guchar)g_rand_int (randomizer); *(guchar*)((guchar*)buffer + sizeof(guchar) * i) = x; } base_stream = g_memory_input_stream_new (); stream = G_INPUT_STREAM (g_data_input_stream_new (base_stream)); g_memory_input_stream_add_data (G_MEMORY_INPUT_STREAM (base_stream), buffer, MAX_BYTES, NULL); for (i = 0; i < 3; i++) { int j; g_data_input_stream_set_byte_order (G_DATA_INPUT_STREAM (stream), i); for (j = 0; j <= TEST_DATA_UINT64; j++) test_data_array (stream, base_stream, buffer, MAX_BYTES, j, i); } g_object_unref (base_stream); g_object_unref (stream); g_rand_free (randomizer); g_free (buffer); } int main (int argc, char *argv[]) { g_test_init (&argc, &argv, NULL); g_test_add_func ("/data-input-stream/basic", test_basic); g_test_add_func ("/data-input-stream/read-lines-LF", test_read_lines_LF); g_test_add_func ("/data-input-stream/read-lines-LF-valid-utf8", test_read_lines_LF_valid_utf8); g_test_add_func ("/data-input-stream/read-lines-LF-invalid-utf8", test_read_lines_LF_invalid_utf8); g_test_add_func ("/data-input-stream/read-lines-CR", test_read_lines_CR); g_test_add_func ("/data-input-stream/read-lines-CR-LF", test_read_lines_CR_LF); g_test_add_func ("/data-input-stream/read-lines-any", test_read_lines_any); g_test_add_func ("/data-input-stream/read-until", test_read_until); g_test_add_func ("/data-input-stream/read-upto", test_read_upto); g_test_add_func ("/data-input-stream/read-int", test_read_int); return g_test_run(); }