/* vim:expandtab:shiftwidth=2:tabstop=2:smarttab: * * Gearmand client and server library. * * Copyright (C) 2011-2013 Data Differential, http://datadifferential.com/ * Copyright (C) 2008 Brian Aker, Eric Day * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following disclaimer * in the documentation and/or other materials provided with the * distribution. * * * The names of its contributors may not be used to endorse or * promote products derived from this software without specific prior * written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER 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 "gear_config.h" #include using namespace libtest; #include #include #include #define GEARMAN_CORE #include "libgearman/common.h" #include "libgearman/packet.hpp" #include "libgearman/universal.hpp" #include "tests/regression.h" #ifndef __INTEL_COMPILER #pragma GCC diagnostic ignored "-Wold-style-cast" #endif struct internal_test_st { pid_t gearmand_pid; internal_test_st() : gearmand_pid(-1) { } ~internal_test_st() { } }; static test_return_t init_test(void *) { gearman_universal_st gear; ASSERT_FALSE(gear.options.non_blocking); ASSERT_FALSE(gear._namespace); gearman_universal_free(gear); return TEST_SUCCESS; } static test_return_t clone_test(void *) { gearman_universal_st gear; /* Can we init from null? */ { gearman_universal_st destination; gearman_universal_clone(destination, gear); { // Test all of the flags ASSERT_EQ(destination.options.non_blocking, gear.options.non_blocking); } ASSERT_EQ(destination._namespace, gear._namespace); ASSERT_EQ(destination.verbose, gear.verbose); ASSERT_EQ(destination.con_count, gear.con_count); ASSERT_EQ(destination.packet_count, gear.packet_count); ASSERT_EQ(destination.pfds_size, gear.pfds_size); ASSERT_EQ(destination.last_errno(), gear.last_errno()); ASSERT_EQ(destination.timeout, gear.timeout); ASSERT_EQ(destination.con_list, gear.con_list); ASSERT_EQ(destination.packet_list, gear.packet_list); ASSERT_EQ(destination.pfds, gear.pfds); ASSERT_EQ(destination.log_fn, gear.log_fn); ASSERT_EQ(destination.log_context, gear.log_context); ASSERT_EQ(destination.allocator.malloc, gear.allocator.malloc); ASSERT_EQ(destination.allocator.context, gear.allocator.context); ASSERT_EQ(destination.allocator.free, gear.allocator.free); gearman_universal_free(gear); } gearman_universal_free(gear); return TEST_SUCCESS; } static test_return_t set_timout_test(void *) { gearman_universal_st universal; ASSERT_EQ(-1, gearman_universal_timeout(universal)); // Current default gearman_universal_set_timeout(universal, 20); ASSERT_EQ(20, gearman_universal_timeout(universal)); // New value of 20 gearman_universal_set_timeout(universal, 10); ASSERT_EQ(10, gearman_universal_timeout(universal)); // New value of 10 gearman_universal_free(universal); return TEST_SUCCESS; } static test_return_t basic_error_test(void *) { gearman_universal_st universal; const char *error= gearman_universal_error(universal); ASSERT_FALSE(error); ASSERT_EQ(0, gearman_universal_errno(universal)); gearman_universal_free(universal); return TEST_SUCCESS; } static test_return_t state_option_test(void *) { gearman_universal_st universal; { // Initial Allocated, no changes ASSERT_FALSE(universal.options.non_blocking); } gearman_universal_free(universal); return TEST_SUCCESS; } static test_return_t state_option_on_create_test(void *) { universal_options_t options[]= { GEARMAN_UNIVERSAL_NON_BLOCKING, GEARMAN_UNIVERSAL_DONT_TRACK_PACKETS, GEARMAN_UNIVERSAL_MAX}; gearman_universal_st universal(options); { // Initial Allocated, no changes ASSERT_TRUE(universal.options.non_blocking); } gearman_universal_free(universal); return TEST_SUCCESS; } static test_return_t gearman_universal_set_namespace_test(void *) { gearman_universal_st universal; ASSERT_FALSE(universal._namespace); gearman_universal_set_namespace(universal, gearman_literal_param("foo23")); ASSERT_TRUE(universal._namespace); gearman_universal_free(universal); return TEST_SUCCESS; } static test_return_t clone_gearman_universal_set_namespace_test(void *) { gearman_universal_st universal; ASSERT_FALSE(universal._namespace); gearman_universal_set_namespace(universal, gearman_literal_param("my_dog")); ASSERT_TRUE(universal._namespace); gearman_universal_st clone; gearman_universal_clone(clone, universal); ASSERT_TRUE(clone._namespace); gearman_universal_free(universal); gearman_universal_free(clone); return TEST_SUCCESS; } static test_return_t state_option_set_test(void *) { universal_options_t options[]= { GEARMAN_UNIVERSAL_NON_BLOCKING, GEARMAN_UNIVERSAL_DONT_TRACK_PACKETS, GEARMAN_UNIVERSAL_MAX}; gearman_universal_st universal(options); { // Initial Allocated, no changes ASSERT_TRUE(universal.options.non_blocking); } ASSERT_TRUE(gearman_universal_is_non_blocking(universal)); { // Initial Allocated, no changes ASSERT_TRUE(universal.options.non_blocking); } gearman_universal_add_options(universal, GEARMAN_UNIVERSAL_DONT_TRACK_PACKETS); { // Initial Allocated, no changes ASSERT_TRUE(universal.options.non_blocking); } gearman_universal_remove_options(universal, GEARMAN_UNIVERSAL_DONT_TRACK_PACKETS); { // Initial Allocated, no changes ASSERT_TRUE(universal.options.non_blocking); } gearman_universal_free(universal); return TEST_SUCCESS; } test_st universal_st_test[] ={ {"init", 0, init_test }, {"clone_test", 0, clone_test }, {"set_timeout", 0, set_timout_test }, {"basic_error", 0, basic_error_test }, {"state_options", 0, state_option_test }, {"state_options_on_create", 0, state_option_on_create_test}, {"state_options_set", 0, state_option_set_test }, {"gearman_universal_set_namespace()", 0, gearman_universal_set_namespace_test }, {"gearman_universal_clone() with gearman_universal_set_namespace()", 0, clone_gearman_universal_set_namespace_test }, {0, 0, 0} }; static test_return_t connection_init_test(void *) { gearman_universal_st universal; gearman_connection_st *connection_ptr= gearman_connection_create(universal, NULL, (const char*)(GEARMAN_DEFAULT_TCP_PORT_STRING)); ASSERT_TRUE(connection_ptr); ASSERT_FALSE(connection_ptr->options.ready); ASSERT_FALSE(connection_ptr->options.packet_in_use); delete connection_ptr; return TEST_SUCCESS; } static test_return_t connection_alloc_test(void *) { gearman_universal_st universal; gearman_connection_st *connection_ptr= gearman_connection_create(universal, NULL, (const char*)(GEARMAN_DEFAULT_TCP_PORT_STRING)); ASSERT_TRUE(connection_ptr); ASSERT_FALSE(connection_ptr->options.ready); ASSERT_FALSE(connection_ptr->options.packet_in_use); delete connection_ptr; return TEST_SUCCESS; } test_st connection_st_test[] ={ {"init", 0, connection_init_test }, {"alloc", 0, connection_alloc_test }, {0, 0, 0} }; static test_return_t packet_init_test(void *) { gearman_universal_st universal; gearman_packet_st packet; gearman_packet_st *packet_ptr; packet_ptr= gearman_packet_create(universal, packet); ASSERT_FALSE(packet.options.is_allocated); ASSERT_FALSE(packet_ptr->options.is_allocated); ASSERT_FALSE(packet.options.complete); ASSERT_FALSE(packet.options.free_data); ASSERT_EQ(packet_ptr, &packet); gearman_packet_free(packet_ptr); ASSERT_FALSE(packet.options.is_allocated); return TEST_SUCCESS; } static test_return_t gearman_packet_give_data_test(void *) { // @note Since this is a give data, ignore any errors that believe there is // an implicit memory leak. size_t data_size= test_literal_param_size("Mine!"); char *data= (char *)calloc(data_size +1, sizeof(char)); ASSERT_TRUE(data); memcpy(data, "Mine!", data_size); gearman_universal_st universal; gearman_packet_st packet; ASSERT_TRUE(gearman_packet_create(universal, packet)); gearman_packet_give_data(packet, data, data_size); ASSERT_EQ(packet.data, data); ASSERT_EQ(packet.data_size, data_size); ASSERT_TRUE(packet.options.free_data); gearman_packet_free(&packet); gearman_universal_free(universal); return TEST_SUCCESS; } static test_return_t gearman_packet_take_data_test(void *) { // Since this is a take data, ignore any errors that believe there is an // implicit memory leak. size_t data_size= test_literal_param_size("Mine!"); char *data= (char *)calloc(data_size +1, sizeof(char)); ASSERT_TRUE(data); memcpy(data, "Mine!", data_size); gearman_universal_st universal; gearman_packet_st packet; gearman_packet_st *packet_ptr= gearman_packet_create(universal, packet); ASSERT_TRUE(packet_ptr); gearman_packet_give_data(packet, data, data_size); ASSERT_EQ(packet_ptr->data, data); ASSERT_EQ(data_size, packet_ptr->data_size); ASSERT_TRUE(packet_ptr->options.free_data); size_t mine_size; char *mine= (char *)gearman_packet_take_data(packet, &mine_size); ASSERT_FALSE(packet_ptr->data); test_zero(packet_ptr->data_size); ASSERT_FALSE(packet_ptr->options.free_data); test_strcmp(mine, "Mine!"); ASSERT_EQ(data_size, mine_size); gearman_packet_free(packet_ptr); gearman_universal_free(universal); free(mine); return TEST_SUCCESS; } test_st packet_st_test[] ={ {"init", 0, packet_init_test }, {"gearman_packet_give_data", 0, gearman_packet_give_data_test }, {"gearman_packet_take_data", 0, gearman_packet_take_data_test }, {0, 0, 0} }; test_st regression_tests[] ={ {"lp:783141, multiple calls for bad host", 0, regression_bug_783141_test }, {"lp:372074", 0, regression_bug_372074_test }, {0, 0, 0} }; collection_st collection[] ={ {"gearman_universal_st", 0, 0, universal_st_test}, {"gearman_connection_st", 0, 0, connection_st_test}, {"gearman_packet_st", 0, 0, packet_st_test}, {"regression", 0, 0, regression_tests}, {0, 0, 0, 0} }; static void *world_create(server_startup_st& servers, test_return_t& error) { /** We start up everything before we allocate so that we don't have to track memory in the forked process. */ if (server_startup(servers, "gearmand", libtest::default_port(), NULL) == false) { error= TEST_SKIPPED; return NULL; } return NULL; } void get_world(libtest::Framework *world) { world->collections(collection); world->create(world_create); }