123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355 |
- /*
- * CDE - Common Desktop Environment
- *
- * Copyright (c) 1993-2012, The Open Group. All rights reserved.
- *
- * These libraries and programs are free software; you can
- * redistribute them and/or modify them under the terms of the GNU
- * Lesser General Public License as published by the Free Software
- * Foundation; either version 2 of the License, or (at your option)
- * any later version.
- *
- * These libraries and programs are distributed in the hope that
- * they will be useful, but WITHOUT ANY WARRANTY; without even the
- * implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
- * PURPOSE. See the GNU Lesser General Public License for more
- * details.
- *
- * You should have received a copy of the GNU Lesser General Public
- * License along with these libraries and programs; if not, write
- * to the Free Software Foundation, Inc., 51 Franklin Street, Fifth
- * Floor, Boston, MA 02110-1301 USA
- */
- /*
- *+SNOTICE
- *
- * $TOG: IMAPServer.C /main/7 1998/11/10 17:08:32 mgreess $
- *
- * RESTRICTED CONFIDENTIAL INFORMATION:
- *
- * The information in this document is subject to special
- * restrictions in a confidential disclosure agreement between
- * HP, IBM, Sun, USL, SCO and Univel. Do not distribute this
- * document outside HP, IBM, Sun, USL, SCO, or Univel without
- * Sun's specific written approval. This document and all copies
- * and derivative works thereof must be returned or destroyed at
- * Sun's request.
- *
- * Copyright 1993, 1995, 1995 Sun Microsystems, Inc. All rights reserved.
- *
- *+ENOTICE
- */
- /*
- * Common Desktop Environment
- *
- * (c) Copyright 1993, 1994, 1995 Hewlett-Packard Company
- * (c) Copyright 1993, 1994, 1995 International Business Machines Corp.
- * (c) Copyright 1993, 1994, 1995 Sun Microsystems, Inc.
- * (c) Copyright 1993, 1994, 1995 Novell, Inc.
- * (c) Copyright 1995 Digital Equipment Corp.
- * (c) Copyright 1995 Fujitsu Limited
- * (c) Copyright 1995 Hitachi, Ltd.
- *
- *
- * RESTRICTED RIGHTS LEGEND
- *
- *Use, duplication, or disclosure by the U.S. Government is subject to
- *restrictions as set forth in subparagraph (c)(1)(ii) of the Rights in
- *Technical Data and Computer Software clause in DFARS 252.227-7013. Rights
- *for non-DOD U.S. Government Departments and Agencies are as set forth in
- *FAR 52.227-19(c)(1,2).
- *Hewlett-Packard Company, 3000 Hanover Street, Palo Alto, CA 94304 U.S.A.
- *International Business Machines Corp., Route 100, Somers, NY 10589 U.S.A.
- *Sun Microsystems, Inc., 2550 Garcia Avenue, Mountain View, CA 94043 U.S.A.
- *Novell, Inc., 190 River Road, Summit, NJ 07901 U.S.A.
- *Digital Equipment Corp., 111 Powdermill Road, Maynard, MA 01754, U.S.A.
- *Fujitsu Limited, 1015, Kamikodanaka Nakahara-Ku, Kawasaki 211, Japan
- *Hitachi, Ltd., 6, Kanda Surugadai 4-Chome, Chiyoda-ku, Tokyo 101, Japan
- */
- #include <stdio.h>
- #include <string.h>
- #include <ctype.h>
- #include <stdlib.h>
- #include <DtMail/DtMailServer.hh>
- #include <DtMail/IO.hh>
- #define dtmasTAGGET() (_transtag)
- IMAPServer::IMAPServer(
- char *folder,
- DtMail::Session *session,
- DtMail::MailBox *mailbox,
- DtMailAppendCallback append_mailbox_cb,
- void *append_mailbox_cb_data)
- : DtMailServer(folder, session, mailbox,
- append_mailbox_cb, append_mailbox_cb_data)
- {
- _append_mailbox_cb = append_mailbox_cb;
- _append_mailbox_cb_data = append_mailbox_cb_data;
- _count = 0;
- _imap4 = 0;
- _recent = 0;
- _seen = 0;
- _unseen = 0;
- }
- IMAPServer::~IMAPServer()
- {
- }
- //
- // Set delete flag for given message.
- //
- DTMailError_t
- IMAPServer::ptrans_delete(int msg)
- {
- // Use SILENT if possible as a minor throughput optimization.
- if (_imap4)
- return do_transaction("STORE %d +FLAGS.SILENT (\\Deleted)", msg);
- else
- return do_transaction("STORE %d +FLAGS (\\Deleted)", msg);
- }
- //
- // Apply for connection authorization
- //
- DTMailError_t
- IMAPServer::ptrans_authorize(char*)
- {
- /* try to get authorized */
- DTMailError_t ok;
-
- ok = do_transaction("LOGIN %s \"%s\"", _username, _password);
- if (DTME_NoError != ok) return DTME_MailServerAccess_AuthorizationFailed;
- // probe to see if we're running IMAP4 and can use RFC822.PEEK
- _imap4 = ((do_transaction("CAPABILITY")) == 0);
- return DTME_NoError;
- }
- //
- // Get range of messages to be fetched
- //
- DTMailError_t
- IMAPServer::ptrans_fldstate_read(int *countp, int *newp)
- {
- DTMailError_t ok;
- /* find out how many messages are waiting */
- _recent = _unseen = 0;
- ok = do_transaction("SELECT %s", _folder);
- if (DTME_NoError != ok) return ok;
- *countp = _count;
- if (_unseen) // Optional response, but better if we see it.
- *newp = _unseen;
- else if (_recent)
- *newp = _recent; // Mandatory
- else
- *newp = -1; // Should never happen, RECENT is mandatory.
- return DTME_NoError;
- }
- //
- // Capture the sizes of all messages.
- //
- DTMailError_t
- IMAPServer::ptrans_msgsizes(int count, int *sizes)
- {
- char buf[DTMAS_POPBUFSIZE+1];
- DTMailError_t ok = DTME_NoError;
- ok = do_send("FETCH 1:%d RFC822.SIZE", count);
- if (DTME_NoError != ok) return ok;
- while (SockGets(buf, sizeof(buf), _sockfp))
- {
- int num, size;
- if (buf[strlen(buf)-1] == '\n')
- buf[strlen(buf)-1] = '\0';
- if (buf[strlen(buf)-1] == '\r')
- buf[strlen(buf)-1] = '\r';
- if (_protologging)
- _logger.logError(DTM_FALSE, "%s< %s", proto_name(), buf);
- if (strstr(buf, "OK"))
- break;
- else if (sscanf(buf, "* %d FETCH (RFC822.SIZE %d)", &num, &size) == 2)
- sizes[num - 1] = size;
- else
- sizes[num - 1] = -1;
- }
- return DTME_NoError;
- }
- //
- // Is the given message old?
- //
- int
- IMAPServer::ptrans_msgisold(int num)
- {
- DTMailError_t ok;
- ok = do_transaction("FETCH %d FLAGS", num);
- if (DTME_NoError != ok) return 0;
- return _seen;
- }
- //
- // request nth message
- //
- DTMailError_t
- IMAPServer::ptrans_retrieve_start(int msg, int *lenp)
- {
- char buf[DTMAS_POPBUFSIZE+1];
- DTMailError_t ok = DTME_NoError;
- int num;
- //
- // If we're using IMAP4, we can fetch the message without setting its
- // seen flag. This is good! It means that if the protocol exchange
- // craps out during the message, it will still be marked `unseen' on
- // the server.
- //
- // However...*don't* do this if we're using keep to suppress deletion!
- // In that case, marking the seen flag is the only way to prevent the
- // message from being re-fetched on subsequent runs.
- //
- if (_imap4 && _removeafterdelivery)
- ok = do_send("FETCH %d RFC822.PEEK", msg);
- else
- ok = do_send("FETCH %d RFC822", msg);
- if (DTME_NoError != ok) return ok;
- // looking for FETCH response
- do
- {
- if (! SockGets(buf, sizeof(buf), _sockfp))
- {
- _logger.logError(DTM_FALSE, "Socket Error fetching message");
- return DTME_MailServerAccess_SocketIOError;
- }
- if (_protologging)
- _logger.logError(DTM_FALSE, "%s< %s", proto_name(), buf);
- } while (sscanf(buf+2, "%d FETCH (RFC822 {%d}", &num, lenp) != 2);
- if (num != msg)
- {
- _logger.logError(DTM_FALSE, "Protocol Error fetching message");
- return DTME_MailServerAccess_Error;
- }
- else
- {
- return DTME_NoError;
- }
- }
- //
- // Parse command response
- //
- DTMailError_t
- IMAPServer::ptrans_parse_response(char *argbuf)
- {
- char buf[DTMAS_POPBUFSIZE+1];
- _seen = 0;
- do
- {
- if (! SockGets(buf, sizeof(buf), _sockfp))
- {
- _logger.logError(DTM_FALSE, "Socket Error reading response");
- return DTME_MailServerAccess_SocketIOError;
- }
- if (buf[strlen(buf)-1] == '\n')
- buf[strlen(buf)-1] = '\0';
- if (buf[strlen(buf)-1] == '\r')
- buf[strlen(buf)-1] = '\r';
- if (_protologging)
- _logger.logError(DTM_FALSE, "%s< %s", proto_name(), buf);
- /* interpret untagged status responses */
- if (strstr(buf, "EXISTS"))
- _count = atoi(buf+2);
- if (strstr(buf, "RECENT"))
- _recent = atoi(buf+2);
- if (strstr(buf, "UNSEEN"))
- _unseen = atoi(buf+2);
- if (strstr(buf, "FLAGS"))
- _seen = (strstr(buf, "Seen") != (char *)NULL);
- } while (strlen(dtmasTAGGET()) &&
- strncmp(buf, dtmasTAGGET(), strlen(dtmasTAGGET())));
- if (! strlen(dtmasTAGGET()))
- {
- strcpy(argbuf, buf);
- return DTME_NoError;
- }
- else
- {
- char *cp;
- /* skip the tag */
- for (cp = buf; !isspace(*cp); cp++)
- continue;
- while (isspace(*cp))
- cp++;
- if (strncmp(cp, "OK", 2) == 0)
- {
- strcpy(argbuf, cp);
- return DTME_NoError;
- }
- else if (strncmp(cp, "BAD", 2) == 0)
- {
- _logger.logError(DTM_FALSE, "Protocol Error reading response");
- return DTME_MailServerAccess_Error;
- }
- else
- {
- _logger.logError(DTM_FALSE, "Protocol Violation reading response");
- return DTME_MailServerAccess_ProtocolViolation;
- }
- }
- }
- //
- // Retrieve messages using IMAP Version 2bis or Version 4.
- //
- void
- IMAPServer::retrieve_messages(DtMailEnv &error)
- {
- DtMailServer::retrieve_messages(error);
- }
- //
- // Discard tail of FETCH response after reading message text.
- //
- DTMailError_t
- IMAPServer::ptrans_retrieve_end(int)
- {
- char buf [DTMAS_POPBUFSIZE+1];
- if (! SockGets(buf, sizeof(buf), _sockfp))
- {
- _logger.logError(DTM_FALSE, "Socket Error reading trail");
- return DTME_MailServerAccess_SocketIOError;
- }
- return DTME_NoError;
- }
|