support backslashes and quotes in quoted IMAP strings
the RFCs require it - well hidden in the BNF at the bottom. patch somewhat inspired by "guns" <self@sungpae.com>.
This commit is contained in:
parent
c0bf867669
commit
2213d6976c
161
src/drv_imap.c
161
src/drv_imap.c
@ -355,6 +355,88 @@ submit_imap_cmd( imap_store_t *ctx, struct imap_cmd *cmd )
|
||||
return send_imap_cmd( ctx, cmd );
|
||||
}
|
||||
|
||||
/* Minimal printf() replacement that supports an %\s format sequence to print backslash-escaped
|
||||
* string literals. Note that this does not automatically add quotes around the printed string,
|
||||
* so it is possible to concatenate multiple segments. */
|
||||
static char *
|
||||
imap_vprintf( const char *fmt, va_list ap )
|
||||
{
|
||||
const char *s, *es;
|
||||
char *d, *ed;
|
||||
int maxlen;
|
||||
char c;
|
||||
char buf[1024]; /* Minimal supported command buffer size per IMAP spec. */
|
||||
|
||||
d = buf;
|
||||
ed = d + sizeof(buf);
|
||||
s = fmt;
|
||||
for (;;) {
|
||||
c = *fmt;
|
||||
if (!c || c == '%') {
|
||||
int l = fmt - s;
|
||||
if (d + l > ed)
|
||||
oob();
|
||||
memcpy( d, s, l );
|
||||
d += l;
|
||||
if (!c) {
|
||||
l = d - buf;
|
||||
ed = nfmalloc( l + 1 );
|
||||
memcpy( ed, buf, l );
|
||||
ed[l] = 0;
|
||||
return ed;
|
||||
}
|
||||
maxlen = INT_MAX;
|
||||
c = *++fmt;
|
||||
if (c == '\\') {
|
||||
c = *++fmt;
|
||||
if (c != 's') {
|
||||
fputs( "Fatal: unsupported escaped format specifier. Please report a bug.\n", stderr );
|
||||
abort();
|
||||
}
|
||||
s = va_arg( ap, const char * );
|
||||
while ((c = *s++)) {
|
||||
if (d + 2 > ed)
|
||||
oob();
|
||||
if (c == '\\' || c == '"')
|
||||
*d++ = '\\';
|
||||
*d++ = c;
|
||||
}
|
||||
} else { /* \\ cannot be combined with anything else. */
|
||||
if (c == '.') {
|
||||
c = *++fmt;
|
||||
if (c != '*') {
|
||||
fputs( "Fatal: unsupported string length specification. Please report a bug.\n", stderr );
|
||||
abort();
|
||||
}
|
||||
maxlen = va_arg( ap , int );
|
||||
c = *++fmt;
|
||||
}
|
||||
if (c == 'c') {
|
||||
if (d + 1 > ed)
|
||||
oob();
|
||||
*d++ = (char)va_arg( ap , int );
|
||||
} else if (c == 's') {
|
||||
s = va_arg( ap, const char * );
|
||||
es = memchr( s, 0, maxlen );
|
||||
l = es ? es - s : maxlen;
|
||||
if (d + l > ed)
|
||||
oob();
|
||||
memcpy( d, s, l );
|
||||
d += l;
|
||||
} else if (c == 'd') {
|
||||
d += nfsnprintf( d, ed - d, "%d", va_arg( ap , int ) );
|
||||
} else {
|
||||
fputs( "Fatal: unsupported format specifier. Please report a bug.\n", stderr );
|
||||
abort();
|
||||
}
|
||||
}
|
||||
s = ++fmt;
|
||||
} else {
|
||||
fmt++;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static int
|
||||
imap_exec( imap_store_t *ctx, struct imap_cmd *cmdp,
|
||||
void (*done)( imap_store_t *ctx, struct imap_cmd *cmd, int response ),
|
||||
@ -366,7 +448,7 @@ imap_exec( imap_store_t *ctx, struct imap_cmd *cmdp,
|
||||
cmdp = new_imap_cmd( sizeof(*cmdp) );
|
||||
cmdp->param.done = done;
|
||||
va_start( ap, fmt );
|
||||
nfvasprintf( &cmdp->cmd, fmt, ap );
|
||||
cmdp->cmd = imap_vprintf( fmt, ap );
|
||||
va_end( ap );
|
||||
return submit_imap_cmd( ctx, cmdp );
|
||||
}
|
||||
@ -457,10 +539,25 @@ imap_refcounted_done_box( imap_store_t *ctx ATTR_UNUSED, struct imap_cmd *cmd, i
|
||||
imap_refcounted_done( sts );
|
||||
}
|
||||
|
||||
static const char *
|
||||
imap_strchr( const char *s, char tc )
|
||||
{
|
||||
for (;; s++) {
|
||||
char c = *s;
|
||||
if (c == '\\')
|
||||
c = *++s;
|
||||
if (!c)
|
||||
return 0;
|
||||
if (c == tc)
|
||||
return s;
|
||||
}
|
||||
}
|
||||
|
||||
static char *
|
||||
next_arg( char **ps )
|
||||
{
|
||||
char *ret, *s;
|
||||
char *ret, *s, *d;
|
||||
char c;
|
||||
|
||||
assert( ps );
|
||||
s = *ps;
|
||||
@ -473,20 +570,30 @@ next_arg( char **ps )
|
||||
return 0;
|
||||
}
|
||||
if (*s == '"') {
|
||||
++s;
|
||||
ret = s;
|
||||
s = strchr( s, '"' );
|
||||
s++;
|
||||
ret = d = s;
|
||||
while ((c = *s++) != '"') {
|
||||
if (c == '\\')
|
||||
c = *s++;
|
||||
if (!c) {
|
||||
*ps = 0;
|
||||
return 0;
|
||||
}
|
||||
*d++ = c;
|
||||
}
|
||||
*d = 0;
|
||||
} else {
|
||||
ret = s;
|
||||
while (*s && !isspace( (unsigned char)*s ))
|
||||
while ((c = *s)) {
|
||||
if (isspace( (unsigned char)c )) {
|
||||
*s++ = 0;
|
||||
break;
|
||||
}
|
||||
s++;
|
||||
}
|
||||
}
|
||||
if (s) {
|
||||
if (*s)
|
||||
*s++ = 0;
|
||||
if (!*s)
|
||||
s = 0;
|
||||
}
|
||||
if (!*s)
|
||||
s = 0;
|
||||
|
||||
*ps = s;
|
||||
return ret;
|
||||
@ -529,8 +636,9 @@ static int
|
||||
parse_imap_list( imap_store_t *ctx, char **sp, parse_list_state_t *sts )
|
||||
{
|
||||
list_t *cur, **curp;
|
||||
char *s = *sp, *p;
|
||||
char *s = *sp, *d, *p;
|
||||
int bytes;
|
||||
char c;
|
||||
|
||||
assert( sts );
|
||||
assert( sts->level > 0 );
|
||||
@ -595,12 +703,15 @@ parse_imap_list( imap_store_t *ctx, char **sp, parse_list_state_t *sts )
|
||||
} else if (*s == '"') {
|
||||
/* quoted string */
|
||||
s++;
|
||||
p = s;
|
||||
for (; *s != '"'; s++)
|
||||
if (!*s)
|
||||
p = d = s;
|
||||
while ((c = *s++) != '"') {
|
||||
if (c == '\\')
|
||||
c = *s++;
|
||||
if (!c)
|
||||
goto bail;
|
||||
cur->len = s - p;
|
||||
s++;
|
||||
*d++ = c;
|
||||
}
|
||||
cur->len = d - p;
|
||||
cur->val = nfmalloc( cur->len + 1 );
|
||||
memcpy( cur->val, p, cur->len );
|
||||
cur->val[cur->len] = 0;
|
||||
@ -1140,7 +1251,7 @@ imap_socket_read( void *aux )
|
||||
cmd2->gen.param.high_prio = 1;
|
||||
p = strchr( cmdp->cmd, '"' );
|
||||
if (imap_exec( ctx, &cmd2->gen, get_cmd_result_p2,
|
||||
"CREATE %.*s", strchr( p + 1, '"' ) - p + 1, p ) < 0)
|
||||
"CREATE %.*s", imap_strchr( p + 1, '"' ) - p + 1, p ) < 0)
|
||||
return;
|
||||
continue;
|
||||
}
|
||||
@ -1571,7 +1682,7 @@ imap_open_store_authenticate2( imap_store_t *ctx )
|
||||
#endif
|
||||
warn( "*** IMAP Warning *** Password is being sent in the clear\n" );
|
||||
imap_exec( ctx, 0, imap_open_store_authenticate2_p2,
|
||||
"LOGIN \"%s\" \"%s\"", srvc->user, srvc->pass );
|
||||
"LOGIN \"%\\s\" \"%\\s\"", srvc->user, srvc->pass );
|
||||
return;
|
||||
|
||||
bail:
|
||||
@ -1695,7 +1806,7 @@ imap_select( store_t *gctx, int create,
|
||||
cmd->gen.param.create = create;
|
||||
cmd->gen.param.trycreate = 1;
|
||||
imap_exec( ctx, &cmd->gen, imap_done_simple_box,
|
||||
"SELECT \"%s\"", buf );
|
||||
"SELECT \"%\\s\"", buf );
|
||||
free( buf );
|
||||
}
|
||||
|
||||
@ -1912,7 +2023,7 @@ imap_trash_msg( store_t *gctx, message_t *msg,
|
||||
return;
|
||||
}
|
||||
imap_exec( ctx, &cmd->gen, imap_done_simple_msg,
|
||||
"UID COPY %d \"%s\"", msg->uid, buf );
|
||||
"UID COPY %d \"%\\s\"", msg->uid, buf );
|
||||
free( buf );
|
||||
}
|
||||
|
||||
@ -1966,10 +2077,10 @@ imap_store_msg( store_t *gctx, msg_data_t *data, int to_trash,
|
||||
# pragma GCC diagnostic pop
|
||||
#endif
|
||||
imap_exec( ctx, &cmd->gen, imap_store_msg_p2,
|
||||
"APPEND \"%s\" %s\"%s\" ", buf, flagstr, datestr );
|
||||
"APPEND \"%\\s\" %s\"%\\s\" ", buf, flagstr, datestr );
|
||||
} else {
|
||||
imap_exec( ctx, &cmd->gen, imap_store_msg_p2,
|
||||
"APPEND \"%s\" %s", buf, flagstr );
|
||||
"APPEND \"%\\s\" %s", buf, flagstr );
|
||||
}
|
||||
free( buf );
|
||||
}
|
||||
@ -2023,7 +2134,7 @@ imap_list( store_t *gctx, int flags,
|
||||
|
||||
if (((flags & LIST_PATH) &&
|
||||
imap_exec( ctx, imap_refcounted_new_cmd( sts ), imap_refcounted_done_box,
|
||||
"LIST \"\" \"%s*\"", ctx->prefix ) < 0) ||
|
||||
"LIST \"\" \"%\\s*\"", ctx->prefix ) < 0) ||
|
||||
((flags & LIST_INBOX) && (!(flags & LIST_PATH) || *ctx->prefix) &&
|
||||
imap_exec( ctx, imap_refcounted_new_cmd( sts ), imap_refcounted_done_box,
|
||||
"LIST \"\" INBOX*" ) < 0))
|
||||
|
Loading…
x
Reference in New Issue
Block a user