Merge remote-tracking branch 'origin/develop' into fork
Signed-off-by: marcin mikołajczak <git@mkljczk.pl>
This commit is contained in:
commit
6f957c94a7
44 changed files with 1048 additions and 662 deletions
|
@ -2,5 +2,8 @@
|
||||||
{"lib/cachex.ex", "Unknown type: Spec.cache/0."},
|
{"lib/cachex.ex", "Unknown type: Spec.cache/0."},
|
||||||
{"lib/pleroma/web/plugs/rate_limiter.ex", "The pattern can never match the type {:commit, _} | {:ignore, _}."},
|
{"lib/pleroma/web/plugs/rate_limiter.ex", "The pattern can never match the type {:commit, _} | {:ignore, _}."},
|
||||||
{"lib/pleroma/web/plugs/rate_limiter.ex", "Function get_scale/2 will never be called."},
|
{"lib/pleroma/web/plugs/rate_limiter.ex", "Function get_scale/2 will never be called."},
|
||||||
{"lib/pleroma/web/plugs/rate_limiter.ex", "Function initialize_buckets!/1 will never be called."}
|
{"lib/pleroma/web/plugs/rate_limiter.ex", "Function initialize_buckets!/1 will never be called."},
|
||||||
|
{"lib/pleroma/workers/receiver_worker.ex", :call},
|
||||||
|
{"lib/pleroma/workers/receiver_worker.ex", :pattern_match},
|
||||||
|
{"lib/pleroma/workers/receiver_worker.ex", :pattern_match_cov},
|
||||||
]
|
]
|
||||||
|
|
1
changelog.d/backups-refactor.change
Normal file
1
changelog.d/backups-refactor.change
Normal file
|
@ -0,0 +1 @@
|
||||||
|
Refactor the user backups code and improve test coverage
|
1
changelog.d/disclose_client.fix
Normal file
1
changelog.d/disclose_client.fix
Normal file
|
@ -0,0 +1 @@
|
||||||
|
Client application data was always missing from the status
|
1
changelog.d/federator_outgoing_increase.change
Normal file
1
changelog.d/federator_outgoing_increase.change
Normal file
|
@ -0,0 +1 @@
|
||||||
|
Increase outgoing federation parallelism
|
0
changelog.d/fix-migration.skip
Normal file
0
changelog.d/fix-migration.skip
Normal file
1
changelog.d/hackney-pool-timeouts.change
Normal file
1
changelog.d/hackney-pool-timeouts.change
Normal file
|
@ -0,0 +1 @@
|
||||||
|
Change Hackney connection pool timeouts to align with the values Gun uses
|
0
changelog.d/oban-transmogrifier.skip
Normal file
0
changelog.d/oban-transmogrifier.skip
Normal file
1
changelog.d/optimistic-inbox-sigs.fix
Normal file
1
changelog.d/optimistic-inbox-sigs.fix
Normal file
|
@ -0,0 +1 @@
|
||||||
|
Fix Optimistic Inbox for failed signatures
|
1
changelog.d/publisher.change
Normal file
1
changelog.d/publisher.change
Normal file
|
@ -0,0 +1 @@
|
||||||
|
Publisher jobs now store the the activity id instead of inserting duplicate JSON data in the Oban queue for each delivery.
|
|
@ -598,10 +598,8 @@
|
||||||
queues: [
|
queues: [
|
||||||
activity_expiration: 10,
|
activity_expiration: 10,
|
||||||
federator_incoming: 5,
|
federator_incoming: 5,
|
||||||
federator_outgoing: 5,
|
federator_outgoing: 25,
|
||||||
web_push: 50,
|
web_push: 50,
|
||||||
transmogrifier: 20,
|
|
||||||
notifications: 20,
|
|
||||||
background: 20,
|
background: 20,
|
||||||
search_indexing: [limit: 10, paused: true],
|
search_indexing: [limit: 10, paused: true],
|
||||||
slow: 5
|
slow: 5
|
||||||
|
@ -882,19 +880,19 @@
|
||||||
config :pleroma, :hackney_pools,
|
config :pleroma, :hackney_pools,
|
||||||
federation: [
|
federation: [
|
||||||
max_connections: 50,
|
max_connections: 50,
|
||||||
timeout: 150_000
|
timeout: 10_000
|
||||||
],
|
],
|
||||||
media: [
|
media: [
|
||||||
max_connections: 50,
|
max_connections: 50,
|
||||||
timeout: 150_000
|
timeout: 15_000
|
||||||
],
|
],
|
||||||
rich_media: [
|
rich_media: [
|
||||||
max_connections: 50,
|
max_connections: 50,
|
||||||
timeout: 150_000
|
timeout: 15_000
|
||||||
],
|
],
|
||||||
upload: [
|
upload: [
|
||||||
max_connections: 25,
|
max_connections: 25,
|
||||||
timeout: 300_000
|
timeout: 15_000
|
||||||
]
|
]
|
||||||
|
|
||||||
config :pleroma, :majic_pool, size: 2
|
config :pleroma, :majic_pool, size: 2
|
||||||
|
@ -933,8 +931,8 @@
|
||||||
purge_after_days: 30,
|
purge_after_days: 30,
|
||||||
limit_days: 7,
|
limit_days: 7,
|
||||||
dir: nil,
|
dir: nil,
|
||||||
process_wait_time: 30_000,
|
process_chunk_size: 100,
|
||||||
process_chunk_size: 100
|
timeout: :timer.minutes(30)
|
||||||
|
|
||||||
config :pleroma, ConcurrentLimiter, [
|
config :pleroma, ConcurrentLimiter, [
|
||||||
{Pleroma.Search, [max_running: 30, max_waiting: 50]},
|
{Pleroma.Search, [max_running: 30, max_waiting: 50]},
|
||||||
|
|
|
@ -3386,20 +3386,19 @@
|
||||||
description: "Limit user to export not more often than once per N days",
|
description: "Limit user to export not more often than once per N days",
|
||||||
suggestions: [7]
|
suggestions: [7]
|
||||||
},
|
},
|
||||||
%{
|
|
||||||
key: :process_wait_time,
|
|
||||||
type: :integer,
|
|
||||||
label: "Process Wait Time",
|
|
||||||
description:
|
|
||||||
"The amount of time to wait for backup to report progress, in milliseconds. If no progress is received from the backup job for that much time, terminate it and deem it failed.",
|
|
||||||
suggestions: [30_000]
|
|
||||||
},
|
|
||||||
%{
|
%{
|
||||||
key: :process_chunk_size,
|
key: :process_chunk_size,
|
||||||
type: :integer,
|
type: :integer,
|
||||||
label: "Process Chunk Size",
|
label: "Process Chunk Size",
|
||||||
description: "The number of activities to fetch in the backup job for each chunk.",
|
description: "The number of activities to fetch in the backup job for each chunk.",
|
||||||
suggestions: [100]
|
suggestions: [100]
|
||||||
|
},
|
||||||
|
%{
|
||||||
|
key: :timeout,
|
||||||
|
type: :integer,
|
||||||
|
label: "Timeout",
|
||||||
|
description: "The amount of time to wait for backup to complete in seconds.",
|
||||||
|
suggestions: [1_800]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
|
|
@ -170,8 +170,7 @@
|
||||||
config :pleroma, Pleroma.Web.Plugs.HTTPSecurityPlug, config_impl: Pleroma.StaticStubbedConfigMock
|
config :pleroma, Pleroma.Web.Plugs.HTTPSecurityPlug, config_impl: Pleroma.StaticStubbedConfigMock
|
||||||
config :pleroma, Pleroma.Web.Plugs.HTTPSignaturePlug, config_impl: Pleroma.StaticStubbedConfigMock
|
config :pleroma, Pleroma.Web.Plugs.HTTPSignaturePlug, config_impl: Pleroma.StaticStubbedConfigMock
|
||||||
|
|
||||||
config :pleroma, Pleroma.Web.Plugs.HTTPSignaturePlug,
|
config :pleroma, Pleroma.Signature, http_signatures_impl: Pleroma.StubbedHTTPSignaturesMock
|
||||||
http_signatures_impl: Pleroma.StubbedHTTPSignaturesMock
|
|
||||||
|
|
||||||
peer_module =
|
peer_module =
|
||||||
if String.to_integer(System.otp_release()) >= 25 do
|
if String.to_integer(System.otp_release()) >= 25 do
|
||||||
|
@ -201,6 +200,8 @@
|
||||||
|
|
||||||
config :pleroma, Pleroma.Web.Plugs.HTTPSecurityPlug, enable: false
|
config :pleroma, Pleroma.Web.Plugs.HTTPSecurityPlug, enable: false
|
||||||
|
|
||||||
|
config :pleroma, Pleroma.User.Backup, tempdir: "test/tmp"
|
||||||
|
|
||||||
if File.exists?("./config/test.secret.exs") do
|
if File.exists?("./config/test.secret.exs") do
|
||||||
import_config "test.secret.exs"
|
import_config "test.secret.exs"
|
||||||
else
|
else
|
||||||
|
|
|
@ -1171,6 +1171,7 @@ Control favicons for instances.
|
||||||
3. the directory named by the TMP environment variable
|
3. the directory named by the TMP environment variable
|
||||||
4. C:\TMP on Windows or /tmp on Unix-like operating systems
|
4. C:\TMP on Windows or /tmp on Unix-like operating systems
|
||||||
5. as a last resort, the current working directory
|
5. as a last resort, the current working directory
|
||||||
|
* `:timeout` an integer representing seconds
|
||||||
|
|
||||||
## Frontend management
|
## Frontend management
|
||||||
|
|
||||||
|
|
|
@ -27,11 +27,3 @@
|
||||||
failed: 4,
|
failed: 4,
|
||||||
manual: 5
|
manual: 5
|
||||||
)
|
)
|
||||||
|
|
||||||
defenum(Pleroma.User.Backup.State,
|
|
||||||
pending: 1,
|
|
||||||
running: 2,
|
|
||||||
complete: 3,
|
|
||||||
failed: 4,
|
|
||||||
invalid: 5
|
|
||||||
)
|
|
||||||
|
|
|
@ -345,37 +345,22 @@ def unsubscribe_url(user, notifications_type) do
|
||||||
Router.Helpers.subscription_url(Endpoint, :unsubscribe, token)
|
Router.Helpers.subscription_url(Endpoint, :unsubscribe, token)
|
||||||
end
|
end
|
||||||
|
|
||||||
def backup_is_ready_email(backup, admin_user_id \\ nil) do
|
def backup_is_ready_email(backup) do
|
||||||
%{user: user} = Pleroma.Repo.preload(backup, :user)
|
%{user: user} = Pleroma.Repo.preload(backup, :user)
|
||||||
|
|
||||||
Gettext.with_locale_or_default user.language do
|
Gettext.with_locale_or_default user.language do
|
||||||
download_url = Pleroma.Web.PleromaAPI.BackupView.download_url(backup)
|
download_url = Pleroma.Web.PleromaAPI.BackupView.download_url(backup)
|
||||||
|
|
||||||
html_body =
|
html_body =
|
||||||
if is_nil(admin_user_id) do
|
Gettext.dpgettext(
|
||||||
Gettext.dpgettext(
|
"static_pages",
|
||||||
"static_pages",
|
"account archive email body",
|
||||||
"account archive email body - self-requested",
|
"""
|
||||||
"""
|
<p>A full backup of your Pleroma account was requested. It's ready for download:</p>
|
||||||
<p>You requested a full backup of your Pleroma account. It's ready for download:</p>
|
<p><a href="%{download_url}">%{download_url}</a></p>
|
||||||
<p><a href="%{download_url}">%{download_url}</a></p>
|
""",
|
||||||
""",
|
download_url: download_url
|
||||||
download_url: download_url
|
)
|
||||||
)
|
|
||||||
else
|
|
||||||
admin = Pleroma.Repo.get(User, admin_user_id)
|
|
||||||
|
|
||||||
Gettext.dpgettext(
|
|
||||||
"static_pages",
|
|
||||||
"account archive email body - admin requested",
|
|
||||||
"""
|
|
||||||
<p>Admin @%{admin_nickname} requested a full backup of your Pleroma account. It's ready for download:</p>
|
|
||||||
<p><a href="%{download_url}">%{download_url}</a></p>
|
|
||||||
""",
|
|
||||||
admin_nickname: admin.nickname,
|
|
||||||
download_url: download_url
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
new()
|
new()
|
||||||
|> to(recipient(user))
|
|> to(recipient(user))
|
||||||
|
|
|
@ -10,6 +10,14 @@ defmodule Pleroma.Signature do
|
||||||
alias Pleroma.User
|
alias Pleroma.User
|
||||||
alias Pleroma.Web.ActivityPub.ActivityPub
|
alias Pleroma.Web.ActivityPub.ActivityPub
|
||||||
|
|
||||||
|
import Plug.Conn, only: [put_req_header: 3]
|
||||||
|
|
||||||
|
@http_signatures_impl Application.compile_env(
|
||||||
|
:pleroma,
|
||||||
|
[__MODULE__, :http_signatures_impl],
|
||||||
|
HTTPSignatures
|
||||||
|
)
|
||||||
|
|
||||||
@known_suffixes ["/publickey", "/main-key"]
|
@known_suffixes ["/publickey", "/main-key"]
|
||||||
|
|
||||||
def key_id_to_actor_id(key_id) do
|
def key_id_to_actor_id(key_id) do
|
||||||
|
@ -85,4 +93,48 @@ def signed_date, do: signed_date(NaiveDateTime.utc_now())
|
||||||
def signed_date(%NaiveDateTime{} = date) do
|
def signed_date(%NaiveDateTime{} = date) do
|
||||||
Timex.format!(date, "{WDshort}, {0D} {Mshort} {YYYY} {h24}:{m}:{s} GMT")
|
Timex.format!(date, "{WDshort}, {0D} {Mshort} {YYYY} {h24}:{m}:{s} GMT")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@spec validate_signature(Plug.Conn.t(), String.t()) :: boolean()
|
||||||
|
def validate_signature(%Plug.Conn{} = conn, request_target) do
|
||||||
|
# Newer drafts for HTTP signatures now use @request-target instead of the
|
||||||
|
# old (request-target). We'll now support both for incoming signatures.
|
||||||
|
conn =
|
||||||
|
conn
|
||||||
|
|> put_req_header("(request-target)", request_target)
|
||||||
|
|> put_req_header("@request-target", request_target)
|
||||||
|
|
||||||
|
@http_signatures_impl.validate_conn(conn)
|
||||||
|
end
|
||||||
|
|
||||||
|
@spec validate_signature(Plug.Conn.t()) :: boolean()
|
||||||
|
def validate_signature(%Plug.Conn{} = conn) do
|
||||||
|
# This (request-target) is non-standard, but many implementations do it
|
||||||
|
# this way due to a misinterpretation of
|
||||||
|
# https://datatracker.ietf.org/doc/html/draft-cavage-http-signatures-06
|
||||||
|
# "path" was interpreted as not having the query, though later examples
|
||||||
|
# show that it must be the absolute path + query. This behavior is kept to
|
||||||
|
# make sure most software (Pleroma itself, Mastodon, and probably others)
|
||||||
|
# do not break.
|
||||||
|
request_target = Enum.join([String.downcase(conn.method), conn.request_path], " ")
|
||||||
|
|
||||||
|
# This is the proper way to build the @request-target, as expected by
|
||||||
|
# many HTTP signature libraries, clarified in the following draft:
|
||||||
|
# https://www.ietf.org/archive/id/draft-ietf-httpbis-message-signatures-11.html#section-2.2.6
|
||||||
|
# It is the same as before, but containing the query part as well.
|
||||||
|
proper_target = Enum.join([request_target, "?", conn.query_string], "")
|
||||||
|
|
||||||
|
cond do
|
||||||
|
# Normal, non-standard behavior but expected by Pleroma and more.
|
||||||
|
validate_signature(conn, request_target) ->
|
||||||
|
true
|
||||||
|
|
||||||
|
# Has query string and the previous one failed: let's try the standard.
|
||||||
|
conn.query_string != "" ->
|
||||||
|
validate_signature(conn, proper_target)
|
||||||
|
|
||||||
|
# If there's no query string and signature fails, it's rotten.
|
||||||
|
true ->
|
||||||
|
false
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -15,9 +15,10 @@ defmodule Pleroma.User.Backup do
|
||||||
alias Pleroma.Activity
|
alias Pleroma.Activity
|
||||||
alias Pleroma.Bookmark
|
alias Pleroma.Bookmark
|
||||||
alias Pleroma.Chat
|
alias Pleroma.Chat
|
||||||
|
alias Pleroma.Config
|
||||||
alias Pleroma.Repo
|
alias Pleroma.Repo
|
||||||
|
alias Pleroma.Uploaders.Uploader
|
||||||
alias Pleroma.User
|
alias Pleroma.User
|
||||||
alias Pleroma.User.Backup.State
|
|
||||||
alias Pleroma.Web.ActivityPub.ActivityPub
|
alias Pleroma.Web.ActivityPub.ActivityPub
|
||||||
alias Pleroma.Web.ActivityPub.Transmogrifier
|
alias Pleroma.Web.ActivityPub.Transmogrifier
|
||||||
alias Pleroma.Web.ActivityPub.UserView
|
alias Pleroma.Web.ActivityPub.UserView
|
||||||
|
@ -30,71 +31,111 @@ defmodule Pleroma.User.Backup do
|
||||||
field(:file_name, :string)
|
field(:file_name, :string)
|
||||||
field(:file_size, :integer, default: 0)
|
field(:file_size, :integer, default: 0)
|
||||||
field(:processed, :boolean, default: false)
|
field(:processed, :boolean, default: false)
|
||||||
field(:state, State, default: :invalid)
|
field(:tempdir, :string)
|
||||||
field(:processed_number, :integer, default: 0)
|
|
||||||
|
|
||||||
belongs_to(:user, User, type: FlakeId.Ecto.CompatType)
|
belongs_to(:user, User, type: FlakeId.Ecto.CompatType)
|
||||||
|
|
||||||
timestamps()
|
timestamps()
|
||||||
end
|
end
|
||||||
|
|
||||||
@config_impl Application.compile_env(:pleroma, [__MODULE__, :config_impl], Pleroma.Config)
|
@doc """
|
||||||
|
Schedules a job to backup a user if the number of backup requests has not exceeded the limit.
|
||||||
|
|
||||||
def create(user, admin_id \\ nil) do
|
Admins can directly call new/1 and schedule_backup/1 to bypass the limit.
|
||||||
with :ok <- validate_limit(user, admin_id),
|
"""
|
||||||
{:ok, backup} <- user |> new() |> Repo.insert() do
|
@spec user(User.t()) :: {:ok, t()} | {:error, any()}
|
||||||
BackupWorker.process(backup, admin_id)
|
def user(user) do
|
||||||
|
days = Config.get([__MODULE__, :limit_days])
|
||||||
|
|
||||||
|
with true <- permitted?(user),
|
||||||
|
%__MODULE__{} = backup <- new(user),
|
||||||
|
{:ok, inserted_backup} <- Repo.insert(backup),
|
||||||
|
{:ok, %Oban.Job{}} <- schedule_backup(inserted_backup) do
|
||||||
|
{:ok, inserted_backup}
|
||||||
|
else
|
||||||
|
false ->
|
||||||
|
{:error,
|
||||||
|
dngettext(
|
||||||
|
"errors",
|
||||||
|
"Last export was less than a day ago",
|
||||||
|
"Last export was less than %{days} days ago",
|
||||||
|
days,
|
||||||
|
days: days
|
||||||
|
)}
|
||||||
|
|
||||||
|
e ->
|
||||||
|
{:error, e}
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@doc "Generates a %Backup{} for a user with a random file name"
|
||||||
|
@spec new(User.t()) :: t()
|
||||||
def new(user) do
|
def new(user) do
|
||||||
rand_str = :crypto.strong_rand_bytes(32) |> Base.url_encode64(padding: false)
|
rand_str = :crypto.strong_rand_bytes(32) |> Base.url_encode64(padding: false)
|
||||||
datetime = Calendar.NaiveDateTime.Format.iso8601_basic(NaiveDateTime.utc_now())
|
datetime = Calendar.NaiveDateTime.Format.iso8601_basic(NaiveDateTime.utc_now())
|
||||||
name = "archive-#{user.nickname}-#{datetime}-#{rand_str}.zip"
|
name = "archive-#{user.nickname}-#{datetime}-#{rand_str}.zip"
|
||||||
|
|
||||||
%__MODULE__{
|
%__MODULE__{
|
||||||
user_id: user.id,
|
|
||||||
content_type: "application/zip",
|
content_type: "application/zip",
|
||||||
file_name: name,
|
file_name: name,
|
||||||
state: :pending
|
tempdir: tempdir(),
|
||||||
|
user: user
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
def delete(backup) do
|
@doc "Schedules the execution of the provided backup"
|
||||||
uploader = Pleroma.Config.get([Pleroma.Upload, :uploader])
|
@spec schedule_backup(t()) :: {:ok, Oban.Job.t()} | {:error, any()}
|
||||||
|
def schedule_backup(backup) do
|
||||||
|
with false <- is_nil(backup.id) do
|
||||||
|
%{"op" => "process", "backup_id" => backup.id}
|
||||||
|
|> BackupWorker.new()
|
||||||
|
|> Oban.insert()
|
||||||
|
else
|
||||||
|
true ->
|
||||||
|
{:error, "Backup is missing id. Please insert it into the Repo first."}
|
||||||
|
|
||||||
|
e ->
|
||||||
|
{:error, e}
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
@doc "Deletes the backup archive file and removes the database record"
|
||||||
|
@spec delete_archive(t()) :: {:ok, Ecto.Schema.t()} | {:error, Ecto.Changeset.t()}
|
||||||
|
def delete_archive(backup) do
|
||||||
|
uploader = Config.get([Pleroma.Upload, :uploader])
|
||||||
|
|
||||||
with :ok <- uploader.delete_file(Path.join("backups", backup.file_name)) do
|
with :ok <- uploader.delete_file(Path.join("backups", backup.file_name)) do
|
||||||
Repo.delete(backup)
|
Repo.delete(backup)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
defp validate_limit(_user, admin_id) when is_binary(admin_id), do: :ok
|
@doc "Schedules a job to delete the backup archive"
|
||||||
|
@spec schedule_delete(t()) :: {:ok, Oban.Job.t()} | {:error, any()}
|
||||||
|
def schedule_delete(backup) do
|
||||||
|
days = Config.get([__MODULE__, :purge_after_days])
|
||||||
|
time = 60 * 60 * 24 * days
|
||||||
|
scheduled_at = Calendar.NaiveDateTime.add!(backup.inserted_at, time)
|
||||||
|
|
||||||
defp validate_limit(user, nil) do
|
%{"op" => "delete", "backup_id" => backup.id}
|
||||||
case get_last(user.id) do
|
|> BackupWorker.new(scheduled_at: scheduled_at)
|
||||||
%__MODULE__{inserted_at: inserted_at} ->
|
|> Oban.insert()
|
||||||
days = Pleroma.Config.get([__MODULE__, :limit_days])
|
end
|
||||||
diff = Timex.diff(NaiveDateTime.utc_now(), inserted_at, :days)
|
|
||||||
|
|
||||||
if diff > days do
|
defp permitted?(user) do
|
||||||
:ok
|
with {_, %__MODULE__{inserted_at: inserted_at}} <- {:last, get_last(user)},
|
||||||
else
|
days = Config.get([__MODULE__, :limit_days]),
|
||||||
{:error,
|
diff = Timex.diff(NaiveDateTime.utc_now(), inserted_at, :days),
|
||||||
dngettext(
|
{_, true} <- {:diff, diff > days} do
|
||||||
"errors",
|
true
|
||||||
"Last export was less than a day ago",
|
else
|
||||||
"Last export was less than %{days} days ago",
|
{:last, nil} -> true
|
||||||
days,
|
{:diff, false} -> false
|
||||||
days: days
|
|
||||||
)}
|
|
||||||
end
|
|
||||||
|
|
||||||
nil ->
|
|
||||||
:ok
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def get_last(user_id) do
|
@doc "Returns last backup for the provided user"
|
||||||
|
@spec get_last(User.t()) :: t()
|
||||||
|
def get_last(%User{id: user_id}) do
|
||||||
__MODULE__
|
__MODULE__
|
||||||
|> where(user_id: ^user_id)
|
|> where(user_id: ^user_id)
|
||||||
|> order_by(desc: :id)
|
|> order_by(desc: :id)
|
||||||
|
@ -102,6 +143,8 @@ def get_last(user_id) do
|
||||||
|> Repo.one()
|
|> Repo.one()
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@doc "Lists all existing backups for a user"
|
||||||
|
@spec list(User.t()) :: [Ecto.Schema.t() | term()]
|
||||||
def list(%User{id: user_id}) do
|
def list(%User{id: user_id}) do
|
||||||
__MODULE__
|
__MODULE__
|
||||||
|> where(user_id: ^user_id)
|
|> where(user_id: ^user_id)
|
||||||
|
@ -109,94 +152,37 @@ def list(%User{id: user_id}) do
|
||||||
|> Repo.all()
|
|> Repo.all()
|
||||||
end
|
end
|
||||||
|
|
||||||
def remove_outdated(%__MODULE__{id: latest_id, user_id: user_id}) do
|
@doc "Schedules deletion of all but the the most recent backup"
|
||||||
__MODULE__
|
@spec remove_outdated(User.t()) :: :ok
|
||||||
|> where(user_id: ^user_id)
|
def remove_outdated(user) do
|
||||||
|> where([b], b.id != ^latest_id)
|
with %__MODULE__{} = latest_backup <- get_last(user) do
|
||||||
|> Repo.all()
|
__MODULE__
|
||||||
|> Enum.each(&BackupWorker.delete/1)
|
|> where(user_id: ^user.id)
|
||||||
end
|
|> where([b], b.id != ^latest_backup.id)
|
||||||
|
|> Repo.all()
|
||||||
def get(id), do: Repo.get(__MODULE__, id)
|
|> Enum.each(&schedule_delete/1)
|
||||||
|
else
|
||||||
defp set_state(backup, state, processed_number \\ nil) do
|
_ -> :ok
|
||||||
struct =
|
|
||||||
%{state: state}
|
|
||||||
|> Pleroma.Maps.put_if_present(:processed_number, processed_number)
|
|
||||||
|
|
||||||
backup
|
|
||||||
|> cast(struct, [:state, :processed_number])
|
|
||||||
|> Repo.update()
|
|
||||||
end
|
|
||||||
|
|
||||||
def process(
|
|
||||||
%__MODULE__{} = backup,
|
|
||||||
processor_module \\ __MODULE__.Processor
|
|
||||||
) do
|
|
||||||
set_state(backup, :running, 0)
|
|
||||||
|
|
||||||
current_pid = self()
|
|
||||||
|
|
||||||
task =
|
|
||||||
Task.Supervisor.async_nolink(
|
|
||||||
Pleroma.TaskSupervisor,
|
|
||||||
processor_module,
|
|
||||||
:do_process,
|
|
||||||
[backup, current_pid]
|
|
||||||
)
|
|
||||||
|
|
||||||
wait_backup(backup, backup.processed_number, task)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp wait_backup(backup, current_processed, task) do
|
|
||||||
wait_time = @config_impl.get([__MODULE__, :process_wait_time])
|
|
||||||
|
|
||||||
receive do
|
|
||||||
{:progress, new_processed} ->
|
|
||||||
total_processed = current_processed + new_processed
|
|
||||||
|
|
||||||
set_state(backup, :running, total_processed)
|
|
||||||
wait_backup(backup, total_processed, task)
|
|
||||||
|
|
||||||
{:DOWN, _ref, _proc, _pid, reason} ->
|
|
||||||
backup = get(backup.id)
|
|
||||||
|
|
||||||
if reason != :normal do
|
|
||||||
Logger.error("Backup #{backup.id} process ended abnormally: #{inspect(reason)}")
|
|
||||||
|
|
||||||
{:ok, backup} = set_state(backup, :failed)
|
|
||||||
|
|
||||||
cleanup(backup)
|
|
||||||
|
|
||||||
{:error,
|
|
||||||
%{
|
|
||||||
backup: backup,
|
|
||||||
reason: :exit,
|
|
||||||
details: reason
|
|
||||||
}}
|
|
||||||
else
|
|
||||||
{:ok, backup}
|
|
||||||
end
|
|
||||||
after
|
|
||||||
wait_time ->
|
|
||||||
Logger.error(
|
|
||||||
"Backup #{backup.id} timed out after no response for #{wait_time}ms, terminating"
|
|
||||||
)
|
|
||||||
|
|
||||||
Task.Supervisor.terminate_child(Pleroma.TaskSupervisor, task.pid)
|
|
||||||
|
|
||||||
{:ok, backup} = set_state(backup, :failed)
|
|
||||||
|
|
||||||
cleanup(backup)
|
|
||||||
|
|
||||||
{:error,
|
|
||||||
%{
|
|
||||||
backup: backup,
|
|
||||||
reason: :timeout
|
|
||||||
}}
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def get_by_id(id), do: Repo.get(__MODULE__, id)
|
||||||
|
|
||||||
|
@doc "Generates changeset for %Pleroma.User.Backup{}"
|
||||||
|
@spec changeset(%__MODULE__{}, map()) :: %Ecto.Changeset{}
|
||||||
|
def changeset(backup \\ %__MODULE__{}, attrs) do
|
||||||
|
backup
|
||||||
|
|> cast(attrs, [:content_type, :file_name, :file_size, :processed, :tempdir])
|
||||||
|
end
|
||||||
|
|
||||||
|
@doc "Updates the backup record"
|
||||||
|
@spec update_record(%__MODULE__{}, map()) :: {:ok, %__MODULE__{}} | {:error, %Ecto.Changeset{}}
|
||||||
|
def update_record(%__MODULE__{} = backup, attrs) do
|
||||||
|
backup
|
||||||
|
|> changeset(attrs)
|
||||||
|
|> Repo.update()
|
||||||
|
end
|
||||||
|
|
||||||
@files [
|
@files [
|
||||||
~c"actor.json",
|
~c"actor.json",
|
||||||
~c"outbox.json",
|
~c"outbox.json",
|
||||||
|
@ -207,55 +193,70 @@ defp wait_backup(backup, current_processed, task) do
|
||||||
~c"chats.json",
|
~c"chats.json",
|
||||||
~c"chat_messages.json"
|
~c"chat_messages.json"
|
||||||
]
|
]
|
||||||
@spec export(Pleroma.User.Backup.t(), pid()) :: {:ok, String.t()} | :error
|
|
||||||
def export(%__MODULE__{} = backup, caller_pid) do
|
|
||||||
backup = Repo.preload(backup, :user)
|
|
||||||
dir = backup_tempdir(backup)
|
|
||||||
|
|
||||||
with :ok <- File.mkdir(dir),
|
@spec run(t()) :: {:ok, t()} | {:error, :failed}
|
||||||
:ok <- actor(dir, backup.user, caller_pid),
|
def run(%__MODULE__{} = backup) do
|
||||||
:ok <- statuses(dir, backup.user, caller_pid),
|
backup = Repo.preload(backup, :user)
|
||||||
:ok <- likes(dir, backup.user, caller_pid),
|
tempfile = Path.join([backup.tempdir, backup.file_name])
|
||||||
:ok <- bookmarks(dir, backup.user, caller_pid),
|
|
||||||
:ok <- followers(dir, backup.user, caller_pid),
|
with {_, :ok} <- {:mkdir, File.mkdir_p(backup.tempdir)},
|
||||||
:ok <- following(dir, backup.user, caller_pid),
|
{_, :ok} <- {:actor, actor(backup.tempdir, backup.user)},
|
||||||
:ok <- chats(dir, backup.user, caller_pid),
|
{_, :ok} <- {:statuses, statuses(backup.tempdir, backup.user)},
|
||||||
:ok <- chat_messages(dir, backup.user, caller_pid),
|
{_, :ok} <- {:likes, likes(backup.tempdir, backup.user)},
|
||||||
{:ok, zip_path} <- :zip.create(backup.file_name, @files, cwd: dir),
|
{_, :ok} <- {:bookmarks, bookmarks(backup.tempdir, backup.user)},
|
||||||
{:ok, _} <- File.rm_rf(dir) do
|
{_, :ok} <- {:followers, followers(backup.tempdir, backup.user)},
|
||||||
{:ok, zip_path}
|
{_, :ok} <- {:following, following(backup.tempdir, backup.user)},
|
||||||
|
{_, :ok} <- {:chats, chats(backup.tempdir, backup.user)},
|
||||||
|
{_, :ok} <- {:chat_messages, chat_messages(backup.tempdir, backup.user)},
|
||||||
|
{_, {:ok, _zip_path}} <-
|
||||||
|
{:zip, :zip.create(to_charlist(tempfile), @files, cwd: to_charlist(backup.tempdir))},
|
||||||
|
{_, {:ok, %File.Stat{size: zip_size}}} <- {:filestat, File.stat(tempfile)},
|
||||||
|
{:ok, updated_backup} <- update_record(backup, %{file_size: zip_size}) do
|
||||||
|
{:ok, updated_backup}
|
||||||
else
|
else
|
||||||
_ -> :error
|
_ ->
|
||||||
|
File.rm_rf(backup.tempdir)
|
||||||
|
{:error, :failed}
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def dir(name) do
|
defp tempdir do
|
||||||
dir = Pleroma.Config.get([__MODULE__, :dir]) || System.tmp_dir!()
|
rand = :crypto.strong_rand_bytes(8) |> Base.url_encode64(padding: false)
|
||||||
Path.join(dir, name)
|
subdir = "backup-#{rand}"
|
||||||
|
|
||||||
|
case Config.get([__MODULE__, :tempdir]) do
|
||||||
|
nil ->
|
||||||
|
Path.join([System.tmp_dir!(), subdir])
|
||||||
|
|
||||||
|
path ->
|
||||||
|
Path.join([path, subdir])
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def upload(%__MODULE__{} = backup, zip_path) do
|
@doc "Uploads the completed backup and marks it as processed"
|
||||||
uploader = Pleroma.Config.get([Pleroma.Upload, :uploader])
|
@spec upload(t()) :: {:ok, t()}
|
||||||
|
def upload(%__MODULE__{tempdir: tempdir} = backup) when is_binary(tempdir) do
|
||||||
|
uploader = Config.get([Pleroma.Upload, :uploader])
|
||||||
|
|
||||||
upload = %Pleroma.Upload{
|
upload = %Pleroma.Upload{
|
||||||
name: backup.file_name,
|
name: backup.file_name,
|
||||||
tempfile: zip_path,
|
tempfile: Path.join([tempdir, backup.file_name]),
|
||||||
content_type: backup.content_type,
|
content_type: backup.content_type,
|
||||||
path: Path.join("backups", backup.file_name)
|
path: Path.join("backups", backup.file_name)
|
||||||
}
|
}
|
||||||
|
|
||||||
with {:ok, _} <- Pleroma.Uploaders.Uploader.put_file(uploader, upload),
|
with {:ok, _} <- Uploader.put_file(uploader, upload),
|
||||||
:ok <- File.rm(zip_path) do
|
{:ok, uploaded_backup} <- update_record(backup, %{processed: true}),
|
||||||
{:ok, upload}
|
{:ok, _} <- File.rm_rf(tempdir) do
|
||||||
|
{:ok, uploaded_backup}
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
defp actor(dir, user, caller_pid) do
|
defp actor(dir, user) do
|
||||||
with {:ok, json} <-
|
with {:ok, json} <-
|
||||||
UserView.render("user.json", %{user: user})
|
UserView.render("user.json", %{user: user})
|
||||||
|> Map.merge(%{"likes" => "likes.json", "bookmarks" => "bookmarks.json"})
|
|> Map.merge(%{"likes" => "likes.json", "bookmarks" => "bookmarks.json"})
|
||||||
|> Jason.encode() do
|
|> Jason.encode() do
|
||||||
send(caller_pid, {:progress, 1})
|
|
||||||
File.write(Path.join(dir, "actor.json"), json)
|
File.write(Path.join(dir, "actor.json"), json)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -274,22 +275,10 @@ defp write_header(file, name) do
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
defp should_report?(num, chunk_size), do: rem(num, chunk_size) == 0
|
defp write(query, dir, name, fun) do
|
||||||
|
|
||||||
defp backup_tempdir(backup) do
|
|
||||||
name = String.trim_trailing(backup.file_name, ".zip")
|
|
||||||
dir(name)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp cleanup(backup) do
|
|
||||||
dir = backup_tempdir(backup)
|
|
||||||
File.rm_rf(dir)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp write(query, dir, name, fun, caller_pid) do
|
|
||||||
path = Path.join(dir, "#{name}.json")
|
path = Path.join(dir, "#{name}.json")
|
||||||
|
|
||||||
chunk_size = Pleroma.Config.get([__MODULE__, :process_chunk_size])
|
chunk_size = Config.get([__MODULE__, :process_chunk_size])
|
||||||
|
|
||||||
with {:ok, file} <- File.open(path, [:write, :utf8]),
|
with {:ok, file} <- File.open(path, [:write, :utf8]),
|
||||||
:ok <- write_header(file, name) do
|
:ok <- write_header(file, name) do
|
||||||
|
@ -305,10 +294,6 @@ defp write(query, dir, name, fun, caller_pid) do
|
||||||
end),
|
end),
|
||||||
{:ok, str} <- Jason.encode(data),
|
{:ok, str} <- Jason.encode(data),
|
||||||
:ok <- IO.write(file, str <> ",\n") do
|
:ok <- IO.write(file, str <> ",\n") do
|
||||||
if should_report?(acc + 1, chunk_size) do
|
|
||||||
send(caller_pid, {:progress, chunk_size})
|
|
||||||
end
|
|
||||||
|
|
||||||
acc + 1
|
acc + 1
|
||||||
else
|
else
|
||||||
{:error, e} ->
|
{:error, e} ->
|
||||||
|
@ -323,31 +308,29 @@ defp write(query, dir, name, fun, caller_pid) do
|
||||||
end
|
end
|
||||||
end)
|
end)
|
||||||
|
|
||||||
send(caller_pid, {:progress, rem(total, chunk_size)})
|
|
||||||
|
|
||||||
with :ok <- :file.pwrite(file, {:eof, -2}, "\n],\n \"totalItems\": #{total}}") do
|
with :ok <- :file.pwrite(file, {:eof, -2}, "\n],\n \"totalItems\": #{total}}") do
|
||||||
File.close(file)
|
File.close(file)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
defp bookmarks(dir, %{id: user_id} = _user, caller_pid) do
|
defp bookmarks(dir, %{id: user_id} = _user) do
|
||||||
Bookmark
|
Bookmark
|
||||||
|> where(user_id: ^user_id)
|
|> where(user_id: ^user_id)
|
||||||
|> join(:inner, [b], activity in assoc(b, :activity))
|
|> join(:inner, [b], activity in assoc(b, :activity))
|
||||||
|> select([b, a], %{id: b.id, object: fragment("(?)->>'object'", a.data)})
|
|> select([b, a], %{id: b.id, object: fragment("(?)->>'object'", a.data)})
|
||||||
|> write(dir, "bookmarks", fn a -> {:ok, a.object} end, caller_pid)
|
|> write(dir, "bookmarks", fn a -> {:ok, a.object} end)
|
||||||
end
|
end
|
||||||
|
|
||||||
defp likes(dir, user, caller_pid) do
|
defp likes(dir, user) do
|
||||||
user.ap_id
|
user.ap_id
|
||||||
|> Activity.Queries.by_actor()
|
|> Activity.Queries.by_actor()
|
||||||
|> Activity.Queries.by_type("Like")
|
|> Activity.Queries.by_type("Like")
|
||||||
|> select([like], %{id: like.id, object: fragment("(?)->>'object'", like.data)})
|
|> select([like], %{id: like.id, object: fragment("(?)->>'object'", like.data)})
|
||||||
|> write(dir, "likes", fn a -> {:ok, a.object} end, caller_pid)
|
|> write(dir, "likes", fn a -> {:ok, a.object} end)
|
||||||
end
|
end
|
||||||
|
|
||||||
defp statuses(dir, user, caller_pid) do
|
defp statuses(dir, user) do
|
||||||
opts =
|
opts =
|
||||||
%{}
|
%{}
|
||||||
|> Map.put(:type, ["Create", "Announce"])
|
|> Map.put(:type, ["Create", "Announce"])
|
||||||
|
@ -367,22 +350,21 @@ defp statuses(dir, user, caller_pid) do
|
||||||
with {:ok, activity} <- Transmogrifier.prepare_outgoing(a.data) do
|
with {:ok, activity} <- Transmogrifier.prepare_outgoing(a.data) do
|
||||||
{:ok, Map.delete(activity, "@context")}
|
{:ok, Map.delete(activity, "@context")}
|
||||||
end
|
end
|
||||||
end,
|
end
|
||||||
caller_pid
|
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
defp followers(dir, user, caller_pid) do
|
defp followers(dir, user) do
|
||||||
User.get_followers_query(user)
|
User.get_followers_query(user)
|
||||||
|> write(dir, "followers", fn a -> {:ok, a.ap_id} end, caller_pid)
|
|> write(dir, "followers", fn a -> {:ok, a.ap_id} end)
|
||||||
end
|
end
|
||||||
|
|
||||||
defp following(dir, user, caller_pid) do
|
defp following(dir, user) do
|
||||||
User.get_friends_query(user)
|
User.get_friends_query(user)
|
||||||
|> write(dir, "following", fn a -> {:ok, a.ap_id} end, caller_pid)
|
|> write(dir, "following", fn a -> {:ok, a.ap_id} end)
|
||||||
end
|
end
|
||||||
|
|
||||||
defp chats(dir, user, caller_pid) do
|
defp chats(dir, user) do
|
||||||
Chat.for_user_query(user.id)
|
Chat.for_user_query(user.id)
|
||||||
|> write(
|
|> write(
|
||||||
dir,
|
dir,
|
||||||
|
@ -397,12 +379,11 @@ defp chats(dir, user, caller_pid) do
|
||||||
"published" =>
|
"published" =>
|
||||||
chat.inserted_at |> DateTime.from_naive!("Etc/UTC") |> DateTime.to_iso8601()
|
chat.inserted_at |> DateTime.from_naive!("Etc/UTC") |> DateTime.to_iso8601()
|
||||||
}}
|
}}
|
||||||
end,
|
end
|
||||||
caller_pid
|
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
def chat_messages(dir, %{id: user_id}, caller_pid) do
|
def chat_messages(dir, %{id: user_id}) do
|
||||||
chats_subquery =
|
chats_subquery =
|
||||||
from(c in Chat,
|
from(c in Chat,
|
||||||
where: c.user_id == ^user_id,
|
where: c.user_id == ^user_id,
|
||||||
|
@ -427,42 +408,7 @@ def chat_messages(dir, %{id: user_id}, caller_pid) do
|
||||||
)} do
|
)} do
|
||||||
{:ok, Map.delete(activity, "@context")}
|
{:ok, Map.delete(activity, "@context")}
|
||||||
end
|
end
|
||||||
end,
|
end
|
||||||
caller_pid
|
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
defmodule Pleroma.User.Backup.ProcessorAPI do
|
|
||||||
@callback do_process(%Pleroma.User.Backup{}, pid()) ::
|
|
||||||
{:ok, %Pleroma.User.Backup{}} | {:error, any()}
|
|
||||||
end
|
|
||||||
|
|
||||||
defmodule Pleroma.User.Backup.Processor do
|
|
||||||
@behaviour Pleroma.User.Backup.ProcessorAPI
|
|
||||||
|
|
||||||
alias Pleroma.Repo
|
|
||||||
alias Pleroma.User.Backup
|
|
||||||
|
|
||||||
import Ecto.Changeset
|
|
||||||
|
|
||||||
@impl true
|
|
||||||
def do_process(backup, current_pid) do
|
|
||||||
with {:ok, zip_file} <- Backup.export(backup, current_pid),
|
|
||||||
{:ok, %{size: size}} <- File.stat(zip_file),
|
|
||||||
{:ok, _upload} <- Backup.upload(backup, zip_file) do
|
|
||||||
backup
|
|
||||||
|> cast(
|
|
||||||
%{
|
|
||||||
file_size: size,
|
|
||||||
processed: true,
|
|
||||||
state: :complete
|
|
||||||
},
|
|
||||||
[:file_size, :processed, :state]
|
|
||||||
)
|
|
||||||
|> Repo.update()
|
|
||||||
else
|
|
||||||
e -> {:error, e}
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
|
@ -303,8 +303,15 @@ def inbox(%{assigns: %{valid_signature: true}} = conn, params) do
|
||||||
json(conn, "ok")
|
json(conn, "ok")
|
||||||
end
|
end
|
||||||
|
|
||||||
def inbox(%{assigns: %{valid_signature: false}, req_headers: req_headers} = conn, params) do
|
def inbox(%{assigns: %{valid_signature: false}} = conn, params) do
|
||||||
Federator.incoming_ap_doc(%{req_headers: req_headers, params: params})
|
Federator.incoming_ap_doc(%{
|
||||||
|
method: conn.method,
|
||||||
|
req_headers: conn.req_headers,
|
||||||
|
request_path: conn.request_path,
|
||||||
|
params: params,
|
||||||
|
query_string: conn.query_string
|
||||||
|
})
|
||||||
|
|
||||||
json(conn, "ok")
|
json(conn, "ok")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -80,13 +80,26 @@ def representable?(%Activity{} = activity) do
|
||||||
parameters set:
|
parameters set:
|
||||||
|
|
||||||
* `inbox`: the inbox to publish to
|
* `inbox`: the inbox to publish to
|
||||||
* `json`: the JSON message body representing the ActivityPub message
|
* `activity_id`: the internal activity id
|
||||||
* `actor`: the actor which is signing the message
|
* `cc`: the cc recipients relevant to this inbox (optional)
|
||||||
* `id`: the ActivityStreams URI of the message
|
|
||||||
"""
|
"""
|
||||||
def publish_one(%{inbox: inbox, json: json, actor: %User{} = actor, id: id} = params) do
|
def publish_one(%{inbox: inbox, activity_id: activity_id} = params) do
|
||||||
Logger.debug("Federating #{id} to #{inbox}")
|
activity = Activity.get_by_id_with_user_actor(activity_id)
|
||||||
|
actor = activity.user_actor
|
||||||
|
|
||||||
|
ap_id = activity.data["id"]
|
||||||
|
Logger.debug("Federating #{ap_id} to #{inbox}")
|
||||||
uri = %{path: path} = URI.parse(inbox)
|
uri = %{path: path} = URI.parse(inbox)
|
||||||
|
|
||||||
|
{:ok, data} = Transmogrifier.prepare_outgoing(activity.data)
|
||||||
|
|
||||||
|
cc = Map.get(params, :cc)
|
||||||
|
|
||||||
|
json =
|
||||||
|
data
|
||||||
|
|> Map.put("cc", cc)
|
||||||
|
|> Jason.encode!()
|
||||||
|
|
||||||
digest = "SHA-256=" <> (:crypto.hash(:sha256, json) |> Base.encode64())
|
digest = "SHA-256=" <> (:crypto.hash(:sha256, json) |> Base.encode64())
|
||||||
|
|
||||||
date = Pleroma.Signature.signed_date()
|
date = Pleroma.Signature.signed_date()
|
||||||
|
@ -119,7 +132,7 @@ def publish_one(%{inbox: inbox, json: json, actor: %User{} = actor, id: id} = pa
|
||||||
else
|
else
|
||||||
{_post_result, %{status: code} = response} = e ->
|
{_post_result, %{status: code} = response} = e ->
|
||||||
unless params[:unreachable_since], do: Instances.set_unreachable(inbox)
|
unless params[:unreachable_since], do: Instances.set_unreachable(inbox)
|
||||||
Logger.metadata(activity: id, inbox: inbox, status: code)
|
Logger.metadata(activity: activity_id, inbox: inbox, status: code)
|
||||||
Logger.error("Publisher failed to inbox #{inbox} with status #{code}")
|
Logger.error("Publisher failed to inbox #{inbox} with status #{code}")
|
||||||
|
|
||||||
case response do
|
case response do
|
||||||
|
@ -136,21 +149,12 @@ def publish_one(%{inbox: inbox, json: json, actor: %User{} = actor, id: id} = pa
|
||||||
|
|
||||||
e ->
|
e ->
|
||||||
unless params[:unreachable_since], do: Instances.set_unreachable(inbox)
|
unless params[:unreachable_since], do: Instances.set_unreachable(inbox)
|
||||||
Logger.metadata(activity: id, inbox: inbox)
|
Logger.metadata(activity: activity_id, inbox: inbox)
|
||||||
Logger.error("Publisher failed to inbox #{inbox} #{inspect(e)}")
|
Logger.error("Publisher failed to inbox #{inbox} #{inspect(e)}")
|
||||||
{:error, e}
|
{:error, e}
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def publish_one(%{actor_id: actor_id} = params) do
|
|
||||||
actor = User.get_cached_by_id(actor_id)
|
|
||||||
|
|
||||||
params
|
|
||||||
|> Map.delete(:actor_id)
|
|
||||||
|> Map.put(:actor, actor)
|
|
||||||
|> publish_one()
|
|
||||||
end
|
|
||||||
|
|
||||||
defp signature_host(%URI{port: port, scheme: scheme, host: host}) do
|
defp signature_host(%URI{port: port, scheme: scheme, host: host}) do
|
||||||
if port == URI.default_port(scheme) do
|
if port == URI.default_port(scheme) do
|
||||||
host
|
host
|
||||||
|
@ -251,7 +255,6 @@ def determine_inbox(
|
||||||
def publish(%User{} = actor, %{data: %{"bcc" => bcc}} = activity)
|
def publish(%User{} = actor, %{data: %{"bcc" => bcc}} = activity)
|
||||||
when is_list(bcc) and bcc != [] do
|
when is_list(bcc) and bcc != [] do
|
||||||
public = public?(activity)
|
public = public?(activity)
|
||||||
{:ok, data} = Transmogrifier.prepare_outgoing(activity.data)
|
|
||||||
|
|
||||||
[priority_recipients, recipients] = recipients(actor, activity)
|
[priority_recipients, recipients] = recipients(actor, activity)
|
||||||
|
|
||||||
|
@ -276,16 +279,10 @@ def publish(%User{} = actor, %{data: %{"bcc" => bcc}} = activity)
|
||||||
# instance would only accept a first message for the first recipient and ignore the rest.
|
# instance would only accept a first message for the first recipient and ignore the rest.
|
||||||
cc = get_cc_ap_ids(ap_id, recipients)
|
cc = get_cc_ap_ids(ap_id, recipients)
|
||||||
|
|
||||||
json =
|
|
||||||
data
|
|
||||||
|> Map.put("cc", cc)
|
|
||||||
|> Jason.encode!()
|
|
||||||
|
|
||||||
__MODULE__.enqueue_one(%{
|
__MODULE__.enqueue_one(%{
|
||||||
inbox: inbox,
|
inbox: inbox,
|
||||||
json: json,
|
cc: cc,
|
||||||
actor_id: actor.id,
|
activity_id: activity.id,
|
||||||
id: activity.data["id"],
|
|
||||||
unreachable_since: unreachable_since
|
unreachable_since: unreachable_since
|
||||||
})
|
})
|
||||||
end)
|
end)
|
||||||
|
@ -302,9 +299,6 @@ def publish(%User{} = actor, %Activity{} = activity) do
|
||||||
Relay.publish(activity)
|
Relay.publish(activity)
|
||||||
end
|
end
|
||||||
|
|
||||||
{:ok, data} = Transmogrifier.prepare_outgoing(activity.data)
|
|
||||||
json = Jason.encode!(data)
|
|
||||||
|
|
||||||
[priority_inboxes, inboxes] =
|
[priority_inboxes, inboxes] =
|
||||||
recipients(actor, activity)
|
recipients(actor, activity)
|
||||||
|> Enum.map(fn recipients ->
|
|> Enum.map(fn recipients ->
|
||||||
|
@ -326,9 +320,7 @@ def publish(%User{} = actor, %Activity{} = activity) do
|
||||||
__MODULE__.enqueue_one(
|
__MODULE__.enqueue_one(
|
||||||
%{
|
%{
|
||||||
inbox: inbox,
|
inbox: inbox,
|
||||||
json: json,
|
activity_id: activity.id,
|
||||||
actor_id: actor.id,
|
|
||||||
id: activity.data["id"],
|
|
||||||
unreachable_since: unreachable_since
|
unreachable_since: unreachable_since
|
||||||
},
|
},
|
||||||
priority: priority
|
priority: priority
|
||||||
|
|
|
@ -13,6 +13,7 @@ defmodule Pleroma.Web.AdminAPI.AdminAPIController do
|
||||||
alias Pleroma.ModerationLog
|
alias Pleroma.ModerationLog
|
||||||
alias Pleroma.Stats
|
alias Pleroma.Stats
|
||||||
alias Pleroma.User
|
alias Pleroma.User
|
||||||
|
alias Pleroma.User.Backup
|
||||||
alias Pleroma.Web.ActivityPub.ActivityPub
|
alias Pleroma.Web.ActivityPub.ActivityPub
|
||||||
alias Pleroma.Web.AdminAPI
|
alias Pleroma.Web.AdminAPI
|
||||||
alias Pleroma.Web.AdminAPI.AccountView
|
alias Pleroma.Web.AdminAPI.AccountView
|
||||||
|
@ -429,7 +430,9 @@ def stats(conn, params) do
|
||||||
|
|
||||||
def create_backup(%{assigns: %{user: admin}} = conn, %{"nickname" => nickname}) do
|
def create_backup(%{assigns: %{user: admin}} = conn, %{"nickname" => nickname}) do
|
||||||
with %User{} = user <- User.get_by_nickname(nickname),
|
with %User{} = user <- User.get_by_nickname(nickname),
|
||||||
{:ok, _} <- Pleroma.User.Backup.create(user, admin.id) do
|
%Backup{} = backup <- Backup.new(user),
|
||||||
|
{:ok, inserted_backup} <- Pleroma.Repo.insert(backup),
|
||||||
|
{:ok, %Oban.Job{}} <- Backup.schedule_backup(inserted_backup) do
|
||||||
ModerationLog.insert_log(%{actor: admin, subject: user, action: "create_backup"})
|
ModerationLog.insert_log(%{actor: admin, subject: user, action: "create_backup"})
|
||||||
|
|
||||||
json(conn, "")
|
json(conn, "")
|
||||||
|
|
|
@ -65,12 +65,7 @@ defp backup do
|
||||||
file_name: %Schema{type: :string},
|
file_name: %Schema{type: :string},
|
||||||
file_size: %Schema{type: :integer},
|
file_size: %Schema{type: :integer},
|
||||||
processed: %Schema{type: :boolean, description: "whether this backup has succeeded"},
|
processed: %Schema{type: :boolean, description: "whether this backup has succeeded"},
|
||||||
state: %Schema{
|
tempdir: %Schema{type: :string}
|
||||||
type: :string,
|
|
||||||
description: "the state of the backup",
|
|
||||||
enum: ["pending", "running", "complete", "failed"]
|
|
||||||
},
|
|
||||||
processed_number: %Schema{type: :integer, description: "the number of records processed"}
|
|
||||||
},
|
},
|
||||||
example: %{
|
example: %{
|
||||||
"content_type" => "application/zip",
|
"content_type" => "application/zip",
|
||||||
|
@ -79,8 +74,7 @@ defp backup do
|
||||||
"file_size" => 4105,
|
"file_size" => 4105,
|
||||||
"inserted_at" => "2020-09-08T16:42:07.000Z",
|
"inserted_at" => "2020-09-08T16:42:07.000Z",
|
||||||
"processed" => true,
|
"processed" => true,
|
||||||
"state" => "complete",
|
"tempdir" => "/tmp/PZIMw40vmpM"
|
||||||
"processed_number" => 20
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
|
@ -850,11 +850,11 @@ defp make_update_event_data(user, orig_object, changes, location) do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
defp maybe_cancel_jobs(%Activity{data: %{"id" => ap_id}}) do
|
defp maybe_cancel_jobs(%Activity{id: activity_id}) do
|
||||||
Oban.Job
|
Oban.Job
|
||||||
|> where([j], j.worker == "Pleroma.Workers.PublisherWorker")
|
|> where([j], j.worker == "Pleroma.Workers.PublisherWorker")
|
||||||
|> where([j], j.args["op"] == "publish_one")
|
|> where([j], j.args["op"] == "publish_one")
|
||||||
|> where([j], j.args["params"]["id"] == ^ap_id)
|
|> where([j], j.args["params"]["activity_id"] == ^activity_id)
|
||||||
|> Oban.cancel_all_jobs()
|
|> Oban.cancel_all_jobs()
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -35,10 +35,12 @@ def allowed_thread_distance?(distance) do
|
||||||
end
|
end
|
||||||
|
|
||||||
# Client API
|
# Client API
|
||||||
def incoming_ap_doc(%{params: params, req_headers: req_headers}) do
|
def incoming_ap_doc(%{params: _params, req_headers: _req_headers} = args) do
|
||||||
|
job_args = Enum.into(args, %{}, fn {k, v} -> {Atom.to_string(k), v} end)
|
||||||
|
|
||||||
ReceiverWorker.enqueue(
|
ReceiverWorker.enqueue(
|
||||||
"incoming_ap_doc",
|
"incoming_ap_doc",
|
||||||
%{"req_headers" => req_headers, "params" => params, "timeout" => :timer.seconds(20)},
|
Map.put(job_args, "timeout", :timer.seconds(20)),
|
||||||
priority: 2
|
priority: 2
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
|
@ -20,7 +20,7 @@ def index(%{assigns: %{user: user}} = conn, _params) do
|
||||||
end
|
end
|
||||||
|
|
||||||
def create(%{assigns: %{user: user}} = conn, _params) do
|
def create(%{assigns: %{user: user}} = conn, _params) do
|
||||||
with {:ok, _} <- Backup.create(user) do
|
with {:ok, _} <- Backup.user(user) do
|
||||||
backups = Backup.list(user)
|
backups = Backup.list(user)
|
||||||
render(conn, "index.json", backups: backups)
|
render(conn, "index.json", backups: backups)
|
||||||
end
|
end
|
||||||
|
|
|
@ -9,22 +9,12 @@ defmodule Pleroma.Web.PleromaAPI.BackupView do
|
||||||
alias Pleroma.Web.CommonAPI.Utils
|
alias Pleroma.Web.CommonAPI.Utils
|
||||||
|
|
||||||
def render("show.json", %{backup: %Backup{} = backup}) do
|
def render("show.json", %{backup: %Backup{} = backup}) do
|
||||||
# To deal with records before the migration
|
|
||||||
state =
|
|
||||||
if backup.state == :invalid do
|
|
||||||
if backup.processed, do: :complete, else: :failed
|
|
||||||
else
|
|
||||||
backup.state
|
|
||||||
end
|
|
||||||
|
|
||||||
%{
|
%{
|
||||||
id: backup.id,
|
id: backup.id,
|
||||||
content_type: backup.content_type,
|
content_type: backup.content_type,
|
||||||
url: download_url(backup),
|
url: download_url(backup),
|
||||||
file_size: backup.file_size,
|
file_size: backup.file_size,
|
||||||
processed: backup.processed,
|
processed: backup.processed,
|
||||||
state: to_string(state),
|
|
||||||
processed_number: backup.processed_number,
|
|
||||||
inserted_at: Utils.to_masto_date(backup.inserted_at)
|
inserted_at: Utils.to_masto_date(backup.inserted_at)
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
|
@ -8,16 +8,12 @@ defmodule Pleroma.Web.Plugs.HTTPSignaturePlug do
|
||||||
import Plug.Conn
|
import Plug.Conn
|
||||||
import Phoenix.Controller, only: [get_format: 1, text: 2]
|
import Phoenix.Controller, only: [get_format: 1, text: 2]
|
||||||
|
|
||||||
|
alias Pleroma.Signature
|
||||||
alias Pleroma.Web.ActivityPub.MRF
|
alias Pleroma.Web.ActivityPub.MRF
|
||||||
|
|
||||||
require Logger
|
require Logger
|
||||||
|
|
||||||
@config_impl Application.compile_env(:pleroma, [__MODULE__, :config_impl], Pleroma.Config)
|
@config_impl Application.compile_env(:pleroma, [__MODULE__, :config_impl], Pleroma.Config)
|
||||||
@http_signatures_impl Application.compile_env(
|
|
||||||
:pleroma,
|
|
||||||
[__MODULE__, :http_signatures_impl],
|
|
||||||
HTTPSignatures
|
|
||||||
)
|
|
||||||
|
|
||||||
def init(options) do
|
def init(options) do
|
||||||
options
|
options
|
||||||
|
@ -39,48 +35,6 @@ def call(conn, _opts) do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
defp validate_signature(conn, request_target) do
|
|
||||||
# Newer drafts for HTTP signatures now use @request-target instead of the
|
|
||||||
# old (request-target). We'll now support both for incoming signatures.
|
|
||||||
conn =
|
|
||||||
conn
|
|
||||||
|> put_req_header("(request-target)", request_target)
|
|
||||||
|> put_req_header("@request-target", request_target)
|
|
||||||
|
|
||||||
@http_signatures_impl.validate_conn(conn)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp validate_signature(conn) do
|
|
||||||
# This (request-target) is non-standard, but many implementations do it
|
|
||||||
# this way due to a misinterpretation of
|
|
||||||
# https://datatracker.ietf.org/doc/html/draft-cavage-http-signatures-06
|
|
||||||
# "path" was interpreted as not having the query, though later examples
|
|
||||||
# show that it must be the absolute path + query. This behavior is kept to
|
|
||||||
# make sure most software (Pleroma itself, Mastodon, and probably others)
|
|
||||||
# do not break.
|
|
||||||
request_target = String.downcase("#{conn.method}") <> " #{conn.request_path}"
|
|
||||||
|
|
||||||
# This is the proper way to build the @request-target, as expected by
|
|
||||||
# many HTTP signature libraries, clarified in the following draft:
|
|
||||||
# https://www.ietf.org/archive/id/draft-ietf-httpbis-message-signatures-11.html#section-2.2.6
|
|
||||||
# It is the same as before, but containing the query part as well.
|
|
||||||
proper_target = request_target <> "?#{conn.query_string}"
|
|
||||||
|
|
||||||
cond do
|
|
||||||
# Normal, non-standard behavior but expected by Pleroma and more.
|
|
||||||
validate_signature(conn, request_target) ->
|
|
||||||
true
|
|
||||||
|
|
||||||
# Has query string and the previous one failed: let's try the standard.
|
|
||||||
conn.query_string != "" ->
|
|
||||||
validate_signature(conn, proper_target)
|
|
||||||
|
|
||||||
# If there's no query string and signature fails, it's rotten.
|
|
||||||
true ->
|
|
||||||
false
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp maybe_assign_valid_signature(conn) do
|
defp maybe_assign_valid_signature(conn) do
|
||||||
if has_signature_header?(conn) do
|
if has_signature_header?(conn) do
|
||||||
# we replace the digest header with the one we computed in DigestPlug
|
# we replace the digest header with the one we computed in DigestPlug
|
||||||
|
@ -90,7 +44,7 @@ defp maybe_assign_valid_signature(conn) do
|
||||||
conn -> conn
|
conn -> conn
|
||||||
end
|
end
|
||||||
|
|
||||||
assign(conn, :valid_signature, validate_signature(conn))
|
assign(conn, :valid_signature, Signature.validate_signature(conn))
|
||||||
else
|
else
|
||||||
Logger.debug("No signature header!")
|
Logger.debug("No signature header!")
|
||||||
conn
|
conn
|
||||||
|
|
|
@ -52,7 +52,7 @@ defp fetch_user_and_token(token) do
|
||||||
where: t.token == ^token
|
where: t.token == ^token
|
||||||
)
|
)
|
||||||
|
|
||||||
with %Token{user_id: user_id} = token_record <- Repo.one(token_query),
|
with %Token{user_id: user_id} = token_record <- Repo.one(token_query) |> Repo.preload(:user),
|
||||||
false <- is_nil(user_id),
|
false <- is_nil(user_id),
|
||||||
%User{} = user <- User.get_cached_by_id(user_id) do
|
%User{} = user <- User.get_cached_by_id(user_id) do
|
||||||
{:ok, user, token_record}
|
{:ok, user, token_record}
|
||||||
|
|
|
@ -6,64 +6,46 @@ defmodule Pleroma.Workers.BackupWorker do
|
||||||
use Oban.Worker, queue: :slow, max_attempts: 1
|
use Oban.Worker, queue: :slow, max_attempts: 1
|
||||||
|
|
||||||
alias Oban.Job
|
alias Oban.Job
|
||||||
|
alias Pleroma.Config.Getting, as: Config
|
||||||
alias Pleroma.User.Backup
|
alias Pleroma.User.Backup
|
||||||
|
|
||||||
def process(backup, admin_user_id \\ nil) do
|
|
||||||
%{"op" => "process", "backup_id" => backup.id, "admin_user_id" => admin_user_id}
|
|
||||||
|> new()
|
|
||||||
|> Oban.insert()
|
|
||||||
end
|
|
||||||
|
|
||||||
def schedule_deletion(backup) do
|
|
||||||
days = Pleroma.Config.get([Backup, :purge_after_days])
|
|
||||||
time = 60 * 60 * 24 * days
|
|
||||||
scheduled_at = Calendar.NaiveDateTime.add!(backup.inserted_at, time)
|
|
||||||
|
|
||||||
%{"op" => "delete", "backup_id" => backup.id}
|
|
||||||
|> new(scheduled_at: scheduled_at)
|
|
||||||
|> Oban.insert()
|
|
||||||
end
|
|
||||||
|
|
||||||
def delete(backup) do
|
|
||||||
%{"op" => "delete", "backup_id" => backup.id}
|
|
||||||
|> new()
|
|
||||||
|> Oban.insert()
|
|
||||||
end
|
|
||||||
|
|
||||||
@impl Oban.Worker
|
@impl Oban.Worker
|
||||||
def perform(%Job{
|
def perform(%Job{
|
||||||
args: %{"op" => "process", "backup_id" => backup_id, "admin_user_id" => admin_user_id}
|
args: %{"op" => "process", "backup_id" => backup_id}
|
||||||
}) do
|
}) do
|
||||||
with {:ok, %Backup{} = backup} <-
|
with {_, %Backup{} = backup} <- {:get, Backup.get_by_id(backup_id)},
|
||||||
backup_id |> Backup.get() |> Backup.process(),
|
{_, {:ok, updated_backup}} <- {:run, Backup.run(backup)},
|
||||||
{:ok, _job} <- schedule_deletion(backup),
|
{_, {:ok, uploaded_backup}} <- {:upload, Backup.upload(updated_backup)},
|
||||||
:ok <- Backup.remove_outdated(backup),
|
{_, {:ok, _job}} <- {:delete, Backup.schedule_delete(uploaded_backup)},
|
||||||
:ok <- maybe_deliver_email(backup, admin_user_id) do
|
{_, :ok} <- {:outdated, Backup.remove_outdated(uploaded_backup.user)},
|
||||||
{:ok, backup}
|
{_, :ok} <- {:email, maybe_deliver_email(uploaded_backup)} do
|
||||||
|
{:ok, uploaded_backup}
|
||||||
|
else
|
||||||
|
e -> {:error, e}
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def perform(%Job{args: %{"op" => "delete", "backup_id" => backup_id}}) do
|
def perform(%Job{args: %{"op" => "delete", "backup_id" => backup_id}}) do
|
||||||
case Backup.get(backup_id) do
|
case Backup.get_by_id(backup_id) do
|
||||||
%Backup{} = backup -> Backup.delete(backup)
|
%Backup{} = backup -> Backup.delete_archive(backup)
|
||||||
nil -> :ok
|
nil -> :ok
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@impl Oban.Worker
|
@impl Oban.Worker
|
||||||
def timeout(_job), do: :infinity
|
def timeout(_job), do: Config.get([Backup, :timeout], :timer.minutes(30))
|
||||||
|
|
||||||
defp has_email?(user) do
|
defp has_email?(user) do
|
||||||
not is_nil(user.email) and user.email != ""
|
not is_nil(user.email) and user.email != ""
|
||||||
end
|
end
|
||||||
|
|
||||||
defp maybe_deliver_email(backup, admin_user_id) do
|
defp maybe_deliver_email(backup) do
|
||||||
has_mailer = Pleroma.Config.get([Pleroma.Emails.Mailer, :enabled])
|
has_mailer = Pleroma.Config.get([Pleroma.Emails.Mailer, :enabled])
|
||||||
backup = backup |> Pleroma.Repo.preload(:user)
|
backup = backup |> Pleroma.Repo.preload(:user)
|
||||||
|
|
||||||
if has_email?(backup.user) and has_mailer do
|
if has_email?(backup.user) and has_mailer do
|
||||||
backup
|
backup
|
||||||
|> Pleroma.Emails.UserEmail.backup_is_ready_email(admin_user_id)
|
|> Pleroma.Emails.UserEmail.backup_is_ready_email()
|
||||||
|> Pleroma.Emails.Mailer.deliver()
|
|> Pleroma.Emails.Mailer.deliver()
|
||||||
|
|
||||||
:ok
|
:ok
|
||||||
|
|
|
@ -12,17 +12,30 @@ defmodule Pleroma.Workers.ReceiverWorker do
|
||||||
@impl Oban.Worker
|
@impl Oban.Worker
|
||||||
|
|
||||||
def perform(%Job{
|
def perform(%Job{
|
||||||
args: %{"op" => "incoming_ap_doc", "req_headers" => req_headers, "params" => params}
|
args: %{
|
||||||
|
"op" => "incoming_ap_doc",
|
||||||
|
"method" => method,
|
||||||
|
"params" => params,
|
||||||
|
"req_headers" => req_headers,
|
||||||
|
"request_path" => request_path,
|
||||||
|
"query_string" => query_string
|
||||||
|
}
|
||||||
}) do
|
}) do
|
||||||
# Oban's serialization converts our tuple headers to lists.
|
# Oban's serialization converts our tuple headers to lists.
|
||||||
# Revert it for the signature validation.
|
# Revert it for the signature validation.
|
||||||
req_headers = Enum.into(req_headers, [], &List.to_tuple(&1))
|
req_headers = Enum.into(req_headers, [], &List.to_tuple(&1))
|
||||||
|
|
||||||
conn_data = %{params: params, req_headers: req_headers}
|
conn_data = %Plug.Conn{
|
||||||
|
method: method,
|
||||||
|
params: params,
|
||||||
|
req_headers: req_headers,
|
||||||
|
request_path: request_path,
|
||||||
|
query_string: query_string
|
||||||
|
}
|
||||||
|
|
||||||
with {:ok, %User{} = _actor} <- User.get_or_fetch_by_ap_id(conn_data.params["actor"]),
|
with {:ok, %User{} = _actor} <- User.get_or_fetch_by_ap_id(conn_data.params["actor"]),
|
||||||
{:ok, _public_key} <- Signature.refetch_public_key(conn_data),
|
{:ok, _public_key} <- Signature.refetch_public_key(conn_data),
|
||||||
{:signature, true} <- {:signature, HTTPSignatures.validate_conn(conn_data)},
|
{:signature, true} <- {:signature, Signature.validate_signature(conn_data)},
|
||||||
{:ok, res} <- Federator.perform(:incoming_ap_doc, params) do
|
{:ok, res} <- Federator.perform(:incoming_ap_doc, params) do
|
||||||
{:ok, res}
|
{:ok, res}
|
||||||
else
|
else
|
||||||
|
|
19
priv/repo/migrations/20240622175346_backup_refactor.exs
Normal file
19
priv/repo/migrations/20240622175346_backup_refactor.exs
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
defmodule Pleroma.Repo.Migrations.BackupRefactor do
|
||||||
|
use Ecto.Migration
|
||||||
|
|
||||||
|
def up do
|
||||||
|
alter table("backups") do
|
||||||
|
remove(:state)
|
||||||
|
remove(:processed_number)
|
||||||
|
add(:tempdir, :string)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def down do
|
||||||
|
alter table("backups") do
|
||||||
|
add(:state, :integer, default: 5)
|
||||||
|
add(:processed_number, :integer, default: 0)
|
||||||
|
remove(:tempdir)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
32
priv/repo/migrations/20240729163838_publisher_job_change.exs
Normal file
32
priv/repo/migrations/20240729163838_publisher_job_change.exs
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
defmodule Pleroma.Repo.Migrations.PublisherJobChange do
|
||||||
|
use Ecto.Migration
|
||||||
|
|
||||||
|
alias Pleroma.Activity
|
||||||
|
alias Pleroma.Repo
|
||||||
|
import Ecto.Query
|
||||||
|
|
||||||
|
def up do
|
||||||
|
query =
|
||||||
|
from(j in Oban.Job,
|
||||||
|
where: j.worker == "Pleroma.Workers.PublisherWorker",
|
||||||
|
where: j.state in ["available", "retryable"]
|
||||||
|
)
|
||||||
|
|
||||||
|
jobs = Repo.all(query)
|
||||||
|
|
||||||
|
Enum.each(jobs, fn job ->
|
||||||
|
args = job.args
|
||||||
|
|
||||||
|
case Activity.get_by_ap_id(args["id"]) do
|
||||||
|
nil ->
|
||||||
|
:ok
|
||||||
|
|
||||||
|
%Activity{id: activity_id} ->
|
||||||
|
updated_args = Map.put(args, "activity_id", activity_id)
|
||||||
|
|
||||||
|
Pleroma.Workers.PublisherWorker.new(updated_args)
|
||||||
|
|> Oban.insert()
|
||||||
|
end
|
||||||
|
end)
|
||||||
|
end
|
||||||
|
end
|
117
test/fixtures/bastianallgeier.json
vendored
Normal file
117
test/fixtures/bastianallgeier.json
vendored
Normal file
|
@ -0,0 +1,117 @@
|
||||||
|
{
|
||||||
|
"@context": [
|
||||||
|
"https://www.w3.org/ns/activitystreams",
|
||||||
|
"https://w3id.org/security/v1",
|
||||||
|
{
|
||||||
|
"Curve25519Key": "toot:Curve25519Key",
|
||||||
|
"Device": "toot:Device",
|
||||||
|
"Ed25519Key": "toot:Ed25519Key",
|
||||||
|
"Ed25519Signature": "toot:Ed25519Signature",
|
||||||
|
"EncryptedMessage": "toot:EncryptedMessage",
|
||||||
|
"PropertyValue": "schema:PropertyValue",
|
||||||
|
"alsoKnownAs": {
|
||||||
|
"@id": "as:alsoKnownAs",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"cipherText": "toot:cipherText",
|
||||||
|
"claim": {
|
||||||
|
"@id": "toot:claim",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"deviceId": "toot:deviceId",
|
||||||
|
"devices": {
|
||||||
|
"@id": "toot:devices",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"discoverable": "toot:discoverable",
|
||||||
|
"featured": {
|
||||||
|
"@id": "toot:featured",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"featuredTags": {
|
||||||
|
"@id": "toot:featuredTags",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"fingerprintKey": {
|
||||||
|
"@id": "toot:fingerprintKey",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"focalPoint": {
|
||||||
|
"@container": "@list",
|
||||||
|
"@id": "toot:focalPoint"
|
||||||
|
},
|
||||||
|
"identityKey": {
|
||||||
|
"@id": "toot:identityKey",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"indexable": "toot:indexable",
|
||||||
|
"manuallyApprovesFollowers": "as:manuallyApprovesFollowers",
|
||||||
|
"memorial": "toot:memorial",
|
||||||
|
"messageFranking": "toot:messageFranking",
|
||||||
|
"messageType": "toot:messageType",
|
||||||
|
"movedTo": {
|
||||||
|
"@id": "as:movedTo",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"publicKeyBase64": "toot:publicKeyBase64",
|
||||||
|
"schema": "http://schema.org#",
|
||||||
|
"suspended": "toot:suspended",
|
||||||
|
"toot": "http://joinmastodon.org/ns#",
|
||||||
|
"value": "schema:value"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"attachment": [
|
||||||
|
{
|
||||||
|
"name": "Website",
|
||||||
|
"type": "PropertyValue",
|
||||||
|
"value": "<a href=\"https://bastianallgeier.com\" target=\"_blank\" rel=\"nofollow noopener noreferrer me\" translate=\"no\"><span class=\"invisible\">https://</span><span class=\"\">bastianallgeier.com</span><span class=\"invisible\"></span></a>"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Project",
|
||||||
|
"type": "PropertyValue",
|
||||||
|
"value": "<a href=\"https://getkirby.com\" target=\"_blank\" rel=\"nofollow noopener noreferrer me\" translate=\"no\"><span class=\"invisible\">https://</span><span class=\"\">getkirby.com</span><span class=\"invisible\"></span></a>"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Github",
|
||||||
|
"type": "PropertyValue",
|
||||||
|
"value": "<a href=\"https://github.com/bastianallgeier\" target=\"_blank\" rel=\"nofollow noopener noreferrer me\" translate=\"no\"><span class=\"invisible\">https://</span><span class=\"\">github.com/bastianallgeier</span><span class=\"invisible\"></span></a>"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"devices": "https://mastodon.social/users/bastianallgeier/collections/devices",
|
||||||
|
"discoverable": true,
|
||||||
|
"endpoints": {
|
||||||
|
"sharedInbox": "https://mastodon.social/inbox"
|
||||||
|
},
|
||||||
|
"featured": "https://mastodon.social/users/bastianallgeier/collections/featured",
|
||||||
|
"featuredTags": "https://mastodon.social/users/bastianallgeier/collections/tags",
|
||||||
|
"followers": "https://mastodon.social/users/bastianallgeier/followers",
|
||||||
|
"following": "https://mastodon.social/users/bastianallgeier/following",
|
||||||
|
"icon": {
|
||||||
|
"mediaType": "image/jpeg",
|
||||||
|
"type": "Image",
|
||||||
|
"url": "https://files.mastodon.social/accounts/avatars/000/007/393/original/0180a20079617c71.jpg"
|
||||||
|
},
|
||||||
|
"id": "https://mastodon.social/users/bastianallgeier",
|
||||||
|
"image": {
|
||||||
|
"mediaType": "image/jpeg",
|
||||||
|
"type": "Image",
|
||||||
|
"url": "https://files.mastodon.social/accounts/headers/000/007/393/original/13d644ab46d50478.jpeg"
|
||||||
|
},
|
||||||
|
"inbox": "https://mastodon.social/users/bastianallgeier/inbox",
|
||||||
|
"indexable": false,
|
||||||
|
"manuallyApprovesFollowers": false,
|
||||||
|
"memorial": false,
|
||||||
|
"name": "Bastian Allgeier",
|
||||||
|
"outbox": "https://mastodon.social/users/bastianallgeier/outbox",
|
||||||
|
"preferredUsername": "bastianallgeier",
|
||||||
|
"publicKey": {
|
||||||
|
"id": "https://mastodon.social/users/bastianallgeier#main-key",
|
||||||
|
"owner": "https://mastodon.social/users/bastianallgeier",
|
||||||
|
"publicKeyPem": "-----BEGIN PUBLIC KEY-----\nMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA3fz+hpgVztO9z6HUhyzv\nwP++ERBBoIwSLKf1TyIM8bvzGFm2YXaO5uxu1HvumYFTYc3ACr3q4j8VUb7NMxkQ\nlzu4QwPjOFJ43O+fY+HSPORXEDW5fXDGC5DGpox4+i08LxRmx7L6YPRUSUuPN8nI\nWyq1Qsq1zOQrNY/rohMXkBdSXxqC3yIRqvtLt4otCgay/5tMogJWkkS6ZKyFhb9z\nwVVy1fsbV10c9C+SHy4NH26CKaTtpTYLRBMjhTCS8bX8iDSjGIf2aZgYs1ir7gEz\n9wf5CvLiENmVWGwm64t6KSEAkA4NJ1hzgHUZPCjPHZE2SmhO/oHaxokTzqtbbENJ\n1QIDAQAB\n-----END PUBLIC KEY-----\n"
|
||||||
|
},
|
||||||
|
"published": "2016-11-01T00:00:00Z",
|
||||||
|
"summary": "<p>Designer & developer. Creator of Kirby CMS</p>",
|
||||||
|
"tag": [],
|
||||||
|
"type": "Person",
|
||||||
|
"url": "https://mastodon.social/@bastianallgeier"
|
||||||
|
}
|
112
test/fixtures/denniskoch.json
vendored
Normal file
112
test/fixtures/denniskoch.json
vendored
Normal file
|
@ -0,0 +1,112 @@
|
||||||
|
{
|
||||||
|
"@context": [
|
||||||
|
"https://www.w3.org/ns/activitystreams",
|
||||||
|
"https://w3id.org/security/v1",
|
||||||
|
{
|
||||||
|
"Curve25519Key": "toot:Curve25519Key",
|
||||||
|
"Device": "toot:Device",
|
||||||
|
"Ed25519Key": "toot:Ed25519Key",
|
||||||
|
"Ed25519Signature": "toot:Ed25519Signature",
|
||||||
|
"EncryptedMessage": "toot:EncryptedMessage",
|
||||||
|
"PropertyValue": "schema:PropertyValue",
|
||||||
|
"alsoKnownAs": {
|
||||||
|
"@id": "as:alsoKnownAs",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"cipherText": "toot:cipherText",
|
||||||
|
"claim": {
|
||||||
|
"@id": "toot:claim",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"deviceId": "toot:deviceId",
|
||||||
|
"devices": {
|
||||||
|
"@id": "toot:devices",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"discoverable": "toot:discoverable",
|
||||||
|
"featured": {
|
||||||
|
"@id": "toot:featured",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"featuredTags": {
|
||||||
|
"@id": "toot:featuredTags",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"fingerprintKey": {
|
||||||
|
"@id": "toot:fingerprintKey",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"focalPoint": {
|
||||||
|
"@container": "@list",
|
||||||
|
"@id": "toot:focalPoint"
|
||||||
|
},
|
||||||
|
"identityKey": {
|
||||||
|
"@id": "toot:identityKey",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"indexable": "toot:indexable",
|
||||||
|
"manuallyApprovesFollowers": "as:manuallyApprovesFollowers",
|
||||||
|
"memorial": "toot:memorial",
|
||||||
|
"messageFranking": "toot:messageFranking",
|
||||||
|
"messageType": "toot:messageType",
|
||||||
|
"movedTo": {
|
||||||
|
"@id": "as:movedTo",
|
||||||
|
"@type": "@id"
|
||||||
|
},
|
||||||
|
"publicKeyBase64": "toot:publicKeyBase64",
|
||||||
|
"schema": "http://schema.org#",
|
||||||
|
"suspended": "toot:suspended",
|
||||||
|
"toot": "http://joinmastodon.org/ns#",
|
||||||
|
"value": "schema:value"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"attachment": [
|
||||||
|
{
|
||||||
|
"name": "GitHub",
|
||||||
|
"type": "PropertyValue",
|
||||||
|
"value": "<a href=\"https://github.com/pxlrbt/\" target=\"_blank\" rel=\"nofollow noopener noreferrer me\" translate=\"no\"><span class=\"invisible\">https://</span><span class=\"\">github.com/pxlrbt/</span><span class=\"invisible\"></span></a>"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Discord",
|
||||||
|
"type": "PropertyValue",
|
||||||
|
"value": "pxlrbt#6029"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"devices": "https://phpc.social/users/denniskoch/collections/devices",
|
||||||
|
"discoverable": true,
|
||||||
|
"endpoints": {
|
||||||
|
"sharedInbox": "https://phpc.social/inbox"
|
||||||
|
},
|
||||||
|
"featured": "https://phpc.social/users/denniskoch/collections/featured",
|
||||||
|
"featuredTags": "https://phpc.social/users/denniskoch/collections/tags",
|
||||||
|
"followers": "https://phpc.social/users/denniskoch/followers",
|
||||||
|
"following": "https://phpc.social/users/denniskoch/following",
|
||||||
|
"icon": {
|
||||||
|
"mediaType": "image/jpeg",
|
||||||
|
"type": "Image",
|
||||||
|
"url": "https://media.phpc.social/accounts/avatars/109/364/097/179/042/485/original/6e770c7b3f5ef72d.jpg"
|
||||||
|
},
|
||||||
|
"id": "https://phpc.social/users/denniskoch",
|
||||||
|
"image": {
|
||||||
|
"mediaType": "image/jpeg",
|
||||||
|
"type": "Image",
|
||||||
|
"url": "https://media.phpc.social/accounts/headers/109/364/097/179/042/485/original/709da24705260c04.jpg"
|
||||||
|
},
|
||||||
|
"inbox": "https://phpc.social/users/denniskoch/inbox",
|
||||||
|
"indexable": true,
|
||||||
|
"manuallyApprovesFollowers": false,
|
||||||
|
"memorial": false,
|
||||||
|
"name": "Dennis Koch",
|
||||||
|
"outbox": "https://phpc.social/users/denniskoch/outbox",
|
||||||
|
"preferredUsername": "denniskoch",
|
||||||
|
"publicKey": {
|
||||||
|
"id": "https://phpc.social/users/denniskoch#main-key",
|
||||||
|
"owner": "https://phpc.social/users/denniskoch",
|
||||||
|
"publicKeyPem": "-----BEGIN PUBLIC KEY-----\nMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA4dmcSlqLj18gPvuslkmt\nQTniZ8ybO4pgvMvPLYtBuTBUjo49vJ/8Sw6jB5zcKb1haqIdny7Rv/vY3kCdCXcP\nloh1I+jthEgqLT8JpZWGwLGwg9piFhrMGADmt3N8du7HfglzuZ8LlVpnZ8feCw7I\nS2ua/ZCxE47mI45Z3ed2kkFYKWopWWqFn2lan/1OyHrcFKtCvaVjRdvo0UUt2tgl\nvyJI4+zN8FnrCbsMtcbI5nSzfJIrOc4LeaGmLJh+0o2rwoOQZc2487XWbeyfhjsq\nPRBpYN7pfHWQDvzQIN075LHTf9zDFsm6+HqY7Zs5rYxr72rvcX7d9JcP6CasIosY\nqwIDAQAB\n-----END PUBLIC KEY-----\n"
|
||||||
|
},
|
||||||
|
"published": "2022-11-18T00:00:00Z",
|
||||||
|
"summary": "<p>🧑💻 Full Stack Developer<br />🚀 Laravel, Filament, Livewire, Vue, Inertia<br />🌍 Germany</p>",
|
||||||
|
"tag": [],
|
||||||
|
"type": "Person",
|
||||||
|
"url": "https://phpc.social/@denniskoch"
|
||||||
|
}
|
62
test/fixtures/receiver_worker_signature_activity.json
vendored
Normal file
62
test/fixtures/receiver_worker_signature_activity.json
vendored
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
{
|
||||||
|
"@context": [
|
||||||
|
"https://www.w3.org/ns/activitystreams",
|
||||||
|
{
|
||||||
|
"atomUri": "ostatus:atomUri",
|
||||||
|
"blurhash": "toot:blurhash",
|
||||||
|
"conversation": "ostatus:conversation",
|
||||||
|
"focalPoint": {
|
||||||
|
"@container": "@list",
|
||||||
|
"@id": "toot:focalPoint"
|
||||||
|
},
|
||||||
|
"inReplyToAtomUri": "ostatus:inReplyToAtomUri",
|
||||||
|
"ostatus": "http://ostatus.org#",
|
||||||
|
"sensitive": "as:sensitive",
|
||||||
|
"toot": "http://joinmastodon.org/ns#",
|
||||||
|
"votersCount": "toot:votersCount"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"atomUri": "https://chaos.social/users/distantnative/statuses/109336635639931467",
|
||||||
|
"attachment": [
|
||||||
|
{
|
||||||
|
"blurhash": "UAK1zS00OXIUxuMxIUM{?b-:-;W:Di?b%2M{",
|
||||||
|
"height": 960,
|
||||||
|
"mediaType": "image/jpeg",
|
||||||
|
"name": null,
|
||||||
|
"type": "Document",
|
||||||
|
"url": "https://assets.chaos.social/media_attachments/files/109/336/634/286/114/657/original/2e6122063d8bfb26.jpeg",
|
||||||
|
"width": 346
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"attributedTo": "https://chaos.social/users/distantnative",
|
||||||
|
"cc": [
|
||||||
|
"https://chaos.social/users/distantnative/followers"
|
||||||
|
],
|
||||||
|
"content": "<p>Favorite piece of anthropology meta discourse.</p>",
|
||||||
|
"contentMap": {
|
||||||
|
"en": "<p>Favorite piece of anthropology meta discourse.</p>"
|
||||||
|
},
|
||||||
|
"conversation": "tag:chaos.social,2022-11-13:objectId=71843781:objectType=Conversation",
|
||||||
|
"id": "https://chaos.social/users/distantnative/statuses/109336635639931467",
|
||||||
|
"inReplyTo": null,
|
||||||
|
"inReplyToAtomUri": null,
|
||||||
|
"published": "2022-11-13T13:04:20Z",
|
||||||
|
"replies": {
|
||||||
|
"first": {
|
||||||
|
"items": [],
|
||||||
|
"next": "https://chaos.social/users/distantnative/statuses/109336635639931467/replies?only_other_accounts=true&page=true",
|
||||||
|
"partOf": "https://chaos.social/users/distantnative/statuses/109336635639931467/replies",
|
||||||
|
"type": "CollectionPage"
|
||||||
|
},
|
||||||
|
"id": "https://chaos.social/users/distantnative/statuses/109336635639931467/replies",
|
||||||
|
"type": "Collection"
|
||||||
|
},
|
||||||
|
"sensitive": false,
|
||||||
|
"summary": null,
|
||||||
|
"tag": [],
|
||||||
|
"to": [
|
||||||
|
"https://www.w3.org/ns/activitystreams#Public"
|
||||||
|
],
|
||||||
|
"type": "Note",
|
||||||
|
"url": "https://chaos.social/@distantnative/109336635639931467"
|
||||||
|
}
|
|
@ -0,0 +1,43 @@
|
||||||
|
# Pleroma: A lightweight social networking server
|
||||||
|
# Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/>
|
||||||
|
# SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
|
||||||
|
defmodule Pleroma.Repo.Migrations.PublisherMigrationChangeTest do
|
||||||
|
use Oban.Testing, repo: Pleroma.Repo
|
||||||
|
use Pleroma.DataCase
|
||||||
|
import Pleroma.Factory
|
||||||
|
import Pleroma.Tests.Helpers
|
||||||
|
|
||||||
|
alias Pleroma.Activity
|
||||||
|
alias Pleroma.Workers.PublisherWorker
|
||||||
|
|
||||||
|
setup_all do: require_migration("20240729163838_publisher_job_change")
|
||||||
|
|
||||||
|
describe "up/0" do
|
||||||
|
test "migrates publisher jobs to new format", %{migration: migration} do
|
||||||
|
user = insert(:user)
|
||||||
|
|
||||||
|
%Activity{id: activity_id, data: %{"id" => ap_id}} =
|
||||||
|
insert(:note_activity, user: user)
|
||||||
|
|
||||||
|
{:ok, %{id: job_id}} =
|
||||||
|
PublisherWorker.new(%{
|
||||||
|
"actor_id" => user.id,
|
||||||
|
"json" => "{}",
|
||||||
|
"id" => ap_id,
|
||||||
|
"inbox" => "https://example.com/inbox",
|
||||||
|
"unreachable_since" => nil
|
||||||
|
})
|
||||||
|
|> Oban.insert()
|
||||||
|
|
||||||
|
assert [%{id: ^job_id, args: %{"id" => ^ap_id}}] = all_enqueued(worker: PublisherWorker)
|
||||||
|
|
||||||
|
assert migration.up() == :ok
|
||||||
|
|
||||||
|
assert_enqueued(
|
||||||
|
worker: PublisherWorker,
|
||||||
|
args: %{"id" => ap_id, "activity_id" => activity_id}
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -1,49 +0,0 @@
|
||||||
# Pleroma: A lightweight social networking server
|
|
||||||
# Copyright © 2017-2023 Pleroma Authors <https://pleroma.social/>
|
|
||||||
# SPDX-License-Identifier: AGPL-3.0-only
|
|
||||||
|
|
||||||
defmodule Pleroma.User.BackupAsyncTest do
|
|
||||||
use Pleroma.DataCase, async: true
|
|
||||||
|
|
||||||
import Pleroma.Factory
|
|
||||||
import Mox
|
|
||||||
|
|
||||||
alias Pleroma.UnstubbedConfigMock, as: ConfigMock
|
|
||||||
alias Pleroma.User.Backup
|
|
||||||
alias Pleroma.User.Backup.ProcessorMock
|
|
||||||
|
|
||||||
setup do
|
|
||||||
user = insert(:user, %{nickname: "cofe", name: "Cofe", ap_id: "http://cofe.io/users/cofe"})
|
|
||||||
|
|
||||||
{:ok, backup} = user |> Backup.new() |> Repo.insert()
|
|
||||||
%{backup: backup}
|
|
||||||
end
|
|
||||||
|
|
||||||
test "it handles unrecoverable exceptions", %{backup: backup} do
|
|
||||||
ProcessorMock
|
|
||||||
|> expect(:do_process, fn _, _ ->
|
|
||||||
raise "mock exception"
|
|
||||||
end)
|
|
||||||
|
|
||||||
ConfigMock
|
|
||||||
|> stub_with(Pleroma.Config)
|
|
||||||
|
|
||||||
{:error, %{backup: backup, reason: :exit}} = Backup.process(backup, ProcessorMock)
|
|
||||||
|
|
||||||
assert backup.state == :failed
|
|
||||||
end
|
|
||||||
|
|
||||||
test "it handles timeouts", %{backup: backup} do
|
|
||||||
ProcessorMock
|
|
||||||
|> expect(:do_process, fn _, _ ->
|
|
||||||
Process.sleep(:timer.seconds(4))
|
|
||||||
end)
|
|
||||||
|
|
||||||
ConfigMock
|
|
||||||
|> expect(:get, fn [Pleroma.User.Backup, :process_wait_time] -> :timer.seconds(2) end)
|
|
||||||
|
|
||||||
{:error, %{backup: backup, reason: :timeout}} = Backup.process(backup, ProcessorMock)
|
|
||||||
|
|
||||||
assert backup.state == :failed
|
|
||||||
end
|
|
||||||
end
|
|
|
@ -6,7 +6,6 @@ defmodule Pleroma.User.BackupTest do
|
||||||
use Oban.Testing, repo: Pleroma.Repo
|
use Oban.Testing, repo: Pleroma.Repo
|
||||||
use Pleroma.DataCase
|
use Pleroma.DataCase
|
||||||
|
|
||||||
import Mock
|
|
||||||
import Pleroma.Factory
|
import Pleroma.Factory
|
||||||
import Swoosh.TestAssertions
|
import Swoosh.TestAssertions
|
||||||
import Mox
|
import Mox
|
||||||
|
@ -18,7 +17,6 @@ defmodule Pleroma.User.BackupTest do
|
||||||
alias Pleroma.Uploaders.S3.ExAwsMock
|
alias Pleroma.Uploaders.S3.ExAwsMock
|
||||||
alias Pleroma.User
|
alias Pleroma.User
|
||||||
alias Pleroma.User.Backup
|
alias Pleroma.User.Backup
|
||||||
alias Pleroma.User.Backup.ProcessorMock
|
|
||||||
alias Pleroma.Web.CommonAPI
|
alias Pleroma.Web.CommonAPI
|
||||||
alias Pleroma.Workers.BackupWorker
|
alias Pleroma.Workers.BackupWorker
|
||||||
|
|
||||||
|
@ -30,79 +28,56 @@ defmodule Pleroma.User.BackupTest do
|
||||||
ConfigMock
|
ConfigMock
|
||||||
|> stub_with(Pleroma.Config)
|
|> stub_with(Pleroma.Config)
|
||||||
|
|
||||||
ProcessorMock
|
|
||||||
|> stub_with(Pleroma.User.Backup.Processor)
|
|
||||||
|
|
||||||
:ok
|
:ok
|
||||||
end
|
end
|
||||||
|
|
||||||
test "it does not requrie enabled email" do
|
test "it does not requrie enabled email" do
|
||||||
clear_config([Pleroma.Emails.Mailer, :enabled], false)
|
clear_config([Pleroma.Emails.Mailer, :enabled], false)
|
||||||
user = insert(:user)
|
user = insert(:user)
|
||||||
assert {:ok, _} = Backup.create(user)
|
assert {:ok, _} = Backup.user(user)
|
||||||
end
|
end
|
||||||
|
|
||||||
test "it does not require user's email" do
|
test "it does not require user's email" do
|
||||||
user = insert(:user, %{email: nil})
|
user = insert(:user, %{email: nil})
|
||||||
assert {:ok, _} = Backup.create(user)
|
assert {:ok, _} = Backup.user(user)
|
||||||
end
|
end
|
||||||
|
|
||||||
test "it creates a backup record and an Oban job" do
|
test "it creates a backup record and an Oban job" do
|
||||||
%{id: user_id} = user = insert(:user)
|
user = insert(:user)
|
||||||
assert {:ok, %Oban.Job{args: args}} = Backup.create(user)
|
assert {:ok, %Backup{} = backup} = Backup.user(user)
|
||||||
|
assert {:ok, %Oban.Job{args: args}} = Backup.schedule_backup(backup)
|
||||||
assert_enqueued(worker: BackupWorker, args: args)
|
assert_enqueued(worker: BackupWorker, args: args)
|
||||||
|
|
||||||
backup = Backup.get(args["backup_id"])
|
backup = Backup.get_by_id(args["backup_id"])
|
||||||
assert %Backup{user_id: ^user_id, processed: false, file_size: 0, state: :pending} = backup
|
assert %Backup{processed: false, file_size: 0} = backup
|
||||||
end
|
end
|
||||||
|
|
||||||
test "it return an error if the export limit is over" do
|
test "it return an error if the export limit is over" do
|
||||||
%{id: user_id} = user = insert(:user)
|
user = insert(:user)
|
||||||
limit_days = Pleroma.Config.get([Backup, :limit_days])
|
limit_days = Pleroma.Config.get([Backup, :limit_days])
|
||||||
assert {:ok, %Oban.Job{args: args}} = Backup.create(user)
|
{:ok, first_backup} = Backup.user(user)
|
||||||
backup = Backup.get(args["backup_id"])
|
{:ok, _run_backup} = Backup.run(first_backup)
|
||||||
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"}
|
assert Backup.user(user) == {:error, "Last export was less than #{limit_days} days ago"}
|
||||||
end
|
end
|
||||||
|
|
||||||
test "it process a backup record" do
|
test "it process a backup record" do
|
||||||
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
||||||
%{id: user_id} = user = insert(:user)
|
%{id: user_id} = user = insert(:user)
|
||||||
|
|
||||||
assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id} = args}} = Backup.create(user)
|
assert {:ok, %Backup{id: backup_id}} = Backup.user(user)
|
||||||
assert {:ok, backup} = perform_job(BackupWorker, args)
|
|
||||||
|
oban_args = %{"op" => "process", "backup_id" => backup_id}
|
||||||
|
|
||||||
|
assert {:ok, backup} = perform_job(BackupWorker, oban_args)
|
||||||
assert backup.file_size > 0
|
assert backup.file_size > 0
|
||||||
assert %Backup{id: ^backup_id, processed: true, user_id: ^user_id, state: :complete} = backup
|
assert match?(%Backup{id: ^backup_id, processed: true, user_id: ^user_id}, backup)
|
||||||
|
|
||||||
delete_job_args = %{"op" => "delete", "backup_id" => backup_id}
|
delete_job_args = %{"op" => "delete", "backup_id" => backup_id}
|
||||||
|
|
||||||
assert_enqueued(worker: BackupWorker, args: delete_job_args)
|
assert_enqueued(worker: BackupWorker, args: delete_job_args)
|
||||||
assert {:ok, backup} = perform_job(BackupWorker, delete_job_args)
|
assert {:ok, backup} = perform_job(BackupWorker, delete_job_args)
|
||||||
refute Backup.get(backup_id)
|
refute Backup.get_by_id(backup_id)
|
||||||
|
|
||||||
email = Pleroma.Emails.UserEmail.backup_is_ready_email(backup)
|
|
||||||
|
|
||||||
assert_email_sent(
|
|
||||||
to: {user.name, user.email},
|
|
||||||
html_body: email.html_body
|
|
||||||
)
|
|
||||||
end
|
|
||||||
|
|
||||||
test "it updates states of the backup" do
|
|
||||||
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
|
||||||
%{id: user_id} = user = insert(:user)
|
|
||||||
|
|
||||||
assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id} = args}} = Backup.create(user)
|
|
||||||
assert {:ok, backup} = perform_job(BackupWorker, args)
|
|
||||||
assert backup.file_size > 0
|
|
||||||
assert %Backup{id: ^backup_id, processed: true, user_id: ^user_id, state: :complete} = backup
|
|
||||||
|
|
||||||
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)
|
|
||||||
|
|
||||||
email = Pleroma.Emails.UserEmail.backup_is_ready_email(backup)
|
email = Pleroma.Emails.UserEmail.backup_is_ready_email(backup)
|
||||||
|
|
||||||
|
@ -116,10 +91,15 @@ test "it does not send an email if the user does not have an email" do
|
||||||
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
||||||
%{id: user_id} = user = insert(:user, %{email: nil})
|
%{id: user_id} = user = insert(:user, %{email: nil})
|
||||||
|
|
||||||
assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id} = args}} = Backup.create(user)
|
assert {:ok, %Backup{} = backup} = Backup.user(user)
|
||||||
assert {:ok, backup} = perform_job(BackupWorker, args)
|
|
||||||
assert backup.file_size > 0
|
expected_args = %{"op" => "process", "backup_id" => backup.id}
|
||||||
assert %Backup{id: ^backup_id, processed: true, user_id: ^user_id} = backup
|
|
||||||
|
assert_enqueued(worker: BackupWorker, args: %{"backup_id" => backup.id})
|
||||||
|
assert {:ok, completed_backup} = perform_job(BackupWorker, expected_args)
|
||||||
|
assert completed_backup.file_size > 0
|
||||||
|
assert completed_backup.processed
|
||||||
|
assert completed_backup.user_id == user_id
|
||||||
|
|
||||||
assert_no_email_sent()
|
assert_no_email_sent()
|
||||||
end
|
end
|
||||||
|
@ -129,10 +109,13 @@ test "it does not send an email if mailer is not on" do
|
||||||
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
||||||
%{id: user_id} = user = insert(:user)
|
%{id: user_id} = user = insert(:user)
|
||||||
|
|
||||||
assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id} = args}} = Backup.create(user)
|
assert {:ok, %Backup{id: backup_id}} = Backup.user(user)
|
||||||
assert {:ok, backup} = perform_job(BackupWorker, args)
|
|
||||||
|
oban_args = %{"op" => "process", "backup_id" => backup_id}
|
||||||
|
|
||||||
|
assert {:ok, backup} = perform_job(BackupWorker, oban_args)
|
||||||
assert backup.file_size > 0
|
assert backup.file_size > 0
|
||||||
assert %Backup{id: ^backup_id, processed: true, user_id: ^user_id} = backup
|
assert match?(%Backup{id: ^backup_id, processed: true, user_id: ^user_id}, backup)
|
||||||
|
|
||||||
assert_no_email_sent()
|
assert_no_email_sent()
|
||||||
end
|
end
|
||||||
|
@ -141,10 +124,15 @@ test "it does not send an email if the user has an empty email" do
|
||||||
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
||||||
%{id: user_id} = user = insert(:user, %{email: ""})
|
%{id: user_id} = user = insert(:user, %{email: ""})
|
||||||
|
|
||||||
assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id} = args}} = Backup.create(user)
|
assert {:ok, %Backup{id: backup_id} = backup} = Backup.user(user)
|
||||||
assert {:ok, backup} = perform_job(BackupWorker, args)
|
|
||||||
|
expected_args = %{"op" => "process", "backup_id" => backup.id}
|
||||||
|
|
||||||
|
assert_enqueued(worker: BackupWorker, args: expected_args)
|
||||||
|
|
||||||
|
assert {:ok, backup} = perform_job(BackupWorker, expected_args)
|
||||||
assert backup.file_size > 0
|
assert backup.file_size > 0
|
||||||
assert %Backup{id: ^backup_id, processed: true, user_id: ^user_id} = backup
|
assert match?(%Backup{id: ^backup_id, processed: true, user_id: ^user_id}, backup)
|
||||||
|
|
||||||
assert_no_email_sent()
|
assert_no_email_sent()
|
||||||
end
|
end
|
||||||
|
@ -154,16 +142,13 @@ test "it removes outdated backups after creating a fresh one" do
|
||||||
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
||||||
user = insert(:user)
|
user = insert(:user)
|
||||||
|
|
||||||
assert {:ok, job1} = Backup.create(user)
|
assert {:ok, %{id: backup_one_id}} = Backup.user(user)
|
||||||
|
assert {:ok, %{id: _backup_two_id}} = Backup.user(user)
|
||||||
assert {:ok, %Backup{}} = ObanHelpers.perform(job1)
|
|
||||||
assert {:ok, job2} = Backup.create(user)
|
|
||||||
assert Pleroma.Repo.aggregate(Backup, :count) == 2
|
|
||||||
assert {:ok, backup2} = ObanHelpers.perform(job2)
|
|
||||||
|
|
||||||
|
# Run the backups
|
||||||
ObanHelpers.perform_all()
|
ObanHelpers.perform_all()
|
||||||
|
|
||||||
assert [^backup2] = Pleroma.Repo.all(Backup)
|
assert_enqueued(worker: BackupWorker, args: %{"op" => "delete", "backup_id" => backup_one_id})
|
||||||
end
|
end
|
||||||
|
|
||||||
test "it creates a zip archive with user data" do
|
test "it creates a zip archive with user data" do
|
||||||
|
@ -194,9 +179,12 @@ test "it creates a zip archive with user data" do
|
||||||
{:ok, _message_1} = CommonAPI.post_chat_message(user, other_user, "hey")
|
{:ok, _message_1} = CommonAPI.post_chat_message(user, other_user, "hey")
|
||||||
{:ok, _message_2} = CommonAPI.post_chat_message(other_user, user, "ho")
|
{:ok, _message_2} = CommonAPI.post_chat_message(other_user, user, "ho")
|
||||||
|
|
||||||
assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
|
assert {:ok, backup} = Backup.user(user)
|
||||||
assert {:ok, path} = Backup.export(backup, self())
|
assert {:ok, run_backup} = Backup.run(backup)
|
||||||
assert {:ok, zipfile} = :zip.zip_open(String.to_charlist(path), [:memory])
|
|
||||||
|
tempfile = Path.join([run_backup.tempdir, run_backup.file_name])
|
||||||
|
|
||||||
|
assert {:ok, zipfile} = :zip.zip_open(String.to_charlist(tempfile), [:memory])
|
||||||
assert {:ok, {~c"actor.json", json}} = :zip.zip_get(~c"actor.json", zipfile)
|
assert {:ok, {~c"actor.json", json}} = :zip.zip_get(~c"actor.json", zipfile)
|
||||||
|
|
||||||
assert %{
|
assert %{
|
||||||
|
@ -330,10 +318,10 @@ test "it creates a zip archive with user data" do
|
||||||
} = Jason.decode!(json)
|
} = Jason.decode!(json)
|
||||||
|
|
||||||
:zip.zip_close(zipfile)
|
:zip.zip_close(zipfile)
|
||||||
File.rm!(path)
|
File.rm_rf!(run_backup.tempdir)
|
||||||
end
|
end
|
||||||
|
|
||||||
test "it counts the correct number processed" do
|
test "correct number processed" do
|
||||||
user = insert(:user, %{nickname: "cofe", name: "Cofe", ap_id: "http://cofe.io/users/cofe"})
|
user = insert(:user, %{nickname: "cofe", name: "Cofe", ap_id: "http://cofe.io/users/cofe"})
|
||||||
|
|
||||||
Enum.map(1..120, fn i ->
|
Enum.map(1..120, fn i ->
|
||||||
|
@ -343,43 +331,21 @@ test "it counts the correct number processed" do
|
||||||
end)
|
end)
|
||||||
|
|
||||||
assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
|
assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
|
||||||
{:ok, backup} = Backup.process(backup)
|
{:ok, backup} = Backup.run(backup)
|
||||||
|
|
||||||
assert backup.processed_number == 1 + 120 + 120 + 120
|
zip_path = Path.join([backup.tempdir, backup.file_name])
|
||||||
|
|
||||||
Backup.delete(backup)
|
assert {:ok, zipfile} = :zip.zip_open(String.to_charlist(zip_path), [:memory])
|
||||||
end
|
|
||||||
|
|
||||||
test "it handles errors" do
|
backup_parts = [~c"likes.json", ~c"bookmarks.json", ~c"outbox.json"]
|
||||||
user = insert(:user, %{nickname: "cofe", name: "Cofe", ap_id: "http://cofe.io/users/cofe"})
|
|
||||||
|
|
||||||
Enum.map(1..120, fn i ->
|
Enum.each(backup_parts, fn part ->
|
||||||
{:ok, _status} = CommonAPI.post(user, %{status: "status #{i}"})
|
assert {:ok, {_part, part_json}} = :zip.zip_get(part, zipfile)
|
||||||
|
{:ok, decoded_part} = Jason.decode(part_json)
|
||||||
|
assert decoded_part["totalItems"] == 120
|
||||||
end)
|
end)
|
||||||
|
|
||||||
assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
|
Backup.delete_archive(backup)
|
||||||
|
|
||||||
with_mock Pleroma.Web.ActivityPub.Transmogrifier,
|
|
||||||
[:passthrough],
|
|
||||||
prepare_outgoing: fn data ->
|
|
||||||
object =
|
|
||||||
data["object"]
|
|
||||||
|> Pleroma.Object.normalize(fetch: false)
|
|
||||||
|> Map.get(:data)
|
|
||||||
|
|
||||||
data = data |> Map.put("object", object)
|
|
||||||
|
|
||||||
if String.contains?(data["object"]["content"], "119"),
|
|
||||||
do: raise(%Postgrex.Error{}),
|
|
||||||
else: {:ok, data}
|
|
||||||
end do
|
|
||||||
{:ok, backup} = Backup.process(backup)
|
|
||||||
assert backup.processed
|
|
||||||
assert backup.state == :complete
|
|
||||||
assert backup.processed_number == 1 + 119
|
|
||||||
|
|
||||||
Backup.delete(backup)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
||||||
describe "it uploads and deletes a backup archive" do
|
describe "it uploads and deletes a backup archive" do
|
||||||
|
@ -398,12 +364,11 @@ test "it handles errors" do
|
||||||
Bookmark.create(user.id, status3.id)
|
Bookmark.create(user.id, status3.id)
|
||||||
|
|
||||||
assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
|
assert {:ok, backup} = user |> Backup.new() |> Repo.insert()
|
||||||
assert {:ok, path} = Backup.export(backup, self())
|
|
||||||
|
|
||||||
[path: path, backup: backup]
|
[backup: backup]
|
||||||
end
|
end
|
||||||
|
|
||||||
test "S3", %{path: path, backup: backup} do
|
test "S3", %{backup: backup} do
|
||||||
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.S3)
|
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.S3)
|
||||||
clear_config([Pleroma.Uploaders.S3, :streaming_enabled], false)
|
clear_config([Pleroma.Uploaders.S3, :streaming_enabled], false)
|
||||||
|
|
||||||
|
@ -413,15 +378,17 @@ test "S3", %{path: path, backup: backup} do
|
||||||
%{http_method: :delete} -> {:ok, %{status_code: 204}}
|
%{http_method: :delete} -> {:ok, %{status_code: 204}}
|
||||||
end)
|
end)
|
||||||
|
|
||||||
assert {:ok, %Pleroma.Upload{}} = Backup.upload(backup, path)
|
assert {:ok, backup} = Backup.run(backup)
|
||||||
assert {:ok, _backup} = Backup.delete(backup)
|
assert {:ok, %Backup{processed: true}} = Backup.upload(backup)
|
||||||
|
assert {:ok, _backup} = Backup.delete_archive(backup)
|
||||||
end
|
end
|
||||||
|
|
||||||
test "Local", %{path: path, backup: backup} do
|
test "Local", %{backup: backup} do
|
||||||
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
clear_config([Pleroma.Upload, :uploader], Pleroma.Uploaders.Local)
|
||||||
|
|
||||||
assert {:ok, %Pleroma.Upload{}} = Backup.upload(backup, path)
|
assert {:ok, backup} = Backup.run(backup)
|
||||||
assert {:ok, _backup} = Backup.delete(backup)
|
assert {:ok, %Backup{processed: true}} = Backup.upload(backup)
|
||||||
|
assert {:ok, _backup} = Backup.delete_archive(backup)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -137,6 +137,7 @@ test "it returns inbox for messages involving single recipients in total" do
|
||||||
test "publish to url with with different ports" do
|
test "publish to url with with different ports" do
|
||||||
inbox80 = "http://42.site/users/nick1/inbox"
|
inbox80 = "http://42.site/users/nick1/inbox"
|
||||||
inbox42 = "http://42.site:42/users/nick1/inbox"
|
inbox42 = "http://42.site:42/users/nick1/inbox"
|
||||||
|
activity = insert(:note_activity)
|
||||||
|
|
||||||
mock(fn
|
mock(fn
|
||||||
%{method: :post, url: "http://42.site:42/users/nick1/inbox"} ->
|
%{method: :post, url: "http://42.site:42/users/nick1/inbox"} ->
|
||||||
|
@ -146,23 +147,19 @@ test "publish to url with with different ports" do
|
||||||
{:ok, %Tesla.Env{status: 200, body: "port 80"}}
|
{:ok, %Tesla.Env{status: 200, body: "port 80"}}
|
||||||
end)
|
end)
|
||||||
|
|
||||||
actor = insert(:user)
|
_actor = insert(:user)
|
||||||
|
|
||||||
assert {:ok, %{body: "port 42"}} =
|
assert {:ok, %{body: "port 42"}} =
|
||||||
Publisher.publish_one(%{
|
Publisher.publish_one(%{
|
||||||
inbox: inbox42,
|
inbox: inbox42,
|
||||||
json: "{}",
|
activity_id: activity.id,
|
||||||
actor: actor,
|
|
||||||
id: 1,
|
|
||||||
unreachable_since: true
|
unreachable_since: true
|
||||||
})
|
})
|
||||||
|
|
||||||
assert {:ok, %{body: "port 80"}} =
|
assert {:ok, %{body: "port 80"}} =
|
||||||
Publisher.publish_one(%{
|
Publisher.publish_one(%{
|
||||||
inbox: inbox80,
|
inbox: inbox80,
|
||||||
json: "{}",
|
activity_id: activity.id,
|
||||||
actor: actor,
|
|
||||||
id: 1,
|
|
||||||
unreachable_since: true
|
unreachable_since: true
|
||||||
})
|
})
|
||||||
end
|
end
|
||||||
|
@ -171,10 +168,13 @@ test "publish to url with with different ports" do
|
||||||
Instances,
|
Instances,
|
||||||
[:passthrough],
|
[:passthrough],
|
||||||
[] do
|
[] do
|
||||||
actor = insert(:user)
|
_actor = insert(:user)
|
||||||
inbox = "http://200.site/users/nick1/inbox"
|
inbox = "http://200.site/users/nick1/inbox"
|
||||||
|
activity = insert(:note_activity)
|
||||||
|
|
||||||
|
assert {:ok, _} =
|
||||||
|
Publisher.publish_one(%{inbox: inbox, activity_id: activity.id})
|
||||||
|
|
||||||
assert {:ok, _} = Publisher.publish_one(%{inbox: inbox, json: "{}", actor: actor, id: 1})
|
|
||||||
assert called(Instances.set_reachable(inbox))
|
assert called(Instances.set_reachable(inbox))
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -182,15 +182,14 @@ test "publish to url with with different ports" do
|
||||||
Instances,
|
Instances,
|
||||||
[:passthrough],
|
[:passthrough],
|
||||||
[] do
|
[] do
|
||||||
actor = insert(:user)
|
_actor = insert(:user)
|
||||||
inbox = "http://200.site/users/nick1/inbox"
|
inbox = "http://200.site/users/nick1/inbox"
|
||||||
|
activity = insert(:note_activity)
|
||||||
|
|
||||||
assert {:ok, _} =
|
assert {:ok, _} =
|
||||||
Publisher.publish_one(%{
|
Publisher.publish_one(%{
|
||||||
inbox: inbox,
|
inbox: inbox,
|
||||||
json: "{}",
|
activity_id: activity.id,
|
||||||
actor: actor,
|
|
||||||
id: 1,
|
|
||||||
unreachable_since: NaiveDateTime.utc_now()
|
unreachable_since: NaiveDateTime.utc_now()
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -201,15 +200,14 @@ test "publish to url with with different ports" do
|
||||||
Instances,
|
Instances,
|
||||||
[:passthrough],
|
[:passthrough],
|
||||||
[] do
|
[] do
|
||||||
actor = insert(:user)
|
_actor = insert(:user)
|
||||||
inbox = "http://200.site/users/nick1/inbox"
|
inbox = "http://200.site/users/nick1/inbox"
|
||||||
|
activity = insert(:note_activity)
|
||||||
|
|
||||||
assert {:ok, _} =
|
assert {:ok, _} =
|
||||||
Publisher.publish_one(%{
|
Publisher.publish_one(%{
|
||||||
inbox: inbox,
|
inbox: inbox,
|
||||||
json: "{}",
|
activity_id: activity.id,
|
||||||
actor: actor,
|
|
||||||
id: 1,
|
|
||||||
unreachable_since: nil
|
unreachable_since: nil
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -220,11 +218,12 @@ test "publish to url with with different ports" do
|
||||||
Instances,
|
Instances,
|
||||||
[:passthrough],
|
[:passthrough],
|
||||||
[] do
|
[] do
|
||||||
actor = insert(:user)
|
_actor = insert(:user)
|
||||||
inbox = "http://404.site/users/nick1/inbox"
|
inbox = "http://404.site/users/nick1/inbox"
|
||||||
|
activity = insert(:note_activity)
|
||||||
|
|
||||||
assert {:cancel, _} =
|
assert {:cancel, _} =
|
||||||
Publisher.publish_one(%{inbox: inbox, json: "{}", actor: actor, id: 1})
|
Publisher.publish_one(%{inbox: inbox, activity_id: activity.id})
|
||||||
|
|
||||||
assert called(Instances.set_unreachable(inbox))
|
assert called(Instances.set_unreachable(inbox))
|
||||||
end
|
end
|
||||||
|
@ -233,12 +232,16 @@ test "publish to url with with different ports" do
|
||||||
Instances,
|
Instances,
|
||||||
[:passthrough],
|
[:passthrough],
|
||||||
[] do
|
[] do
|
||||||
actor = insert(:user)
|
_actor = insert(:user)
|
||||||
inbox = "http://connrefused.site/users/nick1/inbox"
|
inbox = "http://connrefused.site/users/nick1/inbox"
|
||||||
|
activity = insert(:note_activity)
|
||||||
|
|
||||||
assert capture_log(fn ->
|
assert capture_log(fn ->
|
||||||
assert {:error, _} =
|
assert {:error, _} =
|
||||||
Publisher.publish_one(%{inbox: inbox, json: "{}", actor: actor, id: 1})
|
Publisher.publish_one(%{
|
||||||
|
inbox: inbox,
|
||||||
|
activity_id: activity.id
|
||||||
|
})
|
||||||
end) =~ "connrefused"
|
end) =~ "connrefused"
|
||||||
|
|
||||||
assert called(Instances.set_unreachable(inbox))
|
assert called(Instances.set_unreachable(inbox))
|
||||||
|
@ -248,10 +251,12 @@ test "publish to url with with different ports" do
|
||||||
Instances,
|
Instances,
|
||||||
[:passthrough],
|
[:passthrough],
|
||||||
[] do
|
[] do
|
||||||
actor = insert(:user)
|
_actor = insert(:user)
|
||||||
inbox = "http://200.site/users/nick1/inbox"
|
inbox = "http://200.site/users/nick1/inbox"
|
||||||
|
activity = insert(:note_activity)
|
||||||
|
|
||||||
assert {:ok, _} = Publisher.publish_one(%{inbox: inbox, json: "{}", actor: actor, id: 1})
|
assert {:ok, _} =
|
||||||
|
Publisher.publish_one(%{inbox: inbox, activity_id: activity.id})
|
||||||
|
|
||||||
refute called(Instances.set_unreachable(inbox))
|
refute called(Instances.set_unreachable(inbox))
|
||||||
end
|
end
|
||||||
|
@ -260,16 +265,15 @@ test "publish to url with with different ports" do
|
||||||
Instances,
|
Instances,
|
||||||
[:passthrough],
|
[:passthrough],
|
||||||
[] do
|
[] do
|
||||||
actor = insert(:user)
|
_actor = insert(:user)
|
||||||
inbox = "http://connrefused.site/users/nick1/inbox"
|
inbox = "http://connrefused.site/users/nick1/inbox"
|
||||||
|
activity = insert(:note_activity)
|
||||||
|
|
||||||
assert capture_log(fn ->
|
assert capture_log(fn ->
|
||||||
assert {:error, _} =
|
assert {:error, _} =
|
||||||
Publisher.publish_one(%{
|
Publisher.publish_one(%{
|
||||||
inbox: inbox,
|
inbox: inbox,
|
||||||
json: "{}",
|
activity_id: activity.id,
|
||||||
actor: actor,
|
|
||||||
id: 1,
|
|
||||||
unreachable_since: NaiveDateTime.utc_now()
|
unreachable_since: NaiveDateTime.utc_now()
|
||||||
})
|
})
|
||||||
end) =~ "connrefused"
|
end) =~ "connrefused"
|
||||||
|
@ -309,8 +313,7 @@ test "publish to url with with different ports" do
|
||||||
assert not called(
|
assert not called(
|
||||||
Publisher.enqueue_one(%{
|
Publisher.enqueue_one(%{
|
||||||
inbox: "https://domain.com/users/nick1/inbox",
|
inbox: "https://domain.com/users/nick1/inbox",
|
||||||
actor_id: actor.id,
|
activity_id: note_activity.id
|
||||||
id: note_activity.data["id"]
|
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
@ -346,8 +349,7 @@ test "publish to url with with different ports" do
|
||||||
Publisher.enqueue_one(
|
Publisher.enqueue_one(
|
||||||
%{
|
%{
|
||||||
inbox: "https://domain.com/users/nick1/inbox",
|
inbox: "https://domain.com/users/nick1/inbox",
|
||||||
actor_id: actor.id,
|
activity_id: note_activity.id
|
||||||
id: note_activity.data["id"]
|
|
||||||
},
|
},
|
||||||
priority: 1
|
priority: 1
|
||||||
)
|
)
|
||||||
|
@ -370,8 +372,7 @@ test "publish to url with with different ports" do
|
||||||
Publisher.enqueue_one(
|
Publisher.enqueue_one(
|
||||||
%{
|
%{
|
||||||
inbox: :_,
|
inbox: :_,
|
||||||
actor_id: actor.id,
|
activity_id: note_activity.id
|
||||||
id: note_activity.data["id"]
|
|
||||||
},
|
},
|
||||||
priority: 0
|
priority: 0
|
||||||
)
|
)
|
||||||
|
@ -405,8 +406,7 @@ test "publish to url with with different ports" do
|
||||||
assert called(
|
assert called(
|
||||||
Publisher.enqueue_one(%{
|
Publisher.enqueue_one(%{
|
||||||
inbox: "https://domain.com/users/nick1/inbox",
|
inbox: "https://domain.com/users/nick1/inbox",
|
||||||
actor_id: actor.id,
|
activity_id: note_activity.id
|
||||||
id: note_activity.data["id"]
|
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
@ -456,8 +456,7 @@ test "publish to url with with different ports" do
|
||||||
Publisher.enqueue_one(
|
Publisher.enqueue_one(
|
||||||
%{
|
%{
|
||||||
inbox: "https://domain.com/users/nick1/inbox",
|
inbox: "https://domain.com/users/nick1/inbox",
|
||||||
actor_id: actor.id,
|
activity_id: delete.id
|
||||||
id: delete.data["id"]
|
|
||||||
},
|
},
|
||||||
priority: 1
|
priority: 1
|
||||||
)
|
)
|
||||||
|
@ -467,8 +466,7 @@ test "publish to url with with different ports" do
|
||||||
Publisher.enqueue_one(
|
Publisher.enqueue_one(
|
||||||
%{
|
%{
|
||||||
inbox: "https://domain2.com/users/nick1/inbox",
|
inbox: "https://domain2.com/users/nick1/inbox",
|
||||||
actor_id: actor.id,
|
activity_id: delete.id
|
||||||
id: delete.data["id"]
|
|
||||||
},
|
},
|
||||||
priority: 1
|
priority: 1
|
||||||
)
|
)
|
||||||
|
|
|
@ -1096,9 +1096,13 @@ test "it creates a backup", %{conn: conn} do
|
||||||
|
|
||||||
ObanHelpers.perform_all()
|
ObanHelpers.perform_all()
|
||||||
|
|
||||||
email = Pleroma.Emails.UserEmail.backup_is_ready_email(backup, admin.id)
|
email = Pleroma.Emails.UserEmail.backup_is_ready_email(backup)
|
||||||
|
|
||||||
|
assert String.contains?(
|
||||||
|
email.html_body,
|
||||||
|
"A full backup of your Pleroma account was requested"
|
||||||
|
)
|
||||||
|
|
||||||
assert String.contains?(email.html_body, "Admin @#{admin.nickname} requested a full backup")
|
|
||||||
assert_email_sent(to: {user.name, user.email}, html_body: email.html_body)
|
assert_email_sent(to: {user.name, user.email}, html_body: email.html_body)
|
||||||
|
|
||||||
log_message = "@#{admin_nickname} requested account backup for @#{user_nickname}"
|
log_message = "@#{admin_nickname} requested account backup for @#{user_nickname}"
|
||||||
|
|
|
@ -2005,7 +2005,7 @@ test "when deleting posts", %{
|
||||||
{:ok, _, _} = Pleroma.User.follow(remote_one, local_user)
|
{:ok, _, _} = Pleroma.User.follow(remote_one, local_user)
|
||||||
{:ok, _, _} = Pleroma.User.follow(remote_two, local_user)
|
{:ok, _, _} = Pleroma.User.follow(remote_two, local_user)
|
||||||
|
|
||||||
{:ok, %{data: %{"id" => ap_id}} = activity} =
|
{:ok, %{id: activity_id} = _activity} =
|
||||||
CommonAPI.post(local_user, %{status: "Happy Friday everyone!"})
|
CommonAPI.post(local_user, %{status: "Happy Friday everyone!"})
|
||||||
|
|
||||||
# Generate the publish_one jobs
|
# Generate the publish_one jobs
|
||||||
|
@ -2019,7 +2019,7 @@ test "when deleting posts", %{
|
||||||
state: "available",
|
state: "available",
|
||||||
queue: "federator_outgoing",
|
queue: "federator_outgoing",
|
||||||
worker: "Pleroma.Workers.PublisherWorker",
|
worker: "Pleroma.Workers.PublisherWorker",
|
||||||
args: %{"op" => "publish_one", "params" => %{"id" => ^ap_id}}
|
args: %{"op" => "publish_one", "params" => %{"activity_id" => ^activity_id}}
|
||||||
},
|
},
|
||||||
job
|
job
|
||||||
)
|
)
|
||||||
|
@ -2028,7 +2028,7 @@ test "when deleting posts", %{
|
||||||
assert length(publish_one_jobs) == 2
|
assert length(publish_one_jobs) == 2
|
||||||
|
|
||||||
# The delete should have triggered cancelling the publish_one jobs
|
# The delete should have triggered cancelling the publish_one jobs
|
||||||
assert {:ok, _delete} = CommonAPI.delete(activity.id, local_user)
|
assert {:ok, _delete} = CommonAPI.delete(activity_id, local_user)
|
||||||
|
|
||||||
# all_enqueued/1 will not return cancelled jobs
|
# all_enqueued/1 will not return cancelled jobs
|
||||||
cancelled_jobs =
|
cancelled_jobs =
|
||||||
|
@ -2036,7 +2036,7 @@ test "when deleting posts", %{
|
||||||
|> where([j], j.worker == "Pleroma.Workers.PublisherWorker")
|
|> where([j], j.worker == "Pleroma.Workers.PublisherWorker")
|
||||||
|> where([j], j.state == "cancelled")
|
|> where([j], j.state == "cancelled")
|
||||||
|> where([j], j.args["op"] == "publish_one")
|
|> where([j], j.args["op"] == "publish_one")
|
||||||
|> where([j], j.args["params"]["id"] == ^ap_id)
|
|> where([j], j.args["params"]["activity_id"] == ^activity_id)
|
||||||
|> Pleroma.Repo.all()
|
|> Pleroma.Repo.all()
|
||||||
|
|
||||||
assert length(cancelled_jobs) == 2
|
assert length(cancelled_jobs) == 2
|
||||||
|
@ -2049,7 +2049,7 @@ test "when unfavoriting posts", %{
|
||||||
{:ok, activity} =
|
{:ok, activity} =
|
||||||
CommonAPI.post(remote_user, %{status: "I like turtles!"})
|
CommonAPI.post(remote_user, %{status: "I like turtles!"})
|
||||||
|
|
||||||
{:ok, %{data: %{"id" => ap_id}} = _favorite} =
|
{:ok, %{id: favorite_id} = _favorite} =
|
||||||
CommonAPI.favorite(activity.id, local_user)
|
CommonAPI.favorite(activity.id, local_user)
|
||||||
|
|
||||||
# Generate the publish_one jobs
|
# Generate the publish_one jobs
|
||||||
|
@ -2063,7 +2063,7 @@ test "when unfavoriting posts", %{
|
||||||
state: "available",
|
state: "available",
|
||||||
queue: "federator_outgoing",
|
queue: "federator_outgoing",
|
||||||
worker: "Pleroma.Workers.PublisherWorker",
|
worker: "Pleroma.Workers.PublisherWorker",
|
||||||
args: %{"op" => "publish_one", "params" => %{"id" => ^ap_id}}
|
args: %{"op" => "publish_one", "params" => %{"activity_id" => ^favorite_id}}
|
||||||
},
|
},
|
||||||
job
|
job
|
||||||
)
|
)
|
||||||
|
@ -2080,7 +2080,7 @@ test "when unfavoriting posts", %{
|
||||||
|> where([j], j.worker == "Pleroma.Workers.PublisherWorker")
|
|> where([j], j.worker == "Pleroma.Workers.PublisherWorker")
|
||||||
|> where([j], j.state == "cancelled")
|
|> where([j], j.state == "cancelled")
|
||||||
|> where([j], j.args["op"] == "publish_one")
|
|> where([j], j.args["op"] == "publish_one")
|
||||||
|> where([j], j.args["params"]["id"] == ^ap_id)
|
|> where([j], j.args["params"]["activity_id"] == ^favorite_id)
|
||||||
|> Pleroma.Repo.all()
|
|> Pleroma.Repo.all()
|
||||||
|
|
||||||
assert length(cancelled_jobs) == 1
|
assert length(cancelled_jobs) == 1
|
||||||
|
@ -2097,7 +2097,7 @@ test "when unboosting posts", %{
|
||||||
{:ok, activity} =
|
{:ok, activity} =
|
||||||
CommonAPI.post(remote_one, %{status: "This is an unpleasant post"})
|
CommonAPI.post(remote_one, %{status: "This is an unpleasant post"})
|
||||||
|
|
||||||
{:ok, %{data: %{"id" => ap_id}} = _repeat} =
|
{:ok, %{id: repeat_id} = _repeat} =
|
||||||
CommonAPI.repeat(activity.id, local_user)
|
CommonAPI.repeat(activity.id, local_user)
|
||||||
|
|
||||||
# Generate the publish_one jobs
|
# Generate the publish_one jobs
|
||||||
|
@ -2111,7 +2111,7 @@ test "when unboosting posts", %{
|
||||||
state: "available",
|
state: "available",
|
||||||
queue: "federator_outgoing",
|
queue: "federator_outgoing",
|
||||||
worker: "Pleroma.Workers.PublisherWorker",
|
worker: "Pleroma.Workers.PublisherWorker",
|
||||||
args: %{"op" => "publish_one", "params" => %{"id" => ^ap_id}}
|
args: %{"op" => "publish_one", "params" => %{"activity_id" => ^repeat_id}}
|
||||||
},
|
},
|
||||||
job
|
job
|
||||||
)
|
)
|
||||||
|
@ -2128,7 +2128,7 @@ test "when unboosting posts", %{
|
||||||
|> where([j], j.worker == "Pleroma.Workers.PublisherWorker")
|
|> where([j], j.worker == "Pleroma.Workers.PublisherWorker")
|
||||||
|> where([j], j.state == "cancelled")
|
|> where([j], j.state == "cancelled")
|
||||||
|> where([j], j.args["op"] == "publish_one")
|
|> where([j], j.args["op"] == "publish_one")
|
||||||
|> where([j], j.args["params"]["id"] == ^ap_id)
|
|> where([j], j.args["params"]["activity_id"] == ^repeat_id)
|
||||||
|> Pleroma.Repo.all()
|
|> Pleroma.Repo.all()
|
||||||
|
|
||||||
assert length(cancelled_jobs) == 2
|
assert length(cancelled_jobs) == 2
|
||||||
|
@ -2142,11 +2142,11 @@ test "when unreacting to posts", %{
|
||||||
{:ok, _, _} = Pleroma.User.follow(remote_one, local_user)
|
{:ok, _, _} = Pleroma.User.follow(remote_one, local_user)
|
||||||
{:ok, _, _} = Pleroma.User.follow(remote_two, local_user)
|
{:ok, _, _} = Pleroma.User.follow(remote_two, local_user)
|
||||||
|
|
||||||
{:ok, activity} =
|
{:ok, %{id: activity_id}} =
|
||||||
CommonAPI.post(remote_one, %{status: "Gang gang!!!!"})
|
CommonAPI.post(remote_one, %{status: "Gang gang!!!!"})
|
||||||
|
|
||||||
{:ok, %{data: %{"id" => ap_id}} = _react} =
|
{:ok, %{id: react_id} = _react} =
|
||||||
CommonAPI.react_with_emoji(activity.id, local_user, "👍")
|
CommonAPI.react_with_emoji(activity_id, local_user, "👍")
|
||||||
|
|
||||||
# Generate the publish_one jobs
|
# Generate the publish_one jobs
|
||||||
ObanHelpers.perform_all()
|
ObanHelpers.perform_all()
|
||||||
|
@ -2159,7 +2159,7 @@ test "when unreacting to posts", %{
|
||||||
state: "available",
|
state: "available",
|
||||||
queue: "federator_outgoing",
|
queue: "federator_outgoing",
|
||||||
worker: "Pleroma.Workers.PublisherWorker",
|
worker: "Pleroma.Workers.PublisherWorker",
|
||||||
args: %{"op" => "publish_one", "params" => %{"id" => ^ap_id}}
|
args: %{"op" => "publish_one", "params" => %{"activity_id" => ^react_id}}
|
||||||
},
|
},
|
||||||
job
|
job
|
||||||
)
|
)
|
||||||
|
@ -2168,7 +2168,7 @@ test "when unreacting to posts", %{
|
||||||
assert length(publish_one_jobs) == 2
|
assert length(publish_one_jobs) == 2
|
||||||
|
|
||||||
# The unreact should have triggered cancelling the publish_one jobs
|
# The unreact should have triggered cancelling the publish_one jobs
|
||||||
assert {:ok, _unreact} = CommonAPI.unreact_with_emoji(activity.id, local_user, "👍")
|
assert {:ok, _unreact} = CommonAPI.unreact_with_emoji(activity_id, local_user, "👍")
|
||||||
|
|
||||||
# all_enqueued/1 will not return cancelled jobs
|
# all_enqueued/1 will not return cancelled jobs
|
||||||
cancelled_jobs =
|
cancelled_jobs =
|
||||||
|
@ -2176,7 +2176,7 @@ test "when unreacting to posts", %{
|
||||||
|> where([j], j.worker == "Pleroma.Workers.PublisherWorker")
|
|> where([j], j.worker == "Pleroma.Workers.PublisherWorker")
|
||||||
|> where([j], j.state == "cancelled")
|
|> where([j], j.state == "cancelled")
|
||||||
|> where([j], j.args["op"] == "publish_one")
|
|> where([j], j.args["op"] == "publish_one")
|
||||||
|> where([j], j.args["params"]["id"] == ^ap_id)
|
|> where([j], j.args["params"]["activity_id"] == ^react_id)
|
||||||
|> Pleroma.Repo.all()
|
|> Pleroma.Repo.all()
|
||||||
|
|
||||||
assert length(cancelled_jobs) == 2
|
assert length(cancelled_jobs) == 2
|
||||||
|
|
|
@ -20,9 +20,7 @@ defmodule Pleroma.Web.PleromaAPI.BackupControllerTest do
|
||||||
end
|
end
|
||||||
|
|
||||||
test "GET /api/v1/pleroma/backups", %{user: user, conn: conn} do
|
test "GET /api/v1/pleroma/backups", %{user: user, conn: conn} do
|
||||||
assert {:ok, %Oban.Job{args: %{"backup_id" => backup_id}}} = Backup.create(user)
|
assert {:ok, %Backup{} = backup} = Backup.user(user)
|
||||||
|
|
||||||
backup = Backup.get(backup_id)
|
|
||||||
|
|
||||||
response =
|
response =
|
||||||
conn
|
conn
|
||||||
|
|
|
@ -27,42 +27,11 @@ test "it renders the ID" do
|
||||||
assert result.id == backup.id
|
assert result.id == backup.id
|
||||||
end
|
end
|
||||||
|
|
||||||
test "it renders the state and processed_number" do
|
test "it renders the processed state" do
|
||||||
user = insert(:user)
|
user = insert(:user)
|
||||||
backup = Backup.new(user)
|
backup = Backup.new(user)
|
||||||
|
|
||||||
result = BackupView.render("show.json", backup: backup)
|
result = BackupView.render("show.json", backup: backup)
|
||||||
assert result.state == to_string(backup.state)
|
refute result.processed
|
||||||
assert result.processed_number == backup.processed_number
|
|
||||||
end
|
|
||||||
|
|
||||||
test "it renders failed state with legacy records" do
|
|
||||||
backup = %Backup{
|
|
||||||
id: 0,
|
|
||||||
content_type: "application/zip",
|
|
||||||
file_name: "dummy",
|
|
||||||
file_size: 1,
|
|
||||||
state: :invalid,
|
|
||||||
processed: true,
|
|
||||||
processed_number: 1,
|
|
||||||
inserted_at: NaiveDateTime.utc_now()
|
|
||||||
}
|
|
||||||
|
|
||||||
result = BackupView.render("show.json", backup: backup)
|
|
||||||
assert result.state == "complete"
|
|
||||||
|
|
||||||
backup = %Backup{
|
|
||||||
id: 0,
|
|
||||||
content_type: "application/zip",
|
|
||||||
file_name: "dummy",
|
|
||||||
file_size: 1,
|
|
||||||
state: :invalid,
|
|
||||||
processed: false,
|
|
||||||
processed_number: 1,
|
|
||||||
inserted_at: NaiveDateTime.utc_now()
|
|
||||||
}
|
|
||||||
|
|
||||||
result = BackupView.render("show.json", backup: backup)
|
|
||||||
assert result.state == "failed"
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -9,6 +9,7 @@ defmodule Pleroma.Workers.ReceiverWorkerTest do
|
||||||
import Mock
|
import Mock
|
||||||
import Pleroma.Factory
|
import Pleroma.Factory
|
||||||
|
|
||||||
|
alias Pleroma.Web.Federator
|
||||||
alias Pleroma.Workers.ReceiverWorker
|
alias Pleroma.Workers.ReceiverWorker
|
||||||
|
|
||||||
test "it does not retry MRF reject" do
|
test "it does not retry MRF reject" do
|
||||||
|
@ -49,4 +50,199 @@ test "it does not retry duplicates" do
|
||||||
args: %{"op" => "incoming_ap_doc", "params" => params}
|
args: %{"op" => "incoming_ap_doc", "params" => params}
|
||||||
})
|
})
|
||||||
end
|
end
|
||||||
|
|
||||||
|
test "it can validate the signature" do
|
||||||
|
Tesla.Mock.mock(fn
|
||||||
|
%{url: "https://mastodon.social/users/bastianallgeier"} ->
|
||||||
|
%Tesla.Env{
|
||||||
|
status: 200,
|
||||||
|
body: File.read!("test/fixtures/bastianallgeier.json"),
|
||||||
|
headers: [{"content-type", "application/activity+json"}]
|
||||||
|
}
|
||||||
|
|
||||||
|
%{url: "https://mastodon.social/users/bastianallgeier/collections/featured"} ->
|
||||||
|
%Tesla.Env{
|
||||||
|
status: 200,
|
||||||
|
headers: [{"content-type", "application/activity+json"}],
|
||||||
|
body:
|
||||||
|
File.read!("test/fixtures/users_mock/masto_featured.json")
|
||||||
|
|> String.replace("{{domain}}", "mastodon.social")
|
||||||
|
|> String.replace("{{nickname}}", "bastianallgeier")
|
||||||
|
}
|
||||||
|
|
||||||
|
%{url: "https://phpc.social/users/denniskoch"} ->
|
||||||
|
%Tesla.Env{
|
||||||
|
status: 200,
|
||||||
|
body: File.read!("test/fixtures/denniskoch.json"),
|
||||||
|
headers: [{"content-type", "application/activity+json"}]
|
||||||
|
}
|
||||||
|
|
||||||
|
%{url: "https://phpc.social/users/denniskoch/collections/featured"} ->
|
||||||
|
%Tesla.Env{
|
||||||
|
status: 200,
|
||||||
|
headers: [{"content-type", "application/activity+json"}],
|
||||||
|
body:
|
||||||
|
File.read!("test/fixtures/users_mock/masto_featured.json")
|
||||||
|
|> String.replace("{{domain}}", "phpc.social")
|
||||||
|
|> String.replace("{{nickname}}", "denniskoch")
|
||||||
|
}
|
||||||
|
|
||||||
|
%{url: "https://mastodon.social/users/bastianallgeier/statuses/112846516276907281"} ->
|
||||||
|
%Tesla.Env{
|
||||||
|
status: 200,
|
||||||
|
headers: [{"content-type", "application/activity+json"}],
|
||||||
|
body: File.read!("test/fixtures/receiver_worker_signature_activity.json")
|
||||||
|
}
|
||||||
|
end)
|
||||||
|
|
||||||
|
params = %{
|
||||||
|
"@context" => [
|
||||||
|
"https://www.w3.org/ns/activitystreams",
|
||||||
|
"https://w3id.org/security/v1",
|
||||||
|
%{
|
||||||
|
"claim" => %{"@id" => "toot:claim", "@type" => "@id"},
|
||||||
|
"memorial" => "toot:memorial",
|
||||||
|
"atomUri" => "ostatus:atomUri",
|
||||||
|
"manuallyApprovesFollowers" => "as:manuallyApprovesFollowers",
|
||||||
|
"blurhash" => "toot:blurhash",
|
||||||
|
"ostatus" => "http://ostatus.org#",
|
||||||
|
"discoverable" => "toot:discoverable",
|
||||||
|
"focalPoint" => %{"@container" => "@list", "@id" => "toot:focalPoint"},
|
||||||
|
"votersCount" => "toot:votersCount",
|
||||||
|
"Hashtag" => "as:Hashtag",
|
||||||
|
"Emoji" => "toot:Emoji",
|
||||||
|
"alsoKnownAs" => %{"@id" => "as:alsoKnownAs", "@type" => "@id"},
|
||||||
|
"sensitive" => "as:sensitive",
|
||||||
|
"movedTo" => %{"@id" => "as:movedTo", "@type" => "@id"},
|
||||||
|
"inReplyToAtomUri" => "ostatus:inReplyToAtomUri",
|
||||||
|
"conversation" => "ostatus:conversation",
|
||||||
|
"Device" => "toot:Device",
|
||||||
|
"schema" => "http://schema.org#",
|
||||||
|
"toot" => "http://joinmastodon.org/ns#",
|
||||||
|
"cipherText" => "toot:cipherText",
|
||||||
|
"suspended" => "toot:suspended",
|
||||||
|
"messageType" => "toot:messageType",
|
||||||
|
"featuredTags" => %{"@id" => "toot:featuredTags", "@type" => "@id"},
|
||||||
|
"Curve25519Key" => "toot:Curve25519Key",
|
||||||
|
"deviceId" => "toot:deviceId",
|
||||||
|
"Ed25519Signature" => "toot:Ed25519Signature",
|
||||||
|
"featured" => %{"@id" => "toot:featured", "@type" => "@id"},
|
||||||
|
"devices" => %{"@id" => "toot:devices", "@type" => "@id"},
|
||||||
|
"value" => "schema:value",
|
||||||
|
"PropertyValue" => "schema:PropertyValue",
|
||||||
|
"messageFranking" => "toot:messageFranking",
|
||||||
|
"publicKeyBase64" => "toot:publicKeyBase64",
|
||||||
|
"identityKey" => %{"@id" => "toot:identityKey", "@type" => "@id"},
|
||||||
|
"Ed25519Key" => "toot:Ed25519Key",
|
||||||
|
"indexable" => "toot:indexable",
|
||||||
|
"EncryptedMessage" => "toot:EncryptedMessage",
|
||||||
|
"fingerprintKey" => %{"@id" => "toot:fingerprintKey", "@type" => "@id"}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"actor" => "https://phpc.social/users/denniskoch",
|
||||||
|
"cc" => [
|
||||||
|
"https://phpc.social/users/denniskoch/followers",
|
||||||
|
"https://mastodon.social/users/bastianallgeier",
|
||||||
|
"https://chaos.social/users/distantnative",
|
||||||
|
"https://fosstodon.org/users/kev"
|
||||||
|
],
|
||||||
|
"id" => "https://phpc.social/users/denniskoch/statuses/112847382711461301/activity",
|
||||||
|
"object" => %{
|
||||||
|
"atomUri" => "https://phpc.social/users/denniskoch/statuses/112847382711461301",
|
||||||
|
"attachment" => [],
|
||||||
|
"attributedTo" => "https://phpc.social/users/denniskoch",
|
||||||
|
"cc" => [
|
||||||
|
"https://phpc.social/users/denniskoch/followers",
|
||||||
|
"https://mastodon.social/users/bastianallgeier",
|
||||||
|
"https://chaos.social/users/distantnative",
|
||||||
|
"https://fosstodon.org/users/kev"
|
||||||
|
],
|
||||||
|
"content" =>
|
||||||
|
"<p><span class=\"h-card\" translate=\"no\"><a href=\"https://mastodon.social/@bastianallgeier\" class=\"u-url mention\">@<span>bastianallgeier</span></a></span> <span class=\"h-card\" translate=\"no\"><a href=\"https://chaos.social/@distantnative\" class=\"u-url mention\">@<span>distantnative</span></a></span> <span class=\"h-card\" translate=\"no\"><a href=\"https://fosstodon.org/@kev\" class=\"u-url mention\">@<span>kev</span></a></span> Another main argument: Discord is popular. Many people have an account, so you can just join an server quickly. Also you know the app and how to get around.</p>",
|
||||||
|
"contentMap" => %{
|
||||||
|
"en" =>
|
||||||
|
"<p><span class=\"h-card\" translate=\"no\"><a href=\"https://mastodon.social/@bastianallgeier\" class=\"u-url mention\">@<span>bastianallgeier</span></a></span> <span class=\"h-card\" translate=\"no\"><a href=\"https://chaos.social/@distantnative\" class=\"u-url mention\">@<span>distantnative</span></a></span> <span class=\"h-card\" translate=\"no\"><a href=\"https://fosstodon.org/@kev\" class=\"u-url mention\">@<span>kev</span></a></span> Another main argument: Discord is popular. Many people have an account, so you can just join an server quickly. Also you know the app and how to get around.</p>"
|
||||||
|
},
|
||||||
|
"conversation" =>
|
||||||
|
"tag:mastodon.social,2024-07-25:objectId=760068442:objectType=Conversation",
|
||||||
|
"id" => "https://phpc.social/users/denniskoch/statuses/112847382711461301",
|
||||||
|
"inReplyTo" =>
|
||||||
|
"https://mastodon.social/users/bastianallgeier/statuses/112846516276907281",
|
||||||
|
"inReplyToAtomUri" =>
|
||||||
|
"https://mastodon.social/users/bastianallgeier/statuses/112846516276907281",
|
||||||
|
"published" => "2024-07-25T13:33:29Z",
|
||||||
|
"replies" => %{
|
||||||
|
"first" => %{
|
||||||
|
"items" => [],
|
||||||
|
"next" =>
|
||||||
|
"https://phpc.social/users/denniskoch/statuses/112847382711461301/replies?only_other_accounts=true&page=true",
|
||||||
|
"partOf" =>
|
||||||
|
"https://phpc.social/users/denniskoch/statuses/112847382711461301/replies",
|
||||||
|
"type" => "CollectionPage"
|
||||||
|
},
|
||||||
|
"id" => "https://phpc.social/users/denniskoch/statuses/112847382711461301/replies",
|
||||||
|
"type" => "Collection"
|
||||||
|
},
|
||||||
|
"sensitive" => false,
|
||||||
|
"tag" => [
|
||||||
|
%{
|
||||||
|
"href" => "https://mastodon.social/users/bastianallgeier",
|
||||||
|
"name" => "@bastianallgeier@mastodon.social",
|
||||||
|
"type" => "Mention"
|
||||||
|
},
|
||||||
|
%{
|
||||||
|
"href" => "https://chaos.social/users/distantnative",
|
||||||
|
"name" => "@distantnative@chaos.social",
|
||||||
|
"type" => "Mention"
|
||||||
|
},
|
||||||
|
%{
|
||||||
|
"href" => "https://fosstodon.org/users/kev",
|
||||||
|
"name" => "@kev@fosstodon.org",
|
||||||
|
"type" => "Mention"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"to" => ["https://www.w3.org/ns/activitystreams#Public"],
|
||||||
|
"type" => "Note",
|
||||||
|
"url" => "https://phpc.social/@denniskoch/112847382711461301"
|
||||||
|
},
|
||||||
|
"published" => "2024-07-25T13:33:29Z",
|
||||||
|
"signature" => %{
|
||||||
|
"created" => "2024-07-25T13:33:29Z",
|
||||||
|
"creator" => "https://phpc.social/users/denniskoch#main-key",
|
||||||
|
"signatureValue" =>
|
||||||
|
"slz9BKJzd2n1S44wdXGOU+bV/wsskdgAaUpwxj8R16mYOL8+DTpE6VnfSKoZGsBBJT8uG5gnVfVEz1YsTUYtymeUgLMh7cvd8VnJnZPS+oixbmBRVky/Myf91TEgQQE7G4vDmTdB4ii54hZrHcOOYYf5FKPNRSkMXboKA6LMqNtekhbI+JTUJYIB02WBBK6PUyo15f6B1RJ6HGWVgud9NE0y1EZXfrkqUt682p8/9D49ORf7AwjXUJibKic2RbPvhEBj70qUGfBm4vvgdWhSUn1IG46xh+U0+NrTSUED82j1ZVOeua/2k/igkGs8cSBkY35quXTkPz6gbqCCH66CuA==",
|
||||||
|
"type" => "RsaSignature2017"
|
||||||
|
},
|
||||||
|
"to" => ["https://www.w3.org/ns/activitystreams#Public"],
|
||||||
|
"type" => "Create"
|
||||||
|
}
|
||||||
|
|
||||||
|
req_headers = [
|
||||||
|
["accept-encoding", "gzip"],
|
||||||
|
["content-length", "5184"],
|
||||||
|
["content-type", "application/activity+json"],
|
||||||
|
["date", "Thu, 25 Jul 2024 13:33:31 GMT"],
|
||||||
|
["digest", "SHA-256=ouge/6HP2/QryG6F3JNtZ6vzs/hSwMk67xdxe87eH7A="],
|
||||||
|
["host", "bikeshed.party"],
|
||||||
|
[
|
||||||
|
"signature",
|
||||||
|
"keyId=\"https://mastodon.social/users/bastianallgeier#main-key\",algorithm=\"rsa-sha256\",headers=\"(request-target) host date digest content-type\",signature=\"ymE3vn5Iw50N6ukSp8oIuXJB5SBjGAGjBasdTDvn+ahZIzq2SIJfmVCsIIzyqIROnhWyQoTbavTclVojEqdaeOx+Ejz2wBnRBmhz5oemJLk4RnnCH0lwMWyzeY98YAvxi9Rq57Gojuv/1lBqyGa+rDzynyJpAMyFk17XIZpjMKuTNMCbjMDy76ILHqArykAIL/v1zxkgwxY/+ELzxqMpNqtZ+kQ29znNMUBB3eVZ/mNAHAz6o33Y9VKxM2jw+08vtuIZOusXyiHbRiaj2g5HtN2WBUw1MzzfRfHF2/yy7rcipobeoyk5RvP5SyHV3WrIeZ3iyoNfmv33y8fxllF0EA==\""
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"user-agent",
|
||||||
|
"http.rb/5.2.0 (Mastodon/4.3.0-nightly.2024-07-25; +https://mastodon.social/)"
|
||||||
|
]
|
||||||
|
]
|
||||||
|
|
||||||
|
{:ok, oban_job} =
|
||||||
|
Federator.incoming_ap_doc(%{
|
||||||
|
method: "POST",
|
||||||
|
req_headers: req_headers,
|
||||||
|
request_path: "/inbox",
|
||||||
|
params: params,
|
||||||
|
query_string: ""
|
||||||
|
})
|
||||||
|
|
||||||
|
assert {:ok, %Pleroma.Activity{}} = ReceiverWorker.perform(oban_job)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -32,6 +32,4 @@
|
||||||
|
|
||||||
Mox.defmock(Pleroma.LoggerMock, for: Pleroma.Logging)
|
Mox.defmock(Pleroma.LoggerMock, for: Pleroma.Logging)
|
||||||
|
|
||||||
Mox.defmock(Pleroma.User.Backup.ProcessorMock, for: Pleroma.User.Backup.ProcessorAPI)
|
|
||||||
|
|
||||||
Mox.defmock(Pleroma.Uploaders.S3.ExAwsMock, for: Pleroma.Uploaders.S3.ExAwsAPI)
|
Mox.defmock(Pleroma.Uploaders.S3.ExAwsMock, for: Pleroma.Uploaders.S3.ExAwsAPI)
|
||||||
|
|
Loading…
Reference in a new issue