logo

pleroma

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

remote_follow_controller.ex (5625B)


  1. # Pleroma: A lightweight social networking server
  2. # Copyright © 2017-2022 Pleroma Authors <https://pleroma.social/>
  3. # SPDX-License-Identifier: AGPL-3.0-only
  4. defmodule Pleroma.Web.TwitterAPI.RemoteFollowController do
  5. use Pleroma.Web, :controller
  6. require Logger
  7. alias Pleroma.Activity
  8. alias Pleroma.MFA
  9. alias Pleroma.Object.Fetcher
  10. alias Pleroma.User
  11. alias Pleroma.Web.Auth.TOTPAuthenticator
  12. alias Pleroma.Web.Auth.WrapperAuthenticator
  13. alias Pleroma.Web.CommonAPI
  14. @status_types ["Article", "Event", "Note", "Video", "Page", "Question"]
  15. plug(Pleroma.Web.Plugs.FederatingPlug)
  16. # Note: follower can submit the form (with password auth) not being signed in (having no token)
  17. plug(
  18. Pleroma.Web.Plugs.OAuthScopesPlug,
  19. %{fallback: :proceed_unauthenticated, scopes: ["follow", "write:follows"]}
  20. when action in [:do_follow]
  21. )
  22. # GET /ostatus_subscribe
  23. #
  24. def follow(%{assigns: %{user: user}} = conn, %{"acct" => acct}) do
  25. case status?(acct) do
  26. true -> follow_status(conn, user, acct)
  27. _ -> follow_account(conn, user, acct)
  28. end
  29. end
  30. defp follow_status(conn, _user, acct) do
  31. with {:ok, object} <- Fetcher.fetch_object_from_id(acct),
  32. %Activity{id: activity_id} <- Activity.get_create_by_object_ap_id(object.data["id"]) do
  33. redirect(conn, to: Routes.o_status_path(conn, :notice, activity_id))
  34. else
  35. error ->
  36. handle_follow_error(conn, error)
  37. end
  38. end
  39. defp follow_account(conn, user, acct) do
  40. with {:ok, followee} <- User.get_or_fetch(acct) do
  41. render(conn, follow_template(user), %{error: false, followee: followee, acct: acct})
  42. else
  43. {:error, _reason} ->
  44. render(conn, follow_template(user), %{error: :error})
  45. end
  46. end
  47. defp follow_template(%User{} = _user), do: "follow.html"
  48. defp follow_template(_), do: "follow_login.html"
  49. defp status?(acct) do
  50. case Fetcher.fetch_and_contain_remote_object_from_id(acct) do
  51. {:ok, %{"type" => type}} when type in @status_types ->
  52. true
  53. _ ->
  54. false
  55. end
  56. end
  57. # POST /ostatus_subscribe
  58. #
  59. # adds a remote account in followers if user already is signed in.
  60. #
  61. def do_follow(%{assigns: %{user: %User{} = user}} = conn, %{"user" => %{"id" => id}}) do
  62. with {:fetch_user, %User{} = followee} <- {:fetch_user, User.get_cached_by_id(id)},
  63. {:ok, _, _, _} <- CommonAPI.follow(user, followee) do
  64. redirect(conn, to: "/users/#{followee.id}")
  65. else
  66. error ->
  67. handle_follow_error(conn, error)
  68. end
  69. end
  70. # POST /ostatus_subscribe
  71. #
  72. # step 1.
  73. # checks login\password and displays step 2 form of MFA if need.
  74. #
  75. def do_follow(conn, %{"authorization" => %{"name" => _, "password" => _, "id" => id}}) do
  76. with {_, %User{} = followee} <- {:fetch_user, User.get_cached_by_id(id)},
  77. {_, {:ok, user}, _} <- {:auth, WrapperAuthenticator.get_user(conn), followee},
  78. {_, _, _, false} <- {:mfa_required, followee, user, MFA.require?(user)},
  79. {:ok, _, _, _} <- CommonAPI.follow(user, followee) do
  80. redirect(conn, to: "/users/#{followee.id}")
  81. else
  82. error ->
  83. handle_follow_error(conn, error)
  84. end
  85. end
  86. # POST /ostatus_subscribe
  87. #
  88. # step 2
  89. # checks TOTP code. otherwise displays form with errors
  90. #
  91. def do_follow(conn, %{"mfa" => %{"code" => code, "token" => token, "id" => id}}) do
  92. with {_, %User{} = followee} <- {:fetch_user, User.get_cached_by_id(id)},
  93. {_, _, {:ok, %{user: user}}} <- {:mfa_token, followee, MFA.Token.validate(token)},
  94. {_, _, _, {:ok, _}} <-
  95. {:verify_mfa_code, followee, token, TOTPAuthenticator.verify(code, user)},
  96. {:ok, _, _, _} <- CommonAPI.follow(user, followee) do
  97. redirect(conn, to: "/users/#{followee.id}")
  98. else
  99. error ->
  100. handle_follow_error(conn, error)
  101. end
  102. end
  103. def do_follow(%{assigns: %{user: nil}} = conn, _) do
  104. Logger.debug("Insufficient permissions: follow | write:follows.")
  105. render(conn, "followed.html", %{error: "Insufficient permissions: follow | write:follows."})
  106. end
  107. # GET /authorize_interaction
  108. #
  109. def authorize_interaction(conn, %{"uri" => uri}) do
  110. conn
  111. |> redirect(to: Routes.remote_follow_path(conn, :follow, %{acct: uri}))
  112. end
  113. defp handle_follow_error(conn, {:mfa_token, followee, _} = _) do
  114. render(conn, "follow_login.html", %{error: "Wrong username or password", followee: followee})
  115. end
  116. defp handle_follow_error(conn, {:verify_mfa_code, followee, token, _} = _) do
  117. render(conn, "follow_mfa.html", %{
  118. error: "Wrong authentication code",
  119. followee: followee,
  120. mfa_token: token
  121. })
  122. end
  123. defp handle_follow_error(conn, {:mfa_required, followee, user, _} = _) do
  124. {:ok, %{token: token}} = MFA.Token.create(user)
  125. render(conn, "follow_mfa.html", %{followee: followee, mfa_token: token, error: false})
  126. end
  127. defp handle_follow_error(conn, {:auth, _, followee} = _) do
  128. render(conn, "follow_login.html", %{error: "Wrong username or password", followee: followee})
  129. end
  130. defp handle_follow_error(conn, {:fetch_user, error} = _) do
  131. Logger.debug("Remote follow failed with error #{inspect(error)}")
  132. render(conn, "followed.html", %{error: "Could not find user"})
  133. end
  134. defp handle_follow_error(conn, {:error, "Could not follow user:" <> _} = _) do
  135. render(conn, "followed.html", %{error: "Error following account"})
  136. end
  137. defp handle_follow_error(conn, error) do
  138. Logger.debug("Remote follow failed with error #{inspect(error)}")
  139. render(conn, "followed.html", %{error: "Something went wrong."})
  140. end
  141. end