mirror of https://github.com/zulip/zulip.git
actions: Split out zerver.actions.custom_profile_fields.
Signed-off-by: Anders Kaseorg <anders@zulip.com>
This commit is contained in:
parent
f6a06ba6e3
commit
bbce879c81
|
@ -0,0 +1,185 @@
|
||||||
|
from typing import Dict, List, Optional, Union
|
||||||
|
|
||||||
|
import orjson
|
||||||
|
from django.db import transaction
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
|
||||||
|
from zerver.lib.exceptions import JsonableError
|
||||||
|
from zerver.lib.external_accounts import DEFAULT_EXTERNAL_ACCOUNTS
|
||||||
|
from zerver.lib.streams import render_stream_description
|
||||||
|
from zerver.lib.types import ProfileDataElementValue, ProfileFieldData
|
||||||
|
from zerver.models import (
|
||||||
|
CustomProfileField,
|
||||||
|
CustomProfileFieldValue,
|
||||||
|
Realm,
|
||||||
|
UserProfile,
|
||||||
|
active_user_ids,
|
||||||
|
custom_profile_fields_for_realm,
|
||||||
|
)
|
||||||
|
from zerver.tornado.django_api import send_event
|
||||||
|
|
||||||
|
|
||||||
|
def notify_realm_custom_profile_fields(realm: Realm) -> None:
|
||||||
|
fields = custom_profile_fields_for_realm(realm.id)
|
||||||
|
event = dict(type="custom_profile_fields", fields=[f.as_dict() for f in fields])
|
||||||
|
send_event(realm, event, active_user_ids(realm.id))
|
||||||
|
|
||||||
|
|
||||||
|
def try_add_realm_default_custom_profile_field(
|
||||||
|
realm: Realm, field_subtype: str
|
||||||
|
) -> CustomProfileField:
|
||||||
|
field_data = DEFAULT_EXTERNAL_ACCOUNTS[field_subtype]
|
||||||
|
custom_profile_field = CustomProfileField(
|
||||||
|
realm=realm,
|
||||||
|
name=field_data["name"],
|
||||||
|
field_type=CustomProfileField.EXTERNAL_ACCOUNT,
|
||||||
|
hint=field_data["hint"],
|
||||||
|
field_data=orjson.dumps(dict(subtype=field_subtype)).decode(),
|
||||||
|
)
|
||||||
|
custom_profile_field.save()
|
||||||
|
custom_profile_field.order = custom_profile_field.id
|
||||||
|
custom_profile_field.save(update_fields=["order"])
|
||||||
|
notify_realm_custom_profile_fields(realm)
|
||||||
|
return custom_profile_field
|
||||||
|
|
||||||
|
|
||||||
|
def try_add_realm_custom_profile_field(
|
||||||
|
realm: Realm,
|
||||||
|
name: str,
|
||||||
|
field_type: int,
|
||||||
|
hint: str = "",
|
||||||
|
field_data: Optional[ProfileFieldData] = None,
|
||||||
|
) -> CustomProfileField:
|
||||||
|
custom_profile_field = CustomProfileField(realm=realm, name=name, field_type=field_type)
|
||||||
|
custom_profile_field.hint = hint
|
||||||
|
if (
|
||||||
|
custom_profile_field.field_type == CustomProfileField.SELECT
|
||||||
|
or custom_profile_field.field_type == CustomProfileField.EXTERNAL_ACCOUNT
|
||||||
|
):
|
||||||
|
custom_profile_field.field_data = orjson.dumps(field_data or {}).decode()
|
||||||
|
|
||||||
|
custom_profile_field.save()
|
||||||
|
custom_profile_field.order = custom_profile_field.id
|
||||||
|
custom_profile_field.save(update_fields=["order"])
|
||||||
|
notify_realm_custom_profile_fields(realm)
|
||||||
|
return custom_profile_field
|
||||||
|
|
||||||
|
|
||||||
|
def do_remove_realm_custom_profile_field(realm: Realm, field: CustomProfileField) -> None:
|
||||||
|
"""
|
||||||
|
Deleting a field will also delete the user profile data
|
||||||
|
associated with it in CustomProfileFieldValue model.
|
||||||
|
"""
|
||||||
|
field.delete()
|
||||||
|
notify_realm_custom_profile_fields(realm)
|
||||||
|
|
||||||
|
|
||||||
|
def do_remove_realm_custom_profile_fields(realm: Realm) -> None:
|
||||||
|
CustomProfileField.objects.filter(realm=realm).delete()
|
||||||
|
|
||||||
|
|
||||||
|
def try_update_realm_custom_profile_field(
|
||||||
|
realm: Realm,
|
||||||
|
field: CustomProfileField,
|
||||||
|
name: str,
|
||||||
|
hint: str = "",
|
||||||
|
field_data: Optional[ProfileFieldData] = None,
|
||||||
|
) -> None:
|
||||||
|
field.name = name
|
||||||
|
field.hint = hint
|
||||||
|
if (
|
||||||
|
field.field_type == CustomProfileField.SELECT
|
||||||
|
or field.field_type == CustomProfileField.EXTERNAL_ACCOUNT
|
||||||
|
):
|
||||||
|
field.field_data = orjson.dumps(field_data or {}).decode()
|
||||||
|
field.save()
|
||||||
|
notify_realm_custom_profile_fields(realm)
|
||||||
|
|
||||||
|
|
||||||
|
def try_reorder_realm_custom_profile_fields(realm: Realm, order: List[int]) -> None:
|
||||||
|
order_mapping = {_[1]: _[0] for _ in enumerate(order)}
|
||||||
|
custom_profile_fields = CustomProfileField.objects.filter(realm=realm)
|
||||||
|
for custom_profile_field in custom_profile_fields:
|
||||||
|
if custom_profile_field.id not in order_mapping:
|
||||||
|
raise JsonableError(_("Invalid order mapping."))
|
||||||
|
for custom_profile_field in custom_profile_fields:
|
||||||
|
custom_profile_field.order = order_mapping[custom_profile_field.id]
|
||||||
|
custom_profile_field.save(update_fields=["order"])
|
||||||
|
notify_realm_custom_profile_fields(realm)
|
||||||
|
|
||||||
|
|
||||||
|
def notify_user_update_custom_profile_data(
|
||||||
|
user_profile: UserProfile, field: Dict[str, Union[int, str, List[int], None]]
|
||||||
|
) -> None:
|
||||||
|
data = dict(id=field["id"], value=field["value"])
|
||||||
|
|
||||||
|
if field["rendered_value"]:
|
||||||
|
data["rendered_value"] = field["rendered_value"]
|
||||||
|
payload = dict(user_id=user_profile.id, custom_profile_field=data)
|
||||||
|
event = dict(type="realm_user", op="update", person=payload)
|
||||||
|
send_event(user_profile.realm, event, active_user_ids(user_profile.realm.id))
|
||||||
|
|
||||||
|
|
||||||
|
def do_update_user_custom_profile_data_if_changed(
|
||||||
|
user_profile: UserProfile,
|
||||||
|
data: List[Dict[str, Union[int, ProfileDataElementValue]]],
|
||||||
|
) -> None:
|
||||||
|
with transaction.atomic():
|
||||||
|
for custom_profile_field in data:
|
||||||
|
field_value, created = CustomProfileFieldValue.objects.get_or_create(
|
||||||
|
user_profile=user_profile, field_id=custom_profile_field["id"]
|
||||||
|
)
|
||||||
|
|
||||||
|
# field_value.value is a TextField() so we need to have field["value"]
|
||||||
|
# in string form to correctly make comparisons and assignments.
|
||||||
|
if isinstance(custom_profile_field["value"], str):
|
||||||
|
custom_profile_field_value_string = custom_profile_field["value"]
|
||||||
|
else:
|
||||||
|
custom_profile_field_value_string = orjson.dumps(
|
||||||
|
custom_profile_field["value"]
|
||||||
|
).decode()
|
||||||
|
|
||||||
|
if not created and field_value.value == custom_profile_field_value_string:
|
||||||
|
# If the field value isn't actually being changed to a different one,
|
||||||
|
# we have nothing to do here for this field.
|
||||||
|
continue
|
||||||
|
|
||||||
|
field_value.value = custom_profile_field_value_string
|
||||||
|
if field_value.field.is_renderable():
|
||||||
|
field_value.rendered_value = render_stream_description(
|
||||||
|
custom_profile_field_value_string
|
||||||
|
)
|
||||||
|
field_value.save(update_fields=["value", "rendered_value"])
|
||||||
|
else:
|
||||||
|
field_value.save(update_fields=["value"])
|
||||||
|
notify_user_update_custom_profile_data(
|
||||||
|
user_profile,
|
||||||
|
{
|
||||||
|
"id": field_value.field_id,
|
||||||
|
"value": field_value.value,
|
||||||
|
"rendered_value": field_value.rendered_value,
|
||||||
|
"type": field_value.field.field_type,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def check_remove_custom_profile_field_value(user_profile: UserProfile, field_id: int) -> None:
|
||||||
|
try:
|
||||||
|
custom_profile_field = CustomProfileField.objects.get(realm=user_profile.realm, id=field_id)
|
||||||
|
field_value = CustomProfileFieldValue.objects.get(
|
||||||
|
field=custom_profile_field, user_profile=user_profile
|
||||||
|
)
|
||||||
|
field_value.delete()
|
||||||
|
notify_user_update_custom_profile_data(
|
||||||
|
user_profile,
|
||||||
|
{
|
||||||
|
"id": field_id,
|
||||||
|
"value": None,
|
||||||
|
"rendered_value": None,
|
||||||
|
"type": custom_profile_field.field_type,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
except CustomProfileField.DoesNotExist:
|
||||||
|
raise JsonableError(_("Field id {id} not found.").format(id=field_id))
|
||||||
|
except CustomProfileFieldValue.DoesNotExist:
|
||||||
|
pass
|
|
@ -34,6 +34,7 @@ from typing_extensions import TypedDict
|
||||||
from analytics.lib.counts import COUNT_STATS, do_increment_logging_stat
|
from analytics.lib.counts import COUNT_STATS, do_increment_logging_stat
|
||||||
from confirmation import settings as confirmation_settings
|
from confirmation import settings as confirmation_settings
|
||||||
from confirmation.models import Confirmation, create_confirmation_link, generate_key
|
from confirmation.models import Confirmation, create_confirmation_link, generate_key
|
||||||
|
from zerver.actions.custom_profile_fields import do_remove_realm_custom_profile_fields
|
||||||
from zerver.actions.default_streams import (
|
from zerver.actions.default_streams import (
|
||||||
do_remove_default_stream,
|
do_remove_default_stream,
|
||||||
do_remove_streams_from_default_stream_group,
|
do_remove_streams_from_default_stream_group,
|
||||||
|
@ -78,7 +79,6 @@ from zerver.lib.exceptions import (
|
||||||
StreamWithIDDoesNotExistError,
|
StreamWithIDDoesNotExistError,
|
||||||
ZephyrMessageAlreadySentException,
|
ZephyrMessageAlreadySentException,
|
||||||
)
|
)
|
||||||
from zerver.lib.external_accounts import DEFAULT_EXTERNAL_ACCOUNTS
|
|
||||||
from zerver.lib.i18n import get_language_name
|
from zerver.lib.i18n import get_language_name
|
||||||
from zerver.lib.markdown import MessageRenderingResult, topic_links
|
from zerver.lib.markdown import MessageRenderingResult, topic_links
|
||||||
from zerver.lib.markdown import version as markdown_version
|
from zerver.lib.markdown import version as markdown_version
|
||||||
|
@ -154,7 +154,7 @@ from zerver.lib.topic import (
|
||||||
update_edit_history,
|
update_edit_history,
|
||||||
update_messages_for_topic_edit,
|
update_messages_for_topic_edit,
|
||||||
)
|
)
|
||||||
from zerver.lib.types import EditHistoryEvent, ProfileDataElementValue, ProfileFieldData
|
from zerver.lib.types import EditHistoryEvent
|
||||||
from zerver.lib.upload import delete_avatar_image
|
from zerver.lib.upload import delete_avatar_image
|
||||||
from zerver.lib.user_counts import realm_user_count, realm_user_count_by_role
|
from zerver.lib.user_counts import realm_user_count, realm_user_count_by_role
|
||||||
from zerver.lib.user_groups import (
|
from zerver.lib.user_groups import (
|
||||||
|
@ -179,8 +179,6 @@ from zerver.models import (
|
||||||
ArchivedAttachment,
|
ArchivedAttachment,
|
||||||
Attachment,
|
Attachment,
|
||||||
Client,
|
Client,
|
||||||
CustomProfileField,
|
|
||||||
CustomProfileFieldValue,
|
|
||||||
DefaultStream,
|
DefaultStream,
|
||||||
DefaultStreamGroup,
|
DefaultStreamGroup,
|
||||||
Draft,
|
Draft,
|
||||||
|
@ -208,7 +206,6 @@ from zerver.models import (
|
||||||
active_non_guest_user_ids,
|
active_non_guest_user_ids,
|
||||||
active_user_ids,
|
active_user_ids,
|
||||||
bot_owner_user_ids,
|
bot_owner_user_ids,
|
||||||
custom_profile_fields_for_realm,
|
|
||||||
get_bot_dicts_in_realm,
|
get_bot_dicts_in_realm,
|
||||||
get_bot_services,
|
get_bot_services,
|
||||||
get_client,
|
get_client,
|
||||||
|
@ -6480,172 +6477,6 @@ def do_remove_realm_domain(
|
||||||
transaction.on_commit(lambda: send_event(realm, event, active_user_ids(realm.id)))
|
transaction.on_commit(lambda: send_event(realm, event, active_user_ids(realm.id)))
|
||||||
|
|
||||||
|
|
||||||
def notify_realm_custom_profile_fields(realm: Realm) -> None:
|
|
||||||
fields = custom_profile_fields_for_realm(realm.id)
|
|
||||||
event = dict(type="custom_profile_fields", fields=[f.as_dict() for f in fields])
|
|
||||||
send_event(realm, event, active_user_ids(realm.id))
|
|
||||||
|
|
||||||
|
|
||||||
def try_add_realm_default_custom_profile_field(
|
|
||||||
realm: Realm, field_subtype: str
|
|
||||||
) -> CustomProfileField:
|
|
||||||
field_data = DEFAULT_EXTERNAL_ACCOUNTS[field_subtype]
|
|
||||||
custom_profile_field = CustomProfileField(
|
|
||||||
realm=realm,
|
|
||||||
name=field_data["name"],
|
|
||||||
field_type=CustomProfileField.EXTERNAL_ACCOUNT,
|
|
||||||
hint=field_data["hint"],
|
|
||||||
field_data=orjson.dumps(dict(subtype=field_subtype)).decode(),
|
|
||||||
)
|
|
||||||
custom_profile_field.save()
|
|
||||||
custom_profile_field.order = custom_profile_field.id
|
|
||||||
custom_profile_field.save(update_fields=["order"])
|
|
||||||
notify_realm_custom_profile_fields(realm)
|
|
||||||
return custom_profile_field
|
|
||||||
|
|
||||||
|
|
||||||
def try_add_realm_custom_profile_field(
|
|
||||||
realm: Realm,
|
|
||||||
name: str,
|
|
||||||
field_type: int,
|
|
||||||
hint: str = "",
|
|
||||||
field_data: Optional[ProfileFieldData] = None,
|
|
||||||
) -> CustomProfileField:
|
|
||||||
custom_profile_field = CustomProfileField(realm=realm, name=name, field_type=field_type)
|
|
||||||
custom_profile_field.hint = hint
|
|
||||||
if (
|
|
||||||
custom_profile_field.field_type == CustomProfileField.SELECT
|
|
||||||
or custom_profile_field.field_type == CustomProfileField.EXTERNAL_ACCOUNT
|
|
||||||
):
|
|
||||||
custom_profile_field.field_data = orjson.dumps(field_data or {}).decode()
|
|
||||||
|
|
||||||
custom_profile_field.save()
|
|
||||||
custom_profile_field.order = custom_profile_field.id
|
|
||||||
custom_profile_field.save(update_fields=["order"])
|
|
||||||
notify_realm_custom_profile_fields(realm)
|
|
||||||
return custom_profile_field
|
|
||||||
|
|
||||||
|
|
||||||
def do_remove_realm_custom_profile_field(realm: Realm, field: CustomProfileField) -> None:
|
|
||||||
"""
|
|
||||||
Deleting a field will also delete the user profile data
|
|
||||||
associated with it in CustomProfileFieldValue model.
|
|
||||||
"""
|
|
||||||
field.delete()
|
|
||||||
notify_realm_custom_profile_fields(realm)
|
|
||||||
|
|
||||||
|
|
||||||
def do_remove_realm_custom_profile_fields(realm: Realm) -> None:
|
|
||||||
CustomProfileField.objects.filter(realm=realm).delete()
|
|
||||||
|
|
||||||
|
|
||||||
def try_update_realm_custom_profile_field(
|
|
||||||
realm: Realm,
|
|
||||||
field: CustomProfileField,
|
|
||||||
name: str,
|
|
||||||
hint: str = "",
|
|
||||||
field_data: Optional[ProfileFieldData] = None,
|
|
||||||
) -> None:
|
|
||||||
field.name = name
|
|
||||||
field.hint = hint
|
|
||||||
if (
|
|
||||||
field.field_type == CustomProfileField.SELECT
|
|
||||||
or field.field_type == CustomProfileField.EXTERNAL_ACCOUNT
|
|
||||||
):
|
|
||||||
field.field_data = orjson.dumps(field_data or {}).decode()
|
|
||||||
field.save()
|
|
||||||
notify_realm_custom_profile_fields(realm)
|
|
||||||
|
|
||||||
|
|
||||||
def try_reorder_realm_custom_profile_fields(realm: Realm, order: List[int]) -> None:
|
|
||||||
order_mapping = {_[1]: _[0] for _ in enumerate(order)}
|
|
||||||
custom_profile_fields = CustomProfileField.objects.filter(realm=realm)
|
|
||||||
for custom_profile_field in custom_profile_fields:
|
|
||||||
if custom_profile_field.id not in order_mapping:
|
|
||||||
raise JsonableError(_("Invalid order mapping."))
|
|
||||||
for custom_profile_field in custom_profile_fields:
|
|
||||||
custom_profile_field.order = order_mapping[custom_profile_field.id]
|
|
||||||
custom_profile_field.save(update_fields=["order"])
|
|
||||||
notify_realm_custom_profile_fields(realm)
|
|
||||||
|
|
||||||
|
|
||||||
def notify_user_update_custom_profile_data(
|
|
||||||
user_profile: UserProfile, field: Dict[str, Union[int, str, List[int], None]]
|
|
||||||
) -> None:
|
|
||||||
data = dict(id=field["id"], value=field["value"])
|
|
||||||
|
|
||||||
if field["rendered_value"]:
|
|
||||||
data["rendered_value"] = field["rendered_value"]
|
|
||||||
payload = dict(user_id=user_profile.id, custom_profile_field=data)
|
|
||||||
event = dict(type="realm_user", op="update", person=payload)
|
|
||||||
send_event(user_profile.realm, event, active_user_ids(user_profile.realm.id))
|
|
||||||
|
|
||||||
|
|
||||||
def do_update_user_custom_profile_data_if_changed(
|
|
||||||
user_profile: UserProfile,
|
|
||||||
data: List[Dict[str, Union[int, ProfileDataElementValue]]],
|
|
||||||
) -> None:
|
|
||||||
with transaction.atomic():
|
|
||||||
for custom_profile_field in data:
|
|
||||||
field_value, created = CustomProfileFieldValue.objects.get_or_create(
|
|
||||||
user_profile=user_profile, field_id=custom_profile_field["id"]
|
|
||||||
)
|
|
||||||
|
|
||||||
# field_value.value is a TextField() so we need to have field["value"]
|
|
||||||
# in string form to correctly make comparisons and assignments.
|
|
||||||
if isinstance(custom_profile_field["value"], str):
|
|
||||||
custom_profile_field_value_string = custom_profile_field["value"]
|
|
||||||
else:
|
|
||||||
custom_profile_field_value_string = orjson.dumps(
|
|
||||||
custom_profile_field["value"]
|
|
||||||
).decode()
|
|
||||||
|
|
||||||
if not created and field_value.value == custom_profile_field_value_string:
|
|
||||||
# If the field value isn't actually being changed to a different one,
|
|
||||||
# we have nothing to do here for this field.
|
|
||||||
continue
|
|
||||||
|
|
||||||
field_value.value = custom_profile_field_value_string
|
|
||||||
if field_value.field.is_renderable():
|
|
||||||
field_value.rendered_value = render_stream_description(
|
|
||||||
custom_profile_field_value_string
|
|
||||||
)
|
|
||||||
field_value.save(update_fields=["value", "rendered_value"])
|
|
||||||
else:
|
|
||||||
field_value.save(update_fields=["value"])
|
|
||||||
notify_user_update_custom_profile_data(
|
|
||||||
user_profile,
|
|
||||||
{
|
|
||||||
"id": field_value.field_id,
|
|
||||||
"value": field_value.value,
|
|
||||||
"rendered_value": field_value.rendered_value,
|
|
||||||
"type": field_value.field.field_type,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def check_remove_custom_profile_field_value(user_profile: UserProfile, field_id: int) -> None:
|
|
||||||
try:
|
|
||||||
custom_profile_field = CustomProfileField.objects.get(realm=user_profile.realm, id=field_id)
|
|
||||||
field_value = CustomProfileFieldValue.objects.get(
|
|
||||||
field=custom_profile_field, user_profile=user_profile
|
|
||||||
)
|
|
||||||
field_value.delete()
|
|
||||||
notify_user_update_custom_profile_data(
|
|
||||||
user_profile,
|
|
||||||
{
|
|
||||||
"id": field_id,
|
|
||||||
"value": None,
|
|
||||||
"rendered_value": None,
|
|
||||||
"type": custom_profile_field.field_type,
|
|
||||||
},
|
|
||||||
)
|
|
||||||
except CustomProfileField.DoesNotExist:
|
|
||||||
raise JsonableError(_("Field id {id} not found.").format(id=field_id))
|
|
||||||
except CustomProfileFieldValue.DoesNotExist:
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
def do_update_outgoing_webhook_service(
|
def do_update_outgoing_webhook_service(
|
||||||
bot_profile: UserProfile, service_interface: int, service_payload_url: str
|
bot_profile: UserProfile, service_interface: int, service_payload_url: str
|
||||||
) -> None:
|
) -> None:
|
||||||
|
|
|
@ -3,7 +3,7 @@ from unittest import mock
|
||||||
|
|
||||||
import orjson
|
import orjson
|
||||||
|
|
||||||
from zerver.lib.actions import (
|
from zerver.actions.custom_profile_fields import (
|
||||||
do_remove_realm_custom_profile_field,
|
do_remove_realm_custom_profile_field,
|
||||||
do_update_user_custom_profile_data_if_changed,
|
do_update_user_custom_profile_data_if_changed,
|
||||||
try_add_realm_custom_profile_field,
|
try_add_realm_custom_profile_field,
|
||||||
|
@ -709,7 +709,9 @@ class UpdateCustomProfileFieldTest(CustomProfileFieldTestCase):
|
||||||
]
|
]
|
||||||
do_update_user_custom_profile_data_if_changed(iago, data)
|
do_update_user_custom_profile_data_if_changed(iago, data)
|
||||||
|
|
||||||
with mock.patch("zerver.lib.actions.notify_user_update_custom_profile_data") as mock_notify:
|
with mock.patch(
|
||||||
|
"zerver.actions.custom_profile_fields.notify_user_update_custom_profile_data"
|
||||||
|
) as mock_notify:
|
||||||
# Attempting to "update" the field value, when it wouldn't actually change,
|
# Attempting to "update" the field value, when it wouldn't actually change,
|
||||||
# shouldn't trigger notify.
|
# shouldn't trigger notify.
|
||||||
do_update_user_custom_profile_data_if_changed(iago, data)
|
do_update_user_custom_profile_data_if_changed(iago, data)
|
||||||
|
|
|
@ -14,6 +14,12 @@ import orjson
|
||||||
from django.utils.timezone import now as timezone_now
|
from django.utils.timezone import now as timezone_now
|
||||||
|
|
||||||
from zerver.actions.alert_words import do_add_alert_words, do_remove_alert_words
|
from zerver.actions.alert_words import do_add_alert_words, do_remove_alert_words
|
||||||
|
from zerver.actions.custom_profile_fields import (
|
||||||
|
do_remove_realm_custom_profile_field,
|
||||||
|
do_update_user_custom_profile_data_if_changed,
|
||||||
|
try_add_realm_custom_profile_field,
|
||||||
|
try_update_realm_custom_profile_field,
|
||||||
|
)
|
||||||
from zerver.actions.default_streams import (
|
from zerver.actions.default_streams import (
|
||||||
do_add_default_stream,
|
do_add_default_stream,
|
||||||
do_add_streams_to_default_stream_group,
|
do_add_streams_to_default_stream_group,
|
||||||
|
@ -85,7 +91,6 @@ from zerver.lib.actions import (
|
||||||
do_reactivate_user,
|
do_reactivate_user,
|
||||||
do_regenerate_api_key,
|
do_regenerate_api_key,
|
||||||
do_remove_reaction,
|
do_remove_reaction,
|
||||||
do_remove_realm_custom_profile_field,
|
|
||||||
do_remove_realm_domain,
|
do_remove_realm_domain,
|
||||||
do_rename_stream,
|
do_rename_stream,
|
||||||
do_set_realm_authentication_methods,
|
do_set_realm_authentication_methods,
|
||||||
|
@ -99,9 +104,6 @@ from zerver.lib.actions import (
|
||||||
do_update_message,
|
do_update_message,
|
||||||
do_update_message_flags,
|
do_update_message_flags,
|
||||||
do_update_outgoing_webhook_service,
|
do_update_outgoing_webhook_service,
|
||||||
do_update_user_custom_profile_data_if_changed,
|
|
||||||
try_add_realm_custom_profile_field,
|
|
||||||
try_update_realm_custom_profile_field,
|
|
||||||
)
|
)
|
||||||
from zerver.lib.drafts import do_create_drafts, do_delete_draft, do_edit_draft
|
from zerver.lib.drafts import do_create_drafts, do_delete_draft, do_edit_draft
|
||||||
from zerver.lib.event_schema import (
|
from zerver.lib.event_schema import (
|
||||||
|
|
|
@ -11,6 +11,10 @@ from django.utils.timezone import now as timezone_now
|
||||||
|
|
||||||
from analytics.models import UserCount
|
from analytics.models import UserCount
|
||||||
from zerver.actions.alert_words import do_add_alert_words
|
from zerver.actions.alert_words import do_add_alert_words
|
||||||
|
from zerver.actions.custom_profile_fields import (
|
||||||
|
do_update_user_custom_profile_data_if_changed,
|
||||||
|
try_add_realm_custom_profile_field,
|
||||||
|
)
|
||||||
from zerver.actions.presence import do_update_user_presence, do_update_user_status
|
from zerver.actions.presence import do_update_user_presence, do_update_user_status
|
||||||
from zerver.actions.realm_emoji import check_add_realm_emoji
|
from zerver.actions.realm_emoji import check_add_realm_emoji
|
||||||
from zerver.actions.realm_icon import do_change_icon_source
|
from zerver.actions.realm_icon import do_change_icon_source
|
||||||
|
@ -25,8 +29,6 @@ from zerver.lib.actions import (
|
||||||
do_create_user,
|
do_create_user,
|
||||||
do_deactivate_user,
|
do_deactivate_user,
|
||||||
do_mute_user,
|
do_mute_user,
|
||||||
do_update_user_custom_profile_data_if_changed,
|
|
||||||
try_add_realm_custom_profile_field,
|
|
||||||
)
|
)
|
||||||
from zerver.lib.avatar_hash import user_avatar_path
|
from zerver.lib.avatar_hash import user_avatar_path
|
||||||
from zerver.lib.bot_config import set_bot_config
|
from zerver.lib.bot_config import set_bot_config
|
||||||
|
|
|
@ -6,8 +6,7 @@ from django.db import IntegrityError
|
||||||
from django.http import HttpRequest, HttpResponse
|
from django.http import HttpRequest, HttpResponse
|
||||||
from django.utils.translation import gettext as _
|
from django.utils.translation import gettext as _
|
||||||
|
|
||||||
from zerver.decorator import human_users_only, require_realm_admin
|
from zerver.actions.custom_profile_fields import (
|
||||||
from zerver.lib.actions import (
|
|
||||||
check_remove_custom_profile_field_value,
|
check_remove_custom_profile_field_value,
|
||||||
do_remove_realm_custom_profile_field,
|
do_remove_realm_custom_profile_field,
|
||||||
do_update_user_custom_profile_data_if_changed,
|
do_update_user_custom_profile_data_if_changed,
|
||||||
|
@ -16,6 +15,7 @@ from zerver.lib.actions import (
|
||||||
try_reorder_realm_custom_profile_fields,
|
try_reorder_realm_custom_profile_fields,
|
||||||
try_update_realm_custom_profile_field,
|
try_update_realm_custom_profile_field,
|
||||||
)
|
)
|
||||||
|
from zerver.decorator import human_users_only, require_realm_admin
|
||||||
from zerver.lib.exceptions import JsonableError
|
from zerver.lib.exceptions import JsonableError
|
||||||
from zerver.lib.external_accounts import validate_external_account_field_data
|
from zerver.lib.external_accounts import validate_external_account_field_data
|
||||||
from zerver.lib.request import REQ, has_request_variables
|
from zerver.lib.request import REQ, has_request_variables
|
||||||
|
|
|
@ -6,13 +6,16 @@ from django.http import HttpRequest, HttpResponse
|
||||||
from django.shortcuts import redirect
|
from django.shortcuts import redirect
|
||||||
from django.utils.translation import gettext as _
|
from django.utils.translation import gettext as _
|
||||||
|
|
||||||
|
from zerver.actions.custom_profile_fields import (
|
||||||
|
check_remove_custom_profile_field_value,
|
||||||
|
do_update_user_custom_profile_data_if_changed,
|
||||||
|
)
|
||||||
from zerver.context_processors import get_valid_realm_from_request
|
from zerver.context_processors import get_valid_realm_from_request
|
||||||
from zerver.decorator import require_member_or_admin, require_realm_admin
|
from zerver.decorator import require_member_or_admin, require_realm_admin
|
||||||
from zerver.forms import PASSWORD_TOO_WEAK_ERROR, CreateUserForm
|
from zerver.forms import PASSWORD_TOO_WEAK_ERROR, CreateUserForm
|
||||||
from zerver.lib.actions import (
|
from zerver.lib.actions import (
|
||||||
check_change_bot_full_name,
|
check_change_bot_full_name,
|
||||||
check_change_full_name,
|
check_change_full_name,
|
||||||
check_remove_custom_profile_field_value,
|
|
||||||
do_change_avatar_fields,
|
do_change_avatar_fields,
|
||||||
do_change_bot_owner,
|
do_change_bot_owner,
|
||||||
do_change_default_all_public_streams,
|
do_change_default_all_public_streams,
|
||||||
|
@ -25,7 +28,6 @@ from zerver.lib.actions import (
|
||||||
do_regenerate_api_key,
|
do_regenerate_api_key,
|
||||||
do_update_bot_config_data,
|
do_update_bot_config_data,
|
||||||
do_update_outgoing_webhook_service,
|
do_update_outgoing_webhook_service,
|
||||||
do_update_user_custom_profile_data_if_changed,
|
|
||||||
notify_created_bot,
|
notify_created_bot,
|
||||||
)
|
)
|
||||||
from zerver.lib.avatar import avatar_url, get_gravatar_url
|
from zerver.lib.avatar import avatar_url, get_gravatar_url
|
||||||
|
|
|
@ -18,15 +18,17 @@ from django.utils.timezone import now as timezone_now
|
||||||
from django.utils.timezone import timedelta as timezone_timedelta
|
from django.utils.timezone import timedelta as timezone_timedelta
|
||||||
|
|
||||||
from scripts.lib.zulip_tools import get_or_create_dev_uuid_var_path
|
from scripts.lib.zulip_tools import get_or_create_dev_uuid_var_path
|
||||||
|
from zerver.actions.custom_profile_fields import (
|
||||||
|
do_update_user_custom_profile_data_if_changed,
|
||||||
|
try_add_realm_custom_profile_field,
|
||||||
|
try_add_realm_default_custom_profile_field,
|
||||||
|
)
|
||||||
from zerver.actions.realm_emoji import check_add_realm_emoji
|
from zerver.actions.realm_emoji import check_add_realm_emoji
|
||||||
from zerver.lib.actions import (
|
from zerver.lib.actions import (
|
||||||
build_message_send_dict,
|
build_message_send_dict,
|
||||||
do_change_user_role,
|
do_change_user_role,
|
||||||
do_create_realm,
|
do_create_realm,
|
||||||
do_send_messages,
|
do_send_messages,
|
||||||
do_update_user_custom_profile_data_if_changed,
|
|
||||||
try_add_realm_custom_profile_field,
|
|
||||||
try_add_realm_default_custom_profile_field,
|
|
||||||
)
|
)
|
||||||
from zerver.lib.bulk_create import bulk_create_streams
|
from zerver.lib.bulk_create import bulk_create_streams
|
||||||
from zerver.lib.cache import cache_set
|
from zerver.lib.cache import cache_set
|
||||||
|
|
|
@ -58,13 +58,13 @@ from social_core.pipeline.partial import partial
|
||||||
from typing_extensions import TypedDict
|
from typing_extensions import TypedDict
|
||||||
from zxcvbn import zxcvbn
|
from zxcvbn import zxcvbn
|
||||||
|
|
||||||
|
from zerver.actions.custom_profile_fields import do_update_user_custom_profile_data_if_changed
|
||||||
from zerver.decorator import client_is_exempt_from_rate_limiting
|
from zerver.decorator import client_is_exempt_from_rate_limiting
|
||||||
from zerver.lib.actions import (
|
from zerver.lib.actions import (
|
||||||
do_create_user,
|
do_create_user,
|
||||||
do_deactivate_user,
|
do_deactivate_user,
|
||||||
do_reactivate_user,
|
do_reactivate_user,
|
||||||
do_regenerate_api_key,
|
do_regenerate_api_key,
|
||||||
do_update_user_custom_profile_data_if_changed,
|
|
||||||
)
|
)
|
||||||
from zerver.lib.avatar import avatar_url, is_avatar_new
|
from zerver.lib.avatar import avatar_url, is_avatar_new
|
||||||
from zerver.lib.avatar_hash import user_avatar_content_hash
|
from zerver.lib.avatar_hash import user_avatar_content_hash
|
||||||
|
|
Loading…
Reference in New Issue