CaptchaController.java 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. package org.dromara.web.controller;
  2. import cn.dev33.satoken.annotation.SaIgnore;
  3. import cn.hutool.captcha.AbstractCaptcha;
  4. import cn.hutool.captcha.generator.CodeGenerator;
  5. import cn.hutool.core.util.IdUtil;
  6. import cn.hutool.core.util.RandomUtil;
  7. import jakarta.validation.constraints.NotBlank;
  8. import lombok.RequiredArgsConstructor;
  9. import lombok.extern.slf4j.Slf4j;
  10. import org.dromara.common.core.constant.Constants;
  11. import org.dromara.common.core.constant.GlobalConstants;
  12. import org.dromara.common.core.domain.R;
  13. import org.dromara.common.core.utils.SpringUtils;
  14. import org.dromara.common.core.utils.StringUtils;
  15. import org.dromara.common.core.utils.reflect.ReflectUtils;
  16. import org.dromara.common.mail.config.properties.MailProperties;
  17. import org.dromara.common.mail.utils.MailUtils;
  18. import org.dromara.common.ratelimiter.annotation.RateLimiter;
  19. import org.dromara.common.ratelimiter.enums.LimitType;
  20. import org.dromara.common.redis.utils.RedisUtils;
  21. import org.dromara.common.web.config.properties.CaptchaProperties;
  22. import org.dromara.common.web.enums.CaptchaType;
  23. import org.dromara.sms4j.api.SmsBlend;
  24. import org.dromara.sms4j.api.entity.SmsResponse;
  25. import org.dromara.sms4j.core.factory.SmsFactory;
  26. import org.dromara.web.domain.vo.CaptchaVo;
  27. import org.springframework.expression.Expression;
  28. import org.springframework.expression.ExpressionParser;
  29. import org.springframework.expression.spel.standard.SpelExpressionParser;
  30. import org.springframework.validation.annotation.Validated;
  31. import org.springframework.web.bind.annotation.GetMapping;
  32. import org.springframework.web.bind.annotation.RestController;
  33. import java.time.Duration;
  34. import java.util.LinkedHashMap;
  35. /**
  36. * 验证码操作处理
  37. *
  38. * @author Lion Li
  39. */
  40. @SaIgnore
  41. @Slf4j
  42. @Validated
  43. @RequiredArgsConstructor
  44. @RestController
  45. public class CaptchaController {
  46. private final CaptchaProperties captchaProperties;
  47. private final MailProperties mailProperties;
  48. /**
  49. * 短信验证码
  50. *
  51. * @param phonenumber 用户手机号
  52. */
  53. @RateLimiter(key = "#phonenumber", time = 60, count = 1)
  54. @GetMapping("/resource/sms/code")
  55. public R<Void> smsCode(@NotBlank(message = "{user.phonenumber.not.blank}") String phonenumber) {
  56. String key = GlobalConstants.CAPTCHA_CODE_KEY + phonenumber;
  57. String code = RandomUtil.randomNumbers(4);
  58. RedisUtils.setCacheObject(key, code, Duration.ofMinutes(Constants.CAPTCHA_EXPIRATION));
  59. // 验证码模板id 自行处理 (查数据库或写死均可)
  60. String templateId = "";
  61. LinkedHashMap<String, String> map = new LinkedHashMap<>(1);
  62. map.put("code", code);
  63. SmsBlend smsBlend = SmsFactory.getSmsBlend("config1");
  64. SmsResponse smsResponse = smsBlend.sendMessage(phonenumber, templateId, map);
  65. if (!smsResponse.isSuccess()) {
  66. log.error("验证码短信发送异常 => {}", smsResponse);
  67. return R.fail(smsResponse.getData().toString());
  68. }
  69. return R.ok();
  70. }
  71. /**
  72. * 邮箱验证码
  73. *
  74. * @param email 邮箱
  75. */
  76. @RateLimiter(key = "#email", time = 60, count = 1)
  77. @GetMapping("/resource/email/code")
  78. public R<Void> emailCode(@NotBlank(message = "{user.email.not.blank}") String email) {
  79. if (!mailProperties.getEnabled()) {
  80. return R.fail("当前系统没有开启邮箱功能!");
  81. }
  82. String key = GlobalConstants.CAPTCHA_CODE_KEY + email;
  83. String code = RandomUtil.randomNumbers(4);
  84. RedisUtils.setCacheObject(key, code, Duration.ofMinutes(Constants.CAPTCHA_EXPIRATION));
  85. try {
  86. MailUtils.sendText(email, "登录验证码", "您本次验证码为:" + code + ",有效性为" + Constants.CAPTCHA_EXPIRATION + "分钟,请尽快填写。");
  87. } catch (Exception e) {
  88. log.error("验证码短信发送异常 => {}", e.getMessage());
  89. return R.fail(e.getMessage());
  90. }
  91. return R.ok();
  92. }
  93. /**
  94. * 生成验证码
  95. */
  96. @RateLimiter(time = 60, count = 10, limitType = LimitType.IP)
  97. @GetMapping("/auth/code")
  98. public R<CaptchaVo> getCode() {
  99. CaptchaVo captchaVo = new CaptchaVo();
  100. boolean captchaEnabled = captchaProperties.getEnable();
  101. if (!captchaEnabled) {
  102. captchaVo.setCaptchaEnabled(false);
  103. return R.ok(captchaVo);
  104. }
  105. // 保存验证码信息
  106. String uuid = IdUtil.simpleUUID();
  107. String verifyKey = GlobalConstants.CAPTCHA_CODE_KEY + uuid;
  108. // 生成验证码
  109. CaptchaType captchaType = captchaProperties.getType();
  110. boolean isMath = CaptchaType.MATH == captchaType;
  111. Integer length = isMath ? captchaProperties.getNumberLength() : captchaProperties.getCharLength();
  112. CodeGenerator codeGenerator = ReflectUtils.newInstance(captchaType.getClazz(), length);
  113. AbstractCaptcha captcha = SpringUtils.getBean(captchaProperties.getCategory().getClazz());
  114. captcha.setGenerator(codeGenerator);
  115. captcha.createCode();
  116. // 如果是数学验证码,使用SpEL表达式处理验证码结果
  117. String code = captcha.getCode();
  118. if (isMath) {
  119. ExpressionParser parser = new SpelExpressionParser();
  120. Expression exp = parser.parseExpression(StringUtils.remove(code, "="));
  121. code = exp.getValue(String.class);
  122. }
  123. RedisUtils.setCacheObject(verifyKey, code, Duration.ofMinutes(Constants.CAPTCHA_EXPIRATION));
  124. captchaVo.setUuid(uuid);
  125. captchaVo.setImg(captcha.getImageBase64());
  126. return R.ok(captchaVo);
  127. }
  128. }