]>
git.ipfire.org Git - ipfire.org.git/blob - src/web/wiki.py
e5759e375e27ef6680a869aec85f2c7459b5b1b5
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
)
130 class ActionWatchHandler(auth
.CacheMixin
, base
.BaseHandler
):
131 @tornado.web
.authenticated
132 @base.ratelimit(minutes
=60, requests
=180)
133 def get(self
, path
, action
):
137 page
= self
.backend
.wiki
.get_page(path
)
139 raise tornado
.web
.HTTPError(404, "Page does not exist: %s" % path
)
142 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
143 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
145 with self
.db
.transaction():
146 if action
== "watch":
147 page
.add_watcher(self
.current_user
)
148 elif action
== "unwatch":
149 page
.remove_watcher(self
.current_user
)
151 # Redirect back to page
152 self
.redirect(page
.url
)
155 class ActionRenderHandler(auth
.CacheMixin
, base
.BaseHandler
):
156 def check_xsrf_cookie(self
):
159 @tornado.web
.authenticated
160 @base.ratelimit(minutes
=5, requests
=180)
161 def post(self
, path
):
165 content
= self
.get_argument("content")
168 html
= self
.backend
.wiki
.render(path
, content
)
173 class FilesHandler(auth
.CacheMixin
, base
.BaseHandler
):
174 @tornado.web
.authenticated
180 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
181 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
183 files
= self
.backend
.wiki
.get_files(path
)
185 self
.render("wiki/files/index.html", path
=path
, files
=files
)
188 class FileHandler(base
.BaseHandler
):
191 return self
.get_argument("action", None)
195 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
196 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
198 # Check if we are asked to render a certain revision
199 revision
= self
.get_argument("revision", None)
202 file = self
.backend
.wiki
.get_file_by_path(path
, revision
=revision
)
204 raise tornado
.web
.HTTPError(404, "Could not find %s" % path
)
207 if self
.action
== "detail":
210 for breadcrumb
, title
in self
.backend
.wiki
.make_breadcrumbs(path
):
211 page
= self
.backend
.wiki
.get_page(breadcrumb
)
215 self
.render("wiki/files/detail.html", page
=page
, file=file)
218 size
= self
.get_argument_int("s", None)
220 # Check if image should be resized
221 if file.is_image() and size
:
222 blob
= file.get_thumbnail(size
)
227 self
.set_header("Content-Type", file.mimetype
or "application/octet-stream")
228 self
.set_header("Content-Length", len(blob
))
231 self
.set_expires(3600)
237 class PageHandler(auth
.CacheMixin
, base
.BaseHandler
):
240 return self
.get_argument("action", None)
242 def write_error(self
, status_code
, **kwargs
):
243 # Render a custom page for 404
244 if status_code
== 404:
245 self
.render("wiki/404.html", **kwargs
)
248 # Otherwise raise this to one layer above
249 super().write_error(status_code
, **kwargs
)
251 @tornado.web
.removeslash
257 if not self
.backend
.wiki
.check_acl(path
, self
.current_user
):
258 raise tornado
.web
.HTTPError(403, "Access to %s not allowed for %s" % (path
, self
.current_user
))
260 # Check if we are asked to render a certain revision
261 revision
= self
.get_argument("revision", None)
263 # Fetch the wiki page
264 page
= self
.backend
.wiki
.get_page(path
, revision
=revision
)
267 if self
.action
== "diff":
269 a
= self
.get_argument("a")
270 b
= self
.get_argument("b")
272 # Fetch both versions of the page
273 a
= self
.backend
.wiki
.get_page(path
, revision
=a
)
274 b
= self
.backend
.wiki
.get_page(path
, revision
=b
)
276 raise tornado
.web
.HTTPError(404)
278 # Cannot render a diff for the identical page
280 raise tornado
.web
.HTTPError(400)
282 # Make sure that b is newer than a
286 self
.render("wiki/diff.html", page
=page
, a
=a
, b
=b
)
290 elif self
.action
== "revisions":
291 self
.render("wiki/revisions.html", page
=page
)
294 # If the page does not exist, we send 404
295 if not page
or page
.was_deleted():
296 # Handle /start links which were in the format of DokuWiki
297 if path
.endswith("/start"):
298 # Strip /start from path
299 path
= path
[:-6] or "/"
301 # Redirect user to page if it exists
302 page
= self
.backend
.wiki
.page_exists(path
)
306 raise tornado
.web
.HTTPError(404)
308 # Fetch the latest revision
309 latest_revision
= page
.get_latest_revision()
312 self
.render("wiki/page.html", page
=page
, latest_revision
=latest_revision
)
315 class SearchHandler(auth
.CacheMixin
, base
.BaseHandler
):
316 @base.ratelimit(minutes
=5, requests
=25)
318 q
= self
.get_argument("q")
320 pages
= self
.backend
.wiki
.search(q
, account
=self
.current_user
, limit
=50)
322 self
.render("wiki/search-results.html", q
=q
, pages
=pages
)
325 class RecentChangesHandler(auth
.CacheMixin
, base
.BaseHandler
):
327 recent_changes
= self
.backend
.wiki
.get_recent_changes(self
.current_user
, limit
=50)
329 self
.render("wiki/recent-changes.html", recent_changes
=recent_changes
)
332 class WatchlistHandler(auth
.CacheMixin
, base
.BaseHandler
):
333 @tornado.web
.authenticated
335 pages
= self
.backend
.wiki
.get_watchlist(self
.current_user
)
337 self
.render("wiki/watchlist.html", pages
=pages
)
340 class WikiDiffModule(ui_modules
.UIModule
):
341 differ
= difflib
.Differ()
343 def render(self
, a
, b
):
344 diff
= self
.differ
.compare(
345 a
.markdown
.splitlines(),
346 b
.markdown
.splitlines(),
349 return self
.render_string("wiki/modules/diff.html", diff
=diff
)
352 class WikiListModule(ui_modules
.UIModule
):
353 def render(self
, pages
, link_revision
=False, show_breadcrumbs
=True,
354 show_author
=True, show_changes
=False):
355 return self
.render_string("wiki/modules/list.html", link_revision
=link_revision
,
356 pages
=pages
, show_breadcrumbs
=show_breadcrumbs
,
357 show_author
=show_author
, show_changes
=show_changes
)
360 class WikiNavbarModule(ui_modules
.UIModule
):
364 Returns the path of the page (without any actions)
366 path
= self
.request
.path
.split("/")
368 if path
and path
[-1].startswith("_"):
371 return "/".join(path
)
373 def render(self
, suffix
=None):
374 _
= self
.locale
.translate
377 page
= self
.request
.path
.split("/")
379 # Drop the action bit
380 if page
and page
[-1].startswith("_"):
383 page
= "/".join(page
)
385 breadcrumbs
= self
.backend
.wiki
.make_breadcrumbs(page
)
386 title
= self
.backend
.wiki
.get_page_title(page
)
388 if self
.request
.path
.endswith("/_edit"):
390 elif self
.request
.path
.endswith("/_files"):
393 return self
.render_string("wiki/modules/navbar.html",
394 breadcrumbs
=breadcrumbs
, page
=page
, page_title
=title
, suffix
=suffix
)