diff gpp/gcalendar/calendar.py @ 1:dbd703f7d63a

Initial import of sg101 stuff from private repository.
author gremmie
date Mon, 06 Apr 2009 02:43:12 +0000
parents
children cce1fc3f8752
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/gpp/gcalendar/calendar.py	Mon Apr 06 02:43:12 2009 +0000
@@ -0,0 +1,145 @@
+"""
+This file contains the calendar class wich abstracts the Google gdata API for working with
+Google Calendars.
+"""
+import datetime
+import pytz
+
+from django.utils.tzinfo import FixedOffset
+from gdata.calendar.service import CalendarService
+from gdata.calendar import CalendarEventFeed
+from gdata.calendar import CalendarEventEntry
+from gdata.calendar import Who
+from gdata.calendar import Where
+from gdata.calendar import When
+from gdata.service import BadAuthentication
+import atom
+
+from gcalendar.models import Event
+
+
+class CalendarError(Exception):
+    def __init__(self, errs):
+        self.errs = errs
+
+    def __str__(self):
+        return repr(self.errs)
+
+
+class Calendar(object):
+    DATE_FMT = '%Y-%m-%d'
+    DATE_TIME_FMT = DATE_FMT + 'T%H:%M:%S'
+    DATE_TIME_TZ_FMT = DATE_TIME_FMT + '.000Z'
+
+    def __init__(self, email, password, calendar_id='default'): 
+        self.client = CalendarService()
+        self.client.email = email
+        self.client.password = password
+        self.client.source = 'Google-Calendar_Python_GCalendar'
+        self.insert_feed = '/calendar/feeds/%s/private/full' % calendar_id
+        self.batch_feed = '%s/batch' % self.insert_feed
+        try:
+            self.client.ProgrammaticLogin()
+        except BadAuthentication:
+            raise CalendarError(['Incorrect password'])
+        except Exception, e:
+            raise CalendarError([e])
+
+    def sync_events(self, qs):
+        request_feed = CalendarEventFeed()
+        for model in qs:
+            if model.status == Event.NEW_APRV:
+                event = CalendarEventEntry()
+                request_feed.AddInsert(entry=self._populate_event(model, event))
+            elif model.status == Event.EDIT_APRV:
+                event = self._retrieve_event(model)
+                request_feed.AddUpdate(entry=self._populate_event(model, event))
+            elif model.status == Event.DEL_APRV:
+                event = self._retrieve_event(model)
+                request_feed.AddDelete(entry=event)
+            else:
+                assert False, 'unexpected status in sync_events'
+
+        response_feed = self.client.ExecuteBatch(request_feed, self.batch_feed)
+        err_msgs = []
+        for entry in response_feed.entry:
+            i = int(entry.batch_id.text)
+            code = int(entry.batch_status.code)
+
+            error = False
+            if qs[i].status == Event.NEW_APRV or qs[i].status == Event.EDIT_APRV:
+                if (code == 201 and qs[i].status == Event.NEW_APRV) or \
+                        (code == 200 and qs[i].status == Event.EDIT_APRV):
+                    qs[i].google_id = entry.id.text
+                    qs[i].status = Event.ON_CAL
+                    qs[i].save()
+                else:
+                    error = True
+            elif qs[i].status == Event.DEL_APRV:
+                if code == 200:
+                    qs[i].delete()
+                else:
+                    error = True
+
+            if error:
+                err_msgs.append('%s - (%d) %s' % \
+                        (qs[i].title, code, entry.batch_status.reason))
+
+        if len(err_msgs) > 0:
+            raise CalendarError(err_msgs)
+
+    def _retrieve_event(self, model):
+        try:
+            event = self.client.GetCalendarEventEntry(model.google_id)
+        except Exception:
+            raise CalendarError(['Could not retrieve event from Google: %s' % model.what])
+        return event
+
+    def _populate_event(self, model, event):
+        """Populates a gdata event from an Event model object."""
+        event.title = atom.Title(text=model.what)
+        event.content = atom.Content(text=model.html)
+        event.where = [Where(value_string=model.where)]
+        event.who = [Who(name=model.user.username, email=model.user.email)]
+
+        if model.all_day:
+            start_time = self._make_time(model.start_date)
+            if model.start_date == model.end_date:
+                end_time = None
+            else:
+                end_time = self._make_time(model.end_date)
+        else:
+            start_time = self._make_time(model.start_date, model.start_time, model.time_zone)
+            end_time = self._make_time(model.end_date, model.end_time, model.time_zone)
+
+        event.when = [When(start_time=start_time, end_time=end_time)]
+        return event
+    
+    def _make_time(self, date, time=None, tz_name=None):
+        """
+        Returns the gdata formatted date/time string given a date, optional time,
+        and optional time zone name (e.g. 'US/Pacific'). If the time zone name is None,
+        no time zone info will be added to the string.
+        """
+
+        if time is not None:
+            d = datetime.datetime.combine(date, time)
+        else:
+            d = datetime.datetime(date.year, date.month, date.day)
+
+        if time is None:
+            s = d.strftime(self.DATE_FMT)
+        elif tz_name is None:
+            s = d.strftime(self.DATE_TIME_FMT)
+        else:
+            try:
+                tz = pytz.timezone(tz_name)
+            except pytz.UnknownTimeZoneError:
+                raise CalendarError(['Invalid time zone: %s' (tz_name,)])
+            local = tz.localize(d)
+            zulu = local.astimezone(FixedOffset(0))
+            s = zulu.strftime(self.DATE_TIME_TZ_FMT)
+
+        return s
+
+# vim: ts=4 sw=4