aboutsummaryrefslogblamecommitdiffstats
path: root/addressbook/conduit/address-conduit.c
blob: c7f222730006695ef0c5e8ac19982fcc59bb5430 (plain) (tree)
1
2
3
4
5
6
7
8
                                                                           


                                              
                                  

                                              
                                        














                                                                            

                   
 


                             



                      
                         
                               


                                
                         
 

                             
                                   


                          
                            
 
                                                 

                                                         
 
                               


                   
                                   
 

                          
 

                                                              
     
                 
       
 

                                                               
 


                           
                         
 







                       
                    


                                  




































                                               
                    



                                     
 




                                         
                          






                                                                    

                            



                  












                                                               






                                                    
 

                             

                    
 
                      


















                                                                            
 
                     
 
                    
 
 
           
                                                  
 




















                                                                                                      
                                  
                                            





                                                              
 

                                                
 
                      

 
                                
           
                                                                                
 
                                               
 

                                 
        
                        

 





                                                              

           
                                                                                  
 
                                                                  
 


                                              
 
                                                  
 

                                                           


                                                                        
                                                         

                                         

                                                                        
 


                                                             
                                                         
                                                             


         


                                                                
                                                                  
 

                                                                                           

                                               


                                                             

          
                                                    
 

                          

                                            
                                    
 
                                                        
                                                                                 
                                                  


                                                               
 

                                                                    

                    

                     





                                       
                      









                                                                                                                            















                                                                                        













































































































































                                                              





                                                                      
                             

 
           








                                                     
                                                                                    
 

                              

                                      


















                                                             
 
 








                                                     
                       


                                                        
                           
                                   
        

                                        
                                               
 
                               
                                           


                                           
 
                                             
                          
                                                                
 
                        
 
 
           
                                                                                          
 

                                             
                                


                                                                        

              
                                         
                                         
 
                             
                                            

                                           
                                                                              
 
                                                
 
                                                 
 


                                                                      
                                   
                                    
                                    

                            


                                                                  
                                                                      













                                                                                                   
                 

         
                          



                                                                                                
         
 
                                                                                                  
                       




                                                                                             
         

                           






                                                                                                  
                                             
                































































                                                                                  
                                
         
 








                                                                                            
                                                                       











































                                                                                                          

                                                                       
                                                                                          
                         


                 


                                                                            
                                               
 
 

                                               
                                       

                                                 
                            

                 
                                 
 







                                                          
 


                                                             
                                        
                                           

                                                             

 
              


                                                   
 
                               

                            

                                       
                  
                             

                                                                        
              



                                                                  
 
                            
                                       
            
                                                   
                                           
 
                              
              



                                                                            
                              
 


                                                                                                 
 


                                                                  
 

                                                                

                                                                                
                     
 
                     

















                                                                                                 
                           


                                                                           

                                                                                             
                                                             
                
























                                                                                           
                
                                  
         
 




                                                                 
                                         

                                             

                               
                     

 
           
                                                                        
 

                                                                              
 

                                                                
                                                                      
 

                                                              
                                              
                
                                              
         

 





                                                                                

                                               
                




                                                   

                                       
                                                        
                                                                    
                                                                                        
                                                                                                           








                                                                                  




                                                 
                
                                                                
                                             
                                          

                                                        
                                                                    
                                                                                        
                                                                                                           
         





                                                                              
                              




                                                                              
                                                                            
                        
        
                                           

                                      


                                                            
        
                      
                                                                                                   

                                                          




                                                                   
                                                                     

                                                  


           
                                                                              
 

                                         

                                                










                                                                         
                             
           
                                     
                                
                                    
 


                                              
                           
                       
                        


                                                             
 


                                                                            
 
                                
                           
        


                                                                                 


                          
                                                 
                                   
                                                

                          



                                                                                                 
        

                                                      

                           


                                                                                        


                                                                                                 
 








                                                                                                  

                          

                                                      
 
                                               


                                                                              
 


                 
           


                                      
 
                                    

                                                                 
 
                                   
                                                
                          






                                                                                                 
                    

                                                                            
        


                 
           



                                                
 

                                                  
                                                                    
 
                 

 
           





                                                      

                                                                               



                 


                                            
 

                                       
 
                                                 
 

                                           
 




                                                                                 
 
                                                              
                                                                            
                                                              
 









                                                          
 
                                                              
                                                                               
                                                              

                                                
 

                                                                  
 


                                 
 

                 
 
           


                                                     
 
                               
                         
 
                                                
 
                             
                                                      
                
                                         
                
                          
                


                                                               
                        
                                                                                                  
                         
                                                              
                                                                          
                                                              

                                          

                                      


                        


                                                                                  

                                                              
                                                                          
                                                              

                                                         
 

                                                          
 


                                 
 
                 

 
           



                                           
 
                                       
                                     
                       
 

                                                         
 

                                               
 
                                                                 
 

                                                                           
                           
 




                                      
                      

 
           


                                              
 
                     
                                    


                                                  
 
                                                                          
 
                                                              
        







                                                                   

         
                                       
                                                                     
 
                      

 
           



                                                  
 
                         
                                  


                                    


                                                  
 

                                                         
 


                                                               


                                                                          
 



                                                                                          

                                             











                                                                                                      
                                                                                                           




                                            


                                                                          
 
                      

 



                                                 
 




                                                        
 
                                                                
 
                                                                
                                                                                            
        

                                             
                                                                                                    
                                                                         
        
                      

 
           



                                                  
 
                       
        
                                                 
 
                                                             

                                                                                   
        
                      


           



                                         
 
                        





                                                     
 
                      
                                                                   
        

                         
 
                            
        

                                                  

                 


           


                                              
 
                                      
 
                                                 
 
                           
 



                 

                                           
                                  
                                   
 
                                                                  
        
                                                             
        
                 

 







                                                   
 

                   
                                             

                          
                                  
 
                                                          
 


                                                                    
                                     









                                                                          
 
                                                                            
                                  
 
                                             
                                                                               
 
                                                                                


                                                                                        
                                                                                                    





                                                                                                  
                                                                                            
                                                                                          
                                                                                            




                                                                                    






                                                           
                                  
 

                                                           
 
                                      


                                                  
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* Evolution addressbook - Address Conduit
 *
 * Copyright (C) 1998 Free Software Foundation
 * Copyright (C) 2000 Ximian, Inc.
 *
 * Authors: Eskil Heyn Olsen <deity@eskil.dk> 
 *          JP Rosevear <jpr@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 Place, Suite 330, Boston, MA 02111-1307, USA.
 */

#include <config.h>

#include <liboaf/liboaf.h>
#include <bonobo.h>
#include <gnome-xml/parser.h>
#include <pi-source.h>
#include <pi-socket.h>
#include <pi-file.h>
#include <pi-dlp.h>
#include <ebook/e-book.h>
#include <ebook/e-card-types.h>
#include <ebook/e-card-cursor.h>
#include <ebook/e-card.h>
#include <ebook/e-card-simple.h>
#include <e-pilot-util.h>

#define ADDR_CONFIG_LOAD 1
#define ADDR_CONFIG_DESTROY 1
#include "address-conduit-config.h"
#undef ADDR_CONFIG_LOAD
#undef ADDR_CONFIG_DESTROY

#include "address-conduit.h"

static void free_local (EAddrLocalRecord *local);
GnomePilotConduit * conduit_get_gpilot_conduit (guint32);
void conduit_destroy_gpilot_conduit (GnomePilotConduit*);

#define CONDUIT_VERSION "0.1.2"
#ifdef G_LOG_DOMAIN
#undef G_LOG_DOMAIN
#endif
#define G_LOG_DOMAIN "eaddrconduit"

#define DEBUG_CONDUIT 1
/* #undef DEBUG_CONDUIT */

#ifdef DEBUG_CONDUIT
#define LOG(e...) g_log (G_LOG_DOMAIN, G_LOG_LEVEL_MESSAGE, e)
#else
#define LOG(e...)
#endif 

#define WARN(e...) g_log (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING, e)
#define INFO(e...) g_log (G_LOG_DOMAIN, G_LOG_LEVEL_MESSAGE, e)

typedef struct {
    EBookStatus status;
    char *id;
} CardObjectChangeStatus;

typedef enum {
    CARD_ADDED,
    CARD_MODIFIED,
    CARD_DELETED
} CardObjectChangeType;

typedef struct 
{
    ECard *card;
    CardObjectChangeType type;
} CardObjectChange;


static ECardSimpleField priority [] = {
    E_CARD_SIMPLE_FIELD_PHONE_BUSINESS,
    E_CARD_SIMPLE_FIELD_PHONE_HOME,
    E_CARD_SIMPLE_FIELD_PHONE_BUSINESS_FAX,
    E_CARD_SIMPLE_FIELD_EMAIL,
    E_CARD_SIMPLE_FIELD_PHONE_PAGER,
    E_CARD_SIMPLE_FIELD_PHONE_MOBILE,
    E_CARD_SIMPLE_FIELD_PHONE_BUSINESS_2,
    E_CARD_SIMPLE_FIELD_PHONE_HOME_2,
    E_CARD_SIMPLE_FIELD_PHONE_HOME_FAX,
    E_CARD_SIMPLE_FIELD_EMAIL_2,
    E_CARD_SIMPLE_FIELD_PHONE_OTHER,
    E_CARD_SIMPLE_FIELD_PHONE_PRIMARY,
    E_CARD_SIMPLE_FIELD_PHONE_OTHER_FAX,
    E_CARD_SIMPLE_FIELD_EMAIL_3,
    E_CARD_SIMPLE_FIELD_LAST
};

static char *priority_label [] = {
    "Work",
    "Home",
    "Fax",
    "E-mail",
    "Pager",
    "Mobile",
    "Work",
    "Home",
    "Fax",
    "E-mail",
    "Other",
    "Main",
    "Fax",
    "E-Mail",
    NULL
};

/* Debug routines */
static char *
print_local (EAddrLocalRecord *local)
{
    static char buff[ 4096 ];

    if (local == NULL) {
        sprintf (buff, "[NULL]");
        return buff;
    }

    if (local->addr) {
        g_snprintf (buff, 4096, "['%s' '%s' '%s']",
                local->addr->entry[entryLastname] ?
                local->addr->entry[entryLastname] : "",
                local->addr->entry[entryFirstname] ?
                local->addr->entry[entryFirstname] : "",
                local->addr->entry[entryCompany] ?
                local->addr->entry[entryCompany] : "");
        return buff;
    }

    return "";
}

static char *print_remote (GnomePilotRecord *remote)
{
    static char buff[ 4096 ];
    struct Address addr;

    if (remote == NULL) {
        sprintf (buff, "[NULL]");
        return buff;
    }

    memset (&addr, 0, sizeof (struct Address));
    unpack_Address (&addr, remote->record, remote->length);

    g_snprintf (buff, 4096, "['%s' '%s' '%s']",
            addr.entry[entryLastname] ?
            addr.entry[entryLastname] : "",
            addr.entry[entryFirstname] ?
            addr.entry[entryFirstname] : "",
            addr.entry[entryCompany] ?
            addr.entry[entryCompany] : "");

    free_Address (&addr);

    return buff;
}

/* Context Routines */
static EAddrConduitContext *
e_addr_context_new (guint32 pilot_id) 
{
    EAddrConduitContext *ctxt = g_new0 (EAddrConduitContext, 1);

    addrconduit_load_configuration (&ctxt->cfg, pilot_id);

    ctxt->ebook = NULL;
    ctxt->cards = NULL;
    ctxt->changed_hash = NULL;
    ctxt->changed = NULL;
    ctxt->locals = NULL;
    ctxt->map = NULL;

    return ctxt;
}

static gboolean
e_addr_context_foreach_change (gpointer key, gpointer value, gpointer data) 
{
    g_free (key);

    return TRUE;
}

static void
e_addr_context_destroy (EAddrConduitContext *ctxt)
{
    GList *l;
    
    g_return_if_fail (ctxt != NULL);

    if (ctxt->cfg != NULL)
        addrconduit_destroy_configuration (&ctxt->cfg);

    if (ctxt->ebook != NULL)
        gtk_object_unref (GTK_OBJECT (ctxt->ebook));

    if (ctxt->cards != NULL) {
        for (l = ctxt->cards; l != NULL; l = l->next)
            gtk_object_unref (GTK_OBJECT (l->data));
        g_list_free (ctxt->cards);
    }
    
    if (ctxt->changed_hash != NULL) {
        g_hash_table_foreach_remove (ctxt->changed_hash, e_addr_context_foreach_change, NULL);
        g_hash_table_destroy (ctxt->changed_hash);
    }
    
    if (ctxt->changed != NULL)
        g_list_free (ctxt->changed);
    
    if (ctxt->locals != NULL) {
        for (l = ctxt->locals; l != NULL; l = l->next)
            free_local (l->data);
        g_list_free (ctxt->locals);
    }

    if (ctxt->map != NULL)
        e_pilot_map_destroy (ctxt->map);

    g_free (ctxt);
}

/* Addressbok Server routines */
static void
add_card_cb (EBook *ebook, EBookStatus status, const char *id, gpointer closure)
{
    CardObjectChangeStatus *cons = closure;

    cons->status = status;
    cons->id = g_strdup (id);
    
    gtk_main_quit();
}

static void
status_cb (EBook *ebook, EBookStatus status, gpointer closure)
{
    (*(EBookStatus*)closure) = status;
    gtk_main_quit();
}

static void
cursor_cb (EBook *book, EBookStatus status, ECardCursor *cursor, gpointer closure)
{
    EAddrConduitContext *ctxt = (EAddrConduitContext*)closure;

    if (status == E_BOOK_STATUS_SUCCESS) {
        long length;
        int i;

        ctxt->address_load_success = TRUE;

        length = e_card_cursor_get_length (cursor);
        ctxt->cards = NULL;
        for (i = 0; i < length; i ++) {
            ECard *card = e_card_cursor_get_nth (cursor, i);
            
            if (e_card_evolution_list (card))
                continue;

            ctxt->cards = g_list_append (ctxt->cards, card);
        }

        gtk_main_quit(); /* end the sub event loop */
    }
    else {
        WARN (_("Cursor could not be loaded\n"));
        gtk_main_quit(); /* end the sub event loop */
    }
}

static void
book_open_cb (EBook *book, EBookStatus status, gpointer closure)
{
    EAddrConduitContext *ctxt = (EAddrConduitContext*)closure;

    if (status == E_BOOK_STATUS_SUCCESS) {
        e_book_get_cursor (book, "(contains \"full_name\" \"\")", cursor_cb, ctxt);
    } else {
        WARN (_("EBook not loaded\n"));
        gtk_main_quit(); /* end the sub event loop */
    }
}

static int
start_addressbook_server (EAddrConduitContext *ctxt)
{
    gchar *uri, *path;

    g_return_val_if_fail(ctxt!=NULL,-2);

    ctxt->ebook = e_book_new ();

    path = g_concat_dir_and_file (g_get_home_dir (),
                      "evolution/local/Contacts/addressbook.db");
    uri = g_strdup_printf ("file://%s", path);
    g_free (path);

    e_book_load_uri (ctxt->ebook, uri, book_open_cb, ctxt);

    /* run a sub event loop to turn ebook's async loading into a
           synchronous call */
    gtk_main ();

    g_free (uri);

    if (ctxt->address_load_success)
        return 0;

    return -1;
}

/* Utility routines */
static char *
map_name (EAddrConduitContext *ctxt) 
{
    char *filename = NULL;
    
    filename = g_strdup_printf ("%s/evolution/local/Contacts/pilot-map-%d.xml", g_get_home_dir (), ctxt->cfg->pilot_id);

    return filename;
}

static GList *
next_changed_item (EAddrConduitContext *ctxt, GList *changes) 
{
    CardObjectChange *coc;
    GList *l;
    
    for (l = changes; l != NULL; l = l->next) {
        coc = l->data;
        
        if (g_hash_table_lookup (ctxt->changed_hash, e_card_get_id (coc->card)))
            return l;
    }
    
    return NULL;
}

static int
get_label (EAddrConduitContext *ctxt, const char *label)
{
    int i;
    
    for (i = 0; i < 8; i++) {
        if (!strcmp (ctxt->ai.phoneLabels[i], label))
            return i;
    }

    return 0;
}

static ECardSimpleField
get_next_mail (ECardSimpleField *field)
{
    if (field == NULL)
        return E_CARD_SIMPLE_FIELD_EMAIL;
    
    switch (*field) {
    case E_CARD_SIMPLE_FIELD_EMAIL:
        return E_CARD_SIMPLE_FIELD_EMAIL_2;
    case E_CARD_SIMPLE_FIELD_EMAIL_2:
        return E_CARD_SIMPLE_FIELD_EMAIL_3;
    default:
    }

    return E_CARD_SIMPLE_FIELD_LAST;
}

static ECardSimpleField
get_next_home (ECardSimpleField *field)
{
    if (field == NULL)
        return E_CARD_SIMPLE_FIELD_PHONE_HOME;
    
    switch (*field) {
    case E_CARD_SIMPLE_FIELD_PHONE_HOME:
        return E_CARD_SIMPLE_FIELD_PHONE_HOME_2;
    default:
    }

    return E_CARD_SIMPLE_FIELD_LAST;
}

static ECardSimpleField
get_next_work (ECardSimpleField *field)
{
    if (field == NULL)
        return E_CARD_SIMPLE_FIELD_PHONE_BUSINESS;
    
    switch (*field) {
    case E_CARD_SIMPLE_FIELD_PHONE_BUSINESS:
        return E_CARD_SIMPLE_FIELD_PHONE_BUSINESS_2;
    default:
    }

    return E_CARD_SIMPLE_FIELD_LAST;
}

static ECardSimpleField
get_next_fax (ECardSimpleField *field)
{
    if (field == NULL)
        return E_CARD_SIMPLE_FIELD_PHONE_BUSINESS_FAX;
    
    switch (*field) {
    case E_CARD_SIMPLE_FIELD_PHONE_BUSINESS_FAX:
        return E_CARD_SIMPLE_FIELD_PHONE_HOME_FAX;
    case E_CARD_SIMPLE_FIELD_PHONE_HOME_FAX:
        return E_CARD_SIMPLE_FIELD_PHONE_OTHER_FAX;
    default:
    }

    return E_CARD_SIMPLE_FIELD_LAST;
}

static ECardSimpleField
get_next_other (ECardSimpleField *field)
{
    if (field == NULL)
        return E_CARD_SIMPLE_FIELD_PHONE_OTHER;

    return E_CARD_SIMPLE_FIELD_LAST;
}

static ECardSimpleField
get_next_main (ECardSimpleField *field)
{
    if (field == NULL)
        return E_CARD_SIMPLE_FIELD_PHONE_PRIMARY;

    return E_CARD_SIMPLE_FIELD_LAST;
}

static ECardSimpleField
get_next_pager (ECardSimpleField *field)
{
    if (field == NULL)
        return E_CARD_SIMPLE_FIELD_PHONE_PAGER;

    return E_CARD_SIMPLE_FIELD_LAST;
}

static ECardSimpleField
get_next_mobile (ECardSimpleField *field)
{
    if (field == NULL)
        return E_CARD_SIMPLE_FIELD_PHONE_MOBILE;

    return E_CARD_SIMPLE_FIELD_LAST;
}

static void
get_next_init (ECardSimpleField *next_mail,
           ECardSimpleField *next_home,
           ECardSimpleField *next_work,
           ECardSimpleField *next_fax,
           ECardSimpleField *next_other,
           ECardSimpleField *next_main,
           ECardSimpleField *next_pager,
           ECardSimpleField *next_mobile)
{   
    *next_mail = get_next_mail (NULL);
    *next_home = get_next_home (NULL);
    *next_work = get_next_work (NULL);
    *next_fax = get_next_fax (NULL);
    *next_other = get_next_other (NULL);
    *next_main = get_next_main (NULL);
    *next_pager = get_next_pager (NULL);
    *next_mobile = get_next_mobile (NULL);
}

static gboolean
is_next_done (ECardSimpleField field)
{
    if (field == E_CARD_SIMPLE_FIELD_LAST)
        return TRUE;
    
    return FALSE;
}

static char *
get_entry_text (struct Address address, int field)
{
    if (address.entry[field])
        return e_pilot_utf8_from_pchar (address.entry[field]);
    
    return g_strdup ("");
}

static void
clear_entry_text (struct Address address, int field) 
{
    if (address.entry[field]) {
        free (address.entry[field]);
        address.entry[field] = NULL;
    }
}

static void
compute_status (EAddrConduitContext *ctxt, EAddrLocalRecord *local, const char *uid)
{
    CardObjectChange *coc;

    local->local.archived = FALSE;
    local->local.secret = FALSE;

    coc = g_hash_table_lookup (ctxt->changed_hash, uid);
    
    if (coc == NULL) {
        local->local.attr = GnomePilotRecordNothing;
        return;
    }
    
    switch (coc->type) {
    case CARD_ADDED:
        local->local.attr = GnomePilotRecordNew;
        break;  
    case CARD_MODIFIED:
        local->local.attr = GnomePilotRecordModified;
        break;
    case CARD_DELETED:
        local->local.attr = GnomePilotRecordDeleted;
        break;
    }
}

static void
free_local (EAddrLocalRecord *local) 
{
    gtk_object_unref (GTK_OBJECT (local->ecard));
    free_Address (local->addr);
    g_free (local->addr);
    g_free (local);
}

static GnomePilotRecord
local_record_to_pilot_record (EAddrLocalRecord *local,
                  EAddrConduitContext *ctxt)
{
    GnomePilotRecord p;
    static char record[0xffff];
    
    g_assert (local->addr != NULL );
    
    LOG ("local_record_to_pilot_record\n");

    p.ID = local->local.ID;
    p.category = local->local.category;
    p.attr = local->local.attr;
    p.archived = local->local.archived;
    p.secret = local->local.secret;

    /* Generate pilot record structure */
    p.record = record;
    p.length = pack_Address (local->addr, p.record, 0xffff);

    return p;   
}

static void
local_record_from_ecard (EAddrLocalRecord *local, ECard *ecard, EAddrConduitContext *ctxt)
{
    ECardSimple *simple;
    const ECardDeliveryAddress *delivery;
    int phone = entryPhone1;
    ECardSimpleField next_mail, next_home, next_work, next_fax;
    ECardSimpleField next_other, next_main, next_pager, next_mobile;
    gboolean syncable;
    int i;
    
    g_return_if_fail (local != NULL);
    g_return_if_fail (ecard != NULL);

    local->ecard = ecard;
    gtk_object_ref (GTK_OBJECT (ecard));
    simple = e_card_simple_new (ecard);
    
    local->local.ID = e_pilot_map_lookup_pid (ctxt->map, ecard->id, TRUE);

    compute_status (ctxt, local, ecard->id);

    local->addr = g_new0 (struct Address, 1);

    /* Handle the fields and category we don't sync by making sure
         * we don't overwrite them 
     */
    if (local->local.ID != 0) {
        struct Address addr;
        char record[0xffff];
        int cat = 0;
        
        if (dlp_ReadRecordById (ctxt->dbi->pilot_socket, 
                    ctxt->dbi->db_handle,
                    local->local.ID, &record, 
                    NULL, NULL, NULL, &cat) > 0) {
            local->local.category = cat;
            memset (&addr, 0, sizeof (struct Address));
            unpack_Address (&addr, record, 0xffff);
            if (addr.entry[entryPhone1])
                local->addr->entry[entryPhone1] = strdup (addr.entry[entryPhone1]);
            if (addr.entry[entryPhone2])
                local->addr->entry[entryPhone2] = strdup (addr.entry[entryPhone2]);
            if (addr.entry[entryPhone3])
                local->addr->entry[entryPhone3] = strdup (addr.entry[entryPhone3]);
            if (addr.entry[entryPhone4])
                local->addr->entry[entryPhone4] = strdup (addr.entry[entryPhone4]);
            if (addr.entry[entryPhone5])
                local->addr->entry[entryPhone5] = strdup (addr.entry[entryPhone5]);
            free_Address (&addr);
        }
    }

    if (ecard->name) {
        local->addr->entry[entryFirstname] = e_pilot_utf8_to_pchar (ecard->name->given);
        local->addr->entry[entryLastname] = e_pilot_utf8_to_pchar (ecard->name->family);
        local->addr->entry[entryCompany] = e_pilot_utf8_to_pchar (ecard->org);
        local->addr->entry[entryTitle] = e_pilot_utf8_to_pchar (ecard->title);
    }

    delivery = e_card_simple_get_delivery_address (simple, E_CARD_SIMPLE_ADDRESS_ID_BUSINESS);
    if (delivery) {
        local->addr->entry[entryAddress] = e_pilot_utf8_to_pchar (delivery->street);
        local->addr->entry[entryCity] = e_pilot_utf8_to_pchar (delivery->city);
        local->addr->entry[entryState] = e_pilot_utf8_to_pchar (delivery->region);
        local->addr->entry[entryZip] = e_pilot_utf8_to_pchar (delivery->code);
        local->addr->entry[entryCountry] = e_pilot_utf8_to_pchar (delivery->country);
    }

    /* Phone numbers */
    get_next_init (&next_mail, &next_home, &next_work, &next_fax,
               &next_other, &next_main, &next_pager, &next_mobile);

    /* See if everything is syncable */
    syncable = TRUE;
    for (i = entryPhone1; i <= entryPhone5; i++) {
        char *phonelabel = ctxt->ai.phoneLabels[local->addr->phoneLabel[i - entryPhone1]];
        const char *phone_str = NULL;
        
        if (!strcmp (phonelabel, "E-mail")) {
            if (is_next_done (next_mail)) {
                syncable = FALSE;
                break;
            }
            phone_str = e_card_simple_get_const (simple, next_home);
            if (phone_str && *phone_str)
                next_mail = get_next_mail (&next_mail);
        } else if (!strcmp (phonelabel, "Home")) {
            if (is_next_done (next_home)) {
                syncable = FALSE;
                break;
            }
            phone_str = e_card_simple_get_const (simple, next_home);
            if (phone_str && *phone_str)
                next_home = get_next_home (&next_home);
        } else if (!strcmp (phonelabel, "Work")) {
            if (is_next_done (next_work)) {
                syncable = FALSE;
                break;
            }
            phone_str = e_card_simple_get_const (simple, next_work);
            if (phone_str && *phone_str)
                next_work = get_next_work (&next_work);
        } else if (!strcmp (phonelabel, "Fax")) {
            if (is_next_done (next_fax)) {
                syncable = FALSE;
                break;
            }
            phone_str = e_card_simple_get_const (simple, next_fax);
            if (phone_str && *phone_str)
                next_fax = get_next_fax (&next_fax);
        } else if (!strcmp (phonelabel, "Other")) {
            if (is_next_done (next_other)) {
                syncable = FALSE;
                break;
            }
            phone_str = e_card_simple_get_const (simple, next_other);
            if (phone_str && *phone_str)
                next_other = get_next_other (&next_other);
        } else if (!strcmp (phonelabel, "Main")) {
            if (is_next_done (next_main)) {
                syncable = FALSE;
                break;
            }
            phone_str = e_card_simple_get_const (simple, next_main);
            if (phone_str && *phone_str)
                next_main = get_next_main (&next_main);
        } else if (!strcmp (phonelabel, "Pager")) {
            if (is_next_done (next_pager)) {
                syncable = FALSE;
                break;
            }
            phone_str = e_card_simple_get_const (simple, next_pager);
            if (phone_str && *phone_str)
                next_pager = get_next_pager (&next_pager);
        } else if (!strcmp (phonelabel, "Mobile")) {
            if (is_next_done (next_mobile)) {
                syncable = FALSE;
                break;
            }
            phone_str = e_card_simple_get_const (simple, next_mobile);
            if (phone_str && *phone_str)
                next_mobile = get_next_mobile (&next_mobile);
        }       
    }

    if (syncable) {
        INFO ("Syncable");
        /* Sync by priority */
        for (i = 0, phone = entryPhone1; 
             priority[i] != E_CARD_SIMPLE_FIELD_LAST && phone <= entryPhone5; i++) {
            const char *phone_str;
            
            phone_str = e_card_simple_get_const (simple, priority[i]);
            if (phone_str && *phone_str) {
                clear_entry_text (*local->addr, phone);
                local->addr->entry[phone] = e_pilot_utf8_to_pchar (phone_str);
                local->addr->phoneLabel[phone - entryPhone1] = 
                    get_label (ctxt, priority_label[i]);
                phone++;
            }
        }
        for ( ; phone <= entryPhone5; phone++)
                  local->addr->phoneLabel[phone - entryPhone1] = phone - entryPhone1;
    } else {
        INFO ("Not Syncable");
        get_next_init (&next_mail, &next_home, &next_work, &next_fax,
                   &next_other, &next_main, &next_pager, &next_mobile);

        /* Not completely syncable, so do the best we can */
        for (i = entryPhone1; i <= entryPhone5; i++) {
            char *phonelabel = ctxt->ai.phoneLabels[local->addr->phoneLabel[i - entryPhone1]];
            const char *phone_str = NULL;
            
            if (!strcmp (phonelabel, "E-mail") && !is_next_done (next_mail)) {
                phone_str = e_card_simple_get_const (simple, next_mail);
                next_mail = get_next_mail (&next_mail);
            } else if (!strcmp (phonelabel, "Home") && !is_next_done (next_home)) {
                phone_str = e_card_simple_get_const (simple, next_home);
                next_home = get_next_home (&next_home);
            } else if (!strcmp (phonelabel, "Work") && !is_next_done (next_work)) {
                phone_str = e_card_simple_get_const (simple, next_work);
                next_work = get_next_work (&next_work);
            } else if (!strcmp (phonelabel, "Fax") && !is_next_done (next_fax)) {
                phone_str = e_card_simple_get_const (simple, next_fax);
                next_fax = get_next_fax (&next_fax);
            } else if (!strcmp (phonelabel, "Other")  && !is_next_done (next_other)) {
                phone_str = e_card_simple_get_const (simple, next_other);
                next_other = get_next_other (&next_other);
            } else if (!strcmp (phonelabel, "Main") && !is_next_done (next_main)) {
                phone_str = e_card_simple_get_const (simple, next_main);
                next_main = get_next_main (&next_main);
            } else if (!strcmp (phonelabel, "Pager") && !is_next_done (next_pager)) {
                phone_str = e_card_simple_get_const (simple, next_pager);
                next_pager = get_next_pager (&next_pager);
            } else if (!strcmp (phonelabel, "Mobile") && !is_next_done (next_mobile)) {
                phone_str = e_card_simple_get_const (simple, next_mobile);
                next_mobile = get_next_mobile (&next_mobile);
            }
            
            if (phone_str && *phone_str) {
                clear_entry_text (*local->addr, phone);
                local->addr->entry[i] = e_pilot_utf8_to_pchar (phone_str);
            }
        }
    }
    
    /* Note */
    local->addr->entry[entryNote] = e_pilot_utf8_to_pchar (ecard->note);

    gtk_object_unref (GTK_OBJECT (simple));
}

static void 
local_record_from_uid (EAddrLocalRecord *local,
               const char *uid,
               EAddrConduitContext *ctxt)
{
    ECard *ecard = NULL;
    GList *l;
    
    g_assert (local != NULL);

    for (l = ctxt->cards; l != NULL; l = l->next) {
        ecard = l->data;
        
        if (ecard->id && !strcmp (ecard->id, uid))
            break;

        ecard = NULL;
    }

    if (ecard != NULL) {
        local_record_from_ecard (local, ecard, ctxt);
    } else {
        ecard = e_card_new ("");
        e_card_set_id (ecard, uid);
        local_record_from_ecard (local, ecard, ctxt);
    }
}

static ECard *
ecard_from_remote_record(EAddrConduitContext *ctxt,
             GnomePilotRecord *remote,
             ECard *in_card)
{
    struct Address address;
    ECard *ecard;
    ECardSimple *simple;
    ECardDeliveryAddress *delivery;
    ECardAddrLabel *label;
    char *txt;
    char *stringparts[3];
    ECardSimpleField next_mail, next_home, next_work, next_fax;
    ECardSimpleField next_other, next_main, next_pager, next_mobile;
    int i;

    g_return_val_if_fail(remote!=NULL,NULL);
    memset (&address, 0, sizeof (struct Address));
    unpack_Address (&address, remote->record, remote->length);

    if (in_card == NULL)
        ecard = e_card_new("");
    else
        ecard = e_card_duplicate (in_card);
    simple = e_card_simple_new (ecard);

    /* Name and company */
    i = 0;
    if (address.entry[entryFirstname] && *address.entry[entryFirstname])
        stringparts[i++] = address.entry[entryFirstname];
    if (address.entry[entryLastname] && *address.entry[entryLastname])
        stringparts[i++] = address.entry[entryLastname];
    stringparts[i] = NULL;

    txt = g_strjoinv (" ", stringparts);
    e_card_simple_set (simple, E_CARD_SIMPLE_FIELD_FULL_NAME, e_pilot_utf8_from_pchar (txt));
    g_free (txt);

    txt = get_entry_text (address, entryTitle);
    e_card_simple_set(simple, E_CARD_SIMPLE_FIELD_TITLE, txt);
    g_free (txt);

    txt = get_entry_text (address, entryCompany);
    e_card_simple_set(simple, E_CARD_SIMPLE_FIELD_ORG, txt);
    if (i == 0)
        e_card_simple_set(simple, E_CARD_SIMPLE_FIELD_FILE_AS, txt);    
    g_free (txt);

    /* Address */
    delivery = e_card_delivery_address_new ();
    delivery->flags = E_CARD_ADDR_WORK;
    delivery->street = get_entry_text (address, entryAddress);
    delivery->city = get_entry_text (address, entryCity);
    delivery->region = get_entry_text (address, entryState);
    delivery->country = get_entry_text (address, entryCountry);
    delivery->code = get_entry_text (address, entryZip);

    label = e_card_address_label_new ();
    label->flags = E_CARD_ADDR_WORK;
    label->data = e_card_delivery_address_to_string (delivery);

    e_card_simple_set_address (simple, E_CARD_SIMPLE_ADDRESS_ID_BUSINESS, label);
    e_card_simple_set_delivery_address (simple, E_CARD_SIMPLE_ADDRESS_ID_BUSINESS, delivery);

    e_card_delivery_address_unref (delivery);
    e_card_address_label_unref (label);
    
    /* Phone numbers */
    get_next_init (&next_mail, &next_home, &next_work, &next_fax,
               &next_other, &next_main, &next_pager, &next_mobile);

    for (i = entryPhone1; i <= entryPhone5; i++) {
        char *phonelabel = ctxt->ai.phoneLabels[address.phoneLabel[i - entryPhone1]];
        char *phonenum = get_entry_text (address, i);
        
        if (!strcmp (phonelabel, "E-mail") && !is_next_done (next_mail)) {
            e_card_simple_set (simple, next_mail, phonenum);
            next_mail = get_next_mail (&next_mail);
        } else if (!strcmp (phonelabel, "Home") && !is_next_done (next_home)) {
            e_card_simple_set (simple, next_home, phonenum);
            next_home = get_next_home (&next_home);
        } else if (!strcmp (phonelabel, "Work") && !is_next_done (next_work)) {
            e_card_simple_set (simple, next_work, phonenum);
            next_work = get_next_work (&next_work);
        } else if (!strcmp (phonelabel, "Fax") && !is_next_done (next_fax)) {
            e_card_simple_set (simple, next_fax, phonenum);
            next_fax = get_next_fax (&next_fax);
        } else if (!strcmp (phonelabel, "Other") && !is_next_done (next_other)) {
            e_card_simple_set (simple, next_other, phonenum);
            next_other = get_next_other (&next_other);
        } else if (!strcmp (phonelabel, "Main") && !is_next_done (next_main)) {
            e_card_simple_set (simple, next_main, phonenum);
            next_main = get_next_main (&next_main);
        } else if (!strcmp (phonelabel, "Pager") && !is_next_done (next_pager)) {
            e_card_simple_set (simple, next_pager, phonenum);
            next_pager = get_next_pager (&next_pager);
        } else if (!strcmp (phonelabel, "Mobile") && !is_next_done (next_mobile)) {
            e_card_simple_set (simple, next_mobile, phonenum);
            next_mobile = get_next_mobile (&next_mobile);
        }
        
        g_free (phonenum);
    }

    /* Note */
    txt = get_entry_text (address, entryNote);
    e_card_simple_set(simple, E_CARD_SIMPLE_FIELD_NOTE, txt);
    g_free (txt);
    
    e_card_simple_sync_card (simple);
    gtk_object_unref(GTK_OBJECT(simple));

    free_Address(&address);

    return ecard;
}

static void
check_for_slow_setting (GnomePilotConduit *c, EAddrConduitContext *ctxt)
{
    GnomePilotConduitStandard *conduit = GNOME_PILOT_CONDUIT_STANDARD (c);
    int map_count;

    map_count = g_hash_table_size (ctxt->map->pid_map); 
    if (map_count == 0)
        gnome_pilot_conduit_standard_set_slow (conduit, TRUE);

    if (gnome_pilot_conduit_standard_get_slow (conduit)) {
        ctxt->map->write_touched_only = TRUE;
        LOG ("    doing slow sync\n");
    } else {
        LOG ("    doing fast sync\n");
    }
}

static void
card_added (EBookView *book_view, const GList *cards, EAddrConduitContext *ctxt)
{
    const GList *l;

    for (l = cards; l != NULL; l = l->next) {
        ECard *card = E_CARD (l->data);
        CardObjectChange *coc;
        
        if (e_card_evolution_list (card))
            continue;
        
        coc = g_new0 (CardObjectChange, 1);
        coc->card = card;
        coc->type = CARD_ADDED;

        gtk_object_ref (GTK_OBJECT (coc->card));
        ctxt->changed = g_list_prepend (ctxt->changed, coc);
        if (!e_pilot_map_uid_is_archived (ctxt->map, e_card_get_id (coc->card)))
            g_hash_table_insert (ctxt->changed_hash, (gpointer)e_card_get_id (coc->card), coc);
    }
}

static void
card_changed (EBookView *book_view, const GList *cards, EAddrConduitContext *ctxt)
{
    const GList *l;

    for (l = cards; l != NULL; l = l->next) {
        ECard *card = E_CARD (l->data);
        CardObjectChange *coc;

        if (e_card_evolution_list (card))
            continue;
        
        coc = g_new0 (CardObjectChange, 1);     
        coc->card = E_CARD (l->data);
        coc->type = CARD_MODIFIED;

        gtk_object_ref (GTK_OBJECT (coc->card));
        ctxt->changed = g_list_prepend (ctxt->changed, coc);
        if (!e_pilot_map_uid_is_archived (ctxt->map, e_card_get_id (coc->card)))
            g_hash_table_insert (ctxt->changed_hash, (gpointer)e_card_get_id (coc->card), coc);
    }
}


static void
card_removed (EBookView *book_view, const char *id, EAddrConduitContext *ctxt)
{
    CardObjectChange *coc;
    gboolean archived;
    
    archived = e_pilot_map_uid_is_archived (ctxt->map, id);
    
    /* If its deleted, not in the archive and not in the map its a list */
    if (!archived && e_pilot_map_lookup_pid (ctxt->map, id, FALSE) == 0)
        return; 
    
    coc = g_new0 (CardObjectChange, 1);
    coc->card = e_card_new ("");
    e_card_set_id (coc->card, id);
    coc->type = CARD_DELETED;

    ctxt->changed = g_list_prepend (ctxt->changed, coc);
    
    if (!archived)
        g_hash_table_insert (ctxt->changed_hash, (gpointer)e_card_get_id (coc->card), coc);
    else
        e_pilot_map_remove_by_uid (ctxt->map, id);
}

static void
sequence_complete (EBookView *book_view, EAddrConduitContext *ctxt)
{
    gtk_signal_disconnect_by_data (GTK_OBJECT (book_view), ctxt);
    gtk_object_unref (GTK_OBJECT (book_view));
    gtk_main_quit ();
}

static void
view_cb (EBook *book, EBookStatus status, EBookView *book_view, gpointer data)
{
    EAddrConduitContext *ctxt = data;
    
    gtk_object_ref (GTK_OBJECT (book_view));
    
    gtk_signal_connect (GTK_OBJECT (book_view), "card_added", 
                (GtkSignalFunc) card_added, ctxt);
    gtk_signal_connect (GTK_OBJECT (book_view), "card_changed", 
                (GtkSignalFunc) card_changed, ctxt);
    gtk_signal_connect (GTK_OBJECT (book_view), "card_removed", 
                (GtkSignalFunc) card_removed, ctxt);
    gtk_signal_connect (GTK_OBJECT (book_view), "sequence_complete", 
                (GtkSignalFunc) sequence_complete, ctxt);

}

/* Pilot syncing callbacks */
static gint
pre_sync (GnomePilotConduit *conduit,
      GnomePilotDBInfo *dbi,
      EAddrConduitContext *ctxt)
{
    GnomePilotConduitSyncAbs *abs_conduit;
/*      GList *l; */
    int len;
    unsigned char *buf;
    char *filename;
    char *change_id;
/*      gint num_records; */

    abs_conduit = GNOME_PILOT_CONDUIT_SYNC_ABS (conduit);

    LOG ("---------------------------------------------------------\n");
    LOG ("pre_sync: Addressbook Conduit v.%s", CONDUIT_VERSION);
    g_message ("Addressbook Conduit v.%s", CONDUIT_VERSION);

    ctxt->dbi = dbi;    
    ctxt->ebook = NULL;
    
    if (start_addressbook_server (ctxt) != 0) {
        WARN(_("Could not start wombat server"));
        gnome_pilot_conduit_error (conduit, _("Could not start wombat"));
        return -1;
    }

    /* Load the uid <--> pilot id mappings */
    filename = map_name (ctxt);
    e_pilot_map_read (filename, &ctxt->map);
    g_free (filename);

    /* Count and hash the changes */
    change_id = g_strdup_printf ("pilot-sync-evolution-addressbook-%d", ctxt->cfg->pilot_id);
    ctxt->changed_hash = g_hash_table_new (g_str_hash, g_str_equal);
    e_book_get_changes (ctxt->ebook, change_id, view_cb, ctxt);
    
    /* Force the view loading to be synchronous */
    gtk_main ();
    g_free (change_id);
    
    /* Set the count information */
/*      num_records = cal_client_get_n_objects (ctxt->client, CALOBJ_TYPE_TODO); */
/*      gnome_pilot_conduit_sync_abs_set_num_local_records(abs_conduit, num_records); */
/*      gnome_pilot_conduit_sync_abs_set_num_new_local_records (abs_conduit, add_records); */
/*      gnome_pilot_conduit_sync_abs_set_num_updated_local_records (abs_conduit, mod_records); */
/*      gnome_pilot_conduit_sync_abs_set_num_deleted_local_records(abs_conduit, del_records); */

    buf = (unsigned char*)g_malloc (0xffff);
    len = dlp_ReadAppBlock (dbi->pilot_socket, dbi->db_handle, 0,
                  (unsigned char *)buf, 0xffff);
    
    if (len < 0) {
        WARN (_("Could not read pilot's Address application block"));
        WARN ("dlp_ReadAppBlock(...) = %d", len);
        gnome_pilot_conduit_error (conduit,
                       _("Could not read pilot's Address application block"));
        return -1;
    }
    unpack_AddressAppInfo (&(ctxt->ai), buf, len);
    g_free (buf);

    check_for_slow_setting (conduit, ctxt);
    if (ctxt->cfg->sync_type == GnomePilotConduitSyncTypeCopyToPilot
        || ctxt->cfg->sync_type == GnomePilotConduitSyncTypeCopyFromPilot)
        ctxt->map->write_touched_only = TRUE;

    return 0;
}

static gint
post_sync (GnomePilotConduit *conduit,
       GnomePilotDBInfo *dbi,
       EAddrConduitContext *ctxt)
{
    gchar *filename, *change_id;
    
    LOG ("post_sync: Address Conduit v.%s", CONDUIT_VERSION);

    filename = map_name (ctxt);
    e_pilot_map_write (filename, ctxt->map);
    g_free (filename);

    /* FIX ME ugly hack - our changes musn't count, this does introduce
     * a race condition if anyone changes a record elsewhere during sycnc
         */ 
    change_id = g_strdup_printf ("pilot-sync-evolution-addressbook-%d", ctxt->cfg->pilot_id);
    e_book_get_changes (ctxt->ebook, change_id, view_cb, ctxt);
    g_free (change_id);
    gtk_main ();

    LOG ("---------------------------------------------------------\n");
    
    return 0;
}

static gint
set_pilot_id (GnomePilotConduitSyncAbs *conduit,
          EAddrLocalRecord *local,
          guint32 ID,
          EAddrConduitContext *ctxt)
{
    LOG ("set_pilot_id: setting to %d\n", ID);
    
    e_pilot_map_insert (ctxt->map, ID, local->ecard->id, FALSE);

        return 0;
}

static gint
set_status_cleared (GnomePilotConduitSyncAbs *conduit,
            EAddrLocalRecord *local,
            EAddrConduitContext *ctxt)
{
    LOG ("set_status_cleared: clearing status\n");
    
    g_hash_table_remove (ctxt->changed_hash, e_card_get_id (local->ecard));
    
        return 0;
}

static gint
for_each (GnomePilotConduitSyncAbs *conduit,
      EAddrLocalRecord **local,
      EAddrConduitContext *ctxt)
{
    static GList *cards, *iterator;
    static int count;

    g_return_val_if_fail (local != NULL, -1);

    if (*local == NULL) {
        LOG ("beginning for_each");

        cards = ctxt->cards;
        count = 0;
        
        if (cards != NULL) {
            LOG ("iterating over %d records", g_list_length (cards));

            *local = g_new0 (EAddrLocalRecord, 1);
            local_record_from_ecard (*local, cards->data, ctxt);
            g_list_prepend (ctxt->locals, *local);

            iterator = cards;
        } else {
            LOG ("no events");
            (*local) = NULL;
            return 0;
        }
    } else {
        count++;
        if (g_list_next (iterator)) {
            iterator = g_list_next (iterator);

            *local = g_new0 (EAddrLocalRecord, 1);
            local_record_from_ecard (*local, iterator->data, ctxt);
            g_list_prepend (ctxt->locals, *local);
        } else {
            LOG ("for_each ending");

            /* Tell the pilot the iteration is over */
            *local = NULL;

            return 0;
        }
    }

    return 0;
}

static gint
for_each_modified (GnomePilotConduitSyncAbs *conduit,
           EAddrLocalRecord **local,
           EAddrConduitContext *ctxt)
{
    static GList *iterator;
    static int count;

    g_return_val_if_fail (local != NULL, 0);

    if (*local == NULL) {
        LOG ("for_each_modified beginning\n");
        
        iterator = ctxt->changed;
        
        count = 0;
        
        iterator = next_changed_item (ctxt, iterator);
        if (iterator != NULL) {
            CardObjectChange *coc = iterator->data;
            
            LOG ("iterating over %d records", g_hash_table_size (ctxt->changed_hash));
             
            *local = g_new0 (EAddrLocalRecord, 1);
            local_record_from_ecard (*local, coc->card, ctxt);
            g_list_prepend (ctxt->locals, *local);
        } else {
            LOG ("no events");

            *local = NULL;
        }
    } else {
        count++;
        iterator = g_list_next (iterator);
        if (iterator && (iterator = next_changed_item (ctxt, iterator))) {
            CardObjectChange *coc = iterator->data;

            *local = g_new0 (EAddrLocalRecord, 1);
            local_record_from_ecard (*local, coc->card, ctxt);
            g_list_prepend (ctxt->locals, *local);
        } else {
            LOG ("for_each_modified ending");

                /* Signal the iteration is over */
            *local = NULL;

            return 0;
        }
    }

    return 0;
}

static gint
compare (GnomePilotConduitSyncAbs *conduit,
     EAddrLocalRecord *local,
     GnomePilotRecord *remote,
     EAddrConduitContext *ctxt)
{
    /* used by the quick compare */
    GnomePilotRecord local_pilot;
    int retval = 0;

    LOG ("compare: local=%s remote=%s...\n",
         print_local (local), print_remote (remote));

    g_return_val_if_fail (local!=NULL,-1);
    g_return_val_if_fail (remote!=NULL,-1);

    local_pilot = local_record_to_pilot_record (local, ctxt);

    if (remote->length != local_pilot.length
        || memcmp (local_pilot.record, remote->record, remote->length))
        retval = 1;

    if (retval == 0)
        LOG ("    equal");
    else
        LOG ("    not equal");
    
    return retval;
}

static gint
add_record (GnomePilotConduitSyncAbs *conduit,
        GnomePilotRecord *remote,
        EAddrConduitContext *ctxt)
{
    ECard *ecard;
    CardObjectChangeStatus cons;
    int retval = 0;
    
    g_return_val_if_fail (remote != NULL, -1);

    LOG ("add_record: adding %s to desktop\n", print_remote (remote));

    ecard = ecard_from_remote_record (ctxt, remote, NULL);
    
    /* add the ecard to the server */
    e_book_add_card (ctxt->ebook, ecard, add_card_cb, &cons);

    gtk_main(); /* enter sub mainloop */
    
    if (cons.status != E_BOOK_STATUS_SUCCESS) {
        WARN ("add_record: failed to add card to ebook\n");
        return -1;
    }

    e_card_set_id (ecard, cons.id);
    e_pilot_map_insert (ctxt->map, remote->ID, ecard->id, FALSE);

    return retval;
}

static gint
replace_record (GnomePilotConduitSyncAbs *conduit,
        EAddrLocalRecord *local,
        GnomePilotRecord *remote,
        EAddrConduitContext *ctxt)
{
    ECard *new_ecard;
    EBookStatus commit_status;
    CardObjectChange *coc;
    CardObjectChangeStatus cons;
    char *old_id;
    int retval = 0;
    
    g_return_val_if_fail (remote != NULL, -1);

    LOG ("replace_record: replace %s with %s\n",
         print_local (local), print_remote (remote));

    old_id = g_strdup (e_card_get_id (local->ecard));
    coc = g_hash_table_lookup (ctxt->changed_hash, old_id);
    
    new_ecard = ecard_from_remote_record (ctxt, remote, local->ecard);
    gtk_object_unref (GTK_OBJECT (local->ecard));
    local->ecard = new_ecard;

    if (coc && coc->type == CARD_DELETED)
        e_book_add_card (ctxt->ebook, local->ecard, add_card_cb, &cons);
    else
        e_book_commit_card (ctxt->ebook, local->ecard, status_cb, &commit_status);
    
    gtk_main (); /* enter sub mainloop */

    /* Adding a record causes wombat to assign a new uid so we must tidy */
    if (coc && coc->type == CARD_DELETED) {
        gboolean arch = e_pilot_map_uid_is_archived (ctxt->map, e_card_get_id (local->ecard));
        
        e_card_set_id (local->ecard, cons.id);
        e_pilot_map_insert (ctxt->map, remote->ID, cons.id, arch);

        coc = g_hash_table_lookup (ctxt->changed_hash, old_id);
        if (coc) {
            g_hash_table_remove (ctxt->changed_hash, e_card_get_id (coc->card));
            coc->card = local->ecard;
            g_hash_table_insert (ctxt->changed_hash, (gpointer)e_card_get_id (coc->card), coc);
            
        }
        
        commit_status = cons.status;
    }
    
    if (commit_status != E_BOOK_STATUS_SUCCESS)
        WARN ("replace_record: failed to update card in ebook\n");

    return retval;
}

static gint
delete_record (GnomePilotConduitSyncAbs *conduit,
           EAddrLocalRecord *local,
           EAddrConduitContext *ctxt)
{
    EBookStatus commit_status;
    int retval = 0;
    
    g_return_val_if_fail (local != NULL, -1);
    g_return_val_if_fail (local->ecard != NULL, -1);

    LOG ("delete_record: delete %s\n", print_local (local));

    e_pilot_map_remove_by_uid (ctxt->map, local->ecard->id);
    e_book_remove_card_by_id (ctxt->ebook, local->ecard->id, status_cb, &commit_status);
    
    gtk_main (); /* enter sub mainloop */
    
    if (commit_status != E_BOOK_STATUS_SUCCESS && commit_status != E_BOOK_STATUS_CARD_NOT_FOUND)
        WARN ("delete_record: failed to delete card in ebook\n");
    
    return retval;
}

static gint
archive_record (GnomePilotConduitSyncAbs *conduit,
        EAddrLocalRecord *local,
        gboolean archive,
        EAddrConduitContext *ctxt)
{
    int retval = 0;
    
    g_return_val_if_fail (local != NULL, -1);

    LOG ("archive_record: %s\n", archive ? "yes" : "no");

    e_pilot_map_insert (ctxt->map, local->local.ID, local->ecard->id, archive);
    
        return retval;
}

static gint
match (GnomePilotConduitSyncAbs *conduit,
       GnomePilotRecord *remote,
       EAddrLocalRecord **local,
       EAddrConduitContext *ctxt)
{
    const char *uid;
    
    LOG ("match: looking for local copy of %s\n",
         print_remote (remote));    
    
    g_return_val_if_fail (local != NULL, -1);
    g_return_val_if_fail (remote != NULL, -1);

    *local = NULL;
    uid = e_pilot_map_lookup_uid (ctxt->map, remote->ID, TRUE);
    
    if (!uid)
        return 0;

    LOG ("  matched\n");
    
    *local = g_new0 (EAddrLocalRecord, 1);
    local_record_from_uid (*local, uid, ctxt);
    
    return 0;
}

static gint
free_match (GnomePilotConduitSyncAbs *conduit,
        EAddrLocalRecord *local,
        EAddrConduitContext *ctxt)
{
    LOG ("free_match: freeing\n");

    g_return_val_if_fail (local != NULL, -1);

    free_local (local);

    return 0;
}

static gint
prepare (GnomePilotConduitSyncAbs *conduit,
     EAddrLocalRecord *local,
     GnomePilotRecord *remote,
     EAddrConduitContext *ctxt)
{
    LOG ("prepare: encoding local %s\n", print_local (local));
    
    *remote = local_record_to_pilot_record (local, ctxt);
    
    return 0;
}

static ORBit_MessageValidationResult
accept_all_cookies (CORBA_unsigned_long request_id,
            CORBA_Principal *principal,
            CORBA_char *operation)
{
    /* allow ALL cookies */
    return ORBIT_MESSAGE_ALLOW_ALL;
}


GnomePilotConduit *
conduit_get_gpilot_conduit (guint32 pilot_id)
{
    GtkObject *retval;
    EAddrConduitContext *ctxt;

    LOG ("in address's conduit_get_gpilot_conduit\n");

    /* we need to find wombat with oaf, so make sure oaf
       is initialized here.  once the desktop is converted
       to oaf and gpilotd is built with oaf, this can go away */
    if (!oaf_is_initialized ()) {
        char *argv[ 1 ] = {"hi"};
        oaf_init (1, argv);

        if (bonobo_init (CORBA_OBJECT_NIL,
                 CORBA_OBJECT_NIL,
                 CORBA_OBJECT_NIL) == FALSE)
            g_error (_("Could not initialize Bonobo"));

        ORBit_set_request_validation_handler (accept_all_cookies);
    }

    retval = gnome_pilot_conduit_sync_abs_new ("AddressDB", 0x61646472);
    g_assert (retval != NULL);

    ctxt = e_addr_context_new (pilot_id);
    gtk_object_set_data (GTK_OBJECT (retval), "addrconduit_context", ctxt);

    gtk_signal_connect (retval, "pre_sync", (GtkSignalFunc) pre_sync, ctxt);
    gtk_signal_connect (retval, "post_sync", (GtkSignalFunc) post_sync, ctxt);

    gtk_signal_connect (retval, "set_pilot_id", (GtkSignalFunc) set_pilot_id, ctxt);
    gtk_signal_connect (retval, "set_status_cleared", (GtkSignalFunc) set_status_cleared, ctxt);

    gtk_signal_connect (retval, "for_each", (GtkSignalFunc) for_each, ctxt);
    gtk_signal_connect (retval, "for_each_modified", (GtkSignalFunc) for_each_modified, ctxt);
    gtk_signal_connect (retval, "compare", (GtkSignalFunc) compare, ctxt);

    gtk_signal_connect (retval, "add_record", (GtkSignalFunc) add_record, ctxt);
    gtk_signal_connect (retval, "replace_record", (GtkSignalFunc) replace_record, ctxt);
    gtk_signal_connect (retval, "delete_record", (GtkSignalFunc) delete_record, ctxt);
    gtk_signal_connect (retval, "archive_record", (GtkSignalFunc) archive_record, ctxt);

    gtk_signal_connect (retval, "match", (GtkSignalFunc) match, ctxt);
    gtk_signal_connect (retval, "free_match", (GtkSignalFunc) free_match, ctxt);

    gtk_signal_connect (retval, "prepare", (GtkSignalFunc) prepare, ctxt);

    return GNOME_PILOT_CONDUIT (retval);
}

void
conduit_destroy_gpilot_conduit (GnomePilotConduit *conduit)
{ 
    EAddrConduitContext *ctxt;

    ctxt = gtk_object_get_data (GTK_OBJECT (conduit), 
                    "addrconduit_context");

    e_addr_context_destroy (ctxt);

    gtk_object_destroy (GTK_OBJECT (conduit));
}