1 /* $NetBSD: test-mini_inetd.c,v 1.1.1.2 2014/04/24 12:45:52 pettai Exp $ */
3 /***********************************************************************
4 * Copyright (c) 2009, Secure Endpoints Inc.
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
11 * - Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
14 * - Redistributions in binary form must reproduce the above copyright
15 * notice, this list of conditions and the following disclaimer in
16 * the documentation and/or other materials provided with the
19 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
22 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
23 * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
24 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
25 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
26 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
28 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
29 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
30 * OF THE POSSIBILITY OF SUCH DAMAGE.
32 **********************************************************************/
35 #include <krb5/roken.h>
43 char * prog
= "Master";
47 get_address(int flags
, struct addrinfo
** ret
)
52 memset(&ai
, 0, sizeof(ai
));
54 ai
.ai_flags
= flags
| AI_NUMERICHOST
;
55 ai
.ai_family
= AF_INET
;
56 ai
.ai_socktype
= SOCK_STREAM
;
57 ai
.ai_protocol
= PF_UNSPEC
;
59 rv
= getaddrinfo("127.0.0.1", PORT_S
, &ai
, ret
);
61 warnx("getaddrinfo: %s", gai_strerror(rv
));
66 get_connected_socket(rk_socket_t
* s_ret
)
68 struct addrinfo
* ai
= NULL
;
70 rk_socket_t s
= rk_INVALID_SOCKET
;
72 rv
= get_address(0, &ai
);
76 s
= socket(ai
->ai_family
, ai
->ai_socktype
, ai
->ai_protocol
);
77 if (rk_IS_BAD_SOCKET(s
)) {
82 rv
= connect(s
, ai
->ai_addr
, ai
->ai_addrlen
);
83 if (rk_IS_SOCKET_ERROR(rv
))
87 s
= rk_INVALID_SOCKET
;
91 if (!rk_IS_BAD_SOCKET(s
))
97 return (rv
) ? rk_SOCK_ERRNO
: 0;
100 const char * test_strings
[] = {
102 "01234566789012345689012345678901234567890123456789",
108 test_simple_echo_client(void)
110 rk_socket_t s
= rk_INVALID_SOCKET
;
115 fprintf(stderr
, "[%s] Getting connected socket...", getprogname());
116 rv
= get_connected_socket(&s
);
118 fprintf(stderr
, "\n[%s] get_connected_socket() failed (%s)\n",
119 getprogname(), strerror(rk_SOCK_ERRNO
));
123 fprintf(stderr
, "[%s] done\n", getprogname());
125 for (i
=0; i
< sizeof(test_strings
)/sizeof(test_strings
[0]); i
++) {
126 rv
= send(s
, test_strings
[i
], strlen(test_strings
[i
]), 0);
127 if (rk_IS_SOCKET_ERROR(rv
)) {
128 fprintf(stderr
, "[%s] send() failure (%s)\n",
129 getprogname(), strerror(rk_SOCK_ERRNO
));
134 rv
= recv(s
, buf
, sizeof(buf
), 0);
135 if (rk_IS_SOCKET_ERROR(rv
)) {
136 fprintf (stderr
, "[%s] recv() failure (%s)\n",
137 getprogname(), strerror(rk_SOCK_ERRNO
));
143 fprintf (stderr
, "[%s] No data received\n", prog
);
148 if (rv
!= strlen(test_strings
[i
])) {
149 fprintf (stderr
, "[%s] Data length mismatch %d != %d\n", prog
, rv
, strlen(test_strings
[i
]));
155 fprintf (stderr
, "[%s] Done\n", prog
);
161 test_simple_echo_socket(void)
163 fprintf (stderr
, "[%s] Process ID %d\n", prog
, GetCurrentProcessId());
164 fprintf (stderr
, "[%s] Starting echo test with sockets\n", prog
);
167 return test_simple_echo_client();
170 rk_socket_t s
= rk_INVALID_SOCKET
;
172 fprintf (stderr
, "[%s] Listening for connections...\n", prog
);
173 mini_inetd(htons(PORT
), &s
);
174 if (rk_IS_BAD_SOCKET(s
)) {
175 fprintf (stderr
, "[%s] Connect failed (%s)\n",
176 getprogname(), strerror(rk_SOCK_ERRNO
));
178 fprintf (stderr
, "[%s] Connected\n", prog
);
185 while ((rv
= recv(s
, buf
, sizeof(buf
), 0)) != 0 && !rk_IS_SOCKET_ERROR(rv
)) {
187 fprintf(stderr
, "[%s] Received [%s]\n", prog
, buf
);
190 srv
= send(s
, buf
, rv
, 0);
192 if (rk_IS_SOCKET_ERROR(srv
))
193 fprintf(stderr
, "[%s] send() error [%s]\n",
194 getprogname(), strerror(rk_SOCK_ERRNO
));
196 fprintf(stderr
, "[%s] send() size mismatch %d != %d",
197 getprogname(), srv
, rv
);
200 if (!strcmp(buf
, "exit")) {
201 fprintf(stderr
, "[%s] Exiting...\n", prog
);
202 shutdown(s
, SD_SEND
);
208 fprintf(stderr
, "[%s] recv() failed (%s)\n",
210 strerror(rk_SOCK_ERRNO
));
220 test_simple_echo(void)
222 fprintf (stderr
, "[%s] Starting echo test\n", prog
);
226 return test_simple_echo_client();
230 fprintf (stderr
, "[%s] Listening for connections...\n", prog
);
231 mini_inetd(htons(PORT
), NULL
);
232 fprintf (stderr
, "[%s] Connected\n", prog
);
237 fprintf(stderr
, "[%s] Received [%s]\n", prog
, buf
);
239 if (!strcmp(buf
, "exit"))
246 fprintf(stderr
, "[%s] gets() failed (%s)\n", prog
, _strerror("gets"));
263 fprintf(stderr
, "Starting client...\n");
265 rv
= test_simple_echo_socket();
281 fprintf(stderr
, "Starting server...\n");
283 rv
= test_simple_echo_socket();
291 wait_callback(void *p
)
304 p_server
= _spawnl(_P_NOWAIT
, path
, path
, "--server", NULL
);
306 fprintf(stderr
, "%s: %s", path
, _strerror("Can't start server process"));
310 /* On Windows, the _spawn*() functions return a process handle on
311 success. We need a process ID for use with
312 wait_for_process_timed(). */
314 p_server
= GetProcessId((HANDLE
) p_server
);
316 fprintf(stderr
, "Created server process ID %d\n", p_server
);
318 p_client
= _spawnl(_P_NOWAIT
, path
, path
, "--client", NULL
);
320 fprintf(stderr
, "%s: %s", path
, _strerror("Can't start client process"));
321 fprintf(stderr
, "Waiting for server process to terminate ...");
322 wait_for_process_timed(p_server
, wait_callback
, NULL
, 5);
323 fprintf(stderr
, "DONE\n");
327 p_client
= GetProcessId((HANDLE
) p_client
);
329 fprintf(stderr
, "Created client process ID %d\n", p_client
);
331 fprintf(stderr
, "Waiting for client process to terminate ...");
332 client_rv
= wait_for_process_timed(p_client
, wait_callback
, NULL
, 5);
333 if (SE_IS_ERROR(client_rv
)) {
334 fprintf(stderr
, "\nwait_for_process_timed() failed for client. rv=%d\n", client_rv
);
336 fprintf(stderr
, "DONE\n");
339 fprintf(stderr
, "Waiting for server process to terminate ...");
340 server_rv
= wait_for_process_timed(p_server
, wait_callback
, NULL
, 5);
341 if (SE_IS_ERROR(server_rv
)) {
342 fprintf(stderr
, "\nwait_for_process_timed() failed for server. rv=%d\n", server_rv
);
344 fprintf(stderr
, "DONE\n");
347 if (client_rv
== 0 && server_rv
== 0) {
348 fprintf(stderr
, "PASS\n");
351 fprintf(stderr
, "FAIL: Client rv=%d, Server rv=%d\n", client_rv
, server_rv
);
356 int main(int argc
, char ** argv
)
358 setprogname(argv
[0]);
360 if (argc
== 2 && strcmp(argv
[1], "--client") == 0)
362 else if (argc
== 2 && strcmp(argv
[1], "--server") == 0)
365 return do_test(argv
[0]);
367 printf ("%s: Test mini_inetd() function. Run with no arguments to start test\n",