aboutsummaryrefslogblamecommitdiffstats
path: root/mail/mail-guess-servers.c
blob: 21f1868bc2880458c50f0c36852cfd66c2740904 (plain) (tree)









































































































































































































































































































                                                                                             

                                                    
                                                  


















































                                                                                               


                                                         

































                                                                                                   





                                            
                               








                                                                          
                                                                                                     


















































                                                                    
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
 * This program is free software; you can redistribute it and/or
 * modify it 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) version 3.
 *
 * 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with the program; if not, see <http://www.gnu.org/licenses/>
 *
 *
 * Authors:
 *      Srinivasa Ragavan <srini@linux.intel.com>
 *
 * Copyright (C) 2009 Intel Corporation (www.intel.com)
 *
 */

/* Template of the code is taken from libsoup tests/ */

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include <ctype.h>
#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <unistd.h>

#include <libsoup/soup.h>

#include <libxml/parser.h>
#include <libxml/tree.h>
#include <libxml/xmlmemory.h>

#include <libedataserver/e-proxy.h>

#include <shell/e-shell.h>
#include <e-util/e-util-private.h>

#include "mail-guess-servers.h"

static gchar *
xml_to_gchar (xmlChar *xml,
              EmailProvider *provider)
{
    gchar *gxml = NULL;
    gchar *tmp;
    gchar *repl = NULL;
    const gchar *sec_part;

    tmp = xml ? strstr((gchar *) xml, "\%EMAIL") : NULL;

    if (!tmp) {
        gxml = xml ? g_strdup ((gchar *) xml) : NULL;
    } else {
    decodepart:
        *tmp = 0;
        tmp+=6;
        if (*tmp == 'A')
            repl = provider->email;
        else if (*tmp == 'L')
            repl = provider->username;
        else if (*tmp == 'D')
            repl = provider->domain;
        sec_part = strstr(tmp, "\%");
        sec_part++;
        if (!*sec_part)
            sec_part = "";

        gxml = g_strdup_printf("%s%s%s", gxml ? gxml : (gchar *)xml, repl, sec_part);
        tmp = strstr (gxml, "\%EMAIL");
        if (tmp) {
            goto decodepart;
        }
    }

    xmlFree (xml);

    return gxml;
}

static SoupMessage *
get_url (SoupSession *session,
         const gchar *url)
{
    SoupMessage *msg;
    const gchar *header;

    msg = soup_message_new (SOUP_METHOD_GET, url);
    soup_message_set_flags (msg, SOUP_MESSAGE_NO_REDIRECT);

    soup_session_send_message (session, msg);

    if (SOUP_STATUS_IS_REDIRECTION (msg->status_code)) {
        header = soup_message_headers_get_one (msg->response_headers,
                               "Location");
        if (header) {
            return get_url (session, header);
        }
    } else if (SOUP_STATUS_IS_SUCCESSFUL (msg->status_code)) {
        return msg;
    }

    return NULL;
}

static void
handle_incoming (xmlNodePtr head,
                 EmailProvider *provider)
{
    xmlNodePtr node = head->children;

    provider->recv_type = xml_to_gchar (
        xmlGetProp (head, (xmlChar *) "type"), provider);

    while (node != NULL) {
        if (strcmp ((gchar *)node->name, "hostname") == 0) {
            provider->recv_hostname = xml_to_gchar (
                xmlNodeGetContent (node), provider);
        } else if (strcmp ((gchar *)node->name, "port") == 0) {
            provider->recv_port = xml_to_gchar (
                xmlNodeGetContent (node), provider);
        } else if (strcmp ((gchar *)node->name, "socketType") == 0) {
            provider->recv_socket_type = xml_to_gchar (
                xmlNodeGetContent (node), provider);
        } else if (strcmp ((gchar *)node->name, "username") == 0) {
            provider->recv_username = xml_to_gchar (
                xmlNodeGetContent (node), provider);
        } else if (strcmp ((gchar *)node->name, "authentication") == 0) {
            provider->recv_auth = xml_to_gchar (
                xmlNodeGetContent (node), provider);
        }

        node = node->next;
    }
}

static void
handle_outgoing (xmlNodePtr head,
                 EmailProvider *provider)
{
    xmlNodePtr node = head->children;

    provider->send_type = xml_to_gchar (
        xmlGetProp (head, (xmlChar *) "type"), provider);

    while (node != NULL) {
        if (strcmp ((gchar *)node->name, "hostname") == 0) {
            provider->send_hostname = xml_to_gchar (
                xmlNodeGetContent (node), provider);
        } else if (strcmp ((gchar *)node->name, "port") == 0) {
            provider->send_port = xml_to_gchar (
                xmlNodeGetContent (node), provider);
        } else if (strcmp ((gchar *)node->name, "socketType") == 0) {
            provider->send_socket_type = xml_to_gchar (
                xmlNodeGetContent (node), provider);
        } else if (strcmp ((gchar *)node->name, "username") == 0) {
            provider->send_username = xml_to_gchar (
                xmlNodeGetContent (node), provider);
        } else if (strcmp ((gchar *)node->name, "authentication") == 0) {
            provider->send_auth = xml_to_gchar (
                xmlNodeGetContent (node), provider);
        }

        node = node->next;
    }
}

static gboolean
parse_message (const gchar *msg,
               gint length,
               EmailProvider *provider)
{
    xmlDocPtr doc;
    xmlNodePtr node;

    doc = xmlReadMemory (msg, length, "file.xml", NULL, 0);

    node = doc->children;
    while (node) {
        if (strcmp ((gchar *)node->name, "clientConfig") == 0) {
            break;
        }
        node = node->next;
    }

    if (!node) {
        g_warning ("Incorrect data: ClientConfig not found ... Quitting\n");
        return FALSE;
    }

    node = node->children;
    while (node) {
        if (strcmp ((gchar *)node->name, "emailProvider") == 0) {
            break;
        }
        node = node->next;
    }

    if (!node) {
        g_warning ("Incorrect data: ClientConfig not found ... Quitting\n");
        return FALSE;
    }

    node = node->children;
    while (node) {
        if (strcmp ((gchar *)node->name, "incomingServer") == 0) {
            /* Handle Incoming */
            handle_incoming (node, provider);
        } else if (strcmp ((gchar *)node->name, "outgoingServer") == 0) {
            /* Handle Outgoing */
            handle_outgoing (node, provider);
        }

        node = node->next;
    }

    xmlFreeDoc (doc);

    return TRUE;
}

static gboolean
parse_soup_message (SoupMessage *msg,
                    EmailProvider *provider)
{
    return parse_message (
        msg->response_body->data,
        msg->response_body->length, provider);
}

static gboolean
is_online (void)
{
    EShell *shell;

    /* FIXME Pass this in. */
    shell = e_shell_get_default ();

    return e_shell_get_online (shell);
}

static gboolean
guess_when_online (EmailProvider *provider)
{
    const gchar *cafile = NULL;
    gchar *url;
    EProxy *proxy;
    SoupURI *parsed;
    SoupMessage *msg;
    SoupSession *session;

    proxy = e_proxy_new ();
    e_proxy_setup_proxy (proxy);

    url = g_strdup_printf (
        "%s/%s", "http://api.gnome.org/evolution/autoconfig",
        provider->domain);
    parsed = soup_uri_new (url);
    soup_uri_free (parsed);

    session = soup_session_sync_new_with_options (
        SOUP_SESSION_SSL_CA_FILE, cafile,
        SOUP_SESSION_USER_AGENT, "get ",
        NULL);

    if (e_proxy_require_proxy_for_uri (proxy, url)) {
        SoupURI *proxy_uri = e_proxy_peek_uri_for (proxy, url);
/*      fprintf (stderr, "URL '%s' requires a proxy: '%s'\n",
             url, soup_uri_to_string (proxy_uri, FALSE)); */
        g_object_set (session, SOUP_SESSION_PROXY_URI, proxy_uri, NULL);
    }

    msg = get_url (session, url);
    if (!msg)
        return FALSE;

    parse_soup_message (msg, provider);

    g_object_unref (proxy);
    g_object_unref (msg);
    g_object_unref (session);
    g_free (url);

    return TRUE;

}

static gboolean
offline_file_includes_domain (const gchar *filename,
                              const gchar *domain)
{
    gboolean res = FALSE;
    gchar *content = NULL;
    gsize length;
    xmlDocPtr doc;
    xmlNodePtr node;

    g_return_val_if_fail (filename != NULL, FALSE);
    g_return_val_if_fail (domain != NULL, FALSE);

    if (!g_file_get_contents (filename, &content, &length, NULL))
        return FALSE;

    doc = xmlReadMemory (content, length, "file.xml", NULL, 0);

    node = doc->children;
    while (node) {
        if (strcmp ((gchar *) node->name, "clientConfig") == 0) {
            node = node->children;
            while (node) {
                if (strcmp ((gchar *) node->name, "emailProvider") == 0) {
                    break;
                }
                node = node->next;
            }
            break;
        }
        node = node->next;
    }

    if (node) {
        xmlChar *xmlStr;

        for (node = node->children; node && !res;  node = node->next) {
            if (!g_str_equal (node->name, "domain"))
                continue;

            xmlStr = xmlNodeGetContent (node);
            if (xmlStr && g_ascii_strcasecmp ((const gchar *) xmlStr, domain) == 0)
                res = TRUE;

            xmlFree (xmlStr);
        }
    }

    xmlFreeDoc (doc);
    g_free (content);

    return res;
}

static gchar *
get_filename_for_offline_autoconfig (const gchar *domain)
{
    gchar *path;
    GDir *dir;
    const gchar *filename;

    path = g_build_filename (EVOLUTION_PRIVDATADIR, "mail-autoconfig", domain, NULL);
    if (g_file_test (path, G_FILE_TEST_EXISTS))
        return path;

    g_free (path);

    path = g_build_filename (EVOLUTION_PRIVDATADIR, "mail-autoconfig", NULL);
    dir = g_dir_open (path, 0, NULL);
    g_free (path);

    if (!dir)
        return NULL;

    path = NULL;
    while (filename = g_dir_read_name (dir), filename && !path) {
        if (g_str_equal (filename, ".") ||
            g_str_equal (filename, ".."))
            continue;

        path = g_build_filename (EVOLUTION_PRIVDATADIR, "mail-autoconfig", filename, NULL);
        if (offline_file_includes_domain (path, domain))
            break;

        g_free (path);
        path = NULL;
    }

    g_dir_close (dir);

    return path;
}

static gboolean
guess_when_offline (EmailProvider *provider)
{
    gchar *filename;
    gchar *contents = NULL;
    gsize length;
    gboolean success;

    if (!provider->domain || provider->domain[0] == 0)
        return FALSE;

    success = FALSE;

    filename = get_filename_for_offline_autoconfig (provider->domain);
    if (!filename || !g_file_get_contents (filename, &contents, &length, NULL)) /* NULL-GError */
        goto out;

    success = parse_message (contents, (gint) length, provider);

out:
    g_free (filename);
    g_free (contents);

    return success;
}

gboolean
mail_guess_servers (EmailProvider *provider)
{
    if (is_online () && guess_when_online (provider))
        return TRUE;
    else
        return guess_when_offline (provider);
}

#ifdef TEST
gint
main (gint argc,
      gchar **argv)
{
    EmailProvider *provider;

    g_type_init ();

    provider = g_new0 (EmailProvider, 1);

    provider->email = "sragavan@iijmio-mail.jp";
    provider->domain = "iijmio-mail.jp";
    provider->username = "sragavan";

    mail_guess_servers (provider);

    printf (
        "Recv: %s\n%s(%s), %s by %s \n "
        "Send: %s\n%s(%s), %s by %s\n via %s to %s\n",
        provider->recv_type, provider->recv_hostname,
        provider->recv_port, provider->recv_username,
        provider->recv_auth,
        provider->send_type, provider->send_hostname,
        provider->send_port, provider->send_username,
        provider->send_auth,
        provider->recv_socket_type,
        provider->send_socket_type);
    return 0;
}
#endif