From 9be00e843656f203df5361c7b12503b8d59592aa Mon Sep 17 00:00:00 2001 From: Leif Johansson Date: Thu, 5 May 2011 15:57:24 +0200 Subject: simplify creation process --- src/meetingtools/apps/room/forms.py | 39 ++++----- src/meetingtools/apps/room/models.py | 2 +- src/meetingtools/apps/room/views.py | 150 ++++++++++++++++++++--------------- src/meetingtools/urls.py | 2 +- src/site-media/css/style.css | 1 + src/templates/apps/room/list.html | 2 +- src/templates/form.html | 5 +- 7 files changed, 113 insertions(+), 88 deletions(-) diff --git a/src/meetingtools/apps/room/forms.py b/src/meetingtools/apps/room/forms.py index a211b2d..1906922 100644 --- a/src/meetingtools/apps/room/forms.py +++ b/src/meetingtools/apps/room/forms.py @@ -3,18 +3,26 @@ Created on Feb 1, 2011 @author: leifj ''' -from django.forms.models import ModelChoiceField + from meetingtools.apps.room.models import Room -from django.forms.widgets import Select, TextInput -from django.forms.fields import BooleanField +from django.forms.widgets import Select, TextInput, RadioSelect +from django.forms.fields import BooleanField, ChoiceField from django.forms.forms import Form from form_utils.forms import BetterModelForm -from django.contrib.auth.models import Group +from django.utils.safestring import mark_safe PUBLIC = 0 PROTECTED = 1 PRIVATE = 2 +class PrefixTextInput(TextInput): + def __init__(self, attrs=None, prefix=None): + super(PrefixTextInput, self).__init__(attrs) + self.prefix = prefix + + def render(self, name, value, attrs=None): + return mark_safe(""+self.prefix+" "+super(PrefixTextInput, self).render(name, value, attrs)) + class ModifyRoomForm(BetterModelForm): class Meta: model = Room @@ -30,35 +38,28 @@ class ModifyRoomForm(BetterModelForm): 'description': 'These are basic properties for your room. If you set your room to be self-cleaning it will be reset every time the last participant leaves the room.'}), ] widgets = {'source_sco_id': Select(), - 'urlpath': TextInput(attrs={'size': '40'}), + 'urlpath': PrefixTextInput(attrs={'size': '15'}), 'name': TextInput(attrs={'size': '40'})} class CreateRoomForm(BetterModelForm): - - participants = ModelChoiceField(Group,required=False) - presenters = ModelChoiceField(Group,required=False) - hosts = ModelChoiceField(Group,required=False) - + + access = ChoiceField(choices=(('public','Public'),('private','Private'))) class Meta: model = Room - fields = ['name','urlpath','participants','presenters','hosts','source_sco_id','self_cleaning'] + fields = ['name','urlpath','access','self_cleaning'] fieldsets = [('name',{'fields': ['name'], 'classes': ['step'], 'legend': 'Step 1: Room name', 'description': 'The room name should be short and descriptive.' }), - ('properties',{'fields': ['self_cleaning','urlpath','source_sco_id'], + ('properties',{'fields': ['self_cleaning','urlpath','access'], 'classes': ['step'], 'legend': 'Step 2: Room properties', - 'description': 'These are basic properties for your room. If you set your room to be self-cleaning it will be reset every time the last participant leaves the room.'}), - ('rights',{'fields': ['participants','presenters','hosts'], - 'classes': ['step','submit_step'], - 'legend': 'Step 3: Room rights (optional)', - 'description': 'Define the groups that are to have access to your room. If you leave the Participants field empty that implies that anyone who knows the URL may enter the room.'}) + 'description': 'These are basic properties for your room. If you set your room to clean-up after use it will be reset every time the last participant leaves the room. If you create a public room it will be open to anyone who has the room URL. If you create a private room then guests will have to be approved by an active meeting host before being able to join the room.'}), ] - widgets = {'source_sco_id': Select(), - 'urlpath': TextInput(attrs={'size': '40'}), + widgets = {'access': RadioSelect(), + 'urlpath': PrefixTextInput(attrs={'size': '15'}), 'name': TextInput(attrs={'size': '40'})} class DeleteRoomForm(Form): diff --git a/src/meetingtools/apps/room/models.py b/src/meetingtools/apps/room/models.py index 5c3e9a5..6272212 100644 --- a/src/meetingtools/apps/room/models.py +++ b/src/meetingtools/apps/room/models.py @@ -15,7 +15,7 @@ from django_co_acls.models import AccessControlEntry class Room(models.Model): creator = ForeignKey(User,editable=False) name = CharField(max_length=128,unique=True) - urlpath = CharField(max_length=128,unique=True) + urlpath = CharField(verbose_name="Custom URL",max_length=128,unique=True) acc = ForeignKey(ACCluster,verbose_name="Adobe Connect Cluster",editable=False) self_cleaning = BooleanField(verbose_name="Clean-up when empty?") sco_id = IntegerField(verbose_name="Adobe Connect Room") diff --git a/src/meetingtools/apps/room/views.py b/src/meetingtools/apps/room/views.py index 0084018..250e48e 100644 --- a/src/meetingtools/apps/room/views.py +++ b/src/meetingtools/apps/room/views.py @@ -12,7 +12,6 @@ from meetingtools.ac import ac_api_client, api import re from meetingtools.apps import room from django.contrib.auth.decorators import login_required -from meetingtools.apps.auth.utils import groups import logging from pprint import pformat from meetingtools.utils import session @@ -21,6 +20,7 @@ from meetingtools.settings import GRACE from django.utils.datetime_safe import datetime from django.http import HttpResponseRedirect from django.core.exceptions import ObjectDoesNotExist +from django_co_acls.models import allow, deny, acl, clear_acl def _acc_for_user(user): (local,domain) = user.username.split('@') @@ -116,91 +116,113 @@ def view(request,id): }) def _init_update_form(request,form,acc,my_meetings_sco_id): - if form.fields.has_key('participants'): - form.fields['participants'].queryset = request.user.groups - if form.fields.has_key('presenters'): - form.fields['presenters'].queryset = request.user.groups - if form.fields.has_key('hosts'): - form.fields['hosts'].queryset = request.user.groups + if form.fields.has_key('urlpath'): + form.fields['urlpath'].widget.prefix = acc.url if form.fields.has_key('source_sco_id'): form.fields['source_sco_id'].widget.choices = [('','-- select template --')]+[r for r in _user_templates(request,acc,my_meetings_sco_id)] -@login_required -def _update_room(request, room, form=None): +def _update_room(request, room, form=None): api = ac_api_client(request, room.acc) - params = {'type':'meeting', - 'name':room.name, - 'folder-id': room.folder_sco_id, - 'source-sco-id': room.source_sco_id, - 'url-path':room.urlpath} + params = {'type':'meeting'} - if room.sco_id: - params['sco-id'] = room.sco_id - r = api.request('sco-update', params, raise_error=True) + for attr,param in (('sco_id','sco-id'),('folder_sco_id','folder-id'),('source_sco_id','source-sco-id'),('urlpath','url-path'),('name','name')): + v = None + if hasattr(room,attr): + v = getattr(room,attr) + logging.debug("%s,%s = %s" % (attr,param,v)) + if form and form.cleaned_data.has_key(attr) and form.cleaned_data[attr]: + v = form.cleaned_data[attr] + + if v: + if isinstance(v,(str,unicode)): + params[param] = v + elif hasattr(v,'__getitem__'): + params[param] = v[0] + else: + params[param] = repr(v) + + logging.debug(pformat(params)) + r = api.request('sco-update', params, True) - room.sco_id = r.et.find(".//sco").get('sco-id') - room.source_sco_id = r.et.find(".//sco").get('sco-source-id') + sco_id = r.et.find(".//sco").get('sco-id') + if form: + form.cleaned_data['sco_id'] = sco_id + form.cleaned_data['source_sco_id'] = r.et.find(".//sco").get('sco-source-id') + + room.sco_id = sco_id room.save() - logging.debug(pformat(room)) - #r = api.request('permissions-reset',{'acl-id': room.sco_id},True) user_principal = api.find_user(request.user.username) - api.request('permissions-update',{'acl-id': room.sco_id,'principal-id': user_principal.get('principal-id'),'permission-id':'host'},True) # owner is always host + #api.request('permissions-reset',{'acl-id': sco_id},True) + api.request('permissions-update',{'acl-id': sco_id,'principal-id': user_principal.get('principal-id'),'permission-id':'host'},True) # owner is always host - for (key,perm) in [('participants','view'),('presenters','mini-host'),('hosts','host')]: - if form.cleaned_data.has_key(key): - group = form.cleaned_data[key] - if not group and key == 'participants': - permission = "view-hidden" - principal_id = "public-access" - else: - permission = perm - - principal_id = user_principal.get('principal-id') - if group: - group_principal = api.find_group(group.name) - principal_id = group_principal.get('principal-id') - - api.request('permissions-update',{'acl-id': room.sco_id, 'principal-id': principal_id, 'permission-id': permission},True) + if form: + if form.cleaned_data.has_key('access'): + access = form.cleaned_data['access'] + if access == 'public': + allow(room,'anyone','view-hidden') + elif access == 'private': + allow(room,'anyone','remove') + + # XXX figure out how to keep the room permissions in sync with the AC permissions + for ace in acl(room): + principal_id = None + if ace.group: + principal = api.find_group(ace.group.name) + if principal: + principal_id = principal.get('principal-id') + elif ace.user: + principal = api.find_user(ace.user.username) + if principal: + principal_id = principal.get('principal-id') + else: + principal_id = "public-access" + + if principal_id: + api.request('permissions-update',{'acl-id': room.sco_id, 'principal-id': principal_id, 'permission-id': ace.permission},True) return room @login_required -def update(request,id=None): - if id: - room = get_object_or_404(Room,pk=id) - acc = room.acc - what = "Update" - title = "Modify %s" % room.name - formname = "update" - update = True +def create(request): + acc = _acc_for_user(request.user) + my_meetings_sco_id = _user_meeting_folder(request,acc) + room = Room(creator=request.user,acc=acc,folder_sco_id=my_meetings_sco_id) + what = "Create" + title = "Create a new room" + + if request.method == 'POST': + form = CreateRoomForm(request.POST,instance=room) + _init_update_form(request, form, acc, room.folder_sco_id) + if form.is_valid(): + _update_room(request, room, form) + room = form.save() + return redirect_to("/rooms#%d" % room.id) else: - acc = _acc_for_user(request.user) - my_meetings_sco_id = _user_meeting_folder(request,acc) - room = Room(creator=request.user,acc=acc,folder_sco_id=my_meetings_sco_id) - what = "Create" - formname = "create" - title = "Create a new room" - update = False + form = CreateRoomForm(instance=room) + _init_update_form(request, form, acc, room.folder_sco_id) + + return respond_to(request,{'text/html':'apps/room/create.html'},{'form':form,'formtitle': title,'submitname':'%s Room' % what}) + +@login_required +def update(request,id): + room = get_object_or_404(Room,pk=id) + acc = room.acc + what = "Update" + title = "Modify %s" % room.name if request.method == 'POST': - if update: - form = ModifyRoomForm(request.POST,instance=room) - else: - form = CreateRoomForm(request.POST,instance=room) + form = ModifyRoomForm(request.POST,instance=room) _init_update_form(request, form, acc, room.folder_sco_id) if form.is_valid(): - room = _update_room(request, room, form) + _update_room(request, room, form) room = form.save() return redirect_to("/rooms#%d" % room.id) else: - if update: - form = ModifyRoomForm(instance=room) - else: - form = CreateRoomForm(instance=room) + form = ModifyRoomForm(instance=room) _init_update_form(request, form, acc, room.folder_sco_id) - return respond_to(request,{'text/html':'apps/room/%s.html' % formname},{'form':form,'formtitle': title,'submitname':'%s Room' % what}) + return respond_to(request,{'text/html':'apps/room/update.html'},{'form':form,'formtitle': title,'submitname':'%s Room' % what}) def _import_room(request,acc,sco_id,source_sco_id,folder_sco_id,name,urlpath): modified = False @@ -258,7 +280,7 @@ def list(request): for r in Room.objects.filter(creator=request.user).all(): #logging.debug(pformat(r)) - if (not r.sco_id in ar) and (not r.self_cleaning): + if (not r.sco_id in ar): # and (not r.self_cleaning): #XXX this logic isn't right! r.delete() return respond_to(request,{'text/html':'apps/room/list.html'},{'user':request.user,'rooms':Room.objects.filter(creator=request.user).all()}) @@ -274,7 +296,9 @@ def delete(request,id): if form.is_valid(): api = ac_api_client(request,room.acc) api.request('sco-delete',{'sco-id':room.sco_id},raise_error=True) + clear_acl(room) room.delete() + return redirect_to("/rooms") else: form = DeleteRoomForm() diff --git a/src/meetingtools/urls.py b/src/meetingtools/urls.py index 74a2052..568ac83 100644 --- a/src/meetingtools/urls.py +++ b/src/meetingtools/urls.py @@ -23,7 +23,7 @@ urlpatterns = patterns('', (r'^rooms/(.+)(?:\.([^\.]+))?$','meetingtools.apps.room.views.rooms_by_group'), (r'^go/(\d+)$','meetingtools.apps.room.views.go_by_id'), (r'^go/(.+)$','meetingtools.apps.room.views.go_by_path'), - (r'^room/create$','meetingtools.apps.room.views.update'), + (r'^room/create$','meetingtools.apps.room.views.create'), (r'^room/(\d+)$','meetingtools.apps.room.views.view'), (r'^room/(\d+)/modify$','meetingtools.apps.room.views.update'), (r'^room/(\d+)/delete$','meetingtools.apps.room.views.delete'), diff --git a/src/site-media/css/style.css b/src/site-media/css/style.css index deea06e..e241b06 100644 --- a/src/site-media/css/style.css +++ b/src/site-media/css/style.css @@ -58,6 +58,7 @@ form .ui-widget li { float: right; border-right: 1px solid #CECECE; padding-righ .left { float: left; } .fieldinput ul { list-style: none inside; padding: 0; margin: 0; } .fieldinput li { padding: 0; margin: 0; display: inline; float: left; } +div .fieldinput { margin-top: 10px; } a, a:hover, a:visited { text-decoration: none; color: #E37424; diff --git a/src/templates/apps/room/list.html b/src/templates/apps/room/list.html index 203d67b..2ec375e 100644 --- a/src/templates/apps/room/list.html +++ b/src/templates/apps/room/list.html @@ -24,7 +24,7 @@

Created by {{r.creator}} {{r.timecreated|datehumanize}}.

Meeting room URL: {{r|roomurl}}

diff --git a/src/templates/form.html b/src/templates/form.html index 077cd53..2e1e243 100644 --- a/src/templates/form.html +++ b/src/templates/form.html @@ -18,10 +18,9 @@ {% if field.is_hidden %} {{ field }} {% else %} - {% endif %} -- cgit v1.1