aboutsummaryrefslogblamecommitdiffstats
path: root/calendar/calendar-conduit.c
blob: 929527264f61468d9419a80b9bb552ad89942d49 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

















                       



                                    


                                                     


                     
                     


                             


                                                         










                                                                                     
              



                                

                                                                                     




                                                              
                            




                  






                                                               











                                                                         
                                                                                   












































                                                                                 

























                                                             

                                                                    
























                                                                                                   
          




                                                                                                                 
          












































































































































                                                                                







                                  
                                                          

                   
                     











                                                              






                                    





                                                                                              
             



                                                      
           
 
 







                                                    
           

                                                       


                                     
                                       




                                                
 





                                                                 
           

                                                       


                                          


                                                            
                      
                 
 

 
           

                                                     





                                             
 
           

                                                      





                                              
 
           
                                                    


                                  





                                       
 
 
           

                                                                  




                                                          
 
           

                                               

                       
























                                                                                  
 
 
           

                                                        



                                






                                                                           
                                           


                          
 
 
           
                                             

                     




                               
 
 
           

                                                  



                                          
















                                                           
                 
 
 
           

                                                    



                                            






                                                                    

                 
 
           

                                                    



                                            




                                            

                 
 
           

                                               





                                       
 
           

                                                      





                                              
 
           

                                                     





                                             
 
           
                                                  




                                          
 
                    

                                                





                                        
                   


                                            



                                   


                                                                                 







                                                                              

                                                         
















                                                                                                                    
                                                                                

                                         


                                            
 
    
                                                           



                        

                                                       





                                                    
                                                  
 


 
/* $Id$ */

#include <glib.h>
#include <gnome.h>

#include <pi-source.h>
#include <pi-socket.h>
#include <pi-file.h>
#include <pi-dlp.h>
#include <pi-version.h>

#include <sys/stat.h>
#include <sys/types.h>
#include <utime.h>
#include <unistd.h>
#include <pwd.h>
#include <signal.h>
#include <errno.h>

#include <libgnorba/gnorba.h>
#include <libgnorba/gnome-factory.h>

#include <gpilotd/gnome-pilot-conduit.h>
#include <gpilotd/gnome-pilot-conduit-standard-abs.h>

#include "GnomeCal.h"
#include "calobj.h"
#include "calendar.h"
#include "timeutil.h"

#include "calendar-conduit.h"

GnomePilotConduit * conduit_get_gpilot_conduit (guint32);
void conduit_destroy_gpilot_conduit (GnomePilotConduit*);

typedef struct _ConduitData ConduitData;

struct _ConduitData {
  struct AppointmentAppInfo ai;
  Calendar *cal;
};

#define GET_DATA(c) ((ConduitData*)gtk_object_get_data(GTK_OBJECT(c),"conduit_data"))

GNOME_Calendar_Repository calendar;
CORBA_Environment ev;
CORBA_ORB orb;

static GNOME_Calendar_Repository
calendar_server (void)
{
  calendar = goad_server_activate_with_id (NULL, "IDL:GNOME:Calendar:Repository:1.0",
                       0, NULL);
  if (calendar == CORBA_OBJECT_NIL)
    g_error ("Can not communicate with GnomeCalendar server");
  
  if (ev._major != CORBA_NO_EXCEPTION){
    printf ("Exception: %s\n", CORBA_exception_id (&ev));
    return CORBA_OBJECT_NIL;
  }
  
  return calendar;
}


/* Just a stub to link with */
void
calendar_notify (time_t time, CalendarAlarm *which, void *data)
{
}

static void
local_from_ical(CalLocalRecord **local,iCalObject *obj) {
  g_return_if_fail(local!=NULL);
  g_return_if_fail(*local!=NULL);
  g_return_if_fail(obj!=NULL);

  (*local)->ical = obj;
  (*local)->ID = (*local)->ical->pilot_id;
  
  g_message("(*local)->Id = %ld",(*local)->ID);
  switch((*local)->ical->pilot_status) {
  case ICAL_PILOT_SYNC_NONE: (*local)->local.attr = RecordNothing; break;
  case ICAL_PILOT_SYNC_MOD: (*local)->local.attr = RecordNew|RecordModified; break;
  case ICAL_PILOT_SYNC_DEL: (*local)->local.attr = RecordDeleted; break;
  }
  
  (*local)->local.secret = 0;
  if(obj->class!=NULL) 
    if(strcmp(obj->class,"PRIVATE")==0)
      (*local)->local.secret = 1;
 
  (*local)->local.archived = 0;  
  
  /* used by iterations */
  (*local)->list_ptr = NULL;
}

static CalLocalRecord *
match_record_from_repository(PilotRecord *remote) {
  char *vcal_string;
  CalLocalRecord *loc;
  
  g_return_val_if_fail(remote!=NULL,NULL);
  
  printf ("requesting %ld []\n", remote->ID);
 
  /* FIXME: ehm, who frees this string ? */
  vcal_string = 
    GNOME_Calendar_Repository_get_object_by_pilot_id (calendar, remote->ID, &ev);
  
  if (ev._major == CORBA_USER_EXCEPTION){
    printf (_("\tObject did not exist\n"));
    return NULL;
  } else if(ev._major != CORBA_NO_EXCEPTION) {
    printf(_("\tError while communicating with calendar server\n"));
    CORBA_exception_free(&ev); 
    return NULL;
  } else {
    printf ("\tFound\n");
    loc = g_new0(CalLocalRecord,1);
    /* memory allocated in new_from_string is freed in free_match */
    local_from_ical(&loc,ical_object_new_from_string (vcal_string));
    return loc;
  }

  return NULL;
}

/* Code blatantly stolen from
 * calendar-pilot-sync.c:
 *   
 * (C) 1999 International GNOME Support
 *
 * Author:
 *   Miguel de Icaza (miguel@gnome-support.com)
 *
 */
static void
update_record (PilotRecord *remote) 
{
    char *vcal_string;
    iCalObject *obj;
    int i;
    char *str;
    struct Appointment a;

    g_return_if_fail(remote!=NULL);

    unpack_Appointment(&a,remote->record,remote->length);
    
    obj = ical_new (a.note ? a.note : "",
            g_get_user_name (),
            a.description ? a.description : "");

    printf ("requesting %ld [%s]\n", remote->ID, a.description);
    /* FIXME: ehm, who frees this string ? */
    vcal_string = GNOME_Calendar_Repository_get_object_by_pilot_id (calendar, remote->ID, &ev);

    if (ev._major == CORBA_USER_EXCEPTION){
        time_t now = time (NULL);
        
        obj->created = now;
        obj->last_mod = now;
        obj->priority = 0;
        obj->transp = 0;
        obj->related = NULL;
        obj->pilot_id = remote->ID;
        obj->pilot_status = ICAL_PILOT_SYNC_NONE;
        printf (_("\tObject did not exist, creating a new one\n"));
    } else if(ev._major != CORBA_NO_EXCEPTION) {
      printf(_("\tError while communicating with calendar server\n"));
      printf("\texception id = %s\n",CORBA_exception_id(&ev));
      CORBA_exception_free(&ev); 
      ical_object_destroy (obj); 
      return;
    } else {
        printf ("\tFound\n");
        ical_object_destroy (obj);
        obj = ical_object_new_from_string (vcal_string);
    }

    /*
    if (obj->pilot_status == ICAL_PILOT_SYNC_MOD){
        printf (_("\tObject has been modified on desktop and on the pilot, desktop takes precedence\n"));
        ical_object_destroy (obj);
        return;
    }
    */

    /*
     * Begin and end
     */

    if (a.event)
    {
        /* turn day-long events into a full day's appointment */
        a.begin.tm_sec = 0;
        a.begin.tm_min = 0;
        a.begin.tm_hour = 6;

        a.end.tm_sec = 0;
        a.end.tm_min = 0;
        a.end.tm_hour = 10;
    }
    
    obj->dtstart = mktime (&a.begin);
    obj->dtend = mktime (&a.end);

    /* Special case: daily repetitions are converted to a multi-day event */
    if (a.repeatType == repeatDaily){
        time_t newt = time_add_day (obj->dtend, a.repeatFrequency);

        obj->dtend = newt;
    }

    /*
     * Alarm
     */
    if (a.alarm){
        obj->aalarm.type = ALARM_AUDIO;
        obj->aalarm.enabled = 1;
        obj->aalarm.count = a.advance;

        switch (a.advanceUnits){
        case advMinutes:
            obj->aalarm.units = ALARM_MINUTES;
            break;
            
        case advHours:
            obj->aalarm.units = ALARM_HOURS;
            break;
            
        case advDays:
            obj->aalarm.units = ALARM_DAYS;
            break;
        default:
        }
    }

    /*
     * Recurrence
     */
    if (a.repeatFrequency && a.repeatType != repeatDaily){
        obj->recur = g_new0 (Recurrence, 1);
        
        switch (a.repeatType){
        case repeatDaily:
            /*
             * In the Pilot daily repetitions are actually
             * multi-day events
             */
            g_warning ("Should not have got here");
            break;
            
        case repeatMonthlyByDate:
            obj->recur->type = RECUR_MONTHLY_BY_DAY;
            obj->recur->u.month_day = a.repeatFrequency;
            break;
            
        case repeatWeekly:
        {
            int wd;

            obj->recur->type = RECUR_WEEKLY;
            for (wd = 0; wd < 7; wd++)
                if (a.repeatDays [wd])
                    obj->recur->weekday |= 1 << wd;

            if (obj->recur->weekday == 0){
                struct tm *tm = localtime (&obj->dtstart);

                obj->recur->weekday = 1 << tm->tm_wday;
            }
            break;
        }
        
        case repeatMonthlyByDay:
            obj->recur->type = RECUR_MONTHLY_BY_POS;
            obj->recur->u.month_pos = a.repeatFrequency;
            obj->recur->weekday = (a.repeatDay / 7);
            break;
            
        case repeatYearly:
            obj->recur->type = RECUR_YEARLY_BY_DAY;
            break;

        default:
            g_warning ("Unhandled repeate case");
        }

        if (a.repeatForever)
            obj->recur->duration = 0;
        else
            obj->recur->_enddate = mktime (&a.repeatEnd);
    }

    /*
     * Load exception dates 
     */
    obj->exdate = NULL;
    for (i = 0; i < a.exceptions; i++){
        time_t *t = g_new (time_t, 1);

        *t = mktime (&(a.exception [i]));
        obj->exdate = g_list_prepend (obj->exdate, t);
    }

    g_free (obj->class);
    
    if (remote->attr & dlpRecAttrSecret)
        obj->class = g_strdup ("PRIVATE");
    else
        obj->class = g_strdup ("PUBLIC");

    /*
     * Now, convert the in memory iCalObject to a full vCalendar we can send
     */
    str = calendar_string_from_object (obj);

    GNOME_Calendar_Repository_update_object (calendar, obj->uid, str, &ev);

    free (str);
    
    /*
     * Shutdown
     */
    ical_object_destroy (obj);
}

static gint
load_records(GnomePilotConduit *c)
{
  char *vcalendar_string;
  char *error;
  ConduitData *cd;

  vcalendar_string = 
    GNOME_Calendar_Repository_get_objects (calendar, &ev);

  cd = GET_DATA(c);
  g_assert(cd!=NULL);
  cd->cal = calendar_new("Temporary");

  error = calendar_load_from_memory(cd->cal,vcalendar_string);

  return 0;
}

static gint
pre_sync(GnomePilotConduit *c, GnomePilotDBInfo *dbi) 
{
  int l;
  unsigned char *buf;
  
  calendar = CORBA_OBJECT_NIL;
  calendar = calendar_server();
  if(calendar == CORBA_OBJECT_NIL) {
    return 0;
  }
  
  gtk_object_set_data(GTK_OBJECT(c),"dbinfo",dbi);
  
  load_records(c);

  buf = (unsigned char*)g_malloc(0xffff);
  if((l=dlp_ReadAppBlock(dbi->pilot_socket,dbi->db_handle,0,(unsigned char *)buf,0xffff))<0) {
    return 0;
  }
  unpack_AppointmentAppInfo(&(GET_DATA(c)->ai),buf,l);
  g_free(buf);

  return 1;
}

/**
 * Find (if possible) the local record which matches
 * the given PilotRecord.
 * if successfull, return non-zero and set *local to
 * a non-null value (the located local record),
 * otherwise return 0 and set *local = NULL;
 */

static gint
match_record    (GnomePilotConduitStandardAbs *conduit,
         CalLocalRecord **local,
         PilotRecord *remote,
         gpointer data)
{
  g_return_val_if_fail(remote!=NULL,0);
  g_print ("in match_record\n");

  *local = match_record_from_repository(remote);
  
  return 1;
}

/**
 * Free the data allocated by a previous match_record call.
 * If successfull, return non-zero and ser *local=NULL, otherwise
 * return 0.
 */
static gint
free_match  (GnomePilotConduitStandardAbs *conduit,
         CalLocalRecord **local,
         gpointer data)
{
        g_print ("entering free_match\n");
    ical_object_destroy (CALLOCALRECORD(*local)->ical); 
    g_free(*local);
    
        *local = NULL;
    return 1;
}


static gint
archive_local (GnomePilotConduitStandardAbs *conduit,
           CalLocalRecord *local,
           gpointer data)
{
    g_print ("entering archive_local\n");
    return 1;

}

static gint
archive_remote (GnomePilotConduitStandardAbs *conduit,
        CalLocalRecord *local,
        PilotRecord *remote,
        gpointer data)
{
    g_print ("entering archive_remote\n");
    return 1;
}

static gint
store_remote (GnomePilotConduitStandardAbs *conduit,
          PilotRecord *remote,
          gpointer data)
{
  g_print ("entering store_remote\n");
  g_return_val_if_fail(remote!=NULL,0);

  update_record(remote);
  
  return 1;
}

static gint
clear_status_archive_local (GnomePilotConduitStandardAbs *conduit,
                CalLocalRecord *local,
                gpointer data)
{
    g_print ("entering clear_status_archive_local\n");
        return 1;
}

static gint
iterate (GnomePilotConduitStandardAbs *conduit,
     CalLocalRecord **local,
     gpointer data)
{
  g_return_val_if_fail(local!=NULL,0);
  
  if(*local==NULL) {
    g_message("calconduit: beginning iteration");
    if(GET_DATA(conduit)->cal->events!=NULL) {
      *local = g_new0(CalLocalRecord,1);

      local_from_ical(local,(iCalObject*)GET_DATA(conduit)->cal->events->data);
      (*local)->list_ptr = GET_DATA(conduit)->cal->events;
    } else {
      g_message("calconduit: no events");
      (*local) = NULL;
    }
  } else {
    g_message("calconduit: continuing iteration");
    if(g_list_next((*local)->list_ptr)==NULL) {
      g_message("calconduit: ending");
      g_free((*local));
      (*local) = NULL; /* ends iteration */
    } else {
      local_from_ical(local,(iCalObject*)(g_list_next((*local)->list_ptr)->data));
      (*local)->list_ptr = g_list_next((*local)->list_ptr);
    }
  }
  return 1;
}

static gint
iterate_specific (GnomePilotConduitStandardAbs *conduit,
          CalLocalRecord **local,
          gint flag,
          gint archived,
          gpointer data)
{
  g_return_val_if_fail(local!=NULL,0);

  g_print ("entering iterate_specific\n");
  do {
    gnome_pilot_conduit_standard_abs_iterate(conduit,(LocalRecord**)local);
    if((*local)==NULL) break;
    if(archived && ((*local)->local.archived==archived)) break;
    if((*local)->local.attr |= flag) break;
  } while((*local)!=NULL);

  return 1;
}

static gint
purge (GnomePilotConduitStandardAbs *conduit,
       gpointer data)
{
  g_print ("entering purge\n");

  /* HEST, gem posterne her */

  return 1;
}

static gint
set_status (GnomePilotConduitStandardAbs *conduit,
        CalLocalRecord *local,
        gint status,
        gpointer data)
{
    g_print ("entering set_status\n");
    g_return_val_if_fail(local!=NULL,0);
    g_assert(local->ical!=NULL);
    
    local->local.attr = status;
    switch(status) {
    case RecordPending:
    case RecordNothing:
      local->ical->pilot_status = ICAL_PILOT_SYNC_NONE;
      break;
    case RecordDeleted:
      local->ical->pilot_status = ICAL_PILOT_SYNC_DEL;
      break;
    case RecordNew:
    case RecordModified:
      local->ical->pilot_status = ICAL_PILOT_SYNC_MOD;
      break;      
    }
        return 0;
}

static gint
set_archived (GnomePilotConduitStandardAbs *conduit,
          CalLocalRecord *local,
          gint archived,
          gpointer data)
{
    g_print ("entering set_archived\n");
    g_return_val_if_fail(local!=NULL,0);
    g_assert(local->ical!=NULL);

    local->local.archived = archived;
    /* FIXME: This should move the entry into a speciel
       calendar file, eg. Archive, or (by config option), simply
       delete it */
        return 1;
}

static gint
set_pilot_id (GnomePilotConduitStandardAbs *conduit,
          CalLocalRecord *local,
          guint32 ID,
          gpointer data)
{
    g_print ("entering set_pilot_id\n");
    g_return_val_if_fail(local!=NULL,0);
    g_assert(local->ical!=NULL);

    local->ID = ID;
    local->ical->pilot_id = ID;
        return 1;
}

static gint
compare (GnomePilotConduitStandardAbs *conduit,
     CalLocalRecord *local,
     PilotRecord *remote,
     gpointer data)
{
    g_print ("entering compare\n");
        return 1;
}

static gint
compare_backup (GnomePilotConduitStandardAbs *conduit,
        CalLocalRecord *local,
        PilotRecord *remote,
        gpointer data)
{
    g_print ("entering compare_backup\n");
        return 1;
}

static gint
free_transmit (GnomePilotConduitStandardAbs *conduit,
           CalLocalRecord *local,
           PilotRecord *remote,
           gpointer data)
{
    g_print ("entering free_transmit\n");
        return 1;
}

static gint
delete_all (GnomePilotConduitStandardAbs *conduit,
        gpointer data)
{
    g_print ("entering delete_all\n");
        return 1;
}

static PilotRecord *
transmit (GnomePilotConduitStandardAbs *conduit,
      CalLocalRecord *local,
      gpointer data)
{
    g_print ("entering transmit\n");
    return NULL;
}

GnomePilotConduit *
conduit_get_gpilot_conduit (guint32 pilotId)
{
    GtkObject *retval;
    ConduitCfg *cfg;
    ConduitData *cdata;

    CORBA_exception_init (&ev);

    retval = gnome_pilot_conduit_standard_abs_new ("DatebookDB", 0x64617465);
    g_assert (retval != NULL);
    gnome_pilot_conduit_construct(GNOME_PILOT_CONDUIT(retval),"calendar");

    cfg = g_new0(ConduitCfg,1);
    g_assert(cfg != NULL);
    gtk_object_set_data(retval,"conduit_cfg",cfg);

    cdata = g_new0(ConduitData,1);
    g_assert(cdata != NULL);
    gtk_object_set_data(retval,"conduit_data",cdata);

    gtk_signal_connect (retval, "match_record", (GtkSignalFunc) match_record, NULL);
    gtk_signal_connect (retval, "free_match", (GtkSignalFunc) free_match, NULL);
    gtk_signal_connect (retval, "archive_local", (GtkSignalFunc) archive_local, NULL);
    gtk_signal_connect (retval, "archive_remote", (GtkSignalFunc) archive_remote, NULL);
    gtk_signal_connect (retval, "store_remote", (GtkSignalFunc) store_remote, NULL);
    gtk_signal_connect (retval, "clear_status_archive_local", (GtkSignalFunc) clear_status_archive_local, NULL);
    gtk_signal_connect (retval, "iterate", (GtkSignalFunc) iterate, NULL);
    gtk_signal_connect (retval, "iterate_specific", (GtkSignalFunc) iterate_specific, NULL);
    gtk_signal_connect (retval, "purge", (GtkSignalFunc) purge, NULL);
    gtk_signal_connect (retval, "set_status", (GtkSignalFunc) set_status, NULL);
    gtk_signal_connect (retval, "set_archived", (GtkSignalFunc) set_archived, NULL);
    gtk_signal_connect (retval, "set_pilot_id", (GtkSignalFunc) set_pilot_id, NULL);
    gtk_signal_connect (retval, "compare", (GtkSignalFunc) compare, NULL);
    gtk_signal_connect (retval, "compare_backup", (GtkSignalFunc) compare_backup, NULL);
    gtk_signal_connect (retval, "free_transmit", (GtkSignalFunc) free_transmit, NULL);
    gtk_signal_connect (retval, "delete_all", (GtkSignalFunc) delete_all, NULL);
    gtk_signal_connect (retval, "transmit", (GtkSignalFunc) transmit, NULL);
    gtk_signal_connect (retval, "pre_sync", (GtkSignalFunc) pre_sync, NULL);

    load_configuration(&cfg,pilotId);

    return GNOME_PILOT_CONDUIT (retval);
}

void
conduit_destroy_gpilot_conduit (GnomePilotConduit *conduit)
{ 
        ConduitCfg *cc;
    ConduitData *cd;

    GNOME_Calendar_Repository_done (calendar, &ev);

        cc = GET_CONFIG(conduit);
        destroy_configuration(&cc);

    cd = GET_DATA(conduit);
    if(cd->cal!=NULL) calendar_destroy(cd->cal);

    gtk_object_destroy (GTK_OBJECT (conduit));

}