项目中需要用到小程序登录,所以算是走了一遍小程序登录的流程,在这里记录一下
首先需要用到的有: appId和秘钥(秘钥这里简单说下,因为小程序的账号不在我们这边所以资料都是客户提供的,让他给我秘钥,给了我一个text文本,贼长,一直测不通,后来才知秘钥只有11118128191119899ca6d6134829b6a5这么长,这是我替换过的,长度就是这么长。 )
下面先贴上用到的工具类:
1、请求工具类 用来发送post请求
/**
* 发送post请求
*
* @param url
* @param param
* @return
*/
public static String doPost(String url, Map<String, String> param) {
// 创建Httpclient对象
CloseableHttpClient httpClient = HttpClients.createDefault();
CloseableHttpResponse response = null;
String resultString = "";
try {
// 创建Http Post请求
HttpPost httpPost = new HttpPost(url);
// 创建参数列表
if (param != null) {
List<NameValuePair> paramList = new ArrayList<NameValuePair>();
for (String key : param.keySet()) {
paramList.add(new BasicNameValuePair(key, param.get(key)));
}
// 模拟表单
UrlEncodedFormEntity entity = new UrlEncodedFormEntity(paramList);
httpPost.setEntity(entity);
}
// 执行http请求
response = httpClient.execute(httpPost);
resultString = EntityUtils.toString(response.getEntity(), "utf-8");
} catch (Exception e) {
e.printStackTrace();
} finally {
try {
response.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return resultString;
}
2、解密工具类 用来解密用户数据
/**
* AES解密
*
* @param data //密文,被加密的数据
* @param key //秘钥
* @param iv //偏移量
* @param encodingFormat //解密后的结果需要进行的编码
* @return
*/
public static String decrypt(String data, String key, String iv, String encodingFormat) {
// 被加密的数据
byte[] dataByte = Base64Utils.decode(data.getBytes());
// 加密秘钥
byte[] keyByte = Base64Utils.decode(key.getBytes());
// 偏移量
byte[] ivByte = Base64Utils.decode(iv.getBytes());
try {
// 如果密钥不足16位,那么就补足. 这个if 中的内容很重要
int base = 16;
if (keyByte.length % base != 0) {
int groups = keyByte.length / base + (keyByte.length % base != 0 ? 1 : 0);
byte[] temp = new byte[groups * base];
Arrays.fill(temp, (byte) 0);
System.arraycopy(keyByte, 0, temp, 0, keyByte.length);
keyByte = temp;
}
// 初始化
Security.addProvider(new BouncyCastleProvider());
Cipher cipher = Cipher.getInstance("AES/CBC/PKCS7Padding", "BC");
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, encodingFormat);
}
} catch (NoSuchAlgorithmException e) {
e.printStackTrace();
} catch (NoSuchPaddingException e) {
e.printStackTrace();
} catch (InvalidParameterSpecException e) {
e.printStackTrace();
} catch (IllegalBlockSizeException e) {
e.printStackTrace();
} catch (BadPaddingException e) {
e.printStackTrace();
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
} catch (InvalidKeyException e) {
e.printStackTrace();
} catch (InvalidAlgorithmParameterException e) {
e.printStackTrace();
} catch (NoSuchProviderException e) {
e.printStackTrace();
}
return null;
}
以下为请求用户信息的代码
@GetMapping("/login")
public Object chatLogin(String encryptedData, String iv, String code) {
if (StringUtils.isEmpty(code)) {
throw new MyException(Constant.MISS_PARAM);
}
Map<String, String> paramMap = new HashMap<>(16);
paramMap.put("appid", appId);
paramMap.put("secret", appSecret);
paramMap.put("js_code", code);
paramMap.put("grant_type", GRANT_TYPE);
String firstResult = HttpRequestUtil.doPost(GRANT_URL, paramMap);
//发送请求 解析相应内容(转换成json对象)
JSONObject json = JSONObject.parseObject(firstResult);
// 获取会话密钥session_key
String session_key = json.get("session_key").toString();
// 用户的唯一标识openid 这里用不到
//String openid = json.get("openid").toString();
//2、对encryptedData加密数据进行AES解密
String result = AesCbcUtil.decrypt(encryptedData, session_key, iv, "UTF-8");
if (StringUtils.isEmpty(result)) {
log.error("解析微信用户数据时出现异常");
throw new MyException(Constant.UNKNOWN_ERROR);
} else {
JSONObject userInfoJson = JSONObject.parseObject(result);
// String unionId = userInfoJson.get("unionId").toString();
//暂时取openId
String unionId = userInfoJson.get("openId").toString();
ChatUser user = chatUserService.getOne(new QueryWrapper<ChatUser>().eq("union_id", unionId));
//第一次注册 需要不全信息
Map<String, Object> resultMap = new HashMap<>(16);
if (user == null) {
user = new ChatUser();
user.setNickName(userInfoJson.get("nickName").toString());
user.setAvatar(userInfoJson.get("avatarUrl").toString());
user.setUnionId(unionId);
user.setCreateTime(new Date());
user.setIsRegister(0);
chatUserService.save(user);
}
String token = Constant.CHAT_USER + UUID.randomUUID().toString().replaceAll("-", "");
redisTemplate.opsForValue().set(token, user, 30, TimeUnit.MINUTES);
resultMap.put("token", token);
resultMap.put("user", user);
return ResultUtil.success(resultMap);
}
}
这里代码还没有写完,因为涉及到微信开放平台,暂时拿不到unionId,所以暂时先用openId做代替。