aboutsummaryrefslogblamecommitdiffstats
path: root/calendar/gui/e-pub-utils.c
blob: f7191c0708957b34a6a71848a1ecbe6cbbc0286b (plain) (tree)































































































































































































































































































                                                                                                                                


                                                                               



















                                                                              
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */

/* Evolution calendar Free/Busy utilities and types
 *
 * Copyright (C) 2004 Ximian, Inc.
 *
 * Author: Gary Ekker <gekker@novell.com>
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of version 2 of the GNU General Public
 * License as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA.
 */

#include <string.h>
#include <glib.h>
#include <gconf/gconf.h>
#include <gconf/gconf-client.h>
#include <libedataserver/e-source.h>
#include <libedataserver/e-source-list.h>
#include <libecal/e-cal-time-util.h>
#include "calendar-config.h"
#include "common/authentication.h"
#include "itip-utils.h"
#include "e-pub-utils.h"

void
e_pub_uri_from_xml (EPublishUri *uri, const gchar *xml)
{
    xmlDocPtr doc;
    xmlNodePtr root, p;
    xmlChar *location, *enabled, *frequency;
    xmlChar *username, *password, *publish_time;
    GSList *l = NULL;
    
    uri->location = NULL;
    doc = xmlParseDoc ((char *)xml);
    if (doc == NULL) {
        uri->location = NULL;
        return;
    }
    
    root = doc->children;
    if (strcmp (root->name, "uri") != 0) {
        return;
    }
    location = xmlGetProp (root, "location");
    enabled = xmlGetProp (root, "enabled");
    frequency = xmlGetProp (root, "frequency");
    username = xmlGetProp (root, "username");
    password = xmlGetProp (root, "password");
    publish_time = xmlGetProp (root, "publish_time");
    
    if (location != NULL)
        uri->location = g_strdup (location);
    if (enabled != NULL)
        uri->enabled = atoi (enabled);
    if (frequency != NULL)
        uri->publish_freq = atoi (frequency);
    if (username != NULL)
        uri->username = g_strdup (username);
    if (password != NULL)
        uri->password = g_strdup (password);
    if (publish_time != NULL)
        uri->last_pub_time = g_strdup (publish_time);
    
    for (p = root->children; p != NULL; p = p->next) {
        xmlChar *uid = xmlGetProp (p, "uid");

        l = g_slist_append (l, uid);
    }   
    uri->calendars = l;
    
    xmlFree(location);
    xmlFree(enabled);
    xmlFreeDoc(doc);
    return;
}

gchar *
e_pub_uri_to_xml (EPublishUri *uri)
{
    xmlDocPtr doc;
    xmlNodePtr root;
    gchar *enabled, *frequency;
    GSList *cals = NULL;
    xmlChar *xml_buffer;
    char *returned_buffer;
    int xml_buffer_size;

    g_return_val_if_fail (uri != NULL, NULL);
    g_return_val_if_fail (uri->location != NULL, NULL);

    doc = xmlNewDoc ("1.0");

    root = xmlNewDocNode (doc, NULL, "uri", NULL);
    enabled = g_strdup_printf ("%d", uri->enabled);
    frequency = g_strdup_printf ("%d", uri->publish_freq);
    xmlSetProp (root, "location", uri->location);
    xmlSetProp (root, "enabled", enabled);
    xmlSetProp (root, "frequency", frequency);
    xmlSetProp (root, "username", uri->username);
    xmlSetProp (root, "password", uri->password);
    xmlSetProp (root, "publish_time", uri->last_pub_time);
    
    for (cals = uri->calendars; cals != NULL; cals = cals->next) {
        xmlNodePtr node;
        
        node = xmlNewChild (root, NULL, "source", NULL);
        xmlSetProp (node, "uid", cals->data);
    }
    xmlDocSetRootElement (doc, root);

    xmlDocDumpMemory (doc, &xml_buffer, &xml_buffer_size);
    xmlFreeDoc (doc);

    returned_buffer = g_malloc (xml_buffer_size + 1);
    memcpy (returned_buffer, xml_buffer, xml_buffer_size);
    returned_buffer [xml_buffer_size] = '\0';
    xmlFree (xml_buffer);
    g_free (enabled);

    return returned_buffer;
}

static gboolean 
is_publish_time (EPublishUri *uri) {
    icaltimezone *utc;
    struct icaltimetype current_itt, adjust_itt;

    if (!uri->last_pub_time) {
        utc = icaltimezone_get_utc_timezone ();
        current_itt = icaltime_current_time_with_zone (utc);        
        uri->last_pub_time = g_strdup (icaltime_as_ical_string (current_itt));
        return TRUE;
        
    } else {
        if (strlen (uri->last_pub_time) == 0) {
            uri->last_pub_time = g_strdup (icaltime_as_ical_string (current_itt));
            return TRUE;
        }
        
        utc = icaltimezone_get_utc_timezone ();
        current_itt = icaltime_current_time_with_zone (utc);
        adjust_itt = icaltime_from_string (uri->last_pub_time);
        
        switch (uri->publish_freq) {
            case URI_PUBLISH_DAILY:
                icaltime_adjust (&adjust_itt, 1, 0, 0, 0);
                if (icaltime_compare_date_only (adjust_itt, current_itt ) < 0) {
                    uri->last_pub_time = g_strdup (icaltime_as_ical_string (current_itt));
                    return TRUE;
                }
                break;
            case URI_PUBLISH_WEEKLY:
                icaltime_adjust (&adjust_itt, 7, 0, 0, 0);
                if (icaltime_compare_date_only (adjust_itt, current_itt ) < 0) {
                    uri->last_pub_time = g_strdup (icaltime_as_ical_string (current_itt));
                    return TRUE;
                }
                break;
        }
    }
    
    return FALSE;
}

static gboolean
just_published (gchar *last_pub_time) {
    icaltimezone *utc;
    struct icaltimetype pubtime_itt, adjust_itt;
    
    if (strlen (last_pub_time) != 0) {
        utc = icaltimezone_get_utc_timezone ();
        pubtime_itt  = icaltime_from_string (last_pub_time);
        adjust_itt = icaltime_current_time_with_zone (utc);
        icaltime_adjust (&adjust_itt, 0, 0, 0, 3);
        if (icaltime_compare_date_only (pubtime_itt, adjust_itt) < 0)
            return TRUE;
    }

    
    return FALSE;
}

void
e_pub_publish (gboolean publish) {
    icaltimezone *utc;
    time_t start = time (NULL), end;
    GSList *uri_config_list, *l, *uri_list = NULL;
    ESourceList *source_list;
    GConfClient *gconf_client;
    gboolean published = FALSE;
    
    gconf_client = gconf_client_get_default ();
    source_list = e_source_list_new_for_gconf (gconf_client, "/apps/evolution/calendar/sources");
    
    utc = icaltimezone_get_utc_timezone (); 
    start = time_day_begin_with_zone (start, utc);
    end = time_add_week_with_zone (start, 6, utc);
        
    uri_config_list = calendar_config_get_free_busy ();

    for (l = uri_config_list; l != NULL; l = l->next) {
        GSList *p =NULL;
        EPublishUri *uri;
        ECalComponent *clone = NULL;
        gboolean cloned = FALSE;
        ECal *client = NULL;
        gchar *xml = (gchar *)uri_config_list->data;
        
        uri = g_new0 (EPublishUri, 1);      
        e_pub_uri_from_xml (uri, xml);
        
        /* kludge to safeguard against loop from gconf update */
        if (just_published (uri->last_pub_time))
            return;
        
        /* TODO: make sure we're online */
        /* skip this url if it isn't enabled or if it is manual */
        if (!uri->enabled) {
            uri_config_list = g_slist_next (uri_config_list);
            continue;
        }   
        
        if (!publish) {
            /* a g_idle publish, make sure we are not set to user only */
            if (uri->publish_freq == URI_PUBLISH_USER) {
                uri_config_list = g_slist_next (uri_config_list);
                continue;
            }
            
            /* If not is it time to publish again? */
            publish = is_publish_time (uri);
                
        }
        
        /* User published or config change */
        if (publish) {
            /* We still need to set the last_pub_time */
            uri->last_pub_time = 0;
            is_publish_time (uri);
            
            for (p = uri->calendars; p != NULL; p = p->next) {
                GList *comp_list = NULL;
                gchar *source_uid;
                ESource * source;
    
                source_uid = g_strdup (p->data);
                source =  e_source_list_peek_source_by_uid (source_list, source_uid);
    
                client = auth_new_cal_from_uri (e_source_get_uri (source), E_CAL_SOURCE_TYPE_EVENT);

                if (!client) {
                    g_warning (G_STRLOC ": Could not publish Free/Busy: Calendar backend no longer exists");

                    continue;
                }
            
                e_cal_open (client, TRUE, NULL);
                
                if (e_cal_get_free_busy ((ECal *) client, NULL,
                             start, end, 
                             &comp_list, NULL)) {
                    GList *l;
    
                    for (l = comp_list; l; l = l->next) {
                        ECalComponent *comp = E_CAL_COMPONENT (l->data);
                    
                        cloned = itip_publish_begin (comp, (ECal *) client, cloned, &clone);
                        g_object_unref (comp);
                    }
                    g_list_free (comp_list);
                }
                
                g_object_unref (client);
            
                g_free (source_uid);
            }
            
            if (cloned && clone)
                published = itip_publish_comp ((ECal *) client,
                               uri->location,
                               uri->username, 
                               uri->password, &clone);
            
            g_slist_free (p);
        }
        xml = e_pub_uri_to_xml (uri);
        if (xml != NULL) {
            uri_list = g_slist_append (uri_list, xml);
        }
        g_free (uri);
    }
    
    if (published) {
        /* Update gconf so we have the last_pub_time */
        calendar_config_set_free_busy (uri_list);
    }
        
    g_slist_free (uri_config_list);
    g_slist_free (uri_list);
}