Donutsapps 2019-06-29
首先呢,“登录”、“授权”、“授权登录”,是一样的意思,不用纠结。
写小程序授权登录的代码前,需要了解清楚openid与unionid的区别,这里再简单介绍一下:
方式一:通过code调用code2session接口获得message,包含openid、session_key,满足条件的情况下还能直接获得unionid
/** * Author: huanglp * Date: 2018-11-28 */ public class WeiXinUtils { private static Logger log = LoggerFactory.getLogger(WeiXinUtils.class); /** * 通过前端传过来的code, 调用小程序登录接口, 获取到message并返回 (包含openid session_key等) * * @param code * @return */ public static JSONObject login(String code) { log.info("==============小程序登录方法开始================"); WxMiniProperties properties = WeiXinPropertiesUtils.getWxMiniProperties(); String url = properties.getInterfaceUrl() + "/sns/jscode2session?appid=" + properties.getAppId() + "&secret=" + properties.getAppSecret() + "&js_code=" + code + "&grant_type=authorization_code"; JSONObject message; try { // RestTemplate是Spring封装好的, 挺好用, 可做成单例模式 RestTemplate restTemplate = new RestTemplate(); String response = restTemplate.getForObject(url, String.class); message = JSON.parseObject(response); } catch (Exception e) { log.error("微信服务器请求错误", e); message = new JSONObject(); } log.info("message:" + message.toString()); log.info("==============小程序登录方法结束================"); return message; // 后续, 可获取openid session_key等数据, 以下代码一般放在Service层 //if (message.get("errcode") != null) { // throw new ValidationException(message.toString()); //} //String openid = message.get("openid").toString(); //String sessionKey = message.get("session_key").toString(); //... } }
public class WeiXinPropertiesUtils { // 微信小程序配置 private static WxMiniProperties miniProperties; // 微信公众号配置 private static WxProperties wxProperties; private static void init() { if (miniProperties == null) { miniProperties = ContextLoader.getCurrentWebApplicationContext() .getBean(WxMiniProperties.class); } if (wxProperties == null) { wxProperties = ContextLoader.getCurrentWebApplicationContext() .getBean(WxProperties.class); } } public static WxMiniProperties getWxMiniProperties() { init(); return miniProperties; } public static WxProperties getWxProperties() { init(); return wxProperties; } }
@Data @Component @ConfigurationProperties(prefix = "luwei.module.wx-mini") public class WxMiniProperties { private String appId; private String appSecret; private String interfaceUrl; }
到此已能通过code获取到用户的openid和session_key,但若不满足条件,即使将小程序绑定到微信开放平台上,也获取不到unionid,所以此方式不稳定,推荐使用解密的方式获取数据。
/** * 通过encryptedData,sessionKey,iv获得解密信息, 拥有用户丰富的信息, 包含openid,unionid,昵称等 */ public static JSONObject decryptWxData(String encryptedData, String sessionKey, String iv) throws Exception { log.info("============小程序登录解析数据方法开始=========="); String result = AesCbcUtil.decrypt(encryptedData, sessionKey, iv, "UTF-8"); JSONObject userInfo = new JSONObject(); if (null != result && result.length() > 0) { userInfo = JSONObject.parseObject(result); } log.info("result: " + userInfo); log.info("============小程序登录解析数据方法结束=========="); return userInfo; }
package com.luwei.common.utils; import org.bouncycastle.jce.provider.BouncyCastleProvider; import org.apache.commons.codec.binary.Base64; import javax.crypto.Cipher; import javax.crypto.spec.IvParameterSpec; import javax.crypto.spec.SecretKeySpec; import java.security.AlgorithmParameters; import java.security.Security; /** * Updated by huanglp * Date: 2018-11-28 */ public class AesCbcUtil { static { Security.addProvider(new BouncyCastleProvider()); } /** * AES解密 * * @param data //被加密的数据 * @param key //加密秘钥 * @param iv //偏移量 * @param encoding //解密后的结果需要进行的编码 */ public static String decrypt(String data, String key, String iv, String encoding) { // org.apache.commons.codec.binary.Base64 byte[] dataByte = Base64.decodeBase64(data); byte[] keyByte = Base64.decodeBase64(key); byte[] ivByte = Base64.decodeBase64(iv); try { Cipher cipher = Cipher.getInstance("AES/CBC/PKCS7Padding"); SecretKeySpec spec = new SecretKeySpec(keyByte, "AES"); AlgorithmParameters parameters = AlgorithmParameters.getInstance("AES"); parameters.init(new IvParameterSpec(ivByte)); cipher.init(Cipher.DECRYPT_MODE, spec, parameters);// 初始化 byte[] resultByte = cipher.doFinal(dataByte); if (null != resultByte && resultByte.length > 0) { return new String(resultByte, encoding); } return null; } catch (Exception e) { e.printStackTrace(); } return null; } }
到此已经获取到 JSONObject类型的 userInfo,包含openid,unionid,昵称,头像等数据
后续可以将用户信息保存到数据库,再返回给前端一个token即可,shiro经过公司封装了一层,代码如下:
... // 获得用户ID int userId = wxUser.getWxUserId(); shiroTokenService.afterLogout(userId); String uuid = UUID.randomUUID().toString(); String token = StringUtils.deleteAny(uuid, "-") + Long.toString(System.currentTimeMillis(), Character.MAX_RADIX); shiroTokenService.afterLogin(userId, token, null); return token;
网页授权更加简单,可查看 官方文档
需添加 riversoft 相关依赖包,公众号网页授权,只需要将公众号绑定了开放平台,就能获取到unionid及其他用户信息。
public static OpenUser webSiteLogin(String code, String state) { log.info("============微信公众号(网页)授权开始==========="); WxProperties properties = WeiXinPropertiesUtils.getWxProperties(); AppSetting appSetting = new AppSetting(properties.getAppId(), properties.getAppSecret()); OpenOAuth2s openOAuth2s = OpenOAuth2s.with(appSetting); AccessToken accessToken = openOAuth2s.getAccessToken(code); // 获取用户信息 OpenUser openUser = openOAuth2s.userInfo(accessToken.getAccessToken(), accessToken.getOpenId()); log.info("============微信公众号(网页)授权结束==========="); return openUser; // 后续, 可将用户信息保存 // 最后一步, 生成token后, 需重定向回页面 //return "redirect:" + state + "?token=" + token; }
隆鹏
广州芦苇科技Java开发团队
芦苇科技-广州专业互联网软件服务公司
抓住每一处细节 ,创造每一个美好
关注我们的公众号,了解更多
想和我们一起奋斗吗?lagou搜索“ 芦苇科技 ”或者投放简历到 [email protected] 加入我们吧
关注我们,你的评论和点赞对我们最大的支持