Source code for wxc_sdk.person_settings.permissions_out

"""
Outgoing permissions API and datatypes

API is used in:
    * person settings
    * location settings
    * workspace settings
    * virtual line settings
"""
import json
from dataclasses import dataclass
from typing import Optional, Union

from pydantic import field_validator, model_validator

from .common import PersonSettingsApiChild, ApiSelector
from ..base import ApiModel
from ..base import SafeEnum as Enum
from ..common import AuthCode
from ..rest import RestSession

__all__ = ['OutgoingPermissionCallType', 'Action', 'CallTypePermission', 'CallingPermissions',
           'OutgoingPermissions', 'AutoTransferNumbers', 'DigitPattern', 'DigitPatterns', 'DigitPatternsApi',
           'TransferNumbersApi', 'AuthCodes', 'AccessCodesApi', 'OutgoingPermissionsApi']


[docs] class OutgoingPermissionCallType(str, Enum): """ call types for outgoing permissions """ internal_call = 'INTERNAL_CALL' # TODO: remove when CALL-336 is GA local = 'LOCAL' toll_free = 'TOLL_FREE' # TODO: remove when CALL-336 is GA toll = 'TOLL' national = 'NATIONAL' international = 'INTERNATIONAL' operator_assisted = 'OPERATOR_ASSISTED' chargeable_directory_assisted = 'CHARGEABLE_DIRECTORY_ASSISTED' special_services_i = 'SPECIAL_SERVICES_I' special_services_ii = 'SPECIAL_SERVICES_II' premium_services_i = 'PREMIUM_SERVICES_I' premium_services_ii = 'PREMIUM_SERVICES_II'
[docs] class Action(str, Enum): """ Action on a specific call type """ allow = 'ALLOW' block = 'BLOCK' auth_code = 'AUTH_CODE' transfer_number_1 = 'TRANSFER_NUMBER_1' transfer_number_2 = 'TRANSFER_NUMBER_2' transfer_number_3 = 'TRANSFER_NUMBER_3'
[docs] class CallTypePermission(ApiModel): """ Permission for a specific call type """ #: Action on the given call_type. action: Action #: Allow to transfer or forward a call of the specified call type. transfer_enabled: bool #: If `true`, indicates the call restriction is enabled for the specific call type. is_call_type_restriction_enabled: Optional[bool] = None
[docs] @staticmethod def default() -> 'CallTypePermission': return CallTypePermission(action=Action.allow, transfer_enabled=True)
[docs] class CallingPermissions(ApiModel): """ Calling permissions for all call types """
[docs] class Config: # allow undefined attributes (new call types) extra = 'allow' ...
@model_validator(mode='before') def rv(cls, v): """ :meta private: Make sure that values for unknown call types are also parsed into CallTypePermission instances """ for call_type in v: if call_type not in cls.model_fields: # try to parse unknown call type into CallTypePermission instance v[call_type] = CallTypePermission.model_validate(v[call_type]) return v internal_call: Optional[CallTypePermission] = None local: Optional[CallTypePermission] = None toll_free: Optional[CallTypePermission] = None toll: Optional[CallTypePermission] = None national: Optional[CallTypePermission] = None international: Optional[CallTypePermission] = None operator_assisted: Optional[CallTypePermission] = None chargeable_directory_assisted: Optional[CallTypePermission] = None special_services_i: Optional[CallTypePermission] = None special_services_ii: Optional[CallTypePermission] = None premium_services_i: Optional[CallTypePermission] = None premium_services_ii: Optional[CallTypePermission] = None
[docs] def for_call_type(self, call_type: OutgoingPermissionCallType) -> Optional[CallTypePermission]: """ get call type setting for a specific call type :param call_type: call type :type call_type: :class:`OutgoingPermissionCallType` :return: permissions :rtype: :class:`CallTypePermission` """ try: # if parameter is an actual Enum we want to use the name for attribute access call_type = call_type.name except AttributeError: # ignore AttributeError; call_type most probably was a string already -> use lower case as attribute name call_type = call_type.lower() try: return getattr(self, call_type) except AttributeError: return None
[docs] def permissions_dict(self) -> dict[str, CallTypePermission]: """ Get a dictionary of all permissions. Key is the call type and the value is the permission for this call type :return: dict of all permissions """ return {ct: ctp for ct, ctp in self}
[docs] @staticmethod def allow_all() -> 'CallingPermissions': """ most permissive permissions :return: :class:`CallingPermissions` instance allowing all call types :rtype: CallingPermissions """ init_dict = {call_type: CallTypePermission(action=Action.allow, transfer_enabled=True) for call_type in CallingPermissions.model_fields} return CallingPermissions(**init_dict)
[docs] @staticmethod def default() -> 'CallingPermissions': """ default settings :return: :class:`CallingPermissions` :rtype: CallingPermissions """ # allow all call types except for a few r = CallingPermissions.allow_all() for call_type in (OutgoingPermissionCallType.international, OutgoingPermissionCallType.premium_services_i, OutgoingPermissionCallType.premium_services_ii): ctp = r.for_call_type(call_type) ctp.transfer_enabled = False ctp.action = Action.block return r
[docs] class OutgoingPermissions(ApiModel): """ Outgoing Permission Settings """ #: When true, indicates that this user uses the shared control that applies to all outgoing call settings #: categories when placing outbound calls. use_custom_enabled: Optional[bool] = None #: When true, indicates that this user uses the specified outgoing calling permissions when placing outbound calls. use_custom_permissions: Optional[bool] = None #: Specifies the outbound calling permissions settings. calling_permissions: Optional[CallingPermissions] = None @field_validator('calling_permissions', mode='before') def transform_calling_permissions(cls, v): """ calling permissions are returned by the API as a list of triples: "callingPermissions": [ { "action": "ALLOW", "transferEnabled": true, "callType": "INTERNAL_CALL" }, { "action": "ALLOW", "transferEnabled": true, "callType": "LOCAL" }, ... The validator transforms this to a dict that can be deserialized to a :class:`CallingPermissions` instance: "callingPermissions": { "internal_call": { "action": "ALLOW", "transferEnabled": true }, "local": { "action": "ALLOW", "transferEnabled": true } } :meta private: """ if not isinstance(v, list): return v r = {} for entry in v: call_type = entry['callType'] r[call_type.lower()] = {k: v for k, v in entry.items() if k != 'callType'} return r # noinspection PyMethodOverriding def model_dump(self, drop_call_types: set[str] = None) -> dict: """ :meta private: calling permissions are converted back to a list of objects. drop_call_types can be a set of call types to be excluded from callingPermissions """ if drop_call_types is None: # default call types to be excluded from updates drop_call_types = {'url_dialing', 'unknown', 'casual'} # dump w/o calling_permissions; we build that extra data = super().model_dump(exclude={'calling_permissions'}, by_alias=True) if self.calling_permissions is not None: permissions = [] for call_type, call_type_permission in self.calling_permissions: call_type_permission: CallTypePermission if not call_type_permission or (call_type in drop_call_types): continue ct_dict = call_type_permission.model_dump(by_alias=True) ct_dict['callType'] = call_type.upper() permissions.append(ct_dict) data['callingPermissions'] = permissions return data def model_dump_json(self, drop_call_types: set[str] = None) -> str: """ :meta private: calling permissions are converted back to a list of objects. drop_call_types can be a set of call types to be excluded from callingPermissions """ return json.dumps(self.model_dump(drop_call_types))
[docs] class AutoTransferNumbers(ApiModel): """ Outgoing permission auto transfer numbers """ #: When `true`, use custom settings for the transfer numbers category of outgoing call permissions. use_custom_transfer_numbers: Optional[bool] = None #: Calls placed meeting the criteria in an outbound rule whose action is TRANSFER_NUMBER_1 will be transferred to #: this number auto_transfer_number1: Optional[str] = None #: Calls placed meeting the criteria in an outbound rule whose action is TRANSFER_NUMBER_2 will be transferred to #: this number auto_transfer_number2: Optional[str] = None #: Calls placed meeting the criteria in an outbound rule whose action is TRANSFER_NUMBER_3 will be transferred to #: this number auto_transfer_number3: Optional[str] = None @property def configure_unset_numbers(self) -> 'AutoTransferNumbers': """ Unset numbers are returned by the API as null (None). To set a number back to unset an empty strings has to be set. This property returns an :class:`AutoTransferNumbers` instance where the numbers are set to an empty string instead of None :return: auto transfer numbers with empty strings instead of None :rtype: :class:`AutoTransferNumbers` """ data = self.model_dump(by_alias=True) for k in data: if not k.startswith('autoTransferNumber'): continue data[k] = data[k] or '' return AutoTransferNumbers.model_validate(data)
[docs] class DigitPattern(ApiModel): #: A unique identifier for the digit pattern. id: Optional[str] = None #: A unique name for the digit pattern. name: Optional[str] = None #: The digit pattern to be matched with the input number. pattern: Optional[str] = None #: Action to be performed on the input number that matches with the digit pattern. action: Optional[Action] = None #: Option to allow or disallow transfer of calls. transfer_enabled: Optional[bool] = None def create_or_update(self) -> dict: """ data for create or update :meta private: """ return self.model_dump(mode='json', exclude_none=True, exclude={'id'}, by_alias=True)
[docs] class DigitPatterns(ApiModel): #: When `true`, use custom settings for the digit patterns category of outgoing call permissions. use_custom_digit_patterns: Optional[bool] = None #: List of digit patterns. digit_patterns: Optional[list[DigitPattern]] = None
[docs] class TransferNumbersApi(PersonSettingsApiChild): """ API for outgoing permission auto transfer numbers """ feature = 'outgoingPermission/autoTransferNumbers'
[docs] def read(self, entity_id: str, org_id: str = None) -> AutoTransferNumbers: """ Retrieve Transfer Numbers Settings . When calling a specific call type, this entity will be automatically transferred to another number. The person assigned the Auto Transfer Number can then approve the call and send it through or reject the call type. You can add up to 3 numbers. This API requires a full or read-only administrator auth token with a scope of spark-admin:workspaces_read or a user auth token with spark:workspaces_read scope can be used to read entity settings. :param entity_id: Unique identifier for the entity. :type entity_id: str :param org_id: entity is in this organization. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access API. :type org_id: str :return: auto transfer numbers :rtype: :class:`AutoTransferNumbers` """ url = self.f_ep(entity_id) params = org_id and {'orgId': org_id} or None data = self.get(url, params=params) return AutoTransferNumbers.model_validate(data)
[docs] def configure(self, entity_id: str, settings: AutoTransferNumbers, org_id: str = None): """ Modify Transfer Numbers Settings for an entity. When calling a specific call type, this entity will be automatically transferred to another number. The person assigned the Auto Transfer Number can then approve the call and send it through or reject the call type. You can add up to 3 numbers. This API requires a full or user administrator auth token with the spark-admin:workspaces_write scope or a user auth token with spark:workspaces_write scope can be used to update entity settings. :param entity_id: Unique identifier for the entity. :type entity_id: str :param settings: new auto transfer numbers :type settings: :class:`AutoTransferNumbers` :param org_id: entity is in this organization. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access API. :type org_id: str """ url = self.f_ep(entity_id) params = org_id and {'orgId': org_id} or None body = settings.model_dump_json() self.put(url, params=params, data=body)
[docs] class AuthCodes(ApiModel): #: When `true`, use custom settings for the access codes category of outgoing call permissions. use_custom_access_codes: Optional[bool] = None #: Indicates the set of activation codes and description. access_codes: Optional[list[AuthCode]] = None
[docs] class AccessCodesApi(PersonSettingsApiChild): """ API for outgoing permission access codes: locations, persons, workspaces, virtual lines """ feature = 'outgoingPermission/accessCodes'
[docs] def read(self, entity_id: str, org_id: str = None) -> AuthCodes: """ Retrieve Access codes. Access codes are used to bypass permissions. This API requires a full or read-only administrator auth token with a scope of spark-admin:workspaces_read or a user auth token with spark:workspaces_read scope can be used to read entity settings. :param entity_id: Unique identifier for the entity. :type entity_id: str :param org_id: ID of the organization within which the entity resides. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access API. :type org_id: str :return: authorization codes :rtype: :class:`AuthCodes` """ url = self.f_ep(entity_id) params = org_id and {'orgId': org_id} or None data = self.get(url, params=params) return AuthCodes.model_validate(data)
[docs] def modify(self, entity_id: str, use_custom_access_codes: bool = None, delete_codes: list[Union[str, AuthCode]] = None, org_id: str = None): """ Modify Access Codes Access codes are used to bypass permissions. This API requires a full or user administrator or location administrator auth token with the `spark-admin:telephony_config_write` scope. :param entity_id: Unique identifier for the entity. :type entity_id: str :param use_custom_access_codes: When `true`, use custom settings for the access codes category of outgoing call permissions. :type use_custom_access_codes: bool :param delete_codes: Indicates access codes to delete. :type delete_codes: list[str] :param org_id: ID of the organization in which the entity resides. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access the API. :type org_id: str :rtype: None """ url = self.f_ep(entity_id) params = org_id and {'orgId': org_id} or None if org_id is not None: params['orgId'] = org_id body = dict() if use_custom_access_codes is not None: body['useCustomAccessCodes'] = use_custom_access_codes if delete_codes is not None: body['deleteCodes'] = [ac.code if isinstance(ac, AuthCode) else ac for ac in delete_codes] super().put(url, params=params, json=body)
[docs] def create(self, entity_id: str, code: str, description: str, org_id: str = None): """ Create new Access codes. Access codes are used to bypass permissions. This API requires a full or user administrator auth token with the spark-admin:workspaces_write scope or a user auth token with spark:workspaces_write scope can be used to update workspace settings. :param entity_id: Unique identifier for the entity. :type entity_id: str :param code: Indicates an access code. :type code: str :param description: Indicates the description of the access code. :type description: str :param org_id: ID of the organization within which the entity resides. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access API. :type org_id: str """ url = self.f_ep(entity_id) params = org_id and {'orgId': org_id} or None body = {'code': code, 'description': description} self.post(url, params=params, json=body)
[docs] def delete(self, entity_id: str, org_id: str = None): """ Delete Access Code Deletes all Access codes for the given entity. Access codes are used to bypass permissions. This API requires a full or user administrator or location administrator auth token with the `spark-admin:workspaces_write` scope or a user auth token with `spark:workspaces_write` scope can be used to update entity settings. :param entity_id: Unique identifier for the entity. :type entity_id: str :param org_id: ID of the organization within which the entity resides. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access the API. :type org_id: str :rtype: None """ params = {} if org_id is not None: params['orgId'] = org_id url = self.f_ep(entity_id) super().delete(url, params=params)
[docs] class DigitPatternsApi(PersonSettingsApiChild): feature = 'outgoingPermission/digitPatterns'
[docs] def get_digit_patterns(self, entity_id: str, org_id: str = None) -> DigitPatterns: """ Retrieve Digit Patterns Retrieve digit patterns. Digit patterns are used to bypass permissions. Retrieving digit patterns requires a full or user or read-only administrator or location administrator auth token with a scope of `spark-admin:telephony_config_read`. :param entity_id: Unique identifier for location, person, workspace, or virtual line. :type entity_id: str :param org_id: ID of the organization in which the entity resides. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access the API. :type org_id: str :rtype: :class:`UserOutgoingPermissionDigitPatternGetListObject` """ params = org_id and {'orgId': org_id} or None url = self.f_ep(entity_id) data = super().get(url, params=params) r = DigitPatterns.model_validate(data) return r
[docs] def details(self, entity_id: str, digit_pattern_id: str, org_id: str = None) -> DigitPattern: """ Retrieve Digit Pattern Details Retrieve the digit pattern details. Digit patterns are used to bypass permissions. Retrieving the digit pattern details requires a full or user or read-only administrator or location administrator auth token with a scope of `spark-admin:telephony_config_read`. :param entity_id: Unique identifier for location, person, workspace, or virtual line. :type entity_id: str :param digit_pattern_id: Unique identifier for the digit pattern. :type digit_pattern_id: str :param org_id: ID of the organization in which the entity resides. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access the API. :type org_id: str :rtype: :class:`UserDigitPatternObject` """ params = org_id and {'orgId': org_id} or None url = self.f_ep(entity_id, path=digit_pattern_id) data = super().get(url, params=params) r = DigitPattern.model_validate(data) return r
[docs] def create(self, entity_id: str, pattern: DigitPattern, org_id: str = None) -> str: """ Create Digit Patterns Create new digit pattern. Digit patterns are used to bypass permissions. Creating the digit pattern requires a full or user or location administrator auth token with a scope of `spark-admin:telephony_config_write`. :param entity_id: Unique identifier for location, person, workspace, or virtual line. :type entity_id: str :param pattern: new digit pattern :type pattern: :class:`DigitPattern` :param org_id: ID of the organization in which the entity resides. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access the API. :type org_id: str :rtype: str """ params = org_id and {'orgId': org_id} or None body = pattern.create_or_update() url = self.f_ep(entity_id) data = super().post(url, params=params, json=body) r = data['id'] return r
[docs] def update_category_control_settings(self, entity_id: str, use_custom_digit_patterns: bool = None, org_id: str = None): """ Modify the Digit Pattern Category Control Settings for the entity Modifies whether this user uses the specified digit patterns when placing outbound calls or not. Updating the digit pattern category control settings requires a full or user or location administrator auth token with a scope of `spark-admin:telephony_config_write`. :param entity_id: Unique identifier for location, person, workspace, or virtual line. :type entity_id: str :param use_custom_digit_patterns: When `true`, use custom settings for the digit patterns category of outgoing call permissions. :type use_custom_digit_patterns: bool :param org_id: ID of the organization in which the entity resides. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access the API. :type org_id: str :rtype: None """ params = org_id and {'orgId': org_id} or None body = dict() if use_custom_digit_patterns is not None: body['useCustomDigitPatterns'] = use_custom_digit_patterns url = self.f_ep(entity_id) super().put(url, params=params, json=body)
[docs] def update(self, entity_id: str, settings: DigitPattern, org_id: str = None): """ Modify Digit Patterns Modify Digit Patterns Digit patterns are used to bypass permissions. Updating the digit pattern requires a full or user or location administrator auth token with a scope of `spark-admin:telephony_config_write`. :param entity_id: Unique identifier for location, person, workspace, or virtual line. :type entity_id: str :param settings: new digit pattern settings :type settings: :class:`DigitPattern` :param org_id: ID of the organization in which the entity resides. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access the API. :type org_id: str :rtype: None """ params = org_id and {'orgId': org_id} or None body = settings.create_or_update() url = self.f_ep(entity_id, path=settings.id) super().put(url, params=params, json=body)
[docs] def delete(self, entity_id: str, digit_pattern_id: str, org_id: str = None): """ Delete a Digit Pattern Delete Digit Pattern. Digit patterns are used to bypass permissions. Deleting the digit pattern requires a full or user or location administrator auth token with a scope of `spark-admin:telephony_config_write`. :param entity_id: Unique identifier for location, person, workspace, or virtual line. :type entity_id: str :param digit_pattern_id: Unique identifier for the digit pattern. :type digit_pattern_id: str :param org_id: ID of the organization in which the entity resides. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access the API. :type org_id: str :rtype: None """ params = org_id and {'orgId': org_id} or None url = self.f_ep(entity_id, path=digit_pattern_id) super().delete(url, params=params)
[docs] def delete_all(self, entity_id: str, org_id: str = None): """ Delete all Digit Patterns. Digit patterns are used to bypass permissions. Deleting the digit patterns requires a full or user or location administrator auth token with a scope of `spark-admin:telephony_config_write`. :param entity_id: Unique identifier for location, person, workspace, or virtual line. :type entity_id: str :param org_id: ID of the organization in which the entity resides. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access the API. :type org_id: str :rtype: None """ params = org_id and {'orgId': org_id} or None url = self.f_ep(entity_id) super().delete(url, params=params)
[docs] @dataclass(init=False, repr=False) class OutgoingPermissionsApi(PersonSettingsApiChild): """ API for outgoing permissions settings Also used for user, workspace, location, and virtual line outgoing permissions """ transfer_numbers: TransferNumbersApi access_codes: AccessCodesApi digit_patterns: DigitPatternsApi feature = 'outgoingPermission' def __getattribute__(self, item): if item == 'access_codes' and self.selector == ApiSelector.location: raise AttributeError("access_codes API is not available for locations. Use the telephony access_codes API") return super().__getattribute__(item)
[docs] def __init__(self, *, session: RestSession, selector: ApiSelector = 'person'): super().__init__(session=session, selector=selector) self.transfer_numbers = TransferNumbersApi(session=session, selector=selector) if selector == ApiSelector.location: # Apparently there is a difference between access code API for locations on one hand and users, # workspaces, and virtual lines one the other. # For locations, we can create multiple access codes at once. self.access_codes = None # instead use the access_codes API at the telephony level else: self.access_codes = AccessCodesApi(session=session, selector=selector) self.digit_patterns = DigitPatternsApi(session=session, selector=selector)
[docs] def read(self, entity_id: str, org_id: str = None) -> OutgoingPermissions: """ Retrieve Outgoing Calling Permissions Settings You can change the outgoing calling permissions for a person if you want them to be different from your organization's default. This API requires a full, user, or read-only administrator auth token with a scope of spark-admin:people_read. :param entity_id: Unique identifier for the entity. :type entity_id: str :param org_id: Entity is in this organization. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access API. :type org_id: str :return: outgoing permission settings for specific user :rtype: :class:`OutgoingPermissions` """ ep = self.f_ep(entity_id) params = org_id and {'orgId': org_id} or None return OutgoingPermissions.model_validate(self.get(ep, params=params))
[docs] def configure(self, entity_id: str, settings: OutgoingPermissions, drop_call_types: set[str] = None, org_id: str = None): """ Configure Outgoing Calling Permissions Settings Turn on outgoing call settings for this entity to override the calling settings from the location that are used by default. This API requires a full or user administrator auth token with the spark-admin:people_write scope or a user auth token with spark:people_write scope can be used by a person to update their own settings. :param entity_id: Unique identifier for the entity. :type entity_id: str :param settings: new setting to be applied :type settings: :class:`OutgoingPermissions` :param drop_call_types: set of call type names to be excluded from updates. Default is the set of call_types known to be not supported for updates :type drop_call_types: set[str] :param org_id: Entity is in this organization. Only admin users of another organization (such as partners) may use this parameter as the default is the same organization as the token used to access API. """ ep = self.f_ep(entity_id) params = org_id and {'orgId': org_id} or None self.put(ep, params=params, data=settings.model_dump_json(drop_call_types=drop_call_types))