]>
git.ipfire.org Git - ipfire.org.git/blob - src/web/wiki.py
8 from . import ui_modules
10 class ActionEditHandler(auth
.CacheMixin
, base
.BaseHandler
):
11 @tornado.web
.authenticated
17 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
18 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
21 page
= self
.backend
.wiki
.get_page(path
)
23 # Empty page if it was deleted
24 if page
and page
.was_deleted():
28 self
.render("wiki/edit.html", page
=page
, path
=path
)
30 @tornado.web
.authenticated
36 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
37 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
39 content
= self
.get_argument("content", None)
40 changes
= self
.get_argument("changes")
42 # Create a new page in the database
43 with self
.db
.transaction():
44 page
= self
.backend
.wiki
.create_page(path
,
45 self
.current_user
, content
, changes
=changes
, address
=self
.get_remote_ip())
47 # Add user as a watcher if wanted
48 watch
= self
.get_argument("watch", False)
50 page
.add_watcher(self
.current_user
)
53 if page
.was_deleted():
56 self
.redirect(page
.url
)
60 Updates the search index after the page has been edited
62 # This is being executed in the background and after
63 # the response has been set to the client
64 with self
.db
.transaction():
65 self
.backend
.wiki
.refresh()
68 class ActionUploadHandler(auth
.CacheMixin
, base
.BaseHandler
):
69 @tornado.web
.authenticated
70 @base.ratelimit(minutes
=60, requests
=24)
72 path
= self
.get_argument("path")
75 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
76 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
79 filename
, data
, mimetype
= self
.get_file("file")
81 # Use filename from request if any
82 filename
= self
.get_argument("filename", filename
)
84 # XXX check valid mimetypes
86 with self
.db
.transaction():
87 file = self
.backend
.wiki
.upload(path
, filename
, data
,
88 mimetype
=mimetype
, author
=self
.current_user
,
89 address
=self
.get_remote_ip())
91 except TypeError as e
:
94 self
.redirect("%s/_files" % path
)
97 class ActionDeleteHandler(auth
.CacheMixin
, base
.BaseHandler
):
98 @tornado.web
.authenticated
101 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
102 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
105 file = self
.backend
.wiki
.get_file_by_path(path
)
107 raise tornado
.web
.HTTPError(404, "Could not find %s" % path
)
109 self
.render("wiki/confirm-delete.html", file=file)
111 @tornado.web
.authenticated
112 @base.ratelimit(minutes
=60, requests
=24)
113 def post(self
, path
):
115 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
116 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
119 file = self
.backend
.wiki
.get_file_by_path(path
)
121 raise tornado
.web
.HTTPError(404, "Could not find %s" % path
)
123 with self
.db
.transaction():
124 file.delete(self
.current_user
)
126 self
.redirect("%s/_files" % file.path
)
129 class ActionRestoreHandler(auth
.CacheMixin
, base
.BaseHandler
):
130 @tornado.web
.authenticated
131 @base.ratelimit(minutes
=60, requests
=24)
133 path
= self
.get_argument("path")
136 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
137 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
139 # Check if we are asked to render a certain revision
140 revision
= self
.get_argument("revision", None)
142 # Fetch the wiki page
143 page
= self
.backend
.wiki
.get_page(path
, revision
=revision
)
145 with self
.db
.transaction():
147 author
=self
.current_user
, address
=self
.get_remote_ip(),
150 # Redirect back to page
151 self
.redirect(page
.page
)
154 class ActionWatchHandler(auth
.CacheMixin
, base
.BaseHandler
):
155 @tornado.web
.authenticated
156 @base.ratelimit(minutes
=60, requests
=180)
157 def get(self
, path
, action
):
161 page
= self
.backend
.wiki
.get_page(path
)
163 raise tornado
.web
.HTTPError(404, "Page does not exist: %s" % path
)
166 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
167 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
169 with self
.db
.transaction():
170 if action
== "watch":
171 page
.add_watcher(self
.current_user
)
172 elif action
== "unwatch":
173 page
.remove_watcher(self
.current_user
)
175 # Redirect back to page
176 self
.redirect(page
.url
)
179 class ActionRenderHandler(auth
.CacheMixin
, base
.BaseHandler
):
180 def check_xsrf_cookie(self
):
183 @tornado.web
.authenticated
184 @base.ratelimit(minutes
=5, requests
=180)
185 def post(self
, path
):
189 content
= self
.get_argument("content")
192 html
= self
.backend
.wiki
.render(path
, content
)
197 class FilesHandler(auth
.CacheMixin
, base
.BaseHandler
):
198 @tornado.web
.authenticated
204 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
205 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
207 files
= self
.backend
.wiki
.get_files(path
)
209 self
.render("wiki/files/index.html", path
=path
, files
=files
)
212 class FileHandler(base
.BaseHandler
):
215 return self
.get_argument("action", None)
219 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
220 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
222 # Check if we are asked to render a certain revision
223 revision
= self
.get_argument("revision", None)
226 file = self
.backend
.wiki
.get_file_by_path(path
, revision
=revision
)
228 raise tornado
.web
.HTTPError(404, "Could not find %s" % path
)
231 if self
.action
== "detail":
234 for breadcrumb
, title
in self
.backend
.wiki
.make_breadcrumbs(path
):
235 page
= self
.backend
.wiki
.get_page(breadcrumb
)
239 self
.render("wiki/files/detail.html", page
=page
, file=file)
242 size
= self
.get_argument_int("s", None)
244 # Check if image should be resized
245 if file.is_image() and size
:
246 blob
= file.get_thumbnail(size
)
251 self
.set_header("Content-Type", file.mimetype
or "application/octet-stream")
252 self
.set_header("Content-Length", len(blob
))
255 self
.set_expires(3600)
261 class PageHandler(auth
.CacheMixin
, base
.BaseHandler
):
264 return self
.get_argument("action", None)
266 def write_error(self
, status_code
, **kwargs
):
267 # Render a custom page for 404
268 if status_code
== 404:
269 self
.render("wiki/404.html", **kwargs
)
272 # Otherwise raise this to one layer above
273 super().write_error(status_code
, **kwargs
)
275 @tornado.web
.removeslash
281 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
282 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
284 # Check if we are asked to render a certain revision
285 revision
= self
.get_argument("revision", None)
287 # Fetch the wiki page
288 page
= self
.backend
.wiki
.get_page(path
, revision
=revision
)
291 if self
.action
== "diff":
293 a
= self
.get_argument("a")
294 b
= self
.get_argument("b")
296 # Fetch both versions of the page
297 a
= self
.backend
.wiki
.get_page(path
, revision
=a
)
298 b
= self
.backend
.wiki
.get_page(path
, revision
=b
)
300 raise tornado
.web
.HTTPError(404)
302 # Cannot render a diff for the identical page
304 raise tornado
.web
.HTTPError(400)
306 # Make sure that b is newer than a
310 self
.render("wiki/diff.html", page
=page
, a
=a
, b
=b
)
314 elif self
.action
== "restore":
315 self
.render("wiki/confirm-restore.html", page
=page
)
319 elif self
.action
== "revisions":
320 self
.render("wiki/revisions.html", page
=page
)
323 # If the page does not exist, we send 404
324 if not page
or page
.was_deleted():
325 # Handle /start links which were in the format of DokuWiki
326 if path
.endswith("/start"):
327 # Strip /start from path
328 path
= path
[:-6] or "/"
330 # Redirect user to page if it exists
331 page
= self
.backend
.wiki
.page_exists(path
)
335 raise tornado
.web
.HTTPError(404)
337 # Fetch the latest revision
338 latest_revision
= page
.get_latest_revision()
341 self
.render("wiki/page.html", page
=page
, latest_revision
=latest_revision
)
344 class SearchHandler(auth
.CacheMixin
, base
.BaseHandler
):
345 @base.ratelimit(minutes
=5, requests
=25)
347 q
= self
.get_argument("q")
349 pages
= self
.backend
.wiki
.search(q
, account
=self
.current_user
, limit
=50)
351 self
.render("wiki/search-results.html", q
=q
, pages
=pages
)
354 class RecentChangesHandler(auth
.CacheMixin
, base
.BaseHandler
):
356 recent_changes
= self
.backend
.wiki
.get_recent_changes(self
.current_user
, limit
=50)
358 self
.render("wiki/recent-changes.html", recent_changes
=recent_changes
)
361 class WatchlistHandler(auth
.CacheMixin
, base
.BaseHandler
):
362 @tornado.web
.authenticated
364 pages
= self
.backend
.wiki
.get_watchlist(self
.current_user
)
366 self
.render("wiki/watchlist.html", pages
=pages
)
369 class WikiDiffModule(ui_modules
.UIModule
):
370 differ
= difflib
.Differ()
372 def render(self
, a
, b
):
373 diff
= self
.differ
.compare(
374 a
.markdown
.splitlines(),
375 b
.markdown
.splitlines(),
378 return self
.render_string("wiki/modules/diff.html", diff
=diff
)
381 class WikiListModule(ui_modules
.UIModule
):
382 def render(self
, pages
, link_revision
=False, show_breadcrumbs
=True,
383 show_author
=True, show_changes
=False):
384 return self
.render_string("wiki/modules/list.html", link_revision
=link_revision
,
385 pages
=pages
, show_breadcrumbs
=show_breadcrumbs
,
386 show_author
=show_author
, show_changes
=show_changes
)
389 class WikiNavbarModule(ui_modules
.UIModule
):
393 Returns the path of the page (without any actions)
395 path
= self
.request
.path
.split("/")
397 if path
and path
[-1].startswith("_"):
400 return "/".join(path
)
402 def render(self
, suffix
=None):
403 _
= self
.locale
.translate
406 page
= self
.request
.path
.split("/")
408 # Drop the action bit
409 if page
and page
[-1].startswith("_"):
412 page
= "/".join(page
)
414 breadcrumbs
= self
.backend
.wiki
.make_breadcrumbs(page
)
415 title
= self
.backend
.wiki
.get_page_title(page
)
417 if self
.request
.path
.endswith("/_edit"):
419 elif self
.request
.path
.endswith("/_files"):
422 return self
.render_string("wiki/modules/navbar.html",
423 breadcrumbs
=breadcrumbs
, page
=page
, page_title
=title
, suffix
=suffix
)