"""
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
[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]
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)
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)
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))