mongoose/examples/http-reverse-proxy/main.c

88 lines
3.1 KiB
C
Raw Normal View History

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, see https://mongoose.ws/tutorials/tls/#how-to-build
2020-12-25 12:32:56 +08:00
2023-02-23 03:38:21 +08:00
#include "mongoose.h"
2022-08-01 18:19:32 +08:00
static const char *s_backend_url =
2023-07-26 05:41:41 +08:00
#if MG_TLS
2022-08-01 18:19:32 +08:00
"https://cesanta.com";
#else
2022-08-01 18:19:32 +08:00
"http://info.cern.ch";
#endif
2020-12-25 12:32:56 +08:00
static const char *s_listen_url = "http://localhost:8000";
// 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",
2024-03-15 15:42:24 +08:00
(int) (hm->proto.buf + hm->proto.len - hm->message.buf),
hm->message.buf);
2020-12-25 12:32:56 +08:00
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;
2024-03-15 15:42:24 +08:00
mg_printf(c, "%.*s: %.*s\r\n", (int) k->len, k->buf, (int) v->len, v->buf);
2020-12-25 12:32:56 +08:00
}
mg_send(c, "\r\n", 2);
2024-03-15 15:42:24 +08:00
mg_send(c, hm->body.buf, hm->body.len);
MG_DEBUG(("FORWARDING: %.*s %.*s", (int) hm->method.len, hm->method.buf,
(int) hm->uri.len, hm->uri.buf));
2020-12-25 12:32:56 +08:00
}
static void fn2(struct mg_connection *c, int ev, void *ev_data) {
struct mg_connection *c2 = (struct mg_connection *) c->fn_data;
2022-01-05 19:45:22 +08:00
if (ev == MG_EV_READ) {
2021-10-28 02:18:19 +08:00
// All incoming data from the backend, forward to the client
2022-01-05 19:45:22 +08:00
if (c2 != NULL) mg_send(c2, c->recv.buf, c->recv.len);
2021-10-28 02:18:19 +08:00
mg_iobuf_del(&c->recv, 0, c->recv.len);
2022-01-05 19:45:22 +08:00
} else if (ev == MG_EV_CLOSE) {
if (c2 != NULL) c2->fn_data = NULL;
}
2021-10-28 02:18:19 +08:00
(void) ev_data;
}
static void fn(struct mg_connection *c, int ev, void *ev_data) {
struct mg_connection *c2 = c->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.
2021-10-28 02:18:19 +08:00
c2 = mg_connect(c->mgr, s_backend_url, fn2, c);
if (c2 == NULL) {
mg_error(c, "Cannot create backend connection");
} else {
if (mg_url_is_ssl(s_backend_url)) {
struct mg_tls_opts opts = {.ca = mg_unpacked("/certs/ca.pem"),
.name = mg_url_host(s_backend_url)};
mg_tls_init(c2, &opts);
}
2021-10-28 02:18:19 +08:00
c->fn_data = c2;
forward_request(hm, c2);
c->is_resp = 0; // process further msgs in keep-alive connection
2022-01-05 19:45:22 +08:00
c2->is_hexdumping = 1;
}
2020-12-25 12:32:56 +08:00
} else if (ev == MG_EV_CLOSE) {
if (c2 != NULL) c2->is_closing = 1;
2022-01-05 19:45:22 +08:00
if (c2 != NULL) c2->fn_data = NULL;
2020-12-25 12:32:56 +08:00
}
}
int main(void) {
struct mg_mgr mgr;
2022-08-01 18:19:32 +08:00
mg_log_set(MG_LL_DEBUG); // Set log level
2020-12-25 12:32:56 +08:00
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;
}