summaryrefslogtreecommitdiffabout
path: root/libetpan/src/driver/interface/mailstorage_tools.c
Side-by-side diff
Diffstat (limited to 'libetpan/src/driver/interface/mailstorage_tools.c') (more/less context) (ignore whitespace changes)
-rw-r--r--libetpan/src/driver/interface/mailstorage_tools.c372
1 files changed, 372 insertions, 0 deletions
diff --git a/libetpan/src/driver/interface/mailstorage_tools.c b/libetpan/src/driver/interface/mailstorage_tools.c
new file mode 100644
index 0000000..9d39cab
--- a/dev/null
+++ b/libetpan/src/driver/interface/mailstorage_tools.c
@@ -0,0 +1,372 @@
+/*
+ * libEtPan! -- a mail stuff library
+ *
+ * Copyright (C) 2001, 2005 - DINH Viet Hoa
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of the libEtPan! project nor the names of its
+ * contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+/*
+ * $Id$
+ */
+
+#include "mailstorage_tools.h"
+
+#include "libetpan-config.h"
+
+#include <sys/types.h>
+#include <netdb.h>
+#include <netinet/in.h>
+#include <sys/socket.h>
+#include <unistd.h>
+#include <stdlib.h>
+#include <sys/wait.h>
+#include <sys/ioctl.h>
+#include <fcntl.h>
+#include <string.h>
+
+#include "mail.h"
+#include "mailmessage.h"
+#include "maildriver.h"
+
+/* tools */
+
+/* connection to TCP/IP server */
+
+static int tcp_connect(char * server, uint16_t port)
+{
+ struct hostent * remotehost;
+ struct sockaddr_in sa;
+ int s;
+ int r;
+
+ s = socket(PF_INET, SOCK_STREAM, 0);
+ if (s == -1)
+ goto err;
+
+ remotehost = gethostbyname(server);
+ if (remotehost == NULL)
+ goto close_socket;
+
+ sa.sin_family = AF_INET;
+ sa.sin_port = htons(port);
+ memcpy(&sa.sin_addr, remotehost->h_addr, remotehost->h_length);
+
+ r = connect(s, (struct sockaddr *) &sa, sizeof(struct sockaddr_in));
+ if (r == -1)
+ goto close_socket;
+
+ return s;
+
+ close_socket:
+ close(s);
+ err:
+ return -1;
+}
+
+
+/* connection through a shell command */
+
+static void do_exec_command(int fd, const char *command,
+ char *servername, uint16_t port)
+{
+ int i, maxopen;
+
+ if (fork() > 0) {
+ /* Fork again to become a child of init rather than
+ the etpan client. */
+ exit(0);
+ }
+
+ if (servername)
+ setenv("ETPANSERVER", servername, 1);
+ else
+ unsetenv("ETPANSERVER");
+
+ if (port) {
+ char porttext[20];
+
+ snprintf(porttext, sizeof(porttext), "%d", port);
+ setenv("ETPANPORT", porttext, 1);
+ }
+ else {
+ unsetenv("ETPANPORT");
+ }
+
+ /* Not a lot we can do if there's an error other than bail. */
+ if (dup2(fd, 0) == -1)
+ exit(1);
+ if (dup2(fd, 1) == -1)
+ exit(1);
+
+ /* Should we close stderr and reopen /dev/null? */
+
+ maxopen = sysconf(_SC_OPEN_MAX);
+ for (i=3; i < maxopen; i++)
+ close(i);
+
+#ifdef TIOCNOTTY
+ /* Detach from the controlling tty if we have one. Otherwise,
+ SSH might do something stupid like trying to use it instead
+ of running $SSH_ASKPASS. Doh. */
+ fd = open("/dev/tty", O_RDONLY);
+ if (fd != -1) {
+ ioctl(fd, TIOCNOTTY, NULL);
+ close(fd);
+ }
+#endif /* TIOCNOTTY */
+
+ execl("/bin/sh", "/bin/sh", "-c", command, NULL);
+
+ /* Eep. Shouldn't reach this */
+ exit(1);
+}
+
+static int subcommand_connect(char *command, char *servername, uint16_t port)
+{
+ int sockfds[2];
+ pid_t childpid;
+
+ if (socketpair(AF_UNIX, SOCK_STREAM, 0, sockfds))
+ return -1;
+
+ childpid = fork();
+ if (!childpid) {
+ do_exec_command(sockfds[1], command, servername, port);
+ }
+ else if (childpid == -1) {
+ close(sockfds[0]);
+ close(sockfds[1]);
+ return -1;
+ }
+
+ close(sockfds[1]);
+
+ /* Reap child, leaving grandchild process to run */
+ waitpid(childpid, NULL, 0);
+
+ return sockfds[0];
+}
+
+int mailstorage_generic_connect(mailsession_driver * driver,
+ char * servername,
+ uint16_t port,
+ char * command,
+ int connection_type,
+ int cache_function_id,
+ char * cache_directory,
+ int flags_function_id,
+ char * flags_directory,
+ mailsession ** result)
+{
+ int r;
+ int res;
+ mailstream * stream;
+ int fd;
+ mailsession * session;
+ int connect_result;
+
+ switch (connection_type) {
+ case CONNECTION_TYPE_PLAIN:
+ case CONNECTION_TYPE_TRY_STARTTLS:
+ case CONNECTION_TYPE_STARTTLS:
+ case CONNECTION_TYPE_TLS:
+ fd = tcp_connect(servername, port);
+ if (fd == -1) {
+ res = MAIL_ERROR_CONNECT;
+ goto err;
+ }
+ break;
+
+ case CONNECTION_TYPE_COMMAND:
+ case CONNECTION_TYPE_COMMAND_TRY_STARTTLS:
+ case CONNECTION_TYPE_COMMAND_STARTTLS:
+ case CONNECTION_TYPE_COMMAND_TLS:
+ fd = subcommand_connect(command, servername, port);
+ break;
+
+ default:
+ fd = -1;
+ break;
+ }
+
+ if (fd == -1) {
+ res = MAIL_ERROR_INVAL;
+ goto err;
+ }
+
+ switch (connection_type) {
+ case CONNECTION_TYPE_PLAIN:
+ case CONNECTION_TYPE_TRY_STARTTLS:
+ case CONNECTION_TYPE_STARTTLS:
+ case CONNECTION_TYPE_COMMAND:
+ case CONNECTION_TYPE_COMMAND_TRY_STARTTLS:
+ case CONNECTION_TYPE_COMMAND_STARTTLS:
+ stream = mailstream_socket_open(fd);
+ break;
+
+ case CONNECTION_TYPE_TLS:
+ case CONNECTION_TYPE_COMMAND_TLS:
+ stream = mailstream_ssl_open(fd);
+ break;
+
+ default:
+ stream = NULL;
+ break;
+ }
+
+ if (stream == NULL) {
+ res = MAIL_ERROR_STREAM;
+ close(fd);
+ goto err;
+ }
+
+ session = mailsession_new(driver);
+ if (session == NULL) {
+ res = MAIL_ERROR_MEMORY;
+ goto close_stream;
+ }
+
+ if (cache_directory != NULL) {
+ char cache_directory_server[PATH_MAX];
+
+ snprintf(cache_directory_server, PATH_MAX, "%s/%s",
+ cache_directory, servername);
+
+ r = mailsession_parameters(session,
+ cache_function_id,
+ cache_directory_server);
+ if (r != MAIL_NO_ERROR) {
+ res = r;
+ goto close_stream;
+ }
+ }
+
+ if (flags_directory != NULL) {
+ char flags_directory_server[PATH_MAX];
+
+ snprintf(flags_directory_server, PATH_MAX, "%s/%s",
+ flags_directory, servername);
+
+ r = mailsession_parameters(session,
+ flags_function_id,
+ flags_directory_server);
+ if (r != MAIL_NO_ERROR) {
+ res = r;
+ goto close_stream;
+ }
+ }
+
+ r = mailsession_connect_stream(session, stream);
+ switch (r) {
+ case MAIL_NO_ERROR_NON_AUTHENTICATED:
+ case MAIL_NO_ERROR_AUTHENTICATED:
+ case MAIL_NO_ERROR:
+ break;
+ default:
+ res = r;
+ goto free;
+ }
+
+ connect_result = r;
+
+ switch (connection_type) {
+ case CONNECTION_TYPE_TRY_STARTTLS:
+ case CONNECTION_TYPE_COMMAND_TRY_STARTTLS:
+ r = mailsession_starttls(session);
+ if ((r != MAIL_NO_ERROR) && (r != MAIL_ERROR_NO_TLS)) {
+ res = r;
+ goto free;
+ }
+ break;
+
+ case CONNECTION_TYPE_STARTTLS:
+ case CONNECTION_TYPE_COMMAND_STARTTLS:
+ r = mailsession_starttls(session);
+ if (r != MAIL_NO_ERROR) {
+ res = r;
+ goto free;
+ }
+ }
+
+ * result = session;
+
+ return connect_result;
+
+ close_stream:
+ mailstream_close(stream);
+ free:
+ mailsession_free(session);
+ err:
+ return res;
+}
+
+
+
+
+
+int mailstorage_generic_auth(mailsession * session,
+ int connect_result,
+ int auth_type,
+ char * login,
+ char * password)
+{
+ int must_auth;
+ int r;
+ int res;
+
+ r = connect_result;
+
+ must_auth = FALSE;
+ switch (r) {
+ case MAIL_NO_ERROR_NON_AUTHENTICATED:
+ must_auth = TRUE;
+ break;
+ case MAIL_NO_ERROR_AUTHENTICATED:
+ case MAIL_NO_ERROR:
+ break;
+ default:
+ res = r;
+ goto err;
+ }
+
+ if ((login == NULL) || (password == NULL))
+ must_auth = FALSE;
+
+ if (must_auth) {
+ r = mailsession_login(session, login, password);
+ if (r != MAIL_NO_ERROR) {
+ mailsession_logout(session);
+ res = r;
+ goto err;
+ }
+ }
+
+ return MAIL_NO_ERROR;
+
+ err:
+ return res;
+}