2020-12-08 16:26:17 +00:00
|
|
|
# User Guide
|
2020-12-06 21:12:05 +00:00
|
|
|
|
2020-12-08 16:53:35 +00:00
|
|
|
|
2020-12-06 21:12:05 +00:00
|
|
|
## Introduction
|
|
|
|
|
2020-12-08 16:53:35 +00:00
|
|
|
Mongoose is a networking library for C/C++. It implements an event-driven,
|
2020-12-06 21:12:05 +00:00
|
|
|
non-blocking APIs for TCP, UDP, HTTP, WebSocket, MQTT. It has been designed
|
|
|
|
for connecting devices and bringing them online. On the market since 2004, used
|
2021-03-11 13:01:45 +00:00
|
|
|
by a vast number of open source and commercial products - it even runs on the
|
|
|
|
International Space Station! Mongoose makes embedded network programming fast,
|
2020-12-06 21:12:05 +00:00
|
|
|
robust, and easy.
|
|
|
|
|
|
|
|
Mongoose has three basic data structures:
|
|
|
|
|
|
|
|
- `struct mg_mgr` - an event manager that holds all active connections
|
|
|
|
- `struct mg_connection` - describes a connection
|
|
|
|
- `struct mg_iobuf` - describes data buffer (received or sent data)
|
|
|
|
|
|
|
|
Connections could be either 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 a full visibility into the
|
|
|
|
Mongoose's internals.
|
|
|
|
|
|
|
|
An application that uses mongoose should follow a standard pattern of
|
|
|
|
event-driven application:
|
|
|
|
|
2021-08-22 10:51:40 +01:00
|
|
|
**Step 1.** Declare and initialize an event manager:
|
2020-12-06 21:12:05 +00:00
|
|
|
|
2020-12-10 13:26:05 +00:00
|
|
|
```c
|
|
|
|
struct mg_mgr mgr;
|
|
|
|
mg_mgr_init(&mgr);
|
|
|
|
```
|
2020-12-06 21:12:05 +00:00
|
|
|
|
2021-08-22 10:51:40 +01:00
|
|
|
**Step 2.** Create connections. For example, a server application should create listening
|
2020-12-06 21:12:05 +00:00
|
|
|
connections. When any connection is created (listening or outgoing), an
|
|
|
|
event handler function must be specified. An event handler function defines
|
|
|
|
connection's behavior.
|
|
|
|
|
2020-12-10 13:26:05 +00:00
|
|
|
```c
|
|
|
|
struct mg_connection *c = mg_http_listen(&mgr, "0.0.0.0:8000", fn, arg);
|
|
|
|
```
|
2020-12-06 21:12:05 +00:00
|
|
|
|
2021-08-22 10:51:40 +01:00
|
|
|
**Step 3.** Create an event loop by calling `mg_mgr_poll()`:
|
2020-12-06 21:12:05 +00:00
|
|
|
|
2020-12-10 13:26:05 +00:00
|
|
|
```c
|
|
|
|
for (;;) {
|
|
|
|
mg_mgr_poll(&mgr, 1000);
|
|
|
|
}
|
|
|
|
```
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
|
|
`mg_mgr_poll()` iterates over all sockets, accepts new connections, sends and
|
|
|
|
receives data, closes connections and calls event handler functions for the
|
|
|
|
respective events.
|
|
|
|
|
2020-12-10 13:26:05 +00:00
|
|
|
Since the Mongoose's core is not protected against concurrent accesses,
|
2020-12-06 21:12:05 +00:00
|
|
|
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 an
|
2021-05-19 00:19:21 +01:00
|
|
|
`MG_EV_READ` event. The user may send data back by calling one of the output
|
2020-12-06 21:12:05 +00:00
|
|
|
functions, like `mg_send()` or `mg_printf()`. 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. That function
|
|
|
|
must be implemented by the user. Event handler is the key element of the
|
|
|
|
Mongoose application, since it defines the connection's behaviour. This is
|
|
|
|
what an event handler function looks like:
|
|
|
|
|
|
|
|
```c
|
2021-08-20 20:31:32 +01:00
|
|
|
// Event handler function defines connection's behavior
|
2020-12-06 21:12:05 +00:00
|
|
|
static void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) {
|
2021-08-20 20:31:32 +01:00
|
|
|
if (ev == MG_EV_READ) {
|
|
|
|
mg_send(c, c->recv.buf, c->recv.len); // Implement echo server
|
|
|
|
c->recv.len = 0; // Delete received data
|
2020-12-06 21:12:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
- `struct mg_connection *c` - a connection that received an event
|
|
|
|
- `int ev` - an event number, defined in mongoose.h. For example, when data
|
2021-05-19 00:19:21 +01:00
|
|
|
arrives on an inbound connection, ev would be `MG_EV_READ`
|
2020-12-06 21:12:05 +00:00
|
|
|
- `void *ev_data` - points to the event-specific data, and it has a different
|
2021-05-19 00:19:21 +01:00
|
|
|
meaning for different events. For example, for an `MG_EV_READ` event,
|
2020-12-06 21:12:05 +00:00
|
|
|
`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`
|
2020-12-10 13:26:05 +00:00
|
|
|
pointing to structures that hold protocol-specific information
|
2020-12-06 21:12:05 +00:00
|
|
|
- `void *fn_data` - a user-defined pointer for the connection, which is a
|
2020-12-10 13:26:05 +00:00
|
|
|
placeholder for application-specific data
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
|
|
## Events
|
|
|
|
|
2021-03-11 13:01:45 +00:00
|
|
|
Below is the list of events triggered by Mongoose, taken as-is from `mongoose.h`.
|
2020-12-06 21:12:05 +00:00
|
|
|
For each event, a comment describes a meaning of the `ev_data` pointer passed
|
|
|
|
to an event handler:
|
|
|
|
|
|
|
|
```c
|
|
|
|
enum {
|
2021-01-17 11:16:19 +00:00
|
|
|
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_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 *
|
2020-12-06 21:12:05 +00:00
|
|
|
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 `is_udp` flag tells application if
|
|
|
|
that connection is UDP or not. Some flags can be changed by application, for
|
2021-02-02 22:16:48 -05:00
|
|
|
example, `is_draining` flag, if set by an application, tells Mongoose to send
|
2020-12-06 21:12:05 +00:00
|
|
|
the remaining data to peer, and when everything is sent, close the connection.
|
|
|
|
|
2021-02-02 22:16:48 -05:00
|
|
|
User-changeable flags are: `is_hexdumping`, `is_draining`, `is_closing`.
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
|
|
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
|
2021-05-17 17:36:57 +01:00
|
|
|
files to the application's source tree. The `mongoose.c` and `mongoose.h` files
|
|
|
|
are, actually, an amalgamation - a non-amalgamated sources can be found at
|
|
|
|
https://github.com/cesanta/mongoose/tree/master/src
|
2020-12-06 21:12:05 +00:00
|
|
|
|
2021-05-17 17:36:57 +01:00
|
|
|
Mongoose have 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:
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
|
|
```sh
|
2021-05-17 17:36:57 +01:00
|
|
|
$ cc app0.c mongoose.c # Use defaults!
|
|
|
|
$ cc app1.c mongoose.c -D MG_ENABLE_IPV6=1 # Build with IPv6 enabled
|
2021-05-28 23:49:26 +01:00
|
|
|
$ cc app2.c mongoose.c -D MG_ARCH=MG_ARCH_FREERTOS_LWIP # Set architecture
|
2021-07-29 14:21:20 +01:00
|
|
|
$ cc app3.c mongoose.c -D MG_ENABLE_SSI=0 -D MG_ENABLE_LOG=0 # Multiple options
|
2020-12-06 21:12:05 +00:00
|
|
|
```
|
|
|
|
|
2021-05-17 17:36:57 +01:00
|
|
|
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 |
|
2021-05-28 23:49:26 +01:00
|
|
|
|MG_ARCH_FREERTOS_LWIP | All systems with FreeRTOS kernel and LwIP IP stack |
|
2021-05-17 17:36:57 +01:00
|
|
|
|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. They are tunables that include/exclude
|
|
|
|
a certain functionality or change relevant parameters.
|
|
|
|
|
|
|
|
|
2020-12-06 21:12:05 +00:00
|
|
|
Here is a list of build constants and their default values:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
2020-12-06 21:12:05 +00:00
|
|
|
| Name | Default | Description |
|
|
|
|
| ---- | ------- | ----------- |
|
2021-05-17 17:36:57 +01:00
|
|
|
|MG_ENABLE_SOCKET | 1 | Use BSD socket low-level API |
|
|
|
|
|MG_ENABLE_MBEDTLS | 0 | Enable Mbed TLS 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 |
|
2021-07-29 14:21:20 +01:00
|
|
|
|MG_ENABLE_SSI | 1 | Enable serving SSI files by `mg_http_serve_dir()` |
|
|
|
|
|MG_ENABLE_DIRLIST | 0 | Enable directory listing |
|
2021-07-30 13:19:20 +01:00
|
|
|
|MG_ENABLE_CUSTOM_RANDOM | 0 | Provide custom RNG function `mg_random()` |
|
|
|
|
|MG_ENABLE_PACKED_FS | 0 | Enable embedded FS support |
|
2021-07-29 14:21:20 +01:00
|
|
|
|MG_IO_SIZE | 2048 | Granularity of the send/recv IO buffer growth |
|
2021-05-17 17:36:57 +01:00
|
|
|
|MG_MAX_RECV_BUF_SIZE | (3 * 1024 * 1024) | Maximum recv buffer size |
|
|
|
|
|MG_MAX_HTTP_HEADERS | 40 | Maximum number of HTTP headers |
|
2021-07-29 14:21:20 +01:00
|
|
|
|MG_ENABLE_LINES | undefined | If defined, show source file names in logs |
|
2020-12-06 21:12:05 +00:00
|
|
|
|
2021-05-12 11:39:46 +01:00
|
|
|
|
2021-08-20 20:31:32 +01:00
|
|
|
<span class="badge bg-danger">NOTE:</span> the `MG_IO_SIZE` constant also sets
|
|
|
|
maximum UDP message size, see
|
|
|
|
[issues/907](https://github.com/cesanta/mongoose/issues/907) for details. If
|
|
|
|
application uses large UDP messages, increase the `MG_IO_SIZE` limit
|
|
|
|
accordingly.
|
2020-12-17 11:19:13 +00:00
|
|
|
|
2021-01-20 11:28:47 +00:00
|
|
|
## Custom build
|
|
|
|
|
2021-05-17 17:36:57 +01:00
|
|
|
A custom build should be used for cases which is not covered by the
|
|
|
|
existing architecture options. For example, an embedded architecture that
|
|
|
|
uses some proprietary RTOS and network stack. In order to build on such
|
|
|
|
systems, follow the guideline outlined below:
|
2021-05-12 11:39:46 +01:00
|
|
|
|
|
|
|
1. Add `-DMG_ARCH=MG_ARCH_CUSTOM` to your build flags.
|
|
|
|
|
|
|
|
2. Create a file called `mongoose_custom.h`, with defines and includes that
|
2021-01-20 11:28:47 +00:00
|
|
|
are relevant to your platform. Mongoose uses `bool` type, `MG_DIRSEP` define,
|
|
|
|
and optionally other structures like `DIR *` depending on the functionality
|
2021-05-28 23:49:26 +01:00
|
|
|
you have enabled - see previous section. Below is an example:
|
2021-05-12 11:39:46 +01:00
|
|
|
|
|
|
|
```c
|
2021-05-28 23:49:26 +01:00
|
|
|
#include <stdbool.h> // For bool
|
2021-05-12 11:39:46 +01:00
|
|
|
#include <stdarg.h>
|
2021-01-20 11:28:47 +00:00
|
|
|
|
|
|
|
#define MG_DIRSEP '/'
|
2021-05-12 11:39:46 +01:00
|
|
|
#define MG_INT64_FMT "%lld"
|
2021-01-20 11:28:47 +00:00
|
|
|
#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!
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
## Minimal HTTP server
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
|
|
This example is a simple static HTTP server that serves current directory:
|
|
|
|
|
|
|
|
```c
|
|
|
|
#include "mongoose.h"
|
|
|
|
|
2021-08-22 10:51:40 +01:00
|
|
|
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
|
2021-01-02 17:57:51 +00:00
|
|
|
if (ev == MG_EV_HTTP_MSG) mg_http_serve_dir(c, ev_data, &opts);
|
2020-12-06 21:12:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
struct mg_mgr mgr;
|
2021-08-22 10:51:40 +01:00
|
|
|
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
|
2020-12-06 21:12:05 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
## 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) {
|
2021-05-19 00:19:21 +01:00
|
|
|
if (ev == MG_EV_READ) {
|
2020-12-08 16:26:17 +00:00
|
|
|
mg_send(c, c->recv.buf, c->recv.len); // Echo received data back
|
2021-08-28 07:16:38 +01:00
|
|
|
mg_iobuf_del(&c->recv, 0, c->recv.len); // And discard it
|
2020-12-08 16:26:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
struct mg_mgr mgr;
|
|
|
|
mg_mgr_init(&mgr);
|
2021-08-22 10:51:40 +01:00
|
|
|
mg_listen(&mgr, "tcp://0.0.0.0:1234", cb, &mgr);
|
2020-12-08 16:26:17 +00:00
|
|
|
for (;;) mg_mgr_poll(&mgr, 1000);
|
|
|
|
mg_mgr_free(&mgr);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
# API Reference
|
|
|
|
|
|
|
|
## Core
|
|
|
|
|
|
|
|
### struct mg_mgr
|
|
|
|
|
|
|
|
```c
|
2020-12-08 16:53:35 +00:00
|
|
|
struct mg_mgr {
|
2020-12-08 16:26:17 +00:00
|
|
|
struct mg_connection *conns; // List of active connections
|
2021-08-12 12:24:28 +01:00
|
|
|
struct mg_dns dns4; // DNS for IPv4
|
|
|
|
struct mg_dns dns6; // DNS for IPv6
|
2020-12-08 16:26:17 +00:00
|
|
|
int dnstimeout; // DNS resolve timeout in milliseconds
|
2021-08-12 12:24:28 +01:00
|
|
|
unsigned long nextid; // Next connection ID
|
|
|
|
void *userdata; // Arbitrary user data pointer
|
2020-12-08 16:26:17 +00:00
|
|
|
};
|
|
|
|
```
|
|
|
|
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
|
2021-07-19 08:47:38 +01:00
|
|
|
struct mg_addr peer; // Remote address. For listeners, local address
|
2020-12-08 16:26:17 +00:00
|
|
|
void *fd; // Connected socket, or LWIP data
|
2021-05-19 00:19:21 +01:00
|
|
|
unsigned long id; // Auto-incrementing unique connection ID
|
2020-12-08 16:26:17 +00:00
|
|
|
struct mg_iobuf recv; // Incoming data
|
|
|
|
struct mg_iobuf send; // Outgoing data
|
|
|
|
mg_event_handler_t fn; // User-specified event handler function
|
2021-03-11 13:01:45 +00:00
|
|
|
void *fn_data; // User-specified function parameter
|
2020-12-08 16:26:17 +00:00
|
|
|
mg_event_handler_t pfn; // Protocol-specific handler function
|
|
|
|
void *pfn_data; // Protocol-specific function parameter
|
2021-05-19 00:19:21 +01:00
|
|
|
char label[50]; // Arbitrary label
|
2020-12-08 16:26:17 +00:00
|
|
|
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
|
2021-03-11 13:01:45 +00:00
|
|
|
unsigned is_resolving : 1; // Non-blocking DNS resolve is in progress
|
2020-12-08 16:26:17 +00:00
|
|
|
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,
|
2021-03-11 13:01:45 +00:00
|
|
|
or an outbound connection.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
### mg\_mgr_init()
|
|
|
|
|
|
|
|
```c
|
2021-08-12 12:24:28 +01:00
|
|
|
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 `struct mg_mgr` that needs to be initialised
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_mgr mgr;
|
|
|
|
mg_mgr_init(&mgr);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### 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,
|
2021-05-19 00:19:21 +01:00
|
|
|
- See if there is incoming data. If it is, read it into the `c->recv` buffer, send `MG_EV_READ` event
|
2020-12-08 16:26:17 +00:00
|
|
|
- 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
|
|
|
|
|
|
|
|
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 user-specific handler. It parses
|
|
|
|
incoming data and may invoke protocol-specific events like `MG_EV_HTTP_MSG`.
|
|
|
|
|
|
|
|
|
2021-01-23 15:49:55 +08:00
|
|
|
### mg\_mgr\_free()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
```c
|
|
|
|
void mg_mgr_free(struct mg_mgr *mgr);
|
|
|
|
```
|
|
|
|
|
|
|
|
Close all connections, and free all resources.
|
|
|
|
|
|
|
|
|
|
|
|
### 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`.
|
|
|
|
- `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.
|
|
|
|
|
|
|
|
### mg\_connect()
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_connection *mg_connect(struct mg_mgr *mgr, const char *url,
|
|
|
|
mg_event_handler_t fn, void *fn_data);
|
|
|
|
```
|
|
|
|
|
2021-03-11 13:01:45 +00:00
|
|
|
Create an outbound connection, append this connection to `mgr->conns`.
|
2020-12-08 16:26:17 +00:00
|
|
|
- `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.
|
|
|
|
|
|
|
|
|
|
|
|
### 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.
|
|
|
|
|
|
|
|
|
|
|
|
### 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.
|
|
|
|
|
|
|
|
|
|
|
|
### 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.
|
|
|
|
|
|
|
|
|
2021-08-12 12:24:28 +01:00
|
|
|
### mg\_mkpipe()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
```c
|
2021-08-11 19:17:04 +01:00
|
|
|
struct mg_connection *mg_mkpipe(struct mg_mgr *, mg_event_handler_t, void *);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
2021-08-11 19:17:04 +01:00
|
|
|
Create a "pipe" connection which is safe to pass to a different task/thread,
|
2021-08-11 19:24:33 +01:00
|
|
|
and which is 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 assignes 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.
|
2021-08-09 09:43:21 +01:00
|
|
|
|
2021-08-11 19:17:04 +01:00
|
|
|
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.
|
2021-08-09 09:43:21 +01:00
|
|
|
|
|
|
|
See [examples/multi-threaded](../examples/multi-threaded) for a usage example.
|
2021-08-07 17:22:47 +01:00
|
|
|
|
2021-08-12 12:24:28 +01:00
|
|
|
### 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:
|
|
|
|
|
|
|
|
- `pipe` - a special connection created by the `mg_mkpipe()` call
|
|
|
|
|
|
|
|
Return values: none
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
## IO buffers
|
|
|
|
|
|
|
|
### struct mg\_iobuf
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_iobuf {
|
|
|
|
unsigned char *buf;
|
|
|
|
size_t size, len;
|
|
|
|
};
|
|
|
|
```
|
|
|
|
|
|
|
|
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
|
2021-01-29 12:32:34 +00:00
|
|
|
int mg_iobuf_init(struct mg_iobuf *io, size_t size);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
2021-03-11 13:01:45 +00:00
|
|
|
Initialize IO buffer, allocate `size` bytes. Return 1 on success,
|
2021-01-29 12:32:34 +00:00
|
|
|
0 on allocation failure.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
### mg\_iobuf\_resize()
|
|
|
|
|
|
|
|
```c
|
2021-01-29 12:32:34 +00:00
|
|
|
int mg_iobuf_resize(struct mg_iobuf *io, size_t size);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
|
|
|
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.
|
2021-01-29 12:32:34 +00:00
|
|
|
Return 1 on success, 0 on allocation failure.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
### 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.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_iobuf\_append()
|
|
|
|
|
|
|
|
```c
|
|
|
|
size_t mg_iobuf_append(struct mg_iobuf *io, const void *data, size_t data_size, size_t granularity);
|
|
|
|
```
|
|
|
|
|
|
|
|
Append `data` bytes of size `data_size` to the end of the buffer. The buffer
|
|
|
|
is expanded if `data_size` is greater than `io->size - io->len`. If that
|
|
|
|
happens, the `io->buf` can change. The resulting `io->size` is always
|
|
|
|
set to the `granularity` byte boundary. Example:
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_iobuf io;
|
|
|
|
mg_iobuf_init(&io, 0); // Empty buffer
|
|
|
|
mg_iobuf_append(&io, "hi", 2, 1024); // io->len is 2, io->size is 1024
|
|
|
|
```
|
|
|
|
|
2021-08-28 07:16:38 +01:00
|
|
|
### mg\_iobuf\_del()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
```c
|
2021-08-28 07:16:38 +01:00
|
|
|
size_t mg_iobuf_del(struct mg_iobuf *io, size_t offset, size_t len);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
2021-08-28 07:16:38 +01:00
|
|
|
Delete `len` bytes starting from `offset`, and shift the remaining
|
|
|
|
bytes. If `len` is greater than `io->len`, nothing happens,
|
2020-12-08 16:26:17 +00:00
|
|
|
so such call is silently ignored.
|
|
|
|
|
|
|
|
|
|
|
|
## HTTP
|
|
|
|
|
|
|
|
### struct mg\_http\_header
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_http_header {
|
|
|
|
struct mg_str name;
|
|
|
|
struct mg_str value;
|
|
|
|
};
|
|
|
|
```
|
|
|
|
|
|
|
|
### struct mg\_http\_message
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_http_message {
|
|
|
|
// GET /foo/bar/baz?aa=b&cc=ddd HTTP/1.1
|
|
|
|
// method |-| |----uri---| |--query--| |proto-|
|
|
|
|
|
|
|
|
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
|
|
|
|
};
|
|
|
|
```
|
|
|
|
|
|
|
|
### mg\_http\_listen()
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_connection *mg_http_listen(struct mg_mgr *, const char *url,
|
|
|
|
mg_event_handler_t fn, void *fn_data);
|
|
|
|
```
|
|
|
|
|
|
|
|
Create HTTP listener.
|
|
|
|
|
|
|
|
- `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`
|
|
|
|
|
|
|
|
|
|
|
|
### 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.
|
|
|
|
- `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`
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_http\_get\_request\_len()
|
|
|
|
|
|
|
|
```c
|
|
|
|
int mg_http_get_request_len(const unsigned char *buf, size_t buf_len);
|
|
|
|
```
|
|
|
|
|
|
|
|
Return value: -1 on error,
|
|
|
|
0 if a message is incomplete, or the 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.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_http\_parse()
|
|
|
|
|
|
|
|
```c
|
|
|
|
int mg_http_parse(const char *s, size_t len, struct mg_http_message *hm);
|
|
|
|
```
|
|
|
|
|
|
|
|
Parse string `s`, `len` into a structure `hm`. Return request length - see
|
|
|
|
`mg_http_get_request_len()`.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_http\_printf\_chunk()
|
|
|
|
|
|
|
|
```
|
|
|
|
void mg_http_printf_chunk(struct mg_connection *cnn, const char *fmt, ...);
|
|
|
|
```
|
|
|
|
|
|
|
|
Write a chunk of data in chunked encoding format, using `printf()` semantic.
|
|
|
|
|
|
|
|
|
|
|
|
### 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.
|
|
|
|
|
|
|
|
|
2021-01-12 11:28:41 +01:00
|
|
|
### mg\_http\_serve\_dir()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
```c
|
2021-01-02 17:57:51 +00:00
|
|
|
struct mg_http_serve_opts {
|
2021-03-10 07:56:14 +00:00
|
|
|
const char *root_dir; // Web root directory, must be non-NULL
|
2021-03-11 13:01:45 +00:00
|
|
|
const char *ssi_pattern; // SSI file name pattern, e.g. #.shtml
|
2021-03-10 07:56:14 +00:00
|
|
|
const char *extra_headers; // Extra HTTP headers to add in responses
|
2021-01-02 17:57:51 +00:00
|
|
|
};
|
2020-12-08 16:26:17 +00:00
|
|
|
void mg_http_serve_dir(struct mg_connection *, struct mg_http_message *hm,
|
2021-01-02 17:57:51 +00:00
|
|
|
const struct mg_http_serve_opts *opts);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
2021-01-02 17:57:51 +00:00
|
|
|
Serve static files according to the given options. Note that in order to
|
|
|
|
enable SSI, set a `-DMG_ENABLE_SSI=1` build flag.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
2021-01-12 11:28:41 +01:00
|
|
|
### mg\_http\_serve\_file()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
```c
|
|
|
|
void mg_http_serve_file(struct mg_connection *, struct mg_http_message *hm,
|
2021-07-29 14:21:20 +01:00
|
|
|
const char *path, struct mg_http_serve_opts *opts);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
2021-01-07 16:21:01 +00:00
|
|
|
Serve static file. Note that the `extra_headers` must end with `\r\n`. Here
|
|
|
|
is an example call:
|
|
|
|
|
|
|
|
```c
|
2021-07-29 14:21:20 +01:00
|
|
|
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);
|
2021-01-07 16:21:01 +00:00
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
2021-07-29 14:21:20 +01:00
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
### mg\_http\_reply()
|
|
|
|
|
|
|
|
```c
|
2020-12-17 22:45:22 +00:00
|
|
|
void mg_http_reply(struct mg_connection *c, int status_code, const char *headers,
|
|
|
|
const char *body_fmt, ...);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
2020-12-17 22:45:22 +00:00
|
|
|
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.
|
|
|
|
|
|
|
|
- `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
|
2020-12-08 16:26:17 +00:00
|
|
|
|
2021-07-30 13:19:20 +01:00
|
|
|
Example - send a simple JSON respose:
|
|
|
|
```c
|
|
|
|
mg_http_reply(c, 200, "Content-Type: application/json\r\n", "{\"result\": %d}", 123);
|
|
|
|
```
|
|
|
|
|
|
|
|
Example - 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);
|
|
|
|
```
|
|
|
|
|
|
|
|
Example - send a 302 redirect:
|
|
|
|
```c
|
|
|
|
mg_http_reply(c, 302, "Location: /\r\n", "");
|
|
|
|
```
|
|
|
|
|
|
|
|
Example - send error:
|
|
|
|
```c
|
|
|
|
mg_http_reply(c, 403, "", "%s", "Not Authorised\n");
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
2021-04-29 18:16:33 +02:00
|
|
|
### mg\_http\_get\_header()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_str *mg_http_get_header(struct mg_http_message *, const char *name);
|
|
|
|
```
|
|
|
|
|
|
|
|
Return value of HTTP header, or NULL if not found.
|
|
|
|
|
|
|
|
### mg\_http\_get\_var()
|
|
|
|
|
|
|
|
```c
|
|
|
|
int mg_http_get_var(const struct mg_str *, const char *name, char *buf, int len);
|
|
|
|
```
|
|
|
|
|
|
|
|
Decode HTTP variable `name` into a given buffer. Return length of decoded
|
|
|
|
variable. Zero or negative value means error.
|
|
|
|
|
|
|
|
### mg\_url\_decode()
|
|
|
|
|
|
|
|
```c
|
|
|
|
int mg_url_decode(const char *s, size_t n, char *to, size_t to_len, int form);
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
URL-decode string `s`, `n` unto a buffer `buf`, `len`. Return decoded length.
|
|
|
|
If `form` is non-zero, then `+` is decoded as whitespace.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_http\_creds()
|
|
|
|
|
|
|
|
```c
|
|
|
|
void mg_http_creds(struct mg_http_message *, char *user, int userlen, char *pass, int 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.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_http\_match\_uri()
|
|
|
|
|
|
|
|
```c
|
|
|
|
bool mg_http_match_uri(const struct mg_http_message *, const char *glob);
|
|
|
|
```
|
|
|
|
|
|
|
|
Return true if HTTP request matches a given glob pattern; false otherwise.
|
|
|
|
|
|
|
|
### mg\_http\_upload()
|
|
|
|
|
|
|
|
```c
|
|
|
|
int mg_http_upload(struct mg_connection *, struct mg_http_message *hm,
|
|
|
|
const char *dir);
|
|
|
|
```
|
|
|
|
|
2020-12-14 09:31:03 +00:00
|
|
|
Handle file upload. See [file upload example](https://github.com/cesanta/mongoose/tree/master/examples/file-upload).
|
2020-12-08 16:26:17 +00:00
|
|
|
|
2020-12-14 09:36:46 +00:00
|
|
|
This function expects a series of POST requests with file data. POST requests
|
|
|
|
should have `name` and `offset` query string parameters set:
|
2020-12-14 09:31:03 +00:00
|
|
|
|
2020-12-14 09:35:03 +00:00
|
|
|
```text
|
2020-12-14 09:31:03 +00:00
|
|
|
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
|
2020-12-14 09:35:03 +00:00
|
|
|
- Server must call `mg_http_upload()` when `/whatever_uri` is hit
|
2020-12-14 09:31:03 +00:00
|
|
|
|
|
|
|
So, the expected usage of this API function is this:
|
|
|
|
- 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 POSTed, upload finishes
|
|
|
|
- POST data must not be encoded in any way, it it saved as-is
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
### mg\_http\_bauth()
|
|
|
|
|
|
|
|
```c
|
|
|
|
void mg_http_bauth(struct mg_connection *, const char *user, const char *pass);
|
|
|
|
```
|
|
|
|
|
|
|
|
Write a Basic `Authorization` header to the output buffer.
|
|
|
|
|
2021-08-22 10:51:40 +01:00
|
|
|
### mg\_http\_next\_multipart()
|
2021-03-17 07:43:29 +00:00
|
|
|
|
|
|
|
```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
|
|
|
|
};
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
2021-03-17 13:28:36 +00:00
|
|
|
See `form-upload` example for a usage example.
|
2021-03-17 07:43:29 +00:00
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
## Websocket
|
|
|
|
|
|
|
|
### struct mg\_ws\_message
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_ws_message {
|
|
|
|
struct mg_str data;
|
|
|
|
uint8_t flags; // Websocket message flags
|
|
|
|
};
|
|
|
|
```
|
|
|
|
|
|
|
|
### mg\_ws\_connect()
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_connection *mg_ws_connect(struct mg_mgr *, const char *url,
|
|
|
|
mg_event_handler_t fn, void *fn_data,
|
|
|
|
const char *fmt, ...);
|
|
|
|
```
|
|
|
|
|
|
|
|
Create client Websocket connection.
|
|
|
|
- `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
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_ws\_upgrade()
|
|
|
|
|
|
|
|
```c
|
2021-01-16 22:48:43 +00:00
|
|
|
void mg_ws_upgrade(struct mg_connection *, struct mg_http_message *,
|
|
|
|
const char *fmt, ...);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
2021-01-16 22:48:43 +00:00
|
|
|
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 needs to be passed.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
### mg\_ws\_send()
|
|
|
|
|
|
|
|
```c
|
|
|
|
size_t mg_ws_send(struct mg_connection *, const char *buf, size_t len, int op);
|
|
|
|
```
|
|
|
|
|
|
|
|
Send `buf`, `len` to the websocket peer. `op` is the 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
|
|
|
|
```
|
|
|
|
|
|
|
|
## MQTT
|
|
|
|
|
|
|
|
### struct mg\_mqtt\_opts
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_mqtt_opts {
|
|
|
|
struct mg_str client_id;
|
|
|
|
struct mg_str will_topic;
|
|
|
|
struct mg_str 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
|
|
|
|
};
|
|
|
|
```
|
|
|
|
|
|
|
|
### struct mg\_mqtt\_message
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_mqtt_message {
|
|
|
|
struct mg_str topic;
|
|
|
|
struct mg_str data;
|
|
|
|
};
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_mqtt\_connect()
|
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_connection *mg_mqtt_connect(struct mg_mgr *, const char *url,
|
|
|
|
struct mg_mqtt_opts *opts,
|
|
|
|
mg_event_handler_t fn, void *fn_data);
|
|
|
|
```
|
|
|
|
|
|
|
|
Create client MQTT connection.
|
|
|
|
- `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`
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_mqtt\_pub()
|
|
|
|
|
|
|
|
```c
|
|
|
|
void mg_mqtt_pub(struct mg_connection *, struct mg_str *topic,
|
2021-06-10 19:27:19 +01:00
|
|
|
struct mg_str *data, int qos, bool retain);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
2021-06-10 19:27:19 +01:00
|
|
|
Publish message `data` to the topic `topic` with given QoS and retain flag.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
### mg\_mqtt\_sub()
|
|
|
|
|
|
|
|
```c
|
2021-06-10 19:27:19 +01:00
|
|
|
void mg_mqtt_sub(struct mg_connection *, struct mg_str *topic, int qos);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
2021-06-10 19:27:19 +01:00
|
|
|
Subscribe to topic `topic` with given QoS.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
2021-03-09 11:29:53 +00:00
|
|
|
### mg\_mqtt\_next\_sub()
|
|
|
|
|
|
|
|
```c
|
2021-05-28 18:30:42 +01:00
|
|
|
size_t mg_mqtt_next_sub(struct mg_mqtt_message *msg, struct mg_str *topic, uint8_t *qos, size_t pos);
|
2021-03-09 11:29:53 +00:00
|
|
|
```
|
|
|
|
|
|
|
|
Traverse list of subscribed topics.
|
|
|
|
Used to implement MQTT server when `MQTT_CMD_SUBSCRIBE` is received.
|
2021-05-28 18:30:42 +01:00
|
|
|
Return next position, or 0 when done. Initial position `pos` should be 4. Example:
|
2021-03-09 11:29:53 +00:00
|
|
|
|
|
|
|
```c
|
|
|
|
if (ev == MG_EV_MQTT_CMD) {
|
|
|
|
struct mg_mqtt_message *mm = (struct mg_mqtt_message *) ev_data;
|
|
|
|
if (mm->cmd == MQTT_CMD_SUBSCRIBE) {
|
2021-05-28 18:30:42 +01:00
|
|
|
size_t pos = 4;
|
2021-03-09 11:29:53 +00:00
|
|
|
uint8_t qos;
|
|
|
|
struct mg_str topic;
|
|
|
|
while ((pos = mg_mqtt_next_sub(mm, &topic, &qos, pos)) > 0) {
|
2021-03-09 17:44:15 +00:00
|
|
|
LOG(LL_INFO, ("SUB [%.*s]", (int) topic.len, topic.ptr));
|
2021-03-09 11:29:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
### mg\_mqtt\_next\_unsub()
|
|
|
|
|
|
|
|
```c
|
2021-05-28 18:30:42 +01:00
|
|
|
size_t mg_mqtt_next_unsub(struct mg_mqtt_message *msg, struct mg_str *topic, size_t pos);
|
2021-03-09 11:29:53 +00:00
|
|
|
```
|
|
|
|
|
|
|
|
Same as `mg_mqtt_next_sub()`, but for unsubscribed topics. The difference
|
|
|
|
is that there is no QoS in unsubscribe request.
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
## TLS
|
|
|
|
|
2021-01-30 13:03:11 +00:00
|
|
|
### mg\_tls\_init()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_tls_opts {
|
|
|
|
const char *ca; // CA certificate file. For both listeners and clients
|
|
|
|
const char *cert; // Certificate
|
|
|
|
const char *certkey; // Certificate key
|
|
|
|
const char *ciphers; // Cipher list
|
2021-01-30 13:03:11 +00:00
|
|
|
struct mg_str srvname; // If not empty, enables server name verification
|
2020-12-08 16:26:17 +00:00
|
|
|
};
|
|
|
|
int mg_tls_init(struct mg_connection *c, struct mg_tls_opts *opts);
|
|
|
|
```
|
|
|
|
|
2021-01-30 13:03:11 +00:00
|
|
|
Initialise TLS on a given connection.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
2021-04-29 09:09:21 +01:00
|
|
|
IMPORTANT: mbedTLS implementation uses `mg_random` as RNG. The `mg_random`
|
|
|
|
is defined as weak, therefore it is possible to override Mongoose's `mg_random`
|
|
|
|
with a custom implementation. Just create your own `mg_random` function:
|
|
|
|
`void mg_random(void *buf, size_t len)`.
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
## Timers
|
|
|
|
|
|
|
|
|
|
|
|
### 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
|
2021-03-11 13:01:45 +00:00
|
|
|
void *arg; // Function argument
|
2020-12-08 16:26:17 +00:00
|
|
|
unsigned long expire; // Expiration timestamp in milliseconds
|
|
|
|
struct mg_timer *next; // Linkage in g_timers list
|
|
|
|
};
|
2021-08-22 10:51:40 +01:00
|
|
|
|
|
|
|
#define MG_TIMER_REPEAT 1 // Call function periodically, otherwise run once
|
|
|
|
#define MG_TIMER_RUN_NOW 2 // Call immediately when timer is set
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
2021-08-22 10:51:40 +01:00
|
|
|
Timer structure. Describes a software timer. Timer granularity is the same
|
|
|
|
as the `mg_mgr_poll()` timeout argument in the main event loop.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
### mg\_timer\_init()
|
|
|
|
|
|
|
|
```c
|
|
|
|
void mg_timer_init(struct mg_timer *, int ms, int flags, void (*fn)(void *), void *fn_data);
|
|
|
|
```
|
|
|
|
|
|
|
|
Setup a timer.
|
|
|
|
- `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
|
|
|
|
|
2021-03-11 13:01:45 +00:00
|
|
|
A timer gets initialized and linked into the `g_timers` list:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
```c
|
|
|
|
struct mg_timer *g_timers;
|
|
|
|
```
|
|
|
|
|
|
|
|
### mg\_timer\_free()
|
|
|
|
|
|
|
|
```c
|
|
|
|
void mg_timer_free(struct mg_timer *);
|
|
|
|
```
|
|
|
|
|
|
|
|
Free timer, remove it from the `g_timers` list.
|
|
|
|
|
|
|
|
### 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.
|
|
|
|
|
2021-08-28 06:54:56 +01:00
|
|
|
## String
|
|
|
|
|
|
|
|
### mg\_globmatch()
|
|
|
|
|
|
|
|
```c
|
|
|
|
bool mg_globmatch(const char *pattern, int plen, const char *s, int n);
|
|
|
|
```
|
|
|
|
|
|
|
|
Return true if string `s`, `n` matches glob pattern `pattern`, `plen`.
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
|
|
### 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. 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]
|
|
|
|
```
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
### 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.
|
|
|
|
|
|
|
|
### mg\_hex()
|
|
|
|
|
|
|
|
```c
|
|
|
|
char *mg_hex(const void *buf, int 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.
|
2021-03-11 13:01:45 +00:00
|
|
|
Return value: `dst` pointer. The encoded characters are lowercase,
|
2020-12-08 16:26:17 +00:00
|
|
|
for example `mg_hex("hi", 2, buf)` outputs `6869` and 0 byte, 5 bytes overall.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_unhex()
|
|
|
|
|
|
|
|
```c
|
|
|
|
void mg_unhex(const char *buf, int len, unsigned char *to);
|
|
|
|
```
|
|
|
|
|
|
|
|
Hex-decode string `buf`, `len` into a buffer `to`. The `to` buffer should be
|
|
|
|
at least `lsn` / 2 big.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_unhexn()
|
|
|
|
|
|
|
|
```c
|
|
|
|
unsigned long mg_unhexn(const char *s, int len);
|
|
|
|
```
|
|
|
|
|
|
|
|
Parse `len` characters of the hex-encoded string `s`, return parsed value.
|
|
|
|
The maximum value of `len` is the width of the `long` x 2, for example
|
|
|
|
on 32-bit platforms it is 8.
|
|
|
|
|
|
|
|
|
|
|
|
### 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. Return value: number of bytes printed.
|
|
|
|
|
|
|
|
|
|
|
|
### 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.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_to64()
|
|
|
|
|
|
|
|
```
|
|
|
|
int64_t mg_to64(const char *s);
|
|
|
|
```
|
|
|
|
|
|
|
|
Parse 64-bit integer value held by string `s`.
|
|
|
|
|
|
|
|
### mg\_aton()
|
|
|
|
|
|
|
|
```
|
2020-12-20 16:55:33 +00:00
|
|
|
bool mg_aton(struct mg_str str, uint32_t *ipaddr);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
2020-12-20 16:55:33 +00:00
|
|
|
Parse IP address held by `str` and store it in `ipaddr`. Return true on success.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
### mg\_ntoa()
|
|
|
|
|
|
|
|
```c
|
2020-12-20 16:55:33 +00:00
|
|
|
char *mg_ntoa(const struct mg_addr *, char *buf, size_t len);
|
2020-12-08 16:26:17 +00:00
|
|
|
```
|
|
|
|
|
|
|
|
Stringify IP address `ipaddr` into a buffer `buf`, `len`. Return `buf`.
|
|
|
|
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
|
|
|
|
|
|
## Utility
|
|
|
|
|
|
|
|
|
|
|
|
### 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.
|
|
|
|
|
|
|
|
### mg\_file\_write()
|
|
|
|
|
|
|
|
```c
|
|
|
|
bool mg_file_write(const char *path, const void *buf, size_t len);
|
|
|
|
```
|
|
|
|
|
|
|
|
Write data to a file, return `true` if written, `false` otherwise.
|
|
|
|
The write is atomic, i.e. data gets written to a temporary file first,
|
|
|
|
then `rename()-ed` to a destination file name.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_file\_printf()
|
|
|
|
|
|
|
|
```c
|
|
|
|
int mg_file_printf(const char *path, const char *fmt, ...);
|
|
|
|
```
|
|
|
|
|
|
|
|
Write into a file `path` using `printf()` semantics.
|
|
|
|
Return `true` on success, `false` otherwise. This function prints data to
|
|
|
|
a temporary in-memory buffer first, then calls `mg_file_write()`.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_random()
|
|
|
|
|
|
|
|
```c
|
|
|
|
void mg_random(void *buf, size_t len);
|
|
|
|
```
|
|
|
|
|
|
|
|
Fill in buffer `buf`, `len` with random data.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_ntohs()
|
|
|
|
|
|
|
|
```c
|
|
|
|
uint16_t mg_ntohs(uint16_t net);
|
|
|
|
```
|
|
|
|
|
|
|
|
Convert `uint16_t` value to host order.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_ntohl()
|
|
|
|
|
|
|
|
```c
|
|
|
|
uint32_t mg_ntohl(uint32_t net);
|
|
|
|
```
|
|
|
|
|
|
|
|
Convert `uint32_t` value to host order.
|
|
|
|
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
### mg\_time()
|
|
|
|
|
|
|
|
```
|
|
|
|
double mg_time(void);
|
|
|
|
```
|
|
|
|
|
|
|
|
Return current time as UNIX epoch, using `double` value for sub-second accuracy.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_millis()
|
|
|
|
|
|
|
|
```c
|
|
|
|
unsigned long mg_millis(void);
|
|
|
|
```
|
|
|
|
|
|
|
|
Return current uptime in milliseconds.
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_usleep()
|
|
|
|
|
|
|
|
```c
|
|
|
|
void mg_usleep(unsigned long usecs);
|
|
|
|
```
|
|
|
|
|
|
|
|
Block for a given number of microseconds.
|
2021-03-08 15:41:16 +00:00
|
|
|
|
|
|
|
### mg\_crc32()
|
|
|
|
|
|
|
|
```c
|
|
|
|
uint32_t mg_crc32(uint32_t crc, const uint8_t *buf, size_t len);
|
|
|
|
```
|
|
|
|
|
|
|
|
Calculate CRC32 checksum for a given buffer. An initial `crc` value should
|
|
|
|
be `0`.
|
2021-08-25 07:59:31 +01:00
|
|
|
|
|
|
|
### 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:
|
|
|
|
|
|
|
|
- `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:
|
|
|
|
|
2021-08-26 11:40:47 +01:00
|
|
|
```c
|
2021-08-25 07:59:31 +01:00
|
|
|
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!"));
|
|
|
|
}
|
|
|
|
```
|