fix for windows
[webserver.git] / webserver.c
CommitLineData
8512671a
LM
1/* depend: */
2/* cflags: */
06ec8057 3/* linker: color.o debug.o server.o signal.o */
8512671a 4
184be781 5#include <assert.h>
8512671a
LM
6#include <stdio.h>
7#include <stdlib.h>
89f0e084 8#include <unistd.h>
163f56b1 9#include <signal.h>
8512671a
LM
10
11#include "debug.h"
00afbb65 12#include "server.h"
06ec8057 13#include "signal.h"
8512671a 14
184be781
LM
15/* types */
16
8512671a
LM
17/* constants */
18
19#define BUFFER_SIZE 4096
20
21/* macros */
22
23/* gobal variables */
24
25char *progname = NULL;
21ac031b 26int port = 8080;
06ec8057 27socket_t sock = INVALID_SOCKET;
89f0e084 28socket_t conn = INVALID_SOCKET;
8512671a
LM
29
30/* help function */
31
32int usage (int ret)
33{
34 FILE *fid = ret ? stderr : stdout;
35 fprintf (fid, "usage: %s\n", progname);
36 fprintf (fid, " -h : help message\n");
21ac031b 37 fprintf (fid, " -p : port number (%d)\n", port);
8512671a
LM
38 fprintf (fid, " -v : verbose level (%d)\n", verbose);
39
40 return ret;
41}
42
06ec8057
LM
43void stop_server (__attribute__((unused)) int sig)
44{
89f0e084
LM
45 if (conn != INVALID_SOCKET) {
46 close_socket (conn);
47 }
06ec8057 48 if (sock != INVALID_SOCKET) {
89f0e084 49 close_socket (sock);
06ec8057 50 }
89f0e084
LM
51 terminate_network_context ();
52 exit (0);
06ec8057
LM
53}
54
8512671a
LM
55/* main function */
56
57int main (int argc, char *argv[])
58{
59 int i = 0;
8512671a
LM
60
61 /* program name */
62
63 progname = argv[0];
64 while (progname[i] != '\0') {
65 if ((progname[i] == '/') || (progname[i] == '\\')) {
66 progname += i + 1;
67 i = 0;
68 } else {
69 i++;
70 }
71 }
72
73 /* argument processing */
74
75 while (argc-- > 1) {
76 char *arg = *(++argv);
77 if (arg[0] != '-') {
163f56b1 78 VERBOSE (ERROR, PERROR ("%s: invalid option -- '%s'\n", progname, arg); usage (1));
8512671a
LM
79 return 1;
80 }
81 char c = arg[1];
82 switch (c) {
21ac031b 83 case 'p':
8512671a
LM
84 arg = (arg[2]) ? arg + 2 : (--argc > 0) ? *(++argv) : NULL;
85 if (arg == NULL) {
163f56b1 86 VERBOSE (ERROR, PERROR ("%s: missing port number\n", progname); usage (1));
21ac031b
LM
87 return 1;
88 }
89 port = atoi (arg);
90 if (port <= 0) {
163f56b1 91 VERBOSE (ERROR, PERROR ("%s: incorrect port number (%s)\n", progname, arg); usage (1));
8512671a
LM
92 return 1;
93 }
8512671a 94 break;
8512671a
LM
95 case 'v':
96 arg = (arg[2]) ? arg + 2 : (--argc > 0) ? *(++argv) : NULL;
97 if (arg == NULL) {
163f56b1 98 VERBOSE (ERROR, PERROR ("%s: missing verbose level\n", progname); usage (1));
8512671a
LM
99 return 1;
100 }
101 verbose = atoi (arg);
102 break;
103 case 'h':
104 default:
105 return usage (c != 'h');
106 }
107 }
108
163f56b1 109 VERBOSE (DEBUG, PRINT ("Initializing socket\n"));
89f0e084 110 init_network_context ();
06ec8057
LM
111 sock = open_listening_socket (port);
112 if (sock == INVALID_SOCKET) {
163f56b1 113 VERBOSE (ERROR, PERROR ("Can't open listening socket\n"));
184be781
LM
114 return 1;
115 }
163f56b1
LM
116 VERBOSE (INFO, PRINT ("Listening socket on port %d\n", port));
117
118 signal (SIGTERM, &stop_server);
04a2223d
LM
119
120 /* main loop */
121 while (1) {
89f0e084
LM
122 conn = accept_incoming_connection (sock);
123 if (conn == INVALID_SOCKET) {
124 usleep (1e5);
04a2223d
LM
125 continue;
126 }
127
163f56b1 128 VERBOSE (DEBUG, PRINT ("Server connected, waiting for data\n"));
04a2223d 129
89f0e084 130 unsigned char *data = {0};
04a2223d 131
89f0e084
LM
132 int len = receive_data (conn, &data);
133 if (len == 0) {
163f56b1 134 VERBOSE (WARNING, PRINT ("Connection closed by peer (rx)\n"));
89f0e084 135 } else if (len < 0) {
163f56b1 136 VERBOSE (WARNING, PRINT ("Connection in error (rx)\n"));
89f0e084 137 } else {
04a2223d
LM
138
139 // processing
163f56b1 140 VERBOSE (DEBUG, PRINT ("Received:\n%s\n", data));
89f0e084
LM
141
142 int rc = send_data (conn, data, len);
143 if (rc == 0) {
163f56b1 144 VERBOSE (WARNING, PRINT ("Connection closed by peer (tx)\n"));
89f0e084 145 } else if (rc < 0) {
163f56b1 146 VERBOSE (WARNING, PRINT ("Connection in error (tx)\n"));
04a2223d 147 }
89f0e084 148 }
04a2223d 149
163f56b1 150 VERBOSE (DEBUG, PRINT ("Closing connection\n"));
89f0e084
LM
151 if (data) {
152 free (data);
04a2223d 153 }
89f0e084
LM
154 close_socket (conn);
155 conn = INVALID_SOCKET;
04a2223d 156 }
89f0e084 157
163f56b1 158 VERBOSE (DEBUG, PRINT ("Closing socket\n"));
89f0e084
LM
159 close_socket (sock);
160 terminate_network_context ();
184be781 161
06ec8057 162 return 2;
8512671a
LM
163}
164
165// test: webserver.exe -h
166// test: webserver.exe -h | awk '/usage:/ { rc=1 } END { exit (1-rc) }'
21ac031b 167// test: webserver.exe -_ 2> /dev/null | wc -l | xargs test 0 =
8512671a
LM
168// test: webserver.exe -_ 2>&1 | awk '/usage:/ { rc=1 } END { exit (1-rc) }'
169// test: webserver.exe error 2>&1 | grep -q 'invalid option'
21ac031b
LM
170// test: webserver.exe -v 2>&1 | grep -q 'missing verbose level'
171// test: webserver.exe -p 2>&1 | grep -q 'missing port number'
172// test: webserver.exe -p -1 2>&1 | grep -q 'incorrect port number'
163f56b1
LM
173// test: webserver.exe > test.log & pid=$!; sleep 1; kill -9 $pid; grep -q 'Listening socket on port 8080' test.log
174// test: webserver.exe -p 8000 > test.log & pid=$!; sleep 1; kill -9 $pid; grep -q 'Listening socket on port 8000' test.log
175// test: webserver.exe & pid=$!; sleep 1; kill -TERM $pid; ps aux | grep -q [w]ebserver.exe && kill -9 $pid || rc=1; test x$rc = x1
8512671a
LM
176
177/* vim: set ts=4 sw=4 et: */