Agent Skill · Sinch

sinch-elastic-sip-trunking

Provisions SIP trunks, endpoints, ACLs, credential lists, and phone numbers via the Sinch Elastic SIP Trunking REST API. Use when the user needs SIP connectivity, trunk provisioning, inbound/outbound PSTN voice routing, PBX integration, or SIP-to-PSTN bridging.

Provider: Sinch Path in repo: skills/sinch-elastic-sip-trunking/SKILL.md

Skill body

Sinch Elastic SIP Trunking API

Overview

The Sinch Elastic SIP Trunking (EST) API lets you programmatically provision SIP trunks and route voice traffic between customer infrastructure and the PSTN. The core workflow is: create a trunk, authorize it (ACL or credentials), attach endpoints, assign phone numbers.

Agent Instructions

Before generating code, gather from the user (skip any item already specified in the prompt or context):

  1. Direction — inbound (receive calls from PSTN), outbound (send calls to PSTN), or both?
  2. Auth method for the trunk (if outbound or both) — ACL-based (static IPs) or Credential-based (digest auth / dynamic IPs)?
  3. Endpoint type (if inbound or both) — Static endpoint (fixed IP/port) or Registered endpoint (SIP UA registers dynamically)?
  4. Approach — SDK or direct API calls (curl/fetch/requests)?
  5. Language — for SDK: Node.js. For direct API: any language, or curl. Python, Java, and .NET must use direct HTTP — only Node.js has SDK support.

When the user chooses SDK, refer to the sinch-sdks skill for installation and client initialization, then to the API references linked in References.

When the user chooses direct API calls, refer to the API references linked in References for request/response schemas.

Security: See the Security section below for url fetching policy and credential handling.

Decision Tree

User wants EST →
├─ Outbound only
│  ├─ Static IPs   → Workflow A (Trunk + ACL)
│  └─ Dynamic IPs  → Workflow E (Trunk + Credential List / Digest Auth)
├─ Inbound only
│  ├─ Static IP    → Workflow B (Trunk + Static Endpoint + Phone Number)
│  └─ Dynamic IP   → Workflow D (Trunk + Credential List + Registered Endpoint + Phone Number)
└─ Both           → Workflow C (Trunk + ACL/Creds + Endpoint + Phone Number)

Critical Rules

  1. Dependency order matters. Creating resources out of order causes failures. Create TrunkCreate ACL/CredentialsLink to TrunkAssign Phone NumbersCreate Endpoint
  2. The Domain Trap. Never send SIP INVITEs to trunk.pstn.sinch.com. ALWAYS use {your-hostname}.pstn.sinch.com.
  3. 60-second propagation. After linking ACLs or Credentials, wait 60 seconds before testing.
  4. Lower priority = higher preference. Endpoint priority: 1 is primary; priority: 100 is failover.
  5. PUT replaces the entire object. Omitted fields become null.

Getting Started

Agent Credentials handling

Store credentials in environment variables — never hardcode tokens or keys in commands or source code:

export SINCH_PROJECT_ID="your-project-id"
export SINCH_KEY_ID="your-key-id"
export SINCH_KEY_SECRET="your-key-secret"
export SINCH_ACCESS_TOKEN="your-oauth-token"

Authentication

Ensure that authentication headers are properly set when making API calls. The Elastic SIP Trunking API uses Bearer token authentication:

-H "Authorization: Bearer $SINCH_ACCESS_TOKEN"

See sinch-authentication for full setup, most importantly how to obtain {SINCH_ACCESS_TOKEN} (OAuth2 client-credentials — do not mint your own JWT).

SDK Installation

See sinch-sdks for installation and client initialization. Note: EST is only supported in the Node.js SDK — for Java, Python, and .NET, use direct HTTP calls.

First API Call — Create a Trunk

curl -X POST \
  "https://elastic-trunking.api.sinch.com/v1/projects/$SINCH_PROJECT_ID/trunks" \
  -H "Authorization: Bearer $SINCH_ACCESS_TOKEN" \
  -H "Content-Type: application/json" \
  -d '{"name": "my-trunk", "hostName": "my-trunk"}'

Response includes sipTrunkId and hostName — use {hostName}.pstn.sinch.com for all SIP routing.

For SDK examples, see the Getting Started Guide.

Key Concepts

Workflows

Workflow A: Outbound Only (ACL-based)

API docs: Create trunkCreate ACLLink ACL to trunkList ACLs for trunk

Workflow B: Inbound Only (Static Endpoint)

API docs: Create trunkCreate SIP endpointGet phone numbers

Workflow C: Bidirectional (Both Inbound + Outbound)

API docs: Create trunkCreate ACLLink ACL to trunkCreate SIP endpointGet phone numbers

Workflow D: Inbound with Registered Endpoint (Credential-based)

API docs: Create trunkCredential ListsCreate SIP endpointGet phone numbers

Workflow E: Outbound Only (Digest Auth / Credential-based)

API docs: Create trunkCredential ListsAdd credential list to trunkList credential lists for trunk

SIP Header Rules (Outbound)

Header Value Notes
From sip:+1E164@{your-hostname}.pstn.sinch.com Must be your trunk domain. Wrong domain → 403. Use E.164 format.
To sip:+1E164@{your-hostname}.pstn.sinch.com Destination in E.164 + your trunk domain. In most cases, same as Request-URI.
Request-URI sip:+1E164@{your-hostname}.pstn.sinch.com Destination in E.164 + your trunk domain. In most cases, same as To.

Gotchas and Best Practices

  1. CIDR notation — ACL entries require CIDR (/32 for single IP, /24 for range).
  2. Country permissions — US/Canada enabled by default. Other countries blocked; use updateCountryPermissions.
  3. Project ID ≠ App Key — EST uses projectId, not the Voice Application Key.
  4. Default CPS limit — 1 call per second. Exceeding it → 603. Contact Sinch to increase.
  5. Teardown order — Delete in reverse: unassign phone numbers → delete endpoints → unlink ACLs/credentials → delete trunk. Deleting out of order can orphan resources.

Troubleshooting

For SIP error codes and debugging runbooks, see references/diagnostics.md.

Quick reference:

References

Security

Skill frontmatter

metadata: {"author" => "Sinch", "version" => "1.0.4", "category" => "Voice", "tags" => "sip, trunking, est, pstn, voice, pbx, inbound, outbound", "uses" => ["sinch-authentication", "sinch-sdks"]}