glib2.0/glib/tests/io-channel.c

230 lines
6.4 KiB
C

/* GLib testing framework examples and tests
*
* Copyright © 2001 Hidetoshi Tajima
* Copyright © 2001 Ron Steinke
* Copyright © 2001 Owen Taylor
* Copyright © 2002 Manish Singh
* Copyright © 2011 Sjoerd Simons
* Copyright © 2012 Simon McVittie
* Copyright © 2013 Stef Walter
* Copyright © 2005, 2006, 2008, 2012, 2013 Matthias Clasen
* Copyright © 2020 Endless Mobile, Inc.
*
* SPDX-License-Identifier: LGPL-2.1-or-later
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General
* Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
*
* Author: Philip Withnall <withnall@endlessm.com>
*/
#include <glib.h>
#include <glib/gstdio.h>
static void
test_small_writes (void)
{
GIOChannel *io;
GIOStatus status = G_IO_STATUS_ERROR;
guint bytes_remaining;
gchar tmp;
GError *local_error = NULL;
io = g_io_channel_new_file ("iochannel-test-outfile", "w", &local_error);
g_assert_no_error (local_error);
g_io_channel_set_encoding (io, NULL, NULL);
g_io_channel_set_buffer_size (io, 1022);
bytes_remaining = 2 * g_io_channel_get_buffer_size (io);
tmp = 0;
while (bytes_remaining)
{
status = g_io_channel_write_chars (io, &tmp, 1, NULL, NULL);
if (status == G_IO_STATUS_ERROR)
break;
if (status == G_IO_STATUS_NORMAL)
bytes_remaining--;
}
g_assert_cmpint (status, ==, G_IO_STATUS_NORMAL);
g_io_channel_unref (io);
g_remove ("iochannel-test-outfile");
}
static void
test_read_write (void)
{
GIOChannel *gio_r, *gio_w ;
GError *local_error = NULL;
GString *buffer;
char *filename;
gint rlength = 0;
glong wlength = 0;
gsize length_out;
const gchar *encoding = "EUC-JP";
GIOStatus status;
const gsize buffer_size_bytes = 1024;
filename = g_test_build_filename (G_TEST_DIST, "iochannel-test-infile", NULL);
setbuf (stdout, NULL); /* For debugging */
gio_r = g_io_channel_new_file (filename, "r", &local_error);
g_assert_no_error (local_error);
gio_w = g_io_channel_new_file ("iochannel-test-outfile", "w", &local_error);
g_assert_no_error (local_error);
g_io_channel_set_encoding (gio_r, encoding, &local_error);
g_assert_no_error (local_error);
g_io_channel_set_buffer_size (gio_r, buffer_size_bytes);
status = g_io_channel_set_flags (gio_r, G_IO_FLAG_NONBLOCK, &local_error);
if (status == G_IO_STATUS_ERROR)
{
#ifdef G_OS_WIN32
g_test_message ("FIXME: not implemented on win32");
#else
/* Errors should not happen */
g_assert_no_error (local_error);
#endif
g_clear_error (&local_error);
}
buffer = g_string_sized_new (buffer_size_bytes);
while (TRUE)
{
do
status = g_io_channel_read_line_string (gio_r, buffer, NULL, &local_error);
while (status == G_IO_STATUS_AGAIN);
if (status != G_IO_STATUS_NORMAL)
break;
rlength += buffer->len;
do
status = g_io_channel_write_chars (gio_w, buffer->str, buffer->len,
&length_out, &local_error);
while (status == G_IO_STATUS_AGAIN);
if (status != G_IO_STATUS_NORMAL)
break;
wlength += length_out;
/* Ensure the whole line was written */
g_assert_cmpuint (length_out, ==, buffer->len);
g_test_message ("%s", buffer->str);
g_string_truncate (buffer, 0);
}
switch (status)
{
case G_IO_STATUS_EOF:
break;
case G_IO_STATUS_ERROR:
/* Errors should not happen */
g_assert_no_error (local_error);
g_clear_error (&local_error);
break;
default:
g_assert_not_reached ();
break;
}
do
status = g_io_channel_flush (gio_w, &local_error);
while (status == G_IO_STATUS_AGAIN);
if (status == G_IO_STATUS_ERROR)
{
/* Errors should not happen */
g_assert_no_error (local_error);
g_clear_error (&local_error);
}
g_test_message ("read %d bytes, wrote %ld bytes", rlength, wlength);
g_io_channel_unref (gio_r);
g_io_channel_unref (gio_w);
test_small_writes ();
g_free (filename);
g_string_free (buffer, TRUE);
}
static void
test_read_line_embedded_nuls (void)
{
const guint8 test_data[] = { 'H', 'i', '!', '\0', 'y', 'o', 'u', '\n', ':', ')', '\n' };
gint fd;
gchar *filename = NULL;
GIOChannel *channel = NULL;
GError *local_error = NULL;
gchar *line = NULL;
gsize line_length, terminator_pos;
GIOStatus status;
g_test_summary ("Test that reading a line containing embedded nuls works "
"when using non-standard line terminators.");
/* Write out a temporary file. */
fd = g_file_open_tmp ("glib-test-io-channel-XXXXXX", &filename, &local_error);
g_assert_no_error (local_error);
g_close (fd, NULL);
fd = -1;
g_file_set_contents (filename, (const gchar *) test_data, sizeof (test_data), &local_error);
g_assert_no_error (local_error);
/* Create the channel. */
channel = g_io_channel_new_file (filename, "r", &local_error);
g_assert_no_error (local_error);
/* Only break on newline characters, not nuls.
* Use length -1 here to exercise glib#2323; the case where length > 0
* is covered in glib/tests/protocol.c. */
g_io_channel_set_line_term (channel, "\n", -1);
g_io_channel_set_encoding (channel, NULL, &local_error);
g_assert_no_error (local_error);
status = g_io_channel_read_line (channel, &line, &line_length,
&terminator_pos, &local_error);
g_assert_no_error (local_error);
g_assert_cmpint (status, ==, G_IO_STATUS_NORMAL);
g_assert_cmpuint (line_length, ==, 8);
g_assert_cmpuint (terminator_pos, ==, 7);
g_assert_cmpmem (line, line_length, test_data, 8);
g_free (line);
g_io_channel_unref (channel);
g_free (filename);
}
int
main (int argc,
char *argv[])
{
g_test_init (&argc, &argv, NULL);
g_test_add_func ("/io-channel/read-write", test_read_write);
g_test_add_func ("/io-channel/read-line/embedded-nuls", test_read_line_embedded_nuls);
return g_test_run ();
}