Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added SmsGlobal messaging adapter #40

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions .github/workflows/tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,10 @@ jobs:
VONAGE_API_SECRET: ${{ secrets.VONAGE_API_SECRET }}
VONAGE_TO: ${{ secrets.VONAGE_TO }}
VONAGE_FROM: ${{ secrets.VONAGE_FROM }}
SMS_GLOBAL_API_KEY: ${{ secrets.SMS_GLOBAL_API_KEY }}
SMS_GLOBAL_API_SECRET: ${{ secrets.SMS_GLOBAL_API_SECRET }}
SMS_GLOBAL_TO: ${{ secrets.SMS_GLOBAL_TO }}
SMS_GLOBAL_FROM: ${{ secrets.SMS_GLOBAL_FROM }}
run: |
docker compose up -d --build
sleep 5
Expand Down
4 changes: 4 additions & 0 deletions docker-compose.yml
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,10 @@ services:
- VONAGE_API_SECRET
- VONAGE_TO
- VONAGE_FROM
- SMS_GLOBAL_API_KEY
- SMS_GLOBAL_API_SECRET
- SMS_GLOBAL_TO
- SMS_GLOBAL_FROM
build:
context: .
volumes:
Expand Down
150 changes: 150 additions & 0 deletions src/Utopia/Messaging/Adapters/SMS/SmsGlobal.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,150 @@
<?php

namespace Utopia\Messaging\Adapters\SMS;

use Utopia\Messaging\Adapters\SMS as SMSAdapter;
use Utopia\Messaging\Messages\SMS;

// Reference Material
// https://www.smsglobal.com/rest-api/
class SmsGlobal extends SMSAdapter
{
/**
* Hash Algorithm for API Authentication
*/
const HASH_ALGO = 'sha256';

/**
* @param string $apiKey REST API key from MXT https://mxt.smsglobal.com/integrations
* @param string $apiSecret REST API Secret from MXT https://mxt.smsglobal.com/integrations
*/
public function __construct(
private string $apiKey,
private string $apiSecret
) {
}

public function getName(): string
{
return 'SmsGlobal';
}

public function getMaxMessagesPerRequest(): int
{
//TODO:: Didn't find the limit for REST API in SmsGlobal documentation
return 1000;
}

/**
* {@inheritdoc}
*
* @throws \Exception
*/
protected function process(SMS $message): string
{
$authorizationHeader = $this->getAuthorizationHeader(
method: 'POST',
requestUri: '/v2/sms/',
host: 'api.smsglobal.com'
);

return $this->request(
method: 'POST',
url: 'https://api.smsglobal.com/v2/sms/',
headers: [
'Authorization: '.$authorizationHeader,
'Content-Type: application/json',
],
body: \json_encode(
$this->getRequestBody(
to: $message->getTo(),
text: $message->getContent(),
from: $message->getFrom()
)
),
);
}

/**
* Get the value to use for the Authorization header
*
* @param string $method HTTP method (e.g. POST)
* @param string $requestUri Request URI (e.g. /v2/sms/)
* @param string $host Hostname
*/
public function getAuthorizationHeader(
string $method,
string $requestUri,
string $host
): string {
// Server or computer time should match with the
//current Unix timestamp otherwise authentication will fail

$timestamp = time();
$nonce = md5(microtime().mt_rand());

$hash = $this->getRequestHash(
timestamp: $timestamp,
nonce: $nonce,
method: $method,
requestUri: $requestUri,
host: $host
);

$headerFormat = 'MAC id="%s", ts="%s", nonce="%s", mac="%s"';
$header = sprintf($headerFormat, $this->apiKey, $timestamp, $nonce, $hash);

return $header;
}

/**
* Hashes a request using the API secret, to use in the Authorization header
*
* @param int $timestamp Unix timestamp of request time
* @param string $nonce Random unique string
* @param string $method HTTP method (e.g. POST)
* @param string $requestUri Request URI (e.g. /v1/sms/)
* @param string $host Hostname
* @param int $port Port (e.g. 443)
*/
private function getRequestHash(
int $timestamp,
string $nonce,
string $method,
string $requestUri,
string $host,
int $port = 443
) {
$string = [$timestamp, $nonce, $method, $requestUri, $host, $port, ''];
$string = sprintf("%s\n", implode("\n", $string));
$hash = hash_hmac(self::HASH_ALGO, $string, $this->apiSecret, true);
$hash = base64_encode($hash);

return $hash;
}

/**
* Get the request body
*
* @param array $to Phone number
* @param string $text Message to send
* @param string|null $from Origin of the message
*/
private function getRequestBody(array $to, string $text, string $from = null): array
{
$origin = $from ?? '';
if (count($to) == 1) {
return [
'destination' => $to[0],
'message' => $text,
'origin' => $origin,
];
} else {
return [
'destinations' => $to,
'message' => $text,
'origin' => $origin,
];
}
}
}
34 changes: 34 additions & 0 deletions tests/e2e/SMS/SmsGlobalTest.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
<?php

namespace Tests\E2E;

use Utopia\Messaging\Adapters\SMS\SmsGlobal;
use Utopia\Messaging\Messages\SMS;

class SmsGlobalTest extends Base
{
/**
* @throws \Exception
*/
public function testSendSMS()
{
$apiKey = getenv('SMS_GLOBAL_API_KEY');
$apiSecret = getenv('SMS_GLOBAL_API_SECRET');

$to = [getenv('SMS_GLOBAL_TO')];
$from = getenv('SMS_GLOBAL_FROM');

$sender = new SmsGlobal($apiKey, $apiSecret);
$message = new SMS(
to: $to,
content: 'Test Content',
from: $from
);

$response = $sender->send($message);
$result = \json_decode($response, true);

$this->assertArrayHasKey('messages', $result);
$this->assertEquals(count($to), count($result['messages']));
}
}