-
Notifications
You must be signed in to change notification settings - Fork 38
/
pagekitec.c
444 lines (402 loc) · 13.8 KB
/
pagekitec.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
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
/******************************************************************************
pagekitec.c - A high-performance PageKite connector in C.
Usage: pagekitec [options] LPORT PROTO NAME.pagekite.me PPORT SECRET ...
*******************************************************************************
This file is Copyright 2011-2017, The Beanstalks Project ehf.
This program is free software: you can redistribute it and/or modify it under
the terms of the Apache License 2.0 as published by the Apache Software
Foundation.
This program is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
FOR A PARTICULAR PURPOSE. See the Apache License for more details.
You should have received a copy of the Apache License along with this program.
If not, see: <http://www.apache.org/licenses/>
Note: For alternate license terms, see the file COPYING.md.
******************************************************************************/
/* FIXME! */
#define HAVE_IPV6 1
#include <pagekite.h>
#include <pkcommon.h>
#include <unistd.h>
#include <sys/types.h>
#define EXIT_ERR_MANAGER_INIT 1
#define EXIT_ERR_USAGE 2
#define EXIT_ERR_ADD_KITE 3
#define EXIT_ERR_FRONTENDS 4
#define EXIT_ERR_START_THREAD 5
#define EXIT_ERR_ADD_LISTENER 6
#define EXIT_ERR_STATUS_FILE 7
#define MAX_PLUGIN_ARGS 128
/* When the app is idle and status is unchanged, this is the maximum
* frequency for (minimum interval between) "no-op" changes to the status
* summary file (-Y). Actual update frequency is driven by libpagekite's
* internal housekeeping timers. The value is seconds. */
#define STATUS_MIN_INTERVAL 240
/* Enable this format using -Y json:/path/to/file/ */
#define STATUS_FORMAT_JSON (\
"{\n" \
" \"pagekitec_version\": \"%s\",\n" \
" \"pagekitec_status\": \"%s\",\n" \
" \"pagekitec_status_code\": %d,\n" \
" \"pagekitec_pid\": %d,\n" \
" \"pagekitec_update_min_interval\": %d,\n" \
" \"pagekitec_update_ts\": %lld\n" \
"}\n")
/* Enable this format using -Y text:/path/to/file/ */
#define STATUS_FORMAT_TEXT (\
"pagekitec_version: %s\n" \
"pagekitec_status: %s\n" \
"pagekitec_status_code: %d\n" \
"pagekitec_pid: %d\n" \
"pagekitec_update_min_interval: %d\n" \
"pagekitec_update_ts: %lld\n")
pagekite_mgr m;
void usage(int ecode, char* message) {
fprintf(stderr, "This is pagekitec.c from libpagekite %s.\n\n", PK_VERSION);
fprintf(stderr, "Usage:\tpagekitec [options] LPORT PROTO"
" NAME.pagekite.me PPORT SECRET ...\n"
"Options:\n"
"\t-q\tDecrease verbosity (less log output)\n"
"\t-v\tIncrease verbosity (more log output)\n"
"\t-a x\tSet app name to x for logging\n"
"\t-s\tLog to syslog instead of to stderr\n"
"\t-Y f\tPeriodically summarize status to file f (JSON)\n");
#ifdef HAVE_OPENSSL
fprintf(stderr, "\t-I\tConnect insecurely, without SSL.\n");
#endif
fprintf(stderr, "\t-S\tStatic setup, disable FE failover and DDNS updates\n"
"\t-w D\tWhite-label configuration using domain D.\n"
"\t-r U\tSet a custom rejection URL, U.\n"
"\t-l D\tConnect to host D instead of localhost\n"
"\t-c N\tSet max connection count to N (default = 25)\n"
"\t-n N\tAlways connect to N spare frontends (default = 0)\n"
"\t-B N\tBail out (abort) after N logged errors\n"
"\t-E N\tAllow eviction of streams idle for >N seconds\n"
"\t-F x\tUse x (a DNS name) as frontend pool\n"
"\t-P x\tUse x (a port number) as frontend port\n"
"\t-R\tChoose frontends at random, instead of pinging\n"
"\t-N\tDisable DNS-based updates of available frontends\n"
"\t-V 46\tRequest DNS visibility on IPv6 and/or IPv4\n"
"\t-4\tDisable IPv4 frontends\n");
#ifdef HAVE_IPV6
fprintf(stderr, "\t-6\tDisable IPv6 frontends\n");
#endif
fprintf(stderr, "\t-C\tDisable auto-adding current DNS IP as a front-end\n"
"\t-W\tEnable watchdog thread (dumps core if we lock up)\n"
"\t-H\tDo not add X-Forwarded-Proto and X-Forwarded-For\n\n");
if (message) fprintf(stderr, "\nError: %s\n", message);
exit(ecode);
}
void raise_log_level(int sig) {
if (sig) pagekite_set_log_mask(NULL, PK_LOG_ALL);
}
void dump_state_to_log(int sig) {
if (sig) pagekite_dump_state_to_log(m);
}
void shutdown_pagekite(int sig) {
if (sig) pagekite_thread_stop(m);
}
void safe_exit(int code) {
#ifdef _MSC_VER
fprintf(stderr, "Exiting with status code %d.\n", code);
#endif
fclose(stderr);
exit(code);
}
int summarize_status(const char* fpath, const char* format, const char *status_msg, int status) {
static time_t last_update = 0;
static int last_status = 0;
if (!fpath) return 0;
time_t64 now = time(0);
if ((now - last_update > STATUS_MIN_INTERVAL) || (last_status != status)) {
FILE *fd = fopen(fpath, "w+");
if (!fd) {
fprintf(stderr, "Failed to open %s for writing: %s\n", fpath, strerror(errno));
return 0;
}
fseek(fd, 0, SEEK_SET);
switch (status) {
case PK_STATUS_STARTUP: status_msg = "startup"; break;
case PK_STATUS_CONNECTING: status_msg = "connecting"; break;
case PK_STATUS_UPDATING_DNS: status_msg = "updating_dns"; break;
case PK_STATUS_FLYING: status_msg = "flying"; break;
case PK_STATUS_PROBLEMS: status_msg = "problems"; break;
case PK_STATUS_REJECTED: status_msg = "rejected"; break;
case PK_STATUS_NO_NETWORK: status_msg = "no_network"; break;
}
fprintf(fd, format,
PK_VERSION, status_msg, status, getpid(), STATUS_MIN_INTERVAL, now);
if (0 == ftruncate(fileno(fd), ftell(fd))) fflush(fd);
fclose(fd);
last_update = now;
last_status = status;
}
return 1;
}
int main(int argc, char **argv) {
unsigned int bail_on_errors = 0;
unsigned int conn_eviction_idle_s = 0;
char* proto;
char* kitename;
char* secret;
char* whitelabel_tld = NULL;
char* app_name = "pagekitec";
char* localhost = "localhost";
char* rejection_url = NULL;
char* status_summary_path = NULL;
char* status_summary_fmt = NULL;
int gotargs = 0;
int verbosity = 0;
int use_current = 1;
int use_fake_ping = 0;
int use_watchdog = 0;
int modify_http_headers = 1;
int max_conns = 25;
int spare_frontends = 0;
char* fe_hostname = NULL;
int fe_port = 443;
char* ddns_url = PAGEKITE_NET_DDNS;
int ac;
int pport;
int lport;
int flags = (PK_WITH_SSL
|PK_WITH_IPV4
|PK_WITH_DYNAMIC_FE_LIST
|PK_WITH_SRAND_RESEED
|PK_WITHOUT_SERVICE_FRONTENDS);
#ifdef HAVE_IPV6
flags |= PK_WITH_IPV6;
#endif
while (-1 != (ac = getopt(argc, argv,
"46a:B:c:CE:F:P:HIl:Nn:qr:RsSvV:Ww:Y:Z"))) {
switch (ac) {
case '4':
flags &= ~PK_WITH_IPV4;
break;
#ifdef HAVE_IPV6
case '6':
flags &= ~PK_WITH_IPV6;
break;
#endif
case 'V':
gotargs++;
if (strchr(optarg, '4') != NULL) flags |= PK_WITH_IPV4_DNS;
if (strchr(optarg, '6') != NULL) flags |= PK_WITH_IPV6_DNS;
break;
case 'a':
gotargs++;
app_name = strdup(optarg);
break;
case 's':
flags |= PK_WITH_SYSLOG;
break;
case 'Y':
gotargs++;
status_summary_path = strdup(optarg);
if (strncasecmp(status_summary_path, "json:", 5) == 0) {
status_summary_fmt = STATUS_FORMAT_JSON;
status_summary_path += 5;
}
else if (strncasecmp(status_summary_path, "text:", 5) == 0) {
status_summary_fmt = STATUS_FORMAT_TEXT;
status_summary_path += 5;
}
else {
status_summary_fmt = STATUS_FORMAT_JSON;
}
if (!summarize_status(status_summary_path, status_summary_fmt, "startup", 0)) {
usage(EXIT_ERR_STATUS_FILE, strerror(errno));
}
break;
case 'N':
flags &= ~PK_WITH_DYNAMIC_FE_LIST;
break;
case 'C':
use_current = 0;
break;
case 'v':
verbosity++;
break;
case 'q':
verbosity--;
break;
case 'I':
flags &= ~PK_WITH_SSL;
break;
case 'r':
gotargs++;
rejection_url = strdup(optarg);
break;
case 'R':
use_fake_ping = 1;
break;
case 'S':
ddns_url = NULL;
break;
case 'W':
use_watchdog = 1;
break;
case 'H':
modify_http_headers = 0;
break;
case 'w':
gotargs++;
whitelabel_tld = strdup(optarg);
break;
case 'l':
gotargs++;
localhost = strdup(optarg);
break;
case 'F':
gotargs++;
assert(fe_hostname == NULL);
fe_hostname = strdup(optarg);
break;
case 'P':
gotargs++;
if (1 == sscanf(optarg, "%d", &fe_port)) break;
usage(EXIT_ERR_USAGE, "Invalid argument to -P");
case 'B':
gotargs++;
if (1 == sscanf(optarg, "%u", &bail_on_errors)) break;
usage(EXIT_ERR_USAGE, "Invalid argument to -B");
case 'c':
gotargs++;
if (1 == sscanf(optarg, "%d", &max_conns)) break;
usage(EXIT_ERR_USAGE, "Invalid argument to -c");
case 'E':
gotargs++;
if (1 == sscanf(optarg, "%u", &conn_eviction_idle_s)) break;
usage(EXIT_ERR_USAGE, "Invalid argument to -E");
case 'n':
gotargs++;
if (1 == sscanf(optarg, "%d", &spare_frontends)) break;
usage(EXIT_ERR_USAGE, "Invalid argument to -n");
default:
usage(EXIT_ERR_USAGE, "Unrecognized option");
}
gotargs++;
}
if ((argc-1-gotargs) < 5 || ((argc-1-gotargs) % 5) != 0) {
usage(EXIT_ERR_USAGE,
"Invalid kite specification (wrong number of arguments)");
}
#ifndef _MSC_VER
signal(SIGUSR1, &raise_log_level);
signal(SIGUSR2, &dump_state_to_log);
signal(SIGQUIT, &shutdown_pagekite);
signal(SIGINT, &shutdown_pagekite);
signal(SIGPIPE, SIG_IGN);
#endif
if (whitelabel_tld != NULL)
{
if (NULL == (m = pagekite_init_whitelabel(
app_name,
1 + (argc-1-gotargs)/5, /* Kites */
max_conns,
flags,
verbosity,
whitelabel_tld)))
{
pagekite_perror(m, argv[0]);
safe_exit(EXIT_ERR_MANAGER_INIT);
}
}
else if (NULL == (m = pagekite_init(
app_name,
1 + (argc-1-gotargs)/5, /* Kites */
PAGEKITE_NET_FE_MAX,
max_conns,
ddns_url,
flags,
verbosity)))
{
pagekite_perror(m, argv[0]);
safe_exit(EXIT_ERR_MANAGER_INIT);
}
/* Set all the parameters */
pagekite_want_spare_frontends(m, spare_frontends);
pagekite_enable_watchdog(m, use_watchdog);
pagekite_enable_http_forwarding_headers(m, modify_http_headers);
pagekite_enable_fake_ping(m, use_fake_ping);
pagekite_set_bail_on_errors(m, bail_on_errors);
pagekite_set_conn_eviction_idle_s(m, conn_eviction_idle_s);
if (rejection_url != NULL) pagekite_set_rejection_url(m, rejection_url);
/* Move the logging to the event API, mostly for testing. */
if ((verbosity < 4) && (0 == (flags & PK_WITH_SYSLOG))) {
pagekite_set_log_destination(m, PK_LOG_DEST_NONE);
pagekite_set_event_mask(m, PK_EV_MASK_LOGGING | PK_EV_MASK_MISC);
}
else {
pagekite_set_event_mask(m, PK_EV_MASK_MISC);
}
for (ac = gotargs; ac+5 < argc; ac += 5) {
if ((1 != sscanf(argv[ac+1], "%d", &lport)) ||
(1 != sscanf(argv[ac+4], "%d", &pport))) {
pagekite_free(m);
usage(EXIT_ERR_USAGE, "Failed to parse kite specification port numbers");
}
proto = argv[ac+2];
kitename = argv[ac+3];
secret = argv[ac+5];
if ((0 > pagekite_add_kite(m, proto, kitename, pport, secret,
localhost, lport)) ||
(use_current && (0 > (pagekite_add_frontend(m, kitename, pport)))))
{
pagekite_perror(m, argv[0]);
pagekite_free(m);
safe_exit(EXIT_ERR_ADD_KITE);
}
}
/* The API could do this stuff on INIT, but since we allow for manually
specifying a front-end hostname, we do things by hand. */
if (fe_hostname) {
if (0 > pagekite_lookup_and_add_frontend(m, fe_hostname, fe_port,
flags & PK_WITH_DYNAMIC_FE_LIST))
{
pagekite_perror(m, argv[0]);
pagekite_free(m);
safe_exit(EXIT_ERR_FRONTENDS);
}
}
else if (whitelabel_tld != NULL) {
if (0 > pagekite_add_whitelabel_frontends(m, flags, whitelabel_tld)) {
pagekite_perror(m, argv[0]);
pagekite_free(m);
safe_exit(EXIT_ERR_FRONTENDS);
}
}
else if (ddns_url != NULL) {
if (0 > pagekite_add_service_frontends(m, flags)) {
pagekite_perror(m, argv[0]);
pagekite_free(m);
safe_exit(EXIT_ERR_FRONTENDS);
}
}
if (0 > pagekite_thread_start(m)) {
pagekite_perror(m, argv[0]);
pagekite_free(m);
safe_exit(EXIT_ERR_START_THREAD);
}
summarize_status(
status_summary_path, status_summary_fmt, "unknown", pagekite_get_status(m));
unsigned int eid;
while (PK_EV_SHUTDOWN != (
(eid = pagekite_await_event(m, 1)) & PK_EV_TYPE_MASK))
{
switch (eid & PK_EV_TYPE_MASK) {
case PK_EV_LOGGING:
fprintf(stderr, "%s\n", pagekite_get_event_str(m, eid));
break;
default:
if (verbosity > 2) fprintf(stderr, "[Got event 0x%8.8x]\n", eid);
}
pagekite_event_respond(m, eid, PK_EV_RESPOND_DEFAULT);
summarize_status(
status_summary_path, status_summary_fmt, "unknown", pagekite_get_status(m));
}
pagekite_thread_wait(m);
pagekite_free(m);
summarize_status(status_summary_path, status_summary_fmt, "shutdown", 0);
return 0;
}