routes.py 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. from bottle import request, response, abort, static_file
  2. from kddit import app
  3. from urllib.parse import urlparse
  4. from kddit import settings
  5. from kddit.utils import req, success, ydl, req_url
  6. from kddit import html
  7. from kddit.utils import verify_subreddit_option, verify_user_option
  8. from kddit.utils import get_subreddit_url, get_subreddit
  9. from kddit.utils import nsfw_mode, get_query
  10. def subreddit_content(data, subreddit, option, time, safe):
  11. content = (html.subreddit_menu(option, subreddit))
  12. if option in settings.EXPANDED_OPTIONS:
  13. content += (html.subreddit_sort_menu(subreddit, option or settings.DEFAULT_OPTION, time))
  14. content += (html.mixed_content(data, safe) or html.nothing,)
  15. content += html.subreddit_nav(data, subreddit, option, time)
  16. return content
  17. def search_content(data, subreddit, sort, time, query):
  18. content = html.search_sort_menu(subreddit, query)
  19. content += html.search_time_menu(subreddit, query )
  20. content += html.mixed_content(data, True) or html.nothing
  21. content += html.search_nav(data, subreddit, query)
  22. return content
  23. def domain_content(data, domain, option, time):
  24. content = html.domain_menu(option, domain)
  25. if option in settings.EXPANDED_OPTIONS:
  26. content += html.domain_sort_menu(domain, option or settings.DEFAULT_OPTION, time)
  27. content += html.mixed_content(data, True) or html.nothing
  28. content += html.domain_nav(data, domain, option, time)
  29. return content
  30. def user_content(data, user, option, sort):
  31. content = html.user_menu(option, user)
  32. content += html.user_sort_menu(option, sort, user)
  33. content += (html.mixed_content(data, True),)
  34. content += html.user_nav(data, user, option, sort)
  35. return content
  36. def multi_content(data, user, multi, option, sort):
  37. content = html.multi_menu(option, user, multi)
  38. content += html.multi_sort_menu(user, multi, option, sort)
  39. content += (html.mixed_content(data, True),)
  40. content += html.multi_nav(data, user, multi, option, sort)
  41. return content
  42. @app.route("/search", "GET")
  43. @app.route("/r/<subreddit>/search", "GET")
  44. def search_page(subreddit=None):
  45. url = f'{get_subreddit_url()}/search/.json'
  46. query = dict(request.query)
  47. r = req(url, query)
  48. if success(r):
  49. data = r.json()
  50. time = query.get("t")
  51. sort = query.get("sort")
  52. q = query.get("q") or ""
  53. title = f"search results - {q}"
  54. header = html.page_header(subreddit=subreddit, q=q)
  55. content = search_content(data, subreddit, sort, time, query)
  56. return html.page(title, header, content).render()
  57. else:
  58. return abort(r.status_code)
  59. @app.route("/u/<user>/m/<multi>", "GET")
  60. @app.route("/user/<user>/m/<multi>", "GET")
  61. @app.route("/u/<user>/m/<multi>/<option>", "GET")
  62. @app.route("/user/<user>/m/<multi>/<option>", "GET")
  63. def multi_page(user, multi=None ,option=None):
  64. verify_subreddit_option()
  65. url = f"/user/{user}/m/{multi}/{option or settings.DEFAULT_OPTION}/.json"
  66. query = dict(request.query)
  67. r = req(url, query)
  68. if success(r):
  69. data = r.json()
  70. sort = query.get("t")
  71. title = f"m/{multi} by u/{user}"
  72. header = html.page_header(user=user, multi=multi)
  73. content = multi_content(data, user, multi, option, sort)
  74. return html.page(title, header, content).render()
  75. else:
  76. return abort(r.status_code)
  77. @app.route("/u/<user>", "GET")
  78. @app.route("/user/<user>", "GET")
  79. @app.route("/u/<user>/<option>", "GET")
  80. @app.route("/user/<user>/<option>", "GET")
  81. def user_page(user, option="overview"):
  82. verify_user_option()
  83. url = f"/user/{user}/{option}/.json"
  84. query = dict(request.query)
  85. r = req(url, query)
  86. if success(r):
  87. data = r.json()
  88. sort = query.get("sort")
  89. title = f"{option} by u/{user}"
  90. header = html.page_header(user=user)
  91. content = user_content(data, user, option, sort)
  92. return html.page(title, header, content).render()
  93. else:
  94. return abort(r.status_code)
  95. @app.route("/", "GET")
  96. @app.route("/<option>", "GET")
  97. @app.route("/r/<subreddit>", "GET")
  98. @app.route("/r/<subreddit>/<option>", "GET")
  99. def subreddit_page(subreddit=None, option=None):
  100. verify_subreddit_option()
  101. url = f'{get_subreddit_url()}/{option or settings.DEFAULT_OPTION}.json'
  102. query = dict(request.query)
  103. r = req(url, query)
  104. if success(r):
  105. data = r.json()
  106. time = query.get("t")
  107. title = get_subreddit() or "kddit"
  108. header = html.page_header(subreddit=subreddit)
  109. safe = nsfw_mode(subreddit)
  110. content = subreddit_content(data, subreddit, option, time, safe)
  111. return html.page(title, header, content).render()
  112. return abort(r.status_code)
  113. @app.route("/domain/<domain>", "GET")
  114. @app.route("/domain/<domain>/<option>", "GET")
  115. def domain_page(domain, option=None):
  116. verify_subreddit_option()
  117. query = get_query()
  118. time = query.get("t")
  119. url = f'/domain/{domain}/{option or settings.DEFAULT_OPTION}.json'
  120. r = req(url, query)
  121. if success(r):
  122. data = r.json()
  123. title = domain
  124. header = html.page_header(domain=domain)
  125. content = domain_content(data, domain, option, time)
  126. return html.page(title, header, content).render()
  127. else:
  128. return abort(r.status_code)
  129. @app.route("/r/<subreddit>/comments/<post_id>/<path>", "GET")
  130. @app.route("/r/<subreddit>/comments/<post_id>/<path>/<comment_id>", "GET")
  131. def post_page(subreddit, post_id, path, comment_id=""):
  132. u = f"/r/{subreddit}/comments/{post_id}/{path}/{comment_id}.json"
  133. query = get_query()
  134. r = req(u, query)
  135. if success(r):
  136. data = r.json()
  137. post = data[0]["data"]["children"][0]["data"]
  138. comments = data[1]["data"]["children"]
  139. title = post["title"]
  140. content = (html.post(post), html.comments(comments))
  141. header = html.page_header(subreddit=subreddit)
  142. return html.page(title, header, content).render()
  143. else:
  144. return abort(r.status_code)
  145. @app.route("/static/<file>")
  146. def static(file):
  147. return static_file(file, root=f"{settings.ROOT}/static")
  148. @app.route("/video/<url:path>")
  149. def video_proxy(url):
  150. uri = urlparse(url)
  151. if uri.netloc in settings.PROXY_ALLOW["video"]:
  152. with ydl:
  153. result = ydl.extract_info(url, download=True)
  154. return static_file(
  155. f'{result["id"]}.mp4',
  156. root=settings.FILE_PATH)
  157. else:
  158. return abort(403)
  159. @app.route("/proxy/<url:path>")
  160. def proxy(url):
  161. uri = urlparse(url)
  162. netloc = uri.netloc
  163. query = get_query()
  164. if netloc not in settings.PROXY_ALLOW["image"]:
  165. return abort(403)
  166. r = req_url(url, query)
  167. if success(r):
  168. response.set_header("content-type", r.headers["content-type"])
  169. return r.content
  170. else:
  171. return abort(r.status_code)
  172. @app.error(401)
  173. @app.error(403)
  174. @app.error(404)
  175. @app.error(405)
  176. @app.error(406)
  177. @app.error(429)
  178. @app.error(451)
  179. @app.error(500)
  180. @app.error(503)
  181. def error_redirect(error):
  182. return html.error_page(error).render()