2011-07-31 10:46:29 +02:00
#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
openslides . application . views
~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
Views for the application app .
: copyright : 2011 by the OpenSlides team , see AUTHORS .
: license : GNU GPL , see LICENSE for more details .
"""
2011-11-14 16:37:12 +01:00
from __future__ import with_statement
2011-07-31 10:46:29 +02:00
2011-11-09 22:53:10 +01:00
import csv
2011-11-22 18:49:36 +01:00
import utils . csv_ext
2011-11-24 19:38:25 +01:00
2011-07-31 10:46:29 +02:00
from django . shortcuts import redirect
from django . contrib import messages
from django . contrib . auth . decorators import login_required
2011-11-09 22:53:10 +01:00
from django . contrib . auth . models import User , Group
2011-07-31 10:46:29 +02:00
from django . core . urlresolvers import reverse
from django . utils . translation import ugettext as _
2011-11-09 22:53:10 +01:00
from django . utils . translation import ungettext
from django . db import transaction
2011-07-31 10:46:29 +02:00
2011-11-01 14:35:58 +01:00
from openslides . agenda . models import Item
2011-09-03 18:11:42 +02:00
from openslides . application . models import Application , AVersion
2011-07-31 10:46:29 +02:00
from openslides . application . forms import ApplicationForm , \
2011-11-09 22:53:10 +01:00
ApplicationManagerForm , \
ApplicationImportForm
2011-09-08 17:39:57 +02:00
from openslides . participant . models import Profile
2011-07-31 10:46:29 +02:00
from openslides . poll . models import Poll
2011-09-03 13:53:37 +02:00
from openslides . poll . forms import OptionResultForm , PollForm
2011-07-31 10:46:29 +02:00
from openslides . utils . utils import template , permission_required , \
2011-09-04 10:43:25 +02:00
render_to_forbitten , del_confirm_form , gen_confirm_form
2011-07-31 10:46:29 +02:00
from openslides . utils . pdf import print_application , print_application_poll
from openslides . system . api import config_get
2011-11-14 21:54:42 +01:00
from openslides . participant . api import gen_username , gen_password
2011-07-31 10:46:29 +02:00
2011-09-04 12:21:58 +02:00
@permission_required ( ' application.can_see_application ' )
2011-07-31 10:46:29 +02:00
@template ( ' application/overview.html ' )
def overview ( request ) :
"""
View all applications
"""
query = Application . objects
if ' number ' in request . GET :
query = query . filter ( number = None )
if ' status ' in request . GET :
if ' statusvalue ' in request . GET and ' on ' in request . GET [ ' status ' ] :
query = query . filter ( status__iexact = request . GET [ ' statusvalue ' ] )
try :
sort = request . GET [ ' sort ' ]
if sort in [ ' number ' , ' supporter ' , ' status ' , ' submitter ' , \
' aversion__time ' , ' aversion__title ' ] :
query = query . order_by ( sort )
except KeyError :
query = query . order_by ( " number " )
if ' reverse ' in request . GET :
query = query . reverse ( )
if ' needsup ' in request . GET :
applications = [ ]
for application in query . all ( ) :
if not application . enough_supporters :
applications . append ( application )
else :
applications = query . all ( )
return {
' applications ' : applications ,
' min_supporters ' : int ( config_get ( ' application_min_supporters ' ) ) ,
}
2011-09-04 12:21:58 +02:00
@permission_required ( ' application.can_see_application ' )
2011-07-31 10:46:29 +02:00
@template ( ' application/view.html ' )
2011-09-06 10:41:06 +02:00
def view ( request , application_id , newest = False ) :
2011-07-31 10:46:29 +02:00
"""
View one application .
"""
application = Application . objects . get ( pk = application_id )
2011-09-06 10:41:06 +02:00
if newest :
version = application . last_version
else :
version = application . public_version
2011-07-31 10:46:29 +02:00
revisions = application . versions
actions = application . get_allowed_actions ( user = request . user )
2011-11-01 06:47:10 +01:00
2011-07-31 10:46:29 +02:00
return {
' application ' : application ,
' revisions ' : revisions ,
' actions ' : actions ,
' min_supporters ' : int ( config_get ( ' application_min_supporters ' ) ) ,
2011-09-06 22:22:29 +02:00
' version ' : version ,
' results ' : application . results
2011-07-31 10:46:29 +02:00
}
@login_required
@template ( ' application/edit.html ' )
def edit ( request , application_id = None ) :
"""
View a form to edit or create a application .
"""
if request . user . has_perm ( ' application.can_manage_application ' ) :
is_manager = True
else :
is_manager = False
if not is_manager \
2011-09-04 12:21:58 +02:00
and not request . user . has_perm ( ' application.can_create_application ' ) :
messages . error ( request , _ ( " You have not the necessary rights to create or edit applications. " ) )
2011-07-31 10:46:29 +02:00
return redirect ( reverse ( ' application_overview ' ) )
if application_id is not None :
application = Application . objects . get ( id = application_id )
if not request . user == application . submitter and not is_manager :
messages . error ( request , _ ( " You can not edit this application. You are not the submitter. " ) )
return redirect ( reverse ( ' application_view ' , args = [ application . id ] ) )
else :
application = None
if request . method == ' POST ' :
dataform = ApplicationForm ( request . POST , prefix = " data " )
valid = dataform . is_valid ( )
if is_manager :
managerform = ApplicationManagerForm ( request . POST , \
instance = application , \
prefix = " manager " )
valid = valid and managerform . is_valid ( )
else :
managerform = None
if valid :
del_supporters = True
2011-09-08 17:39:57 +02:00
original_supporters = [ ]
2011-07-31 10:46:29 +02:00
if is_manager :
2011-09-08 17:39:57 +02:00
if application :
for s in application . supporter . all ( ) :
original_supporters . append ( s )
2011-07-31 10:46:29 +02:00
application = managerform . save ( )
elif application_id is None :
application = Application ( submitter = request . user )
application . title = dataform . cleaned_data [ ' title ' ]
application . text = dataform . cleaned_data [ ' text ' ]
application . reason = dataform . cleaned_data [ ' reason ' ]
2011-11-09 22:53:10 +01:00
application . save ( request . user , trivial_change = dataform . cleaned_data [ ' trivial_change ' ] )
2011-09-08 17:39:57 +02:00
if is_manager :
# log added supporters
supporters_added = [ ]
for s in application . supporter . all ( ) :
if s not in original_supporters :
try :
supporters_added . append ( unicode ( s . profile ) )
except Profile . DoesNotExist :
pass
if len ( supporters_added ) > 0 :
2011-09-11 22:43:15 +02:00
log_added = " , " . join ( supporters_added )
2011-09-08 17:39:57 +02:00
application . writelog ( _ ( " Supporter: + %s " ) % log_added , request . user )
# log removed supporters
supporters_removed = [ ]
for s in original_supporters :
if s not in application . supporter . all ( ) :
try :
supporters_removed . append ( unicode ( s . profile ) )
except Profile . DoesNotExist :
pass
if len ( supporters_removed ) > 0 :
2011-09-21 21:01:27 +02:00
log_removed = " , " . join ( supporters_removed )
2011-09-08 17:39:57 +02:00
application . writelog ( _ ( " Supporter: - %s " ) % log_removed , request . user )
2011-07-31 10:46:29 +02:00
if application_id is None :
messages . success ( request , _ ( ' New application was successfully created. ' ) )
else :
messages . success ( request , _ ( ' Application was successfully modified. ' ) )
2011-09-04 10:02:54 +02:00
2011-09-03 19:25:05 +02:00
if not ' apply ' in request . POST :
return redirect ( reverse ( ' application_view ' , args = [ application . id ] ) )
if application_id is None :
return redirect ( reverse ( ' application_edit ' , args = [ application . id ] ) )
2011-11-24 19:42:32 +01:00
else :
messages . error ( request , _ ( ' Please check the form for errors. ' ) )
2011-07-31 10:46:29 +02:00
else :
if application_id is None :
initial = { ' text ' : config_get ( ' application_preamble ' ) }
else :
if application . status == " pub " and application . supporter . count ( ) > 0 :
2011-09-07 22:17:19 +02:00
if request . user . has_perm ( ' application.can_manage_application ' ) :
messages . warning ( request , _ ( " Attention: Do you really want to edit this application? The supporters will <b>not</b> be removed automatically because you can manage applications. Please check if the supports are valid after your changing! " ) )
2011-07-31 10:46:29 +02:00
else :
messages . warning ( request , _ ( " Attention: Do you really want to edit this application? All <b> %s </b> supporters will be removed! Try to convince the supporters again. " ) % application . supporter . count ( ) )
initial = { ' title ' : application . title ,
' text ' : application . text ,
' reason ' : application . reason }
dataform = ApplicationForm ( initial = initial , prefix = " data " )
if is_manager :
if application_id is None :
initial = { ' submitter ' : str ( request . user . id ) }
else :
initial = { }
managerform = ApplicationManagerForm ( initial = initial , \
instance = application , \
prefix = " manager " )
else :
managerform = None
return {
' form ' : dataform ,
' managerform ' : managerform ,
' application ' : application ,
}
2011-11-01 08:07:27 +01:00
@login_required
2011-07-31 10:46:29 +02:00
@template ( ' application/view.html ' )
def delete ( request , application_id ) :
"""
delete a application .
"""
application = Application . objects . get ( id = application_id )
if not ' delete ' in application . get_allowed_actions ( user = request . user ) :
messages . error ( request , _ ( " You can not delete application <b> %s </b>. " ) % application )
else :
if request . method == ' POST ' :
try :
title = str ( application )
application . delete ( )
messages . success ( request , _ ( " Application <b> %s </b> was successfully deleted. " ) % title )
except NameError , name :
messages . error ( request , name )
else :
del_confirm_form ( request , application )
return redirect ( reverse ( ' application_overview ' ) )
@permission_required ( ' application.can_manage_application ' )
@template ( ' application/view.html ' )
def set_number ( request , application_id ) :
"""
set a number for an application .
"""
try :
Application . objects . get ( pk = application_id ) . set_number ( user = request . user )
messages . success ( request , _ ( " Application number was successfully set. " ) )
except Application . DoesNotExist :
pass
except NameError :
pass
return redirect ( reverse ( ' application_view ' , args = [ application_id ] ) )
@permission_required ( ' application.can_manage_application ' )
@template ( ' application/view.html ' )
def permit ( request , application_id ) :
"""
permit an application .
"""
try :
Application . objects . get ( pk = application_id ) . permit ( user = request . user )
messages . success ( request , _ ( " Application was successfully permitted. " ) )
except Application . DoesNotExist :
pass
return redirect ( reverse ( ' application_view ' , args = [ application_id ] ) )
@permission_required ( ' application.can_manage_application ' )
@template ( ' application/view.html ' )
def notpermit ( request , application_id ) :
"""
reject ( not permit ) an application .
"""
try :
Application . objects . get ( pk = application_id ) . notpermit ( user = request . user )
messages . success ( request , _ ( " Application was successfully rejected. " ) )
except Application . DoesNotExist :
pass
return redirect ( reverse ( ' application_view ' , args = [ application_id ] ) )
@template ( ' application/view.html ' )
def set_status ( request , application_id = None , status = None ) :
"""
set a status of an application .
"""
try :
if status is not None :
application = Application . objects . get ( pk = application_id )
application . set_status ( user = request . user , status = status )
messages . success ( request , _ ( " Application status was set to: <b> %s </b>. " ) % application . get_status_display ( ) )
except Application . DoesNotExist :
pass
return redirect ( reverse ( ' application_view ' , args = [ application_id ] ) )
@permission_required ( ' application.can_manage_application ' )
@template ( ' application/view.html ' )
def reset ( request , application_id ) :
"""
reset an application .
"""
try :
Application . objects . get ( pk = application_id ) . reset ( user = request . user )
messages . success ( request , _ ( " Application status was reset. " ) )
except Application . DoesNotExist :
pass
return redirect ( reverse ( ' application_view ' , args = [ application_id ] ) )
@permission_required ( ' application.can_support_application ' )
@template ( ' application/view.html ' )
def support ( request , application_id ) :
"""
support an application .
"""
try :
Application . objects . get ( pk = application_id ) . support ( user = request . user )
messages . success ( request , _ ( " You have support the application successfully. " ) )
except Application . DoesNotExist :
pass
return redirect ( reverse ( ' application_view ' , args = [ application_id ] ) )
@permission_required ( ' application.can_support_application ' )
@template ( ' application/view.html ' )
def unsupport ( request , application_id ) :
"""
unsupport an application .
"""
try :
Application . objects . get ( pk = application_id ) . unsupport ( user = request . user )
messages . success ( request , _ ( " You have unsupport the application successfully. " ) )
except Application . DoesNotExist :
pass
return redirect ( reverse ( ' application_view ' , args = [ application_id ] ) )
2011-11-01 14:35:58 +01:00
@permission_required ( ' application.can_manage_application ' )
def set_active ( request , application_id ) :
2012-02-03 23:12:28 +01:00
application = Application . objects . get ( pk = application_id )
application . set_active ( )
2011-11-01 14:35:58 +01:00
return redirect ( reverse ( ' application_view ' , args = [ application_id ] ) )
2011-11-01 14:45:24 +01:00
2011-07-31 10:46:29 +02:00
@permission_required ( ' application.can_manage_application ' )
@template ( ' application/view.html ' )
def gen_poll ( request , application_id ) :
"""
gen a poll for this application .
"""
try :
2011-09-03 16:20:10 +02:00
poll = Application . objects . get ( pk = application_id ) . gen_poll ( user = request . user )
messages . success ( request , _ ( " New vote was successfully created. " ) )
2011-07-31 10:46:29 +02:00
except Application . DoesNotExist :
pass
2011-09-03 16:20:10 +02:00
return redirect ( reverse ( ' application_poll_view ' , args = [ poll . id ] ) )
2011-07-31 10:46:29 +02:00
@permission_required ( ' application.can_manage_application ' )
def delete_poll ( request , poll_id ) :
"""
delete a poll from this application
"""
poll = Poll . objects . get ( pk = poll_id )
application = poll . application
count = application . poll_set . filter ( id__lte = poll_id ) . count ( )
if request . method == ' POST ' :
poll . delete ( )
messages . success ( request , _ ( ' Poll was successfully deleted. ' ) )
else :
del_confirm_form ( request , poll , name = _ ( " the %s . poll " ) % count )
return redirect ( reverse ( ' application_view ' , args = [ application . id ] ) )
2011-09-03 16:20:10 +02:00
@permission_required ( ' application.can_manage_application ' )
2011-07-31 10:46:29 +02:00
@template ( ' application/poll_view.html ' )
def view_poll ( request , poll_id ) :
"""
view a poll for this application .
"""
poll = Poll . objects . get ( pk = poll_id )
2011-09-03 10:52:29 +02:00
ballot = poll . ballot
2011-07-31 10:46:29 +02:00
options = poll . options
2011-09-03 16:20:10 +02:00
if request . user . has_perm ( ' application.can_manage_application ' ) :
2011-07-31 10:46:29 +02:00
if request . method == ' POST ' :
2011-09-03 13:53:37 +02:00
form = PollForm ( request . POST , prefix = " poll " )
2011-07-31 10:46:29 +02:00
if form . is_valid ( ) :
2011-09-03 00:43:09 +02:00
poll . votesinvalid = form . cleaned_data [ ' invalid ' ] or 0
2011-09-03 13:53:37 +02:00
poll . votescast = form . cleaned_data [ ' votescast ' ] or 0
2011-07-31 10:46:29 +02:00
poll . save ( )
for option in options :
option . form = OptionResultForm ( request . POST ,
prefix = " o %d " % option . id )
if option . form . is_valid ( ) :
option . voteyes = option . form . cleaned_data [ ' yes ' ]
option . voteno = option . form . cleaned_data [ ' no ' ] or 0
option . voteundesided = option . form . \
cleaned_data [ ' undesided ' ] or 0
option . save ( )
2011-09-03 16:20:10 +02:00
messages . success ( request , _ ( " Votes are successfully saved. " ) )
if not ' apply ' in request . POST :
return redirect ( reverse ( ' application_view ' , args = [ poll . application . id ] ) )
2011-07-31 10:46:29 +02:00
else :
2011-09-03 13:53:37 +02:00
form = PollForm ( initial = { ' invalid ' : poll . votesinvalid , ' votescast ' : poll . votescast } , prefix = " poll " )
2011-07-31 10:46:29 +02:00
for option in options :
option . form = OptionResultForm ( initial = {
' yes ' : option . voteyes ,
' no ' : option . voteno ,
' undesided ' : option . voteundesided ,
} , prefix = " o %d " % option . id )
return {
' poll ' : poll ,
' form ' : form ,
' options ' : options ,
2011-09-03 10:52:29 +02:00
' ballot ' : ballot ,
2011-07-31 10:46:29 +02:00
}
2011-09-03 18:11:42 +02:00
@permission_required ( ' application.can_manage_application ' )
def permit_version ( request , aversion_id ) :
aversion = AVersion . objects . get ( pk = aversion_id )
application = aversion . application
2011-09-04 10:43:25 +02:00
if request . method == ' POST ' :
application . accept_version ( aversion )
messages . success ( request , _ ( " Version <b> %s </b> accepted. " ) % ( aversion . aid ) )
else :
gen_confirm_form ( request , _ ( ' Do you really want to permit version <b> %s </b>? ' ) % aversion . aid , reverse ( ' application_version_permit ' , args = [ aversion . id ] ) )
2011-09-03 18:11:42 +02:00
return redirect ( reverse ( ' application_view ' , args = [ application . id ] ) )
@permission_required ( ' application.can_manage_application ' )
def reject_version ( request , aversion_id ) :
aversion = AVersion . objects . get ( pk = aversion_id )
2011-09-06 10:41:06 +02:00
application = aversion . application
2011-09-04 19:07:45 +02:00
if request . method == ' POST ' :
if application . reject_version ( aversion ) :
messages . success ( request , _ ( " Version <b> %s </b> rejected. " ) % ( aversion . aid ) )
else :
messages . error ( request , _ ( " ERROR by rejecting the version. " ) )
2011-09-03 18:11:42 +02:00
else :
2011-09-04 19:07:45 +02:00
gen_confirm_form ( request , _ ( ' Do you really want to reject version <b> %s </b>? ' ) % aversion . aid , reverse ( ' application_version_reject ' , args = [ aversion . id ] ) )
2011-09-03 18:11:42 +02:00
return redirect ( reverse ( ' application_view ' , args = [ application . id ] ) )
2011-11-09 22:53:10 +01:00
@permission_required ( ' application.can_manage_applications ' )
@template ( ' application/import.html ' )
def application_import ( request ) :
try :
request . user . profile
messages . error ( request , _ ( ' The import function is available for the superuser (without user profile) only. ' ) )
return redirect ( reverse ( ' application_overview ' ) )
except Profile . DoesNotExist :
pass
2011-11-14 16:37:12 +01:00
except AttributeError :
# AnonymousUser
pass
2011-11-09 22:53:10 +01:00
if request . method == ' POST ' :
form = ApplicationImportForm ( request . POST , request . FILES )
if form . is_valid ( ) :
try :
users_generated = 0
applications_generated = 0
applications_modified = 0
with transaction . commit_on_success ( ) :
2011-11-22 18:49:36 +01:00
dialect = csv . Sniffer ( ) . sniff ( request . FILES [ ' csvfile ' ] . readline ( ) )
2011-11-24 19:38:25 +01:00
dialect = utils . csv_ext . patchup ( dialect )
2011-11-22 18:49:36 +01:00
request . FILES [ ' csvfile ' ] . seek ( 0 )
for ( lno , line ) in enumerate ( csv . reader ( request . FILES [ ' csvfile ' ] , dialect = dialect ) ) :
2011-11-09 22:53:10 +01:00
# basic input verification
if lno < 1 :
continue
try :
( number , title , text , reason , first_name , last_name ) = line [ : 6 ]
except ValueError :
messages . error ( request , _ ( ' Ignoring malformed line %d in import file. ' ) % ( lno + 1 ) )
continue
form = ApplicationForm ( { ' title ' : title , ' text ' : text , ' reason ' : reason } )
if not form . is_valid ( ) :
messages . error ( request , _ ( ' Ignoring malformed line %d in import file. ' ) % ( lno + 1 ) )
continue
if number :
try :
number = abs ( long ( number ) )
if number < 1 :
messages . error ( request , _ ( ' Ignoring malformed line %d in import file. ' ) % ( lno + 1 ) )
continue
except ValueError :
messages . error ( request , _ ( ' Ignoring malformed line %d in import file. ' ) % ( lno + 1 ) )
continue
# fetch existing users or create new users as needed
try :
user = User . objects . get ( first_name = first_name , last_name = last_name )
except User . DoesNotExist :
user = None
if user is None :
user = User ( )
user . last_name = last_name
user . first_name = first_name
user . username = gen_username ( first_name , last_name )
user . save ( )
profile = Profile ( )
profile . user = user
profile . group = ' '
profile . committee = ' '
profile . gender = ' none '
profile . type = ' guest '
2011-11-14 21:54:42 +01:00
profile . firstpassword = gen_password ( )
profile . user . set_password ( profile . firstpassword )
2011-11-09 22:53:10 +01:00
profile . save ( )
users_generated + = 1
# create / modify the application
application = None
if number :
try :
application = Application . objects . get ( number = number )
applications_modified + = 1
except Application . DoesNotExist :
application = None
if application is None :
application = Application ( submitter = user )
if number :
application . number = number
applications_generated + = 1
application . title = form . cleaned_data [ ' title ' ]
application . text = form . cleaned_data [ ' text ' ]
application . reason = form . cleaned_data [ ' reason ' ]
application . save ( user , trivial_change = True )
if applications_generated :
messages . success ( request , ungettext ( ' %d application was successfully imported. ' ,
' %d applications were successfully imported. ' , applications_generated ) % applications_generated )
if applications_modified :
messages . success ( request , ungettext ( ' %d application was successfully modified. ' ,
' %d applications were successfully modified. ' , applications_modified ) % applications_modified )
if users_generated :
messages . success ( request , ungettext ( ' %d new user was added. ' , ' %d new users were added. ' , users_generated ) % users_generated )
return redirect ( reverse ( ' application_overview ' ) )
except csv . Error :
message . error ( request , _ ( ' Import aborted because of severe errors in the input file. ' ) )
else :
messages . error ( request , _ ( ' Please check the form for errors. ' ) )
else :
messages . warning ( request , _ ( " Attention: Existing applications will be modified if you import new applications with the same number. " ) )
messages . warning ( request , _ ( " Attention: Importing an application without a number multiple times will create duplicates. " ) )
form = ApplicationImportForm ( )
return {
' form ' : form ,
}