logo

pleroma

My custom branche(s) on git.pleroma.social/pleroma/pleroma git clone https://hacktivis.me/git/pleroma.git

backup_test.exs (12614B)


  1. # Pleroma: A lightweight social networking server
  2. # Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/>
  3. # SPDX-License-Identifier: AGPL-3.0-only
  4. defmodule Pleroma.User.BackupTest do
  5. use Oban.Testing, repo: Pleroma.Repo
  6. use Pleroma.DataCase
  7. import Mock
  8. import Pleroma.Factory
  9. import Swoosh.TestAssertions
  10. import Mox
  11. alias Pleroma.Bookmark
  12. alias Pleroma.Tests.ObanHelpers
  13. alias Pleroma.UnstubbedConfigMock, as: ConfigMock
  14. alias Pleroma.Uploaders.S3.ExAwsMock
  15. alias Pleroma.User.Backup
  16. alias Pleroma.User.Backup.ProcessorMock
  17. alias Pleroma.Web.CommonAPI
  18. alias Pleroma.Workers.BackupWorker
  19. setup do
  20. clear_config([Pleroma.Upload, :uploader])
  21. clear_config([Backup, :limit_days])
  22. clear_config([Pleroma.Emails.Mailer, :enabled], true)
  23. ConfigMock
  24. |> stub_with(Pleroma.Config)
  25. ProcessorMock
  26. |> stub_with(Pleroma.User.Backup.Processor)
  27. :ok
  28. end
  29. test "it does not requrie enabled email" do
  30. clear_config([Pleroma.Emails.Mailer, :enabled], false)
  31. user = insert(:user)
  32. assert {:ok, _} = Backup.create(user)
  33. end
  34. test "it does not require user's email" do
  35. user = insert(:user, %{email: nil})
  36. assert {:ok, _} = Backup.create(user)
  37. end
  38. test "it creates a backup record and an Oban job" do
  39. %{id: user_id} = user = insert(:user)
  40. assert {:ok, %Oban.Job{args: args}} = Backup.create(user)
  41. assert_enqueued(worker: BackupWorker, args: args)
  42. backup = Backup.get(args["backup_id"])
  43. assert %Backup{user_id: ^user_id, processed: false, file_size: 0, state: :pending} = backup
  44. end
  45. test "it return an error if the export limit is over" do
  46. %{id: user_id} = user = insert(:user)
  47. limit_days = Pleroma.Config.get([Backup, :limit_days])
  48. assert {:ok, %Oban.Job{args: args}} = Backup.create(user)
  49. backup = Backup.get(args["backup_id"])
  50. assert %Backup{user_id: ^user_id, processed: false, file_size: 0} = backup
  51. assert Backup.create(user) == {:error, "Last export was less than #{limit_days} days ago"}
  52. end
  53. test "it process a backup record" do
  54. clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
  55. %{id: user_id} = user = insert(:user)
  56. assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id} = args}} = Backup.create(user)
  57. assert {:ok, backup} = perform_job(BackupWorker, args)
  58. assert backup.file_size > 0
  59. assert %Backup{id: ^backup_id, processed: true, user_id: ^user_id, state: :complete} = backup
  60. delete_job_args = %{"op" => "delete", "backup_id" => backup_id}
  61. assert_enqueued(worker: BackupWorker, args: delete_job_args)
  62. assert {:ok, backup} = perform_job(BackupWorker, delete_job_args)
  63. refute Backup.get(backup_id)
  64. email = Pleroma.Emails.UserEmail.backup_is_ready_email(backup)
  65. assert_email_sent(
  66. to: {user.name, user.email},
  67. html_body: email.html_body
  68. )
  69. end
  70. test "it updates states of the backup" do
  71. clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
  72. %{id: user_id} = user = insert(:user)
  73. assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id} = args}} = Backup.create(user)
  74. assert {:ok, backup} = perform_job(BackupWorker, args)
  75. assert backup.file_size > 0
  76. assert %Backup{id: ^backup_id, processed: true, user_id: ^user_id, state: :complete} = backup
  77. delete_job_args = %{"op" => "delete", "backup_id" => backup_id}
  78. assert_enqueued(worker: BackupWorker, args: delete_job_args)
  79. assert {:ok, backup} = perform_job(BackupWorker, delete_job_args)
  80. refute Backup.get(backup_id)
  81. email = Pleroma.Emails.UserEmail.backup_is_ready_email(backup)
  82. assert_email_sent(
  83. to: {user.name, user.email},
  84. html_body: email.html_body
  85. )
  86. end
  87. test "it does not send an email if the user does not have an email" do
  88. clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
  89. %{id: user_id} = user = insert(:user, %{email: nil})
  90. assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id} = args}} = Backup.create(user)
  91. assert {:ok, backup} = perform_job(BackupWorker, args)
  92. assert backup.file_size > 0
  93. assert %Backup{id: ^backup_id, processed: true, user_id: ^user_id} = backup
  94. assert_no_email_sent()
  95. end
  96. test "it does not send an email if mailer is not on" do
  97. clear_config([Pleroma.Emails.Mailer, :enabled], false)
  98. clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
  99. %{id: user_id} = user = insert(:user)
  100. assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id} = args}} = Backup.create(user)
  101. assert {:ok, backup} = perform_job(BackupWorker, args)
  102. assert backup.file_size > 0
  103. assert %Backup{id: ^backup_id, processed: true, user_id: ^user_id} = backup
  104. assert_no_email_sent()
  105. end
  106. test "it does not send an email if the user has an empty email" do
  107. clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
  108. %{id: user_id} = user = insert(:user, %{email: ""})
  109. assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id} = args}} = Backup.create(user)
  110. assert {:ok, backup} = perform_job(BackupWorker, args)
  111. assert backup.file_size > 0
  112. assert %Backup{id: ^backup_id, processed: true, user_id: ^user_id} = backup
  113. assert_no_email_sent()
  114. end
  115. test "it removes outdated backups after creating a fresh one" do
  116. clear_config([Backup, :limit_days], -1)
  117. clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
  118. user = insert(:user)
  119. assert {:ok, job1} = Backup.create(user)
  120. assert {:ok, %Backup{}} = ObanHelpers.perform(job1)
  121. assert {:ok, job2} = Backup.create(user)
  122. assert Pleroma.Repo.aggregate(Backup, :count) == 2
  123. assert {:ok, backup2} = ObanHelpers.perform(job2)
  124. ObanHelpers.perform_all()
  125. assert [^backup2] = Pleroma.Repo.all(Backup)
  126. end
  127. test "it creates a zip archive with user data" do
  128. user = insert(:user, %{nickname: "cofe", name: "Cofe", ap_id: "http://cofe.io/users/cofe"})
  129. {:ok, %{object: %{data: %{"id" => id1}}} = status1} =
  130. CommonAPI.post(user, %{status: "status1"})
  131. {:ok, %{object: %{data: %{"id" => id2}}} = status2} =
  132. CommonAPI.post(user, %{status: "status2"})
  133. {:ok, %{object: %{data: %{"id" => id3}}} = status3} =
  134. CommonAPI.post(user, %{status: "status3"})
  135. CommonAPI.favorite(user, status1.id)
  136. CommonAPI.favorite(user, status2.id)
  137. Bookmark.create(user.id, status2.id)
  138. Bookmark.create(user.id, status3.id)
  139. assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
  140. assert {:ok, path} = Backup.export(backup, self())
  141. assert {:ok, zipfile} = :zip.zip_open(String.to_charlist(path), [:memory])
  142. assert {:ok, {'actor.json', json}} = :zip.zip_get('actor.json', zipfile)
  143. assert %{
  144. "@context" => [
  145. "https://www.w3.org/ns/activitystreams",
  146. "http://localhost:4001/schemas/litepub-0.1.jsonld",
  147. %{"@language" => "und"}
  148. ],
  149. "bookmarks" => "bookmarks.json",
  150. "followers" => "http://cofe.io/users/cofe/followers",
  151. "following" => "http://cofe.io/users/cofe/following",
  152. "id" => "http://cofe.io/users/cofe",
  153. "inbox" => "http://cofe.io/users/cofe/inbox",
  154. "likes" => "likes.json",
  155. "name" => "Cofe",
  156. "outbox" => "http://cofe.io/users/cofe/outbox",
  157. "preferredUsername" => "cofe",
  158. "publicKey" => %{
  159. "id" => "http://cofe.io/users/cofe#main-key",
  160. "owner" => "http://cofe.io/users/cofe"
  161. },
  162. "type" => "Person",
  163. "url" => "http://cofe.io/users/cofe"
  164. } = Jason.decode!(json)
  165. assert {:ok, {'outbox.json', json}} = :zip.zip_get('outbox.json', zipfile)
  166. assert %{
  167. "@context" => "https://www.w3.org/ns/activitystreams",
  168. "id" => "outbox.json",
  169. "orderedItems" => [
  170. %{
  171. "object" => %{
  172. "actor" => "http://cofe.io/users/cofe",
  173. "content" => "status1",
  174. "type" => "Note"
  175. },
  176. "type" => "Create"
  177. },
  178. %{
  179. "object" => %{
  180. "actor" => "http://cofe.io/users/cofe",
  181. "content" => "status2"
  182. }
  183. },
  184. %{
  185. "actor" => "http://cofe.io/users/cofe",
  186. "object" => %{
  187. "content" => "status3"
  188. }
  189. }
  190. ],
  191. "totalItems" => 3,
  192. "type" => "OrderedCollection"
  193. } = Jason.decode!(json)
  194. assert {:ok, {'likes.json', json}} = :zip.zip_get('likes.json', zipfile)
  195. assert %{
  196. "@context" => "https://www.w3.org/ns/activitystreams",
  197. "id" => "likes.json",
  198. "orderedItems" => [^id1, ^id2],
  199. "totalItems" => 2,
  200. "type" => "OrderedCollection"
  201. } = Jason.decode!(json)
  202. assert {:ok, {'bookmarks.json', json}} = :zip.zip_get('bookmarks.json', zipfile)
  203. assert %{
  204. "@context" => "https://www.w3.org/ns/activitystreams",
  205. "id" => "bookmarks.json",
  206. "orderedItems" => [^id2, ^id3],
  207. "totalItems" => 2,
  208. "type" => "OrderedCollection"
  209. } = Jason.decode!(json)
  210. :zip.zip_close(zipfile)
  211. File.rm!(path)
  212. end
  213. test "it counts the correct number processed" do
  214. user = insert(:user, %{nickname: "cofe", name: "Cofe", ap_id: "http://cofe.io/users/cofe"})
  215. Enum.map(1..120, fn i ->
  216. {:ok, status} = CommonAPI.post(user, %{status: "status #{i}"})
  217. CommonAPI.favorite(user, status.id)
  218. Bookmark.create(user.id, status.id)
  219. end)
  220. assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
  221. {:ok, backup} = Backup.process(backup)
  222. assert backup.processed_number == 1 + 120 + 120 + 120
  223. Backup.delete(backup)
  224. end
  225. test "it handles errors" do
  226. user = insert(:user, %{nickname: "cofe", name: "Cofe", ap_id: "http://cofe.io/users/cofe"})
  227. Enum.map(1..120, fn i ->
  228. {:ok, _status} = CommonAPI.post(user, %{status: "status #{i}"})
  229. end)
  230. assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
  231. with_mock Pleroma.Web.ActivityPub.Transmogrifier,
  232. [:passthrough],
  233. prepare_outgoing: fn data ->
  234. object =
  235. data["object"]
  236. |> Pleroma.Object.normalize(fetch: false)
  237. |> Map.get(:data)
  238. data = data |> Map.put("object", object)
  239. if String.contains?(data["object"]["content"], "119"),
  240. do: raise(%Postgrex.Error{}),
  241. else: {:ok, data}
  242. end do
  243. {:ok, backup} = Backup.process(backup)
  244. assert backup.processed
  245. assert backup.state == :complete
  246. assert backup.processed_number == 1 + 119
  247. Backup.delete(backup)
  248. end
  249. end
  250. describe "it uploads and deletes a backup archive" do
  251. setup do
  252. clear_config([Pleroma.Upload, :base_url], "https://s3.amazonaws.com")
  253. clear_config([Pleroma.Uploaders.S3, :bucket], "test_bucket")
  254. user = insert(:user, %{nickname: "cofe", name: "Cofe", ap_id: "http://cofe.io/users/cofe"})
  255. {:ok, status1} = CommonAPI.post(user, %{status: "status1"})
  256. {:ok, status2} = CommonAPI.post(user, %{status: "status2"})
  257. {:ok, status3} = CommonAPI.post(user, %{status: "status3"})
  258. CommonAPI.favorite(user, status1.id)
  259. CommonAPI.favorite(user, status2.id)
  260. Bookmark.create(user.id, status2.id)
  261. Bookmark.create(user.id, status3.id)
  262. assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
  263. assert {:ok, path} = Backup.export(backup, self())
  264. [path: path, backup: backup]
  265. end
  266. test "S3", %{path: path, backup: backup} do
  267. clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.S3)
  268. clear_config([Pleroma.Uploaders.S3, :streaming_enabled], false)
  269. ExAwsMock
  270. |> expect(:request, 2, fn
  271. %{http_method: :put} -> {:ok, :ok}
  272. %{http_method: :delete} -> {:ok, %{status_code: 204}}
  273. end)
  274. assert {:ok, %Pleroma.Upload{}} = Backup.upload(backup, path)
  275. assert {:ok, _backup} = Backup.delete(backup)
  276. end
  277. test "Local", %{path: path, backup: backup} do
  278. clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
  279. assert {:ok, %Pleroma.Upload{}} = Backup.upload(backup, path)
  280. assert {:ok, _backup} = Backup.delete(backup)
  281. end
  282. end
  283. end