Source code for wxc_sdk.person_settings.permissions_out

"""
Outgoing permissions API and datatypes

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

from pydantic import validator, parse_obj_as, root_validator

from .common import PersonSettingsApiChild
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', 'TransferNumbersApi',
           'AuthCodesApi', 'OutgoingPermissionsApi']


[docs]class OutgoingPermissionCallType(str, Enum): """ call types for outgoing permissions """ internal_call = 'INTERNAL_CALL' local = 'LOCAL' toll_free = 'TOLL_FREE' 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 the person to transfer or forward a call of the specified call type. transfer_enabled: bool
[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' ...
@root_validator 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.__fields__: # try to parse unknown call type into CallTypePermission instance v[call_type] = CallTypePermission.parse_obj(v[call_type]) return v internal_call: Optional[CallTypePermission] local: Optional[CallTypePermission] toll_free: Optional[CallTypePermission] toll: Optional[CallTypePermission] national: Optional[CallTypePermission] international: Optional[CallTypePermission] operator_assisted: Optional[CallTypePermission] chargeable_directory_assisted: Optional[CallTypePermission] special_services_i: Optional[CallTypePermission] special_services_ii: Optional[CallTypePermission] premium_services_i: Optional[CallTypePermission] premium_services_ii: Optional[CallTypePermission]
[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() return self.__dict__.get(call_type, None)
[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.__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): """ Person's Outgoing Permission Settings """ #: When true, indicates that this user uses the specified calling permissions when placing outbound calls. use_custom_enabled: Optional[bool] #: Specifies the outbound calling permissions settings. calling_permissions: Optional[CallingPermissions] @validator('calling_permissions', pre=True) 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.pop('callType') r[call_type.lower()] = entry return r # noinspection PyMethodOverriding def 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 """ if drop_call_types is None: # default call types to be excluded from updates drop_call_types = {'url_dialing', 'unknown', 'casual'} data = self.dict(exclude={'calling_permissions'}, by_alias=True) if self.calling_permissions is not None: permissions = [] for call_type, call_type_permission in self.calling_permissions.__dict__.items(): call_type_permission: CallTypePermission if not call_type_permission or (call_type in drop_call_types): continue ct_dict = call_type_permission.dict(by_alias=True) ct_dict['callType'] = call_type.upper() permissions.append(ct_dict) data['callingPermissions'] = permissions return json.dumps(data)
[docs]class AutoTransferNumbers(ApiModel): """ Outgoing permission auto transfer numbers """ #: 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] #: 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] #: 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] @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.dict() for k in data: data[k] = data[k] or '' return AutoTransferNumbers.parse_obj(data)
[docs]class TransferNumbersApi(PersonSettingsApiChild): """ API for outgoing permission auto transfer numbers """ feature = 'outgoingPermission/autoTransferNumbers'
[docs] def read(self, person_id: str, org_id: str = None) -> AutoTransferNumbers: """ Retrieve Transfer Numbers Settings for a Workspace. When calling a specific call type, this workspace 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 workspace settings. :param person_id: Unique identifier for the workspace. :type person_id: str :param org_id: Workspace 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(person_id=person_id) params = org_id and {'orgId': org_id} or None data = self.get(url, params=params) return AutoTransferNumbers.parse_obj(data)
[docs] def configure(self, person_id: str, settings: AutoTransferNumbers, org_id: str = None): """ Modify Transfer Numbers Settings for a Place. When calling a specific call type, this workspace 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 workspace settings. :param person_id: Unique identifier for the workspace. :type person_id: str :param settings: new auto transfer numbers :type settings: :class:`AutoTransferNumbers` :param org_id: Workspace 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(person_id=person_id) params = org_id and {'orgId': org_id} or None body = settings.json() self.put(url, params=params, data=body)
[docs]class AuthCodesApi(PersonSettingsApiChild): """ API for person's outgoing permission authorization codes """ feature = 'outgoingPermission/authorizationCodes'
[docs] def read(self, person_id: str, org_id: str = None) -> list[AuthCode]: """ Retrieve Authorization codes for a Workspace. Authorization 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 workspace settings. :param person_id: Unique identifier for the workspace. :type person_id: str :param org_id: Workspace 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: list of authorization codes :rtype: list of :class:`AuthCode` """ url = self.f_ep(person_id=person_id) params = org_id and {'orgId': org_id} or None data = self.get(url, params=params) return parse_obj_as(list[AuthCode], data['authorizationCodes'])
[docs] def delete_codes(self, person_id: str, access_codes: list[Union[str, AuthCode]], org_id: str = None): """ Modify Authorization codes for a workspace. Authorization 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 person_id: Unique identifier for the workspace. :type person_id: str :param access_codes: authorization codes to remove :type access_codes: list[str] :param org_id: Workspace 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(person_id=person_id) params = org_id and {'orgId': org_id} or None body = {'deleteCodes': [ac.code if isinstance(ac, AuthCode) else ac for ac in access_codes]} self.put(url, params=params, json=body)
[docs] def create(self, person_id: str, code: str, description: str, org_id: str = None): """ Modify Authorization codes for a workspace. Authorization 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 person_id: Unique identifier for the workspace. :type person_id: str :param code: Indicates an authorization code. :type code: str :param description: Indicates the description of the authorization code. :type description: str :param org_id: Workspace 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(person_id=person_id) params = org_id and {'orgId': org_id} or None body = {'code': code, 'description': description} self.post(url, params=params, json=body)
[docs]@dataclass(init=False) class OutgoingPermissionsApi(PersonSettingsApiChild): """ API for person's outgoing permissions settings also used for workspace and location outgoing permissions """ #: Only available for workspaces and locations transfer_numbers: TransferNumbersApi #: Only available for workspaces auth_codes: AuthCodesApi feature = 'outgoingPermission' def __init__(self, *, session: RestSession, workspaces: bool = False, locations: bool = False): super().__init__(session=session, workspaces=workspaces, locations=locations) if workspaces: # auto transfer numbers API seems to only exist for workspaces self.transfer_numbers = TransferNumbersApi(session=session, workspaces=True) self.auth_codes = AuthCodesApi(session=session, workspaces=True) elif locations: self.transfer_numbers = TransferNumbersApi(session=session, locations=True) self.auth_codes = None else: self.transfer_numbers = None self.auth_codes = None
[docs] def read(self, person_id: str, org_id: str = None) -> OutgoingPermissions: """ Retrieve a Person's 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 person_id: Unique identifier for the person. :type person_id: str :param org_id: Person 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(person_id=person_id) params = org_id and {'orgId': org_id} or None return OutgoingPermissions.parse_obj(self.get(ep, params=params))
[docs] def configure(self, person_id: str, settings: OutgoingPermissions, drop_call_types: set[str] = None, org_id: str = None): """ Configure a Person's Outgoing Calling Permissions Settings Turn on outgoing call settings for this person 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 person_id: Unique identifier for the person. :type person_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: Person 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(person_id=person_id) params = org_id and {'orgId': org_id} or None self.put(ep, params=params, data=settings.json(drop_call_types=drop_call_types))