with one click
telnyx-whatsapp-ruby
// Send WhatsApp messages, manage templates, WABAs, and phone numbers via the Telnyx WhatsApp Business API.
// Send WhatsApp messages, manage templates, WABAs, and phone numbers via the Telnyx WhatsApp Business API.
Migrate from Twilio to Telnyx. Orchestrates a complete 6-phase migration: discovery, planning, setup, code migration, validation, and cleanup. Covers voice (TwiML to TeXML, Call Control API), messaging, WebRTC, SIP trunking, verify, fax, video, IoT, number lookup, and porting. Includes automated scanners, validation scripts, and integration tests.
Send WhatsApp messages, manage templates, WABAs, and phone numbers via the Telnyx WhatsApp Business API.
Send WhatsApp messages, manage templates, WABAs, and phone numbers via the Telnyx WhatsApp Business API.
Send WhatsApp messages, manage templates, WABAs, and phone numbers via the Telnyx WhatsApp Business API.
Send WhatsApp messages, manage templates, WABAs, and phone numbers via the Telnyx WhatsApp Business API.
Send WhatsApp messages, manage templates, WABAs, and phone numbers via the Telnyx WhatsApp Business API.
| name | telnyx-whatsapp-ruby |
| description | Send WhatsApp messages, manage templates, WABAs, and phone numbers via the Telnyx WhatsApp Business API. |
| metadata | {"author":"telnyx","product":"whatsapp","language":"ruby"} |
gem install telnyx
require "telnyx"
client = Telnyx::Client.new(api_key: ENV["TELNYX_API_KEY"])
All examples below assume client is already initialized as shown above.
All API calls can fail with network errors, rate limits (429), validation errors (422), or authentication errors (401). Always handle errors in production code:
begin
response = client.messages.send_whatsapp(
from: "+19452940762",
to: "+18005551234",
type: "WHATSAPP",
whatsapp_message: {
type: "text",
text: { body: "Hello from Telnyx!" }
}
)
rescue Telnyx::APIError => e
puts "API error: #{e.http_status} - #{e.message}"
rescue Telnyx::AuthenticationError
puts "Invalid API key"
rescue Telnyx::RateLimitError
puts "Rate limited - retry with backoff"
end
Common error codes: 401 invalid API key, 403 insufficient permissions,
404 resource not found, 422 validation error (check field formats),
429 rate limited (retry with exponential backoff).
+13125550001). Include the + prefix and country code.template_id) instead of name + language. When template_id is provided, name and language are resolved automatically.response.auto_paging_each { |item| }.APPROVED status before they can be used for sending.order_confirmation). No spaces, hyphens, or uppercase.AUTHENTICATION templates get special pricing but must contain an OTP. UTILITY is for transactional messages. MARKETING for promotional content.Do not invent Telnyx parameters, enums, response fields, or webhook fields.
## Additional Operations, read the optional-parameters section and the response-schemas section.Send a pre-approved template message. Templates can be sent anytime — no 24-hour window restriction.
client.messages.send_whatsapp() — POST /messages/whatsapp
| Parameter | Type | Required | Description |
|---|---|---|---|
from | string (E.164) | Yes | WhatsApp-enabled phone number in +E.164 format |
to | string (E.164) | Yes | Recipient phone number in +E.164 format |
type | string | No | Must be WHATSAPP |
whatsapp_message | object | Yes | WhatsApp message object |
messaging_profile_id | string (UUID) | No | Messaging profile to use |
webhook_url | string (URL) | No | Callback URL for delivery status updates |
# Send by template name + language
response = client.messages.send_whatsapp(
from: "+19452940762",
to: "+18005551234",
type: "WHATSAPP",
whatsapp_message: {
type: "template",
template: {
name: "order_confirmation",
language: { code: "en_US" },
components: [
{
type: "body",
parameters: [
{ type: "text", text: "ORD-12345" },
{ type: "text", text: "March 15, 2026" }
]
}
]
}
}
)
puts response.data.id
# Send by Telnyx template_id (no name/language needed)
response = client.messages.send_whatsapp(
from: "+19452940762",
to: "+18005551234",
type: "WHATSAPP",
whatsapp_message: {
type: "template",
template: {
template_id: "019cd44b-3a1c-781b-956e-bd33e9fd2ac6",
components: [
{
type: "body",
parameters: [
{ type: "text", text: "483291" }
]
}
]
}
}
)
Primary response fields:
response.data.id — Message UUIDresponse.data.to[0].status — queued, sent, delivered, failedresponse.data.from.phone_numberresponse.data.type — WHATSAPPSend a text message within the 24-hour customer service window.
client.messages.send_whatsapp() — POST /messages/whatsapp
response = client.messages.send_whatsapp(
from: "+19452940762",
to: "+18005551234",
type: "WHATSAPP",
whatsapp_message: {
type: "text",
text: { body: "Your order has shipped!" }
}
)
client.whatsapp.business_accounts.list() — GET /v2/whatsapp/business_accounts
response = client.whatsapp.business_accounts.list
response.data.each do |waba|
puts "#{waba.id}: #{waba.name} (#{waba.status})"
end
Primary response fields:
waba.id — Telnyx WABA UUIDwaba.waba_id — Meta WABA IDwaba.name — Business namewaba.status — Account statuswaba.country — WABA countryclient.whatsapp.templates.list() — GET /v2/whatsapp/message_templates
| Parameter | Type | Required | Description |
|---|---|---|---|
waba_id | string (UUID) | Yes | Telnyx WABA UUID |
category | string | No | Filter: AUTHENTICATION, MARKETING, UTILITY |
status | string | No | Filter: APPROVED, PENDING, REJECTED, DISABLED |
response = client.whatsapp.templates.list(
waba_id: "019c1ff0-5c30-7f36-8436-730b1d0b0e56",
status: "APPROVED"
)
response.data.each do |tmpl|
puts "#{tmpl.id}: #{tmpl.name} (#{tmpl.category}) - #{tmpl.status}"
end
Primary response fields:
tmpl.id — Telnyx template UUID (use as template_id when sending)tmpl.name — Template nametmpl.category — AUTHENTICATION, MARKETING, or UTILITYtmpl.language — Language codetmpl.status — APPROVED, PENDING, REJECTED, DISABLEDtmpl.components — Template componentsclient.whatsapp.templates.create() — POST /v2/whatsapp/message_templates
response = client.whatsapp.templates.create(
waba_id: "019c1ff0-5c30-7f36-8436-730b1d0b0e56",
name: "order_shipped",
category: "UTILITY",
language: "en_US",
components: [
{
type: "BODY",
text: "Your order {{1}} has been shipped and will arrive by {{2}}.",
example: {
body_text: [["ORD-12345", "March 20, 2026"]]
}
}
]
)
puts "Template created: #{response.data.id} (status: #{response.data.status})"
client.whatsapp.phone_numbers.list() — GET /v2/whatsapp/phone_numbers
response = client.whatsapp.phone_numbers.list(
waba_id: "019c1ff0-5c30-7f36-8436-730b1d0b0e56"
)
response.data.each do |pn|
puts "#{pn.phone_number} - quality: #{pn.quality_rating}"
end
Telnyx signs webhooks with Ed25519. Always verify signatures in production:
event = Telnyx::Webhook.construct_event(
payload: request.body.read,
sig_header: request.env["HTTP_TELNYX_SIGNATURE_ED25519"],
timestamp: request.env["HTTP_TELNYX_TIMESTAMP"],
public_key: TELNYX_PUBLIC_KEY
)
These webhook payload fields are inline because they are part of the primary integration path.
| Field | Type | Description |
|---|---|---|
data.event_type | enum: message.sent, message.finalized | Delivery status event |
data.payload.id | uuid | Message ID |
data.payload.to[0].status | string | queued, sent, delivered, read, failed |
data.payload.template_id | string | Telnyx template UUID (if template message) |
data.payload.template_name | string | Template name (if template message) |
| Field | Type | Description |
|---|---|---|
event_type | string | whatsapp.template.approved, whatsapp.template.rejected, whatsapp.template.disabled |
payload.template_id | string | Telnyx template UUID |
payload.template_name | string | Template name |
payload.status | string | New template status |
payload.reason | string | Rejection/disable reason |
appointment_reminder, not msg1).example field — Meta reviewers check these.AUTHENTICATION — OTP/verification codes only. Gets special pricing.UTILITY — Transactional (order updates, shipping, account alerts).MARKETING — Promotional content, offers, newsletters.{{1}}, {{2}}, etc. for variable content. Always provide the correct number of parameters when sending.| Operation | SDK Method | Use Case |
|---|---|---|
| Get template details | client.whatsapp_message_templates.retrieve() | Check template status |
| Get business profile | client.whatsapp.phone_numbers.profile.retrieve() | View business profile |
| Configure webhooks | client.whatsapp.business_accounts.settings.update() | Subscribe to events |
| Operation | SDK Method | Endpoint | Required Params |
|---|---|---|---|
| Send WhatsApp message | client.messages.send_whatsapp() | POST /messages/whatsapp | from, to, whatsapp_message |
| List WABAs | client.whatsapp.business_accounts.list() | GET /v2/whatsapp/business_accounts | — |
| Get WABA | client.whatsapp.business_accounts.retrieve() | GET /v2/whatsapp/business_accounts/:id | waba_id |
| List templates | client.whatsapp.templates.list() | GET /v2/whatsapp/message_templates | waba_id |
| Get template | client.whatsapp_message_templates.retrieve() | GET /v2/whatsapp_message_templates/:id | template_id |
| Create template | client.whatsapp.templates.create() | POST /v2/whatsapp/message_templates | waba_id, name, category, language, components |
| List phone numbers | client.whatsapp.phone_numbers.list() | GET /v2/whatsapp/phone_numbers | waba_id |
| Configure webhooks | client.whatsapp.business_accounts.settings.update() | PATCH /v2/whatsapp/business_accounts/:id/settings | waba_id |