mirror of https://github.com/zulip/zulip.git
models: Extract zerver.models.custom_profile_fields.
Signed-off-by: Anders Kaseorg <anders@zulip.com>
This commit is contained in:
parent
d5410b577a
commit
27c0b507af
|
@ -9,13 +9,8 @@ from zerver.lib.external_accounts import DEFAULT_EXTERNAL_ACCOUNTS
|
||||||
from zerver.lib.streams import render_stream_description
|
from zerver.lib.streams import render_stream_description
|
||||||
from zerver.lib.types import ProfileDataElementUpdateDict, ProfileFieldData
|
from zerver.lib.types import ProfileDataElementUpdateDict, ProfileFieldData
|
||||||
from zerver.lib.users import get_user_ids_who_can_access_user
|
from zerver.lib.users import get_user_ids_who_can_access_user
|
||||||
from zerver.models import (
|
from zerver.models import CustomProfileField, CustomProfileFieldValue, Realm, UserProfile
|
||||||
CustomProfileField,
|
from zerver.models.custom_profile_fields import custom_profile_fields_for_realm
|
||||||
CustomProfileFieldValue,
|
|
||||||
Realm,
|
|
||||||
UserProfile,
|
|
||||||
custom_profile_fields_for_realm,
|
|
||||||
)
|
|
||||||
from zerver.models.users import active_user_ids
|
from zerver.models.users import active_user_ids
|
||||||
from zerver.tornado.django_api import send_event
|
from zerver.tornado.django_api import send_event
|
||||||
|
|
||||||
|
|
|
@ -79,9 +79,9 @@ from zerver.models import (
|
||||||
UserProfile,
|
UserProfile,
|
||||||
UserStatus,
|
UserStatus,
|
||||||
UserTopic,
|
UserTopic,
|
||||||
custom_profile_fields_for_realm,
|
|
||||||
)
|
)
|
||||||
from zerver.models.constants import MAX_TOPIC_NAME_LENGTH
|
from zerver.models.constants import MAX_TOPIC_NAME_LENGTH
|
||||||
|
from zerver.models.custom_profile_fields import custom_profile_fields_for_realm
|
||||||
from zerver.models.linkifiers import linkifiers_for_realm
|
from zerver.models.linkifiers import linkifiers_for_realm
|
||||||
from zerver.models.realm_emoji import get_all_custom_emoji_for_realm
|
from zerver.models.realm_emoji import get_all_custom_emoji_for_realm
|
||||||
from zerver.models.realm_playgrounds import get_realm_playgrounds
|
from zerver.models.realm_playgrounds import get_realm_playgrounds
|
||||||
|
|
|
@ -1,15 +1,11 @@
|
||||||
from typing import Any, Callable, Dict, List, Tuple, TypeVar, Union
|
from typing import Dict, List, Tuple, TypeVar, Union
|
||||||
|
|
||||||
import orjson
|
|
||||||
from django.core.exceptions import ValidationError
|
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.db.backends.base.base import BaseDatabaseWrapper
|
from django.db.backends.base.base import BaseDatabaseWrapper
|
||||||
from django.db.models import CASCADE, QuerySet
|
from django.db.models import CASCADE
|
||||||
from django.db.models.signals import post_delete, post_save
|
from django.db.models.signals import post_delete, post_save
|
||||||
from django.db.models.sql.compiler import SQLCompiler
|
from django.db.models.sql.compiler import SQLCompiler
|
||||||
from django.utils.translation import gettext as _
|
from django_stubs_ext import ValuesQuerySet
|
||||||
from django.utils.translation import gettext_lazy
|
|
||||||
from django_stubs_ext import StrPromise, ValuesQuerySet
|
|
||||||
from typing_extensions import override
|
from typing_extensions import override
|
||||||
|
|
||||||
from zerver.lib.cache import (
|
from zerver.lib.cache import (
|
||||||
|
@ -17,26 +13,9 @@ from zerver.lib.cache import (
|
||||||
realm_alert_words_automaton_cache_key,
|
realm_alert_words_automaton_cache_key,
|
||||||
realm_alert_words_cache_key,
|
realm_alert_words_cache_key,
|
||||||
)
|
)
|
||||||
from zerver.lib.types import (
|
|
||||||
ExtendedFieldElement,
|
|
||||||
ExtendedValidator,
|
|
||||||
FieldElement,
|
|
||||||
ProfileDataElementBase,
|
|
||||||
ProfileDataElementValue,
|
|
||||||
RealmUserValidator,
|
|
||||||
UserFieldElement,
|
|
||||||
Validator,
|
|
||||||
)
|
|
||||||
from zerver.lib.validator import (
|
|
||||||
check_date,
|
|
||||||
check_int,
|
|
||||||
check_list,
|
|
||||||
check_long_string,
|
|
||||||
check_short_string,
|
|
||||||
check_url,
|
|
||||||
validate_select_field,
|
|
||||||
)
|
|
||||||
from zerver.models.clients import Client as Client
|
from zerver.models.clients import Client as Client
|
||||||
|
from zerver.models.custom_profile_fields import CustomProfileField as CustomProfileField
|
||||||
|
from zerver.models.custom_profile_fields import CustomProfileFieldValue as CustomProfileFieldValue
|
||||||
from zerver.models.drafts import Draft as Draft
|
from zerver.models.drafts import Draft as Draft
|
||||||
from zerver.models.groups import GroupGroupMembership as GroupGroupMembership
|
from zerver.models.groups import GroupGroupMembership as GroupGroupMembership
|
||||||
from zerver.models.groups import UserGroup as UserGroup
|
from zerver.models.groups import UserGroup as UserGroup
|
||||||
|
@ -96,7 +75,6 @@ from zerver.models.user_topics import UserTopic as UserTopic
|
||||||
from zerver.models.users import RealmUserDefault as RealmUserDefault
|
from zerver.models.users import RealmUserDefault as RealmUserDefault
|
||||||
from zerver.models.users import UserBaseSettings as UserBaseSettings
|
from zerver.models.users import UserBaseSettings as UserBaseSettings
|
||||||
from zerver.models.users import UserProfile as UserProfile
|
from zerver.models.users import UserProfile as UserProfile
|
||||||
from zerver.models.users import get_user_profile_by_id_in_realm
|
|
||||||
|
|
||||||
|
|
||||||
@models.Field.register_lookup
|
@models.Field.register_lookup
|
||||||
|
@ -152,167 +130,6 @@ def query_for_ids(
|
||||||
return query
|
return query
|
||||||
|
|
||||||
|
|
||||||
def check_valid_user_ids(realm_id: int, val: object, allow_deactivated: bool = False) -> List[int]:
|
|
||||||
user_ids = check_list(check_int)("User IDs", val)
|
|
||||||
realm = Realm.objects.get(id=realm_id)
|
|
||||||
for user_id in user_ids:
|
|
||||||
# TODO: Structurally, we should be doing a bulk fetch query to
|
|
||||||
# get the users here, not doing these in a loop. But because
|
|
||||||
# this is a rarely used feature and likely to never have more
|
|
||||||
# than a handful of users, it's probably mostly OK.
|
|
||||||
try:
|
|
||||||
user_profile = get_user_profile_by_id_in_realm(user_id, realm)
|
|
||||||
except UserProfile.DoesNotExist:
|
|
||||||
raise ValidationError(_("Invalid user ID: {user_id}").format(user_id=user_id))
|
|
||||||
|
|
||||||
if not allow_deactivated and not user_profile.is_active:
|
|
||||||
raise ValidationError(
|
|
||||||
_("User with ID {user_id} is deactivated").format(user_id=user_id)
|
|
||||||
)
|
|
||||||
|
|
||||||
if user_profile.is_bot:
|
|
||||||
raise ValidationError(_("User with ID {user_id} is a bot").format(user_id=user_id))
|
|
||||||
|
|
||||||
return user_ids
|
|
||||||
|
|
||||||
|
|
||||||
class CustomProfileField(models.Model):
|
|
||||||
"""Defines a form field for the per-realm custom profile fields feature.
|
|
||||||
|
|
||||||
See CustomProfileFieldValue for an individual user's values for one of
|
|
||||||
these fields.
|
|
||||||
"""
|
|
||||||
|
|
||||||
HINT_MAX_LENGTH = 80
|
|
||||||
NAME_MAX_LENGTH = 40
|
|
||||||
MAX_DISPLAY_IN_PROFILE_SUMMARY_FIELDS = 2
|
|
||||||
|
|
||||||
realm = models.ForeignKey(Realm, on_delete=CASCADE)
|
|
||||||
name = models.CharField(max_length=NAME_MAX_LENGTH)
|
|
||||||
hint = models.CharField(max_length=HINT_MAX_LENGTH, default="")
|
|
||||||
|
|
||||||
# Sort order for display of custom profile fields.
|
|
||||||
order = models.IntegerField(default=0)
|
|
||||||
|
|
||||||
# Whether the field should be displayed in smaller summary
|
|
||||||
# sections of a page displaying custom profile fields.
|
|
||||||
display_in_profile_summary = models.BooleanField(default=False)
|
|
||||||
|
|
||||||
SHORT_TEXT = 1
|
|
||||||
LONG_TEXT = 2
|
|
||||||
SELECT = 3
|
|
||||||
DATE = 4
|
|
||||||
URL = 5
|
|
||||||
USER = 6
|
|
||||||
EXTERNAL_ACCOUNT = 7
|
|
||||||
PRONOUNS = 8
|
|
||||||
|
|
||||||
# These are the fields whose validators require more than var_name
|
|
||||||
# and value argument. i.e. SELECT require field_data, USER require
|
|
||||||
# realm as argument.
|
|
||||||
SELECT_FIELD_TYPE_DATA: List[ExtendedFieldElement] = [
|
|
||||||
(SELECT, gettext_lazy("List of options"), validate_select_field, str, "SELECT"),
|
|
||||||
]
|
|
||||||
USER_FIELD_TYPE_DATA: List[UserFieldElement] = [
|
|
||||||
(USER, gettext_lazy("Person picker"), check_valid_user_ids, orjson.loads, "USER"),
|
|
||||||
]
|
|
||||||
|
|
||||||
SELECT_FIELD_VALIDATORS: Dict[int, ExtendedValidator] = {
|
|
||||||
item[0]: item[2] for item in SELECT_FIELD_TYPE_DATA
|
|
||||||
}
|
|
||||||
USER_FIELD_VALIDATORS: Dict[int, RealmUserValidator] = {
|
|
||||||
item[0]: item[2] for item in USER_FIELD_TYPE_DATA
|
|
||||||
}
|
|
||||||
|
|
||||||
FIELD_TYPE_DATA: List[FieldElement] = [
|
|
||||||
# Type, display name, validator, converter, keyword
|
|
||||||
(SHORT_TEXT, gettext_lazy("Short text"), check_short_string, str, "SHORT_TEXT"),
|
|
||||||
(LONG_TEXT, gettext_lazy("Long text"), check_long_string, str, "LONG_TEXT"),
|
|
||||||
(DATE, gettext_lazy("Date picker"), check_date, str, "DATE"),
|
|
||||||
(URL, gettext_lazy("Link"), check_url, str, "URL"),
|
|
||||||
(
|
|
||||||
EXTERNAL_ACCOUNT,
|
|
||||||
gettext_lazy("External account"),
|
|
||||||
check_short_string,
|
|
||||||
str,
|
|
||||||
"EXTERNAL_ACCOUNT",
|
|
||||||
),
|
|
||||||
(PRONOUNS, gettext_lazy("Pronouns"), check_short_string, str, "PRONOUNS"),
|
|
||||||
]
|
|
||||||
|
|
||||||
ALL_FIELD_TYPES = [*FIELD_TYPE_DATA, *SELECT_FIELD_TYPE_DATA, *USER_FIELD_TYPE_DATA]
|
|
||||||
|
|
||||||
FIELD_VALIDATORS: Dict[int, Validator[ProfileDataElementValue]] = {
|
|
||||||
item[0]: item[2] for item in FIELD_TYPE_DATA
|
|
||||||
}
|
|
||||||
FIELD_CONVERTERS: Dict[int, Callable[[Any], Any]] = {
|
|
||||||
item[0]: item[3] for item in ALL_FIELD_TYPES
|
|
||||||
}
|
|
||||||
FIELD_TYPE_CHOICES: List[Tuple[int, StrPromise]] = [
|
|
||||||
(item[0], item[1]) for item in ALL_FIELD_TYPES
|
|
||||||
]
|
|
||||||
|
|
||||||
field_type = models.PositiveSmallIntegerField(
|
|
||||||
choices=FIELD_TYPE_CHOICES,
|
|
||||||
default=SHORT_TEXT,
|
|
||||||
)
|
|
||||||
|
|
||||||
# A JSON blob of any additional data needed to define the field beyond
|
|
||||||
# type/name/hint.
|
|
||||||
#
|
|
||||||
# The format depends on the type. Field types SHORT_TEXT, LONG_TEXT,
|
|
||||||
# DATE, URL, and USER leave this empty. Fields of type SELECT store the
|
|
||||||
# choices' descriptions.
|
|
||||||
#
|
|
||||||
# Note: There is no performance overhead of using TextField in PostgreSQL.
|
|
||||||
# See https://www.postgresql.org/docs/9.0/static/datatype-character.html
|
|
||||||
field_data = models.TextField(default="")
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
unique_together = ("realm", "name")
|
|
||||||
|
|
||||||
@override
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return f"{self.realm!r} {self.name} {self.field_type} {self.order}"
|
|
||||||
|
|
||||||
def as_dict(self) -> ProfileDataElementBase:
|
|
||||||
data_as_dict: ProfileDataElementBase = {
|
|
||||||
"id": self.id,
|
|
||||||
"name": self.name,
|
|
||||||
"type": self.field_type,
|
|
||||||
"hint": self.hint,
|
|
||||||
"field_data": self.field_data,
|
|
||||||
"order": self.order,
|
|
||||||
}
|
|
||||||
if self.display_in_profile_summary:
|
|
||||||
data_as_dict["display_in_profile_summary"] = True
|
|
||||||
|
|
||||||
return data_as_dict
|
|
||||||
|
|
||||||
def is_renderable(self) -> bool:
|
|
||||||
if self.field_type in [CustomProfileField.SHORT_TEXT, CustomProfileField.LONG_TEXT]:
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
def custom_profile_fields_for_realm(realm_id: int) -> QuerySet[CustomProfileField]:
|
|
||||||
return CustomProfileField.objects.filter(realm=realm_id).order_by("order")
|
|
||||||
|
|
||||||
|
|
||||||
class CustomProfileFieldValue(models.Model):
|
|
||||||
user_profile = models.ForeignKey(UserProfile, on_delete=CASCADE)
|
|
||||||
field = models.ForeignKey(CustomProfileField, on_delete=CASCADE)
|
|
||||||
value = models.TextField()
|
|
||||||
rendered_value = models.TextField(null=True, default=None)
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
unique_together = ("user_profile", "field")
|
|
||||||
|
|
||||||
@override
|
|
||||||
def __str__(self) -> str:
|
|
||||||
return f"{self.user_profile!r} {self.field!r} {self.value}"
|
|
||||||
|
|
||||||
|
|
||||||
# Interfaces for services
|
# Interfaces for services
|
||||||
# They provide additional functionality like parsing message to obtain query URL, data to be sent to URL,
|
# They provide additional functionality like parsing message to obtain query URL, data to be sent to URL,
|
||||||
# and parsing the response.
|
# and parsing the response.
|
||||||
|
|
|
@ -0,0 +1,193 @@
|
||||||
|
from typing import Any, Callable, Dict, List, Tuple
|
||||||
|
|
||||||
|
import orjson
|
||||||
|
from django.core.exceptions import ValidationError
|
||||||
|
from django.db import models
|
||||||
|
from django.db.models import CASCADE, QuerySet
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.utils.translation import gettext_lazy
|
||||||
|
from django_stubs_ext import StrPromise
|
||||||
|
from typing_extensions import override
|
||||||
|
|
||||||
|
from zerver.lib.types import (
|
||||||
|
ExtendedFieldElement,
|
||||||
|
ExtendedValidator,
|
||||||
|
FieldElement,
|
||||||
|
ProfileDataElementBase,
|
||||||
|
ProfileDataElementValue,
|
||||||
|
RealmUserValidator,
|
||||||
|
UserFieldElement,
|
||||||
|
Validator,
|
||||||
|
)
|
||||||
|
from zerver.lib.validator import (
|
||||||
|
check_date,
|
||||||
|
check_int,
|
||||||
|
check_list,
|
||||||
|
check_long_string,
|
||||||
|
check_short_string,
|
||||||
|
check_url,
|
||||||
|
validate_select_field,
|
||||||
|
)
|
||||||
|
from zerver.models.realms import Realm
|
||||||
|
from zerver.models.users import UserProfile, get_user_profile_by_id_in_realm
|
||||||
|
|
||||||
|
|
||||||
|
def check_valid_user_ids(realm_id: int, val: object, allow_deactivated: bool = False) -> List[int]:
|
||||||
|
user_ids = check_list(check_int)("User IDs", val)
|
||||||
|
realm = Realm.objects.get(id=realm_id)
|
||||||
|
for user_id in user_ids:
|
||||||
|
# TODO: Structurally, we should be doing a bulk fetch query to
|
||||||
|
# get the users here, not doing these in a loop. But because
|
||||||
|
# this is a rarely used feature and likely to never have more
|
||||||
|
# than a handful of users, it's probably mostly OK.
|
||||||
|
try:
|
||||||
|
user_profile = get_user_profile_by_id_in_realm(user_id, realm)
|
||||||
|
except UserProfile.DoesNotExist:
|
||||||
|
raise ValidationError(_("Invalid user ID: {user_id}").format(user_id=user_id))
|
||||||
|
|
||||||
|
if not allow_deactivated and not user_profile.is_active:
|
||||||
|
raise ValidationError(
|
||||||
|
_("User with ID {user_id} is deactivated").format(user_id=user_id)
|
||||||
|
)
|
||||||
|
|
||||||
|
if user_profile.is_bot:
|
||||||
|
raise ValidationError(_("User with ID {user_id} is a bot").format(user_id=user_id))
|
||||||
|
|
||||||
|
return user_ids
|
||||||
|
|
||||||
|
|
||||||
|
class CustomProfileField(models.Model):
|
||||||
|
"""Defines a form field for the per-realm custom profile fields feature.
|
||||||
|
|
||||||
|
See CustomProfileFieldValue for an individual user's values for one of
|
||||||
|
these fields.
|
||||||
|
"""
|
||||||
|
|
||||||
|
HINT_MAX_LENGTH = 80
|
||||||
|
NAME_MAX_LENGTH = 40
|
||||||
|
MAX_DISPLAY_IN_PROFILE_SUMMARY_FIELDS = 2
|
||||||
|
|
||||||
|
realm = models.ForeignKey(Realm, on_delete=CASCADE)
|
||||||
|
name = models.CharField(max_length=NAME_MAX_LENGTH)
|
||||||
|
hint = models.CharField(max_length=HINT_MAX_LENGTH, default="")
|
||||||
|
|
||||||
|
# Sort order for display of custom profile fields.
|
||||||
|
order = models.IntegerField(default=0)
|
||||||
|
|
||||||
|
# Whether the field should be displayed in smaller summary
|
||||||
|
# sections of a page displaying custom profile fields.
|
||||||
|
display_in_profile_summary = models.BooleanField(default=False)
|
||||||
|
|
||||||
|
SHORT_TEXT = 1
|
||||||
|
LONG_TEXT = 2
|
||||||
|
SELECT = 3
|
||||||
|
DATE = 4
|
||||||
|
URL = 5
|
||||||
|
USER = 6
|
||||||
|
EXTERNAL_ACCOUNT = 7
|
||||||
|
PRONOUNS = 8
|
||||||
|
|
||||||
|
# These are the fields whose validators require more than var_name
|
||||||
|
# and value argument. i.e. SELECT require field_data, USER require
|
||||||
|
# realm as argument.
|
||||||
|
SELECT_FIELD_TYPE_DATA: List[ExtendedFieldElement] = [
|
||||||
|
(SELECT, gettext_lazy("List of options"), validate_select_field, str, "SELECT"),
|
||||||
|
]
|
||||||
|
USER_FIELD_TYPE_DATA: List[UserFieldElement] = [
|
||||||
|
(USER, gettext_lazy("Person picker"), check_valid_user_ids, orjson.loads, "USER"),
|
||||||
|
]
|
||||||
|
|
||||||
|
SELECT_FIELD_VALIDATORS: Dict[int, ExtendedValidator] = {
|
||||||
|
item[0]: item[2] for item in SELECT_FIELD_TYPE_DATA
|
||||||
|
}
|
||||||
|
USER_FIELD_VALIDATORS: Dict[int, RealmUserValidator] = {
|
||||||
|
item[0]: item[2] for item in USER_FIELD_TYPE_DATA
|
||||||
|
}
|
||||||
|
|
||||||
|
FIELD_TYPE_DATA: List[FieldElement] = [
|
||||||
|
# Type, display name, validator, converter, keyword
|
||||||
|
(SHORT_TEXT, gettext_lazy("Short text"), check_short_string, str, "SHORT_TEXT"),
|
||||||
|
(LONG_TEXT, gettext_lazy("Long text"), check_long_string, str, "LONG_TEXT"),
|
||||||
|
(DATE, gettext_lazy("Date picker"), check_date, str, "DATE"),
|
||||||
|
(URL, gettext_lazy("Link"), check_url, str, "URL"),
|
||||||
|
(
|
||||||
|
EXTERNAL_ACCOUNT,
|
||||||
|
gettext_lazy("External account"),
|
||||||
|
check_short_string,
|
||||||
|
str,
|
||||||
|
"EXTERNAL_ACCOUNT",
|
||||||
|
),
|
||||||
|
(PRONOUNS, gettext_lazy("Pronouns"), check_short_string, str, "PRONOUNS"),
|
||||||
|
]
|
||||||
|
|
||||||
|
ALL_FIELD_TYPES = [*FIELD_TYPE_DATA, *SELECT_FIELD_TYPE_DATA, *USER_FIELD_TYPE_DATA]
|
||||||
|
|
||||||
|
FIELD_VALIDATORS: Dict[int, Validator[ProfileDataElementValue]] = {
|
||||||
|
item[0]: item[2] for item in FIELD_TYPE_DATA
|
||||||
|
}
|
||||||
|
FIELD_CONVERTERS: Dict[int, Callable[[Any], Any]] = {
|
||||||
|
item[0]: item[3] for item in ALL_FIELD_TYPES
|
||||||
|
}
|
||||||
|
FIELD_TYPE_CHOICES: List[Tuple[int, StrPromise]] = [
|
||||||
|
(item[0], item[1]) for item in ALL_FIELD_TYPES
|
||||||
|
]
|
||||||
|
|
||||||
|
field_type = models.PositiveSmallIntegerField(
|
||||||
|
choices=FIELD_TYPE_CHOICES,
|
||||||
|
default=SHORT_TEXT,
|
||||||
|
)
|
||||||
|
|
||||||
|
# A JSON blob of any additional data needed to define the field beyond
|
||||||
|
# type/name/hint.
|
||||||
|
#
|
||||||
|
# The format depends on the type. Field types SHORT_TEXT, LONG_TEXT,
|
||||||
|
# DATE, URL, and USER leave this empty. Fields of type SELECT store the
|
||||||
|
# choices' descriptions.
|
||||||
|
#
|
||||||
|
# Note: There is no performance overhead of using TextField in PostgreSQL.
|
||||||
|
# See https://www.postgresql.org/docs/9.0/static/datatype-character.html
|
||||||
|
field_data = models.TextField(default="")
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
unique_together = ("realm", "name")
|
||||||
|
|
||||||
|
@override
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return f"{self.realm!r} {self.name} {self.field_type} {self.order}"
|
||||||
|
|
||||||
|
def as_dict(self) -> ProfileDataElementBase:
|
||||||
|
data_as_dict: ProfileDataElementBase = {
|
||||||
|
"id": self.id,
|
||||||
|
"name": self.name,
|
||||||
|
"type": self.field_type,
|
||||||
|
"hint": self.hint,
|
||||||
|
"field_data": self.field_data,
|
||||||
|
"order": self.order,
|
||||||
|
}
|
||||||
|
if self.display_in_profile_summary:
|
||||||
|
data_as_dict["display_in_profile_summary"] = True
|
||||||
|
|
||||||
|
return data_as_dict
|
||||||
|
|
||||||
|
def is_renderable(self) -> bool:
|
||||||
|
if self.field_type in [CustomProfileField.SHORT_TEXT, CustomProfileField.LONG_TEXT]:
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def custom_profile_fields_for_realm(realm_id: int) -> QuerySet[CustomProfileField]:
|
||||||
|
return CustomProfileField.objects.filter(realm=realm_id).order_by("order")
|
||||||
|
|
||||||
|
|
||||||
|
class CustomProfileFieldValue(models.Model):
|
||||||
|
user_profile = models.ForeignKey(UserProfile, on_delete=CASCADE)
|
||||||
|
field = models.ForeignKey(CustomProfileField, on_delete=CASCADE)
|
||||||
|
value = models.TextField()
|
||||||
|
rendered_value = models.TextField(null=True, default=None)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
unique_together = ("user_profile", "field")
|
||||||
|
|
||||||
|
@override
|
||||||
|
def __str__(self) -> str:
|
||||||
|
return f"{self.user_profile!r} {self.field!r} {self.value}"
|
|
@ -583,7 +583,8 @@ class UserProfile(AbstractBaseUser, PermissionsMixin, UserBaseSettings):
|
||||||
return str(self.ROLE_ID_TO_NAME_MAP[self.role])
|
return str(self.ROLE_ID_TO_NAME_MAP[self.role])
|
||||||
|
|
||||||
def profile_data(self) -> ProfileData:
|
def profile_data(self) -> ProfileData:
|
||||||
from zerver.models import CustomProfileFieldValue, custom_profile_fields_for_realm
|
from zerver.models import CustomProfileFieldValue
|
||||||
|
from zerver.models.custom_profile_fields import custom_profile_fields_for_realm
|
||||||
|
|
||||||
values = CustomProfileFieldValue.objects.filter(user_profile=self)
|
values = CustomProfileFieldValue.objects.filter(user_profile=self)
|
||||||
user_data = {
|
user_data = {
|
||||||
|
|
|
@ -15,12 +15,8 @@ from zerver.lib.external_accounts import DEFAULT_EXTERNAL_ACCOUNTS
|
||||||
from zerver.lib.markdown import markdown_convert
|
from zerver.lib.markdown import markdown_convert
|
||||||
from zerver.lib.test_classes import ZulipTestCase
|
from zerver.lib.test_classes import ZulipTestCase
|
||||||
from zerver.lib.types import ProfileDataElementUpdateDict, ProfileDataElementValue
|
from zerver.lib.types import ProfileDataElementUpdateDict, ProfileDataElementValue
|
||||||
from zerver.models import (
|
from zerver.models import CustomProfileField, CustomProfileFieldValue, UserProfile
|
||||||
CustomProfileField,
|
from zerver.models.custom_profile_fields import custom_profile_fields_for_realm
|
||||||
CustomProfileFieldValue,
|
|
||||||
UserProfile,
|
|
||||||
custom_profile_fields_for_realm,
|
|
||||||
)
|
|
||||||
from zerver.models.realms import get_realm
|
from zerver.models.realms import get_realm
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -72,9 +72,9 @@ from zerver.models import (
|
||||||
UserGroupMembership,
|
UserGroupMembership,
|
||||||
UserProfile,
|
UserProfile,
|
||||||
UserTopic,
|
UserTopic,
|
||||||
check_valid_user_ids,
|
|
||||||
)
|
)
|
||||||
from zerver.models.clients import get_client
|
from zerver.models.clients import get_client
|
||||||
|
from zerver.models.custom_profile_fields import check_valid_user_ids
|
||||||
from zerver.models.groups import SystemGroups
|
from zerver.models.groups import SystemGroups
|
||||||
from zerver.models.prereg_users import filter_to_valid_prereg_users
|
from zerver.models.prereg_users import filter_to_valid_prereg_users
|
||||||
from zerver.models.realms import InvalidFakeEmailDomainError, get_fake_email_domain, get_realm
|
from zerver.models.realms import InvalidFakeEmailDomainError, get_fake_email_domain, get_realm
|
||||||
|
|
|
@ -33,7 +33,8 @@ from zerver.lib.validator import (
|
||||||
check_union,
|
check_union,
|
||||||
validate_select_field_data,
|
validate_select_field_data,
|
||||||
)
|
)
|
||||||
from zerver.models import CustomProfileField, Realm, UserProfile, custom_profile_fields_for_realm
|
from zerver.models import CustomProfileField, Realm, UserProfile
|
||||||
|
from zerver.models.custom_profile_fields import custom_profile_fields_for_realm
|
||||||
|
|
||||||
|
|
||||||
def list_realm_custom_profile_fields(
|
def list_realm_custom_profile_fields(
|
||||||
|
|
|
@ -109,8 +109,8 @@ from zerver.models import (
|
||||||
UserGroup,
|
UserGroup,
|
||||||
UserGroupMembership,
|
UserGroupMembership,
|
||||||
UserProfile,
|
UserProfile,
|
||||||
custom_profile_fields_for_realm,
|
|
||||||
)
|
)
|
||||||
|
from zerver.models.custom_profile_fields import custom_profile_fields_for_realm
|
||||||
from zerver.models.realms import (
|
from zerver.models.realms import (
|
||||||
DisposableEmailError,
|
DisposableEmailError,
|
||||||
DomainNotAllowedForRealmError,
|
DomainNotAllowedForRealmError,
|
||||||
|
|
Loading…
Reference in New Issue