-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathserver_cli.c
113 lines (92 loc) · 2.47 KB
/
server_cli.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
#include "httpd.h"
#include "cli.h"
#include <stdio.h>
#include <argp.h>
#include <stdlib.h>
#include <stdbool.h>
#define SERVER_DEFAULT_BIND 8080
#define SERVER_DEFAULT_FORKS 1
static struct {
uint16_t bind;
uint8_t forks;
bool dryrun;
} _settings = {
SERVER_DEFAULT_BIND,
SERVER_DEFAULT_FORKS,
false,
};
static char _doc[] = "HTTP echo server using Linux epoll.";
static char _args_doc[] = "";
/* Options definition */
static struct argp_option _options[] = {
ARG_VERBOSITY,
ARG_CONCURRENCY,
ARG_DRYRUN,
{ "bind", 'b', "PORT", 0,
"TCP Port to bind, default: " STR(SERVER_DEFAULT_BIND) "." },
{ 0 }
};
/* Parse a single option. */
static int
_parse_opt(int key, char *arg, struct argp_state *state) {
switch (key) {
case 'c':
_settings.forks = atoi(arg);
if (_settings.forks <= 0) {
ERRORX("Invalid number of forks: %d", _settings.forks);
}
break;
case 'b':
_settings.bind = atoi(arg);
break;
case ARG_DRYRUN_KEY:
_settings.dryrun = true;
break;
case ARGP_KEY_ARG:
if (state->arg_num >= 0) {
/* Too many arguments. */
return ARGP_ERR_UNKNOWN;
}
break;
default:
return parse_common_opts(key, arg, state);
}
return EXIT_SUCCESS;
}
static struct argp _argp = { _options, _parse_opt, _args_doc, _doc };
static int
_dryrun() {
INFO("forks:\t\t%d", _settings.forks);
INFO("bind:\t\t%d", _settings.bind);
INFO("verbosity:\t%s(%d)", log_levelnames[log_level], log_level);
return EXIT_SUCCESS;
}
int
servercli_run(int argc, char **argv) {
/* Set no buffer for stdout */
setvbuf(stdout, NULL, _IONBF, 0);
setvbuf(stderr, NULL, _IONBF, 0);
/* Parse CLI arguments. */
argp_parse(&_argp, argc, argv, 0, 0, NULL);
/* Configure HTTP server */
struct httpd server = {
.forks = _settings.forks,
.max_headers_size = 1024 * 8,
.bind = _settings.bind
};
/* Dry Run. */
if (_settings.dryrun) {
return _dryrun();
}
/* Start it */
if (httpd_start(&server)) {
ERROR("Cannot start HTTP server.");
return EXIT_FAILURE;
}
/* Prompt listening port. */
INFO("Listening on port: %d", server.bind);
/* Wait for server to stop. */
int ret = httpd_join(&server);
//__gcov_flush();
return ret;
}