@ -69,17 +69,23 @@ typedef struct imap_server_conf {
char failed ;
char failed ;
} imap_server_conf_t ;
} imap_server_conf_t ;
typedef struct {
typedef union imap_store_conf {
store_conf_t gen ;
store_conf_t gen ;
imap_server_conf_t * server ;
struct {
char delimiter ;
STORE_CONF
char use_namespace ;
imap_server_conf_t * server ;
char use_lsub ;
char delimiter ;
char use_namespace ;
char use_lsub ;
} ;
} imap_store_conf_t ;
} imap_store_conf_t ;
typedef struct {
typedef union imap_message {
message_t gen ;
message_t gen ;
/* uint seq; will be needed when expunges are tracked */
struct {
MESSAGE ( union imap_message )
// uint seq; will be needed when expunges are tracked
} ;
} imap_message_t ;
} imap_message_t ;
# define NIL (void*)0x1
# define NIL (void*)0x1
@ -93,7 +99,7 @@ typedef struct _list {
# define MAX_LIST_DEPTH 5
# define MAX_LIST_DEPTH 5
typedef struct imap_store imap_store_t ;
typedef union imap_store imap_store_t ;
typedef struct {
typedef struct {
list_t * head , * * stack [ MAX_LIST_DEPTH ] ;
list_t * head , * * stack [ MAX_LIST_DEPTH ] ;
@ -103,116 +109,142 @@ typedef struct {
typedef struct imap_cmd imap_cmd_t ;
typedef struct imap_cmd imap_cmd_t ;
struct imap_store {
union imap_store {
store_t gen ;
store_t gen ;
const char * label ; /* foreign */
struct {
const char * prefix ;
STORE ( union imap_store )
const char * name ;
const char * label ; // foreign
uint ref_count ;
const char * prefix ;
uint opts ;
const char * name ;
enum { SST_BAD , SST_HALF , SST_GOOD } state ;
uint ref_count ;
/* trash folder's existence is not confirmed yet */
uint opts ;
enum { TrashUnknown , TrashChecking , TrashKnown } trashnc ;
enum { SST_BAD , SST_HALF , SST_GOOD } state ;
// What kind of BODY-less FETCH response we're expecting.
// The trash folder's existence is not confirmed yet
enum { FetchNone , FetchMsgs , FetchUidNext } fetch_sts ;
enum { TrashUnknown , TrashChecking , TrashKnown } trashnc ;
uint got_namespace : 1 ;
// What kind of BODY-less FETCH response we're expecting
uint has_forwarded : 1 ;
enum { FetchNone , FetchMsgs , FetchUidNext } fetch_sts ;
char delimiter [ 2 ] ; /* hierarchy delimiter */
uint got_namespace : 1 ;
char * ns_prefix , ns_delimiter ; /* NAMESPACE info */
uint has_forwarded : 1 ;
string_list_t * boxes ; // _list results
char delimiter [ 2 ] ; // Hierarchy delimiter
char listed ; // was _list already run with these flags?
char * ns_prefix , ns_delimiter ; // NAMESPACE info
// note that the message counts do _not_ reflect stats from msgs,
string_list_t * boxes ; // _list results
// but mailbox totals.
char listed ; // was _list already run with these flags?
int total_msgs , recent_msgs ;
// note that the message counts do _not_ reflect stats from msgs,
uint uidvalidity , uidnext ;
// but mailbox totals.
message_t * msgs ;
int total_msgs , recent_msgs ;
message_t * * msgapp ; /* FETCH results */
uint uidvalidity , uidnext ;
uint caps ; /* CAPABILITY results */
imap_message_t * * msgapp , * msgs ; // FETCH results
string_list_t * auth_mechs ;
uint caps ; // CAPABILITY results
parse_list_state_t parse_list_sts ;
string_list_t * auth_mechs ;
/* command queue */
parse_list_state_t parse_list_sts ;
imap_cmd_t * pending , * * pending_append ;
// Command queue
imap_cmd_t * in_progress , * * in_progress_append ;
imap_cmd_t * pending , * * pending_append ;
imap_cmd_t * wait_check , * * wait_check_append ;
imap_cmd_t * in_progress , * * in_progress_append ;
int nexttag , num_in_progress , num_wait_check ;
imap_cmd_t * wait_check , * * wait_check_append ;
uint buffer_mem ; /* memory currently occupied by buffers in the queue */
int nexttag , num_in_progress , num_wait_check ;
uint buffer_mem ; // Memory currently occupied by buffers in the queue
/* Used during sequential operations like connect */
enum { GreetingPending = 0 , GreetingBad , GreetingOk , GreetingPreauth } greeting ;
// Used during sequential operations like connect
int expectBYE ; /* LOGOUT is in progress */
enum { GreetingPending = 0 , GreetingBad , GreetingOk , GreetingPreauth } greeting ;
int expectEOF ; /* received LOGOUT's OK or unsolicited BYE */
int expectBYE ; // LOGOUT is in progress
int canceling ; /* imap_cancel() is in progress */
int expectEOF ; // received LOGOUT's OK or unsolicited BYE
union {
int canceling ; // imap_cancel() is in progress
void ( * imap_open ) ( int sts , void * aux ) ;
union {
void ( * imap_cancel ) ( void * aux ) ;
void ( * imap_open ) ( int sts , void * aux ) ;
} callbacks ;
void ( * imap_cancel ) ( void * aux ) ;
void * callback_aux ;
} callbacks ;
void * callback_aux ;
# ifdef HAVE_LIBSASL
# ifdef HAVE_LIBSASL
sasl_conn_t * sasl ;
sasl_conn_t * sasl ;
int sasl_cont ;
int sasl_cont ;
# endif
# endif
void ( * bad_callback ) ( void * aux ) ;
void ( * bad_callback ) ( void * aux ) ;
void * bad_callback_aux ;
void * bad_callback_aux ;
conn_t conn ; /* this is BIG, so put it last */
conn_t conn ; // This is BIG, so put it last
} ;
} ;
} ;
struct imap_cmd {
# define IMAP_CMD \
struct imap_cmd * next ;
struct imap_cmd * next ; \
char * cmd ;
char * cmd ; \
int tag ;
int tag ; \
\
struct {
struct { \
/* Will be called on each continuation request until it resets this pointer.
/* Will be called on each continuation request until it resets this pointer. \
* Needs to invoke bad_callback and return - 1 on error , otherwise return 0. */
* Needs to invoke bad_callback and return - 1 on error , otherwise return 0. */ \
int ( * cont ) ( imap_store_t * ctx , imap_cmd_t * cmd , const char * prompt ) ;
int ( * cont ) ( imap_store_t * ctx , imap_cmd_t * cmd , const char * prompt ) ; \
void ( * done ) ( imap_store_t * ctx , imap_cmd_t * cmd , int response ) ;
void ( * done ) ( imap_store_t * ctx , imap_cmd_t * cmd , int response ) ; \
char * data ;
char * data ; \
uint data_len ;
uint data_len ; \
uint uid ; /* to identify fetch responses */
uint uid ; /* to identify fetch responses */ \
char high_prio ; /* if command is queued, put it at the front of the queue. */
char high_prio ; /* if command is queued, put it at the front of the queue. */ \
char wait_check ; // Don't report success until subsequent CHECK success.
char wait_check ; /* Don't report success until subsequent CHECK success. */ \
char to_trash ; /* we are storing to trash, not current. */
char to_trash ; /* we are storing to trash, not current. */ \
char create ; /* create the mailbox if we get an error which suggests so. */
char create ; /* create the mailbox if we get an error which suggests so. */ \
char failok ; /* Don't complain about NO response. */
char failok ; /* Don't complain about NO response. */ \
} param ;
} param ;
struct imap_cmd {
IMAP_CMD
} ;
} ;
typedef struct {
# define IMAP_CMD_SIMPLE \
imap_cmd_t gen ;
IMAP_CMD \
void ( * callback ) ( int sts , void * aux ) ;
void ( * callback ) ( int sts , void * aux ) ; \
void * callback_aux ;
void * callback_aux ;
typedef union {
imap_cmd_t gen ;
struct {
IMAP_CMD_SIMPLE
} ;
} imap_cmd_simple_t ;
} imap_cmd_simple_t ;
typedef struct {
typedef union {
imap_cmd_simple_t gen ;
imap_cmd_simple_t gen ;
msg_data_t * msg_data ;
struct {
IMAP_CMD_SIMPLE
msg_data_t * msg_data ;
} ;
} imap_cmd_fetch_msg_t ;
} imap_cmd_fetch_msg_t ;
typedef struct {
typedef union {
imap_cmd_t gen ;
imap_cmd_t gen ;
void ( * callback ) ( int sts , uint uid , void * aux ) ;
struct {
void * callback_aux ;
IMAP_CMD
uint out_uid ;
void ( * callback ) ( int sts , uint uid , void * aux ) ;
void * callback_aux ;
uint out_uid ;
} ;
} imap_cmd_out_uid_t ;
} imap_cmd_out_uid_t ;
typedef struct {
typedef union {
imap_cmd_t gen ;
imap_cmd_t gen ;
void ( * callback ) ( int sts , message_t * msgs , void * aux ) ;
struct {
void * callback_aux ;
IMAP_CMD
message_t * * out_msgs ;
void ( * callback ) ( int sts , message_t * msgs , void * aux ) ;
uint uid ;
void * callback_aux ;
imap_message_t * * out_msgs ;
uint uid ;
} ;
} imap_cmd_find_new_t ;
} imap_cmd_find_new_t ;
typedef struct {
# define IMAP_CMD_REFCOUNTED_STATE \
uint ref_count ;
uint ref_count ; \
int ret_val ;
int ret_val ;
} imap_cmd_refcounted_state_t ;
typedef struct {
typedef struct {
IMAP_CMD_REFCOUNTED_STATE
} imap_cmd_refcounted_state_t ;
typedef union {
imap_cmd_t gen ;
imap_cmd_t gen ;
imap_cmd_refcounted_state_t * state ;
struct {
IMAP_CMD
imap_cmd_refcounted_state_t * state ;
} ;
} imap_cmd_refcounted_t ;
} imap_cmd_refcounted_t ;
# define CAP(cap) (ctx->caps & (1 << (cap)))
# define CAP(cap) (ctx->caps & (1 << (cap)))
@ -283,8 +315,8 @@ new_imap_cmd( uint size )
# define INIT_IMAP_CMD_X(type, cmdp, cb, aux) \
# define INIT_IMAP_CMD_X(type, cmdp, cb, aux) \
cmdp = ( type * ) new_imap_cmd ( sizeof ( * cmdp ) ) ; \
cmdp = ( type * ) new_imap_cmd ( sizeof ( * cmdp ) ) ; \
cmdp - > gen . callback = cb ; \
cmdp - > callback = cb ; \
cmdp - > gen . callback_aux = aux ;
cmdp - > callback_aux = aux ;
static void
static void
done_imap_cmd ( imap_store_t * ctx , imap_cmd_t * cmd , int response )
done_imap_cmd ( imap_store_t * ctx , imap_cmd_t * cmd , int response )
@ -391,7 +423,7 @@ cmd_sendable( imap_store_t *ctx, imap_cmd_t *cmd )
/* Don't build a queue of MOVE/COPY/APPEND commands that may all fail. */
/* Don't build a queue of MOVE/COPY/APPEND commands that may all fail. */
return 0 ;
return 0 ;
}
}
if ( ctx - > num_in_progress > = ( ( imap_store_conf_t * ) ctx - > gen . conf ) - > server - > max_in_progress ) {
if ( ctx - > num_in_progress > = ctx - > conf - > server - > max_in_progress ) {
/* Too many commands in flight. */
/* Too many commands in flight. */
return 0 ;
return 0 ;
}
}
@ -628,15 +660,15 @@ imap_refcounted_new_cmd( imap_cmd_refcounted_state_t *sts )
}
}
# define DONE_REFCOUNTED_STATE(sts) \
# define DONE_REFCOUNTED_STATE(sts) \
if ( ! - - sts - > gen . ref_count ) { \
if ( ! - - sts - > ref_count ) { \
sts - > callback ( sts - > gen . ret_val , sts - > callback_aux ) ; \
sts - > callback ( sts - > ret_val , sts - > callback_aux ) ; \
free ( sts ) ; \
free ( sts ) ; \
}
}
# define DONE_REFCOUNTED_STATE_ARGS(sts, finalize, ...) \
# define DONE_REFCOUNTED_STATE_ARGS(sts, finalize, ...) \
if ( ! - - sts - > gen . ref_count ) { \
if ( ! - - sts - > ref_count ) { \
finalize \
finalize \
sts - > callback ( sts - > gen . ret_val , __VA_ARGS__ , sts - > callback_aux ) ; \
sts - > callback ( sts - > ret_val , __VA_ARGS__ , sts - > callback_aux ) ; \
free ( sts ) ; \
free ( sts ) ; \
}
}
@ -1164,16 +1196,16 @@ parse_fetch_rsp( imap_store_t *ctx, list_t *list, char *s ATTR_UNUSED )
ctx - > uidnext = uid + 1 ;
ctx - > uidnext = uid + 1 ;
} else if ( ctx - > fetch_sts = = FetchMsgs ) {
} else if ( ctx - > fetch_sts = = FetchMsgs ) {
cur = nfcalloc ( sizeof ( * cur ) ) ;
cur = nfcalloc ( sizeof ( * cur ) ) ;
* ctx - > msgapp = & cur - > gen ;
* ctx - > msgapp = cur ;
ctx - > msgapp = & cur - > gen . next ;
ctx - > msgapp = & cur - > next ;
cur - > gen . uid = uid ;
cur - > uid = uid ;
cur - > gen . flags = mask ;
cur - > flags = mask ;
cur - > gen . status = status ;
cur - > status = status ;
cur - > gen . size = size ;
cur - > size = size ;
if ( msgid )
if ( msgid )
cur - > gen . msgid = nfstrndup ( msgid , msgid_len ) ;
cur - > msgid = nfstrndup ( msgid , msgid_len ) ;
if ( tuid )
if ( tuid )
memcpy ( cur - > gen . tuid , tuid , TUIDL ) ;
memcpy ( cur - > tuid , tuid , TUIDL ) ;
status & = ~ ( M_FLAGS | M_RECENT | M_SIZE | M_HEADER ) ;
status & = ~ ( M_FLAGS | M_RECENT | M_SIZE | M_HEADER ) ;
} else {
} else {
badrsp :
badrsp :
@ -1207,7 +1239,7 @@ parse_capability( imap_store_t *ctx, char *cmd )
ctx - > caps | = 1 < < i ;
ctx - > caps | = 1 < < i ;
}
}
}
}
ctx - > caps & = ~ ( ( imap_store_conf_t * ) ctx - > gen . conf ) - > server - > cap_mask ;
ctx - > caps & = ~ ctx - > conf - > server - > cap_mask ;
if ( ! CAP ( NOLOGIN ) )
if ( ! CAP ( NOLOGIN ) )
add_string_list ( & ctx - > auth_mechs , " LOGIN " ) ;
add_string_list ( & ctx - > auth_mechs , " LOGIN " ) ;
}
}
@ -1418,12 +1450,15 @@ prepare_box( char **buf, const imap_store_t *ctx )
static int
static int
prepare_trash ( char * * buf , const imap_store_t * ctx )
prepare_trash ( char * * buf , const imap_store_t * ctx )
{
{
return prepare_name ( buf , ctx , ctx - > prefix , ctx - > gen . conf - > trash ) ;
return prepare_name ( buf , ctx , ctx - > prefix , ctx - > conf - > trash ) ;
}
}
typedef struct {
typedef union {
imap_cmd_t gen ;
imap_cmd_t gen ;
imap_cmd_t * orig_cmd ;
struct {
IMAP_CMD
imap_cmd_t * orig_cmd ;
} ;
} imap_cmd_trycreate_t ;
} imap_cmd_trycreate_t ;
static void imap_open_store_greeted ( imap_store_t * ) ;
static void imap_open_store_greeted ( imap_store_t * ) ;
@ -1589,7 +1624,7 @@ imap_socket_read( void *aux )
imap_cmd_trycreate_t * cmd2 =
imap_cmd_trycreate_t * cmd2 =
( imap_cmd_trycreate_t * ) new_imap_cmd ( sizeof ( * cmd2 ) ) ;
( imap_cmd_trycreate_t * ) new_imap_cmd ( sizeof ( * cmd2 ) ) ;
cmd2 - > orig_cmd = cmdp ;
cmd2 - > orig_cmd = cmdp ;
cmd2 - > gen . param . high_prio = 1 ;
cmd2 - > param . high_prio = 1 ;
p = strchr ( cmdp - > cmd , ' " ' ) ;
p = strchr ( cmdp - > cmd , ' " ' ) ;
imap_exec ( ctx , & cmd2 - > gen , get_cmd_result_p2 ,
imap_exec ( ctx , & cmd2 - > gen , get_cmd_result_p2 ,
" CREATE %.*s " , imap_strchr ( p + 1 , ' " ' ) - p + 1 , p ) ;
" CREATE %.*s " , imap_strchr ( p + 1 , ' " ' ) - p + 1 , p ) ;
@ -1669,7 +1704,7 @@ imap_cancel_store( store_t *gctx )
cancel_pending_imap_cmds ( ctx ) ;
cancel_pending_imap_cmds ( ctx ) ;
free ( ctx - > ns_prefix ) ;
free ( ctx - > ns_prefix ) ;
free_string_list ( ctx - > auth_mechs ) ;
free_string_list ( ctx - > auth_mechs ) ;
free_generic_messages ( ctx - > msgs ) ;
free_generic_messages ( & ctx - > msgs - > gen ) ;
free_string_list ( ctx - > boxes ) ;
free_string_list ( ctx - > boxes ) ;
imap_deref ( ctx ) ;
imap_deref ( ctx ) ;
}
}
@ -1701,12 +1736,12 @@ imap_invoke_bad_callback( imap_store_t *ctx )
/******************* imap_free_store *******************/
/******************* imap_free_store *******************/
static store_t * unowned ;
static imap_ store_t * unowned ;
static void
static void
imap_cancel_unowned ( void * gctx )
imap_cancel_unowned ( void * gctx )
{
{
store_t * store , * * storep ;
imap_ store_t * store , * * storep ;
for ( storep = & unowned ; ( store = * storep ) ; storep = & store - > next )
for ( storep = & unowned ; ( store = * storep ) ; storep = & store - > next )
if ( store = = gctx ) {
if ( store = = gctx ) {
@ -1723,11 +1758,11 @@ imap_free_store( store_t *gctx )
assert ( ! ctx - > pending & & ! ctx - > in_progress & & ! ctx - > wait_check ) ;
assert ( ! ctx - > pending & & ! ctx - > in_progress & & ! ctx - > wait_check ) ;
free_generic_messages ( ctx - > msgs ) ;
free_generic_messages ( & ctx - > msgs - > gen ) ;
ctx - > msgs = NULL ;
ctx - > msgs = NULL ;
imap_set_bad_callback ( gctx , imap_cancel_unowned , gctx ) ;
imap_set_bad_callback ( gctx , imap_cancel_unowned , gctx ) ;
g ctx- > next = unowned ;
ctx - > next = unowned ;
unowned = g ctx;
unowned = ctx ;
}
}
/******************* imap_cleanup *******************/
/******************* imap_cleanup *******************/
@ -1737,16 +1772,16 @@ static void imap_cleanup_p2( imap_store_t *, imap_cmd_t *, int );
static void
static void
imap_cleanup ( void )
imap_cleanup ( void )
{
{
store_t * ctx , * nctx ;
imap_ store_t * ctx , * nctx ;
for ( ctx = unowned ; ctx ; ctx = nctx ) {
for ( ctx = unowned ; ctx ; ctx = nctx ) {
nctx = ctx - > next ;
nctx = ctx - > next ;
imap_set_bad_callback ( ctx , ( void ( * ) ( void * ) ) imap_cancel_store , ctx ) ;
imap_set_bad_callback ( & ctx - > gen , ( void ( * ) ( void * ) ) imap_cancel_store , ctx ) ;
if ( ( ( imap_store_t * ) ctx ) - > state ! = SST_BAD ) {
if ( ( ( imap_store_t * ) ctx ) - > state ! = SST_BAD ) {
( ( imap_store_t * ) ctx ) - > expectBYE = 1 ;
( ( imap_store_t * ) ctx ) - > expectBYE = 1 ;
imap_exec ( ( imap_store_t * ) ctx , NULL , imap_cleanup_p2 , " LOGOUT " ) ;
imap_exec ( ( imap_store_t * ) ctx , NULL , imap_cleanup_p2 , " LOGOUT " ) ;
} else {
} else {
imap_cancel_store ( ctx ) ;
imap_cancel_store ( & ctx - > gen ) ;
}
}
}
}
}
}
@ -1794,20 +1829,19 @@ imap_alloc_store( store_conf_t *conf, const char *label )
{
{
imap_store_conf_t * cfg = ( imap_store_conf_t * ) conf ;
imap_store_conf_t * cfg = ( imap_store_conf_t * ) conf ;
imap_server_conf_t * srvc = cfg - > server ;
imap_server_conf_t * srvc = cfg - > server ;
imap_store_t * ctx ;
imap_store_t * ctx , * * ctxp ;
store_t * * ctxp ;
/* First try to recycle a whole store. */
/* First try to recycle a whole store. */
for ( ctxp = & unowned ; ( ctx = ( imap_store_t * ) * ctxp ) ; ctxp = & ctx - > gen . next )
for ( ctxp = & unowned ; ( ctx = * ctxp ) ; ctxp = & ctx - > next )
if ( ctx - > state = = SST_GOOD & & ctx - > gen . conf = = con f ) {
if ( ctx - > state = = SST_GOOD & & ctx - > conf = = cfg ) {
* ctxp = ctx - > gen . next ;
* ctxp = ctx - > next ;
goto gotstore ;
goto gotstore ;
}
}
/* Then try to recycle a server connection. */
/* Then try to recycle a server connection. */
for ( ctxp = & unowned ; ( ctx = ( imap_store_t * ) * ctxp ) ; ctxp = & ctx - > gen . next )
for ( ctxp = & unowned ; ( ctx = * ctxp ) ; ctxp = & ctx - > next )
if ( ctx - > state ! = SST_BAD & & ( ( imap_store_conf_t * ) ctx - > gen . conf ) - > server = = srvc ) {
if ( ctx - > state ! = SST_BAD & & ctx - > conf - > server = = srvc ) {
* ctxp = ctx - > gen . next ;
* ctxp = ctx - > next ;
free_string_list ( ctx - > boxes ) ;
free_string_list ( ctx - > boxes ) ;
ctx - > boxes = NULL ;
ctx - > boxes = NULL ;
ctx - > listed = 0 ;
ctx - > listed = 0 ;
@ -1819,7 +1853,7 @@ imap_alloc_store( store_conf_t *conf, const char *label )
/* Finally, schedule opening a new server connection. */
/* Finally, schedule opening a new server connection. */
ctx = nfcalloc ( sizeof ( * ctx ) ) ;
ctx = nfcalloc ( sizeof ( * ctx ) ) ;
ctx - > gen . driver = & imap_driver ;
ctx - > driver = & imap_driver ;
ctx - > ref_count = 1 ;
ctx - > ref_count = 1 ;
socket_init ( & ctx - > conn , & srvc - > sconf ,
socket_init ( & ctx - > conn , & srvc - > sconf ,
( void ( * ) ( void * ) ) imap_invoke_bad_callback ,
( void ( * ) ( void * ) ) imap_invoke_bad_callback ,
@ -1829,7 +1863,7 @@ imap_alloc_store( store_conf_t *conf, const char *label )
ctx - > wait_check_append = & ctx - > wait_check ;
ctx - > wait_check_append = & ctx - > wait_check ;
gotsrv :
gotsrv :
ctx - > gen . conf = con f ;
ctx - > conf = cfg ;
gotstore :
gotstore :
ctx - > label = label ;
ctx - > label = label ;
return & ctx - > gen ;
return & ctx - > gen ;
@ -1857,15 +1891,11 @@ static void
imap_open_store_connected ( int ok , void * aux )
imap_open_store_connected ( int ok , void * aux )
{
{
imap_store_t * ctx = ( imap_store_t * ) aux ;
imap_store_t * ctx = ( imap_store_t * ) aux ;
# ifdef HAVE_LIBSSL
imap_store_conf_t * cfg = ( imap_store_conf_t * ) ctx - > gen . conf ;
imap_server_conf_t * srvc = cfg - > server ;
# endif
if ( ! ok )
if ( ! ok )
imap_open_store_bail ( ctx , FAIL_WAIT ) ;
imap_open_store_bail ( ctx , FAIL_WAIT ) ;
# ifdef HAVE_LIBSSL
# ifdef HAVE_LIBSSL
else if ( srvc - > ssl_type = = SSL_IMAPS )
else if ( ctx - > conf - > server - > ssl_type = = SSL_IMAPS )
socket_start_tls ( & ctx - > conn , imap_open_store_tlsstarted1 ) ;
socket_start_tls ( & ctx - > conn , imap_open_store_tlsstarted1 ) ;
# endif
# endif
else
else
@ -1908,8 +1938,7 @@ static void
imap_open_store_authenticate ( imap_store_t * ctx )
imap_open_store_authenticate ( imap_store_t * ctx )
{
{
# ifdef HAVE_LIBSSL
# ifdef HAVE_LIBSSL
imap_store_conf_t * cfg = ( imap_store_conf_t * ) ctx - > gen . conf ;
imap_server_conf_t * srvc = ctx - > conf - > server ;
imap_server_conf_t * srvc = cfg - > server ;
# endif
# endif
if ( ctx - > greeting ! = GreetingPreauth ) {
if ( ctx - > greeting ! = GreetingPreauth ) {
@ -2108,7 +2137,7 @@ static int
process_sasl_step ( imap_store_t * ctx , int rc , const char * in , uint in_len ,
process_sasl_step ( imap_store_t * ctx , int rc , const char * in , uint in_len ,
sasl_interact_t * interact , const char * * out , uint * out_len )
sasl_interact_t * interact , const char * * out , uint * out_len )
{
{
imap_server_conf_t * srvc = ( ( imap_store_conf_t * ) ctx - > gen . conf ) - > server ;
imap_server_conf_t * srvc = ctx - > conf - > server ;
while ( rc = = SASL_INTERACT ) {
while ( rc = = SASL_INTERACT ) {
if ( process_sasl_interact ( interact , srvc ) < 0 )
if ( process_sasl_interact ( interact , srvc ) < 0 )
@ -2237,8 +2266,7 @@ done_sasl_auth( imap_store_t *ctx, imap_cmd_t *cmd ATTR_UNUSED, int response )
static void
static void
imap_open_store_authenticate2 ( imap_store_t * ctx )
imap_open_store_authenticate2 ( imap_store_t * ctx )
{
{
imap_store_conf_t * cfg = ( imap_store_conf_t * ) ctx - > gen . conf ;
imap_server_conf_t * srvc = ctx - > conf - > server ;
imap_server_conf_t * srvc = cfg - > server ;
string_list_t * mech , * cmech ;
string_list_t * mech , * cmech ;
int auth_login = 0 ;
int auth_login = 0 ;
int skipped_login = 0 ;
int skipped_login = 0 ;
@ -2409,10 +2437,10 @@ imap_open_store_compress_p2( imap_store_t *ctx, imap_cmd_t *cmd ATTR_UNUSED, int
static void
static void
imap_open_store_namespace ( imap_store_t * ctx )
imap_open_store_namespace ( imap_store_t * ctx )
{
{
imap_store_conf_t * cfg = ( imap_store_conf_t * ) ctx - > gen . conf ;
imap_store_conf_t * cfg = ctx - > conf ;
ctx - > state = SST_HALF ;
ctx - > state = SST_HALF ;
ctx - > prefix = cfg - > gen . path ;
ctx - > prefix = cfg - > path ;
ctx - > delimiter [ 0 ] = cfg - > delimiter ;
ctx - > delimiter [ 0 ] = cfg - > delimiter ;
if ( ( ( ! ctx - > prefix & & cfg - > use_namespace ) | | ! cfg - > delimiter ) & & CAP ( NAMESPACE ) ) {
if ( ( ( ! ctx - > prefix & & cfg - > use_namespace ) | | ! cfg - > delimiter ) & & CAP ( NAMESPACE ) ) {
/* get NAMESPACE info */
/* get NAMESPACE info */
@ -2439,9 +2467,7 @@ imap_open_store_namespace_p2( imap_store_t *ctx, imap_cmd_t *cmd ATTR_UNUSED, in
static void
static void
imap_open_store_namespace2 ( imap_store_t * ctx )
imap_open_store_namespace2 ( imap_store_t * ctx )
{
{
imap_store_conf_t * cfg = ( imap_store_conf_t * ) ctx - > gen . conf ;
if ( ! ctx - > prefix & & ctx - > conf - > use_namespace )
if ( ! ctx - > prefix & & cfg - > use_namespace )
ctx - > prefix = ctx - > ns_prefix ;
ctx - > prefix = ctx - > ns_prefix ;
if ( ! ctx - > delimiter [ 0 ] )
if ( ! ctx - > delimiter [ 0 ] )
ctx - > delimiter [ 0 ] = ctx - > ns_delimiter ;
ctx - > delimiter [ 0 ] = ctx - > ns_delimiter ;
@ -2471,7 +2497,7 @@ imap_open_store_ssl_bail( imap_store_t *ctx )
static void
static void
imap_open_store_bail ( imap_store_t * ctx , int failed )
imap_open_store_bail ( imap_store_t * ctx , int failed )
{
{
( ( imap_store_conf_t * ) ctx - > gen . conf ) - > server - > failed = ( char ) failed ;
ctx - > conf - > server - > failed = ( char ) failed ;
ctx - > callbacks . imap_open ( DRV_STORE_BAD , ctx - > callback_aux ) ;
ctx - > callbacks . imap_open ( DRV_STORE_BAD , ctx - > callback_aux ) ;
}
}
@ -2484,7 +2510,7 @@ imap_select_box( store_t *gctx, const char *name )
assert ( ! ctx - > pending & & ! ctx - > in_progress & & ! ctx - > wait_check ) ;
assert ( ! ctx - > pending & & ! ctx - > in_progress & & ! ctx - > wait_check ) ;
free_generic_messages ( ctx - > msgs ) ;
free_generic_messages ( & ctx - > msgs - > gen ) ;
ctx - > msgs = NULL ;
ctx - > msgs = NULL ;
ctx - > msgapp = & ctx - > msgs ;
ctx - > msgapp = & ctx - > msgs ;
@ -2498,10 +2524,13 @@ imap_get_box_path( store_t *gctx ATTR_UNUSED )
return NULL ;
return NULL ;
}
}
typedef struct {
typedef union {
imap_cmd_t gen ;
imap_cmd_t gen ;
void ( * callback ) ( int sts , uint uidvalidity , void * aux ) ;
struct {
void * callback_aux ;
IMAP_CMD
void ( * callback ) ( int sts , uint uidvalidity , void * aux ) ;
void * callback_aux ;
} ;
} imap_cmd_open_box_t ;
} imap_cmd_open_box_t ;
static void imap_open_box_p2 ( imap_store_t * , imap_cmd_t * , int ) ;
static void imap_open_box_p2 ( imap_store_t * , imap_cmd_t * , int ) ;
@ -2525,7 +2554,7 @@ imap_open_box( store_t *gctx,
ctx - > uidnext = 0 ;
ctx - > uidnext = 0 ;
INIT_IMAP_CMD ( imap_cmd_open_box_t , cmd , cb , aux )
INIT_IMAP_CMD ( imap_cmd_open_box_t , cmd , cb , aux )
cmd - > gen . param . failok = 1 ;
cmd - > param . failok = 1 ;
imap_exec ( ctx , & cmd - > gen , imap_open_box_p2 ,
imap_exec ( ctx , & cmd - > gen , imap_open_box_p2 ,
" SELECT \" % \\ s \" " , buf ) ;
" SELECT \" % \\ s \" " , buf ) ;
free ( buf ) ;
free ( buf ) ;
@ -2702,10 +2731,13 @@ imap_set_range( imap_range_t *ranges, uint *nranges, int low_flags, int high_fla
ranges [ r ] . flags | = ( ranges [ r ] . last < = maxlow ) ? low_flags : high_flags ;
ranges [ r ] . flags | = ( ranges [ r ] . last < = maxlow ) ? low_flags : high_flags ;
}
}
typedef struct {
typedef union {
imap_cmd_refcounted_state_t gen ;
imap_cmd_refcounted_state_t gen ;
void ( * callback ) ( int sts , message_t * msgs , int total_msgs , int recent_msgs , void * aux ) ;
struct {
void * callback_aux ;
IMAP_CMD_REFCOUNTED_STATE
void ( * callback ) ( int sts , message_t * msgs , int total_msgs , int recent_msgs , void * aux ) ;
void * callback_aux ;
} ;
} imap_load_box_state_t ;
} imap_load_box_state_t ;
static void imap_submit_load ( imap_store_t * , const char * , int , imap_load_box_state_t * ) ;
static void imap_submit_load ( imap_store_t * , const char * , int , imap_load_box_state_t * ) ;
@ -2778,13 +2810,13 @@ imap_sort_msgs_comp( const void *a_, const void *b_ )
static void
static void
imap_sort_msgs ( imap_store_t * ctx )
imap_sort_msgs ( imap_store_t * ctx )
{
{
uint count = count_generic_messages ( ctx - > msgs ) ;
uint count = count_generic_messages ( & ctx - > msgs - > gen ) ;
if ( count < = 1 )
if ( count < = 1 )
return ;
return ;
message_t * * t = nfmalloc ( sizeof ( * t ) * count ) ;
imap_ message_t * * t = nfmalloc ( sizeof ( * t ) * count ) ;
message_t * m = ctx - > msgs ;
imap_ message_t * m = ctx - > msgs ;
for ( uint i = 0 ; i < count ; i + + ) {
for ( uint i = 0 ; i < count ; i + + ) {
t [ i ] = m ;
t [ i ] = m ;
m = m - > next ;
m = m - > next ;
@ -2833,9 +2865,9 @@ imap_submit_load_p3( imap_store_t *ctx, imap_load_box_state_t *sts )
{
{
DONE_REFCOUNTED_STATE_ARGS ( sts , {
DONE_REFCOUNTED_STATE_ARGS ( sts , {
ctx - > fetch_sts = FetchNone ;
ctx - > fetch_sts = FetchNone ;
if ( sts - > gen . ret_val = = DRV_OK )
if ( sts - > ret_val = = DRV_OK )
imap_sort_msgs ( ctx ) ;
imap_sort_msgs ( ctx ) ;
} , ctx - > msgs , ctx - > total_msgs , ctx - > recent_msgs )
} , & ctx - > msgs - > gen , ctx - > total_msgs , ctx - > recent_msgs )
}
}
/******************* imap_fetch_msg *******************/
/******************* imap_fetch_msg *******************/
@ -2849,7 +2881,7 @@ imap_fetch_msg( store_t *ctx, message_t *msg, msg_data_t *data, int minimal,
imap_cmd_fetch_msg_t * cmd ;
imap_cmd_fetch_msg_t * cmd ;
INIT_IMAP_CMD_X ( imap_cmd_fetch_msg_t , cmd , cb , aux )
INIT_IMAP_CMD_X ( imap_cmd_fetch_msg_t , cmd , cb , aux )
cmd - > gen . gen . param . uid = msg - > uid ;
cmd - > param . uid = msg - > uid ;
cmd - > msg_data = data ;
cmd - > msg_data = data ;
data - > data = NULL ;
data - > data = NULL ;
imap_exec ( ( imap_store_t * ) ctx , & cmd - > gen . gen , imap_fetch_msg_p2 ,
imap_exec ( ( imap_store_t * ) ctx , & cmd - > gen . gen , imap_fetch_msg_p2 ,
@ -2890,10 +2922,13 @@ imap_make_flags( int flags, char *buf )
return d ;
return d ;
}
}
typedef struct {
typedef union {
imap_cmd_refcounted_state_t gen ;
imap_cmd_refcounted_state_t gen ;
void ( * callback ) ( int sts , void * aux ) ;
struct {
void * callback_aux ;
IMAP_CMD_REFCOUNTED_STATE
void ( * callback ) ( int sts , void * aux ) ;
void * callback_aux ;
} ;
} imap_set_msg_flags_state_t ;
} imap_set_msg_flags_state_t ;
static void imap_set_flags_p2 ( imap_store_t * , imap_cmd_t * , int ) ;
static void imap_set_flags_p2 ( imap_store_t * , imap_cmd_t * , int ) ;
@ -2954,10 +2989,13 @@ imap_set_flags_p3( imap_set_msg_flags_state_t *sts )
/******************* imap_close_box *******************/
/******************* imap_close_box *******************/
typedef struct {
typedef union {
imap_cmd_refcounted_state_t gen ;
imap_cmd_refcounted_state_t gen ;
void ( * callback ) ( int sts , void * aux ) ;
struct {
void * callback_aux ;
IMAP_CMD_REFCOUNTED_STATE
void ( * callback ) ( int sts , void * aux ) ;
void * callback_aux ;
} ;
} imap_expunge_state_t ;
} imap_expunge_state_t ;
static void imap_close_box_p2 ( imap_store_t * , imap_cmd_t * , int ) ;
static void imap_close_box_p2 ( imap_store_t * , imap_cmd_t * , int ) ;
@ -2971,9 +3009,9 @@ imap_close_box( store_t *gctx,
assert ( ! ctx - > num_wait_check ) ;
assert ( ! ctx - > num_wait_check ) ;
if ( ctx - > gen . conf - > trash & & CAP ( UIDPLUS ) ) {
if ( ctx - > conf - > trash & & CAP ( UIDPLUS ) ) {
INIT_REFCOUNTED_STATE ( imap_expunge_state_t , sts , cb , aux )
INIT_REFCOUNTED_STATE ( imap_expunge_state_t , sts , cb , aux )
message_t * msg , * fmsg , * nmsg ;
imap_ message_t * msg , * fmsg , * nmsg ;
int bl ;
int bl ;
char buf [ 1000 ] ;
char buf [ 1000 ] ;
@ -3030,8 +3068,8 @@ imap_trash_msg( store_t *gctx, message_t *msg,
char * buf ;
char * buf ;
INIT_IMAP_CMD ( imap_cmd_simple_t , cmd , cb , aux )
INIT_IMAP_CMD ( imap_cmd_simple_t , cmd , cb , aux )
cmd - > gen . param . create = 1 ;
cmd - > param . create = 1 ;
cmd - > gen . param . to_trash = 1 ;
cmd - > param . to_trash = 1 ;
if ( prepare_trash ( & buf , ctx ) < 0 ) {
if ( prepare_trash ( & buf , ctx ) < 0 ) {
cb ( DRV_BOX_BAD , aux ) ;
cb ( DRV_BOX_BAD , aux ) ;
return ;
return ;
@ -3064,13 +3102,13 @@ imap_store_msg( store_t *gctx, msg_data_t *data, int to_trash,
INIT_IMAP_CMD ( imap_cmd_out_uid_t , cmd , cb , aux )
INIT_IMAP_CMD ( imap_cmd_out_uid_t , cmd , cb , aux )
ctx - > buffer_mem + = data - > len ;
ctx - > buffer_mem + = data - > len ;
cmd - > gen . param . data_len = data - > len ;
cmd - > param . data_len = data - > len ;
cmd - > gen . param . data = data - > data ;
cmd - > param . data = data - > data ;
cmd - > out_uid = 0 ;
cmd - > out_uid = 0 ;
if ( to_trash ) {
if ( to_trash ) {
cmd - > gen . param . create = 1 ;
cmd - > param . create = 1 ;
cmd - > gen . param . to_trash = 1 ;
cmd - > param . to_trash = 1 ;
if ( prepare_trash ( & buf , ctx ) < 0 ) {
if ( prepare_trash ( & buf , ctx ) < 0 ) {
cb ( DRV_BOX_BAD , 0 , aux ) ;
cb ( DRV_BOX_BAD , 0 , aux ) ;
return ;
return ;
@ -3185,15 +3223,18 @@ imap_find_new_msgs_p4( imap_store_t *ctx ATTR_UNUSED, imap_cmd_t *gcmd, int resp
imap_cmd_find_new_t * cmdp = ( imap_cmd_find_new_t * ) gcmd ;
imap_cmd_find_new_t * cmdp = ( imap_cmd_find_new_t * ) gcmd ;
transform_box_response ( & response ) ;
transform_box_response ( & response ) ;
cmdp - > callback ( response , * cmdp - > out_msgs , cmdp - > callback_aux ) ;
cmdp - > callback ( response , & ( * cmdp - > out_msgs ) - > gen , cmdp - > callback_aux ) ;
}
}
/******************* imap_list_store *******************/
/******************* imap_list_store *******************/
typedef struct {
typedef union {
imap_cmd_refcounted_state_t gen ;
imap_cmd_refcounted_state_t gen ;
void ( * callback ) ( int sts , string_list_t * , void * aux ) ;
struct {
void * callback_aux ;
IMAP_CMD_REFCOUNTED_STATE
void ( * callback ) ( int sts , string_list_t * , void * aux ) ;
void * callback_aux ;
} ;
} imap_list_store_state_t ;
} imap_list_store_state_t ;
static void imap_list_store_p2 ( imap_store_t * , imap_cmd_t * , int ) ;
static void imap_list_store_p2 ( imap_store_t * , imap_cmd_t * , int ) ;
@ -3204,7 +3245,7 @@ imap_list_store( store_t *gctx, int flags,
void ( * cb ) ( int sts , string_list_t * boxes , void * aux ) , void * aux )
void ( * cb ) ( int sts , string_list_t * boxes , void * aux ) , void * aux )
{
{
imap_store_t * ctx = ( imap_store_t * ) gctx ;
imap_store_t * ctx = ( imap_store_t * ) gctx ;
imap_store_conf_t * cfg = ( imap_store_conf_t * ) ctx - > gen . conf ;
imap_store_conf_t * cfg = ctx - > conf ;
INIT_REFCOUNTED_STATE ( imap_list_store_state_t , sts , cb , aux )
INIT_REFCOUNTED_STATE ( imap_list_store_state_t , sts , cb , aux )
// ctx->prefix may be empty, "INBOX.", or something else.
// ctx->prefix may be empty, "INBOX.", or something else.
@ -3341,8 +3382,8 @@ imap_parse_store( conffile_t *cfg, store_conf_t **storep )
type = " IMAP account " ;
type = " IMAP account " ;
} else if ( ! strcasecmp ( " IMAPStore " , cfg - > cmd ) ) {
} else if ( ! strcasecmp ( " IMAPStore " , cfg - > cmd ) ) {
store = nfcalloc ( sizeof ( * store ) ) ;
store = nfcalloc ( sizeof ( * store ) ) ;
store - > gen . driver = & imap_driver ;
store - > driver = & imap_driver ;
name = store - > gen . name = nfstrdup ( cfg - > val ) ;
name = store - > name = nfstrdup ( cfg - > val ) ;
store - > use_namespace = 1 ;
store - > use_namespace = 1 ;
* storep = & store - > gen ;
* storep = & store - > gen ;
memset ( & sserver , 0 , sizeof ( sserver ) ) ;
memset ( & sserver , 0 , sizeof ( sserver ) ) ;
@ -3522,7 +3563,7 @@ imap_parse_store( conffile_t *cfg, store_conf_t **storep )
else if ( ! strcasecmp ( " SubscribedOnly " , cfg - > cmd ) )
else if ( ! strcasecmp ( " SubscribedOnly " , cfg - > cmd ) )
store - > use_lsub = parse_bool ( cfg ) ;
store - > use_lsub = parse_bool ( cfg ) ;
else if ( ! strcasecmp ( " Path " , cfg - > cmd ) )
else if ( ! strcasecmp ( " Path " , cfg - > cmd ) )
store - > gen . path = nfstrdup ( cfg - > val ) ;
store - > path = nfstrdup ( cfg - > val ) ;
else if ( ! strcasecmp ( " PathDelimiter " , cfg - > cmd ) ) {
else if ( ! strcasecmp ( " PathDelimiter " , cfg - > cmd ) ) {
if ( strlen ( cfg - > val ) ! = 1 ) {
if ( strlen ( cfg - > val ) ! = 1 ) {
error ( " %s:%d: Path delimiter must be exactly one character long \n " , cfg - > file , cfg - > line ) ;
error ( " %s:%d: Path delimiter must be exactly one character long \n " , cfg - > file , cfg - > line ) ;
@ -3622,7 +3663,7 @@ imap_parse_store( conffile_t *cfg, store_conf_t **storep )
if ( ! store - > server ) {
if ( ! store - > server ) {
store - > server = nfmalloc ( sizeof ( sserver ) ) ;
store - > server = nfmalloc ( sizeof ( sserver ) ) ;
memcpy ( store - > server , & sserver , sizeof ( sserver ) ) ;
memcpy ( store - > server , & sserver , sizeof ( sserver ) ) ;
store - > server - > name = store - > gen . name ;
store - > server - > name = store - > name ;
} else if ( acc_opt ) {
} else if ( acc_opt ) {
error ( " %s '%s' has both Account and account-specific options \n " , type , name ) ;
error ( " %s '%s' has both Account and account-specific options \n " , type , name ) ;
cfg - > err = 1 ;
cfg - > err = 1 ;