diff options
author | Vincent Richard <[email protected]> | 2013-04-26 20:52:13 +0000 |
---|---|---|
committer | Vincent Richard <[email protected]> | 2013-04-26 20:52:13 +0000 |
commit | ee68f6c06f553b9346ae9143be8efcfff8e902e9 (patch) | |
tree | 0f00f40f4d6fd8445e54705b2f36207d0f0570ea | |
parent | Added unit tests for IMAPTag. Initialize sequence number at 1. (diff) | |
download | vmime-ee68f6c06f553b9346ae9143be8efcfff8e902e9.tar.gz vmime-ee68f6c06f553b9346ae9143be8efcfff8e902e9.zip |
Allow SPACEs at end of response line (Apple iCloud IMAP server).
-rw-r--r-- | SConstruct | 1 | ||||
-rw-r--r-- | tests/net/imap/IMAPParserTest.cpp | 66 | ||||
-rw-r--r-- | vmime/net/imap/IMAPParser.hpp | 30 |
3 files changed, 96 insertions, 1 deletions
@@ -397,6 +397,7 @@ libvmimetest_sources = [ 'tests/net/pop3/POP3ResponseTest.cpp', 'tests/net/pop3/POP3UtilsTest.cpp', 'tests/net/imap/IMAPTagTest.cpp', + 'tests/net/imap/IMAPParserTest.cpp', 'tests/net/smtp/SMTPTransportTest.cpp', 'tests/net/smtp/SMTPCommandTest.cpp', 'tests/net/smtp/SMTPCommandSetTest.cpp', diff --git a/tests/net/imap/IMAPParserTest.cpp b/tests/net/imap/IMAPParserTest.cpp new file mode 100644 index 00000000..93f6e6fd --- /dev/null +++ b/tests/net/imap/IMAPParserTest.cpp @@ -0,0 +1,66 @@ +// +// VMime library (http://www.vmime.org) +// Copyright (C) 2002-2013 Vincent Richard <[email protected]> +// +// This program is free software; you can redistribute it and/or +// modify it under the terms of the GNU General Public License as +// published by the Free Software Foundation; either version 3 of +// the License, or (at your option) any later version. +// +// This program is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU +// General Public License for more details. +// +// You should have received a copy of the GNU General Public License along +// with this program; if not, write to the Free Software Foundation, Inc., +// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. +// +// Linking this library statically or dynamically with other modules is making +// a combined work based on this library. Thus, the terms and conditions of +// the GNU General Public License cover the whole combination. +// + +#include "tests/testUtils.hpp" + +#include "vmime/net/imap/IMAPTag.hpp" +#include "vmime/net/imap/IMAPParser.hpp" + + +VMIME_TEST_SUITE_BEGIN(IMAPParserTest) + + VMIME_TEST_LIST_BEGIN + VMIME_TEST(testExtraSpaceInCapaResponse) + VMIME_TEST_LIST_END + + + // For Apple iCloud IMAP server + void testExtraSpaceInCapaResponse() + { + vmime::ref <testSocket> socket = vmime::create <testSocket>(); + vmime::ref <vmime::net::timeoutHandler> toh = vmime::create <testTimeoutHandler>(); + + vmime::ref <vmime::net::imap::IMAPTag> tag = + vmime::create <vmime::net::imap::IMAPTag>(); + + socket->localSend( + "* CAPABILITY IMAP4rev1 AUTH=ATOKEN AUTH=PLAIN \r\n" // extra space at end + "a001 OK Capability completed.\r\n"); + + vmime::ref <vmime::net::imap::IMAPParser> parser = + vmime::create <vmime::net::imap::IMAPParser>(tag, socket.dynamicCast <vmime::net::socket>(), toh); + + parser->setStrict(false); + VASSERT_NO_THROW("non-strict mode", parser->readResponse(/* literalHandler */ NULL)); + + ++(*tag); + + socket->localSend( + "* CAPABILITY IMAP4rev1 AUTH=ATOKEN AUTH=PLAIN \r\n" // extra space at end + "a002 OK Capability completed.\r\n"); + + parser->setStrict(true); + VASSERT_THROW("strict mode", parser->readResponse(/* literalHandler */ NULL), vmime::exceptions::invalid_response); + } + +VMIME_TEST_SUITE_END diff --git a/vmime/net/imap/IMAPParser.hpp b/vmime/net/imap/IMAPParser.hpp index c8300122..fe05cd62 100644 --- a/vmime/net/imap/IMAPParser.hpp +++ b/vmime/net/imap/IMAPParser.hpp @@ -2111,7 +2111,13 @@ public: while (parser.check <SPACE>(line, &pos, true)) { - capability* cap = parser.get <capability>(line, &pos); + capability* cap; + + if (parser.isStrict() || m_capabilities.empty()) + cap = parser.get <capability>(line, &pos); + else + cap = parser.get <capability>(line, &pos, /* noThrow */ true); // allow SPACE at end of line (Apple iCloud IMAP server) + if (cap == NULL) break; m_capabilities.push_back(cap); @@ -4574,6 +4580,13 @@ public: if (!(m_message_data = parser.get <IMAPParser::message_data>(line, &pos, true))) m_capability_data = parser.get <IMAPParser::capability_data>(line, &pos); + if (!parser.isStrict()) + { + // Allow SPACEs at end of line + while (parser.check <SPACE>(line, &pos, /* noThrow */ true)) + ; + } + parser.check <CRLF>(line, &pos); *currentPos = pos; @@ -4666,6 +4679,13 @@ public: m_resp_cond_bye = parser.get <IMAPParser::resp_cond_bye>(line, &pos); + if (!parser.isStrict()) + { + // Allow SPACEs at end of line + while (parser.check <SPACE>(line, &pos, /* noThrow */ true)) + ; + } + parser.check <CRLF>(line, &pos); *currentPos = pos; @@ -4708,6 +4728,14 @@ public: parser.check <IMAPParser::xtag>(line, &pos); parser.check <SPACE>(line, &pos); m_resp_cond_state = parser.get <IMAPParser::resp_cond_state>(line, &pos); + + if (!parser.isStrict()) + { + // Allow SPACEs at end of line + while (parser.check <SPACE>(line, &pos, /* noThrow */ true)) + ; + } + parser.check <CRLF>(line, &pos); *currentPos = pos; |