2021-05-28 15:57:08 +02:00
|
|
|
from typing import Optional
|
|
|
|
|
|
|
|
from django.conf import settings
|
|
|
|
from django.utils.translation import gettext as _
|
|
|
|
|
2022-08-14 19:42:55 +02:00
|
|
|
from corporate.lib.stripe import LicenseLimitError, get_latest_seat_count, get_seat_count
|
2021-05-28 15:57:08 +02:00
|
|
|
from corporate.models import get_current_plan_by_realm
|
2024-04-02 06:38:38 +02:00
|
|
|
from zerver.actions.create_user import send_group_direct_message_to_admins
|
2021-05-28 15:57:08 +02:00
|
|
|
from zerver.lib.exceptions import InvitationError
|
2023-12-15 01:16:00 +01:00
|
|
|
from zerver.models import Realm, UserProfile
|
|
|
|
from zerver.models.users import get_system_bot
|
2021-05-28 15:57:08 +02:00
|
|
|
|
|
|
|
|
|
|
|
def generate_licenses_low_warning_message_if_required(realm: Realm) -> Optional[str]:
|
|
|
|
plan = get_current_plan_by_realm(realm)
|
|
|
|
if plan is None or plan.automanage_licenses:
|
|
|
|
return None
|
|
|
|
|
|
|
|
licenses_remaining = plan.licenses() - get_latest_seat_count(realm)
|
|
|
|
if licenses_remaining > 3:
|
|
|
|
return None
|
|
|
|
|
|
|
|
format_kwargs = {
|
|
|
|
"billing_page_link": "/billing/#settings",
|
|
|
|
"deactivate_user_help_page_link": "/help/deactivate-or-reactivate-a-user",
|
|
|
|
}
|
|
|
|
|
|
|
|
if licenses_remaining <= 0:
|
|
|
|
return _(
|
|
|
|
"Your organization has no Zulip licenses remaining and can no longer accept new users. "
|
|
|
|
"Please [increase the number of licenses]({billing_page_link}) or "
|
|
|
|
"[deactivate inactive users]({deactivate_user_help_page_link}) to allow new users to join."
|
|
|
|
).format(**format_kwargs)
|
|
|
|
|
|
|
|
return {
|
|
|
|
1: _(
|
|
|
|
"Your organization has only one Zulip license remaining. You can "
|
|
|
|
"[increase the number of licenses]({billing_page_link}) or [deactivate inactive users]({deactivate_user_help_page_link}) "
|
|
|
|
"to allow more than one user to join."
|
|
|
|
),
|
|
|
|
2: _(
|
|
|
|
"Your organization has only two Zulip licenses remaining. You can "
|
|
|
|
"[increase the number of licenses]({billing_page_link}) or [deactivate inactive users]({deactivate_user_help_page_link}) "
|
|
|
|
"to allow more than two users to join."
|
|
|
|
),
|
|
|
|
3: _(
|
|
|
|
"Your organization has only three Zulip licenses remaining. You can "
|
|
|
|
"[increase the number of licenses]({billing_page_link}) or [deactivate inactive users]({deactivate_user_help_page_link}) "
|
|
|
|
"to allow more than three users to join."
|
|
|
|
),
|
|
|
|
}[licenses_remaining].format(**format_kwargs)
|
|
|
|
|
|
|
|
|
2024-04-02 06:38:38 +02:00
|
|
|
def send_user_unable_to_signup_group_direct_message_to_admins(
|
2021-05-28 15:57:08 +02:00
|
|
|
realm: Realm, user_email: str
|
|
|
|
) -> None:
|
|
|
|
message = _(
|
|
|
|
"A new member ({email}) was unable to join your organization because all Zulip licenses "
|
|
|
|
"are in use. Please [increase the number of licenses]({billing_page_link}) or "
|
|
|
|
"[deactivate inactive users]({deactivate_user_help_page_link}) to allow new members to join."
|
|
|
|
).format(
|
|
|
|
email=user_email,
|
|
|
|
billing_page_link="/billing/#settings",
|
|
|
|
deactivate_user_help_page_link="/help/deactivate-or-reactivate-a-user",
|
|
|
|
)
|
|
|
|
|
2024-04-02 06:38:38 +02:00
|
|
|
send_group_direct_message_to_admins(
|
2021-07-09 18:33:09 +02:00
|
|
|
get_system_bot(settings.NOTIFICATION_BOT, realm.id), realm, message
|
2021-05-28 15:57:08 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def check_spare_licenses_available_for_adding_new_users(
|
2022-08-14 19:42:55 +02:00
|
|
|
realm: Realm, extra_non_guests_count: int = 0, extra_guests_count: int = 0
|
2021-05-28 15:57:08 +02:00
|
|
|
) -> None:
|
|
|
|
plan = get_current_plan_by_realm(realm)
|
2023-04-10 20:23:12 +02:00
|
|
|
if plan is None or plan.automanage_licenses or plan.customer.exempt_from_license_number_check:
|
2021-05-28 15:57:08 +02:00
|
|
|
return
|
2021-06-08 12:43:44 +02:00
|
|
|
|
2022-08-14 19:42:55 +02:00
|
|
|
if plan.licenses() < get_seat_count(
|
|
|
|
realm, extra_non_guests_count=extra_non_guests_count, extra_guests_count=extra_guests_count
|
|
|
|
):
|
2023-02-04 02:07:20 +01:00
|
|
|
raise LicenseLimitError
|
2021-05-28 15:57:08 +02:00
|
|
|
|
|
|
|
|
|
|
|
def check_spare_licenses_available_for_registering_new_user(
|
2022-08-14 19:42:55 +02:00
|
|
|
realm: Realm,
|
|
|
|
user_email_to_add: str,
|
|
|
|
role: int,
|
2021-05-28 15:57:08 +02:00
|
|
|
) -> None:
|
|
|
|
try:
|
2022-08-14 19:42:55 +02:00
|
|
|
if role == UserProfile.ROLE_GUEST:
|
|
|
|
check_spare_licenses_available_for_adding_new_users(realm, extra_guests_count=1)
|
|
|
|
else:
|
|
|
|
check_spare_licenses_available_for_adding_new_users(realm, extra_non_guests_count=1)
|
2021-05-28 15:57:08 +02:00
|
|
|
except LicenseLimitError:
|
2024-04-02 06:38:38 +02:00
|
|
|
send_user_unable_to_signup_group_direct_message_to_admins(realm, user_email_to_add)
|
2021-05-28 15:57:08 +02:00
|
|
|
raise
|
|
|
|
|
|
|
|
|
2022-08-14 19:42:55 +02:00
|
|
|
def check_spare_licenses_available_for_inviting_new_users(
|
|
|
|
realm: Realm, extra_non_guests_count: int = 0, extra_guests_count: int = 0
|
|
|
|
) -> None:
|
|
|
|
num_invites = extra_non_guests_count + extra_guests_count
|
2021-05-28 15:57:08 +02:00
|
|
|
try:
|
2022-08-14 19:42:55 +02:00
|
|
|
check_spare_licenses_available_for_adding_new_users(
|
|
|
|
realm, extra_non_guests_count, extra_guests_count
|
|
|
|
)
|
2021-05-28 15:57:08 +02:00
|
|
|
except LicenseLimitError:
|
|
|
|
if num_invites == 1:
|
|
|
|
message = _("All Zulip licenses for this organization are currently in use.")
|
|
|
|
else:
|
|
|
|
message = _(
|
|
|
|
"Your organization does not have enough unused Zulip licenses to invite {num_invites} users."
|
|
|
|
).format(num_invites=num_invites)
|
|
|
|
raise InvitationError(message, [], sent_invitations=False, license_limit_reached=True)
|