test_utils.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. from django.core.exceptions import PermissionDenied
  2. from django.test import TestCase
  3. from django.test.client import RequestFactory
  4. from django.urls import reverse
  5. from misago.core.utils import (
  6. clean_ids_list,
  7. clean_return_path,
  8. format_plaintext_for_html,
  9. get_exception_message,
  10. get_host_from_address,
  11. is_referer_local,
  12. is_request_to_misago,
  13. parse_iso8601_string,
  14. slugify,
  15. )
  16. class IsRequestToMisagoTests(TestCase):
  17. def test_is_request_to_misago(self):
  18. """
  19. is_request_to_misago correctly detects requests directed at Misago
  20. """
  21. VALID_PATHS = ("/", "/threads/")
  22. INVALID_PATHS = ("", "somewhere/")
  23. misago_prefix = reverse("misago:index")
  24. for path in VALID_PATHS:
  25. request = RequestFactory().get("/")
  26. request.path = path
  27. self.assertTrue(
  28. is_request_to_misago(request),
  29. '"%s" is not overlapped by "%s"' % (path, misago_prefix),
  30. )
  31. for path in INVALID_PATHS:
  32. request = RequestFactory().get("/")
  33. request.path = path
  34. self.assertFalse(
  35. is_request_to_misago(request),
  36. '"%s" is overlapped by "%s"' % (path, misago_prefix),
  37. )
  38. class SlugifyTests(TestCase):
  39. def test_valid_slugify_output(self):
  40. """Misago's slugify correctly slugifies string"""
  41. test_cases = [
  42. ("User", "user"),
  43. ("Eric The Fish", "eric-the-fish"),
  44. ("John Snow", "john-snow"),
  45. ("J0n", "j0n"),
  46. ("An###ne", "anne"),
  47. ("S**t", "st"),
  48. ("Łók", "lok"),
  49. ]
  50. for original, slug in test_cases:
  51. self.assertEqual(slugify(original), slug)
  52. class ParseIso8601StringTests(TestCase):
  53. def test_valid_input(self):
  54. """util parses iso 8601 strings"""
  55. INPUTS = [
  56. "2016-10-22T20:55:39.185085Z",
  57. "2016-10-22T20:55:39.185085-01:00",
  58. "2016-10-22T20:55:39-01:00",
  59. "2016-10-22T20:55:39.185085+01:00",
  60. ]
  61. for test_input in INPUTS:
  62. self.assertTrue(parse_iso8601_string(test_input))
  63. def test_invalid_input(self):
  64. """util throws ValueError on invalid input"""
  65. INPUTS = [
  66. "",
  67. "2016-10-22",
  68. "2016-10-22T30:55:39.185085+11:00",
  69. "2016-10-22T20:55:39.18SSSSS5085Z",
  70. ]
  71. for test_input in INPUTS:
  72. with self.assertRaises(ValueError):
  73. self.assertTrue(parse_iso8601_string(test_input))
  74. PLAINTEXT_FORMAT_CASES = [
  75. ("Lorem ipsum.", "<p>Lorem ipsum.</p>"),
  76. ("Lorem <b>ipsum</b>.", "<p>Lorem &lt;b&gt;ipsum&lt;/b&gt;.</p>"),
  77. ('Lorem "ipsum" dolor met.', "<p>Lorem &quot;ipsum&quot; dolor met.</p>"),
  78. ("Lorem ipsum.\nDolor met.", "<p>Lorem ipsum.<br />Dolor met.</p>"),
  79. ("Lorem ipsum.\n\nDolor met.", "<p>Lorem ipsum.</p>\n\n<p>Dolor met.</p>"),
  80. (
  81. "http://misago-project.org/login/",
  82. '<p><a href="http://misago-project.org/login/">http://misago-project.org/login/</a></p>',
  83. ),
  84. ]
  85. class FormatPlaintextForHtmlTests(TestCase):
  86. def test_format_plaintext_for_html(self):
  87. """format_plaintext_for_html correctly formats plaintext for html"""
  88. for plaintext, html in PLAINTEXT_FORMAT_CASES:
  89. output = format_plaintext_for_html(plaintext)
  90. assertion_message = """
  91. format_plaintext_for_html failed to produce expected output:
  92. expected: %s
  93. return: %s
  94. """ % (
  95. html,
  96. output,
  97. )
  98. self.assertEqual(output, html, assertion_message)
  99. class MockRequest(object):
  100. scheme = "http"
  101. def __init__(self, method, meta=None, post=None):
  102. self.method = method
  103. self.META = meta or {}
  104. self.POST = post or {}
  105. class CleanReturnPathTests(TestCase):
  106. def test_get_request(self):
  107. """clean_return_path works for GET requests"""
  108. bad_request = MockRequest(
  109. "GET",
  110. {"HTTP_REFERER": "http://cookies.com", "HTTP_HOST": "misago-project.org"},
  111. )
  112. self.assertIsNone(clean_return_path(bad_request))
  113. bad_request = MockRequest(
  114. "GET",
  115. {
  116. "HTTP_REFERER": "https://misago-project.org/",
  117. "HTTP_HOST": "misago-project.org/",
  118. },
  119. )
  120. self.assertIsNone(clean_return_path(bad_request))
  121. bad_request = MockRequest(
  122. "GET",
  123. {
  124. "HTTP_REFERER": "https://misago-project.org/",
  125. "HTTP_HOST": "misago-project.org/assadsa/",
  126. },
  127. )
  128. self.assertIsNone(clean_return_path(bad_request))
  129. ok_request = MockRequest(
  130. "GET",
  131. {
  132. "HTTP_REFERER": "http://misago-project.org/",
  133. "HTTP_HOST": "misago-project.org/",
  134. },
  135. )
  136. self.assertEqual(clean_return_path(ok_request), "/")
  137. ok_request = MockRequest(
  138. "GET",
  139. {
  140. "HTTP_REFERER": "http://misago-project.org/login/",
  141. "HTTP_HOST": "misago-project.org/",
  142. },
  143. )
  144. self.assertEqual(clean_return_path(ok_request), "/login/")
  145. def test_post_request(self):
  146. """clean_return_path works for POST requests"""
  147. bad_request = MockRequest(
  148. "POST",
  149. {
  150. "HTTP_REFERER": "http://misago-project.org/",
  151. "HTTP_HOST": "misago-project.org/",
  152. },
  153. {"return_path": "/sdasdsa/"},
  154. )
  155. self.assertIsNone(clean_return_path(bad_request))
  156. ok_request = MockRequest(
  157. "POST",
  158. {
  159. "HTTP_REFERER": "http://misago-project.org/",
  160. "HTTP_HOST": "misago-project.org/",
  161. },
  162. {"return_path": "/login/"},
  163. )
  164. self.assertEqual(clean_return_path(ok_request), "/login/")
  165. class IsRefererLocalTests(TestCase):
  166. def test_local_referers(self):
  167. """local referers return true"""
  168. ok_request = MockRequest(
  169. "GET",
  170. {
  171. "HTTP_REFERER": "http://misago-project.org/",
  172. "HTTP_HOST": "misago-project.org/",
  173. },
  174. )
  175. self.assertTrue(is_referer_local(ok_request))
  176. ok_request = MockRequest(
  177. "GET",
  178. {
  179. "HTTP_REFERER": "http://misago-project.org/",
  180. "HTTP_HOST": "misago-project.org/",
  181. },
  182. )
  183. self.assertTrue(is_referer_local(ok_request))
  184. ok_request = MockRequest(
  185. "GET",
  186. {
  187. "HTTP_REFERER": "http://misago-project.org/login/",
  188. "HTTP_HOST": "misago-project.org/",
  189. },
  190. )
  191. self.assertTrue(is_referer_local(ok_request))
  192. def test_foreign_referers(self):
  193. """non-local referers return false"""
  194. bad_request = MockRequest(
  195. "GET",
  196. {
  197. "HTTP_REFERER": "http://else-project.org/",
  198. "HTTP_HOST": "misago-project.org/",
  199. },
  200. )
  201. self.assertFalse(is_referer_local(bad_request))
  202. bad_request = MockRequest(
  203. "GET",
  204. {
  205. "HTTP_REFERER": "https://misago-project.org/",
  206. "HTTP_HOST": "misago-project.org/",
  207. },
  208. )
  209. self.assertFalse(is_referer_local(bad_request))
  210. bad_request = MockRequest(
  211. "GET",
  212. {
  213. "HTTP_REFERER": "http://misago-project.org/",
  214. "HTTP_HOST": "misago-project.org/assadsa/",
  215. },
  216. )
  217. self.assertFalse(is_referer_local(bad_request))
  218. class GetExceptionMessageTests(TestCase):
  219. def test_no_args(self):
  220. """both of helper args are optional"""
  221. message = get_exception_message()
  222. self.assertIsNone(message)
  223. def test_no_default_message(self):
  224. """helper's default message arg is optional"""
  225. message = get_exception_message(PermissionDenied("Lorem Ipsum"))
  226. self.assertEqual(message, "Lorem Ipsum")
  227. message = get_exception_message(PermissionDenied())
  228. self.assertIsNone(message)
  229. def test_default_message(self):
  230. """helper's default message arg is used"""
  231. message = get_exception_message(PermissionDenied("Lorem Ipsum"), "Default")
  232. self.assertEqual(message, "Lorem Ipsum")
  233. message = get_exception_message(PermissionDenied(), "Default")
  234. self.assertEqual(message, "Default")
  235. message = get_exception_message(default_message="Lorem Ipsum")
  236. self.assertEqual(message, "Lorem Ipsum")
  237. class CleanIdsListTests(TestCase):
  238. def test_valid_list(self):
  239. """list of valid ids is cleaned"""
  240. self.assertEqual(clean_ids_list(["1", 3, "42"], None), [1, 3, 42])
  241. def test_empty_list(self):
  242. """empty list passes validation"""
  243. self.assertEqual(clean_ids_list([], None), [])
  244. def test_string_list(self):
  245. """string list passes validation"""
  246. self.assertEqual(clean_ids_list("1234", None), [1, 2, 3, 4])
  247. def test_message(self):
  248. """utility uses passed message for exception"""
  249. with self.assertRaisesMessage(PermissionDenied, "Test error message!"):
  250. clean_ids_list(None, "Test error message!")
  251. def test_invalid_inputs(self):
  252. """utility raises exception for invalid inputs"""
  253. INVALID_INPUTS = (
  254. None,
  255. "abc",
  256. [None],
  257. [1, 2, "a", 4],
  258. [1, None, 3],
  259. {1: 2, "a": 4},
  260. )
  261. for invalid_input in INVALID_INPUTS:
  262. with self.assertRaisesMessage(PermissionDenied, "Test error message!"):
  263. clean_ids_list(invalid_input, "Test error message!")
  264. class GetHostFromAddressTests(TestCase):
  265. def test_none(self):
  266. """get_host_from_address returns None for None"""
  267. result = get_host_from_address(None)
  268. self.assertIsNone(result)
  269. def test_empty_string(self):
  270. """get_host_from_address returns None for empty string"""
  271. result = get_host_from_address("")
  272. self.assertIsNone(result)
  273. def test_hostname(self):
  274. """get_host_from_address returns hostname unchanged"""
  275. result = get_host_from_address("hostname")
  276. self.assertEqual(result, "hostname")
  277. def test_hostname_with_trailing_slash(self):
  278. """get_host_from_address returns hostname for hostname with trailing slash"""
  279. result = get_host_from_address("//hostname")
  280. self.assertEqual(result, "hostname")
  281. def test_hostname_with_port(self):
  282. """get_host_from_address returns hostname for hostname with port"""
  283. result = get_host_from_address("hostname:8888")
  284. self.assertEqual(result, "hostname")
  285. def test_hostname_with_port_path_and_protocol(self):
  286. """get_host_from_address returns hostname for hostname with port and path"""
  287. result = get_host_from_address("https://hostname:8888/somewhere/else/")
  288. self.assertEqual(result, "hostname")