/* $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"
int debug_alarms = 0; /* needed to satisfy some other part of gncal */
/* Default values for alarms */ /* needed to satisfy some other part of gncal */
CalendarAlarm alarm_defaults[4] = {
{ ALARM_MAIL, 0, 15, ALARM_MINUTES },
{ ALARM_PROGRAM, 0, 15, ALARM_MINUTES },
{ ALARM_DISPLAY, 0, 15, ALARM_MINUTES },
{ ALARM_AUDIO, 0, 15, ALARM_MINUTES }
};
GnomePilotConduit * conduit_get_gpilot_conduit (guint32);
void conduit_destroy_gpilot_conduit (GnomePilotConduit*);
void local_record_from_icalobject(GCalLocalRecord *local,iCalObject *obj);
#define CONDUIT_VERSION "0.8.5"
#ifdef G_LOG_DOMAIN
#undef G_LOG_DOMAIN
#endif
#define G_LOG_DOMAIN "gcalconduit"
#define DEBUG_CALCONDUIT
#ifdef DEBUG_CALCONDUIT
#define show_exception(e) g_warning ("Exception: %s\n", CORBA_exception_id (e))
#define LOG(e...) g_log(G_LOG_DOMAIN,G_LOG_LEVEL_MESSAGE, e)
#else
#define show_exception(e)
#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)
static int
start_calendar_server (GnomePilotConduitStandardAbs *conduit,
GCalConduitContext *ctxt)
{
g_return_val_if_fail(conduit!=NULL,-2);
g_return_val_if_fail(ctxt!=NULL,-2);
ctxt->calendar = goad_server_activate_with_id (NULL,
"IDL:GNOME:Calendar:Repository:1.0",
0, NULL);
if (ctxt->calendar == CORBA_OBJECT_NIL) {
g_error ("Can not communicate with GnomeCalendar server");
return -1;
}
if (ctxt->ev._major != CORBA_NO_EXCEPTION){
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
return -1;
}
return 0;
}
/* Just a stub to link with */
void
calendar_notify (time_t time, CalendarAlarm *which, void *data)
{
}
static GSList *
get_calendar_objects(GnomePilotConduitStandardAbs *conduit,
gboolean *status,
GCalConduitContext *ctxt)
{
GSList *result;
GNOME_Calendar_Repository_String_Sequence *uids;
g_return_val_if_fail(conduit!=NULL,NULL);
g_return_val_if_fail(ctxt!=NULL,NULL);
result = NULL;
uids = GNOME_Calendar_Repository_get_object_id_list (ctxt->calendar, &(ctxt->ev));
if (ctxt->ev._major == CORBA_USER_EXCEPTION){
INFO ("Object did not exist");
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
if(status!=NULL) (*status) = FALSE;
return NULL;
} else if(ctxt->ev._major != CORBA_NO_EXCEPTION) {
WARN (_("Error while communicating with calendar server"));
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
if(status!=NULL) (*status) = FALSE;
return NULL;
}
if(status!=NULL) (*status) = TRUE;
if(uids->_length>0) {
int i;
for(i=0;i<uids->_length;i++) {
result = g_slist_prepend(result,g_strdup(uids->_buffer[i]));
}
} else {
INFO ("No entries found");
}
CORBA_free(uids);
return result;
}
#if 0
static GList *
get_calendar_objects(GnomePilotConduitStandardAbs *conduit)
{
char *vcalendar_string;
char *error;
GList *retval,*l;
Calendar *cal;
g_return_val_if_fail(conduit!=NULL,NULL);
vcalendar_string =
GNOME_Calendar_Repository_get_objects (calendar, &ev);
cal = calendar_new("Temporary");
error = calendar_load_from_memory(cal,vcalendar_string);
if (ev._major == CORBA_USER_EXCEPTION){
INFO ("Object did not exist");
show_exception(&ev);
CORBA_exception_free(&ev);
return;
} else if(ev._major != CORBA_NO_EXCEPTION) {
WARN (_("Error while communicating with calendar server"));
show_exception(&ev);
CORBA_exception_free(&ev);
return;
}
if(error != NULL) {
WARN ("Error while converting records");
WARN ("Error : %s",error);
return NULL;
}
retval = NULL;
for(l=cal->events ; l ; l=l->next) {
LOG ("duping %d [%s]",
((iCalObject*)l->data)->pilot_id,
((iCalObject*)l->data)->summary);
retval = g_list_prepend(retval,ical_object_duplicate(l->data));
}
/* g_free(vcalendar_string); FIXME: this coredumps, but won't it leak without ? */
calendar_destroy(cal);
return retval;
}
#endif
static void
local_record_from_ical_uid(GCalLocalRecord *local,
char *uid,
GCalConduitContext *ctxt)
{
iCalObject *obj;
char *vcalendar_string;
g_assert(local!=NULL);
vcalendar_string = GNOME_Calendar_Repository_get_object(ctxt->calendar, uid, &(ctxt->ev));
if (ctxt->ev._major == CORBA_USER_EXCEPTION){
INFO ("Object did not exist");
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
return;
} else if(ctxt->ev._major != CORBA_NO_EXCEPTION) {
WARN (_("Error while communicating with calendar server"));
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
return;
}
g_return_if_fail(vcalendar_string!=NULL);
obj = ical_object_new_from_string (vcalendar_string);
local_record_from_icalobject(local,obj);
return;
}
/*
* converts a iCalObject to a GCalLocalRecord
*/
void
local_record_from_icalobject(GCalLocalRecord *local,
iCalObject *obj)
{
g_return_if_fail(local!=NULL);
g_return_if_fail(obj!=NULL);
local->ical = obj;
local->local.ID = local->ical->pilot_id;
/*
LOG ("local->Id = %ld [%s], status = %d",
local->local.ID,obj->summary,local->ical->pilot_status);
*/
switch(local->ical->pilot_status) {
case ICAL_PILOT_SYNC_NONE:
local->local.attr = GnomePilotRecordNothing;
break;
case ICAL_PILOT_SYNC_MOD:
local->local.attr = GnomePilotRecordModified;
break;
case ICAL_PILOT_SYNC_DEL:
local->local.attr = GnomePilotRecordDeleted;
break;
}
/* Records without a pilot_id are new */
if(local->local.ID == 0)
local->local.attr = GnomePilotRecordNew;
local->local.secret = 0;
if(obj->class!=NULL)
if(strcmp(obj->class,"PRIVATE")==0)
local->local.secret = 1;
local->local.archived = 0;
}
/*
* Given a PilotRecord, find the matching record in
* the calendar repository. If no match, return NULL
*/
static GCalLocalRecord *
find_record_in_repository(GnomePilotConduitStandardAbs *conduit,
PilotRecord *remote,
GCalConduitContext *ctxt)
{
char *vcal_string;
GCalLocalRecord *loc;
g_return_val_if_fail(conduit!=NULL,NULL);
g_return_val_if_fail(remote!=NULL,NULL);
LOG ("requesting %ld", remote->ID);
vcal_string =
GNOME_Calendar_Repository_get_object_by_pilot_id (ctxt->calendar, remote->ID, &(ctxt->ev));
if (ctxt->ev._major == CORBA_USER_EXCEPTION){
INFO ("Object did not exist");
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
return NULL;
} else if(ctxt->ev._major != CORBA_NO_EXCEPTION) {
WARN (_("Error while communicating with calendar server"));
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
return NULL;
} else {
LOG ("Found");
loc = g_new0(GCalLocalRecord,1);
/* memory allocated in new_from_string is freed in free_match */
local_record_from_icalobject(loc,
ical_object_new_from_string (vcal_string));
/* g_free(vcal_string); FIXME: this coredumps, but won't it leak without ? */
return loc;
}
return NULL;
}
/*
* updates an given iCalObject in the repository
*/
static void
update_calendar_entry_in_repository(GnomePilotConduitStandardAbs *conduit,
iCalObject *obj,
GCalConduitContext *ctxt)
{
char *str;
g_return_if_fail(conduit!=NULL);
g_return_if_fail(obj!=NULL);
str = calendar_string_from_object (obj);
GNOME_Calendar_Repository_update_object (ctxt->calendar, obj->uid, str, &(ctxt->ev));
if (ctxt->ev._major == CORBA_USER_EXCEPTION){
INFO ("Object did not exist");
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
return;
} else if(ctxt->ev._major != CORBA_NO_EXCEPTION) {
WARN (_("Error while communicating with calendar server"));
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
return;
}
free (str);
}
static iCalObject *
ical_from_remote_record(GnomePilotConduitStandardAbs *conduit,
PilotRecord *remote,
iCalObject *in_obj)
{
iCalObject *obj;
int i;
struct Appointment a;
time_t now;
now = time (NULL);
g_return_val_if_fail(remote!=NULL,NULL);
memset(&a,0,sizeof(struct Appointment));
unpack_Appointment(&a,remote->record,remote->length);
if (in_obj == NULL)
obj = ical_new (a.note ? a.note : "",
g_get_user_name (),
a.description ? a.description : "");
else
obj = in_obj;
if (a.note) {
g_free(obj->comment);
obj->comment = g_strdup(a.note);
}
if (a.description) {
g_free(obj->summary);
obj->summary = g_strdup(a.description);
}
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;
/*
* Begin and end
*/
if (a.event)
{
/* turn day-long events into a full day's appointment
FIXME: get settings from gnomecal */
a.begin.tm_sec = 0;
a.begin.tm_min = 0;
a.begin.tm_hour = 0;
a.end.tm_sec = 0;
a.end.tm_min =59;
a.end.tm_hour = 23;
}
obj->dtstart = mktime (&a.begin);
obj->dtend = mktime (&a.end);
/* Special case: daily repetitions are converted to a multi-day event */
/* This sucketh, a pilot event scheduled for dailyRepeat, freq 1, end on
whatever is cleary converted wrong
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){
obj->recur = g_new0 (Recurrence, 1);
switch (a.repeatType){
case repeatDaily:
/*
* In the Pilot daily repetitions are actually
* multi-day events
*/
obj->recur->type = RECUR_DAILY;
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_assert_not_reached();
}
if (a.repeatForever)
obj->recur->duration = 0;
else
obj->recur->_enddate = mktime (&a.repeatEnd);
obj->recur->interval = a.repeatFrequency;
}
/*
* 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");
free_Appointment(&a);
return obj;
}
/* Code blatantly stolen from
* calendar-pilot-sync.c:
*
* (C) 1999 International GNOME Support
*
* Author:
* Miguel de Icaza (miguel@gnome-support.com)
*
*/
static gint
update_record (GnomePilotConduitStandardAbs *conduit,
PilotRecord *remote,
GCalConduitContext *ctxt)
{
char *vcal_string;
iCalObject *obj;
struct Appointment a;
g_return_val_if_fail(remote!=NULL,-1);
memset(&a,0,sizeof(struct Appointment));
unpack_Appointment(&a,remote->record,remote->length);
obj = ical_new (a.note ? a.note : "",
g_get_user_name (),
a.description ? a.description : "");
LOG ("requesting %ld [%s]", remote->ID, a.description);
vcal_string = GNOME_Calendar_Repository_get_object_by_pilot_id (ctxt->calendar, remote->ID, &(ctxt->ev));
if (ctxt->ev._major == CORBA_USER_EXCEPTION){
LOG ("Object did not exist, creating a new one");
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
ical_from_remote_record(conduit,remote,obj);
} else if(ctxt->ev._major != CORBA_NO_EXCEPTION) {
WARN (_("Error while communicating with calendar server"));
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
ical_object_destroy (obj);
free_Appointment(&a);
return -1;
} else {
LOG ("Found");
ical_object_destroy (obj);
obj = ical_object_new_from_string (vcal_string);
ical_from_remote_record(conduit,remote,obj);
CORBA_free(vcal_string);
}
/* update record on server */
update_calendar_entry_in_repository(conduit,obj,ctxt);
/*
* Shutdown
*/
ical_object_destroy (obj);
free_Appointment(&a);
return 0;
}
static gint
pre_sync(GnomePilotConduit *c,
GnomePilotDBInfo *dbi,
GCalConduitContext *ctxt)
{
int l;
unsigned char *buf;
g_message ("GnomeCal Conduit v.%s",CONDUIT_VERSION);
ctxt->calendar = CORBA_OBJECT_NIL;
if (start_calendar_server(GNOME_PILOT_CONDUIT_STANDARD_ABS(c),GET_GCALCONTEXT(c)) != 0) {
return -1;
}
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 -1;
}
unpack_AppointmentAppInfo(&(ctxt->ai),buf,l);
g_free(buf);
return 0;
}
/**
* 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,
GCalLocalRecord **local,
PilotRecord *remote,
GCalConduitContext *ctxt)
{
LOG ("in match_record");
g_return_val_if_fail(local!=NULL,-1);
g_return_val_if_fail(remote!=NULL,-1);
*local = find_record_in_repository(conduit,remote,ctxt);
if (*local==NULL) return -1;
return 0;
}
/**
* 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,
GCalLocalRecord **local,
GCalConduitContext *ctxt)
{
LOG ("entering free_match");
g_return_val_if_fail(local!=NULL,-1);
g_return_val_if_fail(*local!=NULL,-1);
ical_object_destroy (GCAL_LOCALRECORD(*local)->ical);
g_free(*local);
*local = NULL;
return 0;
}
/*
Move to archive and set status to Nothing
*/
static gint
archive_local (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord *local,
GCalConduitContext *ctxt)
{
LOG ("entering archive_local");
g_return_val_if_fail(local!=NULL,-1);
return -1;
}
/*
Store in archive and set status to Nothing
*/
static gint
archive_remote (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord *local,
PilotRecord *remote,
GCalConduitContext *ctxt)
{
LOG ("entering archive_remote");
g_return_val_if_fail(remote!=NULL,-1);
g_return_val_if_fail(local!=NULL,-1);
return -1;
}
/*
Store and set status to Nothing
*/
static gint
store_remote (GnomePilotConduitStandardAbs *conduit,
PilotRecord *remote,
GCalConduitContext *ctxt)
{
LOG ("entering store_remote");
g_return_val_if_fail(remote!=NULL,-1);
remote->attr = GnomePilotRecordNothing;
return update_record(conduit,remote,ctxt);
}
static gint
clear_status_archive_local (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord *local,
GCalConduitContext *ctxt)
{
LOG ("entering clear_status_archive_local");
g_return_val_if_fail(local!=NULL,-1);
return -1;
}
static gint
iterate (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord **local,
GCalConduitContext *ctxt)
{
static GSList *events,*iterator;
static int hest;
g_return_val_if_fail(local!=NULL,-1);
if(*local==NULL) {
LOG ("beginning iteration");
events = get_calendar_objects(conduit,NULL,ctxt);
hest = 0;
if(events!=NULL) {
LOG ("iterating over %d records",g_slist_length(events));
*local = g_new0(GCalLocalRecord,1);
local_record_from_ical_uid(*local,(gchar*)events->data,ctxt);
iterator = events;
} else {
LOG ("no events");
(*local) = NULL;
}
} else {
/*LOG ("continuing iteration");*/
hest++;
if(g_slist_next(iterator)==NULL) {
GSList *l;
LOG ("ending");
/** free stuff allocated for iteration */
g_free((*local));
LOG ("iterated over %d records",hest);
for(l=events;l;l=l->next)
g_free(l->data);
g_slist_free(events);
/* ends iteration */
(*local) = NULL;
return 0;
} else {
iterator = g_slist_next(iterator);
local_record_from_ical_uid(*local,(gchar*)(iterator->data),ctxt);
}
}
return 1;
}
static gint
iterate_specific (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord **local,
gint flag,
gint archived,
GCalConduitContext *ctxt)
{
#ifdef DEBUG_CALCONDUIT
{
gchar *tmp;
switch (flag) {
case GnomePilotRecordNothing: tmp = g_strdup("RecordNothing"); break;
case GnomePilotRecordModified: tmp = g_strdup("RecordModified"); break;
case GnomePilotRecordNew: tmp = g_strdup("RecordNew"); break;
default: tmp = g_strdup_printf("0x%x",flag); break;
}
LOG ("entering iterate_specific(flag = %s)",tmp);
g_free(tmp);
}
#endif
g_return_val_if_fail(local!=NULL,-1);
/* iterate until a record meets the criteria */
while(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;
}
return (*local)==NULL?0:1;
}
static gint
purge (GnomePilotConduitStandardAbs *conduit,
GCalConduitContext *ctxt)
{
LOG ("entering purge");
/* HEST, gem posterne her */
return -1;
}
static gint
set_status (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord *local,
gint status,
GCalConduitContext *ctxt)
{
LOG ("entering set_status");
g_return_val_if_fail(local!=NULL,-1);
g_assert(local->ical!=NULL);
local->local.attr = status;
switch(status) {
case GnomePilotRecordPending:
case GnomePilotRecordNothing:
local->ical->pilot_status = ICAL_PILOT_SYNC_NONE;
break;
case GnomePilotRecordDeleted:
break;
case GnomePilotRecordNew:
case GnomePilotRecordModified:
local->ical->pilot_status = ICAL_PILOT_SYNC_MOD;
break;
}
if ( status != GnomePilotRecordDeleted)
GNOME_Calendar_Repository_update_pilot_id(ctxt->calendar,
local->ical->uid,
local->local.ID,
local->ical->pilot_status,
&(ctxt->ev));
else
GNOME_Calendar_Repository_delete_object(ctxt->calendar,local->ical->uid,&(ctxt->ev));
if (ctxt->ev._major == CORBA_USER_EXCEPTION){
LOG ("Object did not exist");
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
return -1;
} else if(ctxt->ev._major != CORBA_NO_EXCEPTION) {
WARN (_("Error while communicating with calendar server"));
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
return -1;
}
return 0;
}
static gint
set_archived (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord *local,
gint archived,
GCalConduitContext *ctxt)
{
LOG ("entering set_archived");
g_return_val_if_fail(local!=NULL,-1);
g_assert(local->ical!=NULL);
local->local.archived = archived;
update_calendar_entry_in_repository(conduit,local->ical,ctxt);
/* FIXME: This should move the entry into a speciel
calendar file, eg. Archive, or (by config option), simply
delete it */
return 0;
}
static gint
set_pilot_id (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord *local,
guint32 ID,
GCalConduitContext *ctxt)
{
LOG ("entering set_pilot_id");
g_return_val_if_fail(local!=NULL,-1);
g_assert(local->ical!=NULL);
local->local.ID = ID;
local->ical->pilot_id = ID;
GNOME_Calendar_Repository_update_pilot_id(ctxt->calendar,
local->ical->uid,
local->local.ID,
local->ical->pilot_status,
&(ctxt->ev));
if (ctxt->ev._major == CORBA_USER_EXCEPTION){
LOG ("Object did not exist");
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
return -1;
} else if(ctxt->ev._major != CORBA_NO_EXCEPTION) {
WARN (_("Error while communicating with calendar server"));
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
return -1;
}
return 0;
}
static gint
transmit (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord *local,
PilotRecord **remote,
GCalConduitContext *ctxt)
{
PilotRecord *p;
int daycount;
LOG ("entering transmit");
g_return_val_if_fail(local!=NULL,-1);
g_return_val_if_fail(remote!=NULL,-1);
g_assert(local->ical!=NULL);
p = g_new0(PilotRecord,1);
p->ID = local->local.ID;
p->attr = local->local.attr;
p->archived = local->local.archived;
p->secret = local->local.secret;
local->a = g_new0(struct Appointment,1);
local->a->event = 0; /* if no start time, leave at 1 */
local->a->begin = *localtime(&local->ical->dtstart);
local->a->end = *localtime(&local->ical->dtend);
/* set the Audio Alarm parameters */
if(local->ical->aalarm.enabled) {
local->a->alarm = 1;
local->a->advance = local->ical->aalarm.count;
switch(local->ical->aalarm.units) {
case ALARM_MINUTES:
local->a->advanceUnits = advMinutes;
break;
case ALARM_HOURS:
local->a->advanceUnits = advHours;
break;
case ALARM_DAYS:
local->a->advanceUnits = advDays;
break;
}
} else {
local->a->alarm = 0;
local->a->advance = 0;
local->a->advanceUnits = advMinutes;
}
/* set the recurrence parameters */
if (local->ical->recur != NULL) {
switch (local->ical->recur->type) {
case RECUR_DAILY:
local->a->repeatType = repeatDaily;
break;
case RECUR_WEEKLY:
local->a->repeatType = repeatWeekly;
break;
case RECUR_MONTHLY_BY_POS:
local->a->repeatType = repeatMonthlyByDate;
break;
case RECUR_MONTHLY_BY_DAY:
local->a->repeatType = repeatMonthlyByDay;
break;
case RECUR_YEARLY_BY_MONTH:
local->a->repeatType = repeatYearly;
break;
case RECUR_YEARLY_BY_DAY:
local->a->repeatType = repeatYearly;
break;
}
if (local->ical->recur->duration == 0) {
local->a->repeatForever = 1;
} else {
local->a->repeatForever = 0;
local->a->repeatEnd = *localtime(&local->ical->recur->_enddate);
}
local->a->repeatFrequency = local->ical->recur->interval;
for ( daycount=0; daycount<7; daycount++ ) {
if (local->ical->recur->weekday & (1 << daycount))
local->a->repeatDays[daycount] = 1;
}
} else {
local->a->repeatType = repeatNone;
local->a->repeatForever = 0;
local->a->repeatEnd = local->a->end;
local->a->repeatFrequency = 0;
local->a->repeatDay = dom1stSun;
local->a->repeatDays[0] = 0;
local->a->repeatDays[1] = 0;
local->a->repeatDays[2] = 0;
local->a->repeatDays[3] = 0;
local->a->repeatDays[4] = 0;
local->a->repeatDays[5] = 0;
local->a->repeatDays[6] = 0;
local->a->repeatWeekstart = 0;
local->a->exceptions = 0;
local->a->exception = NULL;
}
/* STOP: don't replace these with g_strdup, since free_Appointment
uses free to deallocte */
local->a->note =
local->ical->comment==NULL?NULL:strdup(local->ical->comment);
local->a->description =
local->ical->summary==NULL?NULL:strdup(local->ical->summary);
/* Generate pilot record structure */
p->record = g_new0(char,0xffff);
p->length = pack_Appointment(local->a,p->record,0xffff);
#if 0
/* This is some debug code that hexdumps the calendar entry...
You won't need this. */
{
int x,y;
g_message("calconduit: new item from %s to %s",asctime(&(local->a->begin)),asctime(&(local->a->end)));
g_message("local->a->note = %s",local->a->note);
g_message("local->a->description = %s",local->a->description);
g_message("sizeof(p->record) = %d, length is %d",sizeof(p->record),p->length);
for(x=0;x<p->length;x+=32) {
for(y=x;y<x+32;y++)
if(p->record[y]<33 || p->record[y]>128)
printf("%02X",p->record[y]);
else
printf(" %c",p->record[y]);
printf("\n");
}
}
#endif
*remote = p;
return 0;
}
static gint
free_transmit (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord *local,
PilotRecord **remote,
GCalConduitContext *ctxt)
{
LOG ("entering free_transmit");
g_return_val_if_fail(local!=NULL,-1);
g_return_val_if_fail(remote!=NULL,-1);
free_Appointment(local->a);
g_free((*remote)->record);
*remote = NULL;
return 0;
}
static gint
compare (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord *local,
PilotRecord *remote,
GCalConduitContext *ctxt)
{
/* used by the quick compare */
PilotRecord *remoteOfLocal;
int err;
/* used by the tedious compare */
struct Appointment a;
int daycount;
g_message ("entering compare");
g_return_val_if_fail (local!=NULL,-1);
g_return_val_if_fail (remote!=NULL,-1);
#if 1
err = transmit(conduit,local,&remoteOfLocal,ctxt);
if (err != 0) return err;
if (remote->length == remoteOfLocal->length) {
if (memcmp(remoteOfLocal,remote,remote->length)!=0) {
g_message("compare failed on contents");
free_transmit(conduit,local,&remoteOfLocal,ctxt);
return 1;
}
} else {
g_message("compare failed on length");
free_transmit(conduit,local,&remoteOfLocal,ctxt);
return 1;
}
free_transmit(conduit,local,&remoteOfLocal,ctxt);
return 0;
#else
/** FIXME: All the { LOG("yadayada"); return 1; } bloat is for debug purposes.
Once this is known to work, compact to return 1;'s */
/* Check record attributes */
if (local->local.ID != remote->ID) {
LOG("failed local->local.ID == remote->ID");
return 1;
}
if (local->local.attr != remote->attr) {
LOG("failed local->local.attr == remote->attr");
return 1;
}
if (local->local.archived != remote->archived) {
LOG("failed local->local.archived == remote->archived");
return 1;
}
if (local->local.secret != remote->secret) {
LOG("failed local->local.secret == remote->secret");
return 1;
}
unpack_Appointment(&a,remote->record,remote->length);
/* Check records begin/end time */
if (a.event==0) {
/* FIXME
if (a.begin != *localtime(&local->ical->dtstart)) {
LOG("a.begin == *localtime(&local->ical->dtstart)");
return 1;
}
if (a.end != *localtime(&local->ical->dtend)) {
LOG("a.end == *localtime(&local->ical->dtend)");
return 1;
}
*/
} else {
LOG("failed local->a.event != 0, unsupported by gnomecal");
return 1;
}
/* Check records alarm settings */
if(a.alarm == 1) {
if (local->ical->aalarm.enabled == 1) {
if (a.advance != local->ical->aalarm.count) {
LOG("failed a.advance == local->ical->aalarm.count");
return 1;
}
switch(local->ical->aalarm.units) {
case ALARM_MINUTES:
if (a.advanceUnits != advMinutes) {
LOG("failed local->ical->aalarm.units == a.advanceUnits");
return 1;
}
break;
case ALARM_HOURS:
if (a.advanceUnits != advHours) {
LOG("failed local->ical->aalarm.units == a.advanceUnits");
return 1;
}
break;
case ALARM_DAYS:
if (a.advanceUnits != advDays) {
LOG("failed local->ical->aalarm.units == a.advanceUnits");
return 1;
}
break;
}
} else {
LOG("failed a.alarm == 1 && local->ical->aalarm.enabled == 1");
return 1;
}
} else if (local->ical->aalarm.enabled == 1) {
LOG("failed a.alarm != 1 && local->ical->aalarm.enabled != 1");
return 1;
}
/* Check records recurrence settings */
/* If this code is broken, a more or less safe although not efficient
approach is (other the fixing the bug), if either has recurrence,
return 1, thus failing the comparision */
if (local->ical->recur != NULL) {
if (a.repeatType == repeatNone) {
LOG("failed: local->ical->recur != NULL && a.repeatType != repeatNone");
return 1;
}
switch (local->ical->recur->type) {
case RECUR_DAILY:
if (a.repeatType != repeatDaily) {
LOG("failed a.repeatType == repeatDaily");
return 1; }
break;
case RECUR_WEEKLY:
if (a.repeatType != repeatWeekly) {
LOG("failed a.repeatType == repeatWeekly");
return 1; }
break;
case RECUR_MONTHLY_BY_POS:
if (a.repeatType != repeatMonthlyByDate) {
LOG("failed a.repeatType == repeatMonthlyByDate");
return 1; }
break;
case RECUR_MONTHLY_BY_DAY:
if (a.repeatType != repeatMonthlyByDay) {
LOG("failed a.repeatType == repeatMonthlyByDay");
return 1; }
break;
case RECUR_YEARLY_BY_MONTH:
if (a.repeatType != repeatYearly) {
LOG("failed a.repeatType == repeatYearly");
return 1; }
break;
case RECUR_YEARLY_BY_DAY:
if (a.repeatType != repeatYearly) {
LOG("failed a.repeatType == repeatYearly");
return 1; }
break;
}
if (local->ical->recur->duration == 0) {
if(a.repeatForever != 1) {
LOG("failed local->ical->recur->duration == 0 && a.repeatForever == 1");
return 1;
}
} else {
if(a.repeatForever != 0) {
LOG("failed local->ical->recur->duration != 0 && ! a.repeatForever == 0");
return 1;
}
/* FIXME
if(a.repeatEnd != *localtime(&local->ical->recur->_enddate)) {
LOG("failed a.repeatEnd == *localtime(&local->ical->recur->_enddate)");
return 1;
}
*/
}
if (a.repeatFrequency != local->ical->recur->interval) {
LOG("failed a.repeatFrequency == local->ical->recur->interval");
return 1;
}
for (daycount = 0; daycount<7; daycount++) {
if(local->ical->recur->weekday & (1<<daycount)) {
if (a.repeatDays[daycount]!=1) {
LOG("failed local->ical->recur->weekday & (1<<daycount) && a.repeatDays[daycount]==1");
return 1;
}
} else {
if (a.repeatDays[daycount]!=0) {
LOG("failed local->ical->recur->weekday &! (1<<daycount) && a.repeatDays[daycount]==0");
return 1;
}
}
}
} else if (a.repeatType != repeatNone ) {
LOG("failed: local->ical->recur == NULL && a.repeatType == repeatNone");
return 1;
}
/* check the note and description */
if(a.note!=NULL) {
if(local->ical->comment==NULL) {
LOG("failed a.note != NULL && local->ical->coment != NULL");
return 1;
}
if(strcmp(local->ical->comment,a.note)!=0) {
LOG("failed strcmp(local->ical->comment,a.note)==0");
return 1;
}
} if(local->ical->comment!=NULL) {
LOG("failed a.note == NULL && local->ical->coment == NULL");
return 1;
}
if(a.description!=NULL) {
if(local->ical->summary==NULL) {
LOG("failed a.description != NULL && local->ical->coment != NULL");
return 1;
}
if(strcmp(local->ical->summary,a.description)!=0) {
LOG("failed strcmp(local->ical->summary,a.description)==0");
return 1;
}
} if(local->ical->summary!=NULL) {
LOG("failed a.description == NULL && local->ical->coment == NULL");
return 1;
}
#endif
return 0;
}
static gint
compare_backup (GnomePilotConduitStandardAbs *conduit,
GCalLocalRecord *local,
PilotRecord *remote,
GCalConduitContext *ctxt)
{
LOG ("entering compare_backup");
g_return_val_if_fail(local!=NULL,-1);
g_return_val_if_fail(remote!=NULL,-1);
return -1;
}
static gint
delete_all (GnomePilotConduitStandardAbs *conduit,
GCalConduitContext *ctxt)
{
GSList *events,*it;
gboolean error;
events = get_calendar_objects(conduit,&error,ctxt);
if (error == FALSE) return -1;
for (it=events;it;it = g_slist_next(it)) {
GNOME_Calendar_Repository_delete_object(ctxt->calendar,
it->data,
&(ctxt->ev));
if (ctxt->ev._major == CORBA_USER_EXCEPTION){
INFO ("Object did not exist");
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
} else if(ctxt->ev._major != CORBA_NO_EXCEPTION) {
WARN (_("Error while communicating with calendar server"));
show_exception(&(ctxt->ev));
CORBA_exception_free(&(ctxt->ev));
/* destroy loop, free data */
for (it=events;it;it = g_slist_next(it)) g_free(it->data);
g_slist_free(events);
return -1;
}
g_free(it->data);
}
g_slist_free(events);
return -1;
}
GnomePilotConduit *
conduit_get_gpilot_conduit (guint32 pilotId)
{
GtkObject *retval;
GCalConduitCfg *cfg;
GCalConduitContext *ctxt;
retval = gnome_pilot_conduit_standard_abs_new ("DatebookDB", 0x64617465);
g_assert (retval != NULL);
gnome_pilot_conduit_construct(GNOME_PILOT_CONDUIT(retval),"GnomeCalConduit");
gcalconduit_load_configuration(&cfg,pilotId);
gtk_object_set_data(retval,"gcalconduit_cfg",cfg);
gcalconduit_new_context(&ctxt,cfg);
/* No real need to set it, since all signal are given this
as their user data */
gtk_object_set_data(retval,"gcalconduit_context",ctxt);
gtk_signal_connect (retval, "match_record", (GtkSignalFunc) match_record, ctxt);
gtk_signal_connect (retval, "free_match", (GtkSignalFunc) free_match, ctxt);
gtk_signal_connect (retval, "archive_local", (GtkSignalFunc) archive_local, ctxt);
gtk_signal_connect (retval, "archive_remote", (GtkSignalFunc) archive_remote, ctxt);
gtk_signal_connect (retval, "store_remote", (GtkSignalFunc) store_remote, ctxt);
gtk_signal_connect (retval, "clear_status_archive_local", (GtkSignalFunc) clear_status_archive_local, ctxt);
gtk_signal_connect (retval, "iterate", (GtkSignalFunc) iterate, ctxt);
gtk_signal_connect (retval, "iterate_specific", (GtkSignalFunc) iterate_specific, ctxt);
gtk_signal_connect (retval, "purge", (GtkSignalFunc) purge, ctxt);
gtk_signal_connect (retval, "set_status", (GtkSignalFunc) set_status, ctxt);
gtk_signal_connect (retval, "set_archived", (GtkSignalFunc) set_archived, ctxt);
gtk_signal_connect (retval, "set_pilot_id", (GtkSignalFunc) set_pilot_id, ctxt);
gtk_signal_connect (retval, "compare", (GtkSignalFunc) compare, ctxt);
gtk_signal_connect (retval, "compare_backup", (GtkSignalFunc) compare_backup, ctxt);
gtk_signal_connect (retval, "free_transmit", (GtkSignalFunc) free_transmit, ctxt);
gtk_signal_connect (retval, "delete_all", (GtkSignalFunc) delete_all, ctxt);
gtk_signal_connect (retval, "transmit", (GtkSignalFunc) transmit, ctxt);
gtk_signal_connect (retval, "pre_sync", (GtkSignalFunc) pre_sync, ctxt);
return GNOME_PILOT_CONDUIT (retval);
}
void
conduit_destroy_gpilot_conduit (GnomePilotConduit *conduit)
{
GCalConduitCfg *cc;
GCalConduitContext *ctxt;
cc = GET_GCALCONFIG(conduit);
ctxt = GET_GCALCONTEXT(conduit);
if(ctxt->calendar!=CORBA_OBJECT_NIL)
GNOME_Calendar_Repository_done (ctxt->calendar, &(ctxt->ev));
gcalconduit_destroy_configuration(&cc);
gcalconduit_destroy_context(&ctxt);
gtk_object_destroy (GTK_OBJECT (conduit));
}