|
1 #!/usr/bin/python2.5 |
|
2 # |
|
3 # Copyright 2008 the Melange authors. |
|
4 # |
|
5 # Licensed under the Apache License, Version 2.0 (the "License"); |
|
6 # you may not use this file except in compliance with the License. |
|
7 # You may obtain a copy of the License at |
|
8 # |
|
9 # http://www.apache.org/licenses/LICENSE-2.0 |
|
10 # |
|
11 # Unless required by applicable law or agreed to in writing, software |
|
12 # distributed under the License is distributed on an "AS IS" BASIS, |
|
13 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
14 # See the License for the specific language governing permissions and |
|
15 # limitations under the License. |
|
16 |
|
17 """This module contains the view code for Notifications |
|
18 """ |
|
19 |
|
20 __authors__ = [ |
|
21 '"Lennard de Rijk" <ljvderijk@gmail.com>', |
|
22 ] |
|
23 |
|
24 |
|
25 import time |
|
26 |
|
27 from google.appengine.api import users |
|
28 |
|
29 from django import forms |
|
30 from django import http |
|
31 from django.utils.translation import ugettext_lazy |
|
32 |
|
33 from soc.logic import dicts |
|
34 from soc.logic import validate |
|
35 from soc.models import notification as notification_model |
|
36 from soc.views import helper |
|
37 from soc.views import out_of_band |
|
38 from soc.views.helper import access |
|
39 from soc.views.helper import redirects |
|
40 from soc.views.models import base |
|
41 from soc.logic.models import notification as notification_logic |
|
42 from soc.logic.models import user as user_logic |
|
43 |
|
44 |
|
45 class CreateForm(helper.forms.BaseForm): |
|
46 """Form for creating a Notification. |
|
47 """ |
|
48 |
|
49 # to user field |
|
50 to_user = forms.fields.CharField(label='To User') |
|
51 |
|
52 def __init__(self, *args, **kwargs): |
|
53 """ Calls super and then redefines the order in which the fields appear. |
|
54 |
|
55 for parameters see BaseForm.__init__() |
|
56 """ |
|
57 super(CreateForm, self).__init__(*args, **kwargs) |
|
58 |
|
59 # set form fields order |
|
60 self.fields.keyOrder = ['to_user', 'subject', 'message'] |
|
61 |
|
62 class Meta: |
|
63 model = notification_model.Notification |
|
64 |
|
65 # exclude the necessary fields from the form |
|
66 exclude = ['link_id', 'scope', 'scope_path', 'from_user', 'has_been_read'] |
|
67 |
|
68 def clean_to_user(self): |
|
69 """Check if the to_user field has been filled in correctly. |
|
70 """ |
|
71 link_id = self.cleaned_data.get('to_user').lower() |
|
72 |
|
73 if not validate.isLinkIdFormatValid(link_id): |
|
74 raise forms.ValidationError("This link ID is in wrong format.") |
|
75 |
|
76 to_user = user_logic.logic.getForFields({'link_id' : link_id}, unique=True) |
|
77 |
|
78 if not to_user: |
|
79 # user does not exist |
|
80 raise forms.ValidationError("This user does not exist") |
|
81 |
|
82 return link_id |
|
83 |
|
84 |
|
85 class View(base.View): |
|
86 """View methods for the Notification model. |
|
87 """ |
|
88 |
|
89 def __init__(self, params=None): |
|
90 """Defines the fields and methods required for the base View class |
|
91 to provide the user with list, public, create, edit and delete views. |
|
92 |
|
93 Params: |
|
94 params: a dict with params for this View |
|
95 """ |
|
96 |
|
97 new_params = {} |
|
98 new_params['logic'] = notification_logic.logic |
|
99 |
|
100 new_params['name'] = "Notification" |
|
101 new_params['name_short'] = "Notification" |
|
102 new_params['name_plural'] = "Notifications" |
|
103 new_params['url_name'] = "notification" |
|
104 new_params['module_name'] = "notification" |
|
105 |
|
106 new_params['create_form'] = CreateForm |
|
107 |
|
108 # define the django url patterns |
|
109 new_params['django_patterns_defaults'] = [ |
|
110 (r'^%(url_name)s/(?P<access_type>show)/%(key_fields)s$', |
|
111 'soc.views.models.%(module_name)s.public', 'Show %(name_short)s'), |
|
112 (r'^%(url_name)s/(?P<access_type>create)$', |
|
113 'soc.views.models.%(module_name)s.create', 'Create %(name_short)s'), |
|
114 (r'^%(url_name)s/(?P<access_type>create)/%(scope)s$', |
|
115 'soc.views.models.%(module_name)s.create', 'Create %(name_short)s'), |
|
116 (r'^%(url_name)s/(?P<access_type>delete)/%(key_fields)s$', |
|
117 'soc.views.models.%(module_name)s.delete', 'Delete %(name_short)s'), |
|
118 (r'^%(url_name)s/(?P<access_type>list)$', |
|
119 'soc.views.models.%(module_name)s.list', 'List %(name_plural)s'), |
|
120 ] |
|
121 |
|
122 rights = {} |
|
123 rights['unspecified'] = [access.deny] |
|
124 rights['any_access'] = [access.allow] |
|
125 rights['show'] = [access.checkIsMyNotification] |
|
126 rights['delete'] = [access.checkIsDeveloper] |
|
127 rights['list'] = [access.checkIsUser] |
|
128 # create is developer only for the time being to test functionality |
|
129 rights['create'] = [access.checkIsDeveloper] |
|
130 |
|
131 new_params['rights'] = rights |
|
132 |
|
133 params = dicts.merge(params, new_params) |
|
134 |
|
135 super(View, self).__init__(params=params) |
|
136 |
|
137 def create(self, request, access_type, |
|
138 page_name=None, params=None, **kwargs): |
|
139 """On a successful post create redirects the user to the notification list. |
|
140 |
|
141 for parameters see base.create() |
|
142 """ |
|
143 |
|
144 if request.method == 'POST': |
|
145 response = super(View, self).create(request, access_type, |
|
146 page_name, params, **kwargs) |
|
147 |
|
148 if (response.__class__ == http.HttpResponseRedirect and |
|
149 response['location'].startswith( |
|
150 '/%s/edit/' %(self._params['url_name']))): |
|
151 # redirect to list instead of edit view |
|
152 return http.HttpResponseRedirect('/%s/list' %(self._params['url_name'])) |
|
153 else: |
|
154 return response |
|
155 |
|
156 else: |
|
157 # request.method == 'GET' so act normal |
|
158 return super(View, self).create(request, access_type, |
|
159 page_name, params, **kwargs) |
|
160 |
|
161 def list(self, request, access_type, |
|
162 page_name=None, params=None, seed=None, **kwargs): |
|
163 """Lists all notifications that the current logged in user has stored. |
|
164 |
|
165 for parameters see base.list() |
|
166 """ |
|
167 |
|
168 params = dicts.merge(params, self._params) |
|
169 |
|
170 # get the current user |
|
171 properties = {'account': users.get_current_user()} |
|
172 user_entity = user_logic.logic.getForFields(properties, unique=True) |
|
173 |
|
174 # only select the notifications for this user so construct a filter |
|
175 filter = {'scope': user_entity} |
|
176 |
|
177 # create the list parameters |
|
178 list_params = params.copy() |
|
179 |
|
180 # define the list redirect action to show the notification |
|
181 list_params['list_action'] = (redirects.getPublicRedirect, params) |
|
182 list_params['list_description'] = ugettext_lazy( |
|
183 "An overview of your received Notifications.") |
|
184 |
|
185 # TODO(Lennard) when list sorting is implemented sort on descending date |
|
186 |
|
187 # use the generic list method with the filter. The access check in this |
|
188 # method will trigger an errorResponse when user_entity is None |
|
189 return super(View, self).list(request, access_type, |
|
190 page_name, list_params, filter) |
|
191 |
|
192 def _editPost(self, request, entity, fields): |
|
193 """See base.View._editPost(). |
|
194 """ |
|
195 |
|
196 account = users.get_current_user() |
|
197 current_user = user_logic.logic.getForFields({'account': account}, unique=True) |
|
198 |
|
199 to_user = user_logic.logic.getForFields( |
|
200 {'link_id' : fields['to_user']}, unique=True) |
|
201 |
|
202 fields['link_id'] = '%i' %(time.time()) |
|
203 fields['scope'] = to_user |
|
204 fields['from_user'] = current_user |
|
205 fields['scope_path'] = fields['to_user'] |
|
206 |
|
207 def _editSeed(self, request, seed): |
|
208 """Checks if scope_path is seeded and puts it into to_user. |
|
209 |
|
210 for parameters see base._editSeed() |
|
211 """ |
|
212 |
|
213 # if scope_path is present |
|
214 if 'scope_path' in seed.keys(): |
|
215 # fill the to_user field with the scope path |
|
216 seed['to_user'] = seed['scope_path'] |
|
217 |
|
218 def _public(self, request, entity, context): |
|
219 """Marks the Notification as read if that hasn't happened yet. |
|
220 |
|
221 for parameters see base._public() |
|
222 """ |
|
223 |
|
224 # if the user viewing is the user for which this notification is meant |
|
225 # and the notification has not been read yet |
|
226 if not entity.has_been_read: |
|
227 # get the current user |
|
228 account = users.get_current_user() |
|
229 user = user_logic.logic.getForFields({'account': account}, unique=True) |
|
230 |
|
231 if entity.scope.key() == user.key(): |
|
232 # mark the entity as read |
|
233 self._logic.updateModelProperties(entity, {'has_been_read' : True} ) |
|
234 |
|
235 context['entity_type_url'] = self._params['url_name'] |
|
236 context['entity_suffix'] = self._logic.getKeySuffix(entity) |
|
237 |
|
238 |
|
239 view = View() |
|
240 create = view.create |
|
241 edit = view.edit |
|
242 delete = view.delete |
|
243 list = view.list |
|
244 public = view.public |