本文實(shí)例講述了laravel框架使用阿里云短信發(fā)送消息操作。分享給大家供大家參考,具體如下:
最新需要用到發(fā)送短信的功能,所以就在網(wǎng)上搜索一些寫(xiě)好的擴(kuò)展。
擴(kuò)展地址:
https://github.com/MissMyCat/aliyun-sms
通過(guò)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;
/**
* 阿里云短信類(lèi)
*/
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 '類(lèi)型(0:短信驗(yàn)證碼,1:語(yǔ)音驗(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 模型來(lái)管理。
?php
namespace App\Models;
use App\Services\AliyunSms;
class SmsLog extends Model
{
protected $fillable = [
'type',
'mobile',
'code',
'checked',
'status',
'reason',
'remark',
'operator_id',
'ip',
];
//類(lèi)型(0:短信驗(yàn)證碼,1:語(yǔ)音驗(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)目中通過(guò) 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)容感興趣的讀者可查看本站專(zhuān)題:《Laravel框架入門(mén)與進(jìn)階教程》、《php優(yōu)秀開(kāi)發(fā)框架總結(jié)》、《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門(mén)教程》、《php+mysql數(shù)據(jù)庫(kù)操作入門(mén)教程》及《php常見(jiàn)數(shù)據(jù)庫(kù)操作技巧匯總》
希望本文所述對(duì)大家基于Laravel框架的PHP程序設(shè)計(jì)有所幫助。
您可能感興趣的文章:- Laravel使用swoole實(shí)現(xiàn)websocket主動(dòng)消息推送的方法介紹
- Laravel admin實(shí)現(xiàn)消息提醒、播放音頻功能
- Laravel Validator自定義錯(cuò)誤返回提示消息并在前端展示
- Laravel使用消息隊(duì)列需要注意的一些問(wèn)題
- PHP的Laravel框架中使用消息隊(duì)列queue及異步隊(duì)列的方法
- 在Laravel框架里實(shí)現(xiàn)發(fā)送郵件實(shí)例(郵箱驗(yàn)證)
- Laravel框架實(shí)現(xiàn)發(fā)送短信驗(yàn)證功能代碼
- 使用 laravel sms 構(gòu)建短信驗(yàn)證碼發(fā)送校驗(yàn)功能
- laravel5.4利用163郵箱發(fā)送郵件的步驟詳解
- laravel中短信發(fā)送驗(yàn)證碼的實(shí)現(xiàn)方法
- 用Laravel Sms實(shí)現(xiàn)laravel短信驗(yàn)證碼的發(fā)送的實(shí)現(xiàn)
- laravel框架使用極光推送消息操作示例