2020-04-14 17:19:12 +08:00
|
|
|
/***************************************************************************
|
|
|
|
* _ _ ____ _
|
|
|
|
* Project ___| | | | _ \| |
|
|
|
|
* / __| | | | |_) | |
|
|
|
|
* | (__| |_| | _ <| |___
|
|
|
|
* \___|\___/|_| \_\_____|
|
|
|
|
*
|
2023-01-02 20:51:48 +08:00
|
|
|
* Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
|
|
|
|
* Copyright (C) Björn Stenberg, <bjorn@haxx.se>
|
2020-04-14 17:19:12 +08:00
|
|
|
*
|
|
|
|
* This software is licensed as described in the file COPYING, which
|
|
|
|
* you should have received as part of this distribution. The terms
|
2020-11-04 21:02:01 +08:00
|
|
|
* are also available at https://curl.se/docs/copyright.html.
|
2020-04-14 17:19:12 +08:00
|
|
|
*
|
|
|
|
* You may opt to use, copy, modify, merge, publish, distribute and/or sell
|
|
|
|
* copies of the Software, and permit persons to whom the Software is
|
|
|
|
* furnished to do so, under the terms of the COPYING file.
|
|
|
|
*
|
|
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
* KIND, either express or implied.
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: curl
|
2022-05-17 17:16:50 +08:00
|
|
|
*
|
2020-04-14 17:19:12 +08:00
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#include "curl_setup.h"
|
|
|
|
|
2020-08-31 15:45:09 +08:00
|
|
|
#ifndef CURL_DISABLE_MQTT
|
2020-04-14 17:19:12 +08:00
|
|
|
|
|
|
|
#include "urldata.h"
|
|
|
|
#include <curl/curl.h>
|
|
|
|
#include "transfer.h"
|
|
|
|
#include "sendf.h"
|
|
|
|
#include "progress.h"
|
|
|
|
#include "mqtt.h"
|
|
|
|
#include "select.h"
|
|
|
|
#include "strdup.h"
|
|
|
|
#include "url.h"
|
|
|
|
#include "escape.h"
|
|
|
|
#include "warnless.h"
|
|
|
|
#include "curl_printf.h"
|
|
|
|
#include "curl_memory.h"
|
|
|
|
#include "multiif.h"
|
|
|
|
#include "rand.h"
|
|
|
|
|
|
|
|
/* The last #include file should be: */
|
|
|
|
#include "memdebug.h"
|
|
|
|
|
|
|
|
#define MQTT_MSG_CONNECT 0x10
|
|
|
|
#define MQTT_MSG_CONNACK 0x20
|
|
|
|
#define MQTT_MSG_PUBLISH 0x30
|
|
|
|
#define MQTT_MSG_SUBSCRIBE 0x82
|
|
|
|
#define MQTT_MSG_SUBACK 0x90
|
|
|
|
#define MQTT_MSG_DISCONNECT 0xe0
|
|
|
|
|
2020-04-16 19:20:52 +08:00
|
|
|
#define MQTT_CONNACK_LEN 2
|
|
|
|
#define MQTT_SUBACK_LEN 3
|
2020-04-14 17:19:12 +08:00
|
|
|
#define MQTT_CLIENTID_LEN 12 /* "curl0123abcd" */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Forward declarations.
|
|
|
|
*/
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_do(struct Curl_easy *data, bool *done);
|
2022-01-13 21:27:06 +08:00
|
|
|
static CURLcode mqtt_done(struct Curl_easy *data,
|
|
|
|
CURLcode status, bool premature);
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_doing(struct Curl_easy *data, bool *done);
|
|
|
|
static int mqtt_getsock(struct Curl_easy *data, struct connectdata *conn,
|
|
|
|
curl_socket_t *sock);
|
|
|
|
static CURLcode mqtt_setup_conn(struct Curl_easy *data,
|
|
|
|
struct connectdata *conn);
|
2020-04-14 17:19:12 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* MQTT protocol handler.
|
|
|
|
*/
|
|
|
|
|
|
|
|
const struct Curl_handler Curl_handler_mqtt = {
|
2024-05-07 22:55:23 +08:00
|
|
|
"mqtt", /* scheme */
|
2020-04-14 17:19:12 +08:00
|
|
|
mqtt_setup_conn, /* setup_connection */
|
|
|
|
mqtt_do, /* do_it */
|
2022-01-13 21:27:06 +08:00
|
|
|
mqtt_done, /* done */
|
2020-04-14 17:19:12 +08:00
|
|
|
ZERO_NULL, /* do_more */
|
|
|
|
ZERO_NULL, /* connect_it */
|
|
|
|
ZERO_NULL, /* connecting */
|
|
|
|
mqtt_doing, /* doing */
|
|
|
|
ZERO_NULL, /* proto_getsock */
|
|
|
|
mqtt_getsock, /* doing_getsock */
|
|
|
|
ZERO_NULL, /* domore_getsock */
|
|
|
|
ZERO_NULL, /* perform_getsock */
|
|
|
|
ZERO_NULL, /* disconnect */
|
lib: replace readwrite with write_resp
This clarifies the handling of server responses by folding the code for
the complicated protocols into their protocol handlers. This concerns
mainly HTTP and its bastard sibling RTSP.
The terms "read" and "write" are often used without clear context if
they refer to the connect or the client/application side of a
transfer. This PR uses "read/write" for operations on the client side
and "send/receive" for the connection, e.g. server side. If this is
considered useful, we can revisit renaming of further methods in another
PR.
Curl's protocol handler `readwrite()` method been changed:
```diff
- CURLcode (*readwrite)(struct Curl_easy *data, struct connectdata *conn,
- const char *buf, size_t blen,
- size_t *pconsumed, bool *readmore);
+ CURLcode (*write_resp)(struct Curl_easy *data, const char *buf, size_t blen,
+ bool is_eos, bool *done);
```
The name was changed to clarify that this writes reponse data to the
client side. The parameter changes are:
* `conn` removed as it always operates on `data->conn`
* `pconsumed` removed as the method needs to handle all data on success
* `readmore` removed as no longer necessary
* `is_eos` as indicator that this is the last call for the transfer
response (end-of-stream).
* `done` TRUE on return iff the transfer response is to be treated as
finished
This change affects many files only because of updated comments in
handlers that provide no implementation. The real change is that the
HTTP protocol handlers now provide an implementation.
The HTTP protocol handlers `write_resp()` implementation will get passed
**all** raw data of a server response for the transfer. The HTTP/1.x
formatted status and headers, as well as the undecoded response
body. `Curl_http_write_resp_hds()` is used internally to parse the
response headers and pass them on. This method is public as the RTSP
protocol handler also uses it.
HTTP/1.1 "chunked" transport encoding is now part of the general
*content encoding* writer stack, just like other encodings. A new flag
`CLIENTWRITE_EOS` was added for the last client write. This allows
writers to verify that they are in a valid end state. The chunked
decoder will check if it indeed has seen the last chunk.
The general response handling in `transfer.c:466` happens in function
`readwrite_data()`. This mainly operates now like:
```
static CURLcode readwrite_data(data, ...)
{
do {
Curl_xfer_recv_resp(data, buf)
...
Curl_xfer_write_resp(data, buf)
...
} while(interested);
...
}
```
All the response data handling is implemented in
`Curl_xfer_write_resp()`. It calls the protocol handler's `write_resp()`
implementation if available, or does the default behaviour.
All raw response data needs to pass through this function. Which also
means that anyone in possession of such data may call
`Curl_xfer_write_resp()`.
Closes #12480
2023-12-01 20:50:32 +08:00
|
|
|
ZERO_NULL, /* write_resp */
|
2024-03-21 19:15:59 +08:00
|
|
|
ZERO_NULL, /* write_resp_hd */
|
2020-04-14 17:19:12 +08:00
|
|
|
ZERO_NULL, /* connection_check */
|
2021-05-17 14:54:00 +08:00
|
|
|
ZERO_NULL, /* attach connection */
|
2020-04-14 17:19:12 +08:00
|
|
|
PORT_MQTT, /* defport */
|
|
|
|
CURLPROTO_MQTT, /* protocol */
|
2020-09-21 19:45:24 +08:00
|
|
|
CURLPROTO_MQTT, /* family */
|
2020-04-14 17:19:12 +08:00
|
|
|
PROTOPT_NONE /* flags */
|
|
|
|
};
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_setup_conn(struct Curl_easy *data,
|
|
|
|
struct connectdata *conn)
|
2020-04-14 17:19:12 +08:00
|
|
|
{
|
|
|
|
/* allocate the HTTP-specific struct for the Curl_easy, only to survive
|
|
|
|
during this request */
|
|
|
|
struct MQTT *mq;
|
2021-01-09 00:58:15 +08:00
|
|
|
(void)conn;
|
2020-11-23 15:32:41 +08:00
|
|
|
DEBUGASSERT(data->req.p.mqtt == NULL);
|
2020-04-14 17:19:12 +08:00
|
|
|
|
|
|
|
mq = calloc(1, sizeof(struct MQTT));
|
|
|
|
if(!mq)
|
|
|
|
return CURLE_OUT_OF_MEMORY;
|
2023-10-09 17:36:37 +08:00
|
|
|
Curl_dyn_init(&mq->recvbuf, DYN_MQTT_RECV);
|
2020-11-23 15:32:41 +08:00
|
|
|
data->req.p.mqtt = mq;
|
2020-04-14 17:19:12 +08:00
|
|
|
return CURLE_OK;
|
|
|
|
}
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_send(struct Curl_easy *data,
|
2020-04-14 17:19:12 +08:00
|
|
|
char *buf, size_t len)
|
|
|
|
{
|
|
|
|
CURLcode result = CURLE_OK;
|
2020-11-23 15:32:41 +08:00
|
|
|
struct MQTT *mq = data->req.p.mqtt;
|
lib: Curl_read/Curl_write clarifications
- replace `Curl_read()`, `Curl_write()` and `Curl_nwrite()` to
clarify when and at what level they operate
- send/recv of transfer related data is now done via
`Curl_xfer_send()/Curl_xfer_recv()` which no longer has
socket/socketindex as parameter. It decides on the transfer
setup of `conn->sockfd` and `conn->writesockfd` on which
connection filter chain to operate.
- send/recv on a specific connection filter chain is done via
`Curl_conn_send()/Curl_conn_recv()` which get the socket index
as parameter.
- rename `Curl_setup_transfer()` to `Curl_xfer_setup()` for
naming consistency
- clarify that the special CURLE_AGAIN hangling to return
`CURLE_OK` with length 0 only applies to `Curl_xfer_send()`
and CURLE_AGAIN is returned by all other send() variants.
- fix a bug in websocket `curl_ws_recv()` that mixed up data
when it arrived in more than a single chunk
The method for sending not just raw bytes, but bytes that are either
"headers" or "body". The send abstraction stack, to to bottom, now is:
* `Curl_req_send()`: has parameter to indicate amount of header bytes,
buffers all data.
* `Curl_xfer_send()`: knows on which socket index to send, returns
amount of bytes sent.
* `Curl_conn_send()`: called with socket index, returns amount of bytes
sent.
In addition there is `Curl_req_flush()` for writing out all buffered
bytes.
`Curl_req_send()` is active for requests without body,
`Curl_buffer_send()` still being used for others. This is because the
special quirks need to be addressed in future parts:
* `expect-100` handling
* `Curl_fillreadbuffer()` needs to add directly to the new
`data->req.sendbuf`
* special body handlings, like `chunked` encodings and line end
conversions will be moved into something like a Client Reader.
In functions of the pattern `CURLcode xxx_send(..., ssize_t *written)`,
replace the `ssize_t` with a `size_t`. It makes no sense to allow for negative
values as the returned `CURLcode` already specifies error conditions. This
allows easier handling of lengths without casting.
Closes #12964
2024-02-15 23:22:53 +08:00
|
|
|
size_t n;
|
2024-07-19 07:38:05 +08:00
|
|
|
result = Curl_xfer_send(data, buf, len, &n);
|
2023-02-27 17:02:22 +08:00
|
|
|
if(result)
|
|
|
|
return result;
|
|
|
|
Curl_debug(data, CURLINFO_HEADER_OUT, buf, (size_t)n);
|
lib: Curl_read/Curl_write clarifications
- replace `Curl_read()`, `Curl_write()` and `Curl_nwrite()` to
clarify when and at what level they operate
- send/recv of transfer related data is now done via
`Curl_xfer_send()/Curl_xfer_recv()` which no longer has
socket/socketindex as parameter. It decides on the transfer
setup of `conn->sockfd` and `conn->writesockfd` on which
connection filter chain to operate.
- send/recv on a specific connection filter chain is done via
`Curl_conn_send()/Curl_conn_recv()` which get the socket index
as parameter.
- rename `Curl_setup_transfer()` to `Curl_xfer_setup()` for
naming consistency
- clarify that the special CURLE_AGAIN hangling to return
`CURLE_OK` with length 0 only applies to `Curl_xfer_send()`
and CURLE_AGAIN is returned by all other send() variants.
- fix a bug in websocket `curl_ws_recv()` that mixed up data
when it arrived in more than a single chunk
The method for sending not just raw bytes, but bytes that are either
"headers" or "body". The send abstraction stack, to to bottom, now is:
* `Curl_req_send()`: has parameter to indicate amount of header bytes,
buffers all data.
* `Curl_xfer_send()`: knows on which socket index to send, returns
amount of bytes sent.
* `Curl_conn_send()`: called with socket index, returns amount of bytes
sent.
In addition there is `Curl_req_flush()` for writing out all buffered
bytes.
`Curl_req_send()` is active for requests without body,
`Curl_buffer_send()` still being used for others. This is because the
special quirks need to be addressed in future parts:
* `expect-100` handling
* `Curl_fillreadbuffer()` needs to add directly to the new
`data->req.sendbuf`
* special body handlings, like `chunked` encodings and line end
conversions will be moved into something like a Client Reader.
In functions of the pattern `CURLcode xxx_send(..., ssize_t *written)`,
replace the `ssize_t` with a `size_t`. It makes no sense to allow for negative
values as the returned `CURLcode` already specifies error conditions. This
allows easier handling of lengths without casting.
Closes #12964
2024-02-15 23:22:53 +08:00
|
|
|
if(len != n) {
|
2020-04-14 17:19:12 +08:00
|
|
|
size_t nsend = len - n;
|
|
|
|
char *sendleftovers = Curl_memdup(&buf[n], nsend);
|
|
|
|
if(!sendleftovers)
|
|
|
|
return CURLE_OUT_OF_MEMORY;
|
|
|
|
mq->sendleftovers = sendleftovers;
|
|
|
|
mq->nsend = nsend;
|
|
|
|
}
|
2021-08-24 15:50:33 +08:00
|
|
|
else {
|
|
|
|
mq->sendleftovers = NULL;
|
|
|
|
mq->nsend = 0;
|
|
|
|
}
|
2020-04-14 17:19:12 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Generic function called by the multi interface to figure out what socket(s)
|
|
|
|
to wait for and for what actions during the DOING and PROTOCONNECT
|
|
|
|
states */
|
2021-01-09 00:58:15 +08:00
|
|
|
static int mqtt_getsock(struct Curl_easy *data,
|
|
|
|
struct connectdata *conn,
|
2020-04-14 17:19:12 +08:00
|
|
|
curl_socket_t *sock)
|
|
|
|
{
|
2021-01-09 00:58:15 +08:00
|
|
|
(void)data;
|
2020-04-14 17:19:12 +08:00
|
|
|
sock[0] = conn->sock[FIRSTSOCKET];
|
|
|
|
return GETSOCK_READSOCK(FIRSTSOCKET);
|
|
|
|
}
|
|
|
|
|
2021-06-04 15:25:38 +08:00
|
|
|
static int mqtt_encode_len(char *buf, size_t len)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for(i = 0; (len > 0) && (i<4); i++) {
|
2024-06-03 04:30:52 +08:00
|
|
|
unsigned char encoded;
|
2021-06-04 15:25:38 +08:00
|
|
|
encoded = len % 0x80;
|
|
|
|
len /= 0x80;
|
|
|
|
if(len)
|
|
|
|
encoded |= 0x80;
|
2024-06-03 04:30:52 +08:00
|
|
|
buf[i] = (char)encoded;
|
2021-06-04 15:25:38 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* add the passwd to the CONNECT packet */
|
|
|
|
static int add_passwd(const char *passwd, const size_t plen,
|
|
|
|
char *pkt, const size_t start, int remain_pos)
|
|
|
|
{
|
|
|
|
/* magic number that need to be set properly */
|
|
|
|
const size_t conn_flags_pos = remain_pos + 8;
|
|
|
|
if(plen > 0xffff)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* set password flag */
|
|
|
|
pkt[conn_flags_pos] |= 0x40;
|
|
|
|
|
|
|
|
/* length of password provided */
|
|
|
|
pkt[start] = (char)((plen >> 8) & 0xFF);
|
|
|
|
pkt[start + 1] = (char)(plen & 0xFF);
|
|
|
|
memcpy(&pkt[start + 2], passwd, plen);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-10-17 20:48:33 +08:00
|
|
|
/* add user to the CONNECT packet */
|
2021-06-04 15:25:38 +08:00
|
|
|
static int add_user(const char *username, const size_t ulen,
|
|
|
|
unsigned char *pkt, const size_t start, int remain_pos)
|
|
|
|
{
|
|
|
|
/* magic number that need to be set properly */
|
|
|
|
const size_t conn_flags_pos = remain_pos + 8;
|
|
|
|
if(ulen > 0xffff)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* set username flag */
|
|
|
|
pkt[conn_flags_pos] |= 0x80;
|
|
|
|
/* length of username provided */
|
|
|
|
pkt[start] = (unsigned char)((ulen >> 8) & 0xFF);
|
|
|
|
pkt[start + 1] = (unsigned char)(ulen & 0xFF);
|
|
|
|
memcpy(&pkt[start + 2], username, ulen);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-10-17 20:48:33 +08:00
|
|
|
/* add client ID to the CONNECT packet */
|
2021-06-04 15:25:38 +08:00
|
|
|
static int add_client_id(const char *client_id, const size_t client_id_len,
|
|
|
|
char *pkt, const size_t start)
|
|
|
|
{
|
|
|
|
if(client_id_len != MQTT_CLIENTID_LEN)
|
|
|
|
return 1;
|
|
|
|
pkt[start] = 0x00;
|
|
|
|
pkt[start + 1] = MQTT_CLIENTID_LEN;
|
|
|
|
memcpy(&pkt[start + 2], client_id, MQTT_CLIENTID_LEN);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-10-17 20:48:33 +08:00
|
|
|
/* Set initial values of CONNECT packet */
|
2021-06-04 15:25:38 +08:00
|
|
|
static int init_connpack(char *packet, char *remain, int remain_pos)
|
|
|
|
{
|
|
|
|
/* Fixed header starts */
|
|
|
|
/* packet type */
|
|
|
|
packet[0] = MQTT_MSG_CONNECT;
|
|
|
|
/* remaining length field */
|
|
|
|
memcpy(&packet[1], remain, remain_pos);
|
|
|
|
/* Fixed header ends */
|
|
|
|
|
|
|
|
/* Variable header starts */
|
|
|
|
/* protocol length */
|
|
|
|
packet[remain_pos + 1] = 0x00;
|
|
|
|
packet[remain_pos + 2] = 0x04;
|
|
|
|
/* protocol name */
|
|
|
|
packet[remain_pos + 3] = 'M';
|
|
|
|
packet[remain_pos + 4] = 'Q';
|
|
|
|
packet[remain_pos + 5] = 'T';
|
|
|
|
packet[remain_pos + 6] = 'T';
|
|
|
|
/* protocol level */
|
|
|
|
packet[remain_pos + 7] = 0x04;
|
|
|
|
/* CONNECT flag: CleanSession */
|
|
|
|
packet[remain_pos + 8] = 0x02;
|
|
|
|
/* keep-alive 0 = disabled */
|
|
|
|
packet[remain_pos + 9] = 0x00;
|
|
|
|
packet[remain_pos + 10] = 0x3c;
|
2022-10-31 00:38:16 +08:00
|
|
|
/* end of variable header */
|
2021-06-04 15:25:38 +08:00
|
|
|
return remain_pos + 10;
|
|
|
|
}
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_connect(struct Curl_easy *data)
|
2020-04-14 17:19:12 +08:00
|
|
|
{
|
|
|
|
CURLcode result = CURLE_OK;
|
2021-06-04 15:25:38 +08:00
|
|
|
int pos = 0;
|
|
|
|
int rc = 0;
|
2022-10-31 00:38:16 +08:00
|
|
|
/* remain length */
|
2021-06-04 15:25:38 +08:00
|
|
|
int remain_pos = 0;
|
|
|
|
char remain[4] = {0};
|
|
|
|
size_t packetlen = 0;
|
|
|
|
size_t payloadlen = 0;
|
|
|
|
size_t start_user = 0;
|
|
|
|
size_t start_pwd = 0;
|
2020-04-14 17:19:12 +08:00
|
|
|
char client_id[MQTT_CLIENTID_LEN + 1] = "curl";
|
2020-09-02 18:46:48 +08:00
|
|
|
const size_t clen = strlen("curl");
|
2021-06-04 15:25:38 +08:00
|
|
|
char *packet = NULL;
|
|
|
|
|
|
|
|
/* extracting username from request */
|
|
|
|
const char *username = data->state.aptr.user ?
|
|
|
|
data->state.aptr.user : "";
|
|
|
|
const size_t ulen = strlen(username);
|
|
|
|
/* extracting password from request */
|
|
|
|
const char *passwd = data->state.aptr.passwd ?
|
|
|
|
data->state.aptr.passwd : "";
|
|
|
|
const size_t plen = strlen(passwd);
|
|
|
|
|
|
|
|
payloadlen = ulen + plen + MQTT_CLIENTID_LEN + 2;
|
|
|
|
/* The plus 2 are for the MSB and LSB describing the length of the string to
|
|
|
|
* be added on the payload. Refer to spec 1.5.2 and 1.5.4 */
|
|
|
|
if(ulen)
|
|
|
|
payloadlen += 2;
|
|
|
|
if(plen)
|
|
|
|
payloadlen += 2;
|
|
|
|
|
|
|
|
/* getting how much occupy the remain length */
|
|
|
|
remain_pos = mqtt_encode_len(remain, payloadlen + 10);
|
|
|
|
|
|
|
|
/* 10 length of variable header and 1 the first byte of the fixed header */
|
|
|
|
packetlen = payloadlen + 10 + remain_pos + 1;
|
|
|
|
|
|
|
|
/* allocating packet */
|
|
|
|
if(packetlen > 268435455)
|
|
|
|
return CURLE_WEIRD_SERVER_REPLY;
|
|
|
|
packet = malloc(packetlen);
|
|
|
|
if(!packet)
|
|
|
|
return CURLE_OUT_OF_MEMORY;
|
|
|
|
memset(packet, 0, packetlen);
|
|
|
|
|
2022-10-17 20:48:33 +08:00
|
|
|
/* set initial values for the CONNECT packet */
|
2021-06-04 15:25:38 +08:00
|
|
|
pos = init_connpack(packet, remain, remain_pos);
|
2020-04-14 17:19:12 +08:00
|
|
|
|
2023-09-12 18:51:21 +08:00
|
|
|
result = Curl_rand_alnum(data, (unsigned char *)&client_id[clen],
|
|
|
|
MQTT_CLIENTID_LEN - clen + 1);
|
2021-06-04 15:25:38 +08:00
|
|
|
/* add client id */
|
|
|
|
rc = add_client_id(client_id, strlen(client_id), packet, pos + 1);
|
|
|
|
if(rc) {
|
2023-09-17 13:04:17 +08:00
|
|
|
failf(data, "Client ID length mismatched: [%zu]", strlen(client_id));
|
2021-06-04 15:25:38 +08:00
|
|
|
result = CURLE_WEIRD_SERVER_REPLY;
|
|
|
|
goto end;
|
|
|
|
}
|
2021-07-06 23:05:17 +08:00
|
|
|
infof(data, "Using client id '%s'", client_id);
|
2021-06-04 15:25:38 +08:00
|
|
|
|
|
|
|
/* position where starts the user payload */
|
|
|
|
start_user = pos + 3 + MQTT_CLIENTID_LEN;
|
|
|
|
/* position where starts the password payload */
|
|
|
|
start_pwd = start_user + ulen;
|
2024-07-01 22:47:21 +08:00
|
|
|
/* if username was provided, add it to the packet */
|
2021-06-04 15:25:38 +08:00
|
|
|
if(ulen) {
|
|
|
|
start_pwd += 2;
|
|
|
|
|
|
|
|
rc = add_user(username, ulen,
|
|
|
|
(unsigned char *)packet, start_user, remain_pos);
|
|
|
|
if(rc) {
|
2023-09-17 13:04:17 +08:00
|
|
|
failf(data, "Username is too large: [%zu]", ulen);
|
2021-06-04 15:25:38 +08:00
|
|
|
result = CURLE_WEIRD_SERVER_REPLY;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if passwd was provided, add it to the packet */
|
|
|
|
if(plen) {
|
|
|
|
rc = add_passwd(passwd, plen, packet, start_pwd, remain_pos);
|
|
|
|
if(rc) {
|
2023-09-17 13:04:17 +08:00
|
|
|
failf(data, "Password is too large: [%zu]", plen);
|
2021-06-04 15:25:38 +08:00
|
|
|
result = CURLE_WEIRD_SERVER_REPLY;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-14 17:19:12 +08:00
|
|
|
if(!result)
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_send(data, packet, packetlen);
|
2021-06-04 15:25:38 +08:00
|
|
|
|
|
|
|
end:
|
|
|
|
if(packet)
|
|
|
|
free(packet);
|
|
|
|
Curl_safefree(data->state.aptr.user);
|
|
|
|
Curl_safefree(data->state.aptr.passwd);
|
2020-04-14 17:19:12 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_disconnect(struct Curl_easy *data)
|
2020-04-14 17:19:12 +08:00
|
|
|
{
|
|
|
|
CURLcode result = CURLE_OK;
|
2022-02-09 19:29:31 +08:00
|
|
|
struct MQTT *mq = data->req.p.mqtt;
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_send(data, (char *)"\xe0\x00", 2);
|
2022-02-09 19:29:31 +08:00
|
|
|
Curl_safefree(mq->sendleftovers);
|
2023-10-09 17:36:37 +08:00
|
|
|
Curl_dyn_free(&mq->recvbuf);
|
2020-04-14 17:19:12 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-10-09 17:36:37 +08:00
|
|
|
static CURLcode mqtt_recv_atleast(struct Curl_easy *data, size_t nbytes)
|
2020-04-14 17:19:12 +08:00
|
|
|
{
|
2023-10-09 17:36:37 +08:00
|
|
|
struct MQTT *mq = data->req.p.mqtt;
|
|
|
|
size_t rlen = Curl_dyn_len(&mq->recvbuf);
|
2020-04-14 17:19:12 +08:00
|
|
|
CURLcode result;
|
|
|
|
|
2023-10-09 17:36:37 +08:00
|
|
|
if(rlen < nbytes) {
|
|
|
|
unsigned char readbuf[1024];
|
|
|
|
ssize_t nread;
|
|
|
|
|
|
|
|
DEBUGASSERT(nbytes - rlen < sizeof(readbuf));
|
2024-02-14 19:09:32 +08:00
|
|
|
result = Curl_xfer_recv(data, (char *)readbuf, nbytes - rlen, &nread);
|
2023-10-09 17:36:37 +08:00
|
|
|
if(result)
|
|
|
|
return result;
|
|
|
|
DEBUGASSERT(nread >= 0);
|
|
|
|
if(Curl_dyn_addn(&mq->recvbuf, readbuf, (size_t)nread))
|
|
|
|
return CURLE_OUT_OF_MEMORY;
|
|
|
|
rlen = Curl_dyn_len(&mq->recvbuf);
|
|
|
|
}
|
|
|
|
return (rlen >= nbytes)? CURLE_OK : CURLE_AGAIN;
|
|
|
|
}
|
2020-04-14 17:19:12 +08:00
|
|
|
|
2023-10-09 17:36:37 +08:00
|
|
|
static void mqtt_recv_consume(struct Curl_easy *data, size_t nbytes)
|
|
|
|
{
|
|
|
|
struct MQTT *mq = data->req.p.mqtt;
|
|
|
|
size_t rlen = Curl_dyn_len(&mq->recvbuf);
|
|
|
|
if(rlen <= nbytes)
|
|
|
|
Curl_dyn_reset(&mq->recvbuf);
|
|
|
|
else
|
|
|
|
Curl_dyn_tail(&mq->recvbuf, rlen - nbytes);
|
|
|
|
}
|
2020-04-14 17:19:12 +08:00
|
|
|
|
2023-10-09 17:36:37 +08:00
|
|
|
static CURLcode mqtt_verify_connack(struct Curl_easy *data)
|
|
|
|
{
|
|
|
|
struct MQTT *mq = data->req.p.mqtt;
|
|
|
|
CURLcode result;
|
|
|
|
char *ptr;
|
|
|
|
|
|
|
|
result = mqtt_recv_atleast(data, MQTT_CONNACK_LEN);
|
|
|
|
if(result)
|
2020-04-14 17:19:12 +08:00
|
|
|
goto fail;
|
|
|
|
|
|
|
|
/* verify CONNACK */
|
2023-10-09 17:36:37 +08:00
|
|
|
DEBUGASSERT(Curl_dyn_len(&mq->recvbuf) >= MQTT_CONNACK_LEN);
|
|
|
|
ptr = Curl_dyn_ptr(&mq->recvbuf);
|
|
|
|
Curl_debug(data, CURLINFO_HEADER_IN, ptr, MQTT_CONNACK_LEN);
|
|
|
|
|
|
|
|
if(ptr[0] != 0x00 || ptr[1] != 0x00) {
|
2020-04-16 19:20:52 +08:00
|
|
|
failf(data, "Expected %02x%02x but got %02x%02x",
|
2023-10-09 17:36:37 +08:00
|
|
|
0x00, 0x00, ptr[0], ptr[1]);
|
|
|
|
Curl_dyn_reset(&mq->recvbuf);
|
2020-04-14 17:19:12 +08:00
|
|
|
result = CURLE_WEIRD_SERVER_REPLY;
|
2023-10-09 17:36:37 +08:00
|
|
|
goto fail;
|
2020-04-14 17:19:12 +08:00
|
|
|
}
|
2023-10-09 17:36:37 +08:00
|
|
|
mqtt_recv_consume(data, MQTT_CONNACK_LEN);
|
2020-04-14 17:19:12 +08:00
|
|
|
fail:
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_get_topic(struct Curl_easy *data,
|
2020-04-14 17:19:12 +08:00
|
|
|
char **topic, size_t *topiclen)
|
|
|
|
{
|
2021-01-09 00:58:15 +08:00
|
|
|
char *path = data->state.up.path;
|
2022-10-17 00:05:34 +08:00
|
|
|
CURLcode result = CURLE_URL_MALFORMAT;
|
|
|
|
if(strlen(path) > 1) {
|
|
|
|
result = Curl_urldecode(path + 1, 0, topic, topiclen, REJECT_NADA);
|
|
|
|
if(!result && (*topiclen > 0xffff)) {
|
|
|
|
failf(data, "Too long MQTT topic");
|
|
|
|
result = CURLE_URL_MALFORMAT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
failf(data, "No MQTT topic found. Forgot to URL encode it?");
|
|
|
|
|
|
|
|
return result;
|
2020-04-14 17:19:12 +08:00
|
|
|
}
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_subscribe(struct Curl_easy *data)
|
2020-04-14 17:19:12 +08:00
|
|
|
{
|
|
|
|
CURLcode result = CURLE_OK;
|
|
|
|
char *topic = NULL;
|
|
|
|
size_t topiclen;
|
|
|
|
unsigned char *packet = NULL;
|
|
|
|
size_t packetlen;
|
|
|
|
char encodedsize[4];
|
|
|
|
size_t n;
|
2021-01-09 00:58:15 +08:00
|
|
|
struct connectdata *conn = data->conn;
|
2020-04-14 17:19:12 +08:00
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_get_topic(data, &topic, &topiclen);
|
2020-04-14 17:19:12 +08:00
|
|
|
if(result)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
conn->proto.mqtt.packetid++;
|
|
|
|
|
|
|
|
packetlen = topiclen + 5; /* packetid + topic (has a two byte length field)
|
|
|
|
+ 2 bytes topic length + QoS byte */
|
|
|
|
n = mqtt_encode_len((char *)encodedsize, packetlen);
|
|
|
|
packetlen += n + 1; /* add one for the control packet type byte */
|
|
|
|
|
|
|
|
packet = malloc(packetlen);
|
|
|
|
if(!packet) {
|
|
|
|
result = CURLE_OUT_OF_MEMORY;
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
packet[0] = MQTT_MSG_SUBSCRIBE;
|
|
|
|
memcpy(&packet[1], encodedsize, n);
|
|
|
|
packet[1 + n] = (conn->proto.mqtt.packetid >> 8) & 0xff;
|
|
|
|
packet[2 + n] = conn->proto.mqtt.packetid & 0xff;
|
|
|
|
packet[3 + n] = (topiclen >> 8) & 0xff;
|
|
|
|
packet[4 + n ] = topiclen & 0xff;
|
|
|
|
memcpy(&packet[5 + n], topic, topiclen);
|
|
|
|
packet[5 + n + topiclen] = 0; /* QoS zero */
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_send(data, (char *)packet, packetlen);
|
2020-04-14 17:19:12 +08:00
|
|
|
|
|
|
|
fail:
|
|
|
|
free(topic);
|
|
|
|
free(packet);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-04-16 19:20:52 +08:00
|
|
|
/*
|
|
|
|
* Called when the first byte was already read.
|
|
|
|
*/
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_verify_suback(struct Curl_easy *data)
|
2020-04-14 17:19:12 +08:00
|
|
|
{
|
2023-10-09 17:36:37 +08:00
|
|
|
struct MQTT *mq = data->req.p.mqtt;
|
2021-01-09 00:58:15 +08:00
|
|
|
struct connectdata *conn = data->conn;
|
2020-04-14 17:19:12 +08:00
|
|
|
struct mqtt_conn *mqtt = &conn->proto.mqtt;
|
2023-10-09 17:36:37 +08:00
|
|
|
CURLcode result;
|
|
|
|
char *ptr;
|
2020-04-14 17:19:12 +08:00
|
|
|
|
2023-10-09 17:36:37 +08:00
|
|
|
result = mqtt_recv_atleast(data, MQTT_SUBACK_LEN);
|
2020-04-14 17:19:12 +08:00
|
|
|
if(result)
|
|
|
|
goto fail;
|
|
|
|
|
2023-10-09 17:36:37 +08:00
|
|
|
/* verify SUBACK */
|
|
|
|
DEBUGASSERT(Curl_dyn_len(&mq->recvbuf) >= MQTT_SUBACK_LEN);
|
|
|
|
ptr = Curl_dyn_ptr(&mq->recvbuf);
|
|
|
|
Curl_debug(data, CURLINFO_HEADER_IN, ptr, MQTT_SUBACK_LEN);
|
|
|
|
|
|
|
|
if(((unsigned char)ptr[0]) != ((mqtt->packetid >> 8) & 0xff) ||
|
|
|
|
((unsigned char)ptr[1]) != (mqtt->packetid & 0xff) ||
|
|
|
|
ptr[2] != 0x00) {
|
|
|
|
Curl_dyn_reset(&mq->recvbuf);
|
2020-04-14 17:19:12 +08:00
|
|
|
result = CURLE_WEIRD_SERVER_REPLY;
|
|
|
|
goto fail;
|
|
|
|
}
|
2023-10-09 17:36:37 +08:00
|
|
|
mqtt_recv_consume(data, MQTT_SUBACK_LEN);
|
2020-04-14 17:19:12 +08:00
|
|
|
fail:
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_publish(struct Curl_easy *data)
|
2020-04-14 17:19:12 +08:00
|
|
|
{
|
|
|
|
CURLcode result;
|
2021-01-09 00:58:15 +08:00
|
|
|
char *payload = data->set.postfields;
|
2020-12-17 20:34:38 +08:00
|
|
|
size_t payloadlen;
|
2020-04-14 17:19:12 +08:00
|
|
|
char *topic = NULL;
|
|
|
|
size_t topiclen;
|
|
|
|
unsigned char *pkt = NULL;
|
|
|
|
size_t i = 0;
|
|
|
|
size_t remaininglength;
|
|
|
|
size_t encodelen;
|
|
|
|
char encodedbytes[4];
|
2021-01-09 00:58:15 +08:00
|
|
|
curl_off_t postfieldsize = data->set.postfieldsize;
|
2020-12-17 20:34:38 +08:00
|
|
|
|
2024-01-08 17:34:06 +08:00
|
|
|
if(!payload) {
|
|
|
|
DEBUGF(infof(data, "mqtt_publish without payload, return bad arg"));
|
2020-12-17 20:34:38 +08:00
|
|
|
return CURLE_BAD_FUNCTION_ARGUMENT;
|
2024-01-08 17:34:06 +08:00
|
|
|
}
|
2020-12-17 20:34:38 +08:00
|
|
|
if(postfieldsize < 0)
|
|
|
|
payloadlen = strlen(payload);
|
|
|
|
else
|
|
|
|
payloadlen = (size_t)postfieldsize;
|
2020-04-14 17:19:12 +08:00
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_get_topic(data, &topic, &topiclen);
|
2020-04-14 17:19:12 +08:00
|
|
|
if(result)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
remaininglength = payloadlen + 2 + topiclen;
|
|
|
|
encodelen = mqtt_encode_len(encodedbytes, remaininglength);
|
|
|
|
|
|
|
|
/* add the control byte and the encoded remaining length */
|
|
|
|
pkt = malloc(remaininglength + 1 + encodelen);
|
|
|
|
if(!pkt) {
|
|
|
|
result = CURLE_OUT_OF_MEMORY;
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* assemble packet */
|
|
|
|
pkt[i++] = MQTT_MSG_PUBLISH;
|
|
|
|
memcpy(&pkt[i], encodedbytes, encodelen);
|
|
|
|
i += encodelen;
|
|
|
|
pkt[i++] = (topiclen >> 8) & 0xff;
|
|
|
|
pkt[i++] = (topiclen & 0xff);
|
|
|
|
memcpy(&pkt[i], topic, topiclen);
|
|
|
|
i += topiclen;
|
|
|
|
memcpy(&pkt[i], payload, payloadlen);
|
|
|
|
i += payloadlen;
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_send(data, (char *)pkt, i);
|
2020-04-14 17:19:12 +08:00
|
|
|
|
|
|
|
fail:
|
|
|
|
free(pkt);
|
|
|
|
free(topic);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t mqtt_decode_len(unsigned char *buf,
|
|
|
|
size_t buflen, size_t *lenbytes)
|
|
|
|
{
|
|
|
|
size_t len = 0;
|
|
|
|
size_t mult = 1;
|
|
|
|
size_t i;
|
|
|
|
unsigned char encoded = 128;
|
|
|
|
|
|
|
|
for(i = 0; (i < buflen) && (encoded & 128); i++) {
|
|
|
|
encoded = buf[i];
|
|
|
|
len += (encoded & 127) * mult;
|
|
|
|
mult *= 128;
|
|
|
|
}
|
|
|
|
|
2020-04-16 19:20:52 +08:00
|
|
|
if(lenbytes)
|
|
|
|
*lenbytes = i;
|
2020-04-14 17:19:12 +08:00
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2024-05-20 20:21:05 +08:00
|
|
|
#ifdef DEBUGBUILD
|
2020-04-16 19:20:52 +08:00
|
|
|
static const char *statenames[]={
|
|
|
|
"MQTT_FIRST",
|
|
|
|
"MQTT_REMAINING_LENGTH",
|
|
|
|
"MQTT_CONNACK",
|
|
|
|
"MQTT_SUBACK",
|
|
|
|
"MQTT_SUBACK_COMING",
|
|
|
|
"MQTT_PUBWAIT",
|
2020-04-21 05:27:04 +08:00
|
|
|
"MQTT_PUB_REMAIN",
|
|
|
|
|
|
|
|
"NOT A STATE"
|
2020-04-16 19:20:52 +08:00
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* The only way to change state */
|
2021-01-09 00:58:15 +08:00
|
|
|
static void mqstate(struct Curl_easy *data,
|
2020-04-16 19:20:52 +08:00
|
|
|
enum mqttstate state,
|
|
|
|
enum mqttstate nextstate) /* used if state == FIRST */
|
|
|
|
{
|
2021-01-09 00:58:15 +08:00
|
|
|
struct connectdata *conn = data->conn;
|
2020-04-16 19:20:52 +08:00
|
|
|
struct mqtt_conn *mqtt = &conn->proto.mqtt;
|
2024-05-20 20:21:05 +08:00
|
|
|
#ifdef DEBUGBUILD
|
2021-07-06 23:05:17 +08:00
|
|
|
infof(data, "%s (from %s) (next is %s)",
|
2020-04-16 19:20:52 +08:00
|
|
|
statenames[state],
|
|
|
|
statenames[mqtt->state],
|
|
|
|
(state == MQTT_FIRST)? statenames[nextstate] : "");
|
|
|
|
#endif
|
|
|
|
mqtt->state = state;
|
|
|
|
if(state == MQTT_FIRST)
|
|
|
|
mqtt->nextstate = nextstate;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_read_publish(struct Curl_easy *data, bool *done)
|
2020-04-14 17:19:12 +08:00
|
|
|
{
|
2020-04-16 19:20:52 +08:00
|
|
|
CURLcode result = CURLE_OK;
|
2021-01-09 00:58:15 +08:00
|
|
|
struct connectdata *conn = data->conn;
|
2020-04-14 17:19:12 +08:00
|
|
|
ssize_t nread;
|
2020-04-16 19:20:52 +08:00
|
|
|
size_t remlen;
|
2020-04-14 17:19:12 +08:00
|
|
|
struct mqtt_conn *mqtt = &conn->proto.mqtt;
|
2020-11-23 15:32:41 +08:00
|
|
|
struct MQTT *mq = data->req.p.mqtt;
|
2020-04-16 19:20:52 +08:00
|
|
|
unsigned char packet;
|
2020-04-14 17:19:12 +08:00
|
|
|
|
|
|
|
switch(mqtt->state) {
|
2023-05-18 12:54:18 +08:00
|
|
|
MQTT_SUBACK_COMING:
|
2020-04-16 19:20:52 +08:00
|
|
|
case MQTT_SUBACK_COMING:
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_verify_suback(data);
|
2020-04-14 17:19:12 +08:00
|
|
|
if(result)
|
2020-04-16 19:20:52 +08:00
|
|
|
break;
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
mqstate(data, MQTT_FIRST, MQTT_PUBWAIT);
|
2020-04-16 19:20:52 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MQTT_SUBACK:
|
|
|
|
case MQTT_PUBWAIT:
|
|
|
|
/* we are expecting PUBLISH or SUBACK */
|
|
|
|
packet = mq->firstbyte & 0xf0;
|
|
|
|
if(packet == MQTT_MSG_PUBLISH)
|
2021-01-09 00:58:15 +08:00
|
|
|
mqstate(data, MQTT_PUB_REMAIN, MQTT_NOSTATE);
|
2020-04-16 19:20:52 +08:00
|
|
|
else if(packet == MQTT_MSG_SUBACK) {
|
2021-01-09 00:58:15 +08:00
|
|
|
mqstate(data, MQTT_SUBACK_COMING, MQTT_NOSTATE);
|
2020-04-16 19:20:52 +08:00
|
|
|
goto MQTT_SUBACK_COMING;
|
|
|
|
}
|
|
|
|
else if(packet == MQTT_MSG_DISCONNECT) {
|
2021-07-06 23:05:17 +08:00
|
|
|
infof(data, "Got DISCONNECT");
|
2020-04-16 19:20:52 +08:00
|
|
|
*done = TRUE;
|
2020-04-14 17:19:12 +08:00
|
|
|
goto end;
|
2020-04-16 19:20:52 +08:00
|
|
|
}
|
|
|
|
else {
|
2020-04-14 17:19:12 +08:00
|
|
|
result = CURLE_WEIRD_SERVER_REPLY;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* -- switched state -- */
|
2020-04-16 19:20:52 +08:00
|
|
|
remlen = mq->remaining_length;
|
2023-06-16 11:02:48 +08:00
|
|
|
infof(data, "Remaining length: %zu bytes", remlen);
|
2021-06-01 23:22:31 +08:00
|
|
|
if(data->set.max_filesize &&
|
|
|
|
(curl_off_t)remlen > data->set.max_filesize) {
|
|
|
|
failf(data, "Maximum file size exceeded");
|
|
|
|
result = CURLE_FILESIZE_EXCEEDED;
|
|
|
|
goto end;
|
|
|
|
}
|
2020-04-14 17:19:12 +08:00
|
|
|
Curl_pgrsSetDownloadSize(data, remlen);
|
|
|
|
data->req.bytecount = 0;
|
|
|
|
data->req.size = remlen;
|
|
|
|
mq->npacket = remlen; /* get this many bytes */
|
2023-12-08 21:05:09 +08:00
|
|
|
FALLTHROUGH();
|
2020-04-16 19:20:52 +08:00
|
|
|
case MQTT_PUB_REMAIN: {
|
2020-04-14 17:19:12 +08:00
|
|
|
/* read rest of packet, but no more. Cap to buffer size */
|
2024-01-25 22:33:54 +08:00
|
|
|
char buffer[4*1024];
|
2020-04-14 17:19:12 +08:00
|
|
|
size_t rest = mq->npacket;
|
2024-01-25 22:33:54 +08:00
|
|
|
if(rest > sizeof(buffer))
|
|
|
|
rest = sizeof(buffer);
|
2024-02-14 19:09:32 +08:00
|
|
|
result = Curl_xfer_recv(data, buffer, rest, &nread);
|
2020-04-14 17:19:12 +08:00
|
|
|
if(result) {
|
|
|
|
if(CURLE_AGAIN == result) {
|
2021-07-06 23:05:17 +08:00
|
|
|
infof(data, "EEEE AAAAGAIN");
|
2020-04-14 17:19:12 +08:00
|
|
|
}
|
|
|
|
goto end;
|
|
|
|
}
|
2020-04-16 19:20:52 +08:00
|
|
|
if(!nread) {
|
2021-07-06 23:05:17 +08:00
|
|
|
infof(data, "server disconnected");
|
2020-04-16 19:20:52 +08:00
|
|
|
result = CURLE_PARTIAL_FILE;
|
|
|
|
goto end;
|
|
|
|
}
|
2020-04-14 17:19:12 +08:00
|
|
|
|
|
|
|
/* if QoS is set, message contains packet id */
|
2024-01-25 22:33:54 +08:00
|
|
|
result = Curl_client_write(data, CLIENTWRITE_BODY, buffer, nread);
|
2020-04-14 17:19:12 +08:00
|
|
|
if(result)
|
|
|
|
goto end;
|
|
|
|
|
2024-01-25 22:33:54 +08:00
|
|
|
mq->npacket -= nread;
|
2020-04-14 17:19:12 +08:00
|
|
|
if(!mq->npacket)
|
|
|
|
/* no more PUBLISH payload, back to subscribe wait state */
|
2021-01-09 00:58:15 +08:00
|
|
|
mqstate(data, MQTT_FIRST, MQTT_PUBWAIT);
|
2020-04-14 17:19:12 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
DEBUGASSERT(NULL); /* illegal state */
|
|
|
|
result = CURLE_WEIRD_SERVER_REPLY;
|
|
|
|
goto end;
|
|
|
|
}
|
2023-05-18 12:54:18 +08:00
|
|
|
end:
|
2020-04-14 17:19:12 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_do(struct Curl_easy *data, bool *done)
|
2020-04-14 17:19:12 +08:00
|
|
|
{
|
|
|
|
CURLcode result = CURLE_OK;
|
|
|
|
*done = FALSE; /* unconditionally */
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_connect(data);
|
2020-04-14 17:19:12 +08:00
|
|
|
if(result) {
|
2022-10-17 20:48:33 +08:00
|
|
|
failf(data, "Error %d sending MQTT CONNECT request", result);
|
2020-04-14 17:19:12 +08:00
|
|
|
return result;
|
|
|
|
}
|
2021-01-09 00:58:15 +08:00
|
|
|
mqstate(data, MQTT_FIRST, MQTT_CONNACK);
|
2020-04-14 17:19:12 +08:00
|
|
|
return CURLE_OK;
|
|
|
|
}
|
|
|
|
|
2022-01-13 21:27:06 +08:00
|
|
|
static CURLcode mqtt_done(struct Curl_easy *data,
|
|
|
|
CURLcode status, bool premature)
|
|
|
|
{
|
|
|
|
struct MQTT *mq = data->req.p.mqtt;
|
|
|
|
(void)status;
|
|
|
|
(void)premature;
|
|
|
|
Curl_safefree(mq->sendleftovers);
|
2023-10-09 17:36:37 +08:00
|
|
|
Curl_dyn_free(&mq->recvbuf);
|
2022-01-13 21:27:06 +08:00
|
|
|
return CURLE_OK;
|
|
|
|
}
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
static CURLcode mqtt_doing(struct Curl_easy *data, bool *done)
|
2020-04-14 17:19:12 +08:00
|
|
|
{
|
|
|
|
CURLcode result = CURLE_OK;
|
2021-01-09 00:58:15 +08:00
|
|
|
struct connectdata *conn = data->conn;
|
2020-04-14 17:19:12 +08:00
|
|
|
struct mqtt_conn *mqtt = &conn->proto.mqtt;
|
2020-11-23 15:32:41 +08:00
|
|
|
struct MQTT *mq = data->req.p.mqtt;
|
2020-04-16 19:20:52 +08:00
|
|
|
ssize_t nread;
|
2024-06-30 17:37:06 +08:00
|
|
|
unsigned char recvbyte;
|
2020-04-14 17:19:12 +08:00
|
|
|
|
|
|
|
*done = FALSE;
|
|
|
|
|
|
|
|
if(mq->nsend) {
|
|
|
|
/* send the remainder of an outgoing packet */
|
|
|
|
char *ptr = mq->sendleftovers;
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_send(data, mq->sendleftovers, mq->nsend);
|
2020-04-14 17:19:12 +08:00
|
|
|
free(ptr);
|
|
|
|
if(result)
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-07-06 23:05:17 +08:00
|
|
|
infof(data, "mqtt_doing: state [%d]", (int) mqtt->state);
|
2020-04-14 17:19:12 +08:00
|
|
|
switch(mqtt->state) {
|
2020-04-16 19:20:52 +08:00
|
|
|
case MQTT_FIRST:
|
|
|
|
/* Read the initial byte only */
|
2024-02-14 19:09:32 +08:00
|
|
|
result = Curl_xfer_recv(data, (char *)&mq->firstbyte, 1, &nread);
|
2022-03-29 14:43:13 +08:00
|
|
|
if(result)
|
|
|
|
break;
|
|
|
|
else if(!nread) {
|
|
|
|
failf(data, "Connection disconnected");
|
|
|
|
*done = TRUE;
|
|
|
|
result = CURLE_RECV_ERROR;
|
2020-04-16 19:20:52 +08:00
|
|
|
break;
|
2022-03-29 14:43:13 +08:00
|
|
|
}
|
2020-11-03 00:34:04 +08:00
|
|
|
Curl_debug(data, CURLINFO_HEADER_IN, (char *)&mq->firstbyte, 1);
|
2020-04-16 19:20:52 +08:00
|
|
|
/* remember the first byte */
|
|
|
|
mq->npacket = 0;
|
2021-01-09 00:58:15 +08:00
|
|
|
mqstate(data, MQTT_REMAINING_LENGTH, MQTT_NOSTATE);
|
2023-12-08 21:05:09 +08:00
|
|
|
FALLTHROUGH();
|
2020-04-16 19:20:52 +08:00
|
|
|
case MQTT_REMAINING_LENGTH:
|
|
|
|
do {
|
2024-06-30 17:37:06 +08:00
|
|
|
result = Curl_xfer_recv(data, (char *)&recvbyte, 1, &nread);
|
2024-04-19 15:32:23 +08:00
|
|
|
if(result || !nread)
|
2020-04-16 19:20:52 +08:00
|
|
|
break;
|
2024-06-30 17:37:06 +08:00
|
|
|
Curl_debug(data, CURLINFO_HEADER_IN, (char *)&recvbyte, 1);
|
|
|
|
mq->pkt_hd[mq->npacket++] = recvbyte;
|
|
|
|
} while((recvbyte & 0x80) && (mq->npacket < 4));
|
|
|
|
if(!result && nread && (recvbyte & 0x80))
|
2021-06-01 23:22:31 +08:00
|
|
|
/* MQTT supports up to 127 * 128^0 + 127 * 128^1 + 127 * 128^2 +
|
|
|
|
127 * 128^3 bytes. server tried to send more */
|
|
|
|
result = CURLE_WEIRD_SERVER_REPLY;
|
2020-04-20 23:03:18 +08:00
|
|
|
if(result)
|
|
|
|
break;
|
2024-01-26 16:21:37 +08:00
|
|
|
mq->remaining_length = mqtt_decode_len(mq->pkt_hd, mq->npacket, NULL);
|
2020-04-16 19:20:52 +08:00
|
|
|
mq->npacket = 0;
|
|
|
|
if(mq->remaining_length) {
|
2021-01-09 00:58:15 +08:00
|
|
|
mqstate(data, mqtt->nextstate, MQTT_NOSTATE);
|
2020-04-16 19:20:52 +08:00
|
|
|
break;
|
|
|
|
}
|
2021-01-09 00:58:15 +08:00
|
|
|
mqstate(data, MQTT_FIRST, MQTT_FIRST);
|
2020-04-16 19:20:52 +08:00
|
|
|
|
|
|
|
if(mq->firstbyte == MQTT_MSG_DISCONNECT) {
|
2021-07-06 23:05:17 +08:00
|
|
|
infof(data, "Got DISCONNECT");
|
2020-04-16 19:20:52 +08:00
|
|
|
*done = TRUE;
|
|
|
|
}
|
|
|
|
break;
|
2020-04-14 17:19:12 +08:00
|
|
|
case MQTT_CONNACK:
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_verify_connack(data);
|
2020-04-14 17:19:12 +08:00
|
|
|
if(result)
|
|
|
|
break;
|
|
|
|
|
2021-01-09 00:58:15 +08:00
|
|
|
if(data->state.httpreq == HTTPREQ_POST) {
|
|
|
|
result = mqtt_publish(data);
|
2020-04-14 17:19:12 +08:00
|
|
|
if(!result) {
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_disconnect(data);
|
2020-04-14 17:19:12 +08:00
|
|
|
*done = TRUE;
|
|
|
|
}
|
2020-04-16 19:20:52 +08:00
|
|
|
mqtt->nextstate = MQTT_FIRST;
|
2020-04-14 17:19:12 +08:00
|
|
|
}
|
|
|
|
else {
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_subscribe(data);
|
2020-04-16 19:20:52 +08:00
|
|
|
if(!result) {
|
2021-01-09 00:58:15 +08:00
|
|
|
mqstate(data, MQTT_FIRST, MQTT_SUBACK);
|
2020-04-16 19:20:52 +08:00
|
|
|
}
|
2020-04-14 17:19:12 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MQTT_SUBACK:
|
2020-04-16 19:20:52 +08:00
|
|
|
case MQTT_PUBWAIT:
|
|
|
|
case MQTT_PUB_REMAIN:
|
2021-01-09 00:58:15 +08:00
|
|
|
result = mqtt_read_publish(data, done);
|
2020-04-14 17:19:12 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2021-01-09 00:58:15 +08:00
|
|
|
failf(data, "State not handled yet");
|
2020-04-14 17:19:12 +08:00
|
|
|
*done = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(result == CURLE_AGAIN)
|
|
|
|
result = CURLE_OK;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-08-31 15:45:09 +08:00
|
|
|
#endif /* CURL_DISABLE_MQTT */
|