当前位置:首页 > CMS教程 > 其它CMS > 列表

laravel框架使用阿里云短信发送消息操作示例

发布:smiling 来源: PHP粉丝网  添加日期:2022-02-12 10:43:04 浏览: 评论:0 

本文实例讲述了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 文件:

  1. <?php 
  2. return [ 
  3.   'access_key' => env('ALIYUN_SMS_AK'), // accessKey 
  4.   'access_secret' => env('ALIYUN_SMS_AS'), // accessSecret 
  5.   'sign_name' => env('ALIYUN_SMS_SIGN_NAME'), // 签名 
  6. ]; 

然后在 .env 中配置相应参数:

ALIYUN_SMS_AK=

ALIYUN_SMS_AS=

ALIYUN_SMS_SIGN_NAME=

为了能够方便的发送短信,我们可以在 app 目录下,创建一个Services目录,并添加 AliyunSms.php 文件。

  1. <?php 
  2. namespace App\Services; 
  3. use Mrgoon\AliSms\AliSms; 
  4. /** 
  5.  * 阿里云短信类 
  6.  */ 
  7. class AliyunSms 
  8.   //验证码 
  9.   const VERIFICATION_CODE = 'verification_code'
  10.   //模板CODE 
  11.   public static $templateCodes = [ 
  12.     self::VERIFICATION_CODE => 'SMS_XXXXXXXXXX'
  13.   ]; 
  14.   /** 
  15.    * 发送短信 
  16.    */ 
  17.   public static function sendSms($mobile$scene$params = []) 
  18.   { 
  19.     if (emptyempty($mobile)) { 
  20.       throw new \Exception('手机号不能为空'); 
  21.     } 
  22.     if (emptyempty($scene)) { 
  23.       throw new \Exception('场景不能为空'); 
  24.     } 
  25.     if (!isset(self::$templateCodes[$scene])) { 
  26.       throw new \Exception('请配置场景的模板CODE'); 
  27.     } 
  28.     $template_code = self::$templateCodes[$scene]; 
  29.     try { 
  30.       $ali_sms = new AliSms(); 
  31.       $response = $ali_sms->sendSms($mobile$template_code$params); 
  32.       if ($response->Code == 'OK') { 
  33.         return true; 
  34.       } 
  35.       throw new \Exception($response->Message); 
  36.     } catch (\Throwable $e) { 
  37.       throw new \Exception($e->getMessage()); 
  38.     } 
  39.   } 

为了能够记录每次短信发送的状态,我们可以创建一个 sms_logs 表。

  1. CREATE TABLE `sms_logs` ( 
  2.  `id` int(11) unsigned NOT NULL AUTO_INCREMENT COMMENT 'ID'
  3.  `type` tinyint(1) NOT NULL DEFAULT '0' COMMENT '类型(0:短信验证码,1:语音验证码,2:短信消息通知)'
  4.  `mobile` varchar(16) NOT NULL DEFAULT '' COMMENT '手机号'
  5.  `code` varchar(12) NOT NULL DEFAULT '' COMMENT '验证码'
  6.  `checked` tinyint(1) NOT NULL DEFAULT '0' COMMENT '是否验证(0:未验证,1:已验证)'
  7.  `status` tinyint(1) NOT NULL DEFAULT '0' COMMENT '状态(0:未发送,1:已发送,2:发送失败)'
  8.  `reason` varchar(255) NOT NULL DEFAULT '' COMMENT '失败原因'
  9.  `remark` varchar(255) NOT NULL DEFAULT '' COMMENT '备注'
  10.  `operator_id` int(11) NOT NULL DEFAULT '0' COMMENT '操作人ID'
  11.  `ip` varchar(16) NOT NULL DEFAULT '' COMMENT '操作IP'
  12.  `created` int(11) NOT NULL DEFAULT '0' COMMENT '创建时间'
  13.  `updated` int(11) NOT NULL DEFAULT '0' COMMENT '更新时间'
  14.  PRIMARY KEY (`id`) 
  15. ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='短信表'

然后针对该表,我们创建一个 SmsLog 模型来管理。

  1. <?php 
  2. namespace App\Models; 
  3. use App\Services\AliyunSms; 
  4. class SmsLog extends Model 
  5.   protected $fillable = [ 
  6.     'type'
  7.     'mobile'
  8.     'code'
  9.     'checked'
  10.     'status'
  11.     'reason'
  12.     'remark'
  13.     'operator_id'
  14.     'ip'
  15.   ]; 
  16.   //类型(0:短信验证码,1:语音验证码,2:短信消息通知) 
  17.   const TYPE_CODE = 0; 
  18.   const TYPE_VOICE = 1; 
  19.   const TYPE_MESSAGE = 2; 
  20.   //是否验证(0:未验证,1:已验证) 
  21.   const CHECKED_UNVERIFIED = 0; 
  22.   const CHECKED_VERIFIED = 1; 
  23.   //状态(0:未发送,1:已发送,2:发送失败) 
  24.   const STATUS_NO_SEND = 0; 
  25.   const STATUS_SEND = 1; 
  26.   const STATUS_FAIL = 2; 
  27.   //短信发送间隔时间,默认60秒 
  28.   const SEND_INTERVAL_TIME = 60; 
  29.   /** 
  30.    * 检测短信验证码 
  31.    */ 
  32.   protected function checkCode($mobile$code
  33.   { 
  34.     if (!$mobile) { 
  35.       throw new \Exception('手机号不能为空'); 
  36.     } 
  37.     if (!checkMobile($mobile)) { 
  38.       throw new \Exception('手机号不正确'); 
  39.     } 
  40.     if (!$code) { 
  41.       throw new \Exception('验证码不能为空'); 
  42.     } 
  43.     $sms_log = $this->where([ 
  44.       ['type', self::TYPE_CODE], 
  45.       ['mobile'$mobile], 
  46.       ['status', self::STATUS_SEND], 
  47.       ['checked', self::CHECKED_UNVERIFIED], 
  48.     ])->orderBy('created''desc')->first(); 
  49.     if (!$sms_log) { 
  50.       throw new \Exception('验证码不存在,请重新获取'); 
  51.     } 
  52.     if ($code != $sms_log->code) { 
  53.       throw new \Exception('验证码错误'); 
  54.     } 
  55.     $sms_log->checked = self::CHECKED_VERIFIED; 
  56.     $sms_log->save(); 
  57.     return true; 
  58.   } 
  59.   /** 
  60.    * 检测短信频率 
  61.    */ 
  62.   protected function checkRate($mobile
  63.   { 
  64.     if (!$mobile) { 
  65.       throw new \Exception('手机号不能为空'); 
  66.     } 
  67.     $sms_log = $this->where([ 
  68.       ['mobile'$mobile], 
  69.       ['status', self::STATUS_SEND], 
  70.     ])->orderBy('created''desc')->first(); 
  71.     $now = time(); 
  72.     if ($sms_log) { 
  73.       if (($now - strtotime($sms_log->created)) < self::SEND_INTERVAL_TIME) { 
  74.         throw new \Exception('短信发送太频繁,请稍后再试'); 
  75.       } 
  76.     } 
  77.     return true; 
  78.   } 
  79.   /** 
  80.    * 发送短信验证码 
  81.    */ 
  82.   protected function sendVerifyCode($mobile
  83.   { 
  84.     self::checkRate($mobile); 
  85.     $code = mt_rand(1000, 9999); 
  86.     $sms_log = $this->create([ 
  87.       'type' => self::TYPE_CODE, 
  88.       'mobile' => $mobile
  89.       'code' => $code
  90.       'checked' => self::CHECKED_UNVERIFIED, 
  91.       'status' => self::STATUS_NO_SEND, 
  92.       'ip' => getRealIp(), 
  93.     ]); 
  94.     try { 
  95.       AliyunSms::sendSms($mobile, AliyunSms::VERIFICATION_CODE, ['code' => $code]); 
  96.       $sms_log->status = self::STATUS_SEND; 
  97.       $sms_log->save(); 
  98.       return true; 
  99.     } catch (\Exception $e) { 
  100.       $sms_log->status = self::STATUS_FAIL; 
  101.       $sms_log->reason = $e->getMessage(); 
  102.       $sms_log->save(); 
  103.       throw new \Exception($e->getMessage()); 
  104.     } 
  105.   } 

这样,我们就可以在项目中通过 SmsLog::sendVerifyCode() 发送短信了。

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

  1. /** 
  2.  * 获取真实IP地址 
  3.  */ 
  4. function getRealIp() 
  5.   $ip = false; 
  6.   if (getenv("HTTP_CLIENT_IP") && strcasecmp(getenv("HTTP_CLIENT_IP"), "unknown")) { 
  7.     $ip = getenv("HTTP_CLIENT_IP"); 
  8.   } else if (getenv("HTTP_X_FORWARDED_FOR") && strcasecmp(getenv("HTTP_X_FORWARDED_FOR"), "unknown")) { 
  9.     $ips = explode(", "getenv("HTTP_X_FORWARDED_FOR")); 
  10.     if ($ip) { 
  11.       array_unshift($ips$ip); 
  12.       $ip = false; 
  13.     } 
  14.     $ipscount = count($ips); 
  15.     for ($i = 0; $i < $ipscount$i++) { 
  16.       if (!preg_match("/^(10|172\.16|192\.168)\./i"$ips[$i])) { 
  17.         $ip = $ips[$i]; 
  18.         break
  19.       } 
  20.     } 
  21.   } else if (getenv("REMOTE_ADDR") && strcasecmp(getenv("REMOTE_ADDR"), "unknown")) { 
  22.     $ip = getenv("REMOTE_ADDR"); 
  23.   } else if (isset($_SERVER['REMOTE_ADDR']) && $_SERVER['REMOTE_ADDR'] && strcasecmp($_SERVER['REMOTE_ADDR'], "unknown")) { 
  24.     $ip = $_SERVER['REMOTE_ADDR']; 
  25.   } else { 
  26.     $ip = "unknown"
  27.   } 
  28.   return isIp($ip) ? $ip : "unknown"
  29. /** 
  30.  * 检查是否是合法的IP 
  31.  */ 
  32. function isIp($ip
  33.   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)) { 
  34.     return true; 
  35.   } else { 
  36.     return false; 
  37.   } 
  38. /** 
  39.  * 验证手机号 
  40.  */ 
  41. function checkMobile($mobile
  42.   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); 
  43. }

Tags: laravel阿里云短信发送消息

分享到: