#include <stdlib.h> #include <cgreen/cgreen.h> #include <freeradius/libradius.h> #include "radsec/radsec.h" #include "radsec/request.h" #include "udp.h" #define true 1 /* FIXME: Bug report cgreen. */ #define false 0 static void authenticate (struct rs_connection *conn, const char *user, const char *pw) { struct rs_request *req; struct rs_packet *msg, *resp; assert_true (rs_request_create (conn, &req) == 0); assert_true (rs_packet_create_authn_request (conn, &msg, user, pw) == 0); rs_request_add_reqpkt (req, msg); assert_true (rs_request_send (req, &resp) == 0); //printf ("%s\n", rs_err_msg (rs_err_conn_pop (conn), 1)); assert_true (rs_packet_frpkt (resp)->code == PW_AUTHENTICATION_ACK); rs_request_destroy (req); } static void send_more_than_one_msg_in_one_packet (struct rs_connection *conn) { struct rs_packet *msg0, *msg1; assert_true (rs_packet_create_authn_request (conn, &msg0, NULL, NULL) == 0); assert_true (rs_packet_create_authn_request (conn, &msg1, NULL, NULL) == 0); assert_true (rs_packet_send (msg0, NULL) == 0); assert_true (rs_packet_send (msg1, NULL) == 0); } static void send_large_packet (struct rs_connection *conn) { struct rs_packet *msg0; struct radius_packet *frpkt = NULL; char *buf; int f; buf = malloc (4096); assert_true (buf != NULL); memset (buf, 0, 4096); assert_true (rs_packet_create (conn, &msg0) == 0); frpkt = rs_packet_frpkt (msg0); assert_true (frpkt != NULL); /* 16 chunks --> heap corruption in evbuffer_drain detected by free() */ for (f = 0; f < 15; f++) { VALUE_PAIR *vp = NULL; memset (buf, 'a' + f, 252); vp = pairmake ("EAP-Message", buf, T_OP_EQ); assert_true (vp != NULL); pairadd (&frpkt->vps, vp); } assert_true (rs_packet_send (msg0, NULL) == 0); } /* ************************************************************ */ static struct setup { char *config_file; char *config_name; char *username; char *pw; } setup; static void test_auth () { struct rs_context *ctx; struct rs_connection *conn; setup.config_file = "test.conf"; setup.config_name = "test-udp-auth"; setup.username = "molgan@PROJECT-MOONSHOT.ORG"; setup.pw = "password"; assert_true (rs_context_create (&ctx) == 0); assert_true (rs_context_read_config (ctx, setup.config_file) == 0); assert_true (rs_context_init_freeradius_dict (ctx, NULL) == 0); assert_true (rs_conn_create (ctx, &conn, setup.config_name) == 0); authenticate (conn, setup.username, setup.pw); rs_conn_destroy (conn); rs_context_destroy (ctx); } static ssize_t test_buffering_cb (const uint8_t *buf, ssize_t len) { /* "Exactly one RADIUS packet is encapsulated in the UDP Data field" [RFC 2865]*/ #if 0 hd (buf, len); #endif assert_true (len >= 20); assert_true (len <= 4096); assert_true ((buf[2] << 8) + buf[3] == len); return len; } static void test_buffering () { struct rs_context *ctx; struct rs_connection *conn; struct timeval timeout; struct polldata *polldata; assert_true (rs_context_create (&ctx) == 0); assert_true (rs_context_read_config (ctx, "test.conf") == 0); assert_true (rs_context_init_freeradius_dict (ctx, NULL) == 0); assert_true (rs_conn_create (ctx, &conn, "test-udp-buffering") == 0); timeout.tv_sec = 0; timeout.tv_usec = 150000; polldata = udp_server ("11820", &timeout, test_buffering_cb); assert_true (polldata != NULL); send_more_than_one_msg_in_one_packet (conn); assert_true (udp_poll (polldata) > 0); assert_true (udp_poll (polldata) > 0); #if 0 " send_large_packet() disabled, it's hanging after Sending Access-Request of id 1 to (null) port 0 Message-Authenticator = 0x00000000000000000000000000000000 packet_do_send: about to send this to localhost:11820: Code: 1, Identifier: 1, Lenght: 38 rs_packet_send: entering event loop _evcb: fd=5 what = WRITE rs_packet_send: event loop done " send_large_packet (conn); assert_true (udp_poll (polldata) > 0); #endif /* 0 */ udp_free_polldata (polldata); rs_conn_destroy (conn); rs_context_destroy (ctx); } /* ************************************************************ */ static void setup_auth (TestSuite *ts) { add_test (ts, test_auth); } static void setup_buffering (TestSuite *ts) { add_test (ts, test_buffering); } int main (int argc, char *argv[]) { TestSuite *ts = create_test_suite (); setup_auth (ts); setup_buffering (ts); if (argc > 1) return run_single_test (ts, argv[1], create_text_reporter ()); else return run_test_suite (ts, create_text_reporter ()); }