1. 程式人生 > 程式設計 >laravel框架使用阿里雲簡訊傳送訊息操作示例

laravel框架使用阿里雲簡訊傳送訊息操作示例

本文例項講述了laravel框架使用阿里雲簡訊傳送訊息操作。分享給大家供大家參考,具體如下:

最新需要用到傳送簡訊的功能,所以就在網上搜索一些寫好的擴充套件。

擴充套件地址:

https://github.com/MissMyCat/aliyun-sms

通過composer安裝:

composer require mrgoon/aliyun-sms dev-master

在 config/app.php 中 providers 加入:

Mrgoon\AliSms\ServiceProvider::class,

有需求的可以自行新增 aliases。

然後在控制檯執行 :

php artisan vendor:publish

預設會在 config 目錄下建立一個 aliyunsms.php 檔案:

<?php
return [
  'access_key' => env('ALIYUN_SMS_AK'),// accessKey
  'access_secret' => env('ALIYUN_SMS_AS'),// accessSecret
  'sign_name' => env('ALIYUN_SMS_SIGN_NAME'),// 簽名
];

然後在 .env 中配置相應引數:

ALIYUN_SMS_AK=
ALIYUN_SMS_AS=
ALIYUN_SMS_SIGN_NAME=

為了能夠方便的傳送簡訊,我們可以在 app 目錄下,建立一個Services目錄,並新增 AliyunSms.php 檔案。

<?php
namespace App\Services;
use Mrgoon\AliSms\AliSms;
/**
 * 阿里雲簡訊類
 */
class AliyunSms
{
  //驗證碼
  const VERIFICATION_CODE = 'verification_code';
  //模板CODE
  public static $templateCodes = [
    self::VERIFICATION_CODE => 'SMS_XXXXXXXXXX',];
  /**
   * 傳送簡訊
   */
  public static function sendSms($mobile,$scene,$params = [])
  {
    if (empty($mobile)) {
      throw new \Exception('手機號不能為空');
    }
    if (empty($scene)) {
      throw new \Exception('場景不能為空');
    }
    if (!isset(self::$templateCodes[$scene])) {
      throw new \Exception('請配置場景的模板CODE');
    }
    $template_code = self::$templateCodes[$scene];
    try {
      $ali_sms = new AliSms();
      $response = $ali_sms->sendSms($mobile,$template_code,$params);
      if ($response->Code == 'OK') {
        return true;
      }
      throw new \Exception($response->Message);
    } catch (\Throwable $e) {
      throw new \Exception($e->getMessage());
    }
  }
}

為了能夠記錄每次簡訊傳送的狀態,我們可以建立一個 sms_logs 表。

CREATE TABLE `sms_logs` (
 `id` int(11) unsigned NOT NULL AUTO_INCREMENT COMMENT 'ID',`type` tinyint(1) NOT NULL DEFAULT '0' COMMENT '型別(0:簡訊驗證碼,1:語音驗證碼,2:簡訊訊息通知)',`mobile` varchar(16) NOT NULL DEFAULT '' COMMENT '手機號',`code` varchar(12) NOT NULL DEFAULT '' COMMENT '驗證碼',`checked` tinyint(1) NOT NULL DEFAULT '0' COMMENT '是否驗證(0:未驗證,1:已驗證)',`status` tinyint(1) NOT NULL DEFAULT '0' COMMENT '狀態(0:未傳送,1:已傳送,2:傳送失敗)',`reason` varchar(255) NOT NULL DEFAULT '' COMMENT '失敗原因',`remark` varchar(255) NOT NULL DEFAULT '' COMMENT '備註',`operator_id` int(11) NOT NULL DEFAULT '0' COMMENT '操作人ID',`ip` varchar(16) NOT NULL DEFAULT '' COMMENT '操作IP',`created` int(11) NOT NULL DEFAULT '0' COMMENT '建立時間',`updated` int(11) NOT NULL DEFAULT '0' COMMENT '更新時間',PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='簡訊表';

然後針對該表,我們建立一個 SmsLog 模型來管理。

<?php
namespace App\Models;
use App\Services\AliyunSms;
class SmsLog extends Model
{
  protected $fillable = [
    'type','mobile','code','checked','status','reason','remark','operator_id','ip',];
  //型別(0:簡訊驗證碼,2:簡訊訊息通知)
  const TYPE_CODE = 0;
  const TYPE_VOICE = 1;
  const TYPE_MESSAGE = 2;
  //是否驗證(0:未驗證,1:已驗證)
  const CHECKED_UNVERIFIED = 0;
  const CHECKED_VERIFIED = 1;
  //狀態(0:未傳送,2:傳送失敗)
  const STATUS_NO_SEND = 0;
  const STATUS_SEND = 1;
  const STATUS_FAIL = 2;
  //簡訊傳送間隔時間,預設60秒
  const SEND_INTERVAL_TIME = 60;
  /**
   * 檢測簡訊驗證碼
   */
  protected function checkCode($mobile,$code)
  {
    if (!$mobile) {
      throw new \Exception('手機號不能為空');
    }
    if (!checkMobile($mobile)) {
      throw new \Exception('手機號不正確');
    }
    if (!$code) {
      throw new \Exception('驗證碼不能為空');
    }
    $sms_log = $this->where([
      ['type',self::TYPE_CODE],['mobile',$mobile],['status',self::STATUS_SEND],['checked',self::CHECKED_UNVERIFIED],])->orderBy('created','desc')->first();
    if (!$sms_log) {
      throw new \Exception('驗證碼不存在,請重新獲取');
    }
    if ($code != $sms_log->code) {
      throw new \Exception('驗證碼錯誤');
    }
    $sms_log->checked = self::CHECKED_VERIFIED;
    $sms_log->save();
    return true;
  }
  /**
   * 檢測簡訊頻率
   */
  protected function checkRate($mobile)
  {
    if (!$mobile) {
      throw new \Exception('手機號不能為空');
    }
    $sms_log = $this->where([
      ['mobile','desc')->first();
    $now = time();
    if ($sms_log) {
      if (($now - strtotime($sms_log->created)) < self::SEND_INTERVAL_TIME) {
        throw new \Exception('簡訊傳送太頻繁,請稍後再試');
      }
    }
    return true;
  }
  /**
   * 傳送簡訊驗證碼
   */
  protected function sendVerifyCode($mobile)
  {
    self::checkRate($mobile);
    $code = mt_rand(1000,9999);
    $sms_log = $this->create([
      'type' => self::TYPE_CODE,'mobile' => $mobile,'code' => $code,'checked' => self::CHECKED_UNVERIFIED,'status' => self::STATUS_NO_SEND,'ip' => getRealIp(),]);
    try {
      AliyunSms::sendSms($mobile,AliyunSms::VERIFICATION_CODE,['code' => $code]);
      $sms_log->status = self::STATUS_SEND;
      $sms_log->save();
      return true;
    } catch (\Exception $e) {
      $sms_log->status = self::STATUS_FAIL;
      $sms_log->reason = $e->getMessage();
      $sms_log->save();
      throw new \Exception($e->getMessage());
    }
  }
}

這樣,我們就可以在專案中通過 SmsLog::sendVerifyCode() 傳送簡訊了。

getRealIp() 和 checkMobile() 方法為公共方法,存放在 app/Helpers 的 functions.php 中。

/**
 * 獲取真實IP地址
 */
function getRealIp()
{
  $ip = false;
  if (getenv("HTTP_CLIENT_IP") && strcasecmp(getenv("HTTP_CLIENT_IP"),"unknown")) {
    $ip = getenv("HTTP_CLIENT_IP");
  } else if (getenv("HTTP_X_FORWARDED_FOR") && strcasecmp(getenv("HTTP_X_FORWARDED_FOR"),"unknown")) {
    $ips = explode(",",getenv("HTTP_X_FORWARDED_FOR"));
    if ($ip) {
      array_unshift($ips,$ip);
      $ip = false;
    }
    $ipscount = count($ips);
    for ($i = 0; $i < $ipscount; $i++) {
      if (!preg_match("/^(10|172\.16|192\.168)\./i",$ips[$i])) {
        $ip = $ips[$i];
        break;
      }
    }
  } else if (getenv("REMOTE_ADDR") && strcasecmp(getenv("REMOTE_ADDR"),"unknown")) {
    $ip = getenv("REMOTE_ADDR");
  } else if (isset($_SERVER['REMOTE_ADDR']) && $_SERVER['REMOTE_ADDR'] && strcasecmp($_SERVER['REMOTE_ADDR'],"unknown")) {
    $ip = $_SERVER['REMOTE_ADDR'];
  } else {
    $ip = "unknown";
  }
  return isIp($ip) ? $ip : "unknown";
}
/**
 * 檢查是否是合法的IP
 */
function isIp($ip)
{
  if (preg_match('/^((\d|[1-9]\d|2[0-4]\d|25[0-5]|1\d\d)(?:\.(\d|[1-9]\d|2[0-4]\d|25[0-5]|1\d\d)){3})$/',$ip)) {
    return true;
  } else {
    return false;
  }
}
/**
 * 驗證手機號
 */
function checkMobile($mobile)
{
  return preg_match('/^((13[0-9])|(14[5,7])|(15[0-3,5-9])|(17[0,3,5-8])|(18[0-9])|166|198|199|(147))\d{8}$/i',$mobile);
}

更多關於Laravel相關內容感興趣的讀者可檢視本站專題:《Laravel框架入門與進階教程》、《php優秀開發框架總結》、《php面向物件程式設計入門教程》、《php+mysql資料庫操作入門教程》及《php常見資料庫操作技巧彙總》

希望本文所述對大家基於Laravel框架的PHP程式設計有所幫助。