aboutsummaryrefslogblamecommitdiffstats
path: root/camel/camel-sasl-popb4smtp.c
blob: 7acd8e385c4ba9152f887c3a53c6ae858ae9929f (plain) (tree)


























































































































































                                                                                                       
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
 *  Authors: Michael Zucchi <notzed@ximian.com>
 *
 *  Copyright 2001 Ximian, Inc. (www.ximian.com)
 *
 *  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 2 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., 59 Temple Street #330, Boston, MA 02111-1307, USA.
 *
 */

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

#include <string.h>
#include <time.h>
#include "camel-sasl-popb4smtp.h"
#include "camel-service.h"
#include "camel-session.h"

CamelServiceAuthType camel_sasl_popb4smtp_authtype = {
    N_("POP before SMTP"),

    N_("This option will authorise a POP connection before attempting SMTP"),

    "POPB4SMTP",
    FALSE,
};

/* last time the pop was accessed (through the auth method anyway), *time_t */
static GHashTable *poplast;

/* use 1 hour as our pop timeout */
#define POPB4SMTP_TIMEOUT (60*60)

#ifdef ENABLE_THREADS
#include <pthread.h>
static pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER;
#define POPB4SMTP_LOCK(l) pthread_mutex_lock(&l)
#define POPB4SMTP_UNLOCK(l) pthread_mutex_unlock(&l)
#else
#define POPB4SMTP_LOCK(l)
#define POPB4SMTP_UNLOCK(l)
#endif

static CamelSaslClass *parent_class = NULL;

/* Returns the class for a CamelSaslPOPB4SMTP */
#define CSP_CLASS(so) CAMEL_SASL_POPB4SMTP_CLASS (CAMEL_OBJECT_GET_CLASS (so))

static GByteArray *popb4smtp_challenge (CamelSasl *sasl, GByteArray *token, CamelException *ex);

static void
camel_sasl_popb4smtp_class_init (CamelSaslPOPB4SMTPClass *camel_sasl_popb4smtp_class)
{
    CamelSaslClass *camel_sasl_class = CAMEL_SASL_CLASS (camel_sasl_popb4smtp_class);
    
    parent_class = CAMEL_SASL_CLASS (camel_type_get_global_classfuncs (camel_sasl_get_type ()));
    
    /* virtual method overload */
    camel_sasl_class->challenge = popb4smtp_challenge;

    poplast = g_hash_table_new(g_str_hash, g_str_equal);
}

CamelType
camel_sasl_popb4smtp_get_type (void)
{
    static CamelType type = CAMEL_INVALID_TYPE;
    
    if (type == CAMEL_INVALID_TYPE) {
        type = camel_type_register (camel_sasl_get_type (),
                        "CamelSaslPOPB4SMTP",
                        sizeof (CamelSaslPOPB4SMTP),
                        sizeof (CamelSaslPOPB4SMTPClass),
                        (CamelObjectClassInitFunc) camel_sasl_popb4smtp_class_init,
                        NULL,
                        NULL,
                        NULL);
    }
    
    return type;
}

static GByteArray *
popb4smtp_challenge (CamelSasl *sasl, GByteArray *token, CamelException *ex)
{
    char *popuri;
    CamelSession *session = sasl->service->session;
    CamelStore *store;
    time_t now, *timep;

    sasl->authenticated = FALSE;

    popuri = camel_session_get_password(session, _("POP Source URI"), FALSE,
                        sasl->service, "popb4smtp_uri", ex);

    if (popuri == NULL) {
        camel_exception_setv(ex, 1, _("POP Before SMTP auth using an unknown transport"));
        return NULL;
    }

    if (strncasecmp(popuri, "pop:", 4) != 0) {
        camel_exception_setv(ex, 1, _("POP Before SMTP auth using a non-pop source"));
        return NULL;
    }

    /* check if we've done it before recently in this session */
    now = time(0);

    /* need to lock around the whole thing until finished with timep */

    POPB4SMTP_LOCK(lock);
    timep = g_hash_table_lookup(poplast, popuri);
    if (timep) {
        if ((*timep + POPB4SMTP_TIMEOUT) > now) {
            sasl->authenticated = TRUE;
            POPB4SMTP_UNLOCK(lock);
            g_free(popuri);
            return NULL;
        }
    } else {
        timep = g_malloc0(sizeof(*timep));
        g_hash_table_insert(poplast, g_strdup(popuri), timep);
    }

    /* connect to pop session */
    store = camel_session_get_store(session, popuri, ex);
    if (store) {
        sasl->authenticated = TRUE;
        camel_object_unref((CamelObject *)store);
        *timep = now;
    } else {
        sasl->authenticated = FALSE;
        *timep = 0;
    }

    POPB4SMTP_UNLOCK(lock);

    g_free(popuri);
    
    return NULL;
}