"""
Telephony types and API (location and organisation settings)
"""
from collections.abc import Generator
from dataclasses import dataclass
from typing import Optional
from pydantic import Field, TypeAdapter, field_validator
from wxc_sdk.telephony.emergency_address import EmergencyAddressApi
from .caller_reputation import CallerReputationProviderApi
from ..api_child import ApiChild
from ..base import ApiModel, enum_str, plus1, to_camel
from ..base import SafeEnum as Enum
from ..common import (
DeviceCustomization,
IdAndName,
NumberOwner,
NumberState,
OwnerType,
RouteIdentity,
UserType,
ValidateExtensionsResponse,
ValidatePhoneNumbersResponse,
)
from ..common.schedules import ScheduleApi, ScheduleApiBase
from ..person_settings.common import ApiSelector
from ..person_settings.msteams import OrgMSTeamsSettingApi
from ..person_settings.permissions_out import Action, OutgoingPermissionsApi
from ..rest import RestSession
from .access_codes import LocationAccessCodesApi
from .announcements_repo import AnnouncementsRepositoryApi
from .autoattendant import AutoAttendantApi
from .call_contols_members import CallControlsMembersApi
from .call_recording import CallRecordingSettingsApi
from .call_routing import CallRoutingApi
from .callpark import CallParkApi
from .callpark_extension import CallparkExtensionApi
from .callpickup import CallPickupApi
from .callqueue import CallQueueApi
from .calls import CallsApi
from .conference import ConferenceControlsApi
from .cx_essentials import CustomerExperienceEssentialsApi
from .dect_devices import DECTDevicesApi
from .devices import SupportedDevices, TelephonyDevicesApi
from .emergency_services import OrgEmergencyServicesApi
from .guest_calling import GuestCallingApi
from .hotdesk import HotDeskApi
from .hotdesking_voiceportal import HotDeskingSigninViaVoicePortalApi
from .huntgroup import HuntGroupApi
from .jobs import JobsApi
from .location import TelephonyLocationApi
from .location.intercept import LocationInterceptApi
from .operating_modes import OperatingModesApi
from .org_access_codes import OrganisationAccessCodesApi
from .organisation_vm import OrganisationVoicemailSettingsAPI
from .paging import PagingApi
from .playlists import PlayListApi
from .pnc import PrivateNetworkConnectApi
from .prem_pstn import PremisePstnApi
from .pstn import PSTNApi
from .supervisor import SupervisorApi
from .virtual_extensions import VirtualExtensionsApi
from .virtual_line import VirtualLinesApi
from .vm_rules import VoicemailRulesApi
from .voice_messaging import VoiceMessagingApi
from .voicemail_groups import VoicemailGroupsApi
from .voiceportal import VoicePortalApi
__all__ = ['NumberListPhoneNumberType', 'TelephonyType',
'NumberListPhoneNumber',
'NumberType', 'NumberDetails', 'UCMProfile',
'TestCallRoutingResult', 'OriginatorType', 'CallSourceType', 'CallSourceInfo', 'DestinationType',
'LocationAndNumbers', 'HostedUserDestination', 'ServiceType', 'HostedFeatureDestination',
'TrunkDestination', 'PbxUserDestination', 'PstnNumberDestination', 'VirtualExtensionDestination',
'RouteListDestination', 'FeatureAccessCodeDestination', 'EmergencyDestination',
'TranslationPatternConfigurationLevel', 'AppliedServiceTranslationPattern', 'ConfigurationLevel',
'CallInterceptDetailsPermission', 'CallInterceptDetails', 'CallingPlanReason',
'OutgoingCallingPlanPermissionsByType', 'OutgoingCallingPlanPermissionsByDigitPattern', 'AppliedService',
'AnnouncementLanguage', 'TelephonyApi',
'MoHTheme', 'MoHConfig', 'NameAndCode', 'OrgCallCaptions', 'LargeOrgStatus']
[docs]
class NumberListPhoneNumberType(str, Enum):
primary = 'PRIMARY'
alternate = 'ALTERNATE'
fax = 'FAX'
dnis = 'DNIS'
default = 'Default'
[docs]
class TelephonyType(str, Enum):
#: The number is a PSTN number.
pstn_number = 'PSTN_NUMBER'
#: The number is a mobile number.
mobile_number = 'MOBILE_NUMBER'
[docs]
class NumberListPhoneNumber(ApiModel):
"""
Phone Number
"""
#: A unique identifier for the PSTN phone number.
phone_number: Optional[str] = None
#: Extension for a PSTN phone number.
extension: Optional[str] = None
#: Routing prefix of location.
routing_prefix: Optional[str] = None
#: Routing prefix + extension of a person or workspace.
esn: Optional[str] = None
#: Phone number's state.
state: Optional[NumberState] = None
#: Type of phone number.
phone_number_type: Optional[NumberListPhoneNumberType] = None
#: Indicates if the phone number is used as location clid.
main_number: bool
#: Indicates if a phone number is a toll-free number.
toll_free_number: bool
#: Indicates Telephony type for the number.
included_telephony_types: Optional[TelephonyType] = None
#: Mobile Network for the number if number is MOBILE_NUMBER.
mobile_network: Optional[str] = None
#: Routing Profile for the number if number is MOBILE_NUMBER.
routing_profile: Optional[str] = None
#: If `true`, the phone number is a service number; otherwise, it is a standard number.
is_service_number: Optional[bool] = None
location: IdAndName
owner: Optional[NumberOwner] = None
[docs]
class NumberType(str, Enum):
extension = 'EXTENSION'
number = 'NUMBER'
[docs]
class NumberDetails(ApiModel):
#: Count of phone numbers that are in the assigned state.
assigned: Optional[int] = None
#: Count of phone numbers which are in the un-assigned state.
un_assigned: Optional[int] = None
#: Count of phone numbers which are inactive.
in_active: Optional[int] = None
#: Count of extensions only without phone number.
extension_only: Optional[int] = None
#: Count of the toll free numbers.
toll_free_numbers: Optional[int] = None
#: Total phone numbers and extensions available.
total: Optional[int] = None
#: Count of phone numbers of type `MOBILE_NUMBER` only without `PSTN_NUMBER` and extension.
mobile_number: Optional[int] = None
service_number: Optional[int] = None
[docs]
class UCMProfile(ApiModel):
#: A unique identifier for the calling UC Manager Profile.
profile_id: str = Field(alias='id')
#: Unique name for the calling UC Manager Profile.
name: str
[docs]
class OriginatorType(str, Enum):
#: User
user = 'USER'
#: Connection between Webex Calling and the premises.
trunk = 'TRUNK'
[docs]
class CallSourceType(str, Enum):
"""
The type of call source.
"""
#: Indicates that the call source is a route list
route_list = 'ROUTE_LIST'
#: Indicates that the call source is a dial pattern.
dial_pattern = 'DIAL_PATTERN'
#: Indicates that the call source extension is unknown.
unknown_extension = 'UNKNOWN_EXTENSION'
#: Indicates that the call source phone number is unknown.
unknown_number = 'UNKNOWN_NUMBER'
[docs]
class CallSourceInfo(ApiModel):
"""
This data object is only returned when originatorNumber is specified in the request.
"""
#: The type of call source.
call_source_type: CallSourceType
#: When originatorType is "trunk", originatorId is a valid trunk, this trunk belongs to a route group which is
#: assigned to a route list with the name routeListA, and originatorNumber is a number assigned to routeListA.
#: routeListA is returned here. This element is returned when callSourceType is ROUTE_LIST.
route_list_name: Optional[str] = None
#: route list id
route_list_id: Optional[str] = None
#: When originatorType is "trunk", originatorId is a valid trunk with name trunkA, trunkA belongs to a route group
#: which is assigned to a route list with name routeListA, trunkA is also assigned to dialPlanA as routing choice,
#: dialPlanA has dialPattern xxxx assigned. If the originatorNumber match the dialPattern xxxx, dialPlanA is
#: returned. This element is returned when callSourceType is DIAL_PATTERN.
dial_plan_name: Optional[str] = None
#: When originatorType is "trunk", originatorId is a valid trunk with the name trunkA, trunkA belongs to a route
#: group which is assigned to a route list with the name routeListA, trunkA is also assigned to dialPlanA as routing
#: choice, dialPlanA has dialPattern xxxx assigned. If the originatorNumber match the dialPattern xxxx, dialPattern
#: xxxx is returned. This element is returned when callSourceType is DIAL_PATTERN.
dial_pattern: Optional[str] = None
#: dial plan id
dial_plan_id: Optional[str] = None
[docs]
class DestinationType(str, Enum):
"""
Matching destination type for the call.
"""
#: Matching destination is a person or workspace with details in the hosted_user field.
hosted_agent = 'HOSTED_AGENT'
#: Matching destination is a calling feature like auto-attendant or hunt group with details in the hostedFeature
#: field.
hosted_feature = 'HOSTED_FEATURE'
#: Matching destination routes into a separate PBX with details in the pbxUser field.
pbx_user = 'PBX_USER'
#: Matching destination routes into a PSTN phone number with details in the pstnNumber field.
pstn_number = 'PSTN_NUMBER'
#: Matching destination routes into a virtual extension with details in the virtualExtension field.
virtual_extension = 'VIRTUAL_EXTENSION'
#: Matching destination routes into a virtual extension range with details in the virtualExtensionRange field.
virtual_extension_range = 'VIRTUAL_EXTENSION_RANGE'
#: Matching destination routes into a route list with details in the routeList field.
route_list = 'ROUTE_LIST'
#: Matching destination routes into a feature access code (FAC) with details in the featureAccessCode field.
fac = 'FAC'
#: Matching destination routes into an emergency service like Red Sky, with details in the emergency field.
emergency = 'EMERGENCY'
#: The route is in a repair state with routing choice details in the repair field.
repair = 'REPAIR'
#: Target extension is unknown with routing choice details in the unknownExtension field.
unknown_extension = 'UNKNOWN_EXTENSION'
#: The target phone number is unknown with routing choice details in the unknownNumber field.
unknown_number = 'UNKNOWN_NUMBER'
[docs]
class LocationAndNumbers(ApiModel):
@field_validator('phone_number', mode='before')
@classmethod
def e164(cls, v):
"""
:meta private:
"""
return plus1(v)
location_name: str
location_id: str
phone_number: Optional[str] = None
extension: Optional[str] = None
[docs]
class HostedUserDestination(LocationAndNumbers):
"""
This data object is returned when destinationType is HOSTED_USER.
"""
#: person/workspace's id
hu_id: str = Field(alias='id')
#: Type of agent for call destination.
hu_type: UserType = Field(alias='type')
#: Person or workspace's first name.
first_name: str
#: Person or workspace's last name.
last_name: str
[docs]
class ServiceType(str, Enum):
#: Indicates that this destination is an auto attendant.
auto_attendant = 'AUTO_ATTENDANT'
#: Indicates that this destination is the Office (Broadworks) Anywhere feature.
broadworks_anywhere = 'BROADWORKS_ANYWHERE'
#: Indicates that this destination is the Call Queue feature.
call_queue = 'CALL_QUEUE'
#: Indicates that this destination is the Contact Center Link feature.
contact_center_link = 'CONTACT_CENTER_LINK'
#: Indicates that this destination is the Group Paging feature.
group_paging = 'GROUP_PAGING'
#: Indicates that this destination is the Hunt Group feature.
hunt_group = 'HUNT_GROUP'
#: Indicates that this destination is the Voice Messaging feature.
voice_messaging = 'VOICE_MESSAGING'
#: Indicates that this destination is the Voice Mail Group feature.
voice_mail_group = 'VOICE_MAIL_GROUP'
[docs]
class HostedFeatureDestination(LocationAndNumbers):
"""
This data object is returned when destinationType is HOSTED_FEATURE
"""
service_type: ServiceType = Field(alias='type')
#: service instance name
name: str
#: service instance id
service_instance_id: str = Field(alias='id')
[docs]
class TrunkDestination(ApiModel):
trunk_name: Optional[str] = None
trunk_id: Optional[str] = None
route_group_name: Optional[str] = None
route_group_id: Optional[str] = None
#: location of the trunk, required if trunkName is returned
trunk_location_name: Optional[str] = None
#: location id of the trunk, required if trunkName is returned
trunk_location_id: Optional[str] = None
[docs]
class PbxUserDestination(TrunkDestination):
"""
This data object is returned when destinationType is PBX_USER.
"""
#: the dial plan name that the called string matches
dial_plan_name: str
dial_plan_id: str
#: the dial pattern that the called string matches
dial_pattern: str
[docs]
class PstnNumberDestination(TrunkDestination):
pass
[docs]
class VirtualExtensionDestination(LocationAndNumbers, TrunkDestination):
#: Virtual extension ID.
virtual_extension_id: str = Field(alias='id')
#: Virtual extension display first name.
first_name: str
#: Virtual extension display last name.
last_name: str
#: Virtual extension display name.
display_name: str
class VirtualExtensionRange(LocationAndNumbers, TrunkDestination):
#: Virtual extension range ID.
id: Optional[str] = None
#: Virtual extension range name.
name: Optional[str] = None
#: Prefix that the virtual extension range is associated with (Note: Standard mode must have leading '+' in prefix;
#: BCD/Enhanced mode can have any valid prefix).
prefix: Optional[str] = None
#: Pattern associated with the virtual extension range.
pattern: Optional[str] = None
[docs]
class RouteListDestination(ApiModel):
"""
This data object is returned when destinationType is ROUTE_LIST.
"""
route_list_id: str = Field(alias='id')
name: str
route_group_name: str
route_group_id: str
location_name: str
location_id: str
[docs]
class FeatureAccessCodeDestination(ApiModel):
"""
This data object is returned when destinationType is FAC.
"""
code: str
name: str
[docs]
class EmergencyDestination(TrunkDestination):
"""
This data object is returned when destinationType is EMERGENCY.
"""
is_red_sky: bool
[docs]
class TranslationPatternConfigurationLevel(str, Enum):
#: The applied services of location level.
location = 'LOCATION'
#: The applied services of the organization level.
organization = 'ORGANIZATION'
[docs]
class AppliedServiceTranslationPattern(ApiModel):
#: The level from which the configuration is applied.
configuration_level: Optional[TranslationPatternConfigurationLevel] = None
#: Name given to a translation pattern.
name: Optional[str] = None
#: Matching pattern given to a translation pattern.
matching_pattern: Optional[str] = None
#: Replacement pattern given to a translation pattern.
replacement_pattern: Optional[str] = None
#: The original called number.
matched_number: Optional[str] = None
#: The modified number after matching against `matchingPattern` and replacing with corresponding
#: `replacementPattern`.
translated_number: Optional[str] = None
[docs]
class ConfigurationLevel(str, Enum):
#: The applied services at the location level.
location = 'LOCATION'
#: The applied services at the place level.
place = 'PLACE'
#: The applied services at the virtual line level.
virtual_line = 'VIRTUAL_LINE'
#: The applied services at the people level.
people = 'PEOPLE'
[docs]
class CallInterceptDetailsPermission(str, Enum):
#: Call intercept is disabled.
disallow = 'DISALLOW'
#: Call intercept is transferred to a number.
transfer = 'TRANSFER'
[docs]
class CallInterceptDetails(ApiModel):
#: The level from which the configuration is applied.
configuration_level: Optional[ConfigurationLevel] = None
#: The choices that indicate call intercept permissions.
permission: Optional[CallInterceptDetailsPermission] = None
#: The number to which the outgoing permission by type is to be transferred.
transfer_number: Optional[str] = None
[docs]
class CallingPlanReason(str, Enum):
#: Calling plan gives the Fraud Containment reason.
fraud_containment = 'FRAUD_CONTAINMENT'
#: The Cisco calling plan reason.
cisco_calling_plan = 'CISCO_CALLING_PLAN'
#: The reason if the transfer number 1 is not configured.
transfer_number_1_not_configured = 'TRANSFER_NUMBER_1_NOT_CONFIGURED'
#: The reason if the transfer number 2 is not configured.
transfer_number_2_not_configured = 'TRANSFER_NUMBER_2_NOT_CONFIGURED'
#: The reason if the transfer number 3 is not configured.
transfer_number_3_not_configured = 'TRANSFER_NUMBER_3_NOT_CONFIGURED'
#: The reason for Webex mobile international transfer forward.
webex_mobile_premium_international_transfer_forward = 'WEBEX_MOBILE_PREMIUM_INTERNATIONAL_TRANSFER_FORWARD'
[docs]
class OutgoingCallingPlanPermissionsByType(ApiModel):
#: The level from which the configuration is applied.
configuration_level: Optional[ConfigurationLevel] = None
#: Designates the action to be taken for each call type and if transferring the call type is allowed.
call_type: Optional[str] = None
#: Action to be performed on the input number that matches with the OCP.
permission: Optional[Action] = None
#: The number to which the outgoing permission by type is to be transferred.
transfer_number: Optional[str] = None
#: The reason for the result reported for non-standard OCP service.
reason: Optional[CallingPlanReason] = None
#: A transfer number is present in case it gets transferred to some other number.
number: Optional[str] = None
[docs]
class OutgoingCallingPlanPermissionsByDigitPattern(ApiModel):
#: The level from which the configuration is applied.
configuration_level: Optional[ConfigurationLevel] = None
#: Name given to a digit pattern.
name: Optional[str] = None
#: Action to be performed on the input number that matches with the digit pattern.
permission: Optional[Action] = None
transfer_number: Optional[str] = None
#: Pattern is given to a digit pattern.
pattern: Optional[str] = None
#: The reason for the result reported for non-standard OCP service.
reason: Optional[CallingPlanReason] = None
#: A transfer number is present in case it gets transferred to some other number.
number: Optional[str] = None
[docs]
class AppliedService(ApiModel):
#: Returns the details of the Translation Pattern if applied.
translation_pattern: Optional[AppliedServiceTranslationPattern] = None
#: Returns the details of call intercept if applied.
intercept_details: Optional[CallInterceptDetails] = None
#: Returns the details of permissions by type configuration if applied under OCP.
outgoing_calling_plan_permissions_by_type: Optional[OutgoingCallingPlanPermissionsByType] = None
#: Returns the details of digit pattern configuration if applied under OCP.
outgoing_calling_plan_permissions_by_digit_pattern: Optional[OutgoingCallingPlanPermissionsByDigitPattern] = None
[docs]
class TestCallRoutingResult(ApiModel):
#: Language for call queue.
language: Optional[str] = None
#: Time zone for the call queue.
time_zone: Optional[str] = None
#: This data object is only returned when originatorNumber is specified in the request.
call_source_info: Optional[CallSourceInfo] = None
#: Matching destination type for the call.
destination_type: DestinationType
#: FAC code if destinationType is FAC. The routing address will be returned for all other destination types.
routing_address: str
#: Outside access code.
outside_access_code: Optional[str] = None
#: true if the call would be rejected.
is_rejected: bool
#: This data object is returned when destinationType is HOSTED_USER.
hosted_user: Optional[HostedUserDestination] = Field(alias='hostedAgent', default=None)
#: This data object is returned when destinationType is HOSTED_FEATURE.
hosted_feature: Optional[HostedFeatureDestination] = None
#: This data object is returned when destinationType is PBX_USER.
pbx_user: Optional[PbxUserDestination] = None
#: This data object is returned when destinationType is PSTN_NUMBER.
pstn_number: Optional[PstnNumberDestination] = None
#: This data object is returned when destinationType is VIRTUAL_EXTENSION.
virtual_extension: Optional[VirtualExtensionDestination] = None
#: Returned when destinationType is VIRTUAL_EXTENSION_RANGE.
virtual_extension_range: Optional[VirtualExtensionRange] = None
#: This data object is returned when destinationType is ROUTE_LIST.
route_list: Optional[RouteListDestination] = None
#: This data object is returned when destinationType is FAC.
feature_access_code: Optional[FeatureAccessCodeDestination] = None
#: This data object is returned when destinationType is EMERGENCY.
emergency: Optional[EmergencyDestination] = None
#: This data object is returned when destinationType is REPAIR.
repair: Optional[TrunkDestination] = None
#: This data object is returned when destinationType is UNKNOWN_EXTENSION.
unknown_extension: Optional[TrunkDestination] = None
#: This data object is returned when destinationType is UNKNOWN_NUMBER.
unknown_number: Optional[TrunkDestination] = None
#: Returned if any origin is configured with intercept details, outgoing permissions by type, digit pattern, or
#: translation pattern.
applied_services: Optional[list[AppliedService]] = None
[docs]
class AnnouncementLanguage(ApiModel):
#: Language name.
name: Optional[str] = None
#: Language Code
code: Optional[str] = None
NameAndCode = AnnouncementLanguage
[docs]
class MoHTheme(str, Enum):
#: Legacy MoH Theme
LEGACY = 'LEGACY'
#: Opus MoH Theme
OPUS = 'OPUS'
[docs]
class MoHConfig(ApiModel):
#: Default org level Music on Hold option, can be one of two options: Choose between Opus Number 1
#: (Music On Hold used in other Cisco products like UCM) and existing legacy Music On Hold.
default_org_moh: MoHTheme
[docs]
class OrgCallCaptions(ApiModel):
#: Organization-level closed captions are enabled or disabled.
org_closed_captions_enabled: Optional[bool] = None
#: Organization-level transcripts are enabled or disabled.
org_transcripts_enabled: Optional[bool] = None
def update(self) -> dict:
"""
data for update
:meta private:
"""
return self.model_dump(mode='json', by_alias=True, exclude_unset=True, exclude_none=True)
[docs]
class LargeOrgStatus(ApiModel):
#: `true` if the organization is categorized as large organization.
is_large_org: Optional[bool] = None
#: The threshold percentage represents the percentage of the threshold reached to categorize an organization as a
#: large organization.
large_org_threshold_percentage: Optional[int] = None
[docs]
@dataclass(init=False, repr=False)
class TelephonyApi(ApiChild, base='telephony/config'):
"""
The telephony settings (features) API.
"""
#: access or authentication codes at location level
access_codes: LocationAccessCodesApi
announcements_repo: AnnouncementsRepositoryApi
auto_attendant: AutoAttendantApi
#: location call intercept settings
call_controls_members: CallControlsMembersApi
call_intercept: LocationInterceptApi
call_routing: CallRoutingApi
caller_reputation_provider: CallerReputationProviderApi
calls: CallsApi
callpark: CallParkApi
callpark_extension: CallparkExtensionApi
callqueue: CallQueueApi
call_recording: CallRecordingSettingsApi
conference: ConferenceControlsApi
cx_essentials: CustomerExperienceEssentialsApi
dect_devices: DECTDevicesApi
#: WxC device operations
devices: TelephonyDevicesApi
emergency_address: EmergencyAddressApi
#: emergency services
emergency_services: OrgEmergencyServicesApi
guest_calling: GuestCallingApi
hotdesk: HotDeskApi
hotdesking_voiceportal: HotDeskingSigninViaVoicePortalApi
huntgroup: HuntGroupApi
jobs: JobsApi
#: location specific settings
location: TelephonyLocationApi
locations: TelephonyLocationApi
ms_teams: OrgMSTeamsSettingApi
operating_modes: OperatingModesApi
#: organisation access codes
organisation_access_codes: OrganisationAccessCodesApi
#: organisation voicemail settings
organisation_voicemail: OrganisationVoicemailSettingsAPI
paging: PagingApi
#: location level outgoing permissions
permissions_out: OutgoingPermissionsApi
pickup: CallPickupApi
playlist: PlayListApi
pnc: PrivateNetworkConnectApi
prem_pstn: PremisePstnApi
pstn: PSTNApi
schedules: ScheduleApi
supervisors: SupervisorApi
virtual_extensions: VirtualExtensionsApi
virtual_lines: VirtualLinesApi
# location voicemail groups
voicemail_groups: VoicemailGroupsApi
voicemail_rules: VoicemailRulesApi
voice_messaging: VoiceMessagingApi
voiceportal: VoicePortalApi
[docs]
def __init__(self, session: RestSession):
super().__init__(session=session)
self.access_codes = LocationAccessCodesApi(session=session)
self.announcements_repo = AnnouncementsRepositoryApi(session=session)
self.auto_attendant = AutoAttendantApi(session=session)
self.call_controls_members = CallControlsMembersApi(session=session)
self.call_intercept = LocationInterceptApi(session=session)
self.call_routing = CallRoutingApi(session=session)
self.call_recording = CallRecordingSettingsApi(session=session)
self.caller_reputation_provider = CallerReputationProviderApi(session=session)
self.calls = CallsApi(session=session)
self.callpark = CallParkApi(session=session)
self.callpark_extension = CallparkExtensionApi(session=session)
self.callqueue = CallQueueApi(session=session)
self.conference = ConferenceControlsApi(session=session)
self.cx_essentials = CustomerExperienceEssentialsApi(session=session)
self.dect_devices = DECTDevicesApi(session=session)
self.devices = TelephonyDevicesApi(session=session)
self.emergency_address = EmergencyAddressApi(session=session)
self.emergency_services = OrgEmergencyServicesApi(session=session)
self.guest_calling = GuestCallingApi(session=session)
self.hotdesk = HotDeskApi(session=session)
self.hotdesking_voiceportal = HotDeskingSigninViaVoicePortalApi(session=session)
self.huntgroup = HuntGroupApi(session=session)
self.jobs = JobsApi(session=session)
self.location = TelephonyLocationApi(session=session)
self.locations = self.location
self.ms_teams = OrgMSTeamsSettingApi(session=session)
self.operating_modes = OperatingModesApi(session=session)
self.organisation_access_codes = OrganisationAccessCodesApi(session=session)
self.organisation_voicemail = OrganisationVoicemailSettingsAPI(session=session)
self.paging = PagingApi(session=session)
self.permissions_out = OutgoingPermissionsApi(session=session, selector=ApiSelector.location)
self.pickup = CallPickupApi(session=session)
self.playlist = PlayListApi(session=session)
self.pnc = PrivateNetworkConnectApi(session=session)
self.prem_pstn = PremisePstnApi(session=session)
self.pstn = PSTNApi(session=session)
self.schedules = ScheduleApi(session=session, base=ScheduleApiBase.locations)
self.supervisors = SupervisorApi(session=session)
self.virtual_extensions = VirtualExtensionsApi(session=session)
self.virtual_lines = VirtualLinesApi(session=session)
self.voicemail_groups = VoicemailGroupsApi(session=session)
self.voicemail_rules = VoicemailRulesApi(session=session)
self.voice_messaging = VoiceMessagingApi(session=session)
self.voiceportal = VoicePortalApi(session=session)
[docs]
def phone_numbers(self, location_id: str = None, phone_number: str = None, available: bool = None,
order: str = None,
owner_name: str = None, owner_id: str = None, owner_type: OwnerType = None,
extension: str = None, number_type: NumberType = None,
phone_number_type: NumberListPhoneNumberType = None,
state: NumberState = None, details: bool = None, toll_free_numbers: bool = None,
restricted_non_geo_numbers: bool = None,
included_telephony_type: TelephonyType = None,
service_number: bool = None,
org_id: str = None, **params) -> Generator[NumberListPhoneNumber, None, None]:
"""
Get Phone Numbers for an Organization with Given Criterias
List all the phone numbers for the given organization along with the status and owner (if any).
Numbers can be standard, service, or mobile. Both standard and service numbers are PSTN numbers.
Service numbers are considered as high-utilization or high-concurrency phone numbers and can be assigned to
features like auto-attendants, call queues, and hunt groups.
Phone numbers can be linked to a specific location, be active or inactive, and be assigned or unassigned.
The owner of a number is the person, workspace, or feature to which the number is assigned.
Only a person can own a mobile number.
Retrieving this list requires a full or read-only administrator or location administrator auth token with a
scope of `spark-admin:telephony_config_read`.
:param location_id: Return the list of phone numbers for this location within the given organization.
:type location_id: str
:param phone_number: Search for this phone number.
:type phone_number: str
:param available: Search among the available phone numbers. This parameter cannot be used along with owner_type
parameter when set to true.
:type available: bool
:param order: Sort the list of phone numbers based on the following:lastName,dn,extension. Default sort will
be based on number and extension in an Ascending order
:type order: str
:param owner_name: Return the list of phone numbers that is owned by given owner name. Maximum length is 255.
:type owner_name: str
:param owner_id: Returns only the matched number/extension entries assigned to the feature with specified
uuid/broadsoftId.
:type owner_id: str
:param owner_type: Returns the list of phone numbers that are of given owner_type.
:type owner_type: OwnerType
:param extension: Returns the list of PSTN phone numbers with given extension.
:type extension: str
:param number_type: Returns the filtered list of phone numbers that contain a given type of number. `available`
or `state` query parameters cannot be used when `numberType=EXTENSION`.
:type number_type: NumberType
:param phone_number_type: Returns the filtered list of PSTN phone numbers that are of given phoneNumberType.
:type phone_number_type: NumberListPhoneNumberType
:param state: Returns the list of PSTN phone numbers with matching state.
:type state: NumberState
:param details: Returns the overall count of the PSTN phone numbers along with other details for given
organization.
:type details: bool
:param toll_free_numbers: Returns the list of toll free phone numbers.
:type toll_free_numbers: bool
:param restricted_non_geo_numbers: Returns the list of restricted non geographical numbers.
:type restricted_non_geo_numbers: bool
:param included_telephony_type: Returns the list of phone numbers that are of given `includedTelephonyType`.
By default if this query parameter is not provided, it will list both PSTN and Mobile Numbers. Possible
input values are PSTN_NUMBER, MOBILE_NUMBER.
:type included_telephony_type: TelephonyType
:param service_number: Returns the list of service phone numbers.
:type service_number: bool
:param org_id: List numbers for this organization.
:type org_id: str
:return: yields :class:`NumberListPhoneNumber` instances
"""
params.update((to_camel(p), v) for i, (p, v) in enumerate(locals().items())
if i and v is not None and p != 'params')
# parameter is actually called included_telephony_types
if itp := params.pop('includedTelephonyType', None):
params['includedTelephonyTypes'] = itp
for param, value in params.items():
if isinstance(value, bool):
value = 'true' if value else 'false'
params[param] = value
elif isinstance(value, Enum):
value = value.value
params[param] = value
url = self.ep(path='numbers')
# noinspection PyTypeChecker
return self.session.follow_pagination(url=url, model=NumberListPhoneNumber, params=params,
item_key='phoneNumbers')
[docs]
def phone_number_details(self, org_id: str = None) -> NumberDetails:
"""
get summary (counts) of phone numbers
:param org_id: details for numbers in this organization.
:type org_id: str
:return: phone number details
:rtype: :class:`NumberDetails`
"""
params = {to_camel(p): v for i, (p, v) in enumerate(locals().items())
if i and v is not None}
params['details'] = 'true'
params['max'] = 1
url = self.ep(path='numbers')
data = self.get(url, params=params)
return NumberDetails.model_validate(data['count'])
[docs]
def validate_extensions(self, extensions: list[str]) -> ValidateExtensionsResponse:
"""
Validate the List of Extensions
Validates the list of Extensions provided by the customer at the organization level. It checks the extension
meets the current extension length limits and does not conflict with the extensions of organization-level
entities and settings. To check for extension use across all locations, use the `Get Phone Numbers
<https://developer.webex.com/v1/telephony/config/numbers>`_ API. To
validate an extension and check for conflicts for a specific location, use the `Validate Extensions
<https://developer.webex.com/docs/api/v1/location-call-settings/validate-extensions>`_ API.
Retrieving this list requires a full or read-only administrator or location administrator auth token with a
scope of `spark-admin:telephony_config_read`.
:param extensions: Array of Strings of ID of Extensions.
:type extensions: list[str]
:return: validation response
:rtype: :class:`wxc_sdk.common.ValidateExtensionsResponse`
"""
url = self.ep(path='actions/validateExtensions/invoke')
data = self.post(url, json={'extensions': extensions})
return ValidateExtensionsResponse.model_validate(data)
[docs]
def validate_phone_numbers(self, phone_numbers: list[str], org_id: str = None) -> ValidatePhoneNumbersResponse:
"""
Validate phone numbers
Validate the list of phone numbers in an organization. Each phone number's availability is indicated in the
response.
Each location has a set of phone numbers that can be assigned to people, workspaces, or features. Phone numbers
must follow E.164 format for all countries, except for the United States, which can also follow the National
format. Active phone numbers are in service.
Validating a phone number in an organization requires a full administrator auth token with a scope
of spark-admin:telephony_config_write.
:param phone_numbers: List of phone numbers to be validated.
:type phone_numbers: list[str]
:param org_id: Organization of the Route Group.
:type org_id: str
:return: validation result
:rtype: :class:`wxc_sdk.common.ValidatePhoneNumbersResponse`
"""
url = self.ep('actions/validateNumbers/invoke')
body = {'phoneNumbers': phone_numbers}
params = org_id and {'orgId': org_id} or None
data = self.post(url=url, params=params, json=body)
return ValidatePhoneNumbersResponse.model_validate(data)
[docs]
def ucm_profiles(self, org_id: str = None) -> list[UCMProfile]:
"""
Read the List of UC Manager Profiles
List all calling UC Manager Profiles for the organization.
UC Manager Profiles are applicable if your organization uses Jabber in Team Messaging mode or Calling in Webex
(Unified CM).
The UC Manager Profile has an organization-wide default and may be overridden for individual persons, although
currently only setting at a user level is supported by Webex APIs.
Retrieving this list requires a full or read-only administrator auth token with a scope
of `spark-admin:people_read` as this API is designed to be used in conjunction with calling behavior at the user
level.
:param org_id: List manager profiles in this organization.
:type org_id: str
:return: list of :class:`UCMProfile`
"""
params = org_id and {'orgId': org_id} or None
url = self.ep(path='callingProfiles')
data = self.get(url, params=params)
return TypeAdapter(list[UCMProfile]).validate_python(data['callingProfiles'])
[docs]
def route_choices(self, route_group_name: str = None, trunk_name: str = None, order: str = None,
org_id: str = None) -> Generator[RouteIdentity, None, None]:
"""
Read the List of Routing Choices
List all Routes for the organization.
Trunk and Route Group qualify as Route. Trunks and Route Groups provide you the ability to configure Webex
Calling to manage calls between Webex Calling hosted users and premises PBX(s) users. This solution lets you
configure users to use Cloud PSTN (CCP or Cisco PSTN) or Premises-based PSTN.
Retrieving this list requires a full or read-only administrator auth token with a scope
of spark-admin:telephony_config_read.
:param route_group_name: Return the list of route identities matching the route group name.
:param trunk_name: Return the list of route identities matching the trunk name.
:param order: Order the route identities according to the designated fields.
Available sort fields: routeName, routeType.
:param org_id: List route identities for this organization.
:return:
"""
params = {to_camel(p): v for i, (p, v) in enumerate(locals().items())
if i and v is not None}
url = self.ep('routeChoices')
# noinspection PyTypeChecker
return self.session.follow_pagination(url=url, model=RouteIdentity, params=params, item_key='routeIdentities')
[docs]
def test_call_routing(self, originator_id: str, originator_type: OriginatorType, destination: str,
originator_number: str = None, include_applied_services: bool = None,
org_id: str = None) -> TestCallRoutingResult:
"""
Test Call Routing
Validates that an incoming call can be routed.
Dial plans route calls to on-premises destinations by use of trunks or route groups.
They are configured globally for an enterprise and apply to all users, regardless of location.
A dial plan also specifies the routing choice (trunk or route group) for calls that match any of its dial
patterns.
Specific dial patterns can be defined as part of your dial plan.
Test call routing requires a full administrator auth token with a scope of
`spark-admin:telephony_config_write`.
:param originator_id: This element is used to identify the originating party. It can be a person ID or a trunk
ID.
:type originator_id: str
:param originator_type: This element is used to identify if the `originatorId` is of type `PEOPLE` or `TRUNK`.
:type originator_type: OriginatorType
:param destination: This element specifies the called party. It can be any dialable string, for example, an ESN
number, E.164 number, hosted user DN, extension, extension with location code, URL, or FAC code.
:type destination: str
:param originator_number: Only used when `originatorType` is `TRUNK`. The `originatorNumber` can be a phone
number or URI.
:type originator_number: str
:param include_applied_services: This element is used to retrieve if any translation pattern, call intercept,
permission by type or permission by digit pattern is present for the called party.
:type include_applied_services: bool
:param org_id: Organization in which we are validating a call routing.
:type org_id: str
:rtype: :class:`TestCallRoutingResult`
"""
params = org_id and {'orgId': org_id} or None
body = dict()
body['originatorId'] = originator_id
body['originatorType'] = enum_str(originator_type)
if originator_number is not None:
body['originatorNumber'] = originator_number
body['destination'] = destination
if include_applied_services is not None:
body['includeAppliedServices'] = include_applied_services
url = self.ep('actions/testCallRouting/invoke')
data = self.post(url=url, params=params, json=body)
return TestCallRoutingResult.model_validate(data)
[docs]
def supported_devices(self, allow_configure_layout_enabled: bool = None, type_: str = None,
org_id: str = None) -> SupportedDevices:
"""
Read the List of Supported Devices
Gets the list of supported devices for an organization.
Retrieving this list requires a full or read-only administrator auth token with a scope of
`spark-admin:telephony_config_read`.
:param allow_configure_layout_enabled: List supported devices that allow the user to configure the layout.
:type allow_configure_layout_enabled: bool
:param type_: List supported devices of a specific type. To excluded device types from a request or query, add
`type=not:DEVICE_TYPE`. For example, `type=not:MPP`.
:type type_: str
:param org_id: List supported devices for an organization.
:type org_id: str
:rtype: SupportedDevices
"""
params = {}
if org_id is not None:
params['orgId'] = org_id
if allow_configure_layout_enabled is not None:
params['allowConfigureLayoutEnabled'] = str(allow_configure_layout_enabled).lower()
if type is not None:
params['type'] = type_
url = self.ep('supportedDevices')
data = self.get(url=url, params=params)
return SupportedDevices.model_validate(data)
[docs]
def device_settings(self, org_id: str = None) -> DeviceCustomization:
"""
Get device override settings for an organization.
Retrieving this list requires a full or read-only administrator auth token with a scope
of spark-admin:telephony_config_read.
:param org_id: List supported devices for an organization location.
:type org_id: str
:return: device customization response
:rtype: DeviceCustomization
"""
params = org_id and {'orgId': org_id} or None
url = self.ep('devices/settings')
data = self.get(url=url, params=params)
return DeviceCustomization.model_validate(data)
[docs]
def read_list_of_announcement_languages(self) -> list[AnnouncementLanguage]:
"""
Read the List of Announcement Languages
List all languages supported by Webex Calling for announcements and voice prompts.
Retrieving announcement languages requires a full or read-only administrator or location administrator auth
token with a scope of `spark-admin:telephony_config_read`.
"""
url = self.ep('announcementLanguages')
data = super().get(url=url)
return TypeAdapter(list[AnnouncementLanguage]).validate_python(data["languages"])
[docs]
def read_moh(self, org_id: str = None) -> MoHConfig:
"""
Get the organization Music on Hold configuration
Retrieve the organization's Music on Hold settings.
:param org_id: Retrieve Music on Hold settings for this organization.
:type org_id: str
:rtype: MoHConfig
"""
params = {}
if org_id is not None:
params['orgId'] = org_id
url = self.ep('moh/settings')
data = super().get(url, params=params)
return MoHConfig.model_validate(data)
[docs]
def update_moh(self, settings: MoHConfig, org_id: str = None):
"""
Update the organization Music on Hold configuration
Update the organization's Music on Hold settings.
:param settings: Music on Hold settings
:type settings: MoHConfig
:param org_id: Patch Music on Hold for this organization.
:type org_id: str
:rtype: None
"""
params = {}
if org_id is not None:
params['orgId'] = org_id
body = settings.model_dump(mode='json', by_alias=True, exclude_none=True)
url = self.ep('moh/settings')
super().put(url, params=params, json=body)
[docs]
def create_a_call_token(self, called_number: str, guest_name: str = None) -> str:
"""
Create a call token
Create a call token before initiating the click-to-call interaction.
The call token embeds information related to click-to-call interaction into an encrypted JWE token.
This token is used along with the `guest token
<https://developer.webex.com/docs/api/v1/guests-management/create-a-guest>`_ to initialise the web calling SDK
Creating a call token requires a service app access token with a scope of `spark:webrtc_calling`.
:param called_number: Number to be called. This number should be enabled as guest calling destination at
`Webex Control Hub
<https://admin.webex.com>`_ by the administrator.
:type called_number: str
:param guest_name: Name of the guest caller.
:type guest_name: str
:rtype: str
"""
body = dict()
body['calledNumber'] = called_number
if guest_name is not None:
body['guestName'] = guest_name
url = self.session.ep('telephony/click2call/callToken')
data = super().post(url, json=body)
r = data['callToken']
return r
[docs]
def get_call_captions_settings(self, org_id: str = None) -> OrgCallCaptions:
"""
Get the organization call captions settings
Retrieve the organization's call captions settings.
The call caption feature allows the customer to enable and manage closed captions and transcript functionality
(rolling caption panel) in Webex Calling, without requiring the user to escalate the call to a meeting.
This API requires a full or read-only administrator auth token with a scope of
`spark-admin:telephony_config_read`.
:param org_id: Unique identifier for the organization.
:type org_id: str
:rtype: :class:`OrgCallCaptions`
"""
params = {}
if org_id is not None:
params['orgId'] = org_id
url = self.ep('callCaptions')
data = super().get(url, params=params)
r = OrgCallCaptions.model_validate(data)
return r
[docs]
def update_call_captions_settings(self, settings: OrgCallCaptions, org_id: str = None):
"""
Update the organization call captions settings
Update the organization's call captions settings.
The call caption feature allows the customer to enable and manage closed captions and transcript functionality
(rolling caption panel) in Webex Calling, without requiring the user to escalate the call to a meeting.
This API requires a full administrator auth token with a scope of `spark-admin:telephony_config_write`.
:param settings: Organization call captions settings
:type settings: OrgCallCaptions
:param org_id: Unique identifier for the organization.
:type org_id: str
:rtype: None
"""
params = {}
if org_id is not None:
params['orgId'] = org_id
body = settings.update()
url = self.ep('callCaptions')
super().put(url, params=params, json=body)
[docs]
def get_large_organization_status(self, org_id: str = None) -> LargeOrgStatus:
"""
Get Large Organization Status
Get the large organization status for a customer.
Large organization status indicates whether an organization is categorized as a large organization based on the
threshold percentage.
This API requires a full or read-only administrator auth token with the `spark-admin:telephony_config_read`
scope.
:param org_id: Retrieves large organization status for this organization.
:type org_id: str
:rtype: :class:`LargeOrgStatus`
"""
params = {}
if org_id is not None:
params['orgId'] = org_id
url = self.ep('largeOrgStatus')
data = super().get(url, params=params)
r = LargeOrgStatus.model_validate(data)
return r