gremmie@1: """ gremmie@1: Models for the gcalendar application. bgneal@456: gremmie@1: """ bgneal@458: import datetime bgneal@458: gremmie@1: from django.db import models gremmie@1: from django.db.models import Q gremmie@1: from django.contrib.auth.models import User bgneal@124: bgneal@128: from core.markup import site_markup bgneal@228: import forums.tools bgneal@458: from gcalendar.oauth import serialize_token, deserialize_token gremmie@1: gremmie@1: bgneal@228: GIG_FORUM_SLUG = "gigs" bgneal@228: gremmie@1: class PendingEventManager(models.Manager): gremmie@1: """A manager for pending events.""" gremmie@1: gremmie@1: def get_query_set(self): gremmie@1: """Returns a queryset of events that have been approved to update gremmie@1: the Google calendar.""" gremmie@1: return super(PendingEventManager, self).get_query_set().filter( bgneal@456: Q(status=Event.NEW_APRV) | bgneal@456: Q(status=Event.EDIT_APRV) | gremmie@1: Q(status=Event.DEL_APRV) gremmie@1: ) gremmie@1: gremmie@1: gremmie@1: class Event(models.Model): gremmie@1: """Model to represent calendar events.""" gremmie@1: gremmie@1: # Event status codes: gremmie@1: (NEW, NEW_APRV, EDIT_REQ, EDIT_APRV, DEL_REQ, DEL_APRV, ON_CAL) = range(7) gremmie@1: gremmie@1: STATUS_CHOICES = ( gremmie@1: (NEW, 'New'), gremmie@1: (NEW_APRV, 'New Approved'), gremmie@1: (EDIT_REQ, 'Edit Request'), gremmie@1: (EDIT_APRV, 'Edit Approved'), gremmie@1: (DEL_REQ, 'Delete Request'), gremmie@1: (DEL_APRV, 'Delete Approved'), gremmie@1: (ON_CAL, 'On Calendar'), gremmie@1: ) gremmie@1: gremmie@1: user = models.ForeignKey(User) gremmie@1: what = models.CharField(max_length=255) gremmie@1: start_date = models.DateField() gremmie@1: start_time = models.TimeField(null=True, blank=True) gremmie@1: end_date = models.DateField() gremmie@1: end_time = models.TimeField(null=True, blank=True) gremmie@1: time_zone = models.CharField(max_length=64, blank=True) gremmie@1: all_day = models.BooleanField(default=False) gremmie@1: where = models.CharField(max_length=255, blank=True) gremmie@1: description = models.TextField(blank=True) gremmie@1: html = models.TextField(blank=True) gremmie@1: date_submitted = models.DateTimeField(auto_now_add=True) gremmie@1: google_id = models.CharField(max_length=255, blank=True) bgneal@228: google_url = models.URLField(verify_exists=False, max_length=255, bgneal@228: blank=True) bgneal@228: status = models.SmallIntegerField(choices=STATUS_CHOICES, default=NEW, bgneal@228: db_index=True) bgneal@228: create_forum_thread = models.BooleanField(default=False) gremmie@1: gremmie@1: objects = models.Manager() gremmie@1: pending_events = PendingEventManager() gremmie@1: gremmie@1: def __unicode__(self): gremmie@1: return self.what gremmie@1: gremmie@1: class Meta: gremmie@1: ordering = ('-date_submitted', ) gremmie@1: gremmie@1: def save(self, *args, **kwargs): bgneal@128: self.html = site_markup(self.description) gremmie@1: super(Event, self).save(*args, **kwargs) gremmie@1: bgneal@139: def is_approved(self): bgneal@139: return self.status not in (self.NEW, self.EDIT_REQ, self.DEL_REQ) bgneal@139: is_approved.boolean = True gremmie@1: bgneal@228: def google_html(self): bgneal@228: """Returns a HTML tag to the event if it exits.""" bgneal@228: if self.google_url: bgneal@228: return u'On Google' % self.google_url bgneal@228: return u'' bgneal@228: google_html.allow_tags = True bgneal@228: google_html.short_description = 'Google Link' bgneal@228: bgneal@228: def notify_on_calendar(self): bgneal@228: """ bgneal@228: This function should be called when the event has been added to the bgneal@228: Google calendar for the first time. This gives us a chance to perform bgneal@228: any first-time processing, like creating a forum thread. bgneal@228: """ bgneal@228: if self.create_forum_thread: bgneal@228: topic_name = '%s: %s' % (self.start_date.strftime('%m/%d/%Y'), bgneal@228: self.what) bgneal@228: post_body = "%s\n\n[Link to event on Google Calendar](%s)" % ( bgneal@228: self.description, self.google_url) bgneal@228: bgneal@228: forums.tools.create_topic( bgneal@228: forum_slug=GIG_FORUM_SLUG, bgneal@456: user=self.user, bgneal@228: topic_name=topic_name, bgneal@228: post_body=post_body) bgneal@228: bgneal@228: self.create_forum_thread = False bgneal@228: self.save() bgneal@456: bgneal@456: bgneal@458: class AccessTokenManager(models.Manager): bgneal@458: """ bgneal@458: A manager for the AccessToken table. Only one access token is saved in the bgneal@458: database. This manager provides a convenience method to either return that bgneal@458: access token or a brand new one. bgneal@458: bgneal@458: """ bgneal@458: def get_token(self): bgneal@458: try: bgneal@458: token = self.get(pk=1) bgneal@458: except AccessToken.DoesNotExist: bgneal@458: token = AccessToken() bgneal@458: bgneal@458: return token bgneal@458: bgneal@458: bgneal@456: class AccessToken(models.Model): bgneal@456: """ bgneal@456: This model represents serialized OAuth access tokens for reading and bgneal@456: updating the Google Calendar. bgneal@456: bgneal@456: """ bgneal@456: auth_date = models.DateTimeField() bgneal@456: token = models.TextField() bgneal@456: bgneal@458: objects = AccessTokenManager() bgneal@458: bgneal@456: def __unicode__(self): bgneal@456: return u'Access token created on ' + unicode(self.auth_date) bgneal@456: bgneal@458: def update(self, access_token, auth_date=None): bgneal@458: """ bgneal@458: This function updates the AccessToken object with the input parameters: bgneal@458: access_token - an access token from Google's OAuth dance bgneal@458: auth_date - a datetime or None. If None, now() is used. bgneal@458: bgneal@458: """ bgneal@458: self.auth_date = auth_date if auth_date else datetime.datetime.now() bgneal@458: self.token = serialize_token(access_token) bgneal@458: bgneal@458: def access_token(self): bgneal@458: """ bgneal@458: This function returns a Google OAuth access token by deserializing the bgneal@458: token field from the database. bgneal@458: If the token attribute is empty, None is returned. bgneal@458: bgneal@458: """ bgneal@458: return deserialize_token(self.token) if self.token else None