Laravel 集成微信用户登录和绑定的实现


Posted in PHP onDecember 27, 2019

最近主要在忙活微信与支付宝平台的对接与开发,本篇就基于后端层面来讲述一下微信的登录与绑定实现。

(一)申请微信开放平台

最首先的话就是需要去微信开发中心https://open.weixin.qq.com,创建一个账号,然后创建自己的移动或网站应用。

Laravel 集成微信用户登录和绑定的实现

创建完成后,就会需要腾讯的审核,整个过程在1-3天,基本上1天左右就能完成,审核通过如下图所示。

Laravel 集成微信用户登录和绑定的实现

(二) 、封装微信相关接口

微信移动应用开发文档:https://developers.weixin.qq.com/doc/oplatform/Mobile_App/WeChat_Login/Authorized_API_call_UnionID.html

审核通过后,就需要来封装微信授权、可信息获取的接口。

封装微信授权 && 用户信息获取

微信授权接口:https://api.weixin.qq.com/sns/oauth2

需要填写的参数如下:

参数 是否必须 说明
appid 应用唯一标识,在微信开放平台提交应用审核通过后获得
secret 应用密钥 AppSecret,在微信开放平台提交应用审核通过后获得
code 填写第一步获取的 code 参数
grant_type 填 authorization_code

下面通过我们的PHP代码实现:

<?php
namespace App\Helpers;

use GuzzleHttp\Client;
use Illuminate\Support\Arr;

class WechatAppUtils
{
  protected $client = null;

  protected $config = [];

  public function __construct()
  {
    $this->config = [
      'wechat_app' => [
        'appid' => env('WECHAT_APPID'),  //审核通过的APPID
        'secret' => env('WECHAT_SECRET'),  //应用APP SECRET 详情见上图
      ],
      'time_out'  => 5,
    ];
    $this->client = new Client([
      'time_out' => $this->config['time_out'],
    ]);
  }

  /**
   * 获取微信用户access_token
   *
   * @param [String] $code
   * @return Array
   */
  public function accessToken($code)
  {
    $accessTokenUrl = 'https://api.weixin.qq.com/sns/oauth2/access_token';

    $response = $this->client->request('GET', $accessTokenUrl, [
      'query' => [
        'grant_type' => 'authorization_code',
        'code'    => $code,
        'appid'   => Arr::get($this->config, 'wechat_app.appid'),
        'secret'   => Arr::get($this->config, 'wechat_app.secret'),
      ],
    ]);

    $result = $response->getbody()->getContents();

    return empty($result) ? null : json_decode($result, true);
  }

  /**
   * 微信用户信息
   *
   * @param [String] $accessToken
   * @param [String] $openId
   * @return Array
   */
  public function userInfo($accessToken, $openId)
  {
    $userInfoUrl = 'https://api.weixin.qq.com/sns/userinfo';

    $response = $this->client->request('GET', $userInfoUrl, [
      'query' => [
        'access_token' => $accessToken,
        'openid'    => $openId,
        'lang'     => 'zh_CN',
      ],
    ]);

    $result = $response->getbody()->getContents();

    return empty($result) ? null : json_decode($result, true);
  }
}

上面的accessToken方法主要是实现用户授权,效验的code参数是客户端传递过来的,当成功获取收钱用户的授权信息后,可以根据用户的OPENID来调用userInfo方法查询相关用户的信息,包含了用户的昵称、头像、性别等等。

具体客户端开发文档可以参考这篇:https://developers.weixin.qq.com/doc/oplatform/Mobile_App/WeChat_Login/Development_Guide.html。

上面的用到的Http Client是一个第三方拓展包,叫做GuzzleHttp,是一个PHP HTTP客户端,可以轻松发送HTTP请求,并且可以轻松集成Web服务。

我们可以通过composer一键安装:

composer require guzzlehttp/guzzle

(三)、完善用户微信授权登录

完成上述的封装操作后,我们便开始讲微信接入到我们自己的系统中与用户进行关联起来,下面是微信接入的一张时序图。

Laravel 集成微信用户登录和绑定的实现

如果用户想使用微信登录,首先会通过客户端唤起微信,请求登录第三方应用,然后微信会询问用户是否成功授权给XX应用,授权成功后,客户端会得到一个授权码:code,然后客户端携带code请求我们的客户端API,进行授权绑定,授权成功后,会得到授权用户OPENID(应用下的唯一标识),反之抛出异常信息提示用户。

建立OAuth表,用于储存用户的授权信息。

建立一张o_auths table 储存用户的授权信息,设计oauth_type字段使其成为一个多态模型,方便接入以后的微博、支付宝、QQ接入等等。

Schema::create('o_auths', function (Blueprint $table) {
  $table->increments('id');
  $table->unsignedInteger('user_id')->index()->comment('用户ID');
  $table->morphs('o_auth');
  $table->json('data')->nullable()->comment('授权信息');
  $table->timestamps();
});

完善用户授权绑定

建立好o_auths table,下面开始完善用户授权绑定的逻辑:

function wechat(User $user, $code)
{
  $utils = new WechatAppUtils;

  //获取微信token
  $accessTokens = $utils->accessToken($code);
  throw_if(!Arr::has($accessTokens, ['unionid', 'openid']), Exception::class, '授权失败,请稍后再试!');

  //建立oauth关联
  $oAuth = OAuth::firstOrNew(['oauth_type' => 'wechat', 'oauth_id' => $accessTokens['openid']]);
  throw_if(isset($oAuth->id),Exception::class,'该微信已绑定,请直接登录!');
  $oAuth->user_id = $user->id;
  $oAuth->data  = Arr::only($accessTokens, ['openid', 'refresh_token']);
  $oAuth->save();

  return $oAuth;
}

首先会通过客户端传递过来的Code获取当前用户授权,然后查询该用户是否已授权过,已授权过就提醒用户直接去登录,否则绑定授权信息,返回给客户端。

完善微信登录

完善好用户授权后,登录就显得非常容易了,只需要简单查询授权记录,存在则返回对应绑定的用户,否则抛出异常信息提示用户。

public function signIn($user, $code)
{
  $utils = new WechatAppUtils;
  //获取微信token
  $accessTokens = $utils->accessToken($code);
  throw_if(!Arr::has($accessTokens, ['unionid', 'openid']), Exception::class, '授权失败,请稍后再试!');
  $oauth = $this->getUserOauth($user, 'wechat');
  throw_if(is_null($oauth), UserException::class, '授权失败,该账户未绑定!');

  return $oauth;
}

public function getUserOauth(User $user, $oAuthType)
{
  return OAuth::where(['oauth_type' => $oAuthType, 'user_id' => $user->id])->first();
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

PHP 相关文章推荐
php循环输出数据库内容的代码
May 24 PHP
php array_map array_multisort 高效处理多维数组排序
Jun 11 PHP
PHP 时间转换Unix时间戳代码
Jan 22 PHP
第4章 数据处理-php正则表达式-郑阿奇(续)
Jul 04 PHP
Parse正式发布开源PHP SDK
Aug 11 PHP
[原创]ThinkPHP中SHOW_RUN_TIME不能正常显示运行时间的解决方法
Oct 10 PHP
PHP简单的MVC框架实现方法
Dec 01 PHP
Laravel使用memcached缓存对文章增删改查进行优化的方法
Oct 08 PHP
详解PHP数据压缩、加解密(pack, unpack)
Dec 17 PHP
thinkPHP中钩子的使用方法实例分析
Nov 16 PHP
PHP守护进程化在C和PHP环境下的实现
Nov 21 PHP
ThinkPHP5框架中使用JWT的方法示例
Jun 03 PHP
PHP单元测试配置与使用方法详解
Dec 27 #PHP
PHP全局使用Laravel辅助函数dd
Dec 26 #PHP
解决php用mysql方式连接数据库出现Deprecated报错问题
Dec 25 #PHP
PHP高并发和大流量解决方案整理
Dec 24 #PHP
thinkphp5实现微信扫码支付
Dec 23 #PHP
PHP实现笛卡尔积算法的实例讲解
Dec 22 #PHP
PHP架构及原理知识点详解
Dec 22 #PHP
You might like
php empty() 检查一个变量是否为空
2011/11/10 PHP
获取用户Ip地址通用方法与常见安全隐患(HTTP_X_FORWARDED_FOR)
2013/06/01 PHP
PHP substr()函数参数解释及用法讲解
2017/11/23 PHP
php命名空间设计思想、用法与缺点分析
2019/07/17 PHP
PHP时间类完整代码实例
2021/02/26 PHP
event.srcElement+表格应用
2006/08/29 Javascript
JavaScript 拾碎[三] 使用className属性
2010/10/16 Javascript
JavaScript传递变量: 值传递?引用传递?
2011/02/22 Javascript
javascript中动态加载js文件多种解决办法总结
2013/11/15 Javascript
Javascript设计模式之观察者模式的多个实现版本实例
2015/03/03 Javascript
js中创建对象的几种方式
2017/02/05 Javascript
前端主流框架vue学习笔记第一篇
2017/07/26 Javascript
微信小程序中换行空格(多个空格)写法详解
2018/07/10 Javascript
jQuery事件绑定和解绑、事件冒泡与阻止事件冒泡及弹出应用示例
2019/05/13 jQuery
原生js实现ajax请求和JSONP跨域请求操作示例
2020/03/14 Javascript
javascript 易错知识点实例小结
2020/04/25 Javascript
vue-router懒加载的3种方式汇总
2021/02/28 Vue.js
在Python程序中操作MySQL的基本方法
2015/07/29 Python
Python基于Socket实现的简单聊天程序示例
2017/08/05 Python
Python常见排序操作示例【字典、列表、指定元素等】
2018/08/15 Python
Django处理多用户类型的方法介绍
2019/05/18 Python
Python基础学习之函数方法实例详解
2019/06/18 Python
基于Python获取城市近7天天气预报
2019/11/26 Python
深入浅析python变量加逗号,的含义
2020/02/22 Python
英国知名衬衫品牌美国网站:Charles Tyrwhitt美国
2016/08/28 全球购物
澳大利亚婴儿礼品公司:The Baby Gift Company
2018/11/04 全球购物
演讲稿开场白
2014/01/13 职场文书
国庆节演讲稿
2014/05/27 职场文书
2014超市收银员工作总结
2014/11/13 职场文书
2015年植树节活动总结
2015/02/06 职场文书
小学生节水倡议书
2015/04/29 职场文书
幼儿园教师安全责任书
2015/05/08 职场文书
个人收入证明格式
2015/06/24 职场文书
2016年公司“3.12”植树节活动总结
2016/03/16 职场文书
简单介绍 http请求响应参数、无连接无状态、MIME、状态码、端口、telnet、curl
2021/03/31 HTML / CSS
django上传文件的三种方式
2021/04/29 Python