| /********************************************************************************************************* |
| * Software License Agreement (BSD License) * |
| * Author: Sebastien Decugis <sdecugis@freediameter.net> * |
| * * |
| * Copyright (c) 2013, WIDE Project and NICT * |
| * All rights reserved. * |
| * * |
| * Redistribution and use of this software 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. * |
| * * |
| * * Neither the name of the WIDE Project or NICT nor the * |
| * names of its contributors may be used to endorse or * |
| * promote products derived from this software without * |
| * specific prior written permission of WIDE Project and * |
| * NICT. * |
| * * |
| * 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 "tests.h" |
| |
| #define Define_cb( __int, __extra ) \ |
| int cb_##__int( struct msg ** msg, struct avp * avp, struct session * session, void * opaque, enum disp_action * action ) \ |
| { \ |
| CHECK( 1, msg ? 1 : 0 ); \ |
| CHECK( 1, action ? 1 : 0 ); \ |
| CHECK( sess, session ); \ |
| if (opaque) { \ |
| CHECK( 1, opaque == g_opaque ? 1 : 0 ); \ |
| } \ |
| *action = DISP_ACT_CONT; \ |
| cbcalled[__int] += 1; \ |
| do { \ |
| __extra ; \ |
| } while (0); \ |
| return 0; \ |
| } |
| |
| #define NB_CB 10 |
| char cbcalled[NB_CB]; |
| struct session * sess; |
| void * g_opaque = (void *)"test"; |
| |
| /* cb_0 */ Define_cb( 0, ); |
| /* cb_1 */ Define_cb( 1, ); |
| /* cb_2 */ Define_cb( 2, ); |
| /* cb_3 */ Define_cb( 3, ); |
| /* cb_4 */ Define_cb( 4, ); |
| /* cb_5 */ Define_cb( 5, ); |
| /* cb_6 */ Define_cb( 6, return 12345 ); |
| /* cb_7 */ Define_cb( 7, { CHECK( 1, avp ? 1 : 0 ); } ); |
| /* cb_8 */ Define_cb( 8, { CHECK( 0, fd_msg_free( *msg ) ); *msg = NULL; } ); |
| /* cb_9 */ Define_cb( 9, *action = DISP_ACT_SEND ); |
| /* max: cb_<NB_CB - 1> */ |
| |
| /* Create a new message containing what we want */ |
| struct msg * new_msg(int appid, struct dict_object * cmd, struct dict_object * avp1, struct dict_object * avp2, int val) |
| { |
| struct msg *new; |
| struct avp *avp; |
| union avp_value value; |
| struct msg_hdr * msg_hdr = NULL; |
| |
| CHECK( 0, fd_msg_new ( cmd, 0, &new ) ); |
| CHECK( 0, fd_msg_hdr ( new, &msg_hdr ) ); |
| msg_hdr->msg_appl = appid; |
| |
| if (avp1) { |
| CHECK( 0, fd_msg_avp_new ( avp1, 0, &avp ) ); |
| value.u32 = 0; |
| CHECK( 0, fd_msg_avp_setvalue ( avp, &value ) ); |
| CHECK( 0, fd_msg_avp_add ( new, MSG_BRW_LAST_CHILD, avp ) ); |
| } |
| |
| if (avp2) { |
| CHECK( 0, fd_msg_avp_new ( avp2, 0, &avp ) ); |
| value.u32 = val; |
| CHECK( 0, fd_msg_avp_setvalue ( avp, &value ) ); |
| CHECK( 0, fd_msg_avp_add ( new, MSG_BRW_LAST_CHILD, avp ) ); |
| } |
| |
| return new; |
| } |
| |
| /* Main test routine */ |
| int main(int argc, char *argv[]) |
| { |
| struct dict_object * app1, * app2; |
| struct dict_object * cmd1, * cmd2; |
| struct dict_object * avp1, * avp2; /* avp2 is enumerated; they are both unsigned32 types */ |
| struct dict_object * enu1, * enu2; |
| struct msg * msg = NULL, *error; |
| enum disp_action action; |
| struct disp_hdl * hdl[NB_CB]; |
| struct disp_when when; |
| char * ec, *em; |
| |
| /* First, initialize the daemon modules */ |
| INIT_FD(); |
| |
| /* Create a dummy session, we don't use it anyway */ |
| #define DUMMY_SID "test.disp" |
| CHECK( 0, fd_sess_new( &sess, DUMMY_SID, CONSTSTRLEN(DUMMY_SID), NULL, 0 ) ); |
| |
| memset(&when, 0xff, sizeof(when)); /* check that we don't use un-initialized parts */ |
| |
| /* Initialize dictionary objects */ |
| { |
| struct dict_object * enutype; |
| struct dict_application_data app1_data = { 1, "Application test 1" }; |
| struct dict_application_data app2_data = { 2, "Application test 2" }; |
| struct dict_cmd_data cmd1_data = { 1, "Command test 1 (req)", CMD_FLAG_REQUEST, CMD_FLAG_REQUEST }; |
| struct dict_cmd_data cmd2_data = { 1, "Command test 2 (ans)", CMD_FLAG_REQUEST, 0 }; |
| struct dict_type_data type_data = { AVP_TYPE_UNSIGNED32, "Type test", NULL, NULL }; |
| struct dict_avp_data avp1_data = { 10001, 0, "AVP test 1", 0, 0, AVP_TYPE_UNSIGNED32 }; |
| struct dict_avp_data avp2_data = { 10002, 0, "AVP test 2", 0, 0, AVP_TYPE_UNSIGNED32 }; |
| struct dict_enumval_data enu1_data = { "ENU test 1", { .u32 = 1 }}; |
| struct dict_enumval_data enu2_data = { "ENU test 2", { .u32 = 2 }}; |
| |
| CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_APPLICATION, &app1_data, NULL, &app1 ) ); |
| CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_APPLICATION, &app2_data, NULL, &app2 ) ); |
| CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_COMMAND, &cmd1_data, NULL, &cmd1 ) ); |
| CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_COMMAND, &cmd2_data, NULL, &cmd2 ) ); |
| CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_TYPE, &type_data, NULL, &enutype ) ); |
| CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_AVP, &avp1_data, NULL, &avp1 ) ); |
| CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_AVP, &avp2_data, enutype, &avp2 ) ); |
| CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_ENUMVAL, &enu1_data, enutype, &enu1 ) ); |
| CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_ENUMVAL, &enu2_data, enutype, &enu2 ) ); |
| } |
| |
| /* Register first handler, very simple test */ |
| { |
| CHECK( 0, fd_disp_register( cb_0, DISP_HOW_ANY, NULL, NULL, &hdl[0] ) ); |
| CHECK( 1, hdl[0] ? 1 : 0 ); |
| CHECK( 0, fd_disp_unregister( &hdl[0], NULL ) ); |
| CHECK( NULL, hdl[0] ); |
| CHECK( 0, fd_disp_register( cb_0, DISP_HOW_ANY, NULL, NULL, &hdl[0] ) ); |
| |
| /* Check this handler is called for a message */ |
| msg = new_msg( 0, cmd1, avp1, NULL, 0 ); |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( DISP_ACT_CONT, action ); |
| |
| /* Delete the message */ |
| CHECK( 0, fd_msg_free( msg ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[0], NULL ) ); |
| } |
| |
| /* Handlers for applications */ |
| { |
| CHECK( 0, fd_disp_register( cb_0, DISP_HOW_ANY, &when, NULL, &hdl[0] ) ); |
| when.app = app1; |
| CHECK( 0, fd_disp_register( cb_1, DISP_HOW_APPID, &when, NULL, &hdl[1] ) ); |
| when.app = app2; |
| CHECK( 0, fd_disp_register( cb_2, DISP_HOW_APPID, &when, NULL, &hdl[2] ) ); |
| when.avp = avp2; |
| CHECK( 0, fd_disp_register( cb_3, DISP_HOW_APPID, &when, NULL, &hdl[3] ) ); |
| when.avp = avp1; |
| CHECK( 0, fd_disp_register( cb_4, DISP_HOW_APPID, &when, NULL, &hdl[4] ) ); |
| |
| /* Check the callbacks are called as appropriate */ |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 0, cmd1, avp1, NULL, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 0, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( 0, cbcalled[4] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd1, avp1, NULL, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( 0, cbcalled[4] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 2, cmd1, avp1, NULL, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 0, cbcalled[1] ); |
| CHECK( 1, cbcalled[2] ); |
| CHECK( 1, cbcalled[3] ); |
| CHECK( 1, cbcalled[4] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| CHECK( 0, fd_disp_unregister( &hdl[0], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[1], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[2], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[3], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[4], NULL ) ); |
| } |
| |
| /* Handlers for commands */ |
| { |
| when.app = NULL; |
| when.command = NULL; |
| CHECK( 0, fd_disp_register( cb_0, DISP_HOW_ANY, &when, NULL, &hdl[0] ) ); |
| CHECK( EINVAL, fd_disp_register( cb_1, DISP_HOW_CC, &when, NULL, &hdl[1] ) ); |
| when.command = cmd1; |
| CHECK( 0, fd_disp_register( cb_1, DISP_HOW_CC, &when, NULL, &hdl[1] ) ); /* cmd1 */ |
| when.app = app2; |
| CHECK( 0, fd_disp_register( cb_2, DISP_HOW_CC, &when, NULL, &hdl[2] ) ); /* app2 + cmd1 */ |
| when.command = cmd2; |
| when.app = NULL; |
| when.avp = avp1; |
| CHECK( 0, fd_disp_register( cb_3, DISP_HOW_CC, &when, NULL, &hdl[3] ) ); /* cmd2 (avp1 ignored) */ |
| |
| /* Check the callbacks are called as appropriate */ |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 0, cmd1, avp1, NULL, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 2, cmd1, avp1, NULL, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 2, cmd2, avp1, NULL, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 0, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 1, cbcalled[3] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd2, NULL, avp2, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 0, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 1, cbcalled[3] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| CHECK( 0, fd_disp_unregister( &hdl[0], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[1], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[2], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[3], NULL ) ); |
| } |
| |
| /* Handlers for AVPs */ |
| { |
| when.app = NULL; |
| when.command = NULL; |
| when.avp = NULL; |
| |
| CHECK( 0, fd_disp_register( cb_0, DISP_HOW_ANY, &when, NULL, &hdl[0] ) ); /* all */ |
| CHECK( EINVAL, fd_disp_register( cb_1, DISP_HOW_AVP, &when, NULL, &hdl[1] ) ); |
| |
| when.avp = avp1; |
| CHECK( 0, fd_disp_register( cb_1, DISP_HOW_AVP, &when, NULL, &hdl[1] ) ); /* avp1 */ |
| |
| when.command = cmd1; |
| CHECK( 0, fd_disp_register( cb_2, DISP_HOW_AVP, &when, NULL, &hdl[2] ) ); /* avp1 + cmd1 */ |
| |
| when.command = NULL; |
| when.app = app1; |
| CHECK( 0, fd_disp_register( cb_3, DISP_HOW_AVP, &when, NULL, &hdl[3] ) ); /* avp1 + app1 */ |
| |
| when.command = cmd1; |
| CHECK( 0, fd_disp_register( cb_4, DISP_HOW_AVP, &when, NULL, &hdl[4] ) ); /* avp1 + cmd1 + app1 */ |
| |
| when.app = NULL; |
| when.command = NULL; |
| when.avp = avp2; |
| when.value = enu1; |
| CHECK( 0, fd_disp_register( cb_5, DISP_HOW_AVP, &when, NULL, &hdl[5] ) ); /* avp2 */ |
| |
| when.value = enu2; |
| CHECK( 0, fd_disp_register( cb_7, DISP_HOW_AVP, &when, NULL, &hdl[6] ) ); /* avp2 */ |
| |
| |
| |
| /* Check the callbacks are called as appropriate */ |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 0, cmd1, NULL, NULL, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 0, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( 0, cbcalled[4] ); |
| CHECK( 0, cbcalled[5] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 0, cmd1, avp1, NULL, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( 0, cbcalled[4] ); |
| CHECK( 0, cbcalled[5] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd2, avp1, NULL, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 1, cbcalled[3] ); |
| CHECK( 0, cbcalled[4] ); |
| CHECK( 0, cbcalled[5] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd1, avp1, NULL, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[2] ); |
| CHECK( 1, cbcalled[3] ); |
| CHECK( 1, cbcalled[4] ); |
| CHECK( 0, cbcalled[5] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd1, avp1, avp2, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[2] ); |
| CHECK( 1, cbcalled[3] ); |
| CHECK( 1, cbcalled[4] ); |
| CHECK( 1, cbcalled[5] ); |
| CHECK( 1, cbcalled[7] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd1, NULL, avp2, 1 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 0, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( 0, cbcalled[4] ); |
| CHECK( 1, cbcalled[5] ); |
| CHECK( 1, cbcalled[7] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd1, NULL, avp2, 2 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 0, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( 0, cbcalled[4] ); |
| CHECK( 1, cbcalled[5] ); |
| CHECK( 1, cbcalled[7] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| CHECK( 0, fd_disp_unregister( &hdl[0], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[1], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[2], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[3], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[4], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[5], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[6], NULL ) ); |
| } |
| |
| /* Handlers for enum values */ |
| { |
| when.app = NULL; |
| when.command = NULL; |
| when.avp = NULL; |
| when.value = NULL; |
| |
| CHECK( 0, fd_disp_register( cb_0, DISP_HOW_ANY, &when, NULL, &hdl[0] ) ); /* all */ |
| CHECK( EINVAL, fd_disp_register( cb_1, DISP_HOW_AVP_ENUMVAL, &when, NULL, &hdl[1] ) ); |
| when.value = enu1; |
| CHECK( EINVAL, fd_disp_register( cb_1, DISP_HOW_AVP_ENUMVAL, &when, NULL, &hdl[1] ) ); |
| when.avp = avp1; |
| CHECK( EINVAL, fd_disp_register( cb_1, DISP_HOW_AVP_ENUMVAL, &when, NULL, &hdl[1] ) ); |
| when.avp = avp2; |
| CHECK( 0, fd_disp_register( cb_1, DISP_HOW_AVP_ENUMVAL, &when, NULL, &hdl[1] ) ); /* avp2, enu1 */ |
| |
| when.command = cmd1; |
| CHECK( 0, fd_disp_register( cb_2, DISP_HOW_AVP_ENUMVAL, &when, NULL, &hdl[2] ) ); /* avp2, enu1 + cmd1 */ |
| |
| when.command = NULL; |
| when.app = app1; |
| when.value = enu2; |
| CHECK( 0, fd_disp_register( cb_3, DISP_HOW_AVP_ENUMVAL, &when, NULL, &hdl[3] ) ); /* avp2, enu2 + app1 */ |
| |
| /* Check the callbacks are called as appropriate */ |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 0, cmd1, avp1, NULL, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 0, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd2, avp1, avp2, 0 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 0, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd2, avp1, avp2, 1 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd2, avp1, avp2, 2 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 0, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 1, cbcalled[3] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd1, avp1, avp2, 1 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd2, avp1, avp2, 1 ); |
| { |
| struct avp *avp; |
| union avp_value value; |
| CHECK( 0, fd_msg_avp_new ( avp2, 0, &avp ) ); |
| value.u32 = 2; |
| CHECK( 0, fd_msg_avp_setvalue ( avp, &value ) ); |
| CHECK( 0, fd_msg_avp_add ( msg, MSG_BRW_LAST_CHILD, avp ) ); |
| } |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 1, cbcalled[3] ); |
| CHECK( DISP_ACT_CONT, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| CHECK( 0, fd_disp_unregister( &hdl[0], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[1], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[2], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[3], NULL ) ); |
| } |
| |
| /* Test behavior of handlers */ |
| { |
| CHECK( 0, fd_disp_register( cb_0, DISP_HOW_ANY, &when, NULL, &hdl[0] ) ); |
| CHECK( 0, fd_disp_register( cb_1, DISP_HOW_ANY, &when, NULL, &hdl[1] ) ); |
| CHECK( 0, fd_disp_register( cb_6, DISP_HOW_ANY, &when, NULL, &hdl[2] ) ); |
| CHECK( 0, fd_disp_register( cb_2, DISP_HOW_ANY, &when, NULL, &hdl[3] ) ); |
| CHECK( 0, fd_disp_register( cb_3, DISP_HOW_ANY, &when, NULL, &hdl[4] ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd1, avp1, avp2, 1 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[6] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( 0, msg ? 1 : 0); |
| CHECK( 1, em ? 1 : 0); |
| CHECK( 0, fd_msg_free( error ) ); |
| |
| CHECK( 0, fd_disp_unregister( &hdl[0], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[1], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[2], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[3], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[4], NULL ) ); |
| |
| CHECK( 0, fd_disp_register( cb_0, DISP_HOW_ANY, &when, NULL, &hdl[0] ) ); |
| CHECK( 0, fd_disp_register( cb_1, DISP_HOW_ANY, &when, NULL, &hdl[1] ) ); |
| CHECK( 0, fd_disp_register( cb_8, DISP_HOW_ANY, &when, NULL, &hdl[2] ) ); |
| CHECK( 0, fd_disp_register( cb_2, DISP_HOW_ANY, &when, NULL, &hdl[3] ) ); |
| CHECK( 0, fd_disp_register( cb_3, DISP_HOW_ANY, &when, NULL, &hdl[4] ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd1, avp1, avp2, 1 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[8] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( NULL, msg ); |
| CHECK( NULL, em ); |
| |
| CHECK( 0, fd_disp_unregister( &hdl[0], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[1], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[2], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[3], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[4], NULL ) ); |
| |
| CHECK( 0, fd_disp_register( cb_0, DISP_HOW_ANY, &when, NULL, &hdl[0] ) ); |
| CHECK( 0, fd_disp_register( cb_1, DISP_HOW_ANY, &when, NULL, &hdl[1] ) ); |
| CHECK( 0, fd_disp_register( cb_9, DISP_HOW_ANY, &when, NULL, &hdl[2] ) ); |
| CHECK( 0, fd_disp_register( cb_2, DISP_HOW_ANY, &when, NULL, &hdl[3] ) ); |
| CHECK( 0, fd_disp_register( cb_3, DISP_HOW_ANY, &when, NULL, &hdl[4] ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 1, cmd1, avp1, avp2, 1 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[9] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( DISP_ACT_SEND, action ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| CHECK( 0, fd_disp_unregister( &hdl[0], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[1], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[2], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[3], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[4], NULL ) ); |
| } |
| |
| /* Test order of handlers */ |
| { |
| when.app = app2; |
| when.command = cmd2; |
| when.avp = avp2; |
| when.value = enu2; |
| |
| CHECK( 0, fd_disp_register( cb_0, DISP_HOW_ANY, &when, NULL, &hdl[0] ) ); |
| CHECK( 0, fd_disp_register( cb_1, DISP_HOW_AVP_ENUMVAL, &when, NULL, &hdl[1] ) ); |
| CHECK( 0, fd_disp_register( cb_2, DISP_HOW_AVP, &when, NULL, &hdl[2] ) ); |
| CHECK( 0, fd_disp_register( cb_3, DISP_HOW_CC, &when, NULL, &hdl[3] ) ); |
| CHECK( 0, fd_disp_register( cb_4, DISP_HOW_APPID, &when, NULL, &hdl[4] ) ); |
| |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 2, cmd2, avp1, avp2, 2 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[2] ); |
| CHECK( 1, cbcalled[3] ); |
| CHECK( 1, cbcalled[4] ); |
| CHECK( 0, cbcalled[9] ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| |
| CHECK( 0, fd_disp_register( cb_9, DISP_HOW_ANY, &when, NULL, &hdl[5] ) ); |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 2, cmd2, avp1, avp2, 2 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 0, cbcalled[1] ); |
| CHECK( 0, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( 0, cbcalled[4] ); |
| CHECK( 1, cbcalled[9] ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[5], NULL ) ); |
| |
| CHECK( 0, fd_disp_register( cb_9, DISP_HOW_AVP_ENUMVAL, &when, NULL, &hdl[5] ) ); |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 2, cmd2, avp1, avp2, 2 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( 0, cbcalled[4] ); |
| CHECK( 1, cbcalled[9] ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[5], NULL ) ); |
| |
| CHECK( 0, fd_disp_register( cb_9, DISP_HOW_AVP, &when, NULL, &hdl[5] ) ); |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 2, cmd2, avp1, avp2, 2 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[2] ); |
| CHECK( 0, cbcalled[3] ); |
| CHECK( 0, cbcalled[4] ); |
| CHECK( 1, cbcalled[9] ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[5], NULL ) ); |
| |
| CHECK( 0, fd_disp_register( cb_9, DISP_HOW_CC, &when, NULL, &hdl[5] ) ); |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 2, cmd2, avp1, avp2, 2 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[2] ); |
| CHECK( 1, cbcalled[3] ); |
| CHECK( 0, cbcalled[4] ); |
| CHECK( 1, cbcalled[9] ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[5], NULL ) ); |
| |
| CHECK( 0, fd_disp_register( cb_9, DISP_HOW_APPID, &when, NULL, &hdl[5] ) ); |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| msg = new_msg( 2, cmd2, avp1, avp2, 2 ); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( 1, cbcalled[1] ); |
| CHECK( 1, cbcalled[2] ); |
| CHECK( 1, cbcalled[3] ); |
| CHECK( 1, cbcalled[4] ); |
| CHECK( 1, cbcalled[9] ); |
| CHECK( 0, fd_msg_free( msg ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[5], NULL ) ); |
| |
| CHECK( 0, fd_disp_unregister( &hdl[0], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[1], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[2], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[3], NULL ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[4], NULL ) ); |
| } |
| |
| /* Test application support advertisement */ |
| { |
| struct dict_object * vnd; |
| struct dict_vendor_data vnd_data = { 1, "Vendor test" }; |
| struct fd_app * app; |
| |
| CHECK( 0, fd_dict_new ( fd_g_config->cnf_dict, DICT_VENDOR, &vnd_data, NULL, &vnd ) ); |
| |
| CHECK( EINVAL, fd_disp_app_support ( vnd, NULL, 1, 0 ) ); |
| CHECK( EINVAL, fd_disp_app_support ( app1, NULL, 0, 0 ) ); |
| CHECK( 0, fd_disp_app_support ( app1, NULL, 1, 0 ) ); |
| CHECK( 0, fd_disp_app_support ( app1, NULL, 0, 1 ) ); |
| CHECK( 0, fd_disp_app_support ( app2, vnd, 1, 0 ) ); |
| |
| app = (struct fd_app *)(fd_g_config->cnf_apps.next); |
| CHECK( 1, app->appid ); |
| CHECK( 1, app->flags.auth ); |
| CHECK( 1, app->flags.acct ); |
| app = (struct fd_app *)(fd_g_config->cnf_apps.prev); |
| CHECK( 2, app->appid ); |
| CHECK( 1, app->flags.auth ); |
| CHECK( 0, app->flags.acct ); |
| |
| #if 0 |
| fd_log_debug("%s", fd_conf_dump(FD_DUMP_TEST_PARAMS)); |
| #endif |
| } |
| |
| /* Test opaque pointer management */ |
| { |
| void * ptr; |
| CHECK( 0, fd_disp_register( cb_0, DISP_HOW_ANY, NULL, g_opaque, &hdl[0] ) ); |
| |
| /* Check this handler is called for a message */ |
| msg = new_msg( 0, cmd1, avp1, NULL, 0 ); |
| memset(cbcalled, 0, sizeof(cbcalled)); |
| CHECK( 0, fd_msg_dispatch ( &msg, sess, &action, &ec, &em, &error ) ); |
| CHECK( 1, cbcalled[0] ); |
| CHECK( DISP_ACT_CONT, action ); |
| |
| /* Delete the message */ |
| CHECK( 0, fd_msg_free( msg ) ); |
| CHECK( 0, fd_disp_unregister( &hdl[0], &ptr ) ); |
| CHECK( 1, ptr == g_opaque ? 1 : 0 ); |
| } |
| |
| /* That's all for the tests yet */ |
| PASSTEST(); |
| } |
| |