关键词搜索

源码搜索 ×
×

微信小程序获取用户手机号,后端php实现

发布2020-05-25浏览2583次

详情内容

如图:

小程序代码:

第一步,登录,获取用户的 session_key;

第二步,点击按钮调用 bindgetphonenumber 事件,通过该事件得到 encryptedData 和 iv 

第三步,把session_key,encryptedData 和 iv 传递给后端解密得到用户的手机号信息

文章结尾有我的小程序util.js文件。

登录代码:(我是在app.js 里面统一封装的登录模块,如果有自己的登录模块,可以无视这段代码,把session_key放到第二段代码就可以)
 

  1. //app.js
  2. const util = require('./utils/util.js');
  3. App({
  4. onLaunch: function() {
  5. var that = this;
  6. // wx.login
  7. that.login = (resolve) => {
  8. wx.login({
  9. success: (res) => {
  10. if (res.code) {
  11. that.loginRequest(res.code, resolve);
  12. }
  13. },
  14. })
  15. }
  16. // session定时器
  17. that.sessionIsExpire = (resolve) => {
  18. const session = wx.getStorageSync('session_key');
  19. const deadLine = wx.getStorageSync('deadline');
  20. const nowDate = new Date().getTime();
  21. if (session && deadLine) {
  22. if (nowDate - deadLine >= 1000 * 60 * 60 * 24) {
  23. that.login(resolve)
  24. } else {
  25. wx.checkSession({
  26. success() {
  27. // session_key 未过期,并且在本生命周期一直有效
  28. resolve(session)
  29. },
  30. fail() {
  31. // session_key 已经失效,需要重新执行登录流程
  32. that.login(resolve)
  33. }
  34. })
  35. }
  36. } else {
  37. that.login(resolve)
  38. }
  39. }
  40. // 获取session
  41. that.pro_getSession = new Promise((resolve) => {
  42. that.sessionIsExpire(resolve)
  43. })
  44. // 请求login.do接口得到session
  45. that.loginRequest = (code, resolve) => {
  46. var url = '/login.php';
  47. var data = {
  48. code
  49. }
  50. util.request(url, 'post', data, '', (res) => {
  51. console.log('登录了')
  52. if (res.data.success == true) {
  53. const session = res.session_key;
  54. resolve(session)
  55. wx.setStorageSync('openid', res.openid);
  56. wx.setStorageSync('session_key', session);
  57. wx.setStorageSync('deadline', new Date().getTime())
  58. } else {
  59. resolve(1)
  60. }
  61. }, (fail) => {
  62. resolve(1)
  63. })
  64. }
  65. },
  66. globalData: {
  67. userInfo: null
  68. }
  69. })

小程序代码片段2: 获取手机号页面的代码 

 

  1. <button class="motto-font" open-type="getPhoneNumber" bindgetphonenumber="getPhoneNumber">
  2. 获取手机号
  3. </button>
  1. // 获取手机号码
  2. getPhoneNumber: function (e) {
  3. console.log(e)
  4. if (e.detail.errMsg == 'getPhoneNumber:fail') {
  5. console.log(ErrMsg);
  6. that.showToast_fail('未获取到手机号码');
  7. return false;
  8. } else if (e.detail.iv == undefined || !e.detail.iv) {
  9. that.showToast_fail('授权失败');
  10. return false;
  11. } else {
  12. // 解密手机号接口
  13. var url = "/getTelNumber.php";
  14. var params = {
  15. session_key: wx.getStorageSync('session_key'),
  16. encryptedData: e.detail.encryptedData,
  17. iv: e.detail.iv
  18. };
  19. util.request(url, 'post', params, '', (res) => {
  20. that.setData({
  21. phone: res.data.phoneNumber,
  22. })
  23. wx.reLaunch({
  24. url: '../index/index',
  25. })
  26. }, function () {
  27. that.showToast_fail('获取手机号失败');
  28. })
  29. }
  30. },

 

后端实现代码:

登录接口如图

login.php代码 (这个接口返回的是一个JSON对象)

  1. <?php
  2. header("Content-Type:text/html;charset=utf8");
  3. header("Access-Control-Allow-Origin: *"); //解决跨域
  4. header('Access-Control-Allow-Methods:POST');// 响应类型
  5. header('Access-Control-Allow-Headers:*'); // 响应头设置
  6. error_reporting(E_ALL);
  7. $code = $_POST['code'];
  8. $url='https://api.weixin.qq.com/sns/jscode2session?appid=wx4d7fcc145ca8013b&secret=c58ac54c1bf4c2de7c07221e547a4218&js_code='.$code.'&grant_type=authorization_code';
  9. $html = file_get_contents($url);
  10. print($html);
  11. ?>

 getTelNumber.php代码如图

代码:

  1. <?php
  2. header("Content-Type:text/html;charset=utf8");
  3. header("Access-Control-Allow-Origin: *"); //解决跨域
  4. header('Access-Control-Allow-Methods:POST');// 响应类型
  5. header('Access-Control-Allow-Headers:*'); // 响应头设置
  6. $appid = 'wx4d7fcc145ca8013b';
  7. $sessionKey = $_POST['session_key'];
  8. $encryptedData = $_POST['encryptedData'];
  9. $iv = $_POST['iv'];
  10. include_once "wxBizDataCrypt.php";
  11. $data = '';
  12. $pc = new WXBizDataCrypt($appid, $sessionKey);
  13. $errCode = $pc->decryptData($encryptedData, $iv, $data );
  14. if ($errCode == 0) {
  15. print($data . "\n");
  16. } else {
  17. print($errCode . "\n");
  18. }
  19. ?>

这里有引用到两个文件,放到相同的目录下就行

wxBizDataCrypt.php

  1. <?php
  2. /**
  3. * 对微信小程序用户加密数据的解密示例代码.
  4. *
  5. * @copyright Copyright (c) 1998-2014 Tencent Inc.
  6. */
  7. include_once "errorCode.php";
  8. class WXBizDataCrypt
  9. {
  10. private $appid;
  11. private $sessionKey;
  12. /**
  13. * 构造函数
  14. * @param $sessionKey string 用户在小程序登录后获取的会话密钥
  15. * @param $appid string 小程序的appid
  16. */
  17. public function __construct( $appid, $sessionKey)
  18. {
  19. $this->sessionKey = $sessionKey;
  20. $this->appid = $appid;
  21. }
  22. /**
  23. * 检验数据的真实性,并且获取解密后的明文.
  24. * @param $encryptedData string 加密的用户数据
  25. * @param $iv string 与用户数据一同返回的初始向量
  26. * @param $data string 解密后的原文
  27. *
  28. * @return int 成功0,失败返回对应的错误码
  29. */
  30. public function decryptData( $encryptedData, $iv, &$data )
  31. {
  32. if (strlen($this->sessionKey) != 24) {
  33. return ErrorCode::$IllegalAesKey;
  34. }
  35. $aesKey=base64_decode($this->sessionKey);
  36. if (strlen($iv) != 24) {
  37. return ErrorCode::$IllegalIv;
  38. }
  39. $aesIV=base64_decode($iv);
  40. $aesCipher=base64_decode($encryptedData);
  41. $result=openssl_decrypt( $aesCipher, "AES-128-CBC", $aesKey, 1, $aesIV);
  42. $dataObj=json_decode( $result );
  43. if( $dataObj == NULL )
  44. {
  45. return ErrorCode::$IllegalBuffer;
  46. }
  47. if( $dataObj->watermark->appid != $this->appid )
  48. {
  49. return ErrorCode::$IllegalBuffer;
  50. }
  51. $data = $result;
  52. return ErrorCode::$OK;
  53. }
  54. }

errorCode.php

  1. <?php
  2. /**
  3. * error code 说明.
  4. * <ul>
  5. * <li>-41001: encodingAesKey 非法</li>
  6. * <li>-41003: aes 解密失败</li>
  7. * <li>-41004: 解密后得到的buffer非法</li>
  8. * <li>-41005: base64加密失败</li>
  9. * <li>-41016: base64解密失败</li>
  10. * </ul>
  11. */
  12. class ErrorCode
  13. {
  14. public static $OK = 0;
  15. public static $IllegalAesKey = -41001;
  16. public static $IllegalIv = -41002;
  17. public static $IllegalBuffer = -41003;
  18. public static $DecodeBase64Error = -41004;
  19. }
  20. ?>

小程序util.js文件

  1. const formatTime = date => {
  2. const year = date.getFullYear()
  3. const month = date.getMonth() + 1
  4. const day = date.getDate()
  5. const hour = date.getHours()
  6. const minute = date.getMinutes()
  7. const second = date.getSeconds()
  8. return [year, month, day].map(formatNumber).join('/') + ' ' + [hour, minute, second].map(formatNumber).join(':')
  9. }
  10. // 网络请求
  11. const request = function(url, method, data, msg, succ, fail, com) {
  12. // 小程序顶部显示Loading
  13. wx.showNavigationBarLoading();
  14. if (msg != "") {
  15. wx.showLoading({
  16. title: msg
  17. })
  18. }
  19. wx.request({
  20. url: 'http://localhost/shige/minP'+url,
  21. data: data,
  22. header: {
  23. 'content-type': 'application/x-www-form-urlencoded'
  24. },
  25. method: method,
  26. success: res => {
  27. if (succ) succ(res);
  28. },
  29. fail: err => {
  30. wx.showToast({
  31. title: '网络错误,请稍后再试···',
  32. icon:'none'
  33. })
  34. if (fail) fail(err);
  35. },
  36. complete: com => {
  37. wx.hideNavigationBarLoading();
  38. if (msg != "") {
  39. wx.hideLoading();
  40. }
  41. console.log(url + ' 返回的data:', com.data);
  42. }
  43. })
  44. }
  45. const formatNumber = n => {
  46. n = n.toString()
  47. return n[1] ? n : '0' + n
  48. }
  49. module.exports = {
  50. formatTime: formatTime,
  51. request: request
  52. }

 

相关技术文章

点击QQ咨询
开通会员
返回顶部
×
微信扫码支付
微信扫码支付
确定支付下载
请使用微信描二维码支付
×

提示信息

×

选择支付方式

  • 微信支付
  • 支付宝付款
确定支付下载