-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathEditing.py
1019 lines (912 loc) · 44.3 KB
/
Editing.py
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
"""
Editing methods.
"""
from __future__ import nested_scopes
import re, string, time
from string import split,join,find,lower,rfind,atoi,strip
from urllib import quote, unquote
from types import *
from itertools import *
from email.Message import Message
from copy import deepcopy
import os.path
import socket
import urllib2
import ZODB # need this for pychecker
from AccessControl import getSecurityManager, ClassSecurityInfo, Unauthorized
from App.Common import rfc1123_date
from DateTime import DateTime
from Globals import InitializeClass
try:
from zope.contenttype import guess_content_type
except ImportError:
try:
from zope.app.content_types import guess_content_type
except ImportError:
from OFS.content_types import guess_content_type
from OFS.DTMLDocument import DTMLDocument
from OFS.ObjectManager import BadRequestException
from OFS.ObjectManager import checkValidId
from zExceptions import BadRequest, Forbidden
import OFS.Image
from plugins.pagetypes import PAGETYPES
from Defaults import DISABLE_JAVASCRIPT, LARGE_FILE_SIZE, LEAVE_PLACEHOLDER, \
ZWIKI_SPAMPATTERNS_URL, ZWIKI_SPAMPATTERNS_TIMEOUT
import Permissions
from Regexps import javascriptexpr, htmlheaderexpr, htmlfooterexpr
from Utils import get_transaction, BLATHER, INFO, parseHeadersBody, isunicode, \
safe_hasattr, stripList
from i18n import _
from Diff import addedtext, textdiff
class PageEditingSupport:
security = ClassSecurityInfo()
security.declareObjectProtected('View')
def checkPermission(self, permission, object):
return getSecurityManager().checkPermission(permission,object)
security.declarePublic('createform') # check permissions at runtime
def create(self,page=None,text='',type=None,title='',REQUEST=None,log='',
sendmail=1, parents=None, subtopics=None, pagename=None): # -> string; ...
"""Create a new wiki page, and optionally do extra stuff. Normally
called via edit().
The page name arguments are confusing. Here's the situation as I
understand it: XXX cleanup
- page: the "original" name of the page we are to create.
- pagename: an alternate spelling of the above, to ease page
management form implementation. Pass one or the other.
- title: optional "new" name to rename to after creation.
This allows us to handle the zwiki and CMF/plone
edit forms smoothly, supporting rename during creation.
Page names are assumed to come url-quoted. For the new page id, we use
a url-safe canonical id derived from the name. (This name and id are
assumed to be available, or Zwiki would not be calling create.)
Other arguments:
- text: initial content for the new page
- type: id of the page type to use (rst, html...)
- REQUEST: standard zope argument, pass this to preserve any
user authentication. Also, may include file upload
data in which case the file is uploaded to the wiki.
- log: optional note for edit history and mail-out subject
- sendmail: sends mail-out to wiki subscribers, unless disabled
- parents: the names of the new page's parent(s), if any
- subtopics: if non-None, sets the subtopics display property
Other features:
- checks the edit_needs_username property as well as permissions
- redirects to the new page, or to the denied view
- returns the new page's name, or None
- if old revisions with this name exist, takes the next revision number
"""
if not self.checkPermission(Permissions.Add, self.folder()):
raise Unauthorized, (
_('You are not authorized to add pages in this wiki.'))
if not self.checkSufficientId(REQUEST):
if REQUEST: REQUEST.RESPONSE.redirect(self.pageUrl()+'/denied')
return None
# here goes.. sequence is important
name = self.tounicode(self.urlunquote(page or pagename or REQUEST.form.get('title')))
title = title and self.tounicode(self.urlunquote(title))
text = text and self.tounicode(text)
log = log and self.tounicode(log)
id = self.canonicalIdFrom(name)
if not name:
return self.genericerror(shorttitle=_('No Page Name'),
messagetitle=_('Please enter a Page Name'),
messages=[_('You did not enter a name for your new page. Please go Back and enter a name in the form.'), _('Your browser should still have your edits.')])
p = self.__class__(__name__=id)
p.title = name # because manage_afterAdd adds this to the wiki outline
p = self.folder()[self.folder()._setObject(id,p)] # place in folder
p.checkForSpam(text) # now we're acquiring wiki options, check for spam
p.ensureMyRevisionNumberIsLatest()
p.setCreator(REQUEST)
p.setLastEditor(REQUEST)
p.setLastLog(log)
p._setOwnership(REQUEST)
# now really update the wiki outline
# XXX should reuse reparent code to do parents validation etc.
p.parents = (parents==None) and [self.pageName()] or parents
self.wikiOutline().add(p.pageName(), p.parents)
p.setPageType(type or self.defaultPageType())
p.setText(text,REQUEST)
p.handleFileUpload(REQUEST)
p.handleSubtopicsProperty(subtopics,REQUEST)
if p.autoSubscriptionEnabled(): p.subscribeThisUser(REQUEST)
# allow users to alter the page name in the creation form. We do
# a full rename after all the above to make sure everything gets
# handled properly, such as updating backlinks. We must first
# commit though, to get p.cb_isMoveable().
if title and title != p.pageName():
get_transaction().note('rename during creation')
get_transaction().commit()
p.handleRename(title,0,1,REQUEST,log)
else: # reindex now all attributes are set. handleRename does it too.
p.index_object()
if sendmail:
p.sendMailToSubscribers(
p.read(), REQUEST=REQUEST, subjectSuffix='', subject=log,
message_id=self.messageIdFromTime(p.creationTime()))
if REQUEST:
try:
u = (REQUEST.get('redirectURL',None) or
REQUEST['URL2']+'/'+ self.urlquote(p.id()))
REQUEST.RESPONSE.redirect(u)
except KeyError: pass
return name
security.declarePublic('edit') # check permissions at runtime
def edit(self, page=None, text=None, type=None, title='',
timeStamp=None, REQUEST=None,
subjectSuffix='', log='', check_conflict=1, # temp (?)
leaveplaceholder=LEAVE_PLACEHOLDER, updatebacklinks=1,
subtopics=None):
"""General-purpose method for editing & creating zwiki pages.
This method does a lot; combining all this stuff in one powerful
method simplifies the skin layer above, I think.
- changes the text and/or formatting type of this (or another )
page, or creates that page if it doesn't exist.
- when called from a time-stamped web form, detects and warn when
two people attempt to work on a page at the same time
- The username (authenticated user or zwiki_username cookie) and
ip address are saved in page's last_editor, last_editor_ip
attributes if a change is made
- If the text begins with "DeleteMe", delete this page
- If file has been submitted in REQUEST, create a file or image
object and link or inline it on the current page.
- if title differs from page, assume it is the new page name and
do a rename (the argument remains"title" for backwards compatibility)
- may set, clear or remove this page's show_subtopics property
- sends mail notification to subscribers if appropriate
"""
page = page and self.tounicode(self.urlunquote(page))
title = title and self.tounicode(self.urlunquote(title))
text = text and self.tounicode(text)
log = log and self.tounicode(log)
subjectSuffix = subjectSuffix and self.tounicode(subjectSuffix)
# what are we doing ?
if page: page = unquote(page)
if page is None: # changing this page
p = self
elif self.pageWithNameOrId(page): # changing another page
p = self.pageWithNameOrId(page)
else: # creating a new page
return self.create(page,
text or '',
type,
title,
REQUEST,
log,
subtopics=subtopics)
if not self.checkSufficientId(REQUEST):
return self.denied(
_("Sorry, this wiki doesn't allow anonymous edits. Please configure a username in options first."))
if check_conflict:
if self.checkEditConflict(timeStamp, REQUEST):
return self.editConflictDialog()
if self.isDavLocked():
return self.davLockDialog()
# each of these handlers checks relevant permissions and does the necessary
if p.handleDeleteMe(text,REQUEST,log): return
p.saveRevision()
p.handleEditPageType(type,REQUEST,log)
if text != None: p.handleEditText(text,REQUEST,subjectSuffix,log)
p.handleSubtopicsProperty(subtopics,REQUEST)
p.handleFileUpload(REQUEST,log)
p.handleRename(title,leaveplaceholder,updatebacklinks,REQUEST,log)
p.index_object()
if REQUEST:
try:
REQUEST.RESPONSE.redirect(
(REQUEST.get('redirectURL',None) or
REQUEST['URL2']+'/'+ self.urlquote(p.id())))
except KeyError: pass
# This alternate spelling exists so that we can define an "edit" alias
# in Plone 3, needed to work around a createObject bug
update = edit
security.declareProtected(Permissions.Comment, 'comment')
def comment(self, text='', username='', time='',
note=None, use_heading=None,
REQUEST=None, subject_heading='', message_id=None,
in_reply_to=None, exclude_address=None, sendmail=1):
"""Add a comment to this page.
We try to do this efficiently, avoiding re-rendering the full page
if possible. The comment will be mailed out to any subscribers.
If auto-subscription is in effect, we subscribe the poster to this
page.
subject_heading is so named to avoid a clash with some existing
zope subject attribute. note and use_heading are not used and
kept only for backwards compatibility.
"""
if not self.checkSufficientId(REQUEST):
return self.denied(
_("Sorry, this wiki doesn't allow anonymous edits. Please configure a username in options first."))
if self.isDavLocked(): return self.davLockDialog()
# gather info
oldtext = self.read()
text = text and self.cleanupText(text)
subject_heading = subject_heading and self.cleanupText(subject_heading)
if not username:
username = self.usernameFrom(REQUEST)
if re.match(r'^(?:\d{1,3}\.){3}\d{1,3}$',username): username = ''
username = username and self.tounicode(username)
firstcomment = self.messageCount()==0
# ensure the page comment and mail-out will have the same
# message-id, and the same timestamp if possible (helps threading
# & troubleshooting)
if time: dtime = DateTime(time)
else:
dtime = self.ZopeTime()
time = dtime.rfc822()
if not message_id: message_id = self.messageIdFromTime(dtime)
# format this comment as standard rfc2822
m = Message()
m.set_charset(self.encoding())
m.set_payload(self.toencoded(text))
m['From'] = self.toencoded(username)
m['Date'] = time
m['Subject'] = self.toencoded(subject_heading)
m['Message-ID'] = message_id
if in_reply_to: m['In-Reply-To'] = in_reply_to
m.set_unixfrom(self.fromLineFrom(m['From'],m['Date'])[:-1])
t = self.tounicode(str(m))
# discard junk comments
if not (m['Subject'] or m.get_payload()): return
self.checkForSpam(t)
# do it
self.saveRevision()
# append to the raw source
t = '\n\n' + t
self.raw += t
# and to the _prerendered cache, carefully mimicking a full
# prerender. This works with current page types at least.
t = self.pageType().preRenderMessage(self,m)
if firstcomment: t=self.pageType().discussionSeparator(self) + t
t = self.pageType().preRender(self,t)
self.setPreRendered(self.preRendered()+t)
self.cookDtmlIfNeeded()
# extras
self.setLastEditor(REQUEST)
self.setLastLog(subject_heading)
if self.autoSubscriptionEnabled(): self.subscribeThisUser(REQUEST)
self.index_object()
if REQUEST: REQUEST.cookies['zwiki_username'] = m['From'] # use real from address
if sendmail:
self.sendMailToSubscribers(
m.get_payload(), REQUEST, subject=m['Subject'],
message_id=m['Message-ID'], in_reply_to=m['In-Reply-To'],
exclude_address=exclude_address)
if REQUEST: REQUEST.RESPONSE.redirect(REQUEST['URL1']+'#bottom')
security.declareProtected(Permissions.Comment, 'append')
def append(self, text='', separator='\n\n', REQUEST=None, log=''):
"""Appends some text to an existing wiki page and scrolls to the
bottom. May cause subscriber mail notifications.
"""
if not text: return
if REQUEST: REQUEST.set('redirectURL',REQUEST['URL1']+'#bottom')
self.edit(text=self.read()+separator+str(text), REQUEST=REQUEST,log=log)
def handleSubtopicsProperty(self,subtopics,REQUEST=None): # -> none ; modifies: self
if subtopics is None: return
if not self.checkPermission(Permissions.Reparent, self):
raise Unauthorized, (
_('You are not authorized to reparent or change subtopics links on this ZWiki Page.'))
subtopics = int(subtopics or '0')
self.setSubtopicsPropertyStatus(subtopics,REQUEST)
def handleEditPageType(self,type,REQUEST=None,log=''):
if not type or type==self.pageTypeId(): return
if not self.checkPermission(Permissions.ChangeType,self):
raise Unauthorized, (_("You are not authorized to change this ZWiki Page's type."))
self.setPageType(type)
self.preRender(clear_cache=1)
self.setLastEditor(REQUEST)
self.setLastLog(log)
security.declarePrivate('setLastLog')
def setLastLog(self,log):
"""Save an edit log message, if provided."""
if log and string.strip(log):
log = string.strip(log)
get_transaction().note('"%s"' % self.toencoded(log))
self.last_log = log
else:
self.last_log = ''
def lastLog(self):
"""Accessor for this page's last edit log message."""
return self.last_log or ''
def handleEditText(self,text,REQUEST=None, subjectSuffix='', log=''):
old = self.read()
new = self.cleanupText(text)
if new == old: return
appending = find(new,old)==0
if not (self.checkPermission(Permissions.Edit,self) or
(appending and self.checkPermission(Permissions.Comment,self))):
raise Unauthorized, (
_('You are not authorized to edit this ZWiki Page.'))
self.checkForSpam(addedtext(old, new))
# do it
self.setText(text,REQUEST)
self.setLastEditor(REQUEST)
self.setLastLog(log)
self.sendMailToEditSubscribers(
textdiff(a=old,b=self.read()),
REQUEST=REQUEST,
subject='(edit) %s' % log)
def handleRename(self,newname,leaveplaceholder,updatebacklinks,
REQUEST=None,log=''):
return self.rename(newname, leaveplaceholder=leaveplaceholder,
updatebacklinks=updatebacklinks, REQUEST=REQUEST)
def handleDeleteMe(self,text,REQUEST=None,log=''):
if not text or not re.match('(?m)^DeleteMe', text):
return 0
if not self.checkPermission(Permissions.Edit, self):
raise Unauthorized, (
_('You are not authorized to edit this ZWiki Page.'))
if not self.checkPermission(Permissions.Delete, self):
raise Unauthorized, (
_('You are not authorized to delete this ZWiki Page.'))
self.setLastLog(log)
self.delete(REQUEST=REQUEST)
return 1 # terminate edit processing
security.declareProtected(Permissions.Delete, 'delete')
def delete(self,REQUEST=None, pagename=None):
"""Delete this page, after saving a final revision.
Any subtopics will be reparented. If the pagename argument is
provided (so named to help the page management form) we will try
to redirect all incoming wiki links there instead, similar to a
rename.
"""
oldname,oldid = self.pageName(),self.getId()
self.reparentChildren(self.primaryParentName())
if pagename and strip(pagename):
self._replaceLinksEverywhere(oldname,pagename,REQUEST)
self.saveRevision()
# figure out where to go afterward - up, or to default page (which may change)
redirecturl = self.primaryParent() and self.primaryParentUrl() or None
self.folder().manage_delObjects([self.getId()])
redirecturl = redirecturl or self.defaultPageUrl()
self.sendMailToEditSubscribers(
'This page was deleted.\n',
REQUEST=REQUEST,
subjectSuffix='',
subject='(deleted)')
if REQUEST: REQUEST.RESPONSE.redirect(redirecturl)
security.declareProtected(Permissions.Edit, 'revert')
def revert(self, rev, REQUEST=None):
"""Revert this page to the specified revision's state, updating history.
We do this by looking at the old page revision object and applying
a corrective edit to the current page. This records a new page
revision, reverts text, reverts parents, sends a mailout, etc.
Last editor, last editor ip, last edit time are updated, cf #1157.
Page renames are not reverted since the new revisions
implementation, stand by.
"""
if not self.checkSufficientId(REQUEST):
return self.denied(
_("Sorry, this wiki doesn't allow anonymous edits. Please configure a username in options first."))
if not rev: return
rev = int(rev)
old = self.revision(rev)
if not old: return
reparenting = self.getParents() != old.getParents()
if reparenting and not self.checkPermission(Permissions.Reparent, self):
raise Unauthorized, (
_('You are not authorized to reparent this ZWiki Page.'))
# do it
self.saveRevision()
self.setText(old.text())
self.setPageType(old.pageTypeId())
self.setVotes(old.votes())
if reparenting:
self.setParents(old.getParents())
self.updateWikiOutline()
self.setLastEditor(REQUEST) #seems better than self.setLastEditorLike(old)
self.setLastLog('reverted by %s' % self.usernameFrom(REQUEST))
self.index_object()
self.sendMailToEditSubscribers(
'This page was reverted to the %s version.\n' % old.last_edit_time,
REQUEST=REQUEST,subjectSuffix='',subject='(reverted)')
if REQUEST is not None: REQUEST.RESPONSE.redirect(self.pageUrl())
security.declareProtected(Permissions.manage_properties, 'expunge')
def expunge(self, rev, REQUEST=None):
"""Revert myself to the specified revision, discarding later history."""
if not rev: return
rev = int(rev)
oldrevs = self.oldRevisionNumbers()
if not rev in oldrevs: return
id = self.getId()
def replaceMyselfWith(o): # in zodb (self is not changed)
self.folder()._delObject(id)
self.folder()._setObject(id,o)
def replaceMyselfWithRev(r):
newself = self.revision(rev)
newself._setId(id)
replaceMyselfWith(newself)
def deleteRevsSince(r):
for r in oldrevs[oldrevs.index(rev):]:
self.revisionsFolder()._delObject('%s.%d' % (id,r))
replaceMyselfWithRev(rev)
deleteRevsSince(rev)
BLATHER('expunged %s history after revision %d' % (id,rev))
if REQUEST is not None: REQUEST.RESPONSE.redirect(self.pageUrl())
security.declareProtected(Permissions.manage_properties, 'expungeEditsBy')
def expungeEditsBy(self, username, REQUEST=None):
"""Expunge all my recent edits by username, if any."""
self.expunge(self.revisionNumberBefore(username), REQUEST=REQUEST)
security.declareProtected(Permissions.manage_properties, 'expungeEditsEverywhereBy')
def expungeEditsEverywhereBy(self, username, REQUEST=None, batch=0): # -> none
# depends on: all pages, revisions ; modifies: all pages, revisions
"""Expunge all the most recent edits by username throughout the wiki.
This is a powerful spam repair tool for managers. It removes all
recent consecutive edits by username from each page in the
wiki. The corresponding revisions will disappear from the page
history. See #1157.
Should this use the catalog ? Currently uses a more expensive and
failsafe brute force ZODB search.
"""
batch = int(batch)
for n,p in izip(count(1),
[p for p in self.pageObjects() if p.last_editor==username]):
try:
p.expungeEditsBy(username,REQUEST=REQUEST)
except IndexError:
BLATHER('failed to expunge edits by %s at %s: %s' \
% (username,p.id(),formattedTraceback()))
if batch and (n % batch)==0:
BLATHER('committing after %d expunges' % n)
get_transaction().commit()
security.declareProtected(Permissions.manage_properties, 'expungeLastEditor')
def expungeLastEditor(self, REQUEST=None):
"""Expunge all the recent edits to this page by the last editor."""
self.expungeEditsBy(self.last_editor, REQUEST=REQUEST)
security.declareProtected(Permissions.manage_properties, 'expungeLastEditorEverywhere')
def expungeLastEditorEverywhere(self, REQUEST=None):
"""Expunge all the recent edits by this page's last editor throughout the wiki."""
self.expungeEditsEverywhereBy(self.last_editor, REQUEST=REQUEST)
def ensureTitle(self):
if not self.title: self.title = self.getId()
security.declareProtected(Permissions.Rename, 'rename')
def rename(self,pagename,leaveplaceholder=LEAVE_PLACEHOLDER,
updatebacklinks=1,sendmail=1,REQUEST=None):
"""Rename this page, if permissions allow. Also ensure name/id
conformance, keep our children intact, and optionally
- leave a placeholder page
- update links to this page throughout the wiki. Warning, this is
not 100% reliable.
- notify subscribers. Note the sendmail arg can stop the primary
creation mailout but not the ones that may result from
updatebacklinks.
"""
oldname, oldid = self.pageName(), self.getId()
oldnameisfreeform = oldname != oldid
def clean(s): return re.sub(r'[\r\n]','',s)
newname = clean(self.tounicode(pagename))
newid = self.canonicalIdFrom(newname)
namechanged, idchanged = newname != oldname, newid != oldid
if not newname or not (namechanged or idchanged): return
# sequence is important here
BLATHER('renaming %s (%s) to %s (%s)...' % (
self.toencoded(oldname),oldid,self.toencoded(newname),newid))
self.ensureTitle()
if idchanged:
self.changeIdCarefully(newid)
if namechanged:
self.changeNameCarefully(newname)
if (idchanged or namechanged) and updatebacklinks:
self._replaceLinksEverywhere(oldname,newname,REQUEST)
self.index_object() # update catalog XXX manage_renameObject may also, if idchanged
if idchanged and leaveplaceholder:
try: self._makePlaceholder(oldid,newname)
except BadRequestException:
# special case for CMF/Plone: we'll end up here when first
# saving a page that was created via the CMS ui - we can't
# save a placeholder page since the canonical ID hasn't
# really changed
pass
if namechanged and sendmail:
self._sendRenameNotification(oldname,newname,REQUEST)
BLATHER('rename complete')
if REQUEST: REQUEST.RESPONSE.redirect(self.pageUrl())
def _makePlaceholder(self,oldid,newname):
self.create(
oldid,
_("This page was renamed to [%s].\n") % (newname),
sendmail=0)
def _sendRenameNotification(self,oldname,newname,REQUEST):
self.sendMailToEditSubscribers(
'This page was renamed from %s to %s.\n'%(oldname,newname),
REQUEST=REQUEST,
subjectSuffix='',
subject='(renamed)')
def changeNameCarefully(self,newname):
"""Change this page's name, preserving important info."""
self.reparentChildren(newname)
self.wikiOutline().replace(self.pageName(),newname)
self.title = newname
def changeIdCarefully(self,newid):
"""Change this page's id, preserving important info."""
# this object will be replaced with another
creation_time, creator, creator_ip = \
self.creation_time, self.creator, self.creator_ip
# manage_after* has the effect of losing our place in the hierarchy
parentmap = deepcopy(self.wikiOutline().parentmap())
childmap = deepcopy(self.wikiOutline().childmap())
self.folder().manage_renameObject(self.getId(),newid)
self.creation_time, self.creator, self.creator_ip = \
creation_time, creator, creator_ip
self.wikiOutline().setParentmap(parentmap)
self.wikiOutline().setChildmap(childmap)
self.ensureMyRevisionNumberIsLatest()
def reparentChildren(self,newparent):
children = self.childrenIdsAsList()
if children:
BLATHER('reparenting children of',self.getId())
for id in children:
child = getattr(self.folder(),id) # XXX poor caching
child.removeParent(self.pageName())
child.addParent(newparent)
child.index_object() # XXX need only reindex parents
def _replaceLinksEverywhere(self,oldlink,newlink,REQUEST=None):
"""Replace one link with another throughout the wiki.
Freeform links should not be enclosed in brackets.
Comes with an appropriately big scary-sounding name. See
_replaceLinks for more.
"""
BLATHER('replacing all %s links with %s' % (oldlink,newlink))
for p in self.backlinksFor(self.canonicalIdFrom(oldlink)):
# this is an extensive, risky operation which can fail for
# a number of reasons - carry on regardless so we don't
# block renames
# poor caching
try: p.getObject()._replaceLinks(oldlink,newlink,REQUEST)
except:
BLATHER('_replaceLinks failed to update %s links in %s' \
% (oldlink,p.id))
def _replaceLinks(self,oldlink,newlink,REQUEST=None): # modifies: self.text
text = self.text()
replacement_text = self._replaceLinksInSourceText(oldlink,newlink,text)
if replacement_text != text:
self.edit(text=replacement_text, REQUEST=REQUEST)
def folderContains(self,folder,id):
"""check folder contents safely, without acquiring"""
return safe_hasattr(folder.aq_base,id)
def uploadFolder(self):
"""Where to store uploaded files (an 'uploads' subfolder if
present, otherwise the wiki folder)."""
f = self.folder()
if (self.folderContains(f,'uploads') and f.uploads.isPrincipiaFolderish):
f = f.uploads
return f
def checkUploadPermissions(self):
"""Raise an exception if the current user does not have permission
to upload to this wiki page."""
if not (self.checkPermission(Permissions.Upload,self.uploadFolder())):
raise Unauthorized, (_('You are not authorized to upload files here.'))
if not (self.checkPermission(Permissions.Edit, self) or
self.checkPermission(Permissions.Comment, self)):
raise Unauthorized, (_('You are not authorized to add a link on this ZWiki Page.'))
def requestHasFile(self,r):
return (r and safe_hasattr(r,'file') and safe_hasattr(r.file,'filename') and r.file.filename)
def _sendUploadNotification(self,newid,REQUEST):
self.sendMailToEditSubscribers(
'Uploaded file "%s" on page "%s".\n' % (newid,self.pageName()),
REQUEST=REQUEST,
subjectSuffix='',
subject='(upload)')
def handleFileUpload(self,REQUEST,log=''):
if not self.requestHasFile(REQUEST): return
self.checkUploadPermissions()
newid = self._addFileFromRequest(REQUEST,log=log)
if not newid: raise Exception, (_('Sorry, file creation failed for some reason.'))
self._sendUploadNotification(newid,REQUEST)
def _addFileFromRequest(self,REQUEST,log=''):
"""Add and link a new File or Image object, depending on file's filename
suffix. Returns a tuple containing the new id, content type &
size, or (None,None,None).
"""
# ensure we have a suitable file, id and title
file, title = REQUEST.file, str(REQUEST.get('title',''))
id, title = OFS.Image.cookId('', title, file)
if not id: return None
folder = self.uploadFolder()
try: checkValidId(folder,id,allow_dup=1)
except BadRequest:
id, ext = os.path.splitext(id)
id = self.canonicalIdFrom(id)
id = id + ext
# create the file or image object, unless it already exists
# XXX should use CMF/Plone content types when appropriate
if not (self.folderContains(folder,id) and
folder[id].meta_type in ('File','Image')): #'Portal File','Portal Image')):
if guess_content_type(file.filename)[0][0:5] == 'image':
# if self.inCMF():
# #from Products.ATContentTypes import ATFile, ATImage
# #folder.create(ATImage) ...
# else:
id = folder._setObject(id, OFS.Image.Image(id,title,''))
else:
# if self.inCMF():
# else:
id = folder._setObject(id, OFS.Image.File(id,title,''))
# adding the data after creation is more efficient, reportedly
ob = folder._getOb(id)
ob.manage_upload(file)
# and link/inline it
self._addFileOrImageToPage(id,ob.content_type,ob.getSize(),log,REQUEST)
return id
def _addFileOrImageToPage(self, id, content_type, size, log, REQUEST):
"""Add a file link or image to this page, unless it's already
there. Inline images which are not too big.
"""
alreadythere = re.search(r'(src|href)="%s"' % id,self.text())
if alreadythere: return
def fileOrImageLink():
folderurl = self.uploadFolder().absolute_url()
shouldinline = (
(content_type.startswith('image')
and not (safe_hasattr(REQUEST,'dontinline') and REQUEST.dontinline)
and size <= LARGE_FILE_SIZE))
if shouldinline:
return self.pageType().inlineImage(self,id,folderurl+'/'+id)
else:
return self.pageType().linkFile(self,id,folderurl+'/'+id)
def appendQuietly(linktxt,log,REQUEST):
self.setText(self.read()+linktxt,REQUEST)
self.setLastEditor(REQUEST)
self.setLastLog(log)
self.index_object()
appendQuietly(fileOrImageLink(),log,REQUEST)
def _setOwnership(self, REQUEST=None):
"""Set appropriate ownership for a new page. To help control
executable content, we make sure the new page acquires it's owner
from the parent folder.
"""
self._deleteOwnershipAfterAdd()
# for IssueNo0157
_old_read = DTMLDocument.read
security.declareProtected(Permissions.View, 'read')
def read(self):
return re.sub('<!--antidecapitationkludge-->\n\n?','',
self._old_read())
security.declareProtected(Permissions.View, 'text')
def text(self, REQUEST=None, RESPONSE=None):
"""
Return this page's source text, with text/plain content type.
(a permission-free version of document_src)
# XXX security ?
"""
if RESPONSE is not None:
RESPONSE.setHeader('Content-Type', 'text/plain; charset=utf-8')
#RESPONSE.setHeader('Last-Modified', rfc1123_date(self._p_mtime))
return self.read()
# XXX or self.toencoded(self.read()) ? used as both an internal and
# external fn
def setText(self, text='', REQUEST=None):
"""
Change this page's text.
Does cleanups and triggers pre-rendering and DTML re-parsing.
"""
self.raw = self.cleanupText(text)
self.preRender(clear_cache=1)
# re-cook DTML's cached parse data if necessary
# will prevent edit if DTML can't parse.. hopefully no auth trouble
self.cookDtmlIfNeeded()
# try running the DTML, to prevent edits if DTML can't execute
# got authorization problems, commit didn't help..
#if self.supportsDtml() and self.dtmlAllowed():
# #get_transaction().commit()
# DTMLDocument.__call__(self,self,REQUEST,REQUEST.RESPONSE)
def checkForSpam(self, t=''):
"""Check the current request and any provided text for signs
of spam, and raise an error if found.
"""
REQUEST = getattr(self,'REQUEST',None)
ip = getattr(REQUEST,'REMOTE_ADDR','')
username = self.usernameFrom(REQUEST,ip_address=0)
path = self.getPath()
def forbid(reason):
BLATHER('%s blocked edit from %s (%s), %s:\n%s' % (path, ip, username, reason, t))
raise Forbidden, "There was a problem, please contact the site admin."
# content matches a banned pattern ?
for pat in self.getSpamPatterns():
if re.search(pat,t): forbid("spam pattern found")
def getSpamPatterns(self):
"""Fetch spam patterns from the global zwiki spam blacklist,
or a local property. Returns a list of stripped non-empty
regular expression strings.
"""
if safe_hasattr(self.folder(), 'spampatterns'):
return list(getattr(self.folder(),'spampatterns',[]))
else:
BLATHER('checking zwiki.org spam blacklist')
req = urllib2.Request(
ZWIKI_SPAMPATTERNS_URL,
None,
{'User-Agent':'Zwiki %s' % self.zwiki_version()}
)
# have to set timeout this way for python 2.4. XXX safe ?
saved = socket.getdefaulttimeout()
socket.setdefaulttimeout(ZWIKI_SPAMPATTERNS_TIMEOUT)
try:
try:
response = urllib2.urlopen(req)
t = response.read()
except urllib2.URLError, e:
BLATHER('failed to read blacklist, skipping (%s)' % e)
t = ''
finally:
socket.setdefaulttimeout(saved)
return self.parseSpamPatterns(t)
def parseSpamPatterns(self, t):
"""Parse the contents of spampatterns.txt, returning any
patterns as a list of strings.
spampatterns.txt version 1 may contain:
- comments - lines beginning with #
- whitespace at the start or end of lines, or blank lines
- a line of the form "zwiki-spampatterns-version: 1"; assumed if not present
"""
return [p for p in stripList(t.split('\n')) if not (p.startswith('#') or p.startswith('zwiki-spampatterns-version:'))]
def cleanupText(self, t):
"""Clean up incoming text and convert to unicode for internal use."""
def stripcr(t): return re.sub('\r\n','\n',t)
def disablejs(t): return re.sub(javascriptexpr,r'<disabled \1>',t)
return disablejs(stripcr(self.tounicode(t)))
def lastEditor(self):return self.tounicode(self.last_editor)
def lastEditorIp(self):return self.last_editor_ip
def setLastEditor(self, REQUEST=None):
"""Record last editor info based on the current REQUEST and time."""
if REQUEST:
self.last_editor_ip = REQUEST.REMOTE_ADDR
self.last_editor = self.usernameFrom(REQUEST)
else:
# this has been fiddled with before
# if we have no REQUEST, at least update last editor
self.last_editor_ip = ''
self.last_editor = ''
self.last_edit_time = DateTime().ISO8601()
def setLastEditorLike(self,p):
"""Copy last editor info from p."""
self.last_editor = p.last_editor
self.last_editor_ip = p.last_editor_ip
self.last_edit_time = p.last_edit_time
def hasCreatorInfo(self):
"""True if this page already has creator attributes."""
return (safe_hasattr(self,'creator') and
safe_hasattr(self,'creation_time') and
safe_hasattr(self,'creator_ip'))
def setCreator(self, REQUEST=None):
"""Record my creator, creator_ip & creation_time."""
self.creation_time = DateTime().ISO8601()
if REQUEST:
self.creator_ip = REQUEST.REMOTE_ADDR
self.creator = self.usernameFrom(REQUEST)
else:
self.creator_ip = ''
self.creator = ''
def setCreatorLike(self,p):
"""Copy creator info from p."""
self.creator = p.creator
self.creator_ip = p.creator_ip
self.creation_time = p.creation_time
security.declareProtected(Permissions.View, 'checkEditConflict')
def checkEditConflict(self, timeStamp, REQUEST):
"""
Warn if this edit would be in conflict with another.
Edit conflict checking based on timestamps -
things to consider: what if
- we are behind a proxy so all ip's are the same ?
- several people use the same cookie-based username ?
- people use the same cookie-name as an existing member name ?
- no-one is using usernames ?
strategies:
0. no conflict checking
1. strict - require a matching timestamp. Safest but obstructs a
user trying to backtrack & re-edit. This was the behaviour of
early zwiki versions.
2. semi-careful - record username & ip address with the timestamp,
require a matching timestamp or matching non-anonymous username
and ip. There will be no conflict checking amongst users with the
same username (authenticated or cookie) connecting via proxy.
Anonymous users will experience strict checking until they
configure a username.
3. relaxed - require a matching timestamp or a matching, possibly
anonymous, username and ip. There will be no conflict checking
amongst anonymous users connecting via proxy. This is the current
behaviour.
"""
username = self.usernameFrom(REQUEST)
if (timeStamp is not None and
timeStamp != self.timeStamp() and
(not safe_hasattr(self,'last_editor') or
not safe_hasattr(self,'last_editor_ip') or
username != self.lastEditor() or
REQUEST.REMOTE_ADDR != self.lastEditorIp())):
return 1
else:
return 0
security.declareProtected(Permissions.View, 'timeStamp')
def timeStamp(self):
return str(self._p_mtime)
security.declareProtected(Permissions.FTP, 'manage_FTPget')
def manage_FTPget(self):
"""
Get source for FTP download.
"""
#candidates = list(self.allowedPageTypes())
#types = "%s (alternatives:" % self.pageTypeId()
#if self.pageTypeId() in candidates:
# candidates.remove(self.pageTypeId())
#for i in candidates:
# types = types + " %s" % i
#types = types + ")"
types = "%s" % self.pageTypeId()
return "Wiki-Safetybelt: %s\nType: %s\nLog: \n\n%s" % (
self.timeStamp(), types, self.toencoded(self.read()))
security.declarePublic('isDavLocked')
def isDavLocked(self):
return safe_hasattr(self,'wl_isLocked') and self.wl_isLocked()
security.declareProtected(Permissions.Edit, 'PUT')
def PUT(self, REQUEST, RESPONSE):
"""Handle HTTP/FTP/WebDav PUT requests."""
self.dav__init(REQUEST, RESPONSE)
self.dav__simpleifhandler(REQUEST, RESPONSE, refresh=1)
self._validateProxy(REQUEST)
body = REQUEST.get('BODY', '')
headers, body = parseHeadersBody(body)
log = string.strip(headers.get('Log', headers.get('log', ''))) or ''
type = string.strip(headers.get('Type', headers.get('type', ''))) or None
if type is not None: type = string.split(type)[0]
timestamp = string.strip(headers.get('Wiki-Safetybelt', '')) or None
if timestamp and self.checkEditConflict(timestamp, REQUEST):
RESPONSE.setStatus(423) # Resource Locked
return RESPONSE
try:
self.edit(text=body, type=type, timeStamp=timestamp,
REQUEST=REQUEST, log=log, check_conflict=0)
except 'Unauthorized':
RESPONSE.setStatus(401)
return RESPONSE
RESPONSE.setStatus(204)
return RESPONSE
security.declarePublic('isExternalEditEnabled')
def isExternalEditEnabled(self):
return (safe_hasattr(self.getPhysicalRoot().misc_,'ExternalEditor') and
self.checkPermission(Permissions.Edit, self) and
self.checkPermission(Permissions.ExternalEdit, self))
security.declareProtected(Permissions.Edit, 'manage_edit')
def manage_edit(self, data, title, REQUEST=None):
"""Do standard manage_edit kind of stuff, using our edit."""
data = self.tounicode(data)
title = self.tounicode(title)
#self.edit(text=data, title=title, REQUEST=REQUEST, check_conflict=0)
#I think we need to bypass edit to provide correct permissions
self.title = title
self.setText(data,REQUEST)
self.setLastEditor(REQUEST)
self.reindex_object()
if REQUEST:
message="Content changed."
return self.manage_main(self,REQUEST,manage_tabs_message=message)
def allowedPageTypes(self):
"""
List the page type ids which may be selected in this wiki's edit form.
This will be all available page types, unless overridden by an
allowed_page_types property.
"""
return (filter(lambda x:strip(x),getattr(self,'allowed_page_types',[]))
or map(lambda x:x._id, PAGETYPES))
def defaultPageType(self):
"""This wiki's default page type."""
return self.allowedPageTypes()[0]
security.declareProtected(Permissions.Add, 'split')
def split(self):
"""
Move this page's major sections to sub-pages, if supported.
Delegates to the page type; at present only the restructured