)这个错误通常意味着你正在尝试将一个非 JSON 格式的数据转换成 JSON 格式。为了修复这个问题,你需要找到导致错误的代码行并检查它的输入数据是否符合 JSON 格式标准。这个错误通常是因为你的 TypeScript 版本不支持 Node.js ..._uncaught syntaxerror: "undefined" is not valid json at js">
当前位置:   article > 正文

前端系列17集-和公司架构师在学习vue3-springboot

uncaught syntaxerror: "undefined" is not valid json at json.parse (

SyntaxError: "undefined" is not valid JSON at JSON.parse (<anonymous>)

这个错误通常意味着你正在尝试将一个非 JSON 格式的数据转换成 JSON 格式。为了修复这个问题,你需要找到导致错误的代码行并检查它的输入数据是否符合 JSON 格式标准。

这个错误通常是因为你的 TypeScript 版本不支持 Node.js 的类型定义文件。你可以尝试升级 TypeScript 到最新版本,或者使用 @types/node 包来获取 Node.js 的类型定义文件。

具体的步骤如下:

  1. 在命令行中进入项目根目录,执行以下命令安装 @types/node 包:

    npm install --save-dev @types/node
  2. 确认你的 TypeScript 配置文件(一般为 tsconfig.json)中包含了以下配置:

    1. {
    2. "compilerOptions": {
    3. "target": "es6",
    4. "module": "commonjs",
    5. "esModuleInterop": true,
    6. "types": ["node"]
    7. }
    8. }
  3. 重新编译你的代码,如果使用的是 TypeScript 编译器,执行以下命令:

    tsc

    如果使用的是 Webpack 等构建工具进行编译,按照相应的配置进行操作即可。

这样就解决了 Cannot find namespace 'NodeJS' 的问题。

需求:实现一个拖拽指令,可在父元素区域任意拖拽元素。

  1. 思路:
  2. 1、设置需要拖拽的元素为absolute,其父元素为relative。
  3. 2、鼠标按下(onmousedown)时记录目标元素当前的 left 和 top 值。
  4. 3、鼠标移动(onmousemove)时计算每次移动的横向距离和纵向距离的变化值,并改变元素的 left 和 top 值
  5. 4、鼠标松开(onmouseup)时完成一次拖拽
  6. 使用:在 Dom 上加上 v-draggable 即可

Property 'browserLanguage' does not exist on type 'Navigator'.ts

这个错误通常发生在 TypeScript 中,因为 Navigator 类型没有 browserLanguage 属性。相反,应该使用 navigator.language 属性来获取浏览器当前所用的语言。

要解决这个问题,你需要将代码中所有引用 browserLanguage 的地方替换为 language

例如,如果你的代码是这样的:

  1. const language = navigator.browserLanguage;
  2. console.log(`The browser's language is ${language}`);

应该改为:

  1. const language = navigator.language;
  2. console.log(`The browser's language is ${language}`);

这样就不会再出现类型错误了。

Vue.js 应用程序的国际化配置。它依赖于 vue-i18n 包,并且包含以下步骤:

  1. 导入 createI18n 方法和 getBrowserLang 函数,它们分别来自于 vue-i18n 和自定义的 utils 模块。

  2. 导入了两个语言模块:zhen,它们分别存储了中文和英文版本的翻译。

  3. 使用 createI18n 方法创建 i18n 实例,配置了以下属性:

  • allowComposition:设置为 true,表示使用 Composition API。

  • legacy:设置为 false,表示不使用旧版 Vue.js 的选项 API。

  • locale:根据用户浏览器的语言设置,选择合适的语言作为默认语言。

  • messages:将导入的语言模块对象作为消息传入,供 i18n 实例使用。

  1. 导出 i18n 实例作为默认模块。

6a6625d2bed8955b38bea17486406381.png
image.png
  1. package com.xx.app.utils.constants;
  2. public class Constants {
  3. public static final String SUCCESS_CODE = "200";
  4. public static final String SUCCESS_MSG = "操作成功";
  5. }
  1. package com.xx.app.utils.constants;
  2. public enum ErrorEnum {
  3. E_400("400", "请求处理异常,请稍后再试"),
  4. E_500("500", "请求方式有误,请检查 GET/POST"),
  5. E_501("501", "请求路径不存在"),
  6. E_502("502", "权限不足"),
  7. E_10008("10008", "角色删除失败,尚有用户属于此角色"),
  8. E_10009("10009", "账户已存在"),
  9. E_10010("10010", "账号/密码错误"),
  10. E_20011("20011", "登录已过期,请重新登录"),
  11. E_90003("90003", "缺少必填参数");
  12. private final String errorCode;
  13. private final String errorMsg;
  14. ErrorEnum(String errorCode, String errorMsg) {
  15. this.errorCode = errorCode;
  16. this.errorMsg = errorMsg;
  17. }
  18. public String getErrorCode() {
  19. return errorCode;
  20. }
  21. public String getErrorMsg() {
  22. return errorMsg;
  23. }
  24. }
  1. package com.xxx.app.utils.model;
  2. import com.alibaba.fastjson.JSONObject;
  3. import java.util.List;
  4. import java.util.Set;
  5. /**
  6. * MyBatis的一对多JSON返回对象
  7. * 处理嵌套查询结果时,MyBatis会根据bean定义的属性类型来初始化嵌套的成员变量,主要看其是不是Collection
  8. * 如果这里不定义,那么嵌套返回结果里就只能返回一对一的结果,而不是一对多的
  9. * 参见MyBatis DefaultResultSetHandler.instantiateCollectionPropertyIfAppropriate()
  10. */
  11. public class One2Many extends JSONObject {
  12. private Set<String> roleList;
  13. private Set<String> menuList;
  14. private Set<String> permissionList;
  15. private Set<Integer> permissionIds;
  16. private List<JSONObject> picList;
  17. private List<JSONObject> menus;
  18. private List<JSONObject> users;
  19. private List<JSONObject> permissions;
  20. private List<JSONObject> roles;
  21. }
  1. package com.xxx.app.utils;
  2. import com.xxx.app.utils.constants.Constants;
  3. import com.xxx.app.utils.constants.ErrorEnum;
  4. import com.alibaba.fastjson.JSONObject;
  5. import javax.servlet.http.HttpServletRequest;
  6. import java.util.Enumeration;
  7. import java.util.List;
  8. /**
  9. * 后台接口系统常用的json工具类
  10. */
  11. public class CommonUtil {
  12. /**
  13. * 返回一个info为空对象的成功消息的json
  14. */
  15. public static JSONObject successJson() {
  16. return successJson(new JSONObject());
  17. }
  18. public static JSONObject successJson(Object info) {
  19. JSONObject resultJson = new JSONObject();
  20. resultJson.put("code", Constants.SUCCESS_CODE);
  21. resultJson.put("msg", Constants.SUCCESS_MSG);
  22. resultJson.put("info", info);
  23. return resultJson;
  24. }
  25. /**
  26. * 返回错误信息JSON
  27. */
  28. public static JSONObject errorJson(ErrorEnum errorEnum) {
  29. JSONObject resultJson = new JSONObject();
  30. resultJson.put("code", errorEnum.getErrorCode());
  31. resultJson.put("msg", errorEnum.getErrorMsg());
  32. resultJson.put("info", new JSONObject());
  33. return resultJson;
  34. }
  35. /**
  36. * 查询分页结果后的封装工具方法
  37. * @param requestJson 请求参数json,此json在之前调用fillPageParam 方法时,已经将pageRow放入
  38. * @param list 查询分页对象list
  39. * @param totalCount 查询出记录的总条数
  40. */
  41. public static JSONObject successPage(final JSONObject requestJson, List<JSONObject> list, int totalCount) {
  42. int pageRow = requestJson.getIntValue("pageRow");
  43. int totalPage = getPageCounts(pageRow, totalCount);
  44. JSONObject result = successJson();
  45. JSONObject info = new JSONObject();
  46. info.put("list", list);
  47. info.put("totalCount", totalCount);
  48. info.put("totalPage", totalPage);
  49. result.put("info", info);
  50. return result;
  51. }
  52. /**
  53. * 获取总页数
  54. *
  55. * @param pageRow 每页行数
  56. * @param itemCount 结果的总条数
  57. */
  58. private static int getPageCounts(int pageRow, int itemCount) {
  59. if (itemCount == 0) {
  60. return 1;
  61. }
  62. return itemCount % pageRow == 0? itemCount / pageRow : itemCount / pageRow + 1;
  63. //return (itemCount + pageRow - 1) / pageRow;
  64. }
  65. /**
  66. * 查询分页结果后的封装工具方法
  67. *
  68. * @param list 查询分页对象list
  69. */public static JSONObject successPage(List<JSONObject> list) {
  70. JSONObject result = successJson();
  71. JSONObject info = new JSONObject();
  72. info.put("list", list);
  73. result.put("info", info);
  74. return result;
  75. }
  76. /**
  77. * 将request参数值转为json
  78. */
  79. public static JSONObject request2Json(HttpServletRequest request) {
  80. JSONObject requestJson = new JSONObject();
  81. Enumeration<String> paramNames = request.getParameterNames();
  82. while (paramNames.hasMoreElements()) {
  83. String paramName = (String) paramNames.nextElement();
  84. String[] pv = request.getParameterValues(paramName);
  85. StringBuilder sb = new StringBuilder();
  86. for (int i = 0; i < pv.length; i++) {
  87. if (pv[i].length() > 0) {
  88. if (i > 0) {
  89. sb.append(",");
  90. }
  91. sb.append(pv[i]);
  92. }
  93. }
  94. requestJson.put(paramName, sb.toString());
  95. }
  96. return requestJson;
  97. }
  98. /**
  99. * 将request转JSON
  100. * 并且验证非空字段
  101. */
  102. public static JSONObject convert2JsonAndCheckRequiredColumns(HttpServletRequest request, String requiredColumns) {
  103. JSONObject jsonObject = request2Json(request);
  104. hasAllRequired(jsonObject, requiredColumns);
  105. return jsonObject;
  106. }
  107. /**
  108. * 验证是否含有全部必填字段
  109. *
  110. * @param requiredColumns 必填的参数字段名称 逗号隔开 比如"userId,name,telephone"
  111. */
  112. public static void hasAllRequired(final JSONObject jsonObject, String requiredColumns) {
  113. if (!StringTools.isNullOrEmpty(requiredColumns)) {
  114. //验证字段非空
  115. String[] columns = requiredColumns.split(",");
  116. String missCol = "";
  117. for (String column : columns) {
  118. Object val = jsonObject.get(column.trim());
  119. if (StringTools.isNullOrEmpty(val)) {
  120. missCol += column + " ";
  121. }
  122. }
  123. if (!StringTools.isNullOrEmpty(missCol)) {
  124. jsonObject.clear();
  125. jsonObject.put("code", ErrorEnum.E_90003.getErrorCode());
  126. jsonObject.put("msg", "缺少必填参数:" + missCol.trim());
  127. jsonObject.put("info", new JSONObject());
  128. throw null;
  129. }
  130. }
  131. }
  132. /**
  133. * 在分页查询之前,为查询条件里加上分页参数
  134. *
  135. * @param paramObject 查询条件json
  136. * @param defaultPageRow 默认的每页条数,即前端不传pageRow参数时的每页条数
  137. */
  138. private static void fillPageParam(final JSONObject paramObject, int defaultPageRow) {
  139. int pageNum = paramObject.getIntValue("pageNum");
  140. pageNum = pageNum == 0 ? 1 : pageNum;
  141. int pageRow = paramObject.getIntValue("pageRow");
  142. pageRow = pageRow == 0 ? defaultPageRow : pageRow;
  143. paramObject.put("offSet", (pageNum - 1) * pageRow);
  144. paramObject.put("pageRow", pageRow);
  145. paramObject.put("pageNum", pageNum);
  146. //删除此参数,防止前端传了这个参数,pageHelper分页插件检测到之后,拦截导致SQL错误
  147. paramObject.remove("pageSize");
  148. }
  149. /**
  150. * 分页查询之前的处理参数
  151. * 没有传pageRow参数时,默认每页10条.
  152. */
  153. public static void fillPageParam(final JSONObject paramObject) {
  154. fillPageParam(paramObject, 10);
  155. }
  156. }
  1. package com.xxx.app.utils;
  2. public class StringTools {
  3. public static boolean isNullOrEmpty(String str) {
  4. return null == str || "".equals(str) || "null".equals(str);
  5. }
  6. public static boolean isNullOrEmpty(Object obj) {
  7. return null == obj || "".equals(obj);
  8. }
  9. }
  1. package com.xxxx.app.utils;
  2. import com.auth0.jwt.JWT;
  3. import com.auth0.jwt.JWTVerifier;
  4. import com.auth0.jwt.algorithms.Algorithm;
  5. import com.auth0.jwt.exceptions.JWTDecodeException;
  6. import com.auth0.jwt.interfaces.DecodedJWT;
  7. import java.io.UnsupportedEncodingException;
  8. import java.util.Date;
  9. public class JWTUtils {
  10. public static final String AUTH_HEADER_KEY = "Authorization";
  11. public static final String TOKEN_PREFIX = "Bearer ";
  12. //过期时间一周
  13. private static final long EXPIRE_TIME = 7 * 24 * 60 * 60 * 1000;
  14. private static final String secret = "";
  15. /**
  16. * 验证 JWT token 是否合法
  17. *
  18. * @param token 待验证的 JWT token
  19. * @return 如果 token 合法,则返回 true;否则返回 false。
  20. */
  21. public static boolean verify(String token) {
  22. try {
  23. Algorithm algorithm = Algorithm.HMAC256(secret);
  24. // 创建一个 JWT 验证器
  25. JWTVerifier verifier = JWT.require(algorithm).build();
  26. // 使用验证器对 token 进行验证
  27. DecodedJWT jwt = verifier.verify(token);
  28. return true;
  29. } catch (Exception exception) {
  30. return false;
  31. }
  32. }
  33. /**
  34. * 获取token中信息无需secret解密也能获得
  35. * @param token
  36. * @return token中包含的id
  37. */
  38. public static Long getUserId(String token) {
  39. try {
  40. DecodedJWT jwt = JWT.decode(token);
  41. return jwt.getClaim("id").asLong();
  42. } catch (JWTDecodeException e) {
  43. return null;
  44. }
  45. }
  46. /**
  47. * 签发token
  48. * @param id
  49. * @return 加密的token
  50. */
  51. public static String sign(Long id) {
  52. try {
  53. Date date = new Date(System.currentTimeMillis() + EXPIRE_TIME);
  54. Algorithm algorithm = Algorithm.HMAC256(secret);
  55. // 附带username信息
  56. return JWT.create()
  57. .withClaim("id", id)
  58. .withExpiresAt(date)
  59. .sign(algorithm);
  60. } catch (UnsupportedEncodingException e) {
  61. return null;
  62. }
  63. }
  64. }
  1. package com.xxx.app.service;
  2. import cn.hutool.http.HttpUtil;
  3. import com.alibaba.fastjson.JSON;
  4. import com.xxx.app.common.RedisKey;
  5. import com.xxx.app.handler.UserThreadLocal;
  6. import com.xxx.app.model.WXAuth;
  7. import com.xxx.app.pojo.User;
  8. import com.xxx.app.pojo.dto.UserDto;
  9. import com.xxx.app.utils.JWTUtils;
  10. import lombok.extern.slf4j.Slf4j;
  11. import org.springframework.beans.BeanUtils;
  12. import org.springframework.beans.factory.annotation.Autowired;
  13. import org.springframework.beans.factory.annotation.Value;
  14. import org.springframework.data.redis.core.StringRedisTemplate;
  15. import org.springframework.stereotype.Service;
  16. import java.util.HashMap;
  17. import java.util.Map;
  18. import java.util.UUID;
  19. import java.util.concurrent.TimeUnit;
  20. @Slf4j
  21. @Service
  22. public class UserService {
  23. @Value("${wxmini.appid}")
  24. private String appid;
  25. @Value("${wxmini.secret}")
  26. private String secret;
  27. @Autowired
  28. private StringRedisTemplate redisTemplate;
  29. // public Result getSessionId(String code) {
  30. // /**
  31. // * 1. 拼接一个url,微信登录凭证校验接口
  32. // * 2. 发起一个http的调用,获取微信的返回结果
  33. // * 3. 存到redis
  34. // * 4. 生成一个sessionId,返回给前端,作为当前需要登录用户的标识
  35. // * 5. 生成一个sessionId,用户在点击微信登录的时候,我们可以标识是谁点击微信登录
  36. // */
  37. // String url = "https://api.weixin.qq.com/sns/jscode2session?appid={0}&secret={1}&js_code={2}&grant_type=authorization_code";
  38. // String replaceUrl = url.replace("{0}", appid).replace("{1}", secret).replace("{2}", code);
  39. // String res = HttpUtil.get(replaceUrl);
  40. // String uuid = UUID.randomUUID().toString();
  41. // redisTemplate.opsForValue().set(RedisKey.WX_SESSION_ID + uuid, res, 30, TimeUnit.MINUTES);
  42. // Map<String, String> map = new HashMap<>();
  43. // map.put("sessionId", uuid);
  44. // return Result.SUCCESS(map);
  45. // }
  46. @Autowired
  47. private WxService wxService;
  48. // @Resource
  49. // private UserMapper userMapper;
  50. // public Result authLogin(WXAuth wxAuth) {
  51. // /**
  52. // * 1. 通过wxAuth中的值,要对它进行解密
  53. // * 2. 解密完成之后,会获取到微信用户信息 其中包含 openId,性别,昵称,头像等信息
  54. // * 3. openId 是唯一的,需要去user表中查询openId是否存在,存在,已此用户的身份登录成功
  55. // * 4. 不存在,新用户,注册流程,登录成功
  56. // * 5. 使用jwt技术,生成一个token,提供给前端 token 令牌,用户在下次访问的时候,携带token来访问
  57. // * 6. 后端通过对token的验证,找到此用户是否处于登录状态,以及是哪个用户登录的
  58. // */
  59. try {
  60. log.info("获取请求参数:"+wxAuth);
  61. String wxRes = wxService.wxDecrypt(wxAuth.getEncryptedData(), wxAuth.getSessionId(), wxAuth.getIv());
  62. log.info("信息:"+wxRes);
  63. WxUserInfo wxUserInfo = JSON.parseObject(wxRes, WxUserInfo.class);
  64. String openId = wxUserInfo.getOpenId();
  65. User user = userMapper.selectOne(Wrappers.<User>lambdaQuery().eq(User::getOpenId, openId).last("limit 1"));
  66. UserDto userDto = new UserDto();
  67. userDto.from(wxUserInfo);
  68. if (user == null) {
  69. // 注册
  70. return this.register(userDto);
  71. } else {
  72. userDto.setId(user.getId());
  73. // 登录
  74. return this.login(userDto);
  75. }
  76. } catch (Exception e) {
  77. e.printStackTrace();
  78. }
  79. return Result.FAIL();
  80. // return null;
  81. // }
  82. //
  83. // private Result login(UserDto userDto) {
  84. // String token = JWTUtils.sign(userDto.getId());
  85. // userDto.setToken(token);
  86. // userDto.setOpenId(null);
  87. // userDto.setUnionId(null);
  88. // //需要把token 存入redis,value存为 userDto,下次用户访问需要登录资源的时候,可以根据token拿到用户的详细信息
  89. // //缓存
  90. // redisTemplate.opsForValue().set(RedisKey.TOKEN+token, JSON.toJSONString(userDto), 7, TimeUnit.DAYS);
  91. // return Result.SUCCESS(userDto);
  92. // }
  93. //
  94. // private Result register(UserDto userDto) {
  95. // // 注册之前 判断 用户是否存在
  96. // User user = new User();
  97. // BeanUtils.copyProperties(userDto, user);
  98. this.userMapper.insert(user);
  99. // // return Result.SUCCESS(userDto);
  100. // userDto.setId(user.getId());
  101. // return this.login(userDto);
  102. // }
  103. //
  104. // // 参数 String token
  105. // public Result userinfo(Boolean refresh) {
  106. // /**
  107. // * 1. 根据token 来验证此token 是否有效
  108. // * 2. refresh 如果为true 代表刷新 重新生成token和redis里面重新储存 续期
  109. // * 3. false直接返回用户信息 redis中 查询出来 直接返回
  110. // */
  111. // /**
  112. // token = token.replace("Bearer ", "");
  113. // boolean verify = JWTUtils.verify(token);
  114. // if (!verify) {
  115. // return Result.FAIL("未登录");
  116. // }
  117. // String userJson = redisTemplate.opsForValue().get(RedisKey.TOKEN + token);
  118. // if (StringUtils.isBlank(userJson)) {
  119. // return Result.FAIL("未登录");
  120. // }
  121. // */
  122. // UserDto userDto = UserThreadLocal.get();
  123. // if (refresh) {
  124. // String token = JWTUtils.sign(userDto.getId());
  125. // userDto.setToken(token);
  126. // redisTemplate.opsForValue().set(RedisKey.TOKEN+token, JSON.toJSONString(userDto), 7, TimeUnit.DAYS);
  127. // }
  128. // return Result.SUCCESS(userDto);
  129. // }
  130. }
  1. package com.xxx.app.common;
  2. public enum ResultCode {
  3. /* 成功状态码 */
  4. SUCCESS(0, "操作成功!"),
  5. /* 错误状态码 */
  6. FAIL(-1, "操作失败!"),
  7. /* 参数错误:10001-19999 */
  8. PARAM_IS_INVALID(10001, "参数无效"),
  9. PARAM_IS_BLANK(10002, "参数为空"),
  10. PARAM_TYPE_BIND_ERROR(10003, "参数格式错误"),
  11. PARAM_NOT_COMPLETE(10004, "参数缺失"),
  12. /* 用户错误:20001-29999*/
  13. USER_NOT_LOGGED_IN(20001, "用户未登录,请先登录"),
  14. USER_LOGIN_ERROR(20002, "账号不存在或密码错误"),
  15. USER_ACCOUNT_FORBIDDEN(20003, "账号已被禁用"),
  16. USER_NOT_EXIST(20004, "用户不存在"),
  17. USER_HAS_EXISTED(20005, "用户已存在"),
  18. /* 业务错误:30001-39999 */
  19. BUSINESS_GROUP_NO_ALLOWED_DEL(30001, "应用分组已经被应用使用,不能删除"),
  20. BUSINESS_THEME_NO_ALLOWED_DEL(30002, "主题已经被用户使用,不能删除"),
  21. BUSINESS_THEME_NO_ALLOWED_DISABLE(30003, "主题已经被用户使用,不能停用"),
  22. BUSINESS_THEME_DEFAULT_NO_ALLOWED_DEL(30004, "默认主题,不能删除"),
  23. BUSINESS_THEME_NO_ALLOWED_UPDATE(30005, "主题已经被用户使用,不能修改图片信息"),
  24. BUSINESS_IS_TOP(30040, "已经到最顶部"),
  25. BUSINESS_IS_BOTTOM(30041, "已经到最底部"),
  26. BUSINESS_NAME_EXISTED(30051, "名称已存在"),
  27. /* 系统错误:40001-49999 */
  28. SYSTEM_INNER_ERROR(40001, "系统繁忙,请稍后重试"),
  29. UPLOAD_ERROR(40002, "系统异常,上传文件失败"),
  30. FILE_MAX_SIZE_OVERFLOW(40003, "上传尺寸过大"),
  31. FILE_ACCEPT_NOT_SUPPORT(40004, "上传文件格式不支持"),
  32. SET_UP_AT_LEAST_ONE_ADMIN(40005, "至少指定一个管理员"),
  33. URL_INVALID(40006, "地址不合法"),
  34. LINK_AND_LOGOUT_NO_MATCH(40006, "主页地址和注销地址IP不一致"),
  35. IP_AND_PORT_EXISTED(40007, "当前IP和端口已经被占中"),
  36. LINK_IS_REQUIRED(40008, "生成第三方token认证信息: 主页地址不能为空,请完善信息"),
  37. ONLY_ROOT_DEPARTMENT(40009, "组织机构只能存在一个根机构"),
  38. DEPART_CODE_EXISTED(40010, "组织机构编码已存在"),
  39. DEPART_CONTAINS_USERS(40011, "该机构下是存在用户,不允许删除"),
  40. DEPART_CONTAINS_SON(40012, "该机构下是存在子级机构,不允许删除"),
  41. DEPART_PARENT_IS_SELF(40013, "选择的父机构不能为本身"),
  42. DICT_EXIST_DEPEND(40014, "该字典数据存在详情依赖,不允许删除"),
  43. DICT_DETAIL_LOCK(40015, "该字典数据被锁定,不允许修改或删除"),
  44. DEPART_CODE_EXISTED_WITH_ARGS(40016, "组织机构编码【{0}】系统已存在"),
  45. /* 数据错误:50001-599999 */
  46. RESULT_DATA_NONE(50001, "数据未找到"),
  47. DATA_IS_WRONG(50002, "数据有误"),
  48. DATA_ALREADY_EXISTED(50003, "数据已存在"),
  49. AUTH_CODE_ERROR(50004, "验证码错误"),
  50. /* 接口错误:60001-69999 */
  51. INTERFACE_INNER_INVOKE_ERROR(60001, "内部系统接口调用异常"),
  52. INTERFACE_OUTTER_INVOKE_ERROR(60002, "外部系统接口调用异常"),
  53. INTERFACE_FORBID_VISIT(60003, "该接口禁止访问"),
  54. INTERFACE_ADDRESS_INVALID(60004, "接口地址无效"),
  55. INTERFACE_REQUEST_TIMEOUT(60005, "接口请求超时"),
  56. INTERFACE_EXCEED_LOAD(60006, "接口负载过高"),
  57. /* 权限错误:70001-79999 */
  58. PERMISSION_UNAUTHENTICATED(70001, "此操作需要登陆系统!"),
  59. PERMISSION_UNAUTHORISE(70002, "权限不足,无权操作!"),
  60. PERMISSION_EXPIRE(70003, "登录状态过期!"),
  61. PERMISSION_TOKEN_EXPIRED(70004, "token已过期"),
  62. PERMISSION_LIMIT(70005, "访问次数受限制"),
  63. PERMISSION_TOKEN_INVALID(70006, "无效token"),
  64. PERMISSION_SIGNATURE_ERROR(70007, "签名失败"),
  65. PERMISSION_VIP(70000, "权限不足,此影片需要大会员,请开通大会员");
  66. //操作代码
  67. int code;
  68. //提示信息
  69. String message;
  70. ResultCode(int code, String message) {
  71. this.code = code;
  72. this.message = message;
  73. }
  74. public int code() {
  75. return code;
  76. }
  77. public String message() {
  78. return message;
  79. }
  80. public void setCode(int code) {
  81. this.code = code;
  82. }
  83. public void setMessage(String message) {
  84. this.message = message;
  85. }
  86. }
  1. package com.xxx.app.handler;
  2. import com.alibaba.fastjson.JSON;
  3. import com.xxx.app.common.RedisKey;
  4. import com.xxx.app.pojo.dto.UserDto;
  5. import com.xxx.app.utils.JWTUtils;
  6. import org.apache.commons.lang3.StringUtils;
  7. import org.springframework.beans.factory.annotation.Autowired;
  8. import org.springframework.data.redis.core.StringRedisTemplate;
  9. import org.springframework.stereotype.Component;
  10. import org.springframework.web.method.HandlerMethod;
  11. import org.springframework.web.servlet.HandlerInterceptor;
  12. import javax.servlet.http.HttpServletRequest;
  13. import javax.servlet.http.HttpServletResponse;
  14. // 拦截器的使用 要放入 mvc配置当中
  15. // 用于拦截请求并进行身份认证和授权的操作。
  16. @Component
  17. public class LoginHandler implements HandlerInterceptor {
  18. @Autowired
  19. private StringRedisTemplate redisTemplate;
  20. /**
  21. *
  22. * @param request HTTP 请求对象
  23. * @param response HTTP 响应对象
  24. * @param handler 表示待处理的目标对象
  25. * @return
  26. * @throws Exception
  27. */
  28. @Override
  29. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
  30. //return HandlerInterceptor.super.preHandle(request, response, handler);
  31. //方法执行之前 进行拦截
  32. /**
  33. * 1.判断请求是否是请求 controller 方法
  34. * 2.有些接口是不需要登录拦截,需要开放自定义的注解 @NoAuth 此注解标识的 不需要登录
  35. * 3.拿到token
  36. * 4.token认证 -> user信息
  37. * 5.如果 redis 认证通过 就放行 认证不通过 返回未登录
  38. * 6.得到了用户信息,放入 ThreadLocal 当中
  39. */
  40. // 判断 handler 是否为 HandlerMethod 的实例。如果不是,则直接返回 true,表示放行。
  41. if (!(handler instanceof HandlerMethod)) {
  42. return true;
  43. }
  44. // 将 handler 强制转换为 HandlerMethod 对象,以便后面能够获取处理该请求的 Controller 方法及其注解信息。
  45. HandlerMethod handlerMethod = (HandlerMethod) handler;
  46. // 判断该 Controller 方法是否被 @NoAuth 注解标记,如果有,则直接返回 true,表示放行。
  47. if (handlerMethod.hasMethodAnnotation(NoAuth.class)) {
  48. return true;
  49. }
  50. String token = request.getHeader("Authorization");
  51. token = token.replace("Bearer ", "");
  52. // 从 HTTP 请求头中获取认证 Token,由于 Token 一般以 Bearer 开头,所以需要将其去掉。
  53. boolean verify = JWTUtils.verify(token);
  54. // 使用 JWT 进行 Token 认证,如果认证失败,则返回错误信息并直接返回 false,表示拦截该请求。
  55. if (!verify) {
  56. response.setContentType("application/json;charset=utf-8");
  57. //response.getWriter().write(JSON.toJSONString(Result.FAIL("未登录")));
  58. response.getWriter().write(JSON.toJSONString("未登录"));
  59. return false;
  60. }
  61. String userJson = redisTemplate.opsForValue().get(RedisKey.TOKEN + token);
  62. if (StringUtils.isBlank(userJson)) {
  63. response.setContentType("application/json;charset=utf-8");
  64. //response.getWriter().write(JSON.toJSONString(Result.FAIL("未登录")));
  65. response.getWriter().write(JSON.toJSONString("未登录"));
  66. return false;
  67. }
  68. UserDto userDto = JSON.parseObject(userJson, UserDto.class);
  69. UserThreadLocal.put(userDto); // 得到了用户信息,放入ThreadLocal当中
  70. return true;
  71. }
  72. }
  1. package com.xxx.app.config;
  2. import org.springframework.beans.factory.annotation.Autowired;
  3. import org.springframework.context.annotation.Configuration;
  4. import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
  5. import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
  6. @Configuration
  7. public class WebMvcConfig implements WebMvcConfigurer {
  8. @Autowired
  9. private LoginHandler loginHandler;
  10. @Override
  11. public void addInterceptors(InterceptorRegistry registry) {
  12. registry.addInterceptor(loginHandler);
  13. //WebMvcConfigurer.super.addInterceptors(registry);
  14. }
  15. }

//发送手机验证码

auth/send-sms-code

//使用手机 + 验证码登录

auth/sms-login

//刷新令牌

auth/refresh-token

//微信小程序的一键登录

/auth/weixin-mini-app-login

  1. package com.xxx.app.service;
  2. import cn.hutool.core.codec.Base64;
  3. import com.alibaba.fastjson.JSON;
  4. import com.alibaba.fastjson.JSONObject;
  5. import com.xxx.app.common.RedisKey;
  6. import lombok.extern.slf4j.Slf4j;
  7. import org.springframework.beans.factory.annotation.Autowired;
  8. import org.springframework.data.redis.core.StringRedisTemplate;
  9. import org.springframework.stereotype.Component;
  10. import javax.crypto.Cipher;
  11. import javax.crypto.spec.IvParameterSpec;
  12. import javax.crypto.spec.SecretKeySpec;
  13. import java.security.spec.AlgorithmParameterSpec;
  14. @Slf4j
  15. @Component
  16. public class WxService {
  17.     @Autowired
  18.     private StringRedisTemplate redisTemplate;
  19.     public String wxDecrypt(String encryptedData, String sessionId, String vi) throws Exception {
  20.         // 开始解密
  21.         String json =  redisTemplate.opsForValue().get(RedisKey.WX_SESSION_ID + sessionId);
  22.         log.info("信息wxDecrypt:"+json);
  23.         JSONObject jsonObject = JSON.parseObject(json);
  24.         String sessionKey = (String) jsonObject.get("session_key");
  25.         log.info("sessionKey 信息"+sessionKey);
  26.         byte[] encData = cn.hutool.core.codec.Base64.decode(encryptedData);
  27.         byte[] iv = cn.hutool.core.codec.Base64.decode(vi);
  28.         byte[] key = Base64.decode(sessionKey);
  29.         log.info("key 信息"+key);
  30.         AlgorithmParameterSpec ivSpec = new IvParameterSpec(iv);
  31.         Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");
  32.         SecretKeySpec keySpec = new SecretKeySpec(key, "AES");
  33.         cipher.init(Cipher.DECRYPT_MODE, keySpec, ivSpec);
  34.         return new String(cipher.doFinal(encData), "UTF-8");
  35.     }
  36. }
  1. package com.xxx.app.common;
  2. public class RedisKey {
  3.     public static final String WX_SESSION_ID = "wx_session_id";
  4.     public static final String TOKEN = "token_";
  5. }
  1. package com.xxx.app.entity;
  2. import java.util.Date;
  3. public class Login {
  4.     private int id;
  5.     private String phone;
  6.     private String code;
  7.     private byte role;
  8.     private Date ctime;
  9.     public int getId() {
  10.         return id;
  11.     }
  12.     public void setId(int id) {
  13.         this.id = id;
  14.     }
  15.     public String getPhone() {
  16.         return phone;
  17.     }
  18.     public void setPhone(String phone) {
  19.         this.phone = phone;
  20.     }
  21.     public String getCode() {
  22.         return code;
  23.     }
  24.     public void setCode(String code) {
  25.         this.code = code;
  26.     }
  27.     public byte getRole() {
  28.         return role;
  29.     }
  30.     public void setRole(byte role) {
  31.         this.role = role;
  32.     }
  33.     public Date getCtime() {
  34.         return ctime;
  35.     }
  36.     public void setCtime(Date ctime) {
  37.         this.ctime = ctime;
  38.     }
  39.     @Override
  40.     public String toString() {
  41.         return "Login [id=" + id + ", phone=" + phone + ", code=" + code + ", ctime=" + ctime + ", getId()=" + getId() + ", getPhone()=" + getPhone() + ", getCode()=" + getCode() + ", getCtime()=" + getCtime() + ", getClass()=" + getClass() + ", hashCode()=" + hashCode() + ", toString()=" + super.toString() + "]";
  42.     }
  43. }
  1. package com.xxx.app.entity;
  2. public class Sms {
  3.     private String phoneNumber;
  4.     private String code;
  5.     public String getPhoneNumber() {
  6.         return phoneNumber;
  7.     }
  8.     public  void setPhoneNumber(String phoneNumber) {
  9.         this.phoneNumber = phoneNumber;
  10.     }
  11.     public String getCode() {
  12.         return code;
  13.     }
  14.     public void setCode(String code) {
  15.         this.code = code;
  16.     }
  17. }
  1. package com.xxx.app.handler;
  2. import java.lang.annotation.*;
  3. @Target(ElementType.METHOD)
  4. @Retention(RetentionPolicy.RUNTIME)
  5. @Documented
  6. public @interface NoAuth {
  7. }
  1. package com.xxx.app.handler;
  2. import com.xxx.app.pojo.dto.UserDto;
  3. public class UserThreadLocal {
  4.     private static final ThreadLocal<UserDto> LOCAL = new ThreadLocal<>();
  5.     public static void put(UserDto userDto) {
  6.         LOCAL.set(userDto);
  7.     }
  8.     public static UserDto get() {
  9.         return LOCAL.get();
  10.     }
  11.     public static void remove() {
  12.         LOCAL.remove();
  13.     }
  14. }
  1. package com.xxx.app.model;
  2. import lombok.Data;
  3. @Data
  4. public class WXAuth {
  5.     private String encryptedData;
  6.     private String iv;
  7.     private String sessionId;
  8. }
  1. package com.xxx.app.model;
  2. import lombok.Data;
  3. @Data
  4. public class WxUserInfo {
  5.     private String openId;
  6.     private String nickName;
  7.     private String gender;
  8.     private String city;
  9.     private String province;
  10.     private String country;
  11.     private String avatarUrl;
  12.     private String unionId;
  13. }
  1. package com.xxx.app.pojo.dto;
  2. import com.xxx.app.model.WxUserInfo;
  3. import lombok.AllArgsConstructor;
  4. import lombok.Data;
  5. import lombok.NoArgsConstructor;
  6. import java.io.Serializable;
  7. import java.util.List;
  8. @Data
  9. @NoArgsConstructor
  10. @AllArgsConstructor
  11. public class UserDto implements Serializable {
  12.     private Long id;
  13.     private String nickname;
  14.     private String username;
  15.     private String password;
  16.     private String gender;
  17.     private String phoneNumber;
  18.     private String background;
  19.     private String avatar_url;
  20.     private String openId;
  21.     private String unionId;
  22.     //dto拓展属性
  23.     private String token;
  24.     List<String> permissions;
  25.     List<String> roles;
  26.     // 验证码
  27.     private String code;
  28.     public void from(WxUserInfo wxUserInfo) {
  29.         this.nickname = wxUserInfo.getNickName();
  30.         this.avatar_url = wxUserInfo.getAvatarUrl();
  31.         this.username = "";
  32.         this.password = "";
  33.         this.phoneNumber = "";
  34.         this.gender = wxUserInfo.getGender();
  35.         this.openId = wxUserInfo.getOpenId();
  36.         this.unionId = wxUserInfo.getUnionId();
  37.     }
  38. }
  1. package com.xxx.app.pojo;
  2. //import com.baomidou.mybatisplus.annotation.IdType;
  3. //import com.baomidou.mybatisplus.annotation.TableId;
  4. //import com.baomidou.mybatisplus.annotation.TableName;
  5. import lombok.Data;
  6. import java.io.Serializable;
  7. import java.time.LocalDateTime;
  8. @Data
  9. //@TableName("tbl_weixin")
  10. public class User implements Serializable {
  11. //    @TableId(value = "id", type = IdType.AUTO)
  12.     private Long id;
  13.     //用户唯一标识
  14.     private String openId;
  15.     //昵称
  16.     private String nickName;
  17.     private String password;
  18.     //性别
  19.     private String gender;
  20.     //城市
  21.     private String city;
  22.     //省份
  23.     private String province;
  24.     //国家
  25.     private String country;
  26.     //头像
  27.     private String avatarUrl;
  28.     //用户在开放平台的唯一标识
  29.     private String unionId;
  30.     //是否授权
  31.     private Integer isAuth;
  32.     //创建时间
  33.     private LocalDateTime createTime;
  34.     //修改时间
  35.     private LocalDateTime updateTime;
  36. }
UserController
  1. package com.xxx.app.controller;
  2. import com.xxx.app.handler.NoAuth;
  3. import com.xxx.app.model.WXAuth;
  4. import com.xxx.app.service.UserService;
  5. import lombok.extern.slf4j.Slf4j;
  6. import org.springframework.beans.factory.annotation.Autowired;
  7. import org.springframework.web.bind.annotation.*;
  8. @Slf4j
  9. @RestController
  10. public class UserController {
  11. //    @Autowired
  12. //    private UserService userService;
  13.     //getSessionId
  14. //    @GetMapping("/getSessionId")
  15. //    @NoAuth
  16. //    public Result getSessionId(String code) {
  17. //        return userService.getSessionId(code);
  18. //    }
  19. //
  20. //    @PostMapping("/authLogin")
  21. //    @NoAuth
  22. //    public Result authLogin(@RequestBody WXAuth wxAuth) {
  23. //        Result result = userService.authLogin(wxAuth);
  24. //        log.info("{}", result);
  25. //        return result;
  26. //    }
  27. //
  28. //    @GetMapping("userinfo")
  29. //    public Result userinfo(Boolean refresh) {
  30. //        return userService.userinfo(refresh);
  31. //    }
  32. }
  1. package com.xxx.app.controller;
  2. import cn.hutool.json.JSONException;
  3. import com.xxx.app.entity.Sms;
  4. import com.xxx.app.handler.NoAuth;
  5. import com.github.qcloudsms.SmsSingleSender;
  6. import com.github.qcloudsms.SmsSingleSenderResult;
  7. import com.github.qcloudsms.httpclient.HTTPException;
  8. import lombok.extern.slf4j.Slf4j;
  9. import org.apache.commons.lang3.RandomStringUtils;
  10. import org.springframework.web.bind.annotation.RequestBody;
  11. import org.springframework.web.bind.annotation.RequestMapping;
  12. import org.springframework.web.bind.annotation.RequestMethod;
  13. import org.springframework.web.bind.annotation.RestController;
  14. import java.io.IOException;
  15. @RestController
  16. //@RequestMapping(value = "/sms") //sendCode
  17. @Slf4j
  18. public class SmsController {
  19.     @NoAuth
  20.     @RequestMapping(value = "/sendSms", method = RequestMethod.POST)
  21.     public void sendCode(@RequestBody Sms sms) {
  22.         // String 返回
  23.         // String code = generateCode();
  24.         // redisService.set(mobile, code, 300L); // 有效时间5分钟
  25.         // boolean result = smsService.sendSms(mobile, code);
  26.         // 生产6位数的随机验证码
  27.         // String code = String.valueOf((int) ((Math.random() * 9 + 1) * 100000));
  28.         try{
  29. //            String[] params = {sms.getCode()}; //短信中的参数
  30.             String[] params = {"1" + RandomStringUtils.randomNumeric(5)};
  31.             System.out.println("params = " + params);
  32.             //String[] params = ["2344"];
  33.             SmsSingleSender ssender = new SmsSingleSender(appid,appkey);
  34.             SmsSingleSenderResult result = ssender.sendWithParam(
  35.                     "86",
  36.                     sms.getPhoneNumber(),
  37.                     templateId,
  38.                     params,
  39.                     smsSign,
  40.                     "","");
  41.             System.out.println(result);
  42.         }catch (HTTPException e){
  43.             e.printStackTrace();
  44.         }catch (JSONException e){
  45.             e.printStackTrace();
  46.         }catch (IOException e){
  47.             e.printStackTrace();
  48.         }
  49.     }
  50. }
  1. package com.xxx.app.controller;
  2. import java.util.List;
  3. import java.util.Set;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.web.bind.annotation.GetMapping;
  6. import org.springframework.web.bind.annotation.PostMapping;
  7. import org.springframework.web.bind.annotation.RequestBody;
  8. import org.springframework.web.bind.annotation.RestController;
  9. import com.xxx.common.constant.Constants;
  10. import com.xxx.common.core.domain.AjaxResult;
  11. import com.xxx.common.core.domain.entity.SysMenu;
  12. import com.xxx.common.core.domain.entity.SysUser;
  13. import com.xxx.common.core.domain.model.LoginBody;
  14. import com.xxx.common.utils.SecurityUtils;
  15. import com.xxx.framework.web.service.SysLoginService;
  16. import com.xxx.framework.web.service.SysPermissionService;
  17. import com.xxx.system.service.ISysMenuService;
  18. /**
  19.  * 登录验证
  20.  *
  21.  */
  22. @RestController
  23. public class SysLoginController
  24. {
  25.     @Autowired
  26.     private SysLoginService loginService;
  27.     @Autowired
  28.     private ISysMenuService menuService;
  29.     @Autowired
  30.     private SysPermissionService permissionService;
  31.     /**
  32.      * 登录方法
  33.      *
  34.      * @param loginBody 登录信息
  35.      * @return 结果
  36.      */
  37.     @PostMapping("/login")
  38.     public AjaxResult login(@RequestBody LoginBody loginBody)
  39.     {
  40.         AjaxResult ajax = AjaxResult.success();
  41.         // 生成令牌
  42. //        String token = loginService.login(loginBody.getUsername(), loginBody.getPassword(), loginBody.getCode(),
  43. //                loginBody.getUuid());
  44. //        ajax.put(Constants.TOKEN, token);
  45.         return ajax;
  46.     }
  47.     /**
  48.      * 获取用户信息
  49.      *
  50.      * @return 用户信息
  51.      */
  52.     @GetMapping("getInfo")
  53.     public AjaxResult getInfo()
  54.     {
  55.         SysUser user = SecurityUtils.getLoginUser().getUser();
  56.         // 角色集合
  57.         Set<String> roles = permissionService.getRolePermission(user);
  58.         // 权限集合
  59.         Set<String> permissions = permissionService.getMenuPermission(user);
  60.         AjaxResult ajax = AjaxResult.success();
  61.         ajax.put("user", user);
  62.         ajax.put("roles", roles);
  63.         ajax.put("permissions", permissions);
  64.         return ajax;
  65.     }
  66.     /**
  67.      * 获取路由信息
  68.      *
  69.      * @return 路由信息
  70.      */
  71.     @GetMapping("getRouters")
  72.     public AjaxResult getRouters()
  73.     {
  74.         Long userId = SecurityUtils.getUserId();
  75.         List<SysMenu> menus = menuService.selectMenuTreeByUserId(userId);
  76.         return AjaxResult.success(menuService.buildMenus(menus));
  77.     }
  78. }

加群联系作者vx:xiaoda0423

仓库地址:https://github.com/webVueBlog/WebGuideInterview

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/我家小花儿/article/detail/883846
推荐阅读
相关标签