Merge branch '114_email_invites' into 'develop'
[#114] Email invites See merge request pleroma/pleroma!541
This commit is contained in:
commit
bfe27c1b55
|
@ -67,7 +67,8 @@ config :pleroma, Pleroma.Mailer,
|
||||||
* `avatar_upload_limit`: File size limit of user’s profile avatars
|
* `avatar_upload_limit`: File size limit of user’s profile avatars
|
||||||
* `background_upload_limit`: File size limit of user’s profile backgrounds
|
* `background_upload_limit`: File size limit of user’s profile backgrounds
|
||||||
* `banner_upload_limit`: File size limit of user’s profile banners
|
* `banner_upload_limit`: File size limit of user’s profile banners
|
||||||
* `registrations_open`: Enable registrations for anyone, invitations can be used when false.
|
* `registrations_open`: Enable registrations for anyone, invitations can be enabled when false.
|
||||||
|
* `invites_enabled`: Enable user invitations for admins (depends on `registrations_open: false`).
|
||||||
* `federating`: Enable federation with other instances
|
* `federating`: Enable federation with other instances
|
||||||
* `allow_relay`: Enable Pleroma’s Relay, which makes it possible to follow a whole instance
|
* `allow_relay`: Enable Pleroma’s Relay, which makes it possible to follow a whole instance
|
||||||
* `rewrite_policy`: Message Rewrite Policy, either one or a list. Here are the ones available by default:
|
* `rewrite_policy`: Message Rewrite Policy, either one or a list. Here are the ones available by default:
|
||||||
|
|
|
@ -37,4 +37,30 @@ def password_reset_email(user, password_reset_token) when is_binary(password_res
|
||||||
|> subject("Password reset")
|
|> subject("Password reset")
|
||||||
|> html_body(html_body)
|
|> html_body(html_body)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def user_invitation_email(
|
||||||
|
user,
|
||||||
|
%Pleroma.UserInviteToken{} = user_invite_token,
|
||||||
|
to_email,
|
||||||
|
to_name \\ nil
|
||||||
|
) do
|
||||||
|
registration_url =
|
||||||
|
Router.Helpers.redirect_url(
|
||||||
|
Endpoint,
|
||||||
|
:registration_page,
|
||||||
|
user_invite_token.token
|
||||||
|
)
|
||||||
|
|
||||||
|
html_body = """
|
||||||
|
<h3>You are invited to #{instance_name()}</h3>
|
||||||
|
<p>#{user.name} invites you to join #{instance_name()}, an instance of Pleroma federated social networking platform.</p>
|
||||||
|
<p>Click the following link to register: <a href="#{registration_url}">accept invitation</a>.</p>
|
||||||
|
"""
|
||||||
|
|
||||||
|
new()
|
||||||
|
|> to(recipient(to_email, to_name))
|
||||||
|
|> from(sender())
|
||||||
|
|> subject("Invitation to #{instance_name()}")
|
||||||
|
|> html_body(html_body)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -147,6 +147,19 @@ def relay_unfollow(conn, %{"relay_url" => target}) do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@doc "Sends registration invite via email"
|
||||||
|
def email_invite(%{assigns: %{user: user}} = conn, %{"email" => email} = params) do
|
||||||
|
with true <-
|
||||||
|
Pleroma.Config.get([:instance, :invites_enabled]) &&
|
||||||
|
!Pleroma.Config.get([:instance, :registrations_open]),
|
||||||
|
{:ok, invite_token} <- Pleroma.UserInviteToken.create_token(),
|
||||||
|
email <-
|
||||||
|
Pleroma.UserEmail.user_invitation_email(user, invite_token, email, params["name"]),
|
||||||
|
{:ok, _} <- Pleroma.Mailer.deliver(email) do
|
||||||
|
json_response(conn, :no_content, "")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
@doc "Get a account registeration invite token (base64 string)"
|
@doc "Get a account registeration invite token (base64 string)"
|
||||||
def get_invite_token(conn, _params) do
|
def get_invite_token(conn, _params) do
|
||||||
{:ok, token} = Pleroma.UserInviteToken.create_token()
|
{:ok, token} = Pleroma.UserInviteToken.create_token()
|
||||||
|
|
|
@ -132,6 +132,7 @@ def nodeinfo(conn, %{"version" => "2.0"}) do
|
||||||
banner: Keyword.get(instance, :banner_upload_limit),
|
banner: Keyword.get(instance, :banner_upload_limit),
|
||||||
background: Keyword.get(instance, :background_upload_limit)
|
background: Keyword.get(instance, :background_upload_limit)
|
||||||
},
|
},
|
||||||
|
invitesEnabled: Keyword.get(instance, :invites_enabled, false),
|
||||||
features: features
|
features: features
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -117,6 +117,8 @@ defmodule Pleroma.Web.Router do
|
||||||
delete("/relay", AdminAPIController, :relay_unfollow)
|
delete("/relay", AdminAPIController, :relay_unfollow)
|
||||||
|
|
||||||
get("/invite_token", AdminAPIController, :get_invite_token)
|
get("/invite_token", AdminAPIController, :get_invite_token)
|
||||||
|
post("/email_invite", AdminAPIController, :email_invite)
|
||||||
|
|
||||||
get("/password_reset", AdminAPIController, :get_password_reset)
|
get("/password_reset", AdminAPIController, :get_password_reset)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -173,7 +173,8 @@ def config(conn, _params) do
|
||||||
uploadlimit: uploadlimit,
|
uploadlimit: uploadlimit,
|
||||||
closed: if(Keyword.get(instance, :registrations_open), do: "0", else: "1"),
|
closed: if(Keyword.get(instance, :registrations_open), do: "0", else: "1"),
|
||||||
private: if(Keyword.get(instance, :public, true), do: "0", else: "1"),
|
private: if(Keyword.get(instance, :public, true), do: "0", else: "1"),
|
||||||
vapidPublicKey: vapid_public_key
|
vapidPublicKey: vapid_public_key,
|
||||||
|
invitesEnabled: if(Keyword.get(instance, :invites_enabled, false), do: "1", else: "0")
|
||||||
}
|
}
|
||||||
|
|
||||||
pleroma_fe = %{
|
pleroma_fe = %{
|
||||||
|
|
|
@ -154,6 +154,105 @@ test "/:right DELETE, can remove from a permission group" do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
describe "POST /api/pleroma/admin/email_invite, with valid config" do
|
||||||
|
setup do
|
||||||
|
registrations_open = Pleroma.Config.get([:instance, :registrations_open])
|
||||||
|
invites_enabled = Pleroma.Config.get([:instance, :invites_enabled])
|
||||||
|
Pleroma.Config.put([:instance, :registrations_open], false)
|
||||||
|
Pleroma.Config.put([:instance, :invites_enabled], true)
|
||||||
|
|
||||||
|
on_exit(fn ->
|
||||||
|
Pleroma.Config.put([:instance, :registrations_open], registrations_open)
|
||||||
|
Pleroma.Config.put([:instance, :invites_enabled], invites_enabled)
|
||||||
|
:ok
|
||||||
|
end)
|
||||||
|
|
||||||
|
[user: insert(:user, info: %{is_admin: true})]
|
||||||
|
end
|
||||||
|
|
||||||
|
test "sends invitation and returns 204", %{conn: conn, user: user} do
|
||||||
|
recipient_email = "foo@bar.com"
|
||||||
|
recipient_name = "J. D."
|
||||||
|
|
||||||
|
conn =
|
||||||
|
conn
|
||||||
|
|> assign(:user, user)
|
||||||
|
|> post("/api/pleroma/admin/email_invite?email=#{recipient_email}&name=#{recipient_name}")
|
||||||
|
|
||||||
|
assert json_response(conn, :no_content)
|
||||||
|
|
||||||
|
token_record = List.last(Pleroma.Repo.all(Pleroma.UserInviteToken))
|
||||||
|
assert token_record
|
||||||
|
refute token_record.used
|
||||||
|
|
||||||
|
Swoosh.TestAssertions.assert_email_sent(
|
||||||
|
Pleroma.UserEmail.user_invitation_email(
|
||||||
|
user,
|
||||||
|
token_record,
|
||||||
|
recipient_email,
|
||||||
|
recipient_name
|
||||||
|
)
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
test "it returns 403 if requested by a non-admin", %{conn: conn} do
|
||||||
|
non_admin_user = insert(:user)
|
||||||
|
|
||||||
|
conn =
|
||||||
|
conn
|
||||||
|
|> assign(:user, non_admin_user)
|
||||||
|
|> post("/api/pleroma/admin/email_invite?email=foo@bar.com&name=JD")
|
||||||
|
|
||||||
|
assert json_response(conn, :forbidden)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe "POST /api/pleroma/admin/email_invite, with invalid config" do
|
||||||
|
setup do
|
||||||
|
[user: insert(:user, info: %{is_admin: true})]
|
||||||
|
end
|
||||||
|
|
||||||
|
test "it returns 500 if `invites_enabled` is not enabled", %{conn: conn, user: user} do
|
||||||
|
registrations_open = Pleroma.Config.get([:instance, :registrations_open])
|
||||||
|
invites_enabled = Pleroma.Config.get([:instance, :invites_enabled])
|
||||||
|
Pleroma.Config.put([:instance, :registrations_open], false)
|
||||||
|
Pleroma.Config.put([:instance, :invites_enabled], false)
|
||||||
|
|
||||||
|
on_exit(fn ->
|
||||||
|
Pleroma.Config.put([:instance, :registrations_open], registrations_open)
|
||||||
|
Pleroma.Config.put([:instance, :invites_enabled], invites_enabled)
|
||||||
|
:ok
|
||||||
|
end)
|
||||||
|
|
||||||
|
conn =
|
||||||
|
conn
|
||||||
|
|> assign(:user, user)
|
||||||
|
|> post("/api/pleroma/admin/email_invite?email=foo@bar.com&name=JD")
|
||||||
|
|
||||||
|
assert json_response(conn, :internal_server_error)
|
||||||
|
end
|
||||||
|
|
||||||
|
test "it returns 500 if `registrations_open` is enabled", %{conn: conn, user: user} do
|
||||||
|
registrations_open = Pleroma.Config.get([:instance, :registrations_open])
|
||||||
|
invites_enabled = Pleroma.Config.get([:instance, :invites_enabled])
|
||||||
|
Pleroma.Config.put([:instance, :registrations_open], true)
|
||||||
|
Pleroma.Config.put([:instance, :invites_enabled], true)
|
||||||
|
|
||||||
|
on_exit(fn ->
|
||||||
|
Pleroma.Config.put([:instance, :registrations_open], registrations_open)
|
||||||
|
Pleroma.Config.put([:instance, :invites_enabled], invites_enabled)
|
||||||
|
:ok
|
||||||
|
end)
|
||||||
|
|
||||||
|
conn =
|
||||||
|
conn
|
||||||
|
|> assign(:user, user)
|
||||||
|
|> post("/api/pleroma/admin/email_invite?email=foo@bar.com&name=JD")
|
||||||
|
|
||||||
|
assert json_response(conn, :internal_server_error)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
test "/api/pleroma/admin/invite_token" do
|
test "/api/pleroma/admin/invite_token" do
|
||||||
admin = insert(:user, info: %{is_admin: true})
|
admin = insert(:user, info: %{is_admin: true})
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue