aboutsummaryrefslogblamecommitdiffstats
path: root/camel/camel-disco-store.c
blob: 95a35737294302ef4cb70452f3f9bd27d9fe3f02 (plain) (tree)
1
2
3
4
5
6
7
8







                                                                           


                                                                   
  



                                                                    
  



                                                               







                              
                              
                            
                          




                                                                           


                                                                          
                                                                          
                                                         





                                                                                             



                                                                










                                                                                                                          



                                                                     
                                     
                                                         

                                                           
                                                                   




                                                                   











                                                                                
                             





                                      














                                                                                                  



                                                             










                                                                                       
 
                         
                                      















                                                                       




                                                                          

                                

 










                                                                           






                                                                            
                                         







                                                                                  
 









                                                                                         
        


                                                                                     
                

                                                                                      
                

                                                                                                
         
        
                                







                                                                 










                                                                          


                                                                                         
                






                                                                         
                
                                                                                          
                

                                                                                            
         
        
                                



                    





                                                             







                                                                                      




































































                                                                                











                                                                             
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* camel-disco-store.c: abstract class for a disconnectable remote store */

/*
 *  Authors: Dan Winship <danw@ximian.com>
 *
 *  Copyright 2001 Ximian, Inc.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of version 2 of the GNU General Public
 * License as published by the Free Software Foundation.
 *
 * 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 Place - Suite 330,
 * Boston, MA 02111-1307, USA.
 *
 */

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

#include "camel-disco-store.h"
#include "camel-disco-diary.h"
#include "camel-exception.h"
#include "camel-session.h"

#define CDS_CLASS(o) (CAMEL_DISCO_STORE_CLASS (CAMEL_OBJECT_GET_CLASS (o)))

static CamelRemoteStoreClass *remote_store_class = NULL;

static void disco_construct (CamelService *service, CamelSession *session,
                 CamelProvider *provider, CamelURL *url,
                 CamelException *ex);
static gboolean disco_connect (CamelService *service, CamelException *ex);
static void disco_cancel_connect (CamelService *service);
static gboolean disco_disconnect (CamelService *service, gboolean clean, CamelException *ex);
static CamelFolder *disco_get_folder (CamelStore *store, const char *name,
                      guint32 flags, CamelException *ex);
static CamelFolderInfo *disco_get_folder_info (CamelStore *store,
                           const char *top, guint32 flags,
                           CamelException *ex);
static void set_status (CamelDiscoStore *disco_store,
            CamelDiscoStoreStatus status,
            CamelException *ex);
static gboolean can_work_offline (CamelDiscoStore *disco_store);

static void
camel_disco_store_class_init (CamelDiscoStoreClass *camel_disco_store_class)
{
    CamelServiceClass *camel_service_class =
        CAMEL_SERVICE_CLASS (camel_disco_store_class);
    CamelStoreClass *camel_store_class =
        CAMEL_STORE_CLASS (camel_disco_store_class);

    remote_store_class = CAMEL_REMOTE_STORE_CLASS (camel_type_get_global_classfuncs (camel_remote_store_get_type ()));

    /* virtual method definition */
    camel_disco_store_class->set_status = set_status;
    camel_disco_store_class->can_work_offline = can_work_offline;

    /* virtual method overload */
    camel_service_class->construct = disco_construct;
    camel_service_class->connect = disco_connect;
    camel_service_class->disconnect = disco_disconnect;
    camel_service_class->cancel_connect = disco_cancel_connect;

    camel_store_class->get_folder = disco_get_folder;
    camel_store_class->get_folder_info = disco_get_folder_info;
}

CamelType
camel_disco_store_get_type (void)
{
    static CamelType camel_disco_store_type = CAMEL_INVALID_TYPE;

    if (camel_disco_store_type == CAMEL_INVALID_TYPE) {
        camel_disco_store_type = camel_type_register (
            CAMEL_REMOTE_STORE_TYPE, "CamelDiscoStore",
            sizeof (CamelDiscoStore),
            sizeof (CamelDiscoStoreClass),
            (CamelObjectClassInitFunc) camel_disco_store_class_init,
            NULL,
            NULL,
            NULL);
    }

    return camel_disco_store_type;
}

static void
disco_construct (CamelService *service, CamelSession *session,
         CamelProvider *provider, CamelURL *url,
         CamelException *ex)
{
    CamelDiscoStore *disco = CAMEL_DISCO_STORE (service);

    CAMEL_SERVICE_CLASS (remote_store_class)->construct (service, session, provider, url, ex);
    if (camel_exception_is_set (ex))
        return;

    disco->status = camel_session_is_online (session) ?
        CAMEL_DISCO_STORE_ONLINE : CAMEL_DISCO_STORE_OFFLINE;
}

static gboolean
disco_connect (CamelService *service, CamelException *ex)
{
    CamelDiscoStore *store = CAMEL_DISCO_STORE (service);
    CamelDiscoStoreStatus status;

    status = camel_disco_store_status (store);
    if (status != CAMEL_DISCO_STORE_OFFLINE) {
        if (!CAMEL_SERVICE_CLASS (remote_store_class)->connect (service, ex)) {
            status = camel_disco_store_status (store);
            if (status != CAMEL_DISCO_STORE_OFFLINE)
                return FALSE;
            camel_exception_clear (ex);
        }
    }

    switch (status) {
    case CAMEL_DISCO_STORE_ONLINE:
    case CAMEL_DISCO_STORE_RESYNCING:
        if (!CDS_CLASS (service)->connect_online (service, ex))
            return FALSE;
        if (camel_disco_diary_empty (store->diary))
            return TRUE;

        /* Need to resync */
        store->status = CAMEL_DISCO_STORE_RESYNCING;
        camel_disco_diary_replay (store->diary, ex);
        store->status = CAMEL_DISCO_STORE_ONLINE;
        if (camel_exception_is_set (ex))
            return FALSE;

        if (!camel_service_disconnect (service, TRUE, ex))
            return FALSE;
        return camel_service_connect (service, ex);

    case CAMEL_DISCO_STORE_OFFLINE:
        return CDS_CLASS (service)->connect_offline (service, ex);
    }

    g_assert_not_reached ();
    return FALSE;
}

static void
disco_cancel_connect (CamelService *service)
{
    CamelDiscoStore *store = CAMEL_DISCO_STORE (service);

    /* Fall back */
    store->status = CAMEL_DISCO_STORE_OFFLINE;

    CAMEL_SERVICE_CLASS (remote_store_class)->cancel_connect (service);
}

static gboolean
disco_disconnect (CamelService *service, gboolean clean, CamelException *ex)
{
    CamelDiscoStore *store = CAMEL_DISCO_STORE (service);

    switch (camel_disco_store_status (store)) {
    case CAMEL_DISCO_STORE_ONLINE:
    case CAMEL_DISCO_STORE_RESYNCING:
        if (!CDS_CLASS (service)->disconnect_online (service, clean, ex))
            return FALSE;
        break;

    case CAMEL_DISCO_STORE_OFFLINE:
        if (!CDS_CLASS (service)->disconnect_offline (service, clean, ex))
            return FALSE;
        break;

    }

    return CAMEL_SERVICE_CLASS (remote_store_class)->disconnect (service, clean, ex);
}

static CamelFolder *
disco_get_folder (CamelStore *store, const char *name,
          guint32 flags, CamelException *ex)
{
    CamelDiscoStore *disco_store = CAMEL_DISCO_STORE (store);
    
    switch (camel_disco_store_status (disco_store)) {
    case CAMEL_DISCO_STORE_ONLINE:
        return CDS_CLASS (store)->get_folder_online (store, name, flags, ex);
        
    case CAMEL_DISCO_STORE_OFFLINE:
        return CDS_CLASS (store)->get_folder_offline (store, name, flags, ex);
        
    case CAMEL_DISCO_STORE_RESYNCING:
        return CDS_CLASS (store)->get_folder_resyncing (store, name, flags, ex);    
    }
    
    g_assert_not_reached ();
    return NULL;
}

static CamelFolderInfo *
disco_get_folder_info (CamelStore *store, const char *top,
               guint32 flags, CamelException *ex)
{
    CamelDiscoStore *disco_store = CAMEL_DISCO_STORE (store);

    /* Do this first so if we get forced offline, we'll switch to
     * the correct branch below. (FIXME: This only works because
     * we know that get_folder_info is the first call that the
     * mailer makes on a store.)
     */
    if (CAMEL_SERVICE (store)->status == CAMEL_SERVICE_DISCONNECTED) {
        if (!camel_service_connect (CAMEL_SERVICE (store), ex))
            return NULL;
    }

    switch (camel_disco_store_status (disco_store)) {
    case CAMEL_DISCO_STORE_ONLINE:
        return CDS_CLASS (store)->get_folder_info_online (store, top, flags, ex);
        
    case CAMEL_DISCO_STORE_OFFLINE:
        /* Can't edit subscriptions while offline */
        if ((store->flags & CAMEL_STORE_SUBSCRIPTIONS) &&
            !(flags & CAMEL_STORE_FOLDER_INFO_SUBSCRIBED)) {
            camel_disco_store_check_online (disco_store, ex);
            return NULL;
        }
        
        return CDS_CLASS (store)->get_folder_info_offline (store, top, flags, ex);
        
    case CAMEL_DISCO_STORE_RESYNCING:
        return CDS_CLASS (store)->get_folder_info_resyncing (store, top, flags, ex);
    }
    
    g_assert_not_reached ();
    return NULL;
}


/**
 * camel_disco_store_status:
 * @store: a disconnectable store
 *
 * Return value: the current online/offline status of @store.
 **/
CamelDiscoStoreStatus
camel_disco_store_status (CamelDiscoStore *store)
{
    g_return_val_if_fail (CAMEL_IS_DISCO_STORE (store), CAMEL_DISCO_STORE_ONLINE);

    return store->status;
}


static void
set_status (CamelDiscoStore *disco_store, CamelDiscoStoreStatus status,
        CamelException *ex)
{
    if (disco_store->status == status)
        return;

    camel_store_sync (CAMEL_STORE (disco_store), ex);
    if (camel_exception_is_set (ex))
        return;
    if (!camel_service_disconnect (CAMEL_SERVICE (disco_store), TRUE, ex))
        return;

    disco_store->status = status;
    camel_service_connect (CAMEL_SERVICE (disco_store), ex);
}

/**
 * camel_disco_store_set_status:
 * @store: a disconnectable store
 * @status: the new status
 * @ex: a CamelException
 *
 * Sets @store to @status. If an error occurrs and the status cannot
 * be set to @status, @ex will be set.
 **/
void
camel_disco_store_set_status (CamelDiscoStore *store,
                  CamelDiscoStoreStatus status,
                  CamelException *ex)
{
    CDS_CLASS (store)->set_status (store, status, ex);
}


static gboolean
can_work_offline (CamelDiscoStore *disco_store)
{
    g_warning ("CamelDiscoStore::can_work_offline not implemented for `%s'",
           camel_type_to_name (CAMEL_OBJECT_GET_TYPE (disco_store)));
    return FALSE;
}

/**
 * camel_disco_store_can_work_offline:
 * @store: a disconnectable store
 *
 * Return value: whether or not @store can be used offline. (Will be
 * %FALSE if the store is not caching data to local disk, for example.)
 **/
gboolean
camel_disco_store_can_work_offline (CamelDiscoStore *store)
{
    return CDS_CLASS (store)->can_work_offline (store);
}


/**
 * camel_disco_store_check_online:
 * @store: a disconnectable store
 * @ex: a CamelException
 *
 * This checks that @store is online, and sets @ex if it is not. This
 * can be used as a simple way to set a generic error message in @ex
 * for operations that won't work offline.
 *
 * Return value: whether or not @store is online.
 **/
gboolean
camel_disco_store_check_online (CamelDiscoStore *store, CamelException *ex)
{
    if (camel_disco_store_status (store) != CAMEL_DISCO_STORE_ONLINE) {
        camel_exception_set (ex, CAMEL_EXCEPTION_SERVICE_UNAVAILABLE,
                     _("You must be working online to "
                       "complete this operation"));
        return FALSE;
    }

    return TRUE;
}