2020-08-25 16:51:09 +02:00
|
|
|
# Pleroma: A lightweight social networking server
|
|
|
|
# Copyright © 2017-2020 Pleroma Authors <https://pleroma.social/>
|
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2020-10-20 15:16:58 +02:00
|
|
|
defmodule Pleroma.User.BackupTest do
|
2020-09-02 19:45:22 +02:00
|
|
|
use Oban.Testing, repo: Pleroma.Repo
|
2020-08-25 16:51:09 +02:00
|
|
|
use Pleroma.DataCase
|
2020-09-02 19:45:22 +02:00
|
|
|
|
2020-08-31 21:07:14 +02:00
|
|
|
import Mock
|
2020-09-17 16:42:24 +02:00
|
|
|
import Pleroma.Factory
|
|
|
|
import Swoosh.TestAssertions
|
2020-08-25 16:51:09 +02:00
|
|
|
|
|
|
|
alias Pleroma.Bookmark
|
2020-09-04 20:00:26 +02:00
|
|
|
alias Pleroma.Tests.ObanHelpers
|
2020-10-20 15:47:04 +02:00
|
|
|
alias Pleroma.User.Backup
|
2020-08-25 17:16:01 +02:00
|
|
|
alias Pleroma.Web.CommonAPI
|
2020-09-02 19:45:22 +02:00
|
|
|
alias Pleroma.Workers.BackupWorker
|
|
|
|
|
2020-09-04 19:48:52 +02:00
|
|
|
setup do
|
|
|
|
clear_config([Pleroma.Upload, :uploader])
|
2020-10-20 15:16:58 +02:00
|
|
|
clear_config([Backup, :limit_days])
|
2020-10-19 13:30:32 +02:00
|
|
|
clear_config([Pleroma.Emails.Mailer, :enabled], true)
|
2020-09-15 16:07:28 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
test "it requries enabled email" do
|
|
|
|
Pleroma.Config.put([Pleroma.Emails.Mailer, :enabled], false)
|
|
|
|
user = insert(:user)
|
|
|
|
assert {:error, "Backups require enabled email"} == Backup.create(user)
|
|
|
|
end
|
|
|
|
|
|
|
|
test "it requries user's email" do
|
|
|
|
user = insert(:user, %{email: nil})
|
|
|
|
assert {:error, "Email is required"} == Backup.create(user)
|
2020-09-04 19:48:52 +02:00
|
|
|
end
|
2020-08-25 16:51:09 +02:00
|
|
|
|
2020-09-02 19:45:22 +02:00
|
|
|
test "it creates a backup record and an Oban job" do
|
2020-09-02 18:21:33 +02:00
|
|
|
%{id: user_id} = user = insert(:user)
|
2020-09-02 19:45:22 +02:00
|
|
|
assert {:ok, %Oban.Job{args: args}} = Backup.create(user)
|
|
|
|
assert_enqueued(worker: BackupWorker, args: args)
|
2020-09-02 18:21:33 +02:00
|
|
|
|
2020-09-02 19:45:22 +02:00
|
|
|
backup = Backup.get(args["backup_id"])
|
2020-09-02 18:21:33 +02:00
|
|
|
assert %Backup{user_id: ^user_id, processed: false, file_size: 0} = backup
|
|
|
|
end
|
|
|
|
|
|
|
|
test "it return an error if the export limit is over" do
|
|
|
|
%{id: user_id} = user = insert(:user)
|
2020-10-20 15:16:58 +02:00
|
|
|
limit_days = Pleroma.Config.get([Backup, :limit_days])
|
2020-09-02 19:45:22 +02:00
|
|
|
assert {:ok, %Oban.Job{args: args}} = Backup.create(user)
|
|
|
|
backup = Backup.get(args["backup_id"])
|
2020-09-02 18:21:33 +02:00
|
|
|
assert %Backup{user_id: ^user_id, processed: false, file_size: 0} = backup
|
|
|
|
|
|
|
|
assert Backup.create(user) == {:error, "Last export was less than #{limit_days} days ago"}
|
|
|
|
end
|
|
|
|
|
|
|
|
test "it process a backup record" do
|
2020-09-02 19:45:22 +02:00
|
|
|
Pleroma.Config.put([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
2020-09-02 18:21:33 +02:00
|
|
|
%{id: user_id} = user = insert(:user)
|
2020-09-04 19:48:52 +02:00
|
|
|
|
|
|
|
assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id} = args}} = Backup.create(user)
|
|
|
|
assert {:ok, backup} = perform_job(BackupWorker, args)
|
2020-09-02 18:21:33 +02:00
|
|
|
assert backup.file_size > 0
|
|
|
|
assert %Backup{id: ^backup_id, processed: true, user_id: ^user_id} = backup
|
2020-09-04 19:48:52 +02:00
|
|
|
|
|
|
|
delete_job_args = %{"op" => "delete", "backup_id" => backup_id}
|
|
|
|
|
|
|
|
assert_enqueued(worker: BackupWorker, args: delete_job_args)
|
|
|
|
assert {:ok, backup} = perform_job(BackupWorker, delete_job_args)
|
|
|
|
refute Backup.get(backup_id)
|
2020-09-17 16:42:24 +02:00
|
|
|
|
2020-09-23 18:23:11 +02:00
|
|
|
email = Pleroma.Emails.UserEmail.backup_is_ready_email(backup)
|
|
|
|
|
|
|
|
assert_email_sent(
|
|
|
|
to: {user.name, user.email},
|
|
|
|
html_body: email.html_body
|
|
|
|
)
|
2020-09-04 19:48:52 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
test "it removes outdated backups after creating a fresh one" do
|
2020-10-20 15:16:58 +02:00
|
|
|
Pleroma.Config.put([Backup, :limit_days], -1)
|
2020-09-04 19:48:52 +02:00
|
|
|
Pleroma.Config.put([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
|
|
|
user = insert(:user)
|
|
|
|
|
|
|
|
assert {:ok, job1} = Backup.create(user)
|
|
|
|
|
2020-10-28 12:32:44 +01:00
|
|
|
assert {:ok, %Backup{}} = ObanHelpers.perform(job1)
|
2020-09-04 19:48:52 +02:00
|
|
|
assert {:ok, job2} = Backup.create(user)
|
|
|
|
assert Pleroma.Repo.aggregate(Backup, :count) == 2
|
|
|
|
assert {:ok, backup2} = ObanHelpers.perform(job2)
|
|
|
|
|
|
|
|
ObanHelpers.perform_all()
|
|
|
|
|
|
|
|
assert [^backup2] = Pleroma.Repo.all(Backup)
|
2020-09-02 18:21:33 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
test "it creates a zip archive with user data" do
|
2020-08-25 16:51:09 +02:00
|
|
|
user = insert(:user, %{nickname: "cofe", name: "Cofe", ap_id: "http://cofe.io/users/cofe"})
|
|
|
|
|
|
|
|
{:ok, %{object: %{data: %{"id" => id1}}} = status1} =
|
|
|
|
CommonAPI.post(user, %{status: "status1"})
|
|
|
|
|
|
|
|
{:ok, %{object: %{data: %{"id" => id2}}} = status2} =
|
|
|
|
CommonAPI.post(user, %{status: "status2"})
|
|
|
|
|
|
|
|
{:ok, %{object: %{data: %{"id" => id3}}} = status3} =
|
|
|
|
CommonAPI.post(user, %{status: "status3"})
|
|
|
|
|
|
|
|
CommonAPI.favorite(user, status1.id)
|
|
|
|
CommonAPI.favorite(user, status2.id)
|
|
|
|
|
|
|
|
Bookmark.create(user.id, status2.id)
|
|
|
|
Bookmark.create(user.id, status3.id)
|
|
|
|
|
2020-09-02 18:21:33 +02:00
|
|
|
assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
|
2020-09-15 16:07:28 +02:00
|
|
|
assert {:ok, path} = Backup.export(backup)
|
2020-08-31 18:31:21 +02:00
|
|
|
assert {:ok, zipfile} = :zip.zip_open(String.to_charlist(path), [:memory])
|
2020-08-25 16:51:09 +02:00
|
|
|
assert {:ok, {'actor.json', json}} = :zip.zip_get('actor.json', zipfile)
|
|
|
|
|
|
|
|
assert %{
|
|
|
|
"@context" => [
|
|
|
|
"https://www.w3.org/ns/activitystreams",
|
|
|
|
"http://localhost:4001/schemas/litepub-0.1.jsonld",
|
|
|
|
%{"@language" => "und"}
|
|
|
|
],
|
|
|
|
"bookmarks" => "bookmarks.json",
|
|
|
|
"followers" => "http://cofe.io/users/cofe/followers",
|
|
|
|
"following" => "http://cofe.io/users/cofe/following",
|
|
|
|
"id" => "http://cofe.io/users/cofe",
|
|
|
|
"inbox" => "http://cofe.io/users/cofe/inbox",
|
|
|
|
"likes" => "likes.json",
|
|
|
|
"name" => "Cofe",
|
|
|
|
"outbox" => "http://cofe.io/users/cofe/outbox",
|
|
|
|
"preferredUsername" => "cofe",
|
|
|
|
"publicKey" => %{
|
|
|
|
"id" => "http://cofe.io/users/cofe#main-key",
|
|
|
|
"owner" => "http://cofe.io/users/cofe"
|
|
|
|
},
|
|
|
|
"type" => "Person",
|
|
|
|
"url" => "http://cofe.io/users/cofe"
|
|
|
|
} = Jason.decode!(json)
|
|
|
|
|
|
|
|
assert {:ok, {'outbox.json', json}} = :zip.zip_get('outbox.json', zipfile)
|
|
|
|
|
|
|
|
assert %{
|
|
|
|
"@context" => "https://www.w3.org/ns/activitystreams",
|
|
|
|
"id" => "outbox.json",
|
|
|
|
"orderedItems" => [
|
|
|
|
%{
|
|
|
|
"object" => %{
|
|
|
|
"actor" => "http://cofe.io/users/cofe",
|
|
|
|
"content" => "status1",
|
|
|
|
"type" => "Note"
|
|
|
|
},
|
|
|
|
"type" => "Create"
|
|
|
|
},
|
|
|
|
%{
|
|
|
|
"object" => %{
|
|
|
|
"actor" => "http://cofe.io/users/cofe",
|
|
|
|
"content" => "status2"
|
|
|
|
}
|
|
|
|
},
|
|
|
|
%{
|
|
|
|
"actor" => "http://cofe.io/users/cofe",
|
|
|
|
"object" => %{
|
|
|
|
"content" => "status3"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"totalItems" => 3,
|
|
|
|
"type" => "OrderedCollection"
|
|
|
|
} = Jason.decode!(json)
|
|
|
|
|
|
|
|
assert {:ok, {'likes.json', json}} = :zip.zip_get('likes.json', zipfile)
|
|
|
|
|
|
|
|
assert %{
|
|
|
|
"@context" => "https://www.w3.org/ns/activitystreams",
|
|
|
|
"id" => "likes.json",
|
|
|
|
"orderedItems" => [^id1, ^id2],
|
|
|
|
"totalItems" => 2,
|
|
|
|
"type" => "OrderedCollection"
|
|
|
|
} = Jason.decode!(json)
|
|
|
|
|
|
|
|
assert {:ok, {'bookmarks.json', json}} = :zip.zip_get('bookmarks.json', zipfile)
|
|
|
|
|
|
|
|
assert %{
|
|
|
|
"@context" => "https://www.w3.org/ns/activitystreams",
|
|
|
|
"id" => "bookmarks.json",
|
|
|
|
"orderedItems" => [^id2, ^id3],
|
|
|
|
"totalItems" => 2,
|
|
|
|
"type" => "OrderedCollection"
|
|
|
|
} = Jason.decode!(json)
|
|
|
|
|
|
|
|
:zip.zip_close(zipfile)
|
|
|
|
File.rm!(path)
|
|
|
|
end
|
2020-08-31 18:31:21 +02:00
|
|
|
|
2020-09-04 19:48:52 +02:00
|
|
|
describe "it uploads and deletes a backup archive" do
|
2020-08-31 21:07:14 +02:00
|
|
|
setup do
|
|
|
|
clear_config(Pleroma.Uploaders.S3,
|
2021-01-12 23:35:10 +01:00
|
|
|
bucket: "test_bucket"
|
2020-08-31 21:07:14 +02:00
|
|
|
)
|
2020-08-31 18:31:21 +02:00
|
|
|
|
2020-08-31 21:07:14 +02:00
|
|
|
clear_config([Pleroma.Upload, :uploader])
|
2021-01-12 23:35:10 +01:00
|
|
|
clear_config([Pleroma.Upload, base_url: "https://s3.amazonaws.com"])
|
2020-08-31 18:31:21 +02:00
|
|
|
|
2020-08-31 21:07:14 +02:00
|
|
|
user = insert(:user, %{nickname: "cofe", name: "Cofe", ap_id: "http://cofe.io/users/cofe"})
|
|
|
|
|
|
|
|
{:ok, status1} = CommonAPI.post(user, %{status: "status1"})
|
|
|
|
{:ok, status2} = CommonAPI.post(user, %{status: "status2"})
|
|
|
|
{:ok, status3} = CommonAPI.post(user, %{status: "status3"})
|
|
|
|
CommonAPI.favorite(user, status1.id)
|
|
|
|
CommonAPI.favorite(user, status2.id)
|
|
|
|
Bookmark.create(user.id, status2.id)
|
|
|
|
Bookmark.create(user.id, status3.id)
|
|
|
|
|
2020-09-02 18:21:33 +02:00
|
|
|
assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
|
2020-09-15 16:07:28 +02:00
|
|
|
assert {:ok, path} = Backup.export(backup)
|
2020-08-31 21:07:14 +02:00
|
|
|
|
2020-09-02 18:21:33 +02:00
|
|
|
[path: path, backup: backup]
|
2020-08-31 21:07:14 +02:00
|
|
|
end
|
|
|
|
|
2020-09-02 18:21:33 +02:00
|
|
|
test "S3", %{path: path, backup: backup} do
|
2020-08-31 21:07:14 +02:00
|
|
|
Pleroma.Config.put([Pleroma.Upload, :uploader], Pleroma.Uploaders.S3)
|
|
|
|
|
2020-09-04 19:48:52 +02:00
|
|
|
with_mock ExAws,
|
|
|
|
request: fn
|
|
|
|
%{http_method: :put} -> {:ok, :ok}
|
|
|
|
%{http_method: :delete} -> {:ok, %{status_code: 204}}
|
|
|
|
end do
|
2020-09-02 18:21:33 +02:00
|
|
|
assert {:ok, %Pleroma.Upload{}} = Backup.upload(backup, path)
|
2020-09-04 19:48:52 +02:00
|
|
|
assert {:ok, _backup} = Backup.delete(backup)
|
|
|
|
end
|
|
|
|
|
|
|
|
with_mock ExAws, request: fn %{http_method: :delete} -> {:ok, %{status_code: 204}} end do
|
2020-08-31 21:07:14 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-09-02 18:21:33 +02:00
|
|
|
test "Local", %{path: path, backup: backup} do
|
2020-08-31 21:07:14 +02:00
|
|
|
Pleroma.Config.put([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
|
|
|
|
2020-09-02 18:21:33 +02:00
|
|
|
assert {:ok, %Pleroma.Upload{}} = Backup.upload(backup, path)
|
2020-09-04 19:48:52 +02:00
|
|
|
assert {:ok, _backup} = Backup.delete(backup)
|
2020-08-31 21:07:14 +02:00
|
|
|
end
|
2020-08-31 18:31:21 +02:00
|
|
|
end
|
2020-08-25 16:51:09 +02:00
|
|
|
end
|