diff options
author | dartraiden <wowemuh@gmail.com> | 2025-06-04 09:49:23 +0300 |
---|---|---|
committer | dartraiden <wowemuh@gmail.com> | 2025-06-04 10:27:11 +0300 |
commit | 86d1a677fd310d7d90d6f7545c02a4bd68e1d955 (patch) | |
tree | 7fd5f46ef18038d10dcdf9fa19ffee547d51d6ad /libs/libcurl/src/imap.c | |
parent | e8e2a816fbbcec0d6a64496928fecff19c281d82 (diff) |
libcurl: update to 8.14.0
Diffstat (limited to 'libs/libcurl/src/imap.c')
-rw-r--r-- | libs/libcurl/src/imap.c | 682 |
1 files changed, 411 insertions, 271 deletions
diff --git a/libs/libcurl/src/imap.c b/libs/libcurl/src/imap.c index f28c1a7e75..7a84ed473b 100644 --- a/libs/libcurl/src/imap.c +++ b/libs/libcurl/src/imap.c @@ -36,7 +36,7 @@ ***************************************************************************/
#include "curl_setup.h"
-#include "dynbuf.h"
+#include "curlx/dynbuf.h"
#ifndef CURL_DISABLE_IMAP
@@ -65,7 +65,7 @@ #include "socks.h"
#include "imap.h"
#include "mime.h"
-#include "strparse.h"
+#include "curlx/strparse.h"
#include "strcase.h"
#include "vtls/vtls.h"
#include "cfilters.h"
@@ -75,7 +75,7 @@ #include "url.h"
#include "bufref.h"
#include "curl_sasl.h"
-#include "warnless.h"
+#include "curlx/warnless.h"
#include "curl_ctype.h"
/* The last 3 #include files should be in this order */
@@ -83,8 +83,73 @@ #include "curl_memory.h"
#include "memdebug.h"
+
+/* meta key for storing protocol meta at easy handle */
+#define CURL_META_IMAP_EASY "meta:proto:imap:easy"
+/* meta key for storing protocol meta at connection */
+#define CURL_META_IMAP_CONN "meta:proto:imap:conn"
+
+typedef enum {
+ IMAP_STOP, /* do nothing state, stops the state machine */
+ IMAP_SERVERGREET, /* waiting for the initial greeting immediately after
+ a connect */
+ IMAP_CAPABILITY,
+ IMAP_STARTTLS,
+ IMAP_UPGRADETLS, /* asynchronously upgrade the connection to SSL/TLS
+ (multi mode only) */
+ IMAP_AUTHENTICATE,
+ IMAP_LOGIN,
+ IMAP_LIST,
+ IMAP_SELECT,
+ IMAP_FETCH,
+ IMAP_FETCH_FINAL,
+ IMAP_APPEND,
+ IMAP_APPEND_FINAL,
+ IMAP_SEARCH,
+ IMAP_LOGOUT,
+ IMAP_LAST /* never used */
+} imapstate;
+
+/* imap_conn is used for struct connection-oriented data */
+struct imap_conn {
+ struct pingpong pp;
+ struct SASL sasl; /* SASL-related parameters */
+ struct dynbuf dyn; /* for the IMAP commands */
+ char *mailbox; /* The last selected mailbox */
+ char *mailbox_uidvalidity; /* UIDVALIDITY parsed from select response */
+ imapstate state; /* Always use imap.c:state() to change state! */
+ char resptag[5]; /* Response tag to wait for */
+ unsigned char preftype; /* Preferred authentication type */
+ unsigned char cmdid; /* Last used command ID */
+ BIT(ssldone); /* Is connect() over SSL done? */
+ BIT(preauth); /* Is this connection PREAUTH? */
+ BIT(tls_supported); /* StartTLS capability supported by server */
+ BIT(login_disabled); /* LOGIN command disabled by server */
+ BIT(ir_supported); /* Initial response supported by server */
+};
+
+/* This IMAP struct is used in the Curl_easy. All IMAP data that is
+ connection-oriented must be in imap_conn to properly deal with the fact that
+ perhaps the Curl_easy is changed between the times the connection is
+ used. */
+struct IMAP {
+ curl_pp_transfer transfer;
+ char *mailbox; /* Mailbox to select */
+ char *uidvalidity; /* UIDVALIDITY to check in select */
+ char *uid; /* Message UID to fetch */
+ char *mindex; /* Index in mail box of mail to fetch */
+ char *section; /* Message SECTION to fetch */
+ char *partial; /* Message PARTIAL to fetch */
+ char *query; /* Query to search for */
+ char *custom; /* Custom request */
+ char *custom_params; /* Parameters for the custom request */
+};
+
+
/* Local API functions */
-static CURLcode imap_regular_transfer(struct Curl_easy *data, bool *done);
+static CURLcode imap_regular_transfer(struct Curl_easy *data,
+ struct IMAP *imap,
+ bool *done);
static CURLcode imap_do(struct Curl_easy *data, bool *done);
static CURLcode imap_done(struct Curl_easy *data, CURLcode status,
bool premature);
@@ -98,11 +163,15 @@ static CURLcode imap_doing(struct Curl_easy *data, bool *dophase_done); static CURLcode imap_setup_connection(struct Curl_easy *data,
struct connectdata *conn);
static char *imap_atom(const char *str, bool escape_only);
-static CURLcode imap_sendf(struct Curl_easy *data, const char *fmt, ...)
- CURL_PRINTF(2, 3);
-static CURLcode imap_parse_url_options(struct connectdata *conn);
-static CURLcode imap_parse_url_path(struct Curl_easy *data);
-static CURLcode imap_parse_custom_request(struct Curl_easy *data);
+static CURLcode imap_sendf(struct Curl_easy *data,
+ struct imap_conn *imapc,
+ const char *fmt, ...) CURL_PRINTF(3, 4);
+static CURLcode imap_parse_url_options(struct connectdata *conn,
+ struct imap_conn *imapc);
+static CURLcode imap_parse_url_path(struct Curl_easy *data,
+ struct IMAP *imap);
+static CURLcode imap_parse_custom_request(struct Curl_easy *data,
+ struct IMAP *imap);
static CURLcode imap_perform_authenticate(struct Curl_easy *data,
const char *mech,
const struct bufref *initresp);
@@ -112,6 +181,7 @@ static CURLcode imap_continue_authenticate(struct Curl_easy *data, static CURLcode imap_cancel_authenticate(struct Curl_easy *data,
const char *mech);
static CURLcode imap_get_message(struct Curl_easy *data, struct bufref *out);
+static void imap_easy_reset(struct IMAP *imap);
/*
* IMAP protocol handler.
@@ -249,12 +319,19 @@ static bool imap_matchresp(const char *line, size_t len, const char *cmd) static bool imap_endofresp(struct Curl_easy *data, struct connectdata *conn,
const char *line, size_t len, int *resp)
{
- struct IMAP *imap = data->req.p.imap;
- struct imap_conn *imapc = &conn->proto.imapc;
- const char *id = imapc->resptag;
- size_t id_len = strlen(id);
+ struct imap_conn *imapc = Curl_conn_meta_get(conn, CURL_META_IMAP_CONN);
+ struct IMAP *imap = Curl_meta_get(data, CURL_META_IMAP_EASY);
+ const char *id;
+ size_t id_len;
+
+ DEBUGASSERT(imapc);
+ DEBUGASSERT(imap);
+ if(!imapc || !imap)
+ return FALSE;
/* Do we have a tagged command response? */
+ id = imapc->resptag;
+ id_len = strlen(id);
if(len >= id_len + 1 && !memcmp(id, line, id_len) && line[id_len] == ' ') {
line += id_len + 1;
len -= id_len + 1;
@@ -280,17 +357,17 @@ static bool imap_endofresp(struct Curl_easy *data, struct connectdata *conn, case IMAP_LIST:
if((!imap->custom && !imap_matchresp(line, len, "LIST")) ||
- (imap->custom && !imap_matchresp(line, len, imap->custom) &&
- (!strcasecompare(imap->custom, "STORE") ||
- !imap_matchresp(line, len, "FETCH")) &&
- !strcasecompare(imap->custom, "SELECT") &&
- !strcasecompare(imap->custom, "EXAMINE") &&
- !strcasecompare(imap->custom, "SEARCH") &&
- !strcasecompare(imap->custom, "EXPUNGE") &&
- !strcasecompare(imap->custom, "LSUB") &&
- !strcasecompare(imap->custom, "UID") &&
- !strcasecompare(imap->custom, "GETQUOTAROOT") &&
- !strcasecompare(imap->custom, "NOOP")))
+ (imap->custom && !imap_matchresp(line, len, imap->custom) &&
+ (!strcasecompare(imap->custom, "STORE") ||
+ !imap_matchresp(line, len, "FETCH")) &&
+ !strcasecompare(imap->custom, "SELECT") &&
+ !strcasecompare(imap->custom, "EXAMINE") &&
+ !strcasecompare(imap->custom, "SEARCH") &&
+ !strcasecompare(imap->custom, "EXPUNGE") &&
+ !strcasecompare(imap->custom, "LSUB") &&
+ !strcasecompare(imap->custom, "UID") &&
+ !strcasecompare(imap->custom, "GETQUOTAROOT") &&
+ !strcasecompare(imap->custom, "NOOP")))
return FALSE;
break;
@@ -322,8 +399,8 @@ static bool imap_endofresp(struct Curl_easy *data, struct connectdata *conn, a space and optionally some text as per RFC-3501 for the AUTHENTICATE and
APPEND commands and as outlined in Section 4. Examples of RFC-4959 but
some email servers ignore this and only send a single + instead. */
- if(imap && !imap->custom && ((len == 3 && line[0] == '+') ||
- (len >= 2 && !memcmp("+ ", line, 2)))) {
+ if(!imap->custom && ((len == 3 && line[0] == '+') ||
+ (len >= 2 && !memcmp("+ ", line, 2)))) {
switch(imapc->state) {
/* States which are interested in continuation responses */
case IMAP_AUTHENTICATE:
@@ -351,9 +428,16 @@ static bool imap_endofresp(struct Curl_easy *data, struct connectdata *conn, */
static CURLcode imap_get_message(struct Curl_easy *data, struct bufref *out)
{
- char *message = Curl_dyn_ptr(&data->conn->proto.imapc.pp.recvbuf);
- size_t len = data->conn->proto.imapc.pp.nfinal;
+ struct imap_conn *imapc =
+ Curl_conn_meta_get(data->conn, CURL_META_IMAP_CONN);
+ char *message;
+ size_t len;
+
+ if(!imapc)
+ return CURLE_FAILED_INIT;
+ message = curlx_dyn_ptr(&imapc->pp.recvbuf);
+ len = imapc->pp.nfinal;
if(len > 2) {
/* Find the start of the message */
len -= 2;
@@ -383,9 +467,10 @@ static CURLcode imap_get_message(struct Curl_easy *data, struct bufref *out) *
* This is the ONLY way to change IMAP state!
*/
-static void imap_state(struct Curl_easy *data, imapstate newstate)
+static void imap_state(struct Curl_easy *data,
+ struct imap_conn *imapc,
+ imapstate newstate)
{
- struct imap_conn *imapc = &data->conn->proto.imapc;
#if defined(DEBUGBUILD) && !defined(CURL_DISABLE_VERBOSE_STRINGS)
/* for debug purposes */
static const char * const names[]={
@@ -411,7 +496,7 @@ static void imap_state(struct Curl_easy *data, imapstate newstate) infof(data, "IMAP %p state change from %s to %s",
(void *)imapc, names[imapc->state], names[newstate]);
#endif
-
+ (void)data;
imapc->state = newstate;
}
@@ -423,19 +508,19 @@ static void imap_state(struct Curl_easy *data, imapstate newstate) * supported capabilities.
*/
static CURLcode imap_perform_capability(struct Curl_easy *data,
- struct connectdata *conn)
+ struct imap_conn *imapc)
{
CURLcode result = CURLE_OK;
- struct imap_conn *imapc = &conn->proto.imapc;
+
imapc->sasl.authmechs = SASL_AUTH_NONE; /* No known auth. mechanisms yet */
imapc->sasl.authused = SASL_AUTH_NONE; /* Clear the auth. mechanism used */
imapc->tls_supported = FALSE; /* Clear the TLS capability */
/* Send the CAPABILITY command */
- result = imap_sendf(data, "CAPABILITY");
+ result = imap_sendf(data, imapc, "CAPABILITY");
if(!result)
- imap_state(data, IMAP_CAPABILITY);
+ imap_state(data, imapc, IMAP_CAPABILITY);
return result;
}
@@ -446,13 +531,14 @@ static CURLcode imap_perform_capability(struct Curl_easy *data, *
* Sends the STARTTLS command to start the upgrade to TLS.
*/
-static CURLcode imap_perform_starttls(struct Curl_easy *data)
+static CURLcode imap_perform_starttls(struct Curl_easy *data,
+ struct imap_conn *imapc)
{
/* Send the STARTTLS command */
- CURLcode result = imap_sendf(data, "STARTTLS");
+ CURLcode result = imap_sendf(data, imapc, "STARTTLS");
if(!result)
- imap_state(data, IMAP_STARTTLS);
+ imap_state(data, imapc, IMAP_STARTTLS);
return result;
}
@@ -464,11 +550,11 @@ static CURLcode imap_perform_starttls(struct Curl_easy *data) * Performs the upgrade to TLS.
*/
static CURLcode imap_perform_upgrade_tls(struct Curl_easy *data,
+ struct imap_conn *imapc,
struct connectdata *conn)
{
#ifdef USE_SSL
/* Start the SSL connection */
- struct imap_conn *imapc = &conn->proto.imapc;
CURLcode result;
bool ssldone = FALSE;
@@ -487,12 +573,13 @@ static CURLcode imap_perform_upgrade_tls(struct Curl_easy *data, if(!result && ssldone) {
imapc->ssldone = ssldone;
/* perform CAPA now, changes imapc->state out of IMAP_UPGRADETLS */
- result = imap_perform_capability(data, conn);
+ result = imap_perform_capability(data, imapc);
}
out:
return result;
#else
(void)data;
+ (void)imapc;
(void)conn;
return CURLE_NOT_BUILT_IN;
#endif
@@ -505,6 +592,7 @@ out: * Sends a clear text LOGIN command to authenticate with.
*/
static CURLcode imap_perform_login(struct Curl_easy *data,
+ struct imap_conn *imapc,
struct connectdata *conn)
{
CURLcode result = CURLE_OK;
@@ -514,7 +602,7 @@ static CURLcode imap_perform_login(struct Curl_easy *data, /* Check we have a username and password to authenticate with and end the
connect phase if we do not */
if(!data->state.aptr.user) {
- imap_state(data, IMAP_STOP);
+ imap_state(data, imapc, IMAP_STOP);
return result;
}
@@ -524,14 +612,14 @@ static CURLcode imap_perform_login(struct Curl_easy *data, passwd = imap_atom(conn->passwd, FALSE);
/* Send the LOGIN command */
- result = imap_sendf(data, "LOGIN %s %s", user ? user : "",
+ result = imap_sendf(data, imapc, "LOGIN %s %s", user ? user : "",
passwd ? passwd : "");
free(user);
free(passwd);
if(!result)
- imap_state(data, IMAP_LOGIN);
+ imap_state(data, imapc, IMAP_LOGIN);
return result;
}
@@ -547,16 +635,20 @@ static CURLcode imap_perform_authenticate(struct Curl_easy *data, const char *mech,
const struct bufref *initresp)
{
+ struct imap_conn *imapc =
+ Curl_conn_meta_get(data->conn, CURL_META_IMAP_CONN);
CURLcode result = CURLE_OK;
const char *ir = (const char *) Curl_bufref_ptr(initresp);
+ if(!imapc)
+ return CURLE_FAILED_INIT;
if(ir) {
/* Send the AUTHENTICATE command with the initial response */
- result = imap_sendf(data, "AUTHENTICATE %s %s", mech, ir);
+ result = imap_sendf(data, imapc, "AUTHENTICATE %s %s", mech, ir);
}
else {
/* Send the AUTHENTICATE command */
- result = imap_sendf(data, "AUTHENTICATE %s", mech);
+ result = imap_sendf(data, imapc, "AUTHENTICATE %s", mech);
}
return result;
@@ -572,10 +664,12 @@ static CURLcode imap_continue_authenticate(struct Curl_easy *data, const char *mech,
const struct bufref *resp)
{
- struct imap_conn *imapc = &data->conn->proto.imapc;
+ struct imap_conn *imapc =
+ Curl_conn_meta_get(data->conn, CURL_META_IMAP_CONN);
(void)mech;
-
+ if(!imapc)
+ return CURLE_FAILED_INIT;
return Curl_pp_sendf(data, &imapc->pp,
"%s", (const char *) Curl_bufref_ptr(resp));
}
@@ -589,10 +683,12 @@ static CURLcode imap_continue_authenticate(struct Curl_easy *data, static CURLcode imap_cancel_authenticate(struct Curl_easy *data,
const char *mech)
{
- struct imap_conn *imapc = &data->conn->proto.imapc;
+ struct imap_conn *imapc =
+ Curl_conn_meta_get(data->conn, CURL_META_IMAP_CONN);
(void)mech;
-
+ if(!imapc)
+ return CURLE_FAILED_INIT;
return Curl_pp_sendf(data, &imapc->pp, "*");
}
@@ -605,17 +701,16 @@ static CURLcode imap_cancel_authenticate(struct Curl_easy *data, * mechanism not be available between the client and server.
*/
static CURLcode imap_perform_authentication(struct Curl_easy *data,
- struct connectdata *conn)
+ struct imap_conn *imapc)
{
CURLcode result = CURLE_OK;
- struct imap_conn *imapc = &conn->proto.imapc;
saslprogress progress;
/* Check if already authenticated OR if there is enough data to authenticate
with and end the connect phase if we do not */
if(imapc->preauth ||
!Curl_sasl_can_authenticate(&imapc->sasl, data)) {
- imap_state(data, IMAP_STOP);
+ imap_state(data, imapc, IMAP_STOP);
return result;
}
@@ -624,15 +719,12 @@ static CURLcode imap_perform_authentication(struct Curl_easy *data, if(!result) {
if(progress == SASL_INPROGRESS)
- imap_state(data, IMAP_AUTHENTICATE);
+ imap_state(data, imapc, IMAP_AUTHENTICATE);
else if(!imapc->login_disabled && (imapc->preftype & IMAP_TYPE_CLEARTEXT))
/* Perform clear text authentication */
- result = imap_perform_login(data, conn);
- else {
- /* Other mechanisms not supported */
- infof(data, "No known authentication mechanisms supported");
- result = CURLE_LOGIN_DENIED;
- }
+ result = imap_perform_login(data, imapc, data->conn);
+ else
+ result = Curl_sasl_is_blocked(&imapc->sasl, data);
}
return result;
@@ -644,14 +736,15 @@ static CURLcode imap_perform_authentication(struct Curl_easy *data, *
* Sends a LIST command or an alternative custom request.
*/
-static CURLcode imap_perform_list(struct Curl_easy *data)
+static CURLcode imap_perform_list(struct Curl_easy *data,
+ struct imap_conn *imapc,
+ struct IMAP *imap)
{
CURLcode result = CURLE_OK;
- struct IMAP *imap = data->req.p.imap;
if(imap->custom)
/* Send the custom request */
- result = imap_sendf(data, "%s%s", imap->custom,
+ result = imap_sendf(data, imapc, "%s%s", imap->custom,
imap->custom_params ? imap->custom_params : "");
else {
/* Make sure the mailbox is in the correct atom format if necessary */
@@ -661,13 +754,13 @@ static CURLcode imap_perform_list(struct Curl_easy *data) return CURLE_OUT_OF_MEMORY;
/* Send the LIST command */
- result = imap_sendf(data, "LIST \"%s\" *", mailbox);
+ result = imap_sendf(data, imapc, "LIST \"%s\" *", mailbox);
free(mailbox);
}
if(!result)
- imap_state(data, IMAP_LIST);
+ imap_state(data, imapc, IMAP_LIST);
return result;
}
@@ -678,12 +771,11 @@ static CURLcode imap_perform_list(struct Curl_easy *data) *
* Sends a SELECT command to ask the server to change the selected mailbox.
*/
-static CURLcode imap_perform_select(struct Curl_easy *data)
+static CURLcode imap_perform_select(struct Curl_easy *data,
+ struct imap_conn *imapc,
+ struct IMAP *imap)
{
CURLcode result = CURLE_OK;
- struct connectdata *conn = data->conn;
- struct IMAP *imap = data->req.p.imap;
- struct imap_conn *imapc = &conn->proto.imapc;
char *mailbox;
/* Invalidate old information as we are switching mailboxes */
@@ -702,12 +794,12 @@ static CURLcode imap_perform_select(struct Curl_easy *data) return CURLE_OUT_OF_MEMORY;
/* Send the SELECT command */
- result = imap_sendf(data, "SELECT %s", mailbox);
+ result = imap_sendf(data, imapc, "SELECT %s", mailbox);
free(mailbox);
if(!result)
- imap_state(data, IMAP_SELECT);
+ imap_state(data, imapc, IMAP_SELECT);
return result;
}
@@ -718,30 +810,31 @@ static CURLcode imap_perform_select(struct Curl_easy *data) *
* Sends a FETCH command to initiate the download of a message.
*/
-static CURLcode imap_perform_fetch(struct Curl_easy *data)
+static CURLcode imap_perform_fetch(struct Curl_easy *data,
+ struct imap_conn *imapc,
+ struct IMAP *imap)
{
CURLcode result = CURLE_OK;
- struct IMAP *imap = data->req.p.imap;
/* Check we have a UID */
if(imap->uid) {
/* Send the FETCH command */
if(imap->partial)
- result = imap_sendf(data, "UID FETCH %s BODY[%s]<%s>",
+ result = imap_sendf(data, imapc, "UID FETCH %s BODY[%s]<%s>",
imap->uid, imap->section ? imap->section : "",
imap->partial);
else
- result = imap_sendf(data, "UID FETCH %s BODY[%s]",
+ result = imap_sendf(data, imapc, "UID FETCH %s BODY[%s]",
imap->uid, imap->section ? imap->section : "");
}
else if(imap->mindex) {
/* Send the FETCH command */
if(imap->partial)
- result = imap_sendf(data, "FETCH %s BODY[%s]<%s>",
+ result = imap_sendf(data, imapc, "FETCH %s BODY[%s]<%s>",
imap->mindex, imap->section ? imap->section : "",
imap->partial);
else
- result = imap_sendf(data, "FETCH %s BODY[%s]",
+ result = imap_sendf(data, imapc, "FETCH %s BODY[%s]",
imap->mindex, imap->section ? imap->section : "");
}
else {
@@ -749,7 +842,7 @@ static CURLcode imap_perform_fetch(struct Curl_easy *data) return CURLE_URL_MALFORMAT;
}
if(!result)
- imap_state(data, IMAP_FETCH);
+ imap_state(data, imapc, IMAP_FETCH);
return result;
}
@@ -760,10 +853,11 @@ static CURLcode imap_perform_fetch(struct Curl_easy *data) *
* Sends an APPEND command to initiate the upload of a message.
*/
-static CURLcode imap_perform_append(struct Curl_easy *data)
+static CURLcode imap_perform_append(struct Curl_easy *data,
+ struct imap_conn *imapc,
+ struct IMAP *imap)
{
CURLcode result = CURLE_OK;
- struct IMAP *imap = data->req.p.imap;
char *mailbox;
struct dynbuf flags;
@@ -815,7 +909,7 @@ static CURLcode imap_perform_append(struct Curl_easy *data) return CURLE_OUT_OF_MEMORY;
/* Generate flags string and send the APPEND command */
- Curl_dyn_init(&flags, 100);
+ curlx_dyn_init(&flags, 100);
if(data->set.upload_flags) {
int i;
struct ulbits ulflag[] = {
@@ -828,33 +922,34 @@ static CURLcode imap_perform_append(struct Curl_easy *data) };
result = CURLE_OUT_OF_MEMORY;
- if(Curl_dyn_add(&flags, " (")) {
+ if(curlx_dyn_add(&flags, " (")) {
goto cleanup;
}
for(i = 0; ulflag[i].bit; i++) {
if(data->set.upload_flags & ulflag[i].bit) {
- if((Curl_dyn_len(&flags) > 2 && Curl_dyn_add(&flags, " ")) ||
- Curl_dyn_add(&flags, "\\") || Curl_dyn_add(&flags, ulflag[i].flag))
- goto cleanup;
+ if((curlx_dyn_len(&flags) > 2 && curlx_dyn_add(&flags, " ")) ||
+ curlx_dyn_add(&flags, "\\") ||
+ curlx_dyn_add(&flags, ulflag[i].flag))
+ goto cleanup;
}
}
- if(Curl_dyn_add(&flags, ")"))
+ if(curlx_dyn_add(&flags, ")"))
goto cleanup;
}
- else if(Curl_dyn_add(&flags, ""))
+ else if(curlx_dyn_add(&flags, ""))
goto cleanup;
- result = imap_sendf(data, "APPEND %s%s {%" FMT_OFF_T "}",
- mailbox, Curl_dyn_ptr(&flags), data->state.infilesize);
+ result = imap_sendf(data, imapc, "APPEND %s%s {%" FMT_OFF_T "}",
+ mailbox, curlx_dyn_ptr(&flags), data->state.infilesize);
cleanup:
- Curl_dyn_free(&flags);
+ curlx_dyn_free(&flags);
free(mailbox);
if(!result)
- imap_state(data, IMAP_APPEND);
+ imap_state(data, imapc, IMAP_APPEND);
return result;
}
@@ -865,10 +960,11 @@ cleanup: *
* Sends a SEARCH command.
*/
-static CURLcode imap_perform_search(struct Curl_easy *data)
+static CURLcode imap_perform_search(struct Curl_easy *data,
+ struct imap_conn *imapc,
+ struct IMAP *imap)
{
CURLcode result = CURLE_OK;
- struct IMAP *imap = data->req.p.imap;
/* Check we have a query string */
if(!imap->query) {
@@ -877,10 +973,10 @@ static CURLcode imap_perform_search(struct Curl_easy *data) }
/* Send the SEARCH command */
- result = imap_sendf(data, "SEARCH %s", imap->query);
+ result = imap_sendf(data, imapc, "SEARCH %s", imap->query);
if(!result)
- imap_state(data, IMAP_SEARCH);
+ imap_state(data, imapc, IMAP_SEARCH);
return result;
}
@@ -891,28 +987,28 @@ static CURLcode imap_perform_search(struct Curl_easy *data) *
* Performs the logout action prior to sclose() being called.
*/
-static CURLcode imap_perform_logout(struct Curl_easy *data)
+static CURLcode imap_perform_logout(struct Curl_easy *data,
+ struct imap_conn *imapc)
{
/* Send the LOGOUT command */
- CURLcode result = imap_sendf(data, "LOGOUT");
+ CURLcode result = imap_sendf(data, imapc, "LOGOUT");
if(!result)
- imap_state(data, IMAP_LOGOUT);
+ imap_state(data, imapc, IMAP_LOGOUT);
return result;
}
/* For the initial server greeting */
static CURLcode imap_state_servergreet_resp(struct Curl_easy *data,
+ struct imap_conn *imapc,
int imapcode,
imapstate instate)
{
- struct connectdata *conn = data->conn;
(void)instate; /* no use for this yet */
if(imapcode == IMAP_RESP_PREAUTH) {
/* PREAUTH */
- struct imap_conn *imapc = &conn->proto.imapc;
imapc->preauth = TRUE;
infof(data, "PREAUTH connection, already authenticated");
}
@@ -921,18 +1017,18 @@ static CURLcode imap_state_servergreet_resp(struct Curl_easy *data, return CURLE_WEIRD_SERVER_REPLY;
}
- return imap_perform_capability(data, conn);
+ return imap_perform_capability(data, imapc);
}
/* For CAPABILITY responses */
static CURLcode imap_state_capability_resp(struct Curl_easy *data,
+ struct imap_conn *imapc,
int imapcode,
imapstate instate)
{
CURLcode result = CURLE_OK;
struct connectdata *conn = data->conn;
- struct imap_conn *imapc = &conn->proto.imapc;
- const char *line = Curl_dyn_ptr(&imapc->pp.recvbuf);
+ const char *line = curlx_dyn_ptr(&imapc->pp.recvbuf);
(void)instate; /* no use for this yet */
@@ -992,33 +1088,33 @@ static CURLcode imap_state_capability_resp(struct Curl_easy *data, /* PREAUTH is not compatible with STARTTLS. */
if(imapcode == IMAP_RESP_OK && imapc->tls_supported && !imapc->preauth) {
/* Switch to TLS connection now */
- result = imap_perform_starttls(data);
+ result = imap_perform_starttls(data, imapc);
}
else if(data->set.use_ssl <= CURLUSESSL_TRY)
- result = imap_perform_authentication(data, conn);
+ result = imap_perform_authentication(data, imapc);
else {
failf(data, "STARTTLS not available.");
result = CURLE_USE_SSL_FAILED;
}
}
else
- result = imap_perform_authentication(data, conn);
+ result = imap_perform_authentication(data, imapc);
return result;
}
/* For STARTTLS responses */
static CURLcode imap_state_starttls_resp(struct Curl_easy *data,
+ struct imap_conn *imapc,
int imapcode,
imapstate instate)
{
CURLcode result = CURLE_OK;
- struct connectdata *conn = data->conn;
(void)instate; /* no use for this yet */
/* Pipelining in response is forbidden. */
- if(data->conn->proto.imapc.pp.overflow)
+ if(imapc->pp.overflow)
return CURLE_WEIRD_SERVER_REPLY;
if(imapcode != IMAP_RESP_OK) {
@@ -1027,22 +1123,21 @@ static CURLcode imap_state_starttls_resp(struct Curl_easy *data, result = CURLE_USE_SSL_FAILED;
}
else
- result = imap_perform_authentication(data, conn);
+ result = imap_perform_authentication(data, imapc);
}
else
- imap_state(data, IMAP_UPGRADETLS);
+ imap_state(data, imapc, IMAP_UPGRADETLS);
return result;
}
/* For SASL authentication responses */
static CURLcode imap_state_auth_resp(struct Curl_easy *data,
- struct connectdata *conn,
+ struct imap_conn *imapc,
int imapcode,
imapstate instate)
{
CURLcode result = CURLE_OK;
- struct imap_conn *imapc = &conn->proto.imapc;
saslprogress progress;
(void)instate; /* no use for this yet */
@@ -1051,12 +1146,12 @@ static CURLcode imap_state_auth_resp(struct Curl_easy *data, if(!result)
switch(progress) {
case SASL_DONE:
- imap_state(data, IMAP_STOP); /* Authenticated */
+ imap_state(data, imapc, IMAP_STOP); /* Authenticated */
break;
case SASL_IDLE: /* No mechanism left after cancellation */
if((!imapc->login_disabled) && (imapc->preftype & IMAP_TYPE_CLEARTEXT))
/* Perform clear text authentication */
- result = imap_perform_login(data, conn);
+ result = imap_perform_login(data, imapc, data->conn);
else {
failf(data, "Authentication cancelled");
result = CURLE_LOGIN_DENIED;
@@ -1071,6 +1166,7 @@ static CURLcode imap_state_auth_resp(struct Curl_easy *data, /* For LOGIN responses */
static CURLcode imap_state_login_resp(struct Curl_easy *data,
+ struct imap_conn *imapc,
int imapcode,
imapstate instate)
{
@@ -1083,19 +1179,20 @@ static CURLcode imap_state_login_resp(struct Curl_easy *data, }
else
/* End of connect phase */
- imap_state(data, IMAP_STOP);
+ imap_state(data, imapc, IMAP_STOP);
return result;
}
/* For LIST and SEARCH responses */
static CURLcode imap_state_listsearch_resp(struct Curl_easy *data,
+ struct imap_conn *imapc,
int imapcode,
imapstate instate)
{
CURLcode result = CURLE_OK;
- char *line = Curl_dyn_ptr(&data->conn->proto.imapc.pp.recvbuf);
- size_t len = data->conn->proto.imapc.pp.nfinal;
+ char *line = curlx_dyn_ptr(&imapc->pp.recvbuf);
+ size_t len = imapc->pp.nfinal;
(void)instate; /* No use for this yet */
@@ -1105,20 +1202,20 @@ static CURLcode imap_state_listsearch_resp(struct Curl_easy *data, result = CURLE_QUOTE_ERROR;
else
/* End of DO phase */
- imap_state(data, IMAP_STOP);
+ imap_state(data, imapc, IMAP_STOP);
return result;
}
/* For SELECT responses */
-static CURLcode imap_state_select_resp(struct Curl_easy *data, int imapcode,
+static CURLcode imap_state_select_resp(struct Curl_easy *data,
+ struct imap_conn *imapc,
+ struct IMAP *imap,
+ int imapcode,
imapstate instate)
{
CURLcode result = CURLE_OK;
- struct connectdata *conn = data->conn;
- struct IMAP *imap = data->req.p.imap;
- struct imap_conn *imapc = &conn->proto.imapc;
- const char *line = Curl_dyn_ptr(&data->conn->proto.imapc.pp.recvbuf);
+ const char *line = curlx_dyn_ptr(&imapc->pp.recvbuf);
(void)instate; /* no use for this yet */
@@ -1131,11 +1228,11 @@ static CURLcode imap_state_select_resp(struct Curl_easy *data, int imapcode, len++;
if(len && (p[len] == ']')) {
struct dynbuf uid;
- Curl_dyn_init(&uid, 20);
- if(Curl_dyn_addn(&uid, p, len))
+ curlx_dyn_init(&uid, 20);
+ if(curlx_dyn_addn(&uid, p, len))
return CURLE_OUT_OF_MEMORY;
free(imapc->mailbox_uidvalidity);
- imapc->mailbox_uidvalidity = Curl_dyn_ptr(&uid);
+ imapc->mailbox_uidvalidity = curlx_dyn_ptr(&uid);
}
}
}
@@ -1154,11 +1251,11 @@ static CURLcode imap_state_select_resp(struct Curl_easy *data, int imapcode, return CURLE_OUT_OF_MEMORY;
if(imap->custom)
- result = imap_perform_list(data);
+ result = imap_perform_list(data, imapc, imap);
else if(imap->query)
- result = imap_perform_search(data);
+ result = imap_perform_search(data, imapc, imap);
else
- result = imap_perform_fetch(data);
+ result = imap_perform_fetch(data, imapc, imap);
}
}
else {
@@ -1171,14 +1268,14 @@ static CURLcode imap_state_select_resp(struct Curl_easy *data, int imapcode, /* For the (first line of the) FETCH responses */
static CURLcode imap_state_fetch_resp(struct Curl_easy *data,
- struct connectdata *conn, int imapcode,
+ struct imap_conn *imapc,
+ int imapcode,
imapstate instate)
{
CURLcode result = CURLE_OK;
- struct imap_conn *imapc = &conn->proto.imapc;
struct pingpong *pp = &imapc->pp;
- const char *ptr = Curl_dyn_ptr(&data->conn->proto.imapc.pp.recvbuf);
- size_t len = data->conn->proto.imapc.pp.nfinal;
+ const char *ptr = curlx_dyn_ptr(&imapc->pp.recvbuf);
+ size_t len = imapc->pp.nfinal;
bool parsed = FALSE;
curl_off_t size = 0;
@@ -1186,7 +1283,7 @@ static CURLcode imap_state_fetch_resp(struct Curl_easy *data, if(imapcode != '*') {
Curl_pgrsSetDownloadSize(data, -1);
- imap_state(data, IMAP_STOP);
+ imap_state(data, imapc, IMAP_STOP);
return CURLE_REMOTE_FILE_NOT_FOUND;
}
@@ -1195,8 +1292,8 @@ static CURLcode imap_state_fetch_resp(struct Curl_easy *data, ptr = memchr(ptr, '{', len);
if(ptr) {
ptr++;
- if(!Curl_str_number(&ptr, &size, CURL_OFF_T_MAX) &&
- !Curl_str_single(&ptr, '}'))
+ if(!curlx_str_number(&ptr, &size, CURL_OFF_T_MAX) &&
+ !curlx_str_single(&ptr, '}'))
parsed = TRUE;
}
@@ -1211,7 +1308,7 @@ static CURLcode imap_state_fetch_resp(struct Curl_easy *data, size_t chunk = pp->overflow;
/* keep only the overflow */
- Curl_dyn_tail(&pp->recvbuf, chunk);
+ curlx_dyn_tail(&pp->recvbuf, chunk);
pp->nfinal = 0; /* done */
if(chunk > (size_t)size)
@@ -1220,11 +1317,11 @@ static CURLcode imap_state_fetch_resp(struct Curl_easy *data, if(!chunk) {
/* no size, we are done with the data */
- imap_state(data, IMAP_STOP);
+ imap_state(data, imapc, IMAP_STOP);
return CURLE_OK;
}
result = Curl_client_write(data, CLIENTWRITE_BODY,
- Curl_dyn_ptr(&pp->recvbuf), chunk);
+ curlx_dyn_ptr(&pp->recvbuf), chunk);
if(result)
return result;
@@ -1235,12 +1332,12 @@ static CURLcode imap_state_fetch_resp(struct Curl_easy *data, if(pp->overflow > chunk) {
/* remember the remaining trailing overflow data */
pp->overflow -= chunk;
- Curl_dyn_tail(&pp->recvbuf, pp->overflow);
+ curlx_dyn_tail(&pp->recvbuf, pp->overflow);
}
else {
pp->overflow = 0; /* handled */
/* Free the cache */
- Curl_dyn_reset(&pp->recvbuf);
+ curlx_dyn_reset(&pp->recvbuf);
}
}
@@ -1263,13 +1360,14 @@ static CURLcode imap_state_fetch_resp(struct Curl_easy *data, }
/* End of DO phase */
- imap_state(data, IMAP_STOP);
+ imap_state(data, imapc, IMAP_STOP);
return result;
}
/* For final FETCH responses performed after the download */
static CURLcode imap_state_fetch_final_resp(struct Curl_easy *data,
+ struct imap_conn *imapc,
int imapcode,
imapstate instate)
{
@@ -1281,13 +1379,15 @@ static CURLcode imap_state_fetch_final_resp(struct Curl_easy *data, result = CURLE_WEIRD_SERVER_REPLY;
else
/* End of DONE phase */
- imap_state(data, IMAP_STOP);
+ imap_state(data, imapc, IMAP_STOP);
return result;
}
/* For APPEND responses */
-static CURLcode imap_state_append_resp(struct Curl_easy *data, int imapcode,
+static CURLcode imap_state_append_resp(struct Curl_easy *data,
+ struct imap_conn *imapc,
+ int imapcode,
imapstate instate)
{
CURLcode result = CURLE_OK;
@@ -1304,7 +1404,7 @@ static CURLcode imap_state_append_resp(struct Curl_easy *data, int imapcode, Curl_xfer_setup1(data, CURL_XFER_SEND, -1, FALSE);
/* End of DO phase */
- imap_state(data, IMAP_STOP);
+ imap_state(data, imapc, IMAP_STOP);
}
return result;
@@ -1312,6 +1412,7 @@ static CURLcode imap_state_append_resp(struct Curl_easy *data, int imapcode, /* For final APPEND responses performed after the upload */
static CURLcode imap_state_append_final_resp(struct Curl_easy *data,
+ struct imap_conn *imapc,
int imapcode,
imapstate instate)
{
@@ -1323,25 +1424,29 @@ static CURLcode imap_state_append_final_resp(struct Curl_easy *data, result = CURLE_UPLOAD_FAILED;
else
/* End of DONE phase */
- imap_state(data, IMAP_STOP);
+ imap_state(data, imapc, IMAP_STOP);
return result;
}
-static CURLcode imap_statemachine(struct Curl_easy *data,
- struct connectdata *conn)
+static CURLcode imap_pp_statemachine(struct Curl_easy *data,
+ struct connectdata *conn)
{
CURLcode result = CURLE_OK;
int imapcode;
- struct imap_conn *imapc = &conn->proto.imapc;
- struct pingpong *pp = &imapc->pp;
+ struct imap_conn *imapc = Curl_conn_meta_get(conn, CURL_META_IMAP_CONN);
+ struct IMAP *imap = Curl_meta_get(data, CURL_META_IMAP_EASY);
+ struct pingpong *pp;
size_t nread = 0;
- (void)data;
+ (void)data;
+ if(!imapc || !imap)
+ return CURLE_FAILED_INIT;
+ pp = &imapc->pp;
/* Busy upgrading the connection; right now all I/O is SSL/TLS, not IMAP */
upgrade_tls:
if(imapc->state == IMAP_UPGRADETLS) {
- result = imap_perform_upgrade_tls(data, conn);
+ result = imap_perform_upgrade_tls(data, imapc, conn);
if(result || (imapc->state == IMAP_UPGRADETLS))
return result;
}
@@ -1366,15 +1471,16 @@ upgrade_tls: /* We have now received a full IMAP server response */
switch(imapc->state) {
case IMAP_SERVERGREET:
- result = imap_state_servergreet_resp(data, imapcode, imapc->state);
+ result = imap_state_servergreet_resp(data, imapc,
+ imapcode, imapc->state);
break;
case IMAP_CAPABILITY:
- result = imap_state_capability_resp(data, imapcode, imapc->state);
+ result = imap_state_capability_resp(data, imapc, imapcode, imapc->state);
break;
case IMAP_STARTTLS:
- result = imap_state_starttls_resp(data, imapcode, imapc->state);
+ result = imap_state_starttls_resp(data, imapc, imapcode, imapc->state);
/* During UPGRADETLS, leave the read loop as we need to connect
* (e.g. TLS handshake) before we continue sending/receiving. */
if(!result && (imapc->state == IMAP_UPGRADETLS))
@@ -1382,42 +1488,45 @@ upgrade_tls: break;
case IMAP_AUTHENTICATE:
- result = imap_state_auth_resp(data, conn, imapcode, imapc->state);
+ result = imap_state_auth_resp(data, imapc, imapcode, imapc->state);
break;
case IMAP_LOGIN:
- result = imap_state_login_resp(data, imapcode, imapc->state);
+ result = imap_state_login_resp(data, imapc, imapcode, imapc->state);
break;
case IMAP_LIST:
case IMAP_SEARCH:
- result = imap_state_listsearch_resp(data, imapcode, imapc->state);
+ result = imap_state_listsearch_resp(data, imapc, imapcode, imapc->state);
break;
case IMAP_SELECT:
- result = imap_state_select_resp(data, imapcode, imapc->state);
+ result = imap_state_select_resp(data, imapc, imap,
+ imapcode, imapc->state);
break;
case IMAP_FETCH:
- result = imap_state_fetch_resp(data, conn, imapcode, imapc->state);
+ result = imap_state_fetch_resp(data, imapc, imapcode, imapc->state);
break;
case IMAP_FETCH_FINAL:
- result = imap_state_fetch_final_resp(data, imapcode, imapc->state);
+ result = imap_state_fetch_final_resp(data, imapc,
+ imapcode, imapc->state);
break;
case IMAP_APPEND:
- result = imap_state_append_resp(data, imapcode, imapc->state);
+ result = imap_state_append_resp(data, imapc, imapcode, imapc->state);
break;
case IMAP_APPEND_FINAL:
- result = imap_state_append_final_resp(data, imapcode, imapc->state);
+ result = imap_state_append_final_resp(data, imapc,
+ imapcode, imapc->state);
break;
case IMAP_LOGOUT:
default:
/* internal error */
- imap_state(data, IMAP_STOP);
+ imap_state(data, imapc, IMAP_STOP);
break;
}
} while(!result && imapc->state != IMAP_STOP && Curl_pp_moredata(pp));
@@ -1429,9 +1538,12 @@ upgrade_tls: static CURLcode imap_multi_statemach(struct Curl_easy *data, bool *done)
{
CURLcode result = CURLE_OK;
- struct connectdata *conn = data->conn;
- struct imap_conn *imapc = &conn->proto.imapc;
+ struct imap_conn *imapc =
+ Curl_conn_meta_get(data->conn, CURL_META_IMAP_CONN);
+ *done = FALSE;
+ if(!imapc)
+ return CURLE_FAILED_INIT;
result = Curl_pp_statemach(data, &imapc->pp, FALSE, FALSE);
*done = (imapc->state == IMAP_STOP);
@@ -1439,11 +1551,10 @@ static CURLcode imap_multi_statemach(struct Curl_easy *data, bool *done) }
static CURLcode imap_block_statemach(struct Curl_easy *data,
- struct connectdata *conn,
+ struct imap_conn *imapc,
bool disconnecting)
{
CURLcode result = CURLE_OK;
- struct imap_conn *imapc = &conn->proto.imapc;
while(imapc->state != IMAP_STOP && !result)
result = Curl_pp_statemach(data, &imapc->pp, TRUE, disconnecting);
@@ -1451,26 +1562,14 @@ static CURLcode imap_block_statemach(struct Curl_easy *data, return result;
}
-/* Allocate and initialize the struct IMAP for the current Curl_easy if
- required */
-static CURLcode imap_init(struct Curl_easy *data)
-{
- CURLcode result = CURLE_OK;
- struct IMAP *imap;
-
- imap = data->req.p.imap = calloc(1, sizeof(struct IMAP));
- if(!imap)
- result = CURLE_OUT_OF_MEMORY;
-
- return result;
-}
-
/* For the IMAP "protocol connect" and "doing" phases only */
static int imap_getsock(struct Curl_easy *data,
struct connectdata *conn,
curl_socket_t *socks)
{
- return Curl_pp_getsock(data, &conn->proto.imapc.pp, socks);
+ struct imap_conn *imapc = Curl_conn_meta_get(conn, CURL_META_IMAP_CONN);
+ return imapc ?
+ Curl_pp_getsock(data, &imapc->pp, socks) : GETSOCK_BLANK;
}
/***********************************************************************
@@ -1485,35 +1584,24 @@ static int imap_getsock(struct Curl_easy *data, */
static CURLcode imap_connect(struct Curl_easy *data, bool *done)
{
+ struct imap_conn *imapc =
+ Curl_conn_meta_get(data->conn, CURL_META_IMAP_CONN);
CURLcode result = CURLE_OK;
- struct connectdata *conn = data->conn;
- struct imap_conn *imapc = &conn->proto.imapc;
- struct pingpong *pp = &imapc->pp;
*done = FALSE; /* default to not done yet */
+ if(!imapc)
+ return CURLE_FAILED_INIT;
/* We always support persistent connections in IMAP */
- connkeep(conn, "IMAP default");
-
- if(!imapc->initialised) {
- PINGPONG_SETUP(pp, imap_statemachine, imap_endofresp);
-
- /* Set the default preferred authentication type and mechanism */
- imapc->preftype = IMAP_TYPE_ANY;
- Curl_sasl_init(&imapc->sasl, data, &saslimap);
-
- Curl_dyn_init(&imapc->dyn, DYN_IMAP_CMD);
- Curl_pp_init(pp);
- imapc->initialised = TRUE;
- }
+ connkeep(data->conn, "IMAP default");
/* Parse the URL options */
- result = imap_parse_url_options(conn);
+ result = imap_parse_url_options(data->conn, imapc);
if(result)
return result;
/* Start off waiting for the server greeting response */
- imap_state(data, IMAP_SERVERGREET);
+ imap_state(data, imapc, IMAP_SERVERGREET);
/* Start off with an response id of '*' */
strcpy(imapc->resptag, "*");
@@ -1537,10 +1625,13 @@ static CURLcode imap_done(struct Curl_easy *data, CURLcode status, {
CURLcode result = CURLE_OK;
struct connectdata *conn = data->conn;
- struct IMAP *imap = data->req.p.imap;
+ struct imap_conn *imapc = Curl_conn_meta_get(conn, CURL_META_IMAP_CONN);
+ struct IMAP *imap = Curl_meta_get(data, CURL_META_IMAP_EASY);
(void)premature;
+ if(!imapc)
+ return CURLE_FAILED_INIT;
if(!imap)
return CURLE_OK;
@@ -1554,33 +1645,20 @@ static CURLcode imap_done(struct Curl_easy *data, CURLcode status, /* Handle responses after FETCH or APPEND transfer has finished */
if(!data->state.upload && !IS_MIME_POST(data))
- imap_state(data, IMAP_FETCH_FINAL);
+ imap_state(data, imapc, IMAP_FETCH_FINAL);
else {
/* End the APPEND command first by sending an empty line */
- result = Curl_pp_sendf(data, &conn->proto.imapc.pp, "%s", "");
+ result = Curl_pp_sendf(data, &imapc->pp, "%s", "");
if(!result)
- imap_state(data, IMAP_APPEND_FINAL);
+ imap_state(data, imapc, IMAP_APPEND_FINAL);
}
/* Run the state-machine */
if(!result)
- result = imap_block_statemach(data, conn, FALSE);
+ result = imap_block_statemach(data, imapc, FALSE);
}
- /* Cleanup our per-request based variables */
- Curl_safefree(imap->mailbox);
- Curl_safefree(imap->uidvalidity);
- Curl_safefree(imap->uid);
- Curl_safefree(imap->mindex);
- Curl_safefree(imap->section);
- Curl_safefree(imap->partial);
- Curl_safefree(imap->query);
- Curl_safefree(imap->custom);
- Curl_safefree(imap->custom_params);
-
- /* Clear the transfer mode for the next request */
- imap->transfer = PPTRANSFER_BODY;
-
+ imap_easy_reset(imap);
return result;
}
@@ -1597,11 +1675,13 @@ static CURLcode imap_perform(struct Curl_easy *data, bool *connected, /* This is IMAP and no proxy */
CURLcode result = CURLE_OK;
struct connectdata *conn = data->conn;
- struct IMAP *imap = data->req.p.imap;
- struct imap_conn *imapc = &conn->proto.imapc;
+ struct imap_conn *imapc = Curl_conn_meta_get(conn, CURL_META_IMAP_CONN);
+ struct IMAP *imap = Curl_meta_get(data, CURL_META_IMAP_EASY);
bool selected = FALSE;
DEBUGF(infof(data, "DO phase starts"));
+ if(!imapc || !imap)
+ return CURLE_FAILED_INIT;
if(data->req.no_body) {
/* Requested no body means no transfer */
@@ -1621,23 +1701,23 @@ static CURLcode imap_perform(struct Curl_easy *data, bool *connected, /* Start the first command in the DO phase */
if(data->state.upload || IS_MIME_POST(data))
/* APPEND can be executed directly */
- result = imap_perform_append(data);
+ result = imap_perform_append(data, imapc, imap);
else if(imap->custom && (selected || !imap->mailbox))
/* Custom command using the same mailbox or no mailbox */
- result = imap_perform_list(data);
+ result = imap_perform_list(data, imapc, imap);
else if(!imap->custom && selected && (imap->uid || imap->mindex))
/* FETCH from the same mailbox */
- result = imap_perform_fetch(data);
+ result = imap_perform_fetch(data, imapc, imap);
else if(!imap->custom && selected && imap->query)
/* SEARCH the current mailbox */
- result = imap_perform_search(data);
+ result = imap_perform_search(data, imapc, imap);
else if(imap->mailbox && !selected &&
(imap->custom || imap->uid || imap->mindex || imap->query))
/* SELECT the mailbox */
- result = imap_perform_select(data);
+ result = imap_perform_select(data, imapc, imap);
else
/* LIST */
- result = imap_perform_list(data);
+ result = imap_perform_list(data, imapc, imap);
if(result)
return result;
@@ -1664,20 +1744,23 @@ static CURLcode imap_perform(struct Curl_easy *data, bool *connected, */
static CURLcode imap_do(struct Curl_easy *data, bool *done)
{
+ struct IMAP *imap = Curl_meta_get(data, CURL_META_IMAP_EASY);
CURLcode result = CURLE_OK;
*done = FALSE; /* default to false */
+ if(!imap)
+ return CURLE_FAILED_INIT;
/* Parse the URL path */
- result = imap_parse_url_path(data);
+ result = imap_parse_url_path(data, imap);
if(result)
return result;
/* Parse the custom request */
- result = imap_parse_custom_request(data);
+ result = imap_parse_custom_request(data, imap);
if(result)
return result;
- result = imap_regular_transfer(data, done);
+ result = imap_regular_transfer(data, imap, done);
return result;
}
@@ -1692,10 +1775,10 @@ static CURLcode imap_do(struct Curl_easy *data, bool *done) static CURLcode imap_disconnect(struct Curl_easy *data,
struct connectdata *conn, bool dead_connection)
{
- struct imap_conn *imapc = &conn->proto.imapc;
- (void)data;
+ struct imap_conn *imapc = Curl_conn_meta_get(conn, CURL_META_IMAP_CONN);
- if(imapc->initialised) {
+ (void)data;
+ if(imapc) {
/* We cannot send quit unconditionally. If this connection is stale or
bad in any way, sending quit and waiting around here will make the
disconnect wait in vain and cause more problems than we need to. */
@@ -1703,31 +1786,21 @@ static CURLcode imap_disconnect(struct Curl_easy *data, /* The IMAP session may or may not have been allocated/setup at this
point! */
if(!dead_connection && conn->bits.protoconnstart) {
- if(!imap_perform_logout(data))
- (void)imap_block_statemach(data, conn, TRUE); /* ignore errors */
+ if(!imap_perform_logout(data, imapc))
+ (void)imap_block_statemach(data, imapc, TRUE); /* ignore errors */
}
- /* Disconnect from the server */
- Curl_pp_disconnect(&imapc->pp);
- Curl_dyn_free(&imapc->dyn);
-
/* Cleanup the SASL module */
Curl_sasl_cleanup(conn, imapc->sasl.authused);
-
- /* Cleanup our connection based variables */
- Curl_safefree(imapc->mailbox);
- Curl_safefree(imapc->mailbox_uidvalidity);
- memset(imapc, 0, sizeof(*imapc));
}
-
return CURLE_OK;
}
/* Call this when the DO phase has completed */
-static CURLcode imap_dophase_done(struct Curl_easy *data, bool connected)
+static CURLcode imap_dophase_done(struct Curl_easy *data,
+ struct IMAP *imap,
+ bool connected)
{
- struct IMAP *imap = data->req.p.imap;
-
(void)connected;
if(imap->transfer != PPTRANSFER_BODY)
@@ -1740,12 +1813,17 @@ static CURLcode imap_dophase_done(struct Curl_easy *data, bool connected) /* Called from multi.c while DOing */
static CURLcode imap_doing(struct Curl_easy *data, bool *dophase_done)
{
- CURLcode result = imap_multi_statemach(data, dophase_done);
+ struct IMAP *imap = Curl_meta_get(data, CURL_META_IMAP_EASY);
+ CURLcode result;
+ if(!imap)
+ return CURLE_FAILED_INIT;
+
+ result = imap_multi_statemach(data, dophase_done);
if(result)
DEBUGF(infof(data, "DO phase failed"));
else if(*dophase_done) {
- result = imap_dophase_done(data, FALSE /* not connected */);
+ result = imap_dophase_done(data, imap, FALSE /* not connected */);
DEBUGF(infof(data, "DO phase is complete"));
}
@@ -1763,6 +1841,7 @@ static CURLcode imap_doing(struct Curl_easy *data, bool *dophase_done) * remote host.
*/
static CURLcode imap_regular_transfer(struct Curl_easy *data,
+ struct IMAP *imap,
bool *dophase_done)
{
CURLcode result = CURLE_OK;
@@ -1782,17 +1861,77 @@ static CURLcode imap_regular_transfer(struct Curl_easy *data, /* Perform post DO phase operations if necessary */
if(!result && *dophase_done)
- result = imap_dophase_done(data, connected);
+ result = imap_dophase_done(data, imap, connected);
return result;
}
+static void imap_easy_reset(struct IMAP *imap)
+{
+ Curl_safefree(imap->mailbox);
+ Curl_safefree(imap->uidvalidity);
+ Curl_safefree(imap->uid);
+ Curl_safefree(imap->mindex);
+ Curl_safefree(imap->section);
+ Curl_safefree(imap->partial);
+ Curl_safefree(imap->query);
+ Curl_safefree(imap->custom);
+ Curl_safefree(imap->custom_params);
+ /* Clear the transfer mode for the next request */
+ imap->transfer = PPTRANSFER_BODY;
+}
+
+static void imap_easy_dtor(void *key, size_t klen, void *entry)
+{
+ struct IMAP *imap = entry;
+ (void)key;
+ (void)klen;
+ imap_easy_reset(imap);
+ free(imap);
+}
+
+static void imap_conn_dtor(void *key, size_t klen, void *entry)
+{
+ struct imap_conn *imapc = entry;
+ (void)key;
+ (void)klen;
+ Curl_pp_disconnect(&imapc->pp);
+ curlx_dyn_free(&imapc->dyn);
+ Curl_safefree(imapc->mailbox);
+ Curl_safefree(imapc->mailbox_uidvalidity);
+ free(imapc);
+}
+
static CURLcode imap_setup_connection(struct Curl_easy *data,
struct connectdata *conn)
{
- /* Initialise the IMAP layer */
- (void)conn;
- return imap_init(data);
+ struct imap_conn *imapc;
+ struct pingpong *pp;
+ struct IMAP *imap;
+
+ imapc = calloc(1, sizeof(*imapc));
+ if(!imapc)
+ return CURLE_OUT_OF_MEMORY;
+
+ pp = &imapc->pp;
+ PINGPONG_SETUP(pp, imap_pp_statemachine, imap_endofresp);
+
+ /* Set the default preferred authentication type and mechanism */
+ imapc->preftype = IMAP_TYPE_ANY;
+ Curl_sasl_init(&imapc->sasl, data, &saslimap);
+
+ curlx_dyn_init(&imapc->dyn, DYN_IMAP_CMD);
+ Curl_pp_init(pp);
+
+ if(Curl_conn_meta_set(conn, CURL_META_IMAP_CONN, imapc, imap_conn_dtor))
+ return CURLE_OUT_OF_MEMORY;
+
+ imap = calloc(1, sizeof(struct IMAP));
+ if(!imap ||
+ Curl_meta_set(data, CURL_META_IMAP_EASY, imap, imap_easy_dtor))
+ return CURLE_OUT_OF_MEMORY;
+
+ return CURLE_OK;
}
/***********************************************************************
@@ -1803,10 +1942,11 @@ static CURLcode imap_setup_connection(struct Curl_easy *data, *
* Designed to never block.
*/
-static CURLcode imap_sendf(struct Curl_easy *data, const char *fmt, ...)
+static CURLcode imap_sendf(struct Curl_easy *data,
+ struct imap_conn *imapc,
+ const char *fmt, ...)
{
CURLcode result = CURLE_OK;
- struct imap_conn *imapc = &data->conn->proto.imapc;
DEBUGASSERT(fmt);
@@ -1816,10 +1956,10 @@ static CURLcode imap_sendf(struct Curl_easy *data, const char *fmt, ...) ++imapc->cmdid);
/* start with a blank buffer */
- Curl_dyn_reset(&imapc->dyn);
+ curlx_dyn_reset(&imapc->dyn);
/* append tag + space + fmt */
- result = Curl_dyn_addf(&imapc->dyn, "%s %s", imapc->resptag, fmt);
+ result = curlx_dyn_addf(&imapc->dyn, "%s %s", imapc->resptag, fmt);
if(!result) {
va_list ap;
va_start(ap, fmt);
@@ -1827,7 +1967,7 @@ static CURLcode imap_sendf(struct Curl_easy *data, const char *fmt, ...) #pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wformat-nonliteral"
#endif
- result = Curl_pp_vsendf(data, &imapc->pp, Curl_dyn_ptr(&imapc->dyn), ap);
+ result = Curl_pp_vsendf(data, &imapc->pp, curlx_dyn_ptr(&imapc->dyn), ap);
#ifdef __clang__
#pragma clang diagnostic pop
#endif
@@ -1861,24 +2001,24 @@ static char *imap_atom(const char *str, bool escape_only) /* nothing to escape, return a strdup */
return strdup(str);
- Curl_dyn_init(&line, 2000);
+ curlx_dyn_init(&line, 2000);
- if(!escape_only && Curl_dyn_addn(&line, "\"", 1))
+ if(!escape_only && curlx_dyn_addn(&line, "\"", 1))
return NULL;
while(*str) {
if((*str == '\\' || *str == '"') &&
- Curl_dyn_addn(&line, "\\", 1))
+ curlx_dyn_addn(&line, "\\", 1))
return NULL;
- if(Curl_dyn_addn(&line, str, 1))
+ if(curlx_dyn_addn(&line, str, 1))
return NULL;
str++;
}
- if(!escape_only && Curl_dyn_addn(&line, "\"", 1))
+ if(!escape_only && curlx_dyn_addn(&line, "\"", 1))
return NULL;
- return Curl_dyn_ptr(&line);
+ return curlx_dyn_ptr(&line);
}
/***********************************************************************
@@ -1920,10 +2060,10 @@ static bool imap_is_bchar(char ch) *
* Parse the URL login options.
*/
-static CURLcode imap_parse_url_options(struct connectdata *conn)
+static CURLcode imap_parse_url_options(struct connectdata *conn,
+ struct imap_conn *imapc)
{
CURLcode result = CURLE_OK;
- struct imap_conn *imapc = &conn->proto.imapc;
const char *ptr = conn->options;
bool prefer_login = FALSE;
@@ -1984,11 +2124,11 @@ static CURLcode imap_parse_url_options(struct connectdata *conn) * Parse the URL path into separate path components.
*
*/
-static CURLcode imap_parse_url_path(struct Curl_easy *data)
+static CURLcode imap_parse_url_path(struct Curl_easy *data,
+ struct IMAP *imap)
{
/* The imap struct is already initialised in imap_connect() */
CURLcode result = CURLE_OK;
- struct IMAP *imap = data->req.p.imap;
const char *begin = &data->state.up.path[1]; /* skip leading slash */
const char *ptr = begin;
@@ -2116,10 +2256,10 @@ static CURLcode imap_parse_url_path(struct Curl_easy *data) *
* Parse the custom request.
*/
-static CURLcode imap_parse_custom_request(struct Curl_easy *data)
+static CURLcode imap_parse_custom_request(struct Curl_easy *data,
+ struct IMAP *imap)
{
CURLcode result = CURLE_OK;
- struct IMAP *imap = data->req.p.imap;
const char *custom = data->set.str[STRING_CUSTOMREQUEST];
if(custom) {
|