mirror of https://github.com/zulip/zulip.git
371 lines
13 KiB
Python
371 lines
13 KiB
Python
import logging
|
|
from decimal import Decimal
|
|
from typing import Any, Dict, Optional
|
|
|
|
import stripe
|
|
from django import forms
|
|
from django.conf import settings
|
|
from django.core import signing
|
|
from django.db import transaction
|
|
from django.http import HttpRequest, HttpResponse, HttpResponseRedirect
|
|
from django.shortcuts import render
|
|
from django.urls import reverse
|
|
|
|
from corporate.lib.stripe import (
|
|
DEFAULT_INVOICE_DAYS_UNTIL_DUE,
|
|
MIN_INVOICED_LICENSES,
|
|
BillingError,
|
|
compute_plan_parameters,
|
|
ensure_realm_does_not_have_active_plan,
|
|
get_latest_seat_count,
|
|
is_free_trial_offer_enabled,
|
|
is_sponsored_realm,
|
|
process_initial_upgrade,
|
|
sign_string,
|
|
unsign_string,
|
|
update_or_create_stripe_customer,
|
|
update_sponsorship_status,
|
|
validate_licenses,
|
|
)
|
|
from corporate.lib.support import get_support_url
|
|
from corporate.models import (
|
|
CustomerPlan,
|
|
PaymentIntent,
|
|
Session,
|
|
ZulipSponsorshipRequest,
|
|
get_current_plan_by_customer,
|
|
get_customer_by_realm,
|
|
)
|
|
from corporate.views.billing_page import billing_home
|
|
from zerver.actions.users import do_make_user_billing_admin
|
|
from zerver.decorator import require_organization_member, zulip_login_required
|
|
from zerver.lib.request import REQ, has_request_variables
|
|
from zerver.lib.response import json_success
|
|
from zerver.lib.send_email import FromAddress, send_email
|
|
from zerver.lib.validator import check_bool, check_int, check_string_in
|
|
from zerver.models import Realm, UserProfile, get_org_type_display_name
|
|
|
|
billing_logger = logging.getLogger("corporate.stripe")
|
|
|
|
VALID_BILLING_MODALITY_VALUES = ["send_invoice", "charge_automatically"]
|
|
VALID_BILLING_SCHEDULE_VALUES = ["annual", "monthly"]
|
|
VALID_LICENSE_MANAGEMENT_VALUES = ["automatic", "manual"]
|
|
|
|
|
|
def unsign_seat_count(signed_seat_count: str, salt: str) -> int:
|
|
try:
|
|
return int(unsign_string(signed_seat_count, salt))
|
|
except signing.BadSignature:
|
|
raise BillingError("tampered seat count")
|
|
|
|
|
|
def check_upgrade_parameters(
|
|
billing_modality: str,
|
|
schedule: str,
|
|
license_management: Optional[str],
|
|
licenses: Optional[int],
|
|
seat_count: int,
|
|
) -> None:
|
|
if billing_modality not in VALID_BILLING_MODALITY_VALUES: # nocoverage
|
|
raise BillingError("unknown billing_modality", "")
|
|
if schedule not in VALID_BILLING_SCHEDULE_VALUES: # nocoverage
|
|
raise BillingError("unknown schedule")
|
|
if license_management not in VALID_LICENSE_MANAGEMENT_VALUES: # nocoverage
|
|
raise BillingError("unknown license_management")
|
|
validate_licenses(billing_modality == "charge_automatically", licenses, seat_count)
|
|
|
|
|
|
def setup_upgrade_checkout_session_and_payment_intent(
|
|
user: UserProfile,
|
|
seat_count: int,
|
|
licenses: int,
|
|
license_management: str,
|
|
billing_schedule: int,
|
|
billing_modality: str,
|
|
onboarding: bool,
|
|
) -> stripe.checkout.Session:
|
|
customer = update_or_create_stripe_customer(user)
|
|
assert customer is not None # for mypy
|
|
free_trial = is_free_trial_offer_enabled()
|
|
_, _, _, price_per_license = compute_plan_parameters(
|
|
CustomerPlan.STANDARD,
|
|
license_management == "automatic",
|
|
billing_schedule,
|
|
customer.default_discount,
|
|
free_trial,
|
|
)
|
|
metadata = {
|
|
"billing_modality": billing_modality,
|
|
"billing_schedule": billing_schedule,
|
|
"licenses": licenses,
|
|
"license_management": license_management,
|
|
"price_per_license": price_per_license,
|
|
"seat_count": seat_count,
|
|
"type": "upgrade",
|
|
"user_email": user.delivery_email,
|
|
"realm_id": user.realm.id,
|
|
"realm_str": user.realm.string_id,
|
|
"user_id": user.id,
|
|
}
|
|
if free_trial:
|
|
if onboarding:
|
|
session_type = Session.FREE_TRIAL_UPGRADE_FROM_ONBOARDING_PAGE
|
|
else:
|
|
session_type = Session.FREE_TRIAL_UPGRADE_FROM_BILLING_PAGE
|
|
payment_intent = None
|
|
else:
|
|
session_type = Session.UPGRADE_FROM_BILLING_PAGE
|
|
stripe_payment_intent = stripe.PaymentIntent.create(
|
|
amount=price_per_license * licenses,
|
|
currency="usd",
|
|
customer=customer.stripe_customer_id,
|
|
description=f"Upgrade to Zulip Cloud Standard, ${price_per_license/100} x {licenses}",
|
|
receipt_email=user.delivery_email,
|
|
confirm=False,
|
|
statement_descriptor="Zulip Cloud Standard",
|
|
metadata=metadata,
|
|
)
|
|
payment_intent = PaymentIntent.objects.create(
|
|
customer=customer,
|
|
stripe_payment_intent_id=stripe_payment_intent.id,
|
|
status=PaymentIntent.get_status_integer_from_status_text(stripe_payment_intent.status),
|
|
)
|
|
stripe_session = stripe.checkout.Session.create(
|
|
cancel_url=f"{user.realm.uri}/upgrade/",
|
|
customer=customer.stripe_customer_id,
|
|
mode="setup",
|
|
payment_method_types=["card"],
|
|
metadata=metadata,
|
|
setup_intent_data={"metadata": metadata},
|
|
success_url=f"{user.realm.uri}/billing/event_status?stripe_session_id={{CHECKOUT_SESSION_ID}}",
|
|
)
|
|
session = Session.objects.create(
|
|
customer=customer, stripe_session_id=stripe_session.id, type=session_type
|
|
)
|
|
if payment_intent is not None:
|
|
session.payment_intent = payment_intent
|
|
session.save(update_fields=["payment_intent"])
|
|
return stripe_session
|
|
|
|
|
|
@require_organization_member
|
|
@has_request_variables
|
|
def upgrade(
|
|
request: HttpRequest,
|
|
user: UserProfile,
|
|
billing_modality: str = REQ(str_validator=check_string_in(VALID_BILLING_MODALITY_VALUES)),
|
|
schedule: str = REQ(str_validator=check_string_in(VALID_BILLING_SCHEDULE_VALUES)),
|
|
signed_seat_count: str = REQ(),
|
|
salt: str = REQ(),
|
|
onboarding: bool = REQ(default=False, json_validator=check_bool),
|
|
license_management: Optional[str] = REQ(
|
|
default=None, str_validator=check_string_in(VALID_LICENSE_MANAGEMENT_VALUES)
|
|
),
|
|
licenses: Optional[int] = REQ(json_validator=check_int, default=None),
|
|
) -> HttpResponse:
|
|
ensure_realm_does_not_have_active_plan(user.realm)
|
|
try:
|
|
seat_count = unsign_seat_count(signed_seat_count, salt)
|
|
if billing_modality == "charge_automatically" and license_management == "automatic":
|
|
licenses = seat_count
|
|
if billing_modality == "send_invoice":
|
|
schedule = "annual"
|
|
license_management = "manual"
|
|
check_upgrade_parameters(
|
|
billing_modality, schedule, license_management, licenses, seat_count
|
|
)
|
|
assert licenses is not None and license_management is not None
|
|
automanage_licenses = license_management == "automatic"
|
|
charge_automatically = billing_modality == "charge_automatically"
|
|
|
|
billing_schedule = {"annual": CustomerPlan.ANNUAL, "monthly": CustomerPlan.MONTHLY}[
|
|
schedule
|
|
]
|
|
if charge_automatically:
|
|
stripe_checkout_session = setup_upgrade_checkout_session_and_payment_intent(
|
|
user,
|
|
seat_count,
|
|
licenses,
|
|
license_management,
|
|
billing_schedule,
|
|
billing_modality,
|
|
onboarding,
|
|
)
|
|
return json_success(
|
|
request,
|
|
data={
|
|
"stripe_session_url": stripe_checkout_session.url,
|
|
"stripe_session_id": stripe_checkout_session.id,
|
|
},
|
|
)
|
|
else:
|
|
process_initial_upgrade(
|
|
user,
|
|
licenses,
|
|
automanage_licenses,
|
|
billing_schedule,
|
|
False,
|
|
is_free_trial_offer_enabled(),
|
|
)
|
|
return json_success(request)
|
|
|
|
except BillingError as e:
|
|
billing_logger.warning(
|
|
"BillingError during upgrade: %s. user=%s, realm=%s (%s), billing_modality=%s, "
|
|
"schedule=%s, license_management=%s, licenses=%s",
|
|
e.error_description,
|
|
user.id,
|
|
user.realm.id,
|
|
user.realm.string_id,
|
|
billing_modality,
|
|
schedule,
|
|
license_management,
|
|
licenses,
|
|
)
|
|
raise e
|
|
except Exception:
|
|
billing_logger.exception("Uncaught exception in billing:", stack_info=True)
|
|
error_message = BillingError.CONTACT_SUPPORT.format(email=settings.ZULIP_ADMINISTRATOR)
|
|
error_description = "uncaught exception during upgrade"
|
|
raise BillingError(error_description, error_message)
|
|
|
|
|
|
@zulip_login_required
|
|
@has_request_variables
|
|
def initial_upgrade(
|
|
request: HttpRequest, onboarding: bool = REQ(default=False, json_validator=check_bool)
|
|
) -> HttpResponse:
|
|
user = request.user
|
|
assert user.is_authenticated
|
|
|
|
if not settings.BILLING_ENABLED or user.is_guest:
|
|
return render(request, "404.html", status=404)
|
|
|
|
billing_page_url = reverse(billing_home)
|
|
|
|
customer = get_customer_by_realm(user.realm)
|
|
if customer is not None and (
|
|
get_current_plan_by_customer(customer) is not None or customer.sponsorship_pending
|
|
):
|
|
if onboarding:
|
|
billing_page_url = f"{billing_page_url}?onboarding=true"
|
|
return HttpResponseRedirect(billing_page_url)
|
|
|
|
if is_sponsored_realm(user.realm):
|
|
return HttpResponseRedirect(billing_page_url)
|
|
|
|
percent_off = Decimal(0)
|
|
if customer is not None and customer.default_discount is not None:
|
|
percent_off = customer.default_discount
|
|
|
|
seat_count = get_latest_seat_count(user.realm)
|
|
signed_seat_count, salt = sign_string(str(seat_count))
|
|
context: Dict[str, Any] = {
|
|
"realm": user.realm,
|
|
"email": user.delivery_email,
|
|
"seat_count": seat_count,
|
|
"signed_seat_count": signed_seat_count,
|
|
"salt": salt,
|
|
"min_invoiced_licenses": max(seat_count, MIN_INVOICED_LICENSES),
|
|
"default_invoice_days_until_due": DEFAULT_INVOICE_DAYS_UNTIL_DUE,
|
|
"plan": "Zulip Cloud Standard",
|
|
"free_trial_days": settings.FREE_TRIAL_DAYS,
|
|
"onboarding": onboarding,
|
|
"page_params": {
|
|
"seat_count": seat_count,
|
|
"annual_price": 8000,
|
|
"monthly_price": 800,
|
|
"percent_off": float(percent_off),
|
|
"demo_organization_scheduled_deletion_date": user.realm.demo_organization_scheduled_deletion_date,
|
|
},
|
|
"realm_org_type": user.realm.org_type,
|
|
"sorted_org_types": sorted(
|
|
(
|
|
[org_type_name, org_type]
|
|
for (org_type_name, org_type) in Realm.ORG_TYPES.items()
|
|
if not org_type.get("hidden")
|
|
),
|
|
key=lambda d: d[1]["display_order"],
|
|
),
|
|
"is_demo_organization": user.realm.demo_organization_scheduled_deletion_date is not None,
|
|
}
|
|
response = render(request, "corporate/upgrade.html", context=context)
|
|
return response
|
|
|
|
|
|
class SponsorshipRequestForm(forms.Form):
|
|
website = forms.URLField(max_length=ZulipSponsorshipRequest.MAX_ORG_URL_LENGTH, required=False)
|
|
organization_type = forms.IntegerField()
|
|
description = forms.CharField(widget=forms.Textarea)
|
|
|
|
|
|
@require_organization_member
|
|
@has_request_variables
|
|
def sponsorship(
|
|
request: HttpRequest,
|
|
user: UserProfile,
|
|
organization_type: str = REQ("organization-type"),
|
|
website: str = REQ(),
|
|
description: str = REQ(),
|
|
) -> HttpResponse:
|
|
realm = user.realm
|
|
|
|
requested_by = user.full_name
|
|
user_role = user.get_role_name()
|
|
support_url = get_support_url(realm)
|
|
|
|
post_data = request.POST.copy()
|
|
# We need to do this because the field name in the template
|
|
# for organization type contains a hyphen and the form expects
|
|
# an underscore.
|
|
post_data.update(organization_type=organization_type)
|
|
form = SponsorshipRequestForm(post_data)
|
|
|
|
if form.is_valid():
|
|
with transaction.atomic():
|
|
sponsorship_request = ZulipSponsorshipRequest(
|
|
realm=realm,
|
|
requested_by=user,
|
|
org_website=form.cleaned_data["website"],
|
|
org_description=form.cleaned_data["description"],
|
|
org_type=form.cleaned_data["organization_type"],
|
|
)
|
|
sponsorship_request.save()
|
|
|
|
org_type = form.cleaned_data["organization_type"]
|
|
if realm.org_type != org_type:
|
|
realm.org_type = org_type
|
|
realm.save(update_fields=["org_type"])
|
|
|
|
update_sponsorship_status(realm, True, acting_user=user)
|
|
do_make_user_billing_admin(user)
|
|
|
|
org_type_display_name = get_org_type_display_name(org_type)
|
|
|
|
context = {
|
|
"requested_by": requested_by,
|
|
"user_role": user_role,
|
|
"string_id": realm.string_id,
|
|
"support_url": support_url,
|
|
"organization_type": org_type_display_name,
|
|
"website": website,
|
|
"description": description,
|
|
}
|
|
send_email(
|
|
"zerver/emails/sponsorship_request",
|
|
to_emails=[FromAddress.SUPPORT],
|
|
from_name="Zulip sponsorship",
|
|
from_address=FromAddress.tokenized_no_reply_address(),
|
|
reply_to_email=user.delivery_email,
|
|
context=context,
|
|
)
|
|
|
|
return json_success(request)
|
|
else:
|
|
messages = []
|
|
for error_list in form.errors.get_json_data().values():
|
|
for error in error_list:
|
|
messages.append(error["message"])
|
|
message = " ".join(messages)
|
|
raise BillingError("Form validation error", message=message)
|