Changes on server

- bugfixes in gcal callback
- requirements txt
This commit is contained in:
Martin Bauer 2014-04-21 09:21:59 +02:00
parent aac687c0c7
commit 46b4078fac
9 changed files with 84 additions and 56 deletions

View File

@ -1 +1 @@
{"_module": "oauth2client.client", "token_expiry": "2014-04-19T18:37:13Z", "access_token": "ya29.1.AADtN_UNi3N8wJ5AiTZ3ced1ho7-ZOhe9cV0TqJ7lByEy_h5SHGm0EuZBSWVOQ3q3g", "token_uri": "https://accounts.google.com/o/oauth2/token", "invalid": false, "token_response": {"access_token": "ya29.1.AADtN_UNi3N8wJ5AiTZ3ced1ho7-ZOhe9cV0TqJ7lByEy_h5SHGm0EuZBSWVOQ3q3g", "token_type": "Bearer", "expires_in": 3600}, "client_id": "34462582242-4kpdvvbi27ajt4u22uitqurpve9o8ipj.apps.googleusercontent.com", "id_token": null, "client_secret": "y4t9XBrJdCODPTO5UvtONWWn", "revoke_uri": "https://accounts.google.com/o/oauth2/revoke", "_class": "OAuth2Credentials", "refresh_token": "1/7-6-m_lLAKX8IeD7OuGtkcIiprty_nZUSxhMunSC5b0", "user_agent": null} {"_module": "oauth2client.client", "token_expiry": "2014-04-20T19:09:26Z", "access_token": "ya29.1.AADtN_VE0fTw3cu2E9SR0XoblR3gXjnON1u2Qy__EB7E6HyEpaDky0LA-SmZVyok6A", "token_uri": "https://accounts.google.com/o/oauth2/token", "invalid": false, "token_response": {"access_token": "ya29.1.AADtN_VE0fTw3cu2E9SR0XoblR3gXjnON1u2Qy__EB7E6HyEpaDky0LA-SmZVyok6A", "token_type": "Bearer", "expires_in": 3600}, "client_id": "34462582242-4kpdvvbi27ajt4u22uitqurpve9o8ipj.apps.googleusercontent.com", "id_token": null, "client_secret": "y4t9XBrJdCODPTO5UvtONWWn", "revoke_uri": "https://accounts.google.com/o/oauth2/revoke", "_class": "OAuth2Credentials", "refresh_token": "1/7-6-m_lLAKX8IeD7OuGtkcIiprty_nZUSxhMunSC5b0", "user_agent": null}

View File

@ -79,7 +79,7 @@ STATIC_ROOT = PROJECT_PATH + '/static_collection'
STATIC_URL = '/static/' STATIC_URL = '/static/'
LOGIN_URL="/musicians/login" LOGIN_URL="/musicians/login"
PUBLIC_URLS=( "^musicians/login/?$", "^musicians/login/usernames/?$") PUBLIC_URLS=( "^musicians/login/?$", "^musicians/login/usernames/?$", "^eventplanner_gcal/gcalApiCallback*")
# Additional locations of static files # Additional locations of static files
STATICFILES_DIRS = ( STATICFILES_DIRS = (
@ -195,49 +195,29 @@ CRISPY_TEMPLATE_PACK = 'bootstrap'
# the site admins on every HTTP 500 error when DEBUG=False. # the site admins on every HTTP 500 error when DEBUG=False.
# See http://docs.djangoproject.com/en/dev/topics/logging for # See http://docs.djangoproject.com/en/dev/topics/logging for
# more details on how to customize your logging configuration. # more details on how to customize your logging configuration.
LOGGING = { LOGGING = {
'version': 1, 'version': 1,
'disable_existing_loggers': True, 'disable_existing_loggers': False,
'formatters': {
'standard': {
'format' : "[%(asctime)s] %(levelname)s [%(name)s:%(lineno)s] %(message)s",
'datefmt' : "%d/%b/%Y %H:%M:%S"
},
},
'handlers': { 'handlers': {
'null': { 'file': {
'level': 'DEBUG', 'level': 'DEBUG',
'class':'django.utils.log.NullHandler', 'class': 'logging.FileHandler',
}, 'filename': '/srv/test/eventplanner.log',
'logfile': {
'level':'DEBUG',
'class':'logging.handlers.RotatingFileHandler',
'filename': PROJECT_PATH + "/logfile",
'maxBytes': 50000,
'backupCount': 2,
'formatter': 'standard',
},
'console':{
'level':'INFO',
'class':'logging.StreamHandler',
'formatter': 'standard'
}, },
}, },
'loggers': { 'loggers': {
'django': { 'eventplanner_gcal': {
'handlers':['console'], 'handlers': ['file'],
'level': 'DEBUG',
'propagate': True, 'propagate': True,
'level':'WARN',
}, },
'django.db.backends': { 'eventplanner': {
'handlers': ['console'], 'handler': ['file'],
'level': 'DEBUG', 'level': 'DEBUG',
'propagate': False, 'propagate': True,
},
'musicians': {
'handlers': ['console', 'logfile'],
'level': 'DEBUG',
},
} }
},
} }

View File

@ -15,6 +15,13 @@ framework.
""" """
import os import os
import site
import sys
site.addsitedir('/srv/test/env/lib/python2.7/site-packages')
sys.path.append('/srv/test')
# We defer to a DJANGO_SETTINGS_MODULE already in the environment. This breaks # We defer to a DJANGO_SETTINGS_MODULE already in the environment. This breaks
# if running multiple sites in the same mod_wsgi process. To fix this, use # if running multiple sites in the same mod_wsgi process. To fix this, use
# mod_wsgi daemon mode with each site in its own daemon process, or use # mod_wsgi daemon mode with each site in its own daemon process, or use

View File

@ -132,6 +132,12 @@ class EventParticipation( models.Model ):
def get_username(self): def get_username(self):
return self.user.username return self.user.username
def save( self, *args, **kwargs ):
prev = EventParticipation.objects.get( event = self.event, user = self.user )
if prev.status != self.status or prev.comment != self.comment:
super(EventParticipation,self).save( *args,**kwargs)
@staticmethod @staticmethod
def hasUserSetParticipationForAllEvents( user ): def hasUserSetParticipationForAllEvents( user ):
if not EventParticipation.isMember(user): if not EventParticipation.isMember(user):

View File

@ -282,10 +282,10 @@ def checkGCalSubscription():
global service global service
callbackUrl = settings.GCAL_COUPLING['push_url'] callbackUrl = settings.GCAL_COUPLING['push_url']
#timeToLive = 14*24*3600 # how long the channel should be active timeToLive = 14*24*3600 # how long the channel should be active
#renewBeforeExpiry = 2*24*3600 # duration before expiry when channel is renewed renewBeforeExpiry = 2*24*3600 # duration before expiry when channel is renewed
timeToLive = 60*5 #timeToLive = 60*5
renewBeforeExpiry = 60*3 #renewBeforeExpiry = 60*3
# Test if a channel already exists for this callbackURL # Test if a channel already exists for this callbackURL
try: try:

View File

@ -2,9 +2,11 @@ from django.db.models.signals import post_save,pre_delete
from django.dispatch import receiver from django.dispatch import receiver
from eventplanner.models import Event, EventParticipation from eventplanner.models import Event, EventParticipation
from django.contrib.auth.models import User from django.contrib.auth.models import User
import eventplanner_gcal.models import eventplanner_gcal.models
import logging
logger = logging.getLogger( __name__ )
class SignalLock: class SignalLock:
def __init__(self): def __init__(self):
@ -25,10 +27,21 @@ class SignalLock:
signalLock = SignalLock() signalLock = SignalLock()
@receiver( post_save, sender=User )
def user_changed( **kwargs ): def onGoogleCallback():
if not signalLock.isLocked(): if not signalLock.isLocked():
with signalLock: with signalLock:
logger.info( "Sync back from google" )
eventplanner_gcal.models.syncParticipationFromGoogleToLocal()
@receiver( post_save, sender=User )
def user_changed( **kwargs ):
logger.info("User info changed")
if not signalLock.isLocked():
with signalLock:
logger.info("Synchronizing with google - user information changed")
eventplanner_gcal.models.deleteAllGCalEvents()
eventplanner_gcal.models.syncGCalEvents() eventplanner_gcal.models.syncGCalEvents()
@ -39,10 +52,10 @@ def event_post_save_handler( **kwargs):
event = kwargs['instance'] event = kwargs['instance']
created = kwargs['created'] created = kwargs['created']
if created: if created:
print("Creating Gcal event") logger.info("Creating Gcal event")
eventplanner_gcal.models.createGCalEvent( event ).execute() eventplanner_gcal.models.createGCalEvent( event ).execute()
else: else:
print( "Updating Gcal event") logger.info( "Updating Gcal event")
eventplanner_gcal.models.updateGCalEvent( event ).execute() eventplanner_gcal.models.updateGCalEvent( event ).execute()
@ -52,7 +65,7 @@ def event_pre_delete_handler( **kwargs):
if not signalLock.isLocked(): if not signalLock.isLocked():
with signalLock: with signalLock:
event = kwargs['instance'] event = kwargs['instance']
print ("Deleting GCAL event") logger.info ("Deleting GCAL event")
eventplanner_gcal.models.deleteGCalEvent( event ).execute() eventplanner_gcal.models.deleteGCalEvent( event ).execute()
@ -61,7 +74,7 @@ def participation_post_save_handler( **kwargs):
if not signalLock.isLocked(): if not signalLock.isLocked():
with signalLock: with signalLock:
participation = kwargs['instance'] participation = kwargs['instance']
print("Participation post save -> update gcal") logger.info("Participation post save -> update gcal")
eventplanner_gcal.models.updateGCalEvent( participation.event ).execute() eventplanner_gcal.models.updateGCalEvent( participation.event ).execute()

View File

@ -1,16 +1,24 @@
from django.shortcuts import redirect from django.shortcuts import redirect
from eventplanner_gcal.models import syncGCalEvents, syncParticipationFromGoogleToLocal from eventplanner_gcal.models import syncGCalEvents, syncParticipationFromGoogleToLocal
from eventplanner_gcal.signals import onGoogleCallback
from django.http import HttpResponse
from django.views.decorators.csrf import csrf_exempt
import logging
logger = logging.getLogger( __name__ )
def runSync( request ): def runSync( request ):
syncGCalEvents() syncGCalEvents()
return redirect("/") return redirect("/")
@csrf_exempt
def gcalApiCallback( request ): def gcalApiCallback( request ):
syncParticipationFromGoogleToLocal() onGoogleCallback()
print ( "gcalApiCallback called" ) logger.info("Received callback from GCal - updating event participations... ")
return redirect("/") return HttpResponse('<h1>Callback successful</h1>')

14
requirements.txt Normal file
View File

@ -0,0 +1,14 @@
Django==1.5.5
PIL==1.1.7
South==0.8.2
django-classy-tags==0.4
django-crispy-forms==1.4.0
django-sekizai==0.7
django-tagging==0.3.1
djangorestframework==2.3.8
google-api-python-client==1.2
httplib2==0.9
reportlab==3.0
sorl-thumbnail==11.12
tagging==0.2.1
wsgiref==0.1.2

View File

@ -82,7 +82,7 @@
<img class="asset left-30 sp600 t150 z1" src="{{STATIC_URL}}/img/googleCalNew.png" /> <img class="asset left-30 sp600 t150 z1" src="{{STATIC_URL}}/img/googleCalNew.png" />
<div class="info"> <div class="info">
<a href="/musicians/profile">GoogleCalendar</a> <a href="/musicians/profile">GoogleCalendar</a>
<div class="subtitle subtitlebg">Geht ganz einfach: Im Profil eine Google Mail Adresse eintragen</div> <div class="subtitle subtitlebg">Jetzt auch auf dem Handy im GoogleCalendar Termine planen: Einfach im Profil eine Google Email Adresse eintragen...</div>
</div> </div>
</article> </article>