blob: b1fe0657cd8754c807a56684cac456cbf5660b34 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
#include <glib-object.h>
#include <inttypes.h>
#include <signal.h>
#include <stdio.h>
#include <string.h>
#include <sys/time.h>
#include <thrift/c_glib/thrift.h>
#include <thrift/c_glib/protocol/thrift_binary_protocol.h>
#include <thrift/c_glib/transport/thrift_buffered_transport.h>
#include <thrift/c_glib/transport/thrift_framed_transport.h>
#include <thrift/c_glib/transport/thrift_socket.h>
#include <thrift/c_glib/transport/thrift_transport.h>
#include "../gen-c_glib/t_test_thrift_test.h"
/* Handle SIGPIPE signals (indicating the server has closed the
connection prematurely) by outputting an error message before
exiting. */
static void
sigpipe_handler (int signal_number)
{
THRIFT_UNUSED_VAR (signal_number);
/* Flush standard output to make sure the test results so far are
logged */
fflush (stdout);
fputs ("Broken pipe (server closed connection prematurely)\n", stderr);
fflush (stderr);
/* Re-raise the signal, this time invoking the default signal
handler, to terminate the program */
raise (SIGPIPE);
}
/* Compare two gint32 values. Used for sorting and finding integer
values within a GList. */
static gint
gint32_compare (gconstpointer a, gconstpointer b)
{
gint32 int32_a = *(gint32 *)a;
gint32 int32_b = *(gint32 *)b;
int result = 0;
if (int32_a < int32_b)
result = -1;
else if (int32_a > int32_b)
result = 1;
return result;
}
int
main (int argc, char **argv)
{
static gchar *host = NULL;
static gint port = 9090;
static gchar *transport_option = NULL;
static gchar *protocol_option = NULL;
static gint num_tests = 1;
static
GOptionEntry option_entries[] ={
{ "host", 0, 0, G_OPTION_ARG_STRING, &host,
"Host to connect (=localhost)", NULL },
{ "port", 0, 0, G_OPTION_ARG_INT, &port,
"Port number to connect (=9090)", NULL },
{ "transport", 0, 0, G_OPTION_ARG_STRING, &transport_option,
"Transport: buffered, framed (=buffered)", NULL },
{ "protocol", 0, 0, G_OPTION_ARG_STRING, &protocol_option,
"Protocol: binary (=binary)", NULL },
{ "testloops", 'n', 0, G_OPTION_ARG_INT, &num_tests,
"Number of tests (=1)", NULL },
{ NULL }
};
struct sigaction sigpipe_action;
GType transport_type = THRIFT_TYPE_BUFFERED_TRANSPORT;
gchar *transport_name = "buffered";
GType protocol_type = THRIFT_TYPE_BINARY_PROTOCOL;
gchar *protocol_name = "binary";
ThriftSocket *socket;
ThriftTransport *transport;
ThriftProtocol *protocol;
TTestThriftTestIf *test_client;
struct timeval time_start, time_stop, time_elapsed;
guint64 time_elapsed_usec, time_total_usec = 0;
guint64 time_min_usec = G_MAXUINT64, time_max_usec = 0, time_avg_usec;
GOptionContext *option_context;
gboolean options_valid = TRUE;
int test_num = 0;
int fail_count = 0;
GError *error = NULL;
#if (!GLIB_CHECK_VERSION (2, 36, 0))
g_type_init ();
#endif
/* Configure and parse our command-line options */
option_context = g_option_context_new (NULL);
g_option_context_add_main_entries (option_context,
option_entries,
NULL);
if (!g_option_context_parse (option_context,
&argc,
&argv,
&error)) {
fprintf (stderr, "%s\n", error->message);
return 255;
}
g_option_context_free (option_context);
/* Set remaining default values for unspecified options */
if (host == NULL)
host = g_strdup ("localhost");
/* Validate the parsed options */
if (protocol_option != NULL &&
strncmp (protocol_option, "binary", 7) != 0) {
fprintf (stderr, "Unknown protocol type %s\n", protocol_option);
options_valid = FALSE;
}
if (transport_option != NULL) {
if (strncmp (transport_option, "framed", 7) == 0) {
transport_type = THRIFT_TYPE_FRAMED_TRANSPORT;
transport_name = "framed";
}
else if (strncmp (transport_option, "buffered", 9) != 0) {
fprintf (stderr, "Unknown transport type %s\n", transport_option);
options_valid = FALSE;
}
}
if (!options_valid)
return 254;
printf ("Connecting (%s/%s) to: %s:%d\n",
transport_name,
protocol_name,
host,
port);
/* Install our SIGPIPE handler, which outputs an error message to
standard error before exiting so testers can know what
happened */
memset (&sigpipe_action, 0, sizeof (sigpipe_action));
sigpipe_action.sa_handler = sigpipe_handler;
sigpipe_action.sa_flags = SA_RESETHAND;
sigaction (SIGPIPE, &sigpipe_action, NULL);
/* Establish all our connection objects */
socket = g_object_new (THRIFT_TYPE_SOCKET,
"hostname", host,
"port", port,
NULL);
transport = g_object_new (transport_type,
"transport", socket,
NULL);
protocol = g_object_new (protocol_type,
"transport", transport,
NULL);
test_client = g_object_new (T_TEST_TYPE_THRIFT_TEST_CLIENT,
"input_protocol", protocol,
"output_protocol", protocol,
NULL);
/* Execute the actual tests */
for (test_num = 0; test_num < num_tests; ++test_num) {
if (thrift_transport_open (transport, &error)) {
gchar *string = NULL;
gboolean boolean = 0;
gint8 byte = 0;
gint32 int32 = 0;
gint64 int64 = 0;
gdouble dub = 0;
gint byte_thing, i32_thing, inner_byte_thing, inner_i32_thing;
gint64 i64_thing, inner_i64_thing;
TTestXtruct *xtruct_out, *xtruct_in, *inner_xtruct_in;
TTestXtruct2 *xtruct2_out, *xtruct2_in;
GHashTable *map_out, *map_in, *inner_map_in;
GHashTable *set_out, *set_in;
gpointer key, value;
gint32 *i32_key_ptr, *i32_value_ptr;
GHashTableIter hash_table_iter, inner_hash_table_iter;
GList *keys_out, *keys_in, *keys_elem;
GArray *list_out, *list_in;
TTestNumberz numberz;
TTestUserId user_id, *user_id_ptr;
TTestInsanity *insanity_out, *insanity_in;
GHashTable *user_map;
GHashTableIter user_map_iter;
GPtrArray *xtructs;
TTestXception *xception = NULL;
TTestXception2 *xception2 = NULL;
gboolean oneway_result;
struct timeval oneway_start, oneway_end, oneway_elapsed;
gint oneway_elapsed_usec;
gboolean first;
gint32 i, j;
printf ("Test #%d, connect %s:%d\n", test_num + 1, host, port);
gettimeofday (&time_start, NULL);
/* These test routines have been ported from the C++ test
client, care being taken to ensure their output remains as
close as possible to the original to facilitate diffs.
For simplicity comments have been omitted, but every routine
has the same basic structure:
- Create and populate data structures as necessary.
- Format and output (to the console) a representation of the
outgoing data.
- Issue the remote method call to the server.
- Format and output a representation of the returned data.
- Verify the returned data matches what was expected.
- Deallocate any created data structures.
Note the recognized values and expected behaviour of each
remote method are described in ThriftTest.thrift, which
you'll find in the top-level "test" folder. */
/**
* VOID TEST
*/
printf ("testVoid()");
if (t_test_thrift_test_if_test_void (test_client, &error)) {
printf (" = void\n");
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
/**
* STRING TEST
*/
printf ("testString(\"Test\")");
if (t_test_thrift_test_if_test_string (test_client,
&string,
"Test",
&error)) {
printf (" = \"%s\"\n", string);
if (strncmp (string, "Test", 5) != 0)
fail_count++;
g_free (string);
string = NULL;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
/**
* BOOL TEST
*/
printf ("testByte(true)");
if (t_test_thrift_test_if_test_bool (test_client,
&boolean,
1,
&error)) {
printf (" = %s\n", boolean ? "true" : "false");
if (boolean != 1)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
printf ("testByte(false)");
if (t_test_thrift_test_if_test_bool (test_client,
&boolean,
0,
&error)) {
printf (" = %s\n", boolean ? "true" : "false");
if (boolean != 0)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
/**
* BYTE TEST
*/
printf ("testByte(1)");
if (t_test_thrift_test_if_test_byte (test_client,
&byte,
1,
&error)) {
printf (" = %d\n", byte);
if (byte != 1)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
/**
* I32 TEST
*/
printf ("testI32(-1)");
if (t_test_thrift_test_if_test_i32 (test_client,
&int32,
-1,
&error)) {
printf (" = %d\n", int32);
if (int32 != -1)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
/**
* I64 TEST
*/
printf ("testI64(-34359738368)");
if (t_test_thrift_test_if_test_i64 (test_client,
&int64,
(gint64)-34359738368,
&error)) {
printf (" = %" PRId64 "\n", int64);
if (int64 != (gint64)-34359738368)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
/**
* DOUBLE TEST
*/
printf("testDouble(-5.2098523)");
if (t_test_thrift_test_if_test_double (test_client,
&dub,
-5.2098523,
&error)) {
printf (" = %f\n", dub);
if ((dub - (-5.2098523)) > 0.001)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
// TODO: add testBinary()
/**
* STRUCT TEST
*/
printf ("testStruct({\"Zero\", 1, -3, -5})");
xtruct_out = g_object_new (T_TEST_TYPE_XTRUCT,
"string_thing", "Zero",
"byte_thing", 1,
"i32_thing", -3,
"i64_thing", -5LL,
NULL);
xtruct_in = g_object_new (T_TEST_TYPE_XTRUCT, NULL);
if (t_test_thrift_test_if_test_struct (test_client,
&xtruct_in,
xtruct_out,
&error)) {
g_object_get (xtruct_in,
"string_thing", &string,
"byte_thing", &byte_thing,
"i32_thing", &i32_thing,
"i64_thing", &i64_thing,
NULL);
printf (" = {\"%s\", %d, %d, %" PRId64 "}\n",
string,
byte_thing,
i32_thing,
i64_thing);
if ((string == NULL || strncmp (string, "Zero", 5) != 0) ||
byte_thing != 1 ||
i32_thing != -3 ||
i64_thing != (gint64)-5)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
g_object_unref (xtruct_in);
/**
* NESTED STRUCT TEST
*/
printf ("testNest({1, {\"Zero\", 1, -3, -5}), 5}");
xtruct2_out = g_object_new (T_TEST_TYPE_XTRUCT2,
"byte_thing", 1,
"struct_thing", xtruct_out,
"i32_thing", 5,
NULL);
xtruct2_in = g_object_new (T_TEST_TYPE_XTRUCT2, NULL);
if (t_test_thrift_test_if_test_nest (test_client,
&xtruct2_in,
xtruct2_out,
&error)) {
g_object_get (xtruct2_in,
"byte_thing", &byte_thing,
"struct_thing", &xtruct_in,
"i32_thing", &i32_thing,
NULL);
g_object_get (xtruct_in,
"string_thing", &string,
"byte_thing", &inner_byte_thing,
"i32_thing", &inner_i32_thing,
"i64_thing", &inner_i64_thing,
NULL);
printf (" = {%d, {\"%s\", %d, %d, %" PRId64 "}, %d}\n",
byte_thing,
string,
inner_byte_thing,
inner_i32_thing,
inner_i64_thing,
i32_thing);
if (byte_thing != 1 ||
(string == NULL || strncmp (string, "Zero", 5) != 0) ||
inner_byte_thing != 1 ||
inner_i32_thing != -3 ||
inner_i64_thing != (gint64)-5 ||
i32_thing != 5)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
g_object_unref (xtruct_in);
g_object_unref (xtruct2_in);
g_object_unref (xtruct2_out);
g_object_unref (xtruct_out);
/**
* MAP TEST
*/
map_out = g_hash_table_new_full (g_int_hash,
g_int_equal,
g_free,
g_free);
for (i = 0; i < 5; ++i) {
i32_key_ptr = g_malloc (sizeof *i32_key_ptr);
i32_value_ptr = g_malloc (sizeof *i32_value_ptr);
*i32_key_ptr = i;
*i32_value_ptr = i - 10;
g_hash_table_insert (map_out, i32_key_ptr, i32_value_ptr);
}
printf ("testMap({");
first = TRUE;
g_hash_table_iter_init (&hash_table_iter, map_out);
while (g_hash_table_iter_next (&hash_table_iter,
&key,
&value)) {
if (first)
first = FALSE;
else
printf (", ");
printf ("%d => %d", *(gint32 *)key, *(gint32 *)value);
}
printf ("})");
map_in = g_hash_table_new_full (g_int_hash,
g_int_equal,
g_free,
g_free);
if (t_test_thrift_test_if_test_map (test_client,
&map_in,
map_out,
&error)) {
printf (" = {");
first = TRUE;
g_hash_table_iter_init (&hash_table_iter, map_in);
while (g_hash_table_iter_next (&hash_table_iter,
&key,
&value)) {
if (first)
first = FALSE;
else
printf (", ");
printf ("%d => %d", *(gint32 *)key, *(gint32 *)value);
}
printf ("}\n");
if (g_hash_table_size (map_in) != g_hash_table_size (map_out))
fail_count++;
else {
g_hash_table_iter_init (&hash_table_iter, map_out);
while (g_hash_table_iter_next (&hash_table_iter,
&key,
&value)) {
gpointer in_value = g_hash_table_lookup (map_in, key);
if (in_value == NULL ||
*(gint32 *)in_value != *(gint32 *)value) {
fail_count++;
break;
}
}
}
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
g_hash_table_unref (map_in);
g_hash_table_unref (map_out);
/**
* STRING MAP TEST
*/
map_out = g_hash_table_new_full (g_str_hash,
g_str_equal,
NULL,
NULL);
g_hash_table_insert (map_out, "a", "2");
g_hash_table_insert (map_out, "b", "blah");
g_hash_table_insert (map_out, "some", "thing");
printf ("testStringMap({");
first = TRUE;
g_hash_table_iter_init (&hash_table_iter, map_out);
while (g_hash_table_iter_next (&hash_table_iter,
&key,
&value)) {
if (first)
first = FALSE;
else
printf (", ");
printf ("\"%s\" => \"%s\"", (gchar *)key, (gchar *)value);
}
printf (")}");
map_in = g_hash_table_new_full (g_str_hash,
g_str_equal,
g_free,
g_free);
if (t_test_thrift_test_if_test_string_map (test_client,
&map_in,
map_out,
&error)) {
printf (" = {");
first = TRUE;
g_hash_table_iter_init (&hash_table_iter, map_in);
while (g_hash_table_iter_next (&hash_table_iter,
&key,
&value)) {
if (first)
first = FALSE;
else
printf (", ");
printf ("\"%s\" => \"%s\"", (gchar *)key, (gchar *)value);
}
printf ("}\n");
if (g_hash_table_size (map_in) != g_hash_table_size (map_out))
fail_count++;
else {
g_hash_table_iter_init (&hash_table_iter, map_out);
while (g_hash_table_iter_next (&hash_table_iter,
&key,
&value)) {
gpointer in_value = g_hash_table_lookup (map_in, key);
if (in_value == NULL ||
strcmp ((gchar *)in_value, (gchar *)value) != 0) {
fail_count++;
break;
}
}
}
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
g_hash_table_unref (map_in);
g_hash_table_unref (map_out);
/**
* SET TEST
*/
set_out = g_hash_table_new_full (g_int_hash, g_int_equal, g_free, NULL);
for (i = -2; i < 3; ++i) {
i32_key_ptr = g_malloc (sizeof *i32_key_ptr);
*i32_key_ptr = i;
g_hash_table_insert (set_out, i32_key_ptr, NULL);
}
printf ("testSet({");
first = TRUE;
keys_out = g_hash_table_get_keys (set_out);
keys_elem = keys_out;
while (keys_elem != NULL) {
if (first)
first = FALSE;
else
printf (", ");
printf ("%d", *(gint32 *)keys_elem->data);
keys_elem = keys_elem->next;
}
printf ("})");
set_in = g_hash_table_new_full (g_int_hash, g_int_equal, g_free, NULL);
if (t_test_thrift_test_if_test_set (test_client,
&set_in,
set_out,
&error)) {
printf(" = {");
first = TRUE;
keys_in = g_hash_table_get_keys (set_in);
keys_elem = keys_in;
while (keys_elem != NULL) {
if (first)
first = FALSE;
else
printf (", ");
printf ("%d", *(gint32 *)keys_elem->data);
keys_elem = keys_elem->next;
}
printf ("}\n");
if (g_list_length (keys_in) != g_list_length (keys_out))
fail_count++;
else {
keys_elem = keys_out;
while (keys_elem != NULL) {
if (g_list_find_custom (keys_in,
keys_elem->data,
gint32_compare) == NULL) {
fail_count++;
break;
}
keys_elem = keys_elem->next;
}
}
g_list_free (keys_in);
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
g_hash_table_unref (set_in);
g_list_free (keys_out);
g_hash_table_unref (set_out);
/**
* LIST TEST
*/
list_out = g_array_new (FALSE, TRUE, sizeof (gint32));
for (i = -2; i < 3; ++i) {
g_array_append_val (list_out, i);
}
printf ("testList({");
first = TRUE;
for (i = 0; i < (gint32)list_out->len; ++i) {
if (first)
first = FALSE;
else
printf (", ");
printf ("%d", g_array_index (list_out, gint32, i));
}
printf ("})");
list_in = g_array_new (FALSE, TRUE, sizeof (gint32));
if (t_test_thrift_test_if_test_list (test_client,
&list_in,
list_out,
&error)) {
printf (" = {");
first = TRUE;
for (i = 0; i < (gint32)list_in->len; ++i) {
if (first)
first = FALSE;
else
printf (", ");
printf ("%d", g_array_index (list_in, gint32, i));
}
printf ("}\n");
if (list_in->len != list_out->len ||
memcmp (list_in->data,
list_out->data,
list_in->len * sizeof (gint32)) != 0)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
g_array_unref (list_in);
g_array_unref (list_out);
/**
* ENUM TEST
*/
printf("testEnum(ONE)");
if (t_test_thrift_test_if_test_enum (test_client,
&numberz,
T_TEST_NUMBERZ_ONE,
&error)) {
printf(" = %d\n", numberz);
if (numberz != T_TEST_NUMBERZ_ONE)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
printf("testEnum(TWO)");
if (t_test_thrift_test_if_test_enum (test_client,
&numberz,
T_TEST_NUMBERZ_TWO,
&error)) {
printf(" = %d\n", numberz);
if (numberz != T_TEST_NUMBERZ_TWO)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
printf("testEnum(THREE)");
if (t_test_thrift_test_if_test_enum (test_client,
&numberz,
T_TEST_NUMBERZ_THREE,
&error)) {
printf(" = %d\n", numberz);
if (numberz != T_TEST_NUMBERZ_THREE)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
printf("testEnum(FIVE)");
if (t_test_thrift_test_if_test_enum (test_client,
&numberz,
T_TEST_NUMBERZ_FIVE,
&error)) {
printf(" = %d\n", numberz);
if (numberz != T_TEST_NUMBERZ_FIVE)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
printf("testEnum(EIGHT)");
if (t_test_thrift_test_if_test_enum (test_client,
&numberz,
T_TEST_NUMBERZ_EIGHT,
&error)) {
printf(" = %d\n", numberz);
if (numberz != T_TEST_NUMBERZ_EIGHT)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
/**
* TYPEDEF TEST
*/
printf ("testTypedef(309858235082523)");
if (t_test_thrift_test_if_test_typedef (test_client,
&user_id,
309858235082523LL,
&error)) {
printf(" = %" PRId64 "\n", user_id);
if (user_id != 309858235082523LL)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
/**
* NESTED MAP TEST
*/
printf ("testMapMap(1)");
map_in = g_hash_table_new_full (g_int_hash,
g_int_equal,
g_free,
(GDestroyNotify)g_hash_table_unref);
if (t_test_thrift_test_if_test_map_map (test_client,
&map_in,
1,
&error)) {
g_hash_table_iter_init (&hash_table_iter, map_in);
printf (" = {");
while (g_hash_table_iter_next (&hash_table_iter,
&key,
&value)) {
printf ("%d => {", *(gint32 *)key);
g_hash_table_iter_init (&inner_hash_table_iter,
(GHashTable *)value);
while (g_hash_table_iter_next (&inner_hash_table_iter,
&key,
&value)) {
printf ("%d => %d, ", *(gint32 *)key, *(gint32 *)value);
}
printf ("}, ");
}
printf ("}\n");
if (g_hash_table_size (map_in) != 2)
fail_count++;
else {
gint32 inner_keys[] = {1, 2, 3, 4};
gint32 i32_key;
i32_key = -4;
inner_map_in = g_hash_table_lookup (map_in, &i32_key);
if (inner_map_in == NULL ||
g_hash_table_size (inner_map_in) != 4)
fail_count++;
else {
keys_in = g_hash_table_get_keys (inner_map_in);
keys_in = g_list_sort (keys_in, gint32_compare);
for (i = 0; i < 4; i++) {
keys_elem = g_list_nth (keys_in, 3 - i);
if (*(gint32 *)keys_elem->data != (-1 * inner_keys[i]) ||
*(gint32 *)g_hash_table_lookup (inner_map_in,
keys_elem->data) !=
(-1 * inner_keys[i])) {
fail_count++;
break;
}
}
g_list_free (keys_in);
}
i32_key = 4;
inner_map_in = g_hash_table_lookup (map_in, &i32_key);
if (inner_map_in == NULL ||
g_hash_table_size (inner_map_in) != 4)
fail_count++;
else {
keys_in = g_hash_table_get_keys (inner_map_in);
keys_in = g_list_sort (keys_in, gint32_compare);
for (i = 0; i < 4; i++) {
keys_elem = g_list_nth (keys_in, i);
if (*(gint32 *)keys_elem->data != inner_keys[i] ||
*(gint32 *)g_hash_table_lookup (inner_map_in,
keys_elem->data) !=
inner_keys[i]) {
fail_count++;
break;
}
}
g_list_free (keys_in);
}
}
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
g_hash_table_unref (map_in);
/**
* INSANITY TEST
*/
insanity_out = g_object_new (T_TEST_TYPE_INSANITY, NULL);
g_object_get (insanity_out,
"userMap", &user_map,
"xtructs", &xtructs,
NULL);
numberz = T_TEST_NUMBERZ_FIVE;
user_id_ptr = g_malloc (sizeof *user_id_ptr);
*user_id_ptr = 5000;
g_hash_table_insert (user_map, (gpointer)numberz, user_id_ptr);
g_hash_table_unref (user_map);
xtruct_out = g_object_new (T_TEST_TYPE_XTRUCT,
"string_thing", "Truck",
"byte_thing", 8,
"i32_thing", 8,
"i64_thing", 8LL,
NULL);
g_ptr_array_add (xtructs, xtruct_out);
g_ptr_array_unref (xtructs);
map_in = g_hash_table_new_full (g_int64_hash,
g_int64_equal,
g_free,
(GDestroyNotify)g_hash_table_unref);
printf("testInsanity()");
if (t_test_thrift_test_if_test_insanity (test_client,
&map_in,
insanity_out,
&error)) {
printf (" = {");
g_hash_table_iter_init (&hash_table_iter, map_in);
while (g_hash_table_iter_next (&hash_table_iter,
&key,
&value)) {
printf ("%" PRId64 " => {", *(TTestUserId *)key);
g_hash_table_iter_init (&inner_hash_table_iter,
(GHashTable *)value);
while (g_hash_table_iter_next (&inner_hash_table_iter,
&key,
&value)) {
printf ("%d => {", (TTestNumberz)key);
g_object_get ((TTestInsanity *)value,
"userMap", &user_map,
"xtructs", &xtructs,
NULL);
printf ("{");
g_hash_table_iter_init (&user_map_iter, user_map);
while (g_hash_table_iter_next (&user_map_iter,
&key,
&value)) {
printf ("%d => %" PRId64 ", ",
(TTestNumberz)key,
*(TTestUserId *)value);
}
printf ("}, ");
g_hash_table_unref (user_map);
printf("{");
for (i = 0; i < (gint32)xtructs->len; ++i) {
xtruct_in = g_ptr_array_index (xtructs, i);
g_object_get (xtruct_in,
"string_thing", &string,
"byte_thing", &byte_thing,
"i32_thing", &i32_thing,
"i64_thing", &i64_thing,
NULL);
printf ("{\"%s\", %d, %d, %" PRId64 "}, ",
string,
byte_thing,
i32_thing,
i64_thing);
}
printf ("}");
g_ptr_array_unref (xtructs);
printf ("}, ");
}
printf("}, ");
}
printf("}\n");
if (g_hash_table_size (map_in) != 2)
fail_count++;
else {
TTestNumberz numberz_key_values[] = {
T_TEST_NUMBERZ_TWO, T_TEST_NUMBERZ_THREE
};
gint user_map_values[] = { 5, 8 };
TTestUserId user_id_key;
user_id_key = 1;
inner_map_in = g_hash_table_lookup (map_in, &user_id_key);
if (inner_map_in == NULL ||
g_hash_table_size (inner_map_in) != 2)
fail_count++;
else {
TTestNumberz numberz_key;
for (i = 0; i < 2; ++i) {
numberz_key = numberz_key_values[i];
insanity_in =
g_hash_table_lookup (inner_map_in,
(gconstpointer)numberz_key);
if (insanity_in == NULL)
fail_count++;
else {
g_object_get (insanity_in,
"userMap", &user_map,
"xtructs", &xtructs,
NULL);
if (user_map == NULL)
fail_count++;
else {
if (g_hash_table_size (user_map) != 2)
fail_count++;
else {
for (j = 0; j < 2; ++j) {
numberz_key = (TTestNumberz)user_map_values[j];
value =
g_hash_table_lookup (user_map,
(gconstpointer)numberz_key);
if (value == NULL ||
*(TTestUserId *)value != (TTestUserId)user_map_values[j])
fail_count++;
}
}
g_hash_table_unref (user_map);
}
if (xtructs == NULL)
fail_count++;
else {
if (xtructs->len != 2)
fail_count++;
else {
xtruct_in = g_ptr_array_index (xtructs, 0);
g_object_get (xtruct_in,
"string_thing", &string,
"byte_thing", &byte_thing,
"i32_thing", &i32_thing,
"i64_thing", &i64_thing,
NULL);
if ((string == NULL ||
strncmp (string, "Goodbye4", 9) != 0) ||
byte_thing != 4 ||
i32_thing != 4 ||
i64_thing != 4)
fail_count++;
if (string != NULL)
g_free (string);
xtruct_in = g_ptr_array_index (xtructs, 1);
g_object_get (xtruct_in,
"string_thing", &string,
"byte_thing", &byte_thing,
"i32_thing", &i32_thing,
"i64_thing", &i64_thing,
NULL);
if ((string == NULL ||
strncmp (string, "Hello2", 7) != 0) ||
byte_thing != 2 ||
i32_thing != 2 ||
i64_thing != 2)
fail_count++;
if (string != NULL)
g_free (string);
}
g_ptr_array_unref (xtructs);
}
}
}
}
user_id_key = 2;
inner_map_in = g_hash_table_lookup (map_in, &user_id_key);
if (inner_map_in == NULL ||
g_hash_table_size (inner_map_in) != 1)
fail_count++;
else {
insanity_in =
g_hash_table_lookup (inner_map_in,
(gconstpointer)T_TEST_NUMBERZ_SIX);
if (insanity_in == NULL)
fail_count++;
else {
g_object_get (insanity_in,
"userMap", &user_map,
"xtructs", &xtructs,
NULL);
if (user_map == NULL)
fail_count++;
else {
if (g_hash_table_size (user_map) != 0)
fail_count++;
g_hash_table_unref (user_map);
}
if (xtructs == NULL)
fail_count++;
else {
if (xtructs->len != 0)
fail_count++;
g_ptr_array_unref (xtructs);
}
}
}
}
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
g_hash_table_unref (map_in);
g_object_unref (insanity_out);
/* test exception */
printf ("testClient.testException(\"Xception\") =>");
if (!t_test_thrift_test_if_test_exception (test_client,
"Xception",
&xception,
&error) &&
xception != NULL) {
g_object_get (xception,
"errorCode", &int32,
"message", &string,
NULL);
printf (" {%u, \"%s\"}\n", int32, string);
g_free (string);
g_object_unref (xception);
xception = NULL;
g_error_free (error);
error = NULL;
}
else {
printf (" void\nFAILURE\n");
fail_count++;
if (xception != NULL) {
g_object_unref (xception);
xception = NULL;
}
if (error != NULL) {
g_error_free (error);
error = NULL;
}
}
printf ("testClient.testException(\"TException\") =>");
if (!t_test_thrift_test_if_test_exception (test_client,
"TException",
&xception,
&error) &&
xception == NULL &&
error != NULL) {
printf (" Caught TException\n");
g_error_free (error);
error = NULL;
}
else {
printf (" void\nFAILURE\n");
fail_count++;
if (xception != NULL) {
g_object_unref (xception);
xception = NULL;
}
if (error != NULL) {
g_error_free (error);
error = NULL;
}
}
printf ("testClient.testException(\"success\") =>");
if (t_test_thrift_test_if_test_exception (test_client,
"success",
&xception,
&error))
printf (" void\n");
else {
printf (" void\nFAILURE\n");
fail_count++;
if (xception != NULL) {
g_object_unref (xception);
xception = NULL;
}
g_error_free (error);
error = NULL;
}
g_assert (error == NULL);
/* test multi exception */
printf ("testClient.testMultiException(\"Xception\", \"test 1\") =>");
xtruct_in = g_object_new (T_TEST_TYPE_XTRUCT, NULL);
if (!t_test_thrift_test_if_test_multi_exception (test_client,
&xtruct_in,
"Xception",
"test 1",
&xception,
&xception2,
&error) &&
xception != NULL &&
xception2 == NULL) {
g_object_get (xception,
"errorCode", &int32,
"message", &string,
NULL);
printf (" {%u, \"%s\"}\n", int32, string);
g_free (string);
g_object_unref (xception);
xception = NULL;
g_error_free (error);
error = NULL;
}
else {
printf (" result\nFAILURE\n");
fail_count++;
if (xception != NULL) {
g_object_unref (xception);
xception = NULL;
}
if (xception2 != NULL) {
g_object_unref (xception2);
xception = NULL;
}
if (error != NULL) {
g_error_free (error);
error = NULL;
}
}
g_object_unref (xtruct_in);
printf ("testClient.testMultiException(\"Xception2\", \"test 2\") =>");
xtruct_in = g_object_new (T_TEST_TYPE_XTRUCT, NULL);
if (!t_test_thrift_test_if_test_multi_exception (test_client,
&xtruct_in,
"Xception2",
"test 2",
&xception,
&xception2,
&error) &&
xception == NULL &&
xception2 != NULL) {
g_object_get (xception2,
"errorCode", &int32,
"struct_thing", &inner_xtruct_in,
NULL);
g_object_get (inner_xtruct_in,
"string_thing", &string,
NULL);
printf (" {%u, {\"%s\"}}\n", int32, string);
g_free (string);
g_object_unref (inner_xtruct_in);
inner_xtruct_in = NULL;
g_object_unref (xception2);
xception2 = NULL;
g_error_free (error);
error = NULL;
}
else {
printf (" result\nFAILURE\n");
fail_count++;
if (xception != NULL) {
g_object_unref (xception);
xception = NULL;
}
if (xception2 != NULL) {
g_object_unref (xception2);
xception = NULL;
}
if (error != NULL) {
g_error_free (error);
error = NULL;
}
}
g_object_unref (xtruct_in);
printf ("testClient.testMultiException(\"success\", \"test 3\") =>");
xtruct_in = g_object_new (T_TEST_TYPE_XTRUCT, NULL);
if (t_test_thrift_test_if_test_multi_exception (test_client,
&xtruct_in,
"success",
"test 3",
&xception,
&xception2,
&error) &&
xception == NULL &&
xception2 == NULL) {
g_object_get (xtruct_in,
"string_thing", &string,
NULL);
printf (" {{\"%s\"}}\n", string);
g_free (string);
}
else {
printf (" result\nFAILURE\n");
fail_count++;
if (xception != NULL) {
g_object_unref (xception);
xception = NULL;
}
if (xception2 != NULL) {
g_object_unref (xception2);
xception = NULL;
}
if (error != NULL) {
g_error_free (error);
error = NULL;
}
}
g_object_unref (xtruct_in);
/* test oneway void */
printf ("testClient.testOneway(1) =>");
gettimeofday (&oneway_start, NULL);
oneway_result = t_test_thrift_test_if_test_oneway (test_client,
1,
&error);
gettimeofday (&oneway_end, NULL);
timersub (&oneway_end, &oneway_start, &oneway_elapsed);
oneway_elapsed_usec =
oneway_elapsed.tv_sec * 1000 * 1000 + oneway_elapsed.tv_usec;
if (oneway_result) {
if (oneway_elapsed_usec > 200 * 1000) {
printf (" FAILURE - took %.2f ms\n",
(double)oneway_elapsed_usec / 1000.0);
fail_count++;
}
else
printf (" success - took %.2f ms\n",
(double)oneway_elapsed_usec / 1000.0);
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
/**
* redo a simple test after the oneway to make sure we aren't "off by
* one" -- if the server treated oneway void like normal void, this next
* test will fail since it will get the void confirmation rather than
* the correct result. In this circumstance, the client will receive the
* error:
*
* application error: Wrong method name
*/
/**
* I32 TEST
*/
printf ("re-test testI32(-1)");
if (t_test_thrift_test_if_test_i32 (test_client,
&int32,
-1,
&error)) {
printf (" = %d\n", int32);
if (int32 != -1)
fail_count++;
}
else {
printf ("%s\n", error->message);
g_error_free (error);
error = NULL;
fail_count++;
}
gettimeofday (&time_stop, NULL);
timersub (&time_stop, &time_start, &time_elapsed);
time_elapsed_usec =
time_elapsed.tv_sec * 1000 * 1000 + time_elapsed.tv_usec;
printf("Total time: %" PRIu64 " us\n", time_elapsed_usec);
time_total_usec += time_elapsed_usec;
if (time_elapsed_usec < time_min_usec)
time_min_usec = time_elapsed_usec;
if (time_elapsed_usec > time_max_usec)
time_max_usec = time_elapsed_usec;
thrift_transport_close (transport, &error);
}
else {
printf ("Connect failed: %s\n", error->message);
g_error_free (error);
error = NULL;
return 1;
}
}
/* All done---output statistics */
puts ("\nAll tests done.");
time_avg_usec = time_total_usec / num_tests;
printf ("Min time: %" PRIu64 " us\n", time_min_usec);
printf ("Max time: %" PRIu64 " us\n", time_max_usec);
printf ("Avg time: %" PRIu64 " us\n", time_avg_usec);
g_object_unref (test_client);
g_object_unref (protocol);
g_object_unref (transport);
g_free (host);
return fail_count;
}