mirror of
https://github.com/cesanta/mongoose.git
synced 2025-01-08 04:19:10 +08:00
203 lines
6.0 KiB
C
203 lines
6.0 KiB
C
// Copyright (c) 2014 Cesanta Software Limited
|
|
// All rights reserved
|
|
//
|
|
// To build and run this example:
|
|
// git clone https://github.com/cesanta/net_skeleton.git
|
|
// git clone https://github.com/cesanta/mongoose.git
|
|
// cd mongoose/examples
|
|
// make proxy
|
|
// ./proxy
|
|
//
|
|
// Configure your browser to use localhost:2014 as a proxy for all protocols
|
|
// Then, navigate to https://cesanta.com
|
|
|
|
#include <sys/stat.h>
|
|
#include <signal.h>
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
|
|
#ifdef _WIN32
|
|
#define sleep(x) Sleep((x) * 1000)
|
|
#else
|
|
#include <unistd.h>
|
|
#endif
|
|
|
|
#include "mongoose.h"
|
|
|
|
static int s_received_signal = 0;
|
|
static struct mg_server *s_server = NULL;
|
|
|
|
#define SSE_CONNECTION ((void *) 1)
|
|
|
|
static void elog(int do_exit, const char *fmt, ...) {
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
vfprintf(stderr, fmt, ap);
|
|
va_end(ap);
|
|
fputc('\n', stderr);
|
|
if (do_exit) exit(EXIT_FAILURE);
|
|
}
|
|
|
|
static void signal_handler(int sig_num) {
|
|
signal(sig_num, signal_handler);
|
|
s_received_signal = sig_num;
|
|
}
|
|
|
|
static int sse_push(struct mg_connection *conn, enum mg_event ev) {
|
|
if (ev == MG_POLL && conn->connection_param == SSE_CONNECTION) {
|
|
mg_printf(conn, "data: %s\r\n\r\n", (const char *) conn->callback_param);
|
|
}
|
|
return MG_TRUE;
|
|
}
|
|
|
|
static void *sse_pusher_thread_func(void *param) {
|
|
while (s_received_signal == 0) {
|
|
mg_wakeup_server_ex(s_server, sse_push, "%lu %s",
|
|
(unsigned long) time(NULL), (const char *) param);
|
|
sleep(1);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
// Return: 1 if regular file, 2 if directory, 0 if not found
|
|
static int exists(const char *path) {
|
|
struct stat st;
|
|
return stat(path, &st) != 0 ? 0 : S_ISDIR(st.st_mode) == 0 ? 1 : 2;
|
|
}
|
|
|
|
// Return: 1 if regular file, 2 if directory, 0 if not found
|
|
static int is_local_file(const char *uri, char *path, size_t path_len) {
|
|
snprintf(path, path_len, "%s/%s",
|
|
mg_get_option(s_server, "document_root"), uri);
|
|
return exists(path);
|
|
}
|
|
|
|
static int try_to_serve_locally(struct mg_connection *conn) {
|
|
char path[500], buf[2000];
|
|
int n, res;
|
|
FILE *fp = NULL;
|
|
|
|
if ((res = is_local_file(conn->uri, path, sizeof(path))) == 2) {
|
|
strncat(path, "/index.html", sizeof(path) - strlen(path) - 1);
|
|
res = exists(path);
|
|
printf("PATH: [%s]\n", path);
|
|
}
|
|
if (res == 0) return MG_FALSE;
|
|
|
|
if ((fp = fopen(path, "rb")) != NULL) {
|
|
printf("Serving [%s] locally \n", path);
|
|
mg_send_header(conn, "Connection", "close");
|
|
mg_send_header(conn, "Content-Type", mg_get_mime_type(path, "text/plain"));
|
|
while ((n = fread(buf, 1, sizeof(buf), fp)) > 0) {
|
|
mg_send_data(conn, buf, n);
|
|
}
|
|
mg_send_data(conn, "", 0);
|
|
fclose(fp);
|
|
}
|
|
return fp == NULL ? MG_FALSE : MG_TRUE;
|
|
}
|
|
|
|
static int is_resource_present_locally(const char *uri) {
|
|
char path[500];
|
|
return is_local_file(uri, path, sizeof(path)) || strcmp(uri, "/api/sse") == 0;
|
|
}
|
|
|
|
static int proxy_event_handler(struct mg_connection *conn, enum mg_event ev) {
|
|
static const char target_url[] = "http://cesanta.com";
|
|
static int target_url_size = sizeof(target_url) - 1;
|
|
const char *host;
|
|
|
|
switch (ev) {
|
|
case MG_REQUEST:
|
|
host = mg_get_header(conn, "Host");
|
|
printf("[%s] [%s] [%s]\n", conn->request_method, conn->uri,
|
|
host == NULL ? "" : host);
|
|
if (strstr(conn->uri, "/qqq") != NULL) s_received_signal = SIGTERM;
|
|
|
|
// Proxied HTTPS requests use "CONNECT foo.com:443"
|
|
// Proxied HTTP requests use "GET http://..... "
|
|
// Serve requests for target_url from the local FS.
|
|
if (memcmp(conn->uri, target_url, target_url_size) == 0 &&
|
|
is_resource_present_locally(conn->uri + target_url_size)) {
|
|
conn->uri += target_url_size; // Leave only path in the URI
|
|
}
|
|
|
|
if (strcmp(conn->uri, "/api/sse") == 0) {
|
|
conn->connection_param = SSE_CONNECTION;
|
|
mg_printf(conn, "%s", "HTTP/1.0 200 OK\r\n"
|
|
"Content-Type: text/event-stream\r\n"
|
|
"Cache-Control: no-cache\r\n\r\n");
|
|
return MG_MORE;
|
|
}
|
|
|
|
if (host != NULL && strstr(host, "cesanta") != NULL) {
|
|
return try_to_serve_locally(conn);
|
|
}
|
|
|
|
// Enable man-in-the-middle SSL mode for oracle.com
|
|
if (!strcmp(conn->request_method, "CONNECT") &&
|
|
!strcmp(host, "oracle.com")) {
|
|
mg_terminate_ssl(conn, "ssl_cert.pem"); // MUST return MG_MORE after
|
|
return MG_MORE;
|
|
}
|
|
|
|
return MG_FALSE;
|
|
case MG_AUTH:
|
|
return MG_TRUE;
|
|
default:
|
|
return MG_FALSE;
|
|
}
|
|
}
|
|
|
|
static void setopt(struct mg_server *s, const char *opt, const char *val) {
|
|
const char *err_msg = mg_set_option(s, opt, val);
|
|
if (err_msg != NULL) {
|
|
elog(1, "Error setting [%s]: [%s]", opt, err_msg);
|
|
}
|
|
}
|
|
|
|
int main(int argc, char *argv[]) {
|
|
const char *port = "2014", *dump = NULL, *root = "proxy_web_root";
|
|
int i;
|
|
|
|
// Parse command line options
|
|
for (i = 1; i < argc; i++) {
|
|
if (strcmp(argv[i], "-port") == 0 && i + 1 < argc) {
|
|
port = argv[++i];
|
|
} else if (strcmp(argv[i], "-root") == 0 && i + 1 < argc) {
|
|
root = argv[++i];
|
|
} else if (strcmp(argv[i], "-dump") == 0 && i + 1 < argc) {
|
|
dump = argv[++i];
|
|
} else {
|
|
elog(1, "Usage: %s [-cert FILE] [-ca_cert FILE] [-port PORT]", argv[0]);
|
|
}
|
|
}
|
|
|
|
signal(SIGTERM, signal_handler);
|
|
signal(SIGINT, signal_handler);
|
|
|
|
// Create and configure proxy server
|
|
s_server = mg_create_server(NULL, &proxy_event_handler);
|
|
setopt(s_server, "enable_proxy", "yes");
|
|
setopt(s_server, "document_root", root);
|
|
setopt(s_server, "listening_port", port);
|
|
setopt(s_server, "hexdump_file", dump);
|
|
|
|
// Start two SSE pushing threads
|
|
mg_start_thread(sse_pusher_thread_func, (void *) "sse_pusher_thread_1");
|
|
mg_start_thread(sse_pusher_thread_func, (void *) "sse_pusher_thread_2");
|
|
|
|
// Start serving in the main thread
|
|
printf("Starting on port %s\n", mg_get_option(s_server, "listening_port"));
|
|
while (s_received_signal == 0) {
|
|
mg_poll_server(s_server, 1000);
|
|
}
|
|
printf("Existing on signal %d\n", s_received_signal);
|
|
mg_destroy_server(&s_server);
|
|
|
|
return EXIT_SUCCESS;
|
|
}
|