laravel框架使用阿里云短信發(fā)送消息操作示例
本文實(shí)例講述了laravel框架使用阿里云短信發(fā)送消息操作。分享給大家供大家參考,具體如下:
最新需要用到發(fā)送短信的功能,所以就在網(wǎng)上搜索一些寫好的擴(kuò)展。
擴(kuò)展地址:
https://github.com/MissMyCat/aliyun-sms
通過composer安裝:
composer require mrgoon/aliyun-sms dev-master
在 config/app.php 中 providers 加入:
Mrgoon\AliSms\ServiceProvider::class,
有需求的可以自行添加 aliases。
然后在控制臺(tái)運(yùn)行 :
php artisan vendor:publish
默認(rèn)會(huì)在 config 目錄下創(chuàng)建一個(gè) 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 中配置相應(yīng)參數(shù):
ALIYUN_SMS_AK= ALIYUN_SMS_AS= ALIYUN_SMS_SIGN_NAME=
為了能夠方便的發(fā)送短信,我們可以在 app 目錄下,創(chuàng)建一個(gè)Services目錄,并添加 AliyunSms.php 文件。
<?php namespace App\Services; use Mrgoon\AliSms\AliSms; /** * 阿里云短信類 */ class AliyunSms { //驗(yàn)證碼 const VERIFICATION_CODE = 'verification_code'; //模板CODE public static $templateCodes = [ self::VERIFICATION_CODE => 'SMS_XXXXXXXXXX', ]; /** * 發(fā)送短信 */ public static function sendSms($mobile, $scene, $params = []) { if (empty($mobile)) { throw new \Exception('手機(jī)號(hào)不能為空'); } if (empty($scene)) { throw new \Exception('場(chǎng)景不能為空'); } if (!isset(self::$templateCodes[$scene])) { throw new \Exception('請(qǐng)配置場(chǎng)景的模板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()); } } }
為了能夠記錄每次短信發(fā)送的狀態(tài),我們可以創(chuàng)建一個(gè) 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:短信驗(yàn)證碼,1:語音驗(yàn)證碼,2:短信消息通知)', `mobile` varchar(16) NOT NULL DEFAULT '' COMMENT '手機(jī)號(hào)', `code` varchar(12) NOT NULL DEFAULT '' COMMENT '驗(yàn)證碼', `checked` tinyint(1) NOT NULL DEFAULT '0' COMMENT '是否驗(yàn)證(0:未驗(yàn)證,1:已驗(yàn)證)', `status` tinyint(1) NOT NULL DEFAULT '0' COMMENT '狀態(tài)(0:未發(fā)送,1:已發(fā)送,2:發(fā)送失敗)', `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 '創(chuàng)建時(shí)間', `updated` int(11) NOT NULL DEFAULT '0' COMMENT '更新時(shí)間', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='短信表';
然后針對(duì)該表,我們創(chuàng)建一個(gè) 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:短信驗(yàn)證碼,1:語音驗(yàn)證碼,2:短信消息通知) const TYPE_CODE = 0; const TYPE_VOICE = 1; const TYPE_MESSAGE = 2; //是否驗(yàn)證(0:未驗(yàn)證,1:已驗(yàn)證) const CHECKED_UNVERIFIED = 0; const CHECKED_VERIFIED = 1; //狀態(tài)(0:未發(fā)送,1:已發(fā)送,2:發(fā)送失敗) const STATUS_NO_SEND = 0; const STATUS_SEND = 1; const STATUS_FAIL = 2; //短信發(fā)送間隔時(shí)間,默認(rèn)60秒 const SEND_INTERVAL_TIME = 60; /** * 檢測(cè)短信驗(yàn)證碼 */ protected function checkCode($mobile, $code) { if (!$mobile) { throw new \Exception('手機(jī)號(hào)不能為空'); } if (!checkMobile($mobile)) { throw new \Exception('手機(jī)號(hào)不正確'); } if (!$code) { throw new \Exception('驗(yàn)證碼不能為空'); } $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('驗(yàn)證碼不存在,請(qǐng)重新獲取'); } if ($code != $sms_log->code) { throw new \Exception('驗(yàn)證碼錯(cuò)誤'); } $sms_log->checked = self::CHECKED_VERIFIED; $sms_log->save(); return true; } /** * 檢測(cè)短信頻率 */ protected function checkRate($mobile) { if (!$mobile) { throw new \Exception('手機(jī)號(hào)不能為空'); } $sms_log = $this->where([ ['mobile', $mobile], ['status', self::STATUS_SEND], ])->orderBy('created', 'desc')->first(); $now = time(); if ($sms_log) { if (($now - strtotime($sms_log->created)) < self::SEND_INTERVAL_TIME) { throw new \Exception('短信發(fā)送太頻繁,請(qǐng)稍后再試'); } } return true; } /** * 發(fā)送短信驗(yàn)證碼 */ 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()); } } }
這樣,我們就可以在項(xiàng)目中通過 SmsLog::sendVerifyCode() 發(fā)送短信了。
getRealIp() 和 checkMobile() 方法為公共方法,存放在 app/Helpers 的 functions.php 中。
/** * 獲取真實(shí)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; } } /** * 驗(yàn)證手機(jī)號(hào) */ 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); }
更多關(guān)于Laravel相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《Laravel框架入門與進(jìn)階教程》、《php優(yōu)秀開發(fā)框架總結(jié)》、《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門教程》、《php+mysql數(shù)據(jù)庫(kù)操作入門教程》及《php常見數(shù)據(jù)庫(kù)操作技巧匯總》
希望本文所述對(duì)大家基于Laravel框架的PHP程序設(shè)計(jì)有所幫助。
相關(guān)文章
windows的文件系統(tǒng)機(jī)制引發(fā)的PHP路徑爆破問題分析
這篇文章主要介紹了windows的文件系統(tǒng)機(jī)制引發(fā)的PHP路徑爆破問題分析,需要的朋友可以參考下2014-07-07laravel框架關(guān)于搜索功能的實(shí)現(xiàn)
本文是作者整理的關(guān)于laravel框架搜索功能的實(shí)現(xiàn)原理,并附上了詳細(xì)代碼,有需要的小伙伴請(qǐng)持續(xù)關(guān)注!2018-03-03從零開始學(xué)YII2框架(六)高級(jí)應(yīng)用程序模板
這篇文章主要介紹了YII2框架學(xué)習(xí)筆記之高級(jí)應(yīng)用程序模板,深入淺出從安裝,配置到使用方法都做了介紹,希望對(duì)大家有所幫助2014-08-08PHP safe_mode開啟對(duì)于PHP系統(tǒng)函數(shù)有什么影響
這篇文章主要介紹了PHP safe_mode開啟對(duì)于PHP系統(tǒng)函數(shù)有什么影響,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-11-11PHP設(shè)計(jì)模式之觀察者模式入門與應(yīng)用案例詳解
這篇文章主要介紹了PHP設(shè)計(jì)模式之觀察者模式入門與應(yīng)用,結(jié)合具體案例形式詳細(xì)分析了PHP觀察者模式的相關(guān)概念、原理、使用方法及操作注意事項(xiàng),需要的朋友可以參考下2019-12-12php使用正則驗(yàn)證密碼字段的復(fù)雜強(qiáng)度原理詳細(xì)講解
密碼涉及到安全問題,所以密碼設(shè)計(jì)的復(fù)雜程序尤為重要。通常密碼由大小寫字母、數(shù)字、特殊符號(hào)組成。一般情況下前端可不做js判斷,使用ajax提交php后端程序接口,由后端判斷后返回json數(shù)據(jù),進(jìn)行用戶密碼復(fù)雜程序提示。2022-11-11PHP實(shí)現(xiàn)簡(jiǎn)單的計(jì)算器
這篇文章主要為大家詳細(xì)介紹了PHP實(shí)現(xiàn)簡(jiǎn)單的計(jì)算器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-08-08