test_css_files_creation_and_edition.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389
  1. import pytest
  2. from django.urls import reverse
  3. from ....test import assert_contains, assert_has_error_message
  4. @pytest.fixture
  5. def create_link(theme):
  6. return reverse(
  7. "misago:admin:appearance:themes:new-css-file", kwargs={"pk": theme.pk}
  8. )
  9. @pytest.fixture
  10. def edit_link(theme, css):
  11. return reverse(
  12. "misago:admin:appearance:themes:edit-css-file",
  13. kwargs={"pk": theme.pk, "css_pk": css.pk},
  14. )
  15. @pytest.fixture
  16. def data():
  17. return {"name": "test.css", "source": ".page-header { padding: 0}"}
  18. def test_css_creation_form_is_displayed(admin_client, create_link):
  19. response = admin_client.get(create_link)
  20. assert response.status_code == 200
  21. assert_contains(response, "New CSS")
  22. def test_css_can_be_created(theme, admin_client, create_link, data):
  23. admin_client.post(create_link, data)
  24. assert theme.css.exists()
  25. def test_css_is_created_with_entered_name(theme, admin_client, create_link, data):
  26. admin_client.post(create_link, data)
  27. assert theme.css.last().name == data["name"]
  28. def test_created_source_file_contains_css_entered_by_user(
  29. theme, admin_client, create_link, data
  30. ):
  31. admin_client.post(create_link, data)
  32. css = theme.css.last()
  33. assert css.source_file.read().decode("utf-8") == data["source"]
  34. def test_source_file_is_created_in_theme_directory(
  35. theme, admin_client, create_link, data
  36. ):
  37. admin_client.post(create_link, data)
  38. css = theme.css.last()
  39. assert theme.dirname in str(css.source_file)
  40. def test_created_source_file_name_starts_with_asset_name(
  41. theme, admin_client, create_link, data
  42. ):
  43. admin_client.post(create_link, data)
  44. css = theme.css.last()
  45. source_filename = str(css.source_file).split("/")[-1]
  46. assert source_filename.startswith("test.")
  47. def test_created_source_file_has_css_extension(theme, admin_client, create_link, data):
  48. admin_client.post(create_link, data)
  49. css = theme.css.last()
  50. source_filename = str(css.source_file).split("/")[-1]
  51. assert source_filename.endswith(".css")
  52. def test_created_source_file_is_hashed(theme, admin_client, create_link, data):
  53. admin_client.post(create_link, data)
  54. css = theme.css.last()
  55. source_filename = str(css.source_file).split("/")[-1]
  56. assert ".%s." % css.source_hash in source_filename
  57. def test_css_creation_fails_if_name_is_not_given(
  58. theme, admin_client, create_link, data
  59. ):
  60. data["name"] = ""
  61. admin_client.post(create_link, data)
  62. assert not theme.css.exists()
  63. def test_css_creation_fails_if_name_is_not_valid(
  64. theme, admin_client, create_link, data
  65. ):
  66. data["name"] = "missing-extension"
  67. admin_client.post(create_link, data)
  68. assert not theme.css.exists()
  69. def test_css_creation_fails_if_name_is_already_taken_by_other_css_in_theme(
  70. theme, admin_client, create_link, data, css
  71. ):
  72. data["name"] = css.name
  73. admin_client.post(create_link, data)
  74. assert theme.css.count() == 1
  75. def test_css_name_usage_check_passess_if_name_is_used_by_other_theme_css(
  76. other_theme, admin_client, data, css
  77. ):
  78. create_link = reverse(
  79. "misago:admin:appearance:themes:new-css-file", kwargs={"pk": other_theme.pk}
  80. )
  81. data["name"] = css.name
  82. admin_client.post(create_link, data)
  83. assert other_theme.css.exists()
  84. def test_css_creation_fails_if_source_is_not_given(
  85. theme, admin_client, create_link, data
  86. ):
  87. data["source"] = ""
  88. admin_client.post(create_link, data)
  89. assert not theme.css.exists()
  90. def test_css_file_without_url_is_created_without_rebuilding_flag(
  91. theme, admin_client, create_link, data
  92. ):
  93. admin_client.post(create_link, data)
  94. css = theme.css.last()
  95. assert not css.source_needs_building
  96. def test_css_file_with_image_url_is_created_with_rebuilding_flag(
  97. theme, admin_client, create_link, data, image
  98. ):
  99. data["source"] = "body { background-image: url(/static/%s); }" % image.name
  100. admin_client.post(create_link, data)
  101. css = theme.css.last()
  102. assert css.source_needs_building
  103. def test_creating_css_file_without_image_url_doesnt_trigger_single_css_file_rebuild(
  104. theme, admin_client, create_link, data, image, mock_build_single_theme_css
  105. ):
  106. admin_client.post(create_link, data)
  107. mock_build_single_theme_css.assert_not_called()
  108. def test_creating_css_file_with_image_url_triggers_single_css_file_rebuild(
  109. theme, admin_client, create_link, data, image, mock_build_single_theme_css
  110. ):
  111. data["source"] = "body { background-image: url(/static/%s); }" % image.name
  112. admin_client.post(create_link, data)
  113. css = theme.css.last()
  114. mock_build_single_theme_css.assert_called_once_with(css.pk)
  115. def test_css_file_is_created_with_correct_order(
  116. theme, admin_client, create_link, css_link, data
  117. ):
  118. admin_client.post(create_link, data)
  119. css = theme.css.get(name=data["name"])
  120. assert css.order == 1
  121. def test_error_message_is_set_if_user_attempts_to_create_css_in_default_theme(
  122. default_theme, admin_client
  123. ):
  124. create_link = reverse(
  125. "misago:admin:appearance:themes:new-css-file", kwargs={"pk": default_theme.pk}
  126. )
  127. response = admin_client.get(create_link)
  128. assert_has_error_message(response)
  129. def test_error_message_is_set_if_user_attempts_to_create_css_in_nonexisting_theme(
  130. nonexisting_theme, admin_client
  131. ):
  132. create_link = reverse(
  133. "misago:admin:appearance:themes:new-css-file",
  134. kwargs={"pk": nonexisting_theme.pk},
  135. )
  136. response = admin_client.get(create_link)
  137. assert_has_error_message(response)
  138. def test_css_creation_form_redirects_user_to_edition_after_creation(
  139. theme, admin_client, create_link, data
  140. ):
  141. data["stay"] = "1"
  142. response = admin_client.post(create_link, data)
  143. assert response["location"] == reverse(
  144. "misago:admin:appearance:themes:edit-css-file",
  145. kwargs={"pk": theme.pk, "css_pk": theme.css.last().pk},
  146. )
  147. def test_css_edition_form_is_displayed(admin_client, edit_link, css):
  148. response = admin_client.get(edit_link)
  149. assert response.status_code == 200
  150. assert_contains(response, css.name)
  151. def test_css_edition_form_contains_source_file_contents(admin_client, edit_link, css):
  152. response = admin_client.get(edit_link)
  153. assert_contains(response, css.source_file.read().decode("utf-8"))
  154. def test_css_name_can_be_changed(admin_client, edit_link, css, data):
  155. data["name"] = "new-name.css"
  156. admin_client.post(edit_link, data)
  157. css.refresh_from_db()
  158. assert css.name == data["name"]
  159. def test_css_name_change_also_changes_source_file_name(
  160. admin_client, edit_link, css, data
  161. ):
  162. data["name"] = "new-name.css"
  163. admin_client.post(edit_link, data)
  164. css.refresh_from_db()
  165. assert "new-name" in str(css.source_file)
  166. def test_css_source_can_be_changed(admin_client, edit_link, css, data):
  167. data["source"] = ".misago-footer { display: none; }"
  168. admin_client.post(edit_link, data)
  169. css.refresh_from_db()
  170. assert css.source_file.read().decode("utf-8") == data["source"]
  171. def test_changing_css_source_also_changes_source_hash(
  172. admin_client, edit_link, css, data
  173. ):
  174. original_hash = css.source_hash
  175. data["source"] = ".misago-footer { display: none; }"
  176. admin_client.post(edit_link, data)
  177. css.refresh_from_db()
  178. assert css.source_hash != original_hash
  179. def test_changing_css_source_also_changes_hash_in_filename(
  180. admin_client, edit_link, css, data
  181. ):
  182. original_hash = css.source_hash
  183. data["source"] = ".misago-footer { display: none; }"
  184. admin_client.post(edit_link, data)
  185. css.refresh_from_db()
  186. assert original_hash not in str(css.source_file)
  187. assert css.source_hash in str(css.source_file)
  188. def test_hash_stays_same_if_source_is_not_changed(admin_client, edit_link, css, data):
  189. original_hash = css.source_hash
  190. data["name"] = "changed.css"
  191. data["source"] = css.source_file.read().decode("utf-8")
  192. admin_client.post(edit_link, data)
  193. css.refresh_from_db()
  194. assert original_hash == css.source_hash
  195. def test_file_is_not_updated_if_form_data_has_no_changes(
  196. admin_client, edit_link, css, data
  197. ):
  198. original_source_file = str(css.source_file)
  199. data["name"] = css.name
  200. data["source"] = css.source_file.read().decode("utf-8")
  201. admin_client.post(edit_link, data)
  202. css.refresh_from_db()
  203. assert original_source_file == str(css.source_file)
  204. def test_adding_image_url_to_edited_file_sets_rebuilding_flag(
  205. theme, admin_client, edit_link, css, data, image
  206. ):
  207. data["source"] = "body { background-image: url(/static/%s); }" % image.name
  208. admin_client.post(edit_link, data)
  209. css.refresh_from_db()
  210. assert css.source_needs_building
  211. def test_removing_url_from_edited_file_removes_rebuilding_flag(
  212. theme, admin_client, edit_link, css, data
  213. ):
  214. css.source_needs_building = True
  215. css.save()
  216. data["source"] = "body { background-image: none; }"
  217. admin_client.post(edit_link, data)
  218. css.refresh_from_db()
  219. assert not css.source_needs_building
  220. def test_adding_image_url_to_edited_file_triggers_single_css_file_rebuild(
  221. theme, admin_client, edit_link, css, data, image, mock_build_single_theme_css
  222. ):
  223. data["source"] = "body { background-image: url(/static/%s); }" % image.name
  224. admin_client.post(edit_link, data)
  225. mock_build_single_theme_css.assert_called_once_with(css.pk)
  226. def test_removing_url_from_edited_file_deosnt_trigger_single_css_file_rebuild(
  227. theme, admin_client, edit_link, css, data, mock_build_single_theme_css
  228. ):
  229. data["source"] = "body { background-image: none; }"
  230. admin_client.post(edit_link, data)
  231. mock_build_single_theme_css.assert_not_called()
  232. def test_css_order_stays_the_same_after_edit(admin_client, edit_link, css, data):
  233. original_order = css.order
  234. data["name"] = "changed.css"
  235. admin_client.post(edit_link, data)
  236. css.refresh_from_db()
  237. assert css.order == original_order
  238. def test_css_edit_form_redirects_user_to_edition_after_saving(
  239. theme, admin_client, edit_link, css, data
  240. ):
  241. data["stay"] = "1"
  242. response = admin_client.post(edit_link, data)
  243. assert response["location"] == edit_link
  244. def test_error_message_is_set_if_user_attempts_to_edit_css_file_in_default_theme(
  245. default_theme, admin_client
  246. ):
  247. edit_link = reverse(
  248. "misago:admin:appearance:themes:edit-css-file",
  249. kwargs={"pk": default_theme.pk, "css_pk": 1},
  250. )
  251. response = admin_client.get(edit_link)
  252. assert_has_error_message(response)
  253. def test_error_message_is_set_if_user_attempts_to_edit_css_file_in_nonexisting_theme(
  254. nonexisting_theme, admin_client
  255. ):
  256. edit_link = reverse(
  257. "misago:admin:appearance:themes:edit-css-file",
  258. kwargs={"pk": nonexisting_theme.pk, "css_pk": 1},
  259. )
  260. response = admin_client.get(edit_link)
  261. assert_has_error_message(response)
  262. def test_error_message_is_set_if_user_attempts_to_edit_css_belonging_to_other_theme(
  263. other_theme, admin_client, css
  264. ):
  265. edit_link = reverse(
  266. "misago:admin:appearance:themes:edit-css-file",
  267. kwargs={"pk": other_theme.pk, "css_pk": css.pk},
  268. )
  269. response = admin_client.get(edit_link)
  270. assert_has_error_message(response)
  271. def test_error_message_is_set_if_user_attempts_to_edit_nonexisting_css(
  272. theme, admin_client
  273. ):
  274. edit_link = reverse(
  275. "misago:admin:appearance:themes:edit-css-file",
  276. kwargs={"pk": theme.pk, "css_pk": 1},
  277. )
  278. response = admin_client.get(edit_link)
  279. assert_has_error_message(response)
  280. def test_error_message_is_set_if_user_attempts_to_edit_css_link_with_file_form(
  281. theme, admin_client, css_link
  282. ):
  283. edit_link = reverse(
  284. "misago:admin:appearance:themes:edit-css-file",
  285. kwargs={"pk": theme.pk, "css_pk": css_link.pk},
  286. )
  287. response = admin_client.get(edit_link)
  288. assert_has_error_message(response)