You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

183 lines
7.4 KiB

6 years ago
  1. <?php
  2. namespace App\Common\Wechat;
  3. /**
  4. * OAuth2.0微信授权登录实现/微信PC扫码授权登录
  5. * 微信/PC扫码登录,两种的方式是一样的,先跳转到微信网页获取code,通过code获取token,通过token获取用户信息
  6. */
  7. class WechatAuth
  8. {
  9. //高级功能->开发者模式->获取
  10. private $app_id;
  11. private $app_secret;
  12. public function __construct($app_id, $app_secret)
  13. {
  14. $this->app_id = $app_id;
  15. $this->app_secret = $app_secret;
  16. }
  17. /**
  18. * 获取微信授权链接
  19. *
  20. * @param string $redirect_uri 回调地址,授权后重定向的回调链接地址,请使用urlEncode对链接进行处理
  21. * @param mixed $state 可以为空,重定向后会带上state参数,开发者可以填写a-zA-Z0-9的参数值,最多128字节
  22. */
  23. public function get_authorize_url($redirect_uri = '', $state = '')
  24. {
  25. return "https://open.weixin.qq.com/connect/oauth2/authorize?appid=".$this->app_id."&redirect_uri=".urlencode($redirect_uri)."&response_type=code&scope=snsapi_userinfo&state=".$state."#wechat_redirect";
  26. }
  27. /**
  28. * 微信PC扫码授权登录链接
  29. *
  30. * @param string $redirect_uri 回调地址,授权后重定向的回调链接地址,请使用urlEncode对链接进行处理
  31. * @param mixed $state 可以为空,重定向后会带上state参数,开发者可以填写a-zA-Z0-9的参数值,最多128字节
  32. */
  33. public function get_qrconnect_url($redirect_uri = '', $state = '')
  34. {
  35. return "https://open.weixin.qq.com/connect/qrconnect?appid".$this->app_id."&redirect_uri=".urlencode($redirect_uri)."&response_type=code&scope=snsapi_login&state=".$state."#wechat_redirect";
  36. }
  37. /**
  38. * 获取授权token
  39. *
  40. * @param string $code 通过get_authorize_url获取到的code
  41. */
  42. public function get_access_token($code = '')
  43. {
  44. $token_url = "https://api.weixin.qq.com/sns/oauth2/access_token?appid={$this->app_id}&secret={$this->app_secret}&code={$code}&grant_type=authorization_code";
  45. $token_data = $this->http($token_url);
  46. return json_decode($token_data, true);
  47. }
  48. /**
  49. * 获取access_token,access_token是公众号的全局唯一接口调用凭据,公众号调用各接口时都需使用access_token。access_token的存储至少要保留512个字符空间。access_token的有效期目前为2个小时,需定时刷新,重复获取将导致上次获取的access_token失效。
  50. */
  51. public function get_token()
  52. {
  53. $token_url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid={$this->app_id}&secret={$this->app_secret}";
  54. $token_data = $this->http($token_url);
  55. return json_decode($token_data, true);
  56. }
  57. /**
  58. * 获取小程序码,适用于需要的码数量较少的业务场景,通过该接口生成的小程序码,永久有效,数量限制见文末说明,请谨慎使用。
  59. * @param string $path 不能为空,最大长度 128 字节
  60. * @param int $width 二维码的宽度,默认430
  61. */
  62. public function getwxacode($path, $width = 430)
  63. {
  64. $access_token = $this->get_token();
  65. $url = 'https://api.weixin.qq.com/wxa/getwxacode?access_token='.$access_token['access_token'];
  66. $path ="pages/mine/mine/mine?query=1";
  67. $data ='{"path":"'.$path.'","width":'.$width.'}';
  68. $res = $this->http($url, $data);
  69. return $res;
  70. //将生成的小程序码存入相应文件夹下
  71. //file_put_contents('./public/wxyido/img/'.time().'.jpg',$return);
  72. }
  73. /**
  74. * 获取小程序码,通过该接口生成的小程序码,永久有效,数量暂无限制。用户扫描该码进入小程序后,开发者需在对应页面获取的码中 scene 字段的值,再做处理逻辑。
  75. * @param string $data['scene'] 二维码场景值
  76. * @param string $data['page'] 必须是已经发布的小程序存在的页面(否则报错),例如 "pages/index/index" ,根路径前不要填加'/',不能携带参数(参数请放在scene字段里),如果不填写这个字段,默认跳主页面
  77. * @param int $data['width'] 二维码的宽度,默认430
  78. * @param int $data['type'] 0路径存储,1base64
  79. */
  80. public function getwxacodeunlimit($data)
  81. {
  82. $access_token = $this->get_token();
  83. $url = 'https://api.weixin.qq.com/wxa/getwxacodeunlimit?access_token='.$access_token['access_token'];
  84. $post_data = array();
  85. $post_data['scene'] = $data['scene'];
  86. $post_data['page'] = $data['page'];
  87. $post_data['width'] = $data['width'];
  88. $res = $this->http($url, json_encode($post_data));
  89. if($data['type']==0)
  90. {
  91. file_put_contents($data['image_path'], $res);
  92. }
  93. else
  94. {
  95. $res = $this->data_uri($res);
  96. }
  97. return $res;
  98. //将生成的小程序码存入相应文件夹下
  99. //file_put_contents('./public/wxyido/img/'.time().'.jpg',$res);
  100. }
  101. public function data_uri($contents, $mime = 'image/png')
  102. {
  103. $base64 = base64_encode($contents);
  104. return ('data:' . $mime . ';base64,' . $base64);
  105. }
  106. /**
  107. * 获取授权后的微信用户信息
  108. *
  109. * @param string $access_token
  110. * @param string $open_id
  111. */
  112. public function get_user_info($access_token = '', $open_id = '')
  113. {
  114. $info_url = "https://api.weixin.qq.com/sns/userinfo?access_token={$access_token}&openid={$open_id}&lang=zh_CN";
  115. $info_data = $this->http($info_url);
  116. return json_decode($info_data, true);
  117. }
  118. /**
  119. * 获取用户基本信息(包括UnionID机制)
  120. *
  121. * @param string $access_token
  122. * @param string $open_id
  123. */
  124. public function get_user_unionid($access_token = '', $open_id = '')
  125. {
  126. $info_url = "https://api.weixin.qq.com/cgi-bin/user/info?access_token={$access_token}&openid={$open_id}&lang=zh_CN";
  127. $info_data = $this->http($info_url);
  128. return json_decode($info_data, true);
  129. }
  130. /**
  131. * 小程序登录凭证校验
  132. * 小程序调用wx.login() 获取 临时登录凭证code ,并回传到开发者服务器。
  133. * 开发者服务器以code换取 用户唯一标识openid 会话密钥session_key。
  134. * 临时登录凭证校验接口是一个 HTTPS 接口,开发者服务器使用 临时登录凭证code 获取 session_key openid 等。
  135. * @param string $js_code 小程序登录时获取的code
  136. */
  137. public function miniprogram_wxlogin($js_code)
  138. {
  139. $url = "https://api.weixin.qq.com/sns/jscode2session?appid={$this->app_id}&secret={$this->app_secret}&js_code=$js_code&grant_type=authorization_code";
  140. $res = $this->http($url);
  141. return json_decode($res, true);
  142. }
  143. // cURL函数简单封装
  144. public function http($url, $data = null)
  145. {
  146. $curl = curl_init();
  147. curl_setopt($curl, CURLOPT_URL, $url);
  148. curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, FALSE);
  149. curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, FALSE);
  150. if (!empty($data))
  151. {
  152. curl_setopt($curl, CURLOPT_POST, 1);
  153. curl_setopt($curl, CURLOPT_POSTFIELDS, $data);
  154. }
  155. curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
  156. $output = curl_exec($curl);
  157. curl_close($curl);
  158. return $output;
  159. }
  160. }