当前位置:Gxlcms > php框架 > 使用工厂模式实现Thinkphp6.0接入阿里云短信

使用工厂模式实现Thinkphp6.0接入阿里云短信

时间:2021-07-01 10:21:17 帮助过:285人阅读

如今,短信验证码已成为网站、APP的基础必备应用,应用场景十分丰富,随着移动互联网的发展会越来越多。作为一名码农,对第三方短信接口也是必须掌握的。本文Gxlcms将介绍使用工厂模式怎么实现ThinkPHP6.0接入阿里云短信。

一、环境要求

PHP版本 >= 7.1.0

开发环境必须安装有Composer

已开通阿里云短信服务,并且已获取AccessKey,创建模板和签名

最重要的,阿里云账户余额一定要有钱。

这里我就不演示开通短信服务和创建签名模板了,小伙伴们可以查看官方文档:https://help.aliyun.com/document_detail/108072.html?spm=a2c4g.11186623.6.565.1b4825903BoqGV

二、使用Composer安装Thinkphp6.0

如果您是第一次安装,请在命令行中切换到您的web目录执行下面的命令

  1. composer create-project topthink/think sms

本教程将安装在C盘www目录下 aztp60.png三、使用Composer安装 Alibaba Cloud SDK for PHP

进到刚刚创建的sms项目下执行下面的命令

  1. composer require alibabacloud/sdk

sdk.png四、使用编辑器打开项目 ,并在config文件夹下创建sms.php配置文件来管理阿里短信配置信息

  1. <?php
  2. return [
  3. //阿里云短信API接口地址
  4. 'host' => 'dysmsapi.aliyuncs.com',
  5. //AccessKey ID
  6. 'access_key_id' => '您的AccessKey ID',
  7. //Access Key Secret
  8. 'access_key_secret' => '您的Access Key Secret',
  9. //地区ID
  10. 'region_id' => 'cn-hangzhou',
  11. //模板CODE
  12. 'template_code' => '您的模板CODE',
  13. //签名名称
  14. 'sign_name' => '您的短信签名名称',
  15. ];

五、顺便在config文件夹下打开cache.php添加Redis缓存配置,后面发送短信验证码会用到

  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | 缓存设置
  4. // +----------------------------------------------------------------------
  5. return [
  6. // 默认缓存驱动
  7. 'default' => env('cache.driver', 'redis'),
  8. // 缓存连接方式配置
  9. 'stores' => [
  10. 'file' => [
  11. // 驱动方式
  12. 'type' => 'File',
  13. // 缓存保存目录
  14. 'path' => '',
  15. // 缓存前缀
  16. 'prefix' => '',
  17. // 缓存有效期 0表示永久缓存
  18. 'expire' => 0,
  19. // 缓存标签前缀
  20. 'tag_prefix' => 'tag:',
  21. // 序列化机制 例如 ['serialize', 'unserialize']
  22. 'serialize' => [],
  23. ],
  24. // Redis缓存
  25. 'redis' => [
  26. //服务器地址
  27. 'host' => '127.0.0.1',
  28. //redis端口
  29. 'port' => 6379,
  30. //驱动方式
  31. 'type' => 'redis',
  32. //缓存前缀
  33. 'prefix' => 'sms_code_',
  34. ]
  35. ],
  36. ];

六、在app目录下创建common/lib/sms/Sms.php接口类,用来约束发送短信验证码的方法

  1. <?php
  2. namespace app\common\lib\sms;
  3. //定义实现发送短信验证码的接口类,用来约束发送验证码的方法
  4. interface Sms
  5. {
  6. /**
  7. * @desc 发送短信验证码的方法
  8. * @param string $phone 手机号
  9. * @param int $code 验证码
  10. * @return mixed
  11. */
  12. public static function sendCode(string $phone, int $code);
  13. }

七、在common/lib/sms目录下创建AliSms类来实现Sms接口的smsSend()

  1. <?php
  2. namespace app\common\lib\sms;
  3. use AlibabaCloud\Client\AlibabaCloud;
  4. use AlibabaCloud\Client\Exception\ClientException;
  5. use AlibabaCloud\Client\Exception\ServerException;
  6. class AliSms implements Sms
  7. {
  8. /**
  9. * @desc 阿里云发送短信验证码
  10. * @param string $phone 手机号
  11. * @param int $code 验证码
  12. * @return mixed|void
  13. * @throws ClientException
  14. */
  15. public static function sendCode(string $phone, int $code)
  16. {
  17. //判断手机号和验证码是否为空
  18. if (empty($phone) || empty($code)){
  19. return false;
  20. }
  21. AlibabaCloud::accessKeyClient(config('sms.access_key_id'), config('sms.access_key_secret'))->regionId(config('sms.region_id'))->asDefaultClient();
  22. try {
  23. $result = AlibabaCloud::rpc()
  24. ->product('Dysmsapi')
  25. // ->scheme('https') // https | http
  26. ->version('2017-05-25')
  27. ->action('SendSms')
  28. ->method('POST')
  29. ->host(config('sms.host'))
  30. ->options([
  31. 'query' => [
  32. 'RegionId' =>config('sms.region_id'),
  33. 'SignName' => config('sms.sign_name'),
  34. 'PhoneNumbers' => $phone,
  35. 'TemplateCode' => config('sms.template_code'),
  36. 'TemplateParam' => json_encode(['code' => $code]),
  37. ],
  38. ])->request();
  39. } catch (ClientException $e) {
  40. return false;
  41. } catch (ServerException $e) {
  42. return false;
  43. }
  44. return true;
  45. }
  46. }

八、在common\lib目录下创建生成短信验证码的类 Code.php

  1. <?php
  2. namespace app\common\lib;
  3. class Code
  4. {
  5. /**
  6. * @desc 生成4位或6位短信验证码,默认为4位
  7. * @param int $length 验证码长度
  8. * @return int
  9. */
  10. public static function getCode(int $length = 4)
  11. {
  12. $code = rand(1000,9999);
  13. if ($length == 6){
  14. $code = rand(100000,999999);
  15. }
  16. return $code;
  17. }
  18. }

九、在common目录下创建service/Sms.php

  1. <?php
  2. namespace app\common\Service;
  3. use app\common\lib\Code;
  4. class Sms
  5. {
  6. /**
  7. * @param string $phone 手机号
  8. * @param int $lengthCode 验证码长度
  9. * @param string $type 短信厂家,默认选用AliSms
  10. * @return mixed
  11. */
  12. public static function sendCode(string $phone,int $lengthCode,string $type='AliSms')
  13. {
  14. //生成短信验证码
  15. $code = Code::getCode(4);
  16. //使用工厂模式 调用Lib层发送短信
  17. $class = "app\common\lib\sms\\".$type;
  18. $sms = $class::sendCode($phone,$code);
  19. if ($sms){
  20. //发送成功,把短信验证码存储Redis缓存中,并给失效时间
  21. cache($phone,$code,300);
  22. }
  23. return $sms;
  24. }
  25. }

十、在common目录下创建validate/SmsValidate验证器

  1. <?php
  2. namespace app\common\validate;
  3. use think\Validate;
  4. class SmsValidate extends Validate
  5. {
  6. //验证规则
  7. protected $rule = [
  8. 'phone' => 'require|mobile',
  9. 'code' => 'require|number'
  10. ];
  11. //错误信息
  12. protected $message = [
  13. 'phone.require' => '请输入手机号',
  14. 'phone.mobile' => '手机号格式错误',
  15. 'code.require' => '短信验证码不能为空',
  16. 'code.number' => '短信验证码必须为纯数字'
  17. ];
  18. //验证场景
  19. protected $scene = [
  20. 'sendCode' => ['phone']
  21. ];
  22. }

十一、在controller目录下创建Sms.php

  1. <?php
  2. namespace app\controller;
  3. use app\common\validate\SmsValidate;
  4. class Sms
  5. {
  6. /**
  7. * @desc 发送短信验证码
  8. * @return \think\response\Json
  9. */
  10. public function code()
  11. {
  12. if (request()->isPost()){
  13. //获取手机号
  14. $data = [
  15. 'phone' => request()->param('phone','','trim'),
  16. ];
  17. //参数校验
  18. $validate = new SmsValidate();
  19. if (!$validate->scene('sendCode')->check($data)){
  20. return json(['code'=>0,'msg'=>$validate->getError()]);
  21. }
  22. //发送短信验证码
  23. if (\app\common\Service\Sms::sendCode($data['phone'],6,'AliSms')){
  24. return json(['code'=>1,'msg'=>'发送成功,请注意查收。']);
  25. }else{
  26. return json(['code'=>0,'msg'=>'发送失败,请稍后重试!']);
  27. }
  28. }
  29. }
  30. }

十二、使用PostMan测试发送短信验证码

不输入手机号或输入错误手机号会给相应的提示。

sjh格式错误.png

kkkkk.png

输入正确的手机号,提示发送短信验证成功sjhyzm.png这时我们来看看收到的验证码和缓存中的验证码是否一致yzmyz.png到这里我们的发送验证码已经完成

十三、这时我们需要校验验证码是否正确,在app\controller目录下创建Login.php

  1. <?php
  2. namespace app\controller;
  3. use app\common\validate\SmsValidate;
  4. class Login
  5. {
  6. public function index()
  7. {
  8. //接收参数
  9. $data = [
  10. 'phone' => request()->param('phone','','trim'),
  11. 'code' => request()->param('code','','trim'),
  12. ];
  13. //参数校验
  14. $validate = new SmsValidate();
  15. if (!$validate->check($data)){
  16. return json(['code'=>0,'msg'=>$validate->getError()]);
  17. }
  18. //从Redis中获取验证码
  19. $redisCode = cache($data['phone']);
  20. //判断验证码是否正确
  21. if (empty($redisCode)){
  22. return json(['code'=>0,'msg'=>'验证码已过期,请重新发送!']);
  23. }
  24. if ($redisCode != $data['code']){
  25. return json(['code'=>0,'msg'=>'验证码输入错误,请重新输入!']);
  26. }
  27. return '验证成功';
  28. }
  29. }

十四、使用PostMan校验验证码是否正确

输入错误的验证码,会给出响应的提示

yzmcw.png

输入正确的验证码,提示验证成功yzmyzcg.png

看到这里我相信很多小伙伴的验证码都已发送成功了。

不知道小伙伴们有没有发现文中有两处参数校验的代码相识度很高,在后续的thinkphp技术文章中我会对这个问题进行优化,感兴趣的小伙伴请关注。

以上就是使用工厂模式实现Thinkphp6.0接入阿里云短信的详细内容,更多请关注Gxlcms其它相关文章!

人气教程排行