# User Guide ## Introduction Mongoose is a networking library for C/C++. It implements event-driven, non-blocking APIs for TCP, UDP, HTTP, WebSocket, MQTT. It connects devices and brings them online. Since 2004, a number of open source and commercial products have utilized it. It even runs on the International Space Station! Mongoose makes embedded network programming fast, robust, and easy. Mongoose has two basic data structures: - `struct mg_mgr` - An event manager that holds all active connections - `struct mg_connection` - A single connection descriptor Connections could be listening, outbound, or inbound. Outbound connections are created by the `mg_connect()` call. Listening connections are created by the `mg_listen()` call. Inbound connections are those accepted by a listening connection. Each connection is described by a `struct mg_connection` structure, which has a number of fields. All fields are exposed to the application by design, to give an application full visibility into Mongoose's internals. ## 2-minute integration guide In order to integrate Mongoose into an existing C/C++ application or firmware, please use the following steps: **Step 1.** Copy [mongoose.c](https://raw.githubusercontent.com/cesanta/mongoose/master/mongoose.c) and [mongoose.h](https://raw.githubusercontent.com/cesanta/mongoose/master/mongoose.h) into the source code tree **Step 2.** Add the following lines in your `main.c` file: ```c #include "mongoose.h" ... static void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { struct mg_http_serve_opts opts = {.root_dir = "."}; // Serve local dir if (ev == MG_EV_HTTP_MSG) mg_http_serve_dir(c, ev_data, &opts); } ... int main() { ... struct mg_mgr mgr; mg_mgr_init(&mgr); mg_http_listen(&mgr, "0.0.0.0:8000", fn, NULL); // Create listening connection for (;;) mg_mgr_poll(&mgr, 1000); // Block forever } ``` **Step 3.** Add `mongoose.c` to the build and recompile - and that is it! `mg_mgr_poll()` iterates over all connections, accepts new connections, sends and receives data, closes connections, and calls event handler functions for the respective events. Each connection has two event handler functions: `c->fn` and `c->pfn`. The `c->fn` is a user-specified event handler function. The `c->pfn` is a protocol-specific handler function that is set implicitly. For example, a `mg_http_listen()` sets `c->pfn` to a Mongoose's HTTP event handler. A protocol-specific handler is called before a user-specific handler. It parses incoming data and may invoke protocol-specific events like `MG_EV_HTTP_MSG`. NOTE: Since Mongoose's core is not protected against concurrent accesses, make sure that all `mg_*` API functions are called from the same thread or RTOS task. ## Send and receive buffers Each connection has a send and receive buffer: - `struct mg_connection::send` - Data to be sent to a peer - `struct mg_connection::recv` - Data received from a peer When data arrives, Mongoose appends received data to the `recv` and triggers a `MG_EV_READ` event. The user may send data back by calling one of the output functions, like `mg_send()`, `mg_printf()` or a protocol-specific function like `mg_ws_send`. Output functions append data to the `send` buffer. When Mongoose successfully writes data to the socket, it discards data from struct `mg_connection::send` and sends an `MG_EV_SEND` event. ## Event handler function Each connection has an event handler function associated with it, which must be implemented by the user. Event handler is the key element of Mongoose, since it defines the connection's behavior. See below for an example of an event handler function: ```c // Event handler function defines connection's behavior static void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_READ) { mg_send(c, c->recv.buf, c->recv.len); // Implement echo server c->recv.len = 0; // Delete received data } } ``` - `struct mg_connection *c` - A connection that received an event - `int ev` - An event number, defined in mongoose.h. For example, when data arrives on an inbound connection, `ev` would be `MG_EV_READ` - `void *ev_data` - Points to the event-specific data, and it has a different meaning for different events. For example, for an `MG_EV_READ` event, `ev_data` is an `int *` pointing to the number of bytes received from a remote peer and saved into the `c->recv` IO buffer. The exact meaning of `ev_data` is described for each event. Protocol-specific events usually have `ev_data` pointing to structures that hold protocol-specific information - `void *fn_data` - A user-defined pointer for the connection, which is a placeholder for application-specific data. This `fn_data` pointer is set during the `*_listen()` or `*_connect()` call, and it is stored in the `c->fn_data`. Listening connections copy the value of `c->fn_data` to the newly accepted connection, so all accepted connections initially share the same `fn_data` pointer. It is fine to update/replace that pointer for any connection at any time by setting `c->fn_data = new_value;` ## Events Below is the list of events triggered by Mongoose, taken as-is from `mongoose.h`. For each event, a comment describes the meaning of the `ev_data` pointer passed to an event handler: ```c enum { MG_EV_ERROR, // Error char *error_message MG_EV_POLL, // mg_mgr_poll iteration unsigned long *millis MG_EV_RESOLVE, // Host name is resolved NULL MG_EV_CONNECT, // Connection established NULL MG_EV_ACCEPT, // Connection accepted NULL MG_EV_READ, // Data received from socket struct mg_str * MG_EV_WRITE, // Data written to socket int *num_bytes_written MG_EV_CLOSE, // Connection closed NULL MG_EV_HTTP_MSG, // HTTP request/response struct mg_http_message * MG_EV_HTTP_CHUNK, // HTTP chunk (partial msg) struct mg_http_message * MG_EV_WS_OPEN, // Websocket handshake done struct mg_http_message * MG_EV_WS_MSG, // Websocket msg, text or bin struct mg_ws_message * MG_EV_WS_CTL, // Websocket control msg struct mg_ws_message * MG_EV_MQTT_CMD, // MQTT low-level command struct mg_mqtt_message * MG_EV_MQTT_MSG, // MQTT PUBLISH received struct mg_mqtt_message * MG_EV_MQTT_OPEN, // MQTT CONNACK received int *connack_status_code MG_EV_SNTP_TIME, // SNTP time received struct timeval * MG_EV_USER, // Starting ID for user events }; ``` ## Connection flags `struct mg_connection` has a bitfield with connection flags. Flags are binary: they can be either 0 or 1. Some flags are set by Mongoose and must be not changed by an application code. For example, the `is_udp` flag tells the application if that connection is UDP or not. Some flags can be changed by application, for example, the `is_draining` flag, if set by an application, tells Mongoose to send the remaining data to a peer, and when everything is sent, close the connection. NOTE: User-changeable flags are: `is_hexdumping`, `is_draining`, `is_closing`. This is taken from `mongoose.h` as-is: ```c struct mg_connection { ... unsigned is_listening : 1; // Listening connection unsigned is_client : 1; // Outbound (client) connection unsigned is_accepted : 1; // Accepted (server) connection unsigned is_resolving : 1; // Non-blocking DNS resolv is in progress unsigned is_connecting : 1; // Non-blocking connect is in progress unsigned is_tls : 1; // TLS-enabled connection unsigned is_tls_hs : 1; // TLS handshake is in progress unsigned is_udp : 1; // UDP connection unsigned is_websocket : 1; // WebSocket connection unsigned is_hexdumping : 1; // Hexdump in/out traffic unsigned is_draining : 1; // Send remaining data, then close and free unsigned is_closing : 1; // Close and free the connection immediately unsigned is_readable : 1; // Connection is ready to read unsigned is_writable : 1; // Connection is ready to write }; ``` ## Build options Mongoose source code ships in two files: - [mongoose.h](https://github.com/cesanta/mongoose/blob/master/mongoose.h) - API definitions - [mongoose.c](https://github.com/cesanta/mongoose/blob/master/mongoose.c) - Implementation Therefore, to integrate Mongoose into an application, simply copy these two files to the application's source tree. The `mongoose.c` and `mongoose.h` files are actually an amalgamation - non-amalgamated sources can be found at https://github.com/cesanta/mongoose/tree/master/src Mongoose has two types of build constants (preprocessor definitions) that affect the build: a target architecture, and tunables. In order to set the option during build time, use the `-D OPTION` compiler flag: ```sh $ cc app0.c mongoose.c # Use defaults! $ cc app1.c mongoose.c -D MG_ENABLE_IPV6=1 # Build with IPv6 enabled $ cc app2.c mongoose.c -D MG_ARCH=MG_ARCH_FREERTOS_LWIP # Set architecture $ cc app3.c mongoose.c -D MG_ENABLE_SSI=0 -D MG_ENABLE_LOG=0 # Multiple options ``` The list of supported architectures is defined in the [arch.h](https://github.com/cesanta/mongoose/blob/master/src/arch.h) header file. Normally, there is no need to explicitly specify the architecture. The architecture is guessed during the build, so setting it is not usually required. | Name | Description | | ---- | ----------- | |MG_ARCH_UNIX | All UNIX-like systems like Linux, MacOS, FreeBSD, etc | |MG_ARCH_WIN32 | Windows systems | |MG_ARCH_ESP32 | Espressif's ESP32 | |MG_ARCH_ESP8266 | Espressif's ESP8266 | |MG_ARCH_FREERTOS_LWIP | All systems with FreeRTOS kernel and LwIP IP stack | |MG_ARCH_FREERTOS_TCP | All systems with FreeRTOS kernel and FreeRTOS-Plus-TCP IP stack | |MG_ARCH_CUSTOM | A custom architecture, discussed in the next section | The other class of build constants is defined in [src/config.h](https://github.com/cesanta/mongoose/blob/master/src/config.h) together with their default values. These are tunables that include/exclude a certain functionality or change relevant parameters. Here is a list of build constants and their default values: | Name | Default | Description | | ---- | ------- | ----------- | |MG_ENABLE_SOCKET | 1 | Use BSD socket low-level API | |MG_ENABLE_MBEDTLS | 0 | Enable mbedTLS library | |MG_ENABLE_OPENSSL | 0 | Enable OpenSSL library | |MG_ENABLE_IPV6 | 0 | Enable IPv6 | |MG_ENABLE_LOG | 1 | Enable `LOG()` macro | |MG_ENABLE_MD5 | 0 | Use native MD5 implementation | |MG_ENABLE_SSI | 1 | Enable serving SSI files by `mg_http_serve_dir()` | |MG_ENABLE_DIRLIST | 0 | Enable directory listing | |MG_ENABLE_CUSTOM_RANDOM | 0 | Provide custom RNG function `mg_random()` | |MG_ENABLE_PACKED_FS | 0 | Enable embedded FS support | |MG_IO_SIZE | 2048 | Granularity of the send/recv IO buffer growth | |MG_MAX_RECV_BUF_SIZE | (3 * 1024 * 1024) | Maximum recv buffer size | |MG_MAX_HTTP_HEADERS | 40 | Maximum number of HTTP headers | |MG_ENABLE_LINES | undefined | If defined, show source file names in logs | NOTE: the `MG_IO_SIZE` constant also sets maximum UDP message size, see [issues/907](https://github.com/cesanta/mongoose/issues/907) for details. If the application uses large UDP messages, increase the `MG_IO_SIZE` limit accordingly. ## Custom build A custom build should be used for cases which is not covered by the existing architecture options (e.g., an embedded architecture that uses some proprietary RTOS and network stack). In order to build on such systems, follow the outline below: 1. Add `-DMG_ARCH=MG_ARCH_CUSTOM` to your build flags. 2. Create a file called `mongoose_custom.h`, with defines and includes that are relevant to your platform. Mongoose uses `bool` type, `MG_DIRSEP` define, and optionally other structures like `DIR *` depending on the functionality you have enabled - see previous section. Below is an example: ```c #include #include #define MG_DIRSEP '/' #define MG_INT64_FMT "%lld" #define MG_ENABLE_SOCKET 0 // Disable BSD socket API, implement your own ``` 3. This step is optional. If you have disabled BSD socket API, your build is going to fail due to several undefined symbols. Create `mongoose_custom.c` and implement the following functions (take a look at `src/sock.c` for the reference implementation): ```c struct mg_connection *mg_connect(struct mg_mgr *mgr, const char *url, mg_event_handler_t fn, void *fn_data) { // implement this! } void mg_connect_resolved(struct mg_connection *c) { // implement this! } struct mg_connection *mg_listen(struct mg_mgr *mgr, const char *url, mg_event_handler_t fn, void *fn_data) { // implement this! } void mg_mgr_poll(struct mg_mgr *mgr, int ms) { // implement this! } int mg_send(struct mg_connection *c, const void *buf, size_t len) { // implement this! } ``` ## Minimal HTTP server This example is a simple static HTTP server that serves current directory: ```c #include "mongoose.h" static void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { struct mg_http_serve_opts opts = {.root_dir = "."}; // Serve local dir if (ev == MG_EV_HTTP_MSG) mg_http_serve_dir(c, ev_data, &opts); } int main(int argc, char *argv[]) { struct mg_mgr mgr; mg_mgr_init(&mgr); // Init manager mg_http_listen(&mgr, "http://localhost:8000", fn, &mgr); // Setup listener for (;;) mg_mgr_poll(&mgr, 1000); // Event loop mg_mgr_free(&mgr); // Cleanup return 0; } ``` ## Minimal TCP echo server This example is a simple TCP echo server that listens on port 1234: ```c #include "mongoose.h" static void cb(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_READ) { mg_send(c, c->recv.buf, c->recv.len); // Echo received data back mg_iobuf_del(&c->recv, 0, c->recv.len); // And discard it } } int main(int argc, char *argv[]) { struct mg_mgr mgr; mg_mgr_init(&mgr); // Init manager mg_listen(&mgr, "tcp://0.0.0.0:1234", cb, &mgr); // Setup listener for (;;) mg_mgr_poll(&mgr, 1000); // Event loop mg_mgr_free(&mgr); // Cleanup return 0; } ``` # API Reference ## Core ### struct mg\_addr ```c struct mg_addr { uint16_t port; // TCP or UDP port in network byte order uint32_t ip; // IP address in network byte order uint8_t ip6[16]; // IPv6 address bool is_ip6; // True when address is IPv6 address }; ``` This structure contains network address; it can be considered as a Mongoose equivalent for sockets `sockaddr` structure. ### struct mg\_mgr ```c struct mg_mgr { struct mg_connection *conns; // List of active connections struct mg_dns dns4; // DNS for IPv4 struct mg_dns dns6; // DNS for IPv6 int dnstimeout; // DNS resolve timeout in milliseconds unsigned long nextid; // Next connection ID void *userdata; // Arbitrary user data pointer }; ``` Event management structure that holds a list of active connections, together with some housekeeping information. ### struct mg\_connection ```c struct mg_connection { struct mg_connection *next; // Linkage in struct mg_mgr :: connections struct mg_mgr *mgr; // Our container struct mg_addr peer; // Remote address. For listeners, local address void *fd; // Connected socket, or LWIP data unsigned long id; // Auto-incrementing unique connection ID struct mg_iobuf recv; // Incoming data struct mg_iobuf send; // Outgoing data mg_event_handler_t fn; // User-specified event handler function void *fn_data; // User-specified function parameter mg_event_handler_t pfn; // Protocol-specific handler function void *pfn_data; // Protocol-specific function parameter char label[50]; // Arbitrary label void *tls; // TLS specific data unsigned is_listening : 1; // Listening connection unsigned is_client : 1; // Outbound (client) connection unsigned is_accepted : 1; // Accepted (server) connection unsigned is_resolving : 1; // Non-blocking DNS resolve is in progress unsigned is_connecting : 1; // Non-blocking connect is in progress unsigned is_tls : 1; // TLS-enabled connection unsigned is_tls_hs : 1; // TLS handshake is in progress unsigned is_udp : 1; // UDP connection unsigned is_websocket : 1; // WebSocket connection unsigned is_hexdumping : 1; // Hexdump in/out traffic unsigned is_draining : 1; // Send remaining data, then close and free unsigned is_closing : 1; // Close and free the connection immediately unsigned is_readable : 1; // Connection is ready to read unsigned is_writable : 1; // Connection is ready to write }; ``` A connection - either a listening connection, or an accepted connection, or an outbound connection. ### mg\_mgr\_init() ```c void mg_mgr_init(struct mg_mgr *mgr); ``` Initialize event manager structure: - Set a list of active connections to NULL - Set default DNS servers for IPv4 and IPv6 - Set default DNS lookup timeout Parameters: - `mgr` - a pointer to `mg_mgr` structure that needs to be initialized Return value: none Usage example: ```c struct mg_mgr mgr; mg_mgr_init(&mgr); ``` ### mg\_mgr\_poll() ```c void mg_mgr_poll(struct mg_mgr *mgr, int ms); ``` Perform a single poll iteration. For each connection in the `mgr->conns` list: - See if there is incoming data. If there is, read it into the `c->recv` buffer, send `MG_EV_READ` event - See if there is data in the `c->send` buffer, and write it, send `MG_EV_WRITE` event - If a connection is listening, accept an incoming connection if any, and send `MG_EV_ACCEPT` event to it - Send `MG_EV_POLL` event Parameters: - `mgr` - an event manager to use - `ms` - a timeout in milliseconds Return value: none Usage example: ```c while (running == true) mg_mgr_poll(&mgr, 1000 /* 1 sec */); ``` ### mg\_mgr\_free() ```c void mg_mgr_free(struct mg_mgr *mgr); ``` Close all connections, and free all resources. Parameters: - `mgr` - an event manager to cleanup Return value: none Usage example: ```c struct mg_mgr mgr; mg_mgr_init(&mgr); while (running == true) mg_mgr_poll(&mgr, 1000); // Event loop mg_mgr_free(&mgr); ``` ### mg\_listen() ```c struct mg_connection *mg_listen(struct mg_mgr *mgr, const char *url, mg_event_handler_t fn, void *fn_data); ``` Create a listening connection, append this connection to `mgr->conns`. Parameters: - `mgr` - an event manager to use - `url` - a URL. Specifies local IP address and port to listen on, e.g. `tcp://127.0.0.1:1234` or `udp://0.0.0.0:9000` - `fn` - an event handler function - `fn_data` - an arbitrary pointer, which will be passed as `fn_data` when an event handler is called. This pointer is also stored in a connection structure as `c->fn_data` Return value: created connection, or `NULL` on error. Usage example: ```c struct mg_connection *c = mg_listen(&mgr, "tcp://127.0.0.1:8080", fn, NULL); ``` ### mg\_connect() ```c struct mg_connection *mg_connect(struct mg_mgr *mgr, const char *url, mg_event_handler_t fn, void *fn_data); ``` Create an outbound connection, append this connection to `mgr->conns`. Parameters: - `mgr` - An event manager to use - `url` - A URL, specifies remote IP address/port to connect to, e.g. `http://a.com` - `fn` - An event handler function - `fn_data` - An arbitrary pointer, which will be passed as `fn_data` when an event handler is called. This pointer is also stored in a connection structure as `c->fn_data` Return value: created connection, or `NULL` on error. Note: This function does not connect to peer, it allocates required resources and starts connect process. Once peer is really connected, `MG_EV_CONNECT` event is sent to connection event handler. Usage example: ```c struct mg_connection *c = mg_connect(&mgr, "http://example.org", fn, NULL); ``` ### mg\_send() ```c int mg_send(struct mg_connection *c, const void *data, size_t size); ``` Append `data` of size `size` to the `c->send` buffer. Return number of bytes appended. Note: This function does not push data to the network. It only appends data to the output buffer. The data is being sent when `mg_mgr_poll()` is called. If `mg_send()` is called multiple times, the output buffer grows. Parameters: - `c` - A connection pointer - `data` - A pointer to data to append to the send buffer - `size` - A data size Return value: `true` if data appended successfully and `false` otherwise Usage example: ```c mg_send(c, "hi", 2); // Append string "hi" to the output buffer ``` ### mg\_printf() ```c int mg_printf(struct mg_connection *, const char *fmt, ...); ``` Same as `mg_send()`, but formats data using `printf()` semantics. Return number of bytes appended to the output buffer. Parameters: - `c` - a connection pointer - `fmt` - a format string in `printf` semantics Return value: number of bytes appended to the output buffer. Usage example: ```c mg_printf(c, "Hello, %s!", "world"); // Add "Hello, world!" to output buffer ``` ### mg\_vprintf() ```c int mg_vprintf(struct mg_connection *, const char *fmt, va_list ap); ``` Same as `mg_printf()`, but takes `va_list` argument as a parameter. Parameters: - `c` - A connection pointer - `fmt` - A format string in `printf` semantics - `ap` - An arguments list Return value: Number of bytes appended to the output buffer. Usage example: ```c void foo(struct mg_connection *c, const char *fmt, ...) { va_list ap; va_start(ap, fmt); mg_vprintf(c, fmt, ap); va_end(ap); } ``` ### mg\_straddr ```c char *mg_straddr(struct mg_connection *c, char *buf, size_t len); ``` Write stringified IP address, associated with given connection to `buf` (maximum size `len`) Parameters: - `c` - A connection pointer - `buf` - A pointer to a buffer that will hold stringified address - `len` - A buffer size Return value: `buf` value Usage example: ```c char buf[1024]; mg_straddr(c, buf, sizeof(buf)); // `buf` is now IP address string, like "127.0.0.1:8080" ``` ### mg\_mkpipe() ```c struct mg_connection *mg_mkpipe(struct mg_mgr *, mg_event_handler_t, void *); ``` Create a "pipe" connection which is safe to pass to a different task/thread and used to wake up event manager from a different task. These functions are designed to implement multi-threaded support, to handle two common use cases: - There are multiple consumer connections, e.g. connected websocket clients. A server constantly pushes some data to all of them. In this case, a data producer task should call `mg_mgr_wakeup()` as soon as more data is produced. A pipe's event handler should push data to all client connection. Use `c->label` to mark client connections. - In order to serve a request, a long blocking operation should be performed. In this case, request handler assigns some marker to `c->label` and then spawns a handler task and gives a pipe to a handler task. A handler does its job, and when data is ready, wakes up a manager. A pipe's event handler pushes data to a marked connection. Another task can wake up a sleeping event manager (in `mg_mgr_poll()` call) using `mg_mgr_wakeup()`. When an event manager is woken up, a pipe connection event handler function receives `MG_EV_READ` event. Parameters: - `mgr` - An event manager - `fn` - A pointer to event handler function - `ud` - A user data pointer. It will be passed to `fn` as `fn_data` parameter Return value: Pointer to created connection or `NULL` in case of error Usage example: see [examples/multi-threaded](https://github.com/cesanta/mongoose/tree/master/examples/multi-threaded). ### mg\_mgr\_wakeup() ```c void mg_mgr_wakeup(struct mg_connection *pipe); ``` Wake up an event manager that sleeps in `mg_mgr_poll()` call. This function must be called from a separate task/thread. Parameters: Parameters: - `pipe` - a special connection created by the `mg_mkpipe()` call Return value: None Usage example: see [examples/multi-threaded](https://github.com/cesanta/mongoose/tree/master/examples/multi-threaded). ## HTTP ### struct mg\_http\_header ```c struct mg_http_header { struct mg_str name; // Header name struct mg_str value; // Header value }; ``` Structure represents HTTP header, like `Content-Type: text/html`. `Content-Type` is a header name and `text/html` is a header value. ### struct mg\_http\_message ```c struct mg_http_message { struct mg_str method, uri, query, proto; // Request/response line struct mg_http_header headers[MG_MAX_HTTP_HEADERS]; // Headers struct mg_str body; // Body struct mg_str message; // Request line + headers + body }; ``` Structure represents the HTTP message. ### mg\_http\_listen() ```c struct mg_connection *mg_http_listen(struct mg_mgr *mgr, const char *url, mg_event_handler_t fn, void *fn_data); ``` Create HTTP listener. Parameters: - `mgr` - An event manager - `url` - A URL, specifies local IP address and port to listen on, e.g. `http://0.0.0.0:8000` - `fn` - An event handler function - `fn_data` - An arbitrary pointer, which will be passed as `fn_data` when an event handler is called. This pointer is also stored in a connection structure as `c->fn_data` Return value: Pointer to created connection or `NULL` in case of error Usage example: ```c struct mg_connection *c = mg_http_listen(&mgr, "0.0.0.0:8000", fn, arg); if (c == NULL) fatal_error("Cannot create listener"); ``` ### mg\_http\_connect() ```c struct mg_connection *mg_http_connect(struct mg_mgr *, const char *url, mg_event_handler_t fn, void *fn_data); ``` Create HTTP client connection. Note: This function does not connect to peer; it allocates required resources and starts connect process. Once peer is really connected `MG_EV_CONNECT` event is sent to connection event handler. Parameters: - `mgr` - An event manager - `url` - A URL, specifies remote URL, e.g. `http://google.com` - `fn` - An event handler function - `fn_data` - An arbitrary pointer, which will be passed as `fn_data` when an event handler is called. This pointer is also stored in a connection structure as `c->fn_data` Return value: Pointer to created connection or `NULL` in case of error Usage example: ```c struct mg_connection *c = mg_http_connect(&mgr, "http://google.com", fn, NULL); if (c == NULL) fatal_error("Cannot create connection"); ``` ### mg\_http\_get\_request\_len() ```c int mg_http_get_request_len(const unsigned char *buf, size_t buf_len); ``` Get length of request. The length of request is a number of bytes till the end of HTTP headers. It does not include length of HTTP body. Parameters: - `buf` - A pointer to a buffer with request - `buf_len` - Buffer length Return value: -1 on error, 0 if a message is incomplete, or the length of request Usage example: ```c const char *buf = "GET /test \n\nGET /foo\n\n"; int req_len = mg_http_get_request_len(buf, strlen(buf)); // req_len == 12 ``` ### mg\_http\_parse() ```c int mg_http_parse(const char *s, size_t len, struct mg_http_message *hm); ``` Parse string request into `mg_http_message` structure Parameters: - `s` - A request string - `len` - A request string length - `hm` - A pointer to a structure to store parsed request Return value: request length (see `mg_http_get_request_len()`) Usage example: ```c struct mg_http_message hm; const char *buf = "GET / HTTP/1.0\n\n"; if (mg_http_parse(buf, strlen(buf), &hm) > 0) { /* success */ } ``` ### mg\_http\_printf\_chunk() ```c void mg_http_printf_chunk(struct mg_connection *c, const char *fmt, ...); ``` Write a chunk of data in chunked encoding format, using `printf()` semantic. Parameters: - `c` - A connection pointer - `fmt` - A string, format specified in `printf` semantics Return value: None Usage example: ```c mg_http_printf_chunk(c, "Hello, %s!", "world"); ``` ### mg\_http\_write\_chunk() ```c void mg_http_write_chunk(struct mg_connection *c, const char *buf, size_t len); ``` Write a chunk of data in chunked encoding format. Parameters: - `c` - A connection pointer - `buf` - Data to write - `len` - Data length Return value: None Usage example: ```c mg_http_write_chunk(c, "hi", 2); ``` ### mg\_http\_delete\_chunk() ```c void mg_http_delete_chunk(struct mg_connection *c, struct mg_http_message *hm); ``` Remove chunk specified from input buffer. Parameters: - `c` - A connection pointer - `hm` - Chunk to delete Return value: None Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_HTTP_CHUNK) { struct mg_http_message *hm = (struct mg_http_message *) ev_data; mg_http_delete_chunk(c, hm); // Remove received chunk } } ``` ### mg\_http\_serve\_dir() ```c struct mg_http_serve_opts { const char *root_dir; // Web root directory, must be non-NULL const char *ssi_pattern; // SSI file name pattern, e.g. #.shtml const char *extra_headers; // Extra HTTP headers to add in responses }; void mg_http_serve_dir(struct mg_connection *c, struct mg_http_message *hm, const struct mg_http_serve_opts *opts); ``` Serve static files according to the given options. Note that in order to enable SSI, set a `-DMG_ENABLE_SSI=1` build flag. Parameters: - `c` - Connection to use - `hm` - HTTP message, that should be served - `opts` - Serve options. Note that `opts.root_dir` can optionally accept extra comma-separated `uri=path` pairs, see example below Return value: None Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_HTTP_MSG) { struct mg_http_message *hm = (struct mg_http_message *) ev_data; struct mg_http_serve_opts opts; memset(&opts, 0, sizeof(opts)); opts.root_dir = "/var/www,/conf=/etc"; // Serve /var/www. URIs starting with /conf are served from /etc mg_http_serve_dir(c, hm, &opts); } } ``` ### mg\_http\_serve\_file() ```c void mg_http_serve_file(struct mg_connection *c, struct mg_http_message *hm, const char *path, struct mg_http_serve_opts *opts); ``` Serve static file. Note that the `extra_headers` must end with `\r\n`. Parameters: - `c` - Connection to use - `hm` - HTTP message to serve - `path` - Path to file to serve - `opts` - Serve options Return value: None Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_HTTP_MSG) { struct mg_http_message *hm = (struct mg_http_message *) ev_data; struct mg_http_serve_opts opts = {.mime_types = "png=image/png", .extra_headers = "AA: bb\r\nCC: dd\r\n"}; mg_http_serve_file(c, hm, "a.png", &opts); // Send file } } ``` ### mg\_http\_reply() ```c void mg_http_reply(struct mg_connection *c, int status_code, const char *headers, const char *body_fmt, ...); ``` Send simple HTTP response using `printf()` semantic. This function formats response body according to a `body_fmt`, and automatically appends a correct `Content-Length` header. Extra headers could be passed via `headers` parameter. Parameters: - `c` - Connection to use - `status_code` - An HTTP response code - `headers` - Extra headers, default NULL. If not NULL, must end with `\r\n` - `fmt` - A format string for the HTTP body, in a printf semantics Return value: None Usage examples: Send a simple JSON response: ```c mg_http_reply(c, 200, "Content-Type: application/json\r\n", "{\"result\": %d}", 123); ``` Send JSON response using [mjson](https://github.com/cesanta/mjson) library: ```c char *json = NULL; mjson_printf(mjson_print_dynamic_buf, &json, "{%Q:%d}", "name", 123); mg_http_reply(c, 200, "Content-Type: application/json\r\n", "%s", json); free(json); ``` Send a 302 redirect: ```c mg_http_reply(c, 302, "Location: /\r\n", ""); ``` Send error: ```c mg_http_reply(c, 403, "", "%s", "Not Authorized\n"); ``` ### mg\_http\_get\_header() ```c struct mg_str *mg_http_get_header(struct mg_http_message *hm, const char *name); ``` Get HTTP header value Parameters: - `hm` - HTTP message to look for header - `name` - Header name Return value: HTTP header value or `NULL` if not found Usage example: ```c // Mongoose event handler void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_HTTP_MSG) { struct mg_http_message *hm = (struct mg_http_message *) ev_data; struct mg_str *s = mg_http_get_header(hm, "X-Extra-Header"); if (s != NULL) { mg_http_reply(c, 200, "", "Holly molly! Header value: %.*s", (int) s->len, s->ptr); } else { mg_http_reply(c, 200, "", "Oh no, header is not set..."); } } } ``` ### mg\_http\_get\_var() ```c int mg_http_get_var(const struct mg_str *var, const char *name, char *buf, int len); ``` Decode HTTP variable Parameters: - `var` - HTTP request body - `name` - Variable name - `buf` - Buffer to write decoded variable - `len` - Buffer size Return value: Length of decoded variable. A zero or negative value means error Usage example: ```c char buf[100] = ""; mg_http_get_var(&hm->body, "key1", buf, sizeof(buf)) { ``` ### mg\_http\_creds() ```c void mg_http_creds(struct mg_http_message *hm, char *user, size_t userlen, char *pass, size_t passlen); ``` Fetch authentication credential from the request, and store into the `user`, `userlen` and `pass`, `passlen` buffers. The credentials are looked up in the following order: - from the `Authorization` HTTP header, - Basic auth fills both user and pass - Bearer auth fills only pass - from the `access_token` cookie, fills pass - from the `?access_token=...` query string parameter, fills pass If none is found, then both user and pass are set to empty nul-terminated strings. Parameters: - `hm` - HTTP message to look for credentials - `user` - buffer to receive user name - `userlen` - size of `user` buffer - `pass` - buffer to receive password - `passlen` - size of `pass` buffer Return value: None Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_HTTP_MSG) { struct mg_http_message *hm = (struct mg_http_message *) ev_data; char user[100], pass[100]; mg_http_creds(hm, user, sizeof(user), pass, sizeof(pass)); // "user" is now user name and "pass" is now password from request } } ``` ### mg\_http\_match\_uri() ```c bool mg_http_match_uri(const struct mg_http_message *hm, const char *glob); ``` Check if HTTP request matches a given glob pattern. Parameters: - `hm` - HTTP message to match - `glob` - Pattern Return value: True if HTTP request matches a given glob pattern; false otherwise Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_HTTP_MSG) { struct mg_http_message *hm = (struct mg_http_message *) ev_data; if (mg_http_match_uri(hm, "/secret")) { mg_http_reply(c, 200, NULL, "Very big secret!"); } else { mg_http_reply(c, 200, NULL, "hello world.."); } } } ``` ### mg\_http\_upload() ```c int mg_http_upload(struct mg_connection *c, struct mg_http_message *hm, const char *dir); ``` Handle file upload. See [file upload example](https://github.com/cesanta/mongoose/tree/master/examples/file-uploads/). This function expects a series of POST requests with file data. POST requests should have `name` and `offset` query string parameters set: ```text POST /whatever_uri?name=myfile.txt&offset=1234 HTTP/1.0 Content-Length: 5 hello ``` - `name` - A mandatory query string parameter, specifies a file name. It it created in the `dir` directory - `offset` - An optional parameter, default `0`. If it set to `0`, or omitted, then a file gets truncated before write. Otherwise, the body of the POST request gets appended to the file - Server must call `mg_http_upload()` when `/whatever_uri` is hit The expected usage of this API function follows: - A client splits a file into small enough chunks, to ensure that a chunk fits into the server's RAM - Then, each chunk is POST-ed to the server with using URI, like `/some_uri?name=FILENAME&offset=OFFSET` - Initial OFFSET is `0`, and subsequent offsets are non-zero - Each chunk gets appended to the file - When the last chunk is POST-ed, upload finishes - POST data must not be encoded in any way; it it saved as-is Parameters: - `c` - Connection to use - `hm` - POST message, containing parameters described above - `dir` - Path to directory Return value: Request body length or negative value on error Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_HTTP_MSG) { struct mg_http_message *hm = (struct mg_http_message *) ev_data; mg_http_upload(c, hm, "."); // Upload to current folder } ``` ### mg\_http\_bauth() ```c void mg_http_bauth(struct mg_connection *c, const char *user, const char *pass); ``` Write a Basic `Authorization` header to the output buffer. Parameters: - `c` - Connection to use - `user` - User name - `pass` - Password Return value: None Usage example: ```c mg_http_bauth(c, "user_name", "password") // "user_name:password" is now in output buffer ``` ### mg\_http\_next\_multipart() ```c // Parameter for mg_http_next_multipart struct mg_http_part { struct mg_str name; // Form field name struct mg_str filename; // Filename for file uploads struct mg_str body; // Part contents }; ``` ```c size_t mg_http_next_multipart(struct mg_str body, size_t offset, struct mg_http_part *part); ``` Parse the multipart chunk in the `body` at a given `offset`. An initial `offset` should be 0. Fill up parameters in the provided `part`, which could be NULL. Return offset to the next chunk, or 0 if there are no more chunks. Parameters: - `body`- Message body - `offset` - Start offset - `part` - Pointer to `struct mg_http_part` to fill Return value: offset to the next chunk, or 0 if there are no more chunks. See [File upload tutorial](../tutorials/file-uploads/) for full usage example. Usage example: ```c struct mg_http_part part; if(mg_http_next_multipart(body, 0 /* begin */, &part)) { // Use part } ``` ## Websocket ### struct mg\_ws\_message ```c struct mg_ws_message { struct mg_str data; // Websocket message data uint8_t flags; // Websocket message flags }; ``` Structure represents the WebSocket message. ### mg\_ws\_connect() ```c struct mg_connection *mg_ws_connect(struct mg_mgr *mgr, const char *url, mg_event_handler_t fn, void *fn_data, const char *fmt, ...); ``` Create client Websocket connection. Note: this function does not connect to peer, it allocates required resources and starts the connect process. Once peer is really connected, the `MG_EV_CONNECT` event is sent to connection event handler. Parameters: - `mgr` - Event manager to use - `url` - Specifies remote URL, e.g. `http://google.com` - `opts` - MQTT options, with client ID, QoS, etc - `fn` - An event handler function - `fn_data` - An arbitrary pointer, which will be passed as `fn_data` when an event handler is called. This pointer is also stored in a connection structure as `c->fn_data` - `fmt` - printf-like format string for additional HTTP headers, or NULL Return value: Pointer to created connection or `NULL` on error Usage example: ```c struct mg_connection *c = mg_ws_connect(&mgr, "ws://test_ws_server.com:1000", handler, NULL, "%s", "Sec-WebSocket-Protocol: echo\r\n"); if(c == NULL) fatal("Cannot create connection"); ``` ### mg\_ws\_upgrade() ```c void mg_ws_upgrade(struct mg_connection *c, struct mg_http_message *, const char *fmt, ...); ``` Upgrade given HTTP connection to Websocket. The `fmt` is a printf-like format string for the extra HTTP headers returned to the client in a Websocket handshake. Set `fmt` to `NULL` if no extra headers need to be passed. Parameters: - `c` - Connection to use - `hm` - HTTP message - `fmt` - printf-like format string for additional HTTP headers, or NULL Return value: None Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_HTTP_MSG) { struct mg_http_message *hm = (struct mg_http_message *) ev_data; mg_ws_upgrade(c, hm, NULL); // Upgrade HTTP to WS } } ``` ### mg\_ws\_send() ```c size_t mg_ws_send(struct mg_connection *c, const char *buf, size_t len, int op); ``` Send data to websocket peer Parameters: - `c` - Connection to use - `buf` - Data to send - `len` - Data size - `op` - Websocket message type Return value: sent bytes count Possible Websocket message type: ```c #define WEBSOCKET_OP_CONTINUE 0 #define WEBSOCKET_OP_TEXT 1 #define WEBSOCKET_OP_BINARY 2 #define WEBSOCKET_OP_CLOSE 8 #define WEBSOCKET_OP_PING 9 #define WEBSOCKET_OP_PONG 10 ``` Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_WS_OPEN) { struct mg_http_message *hm = (struct mg_http_message *) ev_data; mg_ws_send(c, "opened", 6, WEBSOCKET_OP_BINARY); // Send "opened" to web socket connection } } ``` ### mg\_ws\_wrap() ```c size_t mg_ws_wrap(struct mg_connection *c, size_t len, int op) ``` Convert data in output buffer to WebSocket format. Useful then implementing protocol over WebSocket See [examples/mqtt-over-ws-client](https://github.com/cesanta/mongoose/tree/master/examples/mqtt-over-ws-client) for full example. Parameters: - `c` - Connection to use - `len` - Bytes count to convert - `op` - Websocket message type (see `mg_ws_send`) Return value: New size of connection output buffer Usage example: ```c size_t len = c->send.len; // Store output buffer len mg_mqtt_login(c, s_url, &opts); // Write MQTT login message mg_ws_wrap(c, c->send.len - len, WEBSOCKET_OP_BINARY); // Wrap it into WS ``` ## SNTP ### mg_sntp_connect() ```c struct mg_connection *mg_sntp_connect(struct mg_mgr *mgr, const char *url, mg_event_handler_t fn, void *fn_data) ``` Connect SNTP server. Parameters: - `mgr` - Event manager to use - `url` - Specifies remote URL, `time.google.com` if NULL. - `fn` - An event handler function - `fn_data` - An arbitrary pointer, which will be passed as `fn_data` when an event handler is called. This pointer is also stored in a connection structure as `c->fn_data` Return value: Pointer to created connection or `NULL` on error Usage example: ```c static void sntp_cb(struct mg_connection *c, int ev, void *evd, void *fnd) { if (ev == MG_EV_SNTP_TIME) { // Time received struct timeval *tv = (struct timeval *tv)evd; } } ... mg_sntp_connect(mgr&, NULL /* connect to time.google.com */, sntp_cb, NULL); ``` ### mg_sntp_send() ```c void mg_sntp_send(struct mg_connection *c, unsigned long utc) ``` Send time request to SNTP server. Note: This app can't send SNTP request more often than every one hour. Parameters: - `c` - Connection to use - `utc` - Current time, used to verify if new request is possible. Return value: None Usage example: ```c mg_sntp_send(c, (unsigned long) time(NULL)); ``` ## MQTT ### struct mg\_mqtt\_opts ```c struct mg_mqtt_opts { struct mg_str client_id; // Client id struct mg_str will_topic; // Will Topic struct mg_str will_message; // Will Message uint8_t qos; // Quality of service bool will_retain; // Retain last will bool clean; // Use clean session, 0 or 1 uint16_t keepalive; // Keep-alive timer in seconds }; ``` Structure used to specify MQTT connection options. ### struct mg\_mqtt\_message ```c struct mg_mqtt_message { struct mg_str topic; // Topic struct mg_str data; // Message data }; ``` Structure represents the MQTT message. ### mg\_mqtt\_connect() ```c struct mg_connection *mg_mqtt_connect(struct mg_mgr *mgr, const char *url, struct mg_mqtt_opts *opts, mg_event_handler_t fn, void *fn_data); ``` Create client MQTT connection. Note: This function does not connect to peer; it allocates required resources and starts connect process. Once peer is really connected `MG_EV_CONNECT` event is sent to connection event handler. Parameters: - `mgr` - Event manager to use - `url` - Specifies remote URL, e.g. `http://google.com` - `opts` - MQTT options, with client ID, qos, etc - `fn` - An event handler function - `fn_data` - An arbitrary pointer, which will be passed as `fn_data` when an event handler is called. This pointer is also stored in a connection structure as `c->fn_data` Return value: pointer to created connection or `NULL` on error Usage example: ```c void fn(struct mg_connection *c, int ev, void *evd, void *fnd) { char *buf = (char *) fnd; if (ev == MG_EV_MQTT_OPEN) { // Connection ready } } mg_mqtt_connect(&mgr, "mqtt://test.org:1883", NULL, handler, NULL); ``` ### mg\_mqtt\_listen() ```c struct mg_connection *mg_mqtt_listen(struct mg_mgr *mgr, const char *url, mg_event_handler_t fn, void *fn_data); ``` Create MQTT listener. Parameters: - `mgr` - Event manager to use - `url` - Specifies local IP address and port to listen on, e.g. `mqtt://0.0.0.0:1883` - `fn` - An event handler function - `fn_data` - An arbitrary pointer, which will be passed as `fn_data` when an event handler is called. This pointer is also stored in a connection structure as `c->fn_data` Return value: Pointer to created connection or `NULL` on error Usage example: ```c struct mg_connection *c = mg_mqtt_listen(&mgr, "0.0.0.0:1883", fn, arg); if (c == NULL) fatal("Cannot create connection"); ``` ### mg\_mqtt\_login ```c void mg_mqtt_login(struct mg_connection *c, const char *url, struct mg_mqtt_opts *opts); ``` Send MQTT login request. Parameters: - `c` - Connection to use - `url` - URL, containing user name and password to use - `opts` - Request options Return value: None Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *evd, void *fnd) { char *buf = (char *) fnd; if (ev == MG_EV_MQTT_OPEN) { struct mg_mqtt_opts opts = {.qos = 1, .will_topic = mg_str("my topic"), .will_message = mg_str("goodbye")}; mg_mqtt_login(c, s_url, &opts); } } ``` ### mg\_mqtt\_pub() ```c void mg_mqtt_pub(struct mg_connection *c, struct mg_str *topic, struct mg_str *data, int qos, bool retain); ``` Publish message. Parameters: - `c` - Connection to use - `topic` - Topic to publish data - `data` - Data to publish - `qos` - Required QoS - `retain` - Retain flag Return value: None Usage example: ```c struct mg_str topic = mg_str("topic"); struct mg_str data = mg_str("data"); mg_mqtt_pub(c, &topic, &data, 1, false); ``` ### mg\_mqtt\_sub() ```c void mg_mqtt_sub(struct mg_connection *c, struct mg_str *topic, int qos); ``` Subscribe to topic. Parameters: - `c` - Connection to use - `topic` - Topic to subscribe - `qos` - Required QoS Return value: none ```c struct mg_str topic = mg_str("topic"); mg_mqtt_sub(c, &topic, 1); ``` ### mg\_mqtt\_next\_sub() ```c size_t mg_mqtt_next_sub(struct mg_mqtt_message *msg, struct mg_str *topic, uint8_t *qos, size_t pos); ``` Traverse list of subscribed topics. Used to implement MQTT server when `MQTT_CMD_SUBSCRIBE` is received. Initial position `pos` should be 4. Parameters: - `mgs` - MQTT message - `topic` - Pointer to `mg_str` to receive topic - `qos` - Pointer to `uint8_t` to receive QoS - `pos` - Position to list from Return value: Next position, or 0 when done Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *evd, void *fnd) { if (ev == MG_EV_MQTT_CMD) { struct mg_mqtt_message *mm = (struct mg_mqtt_message *) ev_data; if (mm->cmd == MQTT_CMD_SUBSCRIBE) { size_t pos = 4; uint8_t qos; struct mg_str topic; // Iterate over all subscribed topics while ((pos = mg_mqtt_next_sub(mm, &topic, &qos, pos)) > 0) { LOG(LL_INFO, ("SUB [%.*s]", (int) topic.len, topic.ptr)); } } } } ``` ### mg\_mqtt\_next\_unsub() ```c size_t mg_mqtt_next_unsub(struct mg_mqtt_message *msg, struct mg_str *topic, size_t pos); ``` Same as `mg_mqtt_next_sub()`, but for unsubscribed topics. The difference is that there is no QoS in unsubscribe request. Parameters: - `mgs` - MQTT message - `topic` - Pointer to `mg_str` to receive topic - `pos` - Position from which to list Return value: Next position, or 0 when done Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *evd, void *fnd) { if (ev == MG_EV_MQTT_CMD) { struct mg_mqtt_message *mm = (struct mg_mqtt_message *) ev_data; if (mm->cmd == MQTT_CMD_UNSUBSCRIBE) { size_t pos = 4; struct mg_str topic; if (mm->cmd == MQTT_CMD_UNSUBSCRIBE) { // Iterate over all unsubscribed topics while ((pos = mg_mqtt_next_unsub(mm, &topic, pos)) > 0) { LOG(LL_INFO, ("SUB [%.*s]", (int) topic.len, topic.ptr)); } } } } } ``` ### mg\_mqtt\_send_header() ```c void mg_mqtt_send_header(struct mg_connection *c, uint8_t cmd, uint8_t flags, uint32_t len); ``` Send MQTT command header. Useful in MQTT server implementation. Command can be one of the following value: ```c #define MQTT_CMD_CONNECT 1 #define MQTT_CMD_CONNACK 2 #define MQTT_CMD_PUBLISH 3 #define MQTT_CMD_PUBACK 4 #define MQTT_CMD_PUBREC 5 #define MQTT_CMD_PUBREL 6 #define MQTT_CMD_PUBCOMP 7 #define MQTT_CMD_SUBSCRIBE 8 #define MQTT_CMD_SUBACK 9 #define MQTT_CMD_UNSUBSCRIBE 10 #define MQTT_CMD_UNSUBACK 11 #define MQTT_CMD_PINGREQ 12 #define MQTT_CMD_PINGRESP 13 #define MQTT_CMD_DISCONNECT 14 ``` Parameters: - `c` - Connection to use - `cmd` - Command (see above) - `flags` - Command flags - `len` - Size of the following command Return value: None Usage example: ```c // Mongoose events handler void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) { if (ev == MG_EV_MQTT_CMD) { struct mg_mqtt_message *mm = (struct mg_mqtt_message *) ev_data; if (mm->cmd == MQTT_CMD_CONNECT) { uint8_t response[] = {0, 0}; mg_mqtt_send_header(c, MQTT_CMD_CONNACK, 0, sizeof(response)); // Send acknowledgement mg_send(c, response, sizeof(response)); } } } ``` ### mg\_mqtt\_ping() ```c void mg_mqtt_ping(struct mg_connection *c); ``` Send `MQTT_CMD_PINGREQ` command via `mg_mqtt_send_header` Parameters: - `c` - Connection to use Return value: None Usage example: ```c // Send periodic pings to all WS connections static void timer_fn(void *arg) { struct mg_mgr *mgr = (struct mg_mgr *) arg; for (struct mg_connection *c = mgr->conns; c != NULL; c = c->next) { if (c->is_websocket) mg_mqtt_ping(c); } } ``` ### mg_mqtt_parse ```c int mg_mqtt_parse(const uint8_t *buf, size_t len, struct mg_mqtt_message *m); ``` Parse buffer and fill `mg_mqtt_message` structure if buffer contain MQTT message. Parameters: - `buf` - buffer with MQTT message to parse - `len` - buffer size - `m` - pointer to `mg_mqtt_message` structure to receive parsed message Return value: `MQTT_OK` if message successfully parsed, `MQTT_INCOMPLETE` if message isn't fully received and `MQTT_MALFORMED` if message has wrong format. Usage example: ```c // Iterate over all MQTT frames contained in buf, len struct mg_mqtt_message mm; while ((mg_mqtt_parse(buf, len, &mm)) == 0) { switch (mm.cmd) { case MQTT_CMD_CONNACK: ... } buf += mm.dgram.len; len -= mm.dgram.len; } ``` ## TLS ### struct mg\_tls\_opts ```c struct mg_tls_opts { const char *ca; // CA certificate file. For both listeners and clients const char *crl; // Certificate Revocation List. For clients const char *cert; // Certificate const char *certkey; // Certificate key const char *ciphers; // Cipher list struct mg_str srvname; // If not empty, enables server name verification }; ``` TLS initialisation structure: - `ca` - Certificate Authority. Can be a filename or a string. Used to verify a certificate that the other end sends to us. If NULL, then certificate checking is disabled - `crl` - Certificate Revocation List. Can be a filename or a string. Used to verify a certificate that the other end sends to us. If NULL, then certificate revocation checking is disabled - `cert` - Our own certificate. Can be a filename, or a string. If NULL, then we don't authenticate with the other peer - `certkey` - A key for a `cert`. Sometimes, a certificate and its key are bundled in a single .pem file, in which case the values for `cert` and `certkey` could be the same - `ciphers` - A list of allowed ciphers - `srvname` - Enable server name verification NOTE: if both `ca` and `cert` are set, then so-called two-way TLS is enabled, when both sides authenticate with each other. Usually, server-side connections set both `ca` and `cert`, whilst client-side - only `ca`. ### mg\_tls\_init() ```c void mg_tls_init(struct mg_connection *c, struct mg_tls_opts *opts); ``` Initialise TLS on a given connection. NOTE: mbedTLS implementation uses `mg_random` as RNG. The `mg_random` can be overridden by setting `MG_ENABLE_CUSTOM_RANDOM` and defining your own `mg_random()` implementation. Parameters: - `c` - Connection, for which TLS should be initialized - `opts` - TLS initialization parameters Return value: None Usage example: ```c struct mg_tls_opts opts = {.cert = "ca.pem"}; mg_tls_init(c, &opts); ``` ## Timer ### struct mg\_timer ```c struct mg_timer { int period_ms; // Timer period in milliseconds int flags; // Possible flags values below void (*fn)(void *); // Function to call void *arg; // Function argument unsigned long expire; // Expiration timestamp in milliseconds struct mg_timer *next; // Linkage in g_timers list }; #define MG_TIMER_REPEAT 1 // Call function periodically, otherwise run once #define MG_TIMER_RUN_NOW 2 // Call immediately when timer is set ``` Timer structure. Describes a software timer. Timer granularity is the same as the `mg_mgr_poll()` timeout argument in the main event loop. ### mg\_timer\_init() ```c void mg_timer_init(struct mg_timer *t, unsigned long ms, unsigned flags, void (*fn)(void *), void *fn_data); ``` Setup a timer. Parameters: - `t` - Pointer to `mg_timer` that should be initialized - `ms` - An interval in milliseconds - `flags` - Timer flags bitmask, `MG_TIMER_REPEAT` and `MG_TIMER_RUN_NOW` - `fn` - Function to invoke - `fn_data` - Function argument Return value: None Usage example: ```c void timer_fn(void *data) { // ... } struct mg_timer timer; mg_timer_init(&timer, 1000 /* 1sec */, MG_TIMER_REPEAT, timer_fn, NULL); // A timer gets initialized and linked into the internal timers list ``` ### mg\_timer\_free() ```c void mg_timer_free(struct mg_timer *t); ``` Free timer, remove it from the internal timers list. Parameters: - `t` - Timer to free Return value: None Usage example: ```c struct mg_timer timer; // ... mg_timer_free(&timer); ``` ### mg\_timer\_poll() ```c void mg_timer_poll(unsigned long uptime_ms); ``` Traverse list of timers and call them if current timestamp `uptime_ms` is past the timer's expiration time. Note, that `mg_mgr_poll` function internally calls `mg_timer_poll`; therefore, in most cases it is unnecessary to call it explicitly. Parameters: - `uptime_ms` - current timestamp Return value: None Usage example: ```c unsigned long now = mg_millis(); mg_timer_poll(now); ``` ## Time ### mg\_time() ``` double mg_time(void); ``` Return current time as UNIX epoch, using `double` value for sub-second accuracy. Parameters: None Return value: Current time Usage example: ```c double now = mg_time() ``` ### mg\_millis() ```c unsigned long mg_millis(void); ``` Return current uptime in milliseconds. Parameters: None Return value: Current uptime Usage example: ```c unsigned long uptime = mg_millis(); ``` ### mg\_usleep() ```c void mg_usleep(unsigned long usecs); ``` Block thread execution for a given number of microseconds. Parameters: - `usecs` - number of microseconds to block thread Return value: None Usage example: ```c mg_usleep(1000000 /* 1 sec */) ``` ## String ### struct mg\_str ```c struct mg_str { const char *ptr; // Pointer to string data size_t len; // String len }; ``` This structure represent an arbitrary chunk of memory, not necessarily zero-terminated. This is a "mongoose string", and it gets used extensively in the codebase instead of C zero-terminated strings. For example, when an HTTP request is received, Mongoose created a `struct mg_http_message` which has a collection of `struct mg_str` pointing to request method, URI, headers, and so on. This way, Mongoose avoids any heap allocations and does not modify the received buffer - instead, it uses `struct mg_str` to describe various parts of HTTP request. Same goes with many other cases. NOTE: since `ptr` is not necessarily zero-terminated, do not use libc string functions against it - like `strlen()` or `sscanf()`. ### mg\_str() ```c struct mg_str mg_str(const char *s) ``` Create Mongoose string from NULL-terminated C-string. This function doesn't duplicate provided string, and stores pointer within created `mg_str` structure. Note, that is you have problems in C++ (constructor shadowing), there is `mg_str_s` synonym for this function. Parameters: - `s` - Pointer to NULL-terminated string to store in created mg_str Return value: Created Mongoose string Usage example: ```c struct mg_str str = mg_str("Hello, world!); ``` ### mg\_str\_n() ```c struct mg_str mg_str_n(const char *s, size_t n); ``` Create Mongoose string from C-string `s` (can be non-NULL terminated, length is specified in `n`). Note: This function doesn't duplicate provided string, but stores pointer within created `mg_str` structure. Parameters: - `s` - Pointer to string to store in created `mg_str` - `n` - String length Return value: Created Mongoose string Usage example: ```c struct mg_str str = mg_str_n("hi", 2); ``` ### mg\_casecmp() ```c int mg_casecmp(const char *s1, const char *s2); ``` Case insensitive compare two NULL-terminated strings. Parameters: - `s1`, `s2` - Pointers to strings to compare Return value: Zero if strings are equal, more than zero if first argument is greater then second, and less than zero otherwise Usage example: ```c if (mg_casecmp("hello", "HELLO") == 0) { // Strings are equal } ``` ### mg\_ncasecmp() ```c int mg_ncasecmp(const char *s1, const char *s2, size_t len); ``` Case insensitive compare two C-strings, not more than `len` symbols or until meet `\0` symbol. Parameters: - `s1`, `s2` - Pointers to strings to compare - `len` - Maximum length to compare Return value: Zero if strings are equal, more than zero if first argument is greater then second, and less than zero otherwise Usage example: ```c if (mg_ncasecmp("hello1", "HELLO2", 5) == 0) { // Strings are equal } ``` ### mg\_vcmp() ```c int mg_vcmp(const struct mg_str *s1, const char *s2); ``` Compare mongoose string and C-string. Parameters: - `s1` - Pointer to Mongoose string to compare - `s2` - Pointer to C-string to compare Return value: 0 if strings are equal, more than zero if first argument is greater then second, and less than zero otherwise Usage example: ```c struct mg_str str = mg_str("hello"); if (mg_vcmp(str, "hello") == 0) { // Strings are equal } ``` ### mg\_vcasecmp() ```c int mg_vcasecmp(const struct mg_str *str1, const char *str2); ``` Case insensitive compare mongoose string and C-string. Parameters: - `str1` - Mongoose string to compare - `str2` - C-string to compare Return value: Zero if strings are equal, more than zero if first argument is greater then second, and less than zero otherwise Usage example: ```c struct mg_str str = mg_str("hello"); if (mg_vcasecmp(str, "HELLO") == 0) { // Strings are equal } ``` ### mg\_strcmp() ```c int mg_strcmp(const struct mg_str str1, const struct mg_str str2); ``` Compare two mongoose strings. Parameters: - `str1`, `str2` - Pointers to Mongoose strings to compare Return value: Zero if strings are equal, more than zero if first argument is greater then second, and less than zero otherwise Usage example: ```c struct mg_str str1 = mg_str("hello"); struct mg_str str2 = mg_str("hello"); if (mg_strcmp(str1, str2) == 0) { // Strings are equal } ``` ### mg\_strdup() ```c struct mg_str mg_strdup(const struct mg_str s); ``` Duplicate provided string. Return new string or `MG_NULL_STR` on error. Note: This function allocates memory for returned string. You may need to free it using `free` function. Parameters: - `s` - Mongoose string to duplicate Return value: Duplicated string Usage example: ```c struct mg_str str1 = mg_str("hello"); struct mg_str str2 = mg_strdup(str1); //... free(str1.ptr); ``` ### mg\_strstr() ```c const char *mg_strstr(const struct mg_str haystack, const struct mg_str needle) ``` Search for `needle` substring in `haystack` string. Parameters: - `haystack` - Mongoose sting to search for substring - `needle` - Mongoose string to search Return value: pointer to `needle` occurrence within `haystack` or `NULL` if not found. Usage example: ```c struct mg_str str = mg_str("Hello, world"); struct mg_str sub_str = mg_str("world"); if (mg_strstr(str, sub_str) != NULL) { // Found } ``` ### mg\_strstrip() ```c struct mg_str mg_strstrip(struct mg_str s) ``` Remove heading and trailing whitespace from mongoose string `s`. Parameters: - `s` - Mongoose string for trimming Return value: Input string Usage example: ```c struct mg_str str = mg_strstrip(mg_str(" Hello, world ")); if (mg_vcmp(str, "Hello, world") == 0) { // Strings are equal } ``` ### mg\_globmatch() ```c bool mg_globmatch(const char *pattern, size_t p_len, const char *s, size_t s_len); ``` Check if string `s` (limited to `s_len` symbols) matches glob pattern `pattern`, (limited to `p_len` symbols). The glob pattern matching rules are as follows: - `?` matches any single character - `*` matches zero or more characters except `/` - `#` matches zero or more characters - any other character matches itself Parameters: - `pattern` - Pattern to match for - `p_len` - Pattern length - `s` - String to match - `s_len` - String length Return value: `true` if matches, `false` otherwise Usage example: ```c struct mg_str pattern = mg_str("#, ?????"); struct mg_str s = mg_str("Hello, world"); if (mg_globmatch(pattern.ptr, pattern.len, s.otr, s.len)) { // Match } ``` ### mg\_commalist() ```c bool mg_commalist(struct mg_str *s, struct mg_str *k, struct mg_str *v); ``` Parse string `s`, which is a comma-separated list of entries. An entry could be either an arbitrary string, which gets stored in `v`, or a `KEY=VALUE` which gets stored in `k` and `v` respectively. IMPORTANT: This function modifies `s` by pointing to the next entry. Parameters: - `s` - String to search for entry - `k` - Pointer to `mg_str` to receive entry key - `v` - Pointer to `mg_str` to receive entry value Return value: `true` if entry is found, `false` otherwise Usage example: ```c struct mg_str k, v, s = mg_str("a=333,b=777"); while (mg_commalist(&s, &k, &v)) // This loop output: printf("[%.*s] set to [%.*s]\n", // [a] set to [333] (int) k.len, k.ptr, (int) v.len, v.ptr); // [b] set to [777] ``` ### mg\_hexdump() ```c char *mg_hexdump(const void *buf, int len); ``` Hexdump binary data `buf`, `len` into malloc-ed buffer and return it. It is a caller's responsibility to free() returned pointer. Parameters: - `buf` - Data to hexdump - `len` - Data length Return value: malloc-ed buffer with hexdumped data Usage example: ```c char arr[] = "\0x1\0x2\0x3"; char *hex = mg_hexdump(arr, sizeof(arr)); LOG(LL_INFO, ("%s", hex)); // Output "0000 01 02 03 00"; free(hex); ``` ### mg\_hex() ```c char *mg_hex(const void *buf, size_t len, char *dst); ``` Hex-encode binary data `buf`, `len` into a buffer `dst` and nul-terminate it. The output buffer must be at least 2 x `len` + 1 big. Parameters: - `buf` - Data to hex-encode - `len` - Data length - `dst` - Pointer to output buffer Return value: `dst` pointer. The encoded characters are lowercase Usage example: ```c char data[] = "\x1\x2\x3"; char buf[sizeof(data)*2]; char *hex = mg_hex(data, sizeof(data) - 1, buf); LOG(LL_INFO, ("%s", hex)); // Output "010203"; free(hex); ``` ### mg\_unhex() ```c void mg_unhex(const char *buf, size_t len, unsigned char *to); ``` Hex-decode string `buf`, `len` into a buffer `to`. The `to` buffer should be at least `lsn` / 2 big. Parameters: - `buf` - Data to hex-decode - `len` - Data length - `to` - Pointer to output buffer Return value: None Usage example: ```c char data[] = "010203"; char *buf[sizeof(data)/2]; char *hex = mg_unhex(data, sizeof(data) - 1, buf); // buf is now [1,2,3] free(hex); ``` ### mg\_unhexn() ```c unsigned long mg_unhexn(const char *s, size_t len); ``` Parse `len` characters of the hex-encoded string `s`. The maximum value of `len` is the width of the `long` x 2, for example on 32-bit platforms it is 8. Parameters: - `s` - String to parse - `len` - String length Return value: Return parsed value Usage example: ```c char data[] = "010203"; char *buf[sizeof(data)/2]; unsigned long val = mg_unhex(data, sizeof(data) - 1); // val is now 123 ``` ### mg\_asprintf() ```c int mg_asprintf(char **buf, size_t size, const char *fmt, ...); ``` Print message specified by printf-like format string `fmt` into a buffer pointed by `buf` of size `size`. If `size` is large enough to hold the whole message, then a message is stored in `*buf`. If it does not fit, then a large enough buffer is allocated to hold a message, and `buf` is changed to point to that buffer. Parameters: - `buf` - Pointer to pointer to output buffer - `size` - Pre-allocated buffer size - `fmt` - printf-like format string Return value: Number of bytes printed Usage example: ```c char buf[1024], *pbuf = &buf; mg_asprintf(&pbuf, sizeof(buf), "Hello, %s!", "world"); // buf is now "Hello, world!" ``` ### mg\_vasprintf() ```c int mg_vasprintf(char **buf, size_t size, const char *fmt, va_list ap); ``` Same as `mg_asprintf()` but uses `va_list` argument. Parameters: - `buf` - Pointer to pointer to output buffer - `size` - Pre-allocated buffer size - `fmt` - printf-like format string - `ap` - Parameters list Return value: Number of bytes printed Usage example: ```c void foo(const char *fmt, ...) { va_list ap; va_start(ap, fmt); char buf[1024], *pbuf = buf; mg_vasprintf(&pbuf, sizeof(buf), fmt, ap); va_end(ap); printf("%s\n", buf); } // ... foo("Hello, %s!", "world"); // Print "Hello, world! ``` ### mg\_to64() ```c int64_t mg_to64(struct mg_str str); ``` Parse 64-bit integer value held by string `s`. Parameters: - `str` - String to parse Return value: Parsed value Usage example: ```c int64_t val = mg_to64(mg_str("123")); // Val is now 123 ``` ### mg\_aton() ```c bool mg_aton(struct mg_str str, struct mg_addr *addr); ``` Parse IP address held by `str` and store it in `addr`. Parameters: - `str` - String to parse, for example `1.2.3.4`, `[::1]`, `01:02::03` - `addr` - Pointer to `mg_addr` string to receive parsed value Return value: `true` on success, `false` otherwise Usage example: ```c struct mg_addr addr; if (mg_aton(mg_str("127.0.0.1"), &addr)) { // addr is now binary representation of 127.0.0.1 IP address } ``` ### mg\_ntoa() ```c char *mg_ntoa(const struct mg_addr *addr, char *buf, size_t len); ``` Stringify IP address `ipaddr` into a buffer `buf`, `len` Parameters: - `addr` - Address to stringify - `buf` - Pointer to output buffer - `len` - Output buffer size Return value: `buf` value Usage example: ```c char buf[100]; mg_ntoa(&c->peer, buf, sizeof(buf)); ``` ## Utility ### mg\_call() ```c void mg_call(struct mg_connection *c, int ev, void *ev_data); ``` Send `ev` event to `c` event handler. This function is useful then implementing your own protocol. Parameters: - `c` - Connection to send event - `ev` - Event to send - `ev_data` - Additional event parameter Return value: None Usage example: ```c // In a timer callback, send MG_EV_USER event to all connections static void timer_fn(void *arg) { struct mg_mgr *mgr = (struct mg_mgr *) arg; for (struct mg_connection *c = mgr->conns; c != NULL; c = c->next) { mg_call(c, MG_EV_USER, "hi!"); } } ``` ### mg\_error() ```c void mg_error(struct mg_connection *c, const char *fmt, ...); ``` Send `MG_EV_ERROR` to connection event handler with error message formatted using printf semantics. Parameters: - `c` - Connection to send event - `fmt` - Format string in `printf` semantics Return value: None Usage example: ```c mg_error(c, "Operation failed, error code: %d", errno); ``` ### mg\_md5\_init() ```c void mg_md5_init(mg_md5_ctx *c); ``` Initialize context for MD5 hashing. Parameters: - `c` - Pointer to `mg_md5_ctx` structure to initialize Return value: None Usage example: ```c mg_md5_ctx ctx; mg_md5_init(&ctx); ``` ### mg\_md5\_update() ```c void mg_md5_update(mg_md5_ctx *c, const unsigned char *data, size_t len); ``` Hash `len` bytes of data pointed by `data` using MD5 algorithm. Parameters: - `c` - MD5 context - `data` - Data to hash - `len` - Data length Return value: None Usage example: ```c mg_md5_ctx ctx; // Context initialization // ... mg_md5_update(&ctx, "data", 4); // hash "data" string mg_md5_update(&ctx, "more data", 9); // hash "more data" string ``` ### mg\_md5\_final() ```c void mg_md5_final(mg_md5_ctx *c, unsigned char buf[16]); ``` Get current MD5 hash for context. Parameters: - `c` - MD5 context - `buf` - Pointer to buffer to write MD5 hash value Return value: None Usage example: ```c mg_md5_ctx ctx; // Context initialization // ... unsigned char buf[16]; mg_md5_final(&ctx, buf); // `buf` is now MD5 hash ``` ### mg\_sha1\_init() ```c void mg_sha1_init(mg_sha1_ctx *c); ``` Initialize context for calculating SHA1 hash Parameters: - `c` - pointer to `mg_sha1_ctx` structure to initialize Return value: none Usage example: ```c mg_sha1_ctx ctx; mg_sha1_init(&ctx); ``` ### mg\_sha1\_update() ```c void mg_sha1_update(mg_sha1_ctx *c, const unsigned char *data, size_t len); ``` Hash `len` bytes of `data` using SHA1 algorithm. Parameters: - `c` - SHA1 context - `data` - Data to hash - `len` - Data length Return value: None Usage example: ```c mg_sha1_ctx ctx; // Context initialization // ... mg_sha1_update(&ctx, "data", 4); // hash "data" string mg_sha1_update(&ctx, "more data", 9); // hash "more data" string ``` ### mg\_sha1\_final() ```c void mg_sha1_final(unsigned char digest[20], mg_sha1_ctx *c); ``` Get current SHA1 hash for context. Parameters: - `c` - SHA1 context - `digest` - Pointer to buffer to receive hash value Return value: None Usage example: ```c mg_sha1_ctx ctx; // Context initialization // ... unsigned char buf[20]; mg_sha1_final(buf, &ctx); // `buf` is now SHA1 hash ``` ### mg\_base64\_update() ```c int mg_base64_update(unsigned char p, char *out, int pos); ``` Encode `p` byte to base64 and write result into `out` buffer starting with `pos` position. Parameters: - `p` - Byte to encode - `out` - Pointer to buffer to write result - `pos` - Position in output buffer to write result Return value: New position for further operations Usage example: ```c char buf[10]; mg_base64_update((unsigned char)"a", buf, 0); // Encode "a" into base64 and write it to the beginning of buf ``` ### mg\_base64\_final() ```c int mg_base64_final(char *buf, int pos); ``` Add base64 finish mark and `\0` symbol to `buf` at `pos` position. Parameters: - `buf` - Pointer to buffer to write finish mark - `pos` - Position to write Return value: New position for further operations ```c char buf[10]; int pos; // ... mg_base64_final(buf, pos); ``` ### mg\_base64\_encode() ```c int mg_base64_encode(const unsigned char *p, int n, char *to); ``` Encode `n` bytes data pointed by `p` using base64 and write result into `to`. Parameters: - `p` - Pointer to data to encode - `n` - Data length - `to` - Pointer to buffer to write result Return value: Written symbols number Usage example: ```c char buf[128]; mg_base64_encode((uint8_t *) "abcde", 5, buf); // buf is now YWJjZGU= ``` ### mg\_base64\_decode() ```c int mg_base64_decode(const char *src, int n, char *dst); ``` Decode `n` bytes of base64-ed `src` and write it to `dst`. Parameters: - `src` - Data to decode - `n` - Data length - `dst` - Pointer to output buffer Return value: Number of written symbols. Usage example: ```c char buf[128]; mg_base64_decode("Q2VzYW50YQ==", 12, buf); // buf is now "Cesanta" ``` ### mg\_file\_read() ```c char *mg_file_read(const char *path, size_t *sizep); ``` Read file contents into a nul-terminated malloc-ed string. It is a caller's responsibility to free() a returned pointer. If `sizep` is not NULL, it will return a file size in bytes. Return `NULL` on error. Parameters: - `path` - Path to file to read - `sizep` - Pointer to `size_t` to receive file size Return value: File contents, see function description Usage example: ```c size_t file_size; char* data = mg_file_read("myfile.txt", &file_size); if (data != NULL) { // `data` is now pointer to information readen from file and `file_size` is it size. } free(data); ``` ### mg\_file\_write() ```c bool mg_file_write(const char *path, const void *buf, size_t len); ``` Write data to a file. The write is atomic, i.e. data gets written to a temporary file first, then `rename()-ed` to a destination file name. Parameters: - `path` - Path to file - `buf` - Data to write - `len` - Data length Return value: `true` if written, `false` otherwise Usage example: ```c char data[] = "Hello, world!"; if(mg_file_write("my_file.txt", data, sizeof(data) - 1)) { // File contains "Hello, world!" string } ``` ### mg\_file\_printf() ```c int mg_file_printf(const char *path, const char *fmt, ...); ``` Write into a file `path` using `printf()` semantics. This function prints data to a temporary in-memory buffer first, then calls `mg_file_write()`. Parameters: - `path`- path to file - `fmt` - Format string in `printf()` semantics Return value: `true` on success, `false` otherwise ```c if (mg_file_printf("my_file.txt", "Hello, %s!", "world") { // File contains "Hello, world!" string } ``` ### mg\_random() ```c void mg_random(void *buf, size_t len); ``` Fill in buffer `buf`, `len` with random data. Note: Mongoose uses this function for TLS and some other routines that require RNG (random number generator). It is possible to override a built-in `mg_random()` by specifying a `MG_ENABLE_CUSTOM_RANDOM=1` build preprocessor constant. Parameters: - `buf` - Pointer to buffer to receive random data - `len` - Buffer size Return value: None Usage example: ```c char buf[10]; mg_random(buf, sizeof(buf)); // `buf` is now random bytes ``` ### mg\_ntohs() ```c uint16_t mg_ntohs(uint16_t net); ``` Convert `uint16_t` value to host order. Parameters: - `net` - 16-bit value in network order Return value: 16-bit value in host order Usage example: ```c uint16_t val = mg_ntohs(0x1234); ``` ### mg\_ntohl() ```c uint32_t mg_ntohl(uint32_t net); ``` Convert `uint32_t` value to host order. Parameters: - `net` - 32-bit value in network order Return value: 32-bit value in host order Usage example: ```c uint32_t val = mg_ntohl(0x12345678); ``` ### mg\_ntohs() ```c uint16_t mg_htons(uint16_t h); ``` Convert `uint16_t` value to network order. Parameters: - `h` - 16-bit value in host order Return value: 16-bit value in network order Usage example: ```c uint16_t val = mg_htons(0x1234); ``` ### mg\_htonl() ```c uint32_t mg_ntohl(uint32_t h); ``` Convert `uint32_t` value to network order. Parameters: - `h` - 32-bit value in host order Return value: 32-bit value in network order Usage example: ```c uint32_t val = mg_htonl(0x12345678); ``` ### mg\_crc32() ```c uint32_t mg_crc32(uint32_t crc, const char *buf, size_t len); ``` Calculate CRC32 checksum for a given buffer. An initial `crc` value should be `0`. Parameters: - `crc` - Initial CRC value - `buf` - Data to calculate CRC32 - `len` - Data size Return value: Calculated CRC32 checksum Usage example: ```c char data[] = "hello"; uint32_t crc = mg_crc32(0, data, sizeof(data)); ``` ### mg\_check\_ip\_acl() ```c int mg_check_ip_acl(struct mg_str acl, uint32_t remote_ip); ``` Check IPv4 address `remote_ip` against the IP ACL `acl`. Parameters: Parameters: - `acl` - an ACL string, e.g. `-0.0.0.0/0,+1.2.3.4` - `remote_ip` - IPv4 address in network byte order Return value: 1 if `remote_ip` is allowed, 0 if not, and <0 if `acl` is invalid Usage example: ```c if (mg_check_ip_acl(mg_str("-0.0.0.0/0,+1.2.3.4"), c->peer.ip) != 1) { LOG(LL_INFO, ("NOT ALLOWED!")); } ``` ### mg\_url\_decode() ```c int mg_url_decode(const char *s, size_t n, char *to, size_t to_len, int form); ``` Decode URL-encoded string `s` and write it into `to` buffer. Parameters: - `s` - String to encode - `n` - String to encode length - `to` - Pointer to output buffer - `to_len` - Output buffer size - `form` - If non-zero, then `+` is decoded as whitespace. Return value: Decoded bytes count or negative value on error Usage example: ```c char url[] = "eexample.org%2Ftest"; char buf[1024]; mg_url_encode(url, sizeof(url) - 1, buf, sizeof(buf), 0); // buf is now "example.org/test" ``` ### mg\_url\_encode ```c size_t mg_url_encode(const char *s, size_t n, char *buf, size_t len); ``` Encode `s` string to URL-encoding and write encoded string into `buf`. Parameters: - `s` - String to encode - `n` - String to encode length - `buf` - Output buffer - `len` - Output buffer size Return value: Number of characters written to `buf` Usage example: ```c char url[] = "example.org/test"; char buf[1024]; mg_url_encode(url, sizeof(url) - 1, buf, sizeof(buf)); // buf is now "example.org%2Ftest" ``` ## IO Buffers IO buffer, described by the `struct mg_iobuf`, is a simple data structure that inserts or deletes chunks of data at arbitrary offsets and grows/shrinks automatically. ### struct mg\_iobuf ```c struct mg_iobuf { unsigned char *buf; // Pointer to stored data size_t size; // Total size available size_t len; // Current number of bytes }; ``` Generic IO buffer. The `size` specifies an allocation size of the data pointed by `buf`, and `len` specifies number of bytes currently stored. ### mg\_iobuf\_init() ```c int mg_iobuf_init(struct mg_iobuf *io, size_t size); ``` Initialize IO buffer, allocate `size` bytes. Parameters: - `io` - Pointer to `mg_iobuf` structure to initialize - `size` - Amount of bytes to allocate Return value: 1 on success, 0 on allocation failure Usage example: ```c struct mg_iobuf io; if (mg_iobuf_init(&io)) { // io successfully initialized } ``` ### mg\_iobuf\_resize() ```c int mg_iobuf_resize(struct mg_iobuf *io, size_t size); ``` Resize IO buffer, set the new size to `size`. The `io->buf` pointer could change after this, for example if the buffer grows. If `size` is 0, then the `io->buf` is freed and set to NULL, and both `size` and `len` are set to 0. Parameters: - `io` - iobuf to resize - `size` - New size Return value: 1 on success, 0 on allocation failure Usage example: ```c struct mg_iobuf io; // IO buffer initialization // ... if (mg_iobuf_resize(&io, 1024)) { // New io size is 1024 bytes } ``` ### mg\_iobuf\_free() ```c void mg_iobuf_free(struct mg_iobuf *io); ``` Free memory pointed by `io->buf` and set to NULL. Both `size` and `len` are set to 0. Parameters: - `io` - iobuf to free Return value: None Usage example: ```c struct mg_iobuf io; // IO buffer initialization // ... // Time to cleanup mg_iobuf_free(&io); ``` ### mg\_iobuf\_add() ```c size_t mg_iobuf_add(struct mg_iobuf *io, size_t offset, const void *buf, size_t len, size_t align); ``` Insert data buffer `buf`, `len` at offset `offset`. The iobuf is expanded if required. The resulting `io->size` is always aligned to the `align` byte boundary; therefore, to avoid memory fragmentation and frequent reallocations, set `align` to a higher value. Parameters: - `io` - iobuf to add data - `offset` - Offset to add data - `buf` - Data to add - `len` - Data length - `align` - Align boundary Return value: new `io` length Usage example: ```c struct mg_iobuf io; mg_iobuf_init(&io, 0); // Empty buffer mg_iobuf_add(&io, 0, "hi", 2, 512); // io->len is 2, io->size is 512 ``` ### mg\_iobuf\_del() ```c size_t mg_iobuf_del(struct mg_iobuf *io, size_t offset, size_t len); ``` Delete `len` bytes starting from `offset`, and shift the remaining bytes. If `len` is greater than `io->len`, nothing happens, so such call is silently ignored. Parameters: - `io` - iobuf to delete data - `offset` - Start offset - `len` - Amount of bytes to delete Return value: New `io` length Usage example: ```c struct mg_iobuf io; mg_iobuf_init(&io, 0); // Empty buffer mg_iobuf_add(&io, 0, "hi", 2, 512); // io->len is 2, io->size is 512 // ... mg_iobuf_del(&io, 0, "hi", 2, 512); // io->len is 0, io->size is still 512 ``` ## URL ### mg\_url\_port() ```c unsigned short mg_url_port(const char *url); ``` Return port for given URL Parameters: - `url` - URL to extract port Return value: Port for given URL or `0` if URL doesn't contain port and there isn't default port for URL protocol Usage example: ```c unsigned short port1 = mg_url_port("htts://myhost.com") // port1 is now 443 (default https port) unsigned short port2 = mg_url_port("127.0.0.1:567") // port2 is now 567 ``` ### mg\_url\_is_ssl() ```c int mg_url_is_ssl(const char *url); ``` Check if given URL uses encrypted scheme Parameters: - `url` - URL to check Return value: `0` is given URL uses encrypted scheme and non-zero otherwise Usage example: ```c if (mg_url_is_ssl("https://example.org") == 0) { // scheme is encrypted } ``` ### mg\_url\_host() ```c struct mg_str mg_url_host(const char *url); ``` Extract host name from given URL. Parameters: - `url` - a URL string Return value: host name Usage example: ```c struct mg_str a = mg_url_host("https://my.example.org:1234"); // a == "my.example.org" struct mg_str b = mg_url_host("tcp://[::1]"); // b == "[::1]" ``` ### mg\_url\_user() ```c struct mg_str mg_url_user(const char *url); ``` Extract user name from given URL. Parameters: - `url` - URL to extract user name Return value: User name or empty string if not found Usage example: ```c struct mg_str user_name = mg_url_user("https://user@password@my.example.org"); // user_name is now "user" ``` ### mg\_url\_pass() ```c struct mg_str mg_url_pass(const char *url); ``` Extract password from given URL. Parameters: - `url` - URL to extract password Return value: Password or empty string if not found Usage example: ```c struct mg_str pwd = mg_url_user("https://user@password@my.example.org"); // pwd is now "password" ``` ### mg\_url\_uri() ```c const char *mg_url_uri(const char *url); ``` Extract URI from given URL. Note, that function returns pointer within `url`; there is no need to free() it explicitly Parameters: - `url` - URL to extract URI Return value: URI or `\` if not found Usage example: ```c const char *uri = mg_url_uri("https://example.org/subdir/subsubdir"); // `uri` is now pointer to "subdir/subsubdir" ``` ## Logging Mongoose provides a set of functions and macros for logging. The application can use these functions for its own purposes as well as the rest of Mongoose API. ### LOG() ```c #define LOG(level, args) ``` General way to log is using `LOG` macro. `LOG` prints to log only is `MG_ENABLE_LOG` macro defined, otherwise is does nothing. This macro has two arguments: log level and information to log. The second argument is a printf-alike format string. Log levels defined as: ```c enum { LL_NONE, LL_ERROR, LL_INFO, LL_DEBUG, LL_VERBOSE_DEBUG }; ``` Parameters: - `level` - Log level, see levels above - `args` - Information to log Return value: None Usage example: ```c LOG(LL_ERROR, ("Hello %s!", "world")); // Output "Hello, world" ``` ### mg\_log\_set() ```c void mg_log_set(const char *spec); ``` Set Mongoose logging level. Parameters: - `spec` - String, containing log level, can be one of the following values: - `0` - Disable logging - `1` - Log errors only - `2` - Log errors and info messages - `3` - Log errors, into and debug messages - `4` - Log everything Return value: None It is possible to override log level per source file basis. For example, if there is a file called `foo.c`, and you'd like to set a global level to `2` (info) but increase log level for file foo.c to `debug`, then, a `spec` should look like `"2,foo.c=3"`. There could be several comma-separated overrides. Usage example: ```c mg_log_set("2"); // Set log level to info mg_log_set("2,foo.c=3,bar.c=0"); // Set log level to info, with overrides ``` ### mg\_log\_set\_callback() ```c void mg_log_set_callback(void (*fn)(const void *, size_t, void *), void *fnd); ``` By default, `LOG` writes to standard output stream (aka `stdout`), but this behaviour can be changes via `mg_log_set_callback`. This function allows to set callback, which called once mongoose (or host application) calls `LOG` Parameters: - `fn` - callback function, should be called on logging - `fnd` - user parameter to pass to `fn` Return value: none Usage example: ```c void log_via_printf(const void *buf, size_t len, void *userdata) { (void) userdata; printf("*.%s", buf, len); } // ... mg_log_set_callback(&log_via_printf, NULL); ``` ## Filesystem ### FS virtualisation Mongoose allows to override file i/o operations in order to support different storages, like programmable flash, no-filesystem devices etc. In order to accomplish this, Mongoose provides a `struct mg_fs` API to specify a custom filesystem. In addition to this, Mongoose provides two built-in APIs - a standard POSIX API, and a "packed FS" API. A packed FS allows to embed a filesystem into the application or firmware binary, described below. ```c enum { MG_FS_READ = 1, MG_FS_WRITE = 2, MG_FS_DIR = 4 }; // Filesystem API functions // stat() returns MG_FS_* flags and populates file size and modification time // list() calls fn() for every directory entry, allowing to list a directory struct mg_fs { int (*stat)(const char *path, size_t *size, time_t *mtime); void (*list)(const char *path, void (*fn)(const char *, void *), void *); struct mg_fd *(*open)(const char *path, int flags); // Open file void (*close)(struct mg_fd *fd); // Close file size_t (*read)(void *fd, void *buf, size_t len); // Read file size_t (*write)(void *fd, const void *buf, size_t len); // Write file size_t (*seek)(void *fd, size_t offset); // Set file position }; ``` HTTP server's `struct mg_http_serve_opts` has a `fs` pointer which specifies which filesystem to use when serving static files. By default, `fs` is set to NULL and therefore a standard POSIX API is used. That could be overridden and a packed FS, or any other user-defined custom FS could be used: ```c struct mg_http_serve_opts opts; opts.fs = &mg_fs_posix; mg_http_serve_dir(c, hm, &opts); ``` ### Packed filesystem Packed filesystem allow to "pack" filesystem into single file, for example, into executable or flashable image. This is useful, for example, for implementation of HTTP-server on devices without filesystem. In order to use packed filesystem do the following: 1. Compile file test\pack.c: ```sh $ cc -o pack pack.c ``` 2. Convert list of files into single .c: ```sh $ ./pack file1.data file2.data > fs.c ``` 3. Build your app with fs.c: ```sh $ cc -o my_app my_app.c fs.c ``` 4. In your application code, you can access files using this function:
`const char *mg_unpack(const char *file_name, size_t *size)` or app can also force `mg_http_serve_dir` function to use packed file system: ```c struct mg_http_serve_opts opts; opts.fs = &mg_fs_packed; // Set packed ds as a file system mg_http_serve_dir(c, hm, &opts); ```