1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
1298
1299
1300
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
1325
1326
1327
1328
1329
1330
1331
1332
1333
1334
1335
1336
1337
1338
1339
1340
1341
1342
1343
1344
1345
1346
1347
1348
1349
1350
1351
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
1364
1365
1366
1367
1368
1369
1370
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
1384
|
# Copyright 2009-2011 Canonical Ltd. This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).
"""Question views."""
__metaclass__ = type
__all__ = [
'SearchAllQuestionsView',
'QuestionAddView',
'QuestionBreadcrumb',
'QuestionChangeStatusView',
'QuestionConfirmAnswerView',
'QuestionCreateFAQView',
'QuestionEditMenu',
'QuestionEditView',
'QuestionExtrasMenu',
'QuestionHistoryView',
'QuestionLinkFAQView',
'QuestionMessageDisplayView',
'QuestionSetContextMenu',
'QuestionSetNavigation',
'QuestionRejectView',
'QuestionSetView',
'QuestionSubscriptionView',
'QuestionWorkflowView',
]
from operator import attrgetter
import re
from xml.sax.saxutils import escape
from lazr.lifecycle.event import ObjectModifiedEvent
from lazr.lifecycle.snapshot import Snapshot
from lazr.restful.interface import copy_field
from z3c.ptcompat import ViewPageTemplateFile
from zope.app.form.browser import (
TextAreaWidget,
TextWidget,
)
from zope.app.form.browser.widget import renderElement
from zope.component import getUtility
from zope.event import notify
from zope.formlib import form
from zope.interface import (
alsoProvides,
implements,
providedBy,
)
from zope.schema import Choice
from zope.schema.interfaces import IContextSourceBinder
from zope.schema.vocabulary import (
SimpleTerm,
SimpleVocabulary,
)
import zope.security
from canonical.launchpad import _
from canonical.launchpad.helpers import (
is_english_variant,
preferred_or_request_languages,
)
from canonical.launchpad.interfaces.launchpad import ILaunchpadCelebrities
from canonical.launchpad.interfaces.launchpadstatistic import (
ILaunchpadStatisticSet,
)
from canonical.launchpad.webapp import (
ApplicationMenu,
canonical_url,
ContextMenu,
enabled_with_permission,
LaunchpadView,
Link,
Navigation,
NavigationMenu,
redirection,
)
from canonical.launchpad.webapp.authorization import check_permission
from canonical.launchpad.webapp.breadcrumb import Breadcrumb
from canonical.launchpad.webapp.interfaces import IAlwaysSubmittedWidget
from canonical.launchpad.webapp.menu import structured
from canonical.lazr.utils import smartquote
from lp.answers.browser.questiontarget import SearchQuestionsView
from lp.answers.interfaces.faq import IFAQ
from lp.answers.interfaces.faqtarget import IFAQTarget
from lp.answers.interfaces.question import (
IQuestion,
IQuestionAddMessageForm,
IQuestionChangeStatusForm,
IQuestionLinkFAQForm,
)
from lp.answers.interfaces.questioncollection import IQuestionSet
from lp.answers.enums import (
QuestionAction,
QuestionSort,
QuestionStatus,
)
from lp.answers.interfaces.questiontarget import (
IAnswersFrontPageSearchForm,
IQuestionTarget,
)
from lp.app.browser.launchpadform import (
action,
custom_widget,
LaunchpadEditFormView,
LaunchpadFormView,
safe_action,
)
from lp.app.errors import (
NotFoundError,
UnexpectedFormData,
)
from lp.app.widgets.itemswidgets import LaunchpadRadioWidget
from lp.app.widgets.launchpadtarget import LaunchpadTargetWidget
from lp.app.widgets.project import ProjectScopeWidget
from lp.app.widgets.textwidgets import TokensTextWidget
from lp.registry.interfaces.projectgroup import IProjectGroup
from lp.services.propertycache import cachedproperty
class QuestionLinksMixin:
"""A mixin class that provides links used by more than one menu."""
def subscription(self):
"""Return a Link to the subscription view."""
if self.user is not None and self.context.isSubscribed(self.user):
text = 'Unsubscribe'
icon = 'remove'
else:
text = 'Subscribe'
icon = 'mail'
return Link('+subscribe', text, icon=icon)
def edit(self):
"""Return a Link to the edit view."""
text = 'Edit question'
return Link('+edit', text, icon='edit')
class QuestionEditMenu(NavigationMenu, QuestionLinksMixin):
"""A menu for different aspects of editing a object."""
usedfor = IQuestion
facet = 'answers'
title = 'Edit question'
links = ['edit', 'reject', 'subscription']
def reject(self):
"""Return a Link to the reject view."""
enabled = self.user is not None and self.context.canReject(self.user)
text = 'Reject question'
return Link('+reject', text, icon='edit', enabled=enabled)
class QuestionExtrasMenu(ApplicationMenu, QuestionLinksMixin):
"""Context menu of actions that can be performed upon a Question."""
usedfor = IQuestion
facet = 'answers'
links = [
'history', 'linkbug', 'unlinkbug', 'makebug', 'linkfaq',
'createfaq', 'edit', 'changestatus']
def initialize(self):
"""Initialize the menu from the Question's state."""
self.has_bugs = bool(self.context.bugs)
@enabled_with_permission('launchpad.Admin')
def changestatus(self):
"""Return a Link to the change status view."""
return Link('+change-status', _('Change status'), icon='edit')
def history(self):
"""Return a Link to the history view."""
text = 'History'
return Link('+history', text, icon='list',
enabled=bool(self.context.messages))
def linkbug(self):
"""Return a Link to the link bug view."""
text = 'Link existing bug'
return Link('+linkbug', text, icon='add')
def unlinkbug(self):
"""Return a Link to the unlink bug view."""
text = 'Remove bug link'
return Link('+unlinkbug', text, icon='remove', enabled=self.has_bugs)
def makebug(self):
"""Return a Link to the make bug view."""
text = 'Create bug report'
summary = 'Create a bug report from this question.'
return Link('+makebug', text, summary, icon='add',
enabled=not self.has_bugs)
def linkfaq(self):
"""Link for linking to a FAQ."""
text = 'Link to a FAQ'
summary = 'Link this question to a FAQ.'
if self.context.faq is None:
icon = 'add'
else:
icon = 'edit'
return Link('+linkfaq', text, summary, icon=icon)
def createfaq(self):
"""LInk for creating a FAQ."""
text = 'Create a new FAQ'
summary = 'Create a new FAQ from this question.'
return Link('+createfaq', text, summary, icon='add')
class QuestionSetContextMenu(ContextMenu):
"""Context menu of actions that can be preformed upon a QuestionSet."""
usedfor = IQuestionSet
links = ['findproduct', 'finddistro']
def findproduct(self):
"""Return a Link to the find product view."""
text = 'Find upstream project'
return Link('/projects', text, icon='search')
def finddistro(self):
"""Return a Link to the find distribution view."""
text = 'Find distribution'
return Link('/distros', text, icon='search')
class QuestionSetNavigation(Navigation):
"""Navigation for the IQuestionSet."""
usedfor = IQuestionSet
def traverse(self, name):
"""Traverse to a question by id."""
try:
question = getUtility(IQuestionSet).get(int(name))
except ValueError:
question = None
if question is None:
raise NotFoundError(name)
return redirection(canonical_url(question, self.request), status=301)
class QuestionBreadcrumb(Breadcrumb):
"""Builds a breadcrumb for an `IQuestion`."""
@property
def text(self):
return 'Question #%d' % self.context.id
class QuestionSetView(LaunchpadFormView):
"""View for the Answer Tracker index page."""
schema = IAnswersFrontPageSearchForm
custom_widget('scope', ProjectScopeWidget)
page_title = 'Launchpad Answers'
label = 'Questions and Answers'
@property
def scope_css_class(self):
"""The CSS class for used in the scope widget."""
if self.scope_error:
return 'error'
else:
return None
@property
def scope_error(self):
"""The error message for the scope widget."""
return self.getFieldError('scope')
@safe_action
@action('Find Answers', name="search")
def search_action(self, action, data):
"""Redirect to the proper search page based on the scope widget."""
# For the scope to be absent from the form, the user must
# build the query string themselves - most likely because they
# are a bot. In that case we just assume they want to search
# all projects.
scope = self.widgets['scope'].getScope()
if scope is None or scope == 'all':
# Use 'All projects' scope.
scope = self.context
else:
scope = self.widgets['scope'].getInputValue()
self.next_url = "%s/+tickets?%s" % (
canonical_url(scope), self.request['QUERY_STRING'])
@property
def question_count(self):
"""Return the number of questions in the system."""
return getUtility(ILaunchpadStatisticSet).value('question_count')
@property
def answered_question_count(self):
"""Return the number of answered questions in the system."""
return getUtility(ILaunchpadStatisticSet).value(
'answered_question_count')
@property
def solved_question_count(self):
"""Return the number of solved questions in the system."""
return getUtility(ILaunchpadStatisticSet).value(
'solved_question_count')
@property
def projects_with_questions_count(self):
"""Return the number of projects with questions in the system."""
return getUtility(ILaunchpadStatisticSet).value(
'projects_with_questions_count')
@property
def latest_questions_asked(self):
"""Return the 5 latest questions asked."""
return self.context.searchQuestions(
status=QuestionStatus.OPEN, sort=QuestionSort.NEWEST_FIRST)[:5]
@property
def latest_questions_solved(self):
"""Return the 10 latest questions solved."""
# XXX flacoste 2006-11-28: We should probably define a new
# QuestionSort value allowing us to sort on dateanswered descending.
return self.context.searchQuestions(
status=QuestionStatus.SOLVED, sort=QuestionSort.NEWEST_FIRST)[:5]
@property
def most_active_projects(self):
"""Return the 5 most active projects."""
return self.context.getMostActiveProjects(limit=5)
class QuestionSubscriptionView(LaunchpadView):
"""View for subscribing and unsubscribing from a question."""
def initialize(self):
"""Initialize the view from the request form."""
if not self.user or self.request.method != "POST":
# No post, nothing to do
return
question_unmodified = Snapshot(
self.context, providing=providedBy(self.context))
modified_fields = set()
response = self.request.response
# Establish if a subscription form was posted.
newsub = self.request.form.get('subscribe', None)
if newsub is not None:
if newsub == 'Subscribe':
self.context.subscribe(self.user)
response.addNotification(
_("You have subscribed to this question."))
modified_fields.add('subscribers')
elif newsub == 'Unsubscribe':
self.context.unsubscribe(self.user)
response.addNotification(
_("You have unsubscribed from this question."))
modified_fields.add('subscribers')
response.redirect(canonical_url(self.context))
notify(ObjectModifiedEvent(
self.context, question_unmodified, list(modified_fields)))
@property
def page_title(self):
return 'Subscription'
@property
def label(self):
if self.subscription:
return 'Unsubscribe from question'
else:
return 'Subscribe to question'
@property
def subscription(self):
"""Establish if this user has a subscription"""
if self.user is None:
return False
return self.context.isSubscribed(self.user)
class QuestionLanguageVocabularyFactory:
"""Factory for a vocabulary containing a subset of the possible languages.
The vocabulary will contain only the languages "interesting" for the user.
That's English plus the users preferred languages. These will be guessed
from the request when the preferred languages weren't configured.
It also always include the question's current language and excludes all
English variants.
"""
implements(IContextSourceBinder)
def __init__(self, view):
"""Create a QuestionLanguageVocabularyFactory.
:param view: The view that provides the request used to determine the
user languages. The view contains the Product widget selected by the
user in the case where a question is asked in the context of a
ProjectGroup.
"""
self.view = view
def __call__(self, context):
languages = set()
for lang in preferred_or_request_languages(self.view.request):
if not is_english_variant(lang):
languages.add(lang)
if context is not None and IQuestion.providedBy(context):
languages.add(context.language)
languages = list(languages)
# Insert English as the first element, to make it the default one.
english = getUtility(ILaunchpadCelebrities).english
if english in languages:
languages.remove(english)
languages.insert(0, english)
# The vocabulary indicates which languages are supported.
if context is not None and not IProjectGroup.providedBy(context):
question_target = IQuestionTarget(context)
supported_languages = question_target.getSupportedLanguages()
elif (IProjectGroup.providedBy(context) and
self.view.question_target is not None):
# ProjectGroups do not implement IQuestionTarget--the user must
# choose a product while asking a question.
question_target = IQuestionTarget(self.view.question_target)
supported_languages = question_target.getSupportedLanguages()
else:
supported_languages = set([english])
terms = []
for lang in languages:
label = lang.displayname
if lang in supported_languages:
label = "%s *" % label
terms.append(SimpleTerm(lang, lang.code, label))
return SimpleVocabulary(terms)
class QuestionSupportLanguageMixin:
"""Helper mixin for views manipulating the question language.
It provides a method to check if the selected language is supported
and another to create the form field to select the question language.
This mixin adapts its context to IQuestionTarget, so it will work if
the context either provides IQuestionTarget directly or if an adapter
exists.
"""
supported_languages_macros = ViewPageTemplateFile(
'../templates/question-supported-languages-macros.pt')
@property
def chosen_language(self):
"""Return the language chosen by the user."""
if self.widgets['language'].hasInput():
return self.widgets['language'].getInputValue()
else:
return self.context.language
@property
def unsupported_languages_warning(self):
"""Macro displaying a warning in case of unsupported languages."""
macros = self.supported_languages_macros.macros
return macros['unsupported_languages_warning']
@property
def question_target(self):
"""Return the IQuestionTarget related to the context."""
return IQuestionTarget(self.context)
@cachedproperty
def supported_languages(self):
"""Return the list of supported languages ordered by name."""
return sorted(
self.question_target.getSupportedLanguages(),
key=attrgetter('englishname'))
def createLanguageField(self):
"""Create a field with a vocabulary to edit a question language.
:param the_form: The form that will use this field.
:return: A form.Fields instance containing the language field.
"""
return form.Fields(
Choice(
__name__='language',
source=QuestionLanguageVocabularyFactory(view=self),
title=_('Language'),
description=_(
"The language in which this question is written. "
"The languages marked with a star (*) are the "
"languages spoken by at least one answer contact in "
"the community.")),
render_context=self.render_context)
def shouldWarnAboutUnsupportedLanguage(self):
"""Test if the warning about unsupported language should be displayed.
A warning will be displayed if the request's language is not listed
as a spoken language for any of the support contacts. The warning
will only be displayed one time, except if the user changes the
request language to another unsupported value.
"""
if (self.chosen_language in
self.question_target.getSupportedLanguages()):
return False
old_chosen_language = self.request.form.get('chosen_language')
return self.chosen_language.code != old_chosen_language
class QuestionHistoryView(LaunchpadView):
"""A view for listing the history of a question."""
@property
def page_title(self):
return 'History of question #%s' % self.context.id
label = page_title
class QuestionAddView(QuestionSupportLanguageMixin, LaunchpadFormView):
"""Multi-page add view.
The user enters first his question summary and then he is shown a list
of similar results before adding the question.
"""
label = _('Ask a question')
schema = IQuestion
field_names = ['title', 'description']
# The fields displayed on the search page.
search_field_names = ['language', 'title']
custom_widget('title', TextWidget, displayWidth=40, displayMaxWidth=250)
search_template = ViewPageTemplateFile(
'../templates/question-add-search.pt')
add_template = ViewPageTemplateFile('../templates/question-add.pt')
template = search_template
_MAX_SIMILAR_QUESTIONS = 10
_MAX_SIMILAR_FAQS = 10
# Do not autofocus the title widget
initial_focus_widget = None
# The similar items will be held in the following properties.
similar_questions = None
similar_faqs = None
@property
def cancel_url(self):
"""Return the url `IQuestionTarget`."""
return canonical_url(self.context)
def setUpFields(self):
"""Set up the form_fields from the schema and custom_widgets."""
# Add our language field with a vocabulary specialized for
# display purpose.
LaunchpadFormView.setUpFields(self)
self.form_fields = self.createLanguageField() + self.form_fields
def _getFieldsForWidgets(self):
"""Return fields for which need we widgets.
Depending on the action, not all fields are present on the screen,
and need validation.
"""
# Only setup the widgets that needs validation
if not self.add_action.submitted():
fields = self.form_fields.select(*self.search_field_names)
else:
fields = self.form_fields
for field in fields:
if field.__name__ in self.custom_widgets:
field.custom_widget = self.custom_widgets[field.__name__]
return fields
def setUpWidgets(self):
"""Set up the widgets using the view's form fields and the context."""
fields = self._getFieldsForWidgets()
self.widgets = form.setUpWidgets(
fields, self.prefix, self.context, self.request,
data=self.initial_values, ignore_request=False)
def validate(self, data):
"""Validate hook.
This validation method sets the chosen_language attribute.
"""
if 'title' not in data:
self.setFieldError(
'title', _('You must enter a summary of your problem.'))
else:
if len(data['title']) > 250:
self.setFieldError(
'title', _('The summary cannot exceed 250 characters.'))
if self.widgets.get('description'):
if 'description' not in data:
self.setFieldError(
'description',
_('You must provide details about your problem.'))
@property
def page_title(self):
"""The current page title."""
return _('Ask a question about ${context}',
mapping=dict(context=self.context.displayname))
@property
def has_similar_items(self):
"""Return True if similar FAQs or questions were found."""
return self.similar_questions or self.similar_faqs
@action(_('Continue'))
def continue_action(self, action, data):
"""Search for questions and FAQs similar to the entered summary."""
# If the description widget wasn't setup, add it here
if self.widgets.get('description') is None:
self.widgets += form.setUpWidgets(
self.form_fields.select('description'), self.prefix,
self.context, self.request, data=self.initial_values,
ignore_request=False)
faqs = IFAQTarget(self.question_target).findSimilarFAQs(data['title'])
self.similar_faqs = list(faqs[:self._MAX_SIMILAR_FAQS])
questions = self.question_target.findSimilarQuestions(data['title'])
self.similar_questions = list(questions[:self._MAX_SIMILAR_QUESTIONS])
return self.add_template()
def handleAddError(self, action, data, errors):
"""Handle errors on new question creation submission. Either redirect
to the search template when the summary is missing or delegate to
the continue action handler to do the search.
"""
if 'title' not in data:
# Remove the description widget.
widgets = [(True, self.widgets[name])
for name in self.search_field_names]
self.widgets = form.Widgets(widgets, len(self.prefix) + 1)
return self.search_template()
return self.continue_action.success(data)
@action(_('Post Question'), name='add', failure='handleAddError')
def add_action(self, action, data):
"""Add a Question to an `IQuestionTarget`."""
if self.shouldWarnAboutUnsupportedLanguage():
# Warn the user that the language is not supported, so redisplay
# last step.
return self.continue_action.success(data)
question = self.question_target.newQuestion(
self.user, data['title'], data['description'], data['language'])
self.request.response.redirect(canonical_url(question))
return ''
class QuestionChangeStatusView(LaunchpadFormView):
"""View for changing a question status."""
schema = IQuestionChangeStatusForm
label = 'Change question status'
@property
def page_title(self):
return 'Change status of question #%s' % self.context.id
def validate(self, data):
"""Check that the status and message are valid."""
if data.get('status') == self.context.status:
self.setFieldError(
'status', _("You didn't change the status."))
if not data.get('message'):
self.setFieldError(
'message', _('You must provide an explanation message.'))
@property
def initial_values(self):
"""Return the initial view values."""
return {'status': self.context.status}
@action(_('Change Status'), name='change-status')
def change_status_action(self, action, data):
"""Change the Question status."""
self.context.setStatus(self.user, data['status'], data['message'])
self.request.response.addNotification(
_('Question status updated.'))
@property
def next_url(self):
return canonical_url(self.context)
cancel_url = next_url
class QuestionEditView(LaunchpadEditFormView):
"""View for editing a Question."""
schema = IQuestion
label = 'Edit question'
field_names = [
"language", "title", "description", "target", "assignee",
"whiteboard"]
custom_widget('title', TextWidget, displayWidth=40)
custom_widget('whiteboard', TextAreaWidget, height=5)
custom_widget('target', LaunchpadTargetWidget)
@property
def page_title(self):
return 'Edit question #%s details' % self.context.id
label = page_title
def setUpFields(self):
"""Select the subset of fields to display.
- Exclude fields that the user doesn't have permission to modify.
"""
LaunchpadEditFormView.setUpFields(self)
self.form_fields = self.form_fields.omit("distribution",
"sourcepackagename", "product")
editable_fields = []
for field in self.form_fields:
if zope.security.canWrite(self.context, field.__name__):
editable_fields.append(field.__name__)
self.form_fields = self.form_fields.select(*editable_fields)
@action(_("Save Changes"), name="change")
def change_action(self, action, data):
"""Update the Question from the request form data."""
self.updateContextFromData(data)
@property
def next_url(self):
return canonical_url(self.context)
cancel_url = next_url
class QuestionRejectView(LaunchpadFormView):
"""View for rejecting a question."""
schema = IQuestionChangeStatusForm
field_names = ['message']
label = 'Reject question'
@property
def page_title(self):
return 'Reject question #%s' % self.context.id
def validate(self, data):
"""Check that required information was provided."""
if 'message' not in data:
self.setFieldError(
'message', _('You must provide an explanation message.'))
@action(_('Reject Question'), name="reject")
def reject_action(self, action, data):
"""Reject the Question."""
self.context.reject(self.user, data['message'])
self.request.response.addNotification(
_('You have rejected this question.'))
def initialize(self):
"""See `LaunchpadFormView`.
Abort early if the question is already rejected.
"""
if self.context.status == QuestionStatus.INVALID:
self.request.response.addNotification(
_('The question is already rejected.'))
self.request.response.redirect(canonical_url(self.context))
return
LaunchpadFormView.initialize(self)
@property
def next_url(self):
return canonical_url(self.context)
cancel_url = next_url
class LinkFAQMixin:
"""Mixin that contains common functionality for views linking a FAQ."""
@cachedproperty
def faq_target(self):
"""Return the `IFAQTarget` that should be use for this question."""
return IFAQTarget(self.context)
@property
def default_message(self):
"""The default link message to use."""
return '%s suggests this article as an answer to your question:' % (
self.user.displayname)
def getFAQMessageReference(self, faq):
"""Return the reference for the FAQ to use in the linking message."""
return smartquote('FAQ #%s: "%s".' % (faq.id, faq.title))
class QuestionWorkflowView(LaunchpadFormView, LinkFAQMixin):
"""View managing the question workflow action, i.e. action changing
its status.
"""
schema = IQuestionAddMessageForm
# Do not autofocus the message widget.
initial_focus_widget = None
@property
def label(self):
return self.context.title
@property
def page_title(self):
return smartquote('%s question #%d: "%s"') % (
self.context.target.displayname,
self.context.id,
self.context.title)
def setUpFields(self):
"""See `LaunchpadFormView`."""
LaunchpadFormView.setUpFields(self)
if self.context.isSubscribed(self.user):
self.form_fields = self.form_fields.omit('subscribe_me')
def setUpWidgets(self):
"""See `LaunchpadFormView`."""
LaunchpadFormView.setUpWidgets(self)
alsoProvides(self.widgets['message'], IAlwaysSubmittedWidget)
def validate(self, data):
"""Form validatation hook.
When the action is confirm, find and validate the message
that was selected. When another action is used, only make sure
that a message was provided.
"""
if self.confirm_action.submitted():
self.validateConfirmAnswer(data)
else:
if not data.get('message'):
self.setFieldError('message', _('Please enter a message.'))
@property
def lang(self):
"""The Question's language for the lang and xml:lang attributes."""
return self.context.language.dashedcode
@property
def dir(self):
"""The Question's language direction for the dir attribute."""
return self.context.language.abbreviated_text_dir
@property
def is_question_owner(self):
"""Return True when this user is the question owner."""
return self.user == self.context.owner
def hasActions(self):
"""Return True if some actions are possible for this user."""
for action in self.actions:
if action.available():
return True
return False
def canAddComment(self, action):
"""Return whether the comment action should be displayed.
Comments (message without a status change) can be added at any
time by any logged-in user.
"""
return self.user is not None
@action(_('Just Add a Comment'), name='comment', condition=canAddComment)
def comment_action(self, action, data):
"""Add a comment to a resolved question."""
self.context.addComment(self.user, data['message'])
self._addNotificationAndHandlePossibleSubscription(
_('Thanks for your comment.'), data)
@property
def show_call_to_answer(self):
"""Return whether the call to answer should be displayed."""
return (self.user != self.context.owner and
self.context.can_give_answer)
def canAddAnswer(self, action):
"""Return whether the answer action should be displayed."""
return (self.user is not None and
self.user != self.context.owner and
self.context.can_give_answer)
@action(_('Add Answer'), name='answer', condition=canAddAnswer)
def answer_action(self, action, data):
"""Add an answer to the question."""
self.context.giveAnswer(self.user, data['message'])
self._addNotificationAndHandlePossibleSubscription(
_('Thanks for your answer.'), data)
def canSelfAnswer(self, action):
"""Return whether the selfanswer action should be displayed."""
return (self.user == self.context.owner and
self.context.can_give_answer)
@action(_('Problem Solved'), name="selfanswer",
condition=canSelfAnswer)
def selfanswer_action(self, action, data):
"""Action called when the owner provides the solution."""
self.context.giveAnswer(self.user, data['message'])
# Owners frequently solve their questions, but their messages imply
# that another user provided an answer. When a question has answers
# that can be confirmed, suggest to the owner that he use the
# confirmation button.
if self.context.can_confirm_answer:
msgid = _("Your question is solved. If a particular message "
"helped you solve the problem, use the <em>'This "
"solved my problem'</em> button.")
self._addNotificationAndHandlePossibleSubscription(
structured(msgid), data)
def canRequestInfo(self, action):
"""Return if the requestinfo action should be displayed."""
return (self.user is not None and
self.user != self.context.owner and
self.context.can_request_info)
@action(_('Add Information Request'), name='requestinfo',
condition=canRequestInfo)
def requestinfo_action(self, action, data):
"""Add a request for more information to the question."""
self.context.requestInfo(self.user, data['message'])
self._addNotificationAndHandlePossibleSubscription(
_('Thanks for your information request.'), data)
def canGiveInfo(self, action):
"""Return whether the giveinfo action should be displayed."""
return (self.user == self.context.owner and
self.context.can_give_info)
@action(_("I'm Providing More Information"), name='giveinfo',
condition=canGiveInfo)
def giveinfo_action(self, action, data):
"""Give additional informatin on the request."""
self.context.giveInfo(data['message'])
self._addNotificationAndHandlePossibleSubscription(
_('Thanks for adding more information to your question.'), data)
def validateConfirmAnswer(self, data):
"""Make sure that a valid message id was provided as the confirmed
answer."""
# No widget is used for the answer, we are using hidden fields
# in the template for that. So, if the answer is missing, it's
# either a programming error or an invalid handcrafted URL
msgid = self.request.form.get('answer_id')
if msgid is None:
raise UnexpectedFormData('missing answer_id')
try:
data['answer'] = self.context.messages[int(msgid)]
except ValueError:
raise UnexpectedFormData('invalid answer_id: %s' % msgid)
except IndexError:
raise UnexpectedFormData("unknown answer: %s" % msgid)
def canConfirm(self, action):
"""Return whether the confirm action should be displayed."""
return (self.user == self.context.owner and
self.context.can_confirm_answer)
@action(_("This Solved My Problem"), name='confirm',
condition=canConfirm)
def confirm_action(self, action, data):
"""Confirm that an answer solved the request."""
# The confirmation message is not given by the user when the
# 'This Solved My Problem' button on the main question view.
if not data['message']:
data['message'] = 'Thanks %s, that solved my question.' % (
data['answer'].owner.displayname)
self.context.confirmAnswer(data['message'], answer=data['answer'])
self._addNotificationAndHandlePossibleSubscription(
_('Thanks for your feedback.'), data)
def canReopen(self, action):
"""Return whether the reopen action should be displayed."""
return (self.user == self.context.owner and
self.context.can_reopen)
@action(_("I Still Need an Answer"), name='reopen',
condition=canReopen)
def reopen_action(self, action, data):
"""State that the problem is still occuring and provide new
information about it."""
self.context.reopen(data['message'])
self._addNotificationAndHandlePossibleSubscription(
_('Your question was reopened.'), data)
def _addNotificationAndHandlePossibleSubscription(self, message, data):
"""Post-processing work common to all workflow actions.
Adds a notification, subscribe the user if he checked the
'E-mail me...' option and redirect to the question page.
"""
self.request.response.addNotification(message)
if data.get('subscribe_me'):
self.context.subscribe(self.user)
self.request.response.addNotification(
_("You have subscribed to this question."))
self.next_url = canonical_url(self.context)
@property
def new_question_url(self):
"""Return a URL to add a new question for the QuestionTarget."""
return '%s/+addquestion' % canonical_url(self.context.target,
rootsite='answers')
@property
def original_bug(self):
"""Return the bug that the question was created from or None."""
for buglink in self.context.bug_links:
if (check_permission('launchpad.View', buglink.bug)
and buglink.bug.owner == self.context.owner
and buglink.bug.datecreated == self.context.datecreated):
return buglink.bug
return None
class QuestionConfirmAnswerView(QuestionWorkflowView):
"""Specialized workflow view for the +confirm link sent in email
notifications.
"""
def initialize(self):
"""Initialize the view from the Question state."""
# This page is only accessible when a confirmation is possible.
if not self.context.can_confirm_answer:
self.request.response.addErrorNotification(_(
"The question is not in a state where you can confirm "
"an answer."))
self.request.response.redirect(canonical_url(self.context))
return
QuestionWorkflowView.initialize(self)
def getAnswerMessage(self):
"""Return the message that should be confirmed."""
data = {}
self.validateConfirmAnswer(data)
return data['answer']
class QuestionMessageDisplayView(LaunchpadView):
"""View that renders a QuestionMessage in the context of a Question."""
def __init__(self, context, request):
LaunchpadView.__init__(self, context, request)
self.question = context.question
display_confirm_button = True
@cachedproperty
def isBestAnswer(self):
"""Return True when this message is marked as solving the question."""
return (self.context == self.question.answer
and self.context.action in [
QuestionAction.ANSWER, QuestionAction.CONFIRM])
def renderAnswerIdFormElement(self):
"""Return the hidden form element to refer to that message."""
return '<input type="hidden" name="answer_id" value="%d" />' % list(
self.context.question.messages).index(self.context)
def getBodyCSSClass(self):
"""Return the CSS class to use for this message's body."""
if self.isBestAnswer:
return "boardCommentBody highlighted"
else:
return "boardCommentBody"
def canConfirmAnswer(self):
"""Return True if the user can confirm this answer."""
return (self.display_confirm_button and
self.user == self.question.owner and
self.question.can_confirm_answer and
self.context.action == QuestionAction.ANSWER)
def renderWithoutConfirmButton(self):
"""Display the message without any confirm button."""
self.display_confirm_button = False
return self()
class SearchAllQuestionsView(SearchQuestionsView):
"""View that searches among all questions posted on Launchpad."""
display_target_column = True
# Match contiguous digits, optionally prefixed with a '#'.
id_pattern = re.compile('^#?(\d+)$')
@property
def pageheading(self):
"""See `SearchQuestionsView`."""
if self.search_text:
return _('Questions matching "${search_text}"',
mapping=dict(search_text=self.search_text))
else:
return _('Search all questions')
@property
def empty_listing_message(self):
"""See `SearchQuestionsView`."""
if self.search_text:
return _("There are no questions matching "
'"${search_text}" with the requested statuses.',
mapping=dict(search_text=self.search_text))
else:
return _('There are no questions with the requested statuses.')
@safe_action
@action(_('Search'), name='search')
def search_action(self, action, data):
"""Action executed when the user clicked the 'Find Answers' button.
Saves the user submitted search parameters in an instance
attribute and redirects to questions when the term is a question id.
"""
super(SearchAllQuestionsView, self).search_action.success(data)
if not self.search_text:
return
id_matches = SearchAllQuestionsView.id_pattern.match(self.search_text)
if id_matches is not None:
question = getUtility(IQuestionSet).get(id_matches.group(1))
if question is not None:
self.request.response.redirect(canonical_url(question))
class QuestionCreateFAQView(LinkFAQMixin, LaunchpadFormView):
"""View to create a new FAQ."""
schema = IFAQ
label = _('Create a new FAQ')
@property
def page_title(self):
return 'Create a FAQ for %s' % self.context.product.displayname
field_names = ['title', 'keywords', 'content']
custom_widget('keywords', TokensTextWidget)
custom_widget("message", TextAreaWidget, height=5)
@property
def initial_values(self):
"""Fill title and content based on the question."""
question = self.context
return {
'title': question.title,
'content': question.description,
'message': self.default_message,
}
def setUpFields(self):
"""See `LaunchpadFormView`.
Adds a message field to the form.
"""
super(QuestionCreateFAQView, self).setUpFields()
self.form_fields += form.Fields(
copy_field(IQuestionLinkFAQForm['message']))
self.form_fields['message'].field.title = _(
'Additional comment for question #%s' % self.context.id)
self.form_fields['message'].custom_widget = (
self.custom_widgets['message'])
@action(_('Create and Link'), name='create_and_link')
def create_and_link_action(self, action, data):
"""Creates the FAQ and link it to the question."""
faq = self.faq_target.newFAQ(
self.user, data['title'], data['content'],
keywords=data['keywords'])
# Append FAQ link to message.
data['message'] += '\n' + self.getFAQMessageReference(faq)
self.context.linkFAQ(self.user, faq, data['message'])
# Redirect to the question.
self.next_url = canonical_url(self.context)
class SearchableFAQRadioWidget(LaunchpadRadioWidget):
"""Widget combining a set of radio buttons with a search text field.
The search field content is used to filter the vocabulary. The user can
select an element from this set using the radio buttons.
"""
_messageNoValue = _('No existing FAQs are relevant')
searchDisplayWidth = 30
searchButtonLabel = _('Search')
@property
def search_field_name(self):
"""Return the name to use for the search field."""
return self.name + '-query'
@property
def search_button_name(self):
"""Return the name to use for the search button."""
return self.name + '-search'
def renderValue(self, value):
"""Render the widget with the value."""
content = super(SearchableFAQRadioWidget, self).renderValue(value)
return "<br />".join([content, self.renderSearchWidget()])
def renderItemsWithValues(self, values):
"""Render the list of possible values.
Those found in `values` are marked as selected. The list of rendered
values is controlled by the search query. The currently selected
value is always added the the set.
"""
rendered_items = []
rendered_values = set()
count = 0
# Render normal values.
for term in self.vocabulary.searchForTerms(self.getSearchQuery()):
selected = term.value in values
rendered_items.append(self.renderTerm(count, term, selected))
rendered_values.add(term.value)
count += 1
# Some selected values may not be included in the search results;
# insert them at the beginning of the list.
for missing in set(values).difference(rendered_values):
if missing != self._missing:
term = self.vocabulary.getTerm(missing)
rendered_items.insert(0, self.renderTerm(count, term, True))
count += 1
# Display self._messageNoValue radio button since an existing
# FAQ may not be relevant. This logic is copied from
# zope/app/form/browser/itemswidgets.py except that we have
# to prepend the value at the end of this method to prevent
# the insert in the for-loop above from going to the top of the list.
missing = self._toFormValue(self.context.missing_value)
if self._displayItemForMissingValue and not self.context.required:
if missing in values:
render = self.renderSelectedItem
else:
render = self.renderItem
missing_item = render(count,
self.translate(self._messageNoValue),
missing,
self.name,
self.cssClass)
rendered_items.insert(0, missing_item)
count += 1
return rendered_items
def getSearchQuery(self):
"""Return the search query."""
return self.request.form_ng.getOne(
self.search_field_name, self.default_query)
def renderTerm(self, index, term, selected):
"""Render a term as a radio button.
The term's token is used as the radio button label. A link to the
term's value is added beside the button.
"""
id = '%s.%s' % (self.name, index)
attributes = dict(
value=term.token,
id=id,
name=self.name,
cssClass=self.cssClass,
type='radio')
if selected:
attributes['checked'] = 'checked'
input = renderElement(u'input', **attributes)
button = '<label style="font-weight: normal">%s %s:</label>' % (
input, escape(term.token))
link = '<a href="%s">%s</a>' % (
canonical_url(term.value), escape(term.title))
return "\n".join([button, link])
def renderSearchWidget(self):
"""Render the search entry field and the button."""
return " ".join([
self.renderSearchField(),
self.renderSearchButton()])
def renderSearchField(self):
"""Render the search field."""
return renderElement(
'input',
type="text",
cssClass=self.cssClass,
value=self.getSearchQuery(),
name=self.search_field_name,
size=self.searchDisplayWidth)
def renderSearchButton(self):
"""Render the search button."""
return renderElement(
'input',
type='submit',
name=self.search_button_name,
value=self.searchButtonLabel)
class QuestionLinkFAQView(LinkFAQMixin, LaunchpadFormView):
"""View to search for and link an existing FAQ to a question."""
schema = IQuestionLinkFAQForm
custom_widget('faq', SearchableFAQRadioWidget)
custom_widget("message", TextAreaWidget, height=5)
label = _('Is this a FAQ?')
@property
def page_title(self):
return _('Is question #%s a FAQ?' % self.context.id)
@property
def initial_values(self):
"""Sets initial form values."""
return {
'faq': self.context.faq,
'message': self.default_message,
}
def setUpWidgets(self):
"""Set the query on the search widget to the question title."""
super(QuestionLinkFAQView, self).setUpWidgets()
self.widgets['faq'].default_query = self.context.title
def validate(self, data):
"""Make sure that the FAQ link was changed."""
if self.context.faq == data.get('faq'):
self.setFieldError('faq', _("You didn't modify the linked FAQ."))
@action(_('Link to FAQ'), name="link")
def link_action(self, action, data):
"""Link the selected FAQ to the question."""
if data['faq'] is not None:
data['message'] += '\n' + self.getFAQMessageReference(data['faq'])
self.context.linkFAQ(self.user, data['faq'], data['message'])
@property
def next_url(self):
return canonical_url(self.context)
cancel_url = next_url
|