analytics/lib/plausible_web/controllers/billing_controller.ex
Vinicius Brasil e4230db2d9
Improve subscription status checking code (#3598)
* Improve subscription status checking code

This commit improves the subscription status checking in code, and
creates convinience functions to work with it, including nil-checking
and in?/2 function.

* Change in?/2 to macro
2023-12-07 09:05:07 -03:00

226 lines
7.3 KiB
Elixir

defmodule PlausibleWeb.BillingController do
use PlausibleWeb, :controller
use Plausible.Repo
require Logger
require Plausible.Billing.Subscription.Status
alias Plausible.Billing
alias Plausible.Billing.{Plans, Subscription}
plug PlausibleWeb.RequireAccountPlug
def ping_subscription(%Plug.Conn{} = conn, _params) do
subscribed? = Billing.has_active_subscription?(conn.assigns.current_user.id)
json(conn, %{is_subscribed: subscribed?})
end
def upgrade(conn, _params) do
user = conn.assigns[:current_user]
cond do
Plausible.Auth.enterprise_configured?(user) ->
redirect(conn, to: Routes.billing_path(conn, :upgrade_to_enterprise_plan))
FunWithFlags.enabled?(:business_tier, for: user) ->
redirect(conn, to: Routes.billing_path(conn, :choose_plan))
Subscription.Status.active?(user.subscription) ->
redirect(conn, to: Routes.billing_path(conn, :change_plan_form))
true ->
render(conn, "upgrade.html",
skip_plausible_tracking: true,
usage: Plausible.Billing.Quota.usage_cycle(user, :last_30_days).total,
user: user,
layout: {PlausibleWeb.LayoutView, "focus.html"}
)
end
end
def choose_plan(conn, _params) do
user = conn.assigns.current_user
if FunWithFlags.enabled?(:business_tier, for: user) do
if Plausible.Auth.enterprise_configured?(user) do
redirect(conn, to: Routes.billing_path(conn, :upgrade_to_enterprise_plan))
else
render(conn, "choose_plan.html",
skip_plausible_tracking: true,
user: user,
layout: {PlausibleWeb.LayoutView, "focus.html"},
connect_live_socket: true
)
end
else
# This will be needed in case we need to flip back the flag.
# With the :business_tier flag enabled we'll have sent emails
# linking to `/billing/choose-plan`.
redirect(conn, to: Routes.billing_path(conn, :upgrade))
end
end
def upgrade_to_enterprise_plan(conn, _params) do
user = Plausible.Users.with_subscription(conn.assigns.current_user)
{latest_enterprise_plan, price} = Plans.latest_enterprise_plan_with_price(user)
subscription_resumable? = Plausible.Billing.Subscriptions.resumable?(user.subscription)
subscribed_to_latest? =
subscription_resumable? &&
user.subscription.paddle_plan_id == latest_enterprise_plan.paddle_plan_id
cond do
Subscription.Status.in?(user.subscription, [
Subscription.Status.past_due(),
Subscription.Status.paused()
]) ->
redirect(conn, to: Routes.auth_path(conn, :user_settings))
subscribed_to_latest? ->
render(conn, "change_enterprise_plan_contact_us.html",
skip_plausible_tracking: true,
layout: {PlausibleWeb.LayoutView, "focus.html"}
)
true ->
render(conn, "upgrade_to_enterprise_plan.html",
user: user,
latest_enterprise_plan: latest_enterprise_plan,
price: price,
subscription_resumable: subscription_resumable?,
contact_link: "https://plausible.io/contact",
skip_plausible_tracking: true,
layout: {PlausibleWeb.LayoutView, "focus.html"}
)
end
end
def upgrade_enterprise_plan(conn, _params) do
# DEPRECATED - For some time we need to ensure that the existing
# links sent out to customers will lead the user to the right place
redirect(conn, to: Routes.billing_path(conn, :upgrade_to_enterprise_plan))
end
def upgrade_success(conn, _params) do
render(conn, "upgrade_success.html", layout: {PlausibleWeb.LayoutView, "focus.html"})
end
def change_plan_form(conn, _params) do
user = conn.assigns[:current_user]
subscription = Billing.active_subscription_for(user.id)
cond do
FunWithFlags.enabled?(:business_tier, for: user) ->
render_error(conn, 404)
Plausible.Auth.enterprise_configured?(user) ->
redirect(conn, to: Routes.billing_path(conn, :upgrade_to_enterprise_plan))
subscription ->
render(conn, "change_plan.html",
skip_plausible_tracking: true,
subscription: subscription,
layout: {PlausibleWeb.LayoutView, "focus.html"}
)
true ->
redirect(conn, to: Routes.billing_path(conn, :upgrade))
end
end
def change_enterprise_plan(conn, _params) do
# DEPRECATED - For some time we need to ensure that the existing
# links sent out to customers will lead the user to the right place
redirect(conn, to: Routes.billing_path(conn, :upgrade_to_enterprise_plan))
end
def change_plan_preview(conn, %{"plan_id" => new_plan_id}) do
user = conn.assigns.current_user
business_tier_enabled? = FunWithFlags.enabled?(:business_tier, for: user)
with {:ok, {subscription, preview_info}} <- preview_subscription(user, new_plan_id) do
back_action = if business_tier_enabled?, do: :choose_plan, else: :change_plan_form
render(conn, "change_plan_preview.html",
back_link: Routes.billing_path(conn, back_action),
skip_plausible_tracking: true,
subscription: subscription,
preview_info: preview_info,
layout: {PlausibleWeb.LayoutView, "focus.html"}
)
else
_ ->
msg =
"Something went wrong with loading your plan change information. Please try again, or contact us at support@plausible.io if the issue persists."
Sentry.capture_message("Error loading change plan preview",
extra: %{
message: msg,
new_plan_id: new_plan_id,
user_id: user.id
}
)
conn
|> put_flash(:error, msg)
|> redirect(to: Plausible.Billing.upgrade_route_for(user))
end
end
def change_plan(conn, %{"new_plan_id" => new_plan_id}) do
case Billing.change_plan(conn.assigns.current_user, new_plan_id) do
{:ok, _subscription} ->
conn
|> put_flash(:success, "Plan changed successfully")
|> redirect(to: "/settings")
{:error, e} ->
msg =
case e do
%{exceeded_limits: exceeded_limits} ->
"Unable to subscribe to this plan because the following limits are exceeded: #{inspect(exceeded_limits)}"
%{"code" => 147} ->
# https://developer.paddle.com/api-reference/intro/api-error-codes
"We were unable to charge your card. Click 'update billing info' to update your payment details and try again."
%{"message" => msg} when not is_nil(msg) ->
msg
_ ->
"Something went wrong. Please try again or contact support at support@plausible.io"
end
Sentry.capture_message("Error changing plans",
extra: %{
errors: inspect(e),
message: msg,
new_plan_id: new_plan_id,
user_id: conn.assigns[:current_user].id
}
)
conn
|> put_flash(:error, msg)
|> redirect(to: "/settings")
end
end
defp preview_subscription(%{id: user_id}, new_plan_id) do
subscription = Billing.active_subscription_for(user_id)
if subscription do
with {:ok, preview_info} <- Billing.change_plan_preview(subscription, new_plan_id) do
{:ok, {subscription, preview_info}}
end
else
{:error, :no_subscription}
end
end
def preview_susbcription(_, _) do
{:error, :no_user_id}
end
end