Enhance documentation, implement Twilio OTP delivery, and update payment gateway methods. Updated AGENTS.md and README.md for clarity on ExecPlans and architecture. Added Twilio as a dependency and implemented capture/refund methods in MoyasarGateway. Improved frontend routing with react-router-dom and added authentication context. Updated styles and localization files for better user experience.
This commit is contained in:
@@ -50,6 +50,8 @@ class ConsoleOtpProvider(BaseOtpProvider):
|
||||
|
||||
|
||||
class TwilioOtpProvider(BaseOtpProvider):
|
||||
"""Twilio provider for SMS and WhatsApp OTP delivery. Requires TWILIO_* env vars."""
|
||||
|
||||
def __init__(self) -> None:
|
||||
self.account_sid = os.getenv("TWILIO_ACCOUNT_SID")
|
||||
self.auth_token = os.getenv("TWILIO_AUTH_TOKEN")
|
||||
@@ -60,15 +62,23 @@ class TwilioOtpProvider(BaseOtpProvider):
|
||||
if not self.account_sid or not self.auth_token or not self.from_number:
|
||||
raise ValueError(_("Twilio credentials are not configured"))
|
||||
|
||||
def send_sms(self, to_number: str, message: str) -> None:
|
||||
def _get_client(self):
|
||||
from twilio.rest import Client
|
||||
self._assert_config()
|
||||
raise NotImplementedError(_("Twilio SMS adapter not implemented yet"))
|
||||
return Client(self.account_sid, self.auth_token)
|
||||
|
||||
def send_sms(self, to_number: str, message: str) -> None:
|
||||
client = self._get_client()
|
||||
client.messages.create(body=message, from_=self.from_number, to=to_number)
|
||||
|
||||
def send_whatsapp(self, to_number: str, message: str) -> None:
|
||||
self._assert_config()
|
||||
if not self.whatsapp_from:
|
||||
raise ValueError(_("Twilio WhatsApp sender is not configured"))
|
||||
raise NotImplementedError(_("Twilio WhatsApp adapter not implemented yet"))
|
||||
client = self._get_client()
|
||||
from_ = f"whatsapp:{self.whatsapp_from}"
|
||||
to = f"whatsapp:{to_number}"
|
||||
client.messages.create(body=message, from_=from_, to=to)
|
||||
|
||||
|
||||
class UnifonicOtpProvider(BaseOtpProvider):
|
||||
|
||||
@@ -0,0 +1,51 @@
|
||||
"""Tests for Twilio OTP provider implementation."""
|
||||
|
||||
import pytest
|
||||
from unittest.mock import MagicMock, patch
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
@patch("apps.accounts.services.otp.TwilioOtpProvider._get_client")
|
||||
def test_twilio_send_sms_calls_client(mock_get_client):
|
||||
from apps.accounts.services.otp import TwilioOtpProvider
|
||||
|
||||
mock_client = MagicMock()
|
||||
mock_get_client.return_value = mock_client
|
||||
|
||||
with patch.dict("os.environ", {
|
||||
"TWILIO_ACCOUNT_SID": "AC123",
|
||||
"TWILIO_AUTH_TOKEN": "token",
|
||||
"TWILIO_FROM_NUMBER": "+966500000000",
|
||||
}):
|
||||
provider = TwilioOtpProvider()
|
||||
provider.send_sms("+966512345678", "Your code is 123456")
|
||||
|
||||
mock_client.messages.create.assert_called_once_with(
|
||||
body="Your code is 123456",
|
||||
from_="+966500000000",
|
||||
to="+966512345678",
|
||||
)
|
||||
|
||||
|
||||
@pytest.mark.django_db
|
||||
@patch("apps.accounts.services.otp.TwilioOtpProvider._get_client")
|
||||
def test_twilio_send_whatsapp_calls_client(mock_get_client):
|
||||
from apps.accounts.services.otp import TwilioOtpProvider
|
||||
|
||||
mock_client = MagicMock()
|
||||
mock_get_client.return_value = mock_client
|
||||
|
||||
with patch.dict("os.environ", {
|
||||
"TWILIO_ACCOUNT_SID": "AC123",
|
||||
"TWILIO_AUTH_TOKEN": "token",
|
||||
"TWILIO_FROM_NUMBER": "+966500000000",
|
||||
"TWILIO_WHATSAPP_FROM": "14155238886",
|
||||
}):
|
||||
provider = TwilioOtpProvider()
|
||||
provider.send_whatsapp("+966512345678", "Your code is 123456")
|
||||
|
||||
mock_client.messages.create.assert_called_once_with(
|
||||
body="Your code is 123456",
|
||||
from_="whatsapp:14155238886",
|
||||
to="whatsapp:+966512345678",
|
||||
)
|
||||
Reference in New Issue
Block a user