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.
|
|
|
|
|
|
2021-08-28 08:52:41 +01:00
|
|
|
|
Mongoose has two basic data structures:
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
|
|
|
|
- `struct mg_mgr` - an event manager that holds all active connections
|
2021-08-28 08:52:41 +01:00
|
|
|
|
- `struct mg_connection` - a single connection descriptor
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
2021-09-17 15:59:15 +01:00
|
|
|
|
## 2-minute intergration guide
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
2021-09-17 15:59:15 +01:00
|
|
|
|
In order to integrate Mongoose into an existing C/C++ aplicaction or firmware,
|
|
|
|
|
please follow these steps:
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
2021-09-17 15:59:15 +01:00
|
|
|
|
**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
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
2021-09-17 15:59:15 +01:00
|
|
|
|
**Step 2.** Add the following lines in your `main.c` file:
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```c
|
2021-09-17 15:59:15 +01:00
|
|
|
|
#include "mongoose.h"
|
|
|
|
|
...
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
static void fn(struct mg_connection *c, int ev, void *ev_data, void *fn_data) {
|
2021-09-17 15:59:15 +01:00
|
|
|
|
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);
|
2021-09-15 07:43:24 +01:00
|
|
|
|
}
|
2021-09-17 15:59:15 +01:00
|
|
|
|
...
|
|
|
|
|
|
|
|
|
|
int main() {
|
|
|
|
|
...
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
2021-09-17 15:59:15 +01:00
|
|
|
|
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
|
|
|
|
|
}
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
2021-09-17 15:59:15 +01:00
|
|
|
|
**Step 3.** Add `mongoose.c` to the build and recompile - and that's it!
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
`mg_mgr_poll()` iterates over all connections, accepts new connections, sends and
|
2020-12-06 21:12:05 +00:00
|
|
|
|
receives data, closes connections and calls event handler functions for the
|
2021-09-15 07:43:24 +01:00
|
|
|
|
respective events.
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
2021-09-17 15:59:15 +01:00
|
|
|
|
|
|
|
|
|
<span class="badge bg-danger">NOTE:</span>
|
|
|
|
|
Since the 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.
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
|
|
|
|
## 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
|
2021-09-15 07:43:24 +01:00
|
|
|
|
functions, like `mg_send()`, `mg_printf()` or 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.
|
2020-12-06 21:12:05 +00:00
|
|
|
|
|
|
|
|
|
## 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
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Mongoose application, since it defines the connection's behavior. This is
|
2020-12-06 21:12:05 +00:00
|
|
|
|
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
|
2021-08-28 11:16:45 +01:00
|
|
|
|
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;`
|
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 *
|
2021-09-15 07:43:24 +01:00
|
|
|
|
MG_EV_HTTP_CHUNK, // HTTP chunk (partial msg) struct mg_http_message *
|
2021-01-17 11:16:19 +00:00
|
|
|
|
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-09-15 07:43:24 +01:00
|
|
|
|
<span class="badge bg-danger">NOTE: </span>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-09-15 07:43:24 +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.
|
2021-05-17 17:36:57 +01:00
|
|
|
|
|
|
|
|
|
| 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-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-09-15 07:43:24 +01:00
|
|
|
|
#include <stdbool.h>
|
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
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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):
|
2021-01-20 11:28:47 +00:00
|
|
|
|
|
|
|
|
|
```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;
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
2020-12-08 16:26:17 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# API Reference
|
|
|
|
|
|
|
|
|
|
## Core
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### 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
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```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
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_mgr\_init()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```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:
|
2021-09-15 15:37:57 +03:00
|
|
|
|
- `mgr` - a pointer to `mg_mgr` structure that needs to be initialized
|
2021-08-12 12:24:28 +01:00
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_mgr mgr;
|
|
|
|
|
mg_mgr_init(&mgr);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_mgr\_poll()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```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-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `mgr` - event manager to use
|
|
|
|
|
- `ms` - timeout (maximum time to block current thread), milliseconds
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
while (running == true) mg_mgr_poll(&mgr, 1000 /* 1 sec */);
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `mgr` - event manager to free
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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);
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### 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`.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `mgr` - event manager to use
|
2020-12-08 16:26:17 +00:00
|
|
|
|
- `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.
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_connection *c = mg_listen(&mgr, "tcp://127.0.0.1:8080", fn, NULL);
|
|
|
|
|
```
|
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
### 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`.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `mgr` - event manager to use
|
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.
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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);
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### 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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `data` - data to append
|
|
|
|
|
- `size` - size of data
|
|
|
|
|
|
|
|
|
|
Return value: `true` if data appended successfully and `false` otherwise
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
mg_send(c, "hi", 2); // Append string "hi" to the output buffer
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### 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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `fmt` - format string in `printf` semantics
|
|
|
|
|
|
|
|
|
|
Return value: number of bytes appended to the output buffer.
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
mg_printf(c, "Hello, %s!", "world"); // Add "Hello, world!" to output buffer
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### 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-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `fmt` - format string in `printf` semantics
|
|
|
|
|
- `ap` - arguments list
|
|
|
|
|
|
|
|
|
|
Return value: number of bytes appended to the output buffer.
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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`)
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `buf` - buffer to write data
|
|
|
|
|
- `len` - buffer (`buf`) size
|
|
|
|
|
|
|
|
|
|
Return value: `buf` value
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
char buf[1024];
|
|
|
|
|
mg_straddr(c, buf, sizeof(buf)); // `buf` is now IP address string, like "127.0.0.1:8080"
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
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-09-15 07:43:24 +01:00
|
|
|
|
and which is used to wake up event manager from a different task. These
|
2021-08-11 19:24:33 +01:00
|
|
|
|
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
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `mgr` - event manager to use
|
|
|
|
|
- `fn` - pointer to event handler function
|
|
|
|
|
- `ud` - user data, which will be passed to `fn`
|
|
|
|
|
|
|
|
|
|
Return value: pointer to created connection or `NULL` in case of error
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example: see [examples/multi-threaded](../examples/multi-threaded).
|
2021-08-07 17:22:47 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_mgr\_wakeup()
|
2021-08-12 12:24:28 +01:00
|
|
|
|
|
|
|
|
|
```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:
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
2021-08-12 12:24:28 +01:00
|
|
|
|
- `pipe` - a special connection created by the `mg_mkpipe()` call
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example: see [examples/multi-threaded](../examples/multi-threaded).
|
2021-08-12 12:24:28 +01:00
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
## HTTP
|
|
|
|
|
|
|
|
|
|
### struct mg\_http\_header
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_http_header {
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str name; // Header name
|
|
|
|
|
struct mg_str value; // Header value
|
2020-12-08 16:26:17 +00:00
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Structure represents HTTP header, like `Content-Type: text/html`.
|
|
|
|
|
`Content-Type` is a header name and `text/html` is a header value.
|
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
### struct mg\_http\_message
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_http_message {
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str method, uri, query, proto; // Request/response line
|
2020-12-08 16:26:17 +00:00
|
|
|
|
struct mg_http_header headers[MG_MAX_HTTP_HEADERS]; // Headers
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str body; // Body
|
|
|
|
|
struct mg_str message; // Request line + headers + body
|
2020-12-08 16:26:17 +00:00
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Structure represents the HTTP message.
|
|
|
|
|
|
|
|
|
|
<img src="images/mg_http_message.png">
|
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
### mg\_http\_listen()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
struct mg_connection *mg_http_listen(struct mg_mgr *mgr, const char *url,
|
2020-12-08 16:26:17 +00:00
|
|
|
|
mg_event_handler_t fn, void *fn_data);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Create HTTP listener.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `mgr` - event manager to use
|
2020-12-08 16:26:17 +00:00
|
|
|
|
- `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`
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Return value: pointer to created connection or `NULL` in case of error
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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");
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### 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.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
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
|
2020-12-08 16:26:17 +00:00
|
|
|
|
- `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`
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Return value: pointer to created connection or `NULL` in case of error
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```c
|
|
|
|
|
struct mg_connection *c = mg_http_connect(&mgr, "http://google.com", fn, NULL);
|
|
|
|
|
if (c == NULL) fatal_error("Cannot create connection");
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### mg\_http\_get\_request\_len()
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
int mg_http_get_request_len(const unsigned char *buf, size_t buf_len);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Get length of request.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
The length of request is a number of bytes till the end of HTTP headers. It does
|
|
|
|
|
not include length of HTTP body.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `buf` - pointer to buffer with request
|
|
|
|
|
- `buf_len` - maximum request length
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
<img src="images/mg_http_get_request_len.png">
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### mg\_http\_parse()
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
int mg_http_parse(const char *s, size_t len, struct mg_http_message *hm);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parse string request into `mg_http_message` structure
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `s` - request string
|
|
|
|
|
- `len` - maximum request length
|
|
|
|
|
- `hm` - pointer to `mg_http_message` structure to receive result
|
|
|
|
|
|
|
|
|
|
Return value: request length (see `mg_http_get_request_len()`).
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```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 */ }
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### mg\_http\_printf\_chunk()
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```c
|
|
|
|
|
void mg_http_printf_chunk(struct mg_connection *c, const char *fmt, ...);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Write a chunk of data in chunked encoding format, using `printf()` semantic.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `fmt` - format string in `printf` semantics
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
mg_http_printf_chunk(c, "Hello, %s!", "world");
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### 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-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `buf` - data to write
|
|
|
|
|
- `len` - number of bytes to write
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Remove chunk specified from input buffer.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `hm` - chunk to delete
|
|
|
|
|
|
|
|
|
|
Return value: none
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
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
|
|
|
|
};
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_http_serve_dir(struct mg_connection *c, 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-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `hm` - http message, that should be served
|
2021-09-20 12:24:44 +01:00
|
|
|
|
- `opts` - serve options. Note that `opts.root_dir` can optionally accept
|
|
|
|
|
extra comma-separated `uri=path` pairs, see example below
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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));
|
2021-09-20 12:24:44 +01:00
|
|
|
|
opts.root_dir = "/var/www,/conf=/etc"; // Serve /var/www. URIs starting with /conf are served from /etc
|
2021-09-15 07:43:24 +01:00
|
|
|
|
mg_http_serve_dir(c, hm, &opts);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
```
|
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
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_http_serve_file(struct mg_connection *c, 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-09-15 07:43:24 +01:00
|
|
|
|
Serve static file. Note that the `extra_headers` must end with `\r\n`.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `hm` - http message to serve
|
|
|
|
|
- `path` - path to file to serve
|
|
|
|
|
- `opts` - serve options
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2021-01-07 16:21:01 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
// 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",
|
2021-07-29 14:21:20 +01:00
|
|
|
|
.extra_headers = "AA: bb\r\nCC: dd\r\n"};
|
2021-09-15 07:43:24 +01:00
|
|
|
|
mg_http_serve_file(c, hm, "a.png", &opts); // Send file
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-01-07 16:21:01 +00:00
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### mg\_http\_reply()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01: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
|
2021-09-15 07:43:24 +01:00
|
|
|
|
`Content-Length` header. Extra headers could be passed via `headers` parameter.
|
2020-12-17 22:45:22 +00:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
2020-12-17 22:45:22 +00:00
|
|
|
|
- `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-09-15 15:37:57 +03:00
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
<img src="images/mg_http_reply.png">
|
2021-07-30 13:19:20 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage examples:
|
2021-07-30 13:19:20 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Send a simple JSON respose:
|
|
|
|
|
```c
|
|
|
|
|
mg_http_reply(c, 200, "Content-Type: application/json\r\n", "{\"result\": %d}", 123);
|
|
|
|
|
```
|
2021-07-30 13:19:20 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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");
|
|
|
|
|
```
|
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
|
2021-09-15 15:37:57 +03:00
|
|
|
|
struct mg_str *mg_http_get_header(struct mg_http_message *hm, const char *name);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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...");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### mg\_http\_get\_var()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
int mg_http_get_var(const struct mg_str *var, const char *name, char *buf, int len);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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. Zero or negative value means error.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
char buf[100] = "";
|
|
|
|
|
mg_http_get_var(&hm->body, "key1", buf, sizeof(buf)) {
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### mg\_http\_creds()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_http_creds(struct mg_http_message *hm, char *user, size_t userlen,
|
2021-09-15 07:43:24 +01:00
|
|
|
|
char *pass, size_t passlen);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### mg\_http\_match\_uri()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
bool mg_http_match_uri(const struct mg_http_message *hm, const char *glob);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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..");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
### mg\_http\_upload()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
int mg_http_upload(struct mg_connection *c, struct mg_http_message *hm,
|
2020-12-08 16:26:17 +00:00
|
|
|
|
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
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `hm` - POST message, containing parameters described above
|
|
|
|
|
- `dir` - path to directory
|
|
|
|
|
|
|
|
|
|
Return value: request body len or negative value on error
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
### mg\_http\_bauth()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_http_bauth(struct mg_connection *c, const char *user, const char *pass);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Write a Basic `Authorization` header to the output buffer.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `user` - user name
|
|
|
|
|
- `pass` - password
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
mg_http_bauth(c, "user_name", "password") // "user_name:password" is now in 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
|
|
|
|
|
};
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
2021-03-17 07:43:29 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
<img src="images/mg_http_part.png">
|
|
|
|
|
|
|
|
|
|
```c
|
2021-03-17 07:43:29 +00:00
|
|
|
|
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-09-15 15:37:57 +03:00
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
See [examples/form-upload](../examples/form-upload) for full usage example.
|
2021-03-17 07:43:29 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
<img src="images/mg_http_next_multipart.png">
|
|
|
|
|
|
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_http_part part;
|
|
|
|
|
if(mg_http_next_multipart(body, 0 /* begin */, &part)) {
|
|
|
|
|
// Use part
|
|
|
|
|
}
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
## Websocket
|
|
|
|
|
|
|
|
|
|
### struct mg\_ws\_message
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_ws_message {
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str data; // Websocket message data
|
|
|
|
|
uint8_t flags; // Websocket message flags
|
2020-12-08 16:26:17 +00:00
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Structure represents the WebSocket message.
|
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
### mg\_ws\_connect()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
struct mg_connection *mg_ws_connect(struct mg_mgr *mgr, const char *url,
|
2020-12-08 16:26:17 +00:00
|
|
|
|
mg_event_handler_t fn, void *fn_data,
|
|
|
|
|
const char *fmt, ...);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Create client Websocket connection.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
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
|
2020-12-08 16:26:17 +00:00
|
|
|
|
- `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
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Return value: pointer to created connection or `NULL` on error
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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");
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### mg\_ws\_upgrade()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_ws_upgrade(struct mg_connection *c, struct mg_http_message *,
|
2021-01-16 22:48:43 +00:00
|
|
|
|
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
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `hm` - HTTP message
|
|
|
|
|
- `fmt` - printf-like format string for additional HTTP headers, or NULL
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### mg\_ws\_send()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
size_t mg_ws_send(struct mg_connection *c, const char *buf, size_t len, int op);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```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
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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](../examples/mqtt-over-ws-client) for full example.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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)
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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, that app can't send SNTP request more often than every 1 hour.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `utc` - current time, used to verify if new request is possible.
|
|
|
|
|
|
|
|
|
|
Return value: none
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
mg_sntp_send(c, (unsigned long) time(NULL));
|
|
|
|
|
```
|
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
## MQTT
|
|
|
|
|
|
|
|
|
|
### struct mg\_mqtt\_opts
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_mqtt_opts {
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
2020-12-08 16:26:17 +00:00
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Structure used to specify MQTT connection options.
|
|
|
|
|
|
2020-12-08 16:26:17 +00:00
|
|
|
|
### struct mg\_mqtt\_message
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_mqtt_message {
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str topic; // Topic
|
|
|
|
|
struct mg_str data; // Message data
|
2020-12-08 16:26:17 +00:00
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Structure represents the MQTT message.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### mg\_mqtt\_connect()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
struct mg_connection *mg_mqtt_connect(struct mg_mgr *mgr, const char *url,
|
2020-12-08 16:26:17 +00:00
|
|
|
|
struct mg_mqtt_opts *opts,
|
|
|
|
|
mg_event_handler_t fn, void *fn_data);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Create client MQTT connection.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
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
|
2020-12-08 16:26:17 +00:00
|
|
|
|
- `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`
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Return value: pointer to created connection or `NULL` on error
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
void fn(struct mg_connection *c, int ev, void *evd, void *fnd) {
|
|
|
|
|
char *buf = (char *) fnd;
|
|
|
|
|
if (ev == MG_EV_MQTT_OPEN) {
|
|
|
|
|
// Connection ready
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
mg_mqtt_connect(&mgr, "mqtt://test.org:1883", NULL, handler, NULL);
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_mqtt\_listen()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_connection *mg_mqtt_listen(struct mg_mgr *mgr, const char *url,
|
|
|
|
|
mg_event_handler_t fn, void *fn_data);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Create MQTT listener.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `mgr` - event manager to use
|
2021-09-15 07:43:24 +01:00
|
|
|
|
- `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`
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Return value: pointer to created connection or `NULL` on error
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `url` - url, containing user name and password to use
|
|
|
|
|
- `opts` - request options
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_mqtt_pub(struct mg_connection *c, struct mg_str *topic,
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str *data, int qos, bool retain);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_mqtt_sub(struct mg_connection *c, struct mg_str *topic, int qos);
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Subscribe to topic.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `topic` - topic to subscribe
|
|
|
|
|
- `qos` - required Qos
|
|
|
|
|
|
|
|
|
|
Return value: none
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_str topic = mg_str("topic");
|
|
|
|
|
mg_mqtt_sub(c, &topic, 1);
|
|
|
|
|
```
|
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
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Traverse list of subscribed topics.
|
2021-03-09 11:29:53 +00:00
|
|
|
|
Used to implement MQTT server when `MQTT_CMD_SUBSCRIBE` is received.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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.
|
2021-03-09 11:29:53 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
2021-03-09 11:29:53 +00:00
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
// 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));
|
|
|
|
|
}
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `mgs` - MQTT message
|
|
|
|
|
- `topic` - pointer to `mg_str` to receive topic
|
|
|
|
|
- `pos` - position to list from
|
|
|
|
|
|
|
|
|
|
Return value: next position, or 0 when done.
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_mqtt_send_header(struct mg_connection *c, uint8_t cmd, uint8_t flags, uint32_t len);
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
|
|
|
|
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
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
- `cmd` - command (see above)
|
|
|
|
|
- `flags` - command flags
|
|
|
|
|
- `len` - size of the following command
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_mqtt_ping(struct mg_connection *c);
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Send `MQTT_CMD_PINGREQ` command via `mg_mqtt_send_header`
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to use
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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
|
2021-09-15 07:43:24 +01:00
|
|
|
|
isn't fully receives and `MQTT_MALFORMED` is 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;
|
|
|
|
|
}
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
## TLS
|
|
|
|
|
|
2021-08-28 08:21:29 +01:00
|
|
|
|
### struct mg\_tls\_opts
|
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
|
|
|
|
};
|
2021-08-28 08:21:29 +01:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
TLS initialisation structure:
|
|
|
|
|
- `ca` - Certificate Authority. Can be a filename, or a string. Used to verify
|
|
|
|
|
a certificate that other end sends to us. If NULL, then certificate 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
|
2021-09-15 07:43:24 +01:00
|
|
|
|
void mg_tls_init(struct mg_connection *c, struct mg_tls_opts *opts);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-01-30 13:03:11 +00:00
|
|
|
|
Initialise TLS on a given connection.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
<span class="badge bg-danger">NOTE:</span> 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.
|
2021-04-29 09:09:21 +01:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection, for which TLS should be initialized
|
|
|
|
|
- `opts` - TLS initialization parameters
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```c
|
|
|
|
|
struct mg_tls_opts opts = {.cert = "ca.pem"};
|
|
|
|
|
mg_tls_init(c, &opts);
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
## Timer
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### 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
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_timer_init(struct mg_timer *t, unsigned long ms, unsigned flags,
|
2021-09-15 07:43:24 +01:00
|
|
|
|
void (*fn)(void *), void *fn_data);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Setup a timer.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `t` - pointer to `mg_timer` that should be initialized
|
2020-12-08 16:26:17 +00:00
|
|
|
|
- `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-09-15 15:37:57 +03:00
|
|
|
|
Return value: none
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
void timer_fn(void *data) {
|
|
|
|
|
// ...
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
struct mg_timer timer;
|
|
|
|
|
mg_timer_init(&timer, 1000 /* 1sec */, MG_TIMER_REPEAT, timer_fn, NULL);
|
2021-09-15 15:37:57 +03:00
|
|
|
|
// A timer gets initialized and linked into the internal timers list
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### mg\_timer\_free()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_timer_free(struct mg_timer *t);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Free timer, remove it from the internal timers list.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `t` - timer to free
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
```c
|
|
|
|
|
struct mg_timer timer;
|
|
|
|
|
// ...
|
|
|
|
|
mg_timer_free(&timer);
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
### 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-09-15 07:43:24 +01:00
|
|
|
|
Note, that `mg_mgr_poll` function internally calls `mg_timer_poll`, therefore,
|
|
|
|
|
in most cases it is unnecessary to call it explicitly.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `uptime_ms` - current timestamp
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
unsigned long now = mg_millis();
|
|
|
|
|
mg_timer_poll(now);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
## Time
|
|
|
|
|
|
2021-08-28 08:46:01 +01:00
|
|
|
|
### mg\_time()
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
double mg_time(void);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Return current time as UNIX epoch, using `double` value for sub-second accuracy.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters: none
|
|
|
|
|
|
|
|
|
|
Return value: current time
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
double now = mg_time()
|
|
|
|
|
```
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
|
|
|
|
### mg\_millis()
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
unsigned long mg_millis(void);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Return current uptime in milliseconds.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters: none
|
|
|
|
|
|
|
|
|
|
Return value: current uptime
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
unsigned long uptime = mg_millis();
|
|
|
|
|
```
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
|
|
|
|
### mg\_usleep()
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
void mg_usleep(unsigned long usecs);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Block thread execution for a given number of microseconds.
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `usecs` - number of microseconds to block thread
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```c
|
|
|
|
|
mg_usleep(1000000 /* 1 sec */)
|
|
|
|
|
```
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
2021-08-28 06:54:56 +01:00
|
|
|
|
## String
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_str
|
2021-08-28 06:54:56 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
In most cases, Mongoose uses `mg_str` struct for string representation rather than NULL-terminated C-strings.
|
2021-08-28 06:54:56 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
|
|
|
|
struct mg_str {
|
|
|
|
|
const char *ptr; // Pointer to string data
|
|
|
|
|
size_t len; // String len
|
|
|
|
|
};
|
|
|
|
|
```
|
2021-08-28 06:54:56 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Note, that in general, `ptr` points to non-NULL terminated string, so, do not use functions from C standard library on it.
|
2021-08-28 06:54:56 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_str()
|
2021-08-28 06:54:56 +01:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str mg_str(const char *s)
|
2021-08-28 06:54:56 +01:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
2021-08-28 06:54:56 +01:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `s` - pointer to NULL-terminated string to store in created mg_str
|
|
|
|
|
|
|
|
|
|
Return value: created Mongoose string
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2021-08-28 06:54:56 +01:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str str = mg_str("Hello, world!);
|
2021-08-28 06:54:56 +01:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_str\_n()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str mg_str_n(const char *s, size_t n);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Create Mongoose string from C-string `s` (can be non-NULL terminated, len is specified in `n`). <br>
|
|
|
|
|
Note: this function doesn't duplicate provided string, but stores pointer within created `mg_str` structure.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `s` - pointer to string to store in created `mg_str`
|
|
|
|
|
- `n` - string len
|
|
|
|
|
|
|
|
|
|
Return value: created Mongoose string
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str str = mg_str_n("hi", 2);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_casecmp()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
int mg_casecmp(const char *s1, const char *s2);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Case insensitive compare two NULL-terminated strings.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `s1`, `s2` - pointers to strings to compare
|
|
|
|
|
|
|
|
|
|
Return value: 0 if strings are equal, more than zero if first argument is greater then second, and less than zero otherwise.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```c
|
|
|
|
|
if (mg_casecmp("hello", "HELLO") == 0) {
|
|
|
|
|
// Strings are equal
|
|
|
|
|
}
|
|
|
|
|
```
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_ncasecmp()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
int mg_ncasecmp(const char *s1, const char *s2, size_t len);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Case insensitive compare two C-strings, not more than `len` symbols or until meet `\0` symbol.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `s1`, `s2` - pointers to strings to compare
|
|
|
|
|
- `len` - maximum length to compare
|
|
|
|
|
|
|
|
|
|
Return value: 0 if strings are equal, more than zero if first argument is
|
2021-09-15 07:43:24 +01:00
|
|
|
|
greater then second and less than zero otherwise.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
if (mg_ncasecmp("hello1", "HELLO2", 5) == 0) {
|
|
|
|
|
// Strings are equal
|
|
|
|
|
}
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_vcmp()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
int mg_vcmp(const struct mg_str *s1, const char *s2);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Сompare mongoose string and C-string.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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
|
2021-09-15 07:43:24 +01:00
|
|
|
|
greater then second and less than zero otherwise.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-08-28 20:27:55 +01:00
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str str = mg_str("hello");
|
|
|
|
|
if (mg_vcmp(str, "hello") == 0) {
|
|
|
|
|
// Strings are equal
|
|
|
|
|
}
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_vcasecmp()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-08-28 20:27:55 +01:00
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
int mg_vcasecmp(const struct mg_str *str1, const char *str2);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Case insensitive compare mongoose string and C-string.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `str1` - Mongoose string to compare
|
|
|
|
|
- `str2` - C-string to compare
|
|
|
|
|
|
|
|
|
|
Return value: 0 if strings are equal, more than zero if first argument is
|
2021-09-15 07:43:24 +01:00
|
|
|
|
greater then second and less than zero otherwise.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str str = mg_str("hello");
|
|
|
|
|
if (mg_vcasecmp(str, "HELLO") == 0) {
|
|
|
|
|
// Strings are equal
|
|
|
|
|
}
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_strcmp()
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```c
|
|
|
|
|
int mg_strcmp(const struct mg_str str1, const struct mg_str str2);
|
|
|
|
|
```
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Сompare two mongoose strings.
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `str1`, `str2` - pointers to Mongoose strings to compare
|
|
|
|
|
|
|
|
|
|
Return value: 0 if strings are equal, more than zero if first argument is
|
2021-09-15 07:43:24 +01:00
|
|
|
|
greater then second and less than zero otherwise.
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str str1 = mg_str("hello");
|
|
|
|
|
struct mg_str str2 = mg_str("hello");
|
|
|
|
|
if (mg_strcmp(str1, str2) == 0) {
|
|
|
|
|
// Strings are equal
|
|
|
|
|
}
|
2021-08-28 07:16:38 +01:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_strdup()
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str mg_strdup(const struct mg_str s);
|
2021-08-28 07:16:38 +01:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `s` - Mongoose string to duplicate
|
|
|
|
|
|
|
|
|
|
Return value: duplicated string
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str str1 = mg_str("hello");
|
|
|
|
|
struct mg_str str2 = mg_strdup(str1);
|
|
|
|
|
//...
|
|
|
|
|
free(str1.ptr);
|
2021-08-28 07:16:38 +01:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_strstr()
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
const char *mg_strstr(const struct mg_str haystack, const struct mg_str needle)
|
2021-08-28 07:16:38 +01:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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`
|
2021-09-15 07:43:24 +01:00
|
|
|
|
occurrence within `haystack` or `NULL` if not found.
|
|
|
|
|
|
|
|
|
|
Usage example:
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```c
|
|
|
|
|
struct mg_str str = mg_str("Hello, world");
|
|
|
|
|
struct mg_str sub_str = mg_str("world");
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
if (mg_strstr(str, sub_str) != NULL) {
|
|
|
|
|
// Found
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### mg\_strstrip()
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str mg_strstrip(struct mg_str s)
|
2021-08-28 07:16:38 +01:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Remove heading and trailing whitespace from mongoose string `s`.
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Paramaters:
|
|
|
|
|
- `s` - Mongoose string for trimming
|
|
|
|
|
|
|
|
|
|
Return value: input string
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```c
|
|
|
|
|
struct mg_str str = mg_strstrip(mg_str(" Hello, world "));
|
|
|
|
|
if (mg_vcmp(str, "Hello, world") == 0) {
|
|
|
|
|
// Strings are equal
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### mg\_globmatch()
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
bool mg_globmatch(const char *pattern, size_t p_len, const char *s, size_t s_len);
|
2021-08-28 07:16:38 +01:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Check if string `s` (limited to `s_len` symbols) matches glob pattern `pattern`, (limited to `p_len` symbols).
|
2021-09-15 07:43:24 +01:00
|
|
|
|
The glob pattern matching rules are as follows:
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
- `?` matches any single character
|
|
|
|
|
- `*` matches zero or more characters except `/`
|
|
|
|
|
- `#` matches zero or more characters
|
|
|
|
|
- any other character matches itself
|
2021-08-28 07:16:38 +01:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `pattern` - pattern to match for
|
|
|
|
|
- `p_len` - pattetn lenght
|
|
|
|
|
- `s` - string to match
|
|
|
|
|
- `s_len` - string lenght
|
|
|
|
|
|
|
|
|
|
Return value: `true` is matches, `false` otherwise
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-08-28 08:46:01 +01:00
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
struct mg_str pattern = mg_str("#, ?????");
|
|
|
|
|
struct mg_str s = mg_str("Hello, world");
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
if (mg_globmatch(pattern.ptr, pattern.len, s.otr, s.len)) {
|
|
|
|
|
// Match
|
|
|
|
|
}
|
|
|
|
|
```
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_commalist()
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
bool mg_commalist(struct mg_str *s, struct mg_str *k, struct mg_str *v);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
IMPORTANT: this function modifies `s` by pointing to the next entry.
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
<img src="images/mg_commalist.png">
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-08-28 08:46:01 +01:00
|
|
|
|
Usage example:
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
### mg\_hexdump()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `buf` - data to hexdump
|
|
|
|
|
- `len` - data lenght
|
|
|
|
|
|
|
|
|
|
Return value: malloc-ed buffer with hexdumped data
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `buf` - data to hex-encode
|
|
|
|
|
- `len` - data length
|
|
|
|
|
- `dst` - pointer to output buffer
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `buf` - data to hex-decode
|
|
|
|
|
- `len` - data length
|
|
|
|
|
- `to` - pointer to output buffer
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parse `len` characters of the hex-encoded string `s`.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
The maximum value of `len` is the width of the `long` x 2, for example
|
|
|
|
|
on 32-bit platforms it is 8.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `s` - string to parse
|
|
|
|
|
- `len` - string length
|
|
|
|
|
|
|
|
|
|
Return value: return parsed value
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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`.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `str` - string to parse
|
|
|
|
|
|
|
|
|
|
Return value: parsed value
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parse IP address held by `str` and store it in `addr`.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `str` - string to parse
|
|
|
|
|
- `addr` - pointer to `mg_addr` string to receive parsed value
|
|
|
|
|
|
|
|
|
|
Return value: `true` on success, `false` otherwise.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to send event
|
|
|
|
|
- `ev` - event to send
|
|
|
|
|
- `ev_data` - additional event parameter
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - connection to send event
|
|
|
|
|
- `fmt` - format string in `printf` semantics
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - pointer to `mg_md5_ctx` structure to initialize
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - md5 context
|
|
|
|
|
- `data` - data to hash
|
|
|
|
|
- `len` - data length
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_md5_final(mg_md5_ctx *c, unsigned char buf[16]);
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Get current MD5 hash for context.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - md5 context
|
|
|
|
|
- `buf` - pointer to buffer to write MD5 hash value
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_sha1_init(mg_sha1_ctx *c);
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Initialize context for calculating SHA1 hash
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - pointer to `mg_sha1_ctx` structure to initialize
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
mg_sha1_ctx ctx;
|
|
|
|
|
mg_sha1_init(&ctx);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### mg\_sha1\_update()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_sha1_update(mg_sha1_ctx *c, const unsigned char *data, size_t len);
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Hash `len` bytes of `data` using SHA1 algorithm.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - sha1 context
|
|
|
|
|
- `data` - data to hash
|
|
|
|
|
- `len` - data length
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
2021-09-15 15:37:57 +03:00
|
|
|
|
void mg_sha1_final(unsigned char digest[20], mg_sha1_ctx *c);
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Get current SHA1 hash for context.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `c` - sha1 context
|
|
|
|
|
- `digest` - pointer to buffer to receive hash value
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
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 futher operations.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `buf` - pointer to buffer to write finish mark
|
|
|
|
|
- `pos` - position to write
|
|
|
|
|
|
|
|
|
|
Return value: new position for futher operations.
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```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);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `path` - path to file to read
|
|
|
|
|
- `sizep` - pointer to `size_t` to receive file size
|
|
|
|
|
|
|
|
|
|
Return value: file contents, see function description
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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);
|
|
|
|
|
```
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Write data to a file.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
The write is atomic, i.e. data gets written to a temporary file first, then `rename()-ed` to a destination file name.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `path` - path to file
|
|
|
|
|
- `buf` - data to write
|
|
|
|
|
- `len` - data length
|
|
|
|
|
|
|
|
|
|
Return value: `true` if written, `false` otherwise.
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
This function prints data to a
|
2021-09-15 07:43:24 +01:00
|
|
|
|
temporary in-memory buffer first, then calls `mg_file_write()`.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `path`- path to file
|
|
|
|
|
- `fmt` - format string in `printf()` semantics
|
|
|
|
|
|
|
|
|
|
Return value: `true` on success, `false` otherwise.
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `buf` - pointer to buffer to receive random data
|
|
|
|
|
- `len` - buffer size
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `net` - 16-bit value in network order
|
|
|
|
|
|
|
|
|
|
Return value: 16-bit value in host order
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `net` - 32-bit value in network order
|
|
|
|
|
|
|
|
|
|
Return value: 32-bit value in host order
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
uint32_t val = mg_ntohl(0x12345678);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### mg\_ntohs()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
uint16_t mg_htons(uint16_t h);
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Convert `uint16_t` value to network order.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `h` - 16-bit value in host order
|
|
|
|
|
|
|
|
|
|
Return value: 16-bit value in network order
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
uint16_t val = mg_htons(0x1234);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### mg\_htonl()
|
|
|
|
|
|
|
|
|
|
```c
|
2021-09-15 15:37:57 +03:00
|
|
|
|
uint32_t mg_ntohl(uint32_t h);
|
2021-09-15 07:43:24 +01:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Convert `uint32_t` value to network order.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `h` - 32-bit value in host order
|
|
|
|
|
|
|
|
|
|
Return value: 32-bit value in network order
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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`.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `crc` - initial crc value
|
|
|
|
|
- `buf` - data to calculate CRC32
|
|
|
|
|
- `len` - data size
|
|
|
|
|
|
|
|
|
|
Return value: calculated CRC32 checksum
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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:
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
2021-09-15 07:43:24 +01:00
|
|
|
|
- `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.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `s` - string to encode
|
|
|
|
|
- `n` - string to encode lenght
|
|
|
|
|
- `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
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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`.
|
2021-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
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`
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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 insert or delete chunks of data at arbitrary offsets and grow/shrink
|
2021-08-30 13:06:45 +01:00
|
|
|
|
automatically.
|
|
|
|
|
|
2021-08-28 08:46:01 +01:00
|
|
|
|
### struct mg\_iobuf
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
2021-08-28 08:46:01 +01:00
|
|
|
|
```c
|
|
|
|
|
struct mg_iobuf {
|
2021-08-30 13:06:45 +01:00
|
|
|
|
unsigned char *buf; // Pointer to stored data
|
|
|
|
|
size_t size; // Total size available
|
|
|
|
|
size_t len; // Current number of bytes
|
2021-08-28 08:46:01 +01:00
|
|
|
|
};
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Generic IO buffer. The `size` specifies an allocation size of the data pointed
|
|
|
|
|
by `buf`, and `len` specifies number of bytes currently stored.
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
<img src="images/mg_iobuf.png">
|
|
|
|
|
|
2021-08-28 08:46:01 +01:00
|
|
|
|
### mg\_iobuf\_init()
|
2020-12-08 16:26:17 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-08-28 08:46:01 +01:00
|
|
|
|
int mg_iobuf_init(struct mg_iobuf *io, size_t size);
|
2020-12-08 16:26:17 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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.
|
2021-03-08 15:41:16 +00:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_iobuf io;
|
|
|
|
|
if (mg_iobuf_init(&io)) {
|
|
|
|
|
// io successfully initialized
|
|
|
|
|
}
|
|
|
|
|
```
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
|
|
|
|
### mg\_iobuf\_resize()
|
2021-03-08 15:41:16 +00:00
|
|
|
|
|
|
|
|
|
```c
|
2021-08-28 08:46:01 +01:00
|
|
|
|
int mg_iobuf_resize(struct mg_iobuf *io, size_t size);
|
2021-03-08 15:41:16 +00:00
|
|
|
|
```
|
|
|
|
|
|
2021-08-28 08:46:01 +01: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-09-15 15:37:57 +03:00
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `io` - iobuf to resize
|
|
|
|
|
- `size` - new size
|
|
|
|
|
|
|
|
|
|
Return value: 1 on success, 0 on allocation failure
|
2021-08-25 07:59:31 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_iobuf io;
|
|
|
|
|
// IO buffer initialization
|
|
|
|
|
// ...
|
|
|
|
|
|
|
|
|
|
if (mg_iobuf_resize(&io, 1024)) {
|
|
|
|
|
// New io size is 1024 bytes
|
|
|
|
|
}
|
|
|
|
|
```
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
|
|
|
|
### mg\_iobuf\_free()
|
2021-08-25 07:59:31 +01:00
|
|
|
|
|
|
|
|
|
```c
|
2021-08-28 08:46:01 +01:00
|
|
|
|
void mg_iobuf_free(struct mg_iobuf *io);
|
2021-08-25 07:59:31 +01:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Free memory pointed by `io->buf` and set to NULL. Both `size` and `len` are set to 0.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `io` - iobuf to free
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_iobuf io;
|
|
|
|
|
// IO buffer initialization
|
|
|
|
|
// ...
|
2021-08-25 07:59:31 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
// Time to cleanup
|
|
|
|
|
mg_iobuf_free(&io);
|
|
|
|
|
```
|
2021-08-25 07:59:31 +01:00
|
|
|
|
|
2021-08-28 08:46:01 +01:00
|
|
|
|
### mg\_iobuf\_add()
|
2021-08-25 07:59:31 +01:00
|
|
|
|
|
2021-08-28 08:46:01 +01:00
|
|
|
|
```c
|
2021-08-28 08:48:47 +01:00
|
|
|
|
size_t mg_iobuf_add(struct mg_iobuf *io, size_t offset, const void *buf, size_t len, size_t align);
|
2021-08-28 08:46:01 +01:00
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Insert data buffer `buf`, `len` at offset `offset`. The iobuf gets 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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `io` - iobuf to add data
|
|
|
|
|
- `offset` - offswt to add data
|
|
|
|
|
- `buf` - data to add
|
|
|
|
|
- `len` - data lenth
|
|
|
|
|
- `align` - align boundary
|
|
|
|
|
|
|
|
|
|
Return value: new `io` len
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
2021-08-25 07:59:31 +01:00
|
|
|
|
|
2021-08-26 11:40:47 +01:00
|
|
|
|
```c
|
2021-08-28 08:46:01 +01:00
|
|
|
|
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
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
<img src="images/mg_iobuf_add.png">
|
|
|
|
|
|
2021-08-28 08:46:01 +01:00
|
|
|
|
### mg\_iobuf\_del()
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
size_t mg_iobuf_del(struct mg_iobuf *io, size_t offset, size_t len);
|
2021-08-25 07:59:31 +01:00
|
|
|
|
```
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `io` - iobuf to delete data
|
|
|
|
|
- `offset` - start offset
|
|
|
|
|
- `len` - amount of bytes to delete
|
|
|
|
|
|
|
|
|
|
Return value: new `io` len
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
<img src="images/mg_iobug_del.png">
|
|
|
|
|
|
|
|
|
|
## URL
|
|
|
|
|
|
|
|
|
|
### mg\_url\_port()
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
unsigned short mg_url_port(const char *url);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
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.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `url` - URL to extract host
|
|
|
|
|
|
|
|
|
|
Return value: host name
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_str host = mg_url_host("https://my.example.org:1234"); // host is now "my.example.org"
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### mg\_url\_user()
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
struct mg_str mg_url_user(const char *url);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Extract user name from given URL.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `url` - URL to extract user name
|
|
|
|
|
|
|
|
|
|
Return value: user name or empty string if not found
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Extract password from given URL.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `url` - URL to extract password
|
|
|
|
|
|
|
|
|
|
Return value: password or empty string if not found
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Extract URI from given URL.
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Note, that function returns pointer within `url`, no need to free() it explicitly.
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `url` - URL to extract URI
|
|
|
|
|
|
|
|
|
|
Return value: URI or `\` if not found
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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 macroses for logging. 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. 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 };
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `level` - log level, see levels above
|
|
|
|
|
- `args` - information to log
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
Usage example:
|
|
|
|
|
```c
|
|
|
|
|
LOG(LL_ERROR, ("Hello %s!", "world")); // Output "Hello, world"
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### mg\_log\_set()
|
|
|
|
|
|
|
|
|
|
```c
|
|
|
|
|
void mg_log_set(const char *spec);
|
|
|
|
|
```
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Set mongoose logging level.
|
|
|
|
|
|
|
|
|
|
Parameters:
|
|
|
|
|
- `spec` - string, containing log level, can be one of the following values:
|
2021-09-15 07:43:24 +01:00
|
|
|
|
|
|
|
|
|
- `0` - disable logging
|
|
|
|
|
- `1` - log errors only
|
|
|
|
|
- `2` - log errors and info messages
|
|
|
|
|
- `3` - log errors, into and debug messages
|
|
|
|
|
- `4` - log everything
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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`
|
|
|
|
|
|
2021-09-15 15:37:57 +03:00
|
|
|
|
Parameters:
|
|
|
|
|
- `fn` - callback function, should be called on logging
|
|
|
|
|
- `fnd` - user parameter to pass to `fn`
|
|
|
|
|
|
|
|
|
|
Return value: none
|
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
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,
|
|
|
|
|
and it is 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:<br>
|
|
|
|
|
`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);
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
<img src="images/packed.png">
|
2021-08-28 08:46:01 +01:00
|
|
|
|
|
2021-09-15 07:43:24 +01:00
|
|
|
|
<img src="images/packed2.png">
|