2020-12-25 12:32:56 +08:00
|
|
|
// Copyright (c) 2020 Cesanta Software Limited
|
|
|
|
// All rights reserved
|
|
|
|
//
|
|
|
|
// Example HTTP reverse proxy
|
|
|
|
// 1. Run `make`. This builds and starts a proxy on port 8000
|
|
|
|
// 2. Start your browser, go to https://localhost:8000
|
|
|
|
//
|
|
|
|
// To enable SSL/TLS, build it like this:
|
|
|
|
// make MBEDTLS_DIR=/path/to/your/mbedtls/installation
|
|
|
|
|
|
|
|
static const char *s_backend_url = "https://cesanta.com";
|
|
|
|
static const char *s_listen_url = "http://localhost:8000";
|
|
|
|
|
|
|
|
#include "mongoose.h"
|
|
|
|
|
|
|
|
// Forward client request to the backend connection, rewriting the Host header
|
|
|
|
static void forward_request(struct mg_http_message *hm,
|
|
|
|
struct mg_connection *c) {
|
|
|
|
size_t i, max = sizeof(hm->headers) / sizeof(hm->headers[0]);
|
|
|
|
struct mg_str host = mg_url_host(s_backend_url);
|
|
|
|
mg_printf(c, "%.*s\r\n",
|
|
|
|
(int) (hm->proto.ptr + hm->proto.len - hm->message.ptr),
|
|
|
|
hm->message.ptr);
|
|
|
|
for (i = 0; i < max && hm->headers[i].name.len > 0; i++) {
|
|
|
|
struct mg_str *k = &hm->headers[i].name, *v = &hm->headers[i].value;
|
|
|
|
if (mg_strcmp(*k, mg_str("Host")) == 0) v = &host;
|
|
|
|
mg_printf(c, "%.*s: %.*s\r\n", (int) k->len, k->ptr, (int) v->len, v->ptr);
|
|
|
|
}
|
|
|
|
mg_send(c, "\r\n", 2);
|
|
|
|
mg_send(c, hm->body.ptr, hm->body.len);
|
|
|
|
LOG(LL_DEBUG, ("FORWARDING: %.*s %.*s", (int) hm->method.len, hm->method.ptr,
|
|
|
|
(int) hm->uri.len, hm->uri.ptr));
|
|
|
|
}
|
|
|
|
|
2021-09-24 11:30:21 +08:00
|
|
|
static void forward_fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) {
|
2020-12-25 12:32:56 +08:00
|
|
|
struct mg_connection *c2 = fn_data;
|
2021-09-24 11:30:21 +08:00
|
|
|
if (ev == MG_EV_READ) {
|
|
|
|
if (c2 != NULL) {
|
2020-12-25 12:32:56 +08:00
|
|
|
// All incoming data from the backend, forward to the client
|
|
|
|
mg_send(c2, c->recv.buf, c->recv.len);
|
2021-08-28 14:16:38 +08:00
|
|
|
mg_iobuf_del(&c->recv, 0, c->recv.len);
|
2020-12-25 12:32:56 +08:00
|
|
|
}
|
|
|
|
} else if (ev == MG_EV_CONNECT) {
|
|
|
|
if (mg_url_is_ssl(s_backend_url)) {
|
|
|
|
struct mg_tls_opts opts = {.ca = "ca.pem"};
|
|
|
|
mg_tls_init(c, &opts);
|
|
|
|
}
|
2021-09-24 11:30:21 +08:00
|
|
|
}
|
|
|
|
(void)ev_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) {
|
|
|
|
struct mg_connection *c2 = fn_data;
|
|
|
|
if (ev == MG_EV_HTTP_MSG) {
|
|
|
|
struct mg_http_message *hm = (struct mg_http_message *) ev_data;
|
|
|
|
// Client request, create backend connection Note that we're passing
|
|
|
|
// client connection `c` as fn_data for the created backend connection.
|
|
|
|
c2 = mg_connect(c->mgr, s_backend_url, forward_fn, c);
|
|
|
|
c->fn_data = c2;
|
|
|
|
|
|
|
|
if (c2 != NULL) {
|
|
|
|
forward_request(hm, c2);
|
|
|
|
} else {
|
|
|
|
c->is_closing = 1;
|
|
|
|
}
|
|
|
|
} else if (ev == MG_EV_CONNECT) {
|
|
|
|
if (mg_url_is_ssl(s_backend_url)) {
|
|
|
|
struct mg_tls_opts opts = {.ca = "ca.pem"};
|
|
|
|
mg_tls_init(c, &opts);
|
|
|
|
}
|
2020-12-25 12:32:56 +08:00
|
|
|
} else if (ev == MG_EV_CLOSE) {
|
2021-09-24 11:30:21 +08:00
|
|
|
if (c2 != NULL) c2->is_closing = 1;
|
2020-12-25 12:32:56 +08:00
|
|
|
c->fn_data = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(void) {
|
|
|
|
struct mg_mgr mgr;
|
|
|
|
|
|
|
|
mg_log_set("3"); // Set log level
|
|
|
|
mg_mgr_init(&mgr); // Initialise event manager
|
|
|
|
mg_http_listen(&mgr, s_listen_url, fn, NULL); // Start proxy
|
|
|
|
for (;;) mg_mgr_poll(&mgr, 1000); // Event loop
|
|
|
|
mg_mgr_free(&mgr);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|