Skip to content

sinch/sinch-sdk-python

Repository files navigation

Sinch Python SDK

License

Python 3.9 Python 3.10 Python 3.11 Python 3.12 Python 3.13 Python 3.14

Here you'll find documentation related to the Sinch Python SDK, including how to install it, initialize it, and start developing Python code using Sinch services.

To use Sinch services, you'll need a Sinch account and access keys. You can sign up for an account and create access keys at dashboard.sinch.com.

For more information on the Sinch APIs on which this SDK is based, refer to the official developer documentation portal.

Prerequisites

  • Python in one of the supported versions - 3.9, 3.10, 3.11, 3.12, 3.13, 3.14
  • pip
  • Sinch account

Installation

You can install this package by typing: pip install sinch

Products

The Sinch client provides access to the following Sinch products:

  • Numbers API
  • SMS API
  • Conversation API (beta release)

Getting started

Client initialization

To establish a connection with the Sinch backend, you must provide credentials based on the API you intend to use. For security best practices, avoid hardcoding credentials — retrieve them from environment variables instead.

Note: sms_region and conversation_region no longer have defaults and must be set before calling those APIs—omitting them will cause a runtime error. See MIGRATION_GUIDE.md for details.

SMS API

The SMS API supports two authentication methods. sms_region is required for both and has no default.

Project auth (OAuth2)

The SDK automatically exchanges your key ID and key secret for a short-lived OAuth2 token and refreshes it automatically on expiry. Supported regions: us, eu, br.

In your Account dashboard, you will find your projectId and access keys composed of pairs of keyId / keySecret.

Note: the keySecret is visible only when you create the Access Key. Store it safely and create a new Access Key if you have lost it.

from sinch import SinchClient

sinch_client = SinchClient(
    project_id="project_id",
    key_id="key_id",
    key_secret="key_secret",
    sms_region="us"
)

Service Plan ID auth (legacy)

Uses a static bearer token that never expires. Support all regions: us, eu, br, ca, au.

In your Service APIs dashboard, you will find your servicePlanId and apiToken (bearer token).

from sinch import SinchClient

sinch_client = SinchClient(
    service_plan_id="service_plan_id",
    sms_api_token="api_token",
    sms_region="us"
)

Conversation API - Project auth (OAuth2)

conversation_region is required and has no default. Supported regions: us, eu, br.

Why region matters: The Conversation API stores and routes data within the selected region for regulatory compliance. Choose the region that matches your data residency requirements.

from sinch import SinchClient

sinch_client = SinchClient(
    project_id="project_id",
    key_id="key_id",
    key_secret="key_secret",
    conversation_region="eu"
)

SMS integration note: If you also use the SMS API, sms_region and conversation_region must match. Mismatched regions will cause delivery failures.

Other APIs - Project auth (OAuth2)

These APIs are not regionalized and use project-based auth.

from sinch import SinchClient

sinch_client = SinchClient(
    project_id="project_id",
    key_id="key_id",
    key_secret="key_secret",
)

Logging

Logging configuration for this SDK utilizes following hierarchy:

  1. If no configuration was provided via logger_name or logger configurable, SDK will inherit configuration from the root logger with the Sinch prefix.
  2. If logger_name configurable was provided, SDK will use logger related to that name. For example: myapp.sinch will inherit configuration from the myapp logger.
  3. If logger (logger instance) configurable was provided, SDK will use that particular logger for all its logging operations.

If all logging returned by this SDK needs to be disabled, usage of NullHandler provided by the standard logging module is advised.

Sample apps

Usage example of the Numbers API via VirtualNumbers on the client (sinch_client.numbers)—list() returns your project’s active virtual numbers:

paginator = sinch_client.numbers.list(
    region_code="US",
    number_type="LOCAL",
)
for active_number in paginator.iterator():
    print(active_number)

Returned values are Pydantic model instances (for example ActiveNumber), including fields such as phone_number, region_code, type, and capabilities.

More examples live under examples/snippets on the main branch.

Handling exceptions

Each API throws a custom, API related exception for an unsuccessful backed call.

Example for Numbers API:

from sinch.domains.numbers.api.v1.exceptions import NumbersException

try:
    paginator = sinch_client.numbers.list(
        region_code="US",
        number_type="LOCAL",
    )
except NumbersException as err:
    pass

For handling all possible exceptions thrown by this SDK use SinchException (superclass of all Sinch exceptions) from sinch.core.exceptions.

Custom HTTP client implementation

By default, the HTTP implementation uses the requests library.

To use a custom HTTP client, assign your transport to the client's configuration after initialization.

Custom transports must extend HTTPTransport and implement the send method. The base class provides prepare_request and authenticate helpers, and handles OAuth token refresh automatically.

The following example replaces the default requests backend with httpx and routes traffic through an authenticated proxy:

import httpx
from sinch import SinchClient
from sinch.core.ports.http_transport import HTTPTransport
from sinch.core.endpoint import HTTPEndpoint
from sinch.core.models.http_response import HTTPResponse


class MyHTTPImplementation(HTTPTransport):
    def __init__(self, sinch, proxy_url, proxy_user, proxy_password):
        super().__init__(sinch)
        self.http_client = httpx.Client(
            proxy=f"http://{proxy_user}:{proxy_password}@{proxy_url}"
        )

    def send(self, endpoint: HTTPEndpoint) -> HTTPResponse:
        request_data = self.prepare_request(endpoint)
        request_data = self.authenticate(endpoint, request_data)

        body = request_data.request_body
        response = self.http_client.request(
            method=request_data.http_method,
            url=request_data.url,
            json=body if isinstance(body, dict) else None,
            content=body if not isinstance(body, dict) else None,
            auth=request_data.auth,
            headers=request_data.headers,
            params=request_data.query_params,
            timeout=self.sinch.configuration.connection_timeout,
        )
        response_body = self.deserialize_json_response(response)

        return HTTPResponse(
            status_code=response.status_code,
            body=response_body,
            headers=dict(response.headers),
        )


sinch_client = SinchClient(
    key_id="key_id",
    key_secret="key_secret",
    project_id="some_project",
)
sinch_client.configuration.transport = MyHTTPImplementation(
    sinch_client,
    proxy_url="proxy.example.com:8080",
    proxy_user="proxy_user",
    proxy_password="proxy_password",
)

Note: Asynchronous HTTP clients are not supported. The transport must be a synchronous implementation.

License

This project is licensed under the Apache License. See the LICENSE file for the license text.

About

Sinch Python SDK for SMS, Numbers and ConversationAPI

Resources

License

Stars

Watchers

Forks

Packages

 
 
 

Contributors