mirror of
https://github.com/cesanta/mongoose.git
synced 2025-01-01 02:27:52 +08:00
174 lines
6.3 KiB
C
174 lines
6.3 KiB
C
// Copyright (c) 2020 Cesanta Software Limited
|
|
// All rights reserved
|
|
//
|
|
// Example socks5 server. To test,
|
|
// 1. Run `make` to start this server on port 1080
|
|
// 2. Run `curl`
|
|
|
|
#include "mongoose.h"
|
|
|
|
static const char *s_lsn = "tcp://localhost:1080"; // Listening address
|
|
|
|
enum {
|
|
VERSION = 5, // Socks protocol version
|
|
STATE_HANDSHAKE = 0, // Connection state: in handshake
|
|
STATE_REQUEST = 1, // Connection state: connecting
|
|
STATE_ESTABLISHED = 2, // Connection state: established
|
|
HANDSHAKE_NOAUTH = 0, // Handshake method - no authentication
|
|
HANDSHAKE_GSSAPI = 1, // Handshake method - GSSAPI auth
|
|
HANDSHAKE_USERPASS = 2, // Handshake method - user/password auth
|
|
HANDSHAKE_FAILURE = 0xff, // Handshake method - failure
|
|
CMD_CONNECT = 1, // Command: CONNECT
|
|
CMD_BIND = 2, // Command: BIND
|
|
CMD_UDP_ASSOCIATE = 3, // Command: UDP ASSOCIATE
|
|
ADDR_TYPE_IPV4 = 1, // Address type: IPv4
|
|
ADDR_TYPE_DOMAIN = 3, // Address type: Domain name
|
|
ADDR_TYPE_IPV6 = 4, // Address type: IPv6
|
|
RESP_SUCCESS = 0, // Response: success
|
|
RESP_FAILURE = 1, // Response: failure
|
|
RESP_NOT_ALLOWED = 2, // Response status
|
|
RESP_NET_UNREACHABLE = 3, // Response status
|
|
RESP_HOST_UNREACHABLE = 4, // Response status
|
|
RESP_CONN_REFUSED = 5, // Response status
|
|
RESP_TTL_EXPIRED = 6, // Response status
|
|
RESP_CMD_NOT_SUPPORTED = 7, // Response status
|
|
RESP_ADDR_NOT_SUPPORTED = 8, // Response status
|
|
};
|
|
|
|
// https://www.ietf.org/rfc/rfc1928.txt paragraph 3, handle client handshake
|
|
// +----+----------+----------+
|
|
// |VER | NMETHODS | METHODS |
|
|
// +----+----------+----------+
|
|
// | 1 | 1 | 1 to 255 |
|
|
// +----+----------+----------+
|
|
static void handshake(struct mg_connection *c) {
|
|
struct mg_iobuf *r = &c->recv;
|
|
if (r->buf[0] != VERSION) {
|
|
c->is_closing = 1;
|
|
} else if (r->len > 2 && (size_t) r->buf[1] + 2 <= r->len) {
|
|
/* https://www.ietf.org/rfc/rfc1928.txt paragraph 3 */
|
|
uint8_t reply[2] = {VERSION, HANDSHAKE_FAILURE};
|
|
int i;
|
|
for (i = 2; i < r->buf[1] + 2; i++) {
|
|
// TODO(lsm): support other auth methods
|
|
if (r->buf[i] == HANDSHAKE_NOAUTH) reply[1] = r->buf[i];
|
|
}
|
|
mg_iobuf_del(r, 0, 2 + r->buf[1]);
|
|
mg_send(c, reply, sizeof(reply));
|
|
c->label[0] = STATE_REQUEST;
|
|
}
|
|
}
|
|
|
|
static void disband(struct mg_connection *c) {
|
|
struct mg_connection *c2 = (struct mg_connection *) c->fn_data;
|
|
if (c2 != NULL) {
|
|
c2->is_draining = 1;
|
|
c2->fn_data = NULL;
|
|
}
|
|
c->is_draining = 1;
|
|
c->fn_data = NULL;
|
|
}
|
|
|
|
static void exchange(struct mg_connection *c) {
|
|
struct mg_connection *c2 = (struct mg_connection *) c->fn_data;
|
|
if (c2 != NULL) {
|
|
mg_send(c2, c->recv.buf, c->recv.len);
|
|
mg_iobuf_del(&c->recv, 0, c->recv.len);
|
|
} else {
|
|
c->is_draining = 1;
|
|
}
|
|
}
|
|
|
|
static void fn2(struct mg_connection *c, int ev, void *ev_data, void *fn_data) {
|
|
if (ev == MG_EV_READ) {
|
|
exchange(c);
|
|
} else if (ev == MG_EV_CLOSE) {
|
|
disband(c);
|
|
}
|
|
(void) ev_data;
|
|
(void) fn_data;
|
|
}
|
|
|
|
// Request, https://www.ietf.org/rfc/rfc1928.txt paragraph 4
|
|
// +----+-----+-------+------+----------+----------+
|
|
// |VER | CMD | RSV | ATYP | DST.ADDR | DST.PORT |
|
|
// +----+-----+-------+------+----------+----------+
|
|
// | 1 | 1 | X'00' | 1 | Variable | 2 |
|
|
// +----+-----+-------+------+----------+----------+
|
|
static void request(struct mg_connection *c) {
|
|
struct mg_iobuf *r = &c->recv;
|
|
uint8_t *p = r->buf, addr_len = 4, reply = RESP_SUCCESS;
|
|
int ver, cmd, atyp;
|
|
char addr[1024];
|
|
|
|
if (r->len < 8) return; // return if not fully buffered. min DST.ADDR is 2
|
|
ver = p[0];
|
|
cmd = p[1];
|
|
atyp = p[3];
|
|
|
|
// TODO(lsm): support other commands
|
|
if (ver != VERSION || cmd != CMD_CONNECT) {
|
|
reply = RESP_CMD_NOT_SUPPORTED;
|
|
} else if (atyp == ADDR_TYPE_IPV4) {
|
|
addr_len = 4;
|
|
if (r->len < (size_t) addr_len + 6) return; /* return if not buffered */
|
|
snprintf(addr, sizeof(addr), "tcp://%d.%d.%d.%d:%d", p[4], p[5], p[6], p[7],
|
|
p[8] << 8 | p[9]);
|
|
c->fn_data = mg_connect(c->mgr, addr, fn2, c);
|
|
} else if (atyp == ADDR_TYPE_IPV6) {
|
|
addr_len = 16;
|
|
if (r->len < (size_t) addr_len + 6) return; /* return if not buffered */
|
|
snprintf(addr, sizeof(addr), "tcp://[%x:%x:%x:%x:%x:%x:%x:%x]:%d",
|
|
p[4] << 8 | p[5], p[6] << 8 | p[7], p[8] << 8 | p[9],
|
|
p[10] << 8 | p[11], p[12] << 8 | p[13], p[14] << 8 | p[15],
|
|
p[16] << 8 | p[17], p[18] << 8 | p[19], p[20] << 8 | p[21]);
|
|
c->fn_data = mg_connect(c->mgr, addr, fn2, c);
|
|
} else if (atyp == ADDR_TYPE_DOMAIN) {
|
|
addr_len = p[4] + 1;
|
|
if (r->len < (size_t) addr_len + 6) return; /* return if not buffered */
|
|
snprintf(addr, sizeof(addr), "tcp://%.*s:%d", p[4], p + 5,
|
|
p[4 + addr_len] << 8 | p[4 + addr_len + 1]);
|
|
c->fn_data = mg_connect(c->mgr, addr, fn2, c);
|
|
} else {
|
|
reply = RESP_ADDR_NOT_SUPPORTED;
|
|
}
|
|
|
|
// Reply, https://www.ietf.org/rfc/rfc1928.txt paragraph 5
|
|
//
|
|
// +----+-----+-------+------+----------+----------+
|
|
// |VER | REP | RSV | ATYP | BND.ADDR | BND.PORT |
|
|
// +----+-----+-------+------+----------+----------+
|
|
// | 1 | 1 | X'00' | 1 | Variable | 2 |
|
|
// +----+-----+-------+------+----------+----------+
|
|
{
|
|
uint8_t buf[] = {VERSION, reply, 0};
|
|
mg_send(c, buf, sizeof(buf));
|
|
}
|
|
mg_send(c, r->buf + 3, addr_len + 1 + 2);
|
|
mg_iobuf_del(r, 0, 6 + addr_len); // Remove request from the input stream
|
|
c->label[0] = STATE_ESTABLISHED; // Mark ourselves as connected
|
|
}
|
|
|
|
static void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) {
|
|
if (ev == MG_EV_READ) {
|
|
// We use the first label byte as a state
|
|
if (c->label[0] == STATE_HANDSHAKE) handshake(c);
|
|
if (c->label[0] == STATE_REQUEST) request(c);
|
|
if (c->label[0] == STATE_ESTABLISHED) exchange(c);
|
|
} else if (ev == MG_EV_CLOSE) {
|
|
disband(c);
|
|
}
|
|
(void) fn_data;
|
|
(void) ev_data;
|
|
}
|
|
|
|
int main(void) {
|
|
struct mg_mgr mgr; // Event manager
|
|
mg_log_set(MG_LL_DEBUG); // Set log level
|
|
mg_mgr_init(&mgr); // Initialise event manager
|
|
mg_listen(&mgr, s_lsn, fn, NULL); // Create client connection
|
|
while (true) mg_mgr_poll(&mgr, 1000); // Infinite event loop
|
|
mg_mgr_free(&mgr); // Free resources
|
|
return 0;
|
|
}
|